|
1 | | -import { useCallback, useState } from 'react'; |
| 1 | +import { create } from 'zustand'; |
| 2 | +import { persist } from 'zustand/middleware'; |
2 | 3 |
|
3 | | -const STORAGE_KEY = 'bull-board-menu-state'; |
| 4 | +type MenuState = { |
| 5 | + state: Record<string, boolean>; |
| 6 | + isMenuOpen: (menuPath: string, defaultOpen?: boolean) => boolean; |
| 7 | + toggleMenu: (menuPath: string) => void; |
| 8 | +}; |
4 | 9 |
|
5 | | -type MenuState = Record<string, boolean>; |
6 | | - |
7 | | -export function useMenuState() { |
8 | | - const [menuState, setMenuState] = useState<MenuState>(() => { |
9 | | - try { |
10 | | - const stored = localStorage.getItem(STORAGE_KEY); |
11 | | - return stored ? JSON.parse(stored) : {}; |
12 | | - } catch { |
13 | | - return {}; |
| 10 | +export const useMenuState = create<MenuState>()( |
| 11 | + persist( |
| 12 | + (set, get) => ({ |
| 13 | + state: {}, |
| 14 | + toggleMenu: (menuPath: string) => |
| 15 | + set(({ state: prev }) => ({ state: { ...prev, [menuPath]: !prev[menuPath] } })), |
| 16 | + isMenuOpen: (menuPath: string, defaultOpen = true) => get().state[menuPath] ?? defaultOpen, |
| 17 | + }), |
| 18 | + { |
| 19 | + name: 'bull-board:menu-state', |
14 | 20 | } |
15 | | - }); |
16 | | - |
17 | | - const toggleMenu = useCallback((menuPath: string) => { |
18 | | - setMenuState((prev) => { |
19 | | - const newState = { |
20 | | - ...prev, |
21 | | - [menuPath]: !prev[menuPath] |
22 | | - }; |
23 | | - |
24 | | - try { |
25 | | - localStorage.setItem(STORAGE_KEY, JSON.stringify(newState)); |
26 | | - } catch { |
27 | | - // Ignore localStorage errors |
28 | | - } |
29 | | - |
30 | | - return newState; |
31 | | - }); |
32 | | - }, []); |
33 | | - |
34 | | - const isMenuOpen = useCallback((menuPath: string, defaultOpen = true) => { |
35 | | - return menuState[menuPath] !== undefined ? menuState[menuPath] : defaultOpen; |
36 | | - }, [menuState]); |
37 | | - |
38 | | - return { toggleMenu, isMenuOpen }; |
39 | | -} |
| 21 | + ) |
| 22 | +); |
0 commit comments