bigbuffet-rw/app/soapbox/components/still-image.tsx

99 lines
2.8 KiB
TypeScript
Raw Normal View History

2023-02-06 10:01:03 -08:00
import clsx from 'clsx';
2022-05-07 10:46:21 -07:00
import React, { useRef } from 'react';
import { useSettings } from 'soapbox/hooks';
export interface IStillImage {
2022-05-07 10:46:21 -07:00
/** Image alt text. */
alt?: string
2022-05-07 10:46:21 -07:00
/** Extra class names for the outer <div> container. */
className?: string
2022-05-07 10:46:21 -07:00
/** URL to the image */
src: string
2022-05-07 10:46:21 -07:00
/** Extra CSS styles on the outer <div> element. */
style?: React.CSSProperties
/** Whether to display the image contained vs filled in its container. */
letterboxed?: boolean
/** Whether to show the file extension in the corner. */
showExt?: boolean
2023-04-24 05:17:51 -07:00
/** Callback function if the image fails to load */
onError?(): void
2022-05-07 10:46:21 -07:00
}
/** Renders images on a canvas, only playing GIFs if autoPlayGif is enabled. */
2023-04-24 05:17:51 -07:00
const StillImage: React.FC<IStillImage> = ({ alt, className, src, style, letterboxed = false, showExt = false, onError }) => {
2022-05-07 10:46:21 -07:00
const settings = useSettings();
const autoPlayGif = settings.get('autoPlayGif');
const canvas = useRef<HTMLCanvasElement>(null);
const img = useRef<HTMLImageElement>(null);
const hoverToPlay = (
src && !autoPlayGif && (src.endsWith('.gif') || src.startsWith('blob:'))
);
const handleImageLoad = () => {
if (hoverToPlay && canvas.current && img.current) {
canvas.current.width = img.current.naturalWidth;
canvas.current.height = img.current.naturalHeight;
canvas.current.getContext('2d')?.drawImage(img.current, 0, 0);
}
};
/** ClassNames shared between the `<img>` and `<canvas>` elements. */
2023-02-06 10:06:44 -08:00
const baseClassName = clsx('block h-full w-full', {
'object-contain': letterboxed,
'object-cover': !letterboxed,
});
2022-05-07 10:46:21 -07:00
return (
<div
data-testid='still-image-container'
2023-02-06 10:06:44 -08:00
className={clsx(className, 'group relative isolate overflow-hidden')}
style={style}
>
<img
src={src}
alt={alt}
ref={img}
onLoad={handleImageLoad}
2023-04-24 05:17:51 -07:00
onError={onError}
2023-02-06 10:01:03 -08:00
className={clsx(baseClassName, {
'invisible group-hover:visible': hoverToPlay,
})}
/>
{hoverToPlay && (
<canvas
ref={canvas}
2023-02-06 10:01:03 -08:00
className={clsx(baseClassName, {
'absolute group-hover:invisible top-0': hoverToPlay,
})}
/>
)}
{(hoverToPlay && showExt) && (
<div className='pointer-events-none absolute bottom-2 left-2 opacity-90 group-hover:hidden'>
<ExtensionBadge ext='GIF' />
</div>
)}
2022-05-07 10:46:21 -07:00
</div>
);
};
interface IExtensionBadge {
/** File extension. */
ext: string
}
/** Badge displaying a file extension. */
const ExtensionBadge: React.FC<IExtensionBadge> = ({ ext }) => {
return (
2023-02-01 14:13:42 -08:00
<div className='inline-flex items-center rounded bg-gray-100 px-2 py-0.5 text-sm font-medium text-gray-900 dark:bg-gray-800 dark:text-gray-100'>
{ext}
</div>
);
};
2022-05-07 10:46:21 -07:00
export default StillImage;