Camera.jsx 2.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. import { h } from 'preact';
  2. import Heading from './components/Heading';
  3. import Link from './components/Link';
  4. import Switch from './components/Switch';
  5. import { route } from 'preact-router';
  6. import { useCallback, useContext } from 'preact/hooks';
  7. import { ApiHost, Config } from './context';
  8. export default function Camera({ camera, url }) {
  9. const config = useContext(Config);
  10. const apiHost = useContext(ApiHost);
  11. if (!(camera in config.cameras)) {
  12. return <div>{`No camera named ${camera}`}</div>;
  13. }
  14. const cameraConfig = config.cameras[camera];
  15. const { pathname, searchParams } = new URL(`${window.location.protocol}//${window.location.host}${url}`);
  16. const searchParamsString = searchParams.toString();
  17. const handleSetOption = useCallback(
  18. (id, value) => {
  19. searchParams.set(id, value ? 1 : 0);
  20. route(`${pathname}?${searchParams.toString()}`, true);
  21. },
  22. [searchParams]
  23. );
  24. function getBoolean(id) {
  25. return Boolean(parseInt(searchParams.get(id), 10));
  26. }
  27. return (
  28. <div>
  29. <Heading size="2xl">{camera}</Heading>
  30. <img
  31. width={cameraConfig.width}
  32. height={cameraConfig.height}
  33. key={searchParamsString}
  34. src={`${apiHost}/api/${camera}?${searchParamsString}`}
  35. />
  36. <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4 p-4">
  37. <Switch checked={getBoolean('bbox')} id="bbox" label="Bounding box" onChange={handleSetOption} />
  38. <Switch checked={getBoolean('timestamp')} id="timestamp" label="Timestamp" onChange={handleSetOption} />
  39. <Switch checked={getBoolean('zones')} id="zones" label="Zones" onChange={handleSetOption} />
  40. <Switch checked={getBoolean('mask')} id="mask" label="Masks" onChange={handleSetOption} />
  41. <Switch checked={getBoolean('motion')} id="motion" label="Motion boxes" onChange={handleSetOption} />
  42. <Switch checked={getBoolean('regions')} id="regions" label="Regions" onChange={handleSetOption} />
  43. </div>
  44. <div>
  45. <Heading size="sm">Tracked objects</Heading>
  46. <ul className="flex flex-row flex-wrap space-x-4">
  47. {cameraConfig.objects.track.map((objectType) => {
  48. return (
  49. <li key={objectType}>
  50. <Link href={`/events?camera=${camera}&label=${objectType}`}>
  51. <span className="capitalize">{objectType}</span>
  52. <img src={`${apiHost}/api/${camera}/${objectType}/best.jpg?crop=1&h=150`} />
  53. </Link>
  54. </li>
  55. );
  56. })}
  57. </ul>
  58. </div>
  59. <div>
  60. <Heading size="sm">Options</Heading>
  61. <ul>
  62. <li>
  63. <Link href={`/cameras/${camera}/editor`}>Mask & Zone creator</Link>
  64. </li>
  65. </ul>
  66. </div>
  67. </div>
  68. );
  69. }