stacker.news/components/image.js

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}
/>
)
}