306 lines
8.4 KiB
TypeScript
306 lines
8.4 KiB
TypeScript
// Copyright 2020 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import React from 'react';
|
|
import { minBy, debounce, noop } from 'lodash';
|
|
import { CallingPipRemoteVideo } from './CallingPipRemoteVideo';
|
|
import { LocalizerType } from '../types/Util';
|
|
import {
|
|
ActiveCallType,
|
|
GroupCallVideoRequest,
|
|
VideoFrameSource,
|
|
} from '../types/Calling';
|
|
import {
|
|
HangUpType,
|
|
SetLocalPreviewType,
|
|
SetRendererCanvasType,
|
|
} from '../state/ducks/calling';
|
|
import { missingCaseError } from '../util/missingCaseError';
|
|
|
|
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;
|
|
};
|
|
|
|
interface SnapCandidate {
|
|
mode:
|
|
| PositionMode.SnapToBottom
|
|
| PositionMode.SnapToLeft
|
|
| PositionMode.SnapToRight
|
|
| PositionMode.SnapToTop;
|
|
distanceToEdge: number;
|
|
}
|
|
|
|
export type PropsType = {
|
|
activeCall: ActiveCallType;
|
|
getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource;
|
|
hangUp: (_: HangUpType) => void;
|
|
hasLocalVideo: boolean;
|
|
i18n: LocalizerType;
|
|
setGroupCallVideoRequest: (_: Array<GroupCallVideoRequest>) => void;
|
|
setLocalPreview: (_: SetLocalPreviewType) => void;
|
|
setRendererCanvas: (_: SetRendererCanvasType) => void;
|
|
togglePip: () => void;
|
|
};
|
|
|
|
const PIP_HEIGHT = 156;
|
|
const PIP_WIDTH = 120;
|
|
const PIP_TOP_MARGIN = 56;
|
|
const PIP_PADDING = 8;
|
|
|
|
export const CallingPip = ({
|
|
activeCall,
|
|
getGroupCallVideoFrameSource,
|
|
hangUp,
|
|
hasLocalVideo,
|
|
i18n,
|
|
setGroupCallVideoRequest,
|
|
setLocalPreview,
|
|
setRendererCanvas,
|
|
togglePip,
|
|
}: PropsType): JSX.Element | null => {
|
|
const videoContainerRef = React.useRef<null | HTMLDivElement>(null);
|
|
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: 0,
|
|
});
|
|
|
|
React.useEffect(() => {
|
|
setLocalPreview({ element: localVideoRef });
|
|
}, [setLocalPreview]);
|
|
|
|
const handleMouseMove = React.useCallback(
|
|
(ev: MouseEvent) => {
|
|
if (positionState.mode === PositionMode.BeingDragged) {
|
|
setPositionState(oldState => ({
|
|
...oldState,
|
|
mouseX: ev.screenX,
|
|
mouseY: ev.screenY,
|
|
}));
|
|
}
|
|
},
|
|
[positionState]
|
|
);
|
|
|
|
const handleMouseUp = React.useCallback(() => {
|
|
if (positionState.mode === PositionMode.BeingDragged) {
|
|
const { mouseX, mouseY, dragOffsetX, dragOffsetY } = positionState;
|
|
const { innerHeight, innerWidth } = window;
|
|
|
|
const offsetX = mouseX - dragOffsetX;
|
|
const offsetY = mouseY - dragOffsetY;
|
|
|
|
const snapCandidates: Array<SnapCandidate> = [
|
|
{
|
|
mode: PositionMode.SnapToLeft,
|
|
distanceToEdge: offsetX,
|
|
},
|
|
{
|
|
mode: PositionMode.SnapToRight,
|
|
distanceToEdge: innerWidth - (offsetX + PIP_WIDTH),
|
|
},
|
|
{
|
|
mode: PositionMode.SnapToTop,
|
|
distanceToEdge: offsetY - PIP_TOP_MARGIN,
|
|
},
|
|
{
|
|
mode: PositionMode.SnapToBottom,
|
|
distanceToEdge: innerHeight - (offsetY + PIP_HEIGHT),
|
|
},
|
|
];
|
|
|
|
// 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,
|
|
offsetX,
|
|
});
|
|
break;
|
|
default:
|
|
throw missingCaseError(snapTo.mode);
|
|
}
|
|
}
|
|
}, [positionState, setPositionState]);
|
|
|
|
React.useEffect(() => {
|
|
if (positionState.mode === PositionMode.BeingDragged) {
|
|
document.addEventListener('mousemove', handleMouseMove, false);
|
|
document.addEventListener('mouseup', handleMouseUp, false);
|
|
|
|
return () => {
|
|
document.removeEventListener('mouseup', handleMouseUp, false);
|
|
document.removeEventListener('mousemove', handleMouseMove, false);
|
|
};
|
|
}
|
|
return noop;
|
|
}, [positionState.mode, handleMouseMove, handleMouseUp]);
|
|
|
|
React.useEffect(() => {
|
|
const handleWindowResize = debounce(
|
|
() => {
|
|
setWindowWidth(window.innerWidth);
|
|
setWindowHeight(window.innerHeight);
|
|
},
|
|
100,
|
|
{
|
|
maxWait: 3000,
|
|
}
|
|
);
|
|
|
|
window.addEventListener('resize', handleWindowResize, false);
|
|
return () => {
|
|
window.removeEventListener('resize', handleWindowResize, false);
|
|
};
|
|
}, []);
|
|
|
|
const [translateX, translateY] = React.useMemo<[number, number]>(() => {
|
|
switch (positionState.mode) {
|
|
case PositionMode.BeingDragged:
|
|
return [
|
|
positionState.mouseX - positionState.dragOffsetX,
|
|
positionState.mouseY - positionState.dragOffsetY,
|
|
];
|
|
case PositionMode.SnapToLeft:
|
|
return [
|
|
PIP_PADDING,
|
|
Math.min(
|
|
PIP_TOP_MARGIN + positionState.offsetY,
|
|
windowHeight - PIP_PADDING - PIP_HEIGHT
|
|
),
|
|
];
|
|
case PositionMode.SnapToRight:
|
|
return [
|
|
windowWidth - PIP_PADDING - PIP_WIDTH,
|
|
Math.min(
|
|
PIP_TOP_MARGIN + 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);
|
|
}
|
|
}, [windowWidth, windowHeight, positionState]);
|
|
|
|
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;
|
|
}
|
|
const rect = node.getBoundingClientRect();
|
|
const dragOffsetX = ev.screenX - rect.left;
|
|
const dragOffsetY = ev.screenY - rect.top;
|
|
|
|
setPositionState({
|
|
mode: PositionMode.BeingDragged,
|
|
mouseX: ev.screenX,
|
|
mouseY: ev.screenY,
|
|
dragOffsetX,
|
|
dragOffsetY,
|
|
});
|
|
}}
|
|
ref={videoContainerRef}
|
|
style={{
|
|
cursor:
|
|
positionState.mode === PositionMode.BeingDragged
|
|
? '-webkit-grabbing'
|
|
: '-webkit-grab',
|
|
transform: `translate3d(${translateX}px,${translateY}px, 0)`,
|
|
transition:
|
|
positionState.mode === PositionMode.BeingDragged
|
|
? 'none'
|
|
: 'transform ease-out 300ms',
|
|
}}
|
|
>
|
|
<CallingPipRemoteVideo
|
|
activeCall={activeCall}
|
|
getGroupCallVideoFrameSource={getGroupCallVideoFrameSource}
|
|
i18n={i18n}
|
|
setRendererCanvas={setRendererCanvas}
|
|
setGroupCallVideoRequest={setGroupCallVideoRequest}
|
|
/>
|
|
{hasLocalVideo ? (
|
|
<video
|
|
className="module-calling-pip__video--local"
|
|
ref={localVideoRef}
|
|
autoPlay
|
|
/>
|
|
) : null}
|
|
<div className="module-calling-pip__actions">
|
|
<button
|
|
aria-label={i18n('calling__hangup')}
|
|
className="module-calling-pip__button--hangup"
|
|
onClick={() => {
|
|
hangUp({ conversationId: activeCall.conversation.id });
|
|
}}
|
|
type="button"
|
|
/>
|
|
<button
|
|
aria-label={i18n('calling__pip--off')}
|
|
className="module-calling-pip__button--pip"
|
|
onClick={togglePip}
|
|
type="button"
|
|
>
|
|
<div />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|