mirror of
				https://github.com/blakeblackshear/frigate.git
				synced 2025-10-27 10:52:11 +01:00 
			
		
		
		
	feat(web): persist darkmode preference
This commit is contained in:
		
							parent
							
								
									5ed7a17f46
								
							
						
					
					
						commit
						276ce8710c
					
				
							
								
								
									
										37
									
								
								web/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										37
									
								
								web/package-lock.json
									
									
									
										generated
									
									
									
								
							@ -421,6 +421,17 @@
 | 
				
			|||||||
        "typed-colors": "^1.0.0",
 | 
					        "typed-colors": "^1.0.0",
 | 
				
			||||||
        "typed-figures": "^1.0.0",
 | 
					        "typed-figures": "^1.0.0",
 | 
				
			||||||
        "yargs": "^16.2.0"
 | 
					        "yargs": "^16.2.0"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "enhanced-resolve": {
 | 
				
			||||||
 | 
					          "version": "5.7.0",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.7.0.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-6njwt/NsZFUKhM6j9U8hzVyD4E4r0x7NQzhTCbcWOJ0IQjNSAoalWmb0AE51Wn+fwan5qVESWi7t2ToBxs9vrw==",
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "graceful-fs": "^4.2.4",
 | 
				
			||||||
 | 
					            "tapable": "^2.2.0"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "@typed/fp": {
 | 
					    "@typed/fp": {
 | 
				
			||||||
@ -444,6 +455,13 @@
 | 
				
			|||||||
        "newtype-ts": "^0.3.4",
 | 
					        "newtype-ts": "^0.3.4",
 | 
				
			||||||
        "ts-toolbelt": "^8.0.7",
 | 
					        "ts-toolbelt": "^8.0.7",
 | 
				
			||||||
        "tslib": "^2.0.3"
 | 
					        "tslib": "^2.0.3"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "json-schema": {
 | 
				
			||||||
 | 
					          "version": "0.2.5",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.5.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-gWJOWYFrhQ8j7pVm0EM8Slr+EPVq1Phf6lvzvD/WCeqkrx/f2xBI0xOsRRS9xCn3I4vKtP519dvs3TP09r24wQ=="
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "@types/json-schema": {
 | 
					    "@types/json-schema": {
 | 
				
			||||||
@ -836,15 +854,6 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="
 | 
					      "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "enhanced-resolve": {
 | 
					 | 
				
			||||||
      "version": "5.7.0",
 | 
					 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.7.0.tgz",
 | 
					 | 
				
			||||||
      "integrity": "sha512-6njwt/NsZFUKhM6j9U8hzVyD4E4r0x7NQzhTCbcWOJ0IQjNSAoalWmb0AE51Wn+fwan5qVESWi7t2ToBxs9vrw==",
 | 
					 | 
				
			||||||
      "requires": {
 | 
					 | 
				
			||||||
        "graceful-fs": "^4.2.4",
 | 
					 | 
				
			||||||
        "tapable": "^2.2.0"
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    "error-ex": {
 | 
					    "error-ex": {
 | 
				
			||||||
      "version": "1.3.2",
 | 
					      "version": "1.3.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz",
 | 
				
			||||||
@ -1047,6 +1056,11 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/hyperhtml-style/-/hyperhtml-style-0.1.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/hyperhtml-style/-/hyperhtml-style-0.1.2.tgz",
 | 
				
			||||||
      "integrity": "sha512-ZDRYNClEaqUS0a8RAED0nQRqWmZk7ctdyij3Iw/PqUUef6xhYO87nx9vJNuxg7Yc6J2FdJjXRKbB0iud2ZyzwQ=="
 | 
					      "integrity": "sha512-ZDRYNClEaqUS0a8RAED0nQRqWmZk7ctdyij3Iw/PqUUef6xhYO87nx9vJNuxg7Yc6J2FdJjXRKbB0iud2ZyzwQ=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "idb-keyval": {
 | 
				
			||||||
 | 
					      "version": "5.0.2",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/idb-keyval/-/idb-keyval-5.0.2.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-1DYjY/nX2U9pkTkwFoAmKcK1ZWmkNgO32Oon9tp/9+HURizxUQ4fZRxMJZs093SldP7q6dotVj03kIkiqOILyA=="
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "ignore": {
 | 
					    "ignore": {
 | 
				
			||||||
      "version": "5.1.8",
 | 
					      "version": "5.1.8",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz",
 | 
				
			||||||
@ -1201,11 +1215,6 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz",
 | 
				
			||||||
      "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w=="
 | 
					      "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "json-schema": {
 | 
					 | 
				
			||||||
      "version": "0.2.5",
 | 
					 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.5.tgz",
 | 
					 | 
				
			||||||
      "integrity": "sha512-gWJOWYFrhQ8j7pVm0EM8Slr+EPVq1Phf6lvzvD/WCeqkrx/f2xBI0xOsRRS9xCn3I4vKtP519dvs3TP09r24wQ=="
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    "json5": {
 | 
					    "json5": {
 | 
				
			||||||
      "version": "2.1.3",
 | 
					      "version": "2.1.3",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz",
 | 
				
			||||||
 | 
				
			|||||||
@ -11,6 +11,7 @@
 | 
				
			|||||||
    "@snowpack/plugin-postcss": "^1.1.0",
 | 
					    "@snowpack/plugin-postcss": "^1.1.0",
 | 
				
			||||||
    "autoprefixer": "^10.2.1",
 | 
					    "autoprefixer": "^10.2.1",
 | 
				
			||||||
    "cross-env": "^7.0.3",
 | 
					    "cross-env": "^7.0.3",
 | 
				
			||||||
 | 
					    "idb-keyval": "^5.0.2",
 | 
				
			||||||
    "immer": "^8.0.1",
 | 
					    "immer": "^8.0.1",
 | 
				
			||||||
    "postcss": "^8.2.2",
 | 
					    "postcss": "^8.2.2",
 | 
				
			||||||
    "postcss-cli": "^8.3.1",
 | 
					    "postcss-cli": "^8.3.1",
 | 
				
			||||||
 | 
				
			|||||||
@ -4,6 +4,7 @@ import AppBar from './components/AppBar';
 | 
				
			|||||||
import Camera from './Camera';
 | 
					import Camera from './Camera';
 | 
				
			||||||
import CameraMap from './CameraMap';
 | 
					import CameraMap from './CameraMap';
 | 
				
			||||||
import Cameras from './Cameras';
 | 
					import Cameras from './Cameras';
 | 
				
			||||||
 | 
					import { DarkModeProvider } from './context';
 | 
				
			||||||
import Debug from './Debug';
 | 
					import Debug from './Debug';
 | 
				
			||||||
import Event from './Event';
 | 
					import Event from './Event';
 | 
				
			||||||
import Events from './Events';
 | 
					import Events from './Events';
 | 
				
			||||||
@ -15,6 +16,7 @@ import Api, { FetchStatus, useConfig } from './api';
 | 
				
			|||||||
export default function App() {
 | 
					export default function App() {
 | 
				
			||||||
  const { data, status } = useConfig();
 | 
					  const { data, status } = useConfig();
 | 
				
			||||||
  return (
 | 
					  return (
 | 
				
			||||||
 | 
					    <DarkModeProvider>
 | 
				
			||||||
      <div class="w-full">
 | 
					      <div class="w-full">
 | 
				
			||||||
        <AppBar title="Frigate" />
 | 
					        <AppBar title="Frigate" />
 | 
				
			||||||
        {status !== FetchStatus.LOADED ? (
 | 
					        {status !== FetchStatus.LOADED ? (
 | 
				
			||||||
@ -22,7 +24,7 @@ export default function App() {
 | 
				
			|||||||
            <ActivityIndicator />
 | 
					            <ActivityIndicator />
 | 
				
			||||||
          </div>
 | 
					          </div>
 | 
				
			||||||
        ) : (
 | 
					        ) : (
 | 
				
			||||||
        <div className="md:flex flex-col md:flex-row md:min-h-screen w-full bg-white dark:bg-gray-900 text-gray-900 dark:text-white">
 | 
					          <div className="flex flex-row min-h-screen w-full bg-white dark:bg-gray-900 text-gray-900 dark:text-white">
 | 
				
			||||||
            <Sidebar />
 | 
					            <Sidebar />
 | 
				
			||||||
            <div className="w-full flex-auto p-2 mt-20 md:p-4 lg:pl-8 lg:pr-8 min-w-0">
 | 
					            <div className="w-full flex-auto p-2 mt-20 md:p-4 lg:pl-8 lg:pr-8 min-w-0">
 | 
				
			||||||
              <Router>
 | 
					              <Router>
 | 
				
			||||||
@ -38,5 +40,6 @@ export default function App() {
 | 
				
			|||||||
          </div>
 | 
					          </div>
 | 
				
			||||||
        )}
 | 
					        )}
 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
 | 
					    </DarkModeProvider>
 | 
				
			||||||
  );
 | 
					  );
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										34
									
								
								web/src/Settings.jsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								web/src/Settings.jsx
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,34 @@
 | 
				
			|||||||
 | 
					import { h } from 'preact';
 | 
				
			||||||
 | 
					import { useDarkMode } from './context';
 | 
				
			||||||
 | 
					import { useCallback } from 'preact/hooks';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default function Settings() {
 | 
				
			||||||
 | 
					  const { currentMode, persistedMode, setDarkMode } = useDarkMode();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const handleSelect = useCallback(
 | 
				
			||||||
 | 
					    (event) => {
 | 
				
			||||||
 | 
					      const mode = event.target.value;
 | 
				
			||||||
 | 
					      setDarkMode(mode);
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    [setDarkMode]
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return (
 | 
				
			||||||
 | 
					    <div>
 | 
				
			||||||
 | 
					      <label>
 | 
				
			||||||
 | 
					        <span className="block uppercase text-sm">Dark mode</span>
 | 
				
			||||||
 | 
					        <select className="border-solid border border-gray-500 rounded dark:text-gray-900" onChange={handleSelect}>
 | 
				
			||||||
 | 
					          <option selected={persistedMode === 'media'} value="media">
 | 
				
			||||||
 | 
					            Auto
 | 
				
			||||||
 | 
					          </option>
 | 
				
			||||||
 | 
					          <option selected={persistedMode === 'light'} value="light">
 | 
				
			||||||
 | 
					            Light
 | 
				
			||||||
 | 
					          </option>
 | 
				
			||||||
 | 
					          <option selected={persistedMode === 'dark'} value="dark">
 | 
				
			||||||
 | 
					            Dark
 | 
				
			||||||
 | 
					          </option>
 | 
				
			||||||
 | 
					        </select>
 | 
				
			||||||
 | 
					      </label>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -4,30 +4,6 @@ import LinkedLogo from './components/LinkedLogo';
 | 
				
			|||||||
import { Link as RouterLink } from 'preact-router/match';
 | 
					import { Link as RouterLink } from 'preact-router/match';
 | 
				
			||||||
import { useCallback, useState } from 'preact/hooks';
 | 
					import { useCallback, useState } from 'preact/hooks';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function HamburgerIcon() {
 | 
					 | 
				
			||||||
  return (
 | 
					 | 
				
			||||||
    <svg fill="currentColor" viewBox="0 0 20 20" className="w-6 h-6">
 | 
					 | 
				
			||||||
      <path
 | 
					 | 
				
			||||||
        fill-rule="evenodd"
 | 
					 | 
				
			||||||
        d="M3 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM3 10a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM9 15a1 1 0 011-1h6a1 1 0 110 2h-6a1 1 0 01-1-1z"
 | 
					 | 
				
			||||||
        clip-rule="evenodd"
 | 
					 | 
				
			||||||
      ></path>
 | 
					 | 
				
			||||||
    </svg>
 | 
					 | 
				
			||||||
  );
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function CloseIcon() {
 | 
					 | 
				
			||||||
  return (
 | 
					 | 
				
			||||||
    <svg fill="currentColor" viewBox="0 0 20 20" className="w-6 h-6">
 | 
					 | 
				
			||||||
      <path
 | 
					 | 
				
			||||||
        fill-rule="evenodd"
 | 
					 | 
				
			||||||
        d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z"
 | 
					 | 
				
			||||||
        clip-rule="evenodd"
 | 
					 | 
				
			||||||
      ></path>
 | 
					 | 
				
			||||||
    </svg>
 | 
					 | 
				
			||||||
  );
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function NavLink({ className = '', href, text }) {
 | 
					function NavLink({ className = '', href, text }) {
 | 
				
			||||||
  const external = href.startsWith('http');
 | 
					  const external = href.startsWith('http');
 | 
				
			||||||
  const El = external ? Link : RouterLink;
 | 
					  const El = external ? Link : RouterLink;
 | 
				
			||||||
@ -45,30 +21,14 @@ function NavLink({ className = '', href, text }) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default function Sidebar() {
 | 
					export default function Sidebar() {
 | 
				
			||||||
  const [open, setOpen] = useState(false);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  const handleToggle = useCallback(() => {
 | 
					 | 
				
			||||||
    setOpen(!open);
 | 
					 | 
				
			||||||
  }, [open, setOpen]);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  return (
 | 
					  return (
 | 
				
			||||||
    <div className="sticky top-0 max-h-screen flex flex-col w-full md:w-64 text-gray-700 bg-white dark:text-gray-200 dark:bg-gray-700 flex-shrink-0 border-r border-gray-200 shadow lg:shadow-none z-20 lg:z-0">
 | 
					    <div className="sticky top-0 max-h-screen flex flex-col w-full md:w-64 text-gray-700 bg-white dark:text-gray-200 dark:bg-gray-900 flex-shrink-0 border-r border-gray-200 dark:border-gray-700 shadow lg:shadow-none z-20 lg:z-0">
 | 
				
			||||||
      <div className="flex-shrink-0 p-4 flex flex-row items-center justify-between">
 | 
					      <div className="flex-shrink-0 p-4 flex flex-row items-center justify-between">
 | 
				
			||||||
        <div class="text-gray-500">
 | 
					        <div class="text-gray-500">
 | 
				
			||||||
          <LinkedLogo />
 | 
					          <LinkedLogo />
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
        <button
 | 
					 | 
				
			||||||
          className="rounded-lg md:hidden rounded-lg focus:outline-none focus:shadow-outline"
 | 
					 | 
				
			||||||
          onClick={handleToggle}
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          {open ? <CloseIcon /> : <HamburgerIcon />}
 | 
					 | 
				
			||||||
        </button>
 | 
					 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
      <nav
 | 
					      <nav className="flex-col flex-grow md:block overflow-hidden px-4 pb-4 md:pb-0 md:overflow-y-auto">
 | 
				
			||||||
        className={`flex-col flex-grow md:block overflow-hidden px-4 pb-4 md:pb-0 md:overflow-y-auto ${
 | 
					 | 
				
			||||||
          !open ? 'md:h-0 hidden' : ''
 | 
					 | 
				
			||||||
        }`}
 | 
					 | 
				
			||||||
      >
 | 
					 | 
				
			||||||
        <NavLink href="/" text="Cameras" />
 | 
					        <NavLink href="/" text="Cameras" />
 | 
				
			||||||
        <NavLink href="/events" text="Events" />
 | 
					        <NavLink href="/events" text="Events" />
 | 
				
			||||||
        <NavLink href="/debug" text="Debug" />
 | 
					        <NavLink href="/debug" text="Debug" />
 | 
				
			||||||
 | 
				
			|||||||
@ -1,8 +1,11 @@
 | 
				
			|||||||
import { h } from 'preact';
 | 
					import { h } from 'preact';
 | 
				
			||||||
import Button from './Button';
 | 
					import Button from './Button';
 | 
				
			||||||
import LinkedLogo from './LinkedLogo';
 | 
					import LinkedLogo from './LinkedLogo';
 | 
				
			||||||
 | 
					import Menu, { MenuItem } from './Menu';
 | 
				
			||||||
import MenuIcon from '../icons/Menu';
 | 
					import MenuIcon from '../icons/Menu';
 | 
				
			||||||
import { useLayoutEffect, useCallback, useState } from 'preact/hooks';
 | 
					import MoreIcon from '../icons/More';
 | 
				
			||||||
 | 
					import { useDarkMode } from '../context';
 | 
				
			||||||
 | 
					import { useLayoutEffect, useCallback, useRef, useState } from 'preact/hooks';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// We would typically preserve these in component state
 | 
					// We would typically preserve these in component state
 | 
				
			||||||
// But need to avoid too many re-renders
 | 
					// But need to avoid too many re-renders
 | 
				
			||||||
@ -13,6 +16,18 @@ export default function AppBar({ title }) {
 | 
				
			|||||||
  const [show, setShow] = useState(true);
 | 
					  const [show, setShow] = useState(true);
 | 
				
			||||||
  const [atZero, setAtZero] = useState(window.scrollY === 0);
 | 
					  const [atZero, setAtZero] = useState(window.scrollY === 0);
 | 
				
			||||||
  const [sidebarVisible, setSidebarVisible] = useState(true);
 | 
					  const [sidebarVisible, setSidebarVisible] = useState(true);
 | 
				
			||||||
 | 
					  const [showMoreMenu, setShowMoreMenu] = useState(false);
 | 
				
			||||||
 | 
					  const { currentMode, persistedMode, setDarkMode } = useDarkMode();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const handleSelectDarkMode = useCallback(
 | 
				
			||||||
 | 
					    (value, label) => {
 | 
				
			||||||
 | 
					      setDarkMode(value);
 | 
				
			||||||
 | 
					      setShowMoreMenu(false);
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    [setDarkMode, setShowMoreMenu]
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const moreRef = useRef(null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const scrollListener = useCallback(
 | 
					  const scrollListener = useCallback(
 | 
				
			||||||
    (event) => {
 | 
					    (event) => {
 | 
				
			||||||
@ -38,9 +53,17 @@ export default function AppBar({ title }) {
 | 
				
			|||||||
    };
 | 
					    };
 | 
				
			||||||
  }, []);
 | 
					  }, []);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const handleShowMenu = useCallback(() => {
 | 
				
			||||||
 | 
					    setShowMoreMenu(true);
 | 
				
			||||||
 | 
					  }, [setShowMoreMenu]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const handleDismissMoreMenu = useCallback(() => {
 | 
				
			||||||
 | 
					    setShowMoreMenu(false);
 | 
				
			||||||
 | 
					  }, [setShowMoreMenu]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return (
 | 
					  return (
 | 
				
			||||||
    <div
 | 
					    <div
 | 
				
			||||||
      className={`w-full border-b border-color-gray-100 flex items-center align-middle p-4 space-x-2 fixed left-0 right-0 z-10 bg-white dark:bg-gray-800 transform transition-all duration-200 translate-y-0 ${
 | 
					      className={`w-full border-b border-gray-100 dark:border-gray-700 flex items-center align-middle p-4 space-x-2 fixed left-0 right-0 z-10 bg-white dark:bg-gray-900 transform transition-all duration-200 translate-y-0 ${
 | 
				
			||||||
        !show ? '-translate-y-full' : ''
 | 
					        !show ? '-translate-y-full' : ''
 | 
				
			||||||
      } ${!atZero ? 'shadow' : ''}`}
 | 
					      } ${!atZero ? 'shadow' : ''}`}
 | 
				
			||||||
    >
 | 
					    >
 | 
				
			||||||
@ -50,6 +73,20 @@ export default function AppBar({ title }) {
 | 
				
			|||||||
        </Button>
 | 
					        </Button>
 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
      <LinkedLogo />
 | 
					      <LinkedLogo />
 | 
				
			||||||
 | 
					      <div className="flex-grow-1 flex justify-end w-full">
 | 
				
			||||||
 | 
					        <div ref={moreRef}>
 | 
				
			||||||
 | 
					          <Button className="rounded-full w-12 h-12" onClick={handleShowMenu} type="text">
 | 
				
			||||||
 | 
					            <MoreIcon />
 | 
				
			||||||
 | 
					          </Button>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					      </div>
 | 
				
			||||||
 | 
					      {showMoreMenu ? (
 | 
				
			||||||
 | 
					        <Menu onDismiss={handleDismissMoreMenu} relativeTo={moreRef}>
 | 
				
			||||||
 | 
					          <MenuItem label="Auto" value="media" onSelect={handleSelectDarkMode} />
 | 
				
			||||||
 | 
					          <MenuItem label="Light" value="light" onSelect={handleSelectDarkMode} />
 | 
				
			||||||
 | 
					          <MenuItem label="Dark" value="dark" onSelect={handleSelectDarkMode} />
 | 
				
			||||||
 | 
					        </Menu>
 | 
				
			||||||
 | 
					      ) : null}
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
  );
 | 
					  );
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -76,13 +76,15 @@ export default function RelativeModal({ className, role = 'dialog', children, on
 | 
				
			|||||||
    <Fragment>
 | 
					    <Fragment>
 | 
				
			||||||
      <div className="absolute inset-0" onClick={handleDismiss} />
 | 
					      <div className="absolute inset-0" onClick={handleDismiss} />
 | 
				
			||||||
      <div
 | 
					      <div
 | 
				
			||||||
        className={`bg-white dark:bg-gray-700 dark:text-white absolute shadow-lg rounded w-auto max-h-48 transition-all duration-75 transform scale-90 opacity-0 ${
 | 
					        className={`z-10 bg-white dark:bg-gray-700 dark:text-white absolute shadow-lg rounded w-auto max-h-48 transition-all duration-75 transform scale-90 opacity-0 ${
 | 
				
			||||||
          show ? 'scale-100 opacity-100' : ''
 | 
					          show ? 'scale-100 opacity-100' : ''
 | 
				
			||||||
        } ${className}`}
 | 
					        } ${className}`}
 | 
				
			||||||
        onkeydown={handleKeydown}
 | 
					        onkeydown={handleKeydown}
 | 
				
			||||||
        role={role}
 | 
					        role={role}
 | 
				
			||||||
        ref={ref}
 | 
					        ref={ref}
 | 
				
			||||||
        style={position.width > 0 ? `width: ${position.width}px; top: ${position.top}px; left: ${position.left}px` : ''}
 | 
					        style={
 | 
				
			||||||
 | 
					          position.width > 0 ? `min-width: ${position.width}px; top: ${position.top}px; left: ${position.left}px` : ''
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        {children}
 | 
					        {children}
 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										66
									
								
								web/src/context/index.jsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								web/src/context/index.jsx
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,66 @@
 | 
				
			|||||||
 | 
					import { h, createContext } from 'preact';
 | 
				
			||||||
 | 
					import { get as getData, set as setData } from 'idb-keyval';
 | 
				
			||||||
 | 
					import produce from 'immer';
 | 
				
			||||||
 | 
					import { useCallback, useContext, useEffect, useState } from 'preact/hooks';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const DarkMode = createContext(null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export function DarkModeProvider({ children }) {
 | 
				
			||||||
 | 
					  const [persistedMode, setPersistedMode] = useState(null);
 | 
				
			||||||
 | 
					  const [currentMode, setCurrentMode] = useState(persistedMode !== 'media' ? persistedMode : null);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const setDarkMode = useCallback(
 | 
				
			||||||
 | 
					    (value) => {
 | 
				
			||||||
 | 
					      setPersistedMode(value);
 | 
				
			||||||
 | 
					      setData('darkmode', value);
 | 
				
			||||||
 | 
					      if (value !== 'media') {
 | 
				
			||||||
 | 
					        setCurrentMode(value);
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    [setPersistedMode]
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  useEffect(() => {
 | 
				
			||||||
 | 
					    async function load() {
 | 
				
			||||||
 | 
					      const darkmode = await getData('darkmode');
 | 
				
			||||||
 | 
					      setDarkMode(darkmode || 'media');
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    load();
 | 
				
			||||||
 | 
					  }, []);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if (persistedMode === null) {
 | 
				
			||||||
 | 
					    return null;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const handleMediaMatch = useCallback(
 | 
				
			||||||
 | 
					    ({ matches }) => {
 | 
				
			||||||
 | 
					      if (matches) {
 | 
				
			||||||
 | 
					        setCurrentMode('dark');
 | 
				
			||||||
 | 
					      } else {
 | 
				
			||||||
 | 
					        setCurrentMode('light');
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    [setCurrentMode]
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  useEffect(() => {
 | 
				
			||||||
 | 
					    if (persistedMode !== 'media') {
 | 
				
			||||||
 | 
					      return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const query = window.matchMedia('(prefers-color-scheme: dark)');
 | 
				
			||||||
 | 
					    query.addEventListener('change', handleMediaMatch);
 | 
				
			||||||
 | 
					    handleMediaMatch(query);
 | 
				
			||||||
 | 
					  }, [persistedMode]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return (
 | 
				
			||||||
 | 
					    <DarkMode.Provider value={{ currentMode, persistedMode, setDarkMode }}>
 | 
				
			||||||
 | 
					      <div className={`${currentMode === 'dark' ? 'dark' : ''}`}>{children}</div>
 | 
				
			||||||
 | 
					    </DarkMode.Provider>
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export function useDarkMode() {
 | 
				
			||||||
 | 
					  return useContext(DarkMode);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -2,7 +2,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
module.exports = {
 | 
					module.exports = {
 | 
				
			||||||
  purge: ['./public/**/*.html', './src/**/*.jsx'],
 | 
					  purge: ['./public/**/*.html', './src/**/*.jsx'],
 | 
				
			||||||
  darkMode: 'media',
 | 
					  darkMode: 'class',
 | 
				
			||||||
  theme: {
 | 
					  theme: {
 | 
				
			||||||
    extend: {},
 | 
					    extend: {},
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
		Reference in New Issue
	
	Block a user