signal-desktop/ts/components/IncomingCallBar.tsx

339 lines
8.7 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 type { ReactChild } from 'react';
2022-05-10 18:14:08 +00:00
import React, { useCallback, useEffect, useRef } from 'react';
2022-12-09 20:37:45 +00:00
import { Avatar, AvatarSize } from './Avatar';
import { Tooltip } from './Tooltip';
2024-05-15 21:48:02 +00:00
import { I18n } from './I18n';
import { Theme } from '../util/theme';
2021-08-20 16:06:15 +00:00
import { getParticipantName } from '../util/callingGetParticipantName';
2020-06-04 18:16:19 +00:00
import { ContactName } from './conversation/ContactName';
import type { LocalizerType } from '../types/Util';
2021-08-06 00:17:05 +00:00
import { AvatarColors } from '../types/Colors';
2021-08-20 16:06:15 +00:00
import { CallMode } from '../types/Calling';
import type { ConversationType } from '../state/ducks/conversations';
import type { AcceptCallType, DeclineCallType } from '../state/ducks/calling';
2021-08-20 16:06:15 +00:00
import { missingCaseError } from '../util/missingCaseError';
2022-05-10 18:14:08 +00:00
import {
useIncomingCallShortcuts,
useKeyboardShortcuts,
} from '../hooks/useKeyboardShortcuts';
2023-04-20 17:03:43 +00:00
import { UserText } from './UserText';
2020-06-04 18:16:19 +00:00
export type PropsType = {
acceptCall: (_: AcceptCallType) => void;
declineCall: (_: DeclineCallType) => void;
i18n: LocalizerType;
2021-05-07 22:21:10 +00:00
conversation: Pick<
ConversationType,
| 'acceptedMessageRequest'
2024-07-11 19:44:09 +00:00
| 'avatarUrl'
2021-05-07 22:21:10 +00:00
| 'color'
| 'id'
| 'isMe'
| 'name'
| 'phoneNumber'
| 'profileName'
| 'sharedGroupNames'
| 'title'
2021-08-20 16:06:15 +00:00
| 'type'
2021-05-07 22:21:10 +00:00
>;
2021-08-20 16:06:15 +00:00
bounceAppIconStart(): unknown;
bounceAppIconStop(): unknown;
2023-08-01 16:06:29 +00:00
notifyForCall(
conversationId: string,
conversationTitle: string,
isVideoCall: boolean
): unknown;
2021-08-20 16:06:15 +00:00
} & (
| {
callMode: CallMode.Direct;
isVideoCall: boolean;
}
| {
callMode: CallMode.Group;
otherMembersRung: Array<
Pick<
ConversationType,
'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
>
>;
ringer: Pick<
ConversationType,
'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
>;
2021-08-20 16:06:15 +00:00
}
);
2020-06-04 18:16:19 +00:00
type CallButtonProps = {
classSuffix: string;
tabIndex: number;
tooltipContent: string;
onClick: () => void;
};
2022-11-18 00:45:19 +00:00
function CallButton({
2020-06-04 18:16:19 +00:00
classSuffix,
onClick,
tabIndex,
tooltipContent,
2022-11-18 00:45:19 +00:00
}: CallButtonProps): JSX.Element {
2020-06-04 18:16:19 +00:00
return (
<Tooltip
content={tooltipContent}
theme={Theme.Dark}
wrapperClassName="IncomingCallBar__button__container"
>
2020-11-19 18:11:35 +00:00
<button
aria-label={tooltipContent}
2021-08-05 23:31:34 +00:00
className={`IncomingCallBar__button IncomingCallBar__button--${classSuffix}`}
2020-11-19 18:11:35 +00:00
onClick={onClick}
tabIndex={tabIndex}
type="button"
2020-06-04 18:16:19 +00:00
>
<div />
2020-11-19 18:11:35 +00:00
</button>
</Tooltip>
2020-06-04 18:16:19 +00:00
);
2022-11-18 00:45:19 +00:00
}
2020-06-04 18:16:19 +00:00
2022-11-18 00:45:19 +00:00
function GroupCallMessage({
2020-06-04 18:16:19 +00:00
i18n,
2021-08-20 16:06:15 +00:00
otherMembersRung,
ringer,
}: Readonly<{
i18n: LocalizerType;
otherMembersRung: Array<
Pick<
ConversationType,
'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
>
>;
ringer: Pick<
ConversationType,
'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
>;
2022-11-18 00:45:19 +00:00
}>): JSX.Element {
2021-08-20 16:06:15 +00:00
// As an optimization, we only process the first two names.
const [first, second] = otherMembersRung
.slice(0, 2)
2023-04-20 17:03:43 +00:00
.map(member => <UserText text={getParticipantName(member)} />);
const ringerNode = <UserText text={getParticipantName(ringer)} />;
2021-08-20 16:06:15 +00:00
switch (otherMembersRung.length) {
case 0:
return (
2024-05-15 21:48:02 +00:00
<I18n
2023-03-30 00:03:25 +00:00
id="icu:incomingGroupCall__ringing-you"
2021-08-20 16:06:15 +00:00
i18n={i18n}
components={{ ringer: ringerNode }}
/>
);
case 1:
return (
2024-05-15 21:48:02 +00:00
<I18n
2023-03-30 00:03:25 +00:00
id="icu:incomingGroupCall__ringing-1-other"
2021-08-20 16:06:15 +00:00
i18n={i18n}
components={{
ringer: ringerNode,
otherMember: first,
}}
/>
);
case 2:
return (
2024-05-15 21:48:02 +00:00
<I18n
2023-03-30 00:03:25 +00:00
id="icu:incomingGroupCall__ringing-2-others"
2021-08-20 16:06:15 +00:00
i18n={i18n}
components={{
ringer: ringerNode,
first,
second,
}}
/>
);
case 3:
return (
2024-05-15 21:48:02 +00:00
<I18n
2023-03-30 00:03:25 +00:00
id="icu:incomingGroupCall__ringing-3-others"
2021-08-20 16:06:15 +00:00
i18n={i18n}
components={{
ringer: ringerNode,
first,
second,
}}
/>
);
default:
return (
2024-05-15 21:48:02 +00:00
<I18n
2023-03-30 00:03:25 +00:00
id="icu:incomingGroupCall__ringing-many"
2021-08-20 16:06:15 +00:00
i18n={i18n}
components={{
ringer: ringerNode,
first,
second,
2023-04-03 19:03:00 +00:00
remaining: otherMembersRung.length - 2,
2021-08-20 16:06:15 +00:00
}}
/>
);
}
2022-11-18 00:45:19 +00:00
}
2021-08-20 16:06:15 +00:00
2022-11-18 00:45:19 +00:00
export function IncomingCallBar(props: PropsType): JSX.Element | null {
2021-08-20 16:06:15 +00:00
const {
acceptCall,
bounceAppIconStart,
bounceAppIconStop,
conversation,
declineCall,
i18n,
notifyForCall,
} = props;
2020-06-04 18:16:19 +00:00
const {
id: conversationId,
2021-05-07 22:21:10 +00:00
acceptedMessageRequest,
2024-07-11 19:44:09 +00:00
avatarUrl,
2020-07-24 01:35:32 +00:00
color,
2021-05-07 22:21:10 +00:00
isMe,
2020-06-04 18:16:19 +00:00
phoneNumber,
profileName,
2021-05-07 22:21:10 +00:00
sharedGroupNames,
title,
2021-08-20 16:06:15 +00:00
type: conversationType,
} = conversation;
2020-06-04 18:16:19 +00:00
2021-08-20 16:06:15 +00:00
let isVideoCall: boolean;
let headerNode: ReactChild;
let messageNode: ReactChild;
switch (props.callMode) {
case CallMode.Direct:
({ isVideoCall } = props);
headerNode = <ContactName title={title} />;
messageNode = isVideoCall
2023-03-30 00:03:25 +00:00
? i18n('icu:incomingVideoCall')
: i18n('icu:incomingAudioCall');
2021-08-20 16:06:15 +00:00
break;
case CallMode.Group: {
const { otherMembersRung, ringer } = props;
isVideoCall = true;
2023-04-20 17:03:43 +00:00
headerNode = <UserText text={title} />;
2021-08-20 16:06:15 +00:00
messageNode = (
<GroupCallMessage
i18n={i18n}
otherMembersRung={otherMembersRung}
ringer={ringer}
/>
);
break;
}
default:
throw missingCaseError(props);
}
// We don't want to re-notify if the title changes.
const initialTitleRef = useRef<string>(title);
useEffect(() => {
const initialTitle = initialTitleRef.current;
2023-08-01 16:06:29 +00:00
notifyForCall(conversationId, initialTitle, isVideoCall);
}, [conversationId, isVideoCall, notifyForCall]);
2021-08-20 16:06:15 +00:00
useEffect(() => {
bounceAppIconStart();
return () => {
bounceAppIconStop();
};
}, [bounceAppIconStart, bounceAppIconStop]);
2022-05-10 18:14:08 +00:00
const acceptVideoCall = useCallback(() => {
2023-01-10 00:52:01 +00:00
if (isVideoCall) {
acceptCall({ conversationId, asVideoCall: true });
}
}, [isVideoCall, acceptCall, conversationId]);
2022-05-10 18:14:08 +00:00
const acceptAudioCall = useCallback(() => {
acceptCall({ conversationId, asVideoCall: false });
}, [acceptCall, conversationId]);
const declineIncomingCall = useCallback(() => {
declineCall({ conversationId });
}, [conversationId, declineCall]);
const incomingCallShortcuts = useIncomingCallShortcuts(
acceptAudioCall,
acceptVideoCall,
declineIncomingCall
);
useKeyboardShortcuts(incomingCallShortcuts);
2020-06-04 18:16:19 +00:00
return (
2021-08-05 23:31:34 +00:00
<div className="IncomingCallBar__container">
<div className="IncomingCallBar__bar">
2021-08-20 16:06:15 +00:00
<div className="IncomingCallBar__conversation">
<div className="IncomingCallBar__conversation--avatar">
2021-08-05 23:31:34 +00:00
<Avatar
acceptedMessageRequest={acceptedMessageRequest}
2024-07-11 19:44:09 +00:00
avatarUrl={avatarUrl}
badge={undefined}
2021-08-06 00:17:05 +00:00
color={color || AvatarColors[0]}
2021-08-05 23:31:34 +00:00
noteToSelf={false}
2021-08-20 16:06:15 +00:00
conversationType={conversationType}
2021-08-05 23:31:34 +00:00
i18n={i18n}
isMe={isMe}
2020-07-24 01:35:32 +00:00
phoneNumber={phoneNumber}
2020-06-04 18:16:19 +00:00
profileName={profileName}
2020-07-24 01:35:32 +00:00
title={title}
2021-08-05 23:31:34 +00:00
sharedGroupNames={sharedGroupNames}
2022-12-09 20:37:45 +00:00
size={AvatarSize.FORTY_EIGHT}
2020-06-04 18:16:19 +00:00
/>
</div>
2021-08-20 16:06:15 +00:00
<div className="IncomingCallBar__conversation--name">
<div className="IncomingCallBar__conversation--name-header">
{headerNode}
2021-08-05 23:31:34 +00:00
</div>
2021-08-20 16:06:15 +00:00
<div
dir="auto"
className="IncomingCallBar__conversation--message-text"
>
{messageNode}
2021-08-05 23:31:34 +00:00
</div>
2020-06-04 18:16:19 +00:00
</div>
</div>
2021-08-05 23:31:34 +00:00
<div className="IncomingCallBar__actions">
<CallButton
classSuffix="decline"
2022-05-10 18:14:08 +00:00
onClick={declineIncomingCall}
tabIndex={0}
2023-03-30 00:03:25 +00:00
tooltipContent={i18n('icu:declineCall')}
/>
2021-08-05 23:31:34 +00:00
{isVideoCall ? (
<>
<CallButton
classSuffix="accept-video-as-audio"
2022-05-10 18:14:08 +00:00
onClick={acceptAudioCall}
2021-08-05 23:31:34 +00:00
tabIndex={0}
2023-03-30 00:03:25 +00:00
tooltipContent={i18n('icu:acceptCallWithoutVideo')}
2021-08-05 23:31:34 +00:00
/>
<CallButton
classSuffix="accept-video"
2022-05-10 18:14:08 +00:00
onClick={acceptVideoCall}
2021-08-05 23:31:34 +00:00
tabIndex={0}
2023-03-30 00:03:25 +00:00
tooltipContent={i18n('icu:acceptCall')}
2021-08-05 23:31:34 +00:00
/>
</>
) : (
<CallButton
classSuffix="accept-audio"
2022-05-10 18:14:08 +00:00
onClick={acceptAudioCall}
tabIndex={0}
2023-03-30 00:03:25 +00:00
tooltipContent={i18n('icu:acceptCall')}
/>
2021-08-05 23:31:34 +00:00
)}
</div>
2020-06-04 18:16:19 +00:00
</div>
</div>
);
2022-11-18 00:45:19 +00:00
}