@shareai-lab/kode 1.1.14 → 1.1.16-dev.2

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 (289) hide show
  1. package/cli.js +77 -82
  2. package/dist/entrypoints/cli.js +59 -38
  3. package/dist/entrypoints/cli.js.map +3 -3
  4. package/dist/index.js +5 -26
  5. package/dist/package.json +4 -1
  6. package/package.json +11 -104
  7. package/dist/test/testAdapters.js +0 -88
  8. package/dist/test/testAdapters.js.map +0 -1
  9. package/src/ProjectOnboarding.tsx +0 -198
  10. package/src/Tool.ts +0 -83
  11. package/src/commands/agents.tsx +0 -3416
  12. package/src/commands/approvedTools.ts +0 -53
  13. package/src/commands/bug.tsx +0 -20
  14. package/src/commands/clear.ts +0 -43
  15. package/src/commands/compact.ts +0 -120
  16. package/src/commands/config.tsx +0 -19
  17. package/src/commands/cost.ts +0 -18
  18. package/src/commands/ctx_viz.ts +0 -209
  19. package/src/commands/doctor.ts +0 -24
  20. package/src/commands/help.tsx +0 -19
  21. package/src/commands/init.ts +0 -37
  22. package/src/commands/listen.ts +0 -42
  23. package/src/commands/login.tsx +0 -51
  24. package/src/commands/logout.tsx +0 -40
  25. package/src/commands/mcp.ts +0 -41
  26. package/src/commands/model.tsx +0 -40
  27. package/src/commands/modelstatus.tsx +0 -20
  28. package/src/commands/onboarding.tsx +0 -34
  29. package/src/commands/pr_comments.ts +0 -59
  30. package/src/commands/refreshCommands.ts +0 -54
  31. package/src/commands/release-notes.ts +0 -34
  32. package/src/commands/resume.tsx +0 -31
  33. package/src/commands/review.ts +0 -49
  34. package/src/commands/terminalSetup.ts +0 -221
  35. package/src/commands.ts +0 -139
  36. package/src/components/ApproveApiKey.tsx +0 -93
  37. package/src/components/AsciiLogo.tsx +0 -13
  38. package/src/components/AutoUpdater.tsx +0 -148
  39. package/src/components/Bug.tsx +0 -367
  40. package/src/components/Config.tsx +0 -293
  41. package/src/components/ConsoleOAuthFlow.tsx +0 -327
  42. package/src/components/Cost.tsx +0 -23
  43. package/src/components/CostThresholdDialog.tsx +0 -46
  44. package/src/components/CustomSelect/option-map.ts +0 -42
  45. package/src/components/CustomSelect/select-option.tsx +0 -78
  46. package/src/components/CustomSelect/select.tsx +0 -152
  47. package/src/components/CustomSelect/theme.ts +0 -45
  48. package/src/components/CustomSelect/use-select-state.ts +0 -414
  49. package/src/components/CustomSelect/use-select.ts +0 -35
  50. package/src/components/FallbackToolUseRejectedMessage.tsx +0 -15
  51. package/src/components/FileEditToolUpdatedMessage.tsx +0 -66
  52. package/src/components/Help.tsx +0 -215
  53. package/src/components/HighlightedCode.tsx +0 -33
  54. package/src/components/InvalidConfigDialog.tsx +0 -113
  55. package/src/components/Link.tsx +0 -32
  56. package/src/components/LogSelector.tsx +0 -86
  57. package/src/components/Logo.tsx +0 -170
  58. package/src/components/MCPServerApprovalDialog.tsx +0 -100
  59. package/src/components/MCPServerDialogCopy.tsx +0 -25
  60. package/src/components/MCPServerMultiselectDialog.tsx +0 -109
  61. package/src/components/Message.tsx +0 -221
  62. package/src/components/MessageResponse.tsx +0 -15
  63. package/src/components/MessageSelector.tsx +0 -211
  64. package/src/components/ModeIndicator.tsx +0 -88
  65. package/src/components/ModelConfig.tsx +0 -301
  66. package/src/components/ModelListManager.tsx +0 -227
  67. package/src/components/ModelSelector.tsx +0 -3387
  68. package/src/components/ModelStatusDisplay.tsx +0 -230
  69. package/src/components/Onboarding.tsx +0 -274
  70. package/src/components/PressEnterToContinue.tsx +0 -11
  71. package/src/components/PromptInput.tsx +0 -760
  72. package/src/components/SentryErrorBoundary.ts +0 -39
  73. package/src/components/Spinner.tsx +0 -129
  74. package/src/components/StickerRequestForm.tsx +0 -16
  75. package/src/components/StructuredDiff.tsx +0 -191
  76. package/src/components/TextInput.tsx +0 -259
  77. package/src/components/TodoItem.tsx +0 -47
  78. package/src/components/TokenWarning.tsx +0 -31
  79. package/src/components/ToolUseLoader.tsx +0 -40
  80. package/src/components/TrustDialog.tsx +0 -106
  81. package/src/components/binary-feedback/BinaryFeedback.tsx +0 -63
  82. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +0 -111
  83. package/src/components/binary-feedback/BinaryFeedbackView.tsx +0 -172
  84. package/src/components/binary-feedback/utils.ts +0 -220
  85. package/src/components/messages/AssistantBashOutputMessage.tsx +0 -22
  86. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +0 -49
  87. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +0 -19
  88. package/src/components/messages/AssistantTextMessage.tsx +0 -144
  89. package/src/components/messages/AssistantThinkingMessage.tsx +0 -40
  90. package/src/components/messages/AssistantToolUseMessage.tsx +0 -132
  91. package/src/components/messages/TaskProgressMessage.tsx +0 -32
  92. package/src/components/messages/TaskToolMessage.tsx +0 -58
  93. package/src/components/messages/UserBashInputMessage.tsx +0 -28
  94. package/src/components/messages/UserCommandMessage.tsx +0 -30
  95. package/src/components/messages/UserKodingInputMessage.tsx +0 -28
  96. package/src/components/messages/UserPromptMessage.tsx +0 -35
  97. package/src/components/messages/UserTextMessage.tsx +0 -39
  98. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +0 -12
  99. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +0 -36
  100. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +0 -31
  101. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +0 -57
  102. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +0 -35
  103. package/src/components/messages/UserToolResultMessage/utils.tsx +0 -56
  104. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +0 -121
  105. package/src/components/permissions/FallbackPermissionRequest.tsx +0 -153
  106. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +0 -182
  107. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +0 -77
  108. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +0 -164
  109. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +0 -83
  110. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +0 -240
  111. package/src/components/permissions/PermissionRequest.tsx +0 -101
  112. package/src/components/permissions/PermissionRequestTitle.tsx +0 -69
  113. package/src/components/permissions/hooks.ts +0 -44
  114. package/src/components/permissions/toolUseOptions.ts +0 -59
  115. package/src/components/permissions/utils.ts +0 -23
  116. package/src/constants/betas.ts +0 -5
  117. package/src/constants/claude-asterisk-ascii-art.tsx +0 -238
  118. package/src/constants/figures.ts +0 -4
  119. package/src/constants/keys.ts +0 -3
  120. package/src/constants/macros.ts +0 -11
  121. package/src/constants/modelCapabilities.ts +0 -179
  122. package/src/constants/models.ts +0 -1025
  123. package/src/constants/oauth.ts +0 -18
  124. package/src/constants/product.ts +0 -17
  125. package/src/constants/prompts.ts +0 -168
  126. package/src/constants/releaseNotes.ts +0 -7
  127. package/src/context/PermissionContext.tsx +0 -149
  128. package/src/context.ts +0 -278
  129. package/src/cost-tracker.ts +0 -84
  130. package/src/entrypoints/cli.tsx +0 -1561
  131. package/src/entrypoints/mcp.ts +0 -175
  132. package/src/history.ts +0 -25
  133. package/src/hooks/useApiKeyVerification.ts +0 -59
  134. package/src/hooks/useArrowKeyHistory.ts +0 -55
  135. package/src/hooks/useCanUseTool.ts +0 -138
  136. package/src/hooks/useCancelRequest.ts +0 -39
  137. package/src/hooks/useDoublePress.ts +0 -41
  138. package/src/hooks/useExitOnCtrlCD.ts +0 -31
  139. package/src/hooks/useInterval.ts +0 -25
  140. package/src/hooks/useLogMessages.ts +0 -16
  141. package/src/hooks/useLogStartupTime.ts +0 -12
  142. package/src/hooks/useNotifyAfterTimeout.ts +0 -65
  143. package/src/hooks/usePermissionRequestLogging.ts +0 -44
  144. package/src/hooks/useTerminalSize.ts +0 -49
  145. package/src/hooks/useTextInput.ts +0 -317
  146. package/src/hooks/useUnifiedCompletion.ts +0 -1405
  147. package/src/index.ts +0 -34
  148. package/src/messages.ts +0 -38
  149. package/src/permissions.ts +0 -268
  150. package/src/query.ts +0 -720
  151. package/src/screens/ConfigureNpmPrefix.tsx +0 -197
  152. package/src/screens/Doctor.tsx +0 -219
  153. package/src/screens/LogList.tsx +0 -68
  154. package/src/screens/REPL.tsx +0 -813
  155. package/src/screens/ResumeConversation.tsx +0 -68
  156. package/src/services/adapters/base.ts +0 -38
  157. package/src/services/adapters/chatCompletions.ts +0 -90
  158. package/src/services/adapters/responsesAPI.ts +0 -170
  159. package/src/services/browserMocks.ts +0 -66
  160. package/src/services/claude.ts +0 -2197
  161. package/src/services/customCommands.ts +0 -704
  162. package/src/services/fileFreshness.ts +0 -377
  163. package/src/services/gpt5ConnectionTest.ts +0 -340
  164. package/src/services/mcpClient.ts +0 -564
  165. package/src/services/mcpServerApproval.tsx +0 -50
  166. package/src/services/mentionProcessor.ts +0 -273
  167. package/src/services/modelAdapterFactory.ts +0 -69
  168. package/src/services/notifier.ts +0 -40
  169. package/src/services/oauth.ts +0 -357
  170. package/src/services/openai.ts +0 -1359
  171. package/src/services/responseStateManager.ts +0 -90
  172. package/src/services/sentry.ts +0 -3
  173. package/src/services/statsig.ts +0 -172
  174. package/src/services/statsigStorage.ts +0 -86
  175. package/src/services/systemReminder.ts +0 -507
  176. package/src/services/vcr.ts +0 -161
  177. package/src/test/testAdapters.ts +0 -96
  178. package/src/tools/ArchitectTool/ArchitectTool.tsx +0 -135
  179. package/src/tools/ArchitectTool/prompt.ts +0 -15
  180. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +0 -576
  181. package/src/tools/BashTool/BashTool.tsx +0 -243
  182. package/src/tools/BashTool/BashToolResultMessage.tsx +0 -38
  183. package/src/tools/BashTool/OutputLine.tsx +0 -49
  184. package/src/tools/BashTool/prompt.ts +0 -174
  185. package/src/tools/BashTool/utils.ts +0 -56
  186. package/src/tools/FileEditTool/FileEditTool.tsx +0 -319
  187. package/src/tools/FileEditTool/prompt.ts +0 -51
  188. package/src/tools/FileEditTool/utils.ts +0 -58
  189. package/src/tools/FileReadTool/FileReadTool.tsx +0 -404
  190. package/src/tools/FileReadTool/prompt.ts +0 -7
  191. package/src/tools/FileWriteTool/FileWriteTool.tsx +0 -301
  192. package/src/tools/FileWriteTool/prompt.ts +0 -10
  193. package/src/tools/GlobTool/GlobTool.tsx +0 -119
  194. package/src/tools/GlobTool/prompt.ts +0 -8
  195. package/src/tools/GrepTool/GrepTool.tsx +0 -147
  196. package/src/tools/GrepTool/prompt.ts +0 -11
  197. package/src/tools/MCPTool/MCPTool.tsx +0 -107
  198. package/src/tools/MCPTool/prompt.ts +0 -3
  199. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +0 -127
  200. package/src/tools/MemoryReadTool/prompt.ts +0 -3
  201. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +0 -89
  202. package/src/tools/MemoryWriteTool/prompt.ts +0 -3
  203. package/src/tools/MultiEditTool/MultiEditTool.tsx +0 -388
  204. package/src/tools/MultiEditTool/prompt.ts +0 -45
  205. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +0 -298
  206. package/src/tools/NotebookEditTool/prompt.ts +0 -3
  207. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +0 -258
  208. package/src/tools/NotebookReadTool/prompt.ts +0 -3
  209. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +0 -107
  210. package/src/tools/StickerRequestTool/prompt.ts +0 -19
  211. package/src/tools/TaskTool/TaskTool.tsx +0 -438
  212. package/src/tools/TaskTool/constants.ts +0 -1
  213. package/src/tools/TaskTool/prompt.ts +0 -92
  214. package/src/tools/ThinkTool/ThinkTool.tsx +0 -54
  215. package/src/tools/ThinkTool/prompt.ts +0 -12
  216. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +0 -313
  217. package/src/tools/TodoWriteTool/prompt.ts +0 -63
  218. package/src/tools/URLFetcherTool/URLFetcherTool.tsx +0 -178
  219. package/src/tools/URLFetcherTool/cache.ts +0 -55
  220. package/src/tools/URLFetcherTool/htmlToMarkdown.ts +0 -55
  221. package/src/tools/URLFetcherTool/prompt.ts +0 -17
  222. package/src/tools/WebSearchTool/WebSearchTool.tsx +0 -103
  223. package/src/tools/WebSearchTool/prompt.ts +0 -13
  224. package/src/tools/WebSearchTool/searchProviders.ts +0 -66
  225. package/src/tools/lsTool/lsTool.tsx +0 -272
  226. package/src/tools/lsTool/prompt.ts +0 -2
  227. package/src/tools.ts +0 -67
  228. package/src/types/PermissionMode.ts +0 -120
  229. package/src/types/RequestContext.ts +0 -72
  230. package/src/types/common.d.ts +0 -2
  231. package/src/types/conversation.ts +0 -51
  232. package/src/types/logs.ts +0 -58
  233. package/src/types/modelCapabilities.ts +0 -64
  234. package/src/types/notebook.ts +0 -87
  235. package/src/utils/Cursor.ts +0 -436
  236. package/src/utils/PersistentShell.ts +0 -552
  237. package/src/utils/advancedFuzzyMatcher.ts +0 -290
  238. package/src/utils/agentLoader.ts +0 -278
  239. package/src/utils/agentStorage.ts +0 -97
  240. package/src/utils/array.ts +0 -3
  241. package/src/utils/ask.tsx +0 -99
  242. package/src/utils/auth.ts +0 -13
  243. package/src/utils/autoCompactCore.ts +0 -223
  244. package/src/utils/autoUpdater.ts +0 -458
  245. package/src/utils/betas.ts +0 -20
  246. package/src/utils/browser.ts +0 -14
  247. package/src/utils/cleanup.ts +0 -72
  248. package/src/utils/commands.ts +0 -261
  249. package/src/utils/commonUnixCommands.ts +0 -161
  250. package/src/utils/config.ts +0 -945
  251. package/src/utils/conversationRecovery.ts +0 -55
  252. package/src/utils/debugLogger.ts +0 -1235
  253. package/src/utils/diff.ts +0 -42
  254. package/src/utils/env.ts +0 -57
  255. package/src/utils/errors.ts +0 -21
  256. package/src/utils/exampleCommands.ts +0 -109
  257. package/src/utils/execFileNoThrow.ts +0 -51
  258. package/src/utils/expertChatStorage.ts +0 -136
  259. package/src/utils/file.ts +0 -405
  260. package/src/utils/fileRecoveryCore.ts +0 -71
  261. package/src/utils/format.tsx +0 -44
  262. package/src/utils/fuzzyMatcher.ts +0 -328
  263. package/src/utils/generators.ts +0 -62
  264. package/src/utils/git.ts +0 -92
  265. package/src/utils/globalLogger.ts +0 -77
  266. package/src/utils/http.ts +0 -10
  267. package/src/utils/imagePaste.ts +0 -38
  268. package/src/utils/json.ts +0 -13
  269. package/src/utils/log.ts +0 -382
  270. package/src/utils/markdown.ts +0 -213
  271. package/src/utils/messageContextManager.ts +0 -294
  272. package/src/utils/messages.tsx +0 -945
  273. package/src/utils/model.ts +0 -914
  274. package/src/utils/permissions/filesystem.ts +0 -127
  275. package/src/utils/responseState.ts +0 -23
  276. package/src/utils/ripgrep.ts +0 -167
  277. package/src/utils/secureFile.ts +0 -564
  278. package/src/utils/sessionState.ts +0 -49
  279. package/src/utils/state.ts +0 -25
  280. package/src/utils/style.ts +0 -29
  281. package/src/utils/terminal.ts +0 -50
  282. package/src/utils/theme.ts +0 -127
  283. package/src/utils/thinking.ts +0 -144
  284. package/src/utils/todoStorage.ts +0 -431
  285. package/src/utils/tokens.ts +0 -43
  286. package/src/utils/toolExecutionController.ts +0 -163
  287. package/src/utils/unaryLogging.ts +0 -26
  288. package/src/utils/user.ts +0 -37
  289. package/src/utils/validate.ts +0 -165
