// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useState, useRef, useEffect } from 'react'; import Measure from 'react-measure'; import { Timestamp } from './Timestamp'; import { LocalizerType } from '../../types/Util'; import { CallMode } from '../../types/Calling'; import { CallingNotificationType, getCallingNotificationText, } from '../../util/callingNotification'; import { missingCaseError } from '../../util/missingCaseError'; import { Tooltip, TooltipPlacement } from '../Tooltip'; export type PropsActionsType = { messageSizeChanged: (messageId: string, conversationId: string) => void; returnToActiveCall: () => void; startCallingLobby: (_: { conversationId: string; isVideoCall: boolean; }) => void; }; type PropsHousekeeping = { i18n: LocalizerType; conversationId: string; messageId: string; }; type PropsType = CallingNotificationType & PropsActionsType & PropsHousekeeping; export const CallingNotification: React.FC = React.memo(props => { const { conversationId, i18n, messageId, messageSizeChanged } = props; const previousHeightRef = useRef(null); const [height, setHeight] = useState(null); useEffect(() => { if (height === null) { return; } if ( previousHeightRef.current !== null && height !== previousHeightRef.current ) { messageSizeChanged(messageId, conversationId); } previousHeightRef.current = height; }, [height, conversationId, messageId, messageSizeChanged]); let timestamp: number; let callType: 'audio' | 'video'; switch (props.callMode) { case CallMode.Direct: timestamp = props.acceptedTime || props.endedTime; callType = props.wasVideoCall ? 'video' : 'audio'; break; case CallMode.Group: timestamp = props.startedTime; callType = 'video'; break; default: window.log.error(missingCaseError(props)); return null; } return ( { if (!bounds) { window.log.error('We should be measuring the bounds'); return; } setHeight(bounds.height); }} > {({ measureRef }) => (
{getCallingNotificationText(props, i18n)}
)} ); }); function CallingNotificationButton(props: PropsType) { if (props.callMode !== CallMode.Group || props.ended) { return null; } const { activeCallConversationId, conversationId, deviceCount, i18n, maxDevices, returnToActiveCall, startCallingLobby, } = props; let buttonText: string; let disabledTooltipText: undefined | string; let onClick: undefined | (() => void); if (activeCallConversationId) { if (activeCallConversationId === conversationId) { buttonText = i18n('calling__return'); onClick = returnToActiveCall; } else { buttonText = i18n('calling__join'); disabledTooltipText = i18n( 'calling__call-notification__button__in-another-call-tooltip' ); } } else if (deviceCount >= maxDevices) { buttonText = i18n('calling__call-is-full'); disabledTooltipText = i18n( 'calling__call-notification__button__call-full-tooltip', [String(deviceCount)] ); } else { buttonText = i18n('calling__join'); onClick = () => { startCallingLobby({ conversationId, isVideoCall: true }); }; } const button = ( ); if (disabledTooltipText) { return ( {button} ); } return button; }