// Copyright 2020-2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactNode } from 'react'; import React, { useState, useRef, useEffect, useCallback } from 'react'; import { noop } from 'lodash'; import classNames from 'classnames'; import type { VideoFrameSource } from 'ringrtc'; import type { HangUpType, SetLocalAudioType, SetLocalPreviewType, SetLocalVideoType, SetRendererCanvasType, } from '../state/ducks/calling'; import { Avatar } from './Avatar'; import { CallingHeader } from './CallingHeader'; import { CallingPreCallInfo, RingMode } from './CallingPreCallInfo'; import { CallingButton, CallingButtonType } from './CallingButton'; import { CallBackgroundBlur } from './CallBackgroundBlur'; import type { ActiveCallType, GroupCallVideoRequest, PresentedSource, } from '../types/Calling'; import { CallMode, CallState, GroupCallConnectionState, GroupCallJoinState, } from '../types/Calling'; import type { AvatarColorType } from '../types/Colors'; import { AvatarColors } from '../types/Colors'; import type { ConversationType } from '../state/ducks/conversations'; import { CallingToastManager } from './CallingToastManager'; import { DirectCallRemoteParticipant } from './DirectCallRemoteParticipant'; import { GroupCallRemoteParticipants } from './GroupCallRemoteParticipants'; import type { LocalizerType } from '../types/Util'; import { NeedsScreenRecordingPermissionsModal } from './NeedsScreenRecordingPermissionsModal'; import { missingCaseError } from '../util/missingCaseError'; import type { UUIDStringType } from '../types/UUID'; import * as KeyboardLayout from '../services/keyboardLayout'; import { useActivateSpeakerViewOnPresenting } from '../hooks/useActivateSpeakerViewOnPresenting'; export type PropsType = { activeCall: ActiveCallType; getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource; getPresentingSources: () => void; groupMembers?: Array>; hangUp: (_: HangUpType) => void; i18n: LocalizerType; joinedAt?: number; me: { avatarPath?: string; color?: AvatarColorType; id: string; name?: string; phoneNumber?: string; profileName?: string; title: string; uuid: UUIDStringType; }; openSystemPreferencesAction: () => unknown; setGroupCallVideoRequest: (_: Array) => void; setLocalAudio: (_: SetLocalAudioType) => void; setLocalVideo: (_: SetLocalVideoType) => void; setLocalPreview: (_: SetLocalPreviewType) => void; setPresenting: (_?: PresentedSource) => void; setRendererCanvas: (_: SetRendererCanvasType) => void; stickyControls: boolean; toggleParticipants: () => void; togglePip: () => void; toggleScreenRecordingPermissionsDialog: () => unknown; toggleSettings: () => void; toggleSpeakerView: () => void; }; type DirectCallHeaderMessagePropsType = { i18n: LocalizerType; callState: CallState; joinedAt?: number; }; function DirectCallHeaderMessage({ callState, i18n, joinedAt, }: DirectCallHeaderMessagePropsType): JSX.Element | null { const [acceptedDuration, setAcceptedDuration] = useState< number | undefined >(); useEffect(() => { if (!joinedAt) { return noop; } // It's really jumpy with a value of 500ms. const interval = setInterval(() => { setAcceptedDuration(Date.now() - joinedAt); }, 100); return clearInterval.bind(null, interval); }, [joinedAt]); if (callState === CallState.Reconnecting) { return <>{i18n('callReconnecting')}; } if (callState === CallState.Accepted && acceptedDuration) { return <>{i18n('callDuration', [renderDuration(acceptedDuration)])}; } return null; } export const CallScreen: React.FC = ({ activeCall, getGroupCallVideoFrameSource, getPresentingSources, groupMembers, hangUp, i18n, joinedAt, me, openSystemPreferencesAction, setGroupCallVideoRequest, setLocalAudio, setLocalVideo, setLocalPreview, setPresenting, setRendererCanvas, stickyControls, toggleParticipants, togglePip, toggleScreenRecordingPermissionsDialog, toggleSettings, toggleSpeakerView, }) => { const { conversation, hasLocalAudio, hasLocalVideo, isInSpeakerView, presentingSource, remoteParticipants, showNeedsScreenRecordingPermissionsWarning, showParticipantsList, } = activeCall; useActivateSpeakerViewOnPresenting( remoteParticipants, isInSpeakerView, toggleSpeakerView ); const toggleAudio = useCallback(() => { setLocalAudio({ enabled: !hasLocalAudio, }); }, [setLocalAudio, hasLocalAudio]); const toggleVideo = useCallback(() => { setLocalVideo({ enabled: !hasLocalVideo, }); }, [setLocalVideo, hasLocalVideo]); const togglePresenting = useCallback(() => { if (presentingSource) { setPresenting(); } else { getPresentingSources(); } }, [getPresentingSources, presentingSource, setPresenting]); const [controlsHover, setControlsHover] = useState(false); const onControlsMouseEnter = useCallback(() => { setControlsHover(true); }, [setControlsHover]); const onControlsMouseLeave = useCallback(() => { setControlsHover(false); }, [setControlsHover]); const [showControls, setShowControls] = useState(true); const localVideoRef = useRef(null); useEffect(() => { setLocalPreview({ element: localVideoRef }); return () => { setLocalPreview({ element: undefined }); }; }, [setLocalPreview, setRendererCanvas]); useEffect(() => { if (!showControls || stickyControls || controlsHover) { return noop; } const timer = setTimeout(() => { setShowControls(false); }, 5000); return clearTimeout.bind(null, timer); }, [showControls, stickyControls, controlsHover]); useEffect(() => { const handleKeyDown = (event: KeyboardEvent): void => { let eventHandled = false; const key = KeyboardLayout.lookup(event); if (event.shiftKey && (key === 'V' || key === 'v')) { toggleVideo(); eventHandled = true; } else if (event.shiftKey && (key === 'M' || key === 'm')) { toggleAudio(); eventHandled = true; } if (eventHandled) { event.preventDefault(); event.stopPropagation(); setShowControls(true); } }; document.addEventListener('keydown', handleKeyDown); return () => { document.removeEventListener('keydown', handleKeyDown); }; }, [toggleAudio, toggleVideo]); const currentPresenter = remoteParticipants.find( participant => participant.presenting ); const hasRemoteVideo = remoteParticipants.some( remoteParticipant => remoteParticipant.hasRemoteVideo ); const isSendingVideo = hasLocalVideo || presentingSource; let isRinging: boolean; let hasCallStarted: boolean; let headerMessage: ReactNode | undefined; let headerTitle: string | undefined; let isConnected: boolean; let participantCount: number; let remoteParticipantsElement: ReactNode; switch (activeCall.callMode) { case CallMode.Direct: { isRinging = activeCall.callState === CallState.Prering || activeCall.callState === CallState.Ringing; hasCallStarted = !isRinging; headerMessage = ( ); headerTitle = isRinging ? undefined : conversation.title; isConnected = activeCall.callState === CallState.Accepted; participantCount = isConnected ? 2 : 0; remoteParticipantsElement = hasCallStarted ? ( ) : (
); break; } case CallMode.Group: isRinging = activeCall.outgoingRing && !activeCall.remoteParticipants.length; hasCallStarted = activeCall.joinState !== GroupCallJoinState.NotJoined; participantCount = activeCall.remoteParticipants.length + 1; if (isRinging) { headerTitle = undefined; } else if (currentPresenter) { headerTitle = i18n('calling__presenting--person-ongoing', [ currentPresenter.title, ]); } else if (!activeCall.remoteParticipants.length) { headerTitle = i18n('calling__in-this-call--zero'); } isConnected = activeCall.connectionState === GroupCallConnectionState.Connected; remoteParticipantsElement = ( ); break; default: throw missingCaseError(activeCall); } let lonelyInGroupNode: ReactNode; let localPreviewNode: ReactNode; if ( activeCall.callMode === CallMode.Group && !activeCall.remoteParticipants.length ) { lonelyInGroupNode = (
{isSendingVideo ? (
); } else { localPreviewNode = isSendingVideo ? (