2024-05-22 16:24:27 +00:00
|
|
|
// Copyright 2024 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
2024-08-06 19:29:13 +00:00
|
|
|
import React, { useState } from 'react';
|
2024-09-09 22:09:57 +00:00
|
|
|
import classNames from 'classnames';
|
2024-05-22 16:24:27 +00:00
|
|
|
import type { CallHistoryGroup } from '../types/CallDisposition';
|
|
|
|
import type { LocalizerType } from '../types/I18N';
|
|
|
|
import { CallHistoryGroupPanelSection } from './conversation/conversation-details/CallHistoryGroupPanelSection';
|
|
|
|
import { PanelSection } from './conversation/conversation-details/PanelSection';
|
|
|
|
import {
|
|
|
|
ConversationDetailsIcon,
|
|
|
|
IconType,
|
|
|
|
} from './conversation/conversation-details/ConversationDetailsIcon';
|
|
|
|
import { PanelRow } from './conversation/conversation-details/PanelRow';
|
2024-07-30 18:39:24 +00:00
|
|
|
import type { CallLinkRestrictions, CallLinkType } from '../types/CallLink';
|
2024-05-22 16:24:27 +00:00
|
|
|
import { linkCallRoute } from '../util/signalRoutes';
|
|
|
|
import { drop } from '../util/drop';
|
|
|
|
import { Avatar, AvatarSize } from './Avatar';
|
|
|
|
import { Button, ButtonSize, ButtonVariant } from './Button';
|
|
|
|
import { copyCallLink } from '../util/copyLinksWithToast';
|
2024-06-26 17:35:48 +00:00
|
|
|
import { getColorForCallLink } from '../util/getColorForCallLink';
|
2024-08-06 19:29:13 +00:00
|
|
|
import { isCallLinkAdmin } from '../types/CallLink';
|
2024-07-30 18:39:24 +00:00
|
|
|
import { CallLinkRestrictionsSelect } from './CallLinkRestrictionsSelect';
|
2024-08-06 19:29:13 +00:00
|
|
|
import { ConfirmationDialog } from './ConfirmationDialog';
|
2024-08-26 20:48:41 +00:00
|
|
|
import { InAnotherCallTooltip } from './conversation/InAnotherCallTooltip';
|
2024-09-09 22:09:57 +00:00
|
|
|
import { offsetDistanceModifier } from '../util/popperUtil';
|
|
|
|
import { Tooltip, TooltipPlacement } from './Tooltip';
|
2024-05-22 16:24:27 +00:00
|
|
|
|
|
|
|
function toUrlWithoutProtocol(url: URL): string {
|
|
|
|
return `${url.hostname}${url.pathname}${url.search}${url.hash}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
export type CallLinkDetailsProps = Readonly<{
|
|
|
|
callHistoryGroup: CallHistoryGroup;
|
2024-10-16 17:36:32 +00:00
|
|
|
callLink: CallLinkType | undefined;
|
2024-09-09 22:09:57 +00:00
|
|
|
isAnybodyInCall: boolean;
|
|
|
|
isInCall: boolean;
|
|
|
|
isInAnotherCall: boolean;
|
2024-05-22 16:24:27 +00:00
|
|
|
i18n: LocalizerType;
|
2024-08-06 19:29:13 +00:00
|
|
|
onDeleteCallLink: () => void;
|
2024-07-30 18:39:24 +00:00
|
|
|
onOpenCallLinkAddNameModal: () => void;
|
2024-05-22 16:24:27 +00:00
|
|
|
onStartCallLinkLobby: () => void;
|
|
|
|
onShareCallLinkViaSignal: () => void;
|
2024-07-30 18:39:24 +00:00
|
|
|
onUpdateCallLinkRestrictions: (restrictions: CallLinkRestrictions) => void;
|
2024-05-22 16:24:27 +00:00
|
|
|
}>;
|
|
|
|
|
|
|
|
export function CallLinkDetails({
|
|
|
|
callHistoryGroup,
|
|
|
|
callLink,
|
|
|
|
i18n,
|
2024-09-09 22:09:57 +00:00
|
|
|
isAnybodyInCall,
|
|
|
|
isInCall,
|
|
|
|
isInAnotherCall,
|
2024-08-06 19:29:13 +00:00
|
|
|
onDeleteCallLink,
|
2024-07-30 18:39:24 +00:00
|
|
|
onOpenCallLinkAddNameModal,
|
2024-05-22 16:24:27 +00:00
|
|
|
onStartCallLinkLobby,
|
|
|
|
onShareCallLinkViaSignal,
|
2024-07-30 18:39:24 +00:00
|
|
|
onUpdateCallLinkRestrictions,
|
2024-05-22 16:24:27 +00:00
|
|
|
}: CallLinkDetailsProps): JSX.Element {
|
2024-08-06 19:29:13 +00:00
|
|
|
const [isDeleteCallLinkModalOpen, setIsDeleteCallLinkModalOpen] =
|
|
|
|
useState(false);
|
2024-10-16 17:36:32 +00:00
|
|
|
|
|
|
|
if (!callLink) {
|
|
|
|
return renderMissingCallLink({ callHistoryGroup, i18n });
|
|
|
|
}
|
|
|
|
|
2024-05-22 16:24:27 +00:00
|
|
|
const webUrl = linkCallRoute.toWebUrl({
|
|
|
|
key: callLink.rootKey,
|
|
|
|
});
|
2024-08-26 20:48:41 +00:00
|
|
|
const joinButton = (
|
|
|
|
<Button
|
2024-09-09 22:09:57 +00:00
|
|
|
className={classNames({
|
|
|
|
CallLinkDetails__HeaderButton: true,
|
|
|
|
'CallLinkDetails__HeaderButton--active-call': isAnybodyInCall,
|
|
|
|
})}
|
|
|
|
variant={
|
|
|
|
isAnybodyInCall
|
|
|
|
? ButtonVariant.Calling
|
|
|
|
: ButtonVariant.SecondaryAffirmative
|
|
|
|
}
|
|
|
|
discouraged={isInAnotherCall}
|
2024-08-26 20:48:41 +00:00
|
|
|
size={ButtonSize.Small}
|
|
|
|
onClick={onStartCallLinkLobby}
|
|
|
|
>
|
2024-09-09 22:09:57 +00:00
|
|
|
{isInCall
|
|
|
|
? i18n('icu:CallsNewCallButton--return')
|
|
|
|
: i18n('icu:CallLinkDetails__Join')}
|
2024-08-26 20:48:41 +00:00
|
|
|
</Button>
|
|
|
|
);
|
2024-09-09 22:09:57 +00:00
|
|
|
const callLinkRestrictionsSelect = (
|
|
|
|
<CallLinkRestrictionsSelect
|
|
|
|
disabled={isAnybodyInCall}
|
|
|
|
i18n={i18n}
|
|
|
|
value={callLink.restrictions}
|
|
|
|
onChange={onUpdateCallLinkRestrictions}
|
|
|
|
/>
|
|
|
|
);
|
2024-08-26 20:48:41 +00:00
|
|
|
|
2024-05-22 16:24:27 +00:00
|
|
|
return (
|
|
|
|
<div className="CallLinkDetails__Container">
|
|
|
|
<header className="CallLinkDetails__Header">
|
|
|
|
<Avatar
|
|
|
|
className="CallLinkDetails__HeaderAvatar"
|
|
|
|
i18n={i18n}
|
|
|
|
badge={undefined}
|
2024-06-26 17:35:48 +00:00
|
|
|
color={getColorForCallLink(callLink.rootKey)}
|
2024-05-22 16:24:27 +00:00
|
|
|
conversationType="callLink"
|
|
|
|
size={AvatarSize.SIXTY_FOUR}
|
|
|
|
acceptedMessageRequest
|
|
|
|
isMe={false}
|
|
|
|
sharedGroupNames={[]}
|
|
|
|
title={callLink.name ?? i18n('icu:calling__call-link-default-title')}
|
|
|
|
/>
|
|
|
|
<div className="CallLinkDetails__HeaderDetails">
|
|
|
|
<h1 className="CallLinkDetails__HeaderTitle">
|
|
|
|
{callLink.name === ''
|
|
|
|
? i18n('icu:calling__call-link-default-title')
|
|
|
|
: callLink.name}
|
|
|
|
</h1>
|
|
|
|
<p className="CallLinkDetails__HeaderDescription">
|
|
|
|
{toUrlWithoutProtocol(webUrl)}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
<div className="CallLinkDetails__HeaderActions">
|
2024-09-09 22:09:57 +00:00
|
|
|
{isInAnotherCall ? (
|
2024-08-26 20:48:41 +00:00
|
|
|
<InAnotherCallTooltip i18n={i18n}>
|
|
|
|
{joinButton}
|
|
|
|
</InAnotherCallTooltip>
|
|
|
|
) : (
|
|
|
|
joinButton
|
|
|
|
)}
|
2024-05-22 16:24:27 +00:00
|
|
|
</div>
|
|
|
|
</header>
|
|
|
|
<CallHistoryGroupPanelSection
|
|
|
|
callHistoryGroup={callHistoryGroup}
|
|
|
|
i18n={i18n}
|
|
|
|
/>
|
2024-07-30 18:39:24 +00:00
|
|
|
{isCallLinkAdmin(callLink) && (
|
|
|
|
<PanelSection>
|
|
|
|
<PanelRow
|
|
|
|
icon={
|
|
|
|
<ConversationDetailsIcon
|
|
|
|
ariaLabel={i18n('icu:CallLinkDetails__AddCallNameLabel')}
|
|
|
|
icon={IconType.edit}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
label={
|
|
|
|
callLink.name === ''
|
|
|
|
? i18n('icu:CallLinkDetails__AddCallNameLabel')
|
|
|
|
: i18n('icu:CallLinkDetails__EditCallNameLabel')
|
|
|
|
}
|
|
|
|
onClick={onOpenCallLinkAddNameModal}
|
|
|
|
/>
|
|
|
|
<PanelRow
|
|
|
|
icon={
|
|
|
|
<ConversationDetailsIcon
|
|
|
|
ariaLabel={i18n('icu:CallLinkDetails__ApproveAllMembersLabel')}
|
|
|
|
icon={IconType.approveAllMembers}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
label={i18n('icu:CallLinkDetails__ApproveAllMembersLabel')}
|
|
|
|
right={
|
2024-09-09 22:09:57 +00:00
|
|
|
isAnybodyInCall ? (
|
|
|
|
<Tooltip
|
|
|
|
className="CallLinkDetails__ApproveAllMembersDisabledTooltip"
|
|
|
|
content={i18n(
|
|
|
|
'icu:CallLinkDetails__SettingTooltip--disabled-for-active-call'
|
|
|
|
)}
|
|
|
|
direction={TooltipPlacement.Top}
|
|
|
|
popperModifiers={[offsetDistanceModifier(5)]}
|
|
|
|
>
|
|
|
|
{callLinkRestrictionsSelect}
|
|
|
|
</Tooltip>
|
|
|
|
) : (
|
|
|
|
callLinkRestrictionsSelect
|
|
|
|
)
|
2024-07-30 18:39:24 +00:00
|
|
|
}
|
|
|
|
/>
|
|
|
|
</PanelSection>
|
|
|
|
)}
|
2024-05-22 16:24:27 +00:00
|
|
|
<PanelSection>
|
|
|
|
<PanelRow
|
|
|
|
icon={
|
|
|
|
<ConversationDetailsIcon
|
|
|
|
ariaLabel={i18n('icu:CallLinkDetails__CopyLink')}
|
|
|
|
icon={IconType.share}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
label={i18n('icu:CallLinkDetails__CopyLink')}
|
|
|
|
onClick={() => {
|
|
|
|
drop(copyCallLink(webUrl.toString()));
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<PanelRow
|
|
|
|
icon={
|
|
|
|
<ConversationDetailsIcon
|
|
|
|
ariaLabel={i18n('icu:CallLinkDetails__ShareLinkViaSignal')}
|
|
|
|
icon={IconType.forward}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
label={i18n('icu:CallLinkDetails__ShareLinkViaSignal')}
|
|
|
|
onClick={onShareCallLinkViaSignal}
|
|
|
|
/>
|
|
|
|
</PanelSection>
|
2024-08-06 19:29:13 +00:00
|
|
|
{isCallLinkAdmin(callLink) && (
|
|
|
|
<PanelSection>
|
|
|
|
<PanelRow
|
2024-09-09 22:09:57 +00:00
|
|
|
className={classNames({
|
|
|
|
CallLinkDetails__DeleteLink: true,
|
|
|
|
'CallLinkDetails__DeleteLink--disabled-for-active-call':
|
|
|
|
isAnybodyInCall,
|
|
|
|
})}
|
|
|
|
disabled={isAnybodyInCall}
|
2024-08-06 19:29:13 +00:00
|
|
|
icon={
|
|
|
|
<ConversationDetailsIcon
|
|
|
|
ariaLabel={i18n('icu:CallLinkDetails__DeleteLink')}
|
|
|
|
icon={IconType.trash}
|
|
|
|
/>
|
|
|
|
}
|
2024-09-09 22:09:57 +00:00
|
|
|
label={
|
|
|
|
isAnybodyInCall ? (
|
|
|
|
<Tooltip
|
|
|
|
className="CallLinkDetails__DeleteLinkTooltip"
|
|
|
|
content={i18n(
|
|
|
|
'icu:CallLinkDetails__DeleteLinkTooltip--disabled-for-active-call'
|
|
|
|
)}
|
|
|
|
direction={TooltipPlacement.Top}
|
|
|
|
popperModifiers={[offsetDistanceModifier(5)]}
|
|
|
|
>
|
|
|
|
{i18n('icu:CallLinkDetails__DeleteLink')}
|
|
|
|
</Tooltip>
|
|
|
|
) : (
|
|
|
|
i18n('icu:CallLinkDetails__DeleteLink')
|
|
|
|
)
|
|
|
|
}
|
2024-08-06 19:29:13 +00:00
|
|
|
onClick={() => {
|
|
|
|
setIsDeleteCallLinkModalOpen(true);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</PanelSection>
|
|
|
|
)}
|
|
|
|
{isDeleteCallLinkModalOpen && (
|
|
|
|
<ConfirmationDialog
|
|
|
|
i18n={i18n}
|
|
|
|
dialogName="CallLinkDetails__DeleteLinkModal"
|
|
|
|
title={i18n('icu:CallLinkDetails__DeleteLinkModal__Title')}
|
|
|
|
cancelText={i18n('icu:CallLinkDetails__DeleteLinkModal__Cancel')}
|
|
|
|
actions={[
|
|
|
|
{
|
|
|
|
text: i18n('icu:CallLinkDetails__DeleteLinkModal__Delete'),
|
|
|
|
style: 'affirmative',
|
|
|
|
action: onDeleteCallLink,
|
|
|
|
},
|
|
|
|
]}
|
|
|
|
onClose={() => {
|
|
|
|
setIsDeleteCallLinkModalOpen(false);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{i18n('icu:CallLinkDetails__DeleteLinkModal__Body')}
|
|
|
|
</ConfirmationDialog>
|
|
|
|
)}
|
2024-05-22 16:24:27 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
2024-10-16 17:36:32 +00:00
|
|
|
|
|
|
|
function renderMissingCallLink({
|
|
|
|
callHistoryGroup,
|
|
|
|
i18n,
|
|
|
|
}: Pick<CallLinkDetailsProps, 'callHistoryGroup' | 'i18n'>): JSX.Element {
|
|
|
|
return (
|
|
|
|
<div className="CallLinkDetails__Container">
|
|
|
|
<header className="CallLinkDetails__Header">
|
|
|
|
<Avatar
|
|
|
|
className="CallLinkDetails__HeaderAvatar"
|
|
|
|
i18n={i18n}
|
|
|
|
badge={undefined}
|
|
|
|
conversationType="callLink"
|
|
|
|
size={AvatarSize.SIXTY_FOUR}
|
|
|
|
acceptedMessageRequest
|
|
|
|
isMe={false}
|
|
|
|
sharedGroupNames={[]}
|
|
|
|
title={i18n('icu:calling__call-link-default-title')}
|
|
|
|
/>
|
|
|
|
<div className="CallLinkDetails__HeaderDetails">
|
|
|
|
<h1 className="CallLinkDetails__HeaderTitle">
|
|
|
|
{i18n('icu:calling__call-link-default-title')}
|
|
|
|
</h1>
|
|
|
|
</div>
|
|
|
|
</header>
|
|
|
|
<CallHistoryGroupPanelSection
|
|
|
|
callHistoryGroup={callHistoryGroup}
|
|
|
|
i18n={i18n}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|