react-os-shell 0.14.0 → 1.1.0

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 (38) hide show
  1. package/dist/Files-V4SZKOGZ.js +12 -0
  2. package/dist/{Files-5N64E375.js.map → Files-V4SZKOGZ.js.map} +1 -1
  3. package/dist/{Notepad-AFAUA4KJ.js → Notepad-6JJ4RT3U.js} +110 -101
  4. package/dist/Notepad-6JJ4RT3U.js.map +1 -0
  5. package/dist/{PomodoroTimer-SBBPQMYB.js → PomodoroTimer-HGPJ5R6V.js} +3 -4
  6. package/dist/PomodoroTimer-HGPJ5R6V.js.map +1 -0
  7. package/dist/Stock-XLC3LZJV.js +152 -0
  8. package/dist/Stock-XLC3LZJV.js.map +1 -0
  9. package/dist/apps/index.d.ts +16 -4
  10. package/dist/apps/index.js +10 -11
  11. package/dist/apps/index.js.map +1 -1
  12. package/dist/chunk-4R4SXMDV.js +98 -0
  13. package/dist/chunk-4R4SXMDV.js.map +1 -0
  14. package/dist/{chunk-MK3HLUO4.js → chunk-5X5LQNOX.js} +172 -3
  15. package/dist/chunk-5X5LQNOX.js.map +1 -0
  16. package/dist/chunk-UXEG2NRI.js +682 -0
  17. package/dist/chunk-UXEG2NRI.js.map +1 -0
  18. package/dist/chunk-VGTEM5RZ.js +89 -0
  19. package/dist/chunk-VGTEM5RZ.js.map +1 -0
  20. package/dist/index.d.ts +9 -1
  21. package/dist/index.js +16 -142
  22. package/dist/index.js.map +1 -1
  23. package/dist/styles.css +26 -2
  24. package/package.json +1 -1
  25. package/dist/Files-5N64E375.js +0 -11
  26. package/dist/Notepad-AFAUA4KJ.js.map +0 -1
  27. package/dist/PomodoroTimer-SBBPQMYB.js.map +0 -1
  28. package/dist/Stock-ICDNFM7U.js +0 -234
  29. package/dist/Stock-ICDNFM7U.js.map +0 -1
  30. package/dist/TodoList-26N6ZTLN.js +0 -309
  31. package/dist/TodoList-26N6ZTLN.js.map +0 -1
  32. package/dist/chunk-D4ZM3K2S.js +0 -605
  33. package/dist/chunk-D4ZM3K2S.js.map +0 -1
  34. package/dist/chunk-MK3HLUO4.js.map +0 -1
  35. package/dist/chunk-OB7T3Q5C.js +0 -46
  36. package/dist/chunk-OB7T3Q5C.js.map +0 -1
  37. package/dist/chunk-QTJ2CHJX.js +0 -174
  38. package/dist/chunk-QTJ2CHJX.js.map +0 -1
