signal-desktop/ts/components/CallManager.tsx

802 lines
24 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2020 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
2021-08-20 16:06:15 +00:00
import React, { useCallback, useEffect } from 'react';
import { noop } from 'lodash';
2023-01-09 18:38:57 +00:00
import type { VideoFrameSource } from '@signalapp/ringrtc';
2020-10-01 19:09:15 +00:00
import { CallNeedPermissionScreen } from './CallNeedPermissionScreen';
import { CallScreen } from './CallScreen';
2020-11-17 15:07:53 +00:00
import { CallingLobby } from './CallingLobby';
import { CallingParticipantsList } from './CallingParticipantsList';
import { CallingSelectPresentingSourcesModal } from './CallingSelectPresentingSourcesModal';
2020-11-17 15:07:53 +00:00
import { CallingPip } from './CallingPip';
import { IncomingCallBar } from './IncomingCallBar';
import type {
2020-12-02 18:14:03 +00:00
ActiveCallType,
2024-02-22 21:19:50 +00:00
CallingConversationType,
CallViewMode,
GroupCallVideoRequest,
} from '../types/Calling';
import {
2020-11-17 15:07:53 +00:00
CallEndedReason,
2020-11-13 19:57:55 +00:00
CallState,
GroupCallConnectionState,
2020-11-13 19:57:55 +00:00
GroupCallJoinState,
} from '../types/Calling';
import { CallMode } from '../types/CallDisposition';
import type { ConversationType } from '../state/ducks/conversations';
import type {
AcceptCallType,
BatchUserActionPayloadType,
2020-11-13 19:57:55 +00:00
CancelCallType,
DeclineCallType,
GroupCallParticipantInfoType,
PendingUserActionPayloadType,
RemoveClientType,
2023-12-06 21:52:29 +00:00
SendGroupCallRaiseHandType,
2023-11-16 19:55:35 +00:00
SendGroupCallReactionType,
SetGroupCallVideoRequestType,
2020-11-13 19:57:55 +00:00
SetLocalAudioType,
SetLocalPreviewType,
SetLocalVideoType,
SetRendererCanvasType,
2020-11-13 19:57:55 +00:00
StartCallType,
} from '../state/ducks/calling';
2024-02-22 21:19:50 +00:00
import { CallLinkRestrictions } from '../types/CallLink';
import type { CallLinkType } from '../types/CallLink';
import type { LocalizerType } from '../types/Util';
2020-11-13 19:57:55 +00:00
import { missingCaseError } from '../util/missingCaseError';
import { CallingToastProvider } from './CallingToast';
2023-11-16 19:55:35 +00:00
import type { SmartReactionPicker } from '../state/smart/ReactionPicker';
import type { Props as ReactionPickerProps } from './conversation/ReactionPicker';
import * as log from '../logging/log';
2024-02-22 21:19:50 +00:00
import { isGroupOrAdhocActiveCall } from '../util/isGroupOrAdhocCall';
import { CallingAdhocCallInfo } from './CallingAdhocCallInfo';
import { callLinkRootKeyToUrl } from '../util/callLinkRootKeyToUrl';
import { usePrevious } from '../hooks/usePrevious';
2024-05-22 16:24:27 +00:00
import { copyCallLink } from '../util/copyLinksWithToast';
2020-06-04 18:16:19 +00:00
const GROUP_CALL_RING_DURATION = 60 * 1000;
export type DirectIncomingCall = Readonly<{
callMode: CallMode.Direct;
callState?: CallState;
callEndedReason?: CallEndedReason;
conversation: ConversationType;
isVideoCall: boolean;
}>;
export type GroupIncomingCall = Readonly<{
callMode: CallMode.Group;
connectionState: GroupCallConnectionState;
joinState: GroupCallJoinState;
conversation: ConversationType;
otherMembersRung: Array<Pick<ConversationType, 'firstName' | 'title'>>;
ringer: Pick<ConversationType, 'firstName' | 'title'>;
remoteParticipants: Array<GroupCallParticipantInfoType>;
}>;
export type CallingImageDataCache = Map<number, ImageData>;
export type PropsType = {
2020-11-13 19:57:55 +00:00
activeCall?: ActiveCallType;
availableCameras: Array<MediaDeviceInfo>;
2024-02-22 21:19:50 +00:00
callLink: CallLinkType | undefined;
2020-11-13 19:57:55 +00:00
cancelCall: (_: CancelCallType) => void;
changeCallView: (mode: CallViewMode) => void;
2020-10-01 19:09:15 +00:00
closeNeedPermissionScreen: () => void;
2020-11-13 19:57:55 +00:00
getGroupCallVideoFrameSource: (
conversationId: string,
demuxId: number
) => VideoFrameSource;
getIsSharingPhoneNumberWithEverybody: () => boolean;
getPresentingSources: () => void;
incomingCall: DirectIncomingCall | GroupIncomingCall | null;
2020-08-27 00:03:42 +00:00
renderDeviceSelection: () => JSX.Element;
2023-11-16 19:55:35 +00:00
renderReactionPicker: (
props: React.ComponentProps<typeof SmartReactionPicker>
) => JSX.Element;
showContactModal: (contactId: string, conversationId?: string) => void;
startCall: (payload: StartCallType) => void;
2020-10-08 01:25:33 +00:00
toggleParticipants: () => void;
acceptCall: (_: AcceptCallType) => void;
approveUser: (payload: PendingUserActionPayloadType) => void;
batchUserAction: (payload: BatchUserActionPayloadType) => void;
2021-08-20 16:06:15 +00:00
bounceAppIconStart: () => unknown;
bounceAppIconStop: () => unknown;
2024-09-20 01:03:44 +00:00
cancelPresenting: () => void;
declineCall: (_: DeclineCallType) => void;
denyUser: (payload: PendingUserActionPayloadType) => void;
hasInitialLoadCompleted: boolean;
i18n: LocalizerType;
2024-06-11 23:45:28 +00:00
isGroupCallRaiseHandEnabled: boolean;
me: ConversationType;
2023-08-01 16:06:29 +00:00
notifyForCall: (
conversationId: string,
title: string,
isVideoCall: boolean
) => unknown;
openSystemPreferencesAction: () => unknown;
2021-08-20 16:06:15 +00:00
playRingtone: () => unknown;
removeClient: (payload: RemoveClientType) => void;
2024-06-29 00:13:20 +00:00
blockClient: (payload: RemoveClientType) => void;
2024-09-20 01:03:44 +00:00
selectPresentingSource: (id: string) => void;
2023-12-06 21:52:29 +00:00
sendGroupCallRaiseHand: (payload: SendGroupCallRaiseHandType) => void;
2023-11-16 19:55:35 +00:00
sendGroupCallReaction: (payload: SendGroupCallReactionType) => void;
setGroupCallVideoRequest: (_: SetGroupCallVideoRequestType) => void;
setIsCallActive: (_: boolean) => void;
setLocalAudio: (_: SetLocalAudioType) => void;
setLocalVideo: (_: SetLocalVideoType) => void;
setLocalPreview: (_: SetLocalPreviewType) => void;
setOutgoingRing: (_: boolean) => void;
setRendererCanvas: (_: SetRendererCanvasType) => void;
showShareCallLinkViaSignal: (
callLink: CallLinkType,
i18n: LocalizerType
) => void;
2021-08-20 16:06:15 +00:00
stopRingtone: () => unknown;
switchToPresentationView: () => void;
switchFromPresentationView: () => void;
2022-08-16 23:52:09 +00:00
hangUpActiveCall: (reason: string) => void;
togglePip: () => void;
toggleCallLinkPendingParticipantModal: (contactId: string) => void;
toggleScreenRecordingPermissionsDialog: () => unknown;
toggleSettings: () => void;
isConversationTooBigToRing: boolean;
2023-02-24 23:18:57 +00:00
pauseVoiceNotePlayer: () => void;
2023-11-16 19:55:35 +00:00
} & Pick<ReactionPickerProps, 'renderEmojiPicker'>;
2020-06-04 18:16:19 +00:00
type ActiveCallManagerPropsType = {
2020-11-13 19:57:55 +00:00
activeCall: ActiveCallType;
} & Omit<
PropsType,
| 'acceptCall'
| 'bounceAppIconStart'
| 'bounceAppIconStop'
| 'declineCall'
| 'hasInitialLoadCompleted'
| 'incomingCall'
| 'notifyForCall'
| 'playRingtone'
| 'setIsCallActive'
| 'stopRingtone'
| 'isConversationTooBigToRing'
>;
2020-11-13 19:57:55 +00:00
2022-11-18 00:45:19 +00:00
function ActiveCallManager({
activeCall,
approveUser,
availableCameras,
batchUserAction,
2024-06-29 00:13:20 +00:00
blockClient,
2024-02-22 21:19:50 +00:00
callLink,
2020-10-08 01:25:33 +00:00
cancelCall,
2024-09-20 01:03:44 +00:00
cancelPresenting,
changeCallView,
2020-10-01 19:09:15 +00:00
closeNeedPermissionScreen,
denyUser,
hangUpActiveCall,
2020-06-04 18:16:19 +00:00
i18n,
2024-06-11 23:45:28 +00:00
isGroupCallRaiseHandEnabled,
getIsSharingPhoneNumberWithEverybody,
2020-11-13 19:57:55 +00:00
getGroupCallVideoFrameSource,
getPresentingSources,
me,
openSystemPreferencesAction,
2020-08-27 00:03:42 +00:00
renderDeviceSelection,
2023-11-16 19:55:35 +00:00
renderEmojiPicker,
renderReactionPicker,
removeClient,
2024-09-20 01:03:44 +00:00
selectPresentingSource,
2023-12-06 21:52:29 +00:00
sendGroupCallRaiseHand,
2023-11-16 19:55:35 +00:00
sendGroupCallReaction,
setGroupCallVideoRequest,
2020-06-04 18:16:19 +00:00
setLocalAudio,
2020-08-27 00:03:42 +00:00
setLocalPreview,
2020-06-04 18:16:19 +00:00
setLocalVideo,
2020-08-27 00:03:42 +00:00
setRendererCanvas,
setOutgoingRing,
showContactModal,
showShareCallLinkViaSignal,
2020-10-08 01:25:33 +00:00
startCall,
switchToPresentationView,
switchFromPresentationView,
toggleCallLinkPendingParticipantModal,
2020-10-08 01:25:33 +00:00
toggleParticipants,
2020-10-01 00:43:05 +00:00
togglePip,
toggleScreenRecordingPermissionsDialog,
2020-08-27 00:03:42 +00:00
toggleSettings,
2023-02-24 23:18:57 +00:00
pauseVoiceNotePlayer,
2022-11-18 00:45:19 +00:00
}: ActiveCallManagerPropsType): JSX.Element {
2020-11-13 19:57:55 +00:00
const {
2020-11-17 15:07:53 +00:00
conversation,
2020-11-13 19:57:55 +00:00
hasLocalAudio,
hasLocalVideo,
2020-12-02 18:14:03 +00:00
peekedParticipants,
2020-11-13 19:57:55 +00:00
pip,
presentingSourcesAvailable,
2020-11-17 15:07:53 +00:00
settingsDialogOpen,
showParticipantsList,
outgoingRing,
2020-12-02 18:14:03 +00:00
} = activeCall;
2020-11-13 19:57:55 +00:00
const cancelActiveCall = useCallback(() => {
cancelCall({ conversationId: conversation.id });
}, [cancelCall, conversation.id]);
const joinActiveCall = useCallback(() => {
2023-02-24 23:18:57 +00:00
// pause any voice note playback
pauseVoiceNotePlayer();
2020-11-13 19:57:55 +00:00
startCall({
2020-12-02 18:14:03 +00:00
callMode: activeCall.callMode,
2020-11-13 19:57:55 +00:00
conversationId: conversation.id,
hasLocalAudio,
hasLocalVideo,
2020-11-13 19:57:55 +00:00
});
2020-12-02 18:14:03 +00:00
}, [
startCall,
activeCall.callMode,
conversation.id,
hasLocalAudio,
hasLocalVideo,
2023-02-24 23:18:57 +00:00
pauseVoiceNotePlayer,
2020-12-02 18:14:03 +00:00
]);
2020-11-13 19:57:55 +00:00
// For caching screenshare frames which update slowly, between Pip and CallScreen.
const imageDataCache = React.useRef<CallingImageDataCache>(new Map());
const previousConversationId = usePrevious(conversation.id, conversation.id);
useEffect(() => {
if (conversation.id !== previousConversationId) {
imageDataCache.current.clear();
}
}, [conversation.id, previousConversationId]);
2020-11-13 19:57:55 +00:00
const getGroupCallVideoFrameSourceForActiveCall = useCallback(
(demuxId: number) => {
return getGroupCallVideoFrameSource(conversation.id, demuxId);
},
[getGroupCallVideoFrameSource, conversation.id]
);
const setGroupCallVideoRequestForConversation = useCallback(
2022-09-07 15:52:55 +00:00
(resolutions: Array<GroupCallVideoRequest>, speakerHeight: number) => {
setGroupCallVideoRequest({
conversationId: conversation.id,
resolutions,
2022-09-07 15:52:55 +00:00
speakerHeight,
});
},
[setGroupCallVideoRequest, conversation.id]
);
2024-02-22 21:19:50 +00:00
const onCopyCallLink = useCallback(async () => {
if (!callLink) {
return;
}
const link = callLinkRootKeyToUrl(callLink.rootKey);
if (link) {
2024-05-22 16:24:27 +00:00
await copyCallLink(link);
2024-02-22 21:19:50 +00:00
}
2024-05-22 16:24:27 +00:00
}, [callLink]);
2024-02-22 21:19:50 +00:00
const handleShareCallLinkViaSignal = useCallback(() => {
if (!callLink) {
log.error('Missing call link');
return;
}
showShareCallLinkViaSignal(callLink, i18n);
}, [callLink, i18n, showShareCallLinkViaSignal]);
2020-12-02 18:14:03 +00:00
let isCallFull: boolean;
2020-11-13 19:57:55 +00:00
let showCallLobby: boolean;
let groupMembers:
| undefined
| Array<Pick<ConversationType, 'id' | 'firstName' | 'title'>>;
let isConvoTooBigToRing = false;
let isAdhocAdminApprovalRequired = false;
2024-02-22 21:19:50 +00:00
let isAdhocJoinRequestPending = false;
let isCallLinkAdmin = false;
2020-12-02 18:14:03 +00:00
switch (activeCall.callMode) {
2020-11-13 19:57:55 +00:00
case CallMode.Direct: {
2020-12-02 18:14:03 +00:00
const { callState, callEndedReason } = activeCall;
2020-11-13 19:57:55 +00:00
const ended = callState === CallState.Ended;
if (
ended &&
callEndedReason === CallEndedReason.RemoteHangupNeedPermission
) {
return (
<CallNeedPermissionScreen
close={closeNeedPermissionScreen}
conversation={conversation}
i18n={i18n}
/>
);
}
2020-11-13 19:57:55 +00:00
showCallLobby = !callState;
2020-12-02 18:14:03 +00:00
isCallFull = false;
groupMembers = undefined;
2020-11-13 19:57:55 +00:00
break;
}
2024-02-22 21:19:50 +00:00
case CallMode.Group:
case CallMode.Adhoc: {
showCallLobby = activeCall.joinState !== GroupCallJoinState.Joined;
2020-12-02 18:14:03 +00:00
isCallFull = activeCall.deviceCount >= activeCall.maxDevices;
isConvoTooBigToRing = activeCall.isConversationTooBigToRing;
({ groupMembers } = activeCall);
isAdhocAdminApprovalRequired =
!callLink?.adminKey &&
callLink?.restrictions === CallLinkRestrictions.AdminApproval;
2024-02-22 21:19:50 +00:00
isAdhocJoinRequestPending =
isAdhocAdminApprovalRequired &&
2024-02-22 21:19:50 +00:00
activeCall.joinState === GroupCallJoinState.Pending;
isCallLinkAdmin = Boolean(callLink?.adminKey);
2020-11-13 19:57:55 +00:00
break;
2020-10-01 00:43:05 +00:00
}
2020-11-13 19:57:55 +00:00
default:
2020-12-02 18:14:03 +00:00
throw missingCaseError(activeCall);
2020-11-13 19:57:55 +00:00
}
2020-10-01 00:43:05 +00:00
if (pip) {
return (
<CallingPip
activeCall={activeCall}
getGroupCallVideoFrameSource={getGroupCallVideoFrameSourceForActiveCall}
imageDataCache={imageDataCache}
hangUpActiveCall={hangUpActiveCall}
hasLocalVideo={hasLocalVideo}
i18n={i18n}
setGroupCallVideoRequest={setGroupCallVideoRequestForConversation}
setLocalPreview={setLocalPreview}
setRendererCanvas={setRendererCanvas}
switchToPresentationView={switchToPresentationView}
switchFromPresentationView={switchFromPresentationView}
togglePip={togglePip}
/>
);
}
2020-11-13 19:57:55 +00:00
if (showCallLobby) {
2020-06-04 18:16:19 +00:00
return (
2020-08-27 00:03:42 +00:00
<>
2020-11-13 19:57:55 +00:00
<CallingLobby
availableCameras={availableCameras}
2024-02-22 21:19:50 +00:00
callMode={activeCall.callMode}
conversation={conversation}
groupMembers={groupMembers}
2020-08-27 00:03:42 +00:00
hasLocalAudio={hasLocalAudio}
hasLocalVideo={hasLocalVideo}
i18n={i18n}
isAdhocAdminApprovalRequired={isAdhocAdminApprovalRequired}
2024-02-22 21:19:50 +00:00
isAdhocJoinRequestPending={isAdhocJoinRequestPending}
isCallFull={isCallFull}
isConversationTooBigToRing={isConvoTooBigToRing}
getIsSharingPhoneNumberWithEverybody={
getIsSharingPhoneNumberWithEverybody
}
me={me}
2020-11-13 19:57:55 +00:00
onCallCanceled={cancelActiveCall}
onJoinCall={joinActiveCall}
outgoingRing={outgoingRing}
2020-12-02 18:14:03 +00:00
peekedParticipants={peekedParticipants}
2020-08-27 00:03:42 +00:00
setLocalPreview={setLocalPreview}
setLocalAudio={setLocalAudio}
setLocalVideo={setLocalVideo}
setOutgoingRing={setOutgoingRing}
2020-11-20 19:39:50 +00:00
showParticipantsList={showParticipantsList}
2020-11-13 19:57:55 +00:00
toggleParticipants={toggleParticipants}
togglePip={togglePip}
2020-08-27 00:03:42 +00:00
toggleSettings={toggleSettings}
/>
{settingsDialogOpen && renderDeviceSelection()}
2024-02-22 21:19:50 +00:00
{showParticipantsList &&
(activeCall.callMode === CallMode.Adhoc && callLink ? (
<CallingAdhocCallInfo
callLink={callLink}
i18n={i18n}
isCallLinkAdmin={isCallLinkAdmin}
isUnknownContactDiscrete={false}
2024-02-22 21:19:50 +00:00
ourServiceId={me.serviceId}
participants={peekedParticipants}
onClose={toggleParticipants}
onCopyCallLink={onCopyCallLink}
onShareCallLinkViaSignal={handleShareCallLinkViaSignal}
removeClient={removeClient}
2024-06-29 00:13:20 +00:00
blockClient={blockClient}
showContactModal={showContactModal}
2024-02-22 21:19:50 +00:00
/>
) : (
<CallingParticipantsList
conversationId={conversation.id}
2024-02-22 21:19:50 +00:00
i18n={i18n}
onClose={toggleParticipants}
ourServiceId={me.serviceId}
participants={peekedParticipants}
showContactModal={showContactModal}
2024-02-22 21:19:50 +00:00
/>
))}
2020-08-27 00:03:42 +00:00
</>
2020-06-04 18:16:19 +00:00
);
}
let isHandRaised = false;
2024-02-22 21:19:50 +00:00
if (isGroupOrAdhocActiveCall(activeCall)) {
const { raisedHands, localDemuxId } = activeCall;
if (localDemuxId) {
isHandRaised = raisedHands.has(localDemuxId);
}
}
2024-02-22 21:19:50 +00:00
const groupCallParticipantsForParticipantsList = isGroupOrAdhocActiveCall(
activeCall
)
? [
...activeCall.remoteParticipants,
{
...me,
hasRemoteAudio: hasLocalAudio,
hasRemoteVideo: hasLocalVideo,
isHandRaised,
presenting: Boolean(activeCall.presentingSource),
demuxId: activeCall.localDemuxId,
2024-02-22 21:19:50 +00:00
},
]
: [];
2020-12-02 18:14:03 +00:00
2020-11-13 19:57:55 +00:00
return (
<>
<CallScreen
activeCall={activeCall}
approveUser={approveUser}
batchUserAction={batchUserAction}
2024-09-20 01:03:44 +00:00
cancelPresenting={cancelPresenting}
changeCallView={changeCallView}
denyUser={denyUser}
getPresentingSources={getPresentingSources}
2020-11-13 19:57:55 +00:00
getGroupCallVideoFrameSource={getGroupCallVideoFrameSourceForActiveCall}
groupMembers={groupMembers}
hangUpActiveCall={hangUpActiveCall}
2020-11-13 19:57:55 +00:00
i18n={i18n}
imageDataCache={imageDataCache}
isCallLinkAdmin={isCallLinkAdmin}
2024-06-11 23:45:28 +00:00
isGroupCallRaiseHandEnabled={isGroupCallRaiseHandEnabled}
2020-11-13 19:57:55 +00:00
me={me}
openSystemPreferencesAction={openSystemPreferencesAction}
2023-11-16 19:55:35 +00:00
renderEmojiPicker={renderEmojiPicker}
renderReactionPicker={renderReactionPicker}
2023-12-06 21:52:29 +00:00
sendGroupCallRaiseHand={sendGroupCallRaiseHand}
2023-11-16 19:55:35 +00:00
sendGroupCallReaction={sendGroupCallReaction}
setGroupCallVideoRequest={setGroupCallVideoRequestForConversation}
2020-11-13 19:57:55 +00:00
setLocalPreview={setLocalPreview}
setRendererCanvas={setRendererCanvas}
setLocalAudio={setLocalAudio}
setLocalVideo={setLocalVideo}
2020-11-17 15:07:53 +00:00
stickyControls={showParticipantsList}
switchToPresentationView={switchToPresentationView}
switchFromPresentationView={switchFromPresentationView}
toggleCallLinkPendingParticipantModal={
toggleCallLinkPendingParticipantModal
}
toggleScreenRecordingPermissionsDialog={
toggleScreenRecordingPermissionsDialog
}
2020-11-17 15:07:53 +00:00
toggleParticipants={toggleParticipants}
2020-11-13 19:57:55 +00:00
togglePip={togglePip}
toggleSettings={toggleSettings}
/>
{presentingSourcesAvailable && presentingSourcesAvailable.length ? (
<CallingSelectPresentingSourcesModal
i18n={i18n}
presentingSourcesAvailable={presentingSourcesAvailable}
2024-09-20 01:03:44 +00:00
selectPresentingSource={selectPresentingSource}
cancelPresenting={cancelPresenting}
/>
) : null}
2020-11-13 19:57:55 +00:00
{settingsDialogOpen && renderDeviceSelection()}
2024-02-22 21:19:50 +00:00
{showParticipantsList &&
(activeCall.callMode === CallMode.Adhoc && callLink ? (
<CallingAdhocCallInfo
callLink={callLink}
i18n={i18n}
isCallLinkAdmin={isCallLinkAdmin}
isUnknownContactDiscrete
2024-02-22 21:19:50 +00:00
ourServiceId={me.serviceId}
participants={groupCallParticipantsForParticipantsList}
onClose={toggleParticipants}
onCopyCallLink={onCopyCallLink}
onShareCallLinkViaSignal={handleShareCallLinkViaSignal}
removeClient={removeClient}
2024-06-29 00:13:20 +00:00
blockClient={blockClient}
showContactModal={showContactModal}
2024-02-22 21:19:50 +00:00
/>
) : (
<CallingParticipantsList
conversationId={conversation.id}
2024-02-22 21:19:50 +00:00
i18n={i18n}
onClose={toggleParticipants}
ourServiceId={me.serviceId}
participants={groupCallParticipantsForParticipantsList}
showContactModal={showContactModal}
2024-02-22 21:19:50 +00:00
/>
))}
2020-11-13 19:57:55 +00:00
</>
);
2022-11-18 00:45:19 +00:00
}
2020-11-13 19:57:55 +00:00
export function CallManager({
acceptCall,
activeCall,
approveUser,
availableCameras,
batchUserAction,
2024-06-29 00:13:20 +00:00
blockClient,
bounceAppIconStart,
bounceAppIconStop,
callLink,
cancelCall,
2024-09-20 01:03:44 +00:00
cancelPresenting,
changeCallView,
closeNeedPermissionScreen,
declineCall,
denyUser,
getGroupCallVideoFrameSource,
getPresentingSources,
hangUpActiveCall,
hasInitialLoadCompleted,
i18n,
incomingCall,
isConversationTooBigToRing,
2024-06-11 23:45:28 +00:00
isGroupCallRaiseHandEnabled,
getIsSharingPhoneNumberWithEverybody,
me,
notifyForCall,
openSystemPreferencesAction,
pauseVoiceNotePlayer,
playRingtone,
removeClient,
renderDeviceSelection,
renderEmojiPicker,
renderReactionPicker,
2024-09-20 01:03:44 +00:00
selectPresentingSource,
sendGroupCallRaiseHand,
sendGroupCallReaction,
setGroupCallVideoRequest,
setIsCallActive,
setLocalAudio,
setLocalPreview,
setLocalVideo,
setOutgoingRing,
setRendererCanvas,
showContactModal,
showShareCallLinkViaSignal,
startCall,
stopRingtone,
switchFromPresentationView,
switchToPresentationView,
toggleParticipants,
togglePip,
toggleCallLinkPendingParticipantModal,
toggleScreenRecordingPermissionsDialog,
toggleSettings,
}: PropsType): JSX.Element | null {
const isCallActive = Boolean(activeCall);
useEffect(() => {
setIsCallActive(isCallActive);
}, [isCallActive, setIsCallActive]);
const shouldRing = getShouldRing({
activeCall,
incomingCall,
isConversationTooBigToRing,
hasInitialLoadCompleted,
});
2021-08-20 16:06:15 +00:00
useEffect(() => {
if (shouldRing) {
log.info('CallManager: Playing ringtone');
2021-08-20 16:06:15 +00:00
playRingtone();
return () => {
log.info('CallManager: Stopping ringtone');
2021-08-20 16:06:15 +00:00
stopRingtone();
};
}
stopRingtone();
return noop;
}, [shouldRing, playRingtone, stopRingtone]);
2020-11-13 19:57:55 +00:00
2021-09-09 21:15:05 +00:00
const mightBeRingingOutgoingGroupCall =
2024-02-22 21:19:50 +00:00
isGroupOrAdhocActiveCall(activeCall) &&
2021-09-09 21:15:05 +00:00
activeCall.outgoingRing &&
activeCall.joinState !== GroupCallJoinState.NotJoined;
useEffect(() => {
2021-09-09 21:15:05 +00:00
if (!mightBeRingingOutgoingGroupCall) {
return noop;
}
const timeout = setTimeout(() => {
setOutgoingRing(false);
}, GROUP_CALL_RING_DURATION);
return () => {
clearTimeout(timeout);
};
2021-09-09 21:15:05 +00:00
}, [mightBeRingingOutgoingGroupCall, setOutgoingRing]);
2020-11-13 19:57:55 +00:00
if (activeCall) {
// `props` should logically have an `activeCall` at this point, but TypeScript can't
// figure that out, so we pass it in again.
return (
<CallingToastProvider i18n={i18n}>
<ActiveCallManager
activeCall={activeCall}
availableCameras={availableCameras}
approveUser={approveUser}
batchUserAction={batchUserAction}
2024-06-29 00:13:20 +00:00
blockClient={blockClient}
callLink={callLink}
cancelCall={cancelCall}
2024-09-20 01:03:44 +00:00
cancelPresenting={cancelPresenting}
changeCallView={changeCallView}
closeNeedPermissionScreen={closeNeedPermissionScreen}
denyUser={denyUser}
getGroupCallVideoFrameSource={getGroupCallVideoFrameSource}
getPresentingSources={getPresentingSources}
hangUpActiveCall={hangUpActiveCall}
i18n={i18n}
2024-06-11 23:45:28 +00:00
isGroupCallRaiseHandEnabled={isGroupCallRaiseHandEnabled}
getIsSharingPhoneNumberWithEverybody={
getIsSharingPhoneNumberWithEverybody
}
me={me}
openSystemPreferencesAction={openSystemPreferencesAction}
pauseVoiceNotePlayer={pauseVoiceNotePlayer}
removeClient={removeClient}
renderDeviceSelection={renderDeviceSelection}
renderEmojiPicker={renderEmojiPicker}
renderReactionPicker={renderReactionPicker}
2024-09-20 01:03:44 +00:00
selectPresentingSource={selectPresentingSource}
sendGroupCallRaiseHand={sendGroupCallRaiseHand}
sendGroupCallReaction={sendGroupCallReaction}
setGroupCallVideoRequest={setGroupCallVideoRequest}
setLocalAudio={setLocalAudio}
setLocalPreview={setLocalPreview}
setLocalVideo={setLocalVideo}
setOutgoingRing={setOutgoingRing}
setRendererCanvas={setRendererCanvas}
showContactModal={showContactModal}
showShareCallLinkViaSignal={showShareCallLinkViaSignal}
startCall={startCall}
switchFromPresentationView={switchFromPresentationView}
switchToPresentationView={switchToPresentationView}
toggleCallLinkPendingParticipantModal={
toggleCallLinkPendingParticipantModal
}
toggleParticipants={toggleParticipants}
togglePip={togglePip}
toggleScreenRecordingPermissionsDialog={
toggleScreenRecordingPermissionsDialog
}
toggleSettings={toggleSettings}
/>
</CallingToastProvider>
);
2020-11-13 19:57:55 +00:00
}
// In the future, we may want to show the incoming call bar when a call is active.
if (incomingCall) {
2020-06-04 18:16:19 +00:00
return (
<IncomingCallBar
acceptCall={acceptCall}
2021-08-20 16:06:15 +00:00
bounceAppIconStart={bounceAppIconStart}
bounceAppIconStop={bounceAppIconStop}
2020-06-04 18:16:19 +00:00
declineCall={declineCall}
i18n={i18n}
2021-08-20 16:06:15 +00:00
notifyForCall={notifyForCall}
{...incomingCall}
2020-06-04 18:16:19 +00:00
/>
);
}
return null;
2022-11-18 00:45:19 +00:00
}
2021-08-20 16:06:15 +00:00
function isRinging(callState: CallState | undefined): boolean {
return callState === CallState.Prering || callState === CallState.Ringing;
}
function isConnected(connectionState: GroupCallConnectionState): boolean {
return (
connectionState === GroupCallConnectionState.Connecting ||
connectionState === GroupCallConnectionState.Connected
);
}
function isJoined(joinState: GroupCallJoinState): boolean {
return joinState !== GroupCallJoinState.NotJoined;
}
function hasRemoteParticipants(
remoteParticipants: Array<GroupCallParticipantInfoType>
): boolean {
return remoteParticipants.length > 0;
}
2024-02-22 21:19:50 +00:00
function isLonelyGroup(conversation: CallingConversationType): boolean {
return (conversation.sortedGroupMembers?.length ?? 0) < 2;
}
2021-08-20 16:06:15 +00:00
function getShouldRing({
activeCall,
incomingCall,
isConversationTooBigToRing,
hasInitialLoadCompleted,
}: Readonly<
Pick<
PropsType,
| 'activeCall'
| 'incomingCall'
| 'isConversationTooBigToRing'
| 'hasInitialLoadCompleted'
>
>): boolean {
if (!hasInitialLoadCompleted) {
return false;
}
if (incomingCall != null) {
// don't ring a large group
if (isConversationTooBigToRing) {
return false;
}
if (activeCall != null) {
return false;
}
if (incomingCall.callMode === CallMode.Direct) {
return (
isRinging(incomingCall.callState) &&
incomingCall.callEndedReason == null
);
}
if (incomingCall.callMode === CallMode.Group) {
return (
!isConnected(incomingCall.connectionState) &&
!isJoined(incomingCall.joinState) &&
!isLonelyGroup(incomingCall.conversation)
);
}
2021-08-20 16:06:15 +00:00
2024-02-22 21:19:50 +00:00
// Adhoc calls can't be incoming.
throw missingCaseError(incomingCall);
2021-08-20 16:06:15 +00:00
}
if (activeCall != null) {
2024-02-22 21:19:50 +00:00
switch (activeCall.callMode) {
case CallMode.Direct:
return (
activeCall.callState === CallState.Prering ||
activeCall.callState === CallState.Ringing
);
case CallMode.Group:
case CallMode.Adhoc:
return (
activeCall.outgoingRing &&
isConnected(activeCall.connectionState) &&
isJoined(activeCall.joinState) &&
!hasRemoteParticipants(activeCall.remoteParticipants) &&
!isLonelyGroup(activeCall.conversation)
);
default:
throw missingCaseError(activeCall);
}
2021-08-20 16:06:15 +00:00
}
return false;
2021-08-20 16:06:15 +00:00
}