react-os-shell 0.2.44 → 0.2.46

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 (41) hide show
  1. package/dist/{Calendar-4UQDQ3NT.js → Calendar-PKRZ5MBU.js} +153 -54
  2. package/dist/Calendar-PKRZ5MBU.js.map +1 -0
  3. package/dist/{CurrencyConverter-TXBFDFG2.js → CurrencyConverter-5N44NZ6Z.js} +3 -3
  4. package/dist/{CurrencyConverter-TXBFDFG2.js.map → CurrencyConverter-5N44NZ6Z.js.map} +1 -1
  5. package/dist/{Email-HRBZUWPY.js → Email-CR6XS2AD.js} +4 -4
  6. package/dist/{Email-HRBZUWPY.js.map → Email-CR6XS2AD.js.map} +1 -1
  7. package/dist/{Files-ITIKVHIE.js → Files-CEID4TC3.js} +4 -4
  8. package/dist/{Files-ITIKVHIE.js.map → Files-CEID4TC3.js.map} +1 -1
  9. package/dist/{GeminiChat-ITU46EH4.js → GeminiChat-XTEBZIVK.js} +3 -3
  10. package/dist/{GeminiChat-ITU46EH4.js.map → GeminiChat-XTEBZIVK.js.map} +1 -1
  11. package/dist/{PomodoroTimer-PRP5CZ3S.js → PomodoroTimer-FHSOLF3O.js} +49 -49
  12. package/dist/PomodoroTimer-FHSOLF3O.js.map +1 -0
  13. package/dist/Preview-B5DUW2AR.js +7 -0
  14. package/dist/{Preview-4MBQI66Q.js.map → Preview-B5DUW2AR.js.map} +1 -1
  15. package/dist/TodoList-7JZ2SLDI.js +494 -0
  16. package/dist/TodoList-7JZ2SLDI.js.map +1 -0
  17. package/dist/{WorldClock-QO5PVJQQ.js → WorldClock-XHM7WAUV.js} +43 -97
  18. package/dist/WorldClock-XHM7WAUV.js.map +1 -0
  19. package/dist/apps/index.d.ts +4 -1
  20. package/dist/apps/index.js +14 -12
  21. package/dist/apps/index.js.map +1 -1
  22. package/dist/chunk-25L4DIKH.js +90 -0
  23. package/dist/chunk-25L4DIKH.js.map +1 -0
  24. package/dist/{chunk-62MVMTBT.js → chunk-5VXRBUEH.js} +20 -3
  25. package/dist/chunk-5VXRBUEH.js.map +1 -0
  26. package/dist/{chunk-MTLVXT2C.js → chunk-6AJUSDEM.js} +3 -3
  27. package/dist/{chunk-MTLVXT2C.js.map → chunk-6AJUSDEM.js.map} +1 -1
  28. package/dist/{chunk-46LICZUM.js → chunk-MVWEL34Y.js} +3 -2
  29. package/dist/chunk-MVWEL34Y.js.map +1 -0
  30. package/dist/{chunk-DUUANLLE.js → chunk-QBH7KERS.js} +482 -19
  31. package/dist/chunk-QBH7KERS.js.map +1 -0
  32. package/dist/index.js +3 -3
  33. package/dist/styles.css +6 -4
  34. package/package.json +1 -1
  35. package/dist/Calendar-4UQDQ3NT.js.map +0 -1
  36. package/dist/PomodoroTimer-PRP5CZ3S.js.map +0 -1
  37. package/dist/Preview-4MBQI66Q.js +0 -7
  38. package/dist/WorldClock-QO5PVJQQ.js.map +0 -1
  39. package/dist/chunk-46LICZUM.js.map +0 -1
  40. package/dist/chunk-62MVMTBT.js.map +0 -1
  41. package/dist/chunk-DUUANLLE.js.map +0 -1
@@ -1,8 +1,8 @@
1
- export { Files as default, openFilesInTrashMode } from './chunk-MTLVXT2C.js';
2
- import './chunk-DUUANLLE.js';
1
+ export { Files as default, openFilesInTrashMode } from './chunk-6AJUSDEM.js';
2
+ import './chunk-QBH7KERS.js';
3
3
  import './chunk-KUIPWCTJ.js';
4
4
  import './chunk-WIJ45SYD.js';
5
5
  import './chunk-7M3BBAHQ.js';
6
6
  import './chunk-PLGHQ7QW.js';
7
- //# sourceMappingURL=Files-ITIKVHIE.js.map
8
- //# sourceMappingURL=Files-ITIKVHIE.js.map
7
+ //# sourceMappingURL=Files-CEID4TC3.js.map
8
+ //# sourceMappingURL=Files-CEID4TC3.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"Files-ITIKVHIE.js"}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"Files-CEID4TC3.js"}
@@ -1,4 +1,4 @@
1
- import { useGoogleAuth, getGoogleAccessToken } from './chunk-46LICZUM.js';
1
+ import { useGoogleAuth, getGoogleAccessToken } from './chunk-MVWEL34Y.js';
2
2
  import { toast_default } from './chunk-WIJ45SYD.js';
3
3
  import { useState, useRef, useEffect } from 'react';
4
4
  import { jsx, jsxs } from 'react/jsx-runtime';
@@ -180,5 +180,5 @@ function GeminiChat() {
180
180
  }
181
181
 
182
182
  export { GeminiChat as default };
