@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.
- package/cli.js +77 -82
- package/dist/entrypoints/cli.js +59 -38
- package/dist/entrypoints/cli.js.map +3 -3
- package/dist/index.js +5 -26
- package/dist/package.json +4 -1
- package/package.json +11 -104
- package/dist/test/testAdapters.js +0 -88
- package/dist/test/testAdapters.js.map +0 -1
- package/src/ProjectOnboarding.tsx +0 -198
- package/src/Tool.ts +0 -83
- package/src/commands/agents.tsx +0 -3416
- package/src/commands/approvedTools.ts +0 -53
- package/src/commands/bug.tsx +0 -20
- package/src/commands/clear.ts +0 -43
- package/src/commands/compact.ts +0 -120
- package/src/commands/config.tsx +0 -19
- package/src/commands/cost.ts +0 -18
- package/src/commands/ctx_viz.ts +0 -209
- package/src/commands/doctor.ts +0 -24
- package/src/commands/help.tsx +0 -19
- package/src/commands/init.ts +0 -37
- package/src/commands/listen.ts +0 -42
- package/src/commands/login.tsx +0 -51
- package/src/commands/logout.tsx +0 -40
- package/src/commands/mcp.ts +0 -41
- package/src/commands/model.tsx +0 -40
- package/src/commands/modelstatus.tsx +0 -20
- package/src/commands/onboarding.tsx +0 -34
- package/src/commands/pr_comments.ts +0 -59
- package/src/commands/refreshCommands.ts +0 -54
- package/src/commands/release-notes.ts +0 -34
- package/src/commands/resume.tsx +0 -31
- package/src/commands/review.ts +0 -49
- package/src/commands/terminalSetup.ts +0 -221
- package/src/commands.ts +0 -139
- package/src/components/ApproveApiKey.tsx +0 -93
- package/src/components/AsciiLogo.tsx +0 -13
- package/src/components/AutoUpdater.tsx +0 -148
- package/src/components/Bug.tsx +0 -367
- package/src/components/Config.tsx +0 -293
- package/src/components/ConsoleOAuthFlow.tsx +0 -327
- package/src/components/Cost.tsx +0 -23
- package/src/components/CostThresholdDialog.tsx +0 -46
- package/src/components/CustomSelect/option-map.ts +0 -42
- package/src/components/CustomSelect/select-option.tsx +0 -78
- package/src/components/CustomSelect/select.tsx +0 -152
- package/src/components/CustomSelect/theme.ts +0 -45
- package/src/components/CustomSelect/use-select-state.ts +0 -414
- package/src/components/CustomSelect/use-select.ts +0 -35
- package/src/components/FallbackToolUseRejectedMessage.tsx +0 -15
- package/src/components/FileEditToolUpdatedMessage.tsx +0 -66
- package/src/components/Help.tsx +0 -215
- package/src/components/HighlightedCode.tsx +0 -33
- package/src/components/InvalidConfigDialog.tsx +0 -113
- package/src/components/Link.tsx +0 -32
- package/src/components/LogSelector.tsx +0 -86
- package/src/components/Logo.tsx +0 -170
- package/src/components/MCPServerApprovalDialog.tsx +0 -100
- package/src/components/MCPServerDialogCopy.tsx +0 -25
- package/src/components/MCPServerMultiselectDialog.tsx +0 -109
- package/src/components/Message.tsx +0 -221
- package/src/components/MessageResponse.tsx +0 -15
- package/src/components/MessageSelector.tsx +0 -211
- package/src/components/ModeIndicator.tsx +0 -88
- package/src/components/ModelConfig.tsx +0 -301
- package/src/components/ModelListManager.tsx +0 -227
- package/src/components/ModelSelector.tsx +0 -3387
- package/src/components/ModelStatusDisplay.tsx +0 -230
- package/src/components/Onboarding.tsx +0 -274
- package/src/components/PressEnterToContinue.tsx +0 -11
- package/src/components/PromptInput.tsx +0 -760
- package/src/components/SentryErrorBoundary.ts +0 -39
- package/src/components/Spinner.tsx +0 -129
- package/src/components/StickerRequestForm.tsx +0 -16
- package/src/components/StructuredDiff.tsx +0 -191
- package/src/components/TextInput.tsx +0 -259
- package/src/components/TodoItem.tsx +0 -47
- package/src/components/TokenWarning.tsx +0 -31
- package/src/components/ToolUseLoader.tsx +0 -40
- package/src/components/TrustDialog.tsx +0 -106
- package/src/components/binary-feedback/BinaryFeedback.tsx +0 -63
- package/src/components/binary-feedback/BinaryFeedbackOption.tsx +0 -111
- package/src/components/binary-feedback/BinaryFeedbackView.tsx +0 -172
- package/src/components/binary-feedback/utils.ts +0 -220
- package/src/components/messages/AssistantBashOutputMessage.tsx +0 -22
- package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +0 -49
- package/src/components/messages/AssistantRedactedThinkingMessage.tsx +0 -19
- package/src/components/messages/AssistantTextMessage.tsx +0 -144
- package/src/components/messages/AssistantThinkingMessage.tsx +0 -40
- package/src/components/messages/AssistantToolUseMessage.tsx +0 -132
- package/src/components/messages/TaskProgressMessage.tsx +0 -32
- package/src/components/messages/TaskToolMessage.tsx +0 -58
- package/src/components/messages/UserBashInputMessage.tsx +0 -28
- package/src/components/messages/UserCommandMessage.tsx +0 -30
- package/src/components/messages/UserKodingInputMessage.tsx +0 -28
- package/src/components/messages/UserPromptMessage.tsx +0 -35
- package/src/components/messages/UserTextMessage.tsx +0 -39
- package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +0 -12
- package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +0 -36
- package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +0 -31
- package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +0 -57
- package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +0 -35
- package/src/components/messages/UserToolResultMessage/utils.tsx +0 -56
- package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +0 -121
- package/src/components/permissions/FallbackPermissionRequest.tsx +0 -153
- package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +0 -182
- package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +0 -77
- package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +0 -164
- package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +0 -83
- package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +0 -240
- package/src/components/permissions/PermissionRequest.tsx +0 -101
- package/src/components/permissions/PermissionRequestTitle.tsx +0 -69
- package/src/components/permissions/hooks.ts +0 -44
- package/src/components/permissions/toolUseOptions.ts +0 -59
- package/src/components/permissions/utils.ts +0 -23
- package/src/constants/betas.ts +0 -5
- package/src/constants/claude-asterisk-ascii-art.tsx +0 -238
- package/src/constants/figures.ts +0 -4
- package/src/constants/keys.ts +0 -3
- package/src/constants/macros.ts +0 -11
- package/src/constants/modelCapabilities.ts +0 -179
- package/src/constants/models.ts +0 -1025
- package/src/constants/oauth.ts +0 -18
- package/src/constants/product.ts +0 -17
- package/src/constants/prompts.ts +0 -168
- package/src/constants/releaseNotes.ts +0 -7
- package/src/context/PermissionContext.tsx +0 -149
- package/src/context.ts +0 -278
- package/src/cost-tracker.ts +0 -84
- package/src/entrypoints/cli.tsx +0 -1561
- package/src/entrypoints/mcp.ts +0 -175
- package/src/history.ts +0 -25
- package/src/hooks/useApiKeyVerification.ts +0 -59
- package/src/hooks/useArrowKeyHistory.ts +0 -55
- package/src/hooks/useCanUseTool.ts +0 -138
- package/src/hooks/useCancelRequest.ts +0 -39
- package/src/hooks/useDoublePress.ts +0 -41
- package/src/hooks/useExitOnCtrlCD.ts +0 -31
- package/src/hooks/useInterval.ts +0 -25
- package/src/hooks/useLogMessages.ts +0 -16
- package/src/hooks/useLogStartupTime.ts +0 -12
- package/src/hooks/useNotifyAfterTimeout.ts +0 -65
- package/src/hooks/usePermissionRequestLogging.ts +0 -44
- package/src/hooks/useTerminalSize.ts +0 -49
- package/src/hooks/useTextInput.ts +0 -317
- package/src/hooks/useUnifiedCompletion.ts +0 -1405
- package/src/index.ts +0 -34
- package/src/messages.ts +0 -38
- package/src/permissions.ts +0 -268
- package/src/query.ts +0 -720
- package/src/screens/ConfigureNpmPrefix.tsx +0 -197
- package/src/screens/Doctor.tsx +0 -219
- package/src/screens/LogList.tsx +0 -68
- package/src/screens/REPL.tsx +0 -813
- package/src/screens/ResumeConversation.tsx +0 -68
- package/src/services/adapters/base.ts +0 -38
- package/src/services/adapters/chatCompletions.ts +0 -90
- package/src/services/adapters/responsesAPI.ts +0 -170
- package/src/services/browserMocks.ts +0 -66
- package/src/services/claude.ts +0 -2197
- package/src/services/customCommands.ts +0 -704
- package/src/services/fileFreshness.ts +0 -377
- package/src/services/gpt5ConnectionTest.ts +0 -340
- package/src/services/mcpClient.ts +0 -564
- package/src/services/mcpServerApproval.tsx +0 -50
- package/src/services/mentionProcessor.ts +0 -273
- package/src/services/modelAdapterFactory.ts +0 -69
- package/src/services/notifier.ts +0 -40
- package/src/services/oauth.ts +0 -357
- package/src/services/openai.ts +0 -1359
- package/src/services/responseStateManager.ts +0 -90
- package/src/services/sentry.ts +0 -3
- package/src/services/statsig.ts +0 -172
- package/src/services/statsigStorage.ts +0 -86
- package/src/services/systemReminder.ts +0 -507
- package/src/services/vcr.ts +0 -161
- package/src/test/testAdapters.ts +0 -96
- package/src/tools/ArchitectTool/ArchitectTool.tsx +0 -135
- package/src/tools/ArchitectTool/prompt.ts +0 -15
- package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +0 -576
- package/src/tools/BashTool/BashTool.tsx +0 -243
- package/src/tools/BashTool/BashToolResultMessage.tsx +0 -38
- package/src/tools/BashTool/OutputLine.tsx +0 -49
- package/src/tools/BashTool/prompt.ts +0 -174
- package/src/tools/BashTool/utils.ts +0 -56
- package/src/tools/FileEditTool/FileEditTool.tsx +0 -319
- package/src/tools/FileEditTool/prompt.ts +0 -51
- package/src/tools/FileEditTool/utils.ts +0 -58
- package/src/tools/FileReadTool/FileReadTool.tsx +0 -404
- package/src/tools/FileReadTool/prompt.ts +0 -7
- package/src/tools/FileWriteTool/FileWriteTool.tsx +0 -301
- package/src/tools/FileWriteTool/prompt.ts +0 -10
- package/src/tools/GlobTool/GlobTool.tsx +0 -119
- package/src/tools/GlobTool/prompt.ts +0 -8
- package/src/tools/GrepTool/GrepTool.tsx +0 -147
- package/src/tools/GrepTool/prompt.ts +0 -11
- package/src/tools/MCPTool/MCPTool.tsx +0 -107
- package/src/tools/MCPTool/prompt.ts +0 -3
- package/src/tools/MemoryReadTool/MemoryReadTool.tsx +0 -127
- package/src/tools/MemoryReadTool/prompt.ts +0 -3
- package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +0 -89
- package/src/tools/MemoryWriteTool/prompt.ts +0 -3
- package/src/tools/MultiEditTool/MultiEditTool.tsx +0 -388
- package/src/tools/MultiEditTool/prompt.ts +0 -45
- package/src/tools/NotebookEditTool/NotebookEditTool.tsx +0 -298
- package/src/tools/NotebookEditTool/prompt.ts +0 -3
- package/src/tools/NotebookReadTool/NotebookReadTool.tsx +0 -258
- package/src/tools/NotebookReadTool/prompt.ts +0 -3
- package/src/tools/StickerRequestTool/StickerRequestTool.tsx +0 -107
- package/src/tools/StickerRequestTool/prompt.ts +0 -19
- package/src/tools/TaskTool/TaskTool.tsx +0 -438
- package/src/tools/TaskTool/constants.ts +0 -1
- package/src/tools/TaskTool/prompt.ts +0 -92
- package/src/tools/ThinkTool/ThinkTool.tsx +0 -54
- package/src/tools/ThinkTool/prompt.ts +0 -12
- package/src/tools/TodoWriteTool/TodoWriteTool.tsx +0 -313
- package/src/tools/TodoWriteTool/prompt.ts +0 -63
- package/src/tools/URLFetcherTool/URLFetcherTool.tsx +0 -178
- package/src/tools/URLFetcherTool/cache.ts +0 -55
- package/src/tools/URLFetcherTool/htmlToMarkdown.ts +0 -55
- package/src/tools/URLFetcherTool/prompt.ts +0 -17
- package/src/tools/WebSearchTool/WebSearchTool.tsx +0 -103
- package/src/tools/WebSearchTool/prompt.ts +0 -13
- package/src/tools/WebSearchTool/searchProviders.ts +0 -66
- package/src/tools/lsTool/lsTool.tsx +0 -272
- package/src/tools/lsTool/prompt.ts +0 -2
- package/src/tools.ts +0 -67
- package/src/types/PermissionMode.ts +0 -120
- package/src/types/RequestContext.ts +0 -72
- package/src/types/common.d.ts +0 -2
- package/src/types/conversation.ts +0 -51
- package/src/types/logs.ts +0 -58
- package/src/types/modelCapabilities.ts +0 -64
- package/src/types/notebook.ts +0 -87
- package/src/utils/Cursor.ts +0 -436
- package/src/utils/PersistentShell.ts +0 -552
- package/src/utils/advancedFuzzyMatcher.ts +0 -290
- package/src/utils/agentLoader.ts +0 -278
- package/src/utils/agentStorage.ts +0 -97
- package/src/utils/array.ts +0 -3
- package/src/utils/ask.tsx +0 -99
- package/src/utils/auth.ts +0 -13
- package/src/utils/autoCompactCore.ts +0 -223
- package/src/utils/autoUpdater.ts +0 -458
- package/src/utils/betas.ts +0 -20
- package/src/utils/browser.ts +0 -14
- package/src/utils/cleanup.ts +0 -72
- package/src/utils/commands.ts +0 -261
- package/src/utils/commonUnixCommands.ts +0 -161
- package/src/utils/config.ts +0 -945
- package/src/utils/conversationRecovery.ts +0 -55
- package/src/utils/debugLogger.ts +0 -1235
- package/src/utils/diff.ts +0 -42
- package/src/utils/env.ts +0 -57
- package/src/utils/errors.ts +0 -21
- package/src/utils/exampleCommands.ts +0 -109
- package/src/utils/execFileNoThrow.ts +0 -51
- package/src/utils/expertChatStorage.ts +0 -136
- package/src/utils/file.ts +0 -405
- package/src/utils/fileRecoveryCore.ts +0 -71
- package/src/utils/format.tsx +0 -44
- package/src/utils/fuzzyMatcher.ts +0 -328
- package/src/utils/generators.ts +0 -62
- package/src/utils/git.ts +0 -92
- package/src/utils/globalLogger.ts +0 -77
- package/src/utils/http.ts +0 -10
- package/src/utils/imagePaste.ts +0 -38
- package/src/utils/json.ts +0 -13
- package/src/utils/log.ts +0 -382
- package/src/utils/markdown.ts +0 -213
- package/src/utils/messageContextManager.ts +0 -294
- package/src/utils/messages.tsx +0 -945
- package/src/utils/model.ts +0 -914
- package/src/utils/permissions/filesystem.ts +0 -127
- package/src/utils/responseState.ts +0 -23
- package/src/utils/ripgrep.ts +0 -167
- package/src/utils/secureFile.ts +0 -564
- package/src/utils/sessionState.ts +0 -49
- package/src/utils/state.ts +0 -25
- package/src/utils/style.ts +0 -29
- package/src/utils/terminal.ts +0 -50
- package/src/utils/theme.ts +0 -127
- package/src/utils/thinking.ts +0 -144
- package/src/utils/todoStorage.ts +0 -431
- package/src/utils/tokens.ts +0 -43
- package/src/utils/toolExecutionController.ts +0 -163
- package/src/utils/unaryLogging.ts +0 -26
- package/src/utils/user.ts +0 -37
- 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
|
-
}
|