bingocode 1.0.3 → 1.0.4

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 (184) hide show
  1. package/package.json +1 -1
  2. package/desktop/README.md +0 -30
  3. package/desktop/bunfig.toml +0 -1
  4. package/desktop/index.html +0 -17
  5. package/desktop/package.json +0 -55
  6. package/desktop/pnpm-lock.yaml +0 -3832
  7. package/desktop/public/app-icon.jpg +0 -0
  8. package/desktop/public/fonts/inter-latin-ext.woff2 +0 -0
  9. package/desktop/public/fonts/inter-latin.woff2 +0 -0
  10. package/desktop/public/fonts/jetbrains-mono-latin-ext.woff2 +0 -0
  11. package/desktop/public/fonts/jetbrains-mono-latin.woff2 +0 -0
  12. package/desktop/public/fonts/manrope-latin-ext.woff2 +0 -0
  13. package/desktop/public/fonts/manrope-latin.woff2 +0 -0
  14. package/desktop/public/fonts/material-symbols-outlined.woff2 +0 -0
  15. package/desktop/public/icons/bilibili.svg +0 -1
  16. package/desktop/public/icons/douyin.svg +0 -1
  17. package/desktop/public/icons/github.svg +0 -3
  18. package/desktop/public/icons/xiaohongshu.svg +0 -1
  19. package/desktop/scripts/build-macos-arm64.sh +0 -270
  20. package/desktop/scripts/build-sidecars.ts +0 -183
  21. package/desktop/scripts/build-windows-x64.ps1 +0 -295
  22. package/desktop/scripts/scan-missing-imports.ts +0 -235
  23. package/desktop/sidecars/claude-sidecar.ts +0 -156
  24. package/desktop/src/App.tsx +0 -5
  25. package/desktop/src/__tests__/agentsSettings.test.tsx +0 -349
  26. package/desktop/src/__tests__/pages.test.tsx +0 -290
  27. package/desktop/src/__tests__/skillsSettings.test.tsx +0 -205
  28. package/desktop/src/api/adapters.ts +0 -12
  29. package/desktop/src/api/agents.ts +0 -36
  30. package/desktop/src/api/cliTasks.ts +0 -28
  31. package/desktop/src/api/client.ts +0 -63
  32. package/desktop/src/api/computerUse.ts +0 -76
  33. package/desktop/src/api/filesystem.ts +0 -30
  34. package/desktop/src/api/hahaOAuth.ts +0 -38
  35. package/desktop/src/api/models.ts +0 -28
  36. package/desktop/src/api/providers.ts +0 -63
  37. package/desktop/src/api/search.ts +0 -29
  38. package/desktop/src/api/sessions.ts +0 -56
  39. package/desktop/src/api/settings.ts +0 -20
  40. package/desktop/src/api/skills.ts +0 -19
  41. package/desktop/src/api/tasks.ts +0 -36
  42. package/desktop/src/api/teams.ts +0 -44
  43. package/desktop/src/api/websocket.ts +0 -164
  44. package/desktop/src/components/chat/AskUserQuestion.tsx +0 -268
  45. package/desktop/src/components/chat/AssistantMessage.tsx +0 -29
  46. package/desktop/src/components/chat/AttachmentGallery.tsx +0 -113
  47. package/desktop/src/components/chat/ChatInput.tsx +0 -622
  48. package/desktop/src/components/chat/CodeViewer.tsx +0 -161
  49. package/desktop/src/components/chat/ComputerUsePermissionModal.test.tsx +0 -174
  50. package/desktop/src/components/chat/ComputerUsePermissionModal.tsx +0 -311
  51. package/desktop/src/components/chat/DiffViewer.tsx +0 -157
  52. package/desktop/src/components/chat/FileSearchMenu.tsx +0 -198
  53. package/desktop/src/components/chat/ImageGalleryModal.tsx +0 -91
  54. package/desktop/src/components/chat/InlineImageGallery.tsx +0 -106
  55. package/desktop/src/components/chat/InlineTaskSummary.tsx +0 -60
  56. package/desktop/src/components/chat/MermaidRenderer.test.tsx +0 -98
  57. package/desktop/src/components/chat/MermaidRenderer.tsx +0 -361
  58. package/desktop/src/components/chat/MessageActionBar.tsx +0 -27
  59. package/desktop/src/components/chat/MessageList.test.tsx +0 -313
  60. package/desktop/src/components/chat/MessageList.tsx +0 -249
  61. package/desktop/src/components/chat/PermissionDialog.tsx +0 -262
  62. package/desktop/src/components/chat/SessionTaskBar.test.tsx +0 -99
  63. package/desktop/src/components/chat/SessionTaskBar.tsx +0 -159
  64. package/desktop/src/components/chat/StreamingIndicator.tsx +0 -41
  65. package/desktop/src/components/chat/TerminalChrome.tsx +0 -35
  66. package/desktop/src/components/chat/ThinkingBlock.tsx +0 -87
  67. package/desktop/src/components/chat/ToolCallBlock.tsx +0 -247
  68. package/desktop/src/components/chat/ToolCallGroup.tsx +0 -617
  69. package/desktop/src/components/chat/ToolResultBlock.tsx +0 -107
  70. package/desktop/src/components/chat/UserMessage.tsx +0 -38
  71. package/desktop/src/components/chat/chatBlocks.test.tsx +0 -136
  72. package/desktop/src/components/chat/clipboard.ts +0 -25
  73. package/desktop/src/components/chat/composerUtils.test.ts +0 -55
  74. package/desktop/src/components/chat/composerUtils.ts +0 -149
  75. package/desktop/src/components/controls/ModelSelector.tsx +0 -156
  76. package/desktop/src/components/controls/PermissionModeSelector.tsx +0 -229
  77. package/desktop/src/components/layout/AppShell.tsx +0 -107
  78. package/desktop/src/components/layout/ContentRouter.tsx +0 -27
  79. package/desktop/src/components/layout/ProjectFilter.tsx +0 -126
  80. package/desktop/src/components/layout/Sidebar.test.tsx +0 -158
  81. package/desktop/src/components/layout/Sidebar.tsx +0 -384
  82. package/desktop/src/components/layout/StatusBar.tsx +0 -31
  83. package/desktop/src/components/layout/TabBar.test.tsx +0 -136
  84. package/desktop/src/components/layout/TabBar.tsx +0 -318
  85. package/desktop/src/components/layout/TitleBar.tsx +0 -96
  86. package/desktop/src/components/layout/WindowControls.test.tsx +0 -69
  87. package/desktop/src/components/layout/WindowControls.tsx +0 -89
  88. package/desktop/src/components/markdown/MarkdownRenderer.test.tsx +0 -100
  89. package/desktop/src/components/markdown/MarkdownRenderer.tsx +0 -229
  90. package/desktop/src/components/settings/ClaudeOfficialLogin.tsx +0 -107
  91. package/desktop/src/components/shared/Button.tsx +0 -63
  92. package/desktop/src/components/shared/CopyButton.tsx +0 -58
  93. package/desktop/src/components/shared/DirectoryPicker.tsx +0 -316
  94. package/desktop/src/components/shared/Dropdown.tsx +0 -91
  95. package/desktop/src/components/shared/Input.tsx +0 -38
  96. package/desktop/src/components/shared/Modal.tsx +0 -65
  97. package/desktop/src/components/shared/ProjectContextChip.tsx +0 -30
  98. package/desktop/src/components/shared/Spinner.tsx +0 -30
  99. package/desktop/src/components/shared/Textarea.tsx +0 -38
  100. package/desktop/src/components/shared/Toast.tsx +0 -47
  101. package/desktop/src/components/shared/UpdateChecker.tsx +0 -90
  102. package/desktop/src/components/skills/SkillDetail.test.tsx +0 -89
  103. package/desktop/src/components/skills/SkillDetail.tsx +0 -403
  104. package/desktop/src/components/skills/SkillList.tsx +0 -254
  105. package/desktop/src/components/tasks/DayOfWeekPicker.tsx +0 -57
  106. package/desktop/src/components/tasks/NewTaskModal.tsx +0 -407
  107. package/desktop/src/components/tasks/PromptEditor.tsx +0 -74
  108. package/desktop/src/components/tasks/TaskEmptyState.tsx +0 -30
  109. package/desktop/src/components/tasks/TaskList.tsx +0 -46
  110. package/desktop/src/components/tasks/TaskRow.tsx +0 -253
  111. package/desktop/src/components/tasks/TaskRunsPanel.tsx +0 -195
  112. package/desktop/src/components/teams/TeamStatusBar.tsx +0 -147
  113. package/desktop/src/config/providerPresets.ts +0 -78
  114. package/desktop/src/config/spinnerVerbs.ts +0 -193
  115. package/desktop/src/hooks/useKeyboardShortcuts.ts +0 -60
  116. package/desktop/src/i18n/index.ts +0 -54
  117. package/desktop/src/i18n/locales/en.ts +0 -670
  118. package/desktop/src/i18n/locales/zh.ts +0 -670
  119. package/desktop/src/lib/__tests__/cronDescribe.test.ts +0 -93
  120. package/desktop/src/lib/cronDescribe.ts +0 -188
  121. package/desktop/src/lib/desktopRuntime.ts +0 -54
  122. package/desktop/src/lib/parseRunOutput.ts +0 -79
  123. package/desktop/src/main.tsx +0 -13
  124. package/desktop/src/mocks/data.ts +0 -202
  125. package/desktop/src/pages/ActiveSession.test.tsx +0 -181
  126. package/desktop/src/pages/ActiveSession.tsx +0 -219
  127. package/desktop/src/pages/AdapterSettings.tsx +0 -375
  128. package/desktop/src/pages/AgentTeams.tsx +0 -200
  129. package/desktop/src/pages/ComputerUseSettings.tsx +0 -420
  130. package/desktop/src/pages/EmptySession.tsx +0 -518
  131. package/desktop/src/pages/NewTaskModal.tsx +0 -346
  132. package/desktop/src/pages/ScheduledTasks.tsx +0 -66
  133. package/desktop/src/pages/ScheduledTasksEmpty.tsx +0 -152
  134. package/desktop/src/pages/ScheduledTasksList.tsx +0 -416
  135. package/desktop/src/pages/SessionControls.tsx +0 -460
  136. package/desktop/src/pages/Settings.tsx +0 -1448
  137. package/desktop/src/pages/ToolInspection.tsx +0 -235
  138. package/desktop/src/stores/adapterStore.ts +0 -106
  139. package/desktop/src/stores/agentStore.ts +0 -34
  140. package/desktop/src/stores/chatStore.test.ts +0 -505
  141. package/desktop/src/stores/chatStore.ts +0 -850
  142. package/desktop/src/stores/cliTaskStore.ts +0 -152
  143. package/desktop/src/stores/hahaOAuthStore.test.ts +0 -77
  144. package/desktop/src/stores/hahaOAuthStore.ts +0 -97
  145. package/desktop/src/stores/providerStore.ts +0 -101
  146. package/desktop/src/stores/sessionStore.test.ts +0 -63
  147. package/desktop/src/stores/sessionStore.ts +0 -102
  148. package/desktop/src/stores/settingsStore.ts +0 -120
  149. package/desktop/src/stores/skillStore.ts +0 -51
  150. package/desktop/src/stores/tabStore.ts +0 -169
  151. package/desktop/src/stores/taskStore.ts +0 -68
  152. package/desktop/src/stores/teamStore.ts +0 -344
  153. package/desktop/src/stores/uiStore.ts +0 -100
  154. package/desktop/src/stores/updateStore.test.ts +0 -71
  155. package/desktop/src/stores/updateStore.ts +0 -221
  156. package/desktop/src/theme/globals.css +0 -465
  157. package/desktop/src/types/adapter.ts +0 -33
  158. package/desktop/src/types/chat.ts +0 -152
  159. package/desktop/src/types/cliTask.ts +0 -24
  160. package/desktop/src/types/provider.ts +0 -62
  161. package/desktop/src/types/session.ts +0 -27
  162. package/desktop/src/types/settings.ts +0 -22
  163. package/desktop/src/types/skill.ts +0 -38
  164. package/desktop/src/types/task.ts +0 -56
  165. package/desktop/src/types/team.ts +0 -38
  166. package/desktop/src-tauri/Cargo.lock +0 -5549
  167. package/desktop/src-tauri/Cargo.toml +0 -20
  168. package/desktop/src-tauri/app-icon.svg +0 -13
  169. package/desktop/src-tauri/build.rs +0 -3
  170. package/desktop/src-tauri/capabilities/default.json +0 -106
  171. package/desktop/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml +0 -5
  172. package/desktop/src-tauri/icons/android/values/ic_launcher_background.xml +0 -4
  173. package/desktop/src-tauri/icons/icon.icns +0 -0
  174. package/desktop/src-tauri/icons/icon.ico +0 -0
  175. package/desktop/src-tauri/src/lib.rs +0 -408
  176. package/desktop/src-tauri/src/main.rs +0 -6
  177. package/desktop/src-tauri/tauri.conf.json +0 -78
  178. package/desktop/src-tauri/tauri.macos.conf.json +0 -18
  179. package/desktop/src-tauri/tauri.release-ci.json +0 -5
  180. package/desktop/src-tauri/tauri.windows.conf.json +0 -16
  181. package/desktop/src-tauri/windows-installer-hooks.nsh +0 -17
  182. package/desktop/tsconfig.json +0 -25
  183. package/desktop/vite.config.ts +0 -26
  184. package/desktop/vitest.config.ts +0 -18