183
- //# sourceMappingURL=GeminiChat-ITU46EH4.js.map
184
- //# sourceMappingURL=GeminiChat-ITU46EH4.js.map
183
+ //# sourceMappingURL=GeminiChat-XTEBZIVK.js.map
184
+ //# sourceMappingURL=GeminiChat-XTEBZIVK.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/apps/GeminiChat.tsx"],"names":[],"mappings":"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","file":"GeminiChat-ITU46EH4.js","sourcesContent":["import { useState, useRef, useEffect } from 'react';\nimport useGoogleAuth, { getGoogleAccessToken } from '../hooks/useGoogleAuth';\nimport toast from '../shell/toast';\n\ninterface ChatMessage {\n role: 'user' | 'model';\n content: string;\n timestamp: Date;\n}\n\nconst GEMINI_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-2.5-flash:generateContent';\n\nexport default function GeminiChat() {\n const { isConnected, user, connect, disconnect, hasClientId, setClientId, loading: authLoading, error: authError } = useGoogleAuth();\n const [messages, setMessages] = useState<ChatMessage[]>([]);\n const [input, setInput] = useState('');\n const [loading, setLoading] = useState(false);\n const [clientIdInput, setClientIdInput] = useState('');\n const scrollRef = useRef<HTMLDivElement>(null);\n const inputRef = useRef<HTMLTextAreaElement>(null);\n\n useEffect(() => {\n scrollRef.current?.scrollTo({ top: scrollRef.current.scrollHeight, behavior: 'smooth' });\n }, [messages]);\n\n const sendMessage = async () => {\n const text = input.trim();\n if (!text || loading) return;\n\n const token = getGoogleAccessToken();\n if (!token) { toast.error('Google session expired. Please reconnect.'); return; }\n\n const userMsg: ChatMessage = { role: 'user', content: text, timestamp: new Date() };\n setMessages(prev => [...prev, userMsg]);\n setInput('');\n setLoading(true);\n\n try {\n // Build conversation history for context\n const contents = [...messages, userMsg].map(m => ({\n role: m.role === 'model' ? 'model' : 'user',\n parts: [{ text: m.content }],\n }));\n\n const res = await fetch(`${GEMINI_API}`, {\n method: 'POST',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({ contents }),\n });\n\n if (!res.ok) {\n const err = await res.json().catch(() => ({}));\n throw new Error(err.error?.message || `API error ${res.status}`);\n }\n\n const data = await res.json();\n const reply = data.candidates?.[0]?.content?.parts?.[0]?.text || 'No response.';\n setMessages(prev => [...prev, { role: 'model', content: reply, timestamp: new Date() }]);\n } catch (err: any) {\n toast.error(err.message || 'Failed to get response from Gemini.');\n setMessages(prev => [...prev, { role: 'model', content: `Error: ${err.message}`, timestamp: new Date() }]);\n }\n setLoading(false);\n setTimeout(() => inputRef.current?.focus(), 50);\n };\n\n const clearChat = () => {\n setMessages([]);\n };\n\n // ── Not connected ──\n if (!isConnected) {\n return (\n <div className=\"flex items-center justify-center h-full\">\n <div className=\"text-center max-w-md space-y-4 px-6\">\n <div className=\"h-16 w-16 mx-auto rounded-2xl bg-gradient-to-br from-blue-500 via-purple-500 to-pink-500 flex items-center justify-center\">\n <svg className=\"h-8 w-8 text-white\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9.813 15.904L9 18.75l-.813-2.846a4.5 4.5 0 00-3.09-3.09L2.25 12l2.846-.813a4.5 4.5 0 003.09-3.09L9 5.25l.813 2.846a4.5 4.5 0 003.09 3.09L15.75 12l-2.846.813a4.5 4.5 0 00-3.09 3.09zM18.259 8.715L18 9.75l-.259-1.035a3.375 3.375 0 00-2.455-2.456L14.25 6l1.036-.259a3.375 3.375 0 002.455-2.456L18 2.25l.259 1.035a3.375 3.375 0 002.455 2.456L21.75 6l-1.036.259a3.375 3.375 0 00-2.455 2.456zM16.894 20.567L16.5 21.75l-.394-1.183a2.25 2.25 0 00-1.423-1.423L13.5 18.75l1.183-.394a2.25 2.25 0 001.423-1.423l.394-1.183.394 1.183a2.25 2.25 0 001.423 1.423l1.183.394-1.183.394a2.25 2.25 0 00-1.423 1.423z\" /></svg>\n </div>\n <h2 className=\"text-lg font-semibold text-gray-900\">Gemini AI</h2>\n <p className=\"text-sm text-gray-500\">Connect your Google account to chat with Gemini AI.</p>\n\n {!hasClientId ? (\n <div className=\"text-left space-y-2 bg-gray-50 rounded-lg p-4\">\n <label className=\"block text-xs font-medium text-gray-700\">Google OAuth Client ID</label>\n <input value={clientIdInput} onChange={e => setClientIdInput(e.target.value)} placeholder=\"123456789.apps.googleusercontent.com\"\n className=\"w-full rounded-md border border-gray-300 px-3 py-2 text-sm focus:border-blue-500 focus:ring-blue-500\" />\n <button onClick={() => { if (clientIdInput.trim()) setClientId(clientIdInput.trim()); }} disabled={!clientIdInput.trim()}\n className=\"w-full bg-gray-900 text-white px-4 py-2 text-sm font-medium rounded-lg hover:bg-gray-800 disabled:opacity-40\">Save Client ID</button>\n <p className=\"text-[10px] text-gray-400\">Enable \"Generative Language API\" in your Google Cloud project</p>\n </div>\n ) : (\n <button onClick={connect} disabled={authLoading}\n className=\"inline-flex items-center gap-2 bg-white border border-gray-300 shadow-sm px-6 py-2.5 text-sm font-medium rounded-lg hover:bg-gray-50 disabled:opacity-50\">\n <svg className=\"h-5 w-5\" viewBox=\"0 0 24 24\"><path d=\"M22.56 12.25c0-.78-.07-1.53-.2-2.25H12v4.26h5.92a5.06 5.06 0 01-2.2 3.32v2.77h3.57c2.08-1.92 3.28-4.74 3.28-8.1z\" fill=\"#4285F4\"/><path d=\"M12 23c2.97 0 5.46-.98 7.28-2.66l-3.57-2.77c-.98.66-2.23 1.06-3.71 1.06-2.86 0-5.29-1.93-6.16-4.53H2.18v2.84C3.99 20.53 7.7 23 12 23z\" fill=\"#34A853\"/><path d=\"M5.84 14.09c-.22-.66-.35-1.36-.35-2.09s.13-1.43.35-2.09V7.07H2.18C1.43 8.55 1 10.22 1 12s.43 3.45 1.18 4.93l2.85-2.22.81-.62z\" fill=\"#FBBC05\"/><path d=\"M12 5.38c1.62 0 3.06.56 4.21 1.64l3.15-3.15C17.45 2.09 14.97 1 12 1 7.7 1 3.99 3.47 2.18 7.07l3.66 2.84c.87-2.6 3.3-4.53 6.16-4.53z\" fill=\"#EA4335\"/></svg>\n {authLoading ? 'Connecting...' : 'Sign in with Google'}\n </button>\n )}\n {authError && <p className=\"text-sm text-red-600\">{authError}</p>}\n </div>\n </div>\n );\n }\n\n // ── Connected: Chat UI ──\n return (\n <div className=\"flex flex-col h-full\">\n {/* Header */}\n <div className=\"flex items-center justify-between px-4 py-2 border-b border-gray-200 shrink-0\">\n <div className=\"flex items-center gap-2\">\n <div className=\"h-6 w-6 rounded-lg bg-gradient-to-br from-blue-500 via-purple-500 to-pink-500 flex items-center justify-center\">\n <svg className=\"h-3.5 w-3.5 text-white\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={2}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9.813 15.904L9 18.75l-.813-2.846a4.5 4.5 0 00-3.09-3.09L2.25 12l2.846-.813a4.5 4.5 0 003.09-3.09L9 5.25l.813 2.846a4.5 4.5 0 003.09 3.09L15.75 12l-2.846.813a4.5 4.5 0 00-3.09 3.09z\" /></svg>\n </div>\n <span className=\"text-sm font-semibold text-gray-900\">Gemini</span>\n <span className=\"text-xs text-gray-400\">2.5 Flash</span>\n </div>\n <div className=\"flex items-center gap-2\">\n <button onClick={clearChat} className=\"text-xs text-gray-500 hover:text-gray-700\">Clear</button>\n <button onClick={() => window.dispatchEvent(new Event('open-google-connect'))} title=\"Google Services\"\n className=\"flex items-center gap-2 hover:bg-gray-100 rounded-md px-1.5 py-1 transition-colors\">\n {user?.picture ? (\n <img src={user.picture} alt=\"\" className=\"h-6 w-6 rounded-full\" />\n ) : (\n <div className=\"h-6 w-6 rounded-full bg-gray-200\" />\n )}\n <div className=\"text-left\">\n <p className=\"text-[11px] font-medium text-gray-900\">{user?.name}</p>\n <p className=\"text-[10px] text-gray-500\">{user?.email}</p>\n </div>\n </button>\n </div>\n </div>\n\n {/* Messages */}\n <div ref={scrollRef} className=\"flex-1 overflow-y-auto px-4 py-4 space-y-4\">\n {messages.length === 0 && (\n <div className=\"flex items-center justify-center h-full text-center\">\n <div>\n <div className=\"h-12 w-12 mx-auto mb-3 rounded-xl bg-gradient-to-br from-blue-500/10 via-purple-500/10 to-pink-500/10 flex items-center justify-center\">\n <svg className=\"h-6 w-6 text-purple-400\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9.813 15.904L9 18.75l-.813-2.846a4.5 4.5 0 00-3.09-3.09L2.25 12l2.846-.813a4.5 4.5 0 003.09-3.09L9 5.25l.813 2.846a4.5 4.5 0 003.09 3.09L15.75 12l-2.846.813a4.5 4.5 0 00-3.09 3.09z\" /></svg>\n </div>\n <p className=\"text-sm text-gray-500\">Ask Gemini anything</p>\n <p className=\"text-xs text-gray-400 mt-1\">Powered by Google AI</p>\n </div>\n </div>\n )}\n {messages.map((msg, i) => (\n <div key={i} className={`flex ${msg.role === 'user' ? 'justify-end' : 'justify-start'}`}>\n <div className={`max-w-[80%] rounded-2xl px-4 py-2.5 ${msg.role === 'user' ? 'bg-blue-600 text-white' : 'bg-gray-100 text-gray-900'}`}>\n <div className=\"text-sm whitespace-pre-wrap leading-relaxed\">{msg.content}</div>\n <p className={`text-[10px] mt-1 ${msg.role === 'user' ? 'text-blue-200' : 'text-gray-400'}`}>\n {msg.timestamp.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' })}\n </p>\n </div>\n </div>\n ))}\n {loading && (\n <div className=\"flex justify-start\">\n <div className=\"bg-gray-100 rounded-2xl px-4 py-3\">\n <div className=\"flex items-center gap-1.5\">\n <div className=\"w-2 h-2 rounded-full bg-gray-400 animate-bounce\" style={{ animationDelay: '0ms' }} />\n <div className=\"w-2 h-2 rounded-full bg-gray-400 animate-bounce\" style={{ animationDelay: '150ms' }} />\n <div className=\"w-2 h-2 rounded-full bg-gray-400 animate-bounce\" style={{ animationDelay: '300ms' }} />\n </div>\n </div>\n </div>\n )}\n </div>\n\n {/* Input */}\n <div className=\"px-4 py-3 border-t border-gray-200 shrink-0\">\n <div className=\"flex items-end gap-2\">\n <textarea\n ref={inputRef}\n value={input}\n onChange={e => setInput(e.target.value)}\n onKeyDown={e => { if (e.key === 'Enter' && !e.shiftKey) { e.preventDefault(); sendMessage(); } }}\n placeholder=\"Message Gemini...\"\n rows={1}\n className=\"flex-1 rounded-xl border border-gray-300 px-4 py-2.5 text-sm resize-none focus:border-blue-500 focus:ring-blue-500 max-h-32\"\n style={{ minHeight: '42px' }}\n />\n <button onClick={sendMessage} disabled={loading || !input.trim()}\n className=\"shrink-0 bg-blue-600 text-white rounded-xl p-2.5 hover:bg-blue-700 disabled:opacity-40 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=\"M6 12L3.269 3.126A59.768 59.768 0 0121.485 12 59.77 59.77 0 013.27 20.876L5.999 12zm0 0h7.5\" /></svg>\n </button>\n </div>\n </div>\n </div>\n );\n}\n"]}
1
+ {"version":3,"sources":["../src/apps/GeminiChat.tsx"],"names":[],"mappings":"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","file":"GeminiChat-XTEBZIVK.js","sourcesContent":["import { useState, useRef, useEffect } from 'react';\nimport useGoogleAuth, { getGoogleAccessToken } from '../hooks/useGoogleAuth';\nimport toast from '../shell/toast';\n\ninterface ChatMessage {\n role: 'user' | 'model';\n content: string;\n timestamp: Date;\n}\n\nconst GEMINI_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-2.5-flash:generateContent';\n\nexport default function GeminiChat() {\n const { isConnected, user, connect, disconnect, hasClientId, setClientId, loading: authLoading, error: authError } = useGoogleAuth();\n const [messages, setMessages] = useState<ChatMessage[]>([]);\n const [input, setInput] = useState('');\n const [loading, setLoading] = useState(false);\n const [clientIdInput, setClientIdInput] = useState('');\n const scrollRef = useRef<HTMLDivElement>(null);\n const inputRef = useRef<HTMLTextAreaElement>(null);\n\n useEffect(() => {\n scrollRef.current?.scrollTo({ top: scrollRef.current.scrollHeight, behavior: 'smooth' });\n }, [messages]);\n\n const sendMessage = async () => {\n const text = input.trim();\n if (!text || loading) return;\n\n const token = getGoogleAccessToken();\n if (!token) { toast.error('Google session expired. Please reconnect.'); return; }\n\n const userMsg: ChatMessage = { role: 'user', content: text, timestamp: new Date() };\n setMessages(prev => [...prev, userMsg]);\n setInput('');\n setLoading(true);\n\n try {\n // Build conversation history for context\n const contents = [...messages, userMsg].map(m => ({\n role: m.role === 'model' ? 'model' : 'user',\n parts: [{ text: m.content }],\n }));\n\n const res = await fetch(`${GEMINI_API}`, {\n method: 'POST',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({ contents }),\n });\n\n if (!res.ok) {\n const err = await res.json().catch(() => ({}));\n throw new Error(err.error?.message || `API error ${res.status}`);\n }\n\n const data = await res.json();\n const reply = data.candidates?.[0]?.content?.parts?.[0]?.text || 'No response.';\n setMessages(prev => [...prev, { role: 'model', content: reply, timestamp: new Date() }]);\n } catch (err: any) {\n toast.error(err.message || 'Failed to get response from Gemini.');\n setMessages(prev => [...prev, { role: 'model', content: `Error: ${err.message}`, timestamp: new Date() }]);\n }\n setLoading(false);\n setTimeout(() => inputRef.current?.focus(), 50);\n };\n\n const clearChat = () => {\n setMessages([]);\n };\n\n // ── Not connected ──\n if (!isConnected) {\n return (\n <div className=\"flex items-center justify-center h-full\">\n <div className=\"text-center max-w-md space-y-4 px-6\">\n <div className=\"h-16 w-16 mx-auto rounded-2xl bg-gradient-to-br from-blue-500 via-purple-500 to-pink-500 flex items-center justify-center\">\n <svg className=\"h-8 w-8 text-white\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9.813 15.904L9 18.75l-.813-2.846a4.5 4.5 0 00-3.09-3.09L2.25 12l2.846-.813a4.5 4.5 0 003.09-3.09L9 5.25l.813 2.846a4.5 4.5 0 003.09 3.09L15.75 12l-2.846.813a4.5 4.5 0 00-3.09 3.09zM18.259 8.715L18 9.75l-.259-1.035a3.375 3.375 0 00-2.455-2.456L14.25 6l1.036-.259a3.375 3.375 0 002.455-2.456L18 2.25l.259 1.035a3.375 3.375 0 002.455 2.456L21.75 6l-1.036.259a3.375 3.375 0 00-2.455 2.456zM16.894 20.567L16.5 21.75l-.394-1.183a2.25 2.25 0 00-1.423-1.423L13.5 18.75l1.183-.394a2.25 2.25 0 001.423-1.423l.394-1.183.394 1.183a2.25 2.25 0 001.423 1.423l1.183.394-1.183.394a2.25 2.25 0 00-1.423 1.423z\" /></svg>\n </div>\n <h2 className=\"text-lg font-semibold text-gray-900\">Gemini AI</h2>\n <p className=\"text-sm text-gray-500\">Connect your Google account to chat with Gemini AI.</p>\n\n {!hasClientId ? (\n <div className=\"text-left space-y-2 bg-gray-50 rounded-lg p-4\">\n <label className=\"block text-xs font-medium text-gray-700\">Google OAuth Client ID</label>\n <input value={clientIdInput} onChange={e => setClientIdInput(e.target.value)} placeholder=\"123456789.apps.googleusercontent.com\"\n className=\"w-full rounded-md border border-gray-300 px-3 py-2 text-sm focus:border-blue-500 focus:ring-blue-500\" />\n <button onClick={() => { if (clientIdInput.trim()) setClientId(clientIdInput.trim()); }} disabled={!clientIdInput.trim()}\n className=\"w-full bg-gray-900 text-white px-4 py-2 text-sm font-medium rounded-lg hover:bg-gray-800 disabled:opacity-40\">Save Client ID</button>\n <p className=\"text-[10px] text-gray-400\">Enable \"Generative Language API\" in your Google Cloud project</p>\n </div>\n ) : (\n <button onClick={connect} disabled={authLoading}\n className=\"inline-flex items-center gap-2 bg-white border border-gray-300 shadow-sm px-6 py-2.5 text-sm font-medium rounded-lg hover:bg-gray-50 disabled:opacity-50\">\n <svg className=\"h-5 w-5\" viewBox=\"0 0 24 24\"><path d=\"M22.56 12.25c0-.78-.07-1.53-.2-2.25H12v4.26h5.92a5.06 5.06 0 01-2.2 3.32v2.77h3.57c2.08-1.92 3.28-4.74 3.28-8.1z\" fill=\"#4285F4\"/><path d=\"M12 23c2.97 0 5.46-.98 7.28-2.66l-3.57-2.77c-.98.66-2.23 1.06-3.71 1.06-2.86 0-5.29-1.93-6.16-4.53H2.18v2.84C3.99 20.53 7.7 23 12 23z\" fill=\"#34A853\"/><path d=\"M5.84 14.09c-.22-.66-.35-1.36-.35-2.09s.13-1.43.35-2.09V7.07H2.18C1.43 8.55 1 10.22 1 12s.43 3.45 1.18 4.93l2.85-2.22.81-.62z\" fill=\"#FBBC05\"/><path d=\"M12 5.38c1.62 0 3.06.56 4.21 1.64l3.15-3.15C17.45 2.09 14.97 1 12 1 7.7 1 3.99 3.47 2.18 7.07l3.66 2.84c.87-2.6 3.3-4.53 6.16-4.53z\" fill=\"#EA4335\"/></svg>\n {authLoading ? 'Connecting...' : 'Sign in with Google'}\n </button>\n )}\n {authError && <p className=\"text-sm text-red-600\">{authError}</p>}\n </div>\n </div>\n );\n }\n\n // ── Connected: Chat UI ──\n return (\n <div className=\"flex flex-col h-full\">\n {/* Header */}\n <div className=\"flex items-center justify-between px-4 py-2 border-b border-gray-200 shrink-0\">\n <div className=\"flex items-center gap-2\">\n <div className=\"h-6 w-6 rounded-lg bg-gradient-to-br from-blue-500 via-purple-500 to-pink-500 flex items-center justify-center\">\n <svg className=\"h-3.5 w-3.5 text-white\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={2}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9.813 15.904L9 18.75l-.813-2.846a4.5 4.5 0 00-3.09-3.09L2.25 12l2.846-.813a4.5 4.5 0 003.09-3.09L9 5.25l.813 2.846a4.5 4.5 0 003.09 3.09L15.75 12l-2.846.813a4.5 4.5 0 00-3.09 3.09z\" /></svg>\n </div>\n <span className=\"text-sm font-semibold text-gray-900\">Gemini</span>\n <span className=\"text-xs text-gray-400\">2.5 Flash</span>\n </div>\n <div className=\"flex items-center gap-2\">\n <button onClick={clearChat} className=\"text-xs text-gray-500 hover:text-gray-700\">Clear</button>\n <button onClick={() => window.dispatchEvent(new Event('open-google-connect'))} title=\"Google Services\"\n className=\"flex items-center gap-2 hover:bg-gray-100 rounded-md px-1.5 py-1 transition-colors\">\n {user?.picture ? (\n <img src={user.picture} alt=\"\" className=\"h-6 w-6 rounded-full\" />\n ) : (\n <div className=\"h-6 w-6 rounded-full bg-gray-200\" />\n )}\n <div className=\"text-left\">\n <p className=\"text-[11px] font-medium text-gray-900\">{user?.name}</p>\n <p className=\"text-[10px] text-gray-500\">{user?.email}</p>\n </div>\n </button>\n </div>\n </div>\n\n {/* Messages */}\n <div ref={scrollRef} className=\"flex-1 overflow-y-auto px-4 py-4 space-y-4\">\n {messages.length === 0 && (\n <div className=\"flex items-center justify-center h-full text-center\">\n <div>\n <div className=\"h-12 w-12 mx-auto mb-3 rounded-xl bg-gradient-to-br from-blue-500/10 via-purple-500/10 to-pink-500/10 flex items-center justify-center\">\n <svg className=\"h-6 w-6 text-purple-400\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\" strokeWidth={1.5}><path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9.813 15.904L9 18.75l-.813-2.846a4.5 4.5 0 00-3.09-3.09L2.25 12l2.846-.813a4.5 4.5 0 003.09-3.09L9 5.25l.813 2.846a4.5 4.5 0 003.09 3.09L15.75 12l-2.846.813a4.5 4.5 0 00-3.09 3.09z\" /></svg>\n </div>\n <p className=\"text-sm text-gray-500\">Ask Gemini anything</p>\n <p className=\"text-xs text-gray-400 mt-1\">Powered by Google AI</p>\n </div>\n </div>\n )}\n {messages.map((msg, i) => (\n <div key={i} className={`flex ${msg.role === 'user' ? 'justify-end' : 'justify-start'}`}>\n <div className={`max-w-[80%] rounded-2xl px-4 py-2.5 ${msg.role === 'user' ? 'bg-blue-600 text-white' : 'bg-gray-100 text-gray-900'}`}>\n <div className=\"text-sm whitespace-pre-wrap leading-relaxed\">{msg.content}</div>\n <p className={`text-[10px] mt-1 ${msg.role === 'user' ? 'text-blue-200' : 'text-gray-400'}`}>\n {msg.timestamp.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' })}\n </p>\n </div>\n </div>\n ))}\n {loading && (\n <div className=\"flex justify-start\">\n <div className=\"bg-gray-100 rounded-2xl px-4 py-3\">\n <div className=\"flex items-center gap-1.5\">\n <div className=\"w-2 h-2 rounded-full bg-gray-400 animate-bounce\" style={{ animationDelay: '0ms' }} />\n <div className=\"w-2 h-2 rounded-full bg-gray-400 animate-bounce\" style={{ animationDelay: '150ms' }} />\n <div className=\"w-2 h-2 rounded-full bg-gray-400 animate-bounce\" style={{ animationDelay: '300ms' }} />\n </div>\n </div>\n </div>\n )}\n </div>\n\n {/* Input */}\n <div className=\"px-4 py-3 border-t border-gray-200 shrink-0\">\n <div className=\"flex items-end gap-2\">\n <textarea\n ref={inputRef}\n value={input}\n onChange={e => setInput(e.target.value)}\n onKeyDown={e => { if (e.key === 'Enter' && !e.shiftKey) { e.preventDefault(); sendMessage(); } }}\n placeholder=\"Message Gemini...\"\n rows={1}\n className=\"flex-1 rounded-xl border border-gray-300 px-4 py-2.5 text-sm resize-none focus:border-blue-500 focus:ring-blue-500 max-h-32\"\n style={{ minHeight: '42px' }}\n />\n <button onClick={sendMessage} disabled={loading || !input.trim()}\n className=\"shrink-0 bg-blue-600 text-white rounded-xl p-2.5 hover:bg-blue-700 disabled:opacity-40 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=\"M6 12L3.269 3.126A59.768 59.768 0 0121.485 12 59.77 59.77 0 013.27 20.876L5.999 12zm0 0h7.5\" /></svg>\n </button>\n </div>\n </div>\n </div>\n );\n}\n"]}
@@ -1,14 +1,14 @@
1
+ import { useTodoTasks, migratePomodoroTasksOnce } from './chunk-25L4DIKH.js';
1
2
  import { subscribePomo, getPomoSnapshot, setPomoDurations, setPomoAlarm, setPomoAlarmConfig, setPomoBehaviour, setPomoFocusSound, pomoSwitchMode, pomoPause, pomoStart, ALARM_OPTIONS, playAlarm, FOCUS_SOUND_OPTIONS, previewFocusSound } from './chunk-MK3HLUO4.js';
