2018-07-15 14:50:38 +00:00
|
|
|
// @flow
|
|
|
|
|
|
|
|
import React, { Component } from 'react';
|
|
|
|
import { Image, Text, View } from 'react-native';
|
|
|
|
import LinearGradient from 'react-native-linear-gradient';
|
|
|
|
import { connect } from 'react-redux';
|
|
|
|
|
|
|
|
import { translate } from '../../../base/i18n';
|
|
|
|
import { Avatar } from '../../../base/participants';
|
|
|
|
|
|
|
|
import AnswerButton from './AnswerButton';
|
|
|
|
import DeclineButton from './DeclineButton';
|
|
|
|
import styles, {
|
|
|
|
AVATAR_BORDER_GRADIENT,
|
|
|
|
BACKGROUND_OVERLAY_GRADIENT,
|
|
|
|
CALLER_AVATAR_SIZE
|
|
|
|
} from './styles';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The type of React {@code Component} props of {@link IncomingCallPage}.
|
|
|
|
*/
|
|
|
|
type Props = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Caller's avatar URL.
|
|
|
|
*/
|
2018-07-16 16:36:32 +00:00
|
|
|
_callerAvatarURL: string,
|
2018-07-15 14:50:38 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Caller's name.
|
|
|
|
*/
|
|
|
|
_callerName: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the call has video or not.
|
|
|
|
*/
|
|
|
|
_hasVideo: boolean,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper for translating strings.
|
|
|
|
*/
|
|
|
|
t: Function
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The React {@code Component} displays an incoming call screen.
|
|
|
|
*/
|
|
|
|
class IncomingCallPage extends Component<Props> {
|
|
|
|
/**
|
|
|
|
* Implements React's {@link Component#render()}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
render() {
|
|
|
|
const { t, _callerName, _hasVideo } = this.props;
|
|
|
|
const callTitle
|
|
|
|
= _hasVideo
|
|
|
|
? t('incomingCall.videoCallTitle')
|
|
|
|
: t('incomingCall.audioCallTitle');
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View style = { styles.pageContainer }>
|
|
|
|
<View style = { styles.backgroundAvatar }>
|
|
|
|
<Image
|
2018-07-16 16:36:32 +00:00
|
|
|
source = {{ uri: this.props._callerAvatarURL }}
|
2018-07-15 14:50:38 +00:00
|
|
|
style = { styles.backgroundAvatarImage } />
|
|
|
|
</View>
|
|
|
|
<LinearGradient
|
|
|
|
colors = { BACKGROUND_OVERLAY_GRADIENT }
|
|
|
|
style = { styles.backgroundOverlayGradient } />
|
|
|
|
<Text style = { styles.title }>
|
|
|
|
{ callTitle }
|
|
|
|
</Text>
|
|
|
|
<Text
|
|
|
|
numberOfLines = { 6 }
|
|
|
|
style = { styles.callerName } >
|
|
|
|
{ _callerName }
|
|
|
|
</Text>
|
|
|
|
<Text style = { styles.productLabel }>
|
|
|
|
{ t('incomingCall.productLabel') }
|
|
|
|
</Text>
|
|
|
|
{ this._renderCallerAvatar() }
|
|
|
|
{ this._renderButtons() }
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders buttons.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {React$Node}
|
|
|
|
*/
|
|
|
|
_renderButtons() {
|
|
|
|
const { t } = this.props;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View style = { styles.buttonsContainer }>
|
|
|
|
<View style = { styles.buttonWrapper } >
|
|
|
|
<DeclineButton
|
|
|
|
styles = { styles.declineButtonStyles } />
|
|
|
|
<Text style = { styles.buttonText }>
|
|
|
|
{ t('incomingCall.decline') }
|
|
|
|
</Text>
|
|
|
|
</View>
|
|
|
|
<View style = { styles.buttonWrapper }>
|
|
|
|
<AnswerButton
|
|
|
|
styles = { styles.answerButtonStyles } />
|
|
|
|
<Text style = { styles.buttonText }>
|
|
|
|
{ t('incomingCall.answer') }
|
|
|
|
</Text>
|
|
|
|
</View>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
2018-07-16 16:36:32 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Renders caller avatar.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {React$Node}
|
|
|
|
*/
|
|
|
|
_renderCallerAvatar() {
|
|
|
|
return (
|
|
|
|
<View style = { styles.avatarContainer }>
|
|
|
|
<LinearGradient
|
|
|
|
colors = { AVATAR_BORDER_GRADIENT }
|
|
|
|
style = { styles.avatarBorder } />
|
|
|
|
<View style = { styles.avatar }>
|
|
|
|
<Avatar
|
|
|
|
size = { CALLER_AVATAR_SIZE }
|
|
|
|
uri = { this.props._callerAvatarURL } />
|
|
|
|
</View>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
2018-07-15 14:50:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps (parts of) the redux state to the component's props.
|
|
|
|
*
|
|
|
|
* @param {Object} state - The redux state.
|
|
|
|
* @param {Object} ownProps - The component's own props.
|
|
|
|
* @private
|
|
|
|
* @returns {{
|
2018-07-16 16:36:32 +00:00
|
|
|
* _callerAvatarURL: string,
|
2018-07-15 14:50:38 +00:00
|
|
|
* _callerName: string,
|
|
|
|
* _hasVideo: boolean
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
function _mapStateToProps(state) {
|
|
|
|
const { caller } = state['features/mobile/incoming-call'] || {};
|
|
|
|
|
|
|
|
return {
|
|
|
|
/**
|
|
|
|
* The caller's avatar url.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string}
|
|
|
|
*/
|
2018-07-16 16:36:32 +00:00
|
|
|
_callerAvatarURL: caller.avatarUrl,
|
2018-07-15 14:50:38 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The caller's name.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {string}
|
|
|
|
*/
|
|
|
|
_callerName: caller.name,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the call has video or not.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {boolean}
|
|
|
|
*/
|
|
|
|
_hasVideo: caller.hasVideo
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export default translate(connect(_mapStateToProps)(IncomingCallPage));
|