import styles from './text.module.css'
import { useState, useRef, useEffect, useMemo, useCallback } from 'react'
import { extractUrls } from '../lib/md'
import { IMGPROXY_URL_REGEXP, IMG_URL_REGEXP } from '../lib/url'
import FileMissing from '../svgs/file-warning-line.svg'
import { useShowModal } from './modal'
import { useMe } from './me'
import { Dropdown } from 'react-bootstrap'
export function decodeOriginalUrl (imgproxyUrl) {
const parts = imgproxyUrl.split('/')
// base64url is not a known encoding in browsers
// so we need to replace the invalid chars
const b64Url = parts[parts.length - 1].replace(/-/g, '+').replace(/_/, '/')
const originalUrl = Buffer.from(b64Url, 'base64').toString('utf-8')
return originalUrl
}
export const IMG_CACHE_STATES = {
LOADING: 'IS_LOADING',
LOADED: 'IS_LOADED',
ERROR: 'IS_ERROR'
}
// this is the image at public/placeholder_click_to_load.png as a data URI so we don't have to rely on network to render it
const IMAGE_CLICK_TO_LOAD_DATA_URI = ''
const IMAGE_PROCESSING_DATA_URI = ''
export function useImgUrlCache (text, imgproxyUrls) {
const ref = useRef({})
const [imgUrlCache, setImgUrlCache] = useState({})
const me = useMe()
const updateCache = (url, state) => setImgUrlCache((prev) => ({ ...prev, [url]: state }))
useEffect(() => {
const urls = extractUrls(text)
urls.forEach((url) => {
if (IMG_URL_REGEXP.test(url) || !!imgproxyUrls?.[url]) {
// it's probably an image if the regexp matches or if we processed the URL as an image in the worker
updateCache(url, IMG_CACHE_STATES.LOADED)
} else {
// don't use image detection by trying to load as an image if user opted-out of loading external images automatically
if (me?.clickToLoadImg) return
// make sure it's not a false negative by trying to load URL as
const img = new window.Image()
ref.current[url] = img
updateCache(url, IMG_CACHE_STATES.LOADING)
const callback = (state) => {
updateCache(url, state)
delete ref.current[url]
}
img.onload = () => callback(IMG_CACHE_STATES.LOADED)
img.onerror = () => callback(IMG_CACHE_STATES.ERROR)
img.src = url
}
})
return () => {
Object.values(ref.current).forEach((img) => {
img.onload = null
img.onerror = null
img.src = ''
})
}
}, [text])
return imgUrlCache
}
export function ZoomableImage ({ src, topLevel, srcSet: srcSetObj, ...props }) {
const me = useMe()
const showModal = useShowModal()
const [originalUrlConsent, setOriginalUrlConsent] = useState(!me ? true : !me.clickToLoadImg)
// if there is no srcset obj, image is still processing (srcSetObj === undefined) or it wasn't detected as an image by the worker (srcSetObj === null).
// we handle both cases the same as imgproxy errors.
const [imgproxyErr, setImgproxyErr] = useState(!srcSetObj)
const [originalErr, setOriginalErr] = useState()
// backwards compatibility:
// src may already be imgproxy url since we used to replace image urls with imgproxy urls
const originalUrl = IMGPROXY_URL_REGEXP.test(src) ? decodeOriginalUrl(src) : src
// we will fallback to the original error if there was an error with our image proxy
const loadOriginalUrl = !!imgproxyErr
const srcSet = useMemo(() => {
if (!srcSetObj) return undefined
// srcSetObj shape: { [widthDescriptor]: , ... }
return Object.entries(srcSetObj).reduce((acc, [wDescriptor, url], i, arr) => {
return acc + `${url} ${wDescriptor}` + (i < arr.length - 1 ? ', ' : '')
}, '')
}, [srcSetObj])
const sizes = `${(topLevel ? 100 : 66)}vw`
// get source url in best resolution
const bestResSrc = useMemo(() => {
if (!srcSetObj) return undefined
return Object.entries(srcSetObj).reduce((acc, [wDescriptor, url]) => {
const w = Number(wDescriptor.replace(/w$/, ''))
return w > acc.w ? { w, url } : acc
}, { w: 0, url: undefined }).url
}, [srcSetObj])
const onError = useCallback((err) => {
if (!imgproxyErr) {
// first error is imgproxy error since that was loaded
console.error('imgproxy image error:', err)
setImgproxyErr(true)
} else {
// second error is error from original url
console.error('original image error:', err)
setOriginalErr(true)
}
}, [setImgproxyErr, setOriginalErr, imgproxyErr, originalUrl])
const handleClick = useCallback(() => showModal(close => (
), {
fullScreen: true,
overflow: (
{loadOriginalUrl ? 'open in new tab' : 'open original'}
)
}), [showModal, loadOriginalUrl, originalUrl, bestResSrc, onError, props])
if (!src) return null
if ((srcSetObj === undefined) && originalUrlConsent && !originalErr) {
// image is still processing and user is okay with loading original url automatically
return (
setOriginalErr(true)}
{...props}
/>
)
}
if ((srcSetObj === undefined) && !originalUrlConsent && !originalErr) {
// image is still processing and user is not okay with loading original url automatically
const { host } = new URL(originalUrl)
return (
setOriginalUrlConsent(true)}
/>
click to load original from
{host}
)
}
if (originalErr) {
// we already tried original URL: degrade to tag
return (
<>
failed to load image
{originalUrl}
>
)
}
if (imgproxyErr && !originalUrlConsent) {
// respect privacy setting that external images should not be loaded automatically
const { host } = new URL(originalUrl)
return (