import classNames from 'classnames'; import { useCallback, useEffect, useState } from 'react'; import styled from 'styled-components'; import { isNumber } from 'lodash'; import { useDisableDrag } from '../../hooks/useDisableDrag'; import { AttachmentType, AttachmentTypeWithPath } from '../../types/Attachment'; import { Spinner } from '../loading'; import { MessageModelType } from '../../models/messageType'; import { MessageGenericAttachment } from './message/message-content/MessageGenericAttachment'; import { useEncryptedFileFetch } from '../../hooks/useEncryptedFileFetch'; type Props = { alt: string; attachment: AttachmentTypeWithPath | AttachmentType; /** undefined if the message is not visible yet, '' if the attachment is broken */ url: string | undefined; imageBroken?: boolean; height?: number | string; width?: number | string; overlayText?: string; closeButton?: boolean; darkOverlay?: boolean; playIconOverlay?: boolean; softCorners: boolean; forceSquare?: boolean; dropShadow?: boolean; attachmentIndex?: number; direction?: MessageModelType; highlight?: boolean; onClick?: (attachment: AttachmentTypeWithPath | AttachmentType) => void; onClickClose?: (attachment: AttachmentTypeWithPath | AttachmentType) => void; onError?: () => void; /** used for debugging */ timestamp?: number; }; const StyledOverlay = styled.div>` position: absolute; top: 0; bottom: 0; z-index: 1; left: 0; right: 0; background-color: ${props => props.darkOverlay ? 'var(--message-link-preview-background-color)' : 'unset'}; `; export const Image = (props: Props) => { const { alt, attachment, imageBroken, closeButton, darkOverlay, height: _height, onClick, onClickClose, onError, overlayText, playIconOverlay, softCorners, forceSquare, dropShadow, attachmentIndex, direction, highlight, url, width: _width, timestamp, } = props; const disableDrag = useDisableDrag(); const { loading, urlToLoad } = useEncryptedFileFetch( url, attachment.contentType, false, timestamp ); const { caption } = attachment || { caption: null }; const [pending, setPending] = useState(attachment.pending || !url || true); const [mounted, setMounted] = useState( (!loading || !pending) && urlToLoad === undefined ); const canClick = onClick && !pending; const role = canClick ? 'button' : undefined; const onErrorUrlFilterering = useCallback(() => { if (mounted && url && urlToLoad === '' && onError) { onError(); setPending(false); } }, [mounted, onError, url, urlToLoad]); const width = isNumber(_width) ? `${_width}px` : _width; const height = isNumber(_height) ? `${_height}px` : _height; useEffect(() => { if (mounted && url === '') { setPending(false); onErrorUrlFilterering(); } if (mounted && imageBroken && urlToLoad === '') { setPending(false); onErrorUrlFilterering(); } if (url) { setPending(false); setMounted(!loading && !pending); } }, [imageBroken, loading, mounted, onErrorUrlFilterering, pending, url, urlToLoad]); if (mounted && imageBroken) { return ( ); } return (
{ if (canClick && onClick) { e.stopPropagation(); onClick(attachment); } }} className={classNames( 'module-image', canClick ? 'module-image__with-click-handler' : null, softCorners ? 'module-image--soft-corners' : null )} style={{ maxHeight: height, maxWidth: width, minHeight: height, minWidth: width, boxShadow: dropShadow ? 'var(--drop-shadow)' : undefined, }} data-attachmentindex={attachmentIndex} > {!mounted || loading || pending ? (
) : ( {alt} )} {caption ? ( {window.i18n('imageCaptionIconAlt')} ) : null} {closeButton ? (
{ e.stopPropagation(); if (onClickClose) { onClickClose(attachment); } }} className="module-image__close-button" /> ) : null} {mounted && playIconOverlay ? (
) : null} {overlayText ? (
{overlayText}
) : null}
); };