frigate/web/src/pages/Live.tsx

100 lines
2.7 KiB
TypeScript
Raw Normal View History

2024-02-07 16:46:45 +03:00
import { EventThumbnail } from "@/components/image/EventThumbnail";
import LivePlayer from "@/components/player/LivePlayer";
2024-02-07 00:30:41 +03:00
import { ScrollArea, ScrollBar } from "@/components/ui/scroll-area";
import { Event as FrigateEvent } from "@/types/event";
import { FrigateConfig } from "@/types/frigateConfig";
import axios from "axios";
2024-02-07 20:34:57 +03:00
import { useCallback, useMemo } from "react";
import useSWR from "swr";
function Live() {
const { data: config } = useSWR<FrigateConfig>("config");
2024-02-07 00:30:41 +03:00
// recent events
2024-02-07 20:31:19 +03:00
const { data: allEvents, mutate: updateEvents } = useSWR<FrigateEvent[]>(
["events", { limit: 10 }],
{ revalidateOnFocus: false, refreshInterval: 60000 }
);
const events = useMemo(() => {
if (!allEvents) {
return [];
}
2024-02-07 16:24:29 +03:00
const date = new Date();
2024-02-07 20:34:57 +03:00
date.setHours(date.getHours() - 1);
2024-02-07 20:31:19 +03:00
const cutoff = date.getTime() / 1000;
return allEvents.filter((event) => event.start_time > cutoff);
}, [allEvents]);
2024-02-07 19:42:34 +03:00
const onFavorite = useCallback(async (e: Event, event: FrigateEvent) => {
e.stopPropagation();
let response;
if (!event.retain_indefinitely) {
response = await axios.post(`events/${event.id}/retain`);
} else {
response = await axios.delete(`events/${event.id}/retain`);
}
if (response.status === 200) {
updateEvents();
}
}, []);
2024-02-07 00:30:41 +03:00
// camera live views
const cameras = useMemo(() => {
2024-02-07 00:30:41 +03:00
if (!config) {
return [];
}
return Object.values(config.cameras)
.filter((conf) => conf.ui.dashboard && conf.enabled)
.sort((aConf, bConf) => aConf.ui.order - bConf.ui.order);
2024-02-07 00:30:41 +03:00
}, [config]);
return (
2024-02-07 00:30:41 +03:00
<>
{events && events.length > 0 && (
<ScrollArea>
<div className="flex">
{events.map((event) => {
return (
2024-02-07 16:46:45 +03:00
<EventThumbnail
2024-02-07 16:24:29 +03:00
key={event.id}
2024-02-07 16:46:45 +03:00
event={event}
onFavorite={onFavorite}
/>
2024-02-07 00:30:41 +03:00
);
})}
</div>
2024-02-07 00:30:41 +03:00
<ScrollBar orientation="horizontal" />
</ScrollArea>
2024-02-08 22:39:06 +03:00
)}
2024-02-07 00:30:41 +03:00
<div className="mt-4 md:grid md:grid-cols-2 xl:grid-cols-3 3xl:grid-cols-4 gap-4">
{cameras.map((camera) => {
2024-02-07 18:21:40 +03:00
let grow;
if (camera.detect.width / camera.detect.height > 2) {
2024-02-08 21:22:44 +03:00
grow = "aspect-wide md:col-span-2";
2024-02-08 22:12:47 +03:00
} else if (camera.detect.width / camera.detect.height < 1) {
grow = "aspect-tall md:aspect-auto md:row-span-2";
2024-02-07 18:21:40 +03:00
} else {
2024-02-07 19:29:13 +03:00
grow = "aspect-video";
2024-02-07 18:21:40 +03:00
}
2024-02-07 16:24:29 +03:00
return (
<LivePlayer
key={camera.name}
2024-02-07 18:21:40 +03:00
className={`rounded-2xl bg-black ${grow}`}
2024-02-07 16:24:29 +03:00
cameraConfig={camera}
2024-02-07 22:13:36 +03:00
liveChips
2024-02-07 16:24:29 +03:00
/>
);
2024-02-07 00:30:41 +03:00
})}
</div>
</>
);
}
export default Live;