signal-desktop/ts/components/CallLinkEditModal.tsx

214 lines
5.8 KiB
TypeScript
Raw Normal View History

2024-06-10 15:23:43 +00:00
// Copyright 2024 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
2024-06-25 18:56:28 +00:00
import type { ReactNode } from 'react';
import React, { useMemo, useState } from 'react';
2024-06-10 15:23:43 +00:00
import { v4 as generateUuid } from 'uuid';
import { Modal } from './Modal';
import type { LocalizerType } from '../types/I18N';
import {
CallLinkRestrictions,
toCallLinkRestrictions,
type CallLinkType,
} from '../types/CallLink';
import { Select } from './Select';
import { linkCallRoute } from '../util/signalRoutes';
import { Button, ButtonSize, ButtonVariant } from './Button';
import { Avatar, AvatarSize } from './Avatar';
2024-06-25 18:56:28 +00:00
const CallLinkEditModalRowIconClasses = {
Edit: 'CallLinkEditModal__RowIcon--Edit',
Approve: 'CallLinkEditModal__RowIcon--Approve',
Copy: 'CallLinkEditModal__RowIcon--Copy',
Share: 'CallLinkEditModal__RowIcon--Share',
} as const;
function RowIcon({
icon,
}: {
icon: keyof typeof CallLinkEditModalRowIconClasses;
}) {
return (
<i
role="presentation"
className={`CallLinkEditModal__RowIcon ${CallLinkEditModalRowIconClasses[icon]}`}
/>
);
}
function RowText({ children }: { children: ReactNode }) {
return <div className="CallLinkEditModal__RowLabel">{children}</div>;
}
function Row({ children }: { children: ReactNode }) {
return <div className="CallLinkEditModal__Row">{children}</div>;
}
function RowButton({
onClick,
children,
}: {
onClick: () => void;
children: ReactNode;
}) {
return (
<button
className="CallLinkEditModal__RowButton"
type="button"
onClick={onClick}
>
{children}
</button>
);
}
function Hr() {
return <hr className="CallLinkEditModal__Hr" />;
}
2024-06-10 15:23:43 +00:00
export type CallLinkEditModalProps = {
i18n: LocalizerType;
callLink: CallLinkType;
onClose: () => void;
onCopyCallLink: () => void;
2024-06-25 18:56:28 +00:00
onOpenCallLinkAddNameModal: () => void;
2024-06-10 15:23:43 +00:00
onUpdateCallLinkRestrictions: (restrictions: CallLinkRestrictions) => void;
onShareCallLinkViaSignal: () => void;
onStartCallLinkLobby: () => void;
};
export function CallLinkEditModal({
i18n,
callLink,
onClose,
onCopyCallLink,
2024-06-25 18:56:28 +00:00
onOpenCallLinkAddNameModal,
2024-06-10 15:23:43 +00:00
onUpdateCallLinkRestrictions,
onShareCallLinkViaSignal,
onStartCallLinkLobby,
}: CallLinkEditModalProps): JSX.Element {
const [restrictionsId] = useState(() => generateUuid());
const callLinkWebUrl = useMemo(() => {
2024-06-25 18:56:28 +00:00
return linkCallRoute.toWebUrl({ key: callLink.rootKey }).toString();
2024-06-10 15:23:43 +00:00
}, [callLink.rootKey]);
return (
<Modal
i18n={i18n}
modalName="CallLinkEditModal"
moduleClassName="CallLinkEditModal"
title={i18n('icu:CallLinkEditModal__Title')}
2024-06-25 18:56:28 +00:00
noEscapeClose
noMouseClose
padded={false}
modalFooter={
<Button type="submit" variant={ButtonVariant.Primary} onClick={onClose}>
{i18n('icu:done')}
</Button>
}
onClose={onClose}
2024-06-10 15:23:43 +00:00
>
<div className="CallLinkEditModal__Header">
<Avatar
i18n={i18n}
badge={undefined}
conversationType="callLink"
size={AvatarSize.SIXTY_FOUR}
acceptedMessageRequest
isMe={false}
sharedGroupNames={[]}
2024-06-25 18:56:28 +00:00
title={
callLink.name === ''
? i18n('icu:calling__call-link-default-title')
: callLink.name
}
2024-06-10 15:23:43 +00:00
/>
<div className="CallLinkEditModal__Header__Details">
2024-06-25 18:56:28 +00:00
<div className="CallLinkEditModal__Header__Title">
{callLink.name === ''
? i18n('icu:calling__call-link-default-title')
: callLink.name}
2024-06-10 15:23:43 +00:00
</div>
2024-06-25 18:56:28 +00:00
<button
className="CallLinkEditModal__Header__CallLinkButton"
type="button"
onClick={onCopyCallLink}
aria-label={i18n('icu:CallLinkDetails__CopyLink')}
>
<div className="CallLinkEditModal__Header__CallLinkButton__Text">
{callLinkWebUrl}
</div>
</button>
</div>
<div className="CallLinkEditModal__Header__Actions">
<Button
onClick={onStartCallLinkLobby}
size={ButtonSize.Small}
variant={ButtonVariant.SecondaryAffirmative}
className="CallLinkEditModal__JoinButton"
>
{i18n('icu:CallLinkEditModal__JoinButtonLabel')}
</Button>
2024-06-10 15:23:43 +00:00
</div>
</div>
2024-06-25 18:56:28 +00:00
<Hr />
<RowButton onClick={onOpenCallLinkAddNameModal}>
<Row>
<RowIcon icon="Edit" />
<RowText>{i18n('icu:CallLinkEditModal__AddCallNameLabel')}</RowText>
</Row>
</RowButton>
<Row>
<RowIcon icon="Approve" />
<RowText>
<label htmlFor={restrictionsId}>
{i18n('icu:CallLinkEditModal__InputLabel--ApproveAllMembers')}
</label>
</RowText>
2024-06-10 15:23:43 +00:00
<Select
id={restrictionsId}
2024-06-25 18:56:28 +00:00
value={String(callLink.restrictions)}
moduleClassName="CallLinkEditModal__RowSelect"
2024-06-10 15:23:43 +00:00
options={[
{
2024-06-25 18:56:28 +00:00
value: String(CallLinkRestrictions.None),
2024-06-10 15:23:43 +00:00
text: i18n(
'icu:CallLinkEditModal__ApproveAllMembers__Option--Off'
),
},
{
2024-06-25 18:56:28 +00:00
value: String(CallLinkRestrictions.AdminApproval),
2024-06-10 15:23:43 +00:00
text: i18n(
'icu:CallLinkEditModal__ApproveAllMembers__Option--On'
),
},
]}
onChange={value => {
2024-06-25 18:56:28 +00:00
onUpdateCallLinkRestrictions(toCallLinkRestrictions(value));
2024-06-10 15:23:43 +00:00
}}
/>
2024-06-25 18:56:28 +00:00
</Row>
2024-06-10 15:23:43 +00:00
2024-06-25 18:56:28 +00:00
<Hr />
<RowButton onClick={onCopyCallLink}>
<Row>
<RowIcon icon="Copy" />
<RowText>{i18n('icu:CallLinkDetails__CopyLink')}</RowText>
</Row>
</RowButton>
<RowButton onClick={onShareCallLinkViaSignal}>
<Row>
<RowIcon icon="Share" />
<RowText>{i18n('icu:CallLinkDetails__ShareLinkViaSignal')}</RowText>
</Row>
</RowButton>
2024-06-10 15:23:43 +00:00
</Modal>
);
}