2
3
  import { useShellPrefs } from './chunk-36VM54SC.js';
3
4
  import './chunk-WIJ45SYD.js';
4
5
  import { loadAppearance, WidgetSettingsModal } from './chunk-UK2AA3J6.js';
5
6
  import { useWidgetSettings } from './chunk-7M3BBAHQ.js';
6
7
  import './chunk-PLGHQ7QW.js';
7
- import { useSyncExternalStore, useEffect, useState, useRef, useCallback } from 'react';
8
+ import { useSyncExternalStore, useEffect, useMemo, useState, useRef, useCallback } from 'react';
8
9
  import { jsxs, Fragment, jsx } from 'react/jsx-runtime';
9
10
 
10
11
  var POMO_SETTINGS_KEY = "pomodoro_appearance";
11
- var TASKS_KEY = "pomodoro_tasks";
12
12
  var ACTIVE_TASK_KEY = "pomodoro_active_task_id";
13
13
  var MODE_LABELS = { focus: "Pomodoro", short: "Short Break", long: "Long Break" };
14
14
  var MODE_COLORS = {
@@ -59,23 +59,18 @@ function readPrefs(raw) {
59
59
  focusVolume: intInRange(p.focusVolume, 0, 100, DEFAULT_PREFS.focusVolume)
60
60
  };
61
61
  }