@@ -0,0 +1,152 @@
1
+ import { loadAppearance, WidgetSettingsModal } from './chunk-7TG7VNJS.js';
2
+ import { useWidgetSettings } from './chunk-4RXDOSKZ.js';
3
+ import './chunk-UBN4IUDE.js';
4
+ import './chunk-ZF6AYO4G.js';
5
+ import { useState, useCallback } from 'react';
6
+ import { jsxs, Fragment, jsx } from 'react/jsx-runtime';
7
+
8
+ var MAX_SYMBOLS = 8;
9
+ var DEFAULT_SYMBOLS = ["AAPL", "MSFT", "GOOGL", "TSLA", "NVDA"];
10
+ var SYMBOLS_KEY = "stock_symbols";
11
+ var SETTINGS_KEY = "stock_appearance";
12
+ var DEMO_QUOTES = {
13
+ AAPL: { price: 229.87, change: 1.42, changePct: 0.62 },
14
+ MSFT: { price: 451.16, change: -2.31, changePct: -0.51 },
15
+ GOOGL: { price: 178.34, change: 0.89, changePct: 0.5 },
16
+ AMZN: { price: 201.45, change: -1.08, changePct: -0.53 },
17
+ TSLA: { price: 248.92, change: 6.74, changePct: 2.78 },
18
+ NVDA: { price: 134.81, change: -3.12, changePct: -2.26 },
19
+ META: { price: 563.27, change: 4.55, changePct: 0.81 }
20
+ };
21
+ var DEMO_TICKERS = Object.keys(DEMO_QUOTES);
22
+ function loadSymbols() {
23
+ try {
24
+ const s = JSON.parse(localStorage.getItem(SYMBOLS_KEY) || "");
25
+ if (Array.isArray(s) && s.length) return s.slice(0, MAX_SYMBOLS);
26
+ } catch {
27
+ }
28
+ return DEFAULT_SYMBOLS;
29
+ }
30
+ var fmtPrice = (n) => `$${n.toLocaleString(void 0, { minimumFractionDigits: 2, maximumFractionDigits: 2 })}`;
31
+ var fmtSigned = (n) => `${n >= 0 ? "+" : ""}${n.toFixed(2)}`;
32
+ function Stock() {
33
+ const [symbols, setSymbols] = useState(loadSymbols);
34
+ const [appearance, setAppearance] = useState(() => loadAppearance(SETTINGS_KEY));
35
+ const [settingsOpen, setSettingsOpen] = useState(false);
36
+ const [configSymbols, setConfigSymbols] = useState([]);
37
+ const [configAppearance, setConfigAppearance] = useState(appearance);
38
+ const [newSymbol, setNewSymbol] = useState("");
39
+ const openSettings = useCallback(() => {
40
+ setConfigSymbols([...symbols]);
41
+ setConfigAppearance({ ...appearance });
42
+ setNewSymbol("");
43
+ setSettingsOpen(true);
44
+ }, [symbols, appearance]);
45
+ useWidgetSettings(openSettings);
46
+ const addSymbol = () => {
47
+ const s = newSymbol.trim().toUpperCase();
48
+ if (!s || configSymbols.includes(s) || configSymbols.length >= MAX_SYMBOLS) return;
49
+ setConfigSymbols((p) => [...p, s]);
50
+ setNewSymbol("");
51
+ };
52
+ const saveSettings = () => {
53
+ const cleaned = configSymbols.map((s) => s.toUpperCase()).slice(0, MAX_SYMBOLS);
54
+ setSymbols(cleaned);
55
+ setAppearance(configAppearance);
56
+ localStorage.setItem(SYMBOLS_KEY, JSON.stringify(cleaned));
57
+ localStorage.setItem(SETTINGS_KEY, JSON.stringify(configAppearance));
58
+ setSettingsOpen(false);
59
+ };
60
+ const availableTickers = DEMO_TICKERS.filter((t) => !configSymbols.includes(t));
61
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
62
+ /* @__PURE__ */ jsx(
63
+ "div",
64
+ {
65
+ className: "flex flex-col h-full",
66
+ style: { backgroundColor: `rgb(var(--taskbar-bg-rgb, 243 244 246) / ${appearance.activeOpacity / 100})`, backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : void 0 },
67
+ children: /* @__PURE__ */ jsx("div", { className: "px-4 py-3 flex-1", children: symbols.length === 0 ? /* @__PURE__ */ jsx("div", { className: "text-xs text-gray-500 text-center py-6", children: "No symbols yet \u2014 add some in settings." }) : /* @__PURE__ */ jsx("div", { className: "space-y-0.5", children: symbols.map((sym) => {
68
+ const q = DEMO_QUOTES[sym];
69
+ const up = (q?.change ?? 0) >= 0;
70
+ const color = !q ? "text-gray-400" : q.change > 0 ? "text-green-600" : q.change < 0 ? "text-red-600" : "text-gray-500";
71
+ return /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between px-2 py-2 border-b border-gray-200 last:border-0", children: [
72
+ /* @__PURE__ */ jsx("span", { className: "text-sm font-bold text-gray-800 tracking-tight", children: sym }),
73
+ q ? /* @__PURE__ */ jsxs("div", { className: "flex flex-col items-end leading-tight", children: [
74
+ /* @__PURE__ */ jsx("span", { className: "text-sm font-mono font-semibold text-gray-900 tabular-nums", children: fmtPrice(q.price) }),
75
+ /* @__PURE__ */ jsxs("span", { className: `text-[11px] font-medium tabular-nums ${color}`, children: [
76
+ up ? "\u25B2" : "\u25BC",
77
+ " ",
78
+ fmtSigned(q.change),
79
+ " (",
80
+ fmtSigned(q.changePct),
81
+ "%)"
82
+ ] })
83
+ ] }) : /* @__PURE__ */ jsx("span", { className: "text-sm font-mono text-gray-400 tabular-nums", children: "\u2014" })
84
+ ] }, sym);
85
+ }) }) })
86
+ }
87
+ ),
88
+ /* @__PURE__ */ jsx(
89
+ WidgetSettingsModal,
90
+ {
91
+ open: settingsOpen,
92
+ onClose: () => setSettingsOpen(false),
93
+ title: "Stock Settings",
94
+ appearance: configAppearance,
95
+ onAppearanceChange: setConfigAppearance,
96
+ onSave: saveSettings,
97
+ children: /* @__PURE__ */ jsxs("div", { children: [
98
+ /* @__PURE__ */ jsxs("h3", { className: "text-sm font-semibold text-gray-700 mb-2", children: [
99
+ "Symbols ",
100
+ /* @__PURE__ */ jsxs("span", { className: "text-[11px] font-normal text-gray-400", children: [
101
+ "(",
102
+ configSymbols.length,
103
+ "/",
104
+ MAX_SYMBOLS,
105
+ ")"
106
+ ] })
107
+ ] }),
108
+ /* @__PURE__ */ jsxs("div", { className: "space-y-1 mb-2", children: [
109
+ configSymbols.map((s, i) => /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between py-1 px-2 bg-gray-50 rounded", children: [
110
+ /* @__PURE__ */ jsx("span", { className: "text-sm font-semibold text-gray-700", children: s }),
111
+ /* @__PURE__ */ jsx("button", { onClick: () => setConfigSymbols((p) => p.filter((_, idx) => idx !== i)), className: "text-red-400 hover:text-red-600", children: "\xD7" })
112
+ ] }, i)),
113
+ configSymbols.length === 0 && /* @__PURE__ */ jsx("p", { className: "text-[11px] text-gray-400 italic", children: "No symbols \u2014 add a ticker below." })
114
+ ] }),
115
+ configSymbols.length < MAX_SYMBOLS ? /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
116
+ /* @__PURE__ */ jsx(
117
+ "input",
118
+ {
119
+ value: newSymbol,
120
+ onChange: (e) => setNewSymbol(e.target.value),
121
+ onKeyDown: (e) => {
122
+ if (e.key === "Enter") addSymbol();
123
+ },
124
+ placeholder: "e.g. NVDA",
125
+ className: "flex-1 bg-gray-50 border border-gray-200 rounded px-2 py-1 text-sm uppercase focus:outline-none focus:ring-1 focus:ring-blue-500"
126
+ }
127
+ ),
128
+ /* @__PURE__ */ jsx("button", { onClick: addSymbol, className: "text-sm font-medium text-blue-600 hover:text-blue-800 px-2", children: "+ Add" })
129
+ ] }) : /* @__PURE__ */ jsxs("p", { className: "text-[11px] text-gray-400 italic", children: [
130
+ "Max ",
131
+ MAX_SYMBOLS,
132
+ " symbols \u2014 remove one to add another."
133
+ ] }),
134
+ availableTickers.length > 0 && configSymbols.length < MAX_SYMBOLS && /* @__PURE__ */ jsx("div", { className: "flex flex-wrap gap-1.5 mt-2", children: availableTickers.map((t) => /* @__PURE__ */ jsx(
135
+ "button",
136
+ {
137
+ onClick: () => setConfigSymbols((p) => p.length < MAX_SYMBOLS ? [...p, t] : p),
138
+ className: "text-[11px] font-medium text-gray-600 bg-gray-100 hover:bg-gray-200 rounded px-2 py-0.5 transition-colors",
139
+ children: t
140
+ },
141
+ t
142
+ )) }),
143
+ /* @__PURE__ */ jsx("p", { className: "text-[11px] text-gray-400 mt-2", children: "Demo prices \u2014 static sample data, no live feed." })
144
+ ] })
145
+ }
146
+ )
147
+ ] });
148
+ }
149
+
150
+ export { Stock as default };
151
+ //# sourceMappingURL=Stock-XLC3LZJV.js.map
152
+ //# sourceMappingURL=Stock-XLC3LZJV.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/apps/Stock.tsx"],"names":[],"mappings":";;;;;;;AAYA,IAAM,WAAA,GAAc,CAAA;AACpB,IAAM,kBAAkB,CAAC,MAAA,EAAQ,MAAA,EAAQ,OAAA,EAAS,QAAQ,MAAM,CAAA;AAEhE,IAAM,WAAA,GAAc,eAAA;AACpB,IAAM,YAAA,GAAe,kBAAA;AAKrB,IAAM,WAAA,GAAqC;AAAA,EACzC,MAAM,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,IAAA,EAAM,WAAW,IAAA,EAAK;AAAA,EACrD,MAAM,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,KAAA,EAAO,WAAW,KAAA,EAAM;AAAA,EACvD,OAAO,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,IAAA,EAAM,WAAW,GAAA,EAAK;AAAA,EACtD,MAAM,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,KAAA,EAAO,WAAW,KAAA,EAAM;AAAA,EACvD,MAAM,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,IAAA,EAAM,WAAW,IAAA,EAAK;AAAA,EACrD,MAAM,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,KAAA,EAAO,WAAW,KAAA,EAAM;AAAA,EACvD,MAAM,EAAE,KAAA,EAAO,QAAQ,MAAA,EAAQ,IAAA,EAAM,WAAW,IAAA;AAClD,CAAA;AAGA,IAAM,YAAA,GAAe,MAAA,CAAO,IAAA,CAAK,WAAW,CAAA;AAE5C,SAAS,WAAA,GAAwB;AAC/B,EAAA,IAAI;AAAE,IAAA,MAAM,IAAI,IAAA,CAAK,KAAA,CAAM,aAAa,OAAA,CAAQ,WAAW,KAAK,EAAE,CAAA;AAAG,IAAA,IAAI,KAAA,CAAM,OAAA,CAAQ,CAAC,CAAA,IAAK,CAAA,CAAE,QAAQ,OAAO,CAAA,CAAE,KAAA,CAAM,CAAA,EAAG,WAAW,CAAA;AAAA,EAAG,CAAA,CAAA,MAAQ;AAAA,EAAC;AAChJ,EAAA,OAAO,eAAA;AACT;AAEA,IAAM,QAAA,GAAW,CAAC,CAAA,KAAc,CAAA,CAAA,EAAI,CAAA,CAAE,cAAA,CAAe,MAAA,EAAW,EAAE,qBAAA,EAAuB,CAAA,EAAG,qBAAA,EAAuB,CAAA,EAAG,CAAC,CAAA,CAAA;AACvH,IAAM,SAAA,GAAY,CAAC,CAAA,KAAc,CAAA,EAAG,CAAA,IAAK,CAAA,GAAI,GAAA,GAAM,EAAE,CAAA,EAAG,CAAA,CAAE,OAAA,CAAQ,CAAC,CAAC,CAAA,CAAA;AAErD,SAAR,KAAA,GAAyB;AAC9B,EAAA,MAAM,CAAC,OAAA,EAAS,UAAU,CAAA,GAAI,SAAS,WAAW,CAAA;AAClD,EAAA,MAAM,CAAC,YAAY,aAAa,CAAA,GAAI,SAAS,MAAM,cAAA,CAAe,YAAY,CAAC,CAAA;AAE/E,EAAA,MAAM,CAAC,YAAA,EAAc,eAAe,CAAA,GAAI,SAAS,KAAK,CAAA;AACtD,EAAA,MAAM,CAAC,aAAA,EAAe,gBAAgB,CAAA,GAAI,QAAA,CAAmB,EAAE,CAAA;AAC/D,EAAA,MAAM,CAAC,gBAAA,EAAkB,mBAAmB,CAAA,GAAI,SAA2B,UAAU,CAAA;AACrF,EAAA,MAAM,CAAC,SAAA,EAAW,YAAY,CAAA,GAAI,SAAS,EAAE,CAAA;AAE7C,EAAA,MAAM,YAAA,GAAe,YAAY,MAAM;AACrC,IAAA,gBAAA,CAAiB,CAAC,GAAG,OAAO,CAAC,CAAA;AAC7B,IAAA,mBAAA,CAAoB,EAAE,GAAG,UAAA,EAAY,CAAA;AACrC,IAAA,YAAA,CAAa,EAAE,CAAA;AACf,IAAA,eAAA,CAAgB,IAAI,CAAA;AAAA,EACtB,CAAA,EAAG,CAAC,OAAA,EAAS,UAAU,CAAC,CAAA;AAExB,EAAA,iBAAA,CAAkB,YAAY,CAAA;AAE9B,EAAA,MAAM,YAAY,MAAM;AACtB,IAAA,MAAM,CAAA,GAAI,SAAA,CAAU,IAAA,EAAK,CAAE,WAAA,EAAY;AACvC,IAAA,IAAI,CAAC,KAAK,aAAA,CAAc,QAAA,CAAS,CAAC,CAAA,IAAK,aAAA,CAAc,UAAU,WAAA,EAAa;AAC5E,IAAA,gBAAA,CAAiB,CAAA,CAAA,KAAK,CAAC,GAAG,CAAA,EAAG,CAAC,CAAC,CAAA;AAC/B,IAAA,YAAA,CAAa,EAAE,CAAA;AAAA,EACjB,CAAA;AAEA,EAAA,MAAM,eAAe,MAAM;AACzB,IAAA,MAAM,OAAA,GAAU,aAAA,CAAc,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,aAAa,CAAA,CAAE,KAAA,CAAM,CAAA,EAAG,WAAW,CAAA;AAC5E,IAAA,UAAA,CAAW,OAAO,CAAA;AAAG,IAAA,aAAA,CAAc,gBAAgB,CAAA;AACnD,IAAA,YAAA,CAAa,OAAA,CAAQ,WAAA,EAAa,IAAA,CAAK,SAAA,CAAU,OAAO,CAAC,CAAA;AACzD,IAAA,YAAA,CAAa,OAAA,CAAQ,YAAA,EAAc,IAAA,CAAK,SAAA,CAAU,gBAAgB,CAAC,CAAA;AACnE,IAAA,eAAA,CAAgB,KAAK,CAAA;AAAA,EACvB,CAAA;AAGA,EAAA,MAAM,gBAAA,GAAmB,aAAa,MAAA,CAAO,CAAA,CAAA,KAAK,CAAC,aAAA,CAAc,QAAA,CAAS,CAAC,CAAC,CAAA;AAE5E,EAAA,uBACE,IAAA,CAAA,QAAA,EAAA,EAGE,QAAA,EAAA;AAAA,oBAAA,GAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QAAI,SAAA,EAAU,sBAAA;AAAA,QACb,OAAO,EAAE,eAAA,EAAiB,CAAA,yCAAA,EAA4C,UAAA,CAAW,gBAAgB,GAAG,CAAA,CAAA,CAAA,EAAK,cAAA,EAAgB,UAAA,CAAW,aAAa,CAAA,GAAI,CAAA,KAAA,EAAQ,UAAA,CAAW,UAAU,QAAQ,MAAA,EAAU;AAAA,QACpM,8BAAC,KAAA,EAAA,EAAI,SAAA,EAAU,oBACZ,QAAA,EAAA,OAAA,CAAQ,MAAA,KAAW,oBAClB,GAAA,CAAC,KAAA,EAAA,EAAI,WAAU,wCAAA,EAAyC,QAAA,EAAA,6CAAA,EAAsC,oBAE9F,GAAA,CAAC,KAAA,EAAA,EAAI,WAAU,aAAA,EACZ,QAAA,EAAA,OAAA,CAAQ,IAAI,CAAA,GAAA,KAAO;AAClB,UAAA,MAAM,CAAA,GAAI,YAAY,GAAG,CAAA;AACzB,UAAA,MAAM,EAAA,GAAA,CAAM,CAAA,EAAG,MAAA,IAAU,CAAA,KAAM,CAAA;AAC/B,UAAA,MAAM,KAAA,GAAQ,CAAC,CAAA,GAAI,eAAA,GAAkB,CAAA,CAAE,MAAA,GAAS,CAAA,GAAI,gBAAA,GAAmB,CAAA,CAAE,MAAA,GAAS,CAAA,GAAI,cAAA,GAAiB,eAAA;AACvG,UAAA,uBACE,IAAA,CAAC,KAAA,EAAA,EAAc,SAAA,EAAU,oFAAA,EACvB,QAAA,EAAA;AAAA,4BAAA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,gDAAA,EAAkD,QAAA,EAAA,GAAA,EAAI,CAAA;AAAA,YACrE,CAAA,mBACC,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,uCAAA,EACb,QAAA,EAAA;AAAA,8BAAA,GAAA,CAAC,UAAK,SAAA,EAAU,4DAAA,EAA8D,QAAA,EAAA,QAAA,CAAS,CAAA,CAAE,KAAK,CAAA,EAAE,CAAA;AAAA,8BAChG,IAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAW,CAAA,qCAAA,EAAwC,KAAK,CAAA,CAAA,EAAK,QAAA,EAAA;AAAA,gBAAA,EAAA,GAAK,QAAA,GAAM,QAAA;AAAA,gBAAI,GAAA;AAAA,gBAAE,SAAA,CAAU,EAAE,MAAM,CAAA;AAAA,gBAAE,IAAA;AAAA,gBAAG,SAAA,CAAU,EAAE,SAAS,CAAA;AAAA,gBAAE;AAAA,eAAA,EAAE;AAAA,aAAA,EACtI,CAAA,mBAEA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,gDAA+C,QAAA,EAAA,QAAA,EAAC;AAAA,WAAA,EAAA,EAR1D,GAUV,CAAA;AAAA,QAEJ,CAAC,GACH,CAAA,EAEJ;AAAA;AAAA,KACF;AAAA,oBAEA,GAAA;AAAA,MAAC,mBAAA;AAAA,MAAA;AAAA,QAAoB,IAAA,EAAM,YAAA;AAAA,QAAc,OAAA,EAAS,MAAM,eAAA,CAAgB,KAAK,CAAA;AAAA,QAAG,KAAA,EAAM,gBAAA;AAAA,QACpF,UAAA,EAAY,gBAAA;AAAA,QAAkB,kBAAA,EAAoB,mBAAA;AAAA,QAAqB,MAAA,EAAQ,YAAA;AAAA,QAC/E,+BAAC,KAAA,EAAA,EACC,QAAA,EAAA;AAAA,0BAAA,IAAA,CAAC,IAAA,EAAA,EAAG,WAAU,0CAAA,EAA2C,QAAA,EAAA;AAAA,YAAA,UAAA;AAAA,4BAAQ,IAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,uCAAA,EAAwC,QAAA,EAAA;AAAA,cAAA,GAAA;AAAA,cAAE,aAAA,CAAc,MAAA;AAAA,cAAO,GAAA;AAAA,cAAE,WAAA;AAAA,cAAY;AAAA,aAAA,EAAC;AAAA,WAAA,EAAO,CAAA;AAAA,0BACtK,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,gBAAA,EACZ,QAAA,EAAA;AAAA,YAAA,aAAA,CAAc,IAAI,CAAC,CAAA,EAAG,sBACrB,IAAA,CAAC,KAAA,EAAA,EAAY,WAAU,gEAAA,EACrB,QAAA,EAAA;AAAA,8BAAA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,qCAAA,EAAuC,QAAA,EAAA,CAAA,EAAE,CAAA;AAAA,kCACxD,QAAA,EAAA,EAAO,OAAA,EAAS,MAAM,gBAAA,CAAiB,OAAK,CAAA,CAAE,MAAA,CAAO,CAAC,CAAA,EAAG,QAAQ,GAAA,KAAQ,CAAC,CAAC,CAAA,EAAG,SAAA,EAAU,mCAAkC,QAAA,EAAA,MAAA,EAAO;AAAA,aAAA,EAAA,EAF1H,CAGV,CACD,CAAA;AAAA,YACA,cAAc,MAAA,KAAW,CAAA,wBAAM,GAAA,EAAA,EAAE,SAAA,EAAU,oCAAmC,QAAA,EAAA,uCAAA,EAAgC;AAAA,WAAA,EACjH,CAAA;AAAA,UACC,cAAc,MAAA,GAAS,WAAA,mBACtB,IAAA,CAAC,KAAA,EAAA,EAAI,WAAU,yBAAA,EACb,QAAA,EAAA;AAAA,4BAAA,GAAA;AAAA,cAAC,OAAA;AAAA,cAAA;AAAA,gBAAM,KAAA,EAAO,SAAA;AAAA,gBAAW,QAAA,EAAU,CAAA,CAAA,KAAK,YAAA,CAAa,CAAA,CAAE,OAAO,KAAK,CAAA;AAAA,gBAAG,WAAW,CAAA,CAAA,KAAK;AAAE,kBAAA,IAAI,CAAA,CAAE,GAAA,KAAQ,OAAA,EAAS,SAAA,EAAU;AAAA,gBAAG,CAAA;AAAA,gBAC1H,WAAA,EAAY,WAAA;AAAA,gBAAY,SAAA,EAAU;AAAA;AAAA,aAAmI;AAAA,gCACtK,QAAA,EAAA,EAAO,OAAA,EAAS,SAAA,EAAW,SAAA,EAAU,8DAA6D,QAAA,EAAA,OAAA,EAAK;AAAA,WAAA,EAC1G,CAAA,mBAEA,IAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,kCAAA,EAAmC,QAAA,EAAA;AAAA,YAAA,MAAA;AAAA,YAAK,WAAA;AAAA,YAAY;AAAA,WAAA,EAAqC,CAAA;AAAA,UAEvG,gBAAA,CAAiB,MAAA,GAAS,CAAA,IAAK,aAAA,CAAc,MAAA,GAAS,WAAA,oBACrD,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,6BAAA,EACZ,QAAA,EAAA,gBAAA,CAAiB,GAAA,CAAI,CAAA,CAAA,qBACpB,GAAA;AAAA,YAAC,QAAA;AAAA,YAAA;AAAA,cAAe,OAAA,EAAS,MAAM,gBAAA,CAAiB,CAAA,CAAA,KAAK,CAAA,CAAE,MAAA,GAAS,WAAA,GAAc,CAAC,GAAG,CAAA,EAAG,CAAC,CAAA,GAAI,CAAC,CAAA;AAAA,cACzF,SAAA,EAAU,2GAAA;AAAA,cAA6G,QAAA,EAAA;AAAA,aAAA;AAAA,YAD5G;AAAA,WAEd,CAAA,EACH,CAAA;AAAA,0BAEF,GAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,gCAAA,EAAiC,QAAA,EAAA,sDAAA,EAA+C;AAAA,SAAA,EAC/F;AAAA;AAAA;AACF,GAAA,EACF,CAAA;AAEJ","file":"Stock-XLC3LZJV.js","sourcesContent":["import { useState, useCallback } from 'react';\nimport { useWidgetSettings } from '../shell/Modal';\nimport WidgetSettingsModal, { loadAppearance, type WidgetAppearance } from '../shell/WidgetSettingsModal';\n\n/**\n * Stock ticker widget — track a watchlist of equities with demo prices.\n *\n * Prices are static, in-file demo data (DEMO_QUOTES) so the widget works with\n * no API key and no server. \"Setup\" is just picking which demo tickers to show.\n */\n\n/** Cap the watchlist — keeps the widget compact. */\nconst MAX_SYMBOLS = 8;\nconst DEFAULT_SYMBOLS = ['AAPL', 'MSFT', 'GOOGL', 'TSLA', 'NVDA'];\n\nconst SYMBOLS_KEY = 'stock_symbols';\nconst SETTINGS_KEY = 'stock_appearance';\n\ninterface Quote { price: number; change: number; changePct: number }\n\n/** Static demo quotes — realistic-ish values, a mix of gainers and losers. */\nconst DEMO_QUOTES: Record<string, Quote> = {\n AAPL: { price: 229.87, change: 1.42, changePct: 0.62 },\n MSFT: { price: 451.16, change: -2.31, changePct: -0.51 },\n GOOGL: { price: 178.34, change: 0.89, changePct: 0.50 },\n AMZN: { price: 201.45, change: -1.08, changePct: -0.53 },\n TSLA: { price: 248.92, change: 6.74, changePct: 2.78 },\n NVDA: { price: 134.81, change: -3.12, changePct: -2.26 },\n META: { price: 563.27, change: 4.55, changePct: 0.81 },\n};\n\n/** Demo tickers offered in the symbol picker. */\nconst DEMO_TICKERS = Object.keys(DEMO_QUOTES);\n\nfunction loadSymbols(): string[] {\n try { const s = JSON.parse(localStorage.getItem(SYMBOLS_KEY) || ''); if (Array.isArray(s) && s.length) return s.slice(0, MAX_SYMBOLS); } catch {}\n return DEFAULT_SYMBOLS;\n}\n\nconst fmtPrice = (n: number) => `$${n.toLocaleString(undefined, { minimumFractionDigits: 2, maximumFractionDigits: 2 })}`;\nconst fmtSigned = (n: number) => `${n >= 0 ? '+' : ''}${n.toFixed(2)}`;\n\nexport default function Stock() {\n const [symbols, setSymbols] = useState(loadSymbols);\n const [appearance, setAppearance] = useState(() => loadAppearance(SETTINGS_KEY));\n\n const [settingsOpen, setSettingsOpen] = useState(false);\n const [configSymbols, setConfigSymbols] = useState<string[]>([]);\n const [configAppearance, setConfigAppearance] = useState<WidgetAppearance>(appearance);\n const [newSymbol, setNewSymbol] = useState('');\n\n const openSettings = useCallback(() => {\n setConfigSymbols([...symbols]);\n setConfigAppearance({ ...appearance });\n setNewSymbol('');\n setSettingsOpen(true);\n }, [symbols, appearance]);\n\n useWidgetSettings(openSettings);\n\n const addSymbol = () => {\n const s = newSymbol.trim().toUpperCase();\n if (!s || configSymbols.includes(s) || configSymbols.length >= MAX_SYMBOLS) return;\n setConfigSymbols(p => [...p, s]);\n setNewSymbol('');\n };\n\n const saveSettings = () => {\n const cleaned = configSymbols.map(s => s.toUpperCase()).slice(0, MAX_SYMBOLS);\n setSymbols(cleaned); setAppearance(configAppearance);\n localStorage.setItem(SYMBOLS_KEY, JSON.stringify(cleaned));\n localStorage.setItem(SETTINGS_KEY, JSON.stringify(configAppearance));\n setSettingsOpen(false);\n };\n\n // Tickers the picker can still offer (demo set minus what's already added).\n const availableTickers = DEMO_TICKERS.filter(t => !configSymbols.includes(t));\n\n return (\n <>\n {/* Theme-aware panel — mirrors the taskbar's `--taskbar-bg-rgb` so the\n * widget matches it across themes; gray-* classes auto-invert in dark. */}\n <div className=\"flex flex-col h-full\"\n style={{ backgroundColor: `rgb(var(--taskbar-bg-rgb, 243 244 246) / ${appearance.activeOpacity / 100})`, backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : undefined }}>\n <div className=\"px-4 py-3 flex-1\">\n {symbols.length === 0 ? (\n <div className=\"text-xs text-gray-500 text-center py-6\">No symbols yet — add some in settings.</div>\n ) : (\n <div className=\"space-y-0.5\">\n {symbols.map(sym => {\n const q = DEMO_QUOTES[sym];\n const up = (q?.change ?? 0) >= 0;\n const color = !q ? 'text-gray-400' : q.change > 0 ? 'text-green-600' : q.change < 0 ? 'text-red-600' : 'text-gray-500';\n return (\n <div key={sym} className=\"flex items-center justify-between px-2 py-2 border-b border-gray-200 last:border-0\">\n <span className=\"text-sm font-bold text-gray-800 tracking-tight\">{sym}</span>\n {q ? (\n <div className=\"flex flex-col items-end leading-tight\">\n <span className=\"text-sm font-mono font-semibold text-gray-900 tabular-nums\">{fmtPrice(q.price)}</span>\n <span className={`text-[11px] font-medium tabular-nums ${color}`}>{up ? '▲' : '▼'} {fmtSigned(q.change)} ({fmtSigned(q.changePct)}%)</span>\n </div>\n ) : (\n <span className=\"text-sm font-mono text-gray-400 tabular-nums\">—</span>\n )}\n </div>\n );\n })}\n </div>\n )}\n </div>\n </div>\n\n <WidgetSettingsModal open={settingsOpen} onClose={() => setSettingsOpen(false)} title=\"Stock Settings\"\n appearance={configAppearance} onAppearanceChange={setConfigAppearance} onSave={saveSettings}>\n <div>\n <h3 className=\"text-sm font-semibold text-gray-700 mb-2\">Symbols <span className=\"text-[11px] font-normal text-gray-400\">({configSymbols.length}/{MAX_SYMBOLS})</span></h3>\n <div className=\"space-y-1 mb-2\">\n {configSymbols.map((s, i) => (\n <div key={i} className=\"flex items-center justify-between py-1 px-2 bg-gray-50 rounded\">\n <span className=\"text-sm font-semibold text-gray-700\">{s}</span>\n <button onClick={() => setConfigSymbols(p => p.filter((_, idx) => idx !== i))} className=\"text-red-400 hover:text-red-600\">&times;</button>\n </div>\n ))}\n {configSymbols.length === 0 && <p className=\"text-[11px] text-gray-400 italic\">No symbols — add a ticker below.</p>}\n </div>\n {configSymbols.length < MAX_SYMBOLS ? (\n <div className=\"flex items-center gap-2\">\n <input value={newSymbol} onChange={e => setNewSymbol(e.target.value)} onKeyDown={e => { if (e.key === 'Enter') addSymbol(); }}\n placeholder=\"e.g. NVDA\" className=\"flex-1 bg-gray-50 border border-gray-200 rounded px-2 py-1 text-sm uppercase focus:outline-none focus:ring-1 focus:ring-blue-500\" />\n <button onClick={addSymbol} className=\"text-sm font-medium text-blue-600 hover:text-blue-800 px-2\">+ Add</button>\n </div>\n ) : (\n <p className=\"text-[11px] text-gray-400 italic\">Max {MAX_SYMBOLS} symbols — remove one to add another.</p>\n )}\n {availableTickers.length > 0 && configSymbols.length < MAX_SYMBOLS && (\n <div className=\"flex flex-wrap gap-1.5 mt-2\">\n {availableTickers.map(t => (\n <button key={t} onClick={() => setConfigSymbols(p => p.length < MAX_SYMBOLS ? [...p, t] : p)}\n className=\"text-[11px] font-medium text-gray-600 bg-gray-100 hover:bg-gray-200 rounded px-2 py-0.5 transition-colors\">{t}</button>\n ))}\n </div>\n )}\n <p className=\"text-[11px] text-gray-400 mt-2\">Demo prices — static sample data, no live feed.</p>\n </div>\n </WidgetSettingsModal>\n </>\n );\n}\n"]}
@@ -4,6 +4,21 @@ import { W as WindowRegistry } from '../types-BYiS2cc8.js';
4
4
 
