import React, { useContext, useEffect, useState } from 'react'; import { StyleProp, TextStyle, View } from 'react-native'; import FastImage from 'react-native-fast-image'; import { IAttachment, IUserMessage } from '../../definitions'; import { TGetCustomEmoji } from '../../definitions/IEmoji'; import { fetchAutoDownloadEnabled } from '../../lib/methods/autoDownloadPreference'; import { cancelDownload, downloadMediaFile, getMediaCache, isDownloadActive, resumeMediaFile } from '../../lib/methods/handleMediaDownload'; import { formatAttachmentUrl } from '../../lib/methods/helpers/formatAttachmentUrl'; import { useTheme } from '../../theme'; import Markdown from '../markdown'; import BlurComponent from './Components/OverlayComponent'; import MessageContext from './Context'; import Touchable from './Touchable'; import styles from './styles'; import { isImageBase64 } from '../../lib/methods'; interface IMessageButton { children: React.ReactElement; disabled?: boolean; onPress: () => void; } interface IMessageImage { file: IAttachment; imageUrl?: string; showAttachment?: (file: IAttachment) => void; style?: StyleProp[]; isReply?: boolean; getCustomEmoji?: TGetCustomEmoji; author?: IUserMessage; msg?: string; } const Button = React.memo(({ children, onPress, disabled }: IMessageButton) => { const { colors } = useTheme(); return ( {children} ); }); export const MessageImage = React.memo(({ imgUri, cached, loading }: { imgUri: string; cached: boolean; loading: boolean }) => { const { colors } = useTheme(); return ( <> {!cached ? ( ) : null} ); }); const ImageContainer = ({ file, imageUrl, showAttachment, getCustomEmoji, style, isReply, author, msg }: IMessageImage): React.ReactElement | null => { const [imageCached, setImageCached] = useState(file); const [cached, setCached] = useState(false); const [loading, setLoading] = useState(true); const { theme } = useTheme(); const { baseUrl, user } = useContext(MessageContext); const getUrl = (link?: string) => imageUrl || formatAttachmentUrl(link, user.id, user.token, baseUrl); const img = getUrl(file.image_url); // The param file.title_link is the one that point to image with best quality, however we still need to test the imageUrl // And we cannot be certain whether the file.title_link actually exists. const imgUrlToCache = getUrl(imageCached.title_link || imageCached.image_url); useEffect(() => { const handleCache = async () => { if (img) { const isImageCached = await handleGetMediaCache(); if (isImageCached) { return; } if (isDownloadActive(imgUrlToCache)) { handleResumeDownload(); return; } setLoading(false); await handleAutoDownload(); } }; if (isImageBase64(imgUrlToCache)) { setLoading(false); setCached(true); } else { handleCache(); } }, []); if (!img) { return null; } const handleAutoDownload = async () => { const isCurrentUserAuthor = author?._id === user.id; const isAutoDownloadEnabled = fetchAutoDownloadEnabled('imagesPreferenceDownload'); if (isAutoDownloadEnabled || isCurrentUserAuthor) { await handleDownload(); } }; const updateImageCached = (imgUri: string) => { setImageCached(prev => ({ ...prev, title_link: imgUri })); setCached(true); }; const handleGetMediaCache = async () => { const cachedImageResult = await getMediaCache({ type: 'image', mimeType: imageCached.image_type, urlToCache: imgUrlToCache }); if (cachedImageResult?.exists) { updateImageCached(cachedImageResult.uri); setLoading(false); } return !!cachedImageResult?.exists; }; const handleResumeDownload = async () => { try { setLoading(true); const imageUri = await resumeMediaFile({ downloadUrl: imgUrlToCache }); updateImageCached(imageUri); } catch (e) { setCached(false); } finally { setLoading(false); } }; const handleDownload = async () => { try { setLoading(true); const imageUri = await downloadMediaFile({ downloadUrl: imgUrlToCache, type: 'image', mimeType: imageCached.image_type }); updateImageCached(imageUri); } catch (e) { setCached(false); } finally { setLoading(false); } }; const onPress = async () => { if (loading && isDownloadActive(imgUrlToCache)) { cancelDownload(imgUrlToCache); setLoading(false); setCached(false); return; } if (!cached && !loading) { const isImageCached = await handleGetMediaCache(); if (isImageCached && showAttachment) { showAttachment(imageCached); return; } if (isDownloadActive(imgUrlToCache)) { handleResumeDownload(); return; } handleDownload(); return; } if (!showAttachment) { return; } showAttachment(imageCached); }; if (msg) { return ( ); } return ( ); }; ImageContainer.displayName = 'MessageImageContainer'; MessageImage.displayName = 'MessageImage'; export default ImageContainer;