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
|
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ReactNode } from 'react';
|
2022-03-03 20:23:10 +00:00
|
|
|
import React from 'react';
|
2021-08-26 20:51:55 +00:00
|
|
|
import { noop } from 'lodash';
|
2023-12-12 16:11:39 +00:00
|
|
|
import { ContextMenuTrigger } from 'react-contextmenu';
|
2020-06-04 18:16:19 +00:00
|
|
|
|
2023-01-12 23:33:22 +00:00
|
|
|
import { SystemMessage, SystemMessageKind } from './SystemMessage';
|
2021-08-26 20:51:55 +00:00
|
|
|
import { Button, ButtonSize, ButtonVariant } from '../Button';
|
2022-01-26 23:05:26 +00:00
|
|
|
import { MessageTimestamp } from './MessageTimestamp';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { LocalizerType } from '../../types/Util';
|
2020-12-07 20:43:19 +00:00
|
|
|
import { CallMode } from '../../types/Calling';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { CallingNotificationType } from '../../util/callingNotification';
|
2020-12-07 20:43:19 +00:00
|
|
|
import {
|
2021-08-26 20:51:55 +00:00
|
|
|
getCallingIcon,
|
2020-12-07 20:43:19 +00:00
|
|
|
getCallingNotificationText,
|
|
|
|
} from '../../util/callingNotification';
|
|
|
|
import { missingCaseError } from '../../util/missingCaseError';
|
|
|
|
import { Tooltip, TooltipPlacement } from '../Tooltip';
|
2021-09-17 18:27:53 +00:00
|
|
|
import * as log from '../../logging/log';
|
2023-08-09 00:53:06 +00:00
|
|
|
import {
|
|
|
|
CallDirection,
|
|
|
|
CallType,
|
|
|
|
DirectCallStatus,
|
|
|
|
GroupCallStatus,
|
|
|
|
} from '../../types/CallDisposition';
|
2023-12-12 16:11:39 +00:00
|
|
|
import {
|
|
|
|
type ContextMenuTriggerType,
|
|
|
|
MessageContextMenu,
|
|
|
|
useHandleMessageContextMenu,
|
|
|
|
} from './MessageContextMenu';
|
|
|
|
import type { DeleteMessagesPropsType } from '../../state/ducks/globalModals';
|
|
|
|
import {
|
|
|
|
useKeyboardShortcutsConditionally,
|
|
|
|
useOpenContextMenu,
|
|
|
|
} from '../../hooks/useKeyboardShortcuts';
|
2020-06-04 18:16:19 +00:00
|
|
|
|
2021-01-14 18:07:05 +00:00
|
|
|
export type PropsActionsType = {
|
2023-09-06 00:34:51 +00:00
|
|
|
onOutgoingAudioCallInConversation: (conversationId: string) => void;
|
|
|
|
onOutgoingVideoCallInConversation: (conversationId: string) => void;
|
2020-12-07 20:43:19 +00:00
|
|
|
returnToActiveCall: () => void;
|
2023-12-12 16:11:39 +00:00
|
|
|
toggleDeleteMessagesModal: (props: DeleteMessagesPropsType) => void;
|
2021-01-14 18:07:05 +00:00
|
|
|
};
|
2020-06-04 18:16:19 +00:00
|
|
|
|
|
|
|
type PropsHousekeeping = {
|
|
|
|
i18n: LocalizerType;
|
2023-12-12 16:11:39 +00:00
|
|
|
id: string;
|
2020-12-07 20:43:19 +00:00
|
|
|
conversationId: string;
|
2022-03-28 22:55:12 +00:00
|
|
|
isNextItemCallingNotification: boolean;
|
2020-06-04 18:16:19 +00:00
|
|
|
};
|
|
|
|
|
2023-08-09 00:53:06 +00:00
|
|
|
export type PropsType = CallingNotificationType &
|
|
|
|
PropsActionsType &
|
|
|
|
PropsHousekeeping;
|
2020-06-04 18:16:19 +00:00
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export const CallingNotification: React.FC<PropsType> = React.memo(
|
|
|
|
function CallingNotificationInner(props) {
|
2023-12-12 16:11:39 +00:00
|
|
|
const menuTriggerRef = React.useRef<ContextMenuTriggerType | null>(null);
|
|
|
|
const handleContextMenu = useHandleMessageContextMenu(menuTriggerRef);
|
|
|
|
const openContextMenuKeyboard = useOpenContextMenu(handleContextMenu);
|
|
|
|
useKeyboardShortcutsConditionally(
|
|
|
|
!props.isSelectMode && props.isTargeted,
|
|
|
|
openContextMenuKeyboard
|
|
|
|
);
|
2022-11-18 00:45:19 +00:00
|
|
|
const { i18n } = props;
|
2023-08-17 00:11:09 +00:00
|
|
|
if (props.callHistory == null) {
|
|
|
|
return null;
|
|
|
|
}
|
2023-12-12 16:11:39 +00:00
|
|
|
|
2023-08-09 00:53:06 +00:00
|
|
|
const { type, direction, status, timestamp } = props.callHistory;
|
|
|
|
const icon = getCallingIcon(type, direction, status);
|
2022-11-18 00:45:19 +00:00
|
|
|
return (
|
2023-12-12 16:11:39 +00:00
|
|
|
<>
|
|
|
|
<ContextMenuTrigger
|
|
|
|
id={props.id}
|
2024-02-07 14:47:25 +00:00
|
|
|
// react-contextmenu's typings are incorrect here
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
ref={menuTriggerRef as any}
|
|
|
|
disable={props.isSelectMode}
|
|
|
|
// Immediately hide the context menu on outside click.
|
|
|
|
// This is a bug in react-contextmenu trying to handle touch events.
|
|
|
|
holdToDisplay={-1}
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
// @ts-expect-error -- React/TS doesn't know about inert
|
|
|
|
// eslint-disable-next-line react/no-unknown-property
|
|
|
|
inert={props.isSelectMode ? '' : undefined}
|
|
|
|
>
|
|
|
|
<SystemMessage
|
|
|
|
button={renderCallingNotificationButton(props)}
|
|
|
|
contents={
|
|
|
|
<>
|
|
|
|
{getCallingNotificationText(props, i18n)} ·{' '}
|
|
|
|
<MessageTimestamp
|
|
|
|
direction="outgoing"
|
|
|
|
i18n={i18n}
|
|
|
|
timestamp={timestamp}
|
|
|
|
withImageNoCaption={false}
|
|
|
|
withSticker={false}
|
|
|
|
withTapToViewExpired={false}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
}
|
|
|
|
icon={icon}
|
|
|
|
kind={
|
|
|
|
status === DirectCallStatus.Missed ||
|
|
|
|
status === GroupCallStatus.Missed
|
|
|
|
? SystemMessageKind.Danger
|
|
|
|
: SystemMessageKind.Normal
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</ContextMenuTrigger>
|
2023-12-12 16:11:39 +00:00
|
|
|
<MessageContextMenu
|
|
|
|
i18n={i18n}
|
|
|
|
triggerId={props.id}
|
|
|
|
onDeleteMessage={() => {
|
|
|
|
props.toggleDeleteMessagesModal({
|
|
|
|
conversationId: props.conversationId,
|
|
|
|
messageIds: [props.id],
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
shouldShowAdditional={false}
|
|
|
|
onDownload={undefined}
|
|
|
|
onEdit={undefined}
|
|
|
|
onReplyToMessage={undefined}
|
|
|
|
onReact={undefined}
|
|
|
|
onRetryMessageSend={undefined}
|
|
|
|
onRetryDeleteForEveryone={undefined}
|
|
|
|
onCopy={undefined}
|
|
|
|
onSelect={undefined}
|
|
|
|
onForward={undefined}
|
|
|
|
onMoreInfo={undefined}
|
|
|
|
/>
|
|
|
|
</>
|
2022-11-18 00:45:19 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
2020-12-07 20:43:19 +00:00
|
|
|
|
2021-09-10 23:59:41 +00:00
|
|
|
function renderCallingNotificationButton(
|
|
|
|
props: Readonly<PropsType>
|
|
|
|
): ReactNode {
|
2020-12-07 20:43:19 +00:00
|
|
|
const {
|
|
|
|
conversationId,
|
|
|
|
i18n,
|
2022-03-28 22:55:12 +00:00
|
|
|
isNextItemCallingNotification,
|
2023-09-06 00:34:51 +00:00
|
|
|
onOutgoingAudioCallInConversation,
|
|
|
|
onOutgoingVideoCallInConversation,
|
2020-12-07 20:43:19 +00:00
|
|
|
returnToActiveCall,
|
|
|
|
} = props;
|
|
|
|
|
2022-03-28 22:55:12 +00:00
|
|
|
if (isNextItemCallingNotification) {
|
2021-09-10 23:59:41 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-12-07 20:43:19 +00:00
|
|
|
let buttonText: string;
|
|
|
|
let disabledTooltipText: undefined | string;
|
2021-08-26 20:51:55 +00:00
|
|
|
let onClick: () => void;
|
2021-09-10 23:59:41 +00:00
|
|
|
|
2023-08-17 00:11:09 +00:00
|
|
|
if (props.callHistory == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-08-09 00:53:06 +00:00
|
|
|
switch (props.callHistory.mode) {
|
2021-09-10 23:59:41 +00:00
|
|
|
case CallMode.Direct: {
|
2023-08-09 00:53:06 +00:00
|
|
|
const { direction, type } = props.callHistory;
|
2023-11-28 15:05:52 +00:00
|
|
|
if (props.callHistory.status === DirectCallStatus.Pending) {
|
|
|
|
return null;
|
|
|
|
}
|
2023-08-09 00:53:06 +00:00
|
|
|
buttonText =
|
|
|
|
direction === CallDirection.Incoming
|
|
|
|
? i18n('icu:calling__call-back')
|
|
|
|
: i18n('icu:calling__call-again');
|
2023-08-21 17:09:54 +00:00
|
|
|
if (props.activeConversationId != null) {
|
2023-03-30 00:03:25 +00:00
|
|
|
disabledTooltipText = i18n('icu:calling__in-another-call-tooltip');
|
2022-02-11 18:21:45 +00:00
|
|
|
onClick = noop;
|
|
|
|
} else {
|
|
|
|
onClick = () => {
|
2023-09-06 00:34:51 +00:00
|
|
|
if (type === CallType.Video) {
|
|
|
|
onOutgoingVideoCallInConversation(conversationId);
|
|
|
|
} else {
|
|
|
|
onOutgoingAudioCallInConversation(conversationId);
|
|
|
|
}
|
2022-02-11 18:21:45 +00:00
|
|
|
};
|
|
|
|
}
|
2021-09-10 23:59:41 +00:00
|
|
|
break;
|
2020-12-07 20:43:19 +00:00
|
|
|
}
|
2021-09-10 23:59:41 +00:00
|
|
|
case CallMode.Group: {
|
2023-08-21 17:09:54 +00:00
|
|
|
if (props.groupCallEnded) {
|
2021-09-10 23:59:41 +00:00
|
|
|
return null;
|
|
|
|
}
|
2023-08-21 17:09:54 +00:00
|
|
|
if (props.activeConversationId != null) {
|
|
|
|
if (props.activeConversationId === conversationId) {
|
|
|
|
buttonText = i18n('icu:calling__return');
|
|
|
|
onClick = returnToActiveCall;
|
|
|
|
} else {
|
|
|
|
buttonText = i18n('icu:calling__join');
|
|
|
|
disabledTooltipText = i18n('icu:calling__in-another-call-tooltip');
|
|
|
|
onClick = noop;
|
|
|
|
}
|
|
|
|
} else if (props.deviceCount > props.maxDevices) {
|
2023-03-30 00:03:25 +00:00
|
|
|
buttonText = i18n('icu:calling__call-is-full');
|
2021-09-10 23:59:41 +00:00
|
|
|
disabledTooltipText = i18n(
|
2023-03-30 00:03:25 +00:00
|
|
|
'icu:calling__call-notification__button__call-full-tooltip',
|
2023-03-27 23:37:39 +00:00
|
|
|
{
|
2023-08-09 00:53:06 +00:00
|
|
|
max: props.maxDevices,
|
2023-03-27 23:37:39 +00:00
|
|
|
}
|
2021-09-10 23:59:41 +00:00
|
|
|
);
|
|
|
|
onClick = noop;
|
2023-08-21 17:09:54 +00:00
|
|
|
} else {
|
2023-03-30 00:03:25 +00:00
|
|
|
buttonText = i18n('icu:calling__join');
|
2021-09-10 23:59:41 +00:00
|
|
|
onClick = () => {
|
2023-09-06 00:34:51 +00:00
|
|
|
onOutgoingVideoCallInConversation(conversationId);
|
2021-09-10 23:59:41 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2024-02-22 21:19:50 +00:00
|
|
|
case CallMode.Adhoc:
|
|
|
|
log.warn('CallingNotification for adhoc call, should never happen');
|
|
|
|
return null;
|
2021-09-10 23:59:41 +00:00
|
|
|
default:
|
2023-08-09 00:53:06 +00:00
|
|
|
log.error(missingCaseError(props.callHistory.mode));
|
2021-09-10 23:59:41 +00:00
|
|
|
return null;
|
2020-12-07 20:43:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const button = (
|
2021-08-26 20:51:55 +00:00
|
|
|
<Button
|
2020-12-07 20:43:19 +00:00
|
|
|
disabled={Boolean(disabledTooltipText)}
|
|
|
|
onClick={onClick}
|
2021-08-26 20:51:55 +00:00
|
|
|
size={ButtonSize.Small}
|
|
|
|
variant={ButtonVariant.SystemMessage}
|
2020-06-04 18:16:19 +00:00
|
|
|
>
|
2020-12-07 20:43:19 +00:00
|
|
|
{buttonText}
|
2021-08-26 20:51:55 +00:00
|
|
|
</Button>
|
2020-06-04 18:16:19 +00:00
|
|
|
);
|
2020-12-07 20:43:19 +00:00
|
|
|
|
|
|
|
if (disabledTooltipText) {
|
|
|
|
return (
|
|
|
|
<Tooltip content={disabledTooltipText} direction={TooltipPlacement.Top}>
|
|
|
|
{button}
|
|
|
|
</Tooltip>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return button;
|
|
|
|
}
|