222 lines
31 KiB
JavaScript
222 lines
31 KiB
JavaScript
|
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 <img>
|
||
|
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]: <imgproxyUrl>, ... }
|
||
|
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 => (
|
||
|
<div
|
||
|
className='d-grid w-100 h-100' style={{ placeContent: 'center' }} onClick={close}
|
||
|
>
|
||
|
<img
|
||
|
style={{ cursor: 'zoom-out', maxWidth: '100%', maxHeight: '100%', minHeight: 0, minWidth: 0 }}
|
||
|
// also load original url in fullscreen if the original url was loaded
|
||
|
src={loadOriginalUrl ? originalUrl : bestResSrc}
|
||
|
onError={onError}
|
||
|
{...props}
|
||
|
/>
|
||
|
</div>
|
||
|
), {
|
||
|
fullScreen: true,
|
||
|
overflow: (
|
||
|
<Dropdown.Item
|
||
|
href={originalUrl} target='_blank' rel='noreferrer'
|
||
|
>
|
||
|
{loadOriginalUrl ? 'open in new tab' : 'open original'}
|
||
|
</Dropdown.Item>)
|
||
|
}), [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 (
|
||
|
<img
|
||
|
className={topLevel ? styles.topLevel : undefined}
|
||
|
style={{ cursor: 'zoom-in', maxHeight: topLevel ? '35vh' : '25vh' }}
|
||
|
src={originalUrl}
|
||
|
onClick={handleClick}
|
||
|
onError={() => 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 (
|
||
|
<div style={{ width: '256px' }}>
|
||
|
<img
|
||
|
className={topLevel ? styles.topLevel : undefined}
|
||
|
src={IMAGE_PROCESSING_DATA_URI} width='256px' height='256px'
|
||
|
style={{ cursor: 'pointer' }} onClick={() => setOriginalUrlConsent(true)}
|
||
|
/>
|
||
|
<div className='text-muted fst-italic text-center'>click to load original from</div>
|
||
|
<div className='text-muted fst-italic text-center'>{host}</div>
|
||
|
</div>
|
||
|
)
|
||
|
}
|
||
|
|
||
|
if (originalErr) {
|
||
|
// we already tried original URL: degrade <img> to <a> tag
|
||
|
return (
|
||
|
<>
|
||
|
<span className='d-flex align-items-baseline text-warning-emphasis fw-bold pb-1'>
|
||
|
<FileMissing width={18} height={18} className='fill-warning me-1 align-self-center' />
|
||
|
failed to load image
|
||
|
</span>
|
||
|
<a target='_blank' href={originalUrl} rel='noreferrer'>{originalUrl}</a>
|
||
|
</>
|
||
|
)
|
||
|
}
|
||
|
|
||
|
if (imgproxyErr && !originalUrlConsent) {
|
||
|
// respect privacy setting that external images should not be loaded automatically
|
||
|
const { host } = new URL(originalUrl)
|
||
|
return (
|
||
|
<div style={{ width: '256px' }}>
|
||
|
<div className='d-flex align-items-baseline text-warning-emphasis fw-bold pb-1 justify-content-center'>
|
||
|
<FileMissing width={18} height={18} className='fill-warning me-1 align-self-center' />
|
||
|
image proxy error
|
||
|
</div>
|
||
|
<img
|
||
|
className={topLevel ? styles.topLevel : undefined}
|
||
|
src={IMAGE_CLICK_TO_LOAD_DATA_URI} width='256px' height='256px'
|
||
|
style={{ cursor: 'pointer' }} onClick={() => setOriginalUrlConsent(true)}
|
||
|
/>
|
||
|
<div className='text-muted fst-italic text-center'>from {host}</div>
|
||
|
</div>
|
||
|
)
|
||
|
}
|
||
|
|
||
|
return (
|
||
|
<img
|
||
|
className={topLevel ? styles.topLevel : undefined}
|
||
|
style={{ cursor: 'zoom-in', maxHeight: topLevel ? '35vh' : '25vh' }}
|
||
|
// browsers that don't support srcSet and sizes will use src. use best resolution possible in that case
|
||
|
src={loadOriginalUrl ? originalUrl : bestResSrc}
|
||
|
// we need to disable srcset and sizes to force browsers to use src
|
||
|
srcSet={loadOriginalUrl ? undefined : srcSet}
|
||
|
sizes={loadOriginalUrl ? undefined : sizes}
|
||
|
onClick={handleClick}
|
||
|
onError={onError}
|
||
|
{...props}
|
||
|
/>
|
||
|
)
|
||
|
}
|