|
|
|
@ -1,17 +1,20 @@
|
|
|
|
|
import classNames from 'classnames';
|
|
|
|
|
import { useCallback } from 'react';
|
|
|
|
|
import { useCallback, useEffect, useState } from 'react';
|
|
|
|
|
import styled from 'styled-components';
|
|
|
|
|
|
|
|
|
|
import { isNumber } from 'lodash';
|
|
|
|
|
import { useDisableDrag } from '../../hooks/useDisableDrag';
|
|
|
|
|
import { useEncryptedFileFetch } from '../../hooks/useEncryptedFileFetch';
|
|
|
|
|
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;
|
|
|
|
|
url: string | undefined; // url is undefined if the message is not visible yet
|
|
|
|
|
imageBroken?: boolean;
|
|
|
|
|
|
|
|
|
|
height?: number | string;
|
|
|
|
|
width?: number | string;
|
|
|
|
@ -26,10 +29,14 @@ type Props = {
|
|
|
|
|
forceSquare?: boolean;
|
|
|
|
|
dropShadow?: boolean;
|
|
|
|
|
attachmentIndex?: number;
|
|
|
|
|
direction?: MessageModelType;
|
|
|
|
|
highlight?: boolean;
|
|
|
|
|
|
|
|
|
|
onClick?: (attachment: AttachmentTypeWithPath | AttachmentType) => void;
|
|
|
|
|
onClickClose?: (attachment: AttachmentTypeWithPath | AttachmentType) => void;
|
|
|
|
|
onError?: () => void;
|
|
|
|
|
|
|
|
|
|
timestamp?: number;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const StyledOverlay = styled.div<Pick<Props, 'darkOverlay' | 'softCorners'>>`
|
|
|
|
@ -46,6 +53,7 @@ export const Image = (props: Props) => {
|
|
|
|
|
const {
|
|
|
|
|
alt,
|
|
|
|
|
attachment,
|
|
|
|
|
imageBroken,
|
|
|
|
|
closeButton,
|
|
|
|
|
darkOverlay,
|
|
|
|
|
height: _height,
|
|
|
|
@ -58,35 +66,78 @@ export const Image = (props: Props) => {
|
|
|
|
|
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 };
|
|
|
|
|
let { pending } = attachment || { pending: true };
|
|
|
|
|
|
|
|
|
|
if (!url) {
|
|
|
|
|
// force pending to true if the url is undefined, so we show a loader while decrypting the attachemtn
|
|
|
|
|
pending = true;
|
|
|
|
|
}
|
|
|
|
|
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 { loading, urlToLoad } = useEncryptedFileFetch(url || '', attachment.contentType, false);
|
|
|
|
|
// data will be url if loading is finished and '' if not
|
|
|
|
|
const srcData = !loading ? urlToLoad : '';
|
|
|
|
|
|
|
|
|
|
const onErrorUrlFilterering = useCallback(() => {
|
|
|
|
|
if (!loading && !pending && !url && onError) {
|
|
|
|
|
if (mounted && url && urlToLoad === '' && onError) {
|
|
|
|
|
onError();
|
|
|
|
|
setPending(false);
|
|
|
|
|
}
|
|
|
|
|
}, [loading, pending, url, onError]);
|
|
|
|
|
}, [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();
|
|
|
|
|
window.log.debug(
|
|
|
|
|
`WIP: [Image] timestamp ${timestamp} fail url ${url !== '' ? url : 'empty'} urlToLoad ${urlToLoad !== '' ? urlToLoad : 'empty'} loading ${loading} pending ${pending} imageBroken ${imageBroken}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mounted && imageBroken && urlToLoad === '') {
|
|
|
|
|
setPending(false);
|
|
|
|
|
onErrorUrlFilterering();
|
|
|
|
|
window.log.debug(
|
|
|
|
|
`WIP: [Image] timestamp ${timestamp} fail url ${url !== '' ? url : 'empty'} urlToLoad ${urlToLoad !== '' ? urlToLoad : 'empty'} loading ${loading} pending ${pending} imageBroken ${imageBroken}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (url) {
|
|
|
|
|
setPending(false);
|
|
|
|
|
setMounted(!loading && !pending);
|
|
|
|
|
window.log.debug(
|
|
|
|
|
`WIP: [Image] timestamp ${timestamp} success url ${url !== '' ? url : 'empty'} urlToLoad ${urlToLoad !== '' ? urlToLoad : 'empty'} loading ${loading} pending ${pending} imageBroken ${imageBroken}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}, [imageBroken, loading, mounted, onErrorUrlFilterering, pending, timestamp, 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}
|
|
|
|
@ -110,7 +161,7 @@ export const Image = (props: Props) => {
|
|
|
|
|
}}
|
|
|
|
|
data-attachmentindex={attachmentIndex}
|
|
|
|
|
>
|
|
|
|
|
{pending || loading ? (
|
|
|
|
|
{!mounted || loading || pending ? (
|
|
|
|
|
<div
|
|
|
|
|
className="module-image__loading-placeholder"
|
|
|
|
|
style={{
|
|
|
|
@ -140,7 +191,7 @@ export const Image = (props: Props) => {
|
|
|
|
|
width: forceSquare ? width : '',
|
|
|
|
|
height: forceSquare ? height : '',
|
|
|
|
|
}}
|
|
|
|
|
src={srcData}
|
|
|
|
|
src={urlToLoad}
|
|
|
|
|
onDragStart={disableDrag}
|
|
|
|
|
/>
|
|
|
|
|
)}
|
|
|
|
@ -169,7 +220,7 @@ export const Image = (props: Props) => {
|
|
|
|
|
className="module-image__close-button"
|
|
|
|
|
/>
|
|
|
|
|
) : null}
|
|
|
|
|
{!(pending || loading) && playIconOverlay ? (
|
|
|
|
|
{!pending && playIconOverlay ? (
|
|
|
|
|
<div className="module-image__play-overlay__circle">
|
|
|
|
|
<div className="module-image__play-overlay__icon" />
|
|
|
|
|
</div>
|
|
|
|
|