Files
element-call/src/grid/SpotlightPortraitLayout.tsx

110 lines
2.8 KiB
TypeScript
Raw Normal View History

2024-05-17 16:38:00 -04:00
/*
Copyright 2024 New Vector Ltd.
2024-05-17 16:38:00 -04:00
SPDX-License-Identifier: AGPL-3.0-only
Please see LICENSE in the repository root for full details.
2024-05-17 16:38:00 -04:00
*/
import { CSSProperties, forwardRef, useMemo } from "react";
2024-05-17 16:38:00 -04:00
import { useObservableEagerState } from "observable-hooks";
import classNames from "classnames";
2024-05-17 16:38:00 -04:00
import {
CallLayout,
GridTileModel,
TileModel,
arrangeTiles,
} from "./CallLayout";
import { SpotlightPortraitLayout as SpotlightPortraitLayoutModel } from "../state/CallViewModel";
import styles from "./SpotlightPortraitLayout.module.css";
2024-07-25 12:50:28 -04:00
import { useUpdateLayout } from "./Grid";
2024-05-17 16:38:00 -04:00
interface GridCSSProperties extends CSSProperties {
"--grid-gap": string;
"--grid-tile-width": string;
"--grid-tile-height": string;
2024-05-17 16:38:00 -04:00
}
2024-07-18 11:33:20 -04:00
/**
* An implementation of the "spotlight portrait" layout, in which the spotlight
* tile is shown across the top of the screen, and the grid of participants
* scrolls behind it.
*/
export const makeSpotlightPortraitLayout: CallLayout<
SpotlightPortraitLayoutModel
> = ({ minBounds }) => ({
scrollingOnTop: false,
2024-05-17 16:38:00 -04:00
fixed: forwardRef(function SpotlightPortraitLayoutFixed(
{ model, Slot },
ref,
) {
2024-07-25 12:50:28 -04:00
useUpdateLayout();
const tileModel: TileModel = useMemo(
() => ({
type: "spotlight",
vms: model.spotlight,
maximised: true,
}),
[model.spotlight],
);
2024-05-17 16:38:00 -04:00
return (
<div ref={ref} className={styles.layer}>
<div className={styles.spotlight}>
<Slot className={styles.slot} id="spotlight" model={tileModel} />
2024-05-17 16:38:00 -04:00
</div>
</div>
);
}),
2024-05-17 16:38:00 -04:00
scrolling: forwardRef(function SpotlightPortraitLayoutScrolling(
{ model, Slot },
ref,
) {
2024-07-25 12:50:28 -04:00
useUpdateLayout();
const { width } = useObservableEagerState(minBounds);
const { gap, tileWidth, tileHeight } = arrangeTiles(
width,
// TODO: We pretend that the minimum height is the width, because the
// actual minimum height is difficult to calculate
width,
model.grid.length,
);
const tileModels: GridTileModel[] = useMemo(
() => model.grid.map((vm) => ({ type: "grid", vm })),
[model.grid],
);
2024-05-17 16:38:00 -04:00
return (
<div
ref={ref}
2024-06-07 16:59:56 -04:00
className={styles.layer}
style={
{
"--grid-gap": `${gap}px`,
"--grid-tile-width": `${Math.floor(tileWidth)}px`,
"--grid-tile-height": `${Math.floor(tileHeight)}px`,
} as GridCSSProperties
}
>
2024-05-17 16:38:00 -04:00
<div
className={classNames(styles.spotlight, {
[styles.withIndicators]: model.spotlight.length > 1,
})}
/>
<div className={styles.grid}>
{tileModels.map((m) => (
<Slot
key={m.vm.id}
className={styles.slot}
id={m.vm.id}
model={m}
/>
))}
2024-05-17 16:38:00 -04:00
</div>
</div>
);
}),
2024-05-17 16:38:00 -04:00
});