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.
- package/dist/Files-V4SZKOGZ.js +12 -0
- package/dist/{Files-5N64E375.js.map → Files-V4SZKOGZ.js.map} +1 -1
- package/dist/{Notepad-AFAUA4KJ.js → Notepad-6JJ4RT3U.js} +110 -101
- package/dist/Notepad-6JJ4RT3U.js.map +1 -0
- package/dist/{PomodoroTimer-SBBPQMYB.js → PomodoroTimer-HGPJ5R6V.js} +3 -4
- package/dist/PomodoroTimer-HGPJ5R6V.js.map +1 -0
- package/dist/Stock-XLC3LZJV.js +152 -0
- package/dist/Stock-XLC3LZJV.js.map +1 -0
- package/dist/apps/index.d.ts +16 -4
- package/dist/apps/index.js +10 -11
- package/dist/apps/index.js.map +1 -1
- package/dist/chunk-4R4SXMDV.js +98 -0
- package/dist/chunk-4R4SXMDV.js.map +1 -0
- package/dist/{chunk-MK3HLUO4.js → chunk-5X5LQNOX.js} +172 -3
- package/dist/chunk-5X5LQNOX.js.map +1 -0
- package/dist/chunk-UXEG2NRI.js +682 -0
- package/dist/chunk-UXEG2NRI.js.map +1 -0
- package/dist/chunk-VGTEM5RZ.js +89 -0
- package/dist/chunk-VGTEM5RZ.js.map +1 -0
- package/dist/index.d.ts +9 -1
- package/dist/index.js +16 -142
- package/dist/index.js.map +1 -1
- package/dist/styles.css +26 -2
- package/package.json +1 -1
- package/dist/Files-5N64E375.js +0 -11
- package/dist/Notepad-AFAUA4KJ.js.map +0 -1
- package/dist/PomodoroTimer-SBBPQMYB.js.map +0 -1
- package/dist/Stock-ICDNFM7U.js +0 -234
- package/dist/Stock-ICDNFM7U.js.map +0 -1
- package/dist/TodoList-26N6ZTLN.js +0 -309
- package/dist/TodoList-26N6ZTLN.js.map +0 -1
- package/dist/chunk-D4ZM3K2S.js +0 -605
- package/dist/chunk-D4ZM3K2S.js.map +0 -1
- package/dist/chunk-MK3HLUO4.js.map +0 -1
- package/dist/chunk-OB7T3Q5C.js +0 -46
- package/dist/chunk-OB7T3Q5C.js.map +0 -1
- package/dist/chunk-QTJ2CHJX.js +0 -174
- package/dist/chunk-QTJ2CHJX.js.map +0 -1
package/dist/styles.css
CHANGED
|
@@ -115,12 +115,36 @@
|
|
|
115
115
|
In dark mode swap the near-white bg for a dark blue tint and the dark
|
|
116
116
|
blue text for a light shade so highlighted items stay readable. */
|
|
117
117
|
[data-theme="dark"] .bg-blue-50,
|
|
118
|
-
[data-theme="dark"] .hover\:bg-blue-50:hover { background-color: rgba(59, 130, 246, 0.
|
|
118
|
+
[data-theme="dark"] .hover\:bg-blue-50:hover { background-color: rgba(59, 130, 246, 0.26) !important; }
|
|
119
119
|
[data-theme="dark"] .text-blue-700,
|
|
120
|
-
[data-theme="dark"] .hover\:text-blue-700:hover { color: #
|
|
120
|
+
[data-theme="dark"] .hover\:text-blue-700:hover { color: #bfdbfe !important; }
|
|
121
121
|
[data-theme="dark"] .text-blue-600 { color: #93c5fd !important; }
|
|
122
122
|
[data-theme="dark"] .text-blue-800 { color: #bfdbfe !important; }
|
|
123
123
|
|
|
124
|
+
/* Bare *-100 highlight backgrounds + *-700 text (status pills, badges, avatars,
|
|
125
|
+
dialog icons) have no dark override otherwise, so they keep their bright
|
|
126
|
+
light-mode value and wash out the text on top. Mute the background to a low
|
|
127
|
+
tint and lighten the text so highlighted content stays readable in dark mode. */
|
|
128
|
+
[data-theme="dark"] .bg-blue-100 { background-color: rgba(59, 130, 246, 0.22) !important; }
|
|
129
|
+
[data-theme="dark"] .bg-green-100 { background-color: rgba(34, 197, 94, 0.18) !important; }
|
|
130
|
+
[data-theme="dark"] .bg-emerald-100 { background-color: rgba(16, 185, 129, 0.18) !important; }
|
|
131
|
+
[data-theme="dark"] .bg-sky-100 { background-color: rgba(14, 165, 233, 0.18) !important; }
|
|
132
|
+
[data-theme="dark"] .bg-amber-100 { background-color: rgba(245, 158, 11, 0.18) !important; }
|
|
133
|
+
[data-theme="dark"] .bg-yellow-100 { background-color: rgba(234, 179, 8, 0.18) !important; }
|
|
134
|
+
[data-theme="dark"] .bg-orange-100 { background-color: rgba(249, 115, 22, 0.18) !important; }
|
|
135
|
+
[data-theme="dark"] .bg-red-100 { background-color: rgba(239, 68, 68, 0.18) !important; }
|
|
136
|
+
[data-theme="dark"] .bg-indigo-100 { background-color: rgba(99, 102, 241, 0.18) !important; }
|
|
137
|
+
[data-theme="dark"] .bg-purple-100 { background-color: rgba(168, 85, 247, 0.18) !important; }
|
|
138
|
+
[data-theme="dark"] .text-green-700 { color: #86efac !important; }
|
|
139
|
+
[data-theme="dark"] .text-emerald-700 { color: #6ee7b7 !important; }
|
|
140
|
+
[data-theme="dark"] .text-sky-700 { color: #7dd3fc !important; }
|
|
141
|
+
[data-theme="dark"] .text-amber-700 { color: #fcd34d !important; }
|
|
142
|
+
[data-theme="dark"] .text-yellow-700 { color: #fde047 !important; }
|
|
143
|
+
[data-theme="dark"] .text-orange-700 { color: #fdba74 !important; }
|
|
144
|
+
[data-theme="dark"] .text-red-700 { color: #fca5a5 !important; }
|
|
145
|
+
[data-theme="dark"] .text-indigo-700 { color: #a5b4fc !important; }
|
|
146
|
+
[data-theme="dark"] .text-purple-700 { color: #d8b4fe !important; }
|
|
147
|
+
|
|
124
148
|
/* Brighten taskbar button text so it stays readable on dark glass.
|
|
125
149
|
The default override (#9399b2) is too dim for short button labels. */
|
|
126
150
|
[data-theme="dark"] .text-gray-600 { color: #a6adc8 !important; }
|
package/package.json
CHANGED
package/dist/Files-5N64E375.js
DELETED
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
export { Files as default, openFilesInTrashMode } from './chunk-D4ZM3K2S.js';
|
|
2
|
-
import './chunk-OB7T3Q5C.js';
|
|
3
|
-
import './chunk-K354WXU5.js';
|
|
4
|
-
import './chunk-KUIPWCTJ.js';
|
|
5
|
-
import './chunk-WIJ45SYD.js';
|
|
6
|
-
import './chunk-6DDVFASF.js';
|
|
7
|
-
import './chunk-4RXDOSKZ.js';
|
|
8
|
-
import './chunk-UBN4IUDE.js';
|
|
9
|
-
import './chunk-ZF6AYO4G.js';
|
|
10
|
-
//# sourceMappingURL=Files-5N64E375.js.map
|
|
11
|
-
//# sourceMappingURL=Files-5N64E375.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/api/auth.ts","../src/apps/Notepad.tsx"],"names":[],"mappings":"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","file":"Notepad-AFAUA4KJ.js","sourcesContent":["/** User-profile API — calls below proxy to the consumer-supplied apiClient\n * via setShellApiClient(). Prefs reads/writes go through <ShellPrefsProvider>. */\nimport apiClient from './client';\nexport const getMe = () => apiClient.get('/auth/me/').then((r: any) => r.data);\nexport const updateMe = (patch: any) => apiClient.patch('/auth/me/', patch).then((r: any) => r.data);\nexport const getNumberingConfigs = () => apiClient.get('/auth/numbering-configs/').then((r: any) => r.data?.results ?? r.data ?? []);\n","import { useState, useEffect, useRef, useCallback, type ReactNode } from 'react';\nimport { useQuery } from '@tanstack/react-query';\nimport { getNumberingConfigs } from '../api/auth';\nimport apiClient from '../api/client';\nimport toast from '../shell/toast';\nimport { useWindowManager } from '../shell/WindowManager';\nimport { useShellPrefs } from '../shell/ShellPrefs';\n\ninterface Note {\n id: string;\n title: string;\n content: string;\n color: string;\n sticky: boolean; // pinned to desktop\n updated_at: string;\n}\n\nconst COLORS = [\n { key: 'yellow', bg: 'bg-yellow-100', border: 'border-yellow-300', text: 'text-yellow-900', dot: 'bg-yellow-400' },\n { key: 'blue', bg: 'bg-blue-100', border: 'border-blue-300', text: 'text-blue-900', dot: 'bg-blue-400' },\n { key: 'green', bg: 'bg-green-100', border: 'border-green-300', text: 'text-green-900', dot: 'bg-green-400' },\n { key: 'pink', bg: 'bg-pink-100', border: 'border-pink-300', text: 'text-pink-900', dot: 'bg-pink-400' },\n { key: 'purple', bg: 'bg-purple-100', border: 'border-purple-300', text: 'text-purple-900', dot: 'bg-purple-400' },\n { key: 'orange', bg: 'bg-orange-100', border: 'border-orange-300', text: 'text-orange-900', dot: 'bg-orange-400' },\n { key: 'white', bg: 'bg-white', border: 'border-gray-300', text: 'text-gray-900', dot: 'bg-gray-300' },\n];\n\nfunction getColor(key: string) {\n return COLORS.find(c => c.key === key) || COLORS[0];\n}\n\nfunction newNote(): Note {\n return {\n id: `note-${Date.now()}-${Math.random().toString(36).slice(2, 6)}`,\n title: '',\n content: '',\n color: 'yellow',\n sticky: false,\n updated_at: new Date().toISOString(),\n };\n}\n\n// ── Entity reference mapping ──\n// Maps entity_type from NumberingConfig to modal registry entityType and search endpoint\nconst ENTITY_TYPE_MAP: Record<string, { entityType: string; endpoint: string }> = {\n sales_order: { entityType: 'order', endpoint: '/orders/sales-orders/' },\n purchase_order: { entityType: 'purchase_order', endpoint: '/purchase-orders/' },\n invoice: { entityType: 'invoice', endpoint: '/invoicing/invoices/' },\n vendor_invoice: { entityType: 'vendor_invoice', endpoint: '/invoicing/vendor-invoices/' },\n shipment: { entityType: 'shipment', endpoint: '/shipments/delivery-notes/' },\n receipt: { entityType: 'payment', endpoint: '/invoicing/payments/' },\n vendor_payment: { entityType: 'vendor_payment', endpoint: '/invoicing/vendor-payments/' },\n vendor_price_sheet: { entityType: 'vendor_price_sheet', endpoint: '/pricing/manufacturer-price-sheets/' },\n client_price_sheet: { entityType: 'price_sheet', endpoint: '/pricing/price-sheets/' },\n qc_report: { entityType: 'qc_report', endpoint: '/qc-reports/' },\n warranty_claim: { entityType: 'warranty_claim', endpoint: '/warranty-claims/claims/' },\n vendor_shipment: { entityType: 'vendor_shipment', endpoint: '/shipments/goods-receipts/' },\n};\n\n// Match checkboxes: [] or [x] or [X]\nconst CHECKBOX_REGEX = /\\[([ xX]?)\\]/g;\n\nexport default function Notepad() {\n const { openEntity } = useWindowManager();\n const { prefs, save } = useShellPrefs();\n\n // Fetch numbering configs to build dynamic prefix map (optional — only used\n // for entity-reference autolinking; safe to fail when no apiClient is wired).\n const { data: numberingConfigs } = useQuery({\n queryKey: ['numbering-configs'],\n queryFn: () => getNumberingConfigs(),\n retry: false,\n });\n\n // Build prefix → { entityType, endpoint } map from DB configs\n const prefixMap = useRef<Record<string, { entityType: string; endpoint: string; prefix: string }>>({});\n useEffect(() => {\n if (!numberingConfigs) return;\n const map: Record<string, { entityType: string; endpoint: string; prefix: string }> = {};\n for (const cfg of numberingConfigs) {\n const mapping = ENTITY_TYPE_MAP[cfg.entity_type];\n if (!mapping) continue;\n // prefix from DB is like \"SO#\" — strip the # to get \"SO\"\n const rawPrefix = (cfg.prefix || '').replace('#', '').toUpperCase();\n if (rawPrefix) {\n map[rawPrefix] = { ...mapping, prefix: cfg.prefix };\n }\n // Also register alt_prefix if set\n if (cfg.alt_prefix) {\n const altRaw = cfg.alt_prefix.replace('#', '').toUpperCase();\n if (altRaw) map[altRaw] = { ...mapping, prefix: cfg.alt_prefix };\n }\n }\n prefixMap.current = map;\n }, [numberingConfigs]);\n\n const notes: Note[] = prefs.notepad_notes || [];\n const [selectedId, setSelectedId] = useState<string | null>(null);\n const [editTitle, setEditTitle] = useState('');\n const [editContent, setEditContent] = useState('');\n const [editColor, setEditColor] = useState('yellow');\n const [dirty, setDirty] = useState(false);\n const [editing, setEditing] = useState(false);\n const textareaRef = useRef<HTMLTextAreaElement>(null);\n const saveTimerRef = useRef<ReturnType<typeof setTimeout>>();\n\n const selected = notes.find(n => n.id === selectedId);\n\n // Select first note on load\n useEffect(() => {\n if (!selectedId && notes.length > 0) {\n const n = notes[0];\n setSelectedId(n.id);\n setEditTitle(n.title);\n setEditContent(n.content);\n setEditColor(n.color);\n }\n }, [notes.length]);\n\n const saveNotes = useCallback((updated: Note[]) => {\n save({ notepad_notes: updated });\n }, [save]);\n\n const autoSave = useCallback(() => {\n if (!selectedId || !dirty) return;\n const updated = notes.map(n =>\n n.id === selectedId ? { ...n, title: editTitle, content: editContent, color: editColor, updated_at: new Date().toISOString() } : n\n );\n saveNotes(updated);\n setDirty(false);\n }, [selectedId, dirty, editTitle, editContent, editColor, notes, saveNotes]);\n\n // Debounced auto-save\n useEffect(() => {\n if (!dirty) return;\n if (saveTimerRef.current) clearTimeout(saveTimerRef.current);\n saveTimerRef.current = setTimeout(autoSave, 800);\n return () => { if (saveTimerRef.current) clearTimeout(saveTimerRef.current); };\n }, [dirty, autoSave]);\n\n // Save on unmount\n useEffect(() => () => { if (dirty) autoSave(); }, []);\n\n const selectNote = (n: Note) => {\n if (dirty) autoSave();\n setSelectedId(n.id);\n setEditTitle(n.title);\n setEditContent(n.content);\n setEditColor(n.color);\n setDirty(false);\n setEditing(false);\n };\n\n const createNote = () => {\n if (dirty) autoSave();\n const n = newNote();\n const updated = [n, ...notes];\n saveNotes(updated);\n setSelectedId(n.id);\n setEditTitle('');\n setEditContent('');\n setEditColor('yellow');\n setDirty(false);\n setEditing(true);\n setTimeout(() => textareaRef.current?.focus(), 50);\n };\n\n const deleteNote = (id: string) => {\n const updated = notes.filter(n => n.id !== id);\n saveNotes(updated);\n if (selectedId === id) {\n const next = updated[0];\n if (next) selectNote(next);\n else { setSelectedId(null); setEditTitle(''); setEditContent(''); }\n }\n };\n\n const toggleSticky = (id: string) => {\n const updated = notes.map(n =>\n n.id === id ? { ...n, sticky: !n.sticky } : n\n );\n saveNotes(updated);\n const note = updated.find(n => n.id === id);\n toast.success(note?.sticky ? 'Pinned to desktop' : 'Removed from desktop');\n };\n\n const timeAgo = (iso: string) => {\n const diff = Date.now() - new Date(iso).getTime();\n const mins = Math.floor(diff / 60000);\n if (mins < 1) return 'just now';\n if (mins < 60) return `${mins}m ago`;\n const hrs = Math.floor(mins / 60);\n if (hrs < 24) return `${hrs}h ago`;\n const days = Math.floor(hrs / 24);\n return `${days}d ago`;\n };\n\n // ── Toggle checkbox in content ──\n const toggleCheckbox = (charIndex: number) => {\n const before = editContent.slice(0, charIndex);\n const match = editContent.slice(charIndex).match(/^\\[([ xX]?)\\]/);\n if (!match) return;\n const isChecked = match[1] === 'x' || match[1] === 'X';\n const replacement = isChecked ? '[ ]' : '[x]';\n const after = editContent.slice(charIndex + match[0].length);\n setEditContent(before + replacement + after);\n setDirty(true);\n };\n\n // ── Open entity reference ──\n const openRef = async (prefix: string, number: string) => {\n const mapping = prefixMap.current[prefix];\n if (!mapping) return;\n const refNum = `${prefix}#${number}`;\n try {\n const { data } = await apiClient.get(mapping.endpoint, { params: { search: refNum, page_size: 1 } });\n const results = data?.results ?? data ?? [];\n const entity = results[0];\n if (entity) {\n openEntity(mapping.entityType, entity.id, entity, refNum);\n } else {\n toast.error(`${refNum} not found`);\n }\n } catch {\n toast.error(`Failed to look up ${refNum}`);\n }\n };\n\n // ── Render content with links and checkboxes ──\n const renderContent = (text: string): ReactNode[] => {\n if (!text) return [];\n const lines = text.split('\\n');\n return lines.map((line, li) => {\n const parts: ReactNode[] = [];\n let lastIdx = 0;\n // Find all special tokens in the line\n const tokens: { idx: number; len: number; render: () => ReactNode }[] = [];\n\n // Entity references — match XX#NNNNN (2+ letters + # + 4-6 digits)\n const refRegex = /([A-Z]{2,4})#(\\d{4,6})/g;\n let m: RegExpExecArray | null;\n while ((m = refRegex.exec(line)) !== null) {\n const prefix = m[1];\n const num = m[2];\n if (prefixMap.current[prefix]) {\n const startIdx = m.index;\n const matchText = m[0];\n tokens.push({\n idx: startIdx,\n len: matchText.length,\n render: () => (\n <button key={`ref-${li}-${startIdx}`} onClick={() => openRef(prefix, num)}\n className=\"text-blue-600 hover:text-blue-800 hover:underline font-medium cursor-pointer\">\n {matchText}\n </button>\n ),\n });\n }\n }\n\n // Checkboxes\n CHECKBOX_REGEX.lastIndex = 0;\n // Calculate the character offset in the full content for this line\n let lineStartInContent = 0;\n for (let i = 0; i < li; i++) lineStartInContent += lines[i].length + 1;\n while ((m = CHECKBOX_REGEX.exec(line)) !== null) {\n const isChecked = m[1] === 'x' || m[1] === 'X';\n const startIdx = m.index;\n const contentCharIdx = lineStartInContent + startIdx;\n tokens.push({\n idx: startIdx,\n len: m[0].length,\n render: () => (\n <button key={`cb-${li}-${startIdx}`} onClick={() => toggleCheckbox(contentCharIdx)}\n className={`inline-flex items-center justify-center w-4 h-4 rounded border ${isChecked ? 'bg-blue-500 border-blue-500 text-white' : 'border-gray-400 bg-white hover:border-blue-400'} cursor-pointer align-text-bottom mr-0.5`}>\n {isChecked && <svg className=\"w-3 h-3\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={3}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M4.5 12.75l6 6 9-13.5\" /></svg>}\n </button>\n ),\n });\n }\n\n // Sort tokens by position\n tokens.sort((a, b) => a.idx - b.idx);\n\n // Build line with tokens interspersed\n for (const token of tokens) {\n if (token.idx > lastIdx) {\n // Check if text after a checked checkbox should be struck through\n parts.push(<span key={`t-${li}-${lastIdx}`}>{line.slice(lastIdx, token.idx)}</span>);\n }\n parts.push(token.render());\n lastIdx = token.idx + token.len;\n }\n if (lastIdx < line.length) {\n parts.push(<span key={`t-${li}-${lastIdx}`}>{line.slice(lastIdx)}</span>);\n }\n if (parts.length === 0) parts.push(<span key={`empty-${li}`}>{'\\u200B'}</span>); // zero-width space for empty lines\n\n // Check if line starts with a checked checkbox → strikethrough the rest\n const lineHasChecked = /^\\[x\\]/i.test(line.trimStart());\n return (\n <div key={li} className={lineHasChecked ? 'line-through text-gray-400' : ''}>\n {parts}\n </div>\n );\n });\n };\n\n const startEditing = () => {\n setEditing(true);\n setTimeout(() => textareaRef.current?.focus(), 50);\n };\n\n const stopEditing = () => {\n setEditing(false);\n if (dirty) autoSave();\n };\n\n const PLACEHOLDER_HINT = `Start writing...\n\nTips:\n [] Type [] for a checkbox, click to toggle\n SO#35001 Type XX#NNNNN to link entities (SO, PO, CI, VI, PL, etc.)`;\n\n return (\n <div className=\"flex h-full\">\n {/* Note list sidebar */}\n <div className=\"w-56 shrink-0 border-r border-gray-200 flex flex-col\">\n <div className=\"p-2 border-b border-gray-200\">\n <button onClick={createNote}\n className=\"w-full flex items-center justify-center gap-1.5 rounded-lg bg-blue-600 text-white px-3 py-1.5 text-sm font-medium hover:bg-blue-700 transition-colors\">\n <svg className=\"h-4 w-4\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={2}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M12 4.5v15m7.5-7.5h-15\" /></svg>\n New Note\n </button>\n </div>\n <div className=\"flex-1 overflow-y-auto\">\n {notes.length === 0 ? (\n <p className=\"text-sm text-gray-400 text-center py-8 px-4\">No notes yet. Create one to get started.</p>\n ) : (\n notes.map(n => {\n const c = getColor(n.color);\n return (\n <button key={n.id} onClick={() => selectNote(n)}\n className={`w-full text-left px-3 py-2.5 border-b border-gray-100 transition-colors ${selectedId === n.id ? 'bg-blue-50' : 'hover:bg-gray-50'}`}>\n <div className=\"flex items-center gap-2\">\n <div className={`w-2.5 h-2.5 rounded-full shrink-0 ${c.dot}`} />\n <span className=\"text-sm font-medium text-gray-900 truncate flex-1\">\n {n.title || 'Untitled'}\n </span>\n {n.sticky && (\n <svg className=\"h-3 w-3 text-amber-500 shrink-0\" fill=\"currentColor\" viewBox=\"0 0 24 24\"><path d=\"M5 5a2 2 0 012-2h10a2 2 0 012 2v16l-7-3.5L5 21V5z\" /></svg>\n )}\n </div>\n <p className=\"text-xs text-gray-400 mt-0.5 truncate ml-4.5\">{n.content.replace(/\\[[ xX]?\\]/g, '').slice(0, 60) || 'Empty note'}</p>\n <p className=\"text-[10px] text-gray-300 mt-0.5 ml-4.5\">{timeAgo(n.updated_at)}</p>\n </button>\n );\n })\n )}\n </div>\n </div>\n\n {/* Editor */}\n {selectedId ? (\n <div className=\"flex-1 flex flex-col min-w-0\">\n {/* Title */}\n <div className=\"flex items-center gap-2 px-4 py-2 border-b border-gray-200\">\n <input\n value={editTitle}\n onChange={e => { setEditTitle(e.target.value); setDirty(true); }}\n placeholder=\"Note title...\"\n className=\"flex-1 text-lg font-semibold text-gray-900 outline-none bg-transparent placeholder:text-gray-300\"\n />\n {/* Color picker */}\n <div className=\"flex gap-1\">\n {COLORS.map(c => (\n <button key={c.key} onClick={() => { setEditColor(c.key); setDirty(true); }}\n className={`w-5 h-5 rounded-full border-2 transition-all ${c.dot} ${editColor === c.key ? 'border-gray-600 scale-110' : 'border-transparent hover:border-gray-400'}`}\n title={c.key} />\n ))}\n </div>\n {/* Sticky toggle */}\n <button onClick={() => toggleSticky(selectedId)} title={selected?.sticky ? 'Remove from desktop' : 'Pin to desktop'}\n className={`p-1 rounded transition-colors ${selected?.sticky ? 'text-amber-500 hover:text-amber-600' : 'text-gray-300 hover:text-amber-400'}`}>\n <svg className=\"h-4 w-4\" fill={selected?.sticky ? 'currentColor' : 'none'} viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M5 5a2 2 0 012-2h10a2 2 0 012 2v16l-7-3.5L5 21V5z\" /></svg>\n </button>\n {/* Delete */}\n <button onClick={() => deleteNote(selectedId)} title=\"Delete note\"\n className=\"p-1 rounded text-gray-300 hover:text-red-500 transition-colors\">\n <svg className=\"h-4 w-4\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0\" /></svg>\n </button>\n </div>\n {/* Content — toggle between textarea (edit) and rendered view */}\n {editing ? (\n <textarea\n ref={textareaRef}\n value={editContent}\n onChange={e => { setEditContent(e.target.value); setDirty(true); }}\n onBlur={stopEditing}\n placeholder={PLACEHOLDER_HINT}\n className=\"flex-1 p-4 text-sm text-gray-700 outline-none resize-none bg-transparent leading-relaxed placeholder:text-gray-300 font-mono\"\n />\n ) : (\n <div\n onClick={startEditing}\n className=\"flex-1 p-4 text-sm text-gray-700 overflow-y-auto leading-relaxed cursor-text\"\n >\n {editContent ? renderContent(editContent) : (\n <p className=\"text-gray-300 whitespace-pre-line\">{PLACEHOLDER_HINT}</p>\n )}\n </div>\n )}\n {/* Bottom hint bar */}\n <div className=\"px-4 py-1.5 border-t border-gray-100 flex items-center gap-4 text-[10px] text-gray-400\">\n <span>[] checkbox</span>\n <span>SO#35001 entity link</span>\n <span className=\"ml-auto\">{editing ? 'Editing' : 'Click to edit'}</span>\n </div>\n </div>\n ) : (\n <div className=\"flex-1 flex items-center justify-center\">\n <div className=\"text-center text-gray-400\">\n <svg className=\"h-12 w-12 mx-auto mb-3 text-gray-300\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1}><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\" /></svg>\n <p className=\"text-sm\">Select a note or create a new one</p>\n </div>\n </div>\n )}\n </div>\n );\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/apps/PomodoroTimer.tsx"],"names":[],"mappings":"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","file":"PomodoroTimer-SBBPQMYB.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"]}
|
package/dist/Stock-ICDNFM7U.js
DELETED
|
@@ -1,234 +0,0 @@
|
|
|
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, useEffect } 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"];
|
|
10
|
-
var SYMBOLS_KEY = "stock_symbols";
|
|
11
|
-
var APIKEY_KEY = "stock_api_key";
|
|
12
|
-
var SETTINGS_KEY = "stock_appearance";
|
|
13
|
-
var CACHE_KEY = "stock_quotes_cache";
|
|
14
|
-
var CACHE_TTL = 6e4;
|
|
15
|
-
var REFRESH_MS = 6e4;
|
|
16
|
-
function readCache() {
|
|
17
|
-
try {
|
|
18
|
-
return JSON.parse(localStorage.getItem(CACHE_KEY) || "{}");
|
|
19
|
-
} catch {
|
|
20
|
-
return {};
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
function writeCache(c) {
|
|
24
|
-
try {
|
|
25
|
-
localStorage.setItem(CACHE_KEY, JSON.stringify(c));
|
|
26
|
-
} catch {
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
function loadSymbols() {
|
|
30
|
-
try {
|
|
31
|
-
const s = JSON.parse(localStorage.getItem(SYMBOLS_KEY) || "");
|
|
32
|
-
if (Array.isArray(s) && s.length) return s.slice(0, MAX_SYMBOLS);
|
|
33
|
-
} catch {
|
|
34
|
-
}
|
|
35
|
-
return DEFAULT_SYMBOLS;
|
|
36
|
-
}
|
|
37
|
-
var fmtPrice = (n) => `$${n.toLocaleString(void 0, { minimumFractionDigits: 2, maximumFractionDigits: 2 })}`;
|
|
38
|
-
var fmtSigned = (n) => `${n >= 0 ? "+" : ""}${n.toFixed(2)}`;
|
|
39
|
-
function Stock() {
|
|
40
|
-
const [symbols, setSymbols] = useState(loadSymbols);
|
|
41
|
-
const [apiKey, setApiKey] = useState(() => {
|
|
42
|
-
try {
|
|
43
|
-
return localStorage.getItem(APIKEY_KEY) || "";
|
|
44
|
-
} catch {
|
|
45
|
-
return "";
|
|
46
|
-
}
|
|
47
|
-
});
|
|
48
|
-
const [appearance, setAppearance] = useState(() => loadAppearance(SETTINGS_KEY));
|
|
49
|
-
const [quotes, setQuotes] = useState({});
|
|
50
|
-
const [loading, setLoading] = useState(false);
|
|
51
|
-
const [tick, setTick] = useState(0);
|
|
52
|
-
const [settingsOpen, setSettingsOpen] = useState(false);
|
|
53
|
-
const [configSymbols, setConfigSymbols] = useState([]);
|
|
54
|
-
const [configApiKey, setConfigApiKey] = useState("");
|
|
55
|
-
const [configAppearance, setConfigAppearance] = useState(appearance);
|
|
56
|
-
const [newSymbol, setNewSymbol] = useState("");
|
|
57
|
-
const openSettings = useCallback(() => {
|
|
58
|
-
setConfigSymbols([...symbols]);
|
|
59
|
-
setConfigApiKey(apiKey);
|
|
60
|
-
setConfigAppearance({ ...appearance });
|
|
61
|
-
setNewSymbol("");
|
|
62
|
-
setSettingsOpen(true);
|
|
63
|
-
}, [symbols, apiKey, appearance]);
|
|
64
|
-
useWidgetSettings(openSettings);
|
|
65
|
-
useEffect(() => {
|
|
66
|
-
const t = setInterval(() => setTick((n) => n + 1), REFRESH_MS);
|
|
67
|
-
return () => clearInterval(t);
|
|
68
|
-
}, []);
|
|
69
|
-
useEffect(() => {
|
|
70
|
-
if (!apiKey || symbols.length === 0) {
|
|
71
|
-
setQuotes({});
|
|
72
|
-
return;
|
|
73
|
-
}
|
|
74
|
-
let mounted = true;
|
|
75
|
-
(async () => {
|
|
76
|
-
setLoading(true);
|
|
77
|
-
const cache = readCache();
|
|
78
|
-
const next = {};
|
|
79
|
-
await Promise.all(symbols.map(async (sym) => {
|
|
80
|
-
const hit = cache[sym];
|
|
81
|
-
if (hit && Date.now() - hit.ts < CACHE_TTL) {
|
|
82
|
-
next[sym] = hit.q;
|
|
83
|
-
return;
|
|
84
|
-
}
|
|
85
|
-
try {
|
|
86
|
-
const res = await fetch(`https://finnhub.io/api/v1/quote?symbol=${encodeURIComponent(sym)}&token=${encodeURIComponent(apiKey)}`);
|
|
87
|
-
const d = await res.json();
|
|
88
|
-
if (typeof d.c === "number" && d.c > 0) {
|
|
89
|
-
const q = { price: d.c, change: d.d ?? 0, changePct: d.dp ?? 0 };
|
|
90
|
-
next[sym] = q;
|
|
91
|
-
cache[sym] = { q, ts: Date.now() };
|
|
92
|
-
} else if (hit) {
|
|
93
|
-
next[sym] = hit.q;
|
|
94
|
-
}
|
|
95
|
-
} catch {
|
|
96
|
-
if (hit) next[sym] = hit.q;
|
|
97
|
-
}
|
|
98
|
-
}));
|
|
99
|
-
writeCache(cache);
|
|
100
|
-
if (mounted) {
|
|
101
|
-
setQuotes(next);
|
|
102
|
-
setLoading(false);
|
|
103
|
-
}
|
|
104
|
-
})();
|
|
105
|
-
return () => {
|
|
106
|
-
mounted = false;
|
|
107
|
-
};
|
|
108
|
-
}, [symbols.join(","), apiKey, tick]);
|
|
109
|
-
const addSymbol = () => {
|
|
110
|
-
const s = newSymbol.trim().toUpperCase();
|
|
111
|
-
if (!s || configSymbols.includes(s) || configSymbols.length >= MAX_SYMBOLS) return;
|
|
112
|
-
setConfigSymbols((p) => [...p, s]);
|
|
113
|
-
setNewSymbol("");
|
|
114
|
-
};
|
|
115
|
-
const saveSettings = () => {
|
|
116
|
-
const cleaned = configSymbols.map((s) => s.toUpperCase()).slice(0, MAX_SYMBOLS);
|
|
117
|
-
const key = configApiKey.trim();
|
|
118
|
-
setSymbols(cleaned);
|
|
119
|
-
setApiKey(key);
|
|
120
|
-
setAppearance(configAppearance);
|
|
121
|
-
localStorage.setItem(SYMBOLS_KEY, JSON.stringify(cleaned));
|
|
122
|
-
localStorage.setItem(APIKEY_KEY, key);
|
|
123
|
-
localStorage.setItem(SETTINGS_KEY, JSON.stringify(configAppearance));
|
|
124
|
-
setSettingsOpen(false);
|
|
125
|
-
};
|
|
126
|
-
return /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
127
|
-
/* @__PURE__ */ jsx(
|
|
128
|
-
"div",
|
|
129
|
-
{
|
|
130
|
-
className: "flex flex-col h-full",
|
|
131
|
-
style: { backgroundColor: `rgb(var(--taskbar-bg-rgb, 243 244 246) / ${appearance.activeOpacity / 100})`, backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : void 0 },
|
|
132
|
-
children: /* @__PURE__ */ jsx("div", { className: "px-4 py-3 flex-1", children: !apiKey ? /* @__PURE__ */ jsxs("div", { className: "flex flex-col items-center justify-center text-center gap-2 py-6", children: [
|
|
133
|
-
/* @__PURE__ */ jsx("svg", { className: "h-8 w-8 text-gray-400", fill: "none", viewBox: "0 0 24 24", stroke: "currentColor", strokeWidth: 1.5, children: /* @__PURE__ */ jsx("path", { strokeLinecap: "round", strokeLinejoin: "round", d: "M2.25 18L9 11.25l4.306 4.306a11.95 11.95 0 015.814-5.518l2.74-1.22m0 0l-5.94-2.281m5.94 2.28l-2.28 5.941" }) }),
|
|
134
|
-
/* @__PURE__ */ jsx("p", { className: "text-sm font-semibold text-gray-700", children: "Track live stock prices" }),
|
|
135
|
-
/* @__PURE__ */ jsx("p", { className: "text-xs text-gray-500 max-w-[15rem]", children: "Add a free API key to get started." }),
|
|
136
|
-
/* @__PURE__ */ jsx("button", { onClick: openSettings, className: "mt-1 text-xs font-medium text-white bg-blue-600 hover:bg-blue-700 rounded-lg px-3 py-1.5 transition-colors", children: "Set up" })
|
|
137
|
-
] }) : 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__ */ jsxs("div", { className: "space-y-0.5", children: [
|
|
138
|
-
loading && Object.keys(quotes).length === 0 && /* @__PURE__ */ jsx("div", { className: "text-xs text-gray-400 text-center py-4", children: "Loading quotes\u2026" }),
|
|
139
|
-
symbols.map((sym) => {
|
|
140
|
-
const q = quotes[sym];
|
|
141
|
-
const up = (q?.change ?? 0) >= 0;
|
|
142
|
-
const color = !q ? "text-gray-400" : q.change > 0 ? "text-green-600" : q.change < 0 ? "text-red-600" : "text-gray-500";
|
|
143
|
-
return /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between px-2 py-2 border-b border-gray-200 last:border-0", children: [
|
|
144
|
-
/* @__PURE__ */ jsx("span", { className: "text-sm font-bold text-gray-800 tracking-tight", children: sym }),
|
|
145
|
-
q ? /* @__PURE__ */ jsxs("div", { className: "flex flex-col items-end leading-tight", children: [
|
|
146
|
-
/* @__PURE__ */ jsx("span", { className: "text-sm font-mono font-semibold text-gray-900 tabular-nums", children: fmtPrice(q.price) }),
|
|
147
|
-
/* @__PURE__ */ jsxs("span", { className: `text-[11px] font-medium tabular-nums ${color}`, children: [
|
|
148
|
-
up ? "\u25B2" : "\u25BC",
|
|
149
|
-
" ",
|
|
150
|
-
fmtSigned(q.change),
|
|
151
|
-
" (",
|
|
152
|
-
fmtSigned(q.changePct),
|
|
153
|
-
"%)"
|
|
154
|
-
] })
|
|
155
|
-
] }) : /* @__PURE__ */ jsx("span", { className: "text-sm font-mono text-gray-400 tabular-nums", children: "\u2014" })
|
|
156
|
-
] }, sym);
|
|
157
|
-
})
|
|
158
|
-
] }) })
|
|
159
|
-
}
|
|
160
|
-
),
|
|
161
|
-
/* @__PURE__ */ jsxs(
|
|
162
|
-
WidgetSettingsModal,
|
|
163
|
-
{
|
|
164
|
-
open: settingsOpen,
|
|
165
|
-
onClose: () => setSettingsOpen(false),
|
|
166
|
-
title: "Stock Settings",
|
|
167
|
-
appearance: configAppearance,
|
|
168
|
-
onAppearanceChange: setConfigAppearance,
|
|
169
|
-
onSave: saveSettings,
|
|
170
|
-
children: [
|
|
171
|
-
/* @__PURE__ */ jsxs("div", { children: [
|
|
172
|
-
/* @__PURE__ */ jsx("h3", { className: "text-sm font-semibold text-gray-700 mb-2", children: "API Key" }),
|
|
173
|
-
/* @__PURE__ */ jsx(
|
|
174
|
-
"input",
|
|
175
|
-
{
|
|
176
|
-
value: configApiKey,
|
|
177
|
-
onChange: (e) => setConfigApiKey(e.target.value),
|
|
178
|
-
placeholder: "Finnhub API key",
|
|
179
|
-
className: "w-full bg-gray-50 border border-gray-200 rounded px-2 py-1.5 text-sm font-mono focus:outline-none focus:ring-1 focus:ring-blue-500"
|
|
180
|
-
}
|
|
181
|
-
),
|
|
182
|
-
/* @__PURE__ */ jsxs("p", { className: "text-[11px] text-gray-400 mt-1", children: [
|
|
183
|
-
"Free key from ",
|
|
184
|
-
/* @__PURE__ */ jsx("a", { href: "https://finnhub.io/register", target: "_blank", rel: "noreferrer", className: "text-blue-600 hover:underline", children: "finnhub.io" }),
|
|
185
|
-
" \u2014 quotes update every minute."
|
|
186
|
-
] })
|
|
187
|
-
] }),
|
|
188
|
-
/* @__PURE__ */ jsxs("div", { children: [
|
|
189
|
-
/* @__PURE__ */ jsxs("h3", { className: "text-sm font-semibold text-gray-700 mb-2", children: [
|
|
190
|
-
"Symbols ",
|
|
191
|
-
/* @__PURE__ */ jsxs("span", { className: "text-[11px] font-normal text-gray-400", children: [
|
|
192
|
-
"(",
|
|
193
|
-
configSymbols.length,
|
|
194
|
-
"/",
|
|
195
|
-
MAX_SYMBOLS,
|
|
196
|
-
")"
|
|
197
|
-
] })
|
|
198
|
-
] }),
|
|
199
|
-
/* @__PURE__ */ jsxs("div", { className: "space-y-1 mb-2", children: [
|
|
200
|
-
configSymbols.map((s, i) => /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between py-1 px-2 bg-gray-50 rounded", children: [
|
|
201
|
-
/* @__PURE__ */ jsx("span", { className: "text-sm font-semibold text-gray-700", children: s }),
|
|
202
|
-
/* @__PURE__ */ jsx("button", { onClick: () => setConfigSymbols((p) => p.filter((_, idx) => idx !== i)), className: "text-red-400 hover:text-red-600", children: "\xD7" })
|
|
203
|
-
] }, i)),
|
|
204
|
-
configSymbols.length === 0 && /* @__PURE__ */ jsx("p", { className: "text-[11px] text-gray-400 italic", children: "No symbols \u2014 add a ticker below." })
|
|
205
|
-
] }),
|
|
206
|
-
configSymbols.length < MAX_SYMBOLS ? /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
207
|
-
/* @__PURE__ */ jsx(
|
|
208
|
-
"input",
|
|
209
|
-
{
|
|
210
|
-
value: newSymbol,
|
|
211
|
-
onChange: (e) => setNewSymbol(e.target.value),
|
|
212
|
-
onKeyDown: (e) => {
|
|
213
|
-
if (e.key === "Enter") addSymbol();
|
|
214
|
-
},
|
|
215
|
-
placeholder: "e.g. NVDA",
|
|
216
|
-
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"
|
|
217
|
-
}
|
|
218
|
-
),
|
|
219
|
-
/* @__PURE__ */ jsx("button", { onClick: addSymbol, className: "text-sm font-medium text-blue-600 hover:text-blue-800 px-2", children: "+ Add" })
|
|
220
|
-
] }) : /* @__PURE__ */ jsxs("p", { className: "text-[11px] text-gray-400 italic", children: [
|
|
221
|
-
"Max ",
|
|
222
|
-
MAX_SYMBOLS,
|
|
223
|
-
" symbols \u2014 remove one to add another."
|
|
224
|
-
] })
|
|
225
|
-
] })
|
|
226
|
-
]
|
|
227
|
-
}
|
|
228
|
-
)
|
|
229
|
-
] });
|
|
230
|
-
}
|
|
231
|
-
|
|
232
|
-
export { Stock as default };
|
|
233
|
-
//# sourceMappingURL=Stock-ICDNFM7U.js.map
|
|
234
|
-
//# sourceMappingURL=Stock-ICDNFM7U.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/apps/Stock.tsx"],"names":[],"mappings":";;;;;;;AAeA,IAAM,WAAA,GAAc,CAAA;AACpB,IAAM,eAAA,GAAkB,CAAC,MAAA,EAAQ,MAAA,EAAQ,SAAS,MAAM,CAAA;AAExD,IAAM,WAAA,GAAc,eAAA;AACpB,IAAM,UAAA,GAAa,eAAA;AACnB,IAAM,YAAA,GAAe,kBAAA;AACrB,IAAM,SAAA,GAAY,oBAAA;AAClB,IAAM,SAAA,GAAY,GAAA;AAClB,IAAM,UAAA,GAAa,GAAA;AAMnB,SAAS,SAAA,GAAmB;AAAE,EAAA,IAAI;AAAE,IAAA,OAAO,KAAK,KAAA,CAAM,YAAA,CAAa,OAAA,CAAQ,SAAS,KAAK,IAAI,CAAA;AAAA,EAAG,CAAA,CAAA,MAAQ;AAAE,IAAA,OAAO,EAAC;AAAA,EAAG;AAAE;AACvH,SAAS,WAAW,CAAA,EAAU;AAAE,EAAA,IAAI;AAAE,IAAA,YAAA,CAAa,OAAA,CAAQ,SAAA,EAAW,IAAA,CAAK,SAAA,CAAU,CAAC,CAAC,CAAA;AAAA,EAAG,CAAA,CAAA,MAAQ;AAAA,EAAC;AAAE;AAErG,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,MAAA,EAAQ,SAAS,CAAA,GAAI,SAAS,MAAM;AAAE,IAAA,IAAI;AAAE,MAAA,OAAO,YAAA,CAAa,OAAA,CAAQ,UAAU,CAAA,IAAK,EAAA;AAAA,IAAI,CAAA,CAAA,MAAQ;AAAE,MAAA,OAAO,EAAA;AAAA,IAAI;AAAA,EAAE,CAAC,CAAA;AAC1H,EAAA,MAAM,CAAC,YAAY,aAAa,CAAA,GAAI,SAAS,MAAM,cAAA,CAAe,YAAY,CAAC,CAAA;AAC/E,EAAA,MAAM,CAAC,MAAA,EAAQ,SAAS,CAAA,GAAI,QAAA,CAAgC,EAAE,CAAA;AAC9D,EAAA,MAAM,CAAC,OAAA,EAAS,UAAU,CAAA,GAAI,SAAS,KAAK,CAAA;AAC5C,EAAA,MAAM,CAAC,IAAA,EAAM,OAAO,CAAA,GAAI,SAAS,CAAC,CAAA;AAElC,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,YAAA,EAAc,eAAe,CAAA,GAAI,SAAS,EAAE,CAAA;AACnD,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,eAAA,CAAgB,MAAM,CAAA;AACtB,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,MAAA,EAAQ,UAAU,CAAC,CAAA;AAEhC,EAAA,iBAAA,CAAkB,YAAY,CAAA;AAG9B,EAAA,SAAA,CAAU,MAAM;AACd,IAAA,MAAM,CAAA,GAAI,YAAY,MAAM,OAAA,CAAQ,OAAK,CAAA,GAAI,CAAC,GAAG,UAAU,CAAA;AAC3D,IAAA,OAAO,MAAM,cAAc,CAAC,CAAA;AAAA,EAC9B,CAAA,EAAG,EAAE,CAAA;AAEL,EAAA,SAAA,CAAU,MAAM;AACd,IAAA,IAAI,CAAC,MAAA,IAAU,OAAA,CAAQ,MAAA,KAAW,CAAA,EAAG;AAAE,MAAA,SAAA,CAAU,EAAE,CAAA;AAAG,MAAA;AAAA,IAAQ;AAC9D,IAAA,IAAI,OAAA,GAAU,IAAA;AACd,IAAA,CAAC,YAAY;AACX,MAAA,UAAA,CAAW,IAAI,CAAA;AACf,MAAA,MAAM,QAAQ,SAAA,EAAU;AACxB,MAAA,MAAM,OAA8B,EAAC;AACrC,MAAA,MAAM,OAAA,CAAQ,GAAA,CAAI,OAAA,CAAQ,GAAA,CAAI,OAAM,GAAA,KAAO;AACzC,QAAA,MAAM,GAAA,GAAM,MAAM,GAAG,CAAA;AACrB,QAAA,IAAI,OAAO,IAAA,CAAK,GAAA,EAAI,GAAI,GAAA,CAAI,KAAK,SAAA,EAAW;AAAE,UAAA,IAAA,CAAK,GAAG,IAAI,GAAA,CAAI,CAAA;AAAG,UAAA;AAAA,QAAQ;AACzE,QAAA,IAAI;AACF,UAAA,MAAM,GAAA,GAAM,MAAM,KAAA,CAAM,CAAA,uCAAA,EAA0C,kBAAA,CAAmB,GAAG,CAAC,CAAA,OAAA,EAAU,kBAAA,CAAmB,MAAM,CAAC,CAAA,CAAE,CAAA;AAC/H,UAAA,MAAM,CAAA,GAAI,MAAM,GAAA,CAAI,IAAA,EAAK;AACzB,UAAA,IAAI,OAAO,CAAA,CAAE,CAAA,KAAM,QAAA,IAAY,CAAA,CAAE,IAAI,CAAA,EAAG;AACtC,YAAA,MAAM,CAAA,GAAW,EAAE,KAAA,EAAO,CAAA,CAAE,CAAA,EAAG,MAAA,EAAQ,CAAA,CAAE,CAAA,IAAK,CAAA,EAAG,SAAA,EAAW,CAAA,CAAE,EAAA,IAAM,CAAA,EAAE;AACtE,YAAA,IAAA,CAAK,GAAG,CAAA,GAAI,CAAA;AAAG,YAAA,KAAA,CAAM,GAAG,CAAA,GAAI,EAAE,GAAG,EAAA,EAAI,IAAA,CAAK,KAAI,EAAE;AAAA,UAClD,WAAW,GAAA,EAAK;AAAE,YAAA,IAAA,CAAK,GAAG,IAAI,GAAA,CAAI,CAAA;AAAA,UAAG;AAAA,QACvC,CAAA,CAAA,MAAQ;AAAE,UAAA,IAAI,GAAA,EAAK,IAAA,CAAK,GAAG,CAAA,GAAI,GAAA,CAAI,CAAA;AAAA,QAAG;AAAA,MACxC,CAAC,CAAC,CAAA;AACF,MAAA,UAAA,CAAW,KAAK,CAAA;AAChB,MAAA,IAAI,OAAA,EAAS;AAAE,QAAA,SAAA,CAAU,IAAI,CAAA;AAAG,QAAA,UAAA,CAAW,KAAK,CAAA;AAAA,MAAG;AAAA,IACrD,CAAA,GAAG;AACH,IAAA,OAAO,MAAM;AAAE,MAAA,OAAA,GAAU,KAAA;AAAA,IAAO,CAAA;AAAA,EAClC,CAAA,EAAG,CAAC,OAAA,CAAQ,IAAA,CAAK,GAAG,CAAA,EAAG,MAAA,EAAQ,IAAI,CAAC,CAAA;AAEpC,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,MAAM,GAAA,GAAM,aAAa,IAAA,EAAK;AAC9B,IAAA,UAAA,CAAW,OAAO,CAAA;AAAG,IAAA,SAAA,CAAU,GAAG,CAAA;AAAG,IAAA,aAAA,CAAc,gBAAgB,CAAA;AACnE,IAAA,YAAA,CAAa,OAAA,CAAQ,WAAA,EAAa,IAAA,CAAK,SAAA,CAAU,OAAO,CAAC,CAAA;AACzD,IAAA,YAAA,CAAa,OAAA,CAAQ,YAAY,GAAG,CAAA;AACpC,IAAA,YAAA,CAAa,OAAA,CAAQ,YAAA,EAAc,IAAA,CAAK,SAAA,CAAU,gBAAgB,CAAC,CAAA;AACnE,IAAA,eAAA,CAAgB,KAAK,CAAA;AAAA,EACvB,CAAA;AAEA,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,QAAA,kBAAA,GAAA,CAAC,SAAI,SAAA,EAAU,kBAAA,EACZ,WAAC,MAAA,mBACA,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,kEAAA,EACb,QAAA,EAAA;AAAA,0BAAA,GAAA,CAAC,SAAI,SAAA,EAAU,uBAAA,EAAwB,MAAK,MAAA,EAAO,OAAA,EAAQ,aAAY,MAAA,EAAO,cAAA,EAAe,aAAa,GAAA,EAAK,QAAA,kBAAA,GAAA,CAAC,UAAK,aAAA,EAAc,OAAA,EAAQ,gBAAe,OAAA,EAAQ,CAAA,EAAE,4GAA2G,CAAA,EAAE,CAAA;AAAA,0BACjR,GAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,qCAAA,EAAsC,QAAA,EAAA,yBAAA,EAAuB,CAAA;AAAA,0BAC1E,GAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,qCAAA,EAAsC,QAAA,EAAA,oCAAA,EAAkC,CAAA;AAAA,8BACpF,QAAA,EAAA,EAAO,OAAA,EAAS,YAAA,EAAc,SAAA,EAAU,8GAA6G,QAAA,EAAA,QAAA,EAAM;AAAA,SAAA,EAC9J,CAAA,GACE,OAAA,CAAQ,MAAA,KAAW,CAAA,mBACrB,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,wCAAA,EAAyC,QAAA,EAAA,6CAAA,EAAsC,CAAA,mBAE9F,IAAA,CAAC,KAAA,EAAA,EAAI,WAAU,aAAA,EACZ,QAAA,EAAA;AAAA,UAAA,OAAA,IAAW,MAAA,CAAO,IAAA,CAAK,MAAM,CAAA,CAAE,MAAA,KAAW,qBAAK,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,wCAAA,EAAyC,QAAA,EAAA,sBAAA,EAAe,CAAA;AAAA,UACtH,OAAA,CAAQ,IAAI,CAAA,GAAA,KAAO;AAClB,YAAA,MAAM,CAAA,GAAI,OAAO,GAAG,CAAA;AACpB,YAAA,MAAM,EAAA,GAAA,CAAM,CAAA,EAAG,MAAA,IAAU,CAAA,KAAM,CAAA;AAC/B,YAAA,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,YAAA,uBACE,IAAA,CAAC,KAAA,EAAA,EAAc,SAAA,EAAU,oFAAA,EACvB,QAAA,EAAA;AAAA,8BAAA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,gDAAA,EAAkD,QAAA,EAAA,GAAA,EAAI,CAAA;AAAA,cACrE,CAAA,mBACC,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,uCAAA,EACb,QAAA,EAAA;AAAA,gCAAA,GAAA,CAAC,UAAK,SAAA,EAAU,4DAAA,EAA8D,QAAA,EAAA,QAAA,CAAS,CAAA,CAAE,KAAK,CAAA,EAAE,CAAA;AAAA,gCAChG,IAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAW,CAAA,qCAAA,EAAwC,KAAK,CAAA,CAAA,EAAK,QAAA,EAAA;AAAA,kBAAA,EAAA,GAAK,QAAA,GAAM,QAAA;AAAA,kBAAI,GAAA;AAAA,kBAAE,SAAA,CAAU,EAAE,MAAM,CAAA;AAAA,kBAAE,IAAA;AAAA,kBAAG,SAAA,CAAU,EAAE,SAAS,CAAA;AAAA,kBAAE;AAAA,iBAAA,EAAE;AAAA,eAAA,EACtI,CAAA,mBAEA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,gDAA+C,QAAA,EAAA,QAAA,EAAC;AAAA,aAAA,EAAA,EAR1D,GAUV,CAAA;AAAA,UAEJ,CAAC;AAAA,SAAA,EACH,CAAA,EAEJ;AAAA;AAAA,KACF;AAAA,oBAEA,IAAA;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,QAAA,EAAA;AAAA,0BAAA,IAAA,CAAC,KAAA,EAAA,EACC,QAAA,EAAA;AAAA,4BAAA,GAAA,CAAC,IAAA,EAAA,EAAG,SAAA,EAAU,0CAAA,EAA2C,QAAA,EAAA,SAAA,EAAO,CAAA;AAAA,4BAChE,GAAA;AAAA,cAAC,OAAA;AAAA,cAAA;AAAA,gBAAM,KAAA,EAAO,YAAA;AAAA,gBAAc,QAAA,EAAU,CAAA,CAAA,KAAK,eAAA,CAAgB,CAAA,CAAE,OAAO,KAAK,CAAA;AAAA,gBAAG,WAAA,EAAY,iBAAA;AAAA,gBACtF,SAAA,EAAU;AAAA;AAAA,aAAqI;AAAA,4BACjJ,IAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,gCAAA,EAAiC,QAAA,EAAA;AAAA,cAAA,gBAAA;AAAA,8BAAc,GAAA,CAAC,GAAA,EAAA,EAAE,IAAA,EAAK,6BAAA,EAA8B,MAAA,EAAO,UAAS,GAAA,EAAI,YAAA,EAAa,SAAA,EAAU,+BAAA,EAAgC,QAAA,EAAA,YAAA,EAAU,CAAA;AAAA,cAAI;AAAA,aAAA,EAA8B;AAAA,WAAA,EAC3N,CAAA;AAAA,+BACC,KAAA,EAAA,EACC,QAAA,EAAA;AAAA,4BAAA,IAAA,CAAC,IAAA,EAAA,EAAG,WAAU,0CAAA,EAA2C,QAAA,EAAA;AAAA,cAAA,UAAA;AAAA,8BAAQ,IAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,uCAAA,EAAwC,QAAA,EAAA;AAAA,gBAAA,GAAA;AAAA,gBAAE,aAAA,CAAc,MAAA;AAAA,gBAAO,GAAA;AAAA,gBAAE,WAAA;AAAA,gBAAY;AAAA,eAAA,EAAC;AAAA,aAAA,EAAO,CAAA;AAAA,4BACtK,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,gBAAA,EACZ,QAAA,EAAA;AAAA,cAAA,aAAA,CAAc,IAAI,CAAC,CAAA,EAAG,sBACrB,IAAA,CAAC,KAAA,EAAA,EAAY,WAAU,gEAAA,EACrB,QAAA,EAAA;AAAA,gCAAA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAU,qCAAA,EAAuC,QAAA,EAAA,CAAA,EAAE,CAAA;AAAA,oCACxD,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,eAAA,EAAA,EAF1H,CAGV,CACD,CAAA;AAAA,cACA,cAAc,MAAA,KAAW,CAAA,wBAAM,GAAA,EAAA,EAAE,SAAA,EAAU,oCAAmC,QAAA,EAAA,uCAAA,EAAgC;AAAA,aAAA,EACjH,CAAA;AAAA,YACC,cAAc,MAAA,GAAS,WAAA,mBACtB,IAAA,CAAC,KAAA,EAAA,EAAI,WAAU,yBAAA,EACb,QAAA,EAAA;AAAA,8BAAA,GAAA;AAAA,gBAAC,OAAA;AAAA,gBAAA;AAAA,kBAAM,KAAA,EAAO,SAAA;AAAA,kBAAW,QAAA,EAAU,CAAA,CAAA,KAAK,YAAA,CAAa,CAAA,CAAE,OAAO,KAAK,CAAA;AAAA,kBAAG,WAAW,CAAA,CAAA,KAAK;AAAE,oBAAA,IAAI,CAAA,CAAE,GAAA,KAAQ,OAAA,EAAS,SAAA,EAAU;AAAA,kBAAG,CAAA;AAAA,kBAC1H,WAAA,EAAY,WAAA;AAAA,kBAAY,SAAA,EAAU;AAAA;AAAA,eAAmI;AAAA,kCACtK,QAAA,EAAA,EAAO,OAAA,EAAS,SAAA,EAAW,SAAA,EAAU,8DAA6D,QAAA,EAAA,OAAA,EAAK;AAAA,aAAA,EAC1G,CAAA,mBAEA,IAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,kCAAA,EAAmC,QAAA,EAAA;AAAA,cAAA,MAAA;AAAA,cAAK,WAAA;AAAA,cAAY;AAAA,aAAA,EAAqC;AAAA,WAAA,EAE1G;AAAA;AAAA;AAAA;AACF,GAAA,EACF,CAAA;AAEJ","file":"Stock-ICDNFM7U.js","sourcesContent":["import { useState, useEffect, 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 live prices.\n *\n * Quotes come from Finnhub's keyless-from-the-browser /quote endpoint, which\n * needs a (free) API key the user pastes in settings — there's no reliable\n * keyless + CORS stock feed, so \"setup\" means entering a key + picking\n * symbols. Until a key is set the widget shows a friendly call-to-action.\n */\n\n/** Cap the watchlist — keeps the widget compact and stays well within\n * Finnhub's free tier (one /quote call per symbol, polled each minute). */\nconst MAX_SYMBOLS = 8;\nconst DEFAULT_SYMBOLS = ['AAPL', 'MSFT', 'GOOGL', 'TSLA'];\n\nconst SYMBOLS_KEY = 'stock_symbols';\nconst APIKEY_KEY = 'stock_api_key';\nconst SETTINGS_KEY = 'stock_appearance';\nconst CACHE_KEY = 'stock_quotes_cache';\nconst CACHE_TTL = 60_000; // 1 min — fresh enough for a desktop ticker\nconst REFRESH_MS = 60_000;\n\ninterface Quote { price: number; change: number; changePct: number }\ntype CacheEntry = { q: Quote; ts: number };\ntype Cache = Record<string, CacheEntry>;\n\nfunction readCache(): Cache { try { return JSON.parse(localStorage.getItem(CACHE_KEY) || '{}'); } catch { return {}; } }\nfunction writeCache(c: Cache) { try { localStorage.setItem(CACHE_KEY, JSON.stringify(c)); } catch {} }\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 [apiKey, setApiKey] = useState(() => { try { return localStorage.getItem(APIKEY_KEY) || ''; } catch { return ''; } });\n const [appearance, setAppearance] = useState(() => loadAppearance(SETTINGS_KEY));\n const [quotes, setQuotes] = useState<Record<string, Quote>>({});\n const [loading, setLoading] = useState(false);\n const [tick, setTick] = useState(0);\n\n const [settingsOpen, setSettingsOpen] = useState(false);\n const [configSymbols, setConfigSymbols] = useState<string[]>([]);\n const [configApiKey, setConfigApiKey] = useState('');\n const [configAppearance, setConfigAppearance] = useState<WidgetAppearance>(appearance);\n const [newSymbol, setNewSymbol] = useState('');\n\n const openSettings = useCallback(() => {\n setConfigSymbols([...symbols]);\n setConfigApiKey(apiKey);\n setConfigAppearance({ ...appearance });\n setNewSymbol('');\n setSettingsOpen(true);\n }, [symbols, apiKey, appearance]);\n\n useWidgetSettings(openSettings);\n\n // Poll on a timer so prices stay current while the widget sits on the desktop.\n useEffect(() => {\n const t = setInterval(() => setTick(n => n + 1), REFRESH_MS);\n return () => clearInterval(t);\n }, []);\n\n useEffect(() => {\n if (!apiKey || symbols.length === 0) { setQuotes({}); return; }\n let mounted = true;\n (async () => {\n setLoading(true);\n const cache = readCache();\n const next: Record<string, Quote> = {};\n await Promise.all(symbols.map(async sym => {\n const hit = cache[sym];\n if (hit && Date.now() - hit.ts < CACHE_TTL) { next[sym] = hit.q; return; }\n try {\n const res = await fetch(`https://finnhub.io/api/v1/quote?symbol=${encodeURIComponent(sym)}&token=${encodeURIComponent(apiKey)}`);\n const d = await res.json();\n if (typeof d.c === 'number' && d.c > 0) {\n const q: Quote = { price: d.c, change: d.d ?? 0, changePct: d.dp ?? 0 };\n next[sym] = q; cache[sym] = { q, ts: Date.now() };\n } else if (hit) { next[sym] = hit.q; } // unknown symbol / rate-limited → keep last good\n } catch { if (hit) next[sym] = hit.q; }\n }));\n writeCache(cache);\n if (mounted) { setQuotes(next); setLoading(false); }\n })();\n return () => { mounted = false; };\n }, [symbols.join(','), apiKey, tick]);\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 const key = configApiKey.trim();\n setSymbols(cleaned); setApiKey(key); setAppearance(configAppearance);\n localStorage.setItem(SYMBOLS_KEY, JSON.stringify(cleaned));\n localStorage.setItem(APIKEY_KEY, key);\n localStorage.setItem(SETTINGS_KEY, JSON.stringify(configAppearance));\n setSettingsOpen(false);\n };\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 {!apiKey ? (\n <div className=\"flex flex-col items-center justify-center text-center gap-2 py-6\">\n <svg className=\"h-8 w-8 text-gray-400\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M2.25 18L9 11.25l4.306 4.306a11.95 11.95 0 015.814-5.518l2.74-1.22m0 0l-5.94-2.281m5.94 2.28l-2.28 5.941\" /></svg>\n <p className=\"text-sm font-semibold text-gray-700\">Track live stock prices</p>\n <p className=\"text-xs text-gray-500 max-w-[15rem]\">Add a free API key to get started.</p>\n <button onClick={openSettings} className=\"mt-1 text-xs font-medium text-white bg-blue-600 hover:bg-blue-700 rounded-lg px-3 py-1.5 transition-colors\">Set up</button>\n </div>\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 {loading && Object.keys(quotes).length === 0 && <div className=\"text-xs text-gray-400 text-center py-4\">Loading quotes…</div>}\n {symbols.map(sym => {\n const q = 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\">API Key</h3>\n <input value={configApiKey} onChange={e => setConfigApiKey(e.target.value)} placeholder=\"Finnhub API key\"\n className=\"w-full bg-gray-50 border border-gray-200 rounded px-2 py-1.5 text-sm font-mono focus:outline-none focus:ring-1 focus:ring-blue-500\" />\n <p className=\"text-[11px] text-gray-400 mt-1\">Free key from <a href=\"https://finnhub.io/register\" target=\"_blank\" rel=\"noreferrer\" className=\"text-blue-600 hover:underline\">finnhub.io</a> — quotes update every minute.</p>\n </div>\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\">×</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 </div>\n </WidgetSettingsModal>\n </>\n );\n}\n"]}
|