mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-03-12 19:37:35 +03:00
Some checks are pending
CI / AMD64 Build (push) Waiting to run
CI / ARM Build (push) Waiting to run
CI / Jetson Jetpack 6 (push) Waiting to run
CI / AMD64 Extra Build (push) Blocked by required conditions
CI / ARM Extra Build (push) Blocked by required conditions
CI / Synaptics Build (push) Blocked by required conditions
CI / Assemble and push default build (push) Blocked by required conditions
* debug replay implementation * fix masks after dev rebase * fix squash merge issues * fix * fix * fix * no need to write debug replay camera to config * camera and filter button and dropdown * add filters * add ability to edit motion and object config for debug replay * add debug draw overlay to debug replay * add guard to prevent crash when camera is no longer in camera_states * fix overflow due to radix absolutely positioned elements * increase number of messages * ensure deep_merge replaces existing list values when override is true * add back button * add debug replay to explore and review menus * clean up * clean up * update instructions to prevent exposing exception info * fix typing * refactor output logic * refactor with helper function * move init to function for consistency
100 lines
2.7 KiB
TypeScript
100 lines
2.7 KiB
TypeScript
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
|
import { useWsMessageSubscribe, WsFeedMessage } from "@/api/ws";
|
|
import { extractCameraName } from "@/utils/wsUtil";
|
|
|
|
type UseWsMessageBufferReturn = {
|
|
messages: WsFeedMessage[];
|
|
clear: () => void;
|
|
};
|
|
|
|
type MessageFilter = {
|
|
cameraFilter?: string | string[]; // "all", specific camera name, or array of camera names (undefined in array = all)
|
|
};
|
|
|
|
export function useWsMessageBuffer(
|
|
maxSize: number = 2000,
|
|
paused: boolean = false,
|
|
filter?: MessageFilter,
|
|
): UseWsMessageBufferReturn {
|
|
const bufferRef = useRef<WsFeedMessage[]>([]);
|
|
const [version, setVersion] = useState(0);
|
|
const pausedRef = useRef(paused);
|
|
const filterRef = useRef(filter);
|
|
|
|
pausedRef.current = paused;
|
|
filterRef.current = filter;
|
|
|
|
const batchTimerRef = useRef<ReturnType<typeof setInterval> | null>(null);
|
|
const dirtyRef = useRef(false);
|
|
|
|
useEffect(() => {
|
|
batchTimerRef.current = setInterval(() => {
|
|
if (dirtyRef.current) {
|
|
dirtyRef.current = false;
|
|
setVersion((v) => v + 1);
|
|
}
|
|
}, 200);
|
|
|
|
return () => {
|
|
if (batchTimerRef.current) {
|
|
clearInterval(batchTimerRef.current);
|
|
}
|
|
};
|
|
}, []);
|
|
|
|
const shouldIncludeMessage = useCallback((msg: WsFeedMessage): boolean => {
|
|
const currentFilter = filterRef.current;
|
|
if (!currentFilter) return true;
|
|
|
|
// Check camera filter
|
|
const cf = currentFilter.cameraFilter;
|
|
if (cf !== undefined) {
|
|
if (Array.isArray(cf)) {
|
|
// Array of cameras: include messages matching any camera in the list
|
|
const msgCamera = extractCameraName(msg);
|
|
if (msgCamera && !cf.includes(msgCamera)) {
|
|
return false;
|
|
}
|
|
} else if (cf !== "all") {
|
|
// Single string camera filter
|
|
const msgCamera = extractCameraName(msg);
|
|
if (msgCamera !== cf) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}, []);
|
|
|
|
useWsMessageSubscribe(
|
|
useCallback(
|
|
(msg: WsFeedMessage) => {
|
|
if (pausedRef.current) return;
|
|
if (!shouldIncludeMessage(msg)) return;
|
|
|
|
const buf = bufferRef.current;
|
|
buf.push(msg);
|
|
if (buf.length > maxSize) {
|
|
buf.splice(0, buf.length - maxSize);
|
|
}
|
|
dirtyRef.current = true;
|
|
},
|
|
[shouldIncludeMessage, maxSize],
|
|
),
|
|
);
|
|
|
|
const clear = useCallback(() => {
|
|
bufferRef.current = [];
|
|
setVersion((v) => v + 1);
|
|
}, []);
|
|
|
|
// version is used to trigger re-renders; we spread the buffer
|
|
// into a new array so that downstream useMemo dependencies
|
|
// see a new reference and recompute.
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
const messages = useMemo(() => [...bufferRef.current], [version]);
|
|
|
|
return { messages, clear };
|
|
}
|