All checks were successful
		
		
	
	/ depoly (push) Successful in 1m22s
				
			* add ~platform/DocumentTitle * add titles for some pages
		
			
				
	
	
		
			127 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			127 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import {
 | |
|   JSX,
 | |
|   splitProps,
 | |
|   Component,
 | |
|   createSignal,
 | |
|   onMount,
 | |
|   createRenderEffect,
 | |
|   Show,
 | |
| } from "solid-js";
 | |
| import { css } from "solid-styled";
 | |
| import { decode } from "blurhash";
 | |
| 
 | |
| type ImgProps = {
 | |
|   blurhash?: string;
 | |
|   keepBlur?: boolean;
 | |
| } & JSX.HTMLElementTags["img"];
 | |
| 
 | |
| const Img: Component<ImgProps> = (props) => {
 | |
|   let canvas: HTMLCanvasElement;
 | |
|   let imgE: HTMLImageElement;
 | |
|   const [managed, passthough] = splitProps(props, [
 | |
|     "blurhash",
 | |
|     "keepBlur",
 | |
|     "class",
 | |
|     "classList",
 | |
|     "style",
 | |
|   ]);
 | |
|   const [isImgLoaded, setIsImgLoaded] = createSignal(false);
 | |
|   const [imgSize, setImgSize] = createSignal<{
 | |
|     width: number;
 | |
|     height: number;
 | |
|   }>();
 | |
| 
 | |
|   const isBlurEnabled = () => managed.keepBlur || !isImgLoaded();
 | |
| 
 | |
|   css`
 | |
|     :where(.img-root) {
 | |
|       display: inline-block;
 | |
|       position: relative;
 | |
| 
 | |
|       > img:first-of-type {
 | |
|         object-fit: contain;
 | |
|         object-position: center;
 | |
|         width: 100%;
 | |
|         height: 100%;
 | |
|         visibility: ${isBlurEnabled() ? "hidden" : "initial"};
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     :where(.cover) {
 | |
|       display: ${isBlurEnabled() ? "block" : "none"};
 | |
|       position: absolute;
 | |
|       left: 0;
 | |
|       top: 0;
 | |
|       height: ${`${imgSize()?.height ?? 0}px`};
 | |
|       width: ${`${imgSize()?.width ?? 0}px`};
 | |
|     }
 | |
|   `;
 | |
| 
 | |
|   const onImgLoaded = () => {
 | |
|     setIsImgLoaded(true);
 | |
|     setImgSize({
 | |
|       width: imgE!.width,
 | |
|       height: imgE!.height,
 | |
|     });
 | |
|   };
 | |
| 
 | |
|   const onMetadataLoaded = () => {
 | |
|     setImgSize({
 | |
|       width: imgE!.width,
 | |
|       height: imgE!.height,
 | |
|     });
 | |
|   };
 | |
| 
 | |
|   onMount(() => {
 | |
|     setImgSize((x) => {
 | |
|       const parent = imgE!.parentElement;
 | |
|       if (!parent) return x;
 | |
|       return x
 | |
|         ? x
 | |
|         : {
 | |
|             width: parent.clientWidth,
 | |
|             height: parent.clientHeight,
 | |
|           };
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   return (
 | |
|     <div
 | |
|       classList={{
 | |
|         ...managed.classList,
 | |
|         [managed.class ?? ""]: true,
 | |
|         "img-root": true,
 | |
|       }}
 | |
|       style={managed.style}
 | |
|     >
 | |
|       <Show when={managed.blurhash}>
 | |
|         <canvas
 | |
|           ref={(canvas) => {
 | |
|             createRenderEffect(() => {
 | |
|               if (!managed.blurhash) return;
 | |
|               const ctx = canvas.getContext("2d");
 | |
|               if (!ctx) return;
 | |
|               const size = imgSize();
 | |
|               if (!size) return;
 | |
|               const imgd = ctx?.createImageData(size.width, size.height);
 | |
|               const pixels = decode(managed.blurhash, size.width, size.height);
 | |
|               imgd.data.set(pixels);
 | |
|               ctx.putImageData(imgd, 0, 0);
 | |
|             });
 | |
|           }}
 | |
|           class="cover"
 | |
|           role="presentation"
 | |
|         />
 | |
|       </Show>
 | |
| 
 | |
|       <img
 | |
|         ref={imgE!}
 | |
|         {...passthough}
 | |
|         onLoad={onImgLoaded}
 | |
|         onLoadedMetadata={onMetadataLoaded}
 | |
|       />
 | |
|     </div>
 | |
|   );
 | |
| };
 | |
| 
 | |
| export default Img;
 |