// Copyright 2020-2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useState, useRef, useMemo, useCallback, useEffect, CSSProperties, } from 'react'; import classNames from 'classnames'; import { noop } from 'lodash'; import { GroupCallRemoteParticipantType, VideoFrameSource, } from '../types/Calling'; import { LocalizerType } from '../types/Util'; import { AvatarColors } from '../types/Colors'; import { CallBackgroundBlur } from './CallBackgroundBlur'; import { Avatar, AvatarSize } from './Avatar'; import { ConfirmationDialog } from './ConfirmationDialog'; import { Intl } from './Intl'; import { ContactName } from './conversation/ContactName'; import { useIntersectionObserver } from '../util/hooks'; import { MAX_FRAME_SIZE } from '../calling/constants'; const MAX_TIME_TO_SHOW_STALE_VIDEO_FRAMES = 5000; type BasePropsType = { getFrameBuffer: () => ArrayBuffer; getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource; i18n: LocalizerType; remoteParticipant: GroupCallRemoteParticipantType; }; type InPipPropsType = { isInPip: true; }; type InOverflowAreaPropsType = { height: number; isInPip?: false; width: number; }; type InGridPropsType = InOverflowAreaPropsType & { left: number; top: number; }; export type PropsType = BasePropsType & (InPipPropsType | InOverflowAreaPropsType | InGridPropsType); export const GroupCallRemoteParticipant: React.FC = React.memo( props => { const { getFrameBuffer, getGroupCallVideoFrameSource, i18n } = props; const { acceptedMessageRequest, avatarPath, color, demuxId, hasRemoteAudio, hasRemoteVideo, isBlocked, isMe, profileName, sharedGroupNames, title, videoAspectRatio, } = props.remoteParticipant; const [hasReceivedVideoRecently, setHasReceivedVideoRecently] = useState( false ); const [isWide, setIsWide] = useState( videoAspectRatio ? videoAspectRatio >= 1 : true ); const [hasHover, setHover] = useState(false); const [showBlockInfo, setShowBlockInfo] = useState(false); // We have some state (`hasReceivedVideoRecently`) and this ref. We can't have a // single state value like `lastReceivedVideoAt` because (1) it won't automatically // trigger a re-render after the video has become stale (2) it would cause a full // re-render of the component for every frame, which is way too often. // // Alternatively, we could create a timeout that's reset every time we get a video // frame (perhaps using a debounce function), but that becomes harder to clean up // when the component unmounts. const lastReceivedVideoAt = useRef(-Infinity); const remoteVideoRef = useRef(null); const canvasContextRef = useRef(null); const [ intersectionRef, intersectionObserverEntry, ] = useIntersectionObserver(); const isVisible = intersectionObserverEntry ? intersectionObserverEntry.isIntersecting : true; const wantsToShowVideo = hasRemoteVideo && !isBlocked && isVisible; const hasVideoToShow = wantsToShowVideo && hasReceivedVideoRecently; const videoFrameSource = useMemo( () => getGroupCallVideoFrameSource(demuxId), [getGroupCallVideoFrameSource, demuxId] ); const renderVideoFrame = useCallback(() => { if ( Date.now() - lastReceivedVideoAt.current > MAX_TIME_TO_SHOW_STALE_VIDEO_FRAMES ) { setHasReceivedVideoRecently(false); } const canvasEl = remoteVideoRef.current; if (!canvasEl) { return; } const canvasContext = canvasContextRef.current; if (!canvasContext) { return; } // This frame buffer is shared by all participants, so it may contain pixel data // for other participants, or pixel data from a previous frame. That's why we // return early and use the `frameWidth` and `frameHeight`. const frameBuffer = getFrameBuffer(); const frameDimensions = videoFrameSource.receiveVideoFrame( Buffer.from(frameBuffer) ); if (!frameDimensions) { return; } const [frameWidth, frameHeight] = frameDimensions; if ( frameWidth < 2 || frameHeight < 2 || frameWidth * frameHeight > MAX_FRAME_SIZE ) { return; } canvasEl.width = frameWidth; canvasEl.height = frameHeight; canvasContext.putImageData( new ImageData( new Uint8ClampedArray(frameBuffer, 0, frameWidth * frameHeight * 4), frameWidth, frameHeight ), 0, 0 ); lastReceivedVideoAt.current = Date.now(); setHasReceivedVideoRecently(true); setIsWide(frameWidth > frameHeight); }, [getFrameBuffer, videoFrameSource]); useEffect(() => { if (!hasRemoteVideo) { setHasReceivedVideoRecently(false); } }, [hasRemoteVideo]); useEffect(() => { if (!hasRemoteVideo || !isVisible) { return noop; } let rafId = requestAnimationFrame(tick); function tick() { renderVideoFrame(); rafId = requestAnimationFrame(tick); } return () => { cancelAnimationFrame(rafId); }; }, [hasRemoteVideo, isVisible, renderVideoFrame, videoFrameSource]); let canvasStyles: CSSProperties; let containerStyles: CSSProperties; // 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 (isWide) { canvasStyles = { width: '100%' }; } else { canvasStyles = { height: '100%' }; } let avatarSize: number; // 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; avatarSize = AvatarSize.FIFTY_TWO; } else { const { width, height } = props; const shorterDimension = Math.min(width, height); if (shorterDimension >= 240) { avatarSize = AvatarSize.ONE_HUNDRED_TWELVE; } else if (shorterDimension >= 180) { avatarSize = AvatarSize.EIGHTY; } else { avatarSize = AvatarSize.FIFTY_TWO; } containerStyles = { height, width, }; if ('top' in props) { containerStyles.position = 'absolute'; containerStyles.top = props.top; containerStyles.left = props.left; } } const showHover = hasHover && !props.isInPip; return ( <> {showBlockInfo && ( { setShowBlockInfo(false); }} title={
, ]} />
} > {i18n('calling__block-info')}
)}
setHover(true)} onMouseLeave={() => setHover(false)} style={containerStyles} > {showHover && (
)} {wantsToShowVideo && ( { remoteVideoRef.current = canvasEl; if (canvasEl) { canvasContextRef.current = canvasEl.getContext('2d', { alpha: false, desynchronized: true, storage: 'discardable', } as CanvasRenderingContext2DSettings); } else { canvasContextRef.current = null; } }} /> )} {!hasVideoToShow && ( {isBlocked ? ( <> ) : ( )} )}
); } );