signal-desktop/ts/components/CallingPreCallInfo.tsx

211 lines
6.2 KiB
TypeScript
Raw Normal View History

// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React from 'react';
import type { ConversationType } from '../state/ducks/conversations';
2024-02-22 21:19:50 +00:00
import type { CallingConversationType } from '../types/Calling';
import type { LocalizerType } from '../types/Util';
import { Avatar, AvatarSize } from './Avatar';
2021-08-20 16:06:15 +00:00
import { getParticipantName } from '../util/callingGetParticipantName';
import { missingCaseError } from '../util/missingCaseError';
2023-04-20 17:03:43 +00:00
import { UserText } from './UserText';
export enum RingMode {
WillNotRing,
WillRing,
IsRinging,
}
export type PropsType = {
conversation: Pick<
2024-02-22 21:19:50 +00:00
CallingConversationType,
| 'acceptedMessageRequest'
| 'avatarPath'
| 'color'
| 'isMe'
| 'phoneNumber'
| 'profileName'
| 'sharedGroupNames'
| 'systemGivenName'
| 'systemNickname'
| 'title'
| 'type'
| 'unblurredAvatarPath'
>;
i18n: LocalizerType;
2023-08-16 20:54:39 +00:00
me: Pick<ConversationType, 'id' | 'serviceId'>;
ringMode: RingMode;
// The following should only be set for group conversations.
groupMembers?: Array<
Pick<
ConversationType,
'id' | 'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
>
>;
isCallFull?: boolean;
peekedParticipants?: Array<
Pick<
ConversationType,
'firstName' | 'systemGivenName' | 'systemNickname' | 'title' | 'serviceId'
>
>;
};
2022-11-18 00:45:19 +00:00
export function CallingPreCallInfo({
conversation,
groupMembers = [],
i18n,
isCallFull = false,
me,
peekedParticipants = [],
ringMode,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
let subtitle: string;
if (ringMode === RingMode.IsRinging) {
2023-03-30 00:03:25 +00:00
subtitle = i18n('icu:outgoingCallRinging');
} else if (isCallFull) {
2023-03-30 00:03:25 +00:00
subtitle = i18n('icu:calling__call-is-full');
} else if (peekedParticipants.length) {
// It should be rare to see yourself in this list, but it's possible if (1) you rejoin
// quickly, causing the server to return stale state (2) you have joined on another
// device.
let hasYou = false;
const participantNames = peekedParticipants.map(participant => {
2023-08-16 20:54:39 +00:00
if (participant.serviceId === me.serviceId) {
hasYou = true;
2023-03-30 00:03:25 +00:00
return i18n('icu:you');
}
return getParticipantName(participant);
});
switch (participantNames.length) {
case 1:
subtitle = hasYou
2023-03-30 00:03:25 +00:00
? i18n('icu:calling__pre-call-info--another-device-in-call')
: i18n('icu:calling__pre-call-info--1-person-in-call', {
2023-03-27 23:37:39 +00:00
first: participantNames[0],
});
break;
case 2:
2023-03-30 00:03:25 +00:00
subtitle = i18n('icu:calling__pre-call-info--2-people-in-call', {
first: participantNames[0],
second: participantNames[1],
});
break;
case 3:
2023-03-30 00:03:25 +00:00
subtitle = i18n('icu:calling__pre-call-info--3-people-in-call', {
first: participantNames[0],
second: participantNames[1],
third: participantNames[2],
});
break;
default:
2023-03-30 00:03:25 +00:00
subtitle = i18n('icu:calling__pre-call-info--many-people-in-call', {
first: participantNames[0],
second: participantNames[1],
others: String(participantNames.length - 2),
});
break;
}
} else {
let memberNames: Array<string>;
switch (conversation.type) {
case 'direct':
memberNames = [getParticipantName(conversation)];
break;
case 'group':
2024-02-22 21:19:50 +00:00
case 'callLink':
memberNames = groupMembers
.filter(member => member.id !== me.id)
.map(getParticipantName);
break;
default:
throw missingCaseError(conversation.type);
}
const ring = ringMode === RingMode.WillRing;
switch (memberNames.length) {
case 0:
2023-03-30 00:03:25 +00:00
subtitle = i18n('icu:calling__pre-call-info--empty-group');
break;
case 1: {
subtitle = ring
2023-03-30 00:03:25 +00:00
? i18n('icu:calling__pre-call-info--will-ring-1', {
2023-03-27 23:37:39 +00:00
person: memberNames[0],
})
2023-03-30 00:03:25 +00:00
: i18n('icu:calling__pre-call-info--will-notify-1', {
2023-03-27 23:37:39 +00:00
person: memberNames[0],
});
break;
}
case 2: {
subtitle = ring
2023-03-30 00:03:25 +00:00
? i18n('icu:calling__pre-call-info--will-ring-2', {
2023-03-27 23:37:39 +00:00
first: memberNames[0],
second: memberNames[1],
})
2023-03-30 00:03:25 +00:00
: i18n('icu:calling__pre-call-info--will-notify-2', {
2023-03-27 23:37:39 +00:00
first: memberNames[0],
second: memberNames[1],
});
break;
}
case 3: {
subtitle = ring
2023-03-30 00:03:25 +00:00
? i18n('icu:calling__pre-call-info--will-ring-3', {
2023-03-27 23:37:39 +00:00
first: memberNames[0],
second: memberNames[1],
third: memberNames[2],
})
2023-03-30 00:03:25 +00:00
: i18n('icu:calling__pre-call-info--will-notify-3', {
2023-03-27 23:37:39 +00:00
first: memberNames[0],
second: memberNames[1],
third: memberNames[2],
});
break;
}
default: {
subtitle = ring
2023-03-30 00:03:25 +00:00
? i18n('icu:calling__pre-call-info--will-ring-many', {
2023-03-27 23:37:39 +00:00
first: memberNames[0],
second: memberNames[1],
others: String(memberNames.length - 2),
})
2023-03-30 00:03:25 +00:00
: i18n('icu:calling__pre-call-info--will-notify-many', {
2023-03-27 23:37:39 +00:00
first: memberNames[0],
second: memberNames[1],
others: String(memberNames.length - 2),
});
break;
}
}
}
return (
<div className="module-CallingPreCallInfo">
<Avatar
avatarPath={conversation.avatarPath}
badge={undefined}
color={conversation.color}
acceptedMessageRequest={conversation.acceptedMessageRequest}
conversationType={conversation.type}
isMe={conversation.isMe}
noteToSelf={false}
phoneNumber={conversation.phoneNumber}
profileName={conversation.profileName}
sharedGroupNames={conversation.sharedGroupNames}
2023-10-31 19:32:56 +00:00
size={AvatarSize.NINETY_SIX}
title={conversation.title}
unblurredAvatarPath={conversation.unblurredAvatarPath}
i18n={i18n}
/>
<div className="module-CallingPreCallInfo__title">
2023-04-20 17:03:43 +00:00
<UserText text={conversation.title} />
</div>
<div className="module-CallingPreCallInfo__subtitle">{subtitle}</div>
</div>
);
2022-11-18 00:45:19 +00:00
}