bingocode 1.0.2 → 1.0.3

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 (186) hide show
  1. package/desktop/README.md +30 -0
  2. package/desktop/bunfig.toml +1 -0
  3. package/desktop/index.html +17 -0
  4. package/desktop/package.json +55 -0
  5. package/desktop/pnpm-lock.yaml +3832 -0
  6. package/desktop/public/app-icon.jpg +0 -0
  7. package/desktop/public/fonts/inter-latin-ext.woff2 +0 -0
  8. package/desktop/public/fonts/inter-latin.woff2 +0 -0
  9. package/desktop/public/fonts/jetbrains-mono-latin-ext.woff2 +0 -0
  10. package/desktop/public/fonts/jetbrains-mono-latin.woff2 +0 -0
  11. package/desktop/public/fonts/manrope-latin-ext.woff2 +0 -0
  12. package/desktop/public/fonts/manrope-latin.woff2 +0 -0
  13. package/desktop/public/fonts/material-symbols-outlined.woff2 +0 -0
  14. package/desktop/public/icons/bilibili.svg +1 -0
  15. package/desktop/public/icons/douyin.svg +1 -0
  16. package/desktop/public/icons/github.svg +3 -0
  17. package/desktop/public/icons/xiaohongshu.svg +1 -0
  18. package/desktop/scripts/build-macos-arm64.sh +270 -0
  19. package/desktop/scripts/build-sidecars.ts +183 -0
  20. package/desktop/scripts/build-windows-x64.ps1 +295 -0
  21. package/desktop/scripts/scan-missing-imports.ts +235 -0
  22. package/desktop/sidecars/claude-sidecar.ts +156 -0
  23. package/desktop/src/App.tsx +5 -0
  24. package/desktop/src/__tests__/agentsSettings.test.tsx +349 -0
  25. package/desktop/src/__tests__/pages.test.tsx +290 -0
  26. package/desktop/src/__tests__/skillsSettings.test.tsx +205 -0
  27. package/desktop/src/api/adapters.ts +12 -0
  28. package/desktop/src/api/agents.ts +36 -0
  29. package/desktop/src/api/cliTasks.ts +28 -0
  30. package/desktop/src/api/client.ts +63 -0
  31. package/desktop/src/api/computerUse.ts +76 -0
  32. package/desktop/src/api/filesystem.ts +30 -0
  33. package/desktop/src/api/hahaOAuth.ts +38 -0
  34. package/desktop/src/api/models.ts +28 -0
  35. package/desktop/src/api/providers.ts +63 -0
  36. package/desktop/src/api/search.ts +29 -0
  37. package/desktop/src/api/sessions.ts +56 -0
  38. package/desktop/src/api/settings.ts +20 -0
  39. package/desktop/src/api/skills.ts +19 -0
  40. package/desktop/src/api/tasks.ts +36 -0
  41. package/desktop/src/api/teams.ts +44 -0
  42. package/desktop/src/api/websocket.ts +164 -0
  43. package/desktop/src/components/chat/AskUserQuestion.tsx +268 -0
  44. package/desktop/src/components/chat/AssistantMessage.tsx +29 -0
  45. package/desktop/src/components/chat/AttachmentGallery.tsx +113 -0
  46. package/desktop/src/components/chat/ChatInput.tsx +622 -0
  47. package/desktop/src/components/chat/CodeViewer.tsx +161 -0
  48. package/desktop/src/components/chat/ComputerUsePermissionModal.test.tsx +174 -0
  49. package/desktop/src/components/chat/ComputerUsePermissionModal.tsx +311 -0
  50. package/desktop/src/components/chat/DiffViewer.tsx +157 -0
  51. package/desktop/src/components/chat/FileSearchMenu.tsx +198 -0
  52. package/desktop/src/components/chat/ImageGalleryModal.tsx +91 -0
  53. package/desktop/src/components/chat/InlineImageGallery.tsx +106 -0
  54. package/desktop/src/components/chat/InlineTaskSummary.tsx +60 -0
  55. package/desktop/src/components/chat/MermaidRenderer.test.tsx +98 -0
  56. package/desktop/src/components/chat/MermaidRenderer.tsx +361 -0
  57. package/desktop/src/components/chat/MessageActionBar.tsx +27 -0
  58. package/desktop/src/components/chat/MessageList.test.tsx +313 -0
  59. package/desktop/src/components/chat/MessageList.tsx +249 -0
  60. package/desktop/src/components/chat/PermissionDialog.tsx +262 -0
  61. package/desktop/src/components/chat/SessionTaskBar.test.tsx +99 -0
  62. package/desktop/src/components/chat/SessionTaskBar.tsx +159 -0
  63. package/desktop/src/components/chat/StreamingIndicator.tsx +41 -0
  64. package/desktop/src/components/chat/TerminalChrome.tsx +35 -0
  65. package/desktop/src/components/chat/ThinkingBlock.tsx +87 -0
  66. package/desktop/src/components/chat/ToolCallBlock.tsx +247 -0
  67. package/desktop/src/components/chat/ToolCallGroup.tsx +617 -0
  68. package/desktop/src/components/chat/ToolResultBlock.tsx +107 -0
  69. package/desktop/src/components/chat/UserMessage.tsx +38 -0
  70. package/desktop/src/components/chat/chatBlocks.test.tsx +136 -0
  71. package/desktop/src/components/chat/clipboard.ts +25 -0
  72. package/desktop/src/components/chat/composerUtils.test.ts +55 -0
  73. package/desktop/src/components/chat/composerUtils.ts +149 -0
  74. package/desktop/src/components/controls/ModelSelector.tsx +156 -0
  75. package/desktop/src/components/controls/PermissionModeSelector.tsx +229 -0
  76. package/desktop/src/components/layout/AppShell.tsx +107 -0
  77. package/desktop/src/components/layout/ContentRouter.tsx +27 -0
  78. package/desktop/src/components/layout/ProjectFilter.tsx +126 -0
  79. package/desktop/src/components/layout/Sidebar.test.tsx +158 -0
  80. package/desktop/src/components/layout/Sidebar.tsx +384 -0
  81. package/desktop/src/components/layout/StatusBar.tsx +31 -0
  82. package/desktop/src/components/layout/TabBar.test.tsx +136 -0
  83. package/desktop/src/components/layout/TabBar.tsx +318 -0
  84. package/desktop/src/components/layout/TitleBar.tsx +96 -0
  85. package/desktop/src/components/layout/WindowControls.test.tsx +69 -0
  86. package/desktop/src/components/layout/WindowControls.tsx +89 -0
  87. package/desktop/src/components/markdown/MarkdownRenderer.test.tsx +100 -0
  88. package/desktop/src/components/markdown/MarkdownRenderer.tsx +229 -0
  89. package/desktop/src/components/settings/ClaudeOfficialLogin.tsx +107 -0
  90. package/desktop/src/components/shared/Button.tsx +63 -0
  91. package/desktop/src/components/shared/CopyButton.tsx +58 -0
  92. package/desktop/src/components/shared/DirectoryPicker.tsx +316 -0
  93. package/desktop/src/components/shared/Dropdown.tsx +91 -0
  94. package/desktop/src/components/shared/Input.tsx +38 -0
  95. package/desktop/src/components/shared/Modal.tsx +65 -0
  96. package/desktop/src/components/shared/ProjectContextChip.tsx +30 -0
  97. package/desktop/src/components/shared/Spinner.tsx +30 -0
  98. package/desktop/src/components/shared/Textarea.tsx +38 -0
  99. package/desktop/src/components/shared/Toast.tsx +47 -0
  100. package/desktop/src/components/shared/UpdateChecker.tsx +90 -0
  101. package/desktop/src/components/skills/SkillDetail.test.tsx +89 -0
  102. package/desktop/src/components/skills/SkillDetail.tsx +403 -0
  103. package/desktop/src/components/skills/SkillList.tsx +254 -0
  104. package/desktop/src/components/tasks/DayOfWeekPicker.tsx +57 -0
  105. package/desktop/src/components/tasks/NewTaskModal.tsx +407 -0
  106. package/desktop/src/components/tasks/PromptEditor.tsx +74 -0
  107. package/desktop/src/components/tasks/TaskEmptyState.tsx +30 -0
  108. package/desktop/src/components/tasks/TaskList.tsx +46 -0
  109. package/desktop/src/components/tasks/TaskRow.tsx +253 -0
  110. package/desktop/src/components/tasks/TaskRunsPanel.tsx +195 -0
  111. package/desktop/src/components/teams/TeamStatusBar.tsx +147 -0
  112. package/desktop/src/config/providerPresets.ts +78 -0
  113. package/desktop/src/config/spinnerVerbs.ts +193 -0
  114. package/desktop/src/hooks/useKeyboardShortcuts.ts +60 -0
  115. package/desktop/src/i18n/index.ts +54 -0
  116. package/desktop/src/i18n/locales/en.ts +670 -0
  117. package/desktop/src/i18n/locales/zh.ts +670 -0
  118. package/desktop/src/lib/__tests__/cronDescribe.test.ts +93 -0
  119. package/desktop/src/lib/cronDescribe.ts +188 -0
  120. package/desktop/src/lib/desktopRuntime.ts +54 -0
  121. package/desktop/src/lib/parseRunOutput.ts +79 -0
  122. package/desktop/src/main.tsx +13 -0
  123. package/desktop/src/mocks/data.ts +202 -0
  124. package/desktop/src/pages/ActiveSession.test.tsx +181 -0
  125. package/desktop/src/pages/ActiveSession.tsx +219 -0
  126. package/desktop/src/pages/AdapterSettings.tsx +375 -0
  127. package/desktop/src/pages/AgentTeams.tsx +200 -0
  128. package/desktop/src/pages/ComputerUseSettings.tsx +420 -0
  129. package/desktop/src/pages/EmptySession.tsx +518 -0
  130. package/desktop/src/pages/NewTaskModal.tsx +346 -0
  131. package/desktop/src/pages/ScheduledTasks.tsx +66 -0
  132. package/desktop/src/pages/ScheduledTasksEmpty.tsx +152 -0
  133. package/desktop/src/pages/ScheduledTasksList.tsx +416 -0
  134. package/desktop/src/pages/SessionControls.tsx +460 -0
  135. package/desktop/src/pages/Settings.tsx +1448 -0
  136. package/desktop/src/pages/ToolInspection.tsx +235 -0
  137. package/desktop/src/stores/adapterStore.ts +106 -0
  138. package/desktop/src/stores/agentStore.ts +34 -0
  139. package/desktop/src/stores/chatStore.test.ts +505 -0
  140. package/desktop/src/stores/chatStore.ts +850 -0
  141. package/desktop/src/stores/cliTaskStore.ts +152 -0
  142. package/desktop/src/stores/hahaOAuthStore.test.ts +77 -0
  143. package/desktop/src/stores/hahaOAuthStore.ts +97 -0
  144. package/desktop/src/stores/providerStore.ts +101 -0
  145. package/desktop/src/stores/sessionStore.test.ts +63 -0
  146. package/desktop/src/stores/sessionStore.ts +102 -0
  147. package/desktop/src/stores/settingsStore.ts +120 -0
  148. package/desktop/src/stores/skillStore.ts +51 -0
  149. package/desktop/src/stores/tabStore.ts +169 -0
  150. package/desktop/src/stores/taskStore.ts +68 -0
  151. package/desktop/src/stores/teamStore.ts +344 -0
  152. package/desktop/src/stores/uiStore.ts +100 -0
  153. package/desktop/src/stores/updateStore.test.ts +71 -0
  154. package/desktop/src/stores/updateStore.ts +221 -0
  155. package/desktop/src/theme/globals.css +465 -0
  156. package/desktop/src/types/adapter.ts +33 -0
  157. package/desktop/src/types/chat.ts +152 -0
  158. package/desktop/src/types/cliTask.ts +24 -0
  159. package/desktop/src/types/provider.ts +62 -0
  160. package/desktop/src/types/session.ts +27 -0
  161. package/desktop/src/types/settings.ts +22 -0
  162. package/desktop/src/types/skill.ts +38 -0
  163. package/desktop/src/types/task.ts +56 -0
  164. package/desktop/src/types/team.ts +38 -0
  165. package/desktop/src-tauri/Cargo.lock +5549 -0
  166. package/desktop/src-tauri/Cargo.toml +20 -0
  167. package/desktop/src-tauri/app-icon.svg +13 -0
  168. package/desktop/src-tauri/build.rs +3 -0
  169. package/desktop/src-tauri/capabilities/default.json +106 -0
  170. package/desktop/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml +5 -0
  171. package/desktop/src-tauri/icons/android/values/ic_launcher_background.xml +4 -0
  172. package/desktop/src-tauri/icons/icon.icns +0 -0
  173. package/desktop/src-tauri/icons/icon.ico +0 -0
  174. package/desktop/src-tauri/src/lib.rs +408 -0
  175. package/desktop/src-tauri/src/main.rs +6 -0
  176. package/desktop/src-tauri/tauri.conf.json +78 -0
  177. package/desktop/src-tauri/tauri.macos.conf.json +18 -0
  178. package/desktop/src-tauri/tauri.release-ci.json +5 -0
  179. package/desktop/src-tauri/tauri.windows.conf.json +16 -0
  180. package/desktop/src-tauri/windows-installer-hooks.nsh +17 -0
  181. package/desktop/tsconfig.json +25 -0
  182. package/desktop/vite.config.ts +26 -0
  183. package/desktop/vitest.config.ts +18 -0
  184. package/package.json +1 -1
  185. package/src/commands/desktop/desktop.tsx +9 -0
  186. package/src/commands/desktop/index.ts +26 -0