@@ -1,88 +0,0 @@
1
- import React from 'react'
2
- import { Box, Text } from 'ink'
3
- import { usePermissionContext } from '../context/PermissionContext'
4
- import { getTheme } from '../utils/theme'
5
-
6
- interface ModeIndicatorProps {
7
- showTransitionCount?: boolean
8
- }
9
-
10
- export function ModeIndicator({
11
- showTransitionCount = false,
12
- }: ModeIndicatorProps) {
13
- const { currentMode, permissionContext, getModeConfig } =
14
- usePermissionContext()
15
- const theme = getTheme()
16
- const modeConfig = getModeConfig()
17
-
18
- // Don't show indicator for default mode unless explicitly requested
19
- if (currentMode === 'default' && !showTransitionCount) {
20
- return null
21
- }
22
-
23
- return (
24
- <Box borderStyle="single" padding={1} marginY={1}>
25
- <Box flexDirection="column">
26
- <Box flexDirection="row" alignItems="center">
27
- <Text color={getThemeColor(modeConfig.color, theme)} bold>
28
- {modeConfig.icon} {modeConfig.label}
29
- </Text>
30
- </Box>
31
-
32
- <Text color="gray" dimColor>
33
- {modeConfig.description}
34
- </Text>
35
-
36
- <Box flexDirection="row" justifyContent="space-between" marginTop={1}>
37
- <Text color="gray" dimColor>
38
- Press Shift+Tab to cycle modes
39
- </Text>
40
- {showTransitionCount && (
41
- <Text color="gray" dimColor>
42
- Switches: {permissionContext.metadata.transitionCount}
43
- </Text>
44
- )}
45
- </Box>
46
-
47
- {currentMode === 'plan' && (
48
- <Box marginTop={1}>
49
- <Text color="cyan" dimColor>
50
- Available tools: {permissionContext.allowedTools.join(', ')}
51
- </Text>
52
- <Text color="yellow" dimColor>
53
- Use exit_plan_mode tool when ready to execute
54
- </Text>
55
- </Box>
56
- )}
57
- </Box>
58
- </Box>
59
- )
60
- }
61
-
62
- function getThemeColor(colorName: string, theme: any): string {
63
- const colorMap: Record<string, string> = {
64
- blue: theme.primary || 'blue',
65
- green: theme.success || 'green',
66
- yellow: theme.warning || 'yellow',
67
- red: theme.error || 'red',
68
- }
69
-
70
- return colorMap[colorName] || colorName
71
- }
72
-
73
- // Compact mode indicator for status bar
74
- export function CompactModeIndicator() {
75
- const { currentMode, getModeConfig } = usePermissionContext()
76
- const modeConfig = getModeConfig()
77
- const theme = getTheme()
78
-
79
- if (currentMode === 'default') {
80
- return null
81
- }
82
-
83
- return (
84
- <Text color={getThemeColor(modeConfig.color, theme)}>
85
- {modeConfig.icon} {modeConfig.name}
86
- </Text>
87
- )
88
- }
@@ -1,301 +0,0 @@
1
- import { Box, Text, useInput } from 'ink'
2
- import * as React from 'react'
3
- import { useState, useCallback, useEffect, useRef } from 'react'
4
- import figures from 'figures'
5
- import { getTheme } from '../utils/theme'
6
- import {
7
- getGlobalConfig,
8
- saveGlobalConfig,
9
- ModelPointerType,
10
- setModelPointer,
11
- } from '../utils/config.js'
12
- import { getModelManager } from '../utils/model'
13
- import { useExitOnCtrlCD } from '../hooks/useExitOnCtrlCD'
14
- import { ModelSelector } from './ModelSelector'
15
- import { ModelListManager } from './ModelListManager'
16
-
17
- type Props = {
18
- onClose: () => void
19
- }
20
-
21
- type ModelPointerSetting = {
22
- id: ModelPointerType | 'add-new'
23
- label: string
24
- description: string
25
- value: string
26
- options: Array<{ id: string; name: string }>
27
- type: 'modelPointer' | 'action'
28
- onChange(value?: string): void
29
- }
30
-
31
- export function ModelConfig({ onClose }: Props): React.ReactNode {
32
- const config = getGlobalConfig()
33
- const theme = getTheme()
34
- const [selectedIndex, setSelectedIndex] = useState(0)
35
- const [showModelSelector, setShowModelSelector] = useState(false)
36
- const [showModelListManager, setShowModelListManager] = useState(false)
37
- const [currentPointer, setCurrentPointer] = useState<ModelPointerType | null>(
38
- null,
39
- )
40
- const [refreshKey, setRefreshKey] = useState(0) // 添加刷新键来强制更新
41
- const [isDeleteMode, setIsDeleteMode] = useState(false) // 保留用于清空指针的删除模式
42
- const selectedIndexRef = useRef(selectedIndex) // 用ref保持焦点状态
43
- const exitState = useExitOnCtrlCD(() => process.exit(0))
44
-
45
- const modelManager = getModelManager()
46
-
47
- // 同步 selectedIndex 到 ref
48
- useEffect(() => {
49
- selectedIndexRef.current = selectedIndex
50
- }, [selectedIndex])
51
-
52
- // Get available models for cycling (memoized) - without "Add New Model" option
53
- const availableModels = React.useMemo((): Array<{
54
- id: string
55
- name: string
56
- }> => {
57
- const profiles = modelManager.getAvailableModels()
58
- return profiles.map(p => ({ id: p.modelName, name: p.name }))
59
- }, [modelManager, refreshKey]) // 依赖refreshKey来强制更新
60
-
61
- // Create menu items: model pointers + "Add New Model" as separate item
62
- const menuItems = React.useMemo(() => {
63
- const modelSettings: ModelPointerSetting[] = [
64
- {
65
- id: 'main',
66
- label: 'Main Model',
67
- description: 'Primary model for general tasks and conversations',
68
- value: config.modelPointers?.main || '',
69
- options: availableModels,
70
- type: 'modelPointer' as const,
71
- onChange: (value: string) => handleModelPointerChange('main', value),
72
- },
73
- {
74
- id: 'task',
75
- label: 'Task Model',
76
- description: 'Model for TaskTool sub-agents and automation',
77
- value: config.modelPointers?.task || '',
78
- options: availableModels,
79
- type: 'modelPointer' as const,
80
- onChange: (value: string) => handleModelPointerChange('task', value),
81
- },
82
- {
83
- id: 'reasoning',
84
- label: 'Reasoning Model',
85
- description: 'Model optimized for complex reasoning tasks',
86
- value: config.modelPointers?.reasoning || '',
87
- options: availableModels,
88
- type: 'modelPointer' as const,
89
- onChange: (value: string) =>
90
- handleModelPointerChange('reasoning', value),
91
- },
92
- {
93
- id: 'quick',
94
- label: 'Quick Model',
95
- description: 'Fast model for simple operations and utilities',
96
- value: config.modelPointers?.quick || '',
97
- options: availableModels,
98
- type: 'modelPointer' as const,
99
- onChange: (value: string) => handleModelPointerChange('quick', value),
100
- },
101
- ]
102
-
103
- // Add menu actions as separate menu items
104
- return [
105
- ...modelSettings,
106
- {
107
- id: 'manage-models',
108
- label: 'Manage Model List',
109
- description: 'View, add, and delete model configurations',
110
- value: '',
111
- options: [],
112
- type: 'action' as const,
113
- onChange: () => handleManageModels(),
114
- },
115
- ]
116
- }, [config.modelPointers, availableModels, refreshKey])
117
-
118
- const handleModelPointerChange = (
119
- pointer: ModelPointerType,
120
- modelId: string,
121
- ) => {
122
- // Direct model assignment
123
- setModelPointer(pointer, modelId)
124
- // Force re-render to show updated assignment
125
- setRefreshKey(prev => prev + 1)
126
- }
127
-
128
- const handleManageModels = () => {
129
- // Launch ModelListManager for model library management
130
- setShowModelListManager(true)
131
- }
132
-
133
- const handleModelConfigurationComplete = () => {
134
- // Model configuration is complete, return to model config screen
135
- setShowModelSelector(false)
136
- setShowModelListManager(false)
137
- setCurrentPointer(null)
138
- // 触发组件刷新,重新加载可用模型列表
139
- setRefreshKey(prev => prev + 1)
140
- // 将焦点重置到 "Manage Model Library" 选项
141
- const manageIndex = menuItems.findIndex(item => item.id === 'manage-models')
142
- if (manageIndex !== -1) {
143
- setSelectedIndex(manageIndex)
144
- }
145
- }
146
-
147
- // Handle keyboard input - completely following Config component pattern
148
- const handleInput = useCallback(
149
- (input: string, key: any) => {
150
- if (key.escape) {
151
- if (isDeleteMode) {
152
- setIsDeleteMode(false) // Exit delete mode
153
- } else {
154
- onClose()
155
- }
156
- } else if (input === 'd' && !isDeleteMode) {
157
- setIsDeleteMode(true) // Enter delete mode
158
- } else if (key.upArrow) {
159
- setSelectedIndex(prev => Math.max(0, prev - 1))
160
- } else if (key.downArrow) {
161
- setSelectedIndex(prev => Math.min(menuItems.length - 1, prev + 1))
162
- } else if (key.return || input === ' ') {
163
- const setting = menuItems[selectedIndex]
164
-
165
- if (isDeleteMode && setting.type === 'modelPointer' && setting.value) {
166
- // Delete mode: clear the pointer assignment (not delete the model config)
167
- setModelPointer(setting.id as ModelPointerType, '')
168
- setRefreshKey(prev => prev + 1)
169
- setIsDeleteMode(false) // Exit delete mode after clearing assignment
170
- } else if (setting.type === 'modelPointer') {
171
- // Normal mode: cycle through available models
172
- if (setting.options.length === 0) {
173
- // No models available, redirect to model library management
174
- handleManageModels()
175
- return
176
- }
177
- const currentIndex = setting.options.findIndex(
178
- opt => opt.id === setting.value,
179
- )
180
- const nextIndex = (currentIndex + 1) % setting.options.length
181
- const nextOption = setting.options[nextIndex]
182
- if (nextOption) {
183
- setting.onChange(nextOption.id)
184
- }
185
- } else if (setting.type === 'action') {
186
- // Execute action (like "Add New Model")
187
- setting.onChange()
188
- }
189
- }
190
- },
191
- [selectedIndex, menuItems, onClose, isDeleteMode, modelManager],
192
- )
193
-
194
- useInput(handleInput)
195
-
196
- // If showing ModelListManager, render it directly
197
- if (showModelListManager) {
198
- return <ModelListManager onClose={handleModelConfigurationComplete} />
199
- }
200
-
201
- // If showing ModelSelector, render it directly
202
- if (showModelSelector) {
203
- return (
204
- <ModelSelector
205
- onDone={handleModelConfigurationComplete}
206
- onCancel={handleModelConfigurationComplete} // Same as onDone - return to ModelConfig
207
- skipModelType={true}
208
- targetPointer={currentPointer || undefined}
209
- isOnboarding={false}
210
- abortController={new AbortController()}
211
- />
212
- )
213
- }
214
-
215
- // Main configuration screen - completely following Config component layout
216
- return (
217
- <Box
218
- flexDirection="column"
219
- borderStyle="round"
220
- borderColor={theme.secondaryBorder}
221
- paddingX={1}
222
- marginTop={1}
223
- >
224
- <Box flexDirection="column" minHeight={2} marginBottom={1}>
225
- <Text bold>
226
- Model Configuration{isDeleteMode ? ' - CLEAR MODE' : ''}
227
- </Text>
228
- <Text dimColor>
229
- {isDeleteMode
230
- ? 'Press Enter/Space to clear selected pointer assignment, Esc to cancel'
231
- : availableModels.length === 0
232
- ? 'No models configured. Use "Configure New Model" to add your first model.'
233
- : 'Configure which models to use for different tasks. Space to cycle, Enter to configure.'}
234
- </Text>
235
- </Box>
236
-
237
- {menuItems.map((setting, i) => {
238
- const isSelected = i === selectedIndex
239
- let displayValue = ''
240
- let actionText = ''
241
-
242
- if (setting.type === 'modelPointer') {
243
- const currentModel = setting.options.find(
244
- opt => opt.id === setting.value,
245
- )
246
- displayValue = currentModel?.name || '(not configured)'
247
- actionText = isSelected ? ' [Space to cycle]' : ''
248
- } else if (setting.type === 'action') {
249
- displayValue = ''
250
- actionText = isSelected ? ' [Enter to configure]' : ''
251
- }
252
-
253
- return (
254
- <Box key={setting.id} flexDirection="column">
255
- <Box>
256
- <Box width={44}>
257
- <Text color={isSelected ? 'blue' : undefined}>
258
- {isSelected ? figures.pointer : ' '} {setting.label}
259
- </Text>
260
- </Box>
261
- <Box>
262
- {setting.type === 'modelPointer' && (
263
- <Text
264
- color={
265
- displayValue !== '(not configured)'
266
- ? theme.success
267
- : theme.warning
268
- }
269
- >
270
- {displayValue}
271
- </Text>
272
- )}
273
- {actionText && <Text color="blue">{actionText}</Text>}
274
- </Box>
275
- </Box>
276
- {isSelected && (
277
- <Box paddingLeft={2} marginBottom={1}>
278
- <Text dimColor>{setting.description}</Text>
279
- </Box>
280
- )}
281
- </Box>
282
- )
283
- })}
284
-
285
- <Box
286
- marginTop={1}
287
- paddingTop={1}
288
- borderTopColor={theme.secondaryBorder}
289
- borderTopStyle="single"
290
- >
291
- <Text dimColor>
292
- {isDeleteMode
293
- ? 'CLEAR MODE: Press Enter/Space to clear assignment, Esc to cancel'
294
- : availableModels.length === 0
295
- ? 'Use ↑/↓ to navigate, Enter to configure new model, Esc to exit'
296
- : 'Use ↑/↓ to navigate, Space to cycle models, Enter to configure, d to clear, Esc to exit'}
297
- </Text>
298
- </Box>
299
- </Box>
300
- )
301
- }
@@ -1,227 +0,0 @@
1
- import { Box, Text, useInput } from 'ink'
2
- import * as React from 'react'
3
- import { useState, useCallback } from 'react'
4
- import figures from 'figures'
5
- import { getTheme } from '../utils/theme'
6
- import { getGlobalConfig, ModelPointerType } from '../utils/config.js'
7
- import { getModelManager } from '../utils/model'
8
- import { useExitOnCtrlCD } from '../hooks/useExitOnCtrlCD'
9
- import { ModelSelector } from './ModelSelector'
10
-
11
- type Props = {
12
- onClose: () => void
13
- }
14
-
15
- export function ModelListManager({ onClose }: Props): React.ReactNode {
16
- const config = getGlobalConfig()
17
- const theme = getTheme()
18
- const [selectedIndex, setSelectedIndex] = useState(0)
19
- const [showModelSelector, setShowModelSelector] = useState(false)
20
- const [isDeleteMode, setIsDeleteMode] = useState(false)
21
- const [refreshKey, setRefreshKey] = useState(0)
22
- const exitState = useExitOnCtrlCD(onClose)
23
-
24
- const modelManager = getModelManager()
25
- const availableModels = modelManager.getAvailableModels()
26
-
27
- // Create menu items: existing models + "Add New Model"
28
- const menuItems = React.useMemo(() => {
29
- const modelItems = availableModels.map(model => ({
30
- id: model.modelName,
31
- name: model.name,
32
- provider: model.provider,
33
- usedBy: getModelUsage(model.modelName),
34
- type: 'model' as const,
35
- }))
36
-
37
- return [
38
- ...modelItems,
39
- {
40
- id: 'add-new',
41
- name: '+ Add New Model',
42
- provider: '',
43
- usedBy: [],
44
- type: 'action' as const,
45
- },
46
- ]
47
- }, [availableModels, config.modelPointers, refreshKey])
48
-
49
- // Check which pointers are using this model
50
- function getModelUsage(modelName: string): ModelPointerType[] {
51
- const usage: ModelPointerType[] = []
52
- const pointers: ModelPointerType[] = ['main', 'task', 'reasoning', 'quick']
53
-
54
- pointers.forEach(pointer => {
55
- if (config.modelPointers?.[pointer] === modelName) {
56
- usage.push(pointer)
57
- }
58
- })
59
-
60
- return usage
61
- }
62
-
63
- const handleDeleteModel = (modelName: string) => {
64
- // Remove the model
65
- modelManager.removeModel(modelName)
66
-
67
- // The removeModel function should already clear the pointers,
68
- // but let's ensure UI refreshes
69
- setRefreshKey(prev => prev + 1)
70
- setIsDeleteMode(false)
71
- }
72
-
73
- const handleAddNewModel = () => {
74
- setShowModelSelector(true)
75
- }
76
-
77
- const handleModelConfigurationComplete = () => {
78
- setShowModelSelector(false)
79
- setRefreshKey(prev => prev + 1)
80
- }
81
-
82
- // Handle keyboard input
83
- const handleInput = useCallback(
84
- (input: string, key: any) => {
85
- if (key.escape) {
86
- if (isDeleteMode) {
87
- setIsDeleteMode(false)
88
- } else {
89
- onClose()
90
- }
91
- } else if (input === 'd' && !isDeleteMode && availableModels.length > 1) {
92
- setIsDeleteMode(true)
93
- } else if (key.upArrow) {
94
- setSelectedIndex(prev => Math.max(0, prev - 1))
95
- } else if (key.downArrow) {
96
- setSelectedIndex(prev => Math.min(menuItems.length - 1, prev + 1))
97
- } else if (key.return || input === ' ') {
98
- const item = menuItems[selectedIndex]
99
-
100
- if (isDeleteMode && item.type === 'model') {
101
- // Prevent deleting the last model
102
- if (availableModels.length <= 1) {
103
- setIsDeleteMode(false) // Exit delete mode
104
- return
105
- }
106
- handleDeleteModel(item.id)
107
- } else if (item.type === 'action') {
108
- handleAddNewModel()
109
- }
110
- // Note: Remove any pointer switching functionality here
111
- }
112
- },
113
- [selectedIndex, menuItems, onClose, isDeleteMode, availableModels.length],
114
- )
115
-
116
- useInput(handleInput)
117
-
118
- // If showing ModelSelector, render it directly
119
- if (showModelSelector) {
120
- return (
121
- <ModelSelector
122
- onDone={handleModelConfigurationComplete}
123
- onCancel={handleModelConfigurationComplete}
124
- skipModelType={true}
125
- isOnboarding={false}
126
- abortController={new AbortController()}
127
- />
128
- )
129
- }
130
-
131
- // Main model list screen
132
- return (
133
- <Box
134
- flexDirection="column"
135
- borderStyle="round"
136
- borderColor={isDeleteMode ? 'red' : theme.secondaryBorder}
137
- paddingX={1}
138
- marginTop={1}
139
- >
140
- <Box flexDirection="column" minHeight={2} marginBottom={1}>
141
- <Text bold color={isDeleteMode ? 'red' : undefined}>
142
- Manage Model List{isDeleteMode ? ' - DELETE MODE' : ''}
143
- {exitState.pending
144
- ? ` (press ${exitState.keyName} again to exit)`
145
- : ''}
146
- </Text>
147
- <Text dimColor>
148
- {isDeleteMode
149
- ? availableModels.length <= 1
150
- ? 'Cannot delete the last model, Esc to cancel'
151
- : 'Press Enter/Space to DELETE selected model, Esc to cancel'
152
- : 'Navigate: ↑↓ | Select: Enter | Delete: d | Exit: Esc'}
153
- </Text>
154
- </Box>
155
-
156
- {menuItems.map((item, i) => {
157
- const isSelected = i === selectedIndex
158
-
159
- return (
160
- <Box key={item.id} flexDirection="column" marginBottom={1}>
161
- <Box>
162
- <Box width={50}>
163
- <Text
164
- color={
165
- isSelected ? (isDeleteMode ? 'red' : 'blue') : undefined
166
- }
167
- >
168
- {isSelected ? figures.pointer : ' '} {item.name}
169
- </Text>
170
- </Box>
171
- <Box>
172
- {item.type === 'model' && (
173
- <>
174
- <Text color={theme.secondaryText}>({item.provider})</Text>
175
- {item.usedBy.length > 0 && (
176
- <Box marginLeft={1}>
177
- <Text color={theme.success}>
178
- [Active: {item.usedBy.join(', ')}]
179
- </Text>
180
- </Box>
181
- )}
182
- {item.usedBy.length === 0 && (
183
- <Box marginLeft={1}>
184
- <Text color={theme.secondaryText}>
185
- [Available]
186
- </Text>
187
- </Box>
188
- )}
189
- </>
190
- )}
191
- {item.type === 'action' && (
192
- <Text color={theme.suggestion}>
193
- {isSelected ? '[Press Enter to add new model]' : ''}
194
- </Text>
195
- )}
196
- </Box>
197
- </Box>
198
- {isSelected && item.type === 'action' && (
199
- <Box paddingLeft={2} marginTop={1}>
200
- <Text dimColor>
201
- Configure a new model and add it to your library
202
- </Text>
203
- </Box>
204
- )}
205
- </Box>
206
- )
207
- })}
208
-
209
- <Box
210
- marginTop={1}
211
- paddingTop={1}
212
- borderTopColor={theme.secondaryBorder}
213
- borderTopStyle="single"
214
- >
215
- <Text dimColor>
216
- {isDeleteMode
217
- ? availableModels.length <= 1
218
- ? 'Cannot delete the last model - press Esc to cancel'
219
- : 'DELETE MODE: Press Enter/Space to delete model, Esc to cancel'
220
- : availableModels.length <= 1
221
- ? 'Use ↑/↓ to navigate, Enter to add new, Esc to exit (cannot delete last model)'
222
- : 'Use ↑/↓ to navigate, d to delete model, Enter to add new, Esc to exit'}
223
- </Text>
224
- </Box>
225
- </Box>
226
- )
227
- }