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,518 @@
1
+ import { useEffect, useMemo, useRef, useState } from 'react'
2
+ import { skillsApi } from '../api/skills'
3
+ import { useTranslation } from '../i18n'
4
+ import { useSessionStore } from '../stores/sessionStore'
5
+ import { useChatStore } from '../stores/chatStore'
6
+ import { useUIStore } from '../stores/uiStore'
7
+ import { useTabStore } from '../stores/tabStore'
8
+ import { DirectoryPicker } from '../components/shared/DirectoryPicker'
9
+ import { PermissionModeSelector } from '../components/controls/PermissionModeSelector'
10
+ import { ModelSelector } from '../components/controls/ModelSelector'
11
+ import { AttachmentGallery } from '../components/chat/AttachmentGallery'
12
+ import { FileSearchMenu, type FileSearchMenuHandle } from '../components/chat/FileSearchMenu'
13
+ import {
14
+ FALLBACK_SLASH_COMMANDS,
15
+ findSlashToken,
16
+ insertSlashTrigger,
17
+ mergeSlashCommands,
18
+ replaceSlashCommand,
19
+ } from '../components/chat/composerUtils'
20
+ import type { AttachmentRef } from '../types/chat'
21
+ import type { SlashCommandOption } from '../components/chat/composerUtils'
22
+
23
+ type Attachment = {
24
+ id: string
25
+ name: string
26
+ type: 'image' | 'file'
27
+ mimeType?: string
28
+ previewUrl?: string
29
+ data?: string
30
+ }
31
+
32
+ export function EmptySession() {
33
+ const t = useTranslation()
34
+ const [input, setInput] = useState('')
35
+ const [isSubmitting, setIsSubmitting] = useState(false)
36
+ const [workDir, setWorkDir] = useState('')
37
+ const [attachments, setAttachments] = useState<Attachment[]>([])
38
+ const [plusMenuOpen, setPlusMenuOpen] = useState(false)
39
+ const [slashMenuOpen, setSlashMenuOpen] = useState(false)
40
+ const [fileSearchOpen, setFileSearchOpen] = useState(false)
41
+ const [atFilter, setAtFilter] = useState('')
42
+ const [atCursorPos, setAtCursorPos] = useState(-1)
43
+ const [slashFilter, setSlashFilter] = useState('')
44
+ const [slashSelectedIndex, setSlashSelectedIndex] = useState(0)
45
+ const [slashCommands, setSlashCommands] = useState<SlashCommandOption[]>([])
46
+ const textareaRef = useRef<HTMLTextAreaElement>(null)
47
+ const fileInputRef = useRef<HTMLInputElement>(null)
48
+ const plusMenuRef = useRef<HTMLDivElement>(null)
49
+ const slashMenuRef = useRef<HTMLDivElement>(null)
50
+ const fileSearchRef = useRef<FileSearchMenuHandle>(null)
51
+ const slashItemRefs = useRef<(HTMLButtonElement | null)[]>([])
52
+ const createSession = useSessionStore((state) => state.createSession)
53
+ const sendMessage = useChatStore((state) => state.sendMessage)
54
+ const connectToSession = useChatStore((state) => state.connectToSession)
55
+ const setActiveView = useUIStore((state) => state.setActiveView)
56
+ const addToast = useUIStore((state) => state.addToast)
57
+
58
+ useEffect(() => {
59
+ textareaRef.current?.focus()
60
+ }, [])
61
+
62
+ useEffect(() => {
63
+ if (!plusMenuOpen) return
64
+ const handleClick = (event: MouseEvent) => {
65
+ if (plusMenuRef.current && !plusMenuRef.current.contains(event.target as Node)) {
66
+ setPlusMenuOpen(false)
67
+ }
68
+ }
69
+ document.addEventListener('mousedown', handleClick)
70
+ return () => document.removeEventListener('mousedown', handleClick)
71
+ }, [plusMenuOpen])
72
+
73
+ useEffect(() => {
74
+ if (!slashMenuOpen) return
75
+ const handleClick = (event: MouseEvent) => {
76
+ if (
77
+ slashMenuRef.current &&
78
+ !slashMenuRef.current.contains(event.target as Node) &&
79
+ textareaRef.current &&
80
+ !textareaRef.current.contains(event.target as Node)
81
+ ) {
82
+ setSlashMenuOpen(false)
83
+ }
84
+ }
85
+ document.addEventListener('mousedown', handleClick)
86
+ return () => document.removeEventListener('mousedown', handleClick)
87
+ }, [slashMenuOpen])
88
+
89
+ useEffect(() => {
90
+ if (!fileSearchOpen) return
91
+ const handleClick = (event: MouseEvent) => {
92
+ const menu = document.getElementById('file-search-menu')
93
+ if (
94
+ menu &&
95
+ !menu.contains(event.target as Node) &&
96
+ textareaRef.current &&
97
+ !textareaRef.current.contains(event.target as Node)
98
+ ) {
99
+ setFileSearchOpen(false)
100
+ }
101
+ }
102
+ document.addEventListener('mousedown', handleClick)
103
+ return () => document.removeEventListener('mousedown', handleClick)
104
+ }, [fileSearchOpen])
105
+
106
+ useEffect(() => {
107
+ let cancelled = false
108
+
109
+ skillsApi.list(workDir || undefined)
110
+ .then(({ skills }) => {
111
+ if (cancelled) return
112
+ setSlashCommands(
113
+ skills
114
+ .filter((skill) => skill.userInvocable)
115
+ .map((skill) => ({
116
+ name: skill.name,
117
+ description: skill.description,
118
+ })),
119
+ )
120
+ })
121
+ .catch(() => {
122
+ if (!cancelled) {
123
+ setSlashCommands([])
124
+ }
125
+ })
126
+
127
+ return () => {
128
+ cancelled = true
129
+ }
130
+ }, [workDir])
131
+
132
+ const filteredCommands = useMemo(() => {
133
+ const source = mergeSlashCommands(slashCommands, FALLBACK_SLASH_COMMANDS)
134
+ if (!slashFilter) return source
135
+ const lower = slashFilter.toLowerCase()
136
+ return source.filter((command) => (
137
+ command.name.toLowerCase().includes(lower) ||
138
+ command.description.toLowerCase().includes(lower)
139
+ ))
140
+ }, [slashCommands, slashFilter])
141
+
142
+ useEffect(() => {
143
+ setSlashSelectedIndex(0)
144
+ }, [slashFilter])
145
+
146
+ useEffect(() => {
147
+ const activeItem = slashMenuOpen ? slashItemRefs.current[slashSelectedIndex] : null
148
+ if (typeof activeItem?.scrollIntoView === 'function') {
149
+ activeItem.scrollIntoView({ block: 'nearest' })
150
+ }
151
+ }, [slashMenuOpen, slashSelectedIndex])
152
+
153
+ const handleSubmit = async () => {
154
+ const text = input.trim()
155
+ if ((!text && attachments.length === 0) || isSubmitting) return
156
+
157
+ setIsSubmitting(true)
158
+ try {
159
+ const sessionId = await createSession(workDir || undefined)
160
+ setActiveView('code')
161
+ useTabStore.getState().openTab(sessionId, 'New Session')
162
+ connectToSession(sessionId)
163
+ const attachmentPayload: AttachmentRef[] = attachments.map((attachment) => ({
164
+ type: attachment.type,
165
+ name: attachment.name,
166
+ data: attachment.data,
167
+ mimeType: attachment.mimeType,
168
+ }))
169
+ sendMessage(sessionId, text, attachmentPayload)
170
+ setInput('')
171
+ setAttachments([])
172
+ } catch (error) {
173
+ addToast({
174
+ type: 'error',
175
+ message: error instanceof Error ? error.message : t('empty.failedToCreate'),
176
+ })
177
+ } finally {
178
+ setIsSubmitting(false)
179
+ }
180
+ }
181
+
182
+ const handleInputChange = (value: string, cursorPos: number) => {
183
+ setInput(value)
184
+ const token = findSlashToken(value, cursorPos)
185
+ if (!token) {
186
+ setSlashMenuOpen(false)
187
+ } else {
188
+ setSlashFilter(token.filter)
189
+ setSlashMenuOpen(true)
190
+ }
191
+
192
+ // Detect @ trigger for file search
193
+ const textBeforeCursor = value.slice(0, cursorPos)
194
+ let pos = -1
195
+ for (let i = textBeforeCursor.length - 1; i >= 0; i--) {
196
+ const ch = textBeforeCursor[i]!
197
+ if (ch === '@') {
198
+ if (i === 0 || /\s/.test(textBeforeCursor[i - 1]!)) {
199
+ pos = i
200
+ break
201
+ }
202
+ break
203
+ }
204
+ if (/\s/.test(ch)) {
205
+ break
206
+ }
207
+ }
208
+ if (pos < 0) {
209
+ setFileSearchOpen(false)
210
+ setAtFilter('')
211
+ setAtCursorPos(-1)
212
+ } else {
213
+ setAtFilter(textBeforeCursor.slice(pos + 1))
214
+ setAtCursorPos(cursorPos)
215
+ setSlashMenuOpen(false)
216
+ setFileSearchOpen(true)
217
+ }
218
+ }
219
+
220
+ const handleKeyDown = (event: React.KeyboardEvent) => {
221
+ // Ignore key events during IME composition (e.g. Chinese input method)
222
+ if (event.nativeEvent.isComposing) return
223
+
224
+ // Route file search navigation keys to FileSearchMenu
225
+ if (fileSearchOpen) {
226
+ const key = event.key
227
+ if (key === 'ArrowDown' || key === 'ArrowUp' || key === 'Enter' || key === 'Tab' || key === 'Escape') {
228
+ event.preventDefault()
229
+ if (key === 'Escape') {
230
+ setFileSearchOpen(false)
231
+ setAtFilter('')
232
+ setAtCursorPos(-1)
233
+ return
234
+ }
235
+ fileSearchRef.current?.handleKeyDown(event.nativeEvent)
236
+ return
237
+ }
238
+ return
239
+ }
240
+
241
+ if (slashMenuOpen && filteredCommands.length > 0) {
242
+ if (event.key === 'ArrowDown') {
243
+ event.preventDefault()
244
+ setSlashSelectedIndex((prev) => (prev + 1) % filteredCommands.length)
245
+ return
246
+ }
247
+ if (event.key === 'ArrowUp') {
248
+ event.preventDefault()
249
+ setSlashSelectedIndex((prev) => (prev - 1 + filteredCommands.length) % filteredCommands.length)
250
+ return
251
+ }
252
+ if (event.key === 'Enter' || event.key === 'Tab') {
253
+ event.preventDefault()
254
+ const selected = filteredCommands[slashSelectedIndex]
255
+ if (selected) selectSlashCommand(selected.name)
256
+ return
257
+ }
258
+ if (event.key === 'Escape') {
259
+ event.preventDefault()
260
+ setSlashMenuOpen(false)
261
+ return
262
+ }
263
+ }
264
+
265
+ if (event.key === 'Enter' && !event.shiftKey) {
266
+ event.preventDefault()
267
+ handleSubmit()
268
+ }
269
+ }
270
+
271
+ const handlePaste = (event: React.ClipboardEvent) => {
272
+ const items = event.clipboardData?.items
273
+ if (!items) return
274
+
275
+ let hasImage = false
276
+ for (let i = 0; i < items.length; i += 1) {
277
+ const item = items[i]
278
+ if (!item || !item.type.startsWith('image/')) continue
279
+
280
+ hasImage = true
281
+ event.preventDefault()
282
+ const file = item.getAsFile()
283
+ if (!file) continue
284
+ const id = `att-${Date.now()}-${Math.random().toString(36).slice(2)}`
285
+ const reader = new FileReader()
286
+ reader.onload = () => {
287
+ setAttachments((prev) => [
288
+ ...prev,
289
+ {
290
+ id,
291
+ name: `pasted-image-${Date.now()}.png`,
292
+ type: 'image',
293
+ mimeType: file.type || undefined,
294
+ previewUrl: reader.result as string,
295
+ data: reader.result as string,
296
+ },
297
+ ])
298
+ }
299
+ reader.readAsDataURL(file)
300
+ }
301
+
302
+ if (!hasImage) return
303
+ }
304
+
305
+ const handleFileSelect = (event: React.ChangeEvent<HTMLInputElement>) => {
306
+ const files = event.target.files
307
+ if (!files) return
308
+
309
+ Array.from(files).forEach((file) => {
310
+ const id = `att-${Date.now()}-${Math.random().toString(36).slice(2)}`
311
+ const isImage = file.type.startsWith('image/')
312
+ const reader = new FileReader()
313
+ reader.onload = () => {
314
+ setAttachments((prev) => [
315
+ ...prev,
316
+ {
317
+ id,
318
+ name: file.name,
319
+ type: isImage ? 'image' : 'file',
320
+ mimeType: file.type || undefined,
321
+ previewUrl: isImage ? (reader.result as string) : undefined,
322
+ data: reader.result as string,
323
+ },
324
+ ])
325
+ }
326
+ reader.readAsDataURL(file)
327
+ })
328
+
329
+ event.target.value = ''
330
+ }
331
+
332
+ const handleDrop = (event: React.DragEvent) => {
333
+ event.preventDefault()
334
+ const files = event.dataTransfer.files
335
+ if (files.length > 0) {
336
+ const fakeEvent = { target: { files } } as React.ChangeEvent<HTMLInputElement>
337
+ handleFileSelect(fakeEvent)
338
+ }
339
+ }
340
+
341
+ const removeAttachment = (id: string) => {
342
+ setAttachments((prev) => prev.filter((attachment) => attachment.id !== id))
343
+ }
344
+
345
+ const selectSlashCommand = (command: string) => {
346
+ const el = textareaRef.current
347
+ if (!el) return
348
+ const cursorPos = el.selectionStart ?? input.length
349
+ const replacement = replaceSlashCommand(input, cursorPos, command)
350
+ if (!replacement) return
351
+ setInput(replacement.value)
352
+ setSlashMenuOpen(false)
353
+ requestAnimationFrame(() => {
354
+ el.focus()
355
+ el.setSelectionRange(replacement.cursorPos, replacement.cursorPos)
356
+ })
357
+ }
358
+
359
+ const insertSlashCommand = () => {
360
+ const el = textareaRef.current
361
+ const cursorPos = el?.selectionStart ?? input.length
362
+ const replacement = insertSlashTrigger(input, cursorPos)
363
+ setInput(replacement.value)
364
+ setPlusMenuOpen(false)
365
+ setSlashFilter('')
366
+ setSlashMenuOpen(true)
367
+ requestAnimationFrame(() => {
368
+ textareaRef.current?.focus()
369
+ textareaRef.current?.setSelectionRange(replacement.cursorPos, replacement.cursorPos)
370
+ })
371
+ }
372
+
373
+ return (
374
+ <div className="relative flex flex-1 flex-col overflow-hidden bg-[var(--color-surface)]">
375
+ <div className="flex flex-1 flex-col items-center justify-center p-8 pb-32">
376
+ <div className="flex max-w-md flex-col items-center text-center">
377
+ <img src="/app-icon.jpg" alt="Claude Code Haha" className="mb-6 h-24 w-24 rounded-[22px]" style={{ boxShadow: 'var(--shadow-dropdown)' }} />
378
+ <h1 className="mb-2 text-3xl font-extrabold tracking-tight text-[var(--color-text-primary)]" style={{ fontFamily: 'var(--font-headline)' }}>
379
+ {t('empty.title')}
380
+ </h1>
381
+ <p className="mx-auto max-w-xs text-[var(--color-text-secondary)]" style={{ fontFamily: 'var(--font-body)' }}>
382
+ {t('empty.subtitle')}
383
+ </p>
384
+ </div>
385
+ </div>
386
+
387
+ <div className="absolute bottom-4 left-0 right-0 flex justify-center px-8">
388
+ <div className="flex w-full max-w-3xl flex-col gap-2">
389
+ <div
390
+ className="glass-panel relative flex flex-col gap-3 rounded-xl p-4"
391
+ onDragOver={(event) => event.preventDefault()}
392
+ onDrop={handleDrop}
393
+ >
394
+ {fileSearchOpen && (
395
+ <FileSearchMenu
396
+ ref={fileSearchRef}
397
+ cwd={workDir || ''}
398
+ filter={atFilter}
399
+ onSelect={(_path, name) => {
400
+ if (atCursorPos >= 0) {
401
+ const newValue = `${input.slice(0, atCursorPos)}${name}${input.slice(atCursorPos)}`
402
+ const newCursorPos = atCursorPos + name.length
403
+ setInput(newValue)
404
+ setFileSearchOpen(false)
405
+ setAtFilter('')
406
+ setAtCursorPos(-1)
407
+ void textareaRef.current?.focus()
408
+ requestAnimationFrame(() => {
409
+ textareaRef.current?.setSelectionRange(newCursorPos, newCursorPos)
410
+ })
411
+ }
412
+ }}
413
+ />
414
+ )}
415
+
416
+ {slashMenuOpen && filteredCommands.length > 0 && (
417
+ <div
418
+ ref={slashMenuRef}
419
+ className="absolute bottom-full left-0 right-0 z-50 mb-2 overflow-hidden rounded-xl border border-[var(--color-border)] bg-[var(--color-surface-container-lowest)] shadow-[var(--shadow-dropdown)]"
420
+ >
421
+ <div className="max-h-[260px] overflow-y-auto py-1">
422
+ {filteredCommands.map((command, index) => (
423
+ <button
424
+ key={command.name}
425
+ ref={(el) => { slashItemRefs.current[index] = el }}
426
+ onClick={() => selectSlashCommand(command.name)}
427
+ onMouseEnter={() => setSlashSelectedIndex(index)}
428
+ className={`flex w-full items-center gap-3 px-4 py-2.5 text-left transition-colors ${
429
+ index === slashSelectedIndex ? 'bg-[var(--color-surface-hover)]' : 'hover:bg-[var(--color-surface-hover)]'
430
+ }`}
431
+ >
432
+ <span className="shrink-0 text-sm font-semibold text-[var(--color-text-primary)]">/{command.name}</span>
433
+ <span className="min-w-0 flex-1 truncate text-xs text-[var(--color-text-tertiary)]">{command.description}</span>
434
+ </button>
435
+ ))}
436
+ </div>
437
+ </div>
438
+ )}
439
+
440
+ {attachments.length > 0 && (
441
+ <AttachmentGallery attachments={attachments} variant="composer" onRemove={removeAttachment} />
442
+ )}
443
+
444
+ <div className="flex items-start gap-3">
445
+ <textarea
446
+ ref={textareaRef}
447
+ value={input}
448
+ onChange={(event) => handleInputChange(event.target.value, event.target.selectionStart ?? event.target.value.length)}
449
+ onKeyDown={handleKeyDown}
450
+ onPaste={handlePaste}
451
+ className="flex-1 resize-none border-none bg-transparent py-2 leading-relaxed text-[var(--color-text-primary)] outline-none placeholder:text-[var(--color-text-tertiary)]"
452
+ style={{ fontFamily: 'var(--font-body)' }}
453
+ placeholder={t('empty.placeholder')}
454
+ rows={2}
455
+ />
456
+ </div>
457
+
458
+ <div className="flex items-center justify-between border-t border-[var(--color-border-separator)] pt-3">
459
+ <div className="flex items-center gap-2">
460
+ <div ref={plusMenuRef} className="relative">
461
+ <button
462
+ onClick={() => setPlusMenuOpen((prev) => !prev)}
463
+ aria-label="Open composer tools"
464
+ className="rounded-lg p-1.5 text-[var(--color-text-secondary)] transition-colors hover:bg-[var(--color-surface-hover)]"
465
+ >
466
+ <span className="material-symbols-outlined text-[18px]">add</span>
467
+ </button>
468
+
469
+ {plusMenuOpen && (
470
+ <div className="absolute bottom-full left-0 mb-2 w-[240px] rounded-xl border border-[var(--color-border)] bg-[var(--color-surface-container-lowest)] py-1 shadow-[var(--shadow-dropdown)]">
471
+ <button
472
+ onClick={() => {
473
+ fileInputRef.current?.click()
474
+ setPlusMenuOpen(false)
475
+ }}
476
+ className="flex w-full items-center gap-3 px-4 py-2.5 text-left text-sm text-[var(--color-text-primary)] transition-colors hover:bg-[var(--color-surface-hover)]"
477
+ >
478
+ <span className="material-symbols-outlined text-[18px] text-[var(--color-text-secondary)]">attach_file</span>
479
+ {t('empty.addFiles')}
480
+ </button>
481
+ <button
482
+ onClick={insertSlashCommand}
483
+ className="flex w-full items-center gap-3 px-4 py-2.5 text-left text-sm text-[var(--color-text-primary)] transition-colors hover:bg-[var(--color-surface-hover)]"
484
+ >
485
+ <span className="w-5 text-center text-[18px] font-bold text-[var(--color-text-secondary)]">/</span>
486
+ {t('empty.slashCommands')}
487
+ </button>
488
+ </div>
489
+ )}
490
+ </div>
491
+
492
+ <PermissionModeSelector workDir={workDir} />
493
+ </div>
494
+
495
+ <div className="flex items-center gap-3">
496
+ <ModelSelector />
497
+ <button
498
+ onClick={handleSubmit}
499
+ disabled={(!input.trim() && attachments.length === 0) || isSubmitting}
500
+ className="flex w-[112px] items-center justify-center gap-1 rounded-lg bg-[image:var(--gradient-btn-primary)] px-3 py-1.5 text-xs font-semibold text-[var(--color-btn-primary-fg)] shadow-[var(--shadow-button-primary)] transition-all hover:brightness-105 disabled:opacity-30"
501
+ >
502
+ {t('common.run')}
503
+ <span className="material-symbols-outlined text-[14px]">arrow_forward</span>
504
+ </button>
505
+ </div>
506
+ </div>
507
+ </div>
508
+
509
+ <div>
510
+ <DirectoryPicker value={workDir} onChange={setWorkDir} />
511
+ </div>
512
+ </div>
513
+ </div>
514
+
515
+ <input ref={fileInputRef} type="file" multiple className="hidden" onChange={handleFileSelect} />
516
+ </div>
517
+ )
518
+ }