| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2020 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  | import React, { useCallback, useState, useMemo, useEffect } from 'react'; | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | import { clamp, chunk, maxBy, flatten, noop } from 'lodash'; | 
					
						
							| 
									
										
										
										
											2023-01-09 10:38:57 -08:00
										 |  |  | import type { VideoFrameSource } from '@signalapp/ringrtc'; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | import { GroupCallRemoteParticipant } from './GroupCallRemoteParticipant'; | 
					
						
							| 
									
										
										
										
											2021-01-08 12:58:28 -06:00
										 |  |  | import { | 
					
						
							|  |  |  |   GroupCallOverflowArea, | 
					
						
							|  |  |  |   OVERFLOW_PARTICIPANT_WIDTH, | 
					
						
							|  |  |  | } from './GroupCallOverflowArea'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { | 
					
						
							| 
									
										
										
										
											2020-11-19 13:13:36 -05:00
										 |  |  |   GroupCallRemoteParticipantType, | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |   GroupCallVideoRequest, | 
					
						
							| 
									
										
										
										
											2020-11-19 13:13:36 -05:00
										 |  |  | } from '../types/Calling'; | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | import { CallViewMode } from '../types/Calling'; | 
					
						
							| 
									
										
										
										
											2021-01-08 11:32:49 -06:00
										 |  |  | import { useGetCallingFrameBuffer } from '../calling/useGetCallingFrameBuffer'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { LocalizerType } from '../types/Util'; | 
					
						
							| 
									
										
										
										
											2021-09-17 18:24:21 -04:00
										 |  |  | import { usePageVisibility } from '../hooks/usePageVisibility'; | 
					
						
							| 
									
										
										
										
											2022-01-06 16:00:11 -06:00
										 |  |  | import { useDevicePixelRatio } from '../hooks/useDevicePixelRatio'; | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  | import { nonRenderedRemoteParticipant } from '../util/ringrtc/nonRenderedRemoteParticipant'; | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  | import { missingCaseError } from '../util/missingCaseError'; | 
					
						
							|  |  |  | import { SECOND } from '../util/durations'; | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  | import { filter, join } from '../util/iterables'; | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  | import * as setUtil from '../util/setUtil'; | 
					
						
							| 
									
										
										
										
											2025-06-16 11:59:31 -07:00
										 |  |  | import { createLogger } from '../logging/log'; | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  | import { MAX_FRAME_HEIGHT, MAX_FRAME_WIDTH } from '../calling/constants'; | 
					
						
							| 
									
										
										
										
											2023-07-25 16:56:56 -07:00
										 |  |  | import { SizeObserver } from '../hooks/useSizeObserver'; | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | import { strictAssert } from '../util/assert'; | 
					
						
							| 
									
										
										
										
											2024-05-07 11:21:57 -07:00
										 |  |  | import type { CallingImageDataCache } from './CallManager'; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 11:59:31 -07:00
										 |  |  | const log = createLogger('GroupCallRemoteParticipants'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | const SMALL_TILES_MIN_HEIGHT = 80; | 
					
						
							|  |  |  | const LARGE_TILES_MIN_HEIGHT = 200; | 
					
						
							|  |  |  | const PARTICIPANT_MARGIN = 12; | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  | const TIME_TO_STOP_REQUESTING_VIDEO_WHEN_PAGE_INVISIBLE = 20 * SECOND; | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | const PAGINATION_BUTTON_ASPECT_RATIO = 1; | 
					
						
							|  |  |  | const MAX_PARTICIPANTS_PER_PAGE = 49; // 49 remote + 1 self-video = 50 total
 | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  | // We scale our video requests down for performance. This number is somewhat arbitrary.
 | 
					
						
							|  |  |  | const VIDEO_REQUEST_SCALAR = 0.75; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | type Dimensions = { | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   width: number; | 
					
						
							|  |  |  |   height: number; | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | type GridArrangement = { | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   rows: Array<Array<ParticipantTileType>>; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   scalar: number; | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | type PaginationButtonType = { | 
					
						
							|  |  |  |   isPaginationButton: true; | 
					
						
							|  |  |  |   videoAspectRatio: number; | 
					
						
							|  |  |  |   paginationButtonType: 'prev' | 'next'; | 
					
						
							|  |  |  |   numParticipants: number; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | type ParticipantTileType = | 
					
						
							|  |  |  |   | GroupCallRemoteParticipantType | 
					
						
							|  |  |  |   | PaginationButtonType; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | type PropsType = { | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   callViewMode: CallViewMode; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   getGroupCallVideoFrameSource: (demuxId: number) => VideoFrameSource; | 
					
						
							| 
									
										
										
										
											2020-11-19 13:13:36 -05:00
										 |  |  |   i18n: LocalizerType; | 
					
						
							| 
									
										
										
										
											2024-05-07 11:21:57 -07:00
										 |  |  |   imageDataCache: React.RefObject<CallingImageDataCache>; | 
					
						
							| 
									
										
										
										
											2023-10-16 13:58:51 -04:00
										 |  |  |   isCallReconnecting: boolean; | 
					
						
							| 
									
										
										
										
											2024-11-01 14:12:49 -07:00
										 |  |  |   joinedAt: number | null; | 
					
						
							| 
									
										
										
										
											2020-11-19 13:13:36 -05:00
										 |  |  |   remoteParticipants: ReadonlyArray<GroupCallRemoteParticipantType>; | 
					
						
							| 
									
										
										
										
											2022-09-07 08:52:55 -07:00
										 |  |  |   setGroupCallVideoRequest: ( | 
					
						
							|  |  |  |     _: Array<GroupCallVideoRequest>, | 
					
						
							|  |  |  |     speakerHeight: number | 
					
						
							|  |  |  |   ) => void; | 
					
						
							| 
									
										
										
										
											2022-05-18 20:28:51 -07:00
										 |  |  |   remoteAudioLevels: Map<number, number>; | 
					
						
							| 
									
										
										
										
											2023-12-11 07:10:31 -08:00
										 |  |  |   onClickRaisedHand?: () => void; | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  | enum VideoRequestMode { | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |   Normal = 'Normal', | 
					
						
							|  |  |  |   LowResolution = 'LowResolution', | 
					
						
							|  |  |  |   NoVideo = 'NoVideo', | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | // 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
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | //   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.
 | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | //
 | 
					
						
							|  |  |  | // There should be more specific comments throughout, but the high-level steps are:
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | // 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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function GroupCallRemoteParticipants({ | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   callViewMode, | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   getGroupCallVideoFrameSource, | 
					
						
							| 
									
										
										
										
											2024-05-07 11:21:57 -07:00
										 |  |  |   imageDataCache, | 
					
						
							| 
									
										
										
										
											2020-11-19 13:13:36 -05:00
										 |  |  |   i18n, | 
					
						
							| 
									
										
										
										
											2023-10-16 13:58:51 -04:00
										 |  |  |   isCallReconnecting, | 
					
						
							| 
									
										
										
										
											2024-11-01 14:12:49 -07:00
										 |  |  |   joinedAt, | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   remoteParticipants, | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |   setGroupCallVideoRequest, | 
					
						
							| 
									
										
										
										
											2022-05-18 20:28:51 -07:00
										 |  |  |   remoteAudioLevels, | 
					
						
							| 
									
										
										
										
											2023-12-11 07:10:31 -08:00
										 |  |  |   onClickRaisedHand, | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | }: PropsType): JSX.Element { | 
					
						
							| 
									
										
										
										
											2021-01-08 12:58:28 -06:00
										 |  |  |   const [gridDimensions, setGridDimensions] = useState<Dimensions>({ | 
					
						
							|  |  |  |     width: 0, | 
					
						
							|  |  |  |     height: 0, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   const [pageIndex, setPageIndex] = useState(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-06 16:00:11 -06:00
										 |  |  |   const devicePixelRatio = useDevicePixelRatio(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-08 11:32:49 -06:00
										 |  |  |   const getFrameBuffer = useGetCallingFrameBuffer(); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  |   const { invisibleDemuxIds, onParticipantVisibilityChanged } = | 
					
						
							|  |  |  |     useInvisibleParticipants(remoteParticipants); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   const minRenderedHeight = | 
					
						
							|  |  |  |     callViewMode === CallViewMode.Paginated | 
					
						
							|  |  |  |       ? SMALL_TILES_MIN_HEIGHT | 
					
						
							|  |  |  |       : LARGE_TILES_MIN_HEIGHT; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   const isInSpeakerView = | 
					
						
							|  |  |  |     callViewMode === CallViewMode.Speaker || | 
					
						
							|  |  |  |     callViewMode === CallViewMode.Presentation; | 
					
						
							| 
									
										
										
										
											2021-01-08 16:57:54 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   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) | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |   ); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   // 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<GroupCallRemoteParticipantType> = | 
					
						
							|  |  |  |     useMemo( | 
					
						
							|  |  |  |       () => | 
					
						
							|  |  |  |         remoteParticipants | 
					
						
							|  |  |  |           .concat() | 
					
						
							|  |  |  |           .sort( | 
					
						
							|  |  |  |             (a, b) => | 
					
						
							|  |  |  |               Number(b.presenting || 0) - Number(a.presenting || 0) || | 
					
						
							|  |  |  |               (b.speakerTime || -Infinity) - (a.speakerTime || -Infinity) | 
					
						
							|  |  |  |           ), | 
					
						
							|  |  |  |       [remoteParticipants] | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   // 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<ParticipantsInPageType> = useMemo(() => { | 
					
						
							|  |  |  |     if (!prioritySortedParticipants.length) { | 
					
						
							|  |  |  |       return []; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     if (!maxRowsPerPage) { | 
					
						
							|  |  |  |       return []; | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     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, | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   }, [ | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     maxRowWidth, | 
					
						
							|  |  |  |     isInPaginationView, | 
					
						
							|  |  |  |     isInSpeakerView, | 
					
						
							|  |  |  |     maxRowsPerPage, | 
					
						
							|  |  |  |     minRenderedHeight, | 
					
						
							|  |  |  |     pageIndex, | 
					
						
							|  |  |  |     prioritySortedParticipants, | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   ]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   // 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 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-13 09:39:43 -08:00
										 |  |  |   // In speaker or sidebar views, not all participants will be on the grid; they'll
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   //   get put in the overflow zone.
 | 
					
						
							|  |  |  |   const overflowedParticipants: Array<GroupCallRemoteParticipantType> = 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 | 
					
						
							| 
									
										
										
										
											2020-11-18 13:07:25 -06:00
										 |  |  |   ); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   const gridParticipantHeightWithMargin = | 
					
						
							|  |  |  |     gridParticipantHeight + PARTICIPANT_MARGIN; | 
					
						
							|  |  |  |   const gridTotalRowHeightWithMargin = | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     gridParticipantHeightWithMargin * gridArrangement.rows.length - | 
					
						
							|  |  |  |     PARTICIPANT_MARGIN; | 
					
						
							|  |  |  |   const gridTopOffset = Math.max( | 
					
						
							|  |  |  |     0, | 
					
						
							|  |  |  |     Math.round((gridDimensions.height - gridTotalRowHeightWithMargin) / 2) | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const rowElements: Array<Array<JSX.Element>> = gridArrangement.rows.map( | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     (tiles, index) => { | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |       const top = gridTopOffset + index * gridParticipantHeightWithMargin; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const totalRowWidthWithoutMargins = | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |         totalRowWidthAtHeight(tiles, minRenderedHeight) * | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |         gridArrangement.scalar; | 
					
						
							|  |  |  |       const totalRowWidth = | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |         totalRowWidthWithoutMargins + PARTICIPANT_MARGIN * (tiles.length - 1); | 
					
						
							|  |  |  |       const leftOffset = Math.max( | 
					
						
							|  |  |  |         0, | 
					
						
							|  |  |  |         Math.round((gridDimensions.width - totalRowWidth) / 2) | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |       let rowWidthSoFar = 0; | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |       return tiles.map(tile => { | 
					
						
							|  |  |  |         const left = rowWidthSoFar + leftOffset; | 
					
						
							| 
									
										
										
										
											2022-02-08 12:30:33 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |         const renderedWidth = Math.round( | 
					
						
							|  |  |  |           tile.videoAspectRatio * gridParticipantHeight | 
					
						
							| 
									
										
										
										
											2020-11-18 13:07:25 -06:00
										 |  |  |         ); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         rowWidthSoFar += renderedWidth + PARTICIPANT_MARGIN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |         if (isPaginationButton(tile)) { | 
					
						
							|  |  |  |           const isNextButton = tile.paginationButtonType === 'next'; | 
					
						
							|  |  |  |           const isPrevButton = tile.paginationButtonType === 'prev'; | 
					
						
							|  |  |  |           return ( | 
					
						
							|  |  |  |             <button | 
					
						
							|  |  |  |               key={ | 
					
						
							|  |  |  |                 isNextButton ? 'next-pagination-tile' : 'prev-pagination-tile' | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               onClick={isNextButton ? nextPage : prevPage} | 
					
						
							|  |  |  |               style={{ | 
					
						
							|  |  |  |                 insetInlineStart: left, | 
					
						
							|  |  |  |                 insetBlockStart: top, | 
					
						
							|  |  |  |                 width: renderedWidth, | 
					
						
							|  |  |  |                 height: gridParticipantHeight, | 
					
						
							|  |  |  |               }} | 
					
						
							|  |  |  |               type="button" | 
					
						
							|  |  |  |               className="module-ongoing-call__group-call--pagination-tile" | 
					
						
							|  |  |  |             > | 
					
						
							|  |  |  |               {isPrevButton ? ( | 
					
						
							|  |  |  |                 <div className="module-ongoing-call__group-call--pagination-tile--prev-arrow" /> | 
					
						
							|  |  |  |               ) : null} | 
					
						
							|  |  |  |               +{tile.numParticipants} | 
					
						
							|  |  |  |               {isNextButton ? ( | 
					
						
							|  |  |  |                 <div className="module-ongoing-call__group-call--pagination-tile--next-arrow" /> | 
					
						
							|  |  |  |               ) : null} | 
					
						
							|  |  |  |             </button> | 
					
						
							|  |  |  |           ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |         return ( | 
					
						
							|  |  |  |           <GroupCallRemoteParticipant | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |             key={tile.demuxId} | 
					
						
							| 
									
										
										
										
											2021-01-08 11:32:49 -06:00
										 |  |  |             getFrameBuffer={getFrameBuffer} | 
					
						
							| 
									
										
										
										
											2024-05-07 11:21:57 -07:00
										 |  |  |             imageDataCache={imageDataCache} | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |             getGroupCallVideoFrameSource={getGroupCallVideoFrameSource} | 
					
						
							| 
									
										
										
										
											2023-12-11 07:10:31 -08:00
										 |  |  |             onClickRaisedHand={onClickRaisedHand} | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |             height={gridParticipantHeight} | 
					
						
							| 
									
										
										
										
											2020-11-19 13:13:36 -05:00
										 |  |  |             i18n={i18n} | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |             audioLevel={remoteAudioLevels.get(tile.demuxId) ?? 0} | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |             left={left} | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |             remoteParticipant={tile} | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |             top={top} | 
					
						
							|  |  |  |             width={renderedWidth} | 
					
						
							| 
									
										
										
										
											2023-03-22 10:53:13 -07:00
										 |  |  |             remoteParticipantsCount={remoteParticipants.length} | 
					
						
							| 
									
										
										
										
											2023-03-22 14:54:11 -07:00
										 |  |  |             isActiveSpeakerInSpeakerView={isInSpeakerView} | 
					
						
							| 
									
										
										
										
											2023-10-16 13:58:51 -04:00
										 |  |  |             isCallReconnecting={isCallReconnecting} | 
					
						
							| 
									
										
										
										
											2024-11-01 14:12:49 -07:00
										 |  |  |             joinedAt={joinedAt} | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |           /> | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  |   const videoRequestMode = useVideoRequestMode(); | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     log.info(`Group call now using ${videoRequestMode} video request mode`); | 
					
						
							|  |  |  |   }, [videoRequestMode]); | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |   useEffect(() => { | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  |     let videoRequest: Array<GroupCallVideoRequest>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (videoRequestMode) { | 
					
						
							|  |  |  |       case VideoRequestMode.Normal: | 
					
						
							|  |  |  |         videoRequest = [ | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |           ...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 | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |                 ), | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |                 height: clamp( | 
					
						
							|  |  |  |                   Math.round(gridParticipantHeight * scalar), | 
					
						
							|  |  |  |                   1, | 
					
						
							|  |  |  |                   MAX_FRAME_HEIGHT | 
					
						
							|  |  |  |                 ), | 
					
						
							|  |  |  |               }; | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |           ...participantsOnOtherPages.map(nonRenderedRemoteParticipant), | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  |           ...overflowedParticipants.map(participant => { | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |             if (invisibleDemuxIds.has(participant.demuxId)) { | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  |               return nonRenderedRemoteParticipant(participant); | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |               demuxId: participant.demuxId, | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |               width: clamp( | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |                 Math.round(OVERFLOW_PARTICIPANT_WIDTH * VIDEO_REQUEST_SCALAR), | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |                 1, | 
					
						
							|  |  |  |                 MAX_FRAME_WIDTH | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  |               ), | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |               height: clamp( | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |                 Math.round( | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |                   (OVERFLOW_PARTICIPANT_WIDTH / participant.videoAspectRatio) * | 
					
						
							|  |  |  |                     VIDEO_REQUEST_SCALAR | 
					
						
							|  |  |  |                 ), | 
					
						
							|  |  |  |                 1, | 
					
						
							|  |  |  |                 MAX_FRAME_HEIGHT | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  |               ), | 
					
						
							|  |  |  |             }; | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  |           }), | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  |         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: | 
					
						
							| 
									
										
										
										
											2025-07-24 10:18:29 -07:00
										 |  |  |         throw missingCaseError(videoRequestMode); | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-09-12 12:50:13 -07:00
										 |  |  |     setGroupCallVideoRequest( | 
					
						
							|  |  |  |       videoRequest, | 
					
						
							|  |  |  |       clamp(gridParticipantHeight, 0, MAX_FRAME_HEIGHT) | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |   }, [ | 
					
						
							| 
									
										
										
										
											2022-01-06 16:00:11 -06:00
										 |  |  |     devicePixelRatio, | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     currentPage.rows, | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |     gridParticipantHeight, | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  |     invisibleDemuxIds, | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |     overflowedParticipants, | 
					
						
							|  |  |  |     remoteParticipants, | 
					
						
							|  |  |  |     setGroupCallVideoRequest, | 
					
						
							| 
									
										
										
										
											2022-01-06 16:00:11 -06:00
										 |  |  |     videoRequestMode, | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     participantsOnOtherPages, | 
					
						
							| 
									
										
										
										
											2020-12-01 19:52:01 -06:00
										 |  |  |   ]); | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     <div className="module-ongoing-call__participants"> | 
					
						
							|  |  |  |       <div className="module-ongoing-call__participants__grid--wrapper"> | 
					
						
							|  |  |  |         <SizeObserver | 
					
						
							|  |  |  |           onSizeChange={size => { | 
					
						
							|  |  |  |             setGridDimensions(size); | 
					
						
							|  |  |  |           }} | 
					
						
							|  |  |  |         > | 
					
						
							|  |  |  |           {gridRef => ( | 
					
						
							|  |  |  |             <div | 
					
						
							|  |  |  |               className="module-ongoing-call__participants__grid" | 
					
						
							|  |  |  |               ref={gridRef} | 
					
						
							|  |  |  |             > | 
					
						
							|  |  |  |               {flatten(rowElements)} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               {isInPaginationView && ( | 
					
						
							|  |  |  |                 <> | 
					
						
							|  |  |  |                   {pageIndex > 0 ? ( | 
					
						
							|  |  |  |                     <button | 
					
						
							|  |  |  |                       aria-label="Prev" | 
					
						
							|  |  |  |                       className="module-ongoing-call__prev-page" | 
					
						
							|  |  |  |                       onClick={prevPage} | 
					
						
							|  |  |  |                       type="button" | 
					
						
							|  |  |  |                     > | 
					
						
							|  |  |  |                       <div className="module-ongoing-call__prev-page--arrow" /> | 
					
						
							|  |  |  |                     </button> | 
					
						
							|  |  |  |                   ) : null} | 
					
						
							|  |  |  |                   {pageIndex < gridParticipantsByPage.length - 1 ? ( | 
					
						
							|  |  |  |                     <button | 
					
						
							|  |  |  |                       aria-label="Next" | 
					
						
							|  |  |  |                       className="module-ongoing-call__next-page" | 
					
						
							|  |  |  |                       onClick={nextPage} | 
					
						
							|  |  |  |                       type="button" | 
					
						
							|  |  |  |                     > | 
					
						
							|  |  |  |                       <div className="module-ongoing-call__next-page--arrow" /> | 
					
						
							|  |  |  |                     </button> | 
					
						
							|  |  |  |                   ) : null} | 
					
						
							|  |  |  |                 </> | 
					
						
							|  |  |  |               )} | 
					
						
							|  |  |  |             </div> | 
					
						
							|  |  |  |           )} | 
					
						
							|  |  |  |         </SizeObserver> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       {shouldShowOverflow && overflowedParticipants.length > 0 ? ( | 
					
						
							|  |  |  |         <GroupCallOverflowArea | 
					
						
							|  |  |  |           getFrameBuffer={getFrameBuffer} | 
					
						
							|  |  |  |           getGroupCallVideoFrameSource={getGroupCallVideoFrameSource} | 
					
						
							| 
									
										
										
										
											2024-05-07 11:21:57 -07:00
										 |  |  |           imageDataCache={imageDataCache} | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |           i18n={i18n} | 
					
						
							|  |  |  |           isCallReconnecting={isCallReconnecting} | 
					
						
							| 
									
										
										
										
											2024-11-01 14:12:49 -07:00
										 |  |  |           joinedAt={joinedAt} | 
					
						
							| 
									
										
										
										
											2023-12-11 07:10:31 -08:00
										 |  |  |           onClickRaisedHand={onClickRaisedHand} | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |           onParticipantVisibilityChanged={onParticipantVisibilityChanged} | 
					
						
							|  |  |  |           overflowedParticipants={overflowedParticipants} | 
					
						
							|  |  |  |           remoteAudioLevels={remoteAudioLevels} | 
					
						
							|  |  |  |           remoteParticipantsCount={remoteParticipants.length} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |       ) : null} | 
					
						
							|  |  |  |     </div> | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  | // This function is only meant for use with `useInvisibleParticipants`. It helps avoid
 | 
					
						
							|  |  |  | //   returning new set instances when the underlying values are equal.
 | 
					
						
							|  |  |  | function pickDifferentSet<T>(a: Readonly<Set<T>>, b: Readonly<Set<T>>): Set<T> { | 
					
						
							|  |  |  |   return a.size === b.size ? a : b; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function useInvisibleParticipants( | 
					
						
							|  |  |  |   remoteParticipants: ReadonlyArray<GroupCallRemoteParticipantType> | 
					
						
							|  |  |  | ): Readonly<{ | 
					
						
							|  |  |  |   invisibleDemuxIds: Set<number>; | 
					
						
							|  |  |  |   onParticipantVisibilityChanged: (demuxId: number, isVisible: boolean) => void; | 
					
						
							|  |  |  | }> { | 
					
						
							|  |  |  |   const [invisibleDemuxIds, setInvisibleDemuxIds] = useState(new Set<number>()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const onParticipantVisibilityChanged = useCallback( | 
					
						
							|  |  |  |     (demuxId: number, isVisible: boolean) => { | 
					
						
							|  |  |  |       setInvisibleDemuxIds(oldInvisibleDemuxIds => { | 
					
						
							|  |  |  |         const toggled = setUtil.toggle( | 
					
						
							|  |  |  |           oldInvisibleDemuxIds, | 
					
						
							|  |  |  |           demuxId, | 
					
						
							|  |  |  |           !isVisible | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         return pickDifferentSet(oldInvisibleDemuxIds, toggled); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     [] | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-23 17:16:13 +00:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     log.info( | 
					
						
							|  |  |  |       `Invisible demux IDs changed to [${join(invisibleDemuxIds, ',')}]` | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   }, [invisibleDemuxIds]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-06 17:06:13 -06:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     const remoteParticipantDemuxIds = new Set<number>( | 
					
						
							|  |  |  |       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, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-04 16:55:30 -05:00
										 |  |  | function useVideoRequestMode(): VideoRequestMode { | 
					
						
							|  |  |  |   const isPageVisible = usePageVisibility(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const [result, setResult] = useState<VideoRequestMode>( | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | function totalRowWidthAtHeight( | 
					
						
							|  |  |  |   participantsInRow: ReadonlyArray< | 
					
						
							|  |  |  |     Pick<ParticipantTileType, 'videoAspectRatio'> | 
					
						
							|  |  |  |   >, | 
					
						
							|  |  |  |   height: number | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  | ): number { | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   return participantsInRow.reduce( | 
					
						
							|  |  |  |     (result, participant) => | 
					
						
							|  |  |  |       result + participantWidthAtHeight(participant, height), | 
					
						
							| 
									
										
										
										
											2020-11-13 13:57:55 -06:00
										 |  |  |     0 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-08 12:58:28 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | function participantWidthAtHeight( | 
					
						
							|  |  |  |   participant: Pick<ParticipantTileType, 'videoAspectRatio'>, | 
					
						
							|  |  |  |   height: number | 
					
						
							|  |  |  | ) { | 
					
						
							|  |  |  |   return participant.videoAspectRatio * height; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-08 12:58:28 -06:00
										 |  |  | function stableParticipantComparator( | 
					
						
							|  |  |  |   a: Readonly<{ demuxId: number }>, | 
					
						
							|  |  |  |   b: Readonly<{ demuxId: number }> | 
					
						
							|  |  |  | ): number { | 
					
						
							|  |  |  |   return a.demuxId - b.demuxId; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | type ParticipantsInPageType< | 
					
						
							| 
									
										
										
										
											2024-07-23 17:31:40 -07:00
										 |  |  |   T extends { videoAspectRatio: number } = ParticipantTileType, | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | > = { | 
					
						
							|  |  |  |   rows: Array<Array<T>>; | 
					
						
							|  |  |  |   numParticipants: number; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type PageLayoutPropsType = { | 
					
						
							|  |  |  |   maxRowWidth: number; | 
					
						
							|  |  |  |   minRenderedHeight: number; | 
					
						
							|  |  |  |   maxRowsPerPage: number; | 
					
						
							|  |  |  |   maxParticipantsPerPage: number; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | function getGridParticipantsByPage({ | 
					
						
							|  |  |  |   participants, | 
					
						
							|  |  |  |   maxPages, | 
					
						
							|  |  |  |   currentPage, | 
					
						
							|  |  |  |   ...pageLayoutProps | 
					
						
							|  |  |  | }: PageLayoutPropsType & { | 
					
						
							|  |  |  |   participants: Array<GroupCallRemoteParticipantType>; | 
					
						
							|  |  |  |   maxPages: number; | 
					
						
							|  |  |  |   currentPage?: number; | 
					
						
							|  |  |  | }): Array<ParticipantsInPageType> { | 
					
						
							|  |  |  |   if (!participants.length) { | 
					
						
							|  |  |  |     return []; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const pages: Array<ParticipantsInPageType> = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   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<ParticipantTileType> | 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( | 
					
						
							| 
									
										
										
										
											2025-06-16 11:59:31 -07:00
										 |  |  |           `failed after ${attemptNumber} attempts to layout
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |           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<ParticipantTileType>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 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<GroupCallRemoteParticipantType>; | 
					
						
							|  |  |  |   leaveRoomForPrevPageButton: boolean; | 
					
						
							|  |  |  |   leaveRoomForNextPageButton: boolean; | 
					
						
							|  |  |  |   isSubsetOfAllParticipants?: boolean; | 
					
						
							|  |  |  | }): ParticipantsInPageType<GroupCallRemoteParticipantType> { | 
					
						
							|  |  |  |   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<Array<GroupCallRemoteParticipantType>> = [[]]; | 
					
						
							|  |  |  |   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<Array<ParticipantTileType>>; | 
					
						
							|  |  |  |   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<Array<ParticipantTileType>>; | 
					
						
							|  |  |  | }): 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; | 
					
						
							|  |  |  | } |