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,344 @@
1
+ import { create } from 'zustand'
2
+ import { teamsApi } from '../api/teams'
3
+ import type { TeamSummary, TeamDetail, TeamMember, AgentColor } from '../types/team'
4
+ import { AGENT_COLORS } from '../types/team'
5
+ import type { TeamMemberStatus, UIMessage } from '../types/chat'
6
+ import { useChatStore, mapHistoryMessagesToUiMessages } from './chatStore'
7
+ import { useTabStore } from './tabStore'
8
+
9
+ const MEMBER_POLL_INTERVAL_MS = 1500
10
+ const MEMBER_TRANSCRIPT_MATCH_WINDOW_MS = 120_000
11
+
12
+ /** Generate a synthetic sessionId for team member tabs */
13
+ const memberSessionId = (agentId: string) => `team-member:${agentId}`
14
+
15
+ /** Module-level timer for polling member transcript */
16
+ let memberPollTimer: ReturnType<typeof setInterval> | null = null
17
+ let polledMemberSessionId: string | null = null
18
+
19
+ function createMemberSessionState() {
20
+ return {
21
+ messages: [] as UIMessage[],
22
+ chatState: 'idle' as const,
23
+ connectionState: 'connected' as const,
24
+ streamingText: '',
25
+ streamingToolInput: '',
26
+ activeToolUseId: null,
27
+ activeToolName: null,
28
+ activeThinkingId: null,
29
+ pendingPermission: null,
30
+ pendingComputerUsePermission: null,
31
+ tokenUsage: { input_tokens: 0, output_tokens: 0 },
32
+ elapsedSeconds: 0,
33
+ statusVerb: '',
34
+ slashCommands: [],
35
+ agentTaskNotifications: {},
36
+ elapsedTimer: null,
37
+ }
38
+ }
39
+
40
+ function normalizeMemberStatus(status: string | undefined): TeamMember['status'] {
41
+ if (status === 'running' || status === 'idle' || status === 'completed') {
42
+ return status
43
+ }
44
+ return status === 'failed' ? 'error' : 'idle'
45
+ }
46
+
47
+ function toTeamMember(raw: Record<string, unknown>): TeamMember {
48
+ return {
49
+ agentId: (raw.agentId as string) || '',
50
+ name: raw.name as string | undefined,
51
+ role:
52
+ (raw.name as string) ||
53
+ (raw.agentType as string) ||
54
+ (raw.role as string) ||
55
+ (raw.agentId as string) ||
56
+ '',
57
+ status: normalizeMemberStatus(raw.status as string | undefined),
58
+ currentTask: raw.currentTask as string | undefined,
59
+ color: raw.color as AgentColor | undefined,
60
+ sessionId: raw.sessionId as string | undefined,
61
+ }
62
+ }
63
+
64
+ function isPendingMemberMessage(message: UIMessage): message is Extract<UIMessage, { type: 'user_text' }> & { pending: true } {
65
+ return message.type === 'user_text' && message.pending === true
66
+ }
67
+
68
+ function transcriptAlreadyContainsMessage(
69
+ transcriptMessages: UIMessage[],
70
+ pendingMessage: Extract<UIMessage, { type: 'user_text' }> & { pending: true },
71
+ ): boolean {
72
+ return transcriptMessages.some((message) => (
73
+ message.type === 'user_text' &&
74
+ message.pending !== true &&
75
+ message.content === pendingMessage.content &&
76
+ Math.abs(message.timestamp - pendingMessage.timestamp) <= MEMBER_TRANSCRIPT_MATCH_WINDOW_MS
77
+ ))
78
+ }
79
+
80
+ function mergeMemberTranscriptMessages(
81
+ existingMessages: UIMessage[],
82
+ transcriptMessages: UIMessage[],
83
+ ): UIMessage[] {
84
+ const pendingMessages = existingMessages.filter(isPendingMemberMessage).filter(
85
+ (message) => !transcriptAlreadyContainsMessage(transcriptMessages, message),
86
+ )
87
+
88
+ return pendingMessages.length > 0
89
+ ? [...transcriptMessages, ...pendingMessages]
90
+ : transcriptMessages
91
+ }
92
+
93
+ function syncMemberSessionMessages(
94
+ sessionId: string,
95
+ memberStatus: TeamMember['status'],
96
+ messages: UIMessage[],
97
+ ) {
98
+ const hasPendingMessages = messages.some(isPendingMemberMessage)
99
+ useChatStore.setState((state) => {
100
+ const existing = state.sessions[sessionId]
101
+ const nextState = existing ?? createMemberSessionState()
102
+ return {
103
+ sessions: {
104
+ ...state.sessions,
105
+ [sessionId]: {
106
+ ...nextState,
107
+ messages,
108
+ connectionState: 'connected',
109
+ chatState:
110
+ memberStatus === 'running' || hasPendingMessages
111
+ ? 'thinking'
112
+ : 'idle',
113
+ },
114
+ },
115
+ }
116
+ })
117
+ }
118
+
119
+ type TeamStore = {
120
+ teams: TeamSummary[]
121
+ activeTeam: TeamDetail | null
122
+ memberColors: Map<string, AgentColor>
123
+ error: string | null
124
+
125
+ fetchTeams: () => Promise<void>
126
+ fetchTeamDetail: (name: string) => Promise<void>
127
+ getMemberBySessionId: (sessionId: string) => TeamMember | null
128
+ refreshMemberSession: (sessionId: string) => Promise<void>
129
+ openMemberSession: (member: TeamMember) => void
130
+ sendMessageToMember: (sessionId: string, content: string) => Promise<void>
131
+ startMemberPolling: (sessionId: string, force?: boolean) => void
132
+ stopMemberPolling: () => void
133
+ clearTeam: () => void
134
+
135
+ // WebSocket handlers
136
+ handleTeamCreated: (teamName: string) => void
137
+ handleTeamUpdate: (teamName: string, members: TeamMemberStatus[]) => void
138
+ handleTeamDeleted: (teamName: string) => void
139
+ }
140
+
141
+ export const useTeamStore = create<TeamStore>((set, get) => ({
142
+ teams: [],
143
+ activeTeam: null,
144
+ memberColors: new Map(),
145
+ error: null,
146
+
147
+ fetchTeams: async () => {
148
+ set({ error: null })
149
+ try {
150
+ const { teams } = await teamsApi.list()
151
+ set({ teams })
152
+ } catch (err) {
153
+ set({ error: err instanceof Error ? err.message : String(err) })
154
+ }
155
+ },
156
+
157
+ fetchTeamDetail: async (name: string) => {
158
+ set({ error: null })
159
+ try {
160
+ const raw = await teamsApi.get(name) as Record<string, unknown>
161
+ const rawMembers = Array.isArray(raw.members) ? raw.members : []
162
+ const members: TeamMember[] = rawMembers.map((m: Record<string, unknown>) => toTeamMember(m))
163
+ const detail: TeamDetail = {
164
+ name: raw.name as string,
165
+ leadAgentId: raw.leadAgentId as string | undefined,
166
+ leadSessionId: raw.leadSessionId as string | undefined,
167
+ members,
168
+ createdAt: raw.createdAt != null ? String(raw.createdAt) : undefined,
169
+ }
170
+ // Assign colors to members
171
+ const colors = new Map<string, AgentColor>()
172
+ detail.members.forEach((m, i) => {
173
+ colors.set(m.agentId, AGENT_COLORS[i % AGENT_COLORS.length]!)
174
+ })
175
+ set({ activeTeam: detail, memberColors: colors })
176
+ } catch (err) {
177
+ set({ error: err instanceof Error ? err.message : String(err) })
178
+ }
179
+ },
180
+
181
+ getMemberBySessionId: (sessionId: string) => {
182
+ const team = get().activeTeam
183
+ if (!team) return null
184
+ return team.members.find(
185
+ (m) => m.sessionId === sessionId || memberSessionId(m.agentId) === sessionId,
186
+ ) ?? null
187
+ },
188
+
189
+ refreshMemberSession: async (sessionId) => {
190
+ const team = get().activeTeam
191
+ const member = get().getMemberBySessionId(sessionId)
192
+ if (!team || !member) return
193
+
194
+ try {
195
+ const { messages } = await teamsApi.getMemberTranscript(team.name, member.agentId)
196
+ const asEntries = messages.map((msg) => ({
197
+ id: msg.id,
198
+ type: msg.type,
199
+ content: msg.content,
200
+ timestamp: msg.timestamp,
201
+ model: msg.model,
202
+ parentToolUseId: msg.parentToolUseId,
203
+ }))
204
+ const transcriptMessages = mapHistoryMessagesToUiMessages(
205
+ asEntries as Parameters<typeof mapHistoryMessagesToUiMessages>[0],
206
+ { includeTeammateMessages: true },
207
+ )
208
+ const existingMessages = useChatStore.getState().sessions[sessionId]?.messages ?? []
209
+ const mergedMessages = mergeMemberTranscriptMessages(
210
+ existingMessages,
211
+ transcriptMessages,
212
+ )
213
+ syncMemberSessionMessages(sessionId, member.status, mergedMessages)
214
+ } catch {
215
+ const existingMessages = useChatStore.getState().sessions[sessionId]?.messages ?? []
216
+ syncMemberSessionMessages(sessionId, member.status, existingMessages)
217
+ }
218
+ },
219
+
220
+ openMemberSession: (member: TeamMember) => {
221
+ const team = get().activeTeam
222
+ if (!team) return
223
+
224
+ get().stopMemberPolling()
225
+
226
+ const tabId = memberSessionId(member.agentId)
227
+ useTabStore.getState().openTab(tabId, member.role, 'session')
228
+ void get().refreshMemberSession(tabId)
229
+ get().startMemberPolling(tabId)
230
+ },
231
+
232
+ sendMessageToMember: async (sessionId, content) => {
233
+ const team = get().activeTeam
234
+ const member = get().getMemberBySessionId(sessionId)
235
+ if (!team || !member) {
236
+ throw new Error('Team member session is no longer available')
237
+ }
238
+
239
+ await teamsApi.sendMemberMessage(team.name, member.agentId, content)
240
+ get().startMemberPolling(sessionId, true)
241
+ await get().refreshMemberSession(sessionId)
242
+ },
243
+
244
+ startMemberPolling: (sessionId, force = false) => {
245
+ const member = get().getMemberBySessionId(sessionId)
246
+ if (!member) return
247
+
248
+ const hasPendingMessages =
249
+ useChatStore.getState().sessions[sessionId]?.messages.some(isPendingMemberMessage) ?? false
250
+
251
+ if (!force && polledMemberSessionId === sessionId && memberPollTimer) {
252
+ return
253
+ }
254
+
255
+ if (member.status !== 'running' && !hasPendingMessages) {
256
+ get().stopMemberPolling()
257
+ return
258
+ }
259
+
260
+ get().stopMemberPolling()
261
+ polledMemberSessionId = sessionId
262
+ memberPollTimer = setInterval(() => {
263
+ const currentTabId = useTabStore.getState().activeTabId
264
+ if (currentTabId !== sessionId) {
265
+ get().stopMemberPolling()
266
+ return
267
+ }
268
+ void get().refreshMemberSession(sessionId)
269
+ }, MEMBER_POLL_INTERVAL_MS)
270
+ },
271
+
272
+ stopMemberPolling: () => {
273
+ if (memberPollTimer) {
274
+ clearInterval(memberPollTimer)
275
+ memberPollTimer = null
276
+ }
277
+ polledMemberSessionId = null
278
+ },
279
+
280
+ clearTeam: () => {
281
+ get().stopMemberPolling()
282
+ set({ activeTeam: null, memberColors: new Map() })
283
+ },
284
+
285
+ handleTeamCreated: (teamName: string) => {
286
+ set((s) => ({
287
+ teams: [...s.teams, { name: teamName, memberCount: 0 }],
288
+ }))
289
+ get().fetchTeamDetail(teamName)
290
+ setTimeout(() => get().fetchTeamDetail(teamName), 1500)
291
+ setTimeout(() => get().fetchTeamDetail(teamName), 4000)
292
+ setTimeout(() => get().fetchTeamDetail(teamName), 8000)
293
+ },
294
+
295
+ handleTeamUpdate: (teamName: string, members: TeamMemberStatus[]) => {
296
+ const team = get().activeTeam
297
+ if (team && team.name === teamName) {
298
+ if (members.length === 0) return
299
+
300
+ if (members.length > team.members.length) {
301
+ get().fetchTeamDetail(teamName)
302
+ }
303
+
304
+ const colors = get().memberColors
305
+ const existingMap = new Map(team.members.map((m) => [m.agentId, m]))
306
+ const incomingIds = new Set(members.map((m) => m.agentId))
307
+ const kept = team.members.filter((m) => !incomingIds.has(m.agentId))
308
+ const updatedMembers: TeamMember[] = [
309
+ ...kept,
310
+ ...members.map((m, i) => {
311
+ const existing = existingMap.get(m.agentId)
312
+ return {
313
+ ...(existing ?? {}),
314
+ name: existing?.name,
315
+ agentId: m.agentId,
316
+ role: m.role,
317
+ status: normalizeMemberStatus(m.status),
318
+ currentTask: m.currentTask,
319
+ color: colors.get(m.agentId) ?? AGENT_COLORS[i % AGENT_COLORS.length]!,
320
+ sessionId: existing?.sessionId,
321
+ }
322
+ }),
323
+ ]
324
+ set({ activeTeam: { ...team, members: updatedMembers } })
325
+
326
+ const currentTabId = useTabStore.getState().activeTabId
327
+ if (currentTabId) {
328
+ const viewedMember = get().getMemberBySessionId(currentTabId)
329
+ if (viewedMember) {
330
+ void get().refreshMemberSession(currentTabId)
331
+ get().startMemberPolling(currentTabId)
332
+ }
333
+ }
334
+ }
335
+ },
336
+
337
+ handleTeamDeleted: (teamName: string) => {
338
+ get().stopMemberPolling()
339
+ set((s) => ({
340
+ teams: s.teams.filter((t) => t.name !== teamName),
341
+ activeTeam: s.activeTeam?.name === teamName ? null : s.activeTeam,
342
+ }))
343
+ },
344
+ }))
@@ -0,0 +1,100 @@
1
+ import { create } from 'zustand'
2
+ import type { ThemeMode } from '../types/settings'
3
+
4
+ const THEME_STORAGE_KEY = 'cc-haha-theme'
5
+
6
+ function getStoredTheme(): ThemeMode {
7
+ try {
8
+ const stored = localStorage.getItem(THEME_STORAGE_KEY)
9
+ if (stored === 'light' || stored === 'dark') return stored
10
+ } catch { /* localStorage unavailable */ }
11
+ return 'light'
12
+ }
13
+
14
+ export function applyTheme(theme: ThemeMode) {
15
+ if (typeof document === 'undefined') return
16
+ document.documentElement.setAttribute('data-theme', theme)
17
+ document.documentElement.style.colorScheme = theme
18
+ }
19
+
20
+ export function initializeTheme() {
21
+ applyTheme(getStoredTheme())
22
+ }
23
+
24
+ export type Toast = {
25
+ id: string
26
+ type: 'success' | 'error' | 'warning' | 'info'
27
+ message: string
28
+ duration?: number
29
+ }
30
+
31
+ export type SettingsTab = 'providers' | 'permissions' | 'general' | 'adapters' | 'agents' | 'skills' | 'computerUse' | 'about'
32
+
33
+ type ActiveView = 'code' | 'scheduled' | 'terminal' | 'history' | 'settings'
34
+
35
+ type UIStore = {
36
+ theme: ThemeMode
37
+ sidebarOpen: boolean
38
+ activeView: ActiveView
39
+ pendingSettingsTab: SettingsTab | null
40
+ activeModal: string | null
41
+ toasts: Toast[]
42
+
43
+ setTheme: (theme: ThemeMode) => void
44
+ toggleTheme: () => void
45
+ toggleSidebar: () => void
46
+ setSidebarOpen: (open: boolean) => void
47
+ setActiveView: (view: ActiveView) => void
48
+ setPendingSettingsTab: (tab: SettingsTab | null) => void
49
+ openModal: (id: string) => void
50
+ closeModal: () => void
51
+ addToast: (toast: Omit<Toast, 'id'>) => void
52
+ removeToast: (id: string) => void
53
+ }
54
+
55
+ let toastCounter = 0
56
+
57
+ export const useUIStore = create<UIStore>((set) => ({
58
+ theme: getStoredTheme(),
59
+ sidebarOpen: true,
60
+ activeView: 'code',
61
+ pendingSettingsTab: null,
62
+ activeModal: null,
63
+ toasts: [],
64
+
65
+ setTheme: (theme) => {
66
+ applyTheme(theme)
67
+ try { localStorage.setItem(THEME_STORAGE_KEY, theme) } catch { /* noop */ }
68
+ set({ theme })
69
+ },
70
+
71
+ toggleTheme: () => {
72
+ set((state) => {
73
+ const next = state.theme === 'light' ? 'dark' : 'light'
74
+ applyTheme(next)
75
+ try { localStorage.setItem(THEME_STORAGE_KEY, next) } catch { /* noop */ }
76
+ return { theme: next }
77
+ })
78
+ },
79
+
80
+ toggleSidebar: () => set((s) => ({ sidebarOpen: !s.sidebarOpen })),
81
+ setSidebarOpen: (open) => set({ sidebarOpen: open }),
82
+ setActiveView: (view) => set({ activeView: view }),
83
+ setPendingSettingsTab: (tab) => set({ pendingSettingsTab: tab }),
84
+ openModal: (id) => set({ activeModal: id }),
85
+ closeModal: () => set({ activeModal: null }),
86
+
87
+ addToast: (toast) => {
88
+ const id = `toast-${++toastCounter}`
89
+ set((s) => ({ toasts: [...s.toasts, { ...toast, id }] }))
90
+ // Auto-remove after duration
91
+ const duration = toast.duration ?? 4000
92
+ if (duration > 0) {
93
+ setTimeout(() => {
94
+ set((s) => ({ toasts: s.toasts.filter((t) => t.id !== id) }))
95
+ }, duration)
96
+ }
97
+ },
98
+
99
+ removeToast: (id) => set((s) => ({ toasts: s.toasts.filter((t) => t.id !== id) })),
100
+ }))
@@ -0,0 +1,71 @@
1
+ import { beforeEach, describe, expect, it, vi } from 'vitest'
2
+
3
+ const check = vi.fn()
4
+ const relaunch = vi.fn()
5
+
6
+ vi.mock('@tauri-apps/plugin-updater', () => ({
7
+ check,
8
+ }))
9
+
10
+ vi.mock('@tauri-apps/plugin-process', () => ({
11
+ relaunch,
12
+ }))
13
+
14
+ describe('updateStore', () => {
15
+ beforeEach(() => {
16
+ check.mockReset()
17
+ relaunch.mockReset()
18
+ Object.defineProperty(window, '__TAURI_INTERNALS__', {
19
+ configurable: true,
20
+ value: {},
21
+ })
22
+ })
23
+
24
+ it('stores available update metadata after a successful check', async () => {
25
+ const update = {
26
+ version: '0.2.0',
27
+ body: 'Bug fixes and performance improvements',
28
+ close: vi.fn().mockResolvedValue(undefined),
29
+ }
30
+ check.mockResolvedValue(update)
31
+
32
+ vi.resetModules()
33
+ const { useUpdateStore } = await import('./updateStore')
34
+
35
+ const result = await useUpdateStore.getState().checkForUpdates()
36
+
37
+ expect(result).toBe(update)
38
+ expect(useUpdateStore.getState().status).toBe('available')
39
+ expect(useUpdateStore.getState().availableVersion).toBe('0.2.0')
40
+ expect(useUpdateStore.getState().releaseNotes).toBe('Bug fixes and performance improvements')
41
+ expect(useUpdateStore.getState().shouldPrompt).toBe(true)
42
+ })
43
+
44
+ it('computes download progress from content length and relaunches after install', async () => {
45
+ const downloadAndInstall = vi.fn(async (onEvent?: (event: unknown) => void) => {
46
+ onEvent?.({ event: 'Started', data: { contentLength: 200 } })
47
+ onEvent?.({ event: 'Progress', data: { chunkLength: 50 } })
48
+ onEvent?.({ event: 'Progress', data: { chunkLength: 150 } })
49
+ onEvent?.({ event: 'Finished' })
50
+ })
51
+
52
+ check.mockResolvedValue({
53
+ version: '0.2.0',
54
+ body: 'Notes',
55
+ downloadAndInstall,
56
+ close: vi.fn().mockResolvedValue(undefined),
57
+ })
58
+ relaunch.mockResolvedValue(undefined)
59
+
60
+ vi.resetModules()
61
+ const { useUpdateStore } = await import('./updateStore')
62
+
63
+ await useUpdateStore.getState().checkForUpdates()
64
+ await useUpdateStore.getState().installUpdate()
65
+
66
+ expect(downloadAndInstall).toHaveBeenCalledTimes(1)
67
+ expect(useUpdateStore.getState().progressPercent).toBe(100)
68
+ expect(useUpdateStore.getState().status).toBe('restarting')
69
+ expect(relaunch).toHaveBeenCalledTimes(1)
70
+ })
71
+ })