2020-11-13 19:57:55 +00:00
|
|
|
// Copyright 2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2020-11-17 19:49:48 +00:00
|
|
|
import React, {
|
|
|
|
useState,
|
|
|
|
useRef,
|
|
|
|
useMemo,
|
|
|
|
useCallback,
|
|
|
|
useEffect,
|
|
|
|
CSSProperties,
|
|
|
|
} from 'react';
|
2020-11-16 19:57:58 +00:00
|
|
|
import classNames from 'classnames';
|
2020-11-13 19:57:55 +00:00
|
|
|
import { noop } from 'lodash';
|
2020-11-17 19:49:48 +00:00
|
|
|
import { VideoFrameSource } from '../types/Calling';
|
2020-11-13 19:57:55 +00:00
|
|
|
import { CallBackgroundBlur } from './CallBackgroundBlur';
|
|
|
|
|
2020-11-17 19:49:48 +00:00
|
|
|
// The max size video frame we'll support (in RGBA)
|
|
|
|
const FRAME_BUFFER_SIZE = 1920 * 1080 * 4;
|
|
|
|
|
|
|
|
interface BasePropsType {
|
2020-11-13 19:57:55 +00:00
|
|
|
demuxId: number;
|
|
|
|
getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource;
|
2020-11-16 19:57:58 +00:00
|
|
|
hasRemoteAudio: boolean;
|
2020-11-13 19:57:55 +00:00
|
|
|
hasRemoteVideo: boolean;
|
2020-11-17 19:49:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface InPipPropsType {
|
|
|
|
isInPip: true;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface NotInPipPropsType {
|
|
|
|
isInPip?: false;
|
|
|
|
width: number;
|
|
|
|
height: number;
|
2020-11-13 19:57:55 +00:00
|
|
|
left: number;
|
|
|
|
top: number;
|
|
|
|
}
|
|
|
|
|
2020-11-17 19:49:48 +00:00
|
|
|
type PropsType = BasePropsType & (InPipPropsType | NotInPipPropsType);
|
|
|
|
|
|
|
|
export const GroupCallRemoteParticipant: React.FC<PropsType> = props => {
|
|
|
|
const {
|
|
|
|
demuxId,
|
|
|
|
getGroupCallVideoFrameSource,
|
|
|
|
hasRemoteAudio,
|
|
|
|
hasRemoteVideo,
|
|
|
|
} = props;
|
|
|
|
|
|
|
|
const [canvasStyles, setCanvasStyles] = useState<CSSProperties>({});
|
|
|
|
|
2020-11-13 19:57:55 +00:00
|
|
|
const remoteVideoRef = useRef<HTMLCanvasElement | null>(null);
|
2020-11-17 19:49:48 +00:00
|
|
|
const rafIdRef = useRef<number | null>(null);
|
|
|
|
const frameBufferRef = useRef<ArrayBuffer>(
|
|
|
|
new ArrayBuffer(FRAME_BUFFER_SIZE)
|
|
|
|
);
|
2020-11-13 19:57:55 +00:00
|
|
|
|
2020-11-17 19:49:48 +00:00
|
|
|
const videoFrameSource = useMemo(
|
|
|
|
() => getGroupCallVideoFrameSource(demuxId),
|
|
|
|
[getGroupCallVideoFrameSource, demuxId]
|
|
|
|
);
|
|
|
|
|
|
|
|
const renderVideoFrame = useCallback(() => {
|
|
|
|
const canvasEl = remoteVideoRef.current;
|
|
|
|
if (!canvasEl) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const context = canvasEl.getContext('2d');
|
|
|
|
if (!context) {
|
|
|
|
return;
|
2020-11-13 19:57:55 +00:00
|
|
|
}
|
|
|
|
|
2020-11-17 19:49:48 +00:00
|
|
|
const frameDimensions = videoFrameSource.receiveVideoFrame(
|
|
|
|
frameBufferRef.current
|
|
|
|
);
|
|
|
|
if (!frameDimensions) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const [frameWidth, frameHeight] = frameDimensions;
|
|
|
|
canvasEl.width = frameWidth;
|
|
|
|
canvasEl.height = frameHeight;
|
|
|
|
|
|
|
|
context.putImageData(
|
|
|
|
new ImageData(
|
|
|
|
new Uint8ClampedArray(
|
|
|
|
frameBufferRef.current,
|
|
|
|
0,
|
|
|
|
frameWidth * frameHeight * 4
|
|
|
|
),
|
|
|
|
frameWidth,
|
|
|
|
frameHeight
|
|
|
|
),
|
|
|
|
0,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
|
|
|
// If our `width` and `height` props don't match the canvas's aspect ratio, we want to
|
|
|
|
// fill the container. This can happen when RingRTC gives us an inaccurate
|
|
|
|
// `videoAspectRatio`, or if the container is an unexpected size.
|
|
|
|
if (frameWidth > frameHeight) {
|
|
|
|
setCanvasStyles({ width: '100%' });
|
2020-11-13 19:57:55 +00:00
|
|
|
} else {
|
2020-11-17 19:49:48 +00:00
|
|
|
setCanvasStyles({ height: '100%' });
|
|
|
|
}
|
|
|
|
}, [videoFrameSource]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!hasRemoteVideo) {
|
|
|
|
return noop;
|
2020-11-13 19:57:55 +00:00
|
|
|
}
|
2020-11-17 19:49:48 +00:00
|
|
|
|
|
|
|
const tick = () => {
|
|
|
|
renderVideoFrame();
|
|
|
|
rafIdRef.current = requestAnimationFrame(tick);
|
|
|
|
};
|
|
|
|
|
|
|
|
rafIdRef.current = requestAnimationFrame(tick);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
if (rafIdRef.current) {
|
|
|
|
cancelAnimationFrame(rafIdRef.current);
|
|
|
|
rafIdRef.current = null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, [hasRemoteVideo, renderVideoFrame, videoFrameSource]);
|
|
|
|
|
|
|
|
let containerStyles: CSSProperties;
|
|
|
|
|
|
|
|
// TypeScript isn't smart enough to know that `isInPip` by itself disambiguates the
|
|
|
|
// types, so we have to use `props.isInPip` instead.
|
|
|
|
// eslint-disable-next-line react/destructuring-assignment
|
|
|
|
if (props.isInPip) {
|
|
|
|
containerStyles = canvasStyles;
|
|
|
|
} else {
|
|
|
|
const { top, left, width, height } = props;
|
|
|
|
|
|
|
|
containerStyles = {
|
|
|
|
height,
|
|
|
|
left,
|
|
|
|
position: 'absolute',
|
|
|
|
top,
|
|
|
|
width,
|
|
|
|
};
|
2020-11-13 19:57:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
2020-11-16 19:57:58 +00:00
|
|
|
className={classNames(
|
|
|
|
'module-ongoing-call__group-call-remote-participant',
|
|
|
|
{
|
|
|
|
'module-ongoing-call__group-call-remote-participant--audio-muted': !hasRemoteAudio,
|
|
|
|
}
|
|
|
|
)}
|
2020-11-17 19:49:48 +00:00
|
|
|
style={containerStyles}
|
2020-11-13 19:57:55 +00:00
|
|
|
>
|
|
|
|
{hasRemoteVideo ? (
|
|
|
|
<canvas
|
|
|
|
className="module-ongoing-call__group-call-remote-participant__remote-video"
|
|
|
|
style={canvasStyles}
|
|
|
|
ref={remoteVideoRef}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<CallBackgroundBlur>
|
|
|
|
{/* TODO: Improve the styling here. See DESKTOP-894. */}
|
|
|
|
<span />
|
|
|
|
</CallBackgroundBlur>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|