5
5
  declare function Browser$1(): react_jsx_runtime.JSX.Element;
6
6
 
7
+ /**
8
+ * Demo filesystem. When a consumer (e.g. the demo app) injects a static tree
9
+ * via `setFilesDemoTree`, Files browses it in-memory — no file server needed.
10
+ * Browse-only: upload / new-folder / rename / delete / trash are hidden while a
11
+ * demo tree is set. Consumers with a real file server never call it, so their
12
+ * behaviour is unchanged.
13
+ */
14
+ interface FilesDemoNode {
15
+ name: string;
16
+ kind: 'file' | 'folder';
17
+ size?: number;
18
+ modifiedAt?: string;
19
+ children?: FilesDemoNode[];
20
+ }
21
+ declare function setFilesDemoTree(tree: FilesDemoNode[] | null): void;
7
22
  declare function openFilesInTrashMode(): void;
8
23
  declare function Files$1(): react_jsx_runtime.JSX.Element;
9
24
 
@@ -57,8 +72,6 @@ declare function Checkers$1(): react_jsx_runtime.JSX.Element;
57
72
 
58
73
  declare function Chess$1(): react_jsx_runtime.JSX.Element;
59
74
 
60
- declare function TodoList$1(): react_jsx_runtime.JSX.Element;
61
-
62
75
  declare function Stock$1(): react_jsx_runtime.JSX.Element;
