Move history card view to separate view and create timeline view

This commit is contained in:
Nick Mowen 2023-12-21 12:09:24 -07:00
parent 5c7a596515
commit d5b7adefd9
4 changed files with 327 additions and 111 deletions

View File

@ -4,6 +4,8 @@ import {
TimelineGroup, TimelineGroup,
TimelineItem, TimelineItem,
TimelineOptions, TimelineOptions,
DateType,
IdType,
} from "vis-timeline"; } from "vis-timeline";
import type { DataGroup, DataItem, TimelineEvents } from "vis-timeline/types"; import type { DataGroup, DataItem, TimelineEvents } from "vis-timeline/types";
import "./scrubber.css"; import "./scrubber.css";
@ -73,12 +75,15 @@ const domEvents: TimelineEventsWithMissing[] = [
type ActivityScrubberProps = { type ActivityScrubberProps = {
items: TimelineItem[]; items: TimelineItem[];
midBar: boolean;
timeBars: { time: DateType; id?: IdType | undefined }[];
groups?: TimelineGroup[]; groups?: TimelineGroup[];
options?: TimelineOptions; options?: TimelineOptions;
} & TimelineEventsHandlers; } & TimelineEventsHandlers;
function ActivityScrubber({ function ActivityScrubber({
items, items,
timeBars,
groups, groups,
options, options,
...eventHandlers ...eventHandlers
@ -130,6 +135,12 @@ function ActivityScrubber({
options options
); );
if (timeBars) {
timeBars.forEach((bar) => {
timelineInstance.addCustomTime(bar.time, bar.id);
});
}
domEvents.forEach((event) => { domEvents.forEach((event) => {
const eventHandler = eventHandlers[`${event}Handler`]; const eventHandler = eventHandlers[`${event}Handler`];
if (typeof eventHandler === "function") { if (typeof eventHandler === "function") {
@ -174,7 +185,12 @@ function ActivityScrubber({
if (items) timelineRef.current.timeline.setItems(items); if (items) timelineRef.current.timeline.setItems(items);
}, [items, groups, options, currentTime, eventHandlers]); }, [items, groups, options, currentTime, eventHandlers]);
return <div ref={containerRef} />; return (
<div>
<div ref={containerRef} />
<div className="absolute bg-red-500 w-[2px]" />
</div>
);
} }
export default ActivityScrubber; export default ActivityScrubber;

View File

@ -1,11 +1,9 @@
import { useCallback, useMemo, useRef, useState } from "react"; import { useCallback, useMemo, useState } from "react";
import useSWR from "swr"; import useSWR from "swr";
import useSWRInfinite from "swr/infinite"; import useSWRInfinite from "swr/infinite";
import { FrigateConfig } from "@/types/frigateConfig"; import { FrigateConfig } from "@/types/frigateConfig";
import Heading from "@/components/ui/heading"; import Heading from "@/components/ui/heading";
import ActivityIndicator from "@/components/ui/activity-indicator"; import ActivityIndicator from "@/components/ui/activity-indicator";
import HistoryCard from "@/components/card/HistoryCard";
import { formatUnixTimestampToDateTime } from "@/utils/dateUtil";
import axios from "axios"; import axios from "axios";
import TimelinePlayerCard from "@/components/card/TimelinePlayerCard"; import TimelinePlayerCard from "@/components/card/TimelinePlayerCard";
import { getHourlyTimelineData } from "@/utils/historyUtil"; import { getHourlyTimelineData } from "@/utils/historyUtil";
@ -21,6 +19,8 @@ import {
} from "@/components/ui/alert-dialog"; } from "@/components/ui/alert-dialog";
import HistoryFilterPopover from "@/components/filter/HistoryFilterPopover"; import HistoryFilterPopover from "@/components/filter/HistoryFilterPopover";
import useApiFilter from "@/hooks/use-api-filter"; import useApiFilter from "@/hooks/use-api-filter";
import HistoryCardView from "@/views/history/HistoryCardView";
import HistoryTimelineView from "@/views/history/HistoryTimelineView";
const API_LIMIT = 200; const API_LIMIT = 200;
@ -100,26 +100,6 @@ function History() {
const isDone = const isDone =
(timelinePages?.[timelinePages.length - 1]?.count ?? 0) < API_LIMIT; (timelinePages?.[timelinePages.length - 1]?.count ?? 0) < API_LIMIT;
// hooks for infinite scroll
const observer = useRef<IntersectionObserver | null>();
const lastTimelineRef = useCallback(
(node: HTMLElement | null) => {
if (isValidating) return;
if (observer.current) observer.current.disconnect();
try {
observer.current = new IntersectionObserver((entries) => {
if (entries[0].isIntersecting && !isDone) {
setSize(size + 1);
}
});
if (node) observer.current.observe(node);
} catch (e) {
// no op
}
},
[size, setSize, isValidating, isDone]
);
const [itemsToDelete, setItemsToDelete] = useState<string[] | null>(null); const [itemsToDelete, setItemsToDelete] = useState<string[] | null>(null);
const onDelete = useCallback( const onDelete = useCallback(
async (timeline: Card) => { async (timeline: Card) => {
@ -162,10 +142,12 @@ function History() {
<> <>
<div className="flex justify-between"> <div className="flex justify-between">
<Heading as="h2">History</Heading> <Heading as="h2">History</Heading>
<HistoryFilterPopover {!playback && (
filter={historyFilter} <HistoryFilterPopover
onUpdateFilter={(filter) => setHistoryFilter(filter)} filter={historyFilter}
/> onUpdateFilter={(filter) => setHistoryFilter(filter)}
/>
)}
</div> </div>
<AlertDialog <AlertDialog
@ -194,92 +176,27 @@ function History() {
</AlertDialog> </AlertDialog>
<TimelinePlayerCard <TimelinePlayerCard
timeline={playback} timeline={undefined}
onDismiss={() => setPlayback(undefined)} onDismiss={() => setPlayback(undefined)}
/> />
<div> <>
{Object.entries(timelineCards) {playback == undefined && (
.reverse() <HistoryCardView
.map(([day, timelineDay], dayIdx) => { timelineCards={timelineCards}
return ( allPreviews={allPreviews}
<div key={day}> isMobileView={shouldAutoPlay}
<Heading isValidating={isValidating}
className="sticky py-2 -top-4 left-0 bg-background w-full z-20" isDone={isDone}
as="h3" onNextPage={() => {
> setSize(size + 1);
{formatUnixTimestampToDateTime(parseInt(day), { }}
strftime_fmt: "%A %b %d", onDelete={onDelete}
time_style: "medium", onItemSelected={(card) => setPlayback(card)}
date_style: "medium", />
})} )}
</Heading> {playback != undefined && <HistoryTimelineView card={playback} />}
{Object.entries(timelineDay).map( </>
([hour, timelineHour], hourIdx) => {
if (Object.values(timelineHour).length == 0) {
return <div key={hour}></div>;
}
const lastRow =
dayIdx == Object.values(timelineCards).length - 1 &&
hourIdx == Object.values(timelineDay).length - 1;
const previewMap: { [key: string]: Preview | undefined } =
{};
return (
<div key={hour} ref={lastRow ? lastTimelineRef : null}>
<Heading as="h4">
{formatUnixTimestampToDateTime(parseInt(hour), {
strftime_fmt:
config.ui.time_format == "24hour"
? "%H:00"
: "%I:00 %p",
time_style: "medium",
date_style: "medium",
})}
</Heading>
<div className="flex flex-wrap">
{Object.entries(timelineHour)
.reverse()
.map(([key, timeline]) => {
const startTs = Object.values(timeline.entries)[0]
.timestamp;
let relevantPreview = previewMap[timeline.camera];
if (relevantPreview == undefined) {
relevantPreview = previewMap[timeline.camera] =
Object.values(allPreviews || []).find(
(preview) =>
preview.camera == timeline.camera &&
preview.start < startTs &&
preview.end > startTs
);
}
return (
<HistoryCard
key={key}
timeline={timeline}
shouldAutoPlay={shouldAutoPlay}
relevantPreview={relevantPreview}
onClick={() => {
setPlayback(timeline);
}}
onDelete={() => onDelete(timeline)}
/>
);
})}
</div>
{lastRow && !isDone && <ActivityIndicator />}
</div>
);
}
)}
</div>
);
})}
</div>
</> </>
); );
} }

View File

@ -0,0 +1,135 @@
import HistoryCard from "@/components/card/HistoryCard";
import ActivityIndicator from "@/components/ui/activity-indicator";
import Heading from "@/components/ui/heading";
import { FrigateConfig } from "@/types/frigateConfig";
import { formatUnixTimestampToDateTime } from "@/utils/dateUtil";
import { useCallback, useRef } from "react";
import useSWR from "swr";
type HistoryCardViewProps = {
timelineCards: CardsData | never[];
allPreviews: Preview[] | undefined;
isMobileView: boolean;
isValidating: boolean;
isDone: boolean;
onNextPage: () => void;
onDelete: (card: Card) => void;
onItemSelected: (card: Card) => void;
};
export default function HistoryCardView({
timelineCards,
allPreviews,
isMobileView,
isValidating,
isDone,
onNextPage,
onDelete,
onItemSelected,
}: HistoryCardViewProps) {
const { data: config } = useSWR<FrigateConfig>("config");
// hooks for infinite scroll
const observer = useRef<IntersectionObserver | null>();
const lastTimelineRef = useCallback(
(node: HTMLElement | null) => {
if (isValidating) return;
if (observer.current) observer.current.disconnect();
try {
observer.current = new IntersectionObserver((entries) => {
if (entries[0].isIntersecting && !isDone) {
onNextPage();
}
});
if (node) observer.current.observe(node);
} catch (e) {
// no op
}
},
[isValidating, isDone]
);
return (
<>
{Object.entries(timelineCards)
.reverse()
.map(([day, timelineDay], dayIdx) => {
return (
<div key={day}>
<Heading
className="sticky py-2 -top-4 left-0 bg-background w-full z-20"
as="h3"
>
{formatUnixTimestampToDateTime(parseInt(day), {
strftime_fmt: "%A %b %d",
time_style: "medium",
date_style: "medium",
})}
</Heading>
{Object.entries(timelineDay).map(
([hour, timelineHour], hourIdx) => {
if (Object.values(timelineHour).length == 0) {
return <div key={hour}></div>;
}
const lastRow =
dayIdx == Object.values(timelineCards).length - 1 &&
hourIdx == Object.values(timelineDay).length - 1;
const previewMap: { [key: string]: Preview | undefined } = {};
return (
<div key={hour} ref={lastRow ? lastTimelineRef : null}>
<Heading as="h4">
{formatUnixTimestampToDateTime(parseInt(hour), {
strftime_fmt:
config?.ui.time_format == "24hour"
? "%H:00"
: "%I:00 %p",
time_style: "medium",
date_style: "medium",
})}
</Heading>
<div className="flex flex-wrap">
{Object.entries(timelineHour)
.reverse()
.map(([key, timeline]) => {
const startTs = Object.values(timeline.entries)[0]
.timestamp;
let relevantPreview = previewMap[timeline.camera];
if (relevantPreview == undefined) {
relevantPreview = previewMap[timeline.camera] =
Object.values(allPreviews || []).find(
(preview) =>
preview.camera == timeline.camera &&
preview.start < startTs &&
preview.end > startTs
);
}
return (
<HistoryCard
key={key}
timeline={timeline}
shouldAutoPlay={isMobileView}
relevantPreview={relevantPreview}
onClick={() => {
onItemSelected(timeline);
}}
onDelete={() => onDelete(timeline)}
/>
);
})}
</div>
{lastRow && !isDone && <ActivityIndicator />}
</div>
);
}
)}
</div>
);
})}
</>
);
}

View File

@ -0,0 +1,148 @@
import { useApiHost } from "@/api";
import VideoPlayer from "@/components/player/VideoPlayer";
import ActivityScrubber from "@/components/scrubber/ActivityScrubber";
import {
getTimelineIcon,
getTimelineItemDescription,
} from "@/utils/timelineUtil";
import { useMemo, useRef, useState } from "react";
import { LuDog } from "react-icons/lu";
import Player from "video.js/dist/types/player";
type HistoryTimelineViewProps = {
card: Card;
};
export default function HistoryTimelineView({
card,
}: HistoryTimelineViewProps) {
const apiHost = useApiHost();
const playerRef = useRef<Player | undefined>(undefined);
const previewRef = useRef<Player | undefined>(undefined);
const [scrubbing, setScrubbing] = useState(false);
const relevantPreview = {
src: "http://localhost:5173/clips/previews/side_cam/1703174400.071426-1703178000.011979.mp4",
start: 1703174400.071426,
end: 1703178000.011979,
};
const timelineTime = useMemo(() => card.entries.at(0)!!.timestamp, [card]);
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: startTime.toFixed(1), end: endTime.toFixed(1) };
}, [timelineTime]);
const playbackUri = useMemo(() => {
if (!card) {
return "";
}
return `${apiHost}vod/${card?.camera}/start/${playbackTimes.start}/end/${playbackTimes.end}/master.m3u8`;
}, [card, playbackTimes]);
return (
<>
<div className="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 -mr-[640px]">
<div className={`2xl:w-[1280px]`}>
<div className={`${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 - parseInt(playbackTimes.start)
);
player.on("playing", () => {
//setSelectedItem(undefined);
});
}}
onDispose={() => {
playerRef.current = undefined;
}}
/>
</div>
<div className={`${scrubbing ? "visible" : "hidden"}`}>
<VideoPlayer
options={{
preload: "auto",
autoplay: false,
controls: false,
muted: true,
loadingSpinner: false,
sources: [
{
src: `${relevantPreview.src}`,
type: "video/mp4",
},
],
}}
seekOptions={{}}
onReady={(player) => {
previewRef.current = player;
}}
onDispose={() => {
previewRef.current = undefined;
}}
/>
</div>
<ActivityScrubber
midBar={true}
items={timelineItemsToScrubber(card.entries)}
options={{
min: new Date(parseInt(playbackTimes.start) * 1000),
max: new Date(parseInt(playbackTimes.end) * 1000),
}}
rangechangeHandler={(time) => {
if (!scrubbing) {
playerRef.current?.pause();
setScrubbing(true);
}
const rangeStart = time.start.getTime() / 1000;
const midTime =
rangeStart + (time.end.getTime() / 1000 - rangeStart / 2);
previewRef.current?.currentTime(midTime - relevantPreview.start);
}}
rangechangedHandler={(data) => {
const playbackTime = data.time.getTime() / 1000;
playerRef.current?.currentTime(
playbackTime - parseInt(playbackTimes.start)
);
setScrubbing(false);
playerRef.current?.play();
}}
/>
</div>
</div>
</>
);
}
function timelineItemsToScrubber(items: Timeline[]) {
return items.map((item) => {
return {
id: item.timestamp,
content: `<div class="flex"><span>${getTimelineItemDescription(
item
)}</span></div>`,
start: new Date(item.timestamp * 1000),
end: new Date(item.timestamp * 1000),
type: "box",
};
});
}