@@ -0,0 +1,254 @@
1
+ import { useEffect, useMemo } from 'react'
2
+ import { useSkillStore } from '../../stores/skillStore'
3
+ import { useSessionStore } from '../../stores/sessionStore'
4
+ import { useTranslation } from '../../i18n'
5
+ import type { SkillMeta, SkillSource } from '../../types/skill'
6
+
7
+ const SOURCE_ORDER: SkillSource[] = ['user', 'project', 'plugin', 'mcp', 'bundled']
8
+
9
+ const SOURCE_ICONS: Record<SkillSource, string> = {
10
+ user: 'person',
11
+ project: 'folder',
12
+ plugin: 'extension',
13
+ mcp: 'hub',
14
+ bundled: 'inventory_2',
15
+ }
16
+
17
+ const SOURCE_ACCENT_CLASSES: Record<SkillSource, string> = {
18
+ user: 'bg-[var(--color-primary-fixed)] text-[var(--color-brand)]',
19
+ project: 'bg-[var(--color-success-container)] text-[var(--color-success)]',
20
+ plugin: 'bg-[var(--color-warning-container)] text-[var(--color-warning)]',
21
+ mcp: 'bg-[var(--color-info-container)] text-[var(--color-info)]',
22
+ bundled: 'bg-[var(--color-surface-container-high)] text-[var(--color-text-tertiary)]',
23
+ }
24
+
25
+ function estimateTokens(contentLength: number) {
26
+ return Math.ceil(contentLength / 4)
27
+ }
28
+
29
+ export function SkillList() {
30
+ const { skills, isLoading, error, fetchSkills, fetchSkillDetail } =
31
+ useSkillStore()
32
+ const sessions = useSessionStore((s) => s.sessions)
33
+ const activeSessionId = useSessionStore((s) => s.activeSessionId)
34
+ const t = useTranslation()
35
+ const activeSession = sessions.find((session) => session.id === activeSessionId)
36
+ const currentWorkDir = activeSession?.workDir || undefined
37
+
38
+ useEffect(() => {
39
+ fetchSkills(currentWorkDir)
40
+ }, [fetchSkills, currentWorkDir])
41
+
42
+ const grouped = useMemo(() => {
43
+ const result: Partial<Record<SkillSource, SkillMeta[]>> = {}
44
+ for (const skill of skills) {
45
+ const src = skill.source as SkillSource
46
+ ;(result[src] ??= []).push(skill)
47
+ }
48
+ return result
49
+ }, [skills])
50
+
51
+ const totalTokens = useMemo(
52
+ () => skills.reduce((sum, skill) => sum + estimateTokens(skill.contentLength), 0),
53
+ [skills],
54
+ )
55
+
56
+ const visibleGroupCount = useMemo(
57
+ () => SOURCE_ORDER.filter((source) => (grouped[source] ?? []).length > 0).length,
58
+ [grouped],
59
+ )
60
+
61
+ if (isLoading) {
62
+ return (
63
+ <div className="flex justify-center py-12">
64
+ <div className="animate-spin w-5 h-5 border-2 border-[var(--color-brand)] border-t-transparent rounded-full" />
65
+ </div>
66
+ )
67
+ }
68
+
69
+ if (error) {
70
+ return <div className="text-sm text-[var(--color-error)] py-4">{error}</div>
71
+ }
72
+
73
+ if (skills.length === 0) {
74
+ return (
75
+ <div className="text-center py-12 rounded-2xl border border-dashed border-[var(--color-border)] bg-[var(--color-surface-container-low)] px-6">
76
+ <span className="material-symbols-outlined text-[40px] text-[var(--color-text-tertiary)] mb-2 block">
77
+ auto_awesome
78
+ </span>
79
+ <p className="text-sm text-[var(--color-text-tertiary)]">
80
+ {t('settings.skills.empty')}
81
+ </p>
82
+ <p className="text-xs text-[var(--color-text-tertiary)] mt-1">
83
+ {t('settings.skills.emptyHint')}
84
+ </p>
85
+ </div>
86
+ )
87
+ }
88
+
89
+ return (
90
+ <div className="flex flex-col gap-6 min-w-0">
91
+ <section className="rounded-2xl border border-[var(--color-border)] bg-[var(--color-surface-container-low)] overflow-hidden">
92
+ <div className="grid gap-4 px-5 py-5 min-w-0 xl:grid-cols-[minmax(0,1.6fr)_minmax(320px,1fr)] xl:items-end">
93
+ <div className="min-w-0">
94
+ <div className="text-[11px] font-semibold uppercase tracking-[0.2em] text-[var(--color-text-tertiary)] mb-2">
95
+ {t('settings.skills.browserEyebrow')}
96
+ </div>
97
+ <div className="flex items-center gap-3 mb-2">
98
+ <span className="material-symbols-outlined text-[22px] text-[var(--color-brand)]">
99
+ auto_awesome
100
+ </span>
101
+ <h3 className="text-lg font-semibold text-[var(--color-text-primary)]">
102
+ {t('settings.skills.browserTitle')}
103
+ </h3>
104
+ </div>
105
+ <p className="text-sm leading-6 text-[var(--color-text-secondary)] max-w-3xl">
106
+ {t('settings.skills.browserDescription')}
107
+ </p>
108
+ </div>
109
+
110
+ <div className="grid grid-cols-2 gap-3 min-w-0 sm:grid-cols-3">
111
+ <SummaryCard
112
+ label={t('settings.skills.summary.totalSkills')}
113
+ value={String(skills.length)}
114
+ icon="auto_awesome"
115
+ />
116
+ <SummaryCard
117
+ label={t('settings.skills.summary.sources')}
118
+ value={String(
119
+ SOURCE_ORDER.filter((source) => (grouped[source] ?? []).length > 0)
120
+ .length,
121
+ )}
122
+ icon="layers"
123
+ />
124
+ <SummaryCard
125
+ label={t('settings.skills.summary.tokens')}
126
+ value={t('settings.skills.tokenEstimateShort', { count: String(totalTokens) })}
127
+ icon="notes"
128
+ className="col-span-2 sm:col-span-1"
129
+ />
130
+ </div>
131
+ </div>
132
+ </section>
133
+
134
+ <div className={`grid gap-4 ${visibleGroupCount >= 2 ? 'xl:grid-cols-2' : ''}`}>
135
+ {SOURCE_ORDER.map((source) => {
136
+ const group = grouped[source]
137
+ if (!group?.length) return null
138
+
139
+ const sourceLabel = t(`settings.skills.source.${source}`)
140
+ const sourceTokenCount = group.reduce(
141
+ (sum, skill) => sum + estimateTokens(skill.contentLength),
142
+ 0,
143
+ )
144
+
145
+ return (
146
+ <section
147
+ key={source}
148
+ className="rounded-2xl border border-[var(--color-border)] bg-[var(--color-surface)] overflow-hidden min-w-0"
149
+ >
150
+ <div className="flex items-start justify-between gap-3 px-5 py-4 border-b border-[var(--color-border)] bg-[var(--color-surface-container-low)]">
151
+ <div className="min-w-0">
152
+ <div className="flex items-center gap-2 mb-1">
153
+ <span className={`inline-flex h-7 w-7 items-center justify-center rounded-full ${SOURCE_ACCENT_CLASSES[source]}`}>
154
+ <span className="material-symbols-outlined text-[16px]">
155
+ {SOURCE_ICONS[source]}
156
+ </span>
157
+ </span>
158
+ <h4 className="text-sm font-semibold text-[var(--color-text-primary)]">
159
+ {sourceLabel}
160
+ </h4>
161
+ <span className="text-xs text-[var(--color-text-tertiary)]">
162
+ {group.length}
163
+ </span>
164
+ </div>
165
+ <p className="text-xs leading-5 text-[var(--color-text-tertiary)]">
166
+ {t('settings.skills.groupHint', {
167
+ source: sourceLabel,
168
+ count: String(group.length),
169
+ })}
170
+ </p>
171
+ </div>
172
+ <div className="text-[11px] text-[var(--color-text-tertiary)] whitespace-nowrap">
173
+ {t('settings.skills.tokenEstimateShort', { count: String(sourceTokenCount) })}
174
+ </div>
175
+ </div>
176
+
177
+ <div className="flex flex-col p-2">
178
+ {group.map((skill) => (
179
+ <button
180
+ key={`${skill.source}-${skill.name}`}
181
+ onClick={() =>
182
+ skill.hasDirectory &&
183
+ fetchSkillDetail(skill.source, skill.name, currentWorkDir)
184
+ }
185
+ disabled={!skill.hasDirectory}
186
+ className="group rounded-xl border border-transparent px-3 py-3 text-left transition-all hover:border-[var(--color-border-focus)] hover:bg-[var(--color-surface-hover)] focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-[var(--color-brand)] focus-visible:ring-offset-2 focus-visible:ring-offset-[var(--color-surface)] disabled:opacity-60 disabled:cursor-default disabled:hover:bg-transparent disabled:hover:border-transparent"
187
+ >
188
+ <div className="flex items-start gap-3">
189
+ <span className="mt-0.5 material-symbols-outlined text-[18px] text-[var(--color-text-tertiary)]">
190
+ auto_awesome
191
+ </span>
192
+ <div className="flex-1 min-w-0">
193
+ <div className="flex items-center gap-2 flex-wrap">
194
+ <span className="text-sm font-semibold text-[var(--color-text-primary)] break-all">
195
+ {skill.displayName || skill.name}
196
+ </span>
197
+ {skill.version && (
198
+ <span className="rounded-full bg-[var(--color-surface-container-high)] px-2 py-0.5 text-[10px] font-medium text-[var(--color-text-tertiary)]">
199
+ v{skill.version}
200
+ </span>
201
+ )}
202
+ {skill.userInvocable && (
203
+ <span className="rounded-full border border-[var(--color-border)] px-2 py-0.5 text-[10px] font-medium text-[var(--color-text-tertiary)]">
204
+ {t('settings.skills.slashCommand')}
205
+ </span>
206
+ )}
207
+ </div>
208
+ <p className="mt-1 text-xs leading-5 text-[var(--color-text-secondary)] break-words">
209
+ {skill.description}
210
+ </p>
211
+ <div className="mt-2 flex flex-wrap items-center gap-x-3 gap-y-1 text-[11px] text-[var(--color-text-tertiary)]">
212
+ <span>{sourceLabel}</span>
213
+ <span>{t('settings.skills.tokenEstimateShort', { count: String(estimateTokens(skill.contentLength)) })}</span>
214
+ <span>{skill.hasDirectory ? t('settings.skills.ready') : t('settings.skills.unavailable')}</span>
215
+ </div>
216
+ </div>
217
+ <span className="material-symbols-outlined text-[18px] text-[var(--color-text-tertiary)] opacity-60 transition-transform group-hover:translate-x-0.5 group-hover:opacity-100">
218
+ chevron_right
219
+ </span>
220
+ </div>
221
+ </button>
222
+ ))}
223
+ </div>
224
+ </section>
225
+ )
226
+ })}
227
+ </div>
228
+ </div>
229
+ )
230
+ }
231
+
232
+ function SummaryCard({
233
+ label,
234
+ value,
235
+ icon,
236
+ className = '',
237
+ }: {
238
+ label: string
239
+ value: string
240
+ icon: string
241
+ className?: string
242
+ }) {
243
+ return (
244
+ <div className={`rounded-xl border border-[var(--color-border)] bg-[var(--color-surface)] px-3 py-3 min-w-0 ${className}`}>
245
+ <div className="flex items-center gap-1.5 text-[11px] uppercase tracking-[0.12em] text-[var(--color-text-tertiary)] min-w-0">
246
+ <span className="material-symbols-outlined text-[14px] flex-shrink-0">{icon}</span>
247
+ <span className="truncate">{label}</span>
248
+ </div>
249
+ <div className="mt-2 text-lg font-semibold text-[var(--color-text-primary)] truncate">
250
+ {value}
251
+ </div>
252
+ </div>
253
+ )
254
+ }
@@ -0,0 +1,57 @@
1
+ import { useTranslation } from '../../i18n'
2
+
3
+ type Props = {
4
+ selected: number[]
5
+ onChange: (days: number[]) => void
6
+ }
7
+
8
+ // Display order: Mon(1) → Sun(0), matching Chinese convention
9
+ const DAY_ORDER = [1, 2, 3, 4, 5, 6, 0]
10
+
11
+ const DAY_KEYS = [
12
+ 'newTask.daySun',
13
+ 'newTask.dayMon',
14
+ 'newTask.dayTue',
15
+ 'newTask.dayWed',
16
+ 'newTask.dayThu',
17
+ 'newTask.dayFri',
18
+ 'newTask.daySat',
19
+ ] as const
20
+
21
+ export function DayOfWeekPicker({ selected, onChange }: Props) {
22
+ const t = useTranslation()
23
+
24
+ const toggle = (day: number) => {
25
+ if (selected.includes(day)) {
26
+ // Don't allow deselecting the last day
27
+ if (selected.length <= 1) return
28
+ onChange(selected.filter((d) => d !== day))
29
+ } else {
30
+ onChange([...selected, day])
31
+ }
32
+ }
33
+
34
+ return (
35
+ <div className="flex gap-1.5">
36
+ {DAY_ORDER.map((day) => {
37
+ const isActive = selected.includes(day)
38
+ return (
39
+ <button
40
+ key={day}
41
+ type="button"
42
+ onClick={() => toggle(day)}
43
+ className={`
44
+ w-8 h-8 rounded-full text-xs font-medium transition-colors
45
+ ${isActive
46
+ ? 'bg-[var(--color-surface-selected)] text-[var(--color-text-primary)] border border-[var(--color-border-focus)]'
47
+ : 'bg-[var(--color-surface)] text-[var(--color-text-tertiary)] border border-[var(--color-border)] hover:bg-[var(--color-surface-hover)]'
48
+ }
49
+ `}
50
+ >
51
+ {t(DAY_KEYS[day]!)}
52
+ </button>
53
+ )
54
+ })}
55
+ </div>
56
+ )
57
+ }