All checks were successful
/ depoly (push) Successful in 1m7s
The reworked viewer presentation process delay the creation of the viewer until the user interaction, to reduce the DOM created during viewing the list.
83 lines
2.3 KiB
TypeScript
83 lines
2.3 KiB
TypeScript
import type { mastodon } from "masto";
|
|
import { type Component, For, createSignal, onMount } from "solid-js";
|
|
import { css } from "solid-styled";
|
|
import tootStyle from "./toot.module.css";
|
|
import MediaViewer from "./MediaViewer";
|
|
import { render } from "solid-js/web";
|
|
|
|
const MediaAttachmentGrid: Component<{
|
|
attachments: mastodon.v1.MediaAttachment[];
|
|
}> = (props) => {
|
|
let rootRef: HTMLElement;
|
|
const [viewerIndex, setViewerIndex] = createSignal<number>();
|
|
const viewerOpened = () => typeof viewerIndex() !== "undefined";
|
|
const gridTemplateColumns = () => {
|
|
const l = props.attachments.length;
|
|
if (l < 2) {
|
|
return "minmax(40px, auto)";
|
|
}
|
|
if (l < 4) {
|
|
return "repeat(2, minmax(40px, auto))";
|
|
}
|
|
return "repeat(3, minmax(40px, auto))";
|
|
};
|
|
|
|
const openViewerFor = (index: number) => {
|
|
setViewerIndex(index);
|
|
const container = document.createElement("div");
|
|
container.setAttribute("role", "presentation");
|
|
document.body.appendChild(container);
|
|
let dispose: () => void
|
|
dispose = render(() => {
|
|
return (
|
|
<MediaViewer
|
|
show={viewerOpened()}
|
|
index={viewerIndex() || 0}
|
|
onIndexUpdated={setViewerIndex}
|
|
media={props.attachments}
|
|
onClose={() => {
|
|
setViewerIndex(undefined);
|
|
dispose();
|
|
document.body.removeChild(container)
|
|
}}
|
|
/>
|
|
);
|
|
}, container);
|
|
};
|
|
|
|
css`
|
|
.attachments {
|
|
grid-template-columns: ${gridTemplateColumns()};
|
|
}
|
|
`;
|
|
return (
|
|
<section
|
|
ref={rootRef!}
|
|
class={[tootStyle.tootAttachmentGrp, "attachments"].join(" ")}
|
|
onClick={(e) => e.stopImmediatePropagation()}
|
|
>
|
|
<For each={props.attachments}>
|
|
{(item, index) => {
|
|
switch (item.type) {
|
|
case "image":
|
|
return (
|
|
<img
|
|
src={item.previewUrl}
|
|
alt={item.description || undefined}
|
|
onClick={[openViewerFor, index()]}
|
|
loading="lazy"
|
|
></img>
|
|
);
|
|
case "video":
|
|
case "gifv":
|
|
case "audio":
|
|
case "unknown":
|
|
return <div></div>;
|
|
}
|
|
}}
|
|
</For>
|
|
</section>
|
|
);
|
|
};
|
|
|
|
export default MediaAttachmentGrid;
|