import { h } from 'preact'; import ActivityIndicator from './ActivityIndicator'; import { useApiHost, useConfig } from '../api'; import { useCallback, useEffect, useMemo, useRef, useState } from 'preact/hooks'; export default function CameraImage({ camera, onload, searchParams = '' }) { const { data: config } = useConfig(); const apiHost = useApiHost(); const [availableWidth, setAvailableWidth] = useState(0); const [hasLoaded, setHasLoaded] = useState(false); const containerRef = useRef(null); const canvasRef = useRef(null); const { name, width, height } = config.cameras[camera]; const aspectRatio = width / height; const resizeObserver = useMemo(() => { return new ResizeObserver((entries) => { window.requestAnimationFrame(() => { if (Array.isArray(entries) && entries.length) { setAvailableWidth(entries[0].contentRect.width); } }); }); }, []); useEffect(() => { resizeObserver.observe(containerRef.current); }, [resizeObserver, containerRef]); const scaledHeight = useMemo(() => Math.min(Math.ceil(availableWidth / aspectRatio), height), [ availableWidth, aspectRatio, height, ]); const scaledWidth = useMemo(() => Math.ceil(scaledHeight * aspectRatio), [scaledHeight, aspectRatio]); const img = useMemo(() => new Image(), []); img.onload = useCallback( (event) => { setHasLoaded(true); if (canvasRef.current) { const ctx = canvasRef.current.getContext('2d'); ctx.drawImage(img, 0, 0, scaledWidth, scaledHeight); } onload && onload(event); }, [img, scaledHeight, scaledWidth, setHasLoaded, onload, canvasRef] ); useEffect(() => { if (scaledHeight === 0 || !canvasRef.current) { return; } img.src = `${apiHost}/api/${name}/latest.jpg?h=${scaledHeight}${searchParams ? `&${searchParams}` : ''}`; }, [apiHost, canvasRef, name, img, searchParams, scaledHeight]); return (