2020-10-30 20:34:04 +00:00
|
|
|
// Copyright 2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2020-12-07 20:43:19 +00:00
|
|
|
import React, { useState, useRef, useEffect } from 'react';
|
|
|
|
import Measure from 'react-measure';
|
2020-06-04 18:16:19 +00:00
|
|
|
|
|
|
|
import { Timestamp } from './Timestamp';
|
|
|
|
import { LocalizerType } from '../../types/Util';
|
2020-12-07 20:43:19 +00:00
|
|
|
import { CallMode } from '../../types/Calling';
|
|
|
|
import {
|
|
|
|
CallingNotificationType,
|
|
|
|
getCallingNotificationText,
|
|
|
|
} from '../../util/callingNotification';
|
|
|
|
import { missingCaseError } from '../../util/missingCaseError';
|
|
|
|
import { Tooltip, TooltipPlacement } from '../Tooltip';
|
2020-06-04 18:16:19 +00:00
|
|
|
|
2021-01-14 18:07:05 +00:00
|
|
|
export type PropsActionsType = {
|
2020-12-07 20:43:19 +00:00
|
|
|
messageSizeChanged: (messageId: string, conversationId: string) => void;
|
|
|
|
returnToActiveCall: () => void;
|
|
|
|
startCallingLobby: (_: {
|
|
|
|
conversationId: string;
|
|
|
|
isVideoCall: boolean;
|
|
|
|
}) => void;
|
2021-01-14 18:07:05 +00:00
|
|
|
};
|
2020-06-04 18:16:19 +00:00
|
|
|
|
|
|
|
type PropsHousekeeping = {
|
|
|
|
i18n: LocalizerType;
|
2020-12-07 20:43:19 +00:00
|
|
|
conversationId: string;
|
|
|
|
messageId: string;
|
2020-06-04 18:16:19 +00:00
|
|
|
};
|
|
|
|
|
2020-12-07 20:43:19 +00:00
|
|
|
type PropsType = CallingNotificationType & PropsActionsType & PropsHousekeeping;
|
2020-06-04 18:16:19 +00:00
|
|
|
|
2020-12-07 20:43:19 +00:00
|
|
|
export const CallingNotification: React.FC<PropsType> = React.memo(props => {
|
|
|
|
const { conversationId, i18n, messageId, messageSizeChanged } = props;
|
|
|
|
|
|
|
|
const previousHeightRef = useRef<null | number>(null);
|
|
|
|
const [height, setHeight] = useState<null | number>(null);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (height === null) {
|
|
|
|
return;
|
2020-06-04 18:16:19 +00:00
|
|
|
}
|
2020-12-07 20:43:19 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
previousHeightRef.current !== null &&
|
|
|
|
height !== previousHeightRef.current
|
|
|
|
) {
|
|
|
|
messageSizeChanged(messageId, conversationId);
|
2020-09-14 19:51:27 +00:00
|
|
|
}
|
2020-12-07 20:43:19 +00:00
|
|
|
|
|
|
|
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;
|
2020-06-04 18:16:19 +00:00
|
|
|
}
|
|
|
|
|
2020-12-07 20:43:19 +00:00
|
|
|
return (
|
|
|
|
<Measure
|
|
|
|
bounds
|
|
|
|
onResize={({ bounds }) => {
|
|
|
|
if (!bounds) {
|
|
|
|
window.log.error('We should be measuring the bounds');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setHeight(bounds.height);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{({ measureRef }) => (
|
|
|
|
<div
|
|
|
|
className={`module-message-calling--notification module-message-calling--${callType}`}
|
|
|
|
ref={measureRef}
|
|
|
|
>
|
|
|
|
<div className={`module-message-calling--${callType}__icon`} />
|
|
|
|
{getCallingNotificationText(props, i18n)}
|
|
|
|
<div>
|
|
|
|
<Timestamp
|
|
|
|
i18n={i18n}
|
|
|
|
timestamp={timestamp}
|
|
|
|
extended
|
|
|
|
direction="outgoing"
|
|
|
|
withImageNoCaption={false}
|
|
|
|
withSticker={false}
|
|
|
|
withTapToViewExpired={false}
|
|
|
|
module="module-message__metadata__date"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<CallingNotificationButton {...props} />
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Measure>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
function CallingNotificationButton(props: PropsType) {
|
|
|
|
if (props.callMode !== CallMode.Group || props.ended) {
|
2020-06-04 18:16:19 +00:00
|
|
|
return null;
|
|
|
|
}
|
2020-12-07 20:43:19 +00:00
|
|
|
|
|
|
|
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 = (
|
|
|
|
<button
|
|
|
|
className="module-message-calling--notification__button"
|
|
|
|
disabled={Boolean(disabledTooltipText)}
|
|
|
|
onClick={onClick}
|
|
|
|
type="button"
|
2020-06-04 18:16:19 +00:00
|
|
|
>
|
2020-12-07 20:43:19 +00:00
|
|
|
{buttonText}
|
|
|
|
</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;
|
|
|
|
}
|