// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useCallback, useState, useMemo, useEffect } from 'react'; import { clamp, chunk, maxBy, flatten, noop } from 'lodash'; import type { VideoFrameSource } from '@signalapp/ringrtc'; import { GroupCallRemoteParticipant } from './GroupCallRemoteParticipant'; import { GroupCallOverflowArea, OVERFLOW_PARTICIPANT_WIDTH, } from './GroupCallOverflowArea'; import type { GroupCallRemoteParticipantType, GroupCallVideoRequest, } from '../types/Calling'; import { CallViewMode } from '../types/Calling'; import { useGetCallingFrameBuffer } from '../calling/useGetCallingFrameBuffer'; import type { LocalizerType } from '../types/Util'; import { usePageVisibility } from '../hooks/usePageVisibility'; import { useDevicePixelRatio } from '../hooks/useDevicePixelRatio'; import { nonRenderedRemoteParticipant } from '../util/ringrtc/nonRenderedRemoteParticipant'; import { missingCaseError } from '../util/missingCaseError'; import { SECOND } from '../util/durations'; import { filter, join } from '../util/iterables'; import * as setUtil from '../util/setUtil'; import * as log from '../logging/log'; import { MAX_FRAME_HEIGHT, MAX_FRAME_WIDTH } from '../calling/constants'; import { SizeObserver } from '../hooks/useSizeObserver'; import { strictAssert } from '../util/assert'; import type { CallingImageDataCache } from './CallManager'; const SMALL_TILES_MIN_HEIGHT = 80; const LARGE_TILES_MIN_HEIGHT = 200; const PARTICIPANT_MARGIN = 12; const TIME_TO_STOP_REQUESTING_VIDEO_WHEN_PAGE_INVISIBLE = 20 * SECOND; const PAGINATION_BUTTON_ASPECT_RATIO = 1; const MAX_PARTICIPANTS_PER_PAGE = 49; // 49 remote + 1 self-video = 50 total // We scale our video requests down for performance. This number is somewhat arbitrary. const VIDEO_REQUEST_SCALAR = 0.75; type Dimensions = { width: number; height: number; }; type GridArrangement = { rows: Array>; scalar: number; }; type PaginationButtonType = { isPaginationButton: true; videoAspectRatio: number; paginationButtonType: 'prev' | 'next'; numParticipants: number; }; type ParticipantTileType = | GroupCallRemoteParticipantType | PaginationButtonType; type PropsType = { callViewMode: CallViewMode; getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource; i18n: LocalizerType; imageDataCache: React.RefObject; isCallReconnecting: boolean; joinedAt: number | null; remoteParticipants: ReadonlyArray; setGroupCallVideoRequest: ( _: Array, speakerHeight: number ) => void; remoteAudioLevels: Map; onClickRaisedHand?: () => void; }; enum VideoRequestMode { Normal = 'Normal', LowResolution = 'LowResolution', NoVideo = 'NoVideo', } // This component lays out group call remote participants. It uses a custom layout // algorithm (in other words, nothing that the browser provides, like flexbox) in // order to animate the boxes as they move around, and to figure out the right fits. // // It's worth looking at the UI (or a design of it) to get an idea of how it works. Some // things to notice: // // * Participants are arranged in 0 or more rows. // * Each row is the same height, but each participant may have a different width. // * It's possible, on small screens with lots of participants, to have participants // removed from the grid, or on subsequent pages. This is because participants // have a minimum rendered height. // * Participant videos may have different aspect ratios // * We want to ensure that presenters and recent speakers are shown on the first page, // but we also want to minimize tiles jumping around as much as possible. // // There should be more specific comments throughout, but the high-level steps are: // // 1. Figure out the maximum number of possible rows that could fit on a page; this is // `maxRowsPerPage`. // 2. Sort the participants in priority order: we want to fit presenters and recent // speakers in the grid first // 3. Figure out which participants should go on each page -- for non-paginated views, // this is just one page, but for paginated views, we could have many pages. The // general idea here is to fill up each page row-by-row, with each video as small // as we allow. // 4. Try to distribute the videos throughout the grid to find the largest "scalar": // how much can we scale these boxes up while still fitting them on the screen? // The biggest scalar wins as the "best arrangement". // 5. Lay out this arrangement on the screen. export function GroupCallRemoteParticipants({ callViewMode, getGroupCallVideoFrameSource, imageDataCache, i18n, isCallReconnecting, joinedAt, remoteParticipants, setGroupCallVideoRequest, remoteAudioLevels, onClickRaisedHand, }: PropsType): JSX.Element { const [gridDimensions, setGridDimensions] = useState({ width: 0, height: 0, }); const [pageIndex, setPageIndex] = useState(0); const devicePixelRatio = useDevicePixelRatio(); const getFrameBuffer = useGetCallingFrameBuffer(); const { invisibleDemuxIds, onParticipantVisibilityChanged } = useInvisibleParticipants(remoteParticipants); const minRenderedHeight = callViewMode === CallViewMode.Paginated ? SMALL_TILES_MIN_HEIGHT : LARGE_TILES_MIN_HEIGHT; const isInSpeakerView = callViewMode === CallViewMode.Speaker || callViewMode === CallViewMode.Presentation; const isInPaginationView = callViewMode === CallViewMode.Paginated; const shouldShowOverflow = !isInPaginationView; const maxRowWidth = gridDimensions.width; const maxGridHeight = gridDimensions.height; // 1. Figure out the maximum number of possible rows that could fit on the page. // Could be 0 if (a) there are no participants (b) the container's height is small. const maxRowsPerPage = Math.floor( maxGridHeight / (minRenderedHeight + PARTICIPANT_MARGIN) ); // 2. Sort the participants in priority order: by `presenting` first, since presenters // should be on the main grid, then by `speakerTime` so that the most recent speakers // are next in line for the first pages of the grid const prioritySortedParticipants: Array = useMemo( () => remoteParticipants .concat() .sort( (a, b) => Number(b.presenting || 0) - Number(a.presenting || 0) || (b.speakerTime || -Infinity) - (a.speakerTime || -Infinity) ), [remoteParticipants] ); // 3. Layout the participants on each page. The general algorithm is: first, try to fill // up each page with as many participants as possible at the smallest acceptable video // height. Second, sort the participants that fit on each page by a stable sort order, // and make sure they still fit on the page! Third, add tiles at the beginning and end // of each page (if paginated) to act as back and next buttons. const gridParticipantsByPage: Array = useMemo(() => { if (!prioritySortedParticipants.length) { return []; } if (!maxRowsPerPage) { return []; } if (isInSpeakerView) { return [ { rows: [[prioritySortedParticipants[0]]], hasSpaceRemaining: false, numParticipants: 1, }, ]; } return getGridParticipantsByPage({ participants: prioritySortedParticipants, maxRowWidth, maxPages: isInPaginationView ? Infinity : 1, maxRowsPerPage, minRenderedHeight, maxParticipantsPerPage: MAX_PARTICIPANTS_PER_PAGE, currentPage: pageIndex, }); }, [ maxRowWidth, isInPaginationView, isInSpeakerView, maxRowsPerPage, minRenderedHeight, pageIndex, prioritySortedParticipants, ]); // Make sure we're not on a page that no longer exists (e.g. if people left the call) if ( pageIndex >= gridParticipantsByPage.length && gridParticipantsByPage.length > 0 ) { setPageIndex(gridParticipantsByPage.length - 1); } const totalParticipantsInGrid = gridParticipantsByPage.reduce( (pageCount, { numParticipants }) => pageCount + numParticipants, 0 ); // In speaker or overflow views, not all participants will be on the grid; they'll // get put in the overflow zone. const overflowedParticipants: Array = useMemo( () => isInPaginationView ? [] : prioritySortedParticipants .slice(totalParticipantsInGrid) .sort(stableParticipantComparator), [isInPaginationView, prioritySortedParticipants, totalParticipantsInGrid] ); const participantsOnOtherPages = useMemo( () => gridParticipantsByPage .map((page, index) => { if (index === pageIndex) { return []; } return page.rows.flat(); }) .flat() .filter(isGroupCallRemoteParticipant), [gridParticipantsByPage, pageIndex] ); const currentPage = gridParticipantsByPage.at(pageIndex) ?? { rows: [], }; // 4. Try to arrange the current page such that we can scale the videos up // as much as possible. const gridArrangement = arrangeParticipantsInGrid({ participantsInRows: currentPage.rows, maxRowsPerPage, maxRowWidth, maxGridHeight, minRenderedHeight, }); const nextPage = () => { setPageIndex(index => index + 1); }; const prevPage = () => { setPageIndex(index => Math.max(0, index - 1)); }; // 5. Lay out the current page on the screen. const gridParticipantHeight = Math.round( gridArrangement.scalar * minRenderedHeight ); const gridParticipantHeightWithMargin = gridParticipantHeight + PARTICIPANT_MARGIN; const gridTotalRowHeightWithMargin = gridParticipantHeightWithMargin * gridArrangement.rows.length - PARTICIPANT_MARGIN; const gridTopOffset = Math.max( 0, Math.round((gridDimensions.height - gridTotalRowHeightWithMargin) / 2) ); const rowElements: Array> = gridArrangement.rows.map( (tiles, index) => { const top = gridTopOffset + index * gridParticipantHeightWithMargin; const totalRowWidthWithoutMargins = totalRowWidthAtHeight(tiles, minRenderedHeight) * gridArrangement.scalar; const totalRowWidth = totalRowWidthWithoutMargins + PARTICIPANT_MARGIN * (tiles.length - 1); const leftOffset = Math.max( 0, Math.round((gridDimensions.width - totalRowWidth) / 2) ); let rowWidthSoFar = 0; return tiles.map(tile => { const left = rowWidthSoFar + leftOffset; const renderedWidth = Math.round( tile.videoAspectRatio * gridParticipantHeight ); rowWidthSoFar += renderedWidth + PARTICIPANT_MARGIN; if (isPaginationButton(tile)) { const isNextButton = tile.paginationButtonType === 'next'; const isPrevButton = tile.paginationButtonType === 'prev'; return ( ); } return ( ); }); } ); const videoRequestMode = useVideoRequestMode(); useEffect(() => { log.info(`Group call now using ${videoRequestMode} video request mode`); }, [videoRequestMode]); useEffect(() => { let videoRequest: Array; switch (videoRequestMode) { case VideoRequestMode.Normal: videoRequest = [ ...currentPage.rows .flat() // Filter out any next/previous page buttons .filter(isGroupCallRemoteParticipant) .map(participant => { let scalar: number; if (participant.sharingScreen) { // We want best-resolution video if someone is sharing their screen. scalar = Math.max(devicePixelRatio, 1); } else { scalar = VIDEO_REQUEST_SCALAR; } return { demuxId: participant.demuxId, width: clamp( Math.round( gridParticipantHeight * participant.videoAspectRatio * scalar ), 1, MAX_FRAME_WIDTH ), height: clamp( Math.round(gridParticipantHeight * scalar), 1, MAX_FRAME_HEIGHT ), }; }), ...participantsOnOtherPages.map(nonRenderedRemoteParticipant), ...overflowedParticipants.map(participant => { if (invisibleDemuxIds.has(participant.demuxId)) { return nonRenderedRemoteParticipant(participant); } return { demuxId: participant.demuxId, width: clamp( Math.round(OVERFLOW_PARTICIPANT_WIDTH * VIDEO_REQUEST_SCALAR), 1, MAX_FRAME_WIDTH ), height: clamp( Math.round( (OVERFLOW_PARTICIPANT_WIDTH / participant.videoAspectRatio) * VIDEO_REQUEST_SCALAR ), 1, MAX_FRAME_HEIGHT ), }; }), ]; break; case VideoRequestMode.LowResolution: videoRequest = remoteParticipants.map(participant => participant.hasRemoteVideo ? { demuxId: participant.demuxId, width: 1, height: 1, } : nonRenderedRemoteParticipant(participant) ); break; case VideoRequestMode.NoVideo: videoRequest = remoteParticipants.map(nonRenderedRemoteParticipant); break; default: log.error(missingCaseError(videoRequestMode)); videoRequest = remoteParticipants.map(nonRenderedRemoteParticipant); break; } setGroupCallVideoRequest( videoRequest, clamp(gridParticipantHeight, 0, MAX_FRAME_HEIGHT) ); }, [ devicePixelRatio, currentPage.rows, gridParticipantHeight, invisibleDemuxIds, overflowedParticipants, remoteParticipants, setGroupCallVideoRequest, videoRequestMode, participantsOnOtherPages, ]); return (
{ setGridDimensions(size); }} > {gridRef => (
{flatten(rowElements)} {isInPaginationView && ( <> {pageIndex > 0 ? ( ) : null} {pageIndex < gridParticipantsByPage.length - 1 ? ( ) : null} )}
)}
{shouldShowOverflow && overflowedParticipants.length > 0 ? ( ) : null}
); } // This function is only meant for use with `useInvisibleParticipants`. It helps avoid // returning new set instances when the underlying values are equal. function pickDifferentSet(a: Readonly>, b: Readonly>): Set { return a.size === b.size ? a : b; } function useInvisibleParticipants( remoteParticipants: ReadonlyArray ): Readonly<{ invisibleDemuxIds: Set; onParticipantVisibilityChanged: (demuxId: number, isVisible: boolean) => void; }> { const [invisibleDemuxIds, setInvisibleDemuxIds] = useState(new Set()); const onParticipantVisibilityChanged = useCallback( (demuxId: number, isVisible: boolean) => { setInvisibleDemuxIds(oldInvisibleDemuxIds => { const toggled = setUtil.toggle( oldInvisibleDemuxIds, demuxId, !isVisible ); return pickDifferentSet(oldInvisibleDemuxIds, toggled); }); }, [] ); useEffect(() => { log.info( `Invisible demux IDs changed to [${join(invisibleDemuxIds, ',')}]` ); }, [invisibleDemuxIds]); useEffect(() => { const remoteParticipantDemuxIds = new Set( remoteParticipants.map(r => r.demuxId) ); setInvisibleDemuxIds(oldInvisibleDemuxIds => { const staleIds = filter( oldInvisibleDemuxIds, id => !remoteParticipantDemuxIds.has(id) ); const withoutStaleIds = setUtil.remove(oldInvisibleDemuxIds, ...staleIds); return pickDifferentSet(oldInvisibleDemuxIds, withoutStaleIds); }); }, [remoteParticipants]); return { invisibleDemuxIds, onParticipantVisibilityChanged, }; } function useVideoRequestMode(): VideoRequestMode { const isPageVisible = usePageVisibility(); const [result, setResult] = useState( isPageVisible ? VideoRequestMode.Normal : VideoRequestMode.LowResolution ); useEffect(() => { if (isPageVisible) { setResult(VideoRequestMode.Normal); return noop; } setResult(VideoRequestMode.LowResolution); const timeout = setTimeout(() => { setResult(VideoRequestMode.NoVideo); }, TIME_TO_STOP_REQUESTING_VIDEO_WHEN_PAGE_INVISIBLE); return () => { clearTimeout(timeout); }; }, [isPageVisible]); return result; } function totalRowWidthAtHeight( participantsInRow: ReadonlyArray< Pick >, height: number ): number { return participantsInRow.reduce( (result, participant) => result + participantWidthAtHeight(participant, height), 0 ); } function participantWidthAtHeight( participant: Pick, height: number ) { return participant.videoAspectRatio * height; } function stableParticipantComparator( a: Readonly<{ demuxId: number }>, b: Readonly<{ demuxId: number }> ): number { return a.demuxId - b.demuxId; } type ParticipantsInPageType< T extends { videoAspectRatio: number } = ParticipantTileType, > = { rows: Array>; numParticipants: number; }; type PageLayoutPropsType = { maxRowWidth: number; minRenderedHeight: number; maxRowsPerPage: number; maxParticipantsPerPage: number; }; function getGridParticipantsByPage({ participants, maxPages, currentPage, ...pageLayoutProps }: PageLayoutPropsType & { participants: Array; maxPages: number; currentPage?: number; }): Array { if (!participants.length) { return []; } const pages: Array = []; function getTotalParticipantsOnGrid() { return pages.reduce((count, page) => count + page.numParticipants, 0); } let remainingParticipants = [...participants]; while (remainingParticipants.length) { if (currentPage === pages.length - 1) { // Optimization: we can stop early, we don't have to lay out the remainder of the // pages pages.push({ rows: [remainingParticipants], numParticipants: remainingParticipants.length, }); return pages; } const nextPageInPriorityOrder = getNextPage({ participants: remainingParticipants, leaveRoomForPrevPageButton: pages.length > 0, leaveRoomForNextPageButton: pages.length + 1 < maxPages, ...pageLayoutProps, }); // We got the next page, but it's in priority order; let's see if these participants // also fit in sorted order const priorityParticipantsOnNextPage = nextPageInPriorityOrder.rows.flat(); let sortedParticipantsHopingToFitOnPage = [ ...priorityParticipantsOnNextPage, ].sort(stableParticipantComparator); let nextPageInSortedOrder = getNextPage({ participants: sortedParticipantsHopingToFitOnPage, leaveRoomForPrevPageButton: pages.length > 0, leaveRoomForNextPageButton: pages.length + 1 < maxPages, isSubsetOfAllParticipants: sortedParticipantsHopingToFitOnPage.length < remainingParticipants.length, ...pageLayoutProps, }); let nextPage: ParticipantsInPageType | undefined; if ( nextPageInSortedOrder.numParticipants === nextPageInPriorityOrder.numParticipants ) { // Great, we're able to show everyone. It's possible that there is now extra space // and we could show more people, but let's leave it here for simplicity nextPage = nextPageInSortedOrder; } else { // We weren't able to fit everyone. Let's remove the least-prioritized person and // try again. It's pretty unlikely this will take more than 1 attempt, but // let's take more and more participants off the screen if it takes a lot of // attempts so we don't have to iterate dozens of times. const PARTICIPANTS_TO_REMOVE_PER_ATTEMPT = [1, 1, 1, 2, 5]; const MAX_ATTEMPTS = 5; let attemptNumber = 0; while ( sortedParticipantsHopingToFitOnPage.length && attemptNumber < MAX_ATTEMPTS ) { const numLeastPrioritizedParticipantsToRemove = PARTICIPANTS_TO_REMOVE_PER_ATTEMPT[ Math.min( attemptNumber, PARTICIPANTS_TO_REMOVE_PER_ATTEMPT.length - 1 ) ]; const leastPrioritizedParticipantIds = new Set( priorityParticipantsOnNextPage .splice( -1 * numLeastPrioritizedParticipantsToRemove, numLeastPrioritizedParticipantsToRemove ) .map(participant => participant.demuxId) ); sortedParticipantsHopingToFitOnPage = sortedParticipantsHopingToFitOnPage.filter( participant => !leastPrioritizedParticipantIds.has(participant.demuxId) ); nextPageInSortedOrder = getNextPage({ participants: sortedParticipantsHopingToFitOnPage, leaveRoomForPrevPageButton: pages.length > 0, leaveRoomForNextPageButton: pages.length + 1 < maxPages, ...pageLayoutProps, }); // Are we able to fill all of them now? Great, let's ship it. if ( nextPageInSortedOrder.numParticipants === sortedParticipantsHopingToFitOnPage.length ) { nextPage = nextPageInSortedOrder; break; } attemptNumber += 1; } if (!nextPage) { log.warn( `GroupCallRemoteParticipants: failed after ${attemptNumber} attempts to layout the page; pageIndex: ${pages.length}, \ # fit in priority order: ${nextPageInPriorityOrder.numParticipants}, \ # fit in sorted order: ${nextPageInSortedOrder.numParticipants}` ); nextPage = nextPageInSortedOrder; } } if (!nextPage) { break; } const nextPageTiles = nextPage as ParticipantsInPageType; // Add a previous page tile if needed if (pages.length > 0) { nextPageTiles.rows[0].unshift({ isPaginationButton: true, paginationButtonType: 'prev', videoAspectRatio: PAGINATION_BUTTON_ASPECT_RATIO, numParticipants: getTotalParticipantsOnGrid(), }); } if (!nextPage.numParticipants) { break; } remainingParticipants = remainingParticipants.slice( nextPage.numParticipants ); pages.push(nextPage); if (pages.length === maxPages) { break; } // Add a next page tile if needed if (remainingParticipants.length) { nextPageTiles.rows.at(-1)?.push({ isPaginationButton: true, paginationButtonType: 'next', videoAspectRatio: PAGINATION_BUTTON_ASPECT_RATIO, numParticipants: remainingParticipants.length, }); } } return pages; } /** * Attempt to fill a new page with as many participants as will fit, leaving room for * next/prev page buttons as needed. Participants will be added in the order provided. * * @returns ParticipantsInPageType, representing the participants that fit on this page * assuming they are rendered at minimum height. Does not include prev/next buttons, * but will leave space for them if needed. Participants are not necessarily * returned in the row-distribution that will maximize video scaling; that should * be done subsequently. */ function getNextPage({ participants, maxRowWidth, minRenderedHeight, maxRowsPerPage, maxParticipantsPerPage, leaveRoomForPrevPageButton, leaveRoomForNextPageButton, isSubsetOfAllParticipants, }: PageLayoutPropsType & { participants: Array; leaveRoomForPrevPageButton: boolean; leaveRoomForNextPageButton: boolean; isSubsetOfAllParticipants?: boolean; }): ParticipantsInPageType { const paginationButtonWidth = participantWidthAtHeight( { videoAspectRatio: PAGINATION_BUTTON_ASPECT_RATIO, }, minRenderedHeight ); let rowWidth = leaveRoomForPrevPageButton ? paginationButtonWidth + PARTICIPANT_MARGIN : 0; // Initialize fresh page with empty first row const rows: Array> = [[]]; let row = rows[0]; let numParticipants = 0; // Start looping through participants and adding them to the rows one-by-one for (let i = 0; i < participants.length; i += 1) { const participant = participants[i]; const isLastParticipant = !isSubsetOfAllParticipants && i === participants.length - 1; const participantWidth = participantWidthAtHeight( participant, minRenderedHeight ); const isLastRow = rows.length === maxRowsPerPage; const shouldShowNextButtonInThisRow = isLastRow && !isLastParticipant && leaveRoomForNextPageButton; const currentRowMaxWidth = shouldShowNextButtonInThisRow ? maxRowWidth - (paginationButtonWidth + PARTICIPANT_MARGIN) : maxRowWidth; const participantFitsOnRow = rowWidth + participantWidth + (row.length ? PARTICIPANT_MARGIN : 0) <= currentRowMaxWidth; if (participantFitsOnRow) { rowWidth += participantWidth + (row.length ? PARTICIPANT_MARGIN : 0); row.push(participant); numParticipants += 1; if (numParticipants === maxParticipantsPerPage) { return { rows, numParticipants }; } } else { if (isLastRow) { return { rows, numParticipants }; } // Start a new row! row = [participant]; rows.push(row); numParticipants += 1; rowWidth = participantWidth; } } return { rows, numParticipants, }; } /** * Given an arrangement of participants in rows that we know fits on a page at minimum * rendered height, try to find an arrangement that maximizes video size, or return the * provided arrangement with maximal video size. The result of this is ready to be * laid out on the screen. * * @returns GridArrangement: { * rows: participants in rows, * scalar: the scalar by which can scale every video on the page and still fit * } */ function arrangeParticipantsInGrid({ participantsInRows, maxRowWidth, minRenderedHeight, maxRowsPerPage, maxGridHeight, }: { participantsInRows: Array>; maxRowWidth: number; minRenderedHeight: number; maxRowsPerPage: number; maxGridHeight: number; }): GridArrangement { // Start out with the arrangement that was prepared by getGridParticipantsByPage. // We know this arrangement (added one-by-one) fits, so its scalar is // guaranteed to be >= 1. Our chunking strategy below might not arrive at such // an arrangement. let bestArrangement: GridArrangement = { scalar: getMaximumScaleForRows({ maxRowWidth, minRenderedHeight, rows: participantsInRows, maxGridHeight, }), rows: participantsInRows, }; const participants = participantsInRows.flat(); // For each possible number of rows (starting at 0 and ending at `maxRowCount`), // distribute participants across the rows at the minimum height. Then find the // "scalar": how much can we scale these boxes up while still fitting them on the // screen? The biggest scalar wins as the "best arrangement". for (let rowCount = 1; rowCount <= maxRowsPerPage; rowCount += 1) { // We do something pretty naïve here and chunk the grid's participants into rows. // For example, if there were 12 grid participants and `rowCount === 3`, there // would be 4 participants per row. // // This naïve chunking is suboptimal in terms of absolute best fit, but it is much // faster and simpler than trying to do this perfectly. In practice, this works // fine in the UI from our testing. const numberOfParticipantsInRow = Math.ceil(participants.length / rowCount); const rows = chunk(participants, numberOfParticipantsInRow); const scalar = getMaximumScaleForRows({ maxRowWidth, minRenderedHeight, rows, maxGridHeight, }); if (scalar > bestArrangement.scalar) { bestArrangement = { scalar, rows, }; } } return bestArrangement; } // We need to find the scalar for this arrangement. Imagine that we have these // participants at the minimum heights, and we want to scale everything up until // it's about to overflow. function getMaximumScaleForRows({ maxRowWidth, minRenderedHeight, maxGridHeight, rows, }: { maxRowWidth: number; minRenderedHeight: number; maxGridHeight: number; rows: Array>; }): number { if (!rows.length) { return 0; } const widestRow = maxBy(rows, x => totalRowWidthAtHeight(x, minRenderedHeight) ); strictAssert(widestRow, 'Could not find widestRow'); // We don't want it to overflow horizontally or vertically, so we calculate a // "width scalar" and "height scalar" and choose the smaller of the two. (Choosing // the LARGER of the two could cause overflow.) const widthScalar = (maxRowWidth - (widestRow.length - 1) * PARTICIPANT_MARGIN) / totalRowWidthAtHeight(widestRow, minRenderedHeight); const heightScalar = (maxGridHeight - (rows.length - 1) * PARTICIPANT_MARGIN) / (rows.length * minRenderedHeight); return Math.min(widthScalar, heightScalar); } function isGroupCallRemoteParticipant( tile: ParticipantTileType ): tile is GroupCallRemoteParticipantType { return 'demuxId' in tile; } function isPaginationButton( tile: ParticipantTileType ): tile is PaginationButtonType { return 'isPaginationButton' in tile; }