62
- function loadTasks() {
63
- try {
64
- const raw = JSON.parse(localStorage.getItem(TASKS_KEY) || "[]");
65
- if (Array.isArray(raw)) return raw;
66
- } catch {
67
- }
68
- return [];
69
- }
70
- function uid() {
71
- return Math.random().toString(36).slice(2, 9);
72
- }
73
62
  var sortDoneToBottom = (arr) => {
74
63
  const undone = [];
75
64
  const done = [];
76
65
  for (const t of arr) (t.done ? done : undone).push(t);
77
66
  return [...undone, ...done];
78
67
  };
68
+ function useMemoSortedTasks(rawTasks, checkToBottom) {
69
+ return useMemo(
70
+ () => checkToBottom ? sortDoneToBottom(rawTasks) : rawTasks,
71
+ [rawTasks, checkToBottom]
72
+ );
73
+ }
79
74
  function PomodoroTimer() {
80
75
  const snap = useSyncExternalStore(subscribePomo, getPomoSnapshot, getPomoSnapshot);
81
76
  const { prefs: shellPrefs, save: saveShellPrefs } = useShellPrefs();
@@ -109,16 +104,28 @@ function PomodoroTimer() {
109
104
  }
110
105
  }
111
106
  }, []);
112
- const [tasks, setTasks] = useState(loadTasks);
107
+ const todo = useTodoTasks();
108
+ const rawTasks = todo.tasks;
109
+ const todayStr = useMemo(() => (/* @__PURE__ */ new Date()).toISOString().slice(0, 10), []);
110
+ const visibleRawTasks = useMemo(
111
+ () => rawTasks.filter((t) => {
112
+ if (!t.dueDate) return false;
113
+ if (t.dueDate === todayStr) return true;
114
+ if (t.dueDate < todayStr && !t.done) return true;
115
+ return false;
116
+ }),
117
+ [rawTasks, todayStr]
118
+ );
119
+ const tasks = useMemoSortedTasks(visibleRawTasks, userPrefs.checkToBottom);
113
120
  const [activeTaskId, setActiveTaskId] = useState(() => localStorage.getItem(ACTIVE_TASK_KEY));
114
121
  const [adding, setAdding] = useState(false);
115
122
  const [menuOpenId, setMenuOpenId] = useState(null);
