2023-12-16 00:24:50 +01:00
|
|
|
import { useApiHost } from "@/api";
|
|
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
|
|
|
import useSWR from "swr";
|
|
|
|
import ActivityIndicator from "../ui/activity-indicator";
|
|
|
|
import { useResizeObserver } from "@/hooks/resize-observer";
|
|
|
|
|
|
|
|
type CameraImageProps = {
|
|
|
|
camera: string;
|
|
|
|
onload?: (event: Event) => void;
|
2023-12-16 15:40:00 +01:00
|
|
|
searchParams?: {};
|
|
|
|
stretch?: boolean; // stretch to fit width
|
|
|
|
fitAspect?: number; // shrink to fit height
|
2023-12-16 00:24:50 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
export default function CameraImage({
|
|
|
|
camera,
|
|
|
|
onload,
|
|
|
|
searchParams = "",
|
|
|
|
stretch = false,
|
2023-12-16 15:40:00 +01:00
|
|
|
fitAspect,
|
2023-12-16 00:24:50 +01:00
|
|
|
}: CameraImageProps) {
|
|
|
|
const { data: config } = useSWR("config");
|
|
|
|
const apiHost = useApiHost();
|
|
|
|
const [hasLoaded, setHasLoaded] = useState(false);
|
|
|
|
const containerRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
const canvasRef = useRef<HTMLCanvasElement | null>(null);
|
2023-12-16 15:40:00 +01:00
|
|
|
const [{ width: containerWidth, height: containerHeight }] =
|
|
|
|
useResizeObserver(containerRef);
|
2023-12-16 00:24:50 +01:00
|
|
|
|
|
|
|
// Add scrollbar width (when visible) to the available observer width to eliminate screen juddering.
|
|
|
|
// https://github.com/blakeblackshear/frigate/issues/1657
|
|
|
|
let scrollBarWidth = 0;
|
|
|
|
if (window.innerWidth && document.body.offsetWidth) {
|
|
|
|
scrollBarWidth = window.innerWidth - document.body.offsetWidth;
|
|
|
|
}
|
|
|
|
const availableWidth = scrollBarWidth
|
|
|
|
? containerWidth + scrollBarWidth
|
|
|
|
: containerWidth;
|
|
|
|
|
|
|
|
const { name } = config ? config.cameras[camera] : "";
|
|
|
|
const enabled = config ? config.cameras[camera].enabled : "True";
|
|
|
|
const { width, height } = config
|
|
|
|
? config.cameras[camera].detect
|
|
|
|
: { width: 1, height: 1 };
|
|
|
|
const aspectRatio = width / height;
|
|
|
|
|
|
|
|
const scaledHeight = useMemo(() => {
|
2023-12-16 15:40:00 +01:00
|
|
|
const scaledHeight =
|
|
|
|
aspectRatio < (fitAspect ?? 0)
|
|
|
|
? Math.floor(containerHeight)
|
|
|
|
: Math.floor(availableWidth / aspectRatio);
|
2023-12-16 00:24:50 +01:00
|
|
|
const finalHeight = stretch ? scaledHeight : Math.min(scaledHeight, height);
|
|
|
|
|
|
|
|
if (finalHeight > 0) {
|
|
|
|
return finalHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 100;
|
|
|
|
}, [availableWidth, aspectRatio, height, stretch]);
|
|
|
|
const scaledWidth = useMemo(
|
|
|
|
() => Math.ceil(scaledHeight * aspectRatio - scrollBarWidth),
|
|
|
|
[scaledHeight, aspectRatio, scrollBarWidth]
|
|
|
|
);
|
|
|
|
|
|
|
|
const img = useMemo(() => new Image(), []);
|
|
|
|
img.onload = useCallback(
|
|
|
|
(event: 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 (!config || scaledHeight === 0 || !canvasRef.current) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
img.src = `${apiHost}api/${name}/latest.jpg?h=${scaledHeight}${
|
|
|
|
searchParams ? `&${searchParams}` : ""
|
|
|
|
}`;
|
|
|
|
}, [apiHost, canvasRef, name, img, searchParams, scaledHeight, config]);
|
|
|
|
|
|
|
|
return (
|
2023-12-16 15:40:00 +01:00
|
|
|
<div
|
|
|
|
className={`relative w-full ${
|
|
|
|
fitAspect && aspectRatio < fitAspect ? "h-full flex justify-center" : ""
|
|
|
|
}`}
|
|
|
|
ref={containerRef}
|
|
|
|
>
|
2023-12-16 00:24:50 +01:00
|
|
|
{enabled ? (
|
|
|
|
<canvas
|
|
|
|
data-testid="cameraimage-canvas"
|
|
|
|
height={scaledHeight}
|
|
|
|
ref={canvasRef}
|
|
|
|
width={scaledWidth}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<div className="text-center pt-6">
|
|
|
|
Camera is disabled in config, no stream or snapshot available!
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
{!hasLoaded && enabled ? (
|
|
|
|
<div
|
|
|
|
className="absolute inset-0 flex justify-center"
|
|
|
|
style={{ height: `${scaledHeight}px` }}
|
|
|
|
>
|
|
|
|
<ActivityIndicator />
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|