|
@@ -1,73 +1,99 @@
|
|
import { h } from 'preact';
|
|
import { h } from 'preact';
|
|
import AutoUpdatingCameraImage from './components/AutoUpdatingCameraImage';
|
|
import AutoUpdatingCameraImage from './components/AutoUpdatingCameraImage';
|
|
|
|
+import Button from './components/Button';
|
|
import Card from './components/Card';
|
|
import Card from './components/Card';
|
|
import Heading from './components/Heading';
|
|
import Heading from './components/Heading';
|
|
import Link from './components/Link';
|
|
import Link from './components/Link';
|
|
|
|
+import SettingsIcon from './icons/Settings';
|
|
import Switch from './components/Switch';
|
|
import Switch from './components/Switch';
|
|
import { route } from 'preact-router';
|
|
import { route } from 'preact-router';
|
|
-import { useCallback, useContext } from 'preact/hooks';
|
|
|
|
|
|
+import { usePersistence } from './context';
|
|
|
|
+import { useCallback, useContext, useMemo, useState } from 'preact/hooks';
|
|
import { useApiHost, useConfig } from './api';
|
|
import { useApiHost, useConfig } from './api';
|
|
|
|
|
|
-export default function Camera({ camera, url }) {
|
|
|
|
|
|
+export default function Camera({ camera }) {
|
|
const { data: config } = useConfig();
|
|
const { data: config } = useConfig();
|
|
const apiHost = useApiHost();
|
|
const apiHost = useApiHost();
|
|
|
|
+ const [showSettings, setShowSettings] = useState(false);
|
|
|
|
|
|
if (!config) {
|
|
if (!config) {
|
|
return <div>{`No camera named ${camera}`}</div>;
|
|
return <div>{`No camera named ${camera}`}</div>;
|
|
}
|
|
}
|
|
|
|
|
|
const cameraConfig = config.cameras[camera];
|
|
const cameraConfig = config.cameras[camera];
|
|
- const objectCount = cameraConfig.objects.track.length;
|
|
|
|
|
|
+ const [options, setOptions, optionsLoaded] = usePersistence(`${camera}-feed`, Object.freeze({}));
|
|
|
|
|
|
- const { pathname, searchParams } = new URL(`${window.location.protocol}//${window.location.host}${url}`);
|
|
|
|
- const searchParamsString = searchParams.toString();
|
|
|
|
|
|
+ const objectCount = useMemo(() => cameraConfig.objects.track.length, [cameraConfig]);
|
|
|
|
|
|
const handleSetOption = useCallback(
|
|
const handleSetOption = useCallback(
|
|
(id, value) => {
|
|
(id, value) => {
|
|
- searchParams.set(id, value ? 1 : 0);
|
|
|
|
- route(`${pathname}?${searchParams.toString()}`, true);
|
|
|
|
|
|
+ const newOptions = { ...options, [id]: value };
|
|
|
|
+ setOptions(newOptions);
|
|
},
|
|
},
|
|
- [searchParams]
|
|
|
|
|
|
+ [options]
|
|
);
|
|
);
|
|
|
|
|
|
- function getBoolean(id) {
|
|
|
|
- return Boolean(parseInt(searchParams.get(id), 10));
|
|
|
|
- }
|
|
|
|
|
|
+ const searchParams = useMemo(
|
|
|
|
+ () =>
|
|
|
|
+ new URLSearchParams(
|
|
|
|
+ Object.keys(options).reduce((memo, key) => {
|
|
|
|
+ memo.push([key, options[key] === true ? '1' : '0']);
|
|
|
|
+ return memo;
|
|
|
|
+ }, [])
|
|
|
|
+ ),
|
|
|
|
+ [camera, options]
|
|
|
|
+ );
|
|
|
|
+
|
|
|
|
+ const handleToggleSettings = useCallback(() => {
|
|
|
|
+ setShowSettings(!showSettings);
|
|
|
|
+ }, [showSettings, setShowSettings]);
|
|
|
|
+
|
|
|
|
+ const optionContent = showSettings ? (
|
|
|
|
+ <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">
|
|
|
|
+ <div className="flex space-x-3">
|
|
|
|
+ <Switch checked={options['bbox']} id="bbox" onChange={handleSetOption} />
|
|
|
|
+ <span class="inline-flex">Bounding box</span>
|
|
|
|
+ </div>
|
|
|
|
+ <div className="flex space-x-3">
|
|
|
|
+ <Switch checked={options['timestamp']} id="timestamp" onChange={handleSetOption} />
|
|
|
|
+ <span class="inline-flex">Timestamp</span>
|
|
|
|
+ </div>
|
|
|
|
+ <div className="flex space-x-3">
|
|
|
|
+ <Switch checked={options['zones']} id="zones" onChange={handleSetOption} />
|
|
|
|
+ <span class="inline-flex">Zones</span>
|
|
|
|
+ </div>
|
|
|
|
+ <div className="flex space-x-3">
|
|
|
|
+ <Switch checked={options['mask']} id="mask" onChange={handleSetOption} />
|
|
|
|
+ <span class="inline-flex">Masks</span>
|
|
|
|
+ </div>
|
|
|
|
+ <div className="flex space-x-3">
|
|
|
|
+ <Switch checked={options['motion']} id="motion" onChange={handleSetOption} />
|
|
|
|
+ <span class="inline-flex">Motion boxes</span>
|
|
|
|
+ </div>
|
|
|
|
+ <div className="flex space-x-3">
|
|
|
|
+ <Switch checked={options['regions']} id="regions" onChange={handleSetOption} />
|
|
|
|
+ <span class="inline-flex">Regions</span>
|
|
|
|
+ </div>
|
|
|
|
+ <Link href={`/cameras/${camera}/editor`}>Mask & Zone creator</Link>
|
|
|
|
+ </div>
|
|
|
|
+ ) : null;
|
|
|
|
|
|
return (
|
|
return (
|
|
<div className="space-y-4">
|
|
<div className="space-y-4">
|
|
<Heading size="2xl">{camera}</Heading>
|
|
<Heading size="2xl">{camera}</Heading>
|
|
- <div>
|
|
|
|
- <AutoUpdatingCameraImage camera={camera} searchParams={searchParamsString} />
|
|
|
|
- </div>
|
|
|
|
-
|
|
|
|
- <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4 p-4">
|
|
|
|
- <div className="flex space-x-3">
|
|
|
|
- <Switch checked={getBoolean('bbox')} id="bbox" onChange={handleSetOption} />
|
|
|
|
- <span class="inline-flex">Bounding box</span>
|
|
|
|
- </div>
|
|
|
|
- <div className="flex space-x-3">
|
|
|
|
- <Switch checked={getBoolean('timestamp')} id="timestamp" onChange={handleSetOption} />
|
|
|
|
- <span class="inline-flex">Timestamp</span>
|
|
|
|
- </div>
|
|
|
|
- <div className="flex space-x-3">
|
|
|
|
- <Switch checked={getBoolean('zones')} id="zones" onChange={handleSetOption} />
|
|
|
|
- <span class="inline-flex">Zones</span>
|
|
|
|
|
|
+ {optionsLoaded ? (
|
|
|
|
+ <div>
|
|
|
|
+ <AutoUpdatingCameraImage camera={camera} searchParams={searchParams} />
|
|
</div>
|
|
</div>
|
|
- <div className="flex space-x-3">
|
|
|
|
- <Switch checked={getBoolean('mask')} id="mask" onChange={handleSetOption} />
|
|
|
|
- <span class="inline-flex">Masks</span>
|
|
|
|
- </div>
|
|
|
|
- <div className="flex space-x-3">
|
|
|
|
- <Switch checked={getBoolean('motion')} id="motion" onChange={handleSetOption} />
|
|
|
|
- <span class="inline-flex">Motion boxes</span>
|
|
|
|
- </div>
|
|
|
|
- <div className="flex space-x-3">
|
|
|
|
- <Switch checked={getBoolean('regions')} id="regions" onChange={handleSetOption} />
|
|
|
|
- <span class="inline-flex">Regions</span>
|
|
|
|
- </div>
|
|
|
|
- <Link href={`/cameras/${camera}/editor`}>Mask & Zone creator</Link>
|
|
|
|
- </div>
|
|
|
|
|
|
+ ) : null}
|
|
|
|
+
|
|
|
|
+ <Button onClick={handleToggleSettings} type="text">
|
|
|
|
+ <span class="w-5 h-5">
|
|
|
|
+ <SettingsIcon />
|
|
|
|
+ </span>{' '}
|
|
|
|
+ <span>{showSettings ? 'Hide' : 'Show'} Options</span>
|
|
|
|
+ </Button>
|
|
|
|
+ {showSettings ? <Card header="Options" elevated={false} content={optionContent} /> : null}
|
|
|
|
|
|
<div className="space-y-4">
|
|
<div className="space-y-4">
|
|
<Heading size="sm">Tracked objects</Heading>
|
|
<Heading size="sm">Tracked objects</Heading>
|