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

145 lines
4.5 KiB
React
Raw Normal View History

2021-06-13 21:21:20 +02:00
import { h, Fragment } from 'preact';
2021-02-07 22:46:05 +01:00
import AutoUpdatingCameraImage from '../components/AutoUpdatingCameraImage';
2022-03-06 05:16:31 +01:00
import ActivityIndicator from '../components/ActivityIndicator';
2021-06-13 21:21:20 +02:00
import JSMpegPlayer from '../components/JSMpegPlayer';
2021-02-07 22:46:05 +01:00
import Button from '../components/Button';
import Card from '../components/Card';
import Heading from '../components/Heading';
import Link from '../components/Link';
import SettingsIcon from '../icons/Settings';
import Switch from '../components/Switch';
2021-07-14 15:32:19 +02:00
import ButtonsTabbed from '../components/ButtonsTabbed';
2021-02-07 22:46:05 +01:00
import { usePersistence } from '../context';
import { useCallback, useMemo, useState } from 'preact/hooks';
2022-02-26 20:11:00 +01:00
import { useApiHost } from '../api';
import useSWR from 'swr';
2021-01-09 18:26:46 +01:00
const emptyObject = Object.freeze({});
2021-02-05 00:19:47 +01:00
export default function Camera({ camera }) {
2022-02-26 20:11:00 +01:00
const { data: config } = useSWR('config');
2021-01-26 16:04:03 +01:00
const apiHost = useApiHost();
2021-02-05 00:19:47 +01:00
const [showSettings, setShowSettings] = useState(false);
2021-06-13 21:21:20 +02:00
const [viewMode, setViewMode] = useState('live');
2021-01-09 18:26:46 +01:00
const cameraConfig = config?.cameras[camera];
2022-03-06 05:16:31 +01:00
const liveWidth = cameraConfig
? Math.round(cameraConfig.live.height * (cameraConfig.detect.width / cameraConfig.detect.height))
: 0;
const [options, setOptions] = usePersistence(`${camera}-feed`, emptyObject);
2021-01-09 18:26:46 +01:00
const handleSetOption = useCallback(
(id, value) => {
2021-02-05 00:19:47 +01:00
const newOptions = { ...options, [id]: value };
setOptions(newOptions);
2021-01-09 18:26:46 +01:00
},
[options, setOptions]
2021-01-09 18:26:46 +01:00
);
2021-02-05 00:19:47 +01:00
const searchParams = useMemo(
() =>
new URLSearchParams(
Object.keys(options).reduce((memo, key) => {
memo.push([key, options[key] === true ? '1' : '0']);
return memo;
}, [])
),
[options]
2021-02-05 00:19:47 +01:00
);
const handleToggleSettings = useCallback(() => {
setShowSettings(!showSettings);
}, [showSettings, setShowSettings]);
2022-03-06 05:16:31 +01:00
if (!cameraConfig) {
return <ActivityIndicator />;
}
2021-02-05 00:19:47 +01:00
const optionContent = showSettings ? (
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">
<Switch
checked={options['bbox']}
id="bbox"
onChange={handleSetOption}
label="Bounding box"
labelPosition="after"
/>
<Switch
checked={options['timestamp']}
id="timestamp"
onChange={handleSetOption}
label="Timestamp"
labelPosition="after"
/>
<Switch checked={options['zones']} id="zones" onChange={handleSetOption} label="Zones" labelPosition="after" />
<Switch checked={options['mask']} id="mask" onChange={handleSetOption} label="Masks" labelPosition="after" />
<Switch
checked={options['motion']}
id="motion"
onChange={handleSetOption}
label="Motion boxes"
labelPosition="after"
/>
<Switch
checked={options['regions']}
id="regions"
onChange={handleSetOption}
label="Regions"
labelPosition="after"
/>
2021-02-05 00:19:47 +01:00
<Link href={`/cameras/${camera}/editor`}>Mask & Zone creator</Link>
</div>
) : null;
2021-01-09 18:26:46 +01:00
2021-06-13 21:21:20 +02:00
let player;
2021-06-13 21:24:34 +02:00
if (viewMode === 'live') {
2021-06-13 21:49:13 +02:00
player = (
<Fragment>
<div>
2021-10-17 15:48:59 +02:00
<JSMpegPlayer camera={camera} width={liveWidth} height={cameraConfig.live.height} />
2021-06-13 21:49:13 +02:00
</div>
</Fragment>
);
2022-03-06 05:16:31 +01:00
} else if (viewMode === 'debug') {
2021-06-13 21:49:13 +02:00
player = (
<Fragment>
<div>
<AutoUpdatingCameraImage camera={camera} searchParams={searchParams} />
</div>
2021-02-05 00:19:47 +01:00
2021-06-13 21:49:13 +02:00
<Button onClick={handleToggleSettings} type="text">
<span className="w-5 h-5">
<SettingsIcon />
</span>{' '}
<span>{showSettings ? 'Hide' : 'Show'} Options</span>
</Button>
{showSettings ? <Card header="Options" elevated={false} content={optionContent} /> : null}
</Fragment>
);
2021-06-13 21:21:20 +02:00
}
return (
2022-02-27 15:04:12 +01:00
<div className="space-y-4 p-2 px-4">
<Heading size="2xl">{camera.replaceAll('_', ' ')}</Heading>
2021-07-16 10:51:21 +02:00
<ButtonsTabbed viewModes={['live', 'debug']} setViewMode={setViewMode} />
2021-06-13 21:21:20 +02:00
{player}
<div className="space-y-4">
2021-01-09 18:26:46 +01:00
<Heading size="sm">Tracked objects</Heading>
2021-02-02 05:28:25 +01:00
<div className="flex flex-wrap justify-start">
{cameraConfig.objects.track.map((objectType) => (
<Card
className="mb-4 mr-4"
key={objectType}
header={objectType}
href={`/events?camera=${camera}&label=${objectType}`}
media={<img src={`${apiHost}/api/${camera}/${objectType}/thumbnail.jpg`} />}
2021-02-02 05:28:25 +01:00
/>
))}
</div>
2021-01-09 18:26:46 +01:00
</div>
</div>
);
}