You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-desktop/ts/components/conversation/Image.tsx

229 lines
6.0 KiB
TypeScript

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<Pick<Props, 'darkOverlay' | 'softCorners'>>`
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<boolean>(attachment.pending || !url || true);
const [mounted, setMounted] = useState<boolean>(
(!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 (
<MessageGenericAttachment
attachment={attachment as AttachmentTypeWithPath}
pending={false}
highlight={!!highlight}
selected={!!dropShadow} // dropshadow is selected
direction={direction}
/>
);
}
return (
<div
role={role}
onClick={(e: any) => {
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 ? (
<div
className="module-image__loading-placeholder"
style={{
maxHeight: height,
maxWidth: width,
width,
height,
lineHeight: height,
textAlign: 'center',
}}
>
<Spinner size="normal" />
</div>
) : (
<img
onError={onErrorUrlFilterering}
className={classNames(
'module-image__image',
forceSquare ? 'module-image__image-cover' : ''
)}
alt={alt}
style={{
maxHeight: height,
maxWidth: width,
minHeight: height,
minWidth: width,
width: forceSquare ? width : '',
height: forceSquare ? height : '',
}}
src={urlToLoad}
onDragStart={disableDrag}
/>
)}
{caption ? (
<img
className="module-image__caption-icon"
src="images/caption-shadow.svg"
alt={window.i18n('imageCaptionIconAlt')}
onDragStart={disableDrag}
/>
) : null}
<StyledOverlay
className={classNames(softCorners ? 'module-image--soft-corners' : null)}
darkOverlay={darkOverlay}
softCorners={softCorners}
/>
{closeButton ? (
<div
role="button"
onClick={(e: any) => {
e.stopPropagation();
if (onClickClose) {
onClickClose(attachment);
}
}}
className="module-image__close-button"
/>
) : null}
{mounted && playIconOverlay ? (
<div className="module-image__play-overlay__circle">
<div className="module-image__play-overlay__icon" />
</div>
) : null}
{overlayText ? (
<div className="module-image__text-container" style={{ lineHeight: height }}>
{overlayText}
</div>
) : null}
</div>
);
};