import React from 'react'; import classNames from 'classnames'; import { CallDetailsType, HangUpType, SetLocalAudioType, SetLocalVideoType, SetVideoCapturerType, SetVideoRendererType, } from '../state/ducks/calling'; import { Avatar } from './Avatar'; import { CallState } from '../types/Calling'; import { LocalizerType } from '../types/Util'; import { CanvasVideoRenderer, GumVideoCapturer } from '../window.d'; type CallingButtonProps = { classNameSuffix: string; onClick: () => void; }; const CallingButton = ({ classNameSuffix, onClick, }: CallingButtonProps): JSX.Element => { const className = classNames( 'module-ongoing-call__icon', `module-ongoing-call__icon${classNameSuffix}` ); return ( ); }; export type PropsType = { callDetails?: CallDetailsType; callState?: CallState; getVideoCapturer: ( ref: React.RefObject ) => GumVideoCapturer; getVideoRenderer: ( ref: React.RefObject ) => CanvasVideoRenderer; hangUp: (_: HangUpType) => void; hasLocalAudio: boolean; hasLocalVideo: boolean; hasRemoteVideo: boolean; i18n: LocalizerType; setLocalAudio: (_: SetLocalAudioType) => void; setLocalVideo: (_: SetLocalVideoType) => void; setVideoCapturer: (_: SetVideoCapturerType) => void; setVideoRenderer: (_: SetVideoRendererType) => void; }; type StateType = { acceptedTime: number | null; acceptedDuration: number | null; showControls: boolean; }; export class CallScreen extends React.Component { private interval: any; private controlsFadeTimer: any; private readonly localVideoRef: React.RefObject; private readonly remoteVideoRef: React.RefObject; constructor(props: PropsType) { super(props); this.state = { acceptedTime: null, acceptedDuration: null, showControls: true, }; this.interval = null; this.controlsFadeTimer = null; this.localVideoRef = React.createRef(); this.remoteVideoRef = React.createRef(); this.setVideoCapturerAndRenderer( props.getVideoCapturer(this.localVideoRef), props.getVideoRenderer(this.remoteVideoRef) ); } public componentDidMount() { // It's really jump with a value of 500ms. this.interval = setInterval(this.updateAcceptedTimer, 100); this.fadeControls(); document.addEventListener('keydown', this.handleKeyDown); } public componentWillUnmount() { document.removeEventListener('keydown', this.handleKeyDown); if (this.interval) { clearInterval(this.interval); } if (this.controlsFadeTimer) { clearTimeout(this.controlsFadeTimer); } this.setVideoCapturerAndRenderer(null, null); } updateAcceptedTimer = () => { const { acceptedTime } = this.state; const { callState } = this.props; if (acceptedTime) { this.setState({ acceptedTime, acceptedDuration: Date.now() - acceptedTime, }); } else if ( callState === CallState.Accepted || callState === CallState.Reconnecting ) { this.setState({ acceptedTime: Date.now(), acceptedDuration: 1, }); } }; handleKeyDown = (event: KeyboardEvent) => { const { callDetails } = this.props; if (!callDetails) { return; } let eventHandled = false; if (event.key === 'V') { this.toggleVideo(); eventHandled = true; } else if (event.key === 'M') { this.toggleAudio(); eventHandled = true; } if (eventHandled) { event.preventDefault(); event.stopPropagation(); this.showControls(); } }; showControls = () => { if (!this.state.showControls) { this.setState({ showControls: true, }); } this.fadeControls(); }; fadeControls = () => { if (this.controlsFadeTimer) { clearTimeout(this.controlsFadeTimer); } this.controlsFadeTimer = setTimeout(() => { this.setState({ showControls: false, }); }, 5000); }; toggleAudio = () => { const { callDetails, hasLocalAudio, setLocalAudio } = this.props; if (!callDetails) { return; } setLocalAudio({ callId: callDetails.callId, enabled: !hasLocalAudio, }); }; toggleVideo = () => { const { callDetails, hasLocalVideo, setLocalVideo } = this.props; if (!callDetails) { return; } setLocalVideo({ callId: callDetails.callId, enabled: !hasLocalVideo }); }; public render() { const { callDetails, callState, hangUp, hasLocalAudio, hasLocalVideo, hasRemoteVideo, } = this.props; const { showControls } = this.state; const isAudioOnly = !hasLocalVideo && !hasRemoteVideo; if (!callDetails || !callState) { return null; } const controlsFadeClass = classNames({ 'module-ongoing-call__controls--fadeIn': (showControls || isAudioOnly) && callState !== CallState.Accepted, 'module-ongoing-call__controls--fadeOut': !showControls && !isAudioOnly && callState === CallState.Accepted, }); const toggleAudioSuffix = hasLocalAudio ? '--audio--enabled' : '--audio--disabled'; const toggleVideoSuffix = hasLocalVideo ? '--video--enabled' : '--video--disabled'; return (
{callDetails.title}
{this.renderMessage(callState)}
{hasRemoteVideo ? this.renderRemoteVideo() : this.renderAvatar(callDetails)} {hasLocalVideo ? this.renderLocalVideo() : null}
{ hangUp({ callId: callDetails.callId }); }} />
); } private renderAvatar(callDetails: CallDetailsType) { const { i18n } = this.props; const { avatarPath, color, name, phoneNumber, profileName, title, } = callDetails; return (
); } private renderLocalVideo() { return (