signal-desktop/ts/components/CallingLobby.tsx

385 lines
12 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
import React from 'react';
2021-11-12 19:34:02 +00:00
import FocusTrap from 'focus-trap-react';
import classNames from 'classnames';
import type {
2020-10-08 01:25:33 +00:00
SetLocalAudioType,
SetLocalPreviewType,
SetLocalVideoType,
} from '../state/ducks/calling';
2020-11-19 18:11:35 +00:00
import { CallingButton, CallingButtonType } from './CallingButton';
import { TooltipPlacement } from './Tooltip';
2020-10-08 01:25:33 +00:00
import { CallBackgroundBlur } from './CallBackgroundBlur';
2023-10-25 13:40:22 +00:00
import { CallParticipantCount } from './CallParticipantCount';
2020-11-17 15:07:53 +00:00
import { CallingHeader } from './CallingHeader';
import { CallingPreCallInfo, RingMode } from './CallingPreCallInfo';
import {
CallingLobbyJoinButton,
CallingLobbyJoinButtonVariant,
} from './CallingLobbyJoinButton';
2024-02-22 21:19:50 +00:00
import { CallMode } from '../types/Calling';
import type { CallingConversationType } from '../types/Calling';
import type { LocalizerType } from '../types/Util';
2021-10-19 13:53:11 +00:00
import { useIsOnline } from '../hooks/useIsOnline';
2021-09-29 21:20:52 +00:00
import * as KeyboardLayout from '../services/keyboardLayout';
import type { ConversationType } from '../state/ducks/conversations';
import { useCallingToasts } from './CallingToast';
import { CallingButtonToastsContainer } from './CallingToastManager';
2024-02-22 21:19:50 +00:00
import { isGroupOrAdhocCallMode } from '../util/isGroupOrAdhocCall';
import { isSharingPhoneNumberWithEverybody } from '../util/phoneNumberSharingMode';
2020-10-08 01:25:33 +00:00
export type PropsType = {
availableCameras: Array<MediaDeviceInfo>;
2024-02-22 21:19:50 +00:00
callMode: CallMode;
conversation: Pick<
2024-02-22 21:19:50 +00:00
CallingConversationType,
| 'acceptedMessageRequest'
| 'avatarPath'
| 'color'
| 'isMe'
2021-09-02 22:34:38 +00:00
| 'memberships'
| 'name'
| 'phoneNumber'
| 'profileName'
| 'sharedGroupNames'
| 'systemGivenName'
| 'systemNickname'
| 'title'
| 'type'
| 'unblurredAvatarPath'
>;
groupMembers?: Array<
Pick<
ConversationType,
'id' | 'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
>
>;
2020-10-08 01:25:33 +00:00
hasLocalAudio: boolean;
hasLocalVideo: boolean;
i18n: LocalizerType;
2024-02-22 21:19:50 +00:00
isAdhocJoinRequestPending: boolean;
isConversationTooBigToRing: boolean;
isCallFull?: boolean;
2023-08-16 20:54:39 +00:00
me: Readonly<
Pick<ConversationType, 'avatarPath' | 'color' | 'id' | 'serviceId'>
>;
2020-10-08 01:25:33 +00:00
onCallCanceled: () => void;
onJoinCall: () => void;
outgoingRing: boolean;
peekedParticipants: Array<ConversationType>;
2020-10-08 01:25:33 +00:00
setLocalAudio: (_: SetLocalAudioType) => void;
setLocalVideo: (_: SetLocalVideoType) => void;
setLocalPreview: (_: SetLocalPreviewType) => void;
setOutgoingRing: (_: boolean) => void;
2020-11-20 19:39:50 +00:00
showParticipantsList: boolean;
2020-10-08 01:25:33 +00:00
toggleParticipants: () => void;
toggleSettings: () => void;
};
2022-11-18 00:45:19 +00:00
export function CallingLobby({
availableCameras,
2024-02-22 21:19:50 +00:00
callMode,
conversation,
groupMembers,
2020-10-08 01:25:33 +00:00
hasLocalAudio,
hasLocalVideo,
i18n,
2024-02-22 21:19:50 +00:00
isAdhocJoinRequestPending,
isCallFull = false,
isConversationTooBigToRing,
me,
2020-10-08 01:25:33 +00:00
onCallCanceled,
onJoinCall,
2020-12-02 18:14:03 +00:00
peekedParticipants,
2020-10-08 01:25:33 +00:00
setLocalAudio,
setLocalPreview,
setLocalVideo,
setOutgoingRing,
2020-10-08 01:25:33 +00:00
toggleParticipants,
toggleSettings,
outgoingRing,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
const localVideoRef = React.useRef<null | HTMLVideoElement>(null);
2020-10-08 01:25:33 +00:00
const shouldShowLocalVideo = hasLocalVideo && availableCameras.length > 0;
2024-02-22 21:19:50 +00:00
const isGroupOrAdhocCall = isGroupOrAdhocCallMode(callMode);
2020-10-08 01:25:33 +00:00
const toggleAudio = React.useCallback((): void => {
setLocalAudio({ enabled: !hasLocalAudio });
}, [hasLocalAudio, setLocalAudio]);
2020-10-08 01:25:33 +00:00
const toggleVideo = React.useCallback((): void => {
setLocalVideo({ enabled: !hasLocalVideo });
}, [hasLocalVideo, setLocalVideo]);
2020-10-08 01:25:33 +00:00
const toggleOutgoingRing = React.useCallback((): void => {
setOutgoingRing(!outgoingRing);
}, [outgoingRing, setOutgoingRing]);
2020-10-08 01:25:33 +00:00
React.useEffect(() => {
setLocalPreview({ element: localVideoRef });
return () => {
setLocalPreview({ element: undefined });
};
}, [setLocalPreview]);
React.useEffect(() => {
function handleKeyDown(event: KeyboardEvent): void {
let eventHandled = false;
2021-09-29 21:20:52 +00:00
const key = KeyboardLayout.lookup(event);
if (event.shiftKey && (key === 'V' || key === 'v')) {
2020-10-08 01:25:33 +00:00
toggleVideo();
eventHandled = true;
2021-09-29 21:20:52 +00:00
} else if (event.shiftKey && (key === 'M' || key === 'm')) {
2020-10-08 01:25:33 +00:00
toggleAudio();
eventHandled = true;
}
if (eventHandled) {
event.preventDefault();
event.stopPropagation();
}
}
document.addEventListener('keydown', handleKeyDown);
return () => {
document.removeEventListener('keydown', handleKeyDown);
};
}, [toggleVideo, toggleAudio]);
2021-10-19 13:53:11 +00:00
const isOnline = useIsOnline();
2020-11-17 15:07:53 +00:00
const [isCallConnecting, setIsCallConnecting] = React.useState(false);
// eslint-disable-next-line no-nested-ternary
2020-10-08 01:25:33 +00:00
const videoButtonType = hasLocalVideo
? CallingButtonType.VIDEO_ON
: availableCameras.length === 0
? CallingButtonType.VIDEO_DISABLED
2020-10-08 01:25:33 +00:00
: CallingButtonType.VIDEO_OFF;
2020-10-08 01:25:33 +00:00
const audioButtonType = hasLocalAudio
? CallingButtonType.AUDIO_ON
: CallingButtonType.AUDIO_OFF;
const isRingButtonVisible: boolean =
2024-02-22 21:19:50 +00:00
isGroupOrAdhocCall &&
peekedParticipants.length === 0 &&
(groupMembers || []).length > 1;
2021-09-02 22:34:38 +00:00
let preCallInfoRingMode: RingMode;
2024-02-22 21:19:50 +00:00
if (isGroupOrAdhocCall) {
2021-09-02 22:34:38 +00:00
preCallInfoRingMode =
outgoingRing && !isConversationTooBigToRing
2021-09-02 22:34:38 +00:00
? RingMode.WillRing
: RingMode.WillNotRing;
} else {
preCallInfoRingMode = RingMode.WillRing;
}
let ringButtonType:
| CallingButtonType.RING_DISABLED
| CallingButtonType.RING_ON
| CallingButtonType.RING_OFF;
if (isRingButtonVisible) {
if (isConversationTooBigToRing) {
ringButtonType = CallingButtonType.RING_DISABLED;
} else if (outgoingRing) {
ringButtonType = CallingButtonType.RING_ON;
} else {
ringButtonType = CallingButtonType.RING_OFF;
}
} else {
ringButtonType = CallingButtonType.RING_DISABLED;
}
2021-10-19 13:53:11 +00:00
const canJoin = !isCallFull && !isCallConnecting && isOnline;
let callingLobbyJoinButtonVariant: CallingLobbyJoinButtonVariant;
if (isCallFull) {
callingLobbyJoinButtonVariant = CallingLobbyJoinButtonVariant.CallIsFull;
} else if (isCallConnecting) {
callingLobbyJoinButtonVariant = CallingLobbyJoinButtonVariant.Loading;
} else if (peekedParticipants.length) {
callingLobbyJoinButtonVariant = CallingLobbyJoinButtonVariant.Join;
} else {
callingLobbyJoinButtonVariant = CallingLobbyJoinButtonVariant.Start;
}
2023-10-25 13:40:22 +00:00
const callStatus = React.useMemo(() => {
2024-02-22 21:19:50 +00:00
if (isGroupOrAdhocCall) {
2023-10-25 13:40:22 +00:00
return (
<CallParticipantCount
2024-02-22 21:19:50 +00:00
callMode={callMode}
2023-10-25 13:40:22 +00:00
i18n={i18n}
2024-02-22 21:19:50 +00:00
isAdhocJoinRequestPending={isAdhocJoinRequestPending}
2023-10-25 13:40:22 +00:00
groupMemberCount={groupMembers?.length ?? 0}
participantCount={peekedParticipants.length}
toggleParticipants={toggleParticipants}
/>
);
}
if (hasLocalVideo) {
return i18n('icu:ContactListItem__menu__video-call');
}
if (hasLocalAudio) {
return i18n('icu:CallControls__InfoDisplay--audio-call');
}
return null;
}, [
2024-02-22 21:19:50 +00:00
callMode,
isAdhocJoinRequestPending,
isGroupOrAdhocCall,
2023-10-25 13:40:22 +00:00
peekedParticipants.length,
i18n,
hasLocalVideo,
hasLocalAudio,
groupMembers?.length,
toggleParticipants,
]);
useWasInitiallyMutedToast(hasLocalAudio, i18n);
2020-10-08 01:25:33 +00:00
return (
2021-11-12 19:34:02 +00:00
<FocusTrap>
<div className="module-calling__container dark-theme">
2021-11-12 19:34:02 +00:00
{shouldShowLocalVideo ? (
<video
className="module-CallingLobby__local-preview module-CallingLobby__local-preview--camera-is-on"
ref={localVideoRef}
autoPlay
/>
) : (
<CallBackgroundBlur
className="module-CallingLobby__local-preview module-CallingLobby__local-preview--camera-is-off"
avatarPath={me.avatarPath}
/>
)}
2020-11-17 15:07:53 +00:00
2021-11-12 19:34:02 +00:00
<CallingHeader
i18n={i18n}
2024-02-22 21:19:50 +00:00
isGroupCall={isGroupOrAdhocCall}
2021-11-12 19:34:02 +00:00
participantCount={peekedParticipants.length}
toggleSettings={toggleSettings}
onCancel={onCallCanceled}
/>
2021-11-12 19:34:02 +00:00
2023-10-31 19:32:56 +00:00
<div className="module-calling__spacer module-CallingPreCallInfo-spacer" />
2021-11-12 19:34:02 +00:00
<CallingPreCallInfo
conversation={conversation}
groupMembers={groupMembers}
i18n={i18n}
2021-11-12 19:34:02 +00:00
isCallFull={isCallFull}
me={me}
peekedParticipants={peekedParticipants}
ringMode={preCallInfoRingMode}
/>
2021-11-12 19:34:02 +00:00
<div
className={classNames(
2023-10-31 19:32:56 +00:00
'module-calling__camera-is-off module-CallingLobby__camera-is-off',
2021-11-12 19:34:02 +00:00
`module-CallingLobby__camera-is-off--${
shouldShowLocalVideo ? 'invisible' : 'visible'
}`
)}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:calling__your-video-is-off')}
2021-11-12 19:34:02 +00:00
</div>
2024-02-22 21:19:50 +00:00
{callMode === CallMode.Adhoc && (
<div className="CallingLobby__CallLinkNotice">
{isSharingPhoneNumberWithEverybody()
? i18n('icu:CallingLobby__CallLinkNotice--phone-sharing')
: i18n('icu:CallingLobby__CallLinkNotice')}
</div>
)}
<CallingButtonToastsContainer
hasLocalAudio={hasLocalAudio}
outgoingRing={outgoingRing}
i18n={i18n}
/>
<div className="CallingLobby__Footer">
<div className="module-calling__spacer CallControls__OuterSpacer" />
<div className="CallControls">
<div className="CallControls__InfoDisplay">
<div className="CallControls__CallTitle">
{conversation.title}
</div>
<div className="CallControls__Status">{callStatus}</div>
</div>
<div className="CallControls__ButtonContainer">
<CallingButton
buttonType={videoButtonType}
i18n={i18n}
onClick={toggleVideo}
tooltipDirection={TooltipPlacement.Top}
/>
<CallingButton
buttonType={audioButtonType}
i18n={i18n}
onClick={toggleAudio}
tooltipDirection={TooltipPlacement.Top}
/>
<CallingButton
buttonType={ringButtonType}
i18n={i18n}
isVisible={isRingButtonVisible}
onClick={toggleOutgoingRing}
tooltipDirection={TooltipPlacement.Top}
/>
</div>
<div className="CallControls__JoinLeaveButtonContainer">
<CallingLobbyJoinButton
disabled={!canJoin}
i18n={i18n}
onClick={() => {
setIsCallConnecting(true);
onJoinCall();
}}
variant={callingLobbyJoinButtonVariant}
/>
</div>
2023-10-25 13:40:22 +00:00
</div>
<div className="module-calling__spacer CallControls__OuterSpacer" />
2021-11-12 19:34:02 +00:00
</div>
2020-10-08 01:25:33 +00:00
</div>
2021-11-12 19:34:02 +00:00
</FocusTrap>
2020-10-08 01:25:33 +00:00
);
2022-11-18 00:45:19 +00:00
}
function useWasInitiallyMutedToast(
hasLocalAudio: boolean,
i18n: LocalizerType
) {
const [wasInitiallyMuted] = React.useState(!hasLocalAudio);
const { showToast, hideToast } = useCallingToasts();
const INITIALLY_MUTED_KEY = 'initially-muted-group-size';
React.useEffect(() => {
if (wasInitiallyMuted) {
showToast({
key: INITIALLY_MUTED_KEY,
content: i18n(
'icu:calling__lobby-automatically-muted-because-there-are-a-lot-of-people'
),
autoClose: true,
dismissable: true,
onlyShowOnce: true,
});
}
}, [wasInitiallyMuted, i18n, showToast]);
// Hide this toast if the user unmutes
React.useEffect(() => {
if (wasInitiallyMuted && hasLocalAudio) {
hideToast(INITIALLY_MUTED_KEY);
}
}, [hideToast, wasInitiallyMuted, hasLocalAudio]);
}