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';
|
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';
|
|
|
|
import { usePersistence } from '../context';
|
2021-02-09 20:35:33 +01:00
|
|
|
import { useCallback, useMemo, useState } from 'preact/hooks';
|
2021-02-07 22:46:05 +01:00
|
|
|
import { useApiHost, useConfig } from '../api';
|
2021-01-09 18:26:46 +01:00
|
|
|
|
2021-02-09 20:35:33 +01:00
|
|
|
const emptyObject = Object.freeze({});
|
|
|
|
|
2021-02-05 00:19:47 +01:00
|
|
|
export default function Camera({ camera }) {
|
2021-01-26 16:04:03 +01:00
|
|
|
const { data: config } = useConfig();
|
|
|
|
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
|
|
|
|
2021-02-09 20:35:33 +01:00
|
|
|
const cameraConfig = config?.cameras[camera];
|
|
|
|
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
|
|
|
},
|
2021-02-09 20:35:33 +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;
|
|
|
|
}, [])
|
|
|
|
),
|
2021-02-09 20:35:33 +01:00
|
|
|
[options]
|
2021-02-05 00:19:47 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
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">
|
2021-02-13 01:06:51 +01:00
|
|
|
<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;
|
|
|
|
if (viewMode == 'live') {
|
|
|
|
player = <>
|
|
|
|
<div>
|
|
|
|
<JSMpegPlayer camera={camera} />
|
|
|
|
</div>
|
|
|
|
</>;
|
|
|
|
}
|
|
|
|
else if (viewMode == 'debug') {
|
|
|
|
player = <>
|
2021-02-09 20:35:33 +01:00
|
|
|
<div>
|
|
|
|
<AutoUpdatingCameraImage camera={camera} searchParams={searchParams} />
|
|
|
|
</div>
|
2021-02-05 00:19:47 +01:00
|
|
|
|
|
|
|
<Button onClick={handleToggleSettings} type="text">
|
2021-02-09 20:35:33 +01:00
|
|
|
<span className="w-5 h-5">
|
2021-02-05 00:19:47 +01:00
|
|
|
<SettingsIcon />
|
|
|
|
</span>{' '}
|
|
|
|
<span>{showSettings ? 'Hide' : 'Show'} Options</span>
|
|
|
|
</Button>
|
|
|
|
{showSettings ? <Card header="Options" elevated={false} content={optionContent} /> : null}
|
2021-06-13 21:21:20 +02:00
|
|
|
</>;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="space-y-4">
|
|
|
|
<Heading size="2xl">{camera}</Heading>
|
|
|
|
<div>
|
|
|
|
<nav className="flex justify-end">
|
|
|
|
<button onClick={() => setViewMode('live')} className={viewMode == 'live' ? `text-gray-600 py-0 px-4 block hover:text-gray-500 focus:outline-none border-b-2 font-medium border-gray-500` : `text-gray-600 py-0 px-4 block hover:text-gray-500`}>
|
|
|
|
Live
|
|
|
|
</button>
|
|
|
|
<button onClick={() => setViewMode('debug')} className={viewMode == 'debug' ? `text-gray-600 py-0 px-4 block hover:text-gray-500 focus:outline-none border-b-2 font-medium border-gray-500` : `text-gray-600 py-0 px-4 block hover:text-gray-500`}>
|
|
|
|
Debug
|
|
|
|
</button>
|
|
|
|
</nav>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
{player}
|
2021-01-19 17:44:18 +01:00
|
|
|
|
|
|
|
<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}/best.jpg?crop=1&h=150`} />}
|
|
|
|
/>
|
|
|
|
))}
|
2021-01-19 17:44:18 +01:00
|
|
|
</div>
|
2021-01-09 18:26:46 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|