signal-desktop/ts/components/CallingParticipantsList.tsx

179 lines
6.4 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
2020-10-15 19:53:21 +00:00
/* eslint-disable react/no-array-index-key */
import React, { useContext } from 'react';
2020-10-15 19:53:21 +00:00
import { createPortal } from 'react-dom';
import FocusTrap from 'focus-trap-react';
import classNames from 'classnames';
2022-12-09 20:37:45 +00:00
import { Avatar, AvatarSize } from './Avatar';
2020-10-15 19:53:21 +00:00
import { ContactName } from './conversation/ContactName';
2020-11-20 19:39:50 +00:00
import { InContactsIcon } from './InContactsIcon';
import type { LocalizerType } from '../types/Util';
2023-08-16 20:54:39 +00:00
import type { ServiceIdString } from '../types/ServiceId';
import { sortByTitle } from '../util/sortByTitle';
import type { ConversationType } from '../state/ducks/conversations';
import { isInSystemContacts } from '../util/isInSystemContacts';
import { ModalContainerContext } from './ModalHost';
2020-12-02 18:14:03 +00:00
type ParticipantType = ConversationType & {
hasRemoteAudio?: boolean;
hasRemoteVideo?: boolean;
isHandRaised?: boolean;
presenting?: boolean;
};
2020-10-15 19:53:21 +00:00
export type PropsType = {
readonly i18n: LocalizerType;
readonly onClose: () => void;
2023-08-16 20:54:39 +00:00
readonly ourServiceId: ServiceIdString | undefined;
2020-12-02 18:14:03 +00:00
readonly participants: Array<ParticipantType>;
2020-10-15 19:53:21 +00:00
};
export const CallingParticipantsList = React.memo(
2022-11-18 00:45:19 +00:00
function CallingParticipantsListInner({
i18n,
onClose,
2023-08-16 20:54:39 +00:00
ourServiceId,
2022-11-18 00:45:19 +00:00
participants,
}: PropsType) {
2020-10-15 19:53:21 +00:00
const [root, setRoot] = React.useState<HTMLElement | null>(null);
const modalContainer = useContext(ModalContainerContext) ?? document.body;
2020-12-02 18:14:03 +00:00
const sortedParticipants = React.useMemo<Array<ParticipantType>>(
() => sortByTitle(participants),
2020-12-02 18:14:03 +00:00
[participants]
);
2020-10-15 19:53:21 +00:00
React.useEffect(() => {
const div = document.createElement('div');
modalContainer.appendChild(div);
2020-10-15 19:53:21 +00:00
setRoot(div);
return () => {
modalContainer.removeChild(div);
2020-10-15 19:53:21 +00:00
setRoot(null);
};
}, [modalContainer]);
2020-10-15 19:53:21 +00:00
2020-11-20 19:39:50 +00:00
const handleCancel = React.useCallback(
(e: React.MouseEvent) => {
if (e.target === e.currentTarget) {
onClose();
}
},
[onClose]
);
2020-10-15 19:53:21 +00:00
if (!root) {
return null;
}
return createPortal(
<FocusTrap>
<div
className="module-calling-participants-list__overlay"
onClick={handleCancel}
role="presentation"
>
<div className="module-calling-participants-list">
<div className="module-calling-participants-list__header">
<div className="module-calling-participants-list__title">
2024-03-04 18:03:11 +00:00
{participants.length
? i18n('icu:calling__in-this-call', {
people: participants.length,
})
: i18n('icu:calling__in-this-call--zero')}
</div>
<button
type="button"
className="module-calling-participants-list__close"
onClick={onClose}
tabIndex={0}
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:close')}
/>
2020-10-15 19:53:21 +00:00
</div>
<ul className="module-calling-participants-list__list">
{sortedParticipants.map(
(participant: ParticipantType, index: number) => (
<li
className="module-calling-participants-list__contact"
2023-08-16 20:54:39 +00:00
// It's tempting to use `participant.serviceId` as the `key`
// here, but that can result in duplicate keys for
// participants who have joined on multiple devices.
key={index}
>
<div className="module-calling-participants-list__avatar-and-name">
<Avatar
acceptedMessageRequest={
participant.acceptedMessageRequest
}
avatarPath={participant.avatarPath}
badge={undefined}
color={participant.color}
conversationType="direct"
i18n={i18n}
isMe={participant.isMe}
profileName={participant.profileName}
title={participant.title}
sharedGroupNames={participant.sharedGroupNames}
2022-12-09 20:37:45 +00:00
size={AvatarSize.THIRTY_TWO}
/>
2023-08-16 20:54:39 +00:00
{ourServiceId &&
participant.serviceId === ourServiceId ? (
<span className="module-calling-participants-list__name">
2023-03-30 00:03:25 +00:00
{i18n('icu:you')}
</span>
) : (
<>
<ContactName
module="module-calling-participants-list__name"
title={participant.title}
/>
{isInSystemContacts(participant) ? (
<InContactsIcon
className="module-calling-participants-list__contact-icon"
i18n={i18n}
/>
) : null}
</>
)}
</div>
<span
className={classNames(
'module-calling-participants-list__status-icon',
participant.isHandRaised &&
'module-calling-participants-list__hand-raised'
)}
/>
<span
className={classNames(
'module-calling-participants-list__status-icon',
participant.presenting &&
'module-calling-participants-list__presenting',
!participant.hasRemoteVideo &&
'module-calling-participants-list__muted--video'
)}
/>
<span
className={classNames(
'module-calling-participants-list__status-icon',
!participant.hasRemoteAudio &&
'module-calling-participants-list__muted--audio'
)}
/>
</li>
)
)}
</ul>
2020-10-15 19:53:21 +00:00
</div>
</div>
</FocusTrap>,
2020-10-15 19:53:21 +00:00
root
);
}
);