// Copyright 2018 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React from 'react'; import classNames from 'classnames'; import type { AttachmentType } from '../../types/Attachment'; import { areAllAttachmentsVisual, getAlt, getImageDimensions, getThumbnailUrl, getUrl, isVideoAttachment, } from '../../types/Attachment'; import { Image, CurveType } from './Image'; import type { LocalizerType, ThemeType } from '../../types/Util'; export type DirectionType = 'incoming' | 'outgoing'; export type Props = { attachments: ReadonlyArray; bottomOverlay?: boolean; direction: DirectionType; isSticker?: boolean; shouldCollapseAbove?: boolean; shouldCollapseBelow?: boolean; stickerSize?: number; tabIndex?: number; withContentAbove?: boolean; withContentBelow?: boolean; i18n: LocalizerType; theme?: ThemeType; onError: () => void; onClick?: (attachment: AttachmentType) => void; }; const GAP = 1; function getCurves({ direction, shouldCollapseAbove, shouldCollapseBelow, withContentAbove, withContentBelow, }: { direction: DirectionType; shouldCollapseAbove?: boolean; shouldCollapseBelow?: boolean; withContentAbove?: boolean; withContentBelow?: boolean; }): { curveTopLeft: CurveType; curveTopRight: CurveType; curveBottomLeft: CurveType; curveBottomRight: CurveType; } { let curveTopLeft = CurveType.None; let curveTopRight = CurveType.None; let curveBottomLeft = CurveType.None; let curveBottomRight = CurveType.None; if (shouldCollapseAbove && direction === 'incoming') { curveTopLeft = CurveType.Tiny; curveTopRight = CurveType.Normal; } else if (shouldCollapseAbove && direction === 'outgoing') { curveTopLeft = CurveType.Normal; curveTopRight = CurveType.Tiny; } else if (!withContentAbove) { curveTopLeft = CurveType.Normal; curveTopRight = CurveType.Normal; } if (withContentBelow) { curveBottomLeft = CurveType.None; curveBottomRight = CurveType.None; } else if (shouldCollapseBelow && direction === 'incoming') { curveBottomLeft = CurveType.Tiny; curveBottomRight = CurveType.None; } else if (shouldCollapseBelow && direction === 'outgoing') { curveBottomLeft = CurveType.None; curveBottomRight = CurveType.Tiny; } else { curveBottomLeft = CurveType.Normal; curveBottomRight = CurveType.Normal; } return { curveTopLeft, curveTopRight, curveBottomLeft, curveBottomRight, }; } export function ImageGrid({ attachments, bottomOverlay, direction, i18n, isSticker, stickerSize, onError, onClick, shouldCollapseAbove, shouldCollapseBelow, tabIndex, theme, withContentAbove, withContentBelow, }: Props): JSX.Element | null { const { curveTopLeft, curveTopRight, curveBottomLeft, curveBottomRight } = getCurves({ direction, shouldCollapseAbove, shouldCollapseBelow, withContentAbove, withContentBelow, }); const withBottomOverlay = Boolean(bottomOverlay && !withContentBelow); if (!attachments || !attachments.length) { return null; } if (attachments.length === 1 || !areAllAttachmentsVisual(attachments)) { const { height, width } = getImageDimensions( attachments[0], isSticker ? stickerSize : undefined ); return (
{getAlt(attachments[0],
); } if (attachments.length === 2) { return (
{getAlt(attachments[0], {getAlt(attachments[1],
); } if (attachments.length === 3) { return (
{getAlt(attachments[0],
{getAlt(attachments[1], {getAlt(attachments[2],
); } if (attachments.length === 4) { return (
{getAlt(attachments[0], {getAlt(attachments[1],
{getAlt(attachments[2], {getAlt(attachments[3],
); } const moreMessagesOverlay = attachments.length > 5; const moreMessagesOverlayText = moreMessagesOverlay ? `+${attachments.length - 5}` : undefined; return (
{getAlt(attachments[0], {getAlt(attachments[1],
{getAlt(attachments[2], {getAlt(attachments[3], {getAlt(attachments[4],
); }