blakeblackshear.frigate/web/src/routes/Cameras.jsx

86 lines
2.7 KiB
React
Raw Normal View History

import { h, Fragment } from 'preact';
2021-02-07 22:46:05 +01:00
import ActivityIndicator from '../components/ActivityIndicator';
import Card from '../components/Card';
import CameraImage from '../components/CameraImage';
import ClipIcon from '../icons/Clip';
import MotionIcon from '../icons/Motion';
import SnapshotIcon from '../icons/Snapshot';
import { useDetectState, useRecordingsState, useSnapshotsState } from '../api/mqtt';
2021-02-02 05:28:25 +01:00
import { useMemo } from 'preact/hooks';
2022-02-26 20:11:00 +01:00
import useSWR from 'swr';
2021-01-09 18:26:46 +01:00
export default function Cameras() {
2022-02-26 20:11:00 +01:00
const { data: config } = useSWR('config');
2021-01-09 18:26:46 +01:00
2022-02-26 20:11:00 +01:00
return !config ? (
<ActivityIndicator />
) : (
2022-02-27 15:04:12 +01:00
<div className="grid grid-cols-1 3xl:grid-cols-3 md:grid-cols-2 gap-4 p-2 px-4">
<SortedCameras unsortedCameras={config.cameras} />
</div>
);
}
function SortedCameras({ unsortedCameras }) {
const sortedCameras = useMemo(() =>
Object.entries(unsortedCameras)
.filter(([_, conf]) => conf.ui.dashboard)
.sort(([_, aConf], [__, bConf]) => aConf.ui.order - bConf.ui.order),
[unsortedCameras]);
return (
<Fragment>
{sortedCameras.map(([camera, conf]) => (
2022-02-26 20:11:00 +01:00
<Camera key={camera} name={camera} conf={conf} />
2021-01-09 18:26:46 +01:00
))}
</Fragment>
2021-01-09 18:26:46 +01:00
);
}
2022-02-26 20:11:00 +01:00
function Camera({ name }) {
const { payload: detectValue, send: sendDetect } = useDetectState(name);
const { payload: recordValue, send: sendRecordings } = useRecordingsState(name);
const { payload: snapshotValue, send: sendSnapshots } = useSnapshotsState(name);
2021-01-09 18:26:46 +01:00
const href = `/cameras/${name}`;
const buttons = useMemo(() => {
2022-02-26 20:11:00 +01:00
return [
{ name: 'Events', href: `/events?camera=${name}` },
{ name: 'Recordings', href: `/recording/${name}` },
];
}, [name]);
const icons = useMemo(
() => [
{
name: `Toggle detect ${detectValue === 'ON' ? 'off' : 'on'}`,
icon: MotionIcon,
color: detectValue === 'ON' ? 'blue' : 'gray',
onClick: () => {
sendDetect(detectValue === 'ON' ? 'OFF' : 'ON');
},
},
{
name: `Toggle recordings ${recordValue === 'ON' ? 'off' : 'on'}`,
icon: ClipIcon,
color: recordValue === 'ON' ? 'blue' : 'gray',
onClick: () => {
sendRecordings(recordValue === 'ON' ? 'OFF' : 'ON');
},
},
{
name: `Toggle snapshots ${snapshotValue === 'ON' ? 'off' : 'on'}`,
icon: SnapshotIcon,
color: snapshotValue === 'ON' ? 'blue' : 'gray',
onClick: () => {
sendSnapshots(snapshotValue === 'ON' ? 'OFF' : 'ON');
},
},
],
[detectValue, sendDetect, recordValue, sendRecordings, snapshotValue, sendSnapshots]
);
2021-01-09 18:26:46 +01:00
return (
<Card buttons={buttons} href={href} header={name} icons={icons} media={<CameraImage camera={name} stretch />} />
);
2021-01-09 18:26:46 +01:00
}