2022-12-10 10:14:48 +01:00
|
|
|
import './avatar.css';
|
|
|
|
|
2023-03-13 03:10:21 +01:00
|
|
|
import { useRef } from 'preact/hooks';
|
|
|
|
|
2022-12-10 10:14:48 +01:00
|
|
|
const SIZES = {
|
|
|
|
s: 16,
|
|
|
|
m: 20,
|
|
|
|
l: 24,
|
|
|
|
xl: 32,
|
|
|
|
xxl: 50,
|
2022-12-17 17:38:19 +01:00
|
|
|
xxxl: 64,
|
2022-12-10 10:14:48 +01:00
|
|
|
};
|
|
|
|
|
2023-03-13 07:24:53 +01:00
|
|
|
const alphaCache = {};
|
|
|
|
|
2023-06-14 05:15:05 +02:00
|
|
|
const canvas = window.OffscreenCanvas
|
|
|
|
? new OffscreenCanvas(1, 1)
|
|
|
|
: document.createElement('canvas');
|
2023-09-09 08:26:08 +02:00
|
|
|
const ctx = canvas.getContext('2d', {
|
|
|
|
willReadFrequently: true,
|
|
|
|
});
|
2023-06-14 05:15:05 +02:00
|
|
|
|
2023-04-10 18:26:43 +02:00
|
|
|
function Avatar({ url, size, alt = '', squircle, ...props }) {
|
2022-12-10 10:14:48 +01:00
|
|
|
size = SIZES[size] || size || SIZES.m;
|
2023-03-13 03:10:21 +01:00
|
|
|
const avatarRef = useRef();
|
2023-03-21 08:52:26 +01:00
|
|
|
const isMissing = /missing\.png$/.test(url);
|
2022-12-10 10:14:48 +01:00
|
|
|
return (
|
|
|
|
<span
|
2023-03-13 03:10:21 +01:00
|
|
|
ref={avatarRef}
|
2023-04-10 18:26:43 +02:00
|
|
|
class={`avatar ${squircle ? 'squircle' : ''} ${
|
|
|
|
alphaCache[url] ? 'has-alpha' : ''
|
|
|
|
}`}
|
2022-12-10 10:14:48 +01:00
|
|
|
style={{
|
|
|
|
width: size,
|
|
|
|
height: size,
|
|
|
|
}}
|
2022-12-14 15:46:50 +01:00
|
|
|
title={alt}
|
2023-01-21 17:37:46 +01:00
|
|
|
{...props}
|
2022-12-10 10:14:48 +01:00
|
|
|
>
|
|
|
|
{!!url && (
|
2023-03-13 03:10:21 +01:00
|
|
|
<img
|
|
|
|
src={url}
|
|
|
|
width={size}
|
|
|
|
height={size}
|
|
|
|
alt={alt}
|
|
|
|
loading="lazy"
|
2023-06-14 14:31:02 +02:00
|
|
|
decoding="async"
|
2023-03-21 08:52:26 +01:00
|
|
|
crossOrigin={
|
|
|
|
alphaCache[url] === undefined && !isMissing
|
|
|
|
? 'anonymous'
|
|
|
|
: undefined
|
|
|
|
}
|
2023-03-13 07:24:53 +01:00
|
|
|
onError={(e) => {
|
2023-03-13 12:25:00 +01:00
|
|
|
if (e.target.crossOrigin) {
|
|
|
|
e.target.crossOrigin = null;
|
|
|
|
e.target.src = url;
|
|
|
|
}
|
2023-03-13 07:24:53 +01:00
|
|
|
}}
|
2023-03-13 03:10:21 +01:00
|
|
|
onLoad={(e) => {
|
2023-03-13 09:22:41 +01:00
|
|
|
if (avatarRef.current) avatarRef.current.dataset.loaded = true;
|
2023-03-19 06:38:40 +01:00
|
|
|
if (alphaCache[url] !== undefined) return;
|
2023-03-21 08:52:26 +01:00
|
|
|
if (isMissing) return;
|
2023-03-13 07:24:53 +01:00
|
|
|
try {
|
|
|
|
// Check if image has alpha channel
|
2023-06-14 05:15:05 +02:00
|
|
|
const { width, height } = e.target;
|
|
|
|
if (canvas.width !== width) canvas.width = width;
|
|
|
|
if (canvas.height !== height) canvas.height = height;
|
2023-03-13 07:24:53 +01:00
|
|
|
ctx.drawImage(e.target, 0, 0);
|
2023-06-14 05:15:05 +02:00
|
|
|
const allPixels = ctx.getImageData(0, 0, width, height);
|
2023-03-15 08:48:26 +01:00
|
|
|
// At least 10% of pixels have alpha <= 128
|
|
|
|
const hasAlpha =
|
|
|
|
allPixels.data.filter((pixel, i) => i % 4 === 3 && pixel <= 128)
|
|
|
|
.length /
|
|
|
|
(allPixels.data.length / 4) >
|
|
|
|
0.1;
|
2023-03-13 07:24:53 +01:00
|
|
|
if (hasAlpha) {
|
2023-03-14 10:32:06 +01:00
|
|
|
// console.log('hasAlpha', hasAlpha, allPixels.data);
|
2023-03-13 07:24:53 +01:00
|
|
|
avatarRef.current.classList.add('has-alpha');
|
|
|
|
}
|
2023-03-19 06:38:40 +01:00
|
|
|
alphaCache[url] = hasAlpha;
|
2023-06-14 05:15:05 +02:00
|
|
|
ctx.clearRect(0, 0, width, height);
|
2023-03-13 07:24:53 +01:00
|
|
|
} catch (e) {
|
2023-03-21 15:45:35 +01:00
|
|
|
// Silent fail
|
|
|
|
alphaCache[url] = false;
|
2023-03-13 07:24:53 +01:00
|
|
|
}
|
2023-03-13 03:10:21 +01:00
|
|
|
}}
|
|
|
|
/>
|
2022-12-10 10:14:48 +01:00
|
|
|
)}
|
|
|
|
</span>
|
|
|
|
);
|
2022-12-16 06:27:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export default Avatar;
|