123
+ const migratedRef = useRef(false);
116
124
  useEffect(() => {
117
- try {
118
- localStorage.setItem(TASKS_KEY, JSON.stringify(tasks));
119
- } catch {
120
- }
121
- }, [tasks]);
125
+ if (migratedRef.current) return;
126
+ migratedRef.current = true;
127
+ migratePomodoroTasksOnce(rawTasks, todo.setAllTasks);
128
+ }, []);
122
129
  useEffect(() => {
123
130
  try {
124
131
  if (activeTaskId) localStorage.setItem(ACTIVE_TASK_KEY, activeTaskId);
@@ -137,48 +144,41 @@ function PomodoroTimer() {
137
144
  }, [menuOpenId]);
138
145
  const lastStreakRef = useRef(snap.streak);
139
146
  useEffect(() => {
140
- if (snap.streak > lastStreakRef.current) {
141
- setTasks((prev) => {
142
- const next = prev.map((t) => {
143
- if (t.id !== activeTaskId) return t;
144
- const completed = t.completed + 1;
145
- const shouldAutoCheck = userPrefs.autoCheckTasks && completed >= t.estimated;
146
- return { ...t, completed, done: t.done || shouldAutoCheck };
147
- });
148
- return userPrefs.checkToBottom ? sortDoneToBottom(next) : next;
149
- });
147
+ if (snap.streak > lastStreakRef.current && activeTaskId) {
148
+ const t = rawTasks.find((x) => x.id === activeTaskId);
149
+ if (t) {
150
+ const completed = (t.completed ?? 0) + 1;
151
+ const estimated = t.estimated ?? 0;
152
+ const shouldAutoCheck = userPrefs.autoCheckTasks && estimated > 0 && completed >= estimated;
153
+ todo.updateTask(activeTaskId, { completed, done: t.done || shouldAutoCheck });
154
+ }
150
155
  }
151
156
  lastStreakRef.current = snap.streak;
152
- }, [snap.streak, activeTaskId, userPrefs.autoCheckTasks, userPrefs.checkToBottom]);
157
+ }, [snap.streak, activeTaskId, userPrefs.autoCheckTasks, rawTasks, todo]);
153
158
  const addTask = (name, estimated) => {
154
159
  const trimmed = name.trim();
155
160
  if (!trimmed) return;
156
- const t = { id: uid(), name: trimmed, estimated, completed: 0, done: false };
157
- setTasks((prev) => [...prev, t]);
158
- if (!activeTaskId) setActiveTaskId(t.id);
161
+ const id = todo.addTask({ name: trimmed, estimated, completed: 0, dueDate: todayStr });
162
+ if (!activeTaskId) setActiveTaskId(id);
159
163
  setAdding(false);
160
164
  };
161
165
  const removeTask = (id) => {
162
- setTasks((prev) => prev.filter((t) => t.id !== id));
166
+ todo.removeTask(id);
163
167
  if (activeTaskId === id) setActiveTaskId(null);
164
168
  setMenuOpenId(null);
165
169
  };
166
170
  const toggleDone = (id) => {
167
- setTasks((prev) => {
168
- const next = prev.map((t) => t.id === id ? { ...t, done: !t.done } : t);
169
- return userPrefs.checkToBottom ? sortDoneToBottom(next) : next;
170
- });
171
+ todo.toggleDone(id);
171
172
  };
172
- const totalCompleted = tasks.reduce((acc, t) => acc + t.completed, 0);
173
- const totalEstimated = tasks.reduce((acc, t) => acc + Math.max(t.estimated, t.completed), 0);
174
- const remainingPomos = tasks.reduce((acc, t) => t.done ? acc : acc + Math.max(0, t.estimated - t.completed), 0);
173
+ const totalCompleted = tasks.reduce((acc, t) => acc + (t.completed ?? 0), 0);
174
+ const totalEstimated = tasks.reduce((acc, t) => acc + Math.max(t.estimated ?? 0, t.completed ?? 0), 0);
175
+ const remainingPomos = tasks.reduce((acc, t) => t.done ? acc : acc + Math.max(0, (t.estimated ?? 0) - (t.completed ?? 0)), 0);
175
176
  const remainingSecsFromTimer = snap.running && snap.mode === "focus" ? snap.remaining : 0;
176
177
  const remainingSecs = remainingPomos * userPrefs.focusMinutes * 60 + remainingSecsFromTimer;
177
178
  const finishAt = new Date(Date.now() + remainingSecs * 1e3);
178
179
  const finishAtStr = finishAt.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit", hour12: false });
179
180
  const totalHours = (remainingSecs / 3600).toFixed(1);
180
- const activeIdx = activeTaskId ? tasks.findIndex((t) => t.id === activeTaskId) : -1;
181
- const activeTask = activeIdx >= 0 ? tasks[activeIdx] : null;
181
+ const activeTask = activeTaskId ? rawTasks.find((t) => t.id === activeTaskId) ?? null : null;
182
182
  const mm = String(Math.floor(snap.remaining / 60)).padStart(2, "0");
183
183
  const ss = String(snap.remaining % 60).padStart(2, "0");
184
184
  const [appearance, setAppearance] = useState(() => loadAppearance(POMO_SETTINGS_KEY));
@@ -215,7 +215,7 @@ function PomodoroTimer() {
215
215
  /* @__PURE__ */ jsxs(
216
216
  "div",
217
217
  {
218
- className: `flex flex-col h-full select-none rounded-2xl overflow-hidden transition-colors duration-300 ${tx.primary}`,
218
+ className: `flex flex-col h-full select-none transition-colors duration-300 ${tx.primary}`,
219
219
  style: {
220
220
  backgroundColor: panelBg,
221
221
  backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : void 0
@@ -247,7 +247,7 @@ function PomodoroTimer() {
247
247
  /* @__PURE__ */ jsx("div", { className: "px-3 py-3 text-center", children: activeTask ? /* @__PURE__ */ jsxs(Fragment, { children: [
248
248
  /* @__PURE__ */ jsxs("div", { className: `text-xs leading-tight ${tx.muted}`, children: [
249
249
  "#",
250
- activeTask.completed + 1
250
+ (activeTask.completed ?? 0) + 1
251
251
  ] }),
252
252
  /* @__PURE__ */ jsx("div", { className: `text-[15px] font-medium leading-tight truncate mt-0.5 ${tx.primary}`, children: activeTask.name })
253
253
  ] }) : /* @__PURE__ */ jsx("div", { className: `text-xs italic ${tx.muted}`, children: "No task selected" }) }),
@@ -282,9 +282,9 @@ function PomodoroTimer() {
282
282
  ),
283
283
  /* @__PURE__ */ jsx("span", { className: `ml-3 flex-1 font-semibold truncate ${task.done ? "line-through text-gray-400" : "text-gray-800"}`, children: task.name }),
284
284
  /* @__PURE__ */ jsxs("span", { className: "text-sm tabular-nums mr-1.5 shrink-0", children: [
285
- /* @__PURE__ */ jsx("span", { className: "font-bold text-gray-500", children: task.completed }),
285
+ /* @__PURE__ */ jsx("span", { className: "font-bold text-gray-500", children: task.completed ?? 0 }),
286
286
  /* @__PURE__ */ jsx("span", { className: "text-gray-400", children: " / " }),
287
- /* @__PURE__ */ jsx("span", { className: "text-gray-400", children: task.estimated })
287
+ /* @__PURE__ */ jsx("span", { className: "text-gray-400", children: task.estimated ?? "\u2013" })
288
288
  ] }),
289
289
  /* @__PURE__ */ jsxs("div", { className: "relative", children: [
290
290
  /* @__PURE__ */ jsx(
@@ -623,5 +623,5 @@ function AddTaskForm({ onSubmit, onCancel, accentColor }) {
623
623
  }
624
624
 
625
625
  export { PomodoroTimer as default };
626
- //# sourceMappingURL=PomodoroTimer-PRP5CZ3S.js.map
627
- //# sourceMappingURL=PomodoroTimer-PRP5CZ3S.js.map
626
+ //# sourceMappingURL=PomodoroTimer-FHSOLF3O.js.map
627
+ //# sourceMappingURL=PomodoroTimer-FHSOLF3O.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/apps/PomodoroTimer.tsx"],"names":[],"mappings":"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","file":"PomodoroTimer-FHSOLF3O.js","sourcesContent":["import { useEffect, useState, useCallback, useSyncExternalStore, useRef, useMemo } from 'react';\nimport { useWidgetSettings } from '../shell/Modal';\nimport WidgetSettingsModal, { loadAppearance, type WidgetAppearance } from '../shell/WidgetSettingsModal';\nimport { useShellPrefs } from '../shell/ShellPrefs';\nimport {\n ALARM_OPTIONS, FOCUS_SOUND_OPTIONS,\n getPomoSnapshot, subscribePomo,\n setPomoDurations, setPomoAlarm, setPomoAlarmConfig, setPomoBehaviour, setPomoFocusSound,\n pomoStart, pomoPause, pomoSwitchMode,\n playAlarm, previewFocusSound,\n type AlarmSound, type FocusSound, type Mode,\n} from '../shell/pomodoroStore';\nimport { useTodoTasks, migratePomodoroTasksOnce } from './_todoStore';\nimport type { TodoTask } from './_todoTypes';\n\nconst POMO_SETTINGS_KEY = 'pomodoro_appearance';\nconst ACTIVE_TASK_KEY = 'pomodoro_active_task_id';\n\nconst MODE_LABELS: Record<Mode, string> = { focus: 'Pomodoro', short: 'Short Break', long: 'Long Break' };\n/**\n * Per-mode background colour. The break modes keep their own colours so\n * a glance at the panel tells you what state you're in. Focus mode is\n * special — its panel uses `--taskbar-bg-rgb` so the widget matches the\n * taskbar (and the rest of the dashboard widgets) across light and dark\n * themes. The hex below is only used for the START / Save accent colour\n * in focus mode.\n */\nconst MODE_COLORS: Record<Mode, string> = {\n focus: '#0f172a', // unused for the panel; only for focus-mode accents\n short: '#508a52', // muted green\n long: '#5b7898', // slate blue\n};\n\n/** Convert `#rrggbb` to an `rgba(r, g, b, alpha)` string so the break\n * panels pick up the user's translucency setting via the alpha channel\n * (instead of an `opacity` on the whole element, which would also fade\n * the text and the START button). */\nfunction hexToRgba(hex: string, alpha: number): string {\n const r = parseInt(hex.slice(1, 3), 16);\n const g = parseInt(hex.slice(3, 5), 16);\n const b = parseInt(hex.slice(5, 7), 16);\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n\ninterface PomoPrefs {\n focusMinutes: number;\n shortBreakMinutes: number;\n longBreakMinutes: number;\n autoStartBreaks: boolean;\n autoStartPomodoros: boolean;\n longBreakInterval: number;\n autoCheckTasks: boolean;\n checkToBottom: boolean;\n alarmSound: AlarmSound;\n alarmVolume: number;\n alarmRepeat: number;\n focusSound: FocusSound;\n focusVolume: number;\n}\nconst DEFAULT_PREFS: PomoPrefs = {\n focusMinutes: 25, shortBreakMinutes: 5, longBreakMinutes: 15,\n autoStartBreaks: false, autoStartPomodoros: false, longBreakInterval: 4,\n autoCheckTasks: true, checkToBottom: true,\n alarmSound: 'bell', alarmVolume: 100, alarmRepeat: 1,\n focusSound: 'none', focusVolume: 50,\n};\n\nconst intInRange = (v: unknown, min: number, max: number, fallback: number) =>\n typeof v === 'number' && isFinite(v) ? Math.max(min, Math.min(max, Math.round(v))) : fallback;\n\nfunction readPrefs(raw: unknown): PomoPrefs {\n const p = (raw && typeof raw === 'object') ? raw as Partial<PomoPrefs> : {};\n return {\n focusMinutes: intInRange(p.focusMinutes, 1, 120, DEFAULT_PREFS.focusMinutes),\n shortBreakMinutes: intInRange(p.shortBreakMinutes, 1, 120, DEFAULT_PREFS.shortBreakMinutes),\n longBreakMinutes: intInRange(p.longBreakMinutes, 1, 120, DEFAULT_PREFS.longBreakMinutes),\n autoStartBreaks: typeof p.autoStartBreaks === 'boolean' ? p.autoStartBreaks : DEFAULT_PREFS.autoStartBreaks,\n autoStartPomodoros: typeof p.autoStartPomodoros === 'boolean' ? p.autoStartPomodoros : DEFAULT_PREFS.autoStartPomodoros,\n longBreakInterval: intInRange(p.longBreakInterval, 1, 20, DEFAULT_PREFS.longBreakInterval),\n autoCheckTasks: typeof p.autoCheckTasks === 'boolean' ? p.autoCheckTasks : DEFAULT_PREFS.autoCheckTasks,\n checkToBottom: typeof p.checkToBottom === 'boolean' ? p.checkToBottom : DEFAULT_PREFS.checkToBottom,\n alarmSound: ALARM_OPTIONS.some(o => o.id === p.alarmSound) ? p.alarmSound as AlarmSound : DEFAULT_PREFS.alarmSound,\n alarmVolume: intInRange(p.alarmVolume, 0, 100, DEFAULT_PREFS.alarmVolume),\n alarmRepeat: intInRange(p.alarmRepeat, 1, 5, DEFAULT_PREFS.alarmRepeat),\n focusSound: FOCUS_SOUND_OPTIONS.some(o => o.id === p.focusSound) ? p.focusSound as FocusSound : DEFAULT_PREFS.focusSound,\n focusVolume: intInRange(p.focusVolume, 0, 100, DEFAULT_PREFS.focusVolume),\n };\n}\n\n// The Pomodoro widget treats each TodoTask as if it has the four\n// fields it cares about (name / estimated / completed / done). Tasks\n// without `estimated` show as `?` slots; tasks without `completed`\n// start at 0 and increment when their pomos finish.\nconst sortDoneToBottom = (arr: TodoTask[]): TodoTask[] => {\n const undone: TodoTask[] = [];\n const done: TodoTask[] = [];\n for (const t of arr) (t.done ? done : undone).push(t);\n return [...undone, ...done];\n};\n\nfunction useMemoSortedTasks(rawTasks: TodoTask[], checkToBottom: boolean): TodoTask[] {\n return useMemo(\n () => checkToBottom ? sortDoneToBottom(rawTasks) : rawTasks,\n [rawTasks, checkToBottom],\n );\n}\n\nexport default function PomodoroTimer() {\n const snap = useSyncExternalStore(subscribePomo, getPomoSnapshot, getPomoSnapshot);\n const { prefs: shellPrefs, save: saveShellPrefs } = useShellPrefs();\n const userPrefs = readPrefs(shellPrefs.pomodoro_settings);\n\n // Sync prefs → store. Split into focused effects so each setting only\n // resets the bit it owns (e.g. flipping the alarm volume doesn't tear\n // down the focus-sound playback).\n useEffect(() => {\n setPomoDurations({\n focus: userPrefs.focusMinutes * 60,\n short: userPrefs.shortBreakMinutes * 60,\n long: userPrefs.longBreakMinutes * 60,\n });\n }, [userPrefs.focusMinutes, userPrefs.shortBreakMinutes, userPrefs.longBreakMinutes]);\n\n useEffect(() => {\n setPomoAlarm(userPrefs.alarmSound);\n setPomoAlarmConfig(userPrefs.alarmVolume, userPrefs.alarmRepeat);\n }, [userPrefs.alarmSound, userPrefs.alarmVolume, userPrefs.alarmRepeat]);\n\n useEffect(() => {\n setPomoBehaviour({\n autoStartBreaks: userPrefs.autoStartBreaks,\n autoStartPomodoros: userPrefs.autoStartPomodoros,\n longBreakInterval: userPrefs.longBreakInterval,\n });\n }, [userPrefs.autoStartBreaks, userPrefs.autoStartPomodoros, userPrefs.longBreakInterval]);\n\n useEffect(() => {\n setPomoFocusSound(userPrefs.focusSound, userPrefs.focusVolume);\n }, [userPrefs.focusSound, userPrefs.focusVolume]);\n\n // Ask once for desktop-notification permission.\n useEffect(() => {\n if (typeof Notification !== 'undefined' && Notification.permission === 'default') {\n try { Notification.requestPermission(); } catch {}\n }\n }, []);\n\n // ── Tasks (shared store) ──\n // Tasks now live in `shellPrefs.todo_tasks` so the Todo List app and\n // Calendar app see the same data. The Pomodoro widget keeps a\n // per-instance `activeTaskId` in localStorage so the user's selection\n // doesn't leak across windows.\n //\n // The Pomodoro list is intentionally narrow — it only shows tasks\n // that are *actionable today*: anything due today (pending or just-\n // completed) plus anything overdue and still pending. Far-future\n // tasks live in the Todo List app; completed tasks from earlier days\n // roll off automatically. Tasks added via the Pomodoro's `+ Add Task`\n // button default to today's `dueDate` so they appear immediately in\n // this filtered view.\n const todo = useTodoTasks();\n const rawTasks = todo.tasks;\n const todayStr = useMemo(() => new Date().toISOString().slice(0, 10), []);\n const visibleRawTasks = useMemo(\n () => rawTasks.filter(t => {\n if (!t.dueDate) return false;\n if (t.dueDate === todayStr) return true; // due today — show even when done\n if (t.dueDate < todayStr && !t.done) return true; // overdue and still pending\n return false;\n }),\n [rawTasks, todayStr],\n );\n const tasks = useMemoSortedTasks(visibleRawTasks, userPrefs.checkToBottom);\n\n const [activeTaskId, setActiveTaskId] = useState<string | null>(() => localStorage.getItem(ACTIVE_TASK_KEY));\n const [adding, setAdding] = useState(false);\n const [menuOpenId, setMenuOpenId] = useState<string | null>(null);\n\n // One-shot migration from the legacy localStorage `pomodoro_tasks` key.\n const migratedRef = useRef(false);\n useEffect(() => {\n if (migratedRef.current) return;\n migratedRef.current = true;\n migratePomodoroTasksOnce(rawTasks, todo.setAllTasks);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n try {\n if (activeTaskId) localStorage.setItem(ACTIVE_TASK_KEY, activeTaskId);\n else localStorage.removeItem(ACTIVE_TASK_KEY);\n } catch {}\n }, [activeTaskId]);\n\n // Close the row menu on any outside click.\n useEffect(() => {\n if (!menuOpenId) return;\n const handler = () => setMenuOpenId(null);\n const t = setTimeout(() => document.addEventListener('click', handler), 0);\n return () => { clearTimeout(t); document.removeEventListener('click', handler); };\n }, [menuOpenId]);\n\n // When a focus block completes (`streak` increments), bump the active\n // task's completed count. If `autoCheckTasks` is on and it hits the\n // estimate, mark it done. (`checkToBottom` is honoured by the\n // sortedTasks view; the underlying store doesn't reorder.)\n const lastStreakRef = useRef(snap.streak);\n useEffect(() => {\n if (snap.streak > lastStreakRef.current && activeTaskId) {\n const t = rawTasks.find(x => x.id === activeTaskId);\n if (t) {\n const completed = (t.completed ?? 0) + 1;\n const estimated = t.estimated ?? 0;\n const shouldAutoCheck = userPrefs.autoCheckTasks && estimated > 0 && completed >= estimated;\n todo.updateTask(activeTaskId, { completed, done: t.done || shouldAutoCheck });\n }\n }\n lastStreakRef.current = snap.streak;\n }, [snap.streak, activeTaskId, userPrefs.autoCheckTasks, rawTasks, todo]);\n\n const addTask = (name: string, estimated: number) => {\n const trimmed = name.trim();\n if (!trimmed) return;\n // Default the deadline to today so the new task immediately appears\n // in the Pomodoro's today/overdue list. The user can change the\n // due date later from the Todo List app's edit drawer.\n const id = todo.addTask({ name: trimmed, estimated, completed: 0, dueDate: todayStr });\n if (!activeTaskId) setActiveTaskId(id);\n setAdding(false);\n };\n const removeTask = (id: string) => {\n todo.removeTask(id);\n if (activeTaskId === id) setActiveTaskId(null);\n setMenuOpenId(null);\n };\n const toggleDone = (id: string) => {\n todo.toggleDone(id);\n };\n\n // ── Stats ──\n const totalCompleted = tasks.reduce((acc, t) => acc + (t.completed ?? 0), 0);\n const totalEstimated = tasks.reduce((acc, t) => acc + Math.max(t.estimated ?? 0, t.completed ?? 0), 0);\n const remainingPomos = tasks.reduce((acc, t) => t.done ? acc : acc + Math.max(0, (t.estimated ?? 0) - (t.completed ?? 0)), 0);\n const remainingSecsFromTimer = (snap.running && snap.mode === 'focus') ? snap.remaining : 0;\n const remainingSecs = remainingPomos * userPrefs.focusMinutes * 60 + remainingSecsFromTimer;\n const finishAt = new Date(Date.now() + remainingSecs * 1000);\n const finishAtStr = finishAt.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit', hour12: false });\n const totalHours = (remainingSecs / 3600).toFixed(1);\n\n // Look the active task up in the *full* `rawTasks` list, not the\n // filtered `tasks`. That way the indicator above the list still shows\n // the active task name even if the user moved its due date out of\n // today/overdue range from the Todo List app.\n const activeTask = activeTaskId ? rawTasks.find(t => t.id === activeTaskId) ?? null : null;\n\n const mm = String(Math.floor(snap.remaining / 60)).padStart(2, '0');\n const ss = String(snap.remaining % 60).padStart(2, '0');\n\n // ── Settings modal ──\n const [appearance, setAppearance] = useState(() => loadAppearance(POMO_SETTINGS_KEY));\n const [settingsOpen, setSettingsOpen] = useState(false);\n const [configAppearance, setConfigAppearance] = useState<WidgetAppearance>(appearance);\n const [configPrefs, setConfigPrefs] = useState<PomoPrefs>(userPrefs);\n\n useWidgetSettings(useCallback(() => {\n setConfigAppearance({ ...appearance });\n setConfigPrefs({ ...userPrefs });\n setSettingsOpen(true);\n }, [appearance, userPrefs]));\n\n const onSave = () => {\n setAppearance(configAppearance);\n localStorage.setItem(POMO_SETTINGS_KEY, JSON.stringify(configAppearance));\n saveShellPrefs({ pomodoro_settings: configPrefs });\n setSettingsOpen(false);\n };\n\n // Focus mode: panel takes the taskbar's colour (light or dark per\n // theme), text inherits the system theme via Tailwind gray-* classes.\n // Break modes: keep their hex-based colored panels with white text.\n const isColored = snap.mode !== 'focus';\n const bg = MODE_COLORS[snap.mode];\n const panelBg = isColored\n ? hexToRgba(bg, appearance.activeOpacity / 100)\n : `rgb(var(--taskbar-bg-rgb, 243 244 246) / ${appearance.activeOpacity / 100})`;\n\n // Class helpers — switch text colours based on whether the panel is a\n // bright break colour or the (theme-aware) focus panel.\n const tx = {\n primary: isColored ? 'text-white' : 'text-gray-900',\n secondary: isColored ? 'text-white/85' : 'text-gray-700',\n muted: isColored ? 'text-white/65' : 'text-gray-500',\n faded: isColored ? 'text-white/55' : 'text-gray-400',\n tabActive: isColored ? 'bg-white/15 text-white' : 'bg-gray-200 text-gray-900',\n tabInactive: isColored ? 'text-white/75 hover:bg-white/10' : 'text-gray-500 hover:bg-gray-200',\n divider: isColored ? 'border-white/30' : 'border-gray-200',\n softDivider: isColored ? 'border-white/20' : 'border-gray-200',\n iconBtn: isColored ? 'bg-black/15 hover:bg-black/25 text-white/85' : 'bg-gray-200 hover:bg-gray-300 text-gray-600',\n addTaskBtn: isColored ? 'border-white/45 text-white/90 hover:bg-white/[0.06]' : 'border-gray-300 text-gray-500 hover:bg-gray-200/50',\n };\n // START / Save accent colour: in colored modes use the mode hex; in\n // focus mode use a fixed dark slate so the white pill stays high-\n // contrast on both light and dark themes.\n const accentColor = isColored ? bg : '#0f172a';\n\n return (\n <>\n <div className={`flex flex-col h-full select-none transition-colors duration-300 ${tx.primary}`}\n style={{\n backgroundColor: panelBg,\n backdropFilter: appearance.activeBlur > 0 ? `blur(${appearance.activeBlur}px)` : undefined,\n }}>\n\n {/* Mode tabs */}\n <div className=\"px-3 pt-3 flex justify-center gap-1.5\">\n {(['focus', 'short', 'long'] as Mode[]).map(m => (\n <button key={m} onClick={() => pomoSwitchMode(m)}\n className={`px-2.5 py-1 text-[13px] font-bold rounded transition-colors ${snap.mode === m ? tx.tabActive : tx.tabInactive}`}>\n {MODE_LABELS[m]}\n </button>\n ))}\n </div>\n\n {/* Big timer */}\n <div className=\"text-center font-bold tabular-nums leading-none tracking-tight mt-3\" style={{ fontSize: '4.5rem' }}>\n {mm}:{ss}\n </div>\n\n {/* START / PAUSE button */}\n <div className=\"flex justify-center mt-3 mb-2\">\n <button onClick={() => snap.running ? pomoPause() : pomoStart()}\n className=\"bg-white px-12 py-2.5 rounded-md font-bold text-base shadow-[0_4px_0_rgba(0,0,0,0.12)] hover:shadow-[0_3px_0_rgba(0,0,0,0.12)] active:translate-y-1 active:shadow-none transition-all\"\n style={{ color: accentColor, letterSpacing: '0.08em' }}>\n {snap.running ? 'PAUSE' : 'START'}\n </button>\n </div>\n\n {/* Active task indicator */}\n <div className=\"px-3 py-3 text-center\">\n {activeTask ? (\n <>\n <div className={`text-xs leading-tight ${tx.muted}`}>#{(activeTask.completed ?? 0) + 1}</div>\n <div className={`text-[15px] font-medium leading-tight truncate mt-0.5 ${tx.primary}`}>{activeTask.name}</div>\n </>\n ) : (\n <div className={`text-xs italic ${tx.muted}`}>No task selected</div>\n )}\n </div>\n\n {/* Tasks header */}\n <div className=\"px-3 mt-1 flex items-center justify-between\">\n <h3 className={`text-base font-bold tracking-tight ${tx.primary}`}>Tasks</h3>\n <button className={`rounded p-1 transition-colors ${tx.iconBtn}`} aria-label=\"Tasks menu\" tabIndex={-1}>\n <svg className=\"h-4 w-4\" viewBox=\"0 0 24 24\" fill=\"currentColor\">\n <circle cx=\"12\" cy=\"5\" r=\"1.4\" /><circle cx=\"12\" cy=\"12\" r=\"1.4\" /><circle cx=\"12\" cy=\"19\" r=\"1.4\" />\n </svg>\n </button>\n </div>\n <div className={`border-t mx-3 mt-2 ${tx.divider}`} />\n\n {/* Tasks list */}\n <div className=\"flex-1 px-3 py-3 space-y-2 overflow-y-auto\">\n {tasks.map(task => {\n const isActive = task.id === activeTaskId;\n return (\n <div key={task.id}\n onClick={() => setActiveTaskId(task.id)}\n className={`bg-white text-gray-800 rounded-md flex items-center pr-2 py-2.5 shadow-sm cursor-pointer transition-shadow hover:shadow-md ${isActive ? 'border-l-4 border-gray-700 pl-2' : 'pl-3 border-l-4 border-transparent'}`}>\n <button onClick={(e) => { e.stopPropagation(); toggleDone(task.id); }}\n className=\"shrink-0\">\n {task.done ? (\n <svg className=\"h-5 w-5\" fill=\"currentColor\" viewBox=\"0 0 20 20\" style={{ color: accentColor }}>\n <path d=\"M10 0a10 10 0 100 20 10 10 0 000-20zm-1 14.5l-4.5-4.5 1.4-1.4 3.1 3.1 6.1-6.1 1.4 1.4z\" />\n </svg>\n ) : (\n <svg className=\"h-5 w-5 text-gray-300\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2} viewBox=\"0 0 24 24\">\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n </svg>\n )}\n </button>\n <span className={`ml-3 flex-1 font-semibold truncate ${task.done ? 'line-through text-gray-400' : 'text-gray-800'}`}>\n {task.name}\n </span>\n <span className=\"text-sm tabular-nums mr-1.5 shrink-0\">\n <span className=\"font-bold text-gray-500\">{task.completed ?? 0}</span>\n <span className=\"text-gray-400\"> / </span>\n <span className=\"text-gray-400\">{task.estimated ?? '–'}</span>\n </span>\n <div className=\"relative\">\n <button onClick={(e) => { e.stopPropagation(); setMenuOpenId(menuOpenId === task.id ? null : task.id); }}\n className=\"rounded p-1 bg-gray-100 hover:bg-gray-200\">\n <svg className=\"h-4 w-4 text-gray-500\" viewBox=\"0 0 24 24\" fill=\"currentColor\">\n <circle cx=\"12\" cy=\"5\" r=\"1.4\" /><circle cx=\"12\" cy=\"12\" r=\"1.4\" /><circle cx=\"12\" cy=\"19\" r=\"1.4\" />\n </svg>\n </button>\n {menuOpenId === task.id && (\n <div className=\"absolute right-0 top-full mt-1 bg-white text-gray-800 rounded shadow-lg border border-gray-200 z-10 min-w-[110px] overflow-hidden\"\n onClick={(e) => e.stopPropagation()}>\n <button onClick={(e) => { e.stopPropagation(); removeTask(task.id); }}\n className=\"block w-full px-3 py-1.5 text-sm text-left hover:bg-red-50 text-red-600\">Delete</button>\n </div>\n )}\n </div>\n </div>\n );\n })}\n\n {/* + Add Task */}\n {!adding ? (\n <button onClick={() => setAdding(true)}\n className={`w-full border-2 border-dashed rounded-md py-3 font-semibold flex items-center justify-center gap-2 transition-colors ${tx.addTaskBtn}`}>\n <svg className=\"h-5 w-5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n <path strokeLinecap=\"round\" d=\"M8 12h8M12 8v8\" />\n </svg>\n Add Task\n </button>\n ) : (\n <AddTaskForm onSubmit={addTask} onCancel={() => setAdding(false)} accentColor={accentColor} />\n )}\n </div>\n\n {/* Footer stats */}\n <div className={`border-t px-3 py-2.5 flex items-center justify-center gap-5 text-sm ${tx.softDivider}`}>\n <span className={tx.secondary}>\n Pomos: <span className={`font-bold ${tx.primary}`}>{totalCompleted}</span>\n <span className={tx.muted}> / </span>\n <span className={`font-bold ${tx.primary}`}>{totalEstimated}</span>\n </span>\n {remainingSecs > 0 && (\n <span className={tx.secondary}>\n Finish At: <span className={`font-bold ${tx.primary}`}>{finishAtStr}</span>\n <span className={tx.muted}> ({totalHours}h)</span>\n </span>\n )}\n </div>\n </div>\n\n <PomodoroSettings\n open={settingsOpen} onClose={() => setSettingsOpen(false)} onSave={onSave}\n configAppearance={configAppearance} setConfigAppearance={setConfigAppearance}\n configPrefs={configPrefs} setConfigPrefs={setConfigPrefs} />\n </>\n );\n}\n\n// ─────────────────────────────────────────────────────────────────────────\n// Settings UI — three sectioned cards (TIMER / TASK / SOUND) matching the\n// reference design. Renders into the standard `WidgetSettingsModal` so we\n// inherit its appearance sliders and the modal chrome.\n// ─────────────────────────────────────────────────────────────────────────\n\nfunction PomodoroSettings({\n open, onClose, onSave, configAppearance, setConfigAppearance, configPrefs, setConfigPrefs,\n}: {\n open: boolean;\n onClose: () => void;\n onSave: () => void;\n configAppearance: WidgetAppearance;\n setConfigAppearance: (a: WidgetAppearance) => void;\n configPrefs: PomoPrefs;\n setConfigPrefs: React.Dispatch<React.SetStateAction<PomoPrefs>>;\n}) {\n const set = <K extends keyof PomoPrefs>(k: K, v: PomoPrefs[K]) => setConfigPrefs(p => ({ ...p, [k]: v }));\n\n return (\n <WidgetSettingsModal open={open} onClose={onClose} title=\"Pomodoro Settings\"\n appearance={configAppearance} onAppearanceChange={setConfigAppearance} onSave={onSave}>\n\n {/* ── TIMER ── */}\n <Section icon={<ClockIcon />} label=\"TIMER\">\n <div>\n <p className=\"text-sm font-bold text-gray-800 mb-2\">Time (minutes)</p>\n <div className=\"grid grid-cols-3 gap-3\">\n <NumberField label=\"Pomodoro\" value={configPrefs.focusMinutes} onChange={v => set('focusMinutes', v)} max={120} />\n <NumberField label=\"Short Break\" value={configPrefs.shortBreakMinutes} onChange={v => set('shortBreakMinutes', v)} max={120} />\n <NumberField label=\"Long Break\" value={configPrefs.longBreakMinutes} onChange={v => set('longBreakMinutes', v)} max={120} />\n </div>\n </div>\n\n <Row label=\"Auto Start Breaks\">\n <Toggle checked={configPrefs.autoStartBreaks} onChange={v => set('autoStartBreaks', v)} />\n </Row>\n\n <Row label=\"Auto Start Pomodoros\">\n <Toggle checked={configPrefs.autoStartPomodoros} onChange={v => set('autoStartPomodoros', v)} />\n </Row>\n\n <Row label=\"Long Break interval\">\n <NumberInput value={configPrefs.longBreakInterval} onChange={v => set('longBreakInterval', v)} min={1} max={20} className=\"w-14 text-center\" />\n </Row>\n </Section>\n\n {/* ── TASK ── */}\n <Section icon={<TaskIcon />} label=\"TASK\">\n <Row\n label=\"Auto Check Tasks\"\n info={'If you enable \"Auto Check Tasks\", the active task will be automatically checked when the actual pomodoro count reaches the estimated count.'}>\n <Toggle checked={configPrefs.autoCheckTasks} onChange={v => set('autoCheckTasks', v)} />\n </Row>\n <Row\n label=\"Check to Bottom\"\n info={'If you enable \"Check to Bottom\", the checked task will be automatically moved to the bottom of the task list.'}>\n <Toggle checked={configPrefs.checkToBottom} onChange={v => set('checkToBottom', v)} />\n </Row>\n </Section>\n\n {/* ── SOUND ── */}\n <Section icon={<SpeakerIcon />} label=\"SOUND\">\n <Row label=\"Alarm Sound\">\n <div className=\"flex items-center gap-2\">\n <Dropdown value={configPrefs.alarmSound} onChange={v => set('alarmSound', v as AlarmSound)}\n options={ALARM_OPTIONS.map(o => ({ id: o.id, label: o.label }))} />\n {configPrefs.alarmSound !== 'off' && (\n <button type=\"button\" onClick={() => playAlarm(configPrefs.alarmSound, { volume: configPrefs.alarmVolume, repeat: 1 })}\n className=\"px-2 py-1 text-[11px] rounded border border-gray-300 text-gray-600 hover:bg-gray-100\">\n Play\n </button>\n )}\n </div>\n </Row>\n <SliderRow value={configPrefs.alarmVolume} onChange={v => set('alarmVolume', v)} />\n <Row label=\"repeat\">\n <NumberInput value={configPrefs.alarmRepeat} onChange={v => set('alarmRepeat', v)} min={1} max={5} className=\"w-14 text-center\" />\n </Row>\n\n <Row label=\"Focus Sound\">\n <div className=\"flex items-center gap-2\">\n <Dropdown value={configPrefs.focusSound} onChange={v => set('focusSound', v as FocusSound)}\n options={FOCUS_SOUND_OPTIONS} />\n {configPrefs.focusSound !== 'none' && (\n <button type=\"button\" onClick={() => previewFocusSound(configPrefs.focusSound, configPrefs.focusVolume)}\n className=\"px-2 py-1 text-[11px] rounded border border-gray-300 text-gray-600 hover:bg-gray-100\">\n Play\n </button>\n )}\n </div>\n </Row>\n <SliderRow value={configPrefs.focusVolume} onChange={v => set('focusVolume', v)} />\n </Section>\n </WidgetSettingsModal>\n );\n}\n\n// ─────────────────────────────────────────────────────────────────────────\n// Tiny presentational helpers — kept here (not exported) so the settings\n// markup above reads top-to-bottom without any noise.\n// ─────────────────────────────────────────────────────────────────────────\n\nfunction Section({ icon, label, children }: { icon: React.ReactNode; label: string; children: React.ReactNode }) {\n return (\n <div className=\"border-t border-gray-200 pt-3 first:border-0 first:pt-0\">\n <div className=\"flex items-center gap-1.5 mb-3 text-gray-400\">\n {icon}\n <span className=\"text-xs font-bold tracking-[0.15em]\">{label}</span>\n </div>\n <div className=\"space-y-3\">{children}</div>\n </div>\n );\n}\n\nfunction Row({ label, info, children }: { label: string; info?: string; children: React.ReactNode }) {\n return (\n <div className=\"flex items-center justify-between gap-3\">\n <div className=\"flex items-center gap-1.5 min-w-0\">\n <span className=\"text-sm font-semibold text-gray-800\">{label}</span>\n {info && <InfoIcon title={info} />}\n </div>\n <div className=\"shrink-0\">{children}</div>\n </div>\n );\n}\n\nfunction SliderRow({ value, onChange }: { value: number; onChange: (v: number) => void }) {\n return (\n <div className=\"flex items-center justify-end gap-3\">\n <span className=\"text-xs text-gray-400 w-7 text-right tabular-nums\">{value}</span>\n <input type=\"range\" min={0} max={100} value={value}\n onChange={e => onChange(parseInt(e.target.value, 10))}\n className=\"w-40 accent-blue-500\" />\n </div>\n );\n}\n\nfunction NumberField({ label, value, onChange, min = 1, max = 99 }: { label: string; value: number; onChange: (v: number) => void; min?: number; max?: number }) {\n return (\n <label className=\"flex flex-col\">\n <span className=\"text-xs font-semibold text-gray-500 mb-1\">{label}</span>\n <NumberInput value={value} onChange={onChange} min={min} max={max} />\n </label>\n );\n}\n\nfunction NumberInput({ value, onChange, min = 1, max = 99, className = '' }: { value: number; onChange: (v: number) => void; min?: number; max?: number; className?: string }) {\n return (\n <input type=\"number\" min={min} max={max} value={value}\n onChange={e => onChange(Math.max(min, Math.min(max, parseInt(e.target.value, 10) || min)))}\n className={`bg-gray-100 border-0 rounded-md px-2 py-1.5 text-sm font-medium text-gray-800 focus:outline-none focus:ring-2 focus:ring-blue-500 w-full ${className}`} />\n );\n}\n\nfunction Dropdown<T extends string>({ value, onChange, options }: { value: T; onChange: (v: T) => void; options: { id: T; label: string }[] }) {\n return (\n <div className=\"relative\">\n <select value={value} onChange={e => onChange(e.target.value as T)}\n className=\"appearance-none bg-gray-100 border-0 rounded-md pl-3 pr-8 py-1.5 text-sm font-medium text-gray-800 focus:outline-none focus:ring-2 focus:ring-blue-500 cursor-pointer\">\n {options.map(o => <option key={o.id} value={o.id}>{o.label}</option>)}\n </select>\n <svg className=\"absolute right-2 top-1/2 -translate-y-1/2 h-3.5 w-3.5 text-gray-400 pointer-events-none\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2.5}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M6 9l6 6 6-6\" />\n </svg>\n </div>\n );\n}\n\nfunction Toggle({ checked, onChange }: { checked: boolean; onChange: (v: boolean) => void }) {\n return (\n <button type=\"button\" role=\"switch\" aria-checked={checked} onClick={() => onChange(!checked)}\n className={`relative inline-flex h-6 w-11 items-center rounded-full transition-colors ${checked ? 'bg-emerald-500' : 'bg-gray-300'}`}>\n <span className={`absolute h-5 w-5 rounded-full bg-white shadow transition-transform ${checked ? 'translate-x-[22px]' : 'translate-x-0.5'}`} />\n </button>\n );\n}\n\nfunction InfoIcon({ title }: { title: string }) {\n return (\n <span title={title} className=\"inline-flex items-center justify-center w-4 h-4 rounded-full bg-gray-300 text-white text-[10px] font-bold cursor-help\">i</span>\n );\n}\n\nfunction ClockIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <circle cx=\"12\" cy=\"12\" r=\"9\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M12 7v5l3 2\" />\n </svg>\n );\n}\n\nfunction TaskIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <rect x=\"4\" y=\"4\" width=\"16\" height=\"16\" rx=\"2\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M9 12l2 2 4-4\" />\n </svg>\n );\n}\n\nfunction SpeakerIcon() {\n return (\n <svg className=\"h-3.5 w-3.5\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" strokeWidth={2}>\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M11 5L6 9H2v6h4l5 4V5z\" />\n <path strokeLinecap=\"round\" strokeLinejoin=\"round\" d=\"M15.5 8.5a5 5 0 010 7M19 5a9 9 0 010 14\" />\n </svg>\n );\n}\n\nfunction AddTaskForm({ onSubmit, onCancel, accentColor }: { onSubmit: (name: string, estimated: number) => void; onCancel: () => void; accentColor: string }) {\n const [name, setName] = useState('');\n const [est, setEst] = useState(1);\n const submit = () => onSubmit(name, est);\n return (\n <div className=\"bg-white text-gray-800 rounded-md p-3 shadow-md\">\n <input autoFocus value={name} onChange={e => setName(e.target.value)} placeholder=\"What are you working on?\"\n onKeyDown={e => { if (e.key === 'Enter') submit(); if (e.key === 'Escape') onCancel(); }}\n className=\"w-full text-base font-medium bg-transparent border-0 outline-none placeholder-gray-400\" />\n <div className=\"flex items-center justify-between mt-2\">\n <span className=\"text-xs font-semibold text-gray-500\">Est Pomodoros</span>\n <input type=\"number\" min={1} max={20} value={est}\n onChange={e => setEst(Math.max(1, Math.min(20, parseInt(e.target.value, 10) || 1)))}\n className=\"w-16 bg-gray-100 rounded px-2 py-1 text-sm text-right border-0 focus:outline-none focus:ring-2 focus:ring-blue-500\" />\n </div>\n <div className=\"flex gap-2 mt-3 justify-end\">\n <button onClick={onCancel} className=\"px-3 py-1 text-sm text-gray-500 hover:bg-gray-100 rounded\">Cancel</button>\n <button onClick={submit}\n className=\"px-4 py-1 text-sm font-semibold text-white rounded shadow-sm\"\n style={{ backgroundColor: accentColor }}>Save</button>\n </div>\n </div>\n );\n}\n"]}
@@ -0,0 +1,7 @@
1
+ export { Preview as default, setPdfPreview } from './chunk-QBH7KERS.js';
2
+ import './chunk-KUIPWCTJ.js';
3
+ import './chunk-WIJ45SYD.js';
4
+ import './chunk-7M3BBAHQ.js';
5
+ import './chunk-PLGHQ7QW.js';
6
+ //# sourceMappingURL=Preview-B5DUW2AR.js.map
7
+ //# sourceMappingURL=Preview-B5DUW2AR.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"Preview-4MBQI66Q.js"}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"Preview-B5DUW2AR.js"}