2021-08-17 21:45:18 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import React from 'react';
|
2024-11-14 23:24:04 +00:00
|
|
|
import { partition } from 'lodash';
|
2021-08-17 21:45:18 +00:00
|
|
|
import type { ConversationType } from '../state/ducks/conversations';
|
2024-02-22 21:19:50 +00:00
|
|
|
import type { CallingConversationType } from '../types/Calling';
|
2021-08-17 21:45:18 +00:00
|
|
|
import type { LocalizerType } from '../types/Util';
|
|
|
|
import { Avatar, AvatarSize } from './Avatar';
|
2021-08-20 16:06:15 +00:00
|
|
|
import { getParticipantName } from '../util/callingGetParticipantName';
|
2021-08-17 21:45:18 +00:00
|
|
|
import { missingCaseError } from '../util/missingCaseError';
|
2023-04-20 17:03:43 +00:00
|
|
|
import { UserText } from './UserText';
|
2021-08-17 21:45:18 +00:00
|
|
|
|
2021-08-25 21:42:51 +00:00
|
|
|
export enum RingMode {
|
|
|
|
WillNotRing,
|
|
|
|
WillRing,
|
|
|
|
IsRinging,
|
|
|
|
}
|
|
|
|
|
2024-11-14 23:24:04 +00:00
|
|
|
type PeekedParticipantType = Pick<
|
|
|
|
ConversationType,
|
|
|
|
| 'firstName'
|
|
|
|
| 'systemGivenName'
|
|
|
|
| 'systemNickname'
|
|
|
|
| 'title'
|
|
|
|
| 'serviceId'
|
|
|
|
| 'titleNoDefault'
|
|
|
|
>;
|
|
|
|
|
2023-10-11 19:06:43 +00:00
|
|
|
export type PropsType = {
|
2021-08-17 21:45:18 +00:00
|
|
|
conversation: Pick<
|
2024-02-22 21:19:50 +00:00
|
|
|
CallingConversationType,
|
2021-08-17 21:45:18 +00:00
|
|
|
| 'acceptedMessageRequest'
|
2024-07-11 19:44:09 +00:00
|
|
|
| 'avatarUrl'
|
2021-08-17 21:45:18 +00:00
|
|
|
| 'color'
|
|
|
|
| 'isMe'
|
|
|
|
| 'phoneNumber'
|
|
|
|
| 'profileName'
|
|
|
|
| 'sharedGroupNames'
|
2024-02-08 15:28:34 +00:00
|
|
|
| 'systemGivenName'
|
|
|
|
| 'systemNickname'
|
2021-08-17 21:45:18 +00:00
|
|
|
| 'title'
|
|
|
|
| 'type'
|
2024-07-11 19:44:09 +00:00
|
|
|
| 'unblurredAvatarUrl'
|
2021-08-17 21:45:18 +00:00
|
|
|
>;
|
|
|
|
i18n: LocalizerType;
|
2023-08-16 20:54:39 +00:00
|
|
|
me: Pick<ConversationType, 'id' | 'serviceId'>;
|
2021-08-25 21:42:51 +00:00
|
|
|
ringMode: RingMode;
|
2021-08-17 21:45:18 +00:00
|
|
|
|
|
|
|
// The following should only be set for group conversations.
|
2024-02-08 15:28:34 +00:00
|
|
|
groupMembers?: Array<
|
|
|
|
Pick<
|
|
|
|
ConversationType,
|
|
|
|
'id' | 'firstName' | 'systemGivenName' | 'systemNickname' | 'title'
|
|
|
|
>
|
|
|
|
>;
|
2021-08-17 21:45:18 +00:00
|
|
|
isCallFull?: boolean;
|
2024-11-14 23:24:04 +00:00
|
|
|
peekedParticipants?: Array<PeekedParticipantType>;
|
2021-08-17 21:45:18 +00:00
|
|
|
};
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function CallingPreCallInfo({
|
2021-08-17 21:45:18 +00:00
|
|
|
conversation,
|
|
|
|
groupMembers = [],
|
|
|
|
i18n,
|
|
|
|
isCallFull = false,
|
|
|
|
me,
|
|
|
|
peekedParticipants = [],
|
2021-08-25 21:42:51 +00:00
|
|
|
ringMode,
|
2022-11-18 00:45:19 +00:00
|
|
|
}: PropsType): JSX.Element {
|
2024-11-14 23:24:04 +00:00
|
|
|
const [visibleParticipants, unknownParticipants] = React.useMemo<
|
|
|
|
[Array<PeekedParticipantType>, Array<PeekedParticipantType>]
|
|
|
|
>(
|
|
|
|
() =>
|
|
|
|
partition(peekedParticipants, (participant: PeekedParticipantType) =>
|
|
|
|
Boolean(participant.titleNoDefault)
|
|
|
|
),
|
|
|
|
[peekedParticipants]
|
|
|
|
);
|
|
|
|
|
2021-08-17 21:45:18 +00:00
|
|
|
let subtitle: string;
|
2021-08-25 21:42:51 +00:00
|
|
|
if (ringMode === RingMode.IsRinging) {
|
2023-03-30 00:03:25 +00:00
|
|
|
subtitle = i18n('icu:outgoingCallRinging');
|
2021-08-25 21:42:51 +00:00
|
|
|
} else if (isCallFull) {
|
2023-03-30 00:03:25 +00:00
|
|
|
subtitle = i18n('icu:calling__call-is-full');
|
2021-08-17 21:45:18 +00:00
|
|
|
} else if (peekedParticipants.length) {
|
2024-11-14 23:24:04 +00:00
|
|
|
if (unknownParticipants.length > 0) {
|
|
|
|
subtitle = i18n(
|
|
|
|
'icu:calling__pre-call-info--only-unknown-contacts-in-call',
|
|
|
|
{
|
|
|
|
count: peekedParticipants.length,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// 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 = visibleParticipants.map(participant => {
|
|
|
|
if (participant.serviceId === me.serviceId) {
|
|
|
|
hasYou = true;
|
|
|
|
return i18n('icu:you');
|
|
|
|
}
|
|
|
|
return getParticipantName(participant);
|
|
|
|
});
|
|
|
|
switch (participantNames.length) {
|
|
|
|
case 1:
|
|
|
|
subtitle = hasYou
|
|
|
|
? i18n('icu:calling__pre-call-info--another-device-in-call')
|
|
|
|
: i18n('icu:calling__pre-call-info--1-person-in-call', {
|
|
|
|
first: participantNames[0],
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
subtitle = i18n('icu:calling__pre-call-info--2-people-in-call', {
|
|
|
|
first: participantNames[0],
|
|
|
|
second: participantNames[1],
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
subtitle = i18n('icu:calling__pre-call-info--3-people-in-call', {
|
|
|
|
first: participantNames[0],
|
|
|
|
second: participantNames[1],
|
|
|
|
third: participantNames[2],
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
subtitle = i18n('icu:calling__pre-call-info--many-people-in-call', {
|
|
|
|
first: participantNames[0],
|
|
|
|
second: participantNames[1],
|
|
|
|
others: participantNames.length - 2,
|
|
|
|
});
|
|
|
|
break;
|
2021-08-17 21:45:18 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-25 21:42:51 +00:00
|
|
|
} 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':
|
2021-08-25 21:42:51 +00:00
|
|
|
memberNames = groupMembers
|
|
|
|
.filter(member => member.id !== me.id)
|
|
|
|
.map(getParticipantName);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw missingCaseError(conversation.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ring = ringMode === RingMode.WillRing;
|
2021-08-17 21:45:18 +00:00
|
|
|
|
|
|
|
switch (memberNames.length) {
|
|
|
|
case 0:
|
2023-03-30 00:03:25 +00:00
|
|
|
subtitle = i18n('icu:calling__pre-call-info--empty-group');
|
2021-08-17 21:45:18 +00:00
|
|
|
break;
|
2021-08-25 21:42:51 +00:00
|
|
|
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],
|
|
|
|
});
|
2021-08-17 21:45:18 +00:00
|
|
|
break;
|
2021-08-25 21:42:51 +00:00
|
|
|
}
|
|
|
|
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],
|
|
|
|
});
|
2021-08-17 21:45:18 +00:00
|
|
|
break;
|
2021-08-25 21:42:51 +00:00
|
|
|
}
|
|
|
|
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],
|
|
|
|
});
|
2021-08-17 21:45:18 +00:00
|
|
|
break;
|
2021-08-25 21:42:51 +00:00
|
|
|
}
|
|
|
|
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],
|
2024-02-29 23:07:50 +00:00
|
|
|
others: memberNames.length - 2,
|
2023-03-27 23:37:39 +00:00
|
|
|
})
|
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],
|
2024-02-29 23:07:50 +00:00
|
|
|
others: memberNames.length - 2,
|
2023-03-27 23:37:39 +00:00
|
|
|
});
|
2021-08-17 21:45:18 +00:00
|
|
|
break;
|
2021-08-25 21:42:51 +00:00
|
|
|
}
|
2021-08-17 21:45:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="module-CallingPreCallInfo">
|
|
|
|
<Avatar
|
2024-07-11 19:44:09 +00:00
|
|
|
avatarUrl={conversation.avatarUrl}
|
2021-12-01 17:24:00 +00:00
|
|
|
badge={undefined}
|
2021-08-17 21:45:18 +00:00
|
|
|
color={conversation.color}
|
|
|
|
acceptedMessageRequest={conversation.acceptedMessageRequest}
|
|
|
|
conversationType={conversation.type}
|
|
|
|
isMe={conversation.isMe}
|
|
|
|
noteToSelf={false}
|
|
|
|
phoneNumber={conversation.phoneNumber}
|
|
|
|
profileName={conversation.profileName}
|
|
|
|
sharedGroupNames={conversation.sharedGroupNames}
|
2024-09-24 14:00:03 +00:00
|
|
|
size={AvatarSize.SIXTY_FOUR}
|
2021-08-17 21:45:18 +00:00
|
|
|
title={conversation.title}
|
2024-07-11 19:44:09 +00:00
|
|
|
unblurredAvatarUrl={conversation.unblurredAvatarUrl}
|
2021-08-17 21:45:18 +00:00
|
|
|
i18n={i18n}
|
|
|
|
/>
|
|
|
|
<div className="module-CallingPreCallInfo__title">
|
2023-04-20 17:03:43 +00:00
|
|
|
<UserText text={conversation.title} />
|
2021-08-17 21:45:18 +00:00
|
|
|
</div>
|
|
|
|
<div className="module-CallingPreCallInfo__subtitle">{subtitle}</div>
|
|
|
|
</div>
|
|
|
|
);
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|