mirror of
https://github.com/blakeblackshear/frigate.git
synced 2025-10-27 10:52:11 +01:00
* Move history card view to separate view and create timeline view * Get custom time scrubber working * Add back nav * Show timeline bounding boxes * Implement seeking limiter * Use browser history to allow back button to close timeline viewer * Fix mobile timeline and add more icons for detections * Play when item is initially visible
301 lines
8.6 KiB
TypeScript
301 lines
8.6 KiB
TypeScript
import { useApiHost } from "@/api";
|
|
import TimelineEventOverlay from "@/components/overlay/TimelineDataOverlay";
|
|
import VideoPlayer from "@/components/player/VideoPlayer";
|
|
import ActivityScrubber, {
|
|
ScrubberItem,
|
|
} from "@/components/scrubber/ActivityScrubber";
|
|
import ActivityIndicator from "@/components/ui/activity-indicator";
|
|
import { FrigateConfig } from "@/types/frigateConfig";
|
|
import {
|
|
getTimelineDetectionIcon,
|
|
getTimelineIcon,
|
|
} from "@/utils/timelineUtil";
|
|
import { renderToStaticMarkup } from "react-dom/server";
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
|
import useSWR from "swr";
|
|
import Player from "video.js/dist/types/player";
|
|
|
|
type HistoryTimelineViewProps = {
|
|
playback: TimelinePlayback;
|
|
isMobile: boolean;
|
|
};
|
|
|
|
export default function HistoryTimelineView({
|
|
playback,
|
|
isMobile,
|
|
}: HistoryTimelineViewProps) {
|
|
const apiHost = useApiHost();
|
|
const { data: config } = useSWR<FrigateConfig>("config");
|
|
const timezone = useMemo(
|
|
() =>
|
|
config?.ui?.timezone || Intl.DateTimeFormat().resolvedOptions().timeZone,
|
|
[config]
|
|
);
|
|
|
|
const hasRelevantPreview = playback.relevantPreview != undefined;
|
|
|
|
const playerRef = useRef<Player | undefined>(undefined);
|
|
const previewRef = useRef<Player | undefined>(undefined);
|
|
|
|
const [scrubbing, setScrubbing] = useState(false);
|
|
const [focusedItem, setFocusedItem] = useState<Timeline | undefined>(
|
|
undefined
|
|
);
|
|
|
|
const [seeking, setSeeking] = useState(false);
|
|
const [timeToSeek, setTimeToSeek] = useState<number | undefined>(undefined);
|
|
|
|
const annotationOffset = useMemo(() => {
|
|
if (!config) {
|
|
return 0;
|
|
}
|
|
|
|
return (
|
|
(config.cameras[playback.camera]?.detect?.annotation_offset || 0) / 1000
|
|
);
|
|
}, [config, playback]);
|
|
|
|
const timelineTime = useMemo(() => {
|
|
if (!playback) {
|
|
return 0;
|
|
}
|
|
|
|
return playback.timelineItems.at(0)!!.timestamp;
|
|
}, [playback]);
|
|
const playbackTimes = useMemo(() => {
|
|
const date = new Date(timelineTime * 1000);
|
|
date.setMinutes(0, 0, 0);
|
|
const startTime = date.getTime() / 1000;
|
|
date.setHours(date.getHours() + 1);
|
|
const endTime = date.getTime() / 1000;
|
|
return {
|
|
start: parseInt(startTime.toFixed(1)),
|
|
end: parseInt(endTime.toFixed(1)),
|
|
};
|
|
}, [timelineTime]);
|
|
|
|
const recordingParams = useMemo(() => {
|
|
return {
|
|
before: playbackTimes.end,
|
|
after: playbackTimes.start,
|
|
};
|
|
}, [playbackTimes]);
|
|
const { data: recordings } = useSWR<Recording[]>(
|
|
playback ? [`${playback.camera}/recordings`, recordingParams] : null,
|
|
{ revalidateOnFocus: false }
|
|
);
|
|
|
|
const playbackUri = useMemo(() => {
|
|
if (!playback) {
|
|
return "";
|
|
}
|
|
|
|
const date = new Date(playbackTimes.start * 1000);
|
|
return `${apiHost}vod/${date.getFullYear()}-${
|
|
date.getMonth() + 1
|
|
}/${date.getDate()}/${date.getHours()}/${
|
|
playback.camera
|
|
}/${timezone.replaceAll("/", ",")}/master.m3u8`;
|
|
}, [playbackTimes]);
|
|
|
|
const onSelectItem = useCallback(
|
|
(data: { items: number[] }) => {
|
|
if (data.items.length > 0) {
|
|
const selected = data.items[0];
|
|
setFocusedItem(
|
|
playback.timelineItems.find(
|
|
(timeline) => timeline.timestamp == selected
|
|
)
|
|
);
|
|
playerRef.current?.pause();
|
|
|
|
let seekSeconds = 0;
|
|
(recordings || []).every((segment) => {
|
|
// if the next segment is past the desired time, stop calculating
|
|
if (segment.start_time > selected) {
|
|
return false;
|
|
}
|
|
|
|
if (segment.end_time < selected) {
|
|
seekSeconds += segment.end_time - segment.start_time;
|
|
return true;
|
|
}
|
|
|
|
seekSeconds +=
|
|
segment.end_time -
|
|
segment.start_time -
|
|
(segment.end_time - selected);
|
|
return true;
|
|
});
|
|
playerRef.current?.currentTime(seekSeconds);
|
|
}
|
|
},
|
|
[annotationOffset, recordings, playerRef]
|
|
);
|
|
|
|
const onScrubTime = useCallback(
|
|
(data: { time: Date }) => {
|
|
if (!hasRelevantPreview) {
|
|
return;
|
|
}
|
|
|
|
if (playerRef.current?.paused() == false) {
|
|
setScrubbing(true);
|
|
playerRef.current?.pause();
|
|
}
|
|
|
|
const seekTimestamp = data.time.getTime() / 1000;
|
|
const seekTime = seekTimestamp - playback.relevantPreview!!.start;
|
|
setTimeToSeek(Math.round(seekTime));
|
|
},
|
|
[scrubbing, playerRef]
|
|
);
|
|
|
|
const onStopScrubbing = useCallback(
|
|
(data: { time: Date }) => {
|
|
const playbackTime = data.time.getTime() / 1000;
|
|
playerRef.current?.currentTime(playbackTime - playbackTimes.start);
|
|
setScrubbing(false);
|
|
playerRef.current?.play();
|
|
},
|
|
[playerRef]
|
|
);
|
|
|
|
// handle seeking to next frame when seek is finished
|
|
useEffect(() => {
|
|
if (seeking) {
|
|
return;
|
|
}
|
|
|
|
if (timeToSeek && timeToSeek != previewRef.current?.currentTime()) {
|
|
setSeeking(true);
|
|
previewRef.current?.currentTime(timeToSeek);
|
|
}
|
|
}, [timeToSeek, seeking]);
|
|
|
|
if (!config || !recordings) {
|
|
return <ActivityIndicator />;
|
|
}
|
|
|
|
return (
|
|
<div className="w-full">
|
|
<>
|
|
<div
|
|
className={`relative ${
|
|
hasRelevantPreview && scrubbing ? "hidden" : "visible"
|
|
}`}
|
|
>
|
|
<VideoPlayer
|
|
options={{
|
|
preload: "auto",
|
|
autoplay: true,
|
|
sources: [
|
|
{
|
|
src: playbackUri,
|
|
type: "application/vnd.apple.mpegurl",
|
|
},
|
|
],
|
|
}}
|
|
seekOptions={{ forward: 10, backward: 5 }}
|
|
onReady={(player) => {
|
|
playerRef.current = player;
|
|
player.currentTime(timelineTime - playbackTimes.start);
|
|
player.on("playing", () => {
|
|
setFocusedItem(undefined);
|
|
});
|
|
}}
|
|
onDispose={() => {
|
|
playerRef.current = undefined;
|
|
}}
|
|
>
|
|
{config && focusedItem ? (
|
|
<TimelineEventOverlay
|
|
timeline={focusedItem}
|
|
cameraConfig={config.cameras[playback.camera]}
|
|
/>
|
|
) : undefined}
|
|
</VideoPlayer>
|
|
</div>
|
|
{hasRelevantPreview && (
|
|
<div className={`${scrubbing ? "visible" : "hidden"}`}>
|
|
<VideoPlayer
|
|
options={{
|
|
preload: "auto",
|
|
autoplay: false,
|
|
controls: false,
|
|
muted: true,
|
|
loadingSpinner: false,
|
|
sources: [
|
|
{
|
|
src: `${playback.relevantPreview?.src}`,
|
|
type: "video/mp4",
|
|
},
|
|
],
|
|
}}
|
|
seekOptions={{}}
|
|
onReady={(player) => {
|
|
previewRef.current = player;
|
|
player.on("seeked", () => setSeeking(false));
|
|
}}
|
|
onDispose={() => {
|
|
previewRef.current = undefined;
|
|
}}
|
|
/>
|
|
</div>
|
|
)}
|
|
</>
|
|
<div className="m-1">
|
|
{playback != undefined && (
|
|
<ActivityScrubber
|
|
items={timelineItemsToScrubber(playback.timelineItems)}
|
|
timeBars={
|
|
hasRelevantPreview
|
|
? [{ time: new Date(timelineTime * 1000), id: "playback" }]
|
|
: []
|
|
}
|
|
options={{
|
|
...(isMobile && {
|
|
start: new Date(
|
|
Math.max(playbackTimes.start, timelineTime - 300) * 1000
|
|
),
|
|
end: new Date(
|
|
Math.min(playbackTimes.end, timelineTime + 300) * 1000
|
|
),
|
|
}),
|
|
snap: null,
|
|
min: new Date(playbackTimes.start * 1000),
|
|
max: new Date(playbackTimes.end * 1000),
|
|
timeAxis: isMobile ? { scale: "minute", step: 5 } : {},
|
|
}}
|
|
timechangeHandler={onScrubTime}
|
|
timechangedHandler={onStopScrubbing}
|
|
selectHandler={onSelectItem}
|
|
/>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function timelineItemsToScrubber(items: Timeline[]): ScrubberItem[] {
|
|
return items.map((item) => {
|
|
return {
|
|
id: item.timestamp,
|
|
content: getTimelineContentElement(item),
|
|
start: new Date(item.timestamp * 1000),
|
|
end: new Date(item.timestamp * 1000),
|
|
type: "box",
|
|
};
|
|
});
|
|
}
|
|
|
|
function getTimelineContentElement(item: Timeline): HTMLElement {
|
|
const output = document.createElement(`div-${item.timestamp}`);
|
|
output.innerHTML = renderToStaticMarkup(
|
|
<div className="flex items-center">
|
|
{getTimelineDetectionIcon(item)} : {getTimelineIcon(item)}
|
|
</div>
|
|
);
|
|
return output;
|
|
}
|