signal-desktop/ts/components/CallingPip.tsx

337 lines
9.5 KiB
TypeScript
Raw Normal View History

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
2020-10-01 00:43:05 +00:00
import React from 'react';
import { minBy, debounce, noop } from 'lodash';
2023-01-09 18:38:57 +00:00
import type { VideoFrameSource } from '@signalapp/ringrtc';
2020-11-17 15:07:53 +00:00
import { CallingPipRemoteVideo } from './CallingPipRemoteVideo';
import type { LocalizerType } from '../types/Util';
import type { ActiveCallType, GroupCallVideoRequest } from '../types/Calling';
import type {
2020-10-01 00:43:05 +00:00
SetLocalPreviewType,
SetRendererCanvasType,
} from '../state/ducks/calling';
import { missingCaseError } from '../util/missingCaseError';
import { useActivateSpeakerViewOnPresenting } from '../hooks/useActivateSpeakerViewOnPresenting';
enum PositionMode {
BeingDragged,
SnapToBottom,
SnapToLeft,
SnapToRight,
SnapToTop,
}
type PositionState =
| {
mode: PositionMode.BeingDragged;
mouseX: number;
mouseY: number;
dragOffsetX: number;
dragOffsetY: number;
}
| {
mode: PositionMode.SnapToLeft | PositionMode.SnapToRight;
offsetY: number;
}
| {
mode: PositionMode.SnapToTop | PositionMode.SnapToBottom;
offsetX: number;
};
type SnapCandidate = {
mode:
| PositionMode.SnapToBottom
| PositionMode.SnapToLeft
| PositionMode.SnapToRight
| PositionMode.SnapToTop;
distanceToEdge: number;
};
2020-10-01 00:43:05 +00:00
export type PropsType = {
activeCall: ActiveCallType;
2020-11-17 15:07:53 +00:00
getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource;
2022-08-16 23:52:09 +00:00
hangUpActiveCall: (reason: string) => void;
2020-10-01 00:43:05 +00:00
hasLocalVideo: boolean;
i18n: LocalizerType;
2022-09-07 15:52:55 +00:00
setGroupCallVideoRequest: (
_: Array<GroupCallVideoRequest>,
speakerHeight: number
) => void;
2020-10-01 00:43:05 +00:00
setLocalPreview: (_: SetLocalPreviewType) => void;
setRendererCanvas: (_: SetRendererCanvasType) => void;
switchToPresentationView: () => void;
switchFromPresentationView: () => void;
2020-10-01 00:43:05 +00:00
togglePip: () => void;
};
const PIP_HEIGHT = 156;
const PIP_WIDTH = 120;
const PIP_TOP_MARGIN = 56;
2020-10-01 00:43:05 +00:00
const PIP_PADDING = 8;
2022-11-18 00:45:19 +00:00
export function CallingPip({
activeCall,
2020-11-17 15:07:53 +00:00
getGroupCallVideoFrameSource,
hangUpActiveCall,
2020-10-01 00:43:05 +00:00
hasLocalVideo,
i18n,
setGroupCallVideoRequest,
2020-10-01 00:43:05 +00:00
setLocalPreview,
setRendererCanvas,
switchToPresentationView,
switchFromPresentationView,
2020-10-01 00:43:05 +00:00
togglePip,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
2023-12-21 19:05:59 +00:00
const isRTL = i18n.getLocaleDirection() === 'rtl';
const videoContainerRef = React.useRef<null | HTMLDivElement>(null);
2020-10-01 00:43:05 +00:00
const localVideoRef = React.useRef(null);
const [windowWidth, setWindowWidth] = React.useState(window.innerWidth);
const [windowHeight, setWindowHeight] = React.useState(window.innerHeight);
const [positionState, setPositionState] = React.useState<PositionState>({
mode: PositionMode.SnapToRight,
offsetY: PIP_TOP_MARGIN,
2020-10-01 00:43:05 +00:00
});
useActivateSpeakerViewOnPresenting({
remoteParticipants: activeCall.remoteParticipants,
switchToPresentationView,
switchFromPresentationView,
});
2020-10-01 00:43:05 +00:00
React.useEffect(() => {
setLocalPreview({ element: localVideoRef });
2020-11-17 15:07:53 +00:00
}, [setLocalPreview]);
2020-10-01 00:43:05 +00:00
2022-08-16 23:52:09 +00:00
const hangUp = React.useCallback(() => {
hangUpActiveCall('pip button click');
}, [hangUpActiveCall]);
2020-10-01 00:43:05 +00:00
const handleMouseMove = React.useCallback(
(ev: MouseEvent) => {
if (positionState.mode === PositionMode.BeingDragged) {
setPositionState(oldState => ({
...oldState,
2022-06-21 15:59:47 +00:00
mouseX: ev.clientX,
mouseY: ev.clientY,
}));
2020-10-01 00:43:05 +00:00
}
},
[positionState]
2020-10-01 00:43:05 +00:00
);
const handleMouseUp = React.useCallback(() => {
if (positionState.mode === PositionMode.BeingDragged) {
const { mouseX, mouseY, dragOffsetX, dragOffsetY } = positionState;
2020-10-01 00:43:05 +00:00
const { innerHeight, innerWidth } = window;
const offsetX = mouseX - dragOffsetX;
const offsetY = mouseY - dragOffsetY;
2020-10-01 00:43:05 +00:00
2023-12-21 21:43:28 +00:00
let distanceToLeftEdge: number;
let distanceToRightEdge: number;
if (isRTL) {
distanceToLeftEdge = innerWidth - (offsetX + PIP_WIDTH);
distanceToRightEdge = offsetX;
} else {
distanceToLeftEdge = offsetX;
distanceToRightEdge = innerWidth - (offsetX + PIP_WIDTH);
}
const snapCandidates: Array<SnapCandidate> = [
{
mode: PositionMode.SnapToLeft,
2023-12-21 21:43:28 +00:00
distanceToEdge: distanceToLeftEdge,
},
{
mode: PositionMode.SnapToRight,
2023-12-21 21:43:28 +00:00
distanceToEdge: distanceToRightEdge,
},
{
mode: PositionMode.SnapToTop,
distanceToEdge: offsetY - PIP_TOP_MARGIN,
},
{
mode: PositionMode.SnapToBottom,
distanceToEdge: innerHeight - (offsetY + PIP_HEIGHT),
},
];
2020-10-01 00:43:05 +00:00
// This fallback is mostly for TypeScript, because `minBy` says it can return
// `undefined`.
const snapTo =
minBy(snapCandidates, candidate => candidate.distanceToEdge) ||
snapCandidates[0];
switch (snapTo.mode) {
case PositionMode.SnapToLeft:
case PositionMode.SnapToRight:
setPositionState({
mode: snapTo.mode,
offsetY,
});
break;
case PositionMode.SnapToTop:
case PositionMode.SnapToBottom:
setPositionState({
mode: snapTo.mode,
2023-12-21 19:05:59 +00:00
offsetX: isRTL ? innerWidth - (offsetX + PIP_WIDTH) : offsetX,
});
break;
default:
throw missingCaseError(snapTo.mode);
}
2020-10-01 00:43:05 +00:00
}
2023-12-21 19:05:59 +00:00
}, [isRTL, positionState, setPositionState]);
2020-10-01 00:43:05 +00:00
React.useEffect(() => {
if (positionState.mode === PositionMode.BeingDragged) {
2020-10-01 00:43:05 +00:00
document.addEventListener('mousemove', handleMouseMove, false);
document.addEventListener('mouseup', handleMouseUp, false);
return () => {
document.removeEventListener('mouseup', handleMouseUp, false);
document.removeEventListener('mousemove', handleMouseMove, false);
};
2020-10-01 00:43:05 +00:00
}
return noop;
}, [positionState.mode, handleMouseMove, handleMouseUp]);
React.useEffect(() => {
const handleWindowResize = debounce(
() => {
setWindowWidth(window.innerWidth);
setWindowHeight(window.innerHeight);
},
100,
{
maxWait: 3000,
}
);
2020-10-01 00:43:05 +00:00
window.addEventListener('resize', handleWindowResize, false);
2020-10-01 00:43:05 +00:00
return () => {
window.removeEventListener('resize', handleWindowResize, false);
2020-10-01 00:43:05 +00:00
};
}, []);
const [translateX, translateY] = React.useMemo<[number, number]>(() => {
switch (positionState.mode) {
case PositionMode.BeingDragged:
return [
2023-12-21 19:05:59 +00:00
isRTL
? windowWidth -
positionState.mouseX -
(PIP_WIDTH - positionState.dragOffsetX)
: positionState.mouseX - positionState.dragOffsetX,
positionState.mouseY - positionState.dragOffsetY,
];
case PositionMode.SnapToLeft:
return [
PIP_PADDING,
Math.min(
positionState.offsetY,
windowHeight - PIP_PADDING - PIP_HEIGHT
),
];
case PositionMode.SnapToRight:
return [
windowWidth - PIP_PADDING - PIP_WIDTH,
Math.min(
positionState.offsetY,
windowHeight - PIP_PADDING - PIP_HEIGHT
),
];
case PositionMode.SnapToTop:
return [
Math.min(
positionState.offsetX,
windowWidth - PIP_PADDING - PIP_WIDTH
),
PIP_TOP_MARGIN + PIP_PADDING,
];
case PositionMode.SnapToBottom:
return [
Math.min(
positionState.offsetX,
windowWidth - PIP_PADDING - PIP_WIDTH
),
windowHeight - PIP_PADDING - PIP_HEIGHT,
];
default:
throw missingCaseError(positionState);
}
2023-12-21 19:05:59 +00:00
}, [isRTL, windowWidth, windowHeight, positionState]);
const localizedTranslateX = isRTL ? -translateX : translateX;
2020-10-01 00:43:05 +00:00
return (
// eslint-disable-next-line jsx-a11y/no-static-element-interactions
<div
className="module-calling-pip"
onMouseDown={ev => {
const node = videoContainerRef.current;
if (!node) {
return;
}
2023-12-21 19:05:59 +00:00
const rect = node.getBoundingClientRect();
2022-06-21 15:59:47 +00:00
const dragOffsetX = ev.clientX - rect.left;
const dragOffsetY = ev.clientY - rect.top;
2020-10-01 00:43:05 +00:00
setPositionState({
mode: PositionMode.BeingDragged,
2022-06-21 15:59:47 +00:00
mouseX: ev.clientX,
mouseY: ev.clientY,
dragOffsetX,
dragOffsetY,
2020-10-01 00:43:05 +00:00
});
}}
ref={videoContainerRef}
style={{
cursor:
positionState.mode === PositionMode.BeingDragged
? '-webkit-grabbing'
: '-webkit-grab',
transform: `translate3d(${localizedTranslateX}px,calc(${translateY}px), 0)`,
transition:
positionState.mode === PositionMode.BeingDragged
? 'none'
: 'transform ease-out 300ms',
2020-10-01 00:43:05 +00:00
}}
>
2020-11-17 15:07:53 +00:00
<CallingPipRemoteVideo
activeCall={activeCall}
2020-11-17 15:07:53 +00:00
getGroupCallVideoFrameSource={getGroupCallVideoFrameSource}
i18n={i18n}
setRendererCanvas={setRendererCanvas}
setGroupCallVideoRequest={setGroupCallVideoRequest}
2020-11-17 15:07:53 +00:00
/>
2020-10-01 00:43:05 +00:00
{hasLocalVideo ? (
<video
className="module-calling-pip__video--local"
ref={localVideoRef}
autoPlay
/>
) : null}
<div className="module-calling-pip__actions">
<button
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:calling__hangup')}
2020-10-01 00:43:05 +00:00
className="module-calling-pip__button--hangup"
2022-08-16 23:52:09 +00:00
onClick={hangUp}
2020-10-01 00:43:05 +00:00
type="button"
2020-11-19 18:11:35 +00:00
/>
<button
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:calling__pip--off')}
className="module-calling-pip__button--pip"
onClick={togglePip}
type="button"
>
<div />
</button>
2020-10-01 00:43:05 +00:00
</div>
</div>
);
2022-11-18 00:45:19 +00:00
}