154 lines
4.5 KiB
TypeScript
154 lines
4.5 KiB
TypeScript
// Copyright 2018 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import React from 'react';
|
|
|
|
import { CurveType, Image } from './Image';
|
|
import { StagedGenericAttachment } from './StagedGenericAttachment';
|
|
import { StagedPlaceholderAttachment } from './StagedPlaceholderAttachment';
|
|
import type { LocalizerType } from '../../types/Util';
|
|
import type {
|
|
AttachmentType,
|
|
AttachmentDraftType,
|
|
} from '../../types/Attachment';
|
|
import {
|
|
areAllAttachmentsVisual,
|
|
canDisplayImage,
|
|
isImageAttachment,
|
|
isVideoAttachment,
|
|
} from '../../types/Attachment';
|
|
|
|
export type Props<T extends AttachmentType | AttachmentDraftType> = Readonly<{
|
|
attachments: ReadonlyArray<T>;
|
|
canEditImages?: boolean;
|
|
i18n: LocalizerType;
|
|
onAddAttachment?: () => void;
|
|
onClickAttachment?: (attachment: T) => void;
|
|
onClose?: () => void;
|
|
onCloseAttachment: (attachment: T) => void;
|
|
}>;
|
|
|
|
const IMAGE_WIDTH = 120;
|
|
const IMAGE_HEIGHT = 120;
|
|
|
|
// This is a 1x1 black square.
|
|
const BLANK_VIDEO_THUMBNAIL =
|
|
'';
|
|
|
|
function getUrl(
|
|
attachment: AttachmentType | AttachmentDraftType
|
|
): string | undefined {
|
|
if (attachment.pending) {
|
|
return undefined;
|
|
}
|
|
|
|
if ('screenshot' in attachment) {
|
|
return attachment.screenshot?.url || attachment.url;
|
|
}
|
|
|
|
return attachment.url;
|
|
}
|
|
|
|
export function AttachmentList<T extends AttachmentType | AttachmentDraftType>({
|
|
attachments,
|
|
canEditImages,
|
|
i18n,
|
|
onAddAttachment,
|
|
onClickAttachment,
|
|
onCloseAttachment,
|
|
onClose,
|
|
}: Props<T>): JSX.Element | null {
|
|
if (!attachments.length) {
|
|
return null;
|
|
}
|
|
|
|
const allVisualAttachments = areAllAttachmentsVisual(attachments);
|
|
|
|
return (
|
|
<div className="module-attachments">
|
|
{onClose && attachments.length > 1 ? (
|
|
<div className="module-attachments__header">
|
|
<button
|
|
type="button"
|
|
onClick={onClose}
|
|
className="module-attachments__close-button"
|
|
aria-label={i18n('icu:close')}
|
|
/>
|
|
</div>
|
|
) : null}
|
|
<div className="module-attachments__rail">
|
|
{(attachments || []).map((attachment, index) => {
|
|
const url = getUrl(attachment);
|
|
|
|
const key = url || attachment.path || attachment.fileName || index;
|
|
|
|
const isImage = isImageAttachment(attachment);
|
|
const isVideo = isVideoAttachment(attachment);
|
|
const closeAttachment = () => onCloseAttachment(attachment);
|
|
|
|
if (
|
|
(isImage && canDisplayImage([attachment])) ||
|
|
isVideo ||
|
|
attachment.pending
|
|
) {
|
|
const isDownloaded = !attachment.pending;
|
|
const imageUrl =
|
|
url || (isVideo ? BLANK_VIDEO_THUMBNAIL : undefined);
|
|
|
|
const clickAttachment = onClickAttachment
|
|
? () => onClickAttachment(attachment)
|
|
: undefined;
|
|
|
|
const imgElement = (
|
|
<Image
|
|
key={key}
|
|
alt={i18n('icu:stagedImageAttachment', {
|
|
path: attachment.fileName || url || index.toString(),
|
|
})}
|
|
className="module-staged-attachment"
|
|
i18n={i18n}
|
|
attachment={attachment}
|
|
isDownloaded={isDownloaded}
|
|
curveBottomLeft={CurveType.Tiny}
|
|
curveBottomRight={CurveType.Tiny}
|
|
curveTopLeft={CurveType.Tiny}
|
|
curveTopRight={CurveType.Tiny}
|
|
playIconOverlay={isVideo}
|
|
height={IMAGE_HEIGHT}
|
|
width={IMAGE_WIDTH}
|
|
url={imageUrl}
|
|
closeButton
|
|
onClick={clickAttachment}
|
|
onClickClose={closeAttachment}
|
|
onError={closeAttachment}
|
|
/>
|
|
);
|
|
|
|
if (isImage && canEditImages) {
|
|
return (
|
|
<div className="module-attachments--editable" key={key}>
|
|
{imgElement}
|
|
<div className="module-attachments__edit-icon" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return imgElement;
|
|
}
|
|
|
|
return (
|
|
<StagedGenericAttachment
|
|
key={key}
|
|
attachment={attachment}
|
|
i18n={i18n}
|
|
onClose={closeAttachment}
|
|
/>
|
|
);
|
|
})}
|
|
{allVisualAttachments && onAddAttachment ? (
|
|
<StagedPlaceholderAttachment onClick={onAddAttachment} i18n={i18n} />
|
|
) : null}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|