@@ -1,161 +0,0 @@
1
- import { useState, useRef, useEffect } from 'react'
2
- import { ShikiHighlighter } from 'react-shiki'
3
- import 'react-shiki/css'
4
- import { CopyButton } from '../shared/CopyButton'
5
-
6
- type Props = {
7
- code: string
8
- language?: string
9
- maxLines?: number
10
- showLineNumbers?: boolean
11
- }
12
-
13
- /**
14
- * Custom warm-toned TextMate theme — uses VS Code-quality tokenization
15
- * while harmonizing with the app's cream/terra-cotta design system.
16
- */
17
- const warmCodeTheme = {
18
- name: 'warm-code',
19
- type: 'dark' as const,
20
- fg: 'var(--color-code-fg)',
21
- bg: 'transparent',
22
- tokenColors: [
23
- { scope: ['comment', 'punctuation.definition.comment'], settings: { foreground: 'var(--color-code-comment)', fontStyle: 'italic' } },
24
- { scope: ['string', 'string.quoted', 'string.template', 'string.other.link'], settings: { foreground: 'var(--color-code-string)' } },
25
- { scope: ['string.regexp'], settings: { foreground: 'var(--color-primary-container)' } },
26
- { scope: ['keyword', 'keyword.control', 'storage', 'storage.type', 'storage.modifier'], settings: { foreground: 'var(--color-code-keyword)' } },
27
- { scope: ['keyword.operator'], settings: { foreground: 'var(--color-code-keyword)' } },
28
- { scope: ['entity.name.function', 'support.function'], settings: { foreground: 'var(--color-code-function)' } },
29
- { scope: ['entity.name.type', 'support.type', 'support.class', 'entity.name.class', 'entity.other.inherited-class'], settings: { foreground: 'var(--color-code-type)' } },
30
- { scope: ['entity.name.type.parameter'], settings: { foreground: 'var(--color-code-number)' } },
31
- { scope: ['variable', 'variable.other', 'variable.other.readwrite'], settings: { foreground: 'var(--color-code-fg)' } },
32
- { scope: ['variable.parameter'], settings: { foreground: 'var(--color-code-parameter)' } },
33
- { scope: ['variable.other.property', 'support.type.property-name', 'meta.object-literal.key'], settings: { foreground: 'var(--color-code-property)' } },
34
- { scope: ['variable.other.constant', 'variable.other.enummember'], settings: { foreground: 'var(--color-code-type)' } },
35
- { scope: ['constant.numeric', 'constant.language'], settings: { foreground: 'var(--color-code-number)' } },
36
- { scope: ['punctuation', 'meta.brace', 'meta.bracket'], settings: { foreground: 'var(--color-code-punctuation)' } },
37
- { scope: ['entity.name.tag', 'punctuation.definition.tag'], settings: { foreground: 'var(--color-code-keyword)' } },
38
- { scope: ['entity.other.attribute-name'], settings: { foreground: 'var(--color-code-property)' } },
39
- { scope: ['meta.decorator', 'punctuation.decorator'], settings: { foreground: 'var(--color-code-type)' } },
40
- { scope: ['markup.inserted', 'punctuation.definition.inserted'], settings: { foreground: 'var(--color-code-inserted)' } },
41
- { scope: ['markup.deleted', 'punctuation.definition.deleted'], settings: { foreground: 'var(--color-code-deleted)' } },
42
- { scope: ['markup.heading', 'entity.name.section'], settings: { foreground: 'var(--color-code-function)', fontStyle: 'bold' } },
43
- { scope: ['markup.bold'], settings: { fontStyle: 'bold' } },
44
- { scope: ['markup.italic'], settings: { fontStyle: 'italic' } },
45
- ],
46
- }
47
-
48
- /**
49
- * Wraps ShikiHighlighter with a plain-text fallback so the code area
50
- * is never empty while the async WASM / language-grammar load is in-flight,
51
- * or if highlighting fails entirely.
52
- */
53
- function CodeArea({ code, language, showLineNumbers }: { code: string; language?: string; showLineNumbers: boolean }) {
54
- const containerRef = useRef<HTMLDivElement>(null)
55
- const [loaded, setLoaded] = useState(false)
56
-
57
- useEffect(() => {
58
- // ShikiHighlighter renders `null` until the async highlight completes.
59
- // Watch for real content appearing via MutationObserver so we can hide
60
- // the plain-text fallback as soon as highlighted output is in the DOM.
61
- const el = containerRef.current
62
- if (!el) return
63
- const check = () => {
64
- const shikiContainer = el.querySelector('[data-testid="shiki-container"]')
65
- // shiki renders a <code> element inside its container once highlighting is done
66
- if (shikiContainer?.querySelector('code')) {
67
- setLoaded(true)
68
- }
69
- }
70
- check()
71
- const observer = new MutationObserver(check)
72
- observer.observe(el, { childList: true, subtree: true })
73
- return () => observer.disconnect()
74
- }, [code, language])
75
-
76
- return (
77
- <div ref={containerRef} className="code-viewer-area max-h-[420px] overflow-auto bg-[var(--color-code-bg)]">
78
- {/* Plain-text fallback shown until Shiki finishes highlighting */}
79
- {!loaded && (
80
- <pre
81
- style={{
82
- margin: 0,
83
- padding: '0.5rem 12px',
84
- fontFamily: 'var(--font-mono)',
85
- fontSize: '12px',
86
- lineHeight: '1.45',
87
- whiteSpace: 'pre-wrap',
88
- wordBreak: 'break-word',
89
- color: 'var(--color-code-fg)',
90
- }}
91
- >
92
- {code}
93
- </pre>
94
- )}
95
- <div style={loaded ? undefined : { position: 'absolute', opacity: 0, pointerEvents: 'none' }}>
96
- <ShikiHighlighter
97
- language={language || 'text'}
98
- theme={warmCodeTheme}
99
- showLineNumbers={showLineNumbers}
100
- showLanguage={false}
101
- addDefaultStyles={false}
102
- style={{
103
- margin: 0,
104
- padding: '0.5rem 0',
105
- fontFamily: 'var(--font-mono)',
106
- fontSize: '12px',
107
- lineHeight: '1.45',
108
- }}
109
- >
110
- {code}
111
- </ShikiHighlighter>
112
- </div>
113
- </div>
114
- )
115
- }
116
-
117
- export function CodeViewer({ code, language, maxLines = 20, showLineNumbers = true }: Props) {
118
- const [expanded, setExpanded] = useState(false)
119
-
120
- const allLines = code.split('\n')
121
- const isTruncated = !expanded && allLines.length > maxLines
122
- const visibleCode = isTruncated ? allLines.slice(0, maxLines).join('\n') : code
123
-
124
- const effectiveShowLineNumbers = showLineNumbers && !!language && language !== 'text'
125
- const languageLabel = language || 'code'
126
- const lineCountLabel = `${allLines.length} ${allLines.length === 1 ? 'line' : 'lines'}`
127
- const showExpandToggle = allLines.length > maxLines
128
-
129
- return (
130
- <div className="overflow-hidden rounded-[var(--radius-lg)] border border-[var(--color-outline-variant)]/50 bg-[var(--color-surface-container-low)]">
131
- {/* Header */}
132
- <div className="flex items-center justify-between border-b border-[var(--color-outline-variant)]/40 bg-[var(--color-surface-container)] px-3 py-1.5 text-[11px] text-[var(--color-text-tertiary)]">
133
- <div className="flex items-center gap-3">
134
- <span className="font-semibold uppercase tracking-[0.14em]">{languageLabel}</span>
135
- <span>{lineCountLabel}</span>
136
- </div>
137
- <CopyButton
138
- text={code}
139
- className="rounded-md border border-[var(--color-outline-variant)]/40 bg-[var(--color-surface-container-lowest)] px-2 py-1 text-[11px] text-[var(--color-text-tertiary)] transition-colors hover:bg-[var(--color-surface-container-high)] hover:text-[var(--color-text-primary)]"
140
- />
141
- </div>
142
-
143
- {/* Code area */}
144
- <CodeArea
145
- code={visibleCode}
146
- language={language}
147
- showLineNumbers={effectiveShowLineNumbers}
148
- />
149
-
150
- {/* Expand/collapse toggle */}
151
- {showExpandToggle && (
152
- <button
153
- onClick={() => setExpanded((value) => !value)}
154
- className="w-full border-t border-[var(--color-outline-variant)]/40 bg-[var(--color-surface-container)] py-1.5 text-[10px] font-semibold uppercase tracking-[0.14em] text-[var(--color-text-tertiary)] transition-colors hover:bg-[var(--color-surface-container-high)] hover:text-[var(--color-text-primary)]"
155
- >
156
- {expanded ? 'Collapse' : `Show ${allLines.length - maxLines} more lines`}
157
- </button>
158
- )}
159
- </div>
160
- )
161
- }
@@ -1,174 +0,0 @@
1
- import { beforeEach, describe, expect, it, vi } from 'vitest'
2
- import { act, fireEvent, render, screen } from '@testing-library/react'
3
-
4
- const { sendMock, openSettingsMock } = vi.hoisted(() => ({
5
- sendMock: vi.fn(),
6
- openSettingsMock: vi.fn(async () => ({ ok: true })),
7
- }))
8
-
9
- vi.mock('../../api/websocket', () => ({
10
- wsManager: {
11
- connect: vi.fn(),
12
- disconnect: vi.fn(),
13
- onMessage: vi.fn(() => () => {}),
14
- clearHandlers: vi.fn(),
15
- send: sendMock,
16
- },
17
- }))
18
-
19
- vi.mock('../../api/sessions', () => ({
20
- sessionsApi: {
21
- getMessages: vi.fn(async () => ({ messages: [] })),
22
- getSlashCommands: vi.fn(async () => ({ commands: [] })),
23
- },
24
- }))
25
-
26
- vi.mock('../../stores/teamStore', () => ({
27
- useTeamStore: {
28
- getState: () => ({
29
- getMemberBySessionId: vi.fn(() => null),
30
- sendMessageToMember: vi.fn(async () => {}),
31
- handleTeamCreated: vi.fn(),
32
- handleTeamUpdate: vi.fn(),
33
- handleTeamDeleted: vi.fn(),
34
- }),
35
- },
36
- }))
37
-
38
- vi.mock('../../stores/tabStore', () => ({
39
- useTabStore: {
40
- getState: () => ({
41
- updateTabStatus: vi.fn(),
42
- updateTabTitle: vi.fn(),
43
- }),
44
- },
45
- }))
46
-
47
- vi.mock('../../stores/sessionStore', () => ({
48
- useSessionStore: {
49
- getState: () => ({
50
- updateSessionTitle: vi.fn(),
51
- }),
52
- },
53
- }))
54
-
55
- vi.mock('../../stores/cliTaskStore', () => ({
56
- useCLITaskStore: {
57
- getState: () => ({
58
- fetchSessionTasks: vi.fn(),
59
- tasks: [],
60
- clearTasks: vi.fn(),
61
- setTasksFromTodos: vi.fn(),
62
- markCompletedAndDismissed: vi.fn(),
63
- refreshTasks: vi.fn(),
64
- }),
65
- },
66
- }))
67
-
68
- vi.mock('../../api/computerUse', () => ({
69
- computerUseApi: {
70
- openSettings: openSettingsMock,
71
- },
72
- }))
73
-
74
- import { useChatStore } from '../../stores/chatStore'
75
- import { ComputerUsePermissionModal } from './ComputerUsePermissionModal'
76
-
77
- describe('ComputerUsePermissionModal', () => {
78
- beforeEach(() => {
79
- sendMock.mockReset()
80
- openSettingsMock.mockReset()
81
- useChatStore.setState({ sessions: {} })
82
- })
83
-
84
- it('returns a full approval payload for resolved apps and requested flags', () => {
85
- render(
86
- <ComputerUsePermissionModal
87
- sessionId="session-1"
88
- request={{
89
- requestId: 'cu-1',
90
- reason: 'Open Finder and inspect a file',
91
- apps: [
92
- {
93
- requestedName: 'Finder',
94
- resolved: {
95
- bundleId: 'com.apple.finder',
96
- displayName: 'Finder',
97
- },
98
- isSentinel: false,
99
- alreadyGranted: false,
100
- proposedTier: 'full',
101
- },
102
- {
103
- requestedName: 'Missing App',
104
- isSentinel: false,
105
- alreadyGranted: false,
106
- proposedTier: 'full',
107
- },
108
- ],
109
- requestedFlags: {
110
- clipboardRead: true,
111
- systemKeyCombos: true,
112
- },
113
- screenshotFiltering: 'native',
114
- willHide: [{ bundleId: 'com.apple.TextEdit', displayName: 'TextEdit' }],
115
- autoUnhideEnabled: true,
116
- }}
117
- />,
118
- )
119
-
120
- fireEvent.click(screen.getByRole('button', { name: 'Allow for session' }))
121
-
122
- expect(sendMock).toHaveBeenCalledTimes(1)
123
- expect(sendMock).toHaveBeenCalledWith('session-1', {
124
- type: 'computer_use_permission_response',
125
- requestId: 'cu-1',
126
- response: {
127
- granted: [
128
- expect.objectContaining({
129
- bundleId: 'com.apple.finder',
130
- displayName: 'Finder',
131
- tier: 'full',
132
- }),
133
- ],
134
- denied: [
135
- {
136
- bundleId: 'Missing App',
137
- reason: 'not_installed',
138
- },
139
- ],
140
- flags: {
141
- clipboardRead: true,
142
- clipboardWrite: false,
143
- systemKeyCombos: true,
144
- },
145
- userConsented: true,
146
- },
147
- })
148
- })
149
-
150
- it('opens System Settings from the macOS permission panel', async () => {
151
- render(
152
- <ComputerUsePermissionModal
153
- sessionId="session-1"
154
- request={{
155
- requestId: 'cu-1',
156
- reason: '',
157
- apps: [],
158
- requestedFlags: {},
159
- screenshotFiltering: 'native',
160
- tccState: {
161
- accessibility: false,
162
- screenRecording: true,
163
- },
164
- }}
165
- />,
166
- )
167
-
168
- await act(async () => {
169
- fireEvent.click(screen.getByRole('button', { name: 'Open Accessibility' }))
170
- })
171
-
172
- expect(openSettingsMock).toHaveBeenCalledWith('Privacy_Accessibility')
173
- })
174
- })
@@ -1,311 +0,0 @@
1
- import { useMemo, useState } from 'react'
2
- import { useTranslation } from '../../i18n'
3
- import { computerUseApi } from '../../api/computerUse'
4
- import { useChatStore } from '../../stores/chatStore'
5
- import type {
6
- ComputerUsePermissionRequest,
7
- ComputerUsePermissionResponse,
8
- } from '../../types/chat'
9
- import { Button } from '../shared/Button'
10
- import { Modal } from '../shared/Modal'
11
-
12
- type Props = {
13
- sessionId: string
14
- request: ComputerUsePermissionRequest | null
15
- }
16
-
17
- const DEFAULT_GRANT_FLAGS = {
18
- clipboardRead: false,
19
- clipboardWrite: false,
20
- systemKeyCombos: false,
21
- } as const
22
-
23
- function denyAllResponse(): ComputerUsePermissionResponse {
24
- return {
25
- granted: [],
26
- denied: [],
27
- flags: { ...DEFAULT_GRANT_FLAGS },
28
- userConsented: false,
29
- }
30
- }
31
-
32
- function buildAllowResponse(
33
- request: ComputerUsePermissionRequest,
34
- ): ComputerUsePermissionResponse {
35
- const now = Date.now()
36
- const granted = request.apps.flatMap((app) => {
37
- if (!app.resolved || app.alreadyGranted) return []
38
- return [{
39
- bundleId: app.resolved.bundleId,
40
- displayName: app.resolved.displayName,
41
- grantedAt: now,
42
- tier: app.proposedTier,
43
- }]
44
- })
45
-
46
- const denied = request.apps.flatMap((app) => {
47
- if (app.resolved) return []
48
- return [{
49
- bundleId: app.requestedName,
50
- reason: 'not_installed' as const,
51
- }]
52
- })
53
-
54
- const flags = {
55
- ...DEFAULT_GRANT_FLAGS,
56
- ...Object.fromEntries(
57
- Object.entries(request.requestedFlags).filter(([, value]) => value === true),
58
- ),
59
- }
60
-
61
- return {
62
- granted,
63
- denied,
64
- flags,
65
- userConsented: true,
66
- }
67
- }
68
-
69
- export function ComputerUsePermissionModal({ sessionId, request }: Props) {
70
- const t = useTranslation()
71
- const respondToComputerUsePermission = useChatStore(
72
- (s) => s.respondToComputerUsePermission,
73
- )
74
- const [openingPane, setOpeningPane] = useState<
75
- 'Privacy_Accessibility' | 'Privacy_ScreenCapture' | null
76
- >(null)
77
-
78
- const requestedFlags = useMemo(
79
- () =>
80
- request
81
- ? Object.entries(request.requestedFlags)
82
- .filter(([, enabled]) => enabled)
83
- .map(([flag]) => flag)
84
- : [],
85
- [request],
86
- )
87
-
88
- if (!request) return null
89
-
90
- const handleDeny = () => {
91
- respondToComputerUsePermission(
92
- sessionId,
93
- request.requestId,
94
- denyAllResponse(),
95
- )
96
- }
97
-
98
- const handleAllow = () => {
99
- respondToComputerUsePermission(
100
- sessionId,
101
- request.requestId,
102
- buildAllowResponse(request),
103
- )
104
- }
105
-
106
- const openSettings = async (
107
- pane: 'Privacy_Accessibility' | 'Privacy_ScreenCapture',
108
- ) => {
109
- setOpeningPane(pane)
110
- try {
111
- await computerUseApi.openSettings(pane)
112
- } finally {
113
- setOpeningPane(null)
114
- }
115
- }
116
-
117
- const tccState = request.tccState
118
-
119
- return (
120
- <Modal
121
- open
122
- onClose={handleDeny}
123
- title={
124
- tccState
125
- ? t('computerUseApproval.titleTcc')
126
- : t('computerUseApproval.titleApps')
127
- }
128
- width={640}
129
- footer={
130
- tccState ? (
131
- <Button variant="ghost" onClick={handleDeny}>
132
- {t('computerUseApproval.deny')}
133
- </Button>
134
- ) : (
135
- <>
136
- <Button variant="ghost" onClick={handleDeny}>
137
- {t('computerUseApproval.deny')}
138
- </Button>
139
- <Button variant="primary" onClick={handleAllow}>
140
- {t('computerUseApproval.allow')}
141
- </Button>
142
- </>
143
- )
144
- }
145
- >
146
- {tccState ? (
147
- <div className="space-y-4">
148
- <p className="text-sm text-[var(--color-text-secondary)]">
149
- {t('computerUseApproval.tccHint')}
150
- </p>
151
-
152
- <div className="space-y-3">
153
- <PermissionRow
154
- label={t('computerUseApproval.accessibility')}
155
- granted={tccState.accessibility}
156
- actionLabel={t('computerUseApproval.openAccessibility')}
157
- actionLoading={openingPane === 'Privacy_Accessibility'}
158
- onAction={() => openSettings('Privacy_Accessibility')}
159
- />
160
- <PermissionRow
161
- label={t('computerUseApproval.screenRecording')}
162
- granted={tccState.screenRecording}
163
- actionLabel={t('computerUseApproval.openScreenRecording')}
164
- actionLoading={openingPane === 'Privacy_ScreenCapture'}
165
- onAction={() => openSettings('Privacy_ScreenCapture')}
166
- />
167
- </div>
168
-
169
- <div className="rounded-[var(--radius-lg)] border border-[var(--color-border)] bg-[var(--color-surface-container-low)] p-3 text-xs text-[var(--color-text-tertiary)]">
170
- {t('computerUseApproval.tryAgainHint')}
171
- </div>
172
-
173
- <div className="flex justify-end">
174
- <Button variant="secondary" onClick={handleDeny}>
175
- {t('computerUseApproval.tryAgain')}
176
- </Button>
177
- </div>
178
- </div>
179
- ) : (
180
- <div className="space-y-4">
181
- {request.reason ? (
182
- <div className="rounded-[var(--radius-lg)] border border-[var(--color-border)] bg-[var(--color-surface-container-low)] p-3">
183
- <div className="text-xs font-semibold uppercase tracking-wide text-[var(--color-text-tertiary)]">
184
- {t('computerUseApproval.reason')}
185
- </div>
186
- <div className="mt-1 text-sm text-[var(--color-text-primary)]">
187
- {request.reason}
188
- </div>
189
- </div>
190
- ) : null}
191
-
192
- <div className="space-y-2">
193
- {request.apps.map((app) => {
194
- const resolved = app.resolved
195
- return (
196
- <div
197
- key={resolved?.bundleId ?? app.requestedName}
198
- className="rounded-[var(--radius-lg)] border border-[var(--color-border)] bg-[var(--color-surface-container-low)] p-3"
199
- >
200
- <div className="flex items-start justify-between gap-3">
201
- <div>
202
- <div className="text-sm font-semibold text-[var(--color-text-primary)]">
203
- {resolved?.displayName ?? app.requestedName}
204
- </div>
205
- <div className="mt-1 text-xs text-[var(--color-text-tertiary)]">
206
- {resolved?.bundleId ?? t('computerUseApproval.notInstalled')}
207
- </div>
208
- </div>
209
- <span className="rounded-full bg-[var(--color-surface-container)] px-2 py-1 text-[10px] font-semibold uppercase tracking-wide text-[var(--color-text-secondary)]">
210
- {app.proposedTier}
211
- </span>
212
- </div>
213
-
214
- {!resolved ? (
215
- <p className="mt-2 text-xs text-[var(--color-error)]">
216
- {t('computerUseApproval.notInstalled')}
217
- </p>
218
- ) : null}
219
-
220
- {app.alreadyGranted ? (
221
- <p className="mt-2 text-xs text-[var(--color-success)]">
222
- {t('computerUseApproval.alreadyGranted')}
223
- </p>
224
- ) : null}
225
-
226
- {app.isSentinel ? (
227
- <p className="mt-2 text-xs text-[var(--color-warning)]">
228
- {t('computerUseApproval.sensitiveApp')}
229
- </p>
230
- ) : null}
231
- </div>
232
- )
233
- })}
234
- </div>
235
-
236
- {requestedFlags.length > 0 ? (
237
- <div className="rounded-[var(--radius-lg)] border border-[var(--color-border)] bg-[var(--color-surface-container-low)] p-3">
238
- <div className="text-xs font-semibold uppercase tracking-wide text-[var(--color-text-tertiary)]">
239
- {t('computerUseApproval.alsoRequested')}
240
- </div>
241
- <div className="mt-2 flex flex-wrap gap-2">
242
- {requestedFlags.map((flag) => (
243
- <span
244
- key={flag}
245
- className="rounded-full bg-[var(--color-surface-container)] px-2 py-1 text-[11px] text-[var(--color-text-secondary)]"
246
- >
247
- {flag}
248
- </span>
249
- ))}
250
- </div>
251
- </div>
252
- ) : null}
253
-
254
- {request.willHide && request.willHide.length > 0 ? (
255
- <div className="rounded-[var(--radius-lg)] border border-[var(--color-border)] bg-[var(--color-surface-container-low)] p-3 text-sm text-[var(--color-text-secondary)]">
256
- {request.autoUnhideEnabled
257
- ? t('computerUseApproval.hideWhileWorkingRestore', {
258
- count: request.willHide.length,
259
- })
260
- : t('computerUseApproval.hideWhileWorking', {
261
- count: request.willHide.length,
262
- })}
263
- </div>
264
- ) : null}
265
- </div>
266
- )}
267
- </Modal>
268
- )
269
- }
270
-
271
- function PermissionRow({
272
- label,
273
- granted,
274
- actionLabel,
275
- actionLoading,
276
- onAction,
277
- }: {
278
- label: string
279
- granted: boolean
280
- actionLabel: string
281
- actionLoading: boolean
282
- onAction: () => void
283
- }) {
284
- const t = useTranslation()
285
-
286
- return (
287
- <div className="flex items-center justify-between gap-4 rounded-[var(--radius-lg)] border border-[var(--color-border)] bg-[var(--color-surface-container-low)] p-3">
288
- <div>
289
- <div className="text-sm font-semibold text-[var(--color-text-primary)]">
290
- {label}
291
- </div>
292
- <div className="mt-1 text-xs text-[var(--color-text-tertiary)]">
293
- {granted
294
- ? t('computerUseApproval.granted')
295
- : t('computerUseApproval.notGranted')}
296
- </div>
297
- </div>
298
-
299
- {!granted ? (
300
- <Button
301
- variant="secondary"
302
- size="sm"
303
- loading={actionLoading}
304
- onClick={onAction}
305
- >
306
- {actionLabel}
307
- </Button>
308
- ) : null}
309
- </div>
310
- )
311
- }