63
76
 
64
77
  /**
@@ -108,7 +121,6 @@ declare const PomodoroTimer: react.LazyExoticComponent<typeof PomodoroTimer$1>;
108
121
  declare const Notepad: react.LazyExoticComponent<typeof Notepad$1>;
109
122
  declare const WorldClock: react.LazyExoticComponent<typeof WorldClock$1>;
110
123
  declare const Stock: react.LazyExoticComponent<typeof Stock$1>;
111
- declare const TodoList: react.LazyExoticComponent<typeof TodoList$1>;
112
124
  declare const Chess: react.LazyExoticComponent<typeof Chess$1>;
113
125
  declare const Checkers: react.LazyExoticComponent<typeof Checkers$1>;
114
126
  declare const Sudoku: react.LazyExoticComponent<typeof Sudoku$1>;
@@ -125,4 +137,4 @@ declare const documentApps: WindowRegistry;
125
137
  declare const webApps: WindowRegistry;
126
138
  declare const bundledApps: WindowRegistry;
127
139
 
128
- export { Browser, Calculator, Checkers, Chess, CurrencyConverter, Documents, Files, Game2048, Minesweeper, Notepad, type PdfPreviewData, type PdfPreviewHandle, PomodoroTimer, Preview, Spreadsheet, type SpreadsheetPreviewData, type SpreadsheetPreviewHandle, Stock, Sudoku, Tetris, TodoList, Weather, WorldClock, bundledApps, documentApps, gameApps, openFilesInTrashMode, setPdfPreview, setSpreadsheetPreview, utilityApps, webApps };
140
+ export { Browser, Calculator, Checkers, Chess, CurrencyConverter, Documents, Files, type FilesDemoNode, Game2048, Minesweeper, Notepad, type PdfPreviewData, type PdfPreviewHandle, PomodoroTimer, Preview, Spreadsheet, type SpreadsheetPreviewData, type SpreadsheetPreviewHandle, Stock, Sudoku, Tetris, Weather, WorldClock, bundledApps, documentApps, gameApps, openFilesInTrashMode, setFilesDemoTree, setPdfPreview, setSpreadsheetPreview, utilityApps, webApps };
@@ -1,5 +1,6 @@
1
- export { openFilesInTrashMode } from '../chunk-D4ZM3K2S.js';
2
- import '../chunk-OB7T3Q5C.js';
1
+ export { openFilesInTrashMode, setFilesDemoTree } from '../chunk-UXEG2NRI.js';
2
+ import '../chunk-4R4SXMDV.js';
3
+ import '../chunk-VGTEM5RZ.js';
3
4
  export { setPdfPreview } from '../chunk-K354WXU5.js';
4
5
  import '../chunk-KUIPWCTJ.js';
5
6
  import '../chunk-WIJ45SYD.js';
@@ -13,11 +14,10 @@ var Calculator = lazy(() => import('../Calculator-DJYKVGOK.js'));
13
14
  var Spreadsheet = lazy(() => import('../Spreadsheet-TVP3SXT6.js'));
14
15
  var Weather = lazy(() => import('../Weather-WH3URRSL.js'));
15
16
  var CurrencyConverter = lazy(() => import('../CurrencyConverter-ZJWZCO5U.js'));
16
- var PomodoroTimer = lazy(() => import('../PomodoroTimer-SBBPQMYB.js'));
17
- var Notepad = lazy(() => import('../Notepad-AFAUA4KJ.js'));
17
+ var PomodoroTimer = lazy(() => import('../PomodoroTimer-HGPJ5R6V.js'));
18
+ var Notepad = lazy(() => import('../Notepad-6JJ4RT3U.js'));
18
19
  var WorldClock = lazy(() => import('../WorldClock-LJRWPGJW.js'));
19
- var Stock = lazy(() => import('../Stock-ICDNFM7U.js'));
20
- var TodoList = lazy(() => import('../TodoList-26N6ZTLN.js'));
20
+ var Stock = lazy(() => import('../Stock-XLC3LZJV.js'));
21
21
  var Chess = lazy(() => import('../Chess-C5BY45NA.js'));
22
22
  var Checkers = lazy(() => import('../Checkers-MIAHIKJH.js'));
23
23
  var Sudoku = lazy(() => import('../Sudoku-XHLYCEVT.js'));
@@ -26,18 +26,17 @@ var Game2048 = lazy(() => import('../Game2048-3RH3ELRD.js'));
26
26
  var Minesweeper = lazy(() => import('../Minesweeper-YNXOWMIP.js'));
27
27
  var Preview = lazy(() => import('../Preview-KUAQIPBY.js'));
28
28
  var Documents = lazy(() => import('../Documents-2S4QPAI3.js'));
29
- var Files = lazy(() => import('../Files-5N64E375.js'));
29
+ var Files = lazy(() => import('../Files-V4SZKOGZ.js'));
30
30
  var Browser = lazy(() => import('../Browser-M5Y3ENA7.js'));
31
31
  var utilityApps = {
32
32
  "/calculator": { component: Calculator, label: "Calculator", size: "sm", allowPinOnTop: true, utility: true, widget: true, dimensions: [280, 420] },
33
33
  "/spreadsheet": { component: Spreadsheet, label: "Spreadsheets", size: "2xl", appStyle: true, multiInstance: true },
34
- "/notepad": { component: Notepad, label: "Notepad", size: "lg", allowPinOnTop: true },
34
+ "/notepad": { component: Notepad, label: "Notepad", size: "lg", allowPinOnTop: true, flushBody: true },
35
35
  "/weather": { component: Weather, label: "Weather", size: "sm", utility: true, widget: true, autoHeight: true, dimensions: [320, 400] },
36
36
  "/currency": { component: CurrencyConverter, label: "Currency Converter", size: "sm", utility: true, widget: true, autoHeight: true, dimensions: [320, 480] },
37
37
  "/pomodoro": { component: PomodoroTimer, label: "Pomodoro Timer", size: "sm", utility: true, widget: true, dimensions: [320, 600] },
38
38
  "/world-clock": { component: WorldClock, label: "World Clock", size: "sm", utility: true, widget: true, autoHeight: true, dimensions: [320, 480] },
39
- "/stock": { component: Stock, label: "Stocks", size: "sm", utility: true, widget: true, autoHeight: true, dimensions: [320, 360] },
40
- "/todo": { component: TodoList, label: "Todo List", size: "md", dimensions: [560, 640] }
39
+ "/stock": { component: Stock, label: "Stocks", size: "sm", utility: true, widget: true, autoHeight: true, dimensions: [320, 360] }
41
40
  };
42
41
  var gameApps = {
43
42
  "/chess": { component: Chess, label: "Chess", size: "lg", compact: true },
@@ -62,6 +61,6 @@ var bundledApps = {
62
61
  ...webApps
63
62
  };
64
63
 
65
- export { Browser, Calculator, Checkers, Chess, CurrencyConverter, Documents, Files, Game2048, Minesweeper, Notepad, PomodoroTimer, Preview, Spreadsheet, Stock, Sudoku, Tetris, TodoList, Weather, WorldClock, bundledApps, documentApps, gameApps, utilityApps, webApps };
64
+ export { Browser, Calculator, Checkers, Chess, CurrencyConverter, Documents, Files, Game2048, Minesweeper, Notepad, PomodoroTimer, Preview, Spreadsheet, Stock, Sudoku, Tetris, Weather, WorldClock, bundledApps, documentApps, gameApps, utilityApps, webApps };
66
65
  //# sourceMappingURL=index.js.map
67
66
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/apps/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Bundled apps — pre-built window registry entries for the 14 apps that ship\n * with `react-os-shell`. Consumers compose them into their own registry via\n * `createWindowRegistry`:\n *\n * import { createWindowRegistry } from 'react-os-shell';\n * import { bundledApps } from 'react-os-shell/apps';\n * import { erpEntities } from './shell-config';\n *\n * const windows = createWindowRegistry(bundledApps, erpEntities);\n *\n * Subsets are also exported (`utilityApps`, `gameApps`) so a\n * consumer can pick-and-choose without importing every component.\n *\n * NOTE: 2 apps require consumer-supplied persistence (Notepad for stored\n * content, Minesweeper for leaderboard). They're exported individually\n * but absent from `bundledApps` — wire the prefs provider to opt them in.\n * WorldClock uses `useShellPrefs()` so it lives in `bundledApps`; without a\n * consumer-supplied prefs adapter the city list won't survive a reload.\n */\nimport { lazy } from 'react';\nimport type { WindowRegistry } from '../windowRegistry/types';\n\n// ── Utility apps ──\nconst Calculator = lazy(() => import('./Calculator'));\nconst Spreadsheet = lazy(() => import('./Spreadsheet'));\nconst Weather = lazy(() => import('./Weather'));\nconst CurrencyConverter = lazy(() => import('./CurrencyConverter'));\nconst PomodoroTimer = lazy(() => import('./PomodoroTimer'));\nconst Notepad = lazy(() => import('./Notepad'));\nconst WorldClock = lazy(() => import('./WorldClock'));\nconst Stock = lazy(() => import('./Stock'));\nconst TodoList = lazy(() => import('./TodoList'));\n\n// ── Games ──\nconst Chess = lazy(() => import('./Chess'));\nconst Checkers = lazy(() => import('./Checkers'));\nconst Sudoku = lazy(() => import('./Sudoku'));\nconst Tetris = lazy(() => import('./Tetris'));\nconst Game2048 = lazy(() => import('./Game2048'));\nconst Minesweeper = lazy(() => import('./Minesweeper'));\n\n// ── Document apps ──\nconst Preview = lazy(() => import('./Preview'));\nconst Documents = lazy(() => import('./Documents'));\nconst Files = lazy(() => import('./Files'));\n\n// ── Web ──\nconst Browser = lazy(() => import('./Browser'));\n\nexport const utilityApps: WindowRegistry = {\n '/calculator': { component: Calculator, label: 'Calculator', size: 'sm', allowPinOnTop: true, utility: true, widget: true, dimensions: [280, 420] },\n '/spreadsheet': { component: Spreadsheet, label: 'Spreadsheets', size: '2xl', appStyle: true, multiInstance: true },\n '/notepad': { component: Notepad, label: 'Notepad', size: 'lg', allowPinOnTop: true },\n '/weather': { component: Weather, label: 'Weather', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 400] },\n '/currency': { component: CurrencyConverter, label: 'Currency Converter', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 480] },\n '/pomodoro': { component: PomodoroTimer, label: 'Pomodoro Timer', size: 'sm', utility: true, widget: true, dimensions: [320, 600] },\n '/world-clock': { component: WorldClock, label: 'World Clock', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 480] },\n '/stock': { component: Stock, label: 'Stocks', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 360] },\n '/todo': { component: TodoList, label: 'Todo List', size: 'md', dimensions: [560, 640] },\n};\n\nexport const gameApps: WindowRegistry = {\n '/chess': { component: Chess, label: 'Chess', size: 'lg', compact: true },\n '/checkers': { component: Checkers, label: 'Checkers', size: 'lg', compact: true },\n '/sudoku': { component: Sudoku, label: 'Sudoku', size: 'sm', compact: true, dimensions: [360, 535] },\n '/tetris': { component: Tetris, label: 'Tetris', size: 'md', compact: true, dimensions: [452, 618] },\n '/2048': { component: Game2048, label: '2048', size: 'sm', compact: true },\n '/minesweeper': { component: Minesweeper, label: 'Minesweeper', size: 'sm', compact: true },\n};\n\nexport const documentApps: WindowRegistry = {\n '/preview': { component: Preview, label: 'Preview', size: '2xl', appStyle: true, multiInstance: true },\n '/documents': { component: Documents, label: 'Documents', size: 'xl', appStyle: true, multiInstance: true },\n '/files': { component: Files, label: 'Files', size: 'xl', appStyle: true },\n};\n\nexport const webApps: WindowRegistry = {\n '/browser': { component: Browser, label: 'Browser', size: '2xl', appStyle: true, multiInstance: true },\n};\n\nexport const bundledApps: WindowRegistry = {\n ...utilityApps,\n ...gameApps,\n ...documentApps,\n ...webApps,\n};\n\nexport {\n Calculator,\n Spreadsheet,\n Notepad,\n Weather,\n CurrencyConverter,\n PomodoroTimer,\n WorldClock,\n Stock,\n TodoList,\n Chess,\n Checkers,\n Sudoku,\n Tetris,\n Game2048,\n Minesweeper,\n Preview,\n Documents,\n Files,\n Browser,\n};\n\nexport { setPdfPreview } from './Preview';\nexport { setSpreadsheetPreview } from './Spreadsheet';\nexport { openFilesInTrashMode } from './Files';\nexport type { PdfPreviewData, PdfPreviewHandle } from './Preview';\nexport type { SpreadsheetPreviewData, SpreadsheetPreviewHandle } from './Spreadsheet';\n"]}
1
+ {"version":3,"sources":["../../src/apps/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Bundled apps — pre-built window registry entries for the 14 apps that ship\n * with `react-os-shell`. Consumers compose them into their own registry via\n * `createWindowRegistry`:\n *\n * import { createWindowRegistry } from 'react-os-shell';\n * import { bundledApps } from 'react-os-shell/apps';\n * import { erpEntities } from './shell-config';\n *\n * const windows = createWindowRegistry(bundledApps, erpEntities);\n *\n * Subsets are also exported (`utilityApps`, `gameApps`) so a\n * consumer can pick-and-choose without importing every component.\n *\n * NOTE: 2 apps require consumer-supplied persistence (Notepad for stored\n * content, Minesweeper for leaderboard). They're exported individually\n * but absent from `bundledApps` — wire the prefs provider to opt them in.\n * WorldClock uses `useShellPrefs()` so it lives in `bundledApps`; without a\n * consumer-supplied prefs adapter the city list won't survive a reload.\n */\nimport { lazy } from 'react';\nimport type { WindowRegistry } from '../windowRegistry/types';\n\n// ── Utility apps ──\nconst Calculator = lazy(() => import('./Calculator'));\nconst Spreadsheet = lazy(() => import('./Spreadsheet'));\nconst Weather = lazy(() => import('./Weather'));\nconst CurrencyConverter = lazy(() => import('./CurrencyConverter'));\nconst PomodoroTimer = lazy(() => import('./PomodoroTimer'));\nconst Notepad = lazy(() => import('./Notepad'));\nconst WorldClock = lazy(() => import('./WorldClock'));\nconst Stock = lazy(() => import('./Stock'));\n\n// ── Games ──\nconst Chess = lazy(() => import('./Chess'));\nconst Checkers = lazy(() => import('./Checkers'));\nconst Sudoku = lazy(() => import('./Sudoku'));\nconst Tetris = lazy(() => import('./Tetris'));\nconst Game2048 = lazy(() => import('./Game2048'));\nconst Minesweeper = lazy(() => import('./Minesweeper'));\n\n// ── Document apps ──\nconst Preview = lazy(() => import('./Preview'));\nconst Documents = lazy(() => import('./Documents'));\nconst Files = lazy(() => import('./Files'));\n\n// ── Web ──\nconst Browser = lazy(() => import('./Browser'));\n\nexport const utilityApps: WindowRegistry = {\n '/calculator': { component: Calculator, label: 'Calculator', size: 'sm', allowPinOnTop: true, utility: true, widget: true, dimensions: [280, 420] },\n '/spreadsheet': { component: Spreadsheet, label: 'Spreadsheets', size: '2xl', appStyle: true, multiInstance: true },\n '/notepad': { component: Notepad, label: 'Notepad', size: 'lg', allowPinOnTop: true, flushBody: true },\n '/weather': { component: Weather, label: 'Weather', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 400] },\n '/currency': { component: CurrencyConverter, label: 'Currency Converter', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 480] },\n '/pomodoro': { component: PomodoroTimer, label: 'Pomodoro Timer', size: 'sm', utility: true, widget: true, dimensions: [320, 600] },\n '/world-clock': { component: WorldClock, label: 'World Clock', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 480] },\n '/stock': { component: Stock, label: 'Stocks', size: 'sm', utility: true, widget: true, autoHeight: true, dimensions: [320, 360] },\n};\n\nexport const gameApps: WindowRegistry = {\n '/chess': { component: Chess, label: 'Chess', size: 'lg', compact: true },\n '/checkers': { component: Checkers, label: 'Checkers', size: 'lg', compact: true },\n '/sudoku': { component: Sudoku, label: 'Sudoku', size: 'sm', compact: true, dimensions: [360, 535] },\n '/tetris': { component: Tetris, label: 'Tetris', size: 'md', compact: true, dimensions: [452, 618] },\n '/2048': { component: Game2048, label: '2048', size: 'sm', compact: true },\n '/minesweeper': { component: Minesweeper, label: 'Minesweeper', size: 'sm', compact: true },\n};\n\nexport const documentApps: WindowRegistry = {\n '/preview': { component: Preview, label: 'Preview', size: '2xl', appStyle: true, multiInstance: true },\n '/documents': { component: Documents, label: 'Documents', size: 'xl', appStyle: true, multiInstance: true },\n '/files': { component: Files, label: 'Files', size: 'xl', appStyle: true },\n};\n\nexport const webApps: WindowRegistry = {\n '/browser': { component: Browser, label: 'Browser', size: '2xl', appStyle: true, multiInstance: true },\n};\n\nexport const bundledApps: WindowRegistry = {\n ...utilityApps,\n ...gameApps,\n ...documentApps,\n ...webApps,\n};\n\nexport {\n Calculator,\n Spreadsheet,\n Notepad,\n Weather,\n CurrencyConverter,\n PomodoroTimer,\n WorldClock,\n Stock,\n Chess,\n Checkers,\n Sudoku,\n Tetris,\n Game2048,\n Minesweeper,\n Preview,\n Documents,\n Files,\n Browser,\n};\n\nexport { setPdfPreview } from './Preview';\nexport { setSpreadsheetPreview } from './Spreadsheet';\nexport { openFilesInTrashMode, setFilesDemoTree } from './Files';\nexport type { FilesDemoNode } from './Files';\nexport type { PdfPreviewData, PdfPreviewHandle } from './Preview';\nexport type { SpreadsheetPreviewData, SpreadsheetPreviewHandle } from './Spreadsheet';\n"]}
@@ -0,0 +1,98 @@
1
+ import { setPdfPreview } from './chunk-K354WXU5.js';
2
+ import { toast_default } from './chunk-WIJ45SYD.js';
3
+ import { setSpreadsheetPreview } from './chunk-6DDVFASF.js';
4
+ import { Fragment } from 'react';
5
+ import { jsx, jsxs } from 'react/jsx-runtime';
6
+
7
+ var DEFAULT_SEPARATOR = /* @__PURE__ */ jsx("svg", { className: "h-3.5 w-3.5 shrink-0 text-gray-300", viewBox: "0 0 20 20", fill: "currentColor", "aria-hidden": true, children: /* @__PURE__ */ jsx(
8
+ "path",
9
+ {
10
+ fillRule: "evenodd",
11
+ 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",
12
+ clipRule: "evenodd"
13
+ }
14
+ ) });
15
+ function Breadcrumbs({ items, separator, maxItems = 0, className }) {
16
+ if (items.length === 0) return null;
17
+ const sep = separator ?? DEFAULT_SEPARATOR;
18
+ const crumbs = [];
19
+ const lastIndex = items.length - 1;
20
+ if (maxItems > 0 && items.length > maxItems && maxItems >= 2) {
21
+ const tailCount = Math.max(1, maxItems - 1);
22
+ crumbs.push({ kind: "item", item: items[0], isLast: false });
23
+ crumbs.push({ kind: "ellipsis" });
24
+ for (let i = items.length - tailCount; i <= lastIndex; i++) {
25
+ crumbs.push({ kind: "item", item: items[i], isLast: i === lastIndex });
26
+ }
27
+ } else {
28
+ items.forEach((item, i) => crumbs.push({ kind: "item", item, isLast: i === lastIndex }));
29
+ }
30
+ return /* @__PURE__ */ jsx("nav", { "aria-label": "Breadcrumb", className: `min-w-0 ${className ?? ""}`, children: /* @__PURE__ */ jsx("ol", { className: "flex items-center gap-1.5 text-sm", children: crumbs.map((crumb, i) => /* @__PURE__ */ jsxs(Fragment, { children: [
31
+ i > 0 && /* @__PURE__ */ jsx("li", { "aria-hidden": true, className: "flex items-center", children: sep }),
32
+ /* @__PURE__ */ jsx("li", { className: "flex min-w-0 items-center", children: crumb.kind === "ellipsis" ? /* @__PURE__ */ jsx("span", { className: "px-0.5 text-gray-400 select-none", "aria-label": "Hidden crumbs", children: "\u2026" }) : crumb.isLast || !crumb.item.onClick ? /* @__PURE__ */ jsxs(
33
+ "span",
34
+ {
35
+ "aria-current": crumb.isLast ? "page" : void 0,
36
+ className: `inline-flex min-w-0 items-center gap-1 truncate ${crumb.isLast ? "font-medium text-gray-900" : "text-gray-500"}`,
37
+ children: [
38
+ crumb.item.icon,
39
+ /* @__PURE__ */ jsx("span", { className: "truncate", children: crumb.item.label })
40
+ ]
41
+ }
42
+ ) : /* @__PURE__ */ jsxs(
43
+ "button",
44
+ {
45
+ type: "button",
46
+ onClick: crumb.item.onClick,
47
+ 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",
48
+ children: [
49
+ crumb.item.icon,
50
+ /* @__PURE__ */ jsx("span", { className: "truncate", children: crumb.item.label })
51
+ ]
52
+ }
53
+ ) })
54
+ ] }, i)) }) });
55
+ }
56
+
57
+ // src/utils/openPreviewFile.ts
58
+ var PREVIEW_OPENED_EVENT = "react-os-shell:preview-opened";
59
+ function getServer() {
60
+ const override = typeof window !== "undefined" && window.__REACT_OS_SHELL_FILE_SERVER__;
61
+ return (override || "http://localhost:4000").replace(/\/$/, "");
62
+ }
63
+ async function openPreviewFile(opts) {
64
+ const { filePath, filename, kind, onStaged } = opts;
65
+ try {
66
+ const res = await fetch(
67
+ `${getServer()}/api/file?path=${encodeURIComponent(filePath)}`,
68
+ { credentials: "include" }
69
+ );
70
+ if (!res.ok) {
71
+ toast_default.error(`Download failed (${res.status})`);
72
+ return false;
73
+ }
74
+ const blob = await res.blob();
75
+ if (kind === "csv") {
76
+ const text = await blob.text();
77
+ setSpreadsheetPreview({ csv: text, filename });
78
+ onStaged?.("/spreadsheet");
79
+ } else {
80
+ const url = URL.createObjectURL(blob);
81
+ setPdfPreview({ url, filename, kind });
82
+ onStaged?.("/preview");
83
+ }
84
+ if (typeof window !== "undefined") {
85
+ window.dispatchEvent(new CustomEvent(PREVIEW_OPENED_EVENT, {
86
+ detail: { filePath, filename, kind }
87
+ }));
88
+ }
89
+ return true;
90
+ } catch (e) {
91
+ toast_default.error(e?.message || "Open failed");
92
+ return false;
93
+ }
94
+ }
95
+
96
+ export { Breadcrumbs, PREVIEW_OPENED_EVENT, openPreviewFile };
97
+ //# sourceMappingURL=chunk-4R4SXMDV.js.map
98
+ //# sourceMappingURL=chunk-4R4SXMDV.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/shell/Breadcrumbs.tsx","../src/utils/openPreviewFile.ts"],"names":[],"mappings":"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","file":"chunk-4R4SXMDV.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"]}
@@ -1,4 +1,6 @@
1
+ import { useShellPrefs } from './chunk-36VM54SC.js';
1
2
  import { toast_default } from './chunk-WIJ45SYD.js';
