frigate/web/src/pages/Live.tsx

96 lines
2.6 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 01:36:32 +03:00
import { useCallback, useEffect, useMemo, useState } 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 16:24:29 +03:00
const [recentCutoff, setRecentCutoff] = useState<number>(0);
2024-02-07 01:36:32 +03:00
useEffect(() => {
2024-02-07 16:24:29 +03:00
const date = new Date();
2024-02-07 16:46:45 +03:00
date.setHours(date.getHours() - 4);
2024-02-07 16:24:29 +03:00
setRecentCutoff(date.getTime() / 1000);
2024-02-07 01:36:32 +03:00
const intervalId: NodeJS.Timeout = setInterval(() => {
const date = new Date();
date.setHours(date.getHours() - 4);
2024-02-07 16:24:29 +03:00
setRecentCutoff(date.getTime() / 1000);
}, 30000);
2024-02-07 01:36:32 +03:00
return () => clearInterval(intervalId);
2024-02-07 16:24:29 +03:00
}, [30000]);
2024-02-07 00:30:41 +03:00
const { data: events, mutate: updateEvents } = useSWR<FrigateEvent[]>([
"events",
2024-02-07 01:36:32 +03:00
{ limit: 10, after: recentCutoff },
2024-02-07 00:30:41 +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();
}
},
[event]
);
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-07 00:30:41 +03:00
<div className="mt-4 grid grid-cols-1 md:grid-cols-2 2xl:grid-cols-3 gap-4">
{cameras.map((camera) => {
2024-02-07 16:24:29 +03:00
return (
<LivePlayer
key={camera.name}
2024-02-07 16:46:45 +03:00
className="rounded-2xl bg-black h-[428px]"
2024-02-07 16:24:29 +03:00
cameraConfig={camera}
/>
);
2024-02-07 00:30:41 +03:00
})}
</div>
</>
);
}
export default Live;