react-os-shell 2.8.1 → 2.9.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/dist/{Browser-RNO4KYSE.js → Browser-HDKOMCUA.js} +4 -4
  2. package/dist/{Browser-RNO4KYSE.js.map → Browser-HDKOMCUA.js.map} +1 -1
  3. package/dist/{Documents-L7CTOPIN.js → Documents-JXLU2ILH.js} +4 -4
  4. package/dist/{Documents-L7CTOPIN.js.map → Documents-JXLU2ILH.js.map} +1 -1
  5. package/dist/Files-Q2YCSHYG.js +13 -0
  6. package/dist/{Files-FJL2BZDI.js.map → Files-Q2YCSHYG.js.map} +1 -1
  7. package/dist/ImageAnnotator-55NPWJ2R.js +4 -0
  8. package/dist/{ImageAnnotator-CTTMAY5Z.js.map → ImageAnnotator-55NPWJ2R.js.map} +1 -1
  9. package/dist/{Notepad-2OJ53Q7M.js → Notepad-TBAF2SHJ.js} +4 -4
  10. package/dist/{Notepad-2OJ53Q7M.js.map → Notepad-TBAF2SHJ.js.map} +1 -1
  11. package/dist/{PomodoroTimer-M7MDQTVN.js → PomodoroTimer-CDAJC2NX.js} +4 -4
  12. package/dist/{PomodoroTimer-M7MDQTVN.js.map → PomodoroTimer-CDAJC2NX.js.map} +1 -1
  13. package/dist/Preview-CHYOTNEP.js +9 -0
  14. package/dist/{Preview-65VPPGWD.js.map → Preview-CHYOTNEP.js.map} +1 -1
  15. package/dist/{Spreadsheet-VIYNZ6KJ.js → Spreadsheet-I2GJKC6Y.js} +4 -4
  16. package/dist/{Spreadsheet-VIYNZ6KJ.js.map → Spreadsheet-I2GJKC6Y.js.map} +1 -1
  17. package/dist/apps/index.js +15 -15
  18. package/dist/{chunk-ZEMXT6BR.js → chunk-75YZMM5Q.js} +5 -5
  19. package/dist/{chunk-ZEMXT6BR.js.map → chunk-75YZMM5Q.js.map} +1 -1
  20. package/dist/{chunk-WG3PMYDQ.js → chunk-DF6TNR4B.js} +5 -5
  21. package/dist/{chunk-WG3PMYDQ.js.map → chunk-DF6TNR4B.js.map} +1 -1
  22. package/dist/{chunk-YQWFKXWH.js → chunk-EHB2QYJX.js} +3 -3
  23. package/dist/{chunk-YQWFKXWH.js.map → chunk-EHB2QYJX.js.map} +1 -1
  24. package/dist/{chunk-P75EON66.js → chunk-FIUUGSGJ.js} +3 -3
  25. package/dist/{chunk-P75EON66.js.map → chunk-FIUUGSGJ.js.map} +1 -1
  26. package/dist/{chunk-IXW6775F.js → chunk-IONOMHM2.js} +3 -3
  27. package/dist/{chunk-IXW6775F.js.map → chunk-IONOMHM2.js.map} +1 -1
  28. package/dist/{chunk-KUIPWCTJ.js → chunk-NUPYEVU4.js} +3 -3
  29. package/dist/{chunk-KUIPWCTJ.js.map → chunk-NUPYEVU4.js.map} +1 -1
  30. package/dist/{chunk-5HXHD62G.js → chunk-QFOOSV3I.js} +5 -5
  31. package/dist/{chunk-5HXHD62G.js.map → chunk-QFOOSV3I.js.map} +1 -1
  32. package/dist/{chunk-V6N2NXHQ.js → chunk-RADCIHYG.js} +3 -3
  33. package/dist/{chunk-V6N2NXHQ.js.map → chunk-RADCIHYG.js.map} +1 -1
  34. package/dist/{chunk-WIJ45SYD.js → chunk-VENYVK3L.js} +18 -10
  35. package/dist/chunk-VENYVK3L.js.map +1 -0
  36. package/dist/index.d.ts +12 -3
  37. package/dist/index.js +26 -23
  38. package/dist/index.js.map +1 -1
  39. package/package.json +1 -1
  40. package/dist/Files-FJL2BZDI.js +0 -13
  41. package/dist/ImageAnnotator-CTTMAY5Z.js +0 -4
  42. package/dist/Preview-65VPPGWD.js +0 -9
  43. package/dist/chunk-WIJ45SYD.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/apps/PomodoroTimer.tsx"],"names":[],"mappings":"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","file":"PomodoroTimer-M7MDQTVN.js","sourcesContent":["import { useEffect, useState, useCallback, useSyncExternalStore, useRef, useMemo } from 'react';\nimport { useWidgetSettings } from '../shell/Modal';\nimport WidgetSettingsModal, { loadAppearance, type WidgetAppearance } from '../shell/WidgetSettingsModal';\nimport { useShellPrefs } from '../shell/ShellPrefs';\nimport {\n ALARM_OPTIONS, FOCUS_SOUND_OPTIONS,\n getPomoSnapshot, subscribePomo,\n setPomoDurations, setPomoAlarm, setPomoAlarmConfig, setPomoBehaviour, setPomoFocusSound,\n pomoStart, pomoPause, pomoSwitchMode,\n playAlarm, previewFocusSound,\n type AlarmSound, type FocusSound, type Mode,\n} from '../shell/pomodoroStore';\nimport { useTodoTasks, migratePomodoroTasksOnce } from './_todoStore';\nimport type { TodoTask } from './_todoTypes';\n\nconst POMO_SETTINGS_KEY = 'pomodoro_appearance';\nconst ACTIVE_TASK_KEY = 'pomodoro_active_task_id';\n\nconst MODE_LABELS: Record<Mode, string> = { focus: 'Pomodoro', short: 'Short Break', long: 'Long Break' };\n/**\n * Per-mode background colour. The break modes keep their own colours so\n * a glance at the panel tells you what state you're in. Focus mode is\n * special — its panel uses `--taskbar-bg-rgb` so the widget matches the\n * taskbar (and the rest of the dashboard widgets) across light and dark\n * themes. The hex below is only used for the START / Save accent colour\n * in focus mode.\n */\nconst MODE_COLORS: Record<Mode, string> = {\n focus: '#0f172a', // unused for the panel; only for focus-mode accents\n short: '#508a52', // muted green\n long: '#5b7898', // slate blue\n};\n\n/** Convert `#rrggbb` to an `rgba(r, g, b, alpha)` string so the break\n * panels pick up the user's translucency setting via the alpha channel\n * (instead of an `opacity` on the whole element, which would also fade\n * the text and the START button). */\nfunction hexToRgba(hex: string, alpha: number): string {\n const r = parseInt(hex.slice(1, 3), 16);\n const g = parseInt(hex.slice(3, 5), 16);\n const b = parseInt(hex.slice(5, 7), 16);\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n\ninterface PomoPrefs {\n focusMinutes: number;\n shortBreakMinutes: number;\n longBreakMinutes: number;\n autoStartBreaks: boolean;\n autoStartPomodoros: boolean;\n longBreakInterval: number;\n autoCheckTasks: boolean;\n checkToBottom: boolean;\n alarmSound: AlarmSound;\n alarmVolume: number;\n alarmRepeat: number;\n focusSound: FocusSound;\n focusVolume: number;\n}\nconst DEFAULT_PREFS: PomoPrefs = {\n focusMinutes: 25, shortBreakMinutes: 5, longBreakMinutes: 15,\n autoStartBreaks: false, autoStartPomodoros: false, longBreakInterval: 4,\n autoCheckTasks: true, checkToBottom: true,\n alarmSound: 'bell', alarmVolume: 100, alarmRepeat: 1,\n focusSound: 'none', focusVolume: 50,\n};\n\nconst intInRange = (v: unknown, min: number, max: number, fallback: number) =>\n typeof v === 'number' && isFinite(v) ? Math.max(min, Math.min(max, Math.round(v))) : fallback;\n\nfunction readPrefs(raw: unknown): PomoPrefs {\n const p = (raw && typeof raw === 'object') ? raw as Partial<PomoPrefs> : {};\n return {\n focusMinutes: intInRange(p.focusMinutes, 1, 120, DEFAULT_PREFS.focusMinutes),\n shortBreakMinutes: intInRange(p.shortBreakMinutes, 1, 120, DEFAULT_PREFS.shortBreakMinutes),\n longBreakMinutes: intInRange(p.longBreakMinutes, 1, 120, DEFAULT_PREFS.longBreakMinutes),\n autoStartBreaks: typeof p.autoStartBreaks === 'boolean' ? p.autoStartBreaks : DEFAULT_PREFS.autoStartBreaks,\n autoStartPomodoros: typeof p.autoStartPomodoros === 'boolean' ? p.autoStartPomodoros : DEFAULT_PREFS.autoStartPomodoros,\n longBreakInterval: intInRange(p.longBreakInterval, 1, 20, DEFAULT_PREFS.longBreakInterval),\n autoCheckTasks: typeof p.autoCheckTasks === 'boolean' ? p.autoCheckTasks : DEFAULT_PREFS.autoCheckTasks,\n checkToBottom: typeof p.checkToBottom === 'boolean' ? p.checkToBottom : DEFAULT_PREFS.checkToBottom,\n alarmSound: ALARM_OPTIONS.some(o => o.id === p.alarmSound) ? p.alarmSound as AlarmSound : DEFAULT_PREFS.alarmSound,\n alarmVolume: intInRange(p.alarmVolume, 0, 100, DEFAULT_PREFS.alarmVolume),\n alarmRepeat: intInRange(p.alarmRepeat, 1, 5, DEFAULT_PREFS.alarmRepeat),\n focusSound: FOCUS_SOUND_OPTIONS.some(o => o.id === p.focusSound) ? p.focusSound as FocusSound : DEFAULT_PREFS.focusSound,\n focusVolume: intInRange(p.focusVolume, 0, 100, DEFAULT_PREFS.focusVolume),\n };\n}\n\n// The Pomodoro widget treats each TodoTask as if it has the four\n// fields it cares about (name / estimated / completed / done). Tasks\n// without `estimated` show as `?` slots; tasks without `completed`\n// start at 0 and increment when their pomos finish.\nconst sortDoneToBottom = (arr: TodoTask[]): TodoTask[] => {\n const undone: TodoTask[] = [];\n const done: TodoTask[] = [];\n for (const t of arr) (t.done ? done : undone).push(t);\n return [...undone, ...done];\n};\n\nfunction useMemoSortedTasks(rawTasks: TodoTask[], checkToBottom: boolean): TodoTask[] {\n return useMemo(\n () => checkToBottom ? sortDoneToBottom(rawTasks) : rawTasks,\n [rawTasks, checkToBottom],\n );\n}\n\nexport default function PomodoroTimer() {\n const snap = useSyncExternalStore(subscribePomo, getPomoSnapshot, getPomoSnapshot);\n const { prefs: shellPrefs, save: saveShellPrefs } = useShellPrefs();\n const userPrefs = readPrefs(shellPrefs.pomodoro_settings);\n\n // Sync prefs → store. Split into focused effects so each setting only\n // resets the bit it owns (e.g. flipping the alarm volume doesn't tear\n // down the focus-sound playback).\n useEffect(() => {\n setPomoDurations({\n focus: userPrefs.focusMinutes * 60,\n short: userPrefs.shortBreakMinutes * 60,\n long: userPrefs.longBreakMinutes * 60,\n });\n }, [userPrefs.focusMinutes, userPrefs.shortBreakMinutes, userPrefs.longBreakMinutes]);\n\n useEffect(() => {\n setPomoAlarm(userPrefs.alarmSound);\n setPomoAlarmConfig(userPrefs.alarmVolume, userPrefs.alarmRepeat);\n }, [userPrefs.alarmSound, userPrefs.alarmVolume, userPrefs.alarmRepeat]);\n\n useEffect(() => {\n setPomoBehaviour({\n autoStartBreaks: userPrefs.autoStartBreaks,\n autoStartPomodoros: userPrefs.autoStartPomodoros,\n longBreakInterval: userPrefs.longBreakInterval,\n });\n }, [userPrefs.autoStartBreaks, userPrefs.autoStartPomodoros, userPrefs.longBreakInterval]);\n\n useEffect(() => {\n setPomoFocusSound(userPrefs.focusSound, userPrefs.focusVolume);\n }, [userPrefs.focusSound, userPrefs.focusVolume]);\n\n // Ask once for desktop-notification permission.\n useEffect(() => {\n if (typeof Notification !== 'undefined' && Notification.permission === 'default') {\n try { Notification.requestPermission(); } catch {}\n }\n }, []);\n\n // ── Tasks (shared store) ──\n // Tasks now live in `shellPrefs.todo_tasks` so the Todo List app and\n // Calendar app see the same data. The Pomodoro widget keeps a\n // per-instance `activeTaskId` in localStorage so the user's selection\n // doesn't leak across windows.\n //\n // The Pomodoro list is intentionally narrow — it only shows tasks\n // that are *actionable today*: anything due today (pending or just-\n // completed) plus anything overdue and still pending. Far-future\n // tasks live in the Todo List app; completed tasks from earlier days\n // roll off automatically. Tasks added via the Pomodoro's `+ Add Task`\n // button default to today's `dueDate` so they appear immediately in\n // this filtered view.\n const todo = useTodoTasks();\n const rawTasks = todo.tasks;\n const todayStr = useMemo(() => new Date().toISOString().slice(0, 10), []);\n const visibleRawTasks = useMemo(\n () => rawTasks.filter(t => {\n if (!t.dueDate) return false;\n if (t.dueDate === todayStr) return true; // due today — show even when done\n if (t.dueDate < todayStr && !t.done) return true; // overdue and still pending\n return false;\n }),\n [rawTasks, todayStr],\n );\n const tasks = useMemoSortedTasks(visibleRawTasks, userPrefs.checkToBottom);\n\n const [activeTaskId, setActiveTaskId] = useState<string | null>(() => localStorage.getItem(ACTIVE_TASK_KEY));\n const [adding, setAdding] = useState(false);\n const [menuOpenId, setMenuOpenId] = useState<string | null>(null);\n\n // One-shot migration from the legacy localStorage `pomodoro_tasks` key.\n const migratedRef = useRef(false);\n useEffect(() => {\n if (migratedRef.current) return;\n migratedRef.current = true;\n migratePomodoroTasksOnce(rawTasks, todo.setAllTasks);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n try {\n if (activeTaskId) localStorage.setItem(ACTIVE_TASK_KEY, activeTaskId);\n else localStorage.removeItem(ACTIVE_TASK_KEY);\n } catch {}\n }, [activeTaskId]);\n\n // Close the row menu on any outside click.\n useEffect(() => {\n if (!menuOpenId) return;\n const handler = () => setMenuOpenId(null);\n const t = setTimeout(() => document.addEventListener('click', handler), 0);\n return () => { clearTimeout(t); document.removeEventListener('click', handler); };\n }, [menuOpenId]);\n\n // When a focus block completes (`streak` increments), bump the active\n // task's completed count. If `autoCheckTasks` is on and it hits the\n // estimate, mark it done. (`checkToBottom` is honoured by the\n // sortedTasks view; the underlying store doesn't reorder.)\n const lastStreakRef = useRef(snap.streak);\n useEffect(() => {\n if (snap.streak > lastStreakRef.current && activeTaskId) {\n const t = rawTasks.find(x => x.id === activeTaskId);\n if (t) {\n const completed = (t.completed ?? 0) + 1;\n const estimated = t.estimated ?? 0;\n const shouldAutoCheck = userPrefs.autoCheckTasks && estimated > 0 && completed >= estimated;\n todo.updateTask(activeTaskId, { completed, done: t.done || shouldAutoCheck });\n }\n }\n lastStreakRef.current = snap.streak;\n }, [snap.streak, activeTaskId, userPrefs.autoCheckTasks, rawTasks, todo]);\n\n const addTask = (name: string, estimated: number) => {\n const trimmed = name.trim();\n if (!trimmed) return;\n // Default the deadline to today so the new task immediately appears\n // in the Pomodoro's today/overdue list. The user can change the\n // due date later from the Todo List app's edit drawer.\n const id = todo.addTask({ name: trimmed, estimated, completed: 0, dueDate: todayStr });\n if (!activeTaskId) setActiveTaskId(id);\n setAdding(false);\n };\n const removeTask = (id: string) => {\n todo.removeTask(id);\n if (activeTaskId === id) setActiveTaskId(null);\n setMenuOpenId(null);\n };\n const toggleDone = (id: string) => {\n todo.toggleDone(id);\n };\n\n // ── Stats ──\n const totalCompleted = tasks.reduce((acc, t) => acc + (t.completed ?? 0), 0);\n const totalEstimated = tasks.reduce((acc, t) => acc + Math.max(t.estimated ?? 0, t.completed ?? 0), 0);\n const remainingPomos = tasks.reduce((acc, t) => t.done ? acc : acc + Math.max(0, (t.estimated ?? 0) - (t.completed ?? 0)), 0);\n const remainingSecsFromTimer = (snap.running && snap.mode === 'focus') ? snap.remaining : 0;\n const remainingSecs = remainingPomos * userPrefs.focusMinutes * 60 + remainingSecsFromTimer;\n const finishAt = new Date(Date.now() + remainingSecs * 1000);\n const finishAtStr = finishAt.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit', hour12: false });\n const totalHours = (remainingSecs / 3600).toFixed(1);\n\n // Look the active task up in the *full* `rawTasks` list, not the\n // filtered `tasks`. That way the indicator above the list still shows\n // the active task name even if the user moved its due date out of\n // today/overdue range from the Todo List app.\n const activeTask = activeTaskId ? rawTasks.find(t => t.id === activeTaskId) ?? null : null;\n\n const mm = String(Math.floor(snap.remaining / 60)).padStart(2, '0');\n const ss = String(snap.remaining % 60).padStart(2, '0');\n\n // ── Settings modal ──\n const [appearance, setAppearance] = useState(() => loadAppearance(POMO_SETTINGS_KEY));\n const [settingsOpen, setSettingsOpen] = useState(false);\n const [configAppearance, setConfigAppearance] = useState<WidgetAppearance>(appearance);\n const [configPrefs, setConfigPrefs] = useState<PomoPrefs>(userPrefs);\n\n useWidgetSettings(useCallback(() => {\n setConfigAppearance({ ...appearance });\n setConfigPrefs({ ...userPrefs });\n setSettingsOpen(true);\n }, [appearance, userPrefs]));\n\n const onSave = () => {\n setAppearance(configAppearance);\n localStorage.setItem(POMO_SETTINGS_KEY, JSON.stringify(configAppearance));\n saveShellPrefs({ pomodoro_settings: configPrefs });\n setSettingsOpen(false);\n };\n\n // Focus mode: panel takes the taskbar's colour (light or dark per\n // theme), text inherits the system theme via Tailwind gray-* classes.\n // Break modes: keep their hex-based colored panels with white text.\n const isColored = snap.mode !== 'focus';\n const bg = MODE_COLORS[snap.mode];\n const panelBg = isColored\n ? hexToRgba(bg, appearance.activeOpacity / 100)\n : `rgb(var(--taskbar-bg-rgb, 243 244 246) / ${appearance.activeOpacity / 100})`;\n\n // Class helpers — switch text colours based on whether the panel is a\n // bright break colour or the (theme-aware) focus panel.\n const tx = {\n primary: isColored ? 'text-white' : 'text-gray-900',\n secondary: isColored ? 'text-white/85' : 'text-gray-700',\n muted: isColored ? 'text-white/65' : 'text-gray-500',\n faded: isColored ? 'text-white/55' : 'text-gray-400',\n tabActive: isColored ? 'bg-white/15 text-white' : 'bg-gray-200 text-gray-900',\n tabInactive: isColored ? 'text-white/75 hover:bg-white/10' : 'text-gray-500 hover:bg-gray-200',\n divider: isColored ? 'border-white/30' : 'border-gray-200',\n softDivider: isColored ? 'border-white/20' : 'border-gray-200',\n iconBtn: isColored ? 'bg-black/15 hover:bg-black/25 text-white/85' : 'bg-gray-200 hover:bg-gray-300 text-gray-600',\n addTaskBtn: isColored ? 'border-white/45 text-white/90 hover:bg-white/[0.06]' : 'border-gray-300 text-gray-500 hover:bg-gray-200/50',\n };\n // START / Save accent colour: in colored modes use the mode hex; in\n // focus mode use a fixed dark slate so the white pill stays high-\n // contrast on both light and dark themes.\n const accentColor = isColored ? bg : '#0f172a';\n\n return (\n <>\n <div className={`flex flex-col h-full select-none transition-colors duration-300 ${tx.primary}`}\n style={{\n backgroundColor: panelBg,\n backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : undefined,\n }}>\n\n {/* Mode tabs */}\n <div className=\"px-3 pt-3 flex justify-center gap-1.5\">\n {(['focus', 'short', 'long'] as Mode[]).map(m => (\n <button key={m} onClick={() => pomoSwitchMode(m)}\n className={`px-2.5 py-1 text-[13px] font-bold rounded transition-colors ${snap.mode === m ? tx.tabActive : tx.tabInactive}`}>\n {MODE_LABELS[m]}\n </button>\n ))}\n </div>\n\n {/* Big timer */}\n <div className=\"text-center font-bold tabular-nums leading-none tracking-tight mt-3\" style={{ fontSize: '4.5rem' }}>\n {mm}:{ss}\n </div>\n\n {/* START / PAUSE button */}\n <div className=\"flex justify-center mt-3 mb-2\">\n <button onClick={() => snap.running ? pomoPause() : pomoStart()}\n className=\"bg-white px-12 py-2.5 rounded-md font-bold text-base shadow-[0_4px_0_rgba(0,0,0,0.12)] hover:shadow-[0_3px_0_rgba(0,0,0,0.12)] active:translate-y-1 active:shadow-none transition-all\"\n style={{ color: accentColor, letterSpacing: '0.08em' }}>\n {snap.running ? 'PAUSE' : 'START'}\n </button>\n </div>\n\n {/* Active task indicator */}\n <div className=\"px-3 py-3 text-center\">\n {activeTask ? (\n <>\n <div className={`text-xs leading-tight ${tx.muted}`}>#{(activeTask.completed ?? 0) + 1}</div>\n <div className={`text-[15px] font-medium leading-tight truncate mt-0.5 ${tx.primary}`}>{activeTask.name}</div>\n </>\n ) : (\n <div className={`text-xs italic ${tx.muted}`}>No task selected</div>\n )}\n </div>\n\n {/* Tasks header */}\n <div className=\"px-3 mt-1 flex items-center justify-between\">\n <h3 className={`text-base font-bold tracking-tight ${tx.primary}`}>Tasks</h3>\n <button className={`rounded p-1 transition-colors ${tx.iconBtn}`} aria-label=\"Tasks menu\" tabIndex={-1}>\n <svg className=\"h-4 w-4\" viewBox=\"0 0 24 24\" fill=\"currentColor\">\n <circle cx=\"12\" cy=\"5\" r=\"1.4\" /><circle cx=\"12\" cy=\"12\" r=\"1.4\" /><circle cx=\"12\" cy=\"19\" r=\"1.4\" />\n </svg>\n </button>\n </div>\n <div className={`border-t mx-3 mt-2 ${tx.divider}`} />\n\n {/* Tasks list */}\n <div className=\"flex-1 px-3 py-3 space-y-2 overflow-y-auto\">\n {tasks.map(task => {\n const isActive = task.id === activeTaskId;\n return (\n <div key={task.id}\n onClick={() => setActiveTaskId(task.id)}\n className={`bg-white text-gray-800 rounded-md flex items-center pr-2 py-2.5 shadow-sm cursor-pointer transition-shadow hover:shadow-md ${isActive ? 'border-l-4 border-gray-700 pl-2' : 'pl-3 border-l-4 border-transparent'}`}>\n <button onClick={(e) => { e.stopPropagation(); toggleDone(task.id); }}\n className=\"shrink-0\">\n {task.done ? (\n <svg className=\"h-5 w-5\" fill=\"currentColor\" viewBox=\"0 0 20 20\" style={{ color: accentColor }}>\n <path d=\"M10 0a10 10 0 100 20 10 10 0 000-20zm-1 14.5l-4.5-4.5 1.4-1.4 3.1 3.1 6.1-6.1 1.4 1.4z\" />\n </svg>\n ) : (\n <svg className=\"h-5 w-5 text-gray-300\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2} viewBox=\"0 0 24 24\">\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n </svg>\n )}\n </button>\n <span className={`ml-3 flex-1 font-semibold truncate ${task.done ? 'line-through text-gray-400' : 'text-gray-800'}`}>\n {task.name}\n </span>\n <span className=\"text-sm tabular-nums mr-1.5 shrink-0\">\n <span className=\"font-bold text-gray-500\">{task.completed ?? 0}</span>\n <span className=\"text-gray-400\"> / </span>\n <span className=\"text-gray-400\">{task.estimated ?? '–'}</span>\n </span>\n <div className=\"relative\">\n <button onClick={(e) => { e.stopPropagation(); setMenuOpenId(menuOpenId === task.id ? null : task.id); }}\n className=\"rounded p-1 bg-gray-100 hover:bg-gray-200\">\n <svg className=\"h-4 w-4 text-gray-500\" viewBox=\"0 0 24 24\" fill=\"currentColor\">\n <circle cx=\"12\" cy=\"5\" r=\"1.4\" /><circle cx=\"12\" cy=\"12\" r=\"1.4\" /><circle cx=\"12\" cy=\"19\" r=\"1.4\" />\n </svg>\n </button>\n {menuOpenId === task.id && (\n <div className=\"absolute right-0 top-full mt-1 bg-white text-gray-800 rounded shadow-lg border border-gray-200 z-10 min-w-[110px] overflow-hidden\"\n onClick={(e) => e.stopPropagation()}>\n <button onClick={(e) => { e.stopPropagation(); removeTask(task.id); }}\n className=\"block w-full px-3 py-1.5 text-sm text-left hover:bg-red-50 text-red-600\">Delete</button>\n </div>\n )}\n </div>\n </div>\n );\n })}\n\n {/* + Add Task */}\n {!adding ? (\n <button onClick={() => setAdding(true)}\n className={`w-full border-2 border-dashed rounded-md py-3 font-semibold flex items-center justify-center gap-2 transition-colors ${tx.addTaskBtn}`}>\n <svg className=\"h-5 w-5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n <path strokeLinecap=\"round\" d=\"M8 12h8M12 8v8\" />\n </svg>\n Add Task\n </button>\n ) : (\n <AddTaskForm onSubmit={addTask} onCancel={() => setAdding(false)} accentColor={accentColor} />\n )}\n </div>\n\n {/* Footer stats */}\n <div className={`border-t px-3 py-2.5 flex items-center justify-center gap-5 text-sm ${tx.softDivider}`}>\n <span className={tx.secondary}>\n Pomos: <span className={`font-bold ${tx.primary}`}>{totalCompleted}</span>\n <span className={tx.muted}> / </span>\n <span className={`font-bold ${tx.primary}`}>{totalEstimated}</span>\n </span>\n {remainingSecs > 0 && (\n <span className={tx.secondary}>\n Finish At: <span className={`font-bold ${tx.primary}`}>{finishAtStr}</span>\n <span className={tx.muted}> ({totalHours}h)</span>\n </span>\n )}\n </div>\n </div>\n\n <PomodoroSettings\n open={settingsOpen} onClose={() => setSettingsOpen(false)} onSave={onSave}\n configAppearance={configAppearance} setConfigAppearance={setConfigAppearance}\n configPrefs={configPrefs} setConfigPrefs={setConfigPrefs} />\n </>\n );\n}\n\n// ─────────────────────────────────────────────────────────────────────────\n// Settings UI — three sectioned cards (TIMER / TASK / SOUND) matching the\n// reference design. Renders into the standard `WidgetSettingsModal` so we\n// inherit its appearance sliders and the modal chrome.\n// ─────────────────────────────────────────────────────────────────────────\n\nfunction PomodoroSettings({\n open, onClose, onSave, configAppearance, setConfigAppearance, configPrefs, setConfigPrefs,\n}: {\n open: boolean;\n onClose: () => void;\n onSave: () => void;\n configAppearance: WidgetAppearance;\n setConfigAppearance: (a: WidgetAppearance) => void;\n configPrefs: PomoPrefs;\n setConfigPrefs: React.Dispatch<React.SetStateAction<PomoPrefs>>;\n}) {\n const set = <K extends keyof PomoPrefs>(k: K, v: PomoPrefs[K]) => setConfigPrefs(p => ({ ...p, [k]: v }));\n\n return (\n <WidgetSettingsModal open={open} onClose={onClose} title=\"Pomodoro Settings\"\n appearance={configAppearance} onAppearanceChange={setConfigAppearance} onSave={onSave}>\n\n {/* ── TIMER ── */}\n <Section icon={<ClockIcon />} label=\"TIMER\">\n <div>\n <p className=\"text-sm font-bold text-gray-800 mb-2\">Time (minutes)</p>\n <div className=\"grid grid-cols-3 gap-3\">\n <NumberField label=\"Pomodoro\" value={configPrefs.focusMinutes} onChange={v => set('focusMinutes', v)} max={120} />\n <NumberField label=\"Short Break\" value={configPrefs.shortBreakMinutes} onChange={v => set('shortBreakMinutes', v)} max={120} />\n <NumberField label=\"Long Break\" value={configPrefs.longBreakMinutes} onChange={v => set('longBreakMinutes', v)} max={120} />\n </div>\n </div>\n\n <Row label=\"Auto Start Breaks\">\n <Toggle checked={configPrefs.autoStartBreaks} onChange={v => set('autoStartBreaks', v)} />\n </Row>\n\n <Row label=\"Auto Start Pomodoros\">\n <Toggle checked={configPrefs.autoStartPomodoros} onChange={v => set('autoStartPomodoros', v)} />\n </Row>\n\n <Row label=\"Long Break interval\">\n <NumberInput value={configPrefs.longBreakInterval} onChange={v => set('longBreakInterval', v)} min={1} max={20} className=\"w-14 text-center\" />\n </Row>\n </Section>\n\n {/* ── TASK ── */}\n <Section icon={<TaskIcon />} label=\"TASK\">\n <Row\n label=\"Auto Check Tasks\"\n info={'If you enable \"Auto Check Tasks\", the active task will be automatically checked when the actual pomodoro count reaches the estimated count.'}>\n <Toggle checked={configPrefs.autoCheckTasks} onChange={v => set('autoCheckTasks', v)} />\n </Row>\n <Row\n label=\"Check to Bottom\"\n info={'If you enable \"Check to Bottom\", the checked task will be automatically moved to the bottom of the task list.'}>\n <Toggle checked={configPrefs.checkToBottom} onChange={v => set('checkToBottom', v)} />\n </Row>\n </Section>\n\n {/* ── SOUND ── */}\n <Section icon={<SpeakerIcon />} label=\"SOUND\">\n <Row label=\"Alarm Sound\">\n <div className=\"flex items-center gap-2\">\n <Dropdown value={configPrefs.alarmSound} onChange={v => set('alarmSound', v as AlarmSound)}\n options={ALARM_OPTIONS.map(o => ({ id: o.id, label: o.label }))} />\n {configPrefs.alarmSound !== 'off' && (\n <button type=\"button\" onClick={() => playAlarm(configPrefs.alarmSound, { volume: configPrefs.alarmVolume, repeat: 1 })}\n className=\"px-2 py-1 text-[11px] rounded border border-gray-300 text-gray-600 hover:bg-gray-100\">\n Play\n </button>\n )}\n </div>\n </Row>\n <SliderRow value={configPrefs.alarmVolume} onChange={v => set('alarmVolume', v)} />\n <Row label=\"repeat\">\n <NumberInput value={configPrefs.alarmRepeat} onChange={v => set('alarmRepeat', v)} min={1} max={5} className=\"w-14 text-center\" />\n </Row>\n\n <Row label=\"Focus Sound\">\n <div className=\"flex items-center gap-2\">\n <Dropdown value={configPrefs.focusSound} onChange={v => set('focusSound', v as FocusSound)}\n options={FOCUS_SOUND_OPTIONS} />\n {configPrefs.focusSound !== 'none' && (\n <button type=\"button\" onClick={() => previewFocusSound(configPrefs.focusSound, configPrefs.focusVolume)}\n className=\"px-2 py-1 text-[11px] rounded border border-gray-300 text-gray-600 hover:bg-gray-100\">\n Play\n </button>\n )}\n </div>\n </Row>\n <SliderRow value={configPrefs.focusVolume} onChange={v => set('focusVolume', v)} />\n </Section>\n </WidgetSettingsModal>\n );\n}\n\n// ─────────────────────────────────────────────────────────────────────────\n// Tiny presentational helpers — kept here (not exported) so the settings\n// markup above reads top-to-bottom without any noise.\n// ─────────────────────────────────────────────────────────────────────────\n\nfunction Section({ icon, label, children }: { icon: React.ReactNode; label: string; children: React.ReactNode }) {\n return (\n <div className=\"border-t border-gray-200 pt-3 first:border-0 first:pt-0\">\n <div className=\"flex items-center gap-1.5 mb-3 text-gray-400\">\n {icon}\n <span className=\"text-xs font-bold tracking-[0.15em]\">{label}</span>\n </div>\n <div className=\"space-y-3\">{children}</div>\n </div>\n );\n}\n\nfunction Row({ label, info, children }: { label: string; info?: string; children: React.ReactNode }) {\n return (\n <div className=\"flex items-center justify-between gap-3\">\n <div className=\"flex items-center gap-1.5 min-w-0\">\n <span className=\"text-sm font-semibold text-gray-800\">{label}</span>\n {info && <InfoIcon title={info} />}\n </div>\n <div className=\"shrink-0\">{children}</div>\n </div>\n );\n}\n\nfunction SliderRow({ value, onChange }: { value: number; onChange: (v: number) => void }) {\n return (\n <div className=\"flex items-center justify-end gap-3\">\n <span className=\"text-xs text-gray-400 w-7 text-right tabular-nums\">{value}</span>\n <input type=\"range\" min={0} max={100} value={value}\n onChange={e => onChange(parseInt(e.target.value, 10))}\n className=\"w-40 accent-blue-500\" />\n </div>\n );\n}\n\nfunction NumberField({ label, value, onChange, min = 1, max = 99 }: { label: string; value: number; onChange: (v: number) => void; min?: number; max?: number }) {\n return (\n <label className=\"flex flex-col\">\n <span className=\"text-xs font-semibold text-gray-500 mb-1\">{label}</span>\n <NumberInput value={value} onChange={onChange} min={min} max={max} />\n </label>\n );\n}\n\nfunction NumberInput({ value, onChange, min = 1, max = 99, className = '' }: { value: number; onChange: (v: number) => void; min?: number; max?: number; className?: string }) {\n return (\n <input type=\"number\" min={min} max={max} value={value}\n onChange={e => onChange(Math.max(min, Math.min(max, parseInt(e.target.value, 10) || min)))}\n className={`bg-gray-100 border-0 rounded-md px-2 py-1.5 text-sm font-medium text-gray-800 focus:outline-none focus:ring-2 focus:ring-blue-500 w-full ${className}`} />\n );\n}\n\nfunction Dropdown<T extends string>({ value, onChange, options }: { value: T; onChange: (v: T) => void; options: { id: T; label: string }[] }) {\n return (\n <div className=\"relative\">\n <select value={value} onChange={e => onChange(e.target.value as T)}\n className=\"appearance-none bg-gray-100 border-0 rounded-md pl-3 pr-8 py-1.5 text-sm font-medium text-gray-800 focus:outline-none focus:ring-2 focus:ring-blue-500 cursor-pointer\">\n {options.map(o => <option key={o.id} value={o.id}>{o.label}</option>)}\n </select>\n <svg className=\"absolute right-2 top-1/2 -translate-y-1/2 h-3.5 w-3.5 text-gray-400 pointer-events-none\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M6 9l6 6 6-6\" />\n </svg>\n </div>\n );\n}\n\nfunction Toggle({ checked, onChange }: { checked: boolean; onChange: (v: boolean) => void }) {\n return (\n <button type=\"button\" role=\"switch\" aria-checked={checked} onClick={() => onChange(!checked)}\n className={`relative inline-flex h-6 w-11 items-center rounded-full transition-colors ${checked ? 'bg-emerald-500' : 'bg-gray-300'}`}>\n <span className={`absolute h-5 w-5 rounded-full bg-white shadow transition-transform ${checked ? 'translate-x-[22px]' : 'translate-x-0.5'}`} />\n </button>\n );\n}\n\nfunction InfoIcon({ title }: { title: string }) {\n return (\n <span title={title} className=\"inline-flex items-center justify-center w-4 h-4 rounded-full bg-gray-300 text-white text-[10px] font-bold cursor-help\">i</span>\n );\n}\n\nfunction ClockIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <circle cx=\"12\" cy=\"12\" r=\"9\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M12 7v5l3 2\" />\n </svg>\n );\n}\n\nfunction TaskIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <rect x=\"4\" y=\"4\" width=\"16\" height=\"16\" rx=\"2\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9 12l2 2 4-4\" />\n </svg>\n );\n}\n\nfunction SpeakerIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M11 5L6 9H2v6h4l5 4V5z\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M15.5 8.5a5 5 0 010 7M19 5a9 9 0 010 14\" />\n </svg>\n );\n}\n\nfunction AddTaskForm({ onSubmit, onCancel, accentColor }: { onSubmit: (name: string, estimated: number) => void; onCancel: () => void; accentColor: string }) {\n const [name, setName] = useState('');\n const [est, setEst] = useState(1);\n const submit = () => onSubmit(name, est);\n return (\n <div className=\"bg-white text-gray-800 rounded-md p-3 shadow-md\">\n <input autoFocus value={name} onChange={e => setName(e.target.value)} placeholder=\"What are you working on?\"\n onKeyDown={e => { if (e.key === 'Enter') submit(); if (e.key === 'Escape') onCancel(); }}\n className=\"w-full text-base font-medium bg-transparent border-0 outline-none placeholder-gray-400\" />\n <div className=\"flex items-center justify-between mt-2\">\n <span className=\"text-xs font-semibold text-gray-500\">Est Pomodoros</span>\n <input type=\"number\" min={1} max={20} value={est}\n onChange={e => setEst(Math.max(1, Math.min(20, parseInt(e.target.value, 10) || 1)))}\n className=\"w-16 bg-gray-100 rounded px-2 py-1 text-sm text-right border-0 focus:outline-none focus:ring-2 focus:ring-blue-500\" />\n </div>\n <div className=\"flex gap-2 mt-3 justify-end\">\n <button onClick={onCancel} className=\"px-3 py-1 text-sm text-gray-500 hover:bg-gray-100 rounded\">Cancel</button>\n <button onClick={submit}\n className=\"px-4 py-1 text-sm font-semibold text-white rounded shadow-sm\"\n style={{ backgroundColor: accentColor }}>Save</button>\n </div>\n </div>\n );\n}\n"]}
1
+ {"version":3,"sources":["../src/apps/PomodoroTimer.tsx"],"names":[],"mappings":"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","file":"PomodoroTimer-CDAJC2NX.js","sourcesContent":["import { useEffect, useState, useCallback, useSyncExternalStore, useRef, useMemo } from 'react';\nimport { useWidgetSettings } from '../shell/Modal';\nimport WidgetSettingsModal, { loadAppearance, type WidgetAppearance } from '../shell/WidgetSettingsModal';\nimport { useShellPrefs } from '../shell/ShellPrefs';\nimport {\n ALARM_OPTIONS, FOCUS_SOUND_OPTIONS,\n getPomoSnapshot, subscribePomo,\n setPomoDurations, setPomoAlarm, setPomoAlarmConfig, setPomoBehaviour, setPomoFocusSound,\n pomoStart, pomoPause, pomoSwitchMode,\n playAlarm, previewFocusSound,\n type AlarmSound, type FocusSound, type Mode,\n} from '../shell/pomodoroStore';\nimport { useTodoTasks, migratePomodoroTasksOnce } from './_todoStore';\nimport type { TodoTask } from './_todoTypes';\n\nconst POMO_SETTINGS_KEY = 'pomodoro_appearance';\nconst ACTIVE_TASK_KEY = 'pomodoro_active_task_id';\n\nconst MODE_LABELS: Record<Mode, string> = { focus: 'Pomodoro', short: 'Short Break', long: 'Long Break' };\n/**\n * Per-mode background colour. The break modes keep their own colours so\n * a glance at the panel tells you what state you're in. Focus mode is\n * special — its panel uses `--taskbar-bg-rgb` so the widget matches the\n * taskbar (and the rest of the dashboard widgets) across light and dark\n * themes. The hex below is only used for the START / Save accent colour\n * in focus mode.\n */\nconst MODE_COLORS: Record<Mode, string> = {\n focus: '#0f172a', // unused for the panel; only for focus-mode accents\n short: '#508a52', // muted green\n long: '#5b7898', // slate blue\n};\n\n/** Convert `#rrggbb` to an `rgba(r, g, b, alpha)` string so the break\n * panels pick up the user's translucency setting via the alpha channel\n * (instead of an `opacity` on the whole element, which would also fade\n * the text and the START button). */\nfunction hexToRgba(hex: string, alpha: number): string {\n const r = parseInt(hex.slice(1, 3), 16);\n const g = parseInt(hex.slice(3, 5), 16);\n const b = parseInt(hex.slice(5, 7), 16);\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n\ninterface PomoPrefs {\n focusMinutes: number;\n shortBreakMinutes: number;\n longBreakMinutes: number;\n autoStartBreaks: boolean;\n autoStartPomodoros: boolean;\n longBreakInterval: number;\n autoCheckTasks: boolean;\n checkToBottom: boolean;\n alarmSound: AlarmSound;\n alarmVolume: number;\n alarmRepeat: number;\n focusSound: FocusSound;\n focusVolume: number;\n}\nconst DEFAULT_PREFS: PomoPrefs = {\n focusMinutes: 25, shortBreakMinutes: 5, longBreakMinutes: 15,\n autoStartBreaks: false, autoStartPomodoros: false, longBreakInterval: 4,\n autoCheckTasks: true, checkToBottom: true,\n alarmSound: 'bell', alarmVolume: 100, alarmRepeat: 1,\n focusSound: 'none', focusVolume: 50,\n};\n\nconst intInRange = (v: unknown, min: number, max: number, fallback: number) =>\n typeof v === 'number' && isFinite(v) ? Math.max(min, Math.min(max, Math.round(v))) : fallback;\n\nfunction readPrefs(raw: unknown): PomoPrefs {\n const p = (raw && typeof raw === 'object') ? raw as Partial<PomoPrefs> : {};\n return {\n focusMinutes: intInRange(p.focusMinutes, 1, 120, DEFAULT_PREFS.focusMinutes),\n shortBreakMinutes: intInRange(p.shortBreakMinutes, 1, 120, DEFAULT_PREFS.shortBreakMinutes),\n longBreakMinutes: intInRange(p.longBreakMinutes, 1, 120, DEFAULT_PREFS.longBreakMinutes),\n autoStartBreaks: typeof p.autoStartBreaks === 'boolean' ? p.autoStartBreaks : DEFAULT_PREFS.autoStartBreaks,\n autoStartPomodoros: typeof p.autoStartPomodoros === 'boolean' ? p.autoStartPomodoros : DEFAULT_PREFS.autoStartPomodoros,\n longBreakInterval: intInRange(p.longBreakInterval, 1, 20, DEFAULT_PREFS.longBreakInterval),\n autoCheckTasks: typeof p.autoCheckTasks === 'boolean' ? p.autoCheckTasks : DEFAULT_PREFS.autoCheckTasks,\n checkToBottom: typeof p.checkToBottom === 'boolean' ? p.checkToBottom : DEFAULT_PREFS.checkToBottom,\n alarmSound: ALARM_OPTIONS.some(o => o.id === p.alarmSound) ? p.alarmSound as AlarmSound : DEFAULT_PREFS.alarmSound,\n alarmVolume: intInRange(p.alarmVolume, 0, 100, DEFAULT_PREFS.alarmVolume),\n alarmRepeat: intInRange(p.alarmRepeat, 1, 5, DEFAULT_PREFS.alarmRepeat),\n focusSound: FOCUS_SOUND_OPTIONS.some(o => o.id === p.focusSound) ? p.focusSound as FocusSound : DEFAULT_PREFS.focusSound,\n focusVolume: intInRange(p.focusVolume, 0, 100, DEFAULT_PREFS.focusVolume),\n };\n}\n\n// The Pomodoro widget treats each TodoTask as if it has the four\n// fields it cares about (name / estimated / completed / done). Tasks\n// without `estimated` show as `?` slots; tasks without `completed`\n// start at 0 and increment when their pomos finish.\nconst sortDoneToBottom = (arr: TodoTask[]): TodoTask[] => {\n const undone: TodoTask[] = [];\n const done: TodoTask[] = [];\n for (const t of arr) (t.done ? done : undone).push(t);\n return [...undone, ...done];\n};\n\nfunction useMemoSortedTasks(rawTasks: TodoTask[], checkToBottom: boolean): TodoTask[] {\n return useMemo(\n () => checkToBottom ? sortDoneToBottom(rawTasks) : rawTasks,\n [rawTasks, checkToBottom],\n );\n}\n\nexport default function PomodoroTimer() {\n const snap = useSyncExternalStore(subscribePomo, getPomoSnapshot, getPomoSnapshot);\n const { prefs: shellPrefs, save: saveShellPrefs } = useShellPrefs();\n const userPrefs = readPrefs(shellPrefs.pomodoro_settings);\n\n // Sync prefs → store. Split into focused effects so each setting only\n // resets the bit it owns (e.g. flipping the alarm volume doesn't tear\n // down the focus-sound playback).\n useEffect(() => {\n setPomoDurations({\n focus: userPrefs.focusMinutes * 60,\n short: userPrefs.shortBreakMinutes * 60,\n long: userPrefs.longBreakMinutes * 60,\n });\n }, [userPrefs.focusMinutes, userPrefs.shortBreakMinutes, userPrefs.longBreakMinutes]);\n\n useEffect(() => {\n setPomoAlarm(userPrefs.alarmSound);\n setPomoAlarmConfig(userPrefs.alarmVolume, userPrefs.alarmRepeat);\n }, [userPrefs.alarmSound, userPrefs.alarmVolume, userPrefs.alarmRepeat]);\n\n useEffect(() => {\n setPomoBehaviour({\n autoStartBreaks: userPrefs.autoStartBreaks,\n autoStartPomodoros: userPrefs.autoStartPomodoros,\n longBreakInterval: userPrefs.longBreakInterval,\n });\n }, [userPrefs.autoStartBreaks, userPrefs.autoStartPomodoros, userPrefs.longBreakInterval]);\n\n useEffect(() => {\n setPomoFocusSound(userPrefs.focusSound, userPrefs.focusVolume);\n }, [userPrefs.focusSound, userPrefs.focusVolume]);\n\n // Ask once for desktop-notification permission.\n useEffect(() => {\n if (typeof Notification !== 'undefined' && Notification.permission === 'default') {\n try { Notification.requestPermission(); } catch {}\n }\n }, []);\n\n // ── Tasks (shared store) ──\n // Tasks now live in `shellPrefs.todo_tasks` so the Todo List app and\n // Calendar app see the same data. The Pomodoro widget keeps a\n // per-instance `activeTaskId` in localStorage so the user's selection\n // doesn't leak across windows.\n //\n // The Pomodoro list is intentionally narrow — it only shows tasks\n // that are *actionable today*: anything due today (pending or just-\n // completed) plus anything overdue and still pending. Far-future\n // tasks live in the Todo List app; completed tasks from earlier days\n // roll off automatically. Tasks added via the Pomodoro's `+ Add Task`\n // button default to today's `dueDate` so they appear immediately in\n // this filtered view.\n const todo = useTodoTasks();\n const rawTasks = todo.tasks;\n const todayStr = useMemo(() => new Date().toISOString().slice(0, 10), []);\n const visibleRawTasks = useMemo(\n () => rawTasks.filter(t => {\n if (!t.dueDate) return false;\n if (t.dueDate === todayStr) return true; // due today — show even when done\n if (t.dueDate < todayStr && !t.done) return true; // overdue and still pending\n return false;\n }),\n [rawTasks, todayStr],\n );\n const tasks = useMemoSortedTasks(visibleRawTasks, userPrefs.checkToBottom);\n\n const [activeTaskId, setActiveTaskId] = useState<string | null>(() => localStorage.getItem(ACTIVE_TASK_KEY));\n const [adding, setAdding] = useState(false);\n const [menuOpenId, setMenuOpenId] = useState<string | null>(null);\n\n // One-shot migration from the legacy localStorage `pomodoro_tasks` key.\n const migratedRef = useRef(false);\n useEffect(() => {\n if (migratedRef.current) return;\n migratedRef.current = true;\n migratePomodoroTasksOnce(rawTasks, todo.setAllTasks);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n try {\n if (activeTaskId) localStorage.setItem(ACTIVE_TASK_KEY, activeTaskId);\n else localStorage.removeItem(ACTIVE_TASK_KEY);\n } catch {}\n }, [activeTaskId]);\n\n // Close the row menu on any outside click.\n useEffect(() => {\n if (!menuOpenId) return;\n const handler = () => setMenuOpenId(null);\n const t = setTimeout(() => document.addEventListener('click', handler), 0);\n return () => { clearTimeout(t); document.removeEventListener('click', handler); };\n }, [menuOpenId]);\n\n // When a focus block completes (`streak` increments), bump the active\n // task's completed count. If `autoCheckTasks` is on and it hits the\n // estimate, mark it done. (`checkToBottom` is honoured by the\n // sortedTasks view; the underlying store doesn't reorder.)\n const lastStreakRef = useRef(snap.streak);\n useEffect(() => {\n if (snap.streak > lastStreakRef.current && activeTaskId) {\n const t = rawTasks.find(x => x.id === activeTaskId);\n if (t) {\n const completed = (t.completed ?? 0) + 1;\n const estimated = t.estimated ?? 0;\n const shouldAutoCheck = userPrefs.autoCheckTasks && estimated > 0 && completed >= estimated;\n todo.updateTask(activeTaskId, { completed, done: t.done || shouldAutoCheck });\n }\n }\n lastStreakRef.current = snap.streak;\n }, [snap.streak, activeTaskId, userPrefs.autoCheckTasks, rawTasks, todo]);\n\n const addTask = (name: string, estimated: number) => {\n const trimmed = name.trim();\n if (!trimmed) return;\n // Default the deadline to today so the new task immediately appears\n // in the Pomodoro's today/overdue list. The user can change the\n // due date later from the Todo List app's edit drawer.\n const id = todo.addTask({ name: trimmed, estimated, completed: 0, dueDate: todayStr });\n if (!activeTaskId) setActiveTaskId(id);\n setAdding(false);\n };\n const removeTask = (id: string) => {\n todo.removeTask(id);\n if (activeTaskId === id) setActiveTaskId(null);\n setMenuOpenId(null);\n };\n const toggleDone = (id: string) => {\n todo.toggleDone(id);\n };\n\n // ── Stats ──\n const totalCompleted = tasks.reduce((acc, t) => acc + (t.completed ?? 0), 0);\n const totalEstimated = tasks.reduce((acc, t) => acc + Math.max(t.estimated ?? 0, t.completed ?? 0), 0);\n const remainingPomos = tasks.reduce((acc, t) => t.done ? acc : acc + Math.max(0, (t.estimated ?? 0) - (t.completed ?? 0)), 0);\n const remainingSecsFromTimer = (snap.running && snap.mode === 'focus') ? snap.remaining : 0;\n const remainingSecs = remainingPomos * userPrefs.focusMinutes * 60 + remainingSecsFromTimer;\n const finishAt = new Date(Date.now() + remainingSecs * 1000);\n const finishAtStr = finishAt.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit', hour12: false });\n const totalHours = (remainingSecs / 3600).toFixed(1);\n\n // Look the active task up in the *full* `rawTasks` list, not the\n // filtered `tasks`. That way the indicator above the list still shows\n // the active task name even if the user moved its due date out of\n // today/overdue range from the Todo List app.\n const activeTask = activeTaskId ? rawTasks.find(t => t.id === activeTaskId) ?? null : null;\n\n const mm = String(Math.floor(snap.remaining / 60)).padStart(2, '0');\n const ss = String(snap.remaining % 60).padStart(2, '0');\n\n // ── Settings modal ──\n const [appearance, setAppearance] = useState(() => loadAppearance(POMO_SETTINGS_KEY));\n const [settingsOpen, setSettingsOpen] = useState(false);\n const [configAppearance, setConfigAppearance] = useState<WidgetAppearance>(appearance);\n const [configPrefs, setConfigPrefs] = useState<PomoPrefs>(userPrefs);\n\n useWidgetSettings(useCallback(() => {\n setConfigAppearance({ ...appearance });\n setConfigPrefs({ ...userPrefs });\n setSettingsOpen(true);\n }, [appearance, userPrefs]));\n\n const onSave = () => {\n setAppearance(configAppearance);\n localStorage.setItem(POMO_SETTINGS_KEY, JSON.stringify(configAppearance));\n saveShellPrefs({ pomodoro_settings: configPrefs });\n setSettingsOpen(false);\n };\n\n // Focus mode: panel takes the taskbar's colour (light or dark per\n // theme), text inherits the system theme via Tailwind gray-* classes.\n // Break modes: keep their hex-based colored panels with white text.\n const isColored = snap.mode !== 'focus';\n const bg = MODE_COLORS[snap.mode];\n const panelBg = isColored\n ? hexToRgba(bg, appearance.activeOpacity / 100)\n : `rgb(var(--taskbar-bg-rgb, 243 244 246) / ${appearance.activeOpacity / 100})`;\n\n // Class helpers — switch text colours based on whether the panel is a\n // bright break colour or the (theme-aware) focus panel.\n const tx = {\n primary: isColored ? 'text-white' : 'text-gray-900',\n secondary: isColored ? 'text-white/85' : 'text-gray-700',\n muted: isColored ? 'text-white/65' : 'text-gray-500',\n faded: isColored ? 'text-white/55' : 'text-gray-400',\n tabActive: isColored ? 'bg-white/15 text-white' : 'bg-gray-200 text-gray-900',\n tabInactive: isColored ? 'text-white/75 hover:bg-white/10' : 'text-gray-500 hover:bg-gray-200',\n divider: isColored ? 'border-white/30' : 'border-gray-200',\n softDivider: isColored ? 'border-white/20' : 'border-gray-200',\n iconBtn: isColored ? 'bg-black/15 hover:bg-black/25 text-white/85' : 'bg-gray-200 hover:bg-gray-300 text-gray-600',\n addTaskBtn: isColored ? 'border-white/45 text-white/90 hover:bg-white/[0.06]' : 'border-gray-300 text-gray-500 hover:bg-gray-200/50',\n };\n // START / Save accent colour: in colored modes use the mode hex; in\n // focus mode use a fixed dark slate so the white pill stays high-\n // contrast on both light and dark themes.\n const accentColor = isColored ? bg : '#0f172a';\n\n return (\n <>\n <div className={`flex flex-col h-full select-none transition-colors duration-300 ${tx.primary}`}\n style={{\n backgroundColor: panelBg,\n backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : undefined,\n }}>\n\n {/* Mode tabs */}\n <div className=\"px-3 pt-3 flex justify-center gap-1.5\">\n {(['focus', 'short', 'long'] as Mode[]).map(m => (\n <button key={m} onClick={() => pomoSwitchMode(m)}\n className={`px-2.5 py-1 text-[13px] font-bold rounded transition-colors ${snap.mode === m ? tx.tabActive : tx.tabInactive}`}>\n {MODE_LABELS[m]}\n </button>\n ))}\n </div>\n\n {/* Big timer */}\n <div className=\"text-center font-bold tabular-nums leading-none tracking-tight mt-3\" style={{ fontSize: '4.5rem' }}>\n {mm}:{ss}\n </div>\n\n {/* START / PAUSE button */}\n <div className=\"flex justify-center mt-3 mb-2\">\n <button onClick={() => snap.running ? pomoPause() : pomoStart()}\n className=\"bg-white px-12 py-2.5 rounded-md font-bold text-base shadow-[0_4px_0_rgba(0,0,0,0.12)] hover:shadow-[0_3px_0_rgba(0,0,0,0.12)] active:translate-y-1 active:shadow-none transition-all\"\n style={{ color: accentColor, letterSpacing: '0.08em' }}>\n {snap.running ? 'PAUSE' : 'START'}\n </button>\n </div>\n\n {/* Active task indicator */}\n <div className=\"px-3 py-3 text-center\">\n {activeTask ? (\n <>\n <div className={`text-xs leading-tight ${tx.muted}`}>#{(activeTask.completed ?? 0) + 1}</div>\n <div className={`text-[15px] font-medium leading-tight truncate mt-0.5 ${tx.primary}`}>{activeTask.name}</div>\n </>\n ) : (\n <div className={`text-xs italic ${tx.muted}`}>No task selected</div>\n )}\n </div>\n\n {/* Tasks header */}\n <div className=\"px-3 mt-1 flex items-center justify-between\">\n <h3 className={`text-base font-bold tracking-tight ${tx.primary}`}>Tasks</h3>\n <button className={`rounded p-1 transition-colors ${tx.iconBtn}`} aria-label=\"Tasks menu\" tabIndex={-1}>\n <svg className=\"h-4 w-4\" viewBox=\"0 0 24 24\" fill=\"currentColor\">\n <circle cx=\"12\" cy=\"5\" r=\"1.4\" /><circle cx=\"12\" cy=\"12\" r=\"1.4\" /><circle cx=\"12\" cy=\"19\" r=\"1.4\" />\n </svg>\n </button>\n </div>\n <div className={`border-t mx-3 mt-2 ${tx.divider}`} />\n\n {/* Tasks list */}\n <div className=\"flex-1 px-3 py-3 space-y-2 overflow-y-auto\">\n {tasks.map(task => {\n const isActive = task.id === activeTaskId;\n return (\n <div key={task.id}\n onClick={() => setActiveTaskId(task.id)}\n className={`bg-white text-gray-800 rounded-md flex items-center pr-2 py-2.5 shadow-sm cursor-pointer transition-shadow hover:shadow-md ${isActive ? 'border-l-4 border-gray-700 pl-2' : 'pl-3 border-l-4 border-transparent'}`}>\n <button onClick={(e) => { e.stopPropagation(); toggleDone(task.id); }}\n className=\"shrink-0\">\n {task.done ? (\n <svg className=\"h-5 w-5\" fill=\"currentColor\" viewBox=\"0 0 20 20\" style={{ color: accentColor }}>\n <path d=\"M10 0a10 10 0 100 20 10 10 0 000-20zm-1 14.5l-4.5-4.5 1.4-1.4 3.1 3.1 6.1-6.1 1.4 1.4z\" />\n </svg>\n ) : (\n <svg className=\"h-5 w-5 text-gray-300\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2} viewBox=\"0 0 24 24\">\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n </svg>\n )}\n </button>\n <span className={`ml-3 flex-1 font-semibold truncate ${task.done ? 'line-through text-gray-400' : 'text-gray-800'}`}>\n {task.name}\n </span>\n <span className=\"text-sm tabular-nums mr-1.5 shrink-0\">\n <span className=\"font-bold text-gray-500\">{task.completed ?? 0}</span>\n <span className=\"text-gray-400\"> / </span>\n <span className=\"text-gray-400\">{task.estimated ?? '–'}</span>\n </span>\n <div className=\"relative\">\n <button onClick={(e) => { e.stopPropagation(); setMenuOpenId(menuOpenId === task.id ? null : task.id); }}\n className=\"rounded p-1 bg-gray-100 hover:bg-gray-200\">\n <svg className=\"h-4 w-4 text-gray-500\" viewBox=\"0 0 24 24\" fill=\"currentColor\">\n <circle cx=\"12\" cy=\"5\" r=\"1.4\" /><circle cx=\"12\" cy=\"12\" r=\"1.4\" /><circle cx=\"12\" cy=\"19\" r=\"1.4\" />\n </svg>\n </button>\n {menuOpenId === task.id && (\n <div className=\"absolute right-0 top-full mt-1 bg-white text-gray-800 rounded shadow-lg border border-gray-200 z-10 min-w-[110px] overflow-hidden\"\n onClick={(e) => e.stopPropagation()}>\n <button onClick={(e) => { e.stopPropagation(); removeTask(task.id); }}\n className=\"block w-full px-3 py-1.5 text-sm text-left hover:bg-red-50 text-red-600\">Delete</button>\n </div>\n )}\n </div>\n </div>\n );\n })}\n\n {/* + Add Task */}\n {!adding ? (\n <button onClick={() => setAdding(true)}\n className={`w-full border-2 border-dashed rounded-md py-3 font-semibold flex items-center justify-center gap-2 transition-colors ${tx.addTaskBtn}`}>\n <svg className=\"h-5 w-5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n <path strokeLinecap=\"round\" d=\"M8 12h8M12 8v8\" />\n </svg>\n Add Task\n </button>\n ) : (\n <AddTaskForm onSubmit={addTask} onCancel={() => setAdding(false)} accentColor={accentColor} />\n )}\n </div>\n\n {/* Footer stats */}\n <div className={`border-t px-3 py-2.5 flex items-center justify-center gap-5 text-sm ${tx.softDivider}`}>\n <span className={tx.secondary}>\n Pomos: <span className={`font-bold ${tx.primary}`}>{totalCompleted}</span>\n <span className={tx.muted}> / </span>\n <span className={`font-bold ${tx.primary}`}>{totalEstimated}</span>\n </span>\n {remainingSecs > 0 && (\n <span className={tx.secondary}>\n Finish At: <span className={`font-bold ${tx.primary}`}>{finishAtStr}</span>\n <span className={tx.muted}> ({totalHours}h)</span>\n </span>\n )}\n </div>\n </div>\n\n <PomodoroSettings\n open={settingsOpen} onClose={() => setSettingsOpen(false)} onSave={onSave}\n configAppearance={configAppearance} setConfigAppearance={setConfigAppearance}\n configPrefs={configPrefs} setConfigPrefs={setConfigPrefs} />\n </>\n );\n}\n\n// ─────────────────────────────────────────────────────────────────────────\n// Settings UI — three sectioned cards (TIMER / TASK / SOUND) matching the\n// reference design. Renders into the standard `WidgetSettingsModal` so we\n// inherit its appearance sliders and the modal chrome.\n// ─────────────────────────────────────────────────────────────────────────\n\nfunction PomodoroSettings({\n open, onClose, onSave, configAppearance, setConfigAppearance, configPrefs, setConfigPrefs,\n}: {\n open: boolean;\n onClose: () => void;\n onSave: () => void;\n configAppearance: WidgetAppearance;\n setConfigAppearance: (a: WidgetAppearance) => void;\n configPrefs: PomoPrefs;\n setConfigPrefs: React.Dispatch<React.SetStateAction<PomoPrefs>>;\n}) {\n const set = <K extends keyof PomoPrefs>(k: K, v: PomoPrefs[K]) => setConfigPrefs(p => ({ ...p, [k]: v }));\n\n return (\n <WidgetSettingsModal open={open} onClose={onClose} title=\"Pomodoro Settings\"\n appearance={configAppearance} onAppearanceChange={setConfigAppearance} onSave={onSave}>\n\n {/* ── TIMER ── */}\n <Section icon={<ClockIcon />} label=\"TIMER\">\n <div>\n <p className=\"text-sm font-bold text-gray-800 mb-2\">Time (minutes)</p>\n <div className=\"grid grid-cols-3 gap-3\">\n <NumberField label=\"Pomodoro\" value={configPrefs.focusMinutes} onChange={v => set('focusMinutes', v)} max={120} />\n <NumberField label=\"Short Break\" value={configPrefs.shortBreakMinutes} onChange={v => set('shortBreakMinutes', v)} max={120} />\n <NumberField label=\"Long Break\" value={configPrefs.longBreakMinutes} onChange={v => set('longBreakMinutes', v)} max={120} />\n </div>\n </div>\n\n <Row label=\"Auto Start Breaks\">\n <Toggle checked={configPrefs.autoStartBreaks} onChange={v => set('autoStartBreaks', v)} />\n </Row>\n\n <Row label=\"Auto Start Pomodoros\">\n <Toggle checked={configPrefs.autoStartPomodoros} onChange={v => set('autoStartPomodoros', v)} />\n </Row>\n\n <Row label=\"Long Break interval\">\n <NumberInput value={configPrefs.longBreakInterval} onChange={v => set('longBreakInterval', v)} min={1} max={20} className=\"w-14 text-center\" />\n </Row>\n </Section>\n\n {/* ── TASK ── */}\n <Section icon={<TaskIcon />} label=\"TASK\">\n <Row\n label=\"Auto Check Tasks\"\n info={'If you enable \"Auto Check Tasks\", the active task will be automatically checked when the actual pomodoro count reaches the estimated count.'}>\n <Toggle checked={configPrefs.autoCheckTasks} onChange={v => set('autoCheckTasks', v)} />\n </Row>\n <Row\n label=\"Check to Bottom\"\n info={'If you enable \"Check to Bottom\", the checked task will be automatically moved to the bottom of the task list.'}>\n <Toggle checked={configPrefs.checkToBottom} onChange={v => set('checkToBottom', v)} />\n </Row>\n </Section>\n\n {/* ── SOUND ── */}\n <Section icon={<SpeakerIcon />} label=\"SOUND\">\n <Row label=\"Alarm Sound\">\n <div className=\"flex items-center gap-2\">\n <Dropdown value={configPrefs.alarmSound} onChange={v => set('alarmSound', v as AlarmSound)}\n options={ALARM_OPTIONS.map(o => ({ id: o.id, label: o.label }))} />\n {configPrefs.alarmSound !== 'off' && (\n <button type=\"button\" onClick={() => playAlarm(configPrefs.alarmSound, { volume: configPrefs.alarmVolume, repeat: 1 })}\n className=\"px-2 py-1 text-[11px] rounded border border-gray-300 text-gray-600 hover:bg-gray-100\">\n Play\n </button>\n )}\n </div>\n </Row>\n <SliderRow value={configPrefs.alarmVolume} onChange={v => set('alarmVolume', v)} />\n <Row label=\"repeat\">\n <NumberInput value={configPrefs.alarmRepeat} onChange={v => set('alarmRepeat', v)} min={1} max={5} className=\"w-14 text-center\" />\n </Row>\n\n <Row label=\"Focus Sound\">\n <div className=\"flex items-center gap-2\">\n <Dropdown value={configPrefs.focusSound} onChange={v => set('focusSound', v as FocusSound)}\n options={FOCUS_SOUND_OPTIONS} />\n {configPrefs.focusSound !== 'none' && (\n <button type=\"button\" onClick={() => previewFocusSound(configPrefs.focusSound, configPrefs.focusVolume)}\n className=\"px-2 py-1 text-[11px] rounded border border-gray-300 text-gray-600 hover:bg-gray-100\">\n Play\n </button>\n )}\n </div>\n </Row>\n <SliderRow value={configPrefs.focusVolume} onChange={v => set('focusVolume', v)} />\n </Section>\n </WidgetSettingsModal>\n );\n}\n\n// ─────────────────────────────────────────────────────────────────────────\n// Tiny presentational helpers — kept here (not exported) so the settings\n// markup above reads top-to-bottom without any noise.\n// ─────────────────────────────────────────────────────────────────────────\n\nfunction Section({ icon, label, children }: { icon: React.ReactNode; label: string; children: React.ReactNode }) {\n return (\n <div className=\"border-t border-gray-200 pt-3 first:border-0 first:pt-0\">\n <div className=\"flex items-center gap-1.5 mb-3 text-gray-400\">\n {icon}\n <span className=\"text-xs font-bold tracking-[0.15em]\">{label}</span>\n </div>\n <div className=\"space-y-3\">{children}</div>\n </div>\n );\n}\n\nfunction Row({ label, info, children }: { label: string; info?: string; children: React.ReactNode }) {\n return (\n <div className=\"flex items-center justify-between gap-3\">\n <div className=\"flex items-center gap-1.5 min-w-0\">\n <span className=\"text-sm font-semibold text-gray-800\">{label}</span>\n {info && <InfoIcon title={info} />}\n </div>\n <div className=\"shrink-0\">{children}</div>\n </div>\n );\n}\n\nfunction SliderRow({ value, onChange }: { value: number; onChange: (v: number) => void }) {\n return (\n <div className=\"flex items-center justify-end gap-3\">\n <span className=\"text-xs text-gray-400 w-7 text-right tabular-nums\">{value}</span>\n <input type=\"range\" min={0} max={100} value={value}\n onChange={e => onChange(parseInt(e.target.value, 10))}\n className=\"w-40 accent-blue-500\" />\n </div>\n );\n}\n\nfunction NumberField({ label, value, onChange, min = 1, max = 99 }: { label: string; value: number; onChange: (v: number) => void; min?: number; max?: number }) {\n return (\n <label className=\"flex flex-col\">\n <span className=\"text-xs font-semibold text-gray-500 mb-1\">{label}</span>\n <NumberInput value={value} onChange={onChange} min={min} max={max} />\n </label>\n );\n}\n\nfunction NumberInput({ value, onChange, min = 1, max = 99, className = '' }: { value: number; onChange: (v: number) => void; min?: number; max?: number; className?: string }) {\n return (\n <input type=\"number\" min={min} max={max} value={value}\n onChange={e => onChange(Math.max(min, Math.min(max, parseInt(e.target.value, 10) || min)))}\n className={`bg-gray-100 border-0 rounded-md px-2 py-1.5 text-sm font-medium text-gray-800 focus:outline-none focus:ring-2 focus:ring-blue-500 w-full ${className}`} />\n );\n}\n\nfunction Dropdown<T extends string>({ value, onChange, options }: { value: T; onChange: (v: T) => void; options: { id: T; label: string }[] }) {\n return (\n <div className=\"relative\">\n <select value={value} onChange={e => onChange(e.target.value as T)}\n className=\"appearance-none bg-gray-100 border-0 rounded-md pl-3 pr-8 py-1.5 text-sm font-medium text-gray-800 focus:outline-none focus:ring-2 focus:ring-blue-500 cursor-pointer\">\n {options.map(o => <option key={o.id} value={o.id}>{o.label}</option>)}\n </select>\n <svg className=\"absolute right-2 top-1/2 -translate-y-1/2 h-3.5 w-3.5 text-gray-400 pointer-events-none\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M6 9l6 6 6-6\" />\n </svg>\n </div>\n );\n}\n\nfunction Toggle({ checked, onChange }: { checked: boolean; onChange: (v: boolean) => void }) {\n return (\n <button type=\"button\" role=\"switch\" aria-checked={checked} onClick={() => onChange(!checked)}\n className={`relative inline-flex h-6 w-11 items-center rounded-full transition-colors ${checked ? 'bg-emerald-500' : 'bg-gray-300'}`}>\n <span className={`absolute h-5 w-5 rounded-full bg-white shadow transition-transform ${checked ? 'translate-x-[22px]' : 'translate-x-0.5'}`} />\n </button>\n );\n}\n\nfunction InfoIcon({ title }: { title: string }) {\n return (\n <span title={title} className=\"inline-flex items-center justify-center w-4 h-4 rounded-full bg-gray-300 text-white text-[10px] font-bold cursor-help\">i</span>\n );\n}\n\nfunction ClockIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <circle cx=\"12\" cy=\"12\" r=\"9\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M12 7v5l3 2\" />\n </svg>\n );\n}\n\nfunction TaskIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <rect x=\"4\" y=\"4\" width=\"16\" height=\"16\" rx=\"2\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9 12l2 2 4-4\" />\n </svg>\n );\n}\n\nfunction SpeakerIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M11 5L6 9H2v6h4l5 4V5z\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M15.5 8.5a5 5 0 010 7M19 5a9 9 0 010 14\" />\n </svg>\n );\n}\n\nfunction AddTaskForm({ onSubmit, onCancel, accentColor }: { onSubmit: (name: string, estimated: number) => void; onCancel: () => void; accentColor: string }) {\n const [name, setName] = useState('');\n const [est, setEst] = useState(1);\n const submit = () => onSubmit(name, est);\n return (\n <div className=\"bg-white text-gray-800 rounded-md p-3 shadow-md\">\n <input autoFocus value={name} onChange={e => setName(e.target.value)} placeholder=\"What are you working on?\"\n onKeyDown={e => { if (e.key === 'Enter') submit(); if (e.key === 'Escape') onCancel(); }}\n className=\"w-full text-base font-medium bg-transparent border-0 outline-none placeholder-gray-400\" />\n <div className=\"flex items-center justify-between mt-2\">\n <span className=\"text-xs font-semibold text-gray-500\">Est Pomodoros</span>\n <input type=\"number\" min={1} max={20} value={est}\n onChange={e => setEst(Math.max(1, Math.min(20, parseInt(e.target.value, 10) || 1)))}\n className=\"w-16 bg-gray-100 rounded px-2 py-1 text-sm text-right border-0 focus:outline-none focus:ring-2 focus:ring-blue-500\" />\n </div>\n <div className=\"flex gap-2 mt-3 justify-end\">\n <button onClick={onCancel} className=\"px-3 py-1 text-sm text-gray-500 hover:bg-gray-100 rounded\">Cancel</button>\n <button onClick={submit}\n className=\"px-4 py-1 text-sm font-semibold text-white rounded shadow-sm\"\n style={{ backgroundColor: accentColor }}>Save</button>\n </div>\n </div>\n );\n}\n"]}
@@ -0,0 +1,9 @@
1
+ export { Preview as default, setPdfPreview } from './chunk-DF6TNR4B.js';
2
+ import './chunk-NUPYEVU4.js';
3
+ import './chunk-VENYVK3L.js';
4
+ import './chunk-IONOMHM2.js';
5
+ import './chunk-JNF5VRPB.js';
6
+ import './chunk-UBN4IUDE.js';
7
+ import './chunk-ZF6AYO4G.js';
8
+ //# sourceMappingURL=Preview-CHYOTNEP.js.map
9
+ //# sourceMappingURL=Preview-CHYOTNEP.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"Preview-65VPPGWD.js"}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"Preview-CHYOTNEP.js"}
@@ -1,7 +1,7 @@
1
- export { Spreadsheet as default, setSpreadsheetPreview } from './chunk-V6N2NXHQ.js';
2
- import './chunk-IXW6775F.js';
1
+ export { Spreadsheet as default, setSpreadsheetPreview } from './chunk-RADCIHYG.js';
2
+ import './chunk-IONOMHM2.js';
3
3
  import './chunk-JNF5VRPB.js';
4
4
  import './chunk-UBN4IUDE.js';
5
5
  import './chunk-ZF6AYO4G.js';
6
- //# sourceMappingURL=Spreadsheet-VIYNZ6KJ.js.map
7
- //# sourceMappingURL=Spreadsheet-VIYNZ6KJ.js.map
6
+ //# sourceMappingURL=Spreadsheet-I2GJKC6Y.js.map
7
+ //# sourceMappingURL=Spreadsheet-I2GJKC6Y.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"Spreadsheet-VIYNZ6KJ.js"}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"Spreadsheet-I2GJKC6Y.js"}
@@ -1,29 +1,29 @@
1
- export { setBrowserStartUrl } from '../chunk-YQWFKXWH.js';
2
- export { openFilesInTrashMode, setFilesDemoTree } from '../chunk-5HXHD62G.js';
3
- import '../chunk-ZEMXT6BR.js';
1
+ export { setBrowserStartUrl } from '../chunk-EHB2QYJX.js';
2
+ export { openFilesInTrashMode, setFilesDemoTree } from '../chunk-QFOOSV3I.js';
3
+ import '../chunk-75YZMM5Q.js';
4
4
  import '../chunk-VGTEM5RZ.js';
5
- export { setPdfPreview } from '../chunk-WG3PMYDQ.js';
6
- import '../chunk-KUIPWCTJ.js';
7
- import '../chunk-WIJ45SYD.js';
8
- export { setSpreadsheetPreview } from '../chunk-V6N2NXHQ.js';
9
- export { BUILTIN_APP_INFO } from '../chunk-IXW6775F.js';
5
+ export { setPdfPreview } from '../chunk-DF6TNR4B.js';
6
+ import '../chunk-NUPYEVU4.js';
7
+ import '../chunk-VENYVK3L.js';
8
+ export { setSpreadsheetPreview } from '../chunk-RADCIHYG.js';
9
+ export { BUILTIN_APP_INFO } from '../chunk-IONOMHM2.js';
10
10
  import '../chunk-JNF5VRPB.js';
11
11
  import '../chunk-UBN4IUDE.js';
12
12
  import '../chunk-ZF6AYO4G.js';
13
13
  import { lazy } from 'react';
14
14
 
15
15
  var Calculator = lazy(() => import('../Calculator-LF2IH7NZ.js'));
16
- var Spreadsheet = lazy(() => import('../Spreadsheet-VIYNZ6KJ.js'));
16
+ var Spreadsheet = lazy(() => import('../Spreadsheet-I2GJKC6Y.js'));
17
17
  var Weather = lazy(() => import('../Weather-B55N3VEX.js'));
18
18
  var CurrencyConverter = lazy(() => import('../CurrencyConverter-AZ5RBU7E.js'));
19
- var PomodoroTimer = lazy(() => import('../PomodoroTimer-M7MDQTVN.js'));
20
- var Notepad = lazy(() => import('../Notepad-2OJ53Q7M.js'));
19
+ var PomodoroTimer = lazy(() => import('../PomodoroTimer-CDAJC2NX.js'));
20
+ var Notepad = lazy(() => import('../Notepad-TBAF2SHJ.js'));
21
21
  var WorldClock = lazy(() => import('../WorldClock-SARH4X3Y.js'));
22
22
  var Stock = lazy(() => import('../Stock-7JFY7KUK.js'));
23
- var Preview = lazy(() => import('../Preview-65VPPGWD.js'));
24
- var Documents = lazy(() => import('../Documents-L7CTOPIN.js'));
25
- var Files = lazy(() => import('../Files-FJL2BZDI.js'));
26
- var Browser = lazy(() => import('../Browser-RNO4KYSE.js'));
23
+ var Preview = lazy(() => import('../Preview-CHYOTNEP.js'));
24
+ var Documents = lazy(() => import('../Documents-JXLU2ILH.js'));
25
+ var Files = lazy(() => import('../Files-Q2YCSHYG.js'));
26
+ var Browser = lazy(() => import('../Browser-HDKOMCUA.js'));
27
27
  var utilityApps = {
28
28
  "/calculator": { component: Calculator, label: "Calculator", size: "sm", allowPinOnTop: true, utility: true, widget: true, dimensions: [280, 420] },
29
29
  "/spreadsheet": { component: Spreadsheet, label: "Spreadsheets", size: "2xl", appStyle: true, multiInstance: true },
@@ -1,6 +1,6 @@
1
- import { setPdfPreview } from './chunk-WG3PMYDQ.js';
2
- import { toast_default } from './chunk-WIJ45SYD.js';
3
- import { setSpreadsheetPreview } from './chunk-V6N2NXHQ.js';
1
+ import { setPdfPreview } from './chunk-DF6TNR4B.js';
2
+ import { toast_default } from './chunk-VENYVK3L.js';
3
+ import { setSpreadsheetPreview } from './chunk-RADCIHYG.js';
4
4
  import { navIcons } from './chunk-ZF6AYO4G.js';
5
5
  import { Fragment, isValidElement, cloneElement } from 'react';
6
6
  import { jsx, jsxs } from 'react/jsx-runtime';
@@ -258,5 +258,5 @@ function consumeFilesViewRequest() {
258
258
  }
259
259
 
260
260
  export { Breadcrumbs, DesktopItemMiniIcon, FILES_OPEN_DESKTOP_FOLDER_EVENT, FILES_SHOW_TRASH_EVENT, FileIconTile, FolderGlyph, PREVIEW_OPENED_EVENT, consumeFilesViewRequest, desktopItemTypeLabel, getDesktopFoldersSnapshot, hashGradient, openPreviewFile, peekFilesViewRequest, publishDesktopFolders, requestFilesDesktopFolderView, requestFilesTrashView, subscribeDesktopFolders };
261
- //# sourceMappingURL=chunk-ZEMXT6BR.js.map
262
- //# sourceMappingURL=chunk-ZEMXT6BR.js.map
261
+ //# sourceMappingURL=chunk-75YZMM5Q.js.map
262
+ //# sourceMappingURL=chunk-75YZMM5Q.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/shell/Breadcrumbs.tsx","../src/utils/openPreviewFile.ts","../src/shell/desktopIcons.tsx"],"names":["jsx","jsxs"],"mappings":";;;;;;;AAgCA,IAAM,iBAAA,mBACJ,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,oCAAA,EAAqC,SAAQ,WAAA,EAAY,IAAA,EAAK,cAAA,EAAe,aAAA,EAAW,IAAA,EACrG,QAAA,kBAAA,GAAA;AAAA,EAAC,MAAA;AAAA,EAAA;AAAA,IACC,QAAA,EAAS,SAAA;AAAA,IACT,CAAA,EAAE,4IAAA;AAAA,IACF,QAAA,EAAS;AAAA;AACX,CAAA,EACF,CAAA;AAKa,SAAR,YAA6B,EAAE,KAAA,EAAO,WAAW,QAAA,GAAW,CAAA,EAAG,WAAU,EAAqB;AACnG,EAAA,IAAI,KAAA,CAAM,MAAA,KAAW,CAAA,EAAG,OAAO,IAAA;AAC/B,EAAA,MAAM,MAAM,SAAA,IAAa,iBAAA;AAGzB,EAAA,MAAM,SAAkB,EAAC;AACzB,EAAA,MAAM,SAAA,GAAY,MAAM,MAAA,GAAS,CAAA;AACjC,EAAA,IAAI,WAAW,CAAA,IAAK,KAAA,CAAM,MAAA,GAAS,QAAA,IAAY,YAAY,CAAA,EAAG;AAC5D,IAAA,MAAM,SAAA,GAAY,IAAA,CAAK,GAAA,CAAI,CAAA,EAAG,WAAW,CAAC,CAAA;AAC1C,IAAA,MAAA,CAAO,IAAA,CAAK,EAAE,IAAA,EAAM,MAAA,EAAQ,IAAA,EAAM,MAAM,CAAC,CAAA,EAAG,MAAA,EAAQ,KAAA,EAAO,CAAA;AAC3D,IAAA,MAAA,CAAO,IAAA,CAAK,EAAE,IAAA,EAAM,UAAA,EAAY,CAAA;AAChC,IAAA,KAAA,IAAS,IAAI,KAAA,CAAM,MAAA,GAAS,SAAA,EAAW,CAAA,IAAK,WAAW,CAAA,EAAA,EAAK;AAC1D,MAAA,MAAA,CAAO,IAAA,CAAK,EAAE,IAAA,EAAM,MAAA,EAAQ,IAAA,EAAM,KAAA,CAAM,CAAC,CAAA,EAAG,MAAA,EAAQ,CAAA,KAAM,SAAA,EAAW,CAAA;AAAA,IACvE;AAAA,EACF,CAAA,MAAO;AACL,IAAA,KAAA,CAAM,OAAA,CAAQ,CAAC,IAAA,EAAM,CAAA,KAAM,OAAO,IAAA,CAAK,EAAE,IAAA,EAAM,MAAA,EAAQ,IAAA,EAAM,MAAA,EAAQ,CAAA,KAAM,SAAA,EAAW,CAAC,CAAA;AAAA,EACzF;AAEA,EAAA,2BACG,KAAA,EAAA,EAAI,YAAA,EAAW,cAAa,SAAA,EAAW,CAAA,QAAA,EAAW,aAAa,EAAE,CAAA,CAAA,EAChE,8BAAC,IAAA,EAAA,EAAG,SAAA,EAAU,qCACX,QAAA,EAAA,MAAA,CAAO,GAAA,CAAI,CAAC,KAAA,EAAO,CAAA,0BACjB,QAAA,EAAA,EACE,QAAA,EAAA;AAAA,IAAA,CAAA,GAAI,qBAAK,GAAA,CAAC,IAAA,EAAA,EAAG,eAAW,IAAA,EAAC,SAAA,EAAU,qBAAqB,QAAA,EAAA,GAAA,EAAI,CAAA;AAAA,oBAC7D,GAAA,CAAC,QAAG,SAAA,EAAU,2BAAA,EACX,gBAAM,IAAA,KAAS,UAAA,uBACb,MAAA,EAAA,EAAK,SAAA,EAAU,oCAAmC,YAAA,EAAW,eAAA,EAAgB,oBAAC,CAAA,GAC7E,KAAA,CAAM,UAAU,CAAC,KAAA,CAAM,KAAK,OAAA,mBAC9B,IAAA;AAAA,MAAC,MAAA;AAAA,MAAA;AAAA,QACC,cAAA,EAAc,KAAA,CAAM,MAAA,GAAS,MAAA,GAAS,MAAA;AAAA,QACtC,SAAA,EAAW,CAAA,gDAAA,EACT,KAAA,CAAM,MAAA,GAAS,8BAA8B,eAC/C,CAAA,CAAA;AAAA,QAEC,QAAA,EAAA;AAAA,UAAA,KAAA,CAAM,IAAA,CAAK,IAAA;AAAA,8BACX,MAAA,EAAA,EAAK,SAAA,EAAU,UAAA,EAAY,QAAA,EAAA,KAAA,CAAM,KAAK,KAAA,EAAM;AAAA;AAAA;AAAA,KAC/C,mBAEA,IAAA;AAAA,MAAC,QAAA;AAAA,MAAA;AAAA,QACC,IAAA,EAAK,QAAA;AAAA,QACL,OAAA,EAAS,MAAM,IAAA,CAAK,OAAA;AAAA,QACpB,SAAA,EAAU,4MAAA;AAAA,QAET,QAAA,EAAA;AAAA,UAAA,KAAA,CAAM,IAAA,CAAK,IAAA;AAAA,8BACX,MAAA,EAAA,EAAK,SAAA,EAAU,UAAA,EAAY,QAAA,EAAA,KAAA,CAAM,KAAK,KAAA,EAAM;AAAA;AAAA;AAAA,KAC/C,EAEJ;AAAA,GAAA,EAAA,EAzBa,CA0Bf,CACD,CAAA,EACH,CAAA,EACF,CAAA;AAEJ;;;ACpEO,IAAM,oBAAA,GAAuB;AAQpC,SAAS,SAAA,GAAY;AACnB,EAAA,MAAM,QAAA,GAAY,OAAO,MAAA,KAAW,WAAA,IAAgB,MAAA,CAAe,8BAAA;AACnE,EAAA,OAAA,CAAQ,QAAA,IAAY,uBAAA,EAAyB,OAAA,CAAQ,KAAA,EAAO,EAAE,CAAA;AAChE;AAEA,eAAsB,gBAAgB,IAAA,EAA6C;AACjF,EAAA,MAAM,EAAE,QAAA,EAAU,QAAA,EAAU,IAAA,EAAM,UAAS,GAAI,IAAA;AAC/C,EAAA,IAAI;AACF,IAAA,MAAM,MAAM,MAAM,KAAA;AAAA,MAChB,GAAG,SAAA,EAAW,CAAA,eAAA,EAAkB,kBAAA,CAAmB,QAAQ,CAAC,CAAA,CAAA;AAAA,MAC5D,EAAE,aAAa,SAAA;AAAU,KAC3B;AACA,IAAA,IAAI,CAAC,IAAI,EAAA,EAAI;AACX,MAAA,aAAA,CAAM,KAAA,CAAM,CAAA,iBAAA,EAAoB,GAAA,CAAI,MAAM,CAAA,CAAA,CAAG,CAAA;AAC7C,MAAA,OAAO,KAAA;AAAA,IACT;AACA,IAAA,MAAM,IAAA,GAAO,MAAM,GAAA,CAAI,IAAA,EAAK;AAC5B,IAAA,IAAI,SAAS,KAAA,EAAO;AAClB,MAAA,MAAM,IAAA,GAAO,MAAM,IAAA,CAAK,IAAA,EAAK;AAC7B,MAAA,qBAAA,CAAsB,EAAE,GAAA,EAAK,IAAA,EAAM,QAAA,EAAU,CAAA;AAC7C,MAAA,QAAA,GAAW,cAAc,CAAA;AAAA,IAC3B,CAAA,MAAO;AACL,MAAA,MAAM,GAAA,GAAM,GAAA,CAAI,eAAA,CAAgB,IAAI,CAAA;AACpC,MAAA,aAAA,CAAc,EAAE,GAAA,EAAK,QAAA,EAAU,IAAA,EAAM,CAAA;AACrC,MAAA,QAAA,GAAW,UAAU,CAAA;AAAA,IACvB;AACA,IAAA,IAAI,OAAO,WAAW,WAAA,EAAa;AACjC,MAAA,MAAA,CAAO,aAAA,CAAc,IAAI,WAAA,CAAiC,oBAAA,EAAsB;AAAA,QAC9E,MAAA,EAAQ,EAAE,QAAA,EAAU,QAAA,EAAU,IAAA;AAAK,OACpC,CAAC,CAAA;AAAA,IACJ;AACA,IAAA,OAAO,IAAA;AAAA,EACT,SAAS,CAAA,EAAQ;AACf,IAAA,aAAA,CAAM,KAAA,CAAM,CAAA,EAAG,OAAA,IAAW,aAAa,CAAA;AACvC,IAAA,OAAO,KAAA;AAAA,EACT;AACF;ACxDO,IAAM,cAAA,GAAiB;AAAA,EAC5B,2BAAA;AAAA,EACA,+BAAA;AAAA,EACA,6BAAA;AAAA,EACA,2BAAA;AAAA,EACA,0BAAA;AAAA,EACA,4BAAA;AAAA,EACA,8BAAA;AAAA,EACA,8BAAA;AAAA,EACA,4BAAA;AAAA,EACA,+BAAA;AAAA,EACA,8BAAA;AAAA,EACA,2BAAA;AAAA,EACA,0BAAA;AAAA,EACA,0BAAA;AAAA,EACA;AACF,CAAA;AAEO,SAAS,aAAa,IAAA,EAAsB;AACjD,EAAA,IAAI,CAAA,GAAI,CAAA;AACR,EAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,IAAA,CAAK,MAAA,EAAQ,CAAA,EAAA,EAAK,CAAA,GAAA,CAAK,CAAA,IAAK,CAAA,IAAK,CAAA,GAAI,IAAA,CAAK,UAAA,CAAW,CAAC,CAAA;AAC1E,EAAA,OAAO,eAAe,IAAA,CAAK,GAAA,CAAI,CAAC,CAAA,GAAI,eAAe,MAAM,CAAA;AAC3D;AAGO,IAAM,kBAAA,GAA6C;AAAA,EACxD,KAAA,EAAO,eAAA;AAAA,EAAiB,cAAA,EAAgB,iBAAA;AAAA,EAAmB,OAAA,EAAS,gBAAA;AAAA,EACpE,MAAA,EAAQ,iBAAA;AAAA,EAAmB,YAAA,EAAc,iBAAA;AAAA,EAAmB,QAAA,EAAU,eAAA;AAAA,EACtE,WAAA,EAAa,eAAA;AAAA,EAAiB,OAAA,EAAS,eAAA;AAAA,EAAiB,KAAA,EAAO,cAAA;AAAA,EAC/D,MAAA,EAAQ,eAAA;AAAA,EAAiB,KAAA,EAAO,gBAAA;AAAA,EAAkB,WAAA,EAAa,kBAAA;AAAA,EAC/D,MAAA,EAAQ,iBAAA;AAAA,EAAmB,IAAA,EAAM;AACnC,CAAA;AACO,IAAM,YAAA,GAAuC;AAAA,EAClD,KAAA,EAAO,IAAA;AAAA,EAAM,cAAA,EAAgB,IAAA;AAAA,EAAM,OAAA,EAAS,KAAA;AAAA,EAAO,MAAA,EAAQ,KAAA;AAAA,EAC3D,YAAA,EAAc,KAAA;AAAA,EAAO,QAAA,EAAU,IAAA;AAAA,EAAM,WAAA,EAAa,IAAA;AAAA,EAAM,OAAA,EAAS,KAAA;AAAA,EACjE,KAAA,EAAO,KAAA;AAAA,EAAO,MAAA,EAAQ,KAAA;AAAA,EAAO,KAAA,EAAO,KAAA;AAAA,EAAO,WAAA,EAAa,IAAA;AAAA,EACxD,cAAA,EAAgB,IAAA;AAAA,EAAM,cAAA,EAAgB,IAAA;AAAA,EAAM,cAAA,EAAgB,IAAA;AAAA,EAC5D,SAAA,EAAW,IAAA;AAAA,EAAM,eAAA,EAAiB,KAAA;AAAA,EAAO,YAAA,EAAc,IAAA;AAAA,EACvD,YAAA,EAAc,IAAA;AAAA,EAAM,UAAA,EAAY,IAAA;AAAA,EAAM,mBAAA,EAAqB,IAAA;AAAA,EAC3D,kBAAA,EAAoB,KAAA;AAAA,EAAO,QAAA,EAAU,IAAA;AAAA,EAAM,MAAA,EAAQ;AACrD,CAAA;AAIO,IAAM,kBAAA,GAAsD;AAAA,EACjE,GAAA,EAAK,KAAA;AAAA,EAAO,GAAA,EAAK,KAAA;AAAA,EAAO,IAAA,EAAM,KAAA;AAAA,EAAO,KAAA,EAAO,KAAA;AAAA,EAAO,GAAA,EAAK;AAC1D,CAAA;AACO,IAAM,mBAAA,GAAuD;AAAA,EAClE,GAAA,EAAK,cAAA;AAAA,EAAgB,GAAA,EAAK,eAAA;AAAA,EAAiB,IAAA,EAAM,iBAAA;AAAA,EACjD,KAAA,EAAO,kBAAA;AAAA,EAAoB,GAAA,EAAK;AAClC,CAAA;AAuBO,SAAS,WAAA,CAAY,EAAE,SAAA,EAAU,EAA2B;AACjE,EAAA,uBACEA,GAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAsB,IAAA,EAAK,cAAA,EAAe,OAAA,EAAQ,WAAA,EACrD,QAAA,kBAAAA,GAAAA,CAAC,MAAA,EAAA,EAAK,CAAA,EAAE,qMAAoM,CAAA,EAC9M,CAAA;AAEJ;AAIA,SAAS,SAAA,CAAU,UAAmB,KAAA,EAA2B;AAC/D,EAAA,OAAQ,SAAS,QAAA,CAAS,KAAK,MAAO,QAAA,GAAW,QAAA,CAAS,QAAQ,CAAA,GAAI,MAAA,CAAA;AACxE;AAEA,IAAM,mBAAA,mBACJA,GAAAA,CAAC,KAAA,EAAA,EAAI,WAAU,SAAA,EAAU,IAAA,EAAK,MAAA,EAAO,OAAA,EAAQ,WAAA,EAAY,MAAA,EAAO,gBAAe,WAAA,EAAa,GAAA,EAC1F,QAAA,kBAAAA,GAAAA,CAAC,MAAA,EAAA,EAAK,aAAA,EAAc,SAAQ,cAAA,EAAe,OAAA,EAAQ,CAAA,EAAE,oHAAA,EAAqH,CAAA,EAC5K,CAAA;AAKK,SAAS,aAAa,EAAE,UAAA,EAAY,YAAY,QAAA,EAAU,KAAA,EAAO,UAAS,EAM9E;AACD,EAAA,MAAM,aAAA,GAAgB,eAAe,cAAA,IAAkB,QAAA;AACvD,EAAA,MAAM,YAAA,GAAe,aAAA,GAAgB,mBAAA,CAAoB,QAAS,CAAA,GAAI,IAAA;AACtE,EAAA,MAAM,WAAA,GAAc,aAAA,GAAgB,kBAAA,CAAmB,QAAS,CAAA,GAAI,IAAA;AACpE,EAAA,IAAI,eAAe,QAAA,EAAU;AAC3B,IAAA,uBACEA,GAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAW,8CAA8C,UAAA,GAAa,gDAAA,GAAmD,EAAE,CAAA,CAAA,EAC9H,QAAA,kBAAAA,GAAAA,CAAC,WAAA,EAAA,EAAY,SAAA,EAAU,oEAAmE,CAAA,EAC5F,CAAA;AAAA,EAEJ;AACA,EAAA,IAAI,eAAe,MAAA,EAAQ;AAIzB,IAAA,MAAM,IAAA,GAAO,SAAA,CAAU,QAAA,EAAU,KAAK,CAAA;AACtC,IAAA,MAAM,QAAA,GAAW,YAAA,CAAa,QAAA,IAAY,KAAA,IAAS,MAAM,CAAA;AACzD,IAAA,uBACEA,GAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAW,CAAA,2CAAA,EAA8C,UAAA,GAAa,gDAAA,GAAmD,EAAE,CAAA,CAAA,EAC9H,QAAA,kBAAAA,GAAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAW,CAAA,uCAAA,EAA0C,QAAQ,CAAA,2GAAA,CAAA,EAChE,QAAA,EAAA,IAAA,IAAQ,cAAA,CAAe,IAAI,CAAA,GACxB,YAAA,CAAa,IAAA,EAAsB,EAAE,SAAA,EAAW,oBAAA,EAAsB,CAAA,GACtE,mBAAA,EACN,CAAA,EACF,CAAA;AAAA,EAEJ;AACA,EAAA,uBACEC,KAAC,KAAA,EAAA,EAAI,SAAA,EAAW,uDAAuD,UAAA,GAAa,gDAAA,GAAmD,EAAE,CAAA,CAAA,EACvI,QAAA,EAAA;AAAA,oBAAAA,IAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAW,CAAA,kDAAA,EAAqD,YAAA,IAAgB,kBAAA,CAAmB,UAAU,CAAA,IAAK,eAAe,CAAA,CAAA,EAAI,OAAA,EAAQ,WAAA,EAAY,MAAK,MAAA,EACjK,QAAA,EAAA;AAAA,sBAAAD,IAAC,MAAA,EAAA,EAAK,CAAA,EAAE,+DAA8D,IAAA,EAAK,OAAA,EAAQ,aAAY,MAAA,EAAO,CAAA;AAAA,sBACtGA,IAAC,MAAA,EAAA,EAAK,CAAA,EAAE,kCAAiC,IAAA,EAAK,cAAA,EAAe,aAAY,KAAA,EAAM,CAAA;AAAA,sBAC/EA,GAAAA,CAAC,MAAA,EAAA,EAAK,CAAA,EAAE,6DAAA,EAA8D,QAAO,cAAA,EAAe,WAAA,EAAY,KAAA,EAAM,aAAA,EAAc,KAAA,EAAM;AAAA,KAAA,EACpI,CAAA;AAAA,oBACAA,IAAC,MAAA,EAAA,EAAK,SAAA,EAAW,+EAA+E,YAAA,IAAgB,kBAAA,CAAmB,UAAU,CAAA,IAAK,eAAe,IAC9J,QAAA,EAAA,WAAA,IAAe,YAAA,CAAa,UAAU,CAAA,IAAK,UAAA,CAAW,MAAM,CAAA,EAAG,CAAC,CAAA,CAAE,WAAA,EAAY,EACjF;AAAA,GAAA,EACF,CAAA;AAEJ;AAGO,SAAS,mBAAA,CAAoB,EAAE,IAAA,EAAK,EAA0B;AACnE,EAAA,IAAI,IAAA,CAAK,eAAe,QAAA,EAAU;AAChC,IAAA,uBAAOA,GAAAA,CAAC,WAAA,EAAA,EAAY,SAAA,EAAU,iCAAA,EAAkC,CAAA;AAAA,EAClE;AACA,EAAA,IAAI,IAAA,CAAK,eAAe,MAAA,EAAQ;AAC9B,IAAA,MAAM,IAAA,GAAO,SAAA,CAAU,IAAA,CAAK,QAAA,EAAU,KAAK,KAAK,CAAA;AAChD,IAAA,uBACEA,GAAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAW,CAAA,kCAAA,EAAqC,YAAA,CAAa,IAAA,CAAK,QAAA,IAAY,IAAA,CAAK,KAAA,IAAS,MAAM,CAAC,CAAA,qDAAA,CAAA,EACtG,QAAA,EAAA,IAAA,IAAQ,cAAA,CAAe,IAAI,CAAA,GACxB,YAAA,CAAa,IAAA,EAAsB,EAAE,SAAA,EAAW,oBAAA,EAAsB,CAAA,mBACtEA,GAAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,oDAAmD,CAAA,EACzE,CAAA;AAAA,EAEJ;AACA,EAAA,MAAM,KAAA,GAAA,CAAS,IAAA,CAAK,UAAA,KAAe,cAAA,IAAkB,IAAA,CAAK,QAAA,GACtD,mBAAA,CAAoB,IAAA,CAAK,QAAQ,CAAA,GACjC,kBAAA,CAAmB,IAAA,CAAK,UAAU,CAAA,KAAM,eAAA;AAC5C,EAAA,uBACEA,GAAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAW,CAAA,iBAAA,EAAoB,KAAK,CAAA,CAAA,EAAI,IAAA,EAAK,MAAA,EAAO,OAAA,EAAQ,WAAA,EAAY,MAAA,EAAO,gBAAe,WAAA,EAAa,GAAA,EAC9G,QAAA,kBAAAA,GAAAA,CAAC,MAAA,EAAA,EAAK,aAAA,EAAc,SAAQ,cAAA,EAAe,OAAA,EAAQ,CAAA,EAAE,8PAAA,EAA+P,CAAA,EACtT,CAAA;AAEJ;AAGO,SAAS,qBAAqB,IAAA,EAA2B;AAC9D,EAAA,IAAI,IAAA,CAAK,eAAe,cAAA,IAAkB,IAAA,CAAK,UAAU,OAAO,kBAAA,CAAmB,KAAK,QAAQ,CAAA;AAChG,EAAA,IAAI,IAAA,CAAK,UAAA,KAAe,MAAA,EAAQ,OAAO,KAAA;AACvC,EAAA,IAAI,IAAA,CAAK,UAAA,KAAe,QAAA,EAAU,OAAO,QAAA;AACzC,EAAA,OAAO,YAAA,CAAa,IAAA,CAAK,UAAU,CAAA,IAAK,IAAA,CAAK,WAAW,KAAA,CAAM,CAAA,EAAG,CAAC,CAAA,CAAE,WAAA,EAAY;AAClF;AAoBA,IAAI,sBAAA,GAAwD,IAAA;AAC5D,IAAM,uBAAA,uBAA8B,GAAA,EAAgB;AAE7C,SAAS,sBAAsB,IAAA,EAAqC;AACzE,EAAA,sBAAA,GAAyB,IAAA;AACzB,EAAA,uBAAA,CAAwB,OAAA,CAAQ,CAAA,CAAA,KAAK,CAAA,EAAG,CAAA;AAC1C;AAEO,SAAS,wBAAwB,QAAA,EAAkC;AACxE,EAAA,uBAAA,CAAwB,IAAI,QAAQ,CAAA;AACpC,EAAA,OAAO,MAAM;AAAE,IAAA,uBAAA,CAAwB,OAAO,QAAQ,CAAA;AAAA,EAAG,CAAA;AAC3D;AAEO,SAAS,yBAAA,GAA2D;AACzE,EAAA,OAAO,sBAAA;AACT;AAYO,IAAM,eAAA,GAAkB,+BAAA;AACxB,IAAM,sBAAA,GAAyB;AAC/B,IAAM,+BAAA,GAAkC;AAExC,SAAS,qBAAA,GAAwB;AACtC,EAAA,IAAI,OAAO,WAAW,WAAA,EAAa;AACnC,EAAC,MAAA,CAAe,eAAe,CAAA,GAAI,OAAA;AACnC,EAAA,MAAA,CAAO,aAAA,CAAc,IAAI,WAAA,CAAY,sBAAsB,CAAC,CAAA;AAC9D;AAEO,SAAS,8BAA8B,QAAA,EAAkB;AAC9D,EAAA,IAAI,OAAO,WAAW,WAAA,EAAa;AACnC,EAAC,OAAe,eAAe,CAAA,GAAI,EAAE,IAAA,EAAM,kBAAkB,QAAA,EAAS;AACtE,EAAA,MAAA,CAAO,aAAA,CAAc,IAAI,WAAA,CAAY,+BAAA,EAAiC,EAAE,QAAQ,EAAE,QAAA,EAAS,EAAG,CAAC,CAAA;AACjG;AAKO,SAAS,oBAAA,GAAgD;AAC9D,EAAA,IAAI,OAAO,MAAA,KAAW,WAAA,EAAa,OAAO,IAAA;AAC1C,EAAA,MAAM,GAAA,GAAO,OAAe,eAAe,CAAA;AAC3C,EAAA,IAAI,GAAA,KAAQ,SAAS,OAAO,OAAA;AAC5B,EAAA,IAAI,GAAA,IAAO,OAAO,GAAA,KAAQ,QAAA,IAAY,GAAA,CAAI,IAAA,KAAS,gBAAA,IAAoB,OAAO,GAAA,CAAI,QAAA,KAAa,QAAA,EAAU,OAAO,GAAA;AAChH,EAAA,OAAO,IAAA;AACT;AAGO,SAAS,uBAAA,GAAmD;AACjE,EAAA,MAAM,MAAM,oBAAA,EAAqB;AACjC,EAAA,IAAI,OAAO,MAAA,KAAW,WAAA,EAAc,MAAA,CAAe,eAAe,CAAA,GAAI,IAAA;AACtE,EAAA,OAAO,GAAA;AACT","file":"chunk-ZEMXT6BR.js","sourcesContent":["import { Fragment } from 'react';\nimport type { ReactNode } from 'react';\n\n/**\n * Generic breadcrumb trail. Self-contained and styled with the same Tailwind\n * utilities the shell already ships, so consumers get it for free.\n *\n * Pass an ordered list of `items` from root → current. Every crumb except the\n * last renders as a button when it has an `onClick`; the last crumb is treated\n * as the current location — rendered inert with `aria-current=\"page\"`. When the\n * trail is long, set `maxItems` to collapse the middle into an ellipsis\n * (`first … last-n` ), keeping the first and the tail visible.\n */\nexport interface BreadcrumbItem {\n /** Visible label. */\n label: ReactNode;\n /** Optional leading icon (typically a 3.5×3.5 svg). */\n icon?: ReactNode;\n /** Navigate to this crumb. Omitted on the current (last) crumb. */\n onClick?: () => void;\n}\n\nexport interface BreadcrumbsProps {\n items: BreadcrumbItem[];\n /** Node rendered between crumbs. Defaults to a chevron. */\n separator?: ReactNode;\n /** Collapse the middle to an ellipsis when there are more than this many\n * crumbs. `0` (default) never collapses. */\n maxItems?: number;\n className?: string;\n}\n\nconst DEFAULT_SEPARATOR = (\n <svg className=\"h-3.5 w-3.5 shrink-0 text-gray-300\" viewBox=\"0 0 20 20\" fill=\"currentColor\" aria-hidden>\n <path\n fillRule=\"evenodd\"\n d=\"M7.21 14.77a.75.75 0 0 1 .02-1.06L11.168 10 7.23 6.29a.75.75 0 1 1 1.04-1.08l4.5 4.25a.75.75 0 0 1 0 1.08l-4.5 4.25a.75.75 0 0 1-1.06-.02Z\"\n clipRule=\"evenodd\"\n />\n </svg>\n);\n\ntype Crumb = { kind: 'item'; item: BreadcrumbItem; isLast: boolean } | { kind: 'ellipsis' };\n\nexport default function Breadcrumbs({ items, separator, maxItems = 0, className }: BreadcrumbsProps) {\n if (items.length === 0) return null;\n const sep = separator ?? DEFAULT_SEPARATOR;\n\n // Build the display list, collapsing the middle if `maxItems` is exceeded.\n const crumbs: Crumb[] = [];\n const lastIndex = items.length - 1;\n if (maxItems > 0 && items.length > maxItems && maxItems >= 2) {\n const tailCount = Math.max(1, maxItems - 1); // keep the first + this many from the end\n crumbs.push({ kind: 'item', item: items[0], isLast: false });\n crumbs.push({ kind: 'ellipsis' });\n for (let i = items.length - tailCount; i <= lastIndex; i++) {\n crumbs.push({ kind: 'item', item: items[i], isLast: i === lastIndex });\n }\n } else {\n items.forEach((item, i) => crumbs.push({ kind: 'item', item, isLast: i === lastIndex }));\n }\n\n return (\n <nav aria-label=\"Breadcrumb\" className={`min-w-0 ${className ?? ''}`}>\n <ol className=\"flex items-center gap-1.5 text-sm\">\n {crumbs.map((crumb, i) => (\n <Fragment key={i}>\n {i > 0 && <li aria-hidden className=\"flex items-center\">{sep}</li>}\n <li className=\"flex min-w-0 items-center\">\n {crumb.kind === 'ellipsis' ? (\n <span className=\"px-0.5 text-gray-400 select-none\" aria-label=\"Hidden crumbs\">…</span>\n ) : crumb.isLast || !crumb.item.onClick ? (\n <span\n aria-current={crumb.isLast ? 'page' : undefined}\n className={`inline-flex min-w-0 items-center gap-1 truncate ${\n crumb.isLast ? 'font-medium text-gray-900' : 'text-gray-500'\n }`}\n >\n {crumb.item.icon}\n <span className=\"truncate\">{crumb.item.label}</span>\n </span>\n ) : (\n <button\n type=\"button\"\n onClick={crumb.item.onClick}\n className=\"inline-flex min-w-0 items-center gap-1 truncate rounded px-1 -mx-1 text-gray-500 transition-colors hover:text-gray-900 hover:underline focus:outline-none focus-visible:ring-2 focus-visible:ring-blue-400\"\n >\n {crumb.item.icon}\n <span className=\"truncate\">{crumb.item.label}</span>\n </button>\n )}\n </li>\n </Fragment>\n ))}\n </ol>\n </nav>\n );\n}\n","/**\n * Fetch a file from the user's file-server and route it into the right\n * preview window. Shared by the Files app and the desktop Documents\n * folder's shortcut click-through.\n *\n * On success, dispatches `react-os-shell:preview-opened` so Desktop can\n * record the file as a shortcut without this util needing access to the\n * prefs adapter.\n */\nimport toast from '../shell/toast';\nimport { setPdfPreview } from '../apps/Preview';\nimport { setSpreadsheetPreview } from '../apps/Spreadsheet';\n\nexport type PreviewFileKind = 'pdf' | 'dxf' | '3d' | 'image' | 'csv';\n\nexport interface OpenPreviewFileOpts {\n /** Server-relative path, e.g. \"/reports/Q1.pdf\". */\n filePath: string;\n /** Display name (also used as the download filename). */\n filename: string;\n /** Which viewer to route into. CSV opens in Spreadsheet; the rest open in Preview. */\n kind: PreviewFileKind;\n /** Optional callback invoked after staging the preview, with the route to\n * open (e.g. '/preview' or '/spreadsheet'). The caller is responsible\n * for actually opening the page since the window manager hook is\n * React-scoped. */\n onStaged?: (route: '/preview' | '/spreadsheet') => void;\n}\n\nexport const PREVIEW_OPENED_EVENT = 'react-os-shell:preview-opened';\n\nexport interface PreviewOpenedDetail {\n filePath: string;\n filename: string;\n kind: PreviewFileKind;\n}\n\nfunction getServer() {\n const override = (typeof window !== 'undefined' && (window as any).__REACT_OS_SHELL_FILE_SERVER__) as string | undefined;\n return (override || 'http://localhost:4000').replace(/\\/$/, '');\n}\n\nexport async function openPreviewFile(opts: OpenPreviewFileOpts): Promise<boolean> {\n const { filePath, filename, kind, onStaged } = opts;\n try {\n const res = await fetch(\n `${getServer()}/api/file?path=${encodeURIComponent(filePath)}`,\n { credentials: 'include' },\n );\n if (!res.ok) {\n toast.error(`Download failed (${res.status})`);\n return false;\n }\n const blob = await res.blob();\n if (kind === 'csv') {\n const text = await blob.text();\n setSpreadsheetPreview({ csv: text, filename });\n onStaged?.('/spreadsheet');\n } else {\n const url = URL.createObjectURL(blob);\n setPdfPreview({ url, filename, kind });\n onStaged?.('/preview');\n }\n if (typeof window !== 'undefined') {\n window.dispatchEvent(new CustomEvent<PreviewOpenedDetail>(PREVIEW_OPENED_EVENT, {\n detail: { filePath, filename, kind },\n }));\n }\n return true;\n } catch (e: any) {\n toast.error(e?.message || 'Open failed');\n return false;\n }\n}\n","/**\n * Shared desktop-icon building blocks.\n *\n * Desktop.tsx renders the desktop surface; Files.tsx renders desktop shortcut\n * folders inside the Files app; MobileHome.tsx renders the mobile home grid.\n * All three import the icon tiles, entity-icon config and the desktop-folder\n * bridge from here so the surfaces never visually diverge — and so Desktop\n * and Files never import each other (that would be an import cycle).\n */\nimport { isValidElement, cloneElement, type ReactElement, type ReactNode } from 'react';\nimport { navIcons } from '../shell-config/nav';\nimport type { PreviewFileKind } from '../utils/openPreviewFile';\n\n// ── Gradient tiles ──────────────────────────────────────────────────────────\n// Per-app colored tile background, shared by the mobile home grid and the\n// desktop \"page\" shortcuts. Tailwind's JIT scans the source so each gradient\n// class string must appear in full somewhere — keep them inline.\nexport const ICON_GRADIENTS = [\n 'from-blue-500 to-blue-700',\n 'from-indigo-500 to-purple-600',\n 'from-purple-500 to-pink-600',\n 'from-pink-500 to-rose-600',\n 'from-red-500 to-rose-600',\n 'from-orange-500 to-red-600',\n 'from-amber-500 to-orange-600',\n 'from-yellow-500 to-amber-500',\n 'from-lime-500 to-green-600',\n 'from-green-500 to-emerald-600',\n 'from-emerald-500 to-teal-600',\n 'from-teal-500 to-cyan-600',\n 'from-cyan-500 to-sky-600',\n 'from-sky-500 to-blue-600',\n 'from-violet-500 to-fuchsia-600',\n];\n\nexport function hashGradient(seed: string): string {\n let h = 0;\n for (let i = 0; i < seed.length; i++) h = (h << 5) - h + seed.charCodeAt(i);\n return ICON_GRADIENTS[Math.abs(h) % ICON_GRADIENTS.length];\n}\n\n// ── Entity icon config ──────────────────────────────────────────────────────\nexport const ENTITY_ICON_COLORS: Record<string, string> = {\n order: 'text-blue-600', purchase_order: 'text-purple-600', invoice: 'text-green-600',\n client: 'text-indigo-600', manufacturer: 'text-orange-600', shipment: 'text-teal-600',\n part_number: 'text-gray-600', project: 'text-pink-600', mould: 'text-red-600',\n design: 'text-cyan-600', brand: 'text-amber-600', price_sheet: 'text-emerald-600',\n folder: 'text-yellow-600', page: 'text-blue-500',\n};\nexport const ENTITY_ICONS: Record<string, string> = {\n order: 'SO', purchase_order: 'PO', invoice: 'INV', client: 'CLI',\n manufacturer: 'MFR', shipment: 'DN', part_number: 'PN', project: 'PRJ',\n mould: 'MLD', design: 'DSN', brand: 'BRD', price_sheet: 'PS',\n vendor_invoice: 'VI', vendor_payment: 'VP', warranty_claim: 'WC',\n qc_report: 'QC', vendor_shipment: 'GRN', bank_account: 'BA',\n wheel_finish: 'WF', weight_log: 'WL', production_progress: 'PP',\n vendor_price_sheet: 'VPS', proposal: 'PR', folder: 'FLD',\n};\n\n// Glyphs and colors for the auto-tracked preview shortcuts that live in the\n// Documents folder. Keyed by `fileKind`, not `entityType`.\nexport const PREVIEW_FILE_CODES: Record<PreviewFileKind, string> = {\n pdf: 'PDF', dxf: 'DXF', '3d': 'STP', image: 'IMG', csv: 'CSV',\n};\nexport const PREVIEW_FILE_COLORS: Record<PreviewFileKind, string> = {\n pdf: 'text-red-600', dxf: 'text-blue-600', '3d': 'text-purple-600',\n image: 'text-emerald-600', csv: 'text-green-600',\n};\n\nexport interface DesktopItem {\n entityType: string;\n entityId: string;\n label: string;\n x?: number;\n y?: number;\n folderId?: string; // if inside a folder\n // Legacy free-form position inside the old folder window (pixels, relative\n // to the folder body). Folders now open in the Files app, which lists items\n // in a table — kept so previously-persisted shortcuts keep parsing.\n folderX?: number;\n folderY?: number;\n // Set for `entityType: 'preview-file'` shortcuts auto-recorded when the\n // user previews a file. `filePath` is the server-relative path passed\n // back into `openPreviewFile` on shortcut click.\n filePath?: string;\n fileKind?: PreviewFileKind;\n}\n\n/** Solid amber folder — the exact glyph the Files app uses for folders, so\n * desktop folders and Files-app folders read as the same object. */\nexport function FolderGlyph({ className }: { className?: string }) {\n return (\n <svg className={className} fill=\"currentColor\" viewBox=\"0 0 24 24\">\n <path d=\"M2.25 7.125A2.25 2.25 0 014.5 4.875h4.504c.61 0 1.193.243 1.624.673l1.494 1.494a.75.75 0 00.53.22h7.098A2.25 2.25 0 0122 9.51v8.366A2.25 2.25 0 0119.75 20.125H4.25A2.25 2.25 0 012 17.875V7.125z\" />\n </svg>\n );\n}\n\n/** Page shortcuts reuse the consumer-registered nav icons; historically the\n * label was tried first, then the route. */\nfunction pageGlyph(entityId?: string, label?: string): ReactNode {\n return (label && navIcons[label]) || (entityId ? navIcons[entityId] : undefined);\n}\n\nconst PAGE_FALLBACK_GLYPH = (\n <svg className=\"h-7 w-7\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M3.75 6A2.25 2.25 0 016 3.75h2.25A2.25 2.25 0 0110.5 6v2.25a2.25 2.25 0 01-2.25 2.25H6a2.25 2.25 0 01-2.25-2.25V6z\" />\n </svg>\n);\n\n// Shared 48×48 tile used by the desktop icon renderer (and anything else\n// that needs a full-size shortcut tile), so all surfaces stay in sync.\nexport function FileIconTile({ entityType, isSelected, entityId, label, fileKind }: {\n entityType: string;\n isSelected: boolean;\n entityId?: string;\n label?: string;\n fileKind?: PreviewFileKind;\n}) {\n const isPreviewFile = entityType === 'preview-file' && fileKind;\n const previewColor = isPreviewFile ? PREVIEW_FILE_COLORS[fileKind!] : null;\n const previewCode = isPreviewFile ? PREVIEW_FILE_CODES[fileKind!] : null;\n if (entityType === 'folder') {\n return (\n <div className={`w-12 h-12 flex items-center justify-center ${isSelected ? 'rounded-lg bg-blue-400/30 ring-2 ring-blue-500' : ''}`}>\n <FolderGlyph className=\"h-12 w-12 text-amber-500 drop-shadow-[0_2px_3px_rgba(0,0,0,0.3)]\" />\n </div>\n );\n }\n if (entityType === 'page') {\n // App shortcut — colored gradient tile with the white nav glyph, same\n // treatment as the mobile home grid (and the same hash seed, the route,\n // so an app keeps its color across surfaces).\n const icon = pageGlyph(entityId, label);\n const gradient = hashGradient(entityId || label || 'page');\n return (\n <div className={`w-12 h-12 flex items-center justify-center ${isSelected ? 'rounded-lg bg-blue-400/30 ring-2 ring-blue-500' : ''}`}>\n <span className={`h-11 w-11 rounded-xl bg-gradient-to-br ${gradient} flex items-center justify-center text-white border border-white/30 drop-shadow-[0_2px_3px_rgba(0,0,0,0.3)]`}>\n {icon && isValidElement(icon)\n ? cloneElement(icon as ReactElement, { className: 'h-7 w-7 text-white' })\n : PAGE_FALLBACK_GLYPH}\n </span>\n </div>\n );\n }\n return (\n <div className={`w-12 h-12 relative flex items-center justify-center ${isSelected ? 'rounded-lg bg-blue-400/30 ring-2 ring-blue-500' : ''}`}>\n <svg className={`w-10 h-12 drop-shadow-[0_2px_3px_rgba(0,0,0,0.3)] ${previewColor ?? ENTITY_ICON_COLORS[entityType] ?? 'text-gray-500'}`} viewBox=\"0 0 40 48\" fill=\"none\">\n <path d=\"M4 0h22l10 10v34a4 4 0 01-4 4H4a4 4 0 01-4-4V4a4 4 0 014-4z\" fill=\"white\" fillOpacity=\"0.92\" />\n <path d=\"M26 0l10 10H30a4 4 0 01-4-4V0z\" fill=\"currentColor\" fillOpacity=\"0.2\" />\n <path d=\"M4 0h22l10 10v34a4 4 0 01-4 4H4a4 4 0 01-4-4V4a4 4 0 014-4z\" stroke=\"currentColor\" strokeWidth=\"1.5\" strokeOpacity=\"0.5\" />\n </svg>\n <span className={`absolute inset-0 flex items-center justify-center text-[9px] font-bold pt-2 ${previewColor ?? ENTITY_ICON_COLORS[entityType] ?? 'text-gray-600'}`}>\n {previewCode ?? ENTITY_ICONS[entityType] ?? entityType.slice(0, 3).toUpperCase()}\n </span>\n </div>\n );\n}\n\n/** Compact row icon for desktop shortcuts listed inside the Files app. */\nexport function DesktopItemMiniIcon({ item }: { item: DesktopItem }) {\n if (item.entityType === 'folder') {\n return <FolderGlyph className=\"h-4 w-4 shrink-0 text-amber-500\" />;\n }\n if (item.entityType === 'page') {\n const icon = pageGlyph(item.entityId, item.label);\n return (\n <span className={`h-4 w-4 rounded bg-gradient-to-br ${hashGradient(item.entityId || item.label || 'page')} flex items-center justify-center text-white shrink-0`}>\n {icon && isValidElement(icon)\n ? cloneElement(icon as ReactElement, { className: 'h-3 w-3 text-white' })\n : <span className=\"h-1.5 w-1.5 rounded-[2px] border border-white/80\" />}\n </span>\n );\n }\n const color = (item.entityType === 'preview-file' && item.fileKind\n ? PREVIEW_FILE_COLORS[item.fileKind]\n : ENTITY_ICON_COLORS[item.entityType]) ?? 'text-gray-400';\n return (\n <svg className={`h-4 w-4 shrink-0 ${color}`} fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M19.5 14.25v-2.625a3.375 3.375 0 00-3.375-3.375h-1.5A1.125 1.125 0 0113.5 7.125v-1.5a3.375 3.375 0 00-3.375-3.375H8.25m2.25 0H5.625c-.621 0-1.125.504-1.125 1.125v17.25c0 .621.504 1.125 1.125 1.125h12.75c.621 0 1.125-.504 1.125-1.125V11.25a9 9 0 00-9-9z\" />\n </svg>\n );\n}\n\n/** Short type tag shown in the Files-app listing of a desktop folder. */\nexport function desktopItemTypeLabel(item: DesktopItem): string {\n if (item.entityType === 'preview-file' && item.fileKind) return PREVIEW_FILE_CODES[item.fileKind];\n if (item.entityType === 'page') return 'App';\n if (item.entityType === 'folder') return 'Folder';\n return ENTITY_ICONS[item.entityType] ?? item.entityType.slice(0, 3).toUpperCase();\n}\n\n// ── Desktop-folder bridge ───────────────────────────────────────────────────\n//\n// Desktop folders are virtual — shortcut collections persisted in the user's\n// prefs, not directories on the file server. The Files app shows them anyway\n// (sidebar section + folder listing), so Desktop publishes its live folder\n// state here and Files subscribes via useSyncExternalStore. Mutations route\n// back through the callbacks so persistence stays in Desktop's host-aware\n// save path.\n\nexport interface DesktopFoldersSnapshot {\n folders: { id: string; name: string; itemCount: number }[];\n itemsByFolder: Record<string, DesktopItem[]>;\n /** Clear `folderId` on these shortcuts — they reappear on the desktop. */\n moveToDesktop: (items: DesktopItem[]) => void;\n /** Delete these shortcuts entirely. */\n removeShortcuts: (items: DesktopItem[]) => void;\n}\n\nlet desktopFoldersSnapshot: DesktopFoldersSnapshot | null = null;\nconst desktopFoldersListeners = new Set<() => void>();\n\nexport function publishDesktopFolders(next: DesktopFoldersSnapshot | null) {\n desktopFoldersSnapshot = next;\n desktopFoldersListeners.forEach(l => l());\n}\n\nexport function subscribeDesktopFolders(listener: () => void): () => void {\n desktopFoldersListeners.add(listener);\n return () => { desktopFoldersListeners.delete(listener); };\n}\n\nexport function getDesktopFoldersSnapshot(): DesktopFoldersSnapshot | null {\n return desktopFoldersSnapshot;\n}\n\n// ── Files-app view side channel ─────────────────────────────────────────────\n//\n// Used by the desktop Trash icon and desktop folders to open the Files app\n// on a specific view. Two cases handled:\n// 1) Files isn't open yet — the flag below is read once on first mount.\n// 2) Files is already open — the event tells the live instance to flip view.\n// The caller follows up with `openPage('/files')` either way.\n\nexport type FilesViewRequest = 'trash' | { view: 'desktop-folder'; folderId: string };\n\nexport const FILES_VIEW_FLAG = '__REACT_OS_SHELL_FILES_VIEW__';\nexport const FILES_SHOW_TRASH_EVENT = 'react-os-shell:files-show-trash';\nexport const FILES_OPEN_DESKTOP_FOLDER_EVENT = 'react-os-shell:files-open-desktop-folder';\n\nexport function requestFilesTrashView() {\n if (typeof window === 'undefined') return;\n (window as any)[FILES_VIEW_FLAG] = 'trash' satisfies FilesViewRequest;\n window.dispatchEvent(new CustomEvent(FILES_SHOW_TRASH_EVENT));\n}\n\nexport function requestFilesDesktopFolderView(folderId: string) {\n if (typeof window === 'undefined') return;\n (window as any)[FILES_VIEW_FLAG] = { view: 'desktop-folder', folderId } satisfies FilesViewRequest;\n window.dispatchEvent(new CustomEvent(FILES_OPEN_DESKTOP_FOLDER_EVENT, { detail: { folderId } }));\n}\n\n/** Read the pending view request without clearing it. Side-effect free so\n * it is safe inside a `useState` initializer (StrictMode double-invokes\n * those; a read-and-clear there would hand the second invocation null). */\nexport function peekFilesViewRequest(): FilesViewRequest | null {\n if (typeof window === 'undefined') return null;\n const raw = (window as any)[FILES_VIEW_FLAG];\n if (raw === 'trash') return 'trash';\n if (raw && typeof raw === 'object' && raw.view === 'desktop-folder' && typeof raw.folderId === 'string') return raw;\n return null;\n}\n\n/** Read-and-clear the pending view request (Files calls this once mounted). */\nexport function consumeFilesViewRequest(): FilesViewRequest | null {\n const req = peekFilesViewRequest();\n if (typeof window !== 'undefined') (window as any)[FILES_VIEW_FLAG] = null;\n return req;\n}\n"]}
1
+ {"version":3,"sources":["../src/shell/Breadcrumbs.tsx","../src/utils/openPreviewFile.ts","../src/shell/desktopIcons.tsx"],"names":["jsx","jsxs"],"mappings":"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","file":"chunk-75YZMM5Q.js","sourcesContent":["import { Fragment } from 'react';\nimport type { ReactNode } from 'react';\n\n/**\n * Generic breadcrumb trail. Self-contained and styled with the same Tailwind\n * utilities the shell already ships, so consumers get it for free.\n *\n * Pass an ordered list of `items` from root → current. Every crumb except the\n * last renders as a button when it has an `onClick`; the last crumb is treated\n * as the current location — rendered inert with `aria-current=\"page\"`. When the\n * trail is long, set `maxItems` to collapse the middle into an ellipsis\n * (`first … last-n` ), keeping the first and the tail visible.\n */\nexport interface BreadcrumbItem {\n /** Visible label. */\n label: ReactNode;\n /** Optional leading icon (typically a 3.5×3.5 svg). */\n icon?: ReactNode;\n /** Navigate to this crumb. Omitted on the current (last) crumb. */\n onClick?: () => void;\n}\n\nexport interface BreadcrumbsProps {\n items: BreadcrumbItem[];\n /** Node rendered between crumbs. Defaults to a chevron. */\n separator?: ReactNode;\n /** Collapse the middle to an ellipsis when there are more than this many\n * crumbs. `0` (default) never collapses. */\n maxItems?: number;\n className?: string;\n}\n\nconst DEFAULT_SEPARATOR = (\n <svg className=\"h-3.5 w-3.5 shrink-0 text-gray-300\" viewBox=\"0 0 20 20\" fill=\"currentColor\" aria-hidden>\n <path\n fillRule=\"evenodd\"\n d=\"M7.21 14.77a.75.75 0 0 1 .02-1.06L11.168 10 7.23 6.29a.75.75 0 1 1 1.04-1.08l4.5 4.25a.75.75 0 0 1 0 1.08l-4.5 4.25a.75.75 0 0 1-1.06-.02Z\"\n clipRule=\"evenodd\"\n />\n </svg>\n);\n\ntype Crumb = { kind: 'item'; item: BreadcrumbItem; isLast: boolean } | { kind: 'ellipsis' };\n\nexport default function Breadcrumbs({ items, separator, maxItems = 0, className }: BreadcrumbsProps) {\n if (items.length === 0) return null;\n const sep = separator ?? DEFAULT_SEPARATOR;\n\n // Build the display list, collapsing the middle if `maxItems` is exceeded.\n const crumbs: Crumb[] = [];\n const lastIndex = items.length - 1;\n if (maxItems > 0 && items.length > maxItems && maxItems >= 2) {\n const tailCount = Math.max(1, maxItems - 1); // keep the first + this many from the end\n crumbs.push({ kind: 'item', item: items[0], isLast: false });\n crumbs.push({ kind: 'ellipsis' });\n for (let i = items.length - tailCount; i <= lastIndex; i++) {\n crumbs.push({ kind: 'item', item: items[i], isLast: i === lastIndex });\n }\n } else {\n items.forEach((item, i) => crumbs.push({ kind: 'item', item, isLast: i === lastIndex }));\n }\n\n return (\n <nav aria-label=\"Breadcrumb\" className={`min-w-0 ${className ?? ''}`}>\n <ol className=\"flex items-center gap-1.5 text-sm\">\n {crumbs.map((crumb, i) => (\n <Fragment key={i}>\n {i > 0 && <li aria-hidden className=\"flex items-center\">{sep}</li>}\n <li className=\"flex min-w-0 items-center\">\n {crumb.kind === 'ellipsis' ? (\n <span className=\"px-0.5 text-gray-400 select-none\" aria-label=\"Hidden crumbs\">…</span>\n ) : crumb.isLast || !crumb.item.onClick ? (\n <span\n aria-current={crumb.isLast ? 'page' : undefined}\n className={`inline-flex min-w-0 items-center gap-1 truncate ${\n crumb.isLast ? 'font-medium text-gray-900' : 'text-gray-500'\n }`}\n >\n {crumb.item.icon}\n <span className=\"truncate\">{crumb.item.label}</span>\n </span>\n ) : (\n <button\n type=\"button\"\n onClick={crumb.item.onClick}\n className=\"inline-flex min-w-0 items-center gap-1 truncate rounded px-1 -mx-1 text-gray-500 transition-colors hover:text-gray-900 hover:underline focus:outline-none focus-visible:ring-2 focus-visible:ring-blue-400\"\n >\n {crumb.item.icon}\n <span className=\"truncate\">{crumb.item.label}</span>\n </button>\n )}\n </li>\n </Fragment>\n ))}\n </ol>\n </nav>\n );\n}\n","/**\n * Fetch a file from the user's file-server and route it into the right\n * preview window. Shared by the Files app and the desktop Documents\n * folder's shortcut click-through.\n *\n * On success, dispatches `react-os-shell:preview-opened` so Desktop can\n * record the file as a shortcut without this util needing access to the\n * prefs adapter.\n */\nimport toast from '../shell/toast';\nimport { setPdfPreview } from '../apps/Preview';\nimport { setSpreadsheetPreview } from '../apps/Spreadsheet';\n\nexport type PreviewFileKind = 'pdf' | 'dxf' | '3d' | 'image' | 'csv';\n\nexport interface OpenPreviewFileOpts {\n /** Server-relative path, e.g. \"/reports/Q1.pdf\". */\n filePath: string;\n /** Display name (also used as the download filename). */\n filename: string;\n /** Which viewer to route into. CSV opens in Spreadsheet; the rest open in Preview. */\n kind: PreviewFileKind;\n /** Optional callback invoked after staging the preview, with the route to\n * open (e.g. '/preview' or '/spreadsheet'). The caller is responsible\n * for actually opening the page since the window manager hook is\n * React-scoped. */\n onStaged?: (route: '/preview' | '/spreadsheet') => void;\n}\n\nexport const PREVIEW_OPENED_EVENT = 'react-os-shell:preview-opened';\n\nexport interface PreviewOpenedDetail {\n filePath: string;\n filename: string;\n kind: PreviewFileKind;\n}\n\nfunction getServer() {\n const override = (typeof window !== 'undefined' && (window as any).__REACT_OS_SHELL_FILE_SERVER__) as string | undefined;\n return (override || 'http://localhost:4000').replace(/\\/$/, '');\n}\n\nexport async function openPreviewFile(opts: OpenPreviewFileOpts): Promise<boolean> {\n const { filePath, filename, kind, onStaged } = opts;\n try {\n const res = await fetch(\n `${getServer()}/api/file?path=${encodeURIComponent(filePath)}`,\n { credentials: 'include' },\n );\n if (!res.ok) {\n toast.error(`Download failed (${res.status})`);\n return false;\n }\n const blob = await res.blob();\n if (kind === 'csv') {\n const text = await blob.text();\n setSpreadsheetPreview({ csv: text, filename });\n onStaged?.('/spreadsheet');\n } else {\n const url = URL.createObjectURL(blob);\n setPdfPreview({ url, filename, kind });\n onStaged?.('/preview');\n }\n if (typeof window !== 'undefined') {\n window.dispatchEvent(new CustomEvent<PreviewOpenedDetail>(PREVIEW_OPENED_EVENT, {\n detail: { filePath, filename, kind },\n }));\n }\n return true;\n } catch (e: any) {\n toast.error(e?.message || 'Open failed');\n return false;\n }\n}\n","/**\n * Shared desktop-icon building blocks.\n *\n * Desktop.tsx renders the desktop surface; Files.tsx renders desktop shortcut\n * folders inside the Files app; MobileHome.tsx renders the mobile home grid.\n * All three import the icon tiles, entity-icon config and the desktop-folder\n * bridge from here so the surfaces never visually diverge — and so Desktop\n * and Files never import each other (that would be an import cycle).\n */\nimport { isValidElement, cloneElement, type ReactElement, type ReactNode } from 'react';\nimport { navIcons } from '../shell-config/nav';\nimport type { PreviewFileKind } from '../utils/openPreviewFile';\n\n// ── Gradient tiles ──────────────────────────────────────────────────────────\n// Per-app colored tile background, shared by the mobile home grid and the\n// desktop \"page\" shortcuts. Tailwind's JIT scans the source so each gradient\n// class string must appear in full somewhere — keep them inline.\nexport const ICON_GRADIENTS = [\n 'from-blue-500 to-blue-700',\n 'from-indigo-500 to-purple-600',\n 'from-purple-500 to-pink-600',\n 'from-pink-500 to-rose-600',\n 'from-red-500 to-rose-600',\n 'from-orange-500 to-red-600',\n 'from-amber-500 to-orange-600',\n 'from-yellow-500 to-amber-500',\n 'from-lime-500 to-green-600',\n 'from-green-500 to-emerald-600',\n 'from-emerald-500 to-teal-600',\n 'from-teal-500 to-cyan-600',\n 'from-cyan-500 to-sky-600',\n 'from-sky-500 to-blue-600',\n 'from-violet-500 to-fuchsia-600',\n];\n\nexport function hashGradient(seed: string): string {\n let h = 0;\n for (let i = 0; i < seed.length; i++) h = (h << 5) - h + seed.charCodeAt(i);\n return ICON_GRADIENTS[Math.abs(h) % ICON_GRADIENTS.length];\n}\n\n// ── Entity icon config ──────────────────────────────────────────────────────\nexport const ENTITY_ICON_COLORS: Record<string, string> = {\n order: 'text-blue-600', purchase_order: 'text-purple-600', invoice: 'text-green-600',\n client: 'text-indigo-600', manufacturer: 'text-orange-600', shipment: 'text-teal-600',\n part_number: 'text-gray-600', project: 'text-pink-600', mould: 'text-red-600',\n design: 'text-cyan-600', brand: 'text-amber-600', price_sheet: 'text-emerald-600',\n folder: 'text-yellow-600', page: 'text-blue-500',\n};\nexport const ENTITY_ICONS: Record<string, string> = {\n order: 'SO', purchase_order: 'PO', invoice: 'INV', client: 'CLI',\n manufacturer: 'MFR', shipment: 'DN', part_number: 'PN', project: 'PRJ',\n mould: 'MLD', design: 'DSN', brand: 'BRD', price_sheet: 'PS',\n vendor_invoice: 'VI', vendor_payment: 'VP', warranty_claim: 'WC',\n qc_report: 'QC', vendor_shipment: 'GRN', bank_account: 'BA',\n wheel_finish: 'WF', weight_log: 'WL', production_progress: 'PP',\n vendor_price_sheet: 'VPS', proposal: 'PR', folder: 'FLD',\n};\n\n// Glyphs and colors for the auto-tracked preview shortcuts that live in the\n// Documents folder. Keyed by `fileKind`, not `entityType`.\nexport const PREVIEW_FILE_CODES: Record<PreviewFileKind, string> = {\n pdf: 'PDF', dxf: 'DXF', '3d': 'STP', image: 'IMG', csv: 'CSV',\n};\nexport const PREVIEW_FILE_COLORS: Record<PreviewFileKind, string> = {\n pdf: 'text-red-600', dxf: 'text-blue-600', '3d': 'text-purple-600',\n image: 'text-emerald-600', csv: 'text-green-600',\n};\n\nexport interface DesktopItem {\n entityType: string;\n entityId: string;\n label: string;\n x?: number;\n y?: number;\n folderId?: string; // if inside a folder\n // Legacy free-form position inside the old folder window (pixels, relative\n // to the folder body). Folders now open in the Files app, which lists items\n // in a table — kept so previously-persisted shortcuts keep parsing.\n folderX?: number;\n folderY?: number;\n // Set for `entityType: 'preview-file'` shortcuts auto-recorded when the\n // user previews a file. `filePath` is the server-relative path passed\n // back into `openPreviewFile` on shortcut click.\n filePath?: string;\n fileKind?: PreviewFileKind;\n}\n\n/** Solid amber folder — the exact glyph the Files app uses for folders, so\n * desktop folders and Files-app folders read as the same object. */\nexport function FolderGlyph({ className }: { className?: string }) {\n return (\n <svg className={className} fill=\"currentColor\" viewBox=\"0 0 24 24\">\n <path d=\"M2.25 7.125A2.25 2.25 0 014.5 4.875h4.504c.61 0 1.193.243 1.624.673l1.494 1.494a.75.75 0 00.53.22h7.098A2.25 2.25 0 0122 9.51v8.366A2.25 2.25 0 0119.75 20.125H4.25A2.25 2.25 0 012 17.875V7.125z\" />\n </svg>\n );\n}\n\n/** Page shortcuts reuse the consumer-registered nav icons; historically the\n * label was tried first, then the route. */\nfunction pageGlyph(entityId?: string, label?: string): ReactNode {\n return (label && navIcons[label]) || (entityId ? navIcons[entityId] : undefined);\n}\n\nconst PAGE_FALLBACK_GLYPH = (\n <svg className=\"h-7 w-7\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M3.75 6A2.25 2.25 0 016 3.75h2.25A2.25 2.25 0 0110.5 6v2.25a2.25 2.25 0 01-2.25 2.25H6a2.25 2.25 0 01-2.25-2.25V6z\" />\n </svg>\n);\n\n// Shared 48×48 tile used by the desktop icon renderer (and anything else\n// that needs a full-size shortcut tile), so all surfaces stay in sync.\nexport function FileIconTile({ entityType, isSelected, entityId, label, fileKind }: {\n entityType: string;\n isSelected: boolean;\n entityId?: string;\n label?: string;\n fileKind?: PreviewFileKind;\n}) {\n const isPreviewFile = entityType === 'preview-file' && fileKind;\n const previewColor = isPreviewFile ? PREVIEW_FILE_COLORS[fileKind!] : null;\n const previewCode = isPreviewFile ? PREVIEW_FILE_CODES[fileKind!] : null;\n if (entityType === 'folder') {\n return (\n <div className={`w-12 h-12 flex items-center justify-center ${isSelected ? 'rounded-lg bg-blue-400/30 ring-2 ring-blue-500' : ''}`}>\n <FolderGlyph className=\"h-12 w-12 text-amber-500 drop-shadow-[0_2px_3px_rgba(0,0,0,0.3)]\" />\n </div>\n );\n }\n if (entityType === 'page') {\n // App shortcut — colored gradient tile with the white nav glyph, same\n // treatment as the mobile home grid (and the same hash seed, the route,\n // so an app keeps its color across surfaces).\n const icon = pageGlyph(entityId, label);\n const gradient = hashGradient(entityId || label || 'page');\n return (\n <div className={`w-12 h-12 flex items-center justify-center ${isSelected ? 'rounded-lg bg-blue-400/30 ring-2 ring-blue-500' : ''}`}>\n <span className={`h-11 w-11 rounded-xl bg-gradient-to-br ${gradient} flex items-center justify-center text-white border border-white/30 drop-shadow-[0_2px_3px_rgba(0,0,0,0.3)]`}>\n {icon && isValidElement(icon)\n ? cloneElement(icon as ReactElement, { className: 'h-7 w-7 text-white' })\n : PAGE_FALLBACK_GLYPH}\n </span>\n </div>\n );\n }\n return (\n <div className={`w-12 h-12 relative flex items-center justify-center ${isSelected ? 'rounded-lg bg-blue-400/30 ring-2 ring-blue-500' : ''}`}>\n <svg className={`w-10 h-12 drop-shadow-[0_2px_3px_rgba(0,0,0,0.3)] ${previewColor ?? ENTITY_ICON_COLORS[entityType] ?? 'text-gray-500'}`} viewBox=\"0 0 40 48\" fill=\"none\">\n <path d=\"M4 0h22l10 10v34a4 4 0 01-4 4H4a4 4 0 01-4-4V4a4 4 0 014-4z\" fill=\"white\" fillOpacity=\"0.92\" />\n <path d=\"M26 0l10 10H30a4 4 0 01-4-4V0z\" fill=\"currentColor\" fillOpacity=\"0.2\" />\n <path d=\"M4 0h22l10 10v34a4 4 0 01-4 4H4a4 4 0 01-4-4V4a4 4 0 014-4z\" stroke=\"currentColor\" strokeWidth=\"1.5\" strokeOpacity=\"0.5\" />\n </svg>\n <span className={`absolute inset-0 flex items-center justify-center text-[9px] font-bold pt-2 ${previewColor ?? ENTITY_ICON_COLORS[entityType] ?? 'text-gray-600'}`}>\n {previewCode ?? ENTITY_ICONS[entityType] ?? entityType.slice(0, 3).toUpperCase()}\n </span>\n </div>\n );\n}\n\n/** Compact row icon for desktop shortcuts listed inside the Files app. */\nexport function DesktopItemMiniIcon({ item }: { item: DesktopItem }) {\n if (item.entityType === 'folder') {\n return <FolderGlyph className=\"h-4 w-4 shrink-0 text-amber-500\" />;\n }\n if (item.entityType === 'page') {\n const icon = pageGlyph(item.entityId, item.label);\n return (\n <span className={`h-4 w-4 rounded bg-gradient-to-br ${hashGradient(item.entityId || item.label || 'page')} flex items-center justify-center text-white shrink-0`}>\n {icon && isValidElement(icon)\n ? cloneElement(icon as ReactElement, { className: 'h-3 w-3 text-white' })\n : <span className=\"h-1.5 w-1.5 rounded-[2px] border border-white/80\" />}\n </span>\n );\n }\n const color = (item.entityType === 'preview-file' && item.fileKind\n ? PREVIEW_FILE_COLORS[item.fileKind]\n : ENTITY_ICON_COLORS[item.entityType]) ?? 'text-gray-400';\n return (\n <svg className={`h-4 w-4 shrink-0 ${color}`} fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M19.5 14.25v-2.625a3.375 3.375 0 00-3.375-3.375h-1.5A1.125 1.125 0 0113.5 7.125v-1.5a3.375 3.375 0 00-3.375-3.375H8.25m2.25 0H5.625c-.621 0-1.125.504-1.125 1.125v17.25c0 .621.504 1.125 1.125 1.125h12.75c.621 0 1.125-.504 1.125-1.125V11.25a9 9 0 00-9-9z\" />\n </svg>\n );\n}\n\n/** Short type tag shown in the Files-app listing of a desktop folder. */\nexport function desktopItemTypeLabel(item: DesktopItem): string {\n if (item.entityType === 'preview-file' && item.fileKind) return PREVIEW_FILE_CODES[item.fileKind];\n if (item.entityType === 'page') return 'App';\n if (item.entityType === 'folder') return 'Folder';\n return ENTITY_ICONS[item.entityType] ?? item.entityType.slice(0, 3).toUpperCase();\n}\n\n// ── Desktop-folder bridge ───────────────────────────────────────────────────\n//\n// Desktop folders are virtual — shortcut collections persisted in the user's\n// prefs, not directories on the file server. The Files app shows them anyway\n// (sidebar section + folder listing), so Desktop publishes its live folder\n// state here and Files subscribes via useSyncExternalStore. Mutations route\n// back through the callbacks so persistence stays in Desktop's host-aware\n// save path.\n\nexport interface DesktopFoldersSnapshot {\n folders: { id: string; name: string; itemCount: number }[];\n itemsByFolder: Record<string, DesktopItem[]>;\n /** Clear `folderId` on these shortcuts — they reappear on the desktop. */\n moveToDesktop: (items: DesktopItem[]) => void;\n /** Delete these shortcuts entirely. */\n removeShortcuts: (items: DesktopItem[]) => void;\n}\n\nlet desktopFoldersSnapshot: DesktopFoldersSnapshot | null = null;\nconst desktopFoldersListeners = new Set<() => void>();\n\nexport function publishDesktopFolders(next: DesktopFoldersSnapshot | null) {\n desktopFoldersSnapshot = next;\n desktopFoldersListeners.forEach(l => l());\n}\n\nexport function subscribeDesktopFolders(listener: () => void): () => void {\n desktopFoldersListeners.add(listener);\n return () => { desktopFoldersListeners.delete(listener); };\n}\n\nexport function getDesktopFoldersSnapshot(): DesktopFoldersSnapshot | null {\n return desktopFoldersSnapshot;\n}\n\n// ── Files-app view side channel ─────────────────────────────────────────────\n//\n// Used by the desktop Trash icon and desktop folders to open the Files app\n// on a specific view. Two cases handled:\n// 1) Files isn't open yet — the flag below is read once on first mount.\n// 2) Files is already open — the event tells the live instance to flip view.\n// The caller follows up with `openPage('/files')` either way.\n\nexport type FilesViewRequest = 'trash' | { view: 'desktop-folder'; folderId: string };\n\nexport const FILES_VIEW_FLAG = '__REACT_OS_SHELL_FILES_VIEW__';\nexport const FILES_SHOW_TRASH_EVENT = 'react-os-shell:files-show-trash';\nexport const FILES_OPEN_DESKTOP_FOLDER_EVENT = 'react-os-shell:files-open-desktop-folder';\n\nexport function requestFilesTrashView() {\n if (typeof window === 'undefined') return;\n (window as any)[FILES_VIEW_FLAG] = 'trash' satisfies FilesViewRequest;\n window.dispatchEvent(new CustomEvent(FILES_SHOW_TRASH_EVENT));\n}\n\nexport function requestFilesDesktopFolderView(folderId: string) {\n if (typeof window === 'undefined') return;\n (window as any)[FILES_VIEW_FLAG] = { view: 'desktop-folder', folderId } satisfies FilesViewRequest;\n window.dispatchEvent(new CustomEvent(FILES_OPEN_DESKTOP_FOLDER_EVENT, { detail: { folderId } }));\n}\n\n/** Read the pending view request without clearing it. Side-effect free so\n * it is safe inside a `useState` initializer (StrictMode double-invokes\n * those; a read-and-clear there would hand the second invocation null). */\nexport function peekFilesViewRequest(): FilesViewRequest | null {\n if (typeof window === 'undefined') return null;\n const raw = (window as any)[FILES_VIEW_FLAG];\n if (raw === 'trash') return 'trash';\n if (raw && typeof raw === 'object' && raw.view === 'desktop-folder' && typeof raw.folderId === 'string') return raw;\n return null;\n}\n\n/** Read-and-clear the pending view request (Files calls this once mounted). */\nexport function consumeFilesViewRequest(): FilesViewRequest | null {\n const req = peekFilesViewRequest();\n if (typeof window !== 'undefined') (window as any)[FILES_VIEW_FLAG] = null;\n return req;\n}\n"]}
@@ -1,6 +1,6 @@
1
- import { ImageAnnotator_default } from './chunk-KUIPWCTJ.js';
2
- import { toast_default } from './chunk-WIJ45SYD.js';
3
- import { AboutApp } from './chunk-IXW6775F.js';
1
+ import { ImageAnnotator_default } from './chunk-NUPYEVU4.js';
2
+ import { toast_default } from './chunk-VENYVK3L.js';
3
+ import { AboutApp } from './chunk-IONOMHM2.js';
4
4
  import { WindowTitle, registerModalEscapeInterceptor, getActiveModalId } from './chunk-JNF5VRPB.js';
5
5
  import { createContext, useRef, useEffect, useState, useContext } from 'react';
6
6
  import { createPortal } from 'react-dom';
@@ -3299,5 +3299,5 @@ function ImagePanel({ url, filename, onDownload, onEmail }) {
3299
3299
  }
3300
3300
 
3301
3301
  export { Preview, setPdfPreview };
3302
- //# sourceMappingURL=chunk-WG3PMYDQ.js.map
3303
- //# sourceMappingURL=chunk-WG3PMYDQ.js.map
3302
+ //# sourceMappingURL=chunk-DF6TNR4B.js.map
3303
+ //# sourceMappingURL=chunk-DF6TNR4B.js.map