3
+ import { useSyncExternalStore, useCallback, useMemo } from 'react';
2
4
 
3
5
  // src/shell/pomodoroStore.ts
4
6
  var ALARM_OPTIONS = [
@@ -374,7 +376,174 @@ function previewFocusSound(type, volume) {
374
376
  } catch {
375
377
  }
376
378
  }
379
+ var POMODORO_LEGACY_KEY = "pomodoro_tasks";
380
+ var POMODORO_LEGACY_MIGRATED_FLAG = "pomodoro_tasks_migrated";
381
+ function uid() {
382
+ return Math.random().toString(36).slice(2, 11);
383
+ }
384
+ function nowIso() {
385
+ return (/* @__PURE__ */ new Date()).toISOString();
386
+ }
387
+ var provider = null;
388
+ var cache = [];
389
+ var loaded = false;
390
+ var inflight = null;
391
+ var listeners = /* @__PURE__ */ new Set();
392
+ function emit() {
393
+ for (const l of listeners) l();
394
+ }
395
+ function setCache(next) {
396
+ cache = next;
397
+ emit();
398
+ }
399
+ function refresh() {
400
+ const p = provider;
401
+ if (!p) {
402
+ cache = [];
403
+ loaded = true;
404
+ emit();
405
+ return Promise.resolve();
406
+ }
407
+ inflight = p.list().then((rows) => {
408
+ if (provider === p) {
409
+ cache = rows;
410
+ loaded = true;
411
+ emit();
412
+ }
413
+ }).catch(() => {
414
+ if (provider === p) {
415
+ loaded = true;
416
+ emit();
417
+ }
418
+ }).finally(() => {
419
+ inflight = null;
420
+ });
421
+ return inflight;
422
+ }
423
+ function setShellTodoProvider(next) {
424
+ provider = next;
425
+ cache = [];
426
+ loaded = false;
427
+ if (next) refresh();
428
+ else emit();
429
+ }
430
+ function hasShellTodoProvider() {
431
+ return provider !== null;
432
+ }
433
+ function subscribe(cb) {
434
+ listeners.add(cb);
435
+ if (provider && !loaded && !inflight) refresh();
436
+ return () => {
437
+ listeners.delete(cb);
438
+ };
439
+ }
440
+ function getSnapshot() {
441
+ return cache;
442
+ }
443
+ function usePrefsTodoTasks() {
444
+ const { prefs, save } = useShellPrefs();
445
+ const tasks = useMemo(() => {
446
+ const raw = prefs.todo_tasks;
447
+ return Array.isArray(raw) ? raw : [];
448
+ }, [prefs.todo_tasks]);
449
+ const writeTasks = useCallback((next) => {
450
+ save({ todo_tasks: next });
451
+ }, [save]);
452
+ const addTask = useCallback((input) => {
453
+ const id = uid();
454
+ const ts = nowIso();
455
+ const task = { done: false, ...input, name: input.name.trim(), id, createdAt: ts, updatedAt: ts };
456
+ writeTasks([...tasks, task]);
457
+ return id;
458
+ }, [tasks, writeTasks]);
459
+ const updateTask = useCallback((id, patch) => {
460
+ writeTasks(tasks.map((t) => t.id === id ? { ...t, ...patch, id: t.id, updatedAt: nowIso() } : t));
461
+ }, [tasks, writeTasks]);
462
+ const removeTask = useCallback((id) => {
463
+ writeTasks(tasks.filter((t) => t.id !== id));
464
+ }, [tasks, writeTasks]);
465
+ const toggleDone = useCallback((id) => {
466
+ const target = tasks.find((t) => t.id === id);
467
+ if (target) updateTask(id, { done: !target.done });
468
+ }, [tasks, updateTask]);
469
+ return { tasks, addTask, updateTask, removeTask, toggleDone, setAllTasks: writeTasks };
470
+ }
471
+ function useTodoTasks() {
472
+ const prefsImpl = usePrefsTodoTasks();
473
+ const providerTasks = useSyncExternalStore(subscribe, getSnapshot, getSnapshot);
474
+ const addTask = useCallback((input) => {
475
+ const p = provider;
476
+ if (!p) return prefsImpl.addTask(input);
477
+ const tempId = "tmp-" + uid();
478
+ const ts = nowIso();
479
+ const optimistic = { done: false, ...input, name: input.name.trim(), id: tempId, createdAt: ts, updatedAt: ts };
480
+ setCache([...cache, optimistic]);
481
+ p.create({ ...input, name: input.name.trim() }).then((saved) => setCache(cache.map((t) => t.id === tempId ? { ...optimistic, ...saved } : t))).catch(() => setCache(cache.filter((t) => t.id !== tempId)));
482
+ return tempId;
483
+ }, [prefsImpl]);
484
+ const updateTask = useCallback((id, patch) => {
485
+ const p = provider;
486
+ if (!p) return prefsImpl.updateTask(id, patch);
487
+ setCache(cache.map((t) => t.id === id ? { ...t, ...patch, id, updatedAt: nowIso() } : t));
488
+ p.update(id, patch).catch(() => {
489
+ refresh();
490
+ });
491
+ }, [prefsImpl]);
492
+ const removeTask = useCallback((id) => {
493
+ const p = provider;
494
+ if (!p) return prefsImpl.removeTask(id);
495
+ const prev = cache;
496
+ setCache(cache.filter((t) => t.id !== id));
497
+ p.remove(id).catch(() => setCache(prev));
498
+ }, [prefsImpl]);
499
+ const toggleDone = useCallback((id) => {
500
+ if (!provider) return prefsImpl.toggleDone(id);
501
+ const target = cache.find((t) => t.id === id);
502
+ if (target) updateTask(id, { done: !target.done });
503
+ }, [prefsImpl, updateTask]);
504
+ const setAllTasks = useCallback((next) => {
505
+ if (!provider) prefsImpl.setAllTasks(next);
506
+ }, [prefsImpl]);
507
+ if (!hasShellTodoProvider()) return prefsImpl;
508
+ return { tasks: providerTasks, addTask, updateTask, removeTask, toggleDone, setAllTasks };
509
+ }
510
+ function migratePomodoroTasksOnce(currentTasks, setAllTasks) {
511
+ try {
512
+ if (localStorage.getItem(POMODORO_LEGACY_MIGRATED_FLAG)) return;
513
+ const raw = localStorage.getItem(POMODORO_LEGACY_KEY);
514
+ if (!raw) {
515
+ localStorage.setItem(POMODORO_LEGACY_MIGRATED_FLAG, "1");
516
+ return;
517
+ }
518
+ const legacy = JSON.parse(raw);
519
+ if (!Array.isArray(legacy) || legacy.length === 0) {
520
+ localStorage.removeItem(POMODORO_LEGACY_KEY);
521
+ localStorage.setItem(POMODORO_LEGACY_MIGRATED_FLAG, "1");
522
+ return;
523
+ }
524
+ const ts = nowIso();
525
+ const migrated = legacy.filter((t) => typeof t?.name === "string").map((t) => ({
526
+ id: t.id || uid(),
527
+ name: String(t.name),
528
+ done: !!t.done,
529
+ estimated: typeof t.estimated === "number" ? t.estimated : void 0,
530
+ completed: typeof t.completed === "number" ? t.completed : void 0,
531
+ createdAt: ts,
532
+ updatedAt: ts
533
+ }));
534
+ const existingIds = new Set(currentTasks.map((t) => t.id));
535
+ const additions = migrated.filter((t) => !existingIds.has(t.id));
536
+ setAllTasks([...additions, ...currentTasks]);
537
+ localStorage.removeItem(POMODORO_LEGACY_KEY);
538
+ localStorage.setItem(POMODORO_LEGACY_MIGRATED_FLAG, "1");
539
+ } catch {
540
+ try {
541
+ localStorage.setItem(POMODORO_LEGACY_MIGRATED_FLAG, "1");
542
+ } catch {
543
+ }
544
+ }
545
+ }
377
546
 
378
- export { ALARM_OPTIONS, FOCUS_SOUND_OPTIONS, getPomoSnapshot, playAlarm, pomoPause, pomoStart, pomoSwitchMode, previewFocusSound, setPomoAlarm, setPomoAlarmConfig, setPomoBehaviour, setPomoDurations, setPomoFocusSound, subscribePomo };
379
- //# sourceMappingURL=chunk-MK3HLUO4.js.map
380
- //# sourceMappingURL=chunk-MK3HLUO4.js.map
547
+ export { ALARM_OPTIONS, FOCUS_SOUND_OPTIONS, getPomoSnapshot, migratePomodoroTasksOnce, playAlarm, pomoPause, pomoStart, pomoSwitchMode, previewFocusSound, setPomoAlarm, setPomoAlarmConfig, setPomoBehaviour, setPomoDurations, setPomoFocusSound, setShellTodoProvider, subscribePomo, useTodoTasks };
548
+ //# sourceMappingURL=chunk-5X5LQNOX.js.map
549
+ //# sourceMappingURL=chunk-5X5LQNOX.js.map