// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactNode } from 'react'; import React from 'react'; import { noop } from 'lodash'; import { ContextMenuTrigger } from 'react-contextmenu'; import { SystemMessage, SystemMessageKind } from './SystemMessage'; import { Button, ButtonSize, ButtonVariant } from '../Button'; import { MessageTimestamp } from './MessageTimestamp'; import type { LocalizerType } from '../../types/Util'; import { CallMode, CallDirection, CallType, DirectCallStatus, GroupCallStatus, } from '../../types/CallDisposition'; import type { CallingNotificationType } from '../../util/callingNotification'; import { getCallingIcon, getCallingNotificationText, } from '../../util/callingNotification'; import { missingCaseError } from '../../util/missingCaseError'; import { Tooltip, TooltipPlacement } from '../Tooltip'; import * as log from '../../logging/log'; import { type ContextMenuTriggerType, MessageContextMenu, useHandleMessageContextMenu, } from './MessageContextMenu'; import type { DeleteMessagesPropsType } from '../../state/ducks/globalModals'; import { useKeyboardShortcutsConditionally, useOpenContextMenu, } from '../../hooks/useKeyboardShortcuts'; import { MINUTE } from '../../util/durations'; import { isMoreRecentThan } from '../../util/timestamp'; export type PropsActionsType = { onOutgoingAudioCallInConversation: (conversationId: string) => void; onOutgoingVideoCallInConversation: (conversationId: string) => void; returnToActiveCall: () => void; toggleDeleteMessagesModal: (props: DeleteMessagesPropsType) => void; }; type PropsHousekeeping = { i18n: LocalizerType; id: string; conversationId: string; isNextItemCallingNotification: boolean; }; export type PropsType = CallingNotificationType & PropsActionsType & PropsHousekeeping; export const CallingNotification: React.FC = React.memo( function CallingNotificationInner(props) { const menuTriggerRef = React.useRef(null); const handleContextMenu = useHandleMessageContextMenu(menuTriggerRef); const openContextMenuKeyboard = useOpenContextMenu(handleContextMenu); useKeyboardShortcutsConditionally( !props.isSelectMode && props.isTargeted, openContextMenuKeyboard ); const { i18n } = props; if (props.callHistory == null) { return null; } const { type, direction, status, timestamp } = props.callHistory; const icon = getCallingIcon(type, direction, status); return ( <>
{getCallingNotificationText(props, i18n)} ·{' '} } icon={icon} kind={ status === DirectCallStatus.Missed || status === GroupCallStatus.Missed || status === DirectCallStatus.Declined || status === GroupCallStatus.Declined ? SystemMessageKind.Danger : SystemMessageKind.Normal } />
{ 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} /> ); } ); function renderCallingNotificationButton( props: Readonly ): ReactNode { const { conversationId, i18n, isNextItemCallingNotification, onOutgoingAudioCallInConversation, onOutgoingVideoCallInConversation, returnToActiveCall, } = props; if (isNextItemCallingNotification) { return null; } let buttonText: string; let disabledTooltipText: undefined | string; let onClick: () => void; if (props.callHistory == null) { return null; } switch (props.callHistory.mode) { case CallMode.Direct: { const { direction, type } = props.callHistory; if (props.callHistory.status === DirectCallStatus.Pending) { return null; } buttonText = direction === CallDirection.Incoming ? i18n('icu:calling__call-back') : i18n('icu:calling__call-again'); if (props.activeConversationId != null) { disabledTooltipText = i18n('icu:calling__in-another-call-tooltip'); onClick = noop; } else { onClick = () => { if (type === CallType.Video) { onOutgoingVideoCallInConversation(conversationId); } else { onOutgoingAudioCallInConversation(conversationId); } }; } break; } case CallMode.Group: { if (props.groupCallEnded) { const { direction, status, timestamp } = props.callHistory; if ( (direction === CallDirection.Incoming && (status === GroupCallStatus.Declined || status === GroupCallStatus.Missed)) || isMoreRecentThan(timestamp, 5 * MINUTE) ) { buttonText = i18n('icu:calling__call-back'); onClick = () => { onOutgoingVideoCallInConversation(conversationId); }; } else { return null; } } else 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) { buttonText = i18n('icu:calling__call-is-full'); disabledTooltipText = i18n( 'icu:calling__call-notification__button__call-full-tooltip', { max: props.maxDevices, } ); onClick = noop; } else { buttonText = i18n('icu:calling__join'); onClick = () => { onOutgoingVideoCallInConversation(conversationId); }; } break; } case CallMode.Adhoc: log.warn('CallingNotification for adhoc call, should never happen'); return null; default: log.error(missingCaseError(props.callHistory.mode)); return null; } const button = ( ); if (disabledTooltipText) { return ( {button} ); } return button; }