// Copyright 2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactNode } from 'react'; import React, { useEffect, useRef } from 'react'; import classNames from 'classnames'; import { Blurhash } from 'react-blurhash'; import type { AttachmentType } from '../types/Attachment'; import type { LocalizerType } from '../types/Util'; import { Spinner } from './Spinner'; import { TextAttachment } from './TextAttachment'; import { ThemeType } from '../types/Util'; import { defaultBlurHash, hasNotResolved, isDownloaded, isDownloading, isGIF, } from '../types/Attachment'; import { getClassNamesFor } from '../util/getClassNamesFor'; import { isVideoTypeSupported } from '../util/GoogleChrome'; export type PropsType = { readonly attachment?: AttachmentType; readonly children?: ReactNode; readonly i18n: LocalizerType; readonly isMuted?: boolean; readonly isPaused?: boolean; readonly isThumbnail?: boolean; readonly label: string; readonly moduleClassName?: string; readonly queueStoryDownload: (storyId: string) => unknown; readonly storyId: string; }; export const StoryImage = ({ attachment, children, i18n, isMuted, isPaused, isThumbnail, label, moduleClassName, queueStoryDownload, storyId, }: PropsType): JSX.Element | null => { const shouldDownloadAttachment = !isDownloaded(attachment) && !isDownloading(attachment) && !hasNotResolved(attachment); const videoRef = useRef(null); useEffect(() => { if (shouldDownloadAttachment) { queueStoryDownload(storyId); } }, [queueStoryDownload, shouldDownloadAttachment, storyId]); useEffect(() => { if (!videoRef.current) { return; } if (isPaused) { videoRef.current.pause(); } else { videoRef.current.play(); } }, [isPaused]); if (!attachment) { return null; } const isPending = Boolean(attachment.pending) && !attachment.textAttachment; const isNotReadyToShow = hasNotResolved(attachment) || isPending; const isSupportedVideo = isVideoTypeSupported(attachment.contentType); const getClassName = getClassNamesFor('StoryImage', moduleClassName); let storyElement: JSX.Element; if (attachment.textAttachment) { storyElement = ( ); } else if (isNotReadyToShow) { storyElement = ( ); } else if (!isThumbnail && isSupportedVideo) { const shouldLoop = isGIF(attachment ? [attachment] : undefined); storyElement = ( ); } else { storyElement = ( {label} ); } let overlay: JSX.Element | undefined; if (isPending) { overlay = (
); } return (
{storyElement} {overlay} {children}
); };