snow-ai 0.4.16 → 0.4.18
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/bundle/cli.mjs +477445 -0
- package/bundle/sql-wasm.wasm +0 -0
- package/bundle/tiktoken_bg.wasm +0 -0
- package/package.json +31 -26
- package/dist/agents/codebaseIndexAgent.d.ts +0 -102
- package/dist/agents/codebaseIndexAgent.js +0 -641
- package/dist/agents/codebaseReviewAgent.d.ts +0 -61
- package/dist/agents/codebaseReviewAgent.js +0 -301
- package/dist/agents/compactAgent.d.ts +0 -55
- package/dist/agents/compactAgent.js +0 -306
- package/dist/agents/promptOptimizeAgent.d.ts +0 -54
- package/dist/agents/promptOptimizeAgent.js +0 -268
- package/dist/agents/reviewAgent.d.ts +0 -50
- package/dist/agents/reviewAgent.js +0 -265
- package/dist/agents/summaryAgent.d.ts +0 -57
- package/dist/agents/summaryAgent.js +0 -260
- package/dist/api/anthropic.d.ts +0 -44
- package/dist/api/anthropic.js +0 -598
- package/dist/api/chat.d.ts +0 -73
- package/dist/api/chat.js +0 -386
- package/dist/api/embedding.d.ts +0 -34
- package/dist/api/embedding.js +0 -80
- package/dist/api/gemini.d.ts +0 -31
- package/dist/api/gemini.js +0 -445
- package/dist/api/models.d.ts +0 -15
- package/dist/api/models.js +0 -139
- package/dist/api/responses.d.ts +0 -38
- package/dist/api/responses.js +0 -515
- package/dist/api/systemPrompt.d.ts +0 -4
- package/dist/api/systemPrompt.js +0 -408
- package/dist/api/types.d.ts +0 -53
- package/dist/api/types.js +0 -4
- package/dist/app.d.ts +0 -8
- package/dist/app.js +0 -112
- package/dist/cli.d.ts +0 -2
- package/dist/cli.js +0 -199
- package/dist/hooks/useAgentPicker.d.ts +0 -14
- package/dist/hooks/useAgentPicker.js +0 -119
- package/dist/hooks/useClipboard.d.ts +0 -4
- package/dist/hooks/useClipboard.js +0 -175
- package/dist/hooks/useCommandHandler.d.ts +0 -35
- package/dist/hooks/useCommandHandler.js +0 -346
- package/dist/hooks/useCommandPanel.d.ts +0 -17
- package/dist/hooks/useCommandPanel.js +0 -114
- package/dist/hooks/useConversation.d.ts +0 -49
- package/dist/hooks/useConversation.js +0 -1052
- package/dist/hooks/useFilePicker.d.ts +0 -18
- package/dist/hooks/useFilePicker.js +0 -224
- package/dist/hooks/useGlobalExit.d.ts +0 -5
- package/dist/hooks/useGlobalExit.js +0 -34
- package/dist/hooks/useGlobalNavigation.d.ts +0 -6
- package/dist/hooks/useGlobalNavigation.js +0 -17
- package/dist/hooks/useHistoryNavigation.d.ts +0 -35
- package/dist/hooks/useHistoryNavigation.js +0 -133
- package/dist/hooks/useInputBuffer.d.ts +0 -6
- package/dist/hooks/useInputBuffer.js +0 -45
- package/dist/hooks/useKeyboardInput.d.ts +0 -80
- package/dist/hooks/useKeyboardInput.js +0 -608
- package/dist/hooks/useSessionManagement.d.ts +0 -10
- package/dist/hooks/useSessionManagement.js +0 -43
- package/dist/hooks/useSessionSave.d.ts +0 -8
- package/dist/hooks/useSessionSave.js +0 -63
- package/dist/hooks/useSnapshotState.d.ts +0 -26
- package/dist/hooks/useSnapshotState.js +0 -28
- package/dist/hooks/useStreamingState.d.ts +0 -33
- package/dist/hooks/useStreamingState.js +0 -105
- package/dist/hooks/useTerminalFocus.d.ts +0 -28
- package/dist/hooks/useTerminalFocus.js +0 -87
- package/dist/hooks/useTerminalSize.d.ts +0 -4
- package/dist/hooks/useTerminalSize.js +0 -20
- package/dist/hooks/useTodoPicker.d.ts +0 -16
- package/dist/hooks/useTodoPicker.js +0 -94
- package/dist/hooks/useToolConfirmation.d.ts +0 -19
- package/dist/hooks/useToolConfirmation.js +0 -61
- package/dist/hooks/useVSCodeState.d.ts +0 -8
- package/dist/hooks/useVSCodeState.js +0 -81
- package/dist/i18n/I18nContext.d.ts +0 -14
- package/dist/i18n/I18nContext.js +0 -24
- package/dist/i18n/index.d.ts +0 -3
- package/dist/i18n/index.js +0 -2
- package/dist/i18n/lang/en.d.ts +0 -2
- package/dist/i18n/lang/en.js +0 -502
- package/dist/i18n/lang/es.d.ts +0 -2
- package/dist/i18n/lang/es.js +0 -502
- package/dist/i18n/lang/ja.d.ts +0 -2
- package/dist/i18n/lang/ja.js +0 -502
- package/dist/i18n/lang/ko.d.ts +0 -2
- package/dist/i18n/lang/ko.js +0 -502
- package/dist/i18n/lang/zh-TW.d.ts +0 -2
- package/dist/i18n/lang/zh-TW.js +0 -502
- package/dist/i18n/lang/zh.d.ts +0 -2
- package/dist/i18n/lang/zh.js +0 -502
- package/dist/i18n/translations.d.ts +0 -2
- package/dist/i18n/translations.js +0 -14
- package/dist/i18n/types.d.ts +0 -478
- package/dist/i18n/types.js +0 -1
- package/dist/mcp/aceCodeSearch.d.ts +0 -247
- package/dist/mcp/aceCodeSearch.js +0 -1058
- package/dist/mcp/bash.d.ts +0 -50
- package/dist/mcp/bash.js +0 -153
- package/dist/mcp/codebaseSearch.d.ts +0 -44
- package/dist/mcp/codebaseSearch.js +0 -275
- package/dist/mcp/filesystem.d.ts +0 -392
- package/dist/mcp/filesystem.js +0 -1445
- package/dist/mcp/ideDiagnostics.d.ts +0 -36
- package/dist/mcp/ideDiagnostics.js +0 -90
- package/dist/mcp/notebook.d.ts +0 -10
- package/dist/mcp/notebook.js +0 -367
- package/dist/mcp/subagent.d.ts +0 -37
- package/dist/mcp/subagent.js +0 -113
- package/dist/mcp/todo.d.ts +0 -46
- package/dist/mcp/todo.js +0 -511
- package/dist/mcp/types/aceCodeSearch.types.d.ts +0 -92
- package/dist/mcp/types/aceCodeSearch.types.js +0 -4
- package/dist/mcp/types/bash.types.d.ts +0 -13
- package/dist/mcp/types/bash.types.js +0 -4
- package/dist/mcp/types/filesystem.types.d.ts +0 -210
- package/dist/mcp/types/filesystem.types.js +0 -27
- package/dist/mcp/types/todo.types.d.ts +0 -27
- package/dist/mcp/types/todo.types.js +0 -4
- package/dist/mcp/types/websearch.types.d.ts +0 -30
- package/dist/mcp/types/websearch.types.js +0 -4
- package/dist/mcp/utils/aceCodeSearch/filesystem.utils.d.ts +0 -34
- package/dist/mcp/utils/aceCodeSearch/filesystem.utils.js +0 -146
- package/dist/mcp/utils/aceCodeSearch/language.utils.d.ts +0 -14
- package/dist/mcp/utils/aceCodeSearch/language.utils.js +0 -418
- package/dist/mcp/utils/aceCodeSearch/search.utils.d.ts +0 -31
- package/dist/mcp/utils/aceCodeSearch/search.utils.js +0 -136
- package/dist/mcp/utils/aceCodeSearch/symbol.utils.d.ts +0 -20
- package/dist/mcp/utils/aceCodeSearch/symbol.utils.js +0 -141
- package/dist/mcp/utils/bash/security.utils.d.ts +0 -20
- package/dist/mcp/utils/bash/security.utils.js +0 -34
- package/dist/mcp/utils/filesystem/batch-operations.utils.d.ts +0 -39
- package/dist/mcp/utils/filesystem/batch-operations.utils.js +0 -182
- package/dist/mcp/utils/filesystem/code-analysis.utils.d.ts +0 -18
- package/dist/mcp/utils/filesystem/code-analysis.utils.js +0 -165
- package/dist/mcp/utils/filesystem/match-finder.utils.d.ts +0 -16
- package/dist/mcp/utils/filesystem/match-finder.utils.js +0 -85
- package/dist/mcp/utils/filesystem/office-parser.utils.d.ts +0 -43
- package/dist/mcp/utils/filesystem/office-parser.utils.js +0 -163
- package/dist/mcp/utils/filesystem/path-fixer.utils.d.ts +0 -7
- package/dist/mcp/utils/filesystem/path-fixer.utils.js +0 -60
- package/dist/mcp/utils/filesystem/similarity.utils.d.ts +0 -22
- package/dist/mcp/utils/filesystem/similarity.utils.js +0 -75
- package/dist/mcp/utils/todo/date.utils.d.ts +0 -9
- package/dist/mcp/utils/todo/date.utils.js +0 -14
- package/dist/mcp/utils/websearch/browser.utils.d.ts +0 -8
- package/dist/mcp/utils/websearch/browser.utils.js +0 -58
- package/dist/mcp/utils/websearch/text.utils.d.ts +0 -16
- package/dist/mcp/utils/websearch/text.utils.js +0 -39
- package/dist/mcp/websearch.d.ts +0 -88
- package/dist/mcp/websearch.js +0 -375
- package/dist/test/logger-test.d.ts +0 -1
- package/dist/test/logger-test.js +0 -7
- package/dist/types/index.d.ts +0 -15
- package/dist/types/index.js +0 -1
- package/dist/ui/components/AgentPickerPanel.d.ts +0 -10
- package/dist/ui/components/AgentPickerPanel.js +0 -74
- package/dist/ui/components/ChatInput.d.ts +0 -46
- package/dist/ui/components/ChatInput.js +0 -384
- package/dist/ui/components/CommandPanel.d.ts +0 -15
- package/dist/ui/components/CommandPanel.js +0 -80
- package/dist/ui/components/DiffViewer.d.ts +0 -11
- package/dist/ui/components/DiffViewer.js +0 -178
- package/dist/ui/components/FileList.d.ts +0 -15
- package/dist/ui/components/FileList.js +0 -360
- package/dist/ui/components/FileRollbackConfirmation.d.ts +0 -8
- package/dist/ui/components/FileRollbackConfirmation.js +0 -108
- package/dist/ui/components/HelpPanel.d.ts +0 -2
- package/dist/ui/components/HelpPanel.js +0 -67
- package/dist/ui/components/MCPInfoPanel.d.ts +0 -2
- package/dist/ui/components/MCPInfoPanel.js +0 -108
- package/dist/ui/components/MCPInfoScreen.d.ts +0 -7
- package/dist/ui/components/MCPInfoScreen.js +0 -115
- package/dist/ui/components/MarkdownRenderer.d.ts +0 -6
- package/dist/ui/components/MarkdownRenderer.js +0 -70
- package/dist/ui/components/Menu.d.ts +0 -17
- package/dist/ui/components/Menu.js +0 -88
- package/dist/ui/components/MessageList.d.ts +0 -56
- package/dist/ui/components/MessageList.js +0 -97
- package/dist/ui/components/PendingMessages.d.ts +0 -13
- package/dist/ui/components/PendingMessages.js +0 -29
- package/dist/ui/components/PendingToolCalls.d.ts +0 -11
- package/dist/ui/components/PendingToolCalls.js +0 -35
- package/dist/ui/components/ScrollableSelectInput.d.ts +0 -29
- package/dist/ui/components/ScrollableSelectInput.js +0 -157
- package/dist/ui/components/SessionListPanel.d.ts +0 -7
- package/dist/ui/components/SessionListPanel.js +0 -175
- package/dist/ui/components/SessionListScreen.d.ts +0 -7
- package/dist/ui/components/SessionListScreen.js +0 -217
- package/dist/ui/components/SessionListScreenWrapper.d.ts +0 -7
- package/dist/ui/components/SessionListScreenWrapper.js +0 -14
- package/dist/ui/components/ShimmerText.d.ts +0 -9
- package/dist/ui/components/ShimmerText.js +0 -30
- package/dist/ui/components/TodoPickerPanel.d.ts +0 -14
- package/dist/ui/components/TodoPickerPanel.js +0 -119
- package/dist/ui/components/TodoTree.d.ts +0 -15
- package/dist/ui/components/TodoTree.js +0 -60
- package/dist/ui/components/ToolConfirmation.d.ts +0 -21
- package/dist/ui/components/ToolConfirmation.js +0 -204
- package/dist/ui/components/ToolResultPreview.d.ts +0 -13
- package/dist/ui/components/ToolResultPreview.js +0 -337
- package/dist/ui/components/UsagePanel.d.ts +0 -2
- package/dist/ui/components/UsagePanel.js +0 -394
- package/dist/ui/contexts/ThemeContext.d.ts +0 -13
- package/dist/ui/contexts/ThemeContext.js +0 -28
- package/dist/ui/pages/ChatScreen.d.ts +0 -6
- package/dist/ui/pages/ChatScreen.js +0 -1519
- package/dist/ui/pages/CodeBaseConfigScreen.d.ts +0 -8
- package/dist/ui/pages/CodeBaseConfigScreen.js +0 -350
- package/dist/ui/pages/ConfigScreen.d.ts +0 -8
- package/dist/ui/pages/ConfigScreen.js +0 -1101
- package/dist/ui/pages/CustomHeadersScreen.d.ts +0 -6
- package/dist/ui/pages/CustomHeadersScreen.js +0 -502
- package/dist/ui/pages/HeadlessModeScreen.d.ts +0 -7
- package/dist/ui/pages/HeadlessModeScreen.js +0 -381
- package/dist/ui/pages/LanguageSettingsScreen.d.ts +0 -7
- package/dist/ui/pages/LanguageSettingsScreen.js +0 -91
- package/dist/ui/pages/MCPConfigScreen.d.ts +0 -6
- package/dist/ui/pages/MCPConfigScreen.js +0 -55
- package/dist/ui/pages/ProxyConfigScreen.d.ts +0 -8
- package/dist/ui/pages/ProxyConfigScreen.js +0 -149
- package/dist/ui/pages/SensitiveCommandConfigScreen.d.ts +0 -7
- package/dist/ui/pages/SensitiveCommandConfigScreen.js +0 -271
- package/dist/ui/pages/SubAgentConfigScreen.d.ts +0 -9
- package/dist/ui/pages/SubAgentConfigScreen.js +0 -435
- package/dist/ui/pages/SubAgentListScreen.d.ts +0 -9
- package/dist/ui/pages/SubAgentListScreen.js +0 -131
- package/dist/ui/pages/SystemPromptConfigScreen.d.ts +0 -6
- package/dist/ui/pages/SystemPromptConfigScreen.js +0 -326
- package/dist/ui/pages/ThemeSettingsScreen.d.ts +0 -7
- package/dist/ui/pages/ThemeSettingsScreen.js +0 -106
- package/dist/ui/pages/WelcomeScreen.d.ts +0 -7
- package/dist/ui/pages/WelcomeScreen.js +0 -217
- package/dist/ui/themes/index.d.ts +0 -23
- package/dist/ui/themes/index.js +0 -140
- package/dist/utils/apiConfig.d.ts +0 -126
- package/dist/utils/apiConfig.js +0 -423
- package/dist/utils/autoCompress.d.ts +0 -15
- package/dist/utils/autoCompress.js +0 -24
- package/dist/utils/chatExporter.d.ts +0 -9
- package/dist/utils/chatExporter.js +0 -118
- package/dist/utils/checkpointManager.d.ts +0 -74
- package/dist/utils/checkpointManager.js +0 -181
- package/dist/utils/codebaseConfig.d.ts +0 -16
- package/dist/utils/codebaseConfig.js +0 -67
- package/dist/utils/codebaseDatabase.d.ts +0 -102
- package/dist/utils/codebaseDatabase.js +0 -333
- package/dist/utils/codebaseSearchEvents.d.ts +0 -16
- package/dist/utils/codebaseSearchEvents.js +0 -13
- package/dist/utils/commandExecutor.d.ts +0 -13
- package/dist/utils/commandExecutor.js +0 -26
- package/dist/utils/commands/agent.d.ts +0 -2
- package/dist/utils/commands/agent.js +0 -12
- package/dist/utils/commands/clear.d.ts +0 -2
- package/dist/utils/commands/clear.js +0 -12
- package/dist/utils/commands/compact.d.ts +0 -2
- package/dist/utils/commands/compact.js +0 -12
- package/dist/utils/commands/export.d.ts +0 -2
- package/dist/utils/commands/export.js +0 -12
- package/dist/utils/commands/help.d.ts +0 -2
- package/dist/utils/commands/help.js +0 -11
- package/dist/utils/commands/home.d.ts +0 -2
- package/dist/utils/commands/home.js +0 -34
- package/dist/utils/commands/ide.d.ts +0 -2
- package/dist/utils/commands/ide.js +0 -32
- package/dist/utils/commands/init.d.ts +0 -2
- package/dist/utils/commands/init.js +0 -93
- package/dist/utils/commands/mcp.d.ts +0 -2
- package/dist/utils/commands/mcp.js +0 -12
- package/dist/utils/commands/resume.d.ts +0 -2
- package/dist/utils/commands/resume.js +0 -12
- package/dist/utils/commands/review.d.ts +0 -2
- package/dist/utils/commands/review.js +0 -81
- package/dist/utils/commands/role.d.ts +0 -2
- package/dist/utils/commands/role.js +0 -37
- package/dist/utils/commands/todoPicker.d.ts +0 -2
- package/dist/utils/commands/todoPicker.js +0 -12
- package/dist/utils/commands/usage.d.ts +0 -2
- package/dist/utils/commands/usage.js +0 -12
- package/dist/utils/commands/yolo.d.ts +0 -2
- package/dist/utils/commands/yolo.js +0 -12
- package/dist/utils/configManager.d.ts +0 -45
- package/dist/utils/configManager.js +0 -303
- package/dist/utils/contextCompressor.d.ts +0 -16
- package/dist/utils/contextCompressor.js +0 -334
- package/dist/utils/devMode.d.ts +0 -13
- package/dist/utils/devMode.js +0 -54
- package/dist/utils/escapeHandler.d.ts +0 -79
- package/dist/utils/escapeHandler.js +0 -153
- package/dist/utils/fileDialog.d.ts +0 -9
- package/dist/utils/fileDialog.js +0 -74
- package/dist/utils/fileUtils.d.ts +0 -40
- package/dist/utils/fileUtils.js +0 -185
- package/dist/utils/historyManager.d.ts +0 -45
- package/dist/utils/historyManager.js +0 -159
- package/dist/utils/incrementalSnapshot.d.ts +0 -109
- package/dist/utils/incrementalSnapshot.js +0 -383
- package/dist/utils/index.d.ts +0 -11
- package/dist/utils/index.js +0 -18
- package/dist/utils/languageConfig.d.ts +0 -21
- package/dist/utils/languageConfig.js +0 -61
- package/dist/utils/logger.d.ts +0 -37
- package/dist/utils/logger.js +0 -122
- package/dist/utils/mcpToolsManager.d.ts +0 -52
- package/dist/utils/mcpToolsManager.js +0 -878
- package/dist/utils/messageFormatter.d.ts +0 -12
- package/dist/utils/messageFormatter.js +0 -115
- package/dist/utils/notebookManager.d.ts +0 -59
- package/dist/utils/notebookManager.js +0 -213
- package/dist/utils/patch-highlight.d.ts +0 -5
- package/dist/utils/patch-highlight.js +0 -23
- package/dist/utils/processManager.d.ts +0 -27
- package/dist/utils/processManager.js +0 -75
- package/dist/utils/proxyUtils.d.ts +0 -15
- package/dist/utils/proxyUtils.js +0 -50
- package/dist/utils/resourceMonitor.d.ts +0 -65
- package/dist/utils/resourceMonitor.js +0 -175
- package/dist/utils/retryUtils.d.ts +0 -49
- package/dist/utils/retryUtils.js +0 -303
- package/dist/utils/sensitiveCommandManager.d.ts +0 -53
- package/dist/utils/sensitiveCommandManager.js +0 -308
- package/dist/utils/sessionConverter.d.ts +0 -7
- package/dist/utils/sessionConverter.js +0 -306
- package/dist/utils/sessionManager.d.ts +0 -53
- package/dist/utils/sessionManager.js +0 -371
- package/dist/utils/subAgentConfig.d.ts +0 -50
- package/dist/utils/subAgentConfig.js +0 -221
- package/dist/utils/subAgentExecutor.d.ts +0 -40
- package/dist/utils/subAgentExecutor.js +0 -434
- package/dist/utils/terminal.d.ts +0 -5
- package/dist/utils/terminal.js +0 -13
- package/dist/utils/textBuffer.d.ts +0 -99
- package/dist/utils/textBuffer.js +0 -547
- package/dist/utils/textUtils.d.ts +0 -37
- package/dist/utils/textUtils.js +0 -102
- package/dist/utils/themeConfig.d.ts +0 -21
- package/dist/utils/themeConfig.js +0 -61
- package/dist/utils/todoPreprocessor.d.ts +0 -5
- package/dist/utils/todoPreprocessor.js +0 -18
- package/dist/utils/todoScanner.d.ts +0 -8
- package/dist/utils/todoScanner.js +0 -148
- package/dist/utils/toolDisplayConfig.d.ts +0 -16
- package/dist/utils/toolDisplayConfig.js +0 -47
- package/dist/utils/toolExecutor.d.ts +0 -37
- package/dist/utils/toolExecutor.js +0 -224
- package/dist/utils/usageLogger.d.ts +0 -11
- package/dist/utils/usageLogger.js +0 -114
- package/dist/utils/vscodeConnection.d.ts +0 -76
- package/dist/utils/vscodeConnection.js +0 -430
- package/dist/utils/workspaceSnapshot.d.ts +0 -63
- package/dist/utils/workspaceSnapshot.js +0 -300
|
@@ -1,29 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
import { Box, Text } from 'ink';
|
|
3
|
-
import { useTheme } from '../contexts/ThemeContext.js';
|
|
4
|
-
export default function PendingMessages({ pendingMessages }) {
|
|
5
|
-
const { theme } = useTheme();
|
|
6
|
-
if (pendingMessages.length === 0) {
|
|
7
|
-
return null;
|
|
8
|
-
}
|
|
9
|
-
return (React.createElement(Box, { flexDirection: "column", borderStyle: "round", borderColor: theme.colors.warning, paddingX: 1 },
|
|
10
|
-
React.createElement(Text, { color: theme.colors.warning, bold: true },
|
|
11
|
-
"\u2B11 Pending Messages (",
|
|
12
|
-
pendingMessages.length,
|
|
13
|
-
")"),
|
|
14
|
-
pendingMessages.map((message, index) => (React.createElement(Box, { key: index, marginLeft: 1, marginY: 0, flexDirection: "column" },
|
|
15
|
-
React.createElement(Box, null,
|
|
16
|
-
React.createElement(Text, { color: "blue", bold: true },
|
|
17
|
-
index + 1,
|
|
18
|
-
"."),
|
|
19
|
-
React.createElement(Box, { marginLeft: 1 },
|
|
20
|
-
React.createElement(Text, { color: theme.colors.menuSecondary }, message.text.length > 60 ? `${message.text.substring(0, 60)}...` : message.text))),
|
|
21
|
-
message.images && message.images.length > 0 && (React.createElement(Box, { marginLeft: 3 },
|
|
22
|
-
React.createElement(Text, { color: theme.colors.menuSecondary, dimColor: true },
|
|
23
|
-
"\u2514\u2500 ",
|
|
24
|
-
message.images.length,
|
|
25
|
-
" image",
|
|
26
|
-
message.images.length > 1 ? 's' : '',
|
|
27
|
-
" attached")))))),
|
|
28
|
-
React.createElement(Text, { color: theme.colors.warning, dimColor: true }, "Will be sent after tool execution completes")));
|
|
29
|
-
}
|
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
import type { Message } from './MessageList.js';
|
|
3
|
-
interface Props {
|
|
4
|
-
messages: Message[];
|
|
5
|
-
}
|
|
6
|
-
/**
|
|
7
|
-
* 显示正在执行的工具调用(只显示耗时工具)
|
|
8
|
-
* 这些消息有 toolPending: true 标记
|
|
9
|
-
*/
|
|
10
|
-
export default function PendingToolCalls({ messages }: Props): React.JSX.Element | null;
|
|
11
|
-
export {};
|
|
@@ -1,35 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
import { Box, Text } from 'ink';
|
|
3
|
-
import Spinner from 'ink-spinner';
|
|
4
|
-
/**
|
|
5
|
-
* 显示正在执行的工具调用(只显示耗时工具)
|
|
6
|
-
* 这些消息有 toolPending: true 标记
|
|
7
|
-
*/
|
|
8
|
-
export default function PendingToolCalls({ messages }) {
|
|
9
|
-
// 筛选出正在执行的工具调用消息
|
|
10
|
-
const pendingTools = messages.filter(msg => msg.role === 'assistant' && msg.toolPending === true);
|
|
11
|
-
if (pendingTools.length === 0) {
|
|
12
|
-
return null;
|
|
13
|
-
}
|
|
14
|
-
return (React.createElement(Box, { flexDirection: "column", borderStyle: "round", borderColor: "cyan", paddingX: 1 },
|
|
15
|
-
React.createElement(Text, { color: "cyan", bold: true },
|
|
16
|
-
React.createElement(Spinner, { type: "dots" }),
|
|
17
|
-
" Executing Tools (",
|
|
18
|
-
pendingTools.length,
|
|
19
|
-
")"),
|
|
20
|
-
pendingTools.map((tool, index) => (React.createElement(Box, { key: index, marginLeft: 1, marginY: 0 },
|
|
21
|
-
React.createElement(Text, { color: "yellow", bold: true },
|
|
22
|
-
index + 1,
|
|
23
|
-
"."),
|
|
24
|
-
React.createElement(Box, { marginLeft: 1 },
|
|
25
|
-
React.createElement(Text, { color: "gray" }, tool.content)),
|
|
26
|
-
tool.toolDisplay && tool.toolDisplay.args.length > 0 && (React.createElement(Box, { flexDirection: "column", marginLeft: 2 },
|
|
27
|
-
tool.toolDisplay.args.slice(0, 3).map((arg, argIndex) => (React.createElement(Text, { key: argIndex, color: "gray", dimColor: true },
|
|
28
|
-
arg.key,
|
|
29
|
-
": ",
|
|
30
|
-
arg.value.length > 50 ? `${arg.value.substring(0, 50)}...` : arg.value))),
|
|
31
|
-
tool.toolDisplay.args.length > 3 && (React.createElement(Text, { color: "gray", dimColor: true },
|
|
32
|
-
"... and ",
|
|
33
|
-
tool.toolDisplay.args.length - 3,
|
|
34
|
-
" more args")))))))));
|
|
35
|
-
}
|
|
@@ -1,29 +0,0 @@
|
|
|
1
|
-
import React from 'react';
|
|
2
|
-
type SelectItem = {
|
|
3
|
-
label: string;
|
|
4
|
-
value: string;
|
|
5
|
-
key?: string;
|
|
6
|
-
[index: string]: unknown;
|
|
7
|
-
};
|
|
8
|
-
type IndicatorProps = {
|
|
9
|
-
isSelected: boolean;
|
|
10
|
-
};
|
|
11
|
-
type RenderItemProps<T extends SelectItem> = T & {
|
|
12
|
-
isSelected: boolean;
|
|
13
|
-
isMarked: boolean;
|
|
14
|
-
};
|
|
15
|
-
type Props<T extends SelectItem> = {
|
|
16
|
-
items: readonly T[];
|
|
17
|
-
limit?: number;
|
|
18
|
-
initialIndex?: number;
|
|
19
|
-
isFocused?: boolean;
|
|
20
|
-
indicator?: (props: IndicatorProps) => React.ReactNode;
|
|
21
|
-
renderItem?: (props: RenderItemProps<T>) => React.ReactNode;
|
|
22
|
-
onSelect?: (item: T) => void;
|
|
23
|
-
onHighlight?: (item: T) => void;
|
|
24
|
-
selectedValues?: ReadonlySet<string> | readonly string[];
|
|
25
|
-
onToggleItem?: (item: T) => void;
|
|
26
|
-
onDeleteSelection?: () => void;
|
|
27
|
-
};
|
|
28
|
-
export default function ScrollableSelectInput<T extends SelectItem>({ items, limit, initialIndex, isFocused, indicator, renderItem, onSelect, onHighlight, selectedValues, onToggleItem, onDeleteSelection }: Props<T>): React.JSX.Element | null;
|
|
29
|
-
export {};
|
|
@@ -1,157 +0,0 @@
|
|
|
1
|
-
import React, { useState, useMemo, useEffect, useCallback } from 'react';
|
|
2
|
-
import { Box, Text, useInput } from 'ink';
|
|
3
|
-
function DefaultIndicator({ isSelected }) {
|
|
4
|
-
return (React.createElement(Box, { marginRight: 1 },
|
|
5
|
-
React.createElement(Text, { color: isSelected ? 'cyan' : 'gray' }, isSelected ? '>' : ' ')));
|
|
6
|
-
}
|
|
7
|
-
function DefaultItem({ label, isSelected }) {
|
|
8
|
-
return React.createElement(Text, { color: isSelected ? 'cyan' : 'white' }, label);
|
|
9
|
-
}
|
|
10
|
-
export default function ScrollableSelectInput({ items, limit, initialIndex = 0, isFocused = true, indicator = DefaultIndicator, renderItem, onSelect, onHighlight, selectedValues, onToggleItem, onDeleteSelection }) {
|
|
11
|
-
const totalItems = items.length;
|
|
12
|
-
const windowSize = totalItems === 0 ? 0 : Math.min(Math.max(limit ?? totalItems, 1), totalItems);
|
|
13
|
-
const selectedValueSet = useMemo(() => {
|
|
14
|
-
if (!selectedValues) {
|
|
15
|
-
return new Set();
|
|
16
|
-
}
|
|
17
|
-
if (selectedValues instanceof Set) {
|
|
18
|
-
return selectedValues;
|
|
19
|
-
}
|
|
20
|
-
return new Set(selectedValues);
|
|
21
|
-
}, [selectedValues]);
|
|
22
|
-
const clampCursor = useCallback((value) => {
|
|
23
|
-
if (totalItems === 0) {
|
|
24
|
-
return 0;
|
|
25
|
-
}
|
|
26
|
-
if (value < 0) {
|
|
27
|
-
return 0;
|
|
28
|
-
}
|
|
29
|
-
if (value > totalItems - 1) {
|
|
30
|
-
return totalItems - 1;
|
|
31
|
-
}
|
|
32
|
-
return value;
|
|
33
|
-
}, [totalItems]);
|
|
34
|
-
const computeOffset = useCallback((currentOffset, targetCursor) => {
|
|
35
|
-
if (totalItems === 0 || windowSize === 0) {
|
|
36
|
-
return 0;
|
|
37
|
-
}
|
|
38
|
-
const maxOffset = Math.max(0, totalItems - windowSize);
|
|
39
|
-
let nextOffset = Math.min(Math.max(currentOffset, 0), maxOffset);
|
|
40
|
-
if (targetCursor < nextOffset) {
|
|
41
|
-
nextOffset = targetCursor;
|
|
42
|
-
}
|
|
43
|
-
else if (targetCursor >= nextOffset + windowSize) {
|
|
44
|
-
nextOffset = targetCursor - windowSize + 1;
|
|
45
|
-
}
|
|
46
|
-
return Math.min(Math.max(nextOffset, 0), maxOffset);
|
|
47
|
-
}, [totalItems, windowSize]);
|
|
48
|
-
const [cursor, setCursor] = useState(() => clampCursor(initialIndex));
|
|
49
|
-
const [offset, setOffset] = useState(() => computeOffset(clampCursor(initialIndex), clampCursor(initialIndex)));
|
|
50
|
-
useEffect(() => {
|
|
51
|
-
if (totalItems === 0) {
|
|
52
|
-
if (cursor !== 0) {
|
|
53
|
-
setCursor(0);
|
|
54
|
-
}
|
|
55
|
-
if (offset !== 0) {
|
|
56
|
-
setOffset(0);
|
|
57
|
-
}
|
|
58
|
-
return;
|
|
59
|
-
}
|
|
60
|
-
const clampedCursor = clampCursor(cursor);
|
|
61
|
-
if (clampedCursor !== cursor) {
|
|
62
|
-
setCursor(clampedCursor);
|
|
63
|
-
return;
|
|
64
|
-
}
|
|
65
|
-
const clampedOffset = computeOffset(offset, clampedCursor);
|
|
66
|
-
if (clampedOffset !== offset) {
|
|
67
|
-
setOffset(clampedOffset);
|
|
68
|
-
}
|
|
69
|
-
}, [clampCursor, computeOffset, cursor, offset, totalItems]);
|
|
70
|
-
const visibleItems = useMemo(() => {
|
|
71
|
-
if (windowSize === 0) {
|
|
72
|
-
return [];
|
|
73
|
-
}
|
|
74
|
-
return items.slice(offset, offset + windowSize);
|
|
75
|
-
}, [items, offset, windowSize]);
|
|
76
|
-
const selectedItem = totalItems === 0 ? undefined : items[cursor];
|
|
77
|
-
useEffect(() => {
|
|
78
|
-
if (selectedItem && onHighlight) {
|
|
79
|
-
onHighlight(selectedItem);
|
|
80
|
-
}
|
|
81
|
-
}, [onHighlight, selectedItem]);
|
|
82
|
-
const moveCursor = useCallback((direction) => {
|
|
83
|
-
if (totalItems === 0) {
|
|
84
|
-
return;
|
|
85
|
-
}
|
|
86
|
-
setCursor(previousCursor => {
|
|
87
|
-
const nextCursor = clampCursor(previousCursor + direction);
|
|
88
|
-
if (nextCursor === previousCursor) {
|
|
89
|
-
return previousCursor;
|
|
90
|
-
}
|
|
91
|
-
setOffset(previousOffset => computeOffset(previousOffset, nextCursor));
|
|
92
|
-
return nextCursor;
|
|
93
|
-
});
|
|
94
|
-
}, [clampCursor, computeOffset, totalItems]);
|
|
95
|
-
const selectIndex = useCallback((targetIndex) => {
|
|
96
|
-
if (totalItems === 0) {
|
|
97
|
-
return;
|
|
98
|
-
}
|
|
99
|
-
const boundedIndex = clampCursor(targetIndex);
|
|
100
|
-
setCursor(boundedIndex);
|
|
101
|
-
setOffset(previousOffset => computeOffset(previousOffset, boundedIndex));
|
|
102
|
-
const item = items[boundedIndex];
|
|
103
|
-
if (item) {
|
|
104
|
-
onSelect?.(item);
|
|
105
|
-
}
|
|
106
|
-
}, [clampCursor, computeOffset, items, onSelect, totalItems]);
|
|
107
|
-
const handleInput = useCallback((input, key) => {
|
|
108
|
-
if (!isFocused || totalItems === 0) {
|
|
109
|
-
return;
|
|
110
|
-
}
|
|
111
|
-
if (key['upArrow']) {
|
|
112
|
-
moveCursor(-1);
|
|
113
|
-
return;
|
|
114
|
-
}
|
|
115
|
-
if (key['downArrow']) {
|
|
116
|
-
moveCursor(1);
|
|
117
|
-
return;
|
|
118
|
-
}
|
|
119
|
-
if (key['return'] && selectedItem) {
|
|
120
|
-
onSelect?.(selectedItem);
|
|
121
|
-
return;
|
|
122
|
-
}
|
|
123
|
-
if (input === ' ' && selectedItem) {
|
|
124
|
-
onToggleItem?.(selectedItem);
|
|
125
|
-
return;
|
|
126
|
-
}
|
|
127
|
-
if ((input === 'd' || input === 'D') && onDeleteSelection) {
|
|
128
|
-
onDeleteSelection();
|
|
129
|
-
return;
|
|
130
|
-
}
|
|
131
|
-
if (/^[1-9]$/.test(input) && windowSize > 0) {
|
|
132
|
-
const target = Number.parseInt(input, 10) - 1;
|
|
133
|
-
if (target >= 0 && target < visibleItems.length) {
|
|
134
|
-
selectIndex(offset + target);
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
}, [isFocused, moveCursor, offset, onDeleteSelection, onSelect, onToggleItem, selectIndex, selectedItem, totalItems, visibleItems.length, windowSize]);
|
|
138
|
-
useInput(handleInput, { isActive: isFocused });
|
|
139
|
-
if (windowSize === 0) {
|
|
140
|
-
return null;
|
|
141
|
-
}
|
|
142
|
-
const renderRow = useCallback((row) => {
|
|
143
|
-
if (renderItem) {
|
|
144
|
-
return renderItem(row);
|
|
145
|
-
}
|
|
146
|
-
return DefaultItem(row);
|
|
147
|
-
}, [renderItem]);
|
|
148
|
-
return (React.createElement(Box, { flexDirection: "column" }, visibleItems.map((item, index) => {
|
|
149
|
-
const absoluteIndex = offset + index;
|
|
150
|
-
const isSelected = absoluteIndex === cursor;
|
|
151
|
-
const isMarked = selectedValueSet.has(item.value);
|
|
152
|
-
const key = (item.key ?? item.value);
|
|
153
|
-
return (React.createElement(Box, { key: key },
|
|
154
|
-
indicator({ isSelected }),
|
|
155
|
-
renderRow({ ...item, isSelected, isMarked })));
|
|
156
|
-
})));
|
|
157
|
-
}
|
|
@@ -1,175 +0,0 @@
|
|
|
1
|
-
import React, { useState, useEffect, useCallback } from 'react';
|
|
2
|
-
import { Box, Text, useInput } from 'ink';
|
|
3
|
-
import { sessionManager, } from '../../utils/sessionManager.js';
|
|
4
|
-
export default function SessionListPanel({ onSelectSession, onClose }) {
|
|
5
|
-
const [sessions, setSessions] = useState([]);
|
|
6
|
-
const [loading, setLoading] = useState(true);
|
|
7
|
-
const [selectedIndex, setSelectedIndex] = useState(0);
|
|
8
|
-
const [scrollOffset, setScrollOffset] = useState(0);
|
|
9
|
-
const [markedSessions, setMarkedSessions] = useState(new Set());
|
|
10
|
-
const VISIBLE_ITEMS = 5; // Number of items to show at once
|
|
11
|
-
// Load sessions on mount
|
|
12
|
-
useEffect(() => {
|
|
13
|
-
const loadSessions = async () => {
|
|
14
|
-
setLoading(true);
|
|
15
|
-
try {
|
|
16
|
-
const sessionList = await sessionManager.listSessions();
|
|
17
|
-
setSessions(sessionList);
|
|
18
|
-
}
|
|
19
|
-
catch (error) {
|
|
20
|
-
console.error('Failed to load sessions:', error);
|
|
21
|
-
setSessions([]);
|
|
22
|
-
}
|
|
23
|
-
finally {
|
|
24
|
-
setLoading(false);
|
|
25
|
-
}
|
|
26
|
-
};
|
|
27
|
-
void loadSessions();
|
|
28
|
-
}, []);
|
|
29
|
-
// Format date to relative time
|
|
30
|
-
const formatDate = useCallback((timestamp) => {
|
|
31
|
-
const date = new Date(timestamp);
|
|
32
|
-
const now = new Date();
|
|
33
|
-
const diffMs = now.getTime() - date.getTime();
|
|
34
|
-
const diffMinutes = Math.floor(diffMs / (1000 * 60));
|
|
35
|
-
const diffHours = Math.floor(diffMinutes / 60);
|
|
36
|
-
const diffDays = Math.floor(diffHours / 24);
|
|
37
|
-
if (diffMinutes < 1)
|
|
38
|
-
return 'now';
|
|
39
|
-
if (diffMinutes < 60)
|
|
40
|
-
return `${diffMinutes}m`;
|
|
41
|
-
if (diffHours < 24)
|
|
42
|
-
return `${diffHours}h`;
|
|
43
|
-
if (diffDays < 7)
|
|
44
|
-
return `${diffDays}d`;
|
|
45
|
-
return date.toLocaleDateString('en-US', { month: 'short', day: 'numeric' });
|
|
46
|
-
}, []);
|
|
47
|
-
// Handle keyboard input
|
|
48
|
-
useInput((input, key) => {
|
|
49
|
-
if (loading)
|
|
50
|
-
return;
|
|
51
|
-
if (key.escape) {
|
|
52
|
-
onClose();
|
|
53
|
-
return;
|
|
54
|
-
}
|
|
55
|
-
if (key.upArrow) {
|
|
56
|
-
setSelectedIndex(prev => {
|
|
57
|
-
const newIndex = Math.max(0, prev - 1);
|
|
58
|
-
// Adjust scroll offset if needed
|
|
59
|
-
if (newIndex < scrollOffset) {
|
|
60
|
-
setScrollOffset(newIndex);
|
|
61
|
-
}
|
|
62
|
-
return newIndex;
|
|
63
|
-
});
|
|
64
|
-
return;
|
|
65
|
-
}
|
|
66
|
-
if (key.downArrow) {
|
|
67
|
-
setSelectedIndex(prev => {
|
|
68
|
-
const newIndex = Math.min(sessions.length - 1, prev + 1);
|
|
69
|
-
// Adjust scroll offset if needed
|
|
70
|
-
if (newIndex >= scrollOffset + VISIBLE_ITEMS) {
|
|
71
|
-
setScrollOffset(newIndex - VISIBLE_ITEMS + 1);
|
|
72
|
-
}
|
|
73
|
-
return newIndex;
|
|
74
|
-
});
|
|
75
|
-
return;
|
|
76
|
-
}
|
|
77
|
-
// Space to toggle mark
|
|
78
|
-
if (input === ' ') {
|
|
79
|
-
const currentSession = sessions[selectedIndex];
|
|
80
|
-
if (currentSession) {
|
|
81
|
-
setMarkedSessions(prev => {
|
|
82
|
-
const next = new Set(prev);
|
|
83
|
-
if (next.has(currentSession.id)) {
|
|
84
|
-
next.delete(currentSession.id);
|
|
85
|
-
}
|
|
86
|
-
else {
|
|
87
|
-
next.add(currentSession.id);
|
|
88
|
-
}
|
|
89
|
-
return next;
|
|
90
|
-
});
|
|
91
|
-
}
|
|
92
|
-
return;
|
|
93
|
-
}
|
|
94
|
-
// D to delete marked sessions
|
|
95
|
-
if (input === 'd' || input === 'D') {
|
|
96
|
-
if (markedSessions.size > 0) {
|
|
97
|
-
const deleteMarked = async () => {
|
|
98
|
-
const ids = Array.from(markedSessions);
|
|
99
|
-
await Promise.all(ids.map(id => sessionManager.deleteSession(id)));
|
|
100
|
-
// Reload sessions
|
|
101
|
-
const sessionList = await sessionManager.listSessions();
|
|
102
|
-
setSessions(sessionList);
|
|
103
|
-
setMarkedSessions(new Set());
|
|
104
|
-
// Reset selection if needed
|
|
105
|
-
if (selectedIndex >= sessionList.length && sessionList.length > 0) {
|
|
106
|
-
setSelectedIndex(sessionList.length - 1);
|
|
107
|
-
}
|
|
108
|
-
};
|
|
109
|
-
void deleteMarked();
|
|
110
|
-
}
|
|
111
|
-
return;
|
|
112
|
-
}
|
|
113
|
-
if (key.return && sessions.length > 0) {
|
|
114
|
-
const selectedSession = sessions[selectedIndex];
|
|
115
|
-
if (selectedSession) {
|
|
116
|
-
onSelectSession(selectedSession.id);
|
|
117
|
-
}
|
|
118
|
-
return;
|
|
119
|
-
}
|
|
120
|
-
});
|
|
121
|
-
if (loading) {
|
|
122
|
-
return (React.createElement(Box, { borderStyle: "round", borderColor: "cyan", paddingX: 1 },
|
|
123
|
-
React.createElement(Text, { color: "gray", dimColor: true }, "Loading sessions...")));
|
|
124
|
-
}
|
|
125
|
-
if (sessions.length === 0) {
|
|
126
|
-
return (React.createElement(Box, { borderStyle: "round", borderColor: "yellow", paddingX: 1 },
|
|
127
|
-
React.createElement(Text, { color: "gray", dimColor: true }, "No conversations found \u2022 Press ESC to close")));
|
|
128
|
-
}
|
|
129
|
-
// Calculate visible sessions based on scroll offset
|
|
130
|
-
const visibleSessions = sessions.slice(scrollOffset, scrollOffset + VISIBLE_ITEMS);
|
|
131
|
-
const hasMore = sessions.length > scrollOffset + VISIBLE_ITEMS;
|
|
132
|
-
const hasPrevious = scrollOffset > 0;
|
|
133
|
-
const currentSession = sessions[selectedIndex];
|
|
134
|
-
return (React.createElement(Box, { borderStyle: "round", borderColor: "cyan", paddingX: 1, flexDirection: "column" },
|
|
135
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
136
|
-
React.createElement(Text, { color: "cyan", dimColor: true },
|
|
137
|
-
"Resume (",
|
|
138
|
-
selectedIndex + 1,
|
|
139
|
-
"/",
|
|
140
|
-
sessions.length,
|
|
141
|
-
")",
|
|
142
|
-
currentSession && ` • ${currentSession.messageCount} msgs`,
|
|
143
|
-
markedSessions.size > 0 && (React.createElement(Text, { color: "yellow" },
|
|
144
|
-
" \u2022 ",
|
|
145
|
-
markedSessions.size,
|
|
146
|
-
" marked"))),
|
|
147
|
-
React.createElement(Text, { color: "gray", dimColor: true }, "\u2191\u2193 navigate \u2022 Space mark \u2022 D delete \u2022 Enter select \u2022 ESC close")),
|
|
148
|
-
hasPrevious && (React.createElement(Text, { color: "gray", dimColor: true },
|
|
149
|
-
' ',
|
|
150
|
-
"\u2191 ",
|
|
151
|
-
scrollOffset,
|
|
152
|
-
" more above")),
|
|
153
|
-
visibleSessions.map((session, index) => {
|
|
154
|
-
const actualIndex = scrollOffset + index;
|
|
155
|
-
const isSelected = actualIndex === selectedIndex;
|
|
156
|
-
const isMarked = markedSessions.has(session.id);
|
|
157
|
-
// Remove newlines and other whitespace characters from title
|
|
158
|
-
const cleanTitle = (session.title || 'Untitled').replace(/[\r\n\t]+/g, ' ');
|
|
159
|
-
const timeStr = formatDate(session.updatedAt);
|
|
160
|
-
const truncatedLabel = cleanTitle.length > 50 ? cleanTitle.slice(0, 47) + '...' : cleanTitle;
|
|
161
|
-
return (React.createElement(Box, { key: session.id },
|
|
162
|
-
React.createElement(Text, { color: isMarked ? 'green' : 'gray' }, isMarked ? '✔ ' : ' '),
|
|
163
|
-
React.createElement(Text, { color: isSelected ? 'green' : 'gray' }, isSelected ? '❯ ' : ' '),
|
|
164
|
-
React.createElement(Text, { color: isSelected ? 'cyan' : isMarked ? 'green' : 'white' }, truncatedLabel),
|
|
165
|
-
React.createElement(Text, { color: "gray", dimColor: true },
|
|
166
|
-
' ',
|
|
167
|
-
"\u2022 ",
|
|
168
|
-
timeStr)));
|
|
169
|
-
}),
|
|
170
|
-
hasMore && (React.createElement(Text, { color: "gray", dimColor: true },
|
|
171
|
-
' ',
|
|
172
|
-
"\u2193 ",
|
|
173
|
-
sessions.length - scrollOffset - VISIBLE_ITEMS,
|
|
174
|
-
" more below"))));
|
|
175
|
-
}
|
|
@@ -1,217 +0,0 @@
|
|
|
1
|
-
import React, { useState, useEffect, useMemo, useCallback } from 'react';
|
|
2
|
-
import { Box, Text, useInput, useStdout } from 'ink';
|
|
3
|
-
import Gradient from 'ink-gradient';
|
|
4
|
-
import { Alert } from '@inkjs/ui';
|
|
5
|
-
import ScrollableSelectInput from './ScrollableSelectInput.js';
|
|
6
|
-
import { sessionManager, } from '../../utils/sessionManager.js';
|
|
7
|
-
export default function SessionListScreen({ onBack, onSelectSession }) {
|
|
8
|
-
const [sessions, setSessions] = useState([]);
|
|
9
|
-
const [loading, setLoading] = useState(true);
|
|
10
|
-
const [error, setError] = useState('');
|
|
11
|
-
const [selectedSessions, setSelectedSessions] = useState(new Set());
|
|
12
|
-
const [actionMessage, setActionMessage] = useState(null);
|
|
13
|
-
const { stdout } = useStdout();
|
|
14
|
-
const loadSessions = useCallback(async () => {
|
|
15
|
-
setLoading(true);
|
|
16
|
-
setError('');
|
|
17
|
-
try {
|
|
18
|
-
const sessionList = await sessionManager.listSessions();
|
|
19
|
-
setSessions(sessionList);
|
|
20
|
-
setSelectedSessions(new Set());
|
|
21
|
-
setActionMessage(null);
|
|
22
|
-
}
|
|
23
|
-
catch (err) {
|
|
24
|
-
setError(err instanceof Error ? err.message : 'Failed to load sessions');
|
|
25
|
-
}
|
|
26
|
-
finally {
|
|
27
|
-
setLoading(false);
|
|
28
|
-
}
|
|
29
|
-
}, []);
|
|
30
|
-
useEffect(() => {
|
|
31
|
-
void loadSessions();
|
|
32
|
-
}, [loadSessions]);
|
|
33
|
-
const formatDate = useCallback((timestamp) => {
|
|
34
|
-
const date = new Date(timestamp);
|
|
35
|
-
const now = new Date();
|
|
36
|
-
const diffMs = now.getTime() - date.getTime();
|
|
37
|
-
const diffHours = Math.floor(diffMs / (1000 * 60 * 60));
|
|
38
|
-
const diffDays = Math.floor(diffHours / 24);
|
|
39
|
-
if (diffHours < 1) {
|
|
40
|
-
const diffMinutes = Math.floor(diffMs / (1000 * 60));
|
|
41
|
-
return diffMinutes < 1 ? 'Just now' : `${diffMinutes}m ago`;
|
|
42
|
-
}
|
|
43
|
-
else if (diffHours < 24) {
|
|
44
|
-
return `${diffHours}h ago`;
|
|
45
|
-
}
|
|
46
|
-
else if (diffDays < 7) {
|
|
47
|
-
return `${diffDays}d ago`;
|
|
48
|
-
}
|
|
49
|
-
else {
|
|
50
|
-
return date.toLocaleDateString();
|
|
51
|
-
}
|
|
52
|
-
}, []);
|
|
53
|
-
// Create select items with truncated labels
|
|
54
|
-
const selectItems = useMemo(() => {
|
|
55
|
-
const terminalWidth = stdout?.columns || 80;
|
|
56
|
-
// Reserve space for indicators (✔ + ❯ + spacing) and margins
|
|
57
|
-
const reservedSpace = 30;
|
|
58
|
-
const maxLabelWidth = Math.max(30, terminalWidth - reservedSpace);
|
|
59
|
-
return sessions.map(session => {
|
|
60
|
-
const timeString = formatDate(session.updatedAt);
|
|
61
|
-
// Remove newlines and other whitespace characters from title
|
|
62
|
-
const title = (session.title || 'Untitled').replace(/[\r\n\t]+/g, ' ');
|
|
63
|
-
// Format: "Title • 5 msgs • 2h ago"
|
|
64
|
-
const messageInfo = `${session.messageCount} msg${session.messageCount !== 1 ? 's' : ''}`;
|
|
65
|
-
const fullLabel = `${title} • ${messageInfo} • ${timeString}`;
|
|
66
|
-
// Truncate if too long - prioritize showing the title
|
|
67
|
-
let truncatedLabel = fullLabel;
|
|
68
|
-
if (fullLabel.length > maxLabelWidth) {
|
|
69
|
-
const ellipsis = '...';
|
|
70
|
-
const suffixLength = messageInfo.length + timeString.length + 6; // " • " x2 + "..."
|
|
71
|
-
const availableForTitle = maxLabelWidth - suffixLength - ellipsis.length;
|
|
72
|
-
if (availableForTitle > 10) {
|
|
73
|
-
const truncatedTitle = title.substring(0, availableForTitle);
|
|
74
|
-
truncatedLabel = `${truncatedTitle}${ellipsis} • ${messageInfo} • ${timeString}`;
|
|
75
|
-
}
|
|
76
|
-
else {
|
|
77
|
-
// If terminal is too narrow, just truncate the whole thing
|
|
78
|
-
truncatedLabel =
|
|
79
|
-
fullLabel.substring(0, maxLabelWidth - ellipsis.length) + ellipsis;
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
return {
|
|
83
|
-
label: truncatedLabel,
|
|
84
|
-
value: session.id,
|
|
85
|
-
isMarked: selectedSessions.has(session.id),
|
|
86
|
-
};
|
|
87
|
-
});
|
|
88
|
-
}, [sessions, formatDate, stdout?.columns, selectedSessions]);
|
|
89
|
-
const handleSelect = useCallback((item) => {
|
|
90
|
-
onSelectSession(item.value);
|
|
91
|
-
}, [onSelectSession]);
|
|
92
|
-
const handleToggleItem = useCallback((item) => {
|
|
93
|
-
setSelectedSessions(previous => {
|
|
94
|
-
const next = new Set(previous);
|
|
95
|
-
if (next.has(item.value)) {
|
|
96
|
-
next.delete(item.value);
|
|
97
|
-
}
|
|
98
|
-
else {
|
|
99
|
-
next.add(item.value);
|
|
100
|
-
}
|
|
101
|
-
return next;
|
|
102
|
-
});
|
|
103
|
-
setActionMessage(null);
|
|
104
|
-
}, []);
|
|
105
|
-
const handleDeleteSelected = useCallback(async () => {
|
|
106
|
-
if (selectedSessions.size === 0) {
|
|
107
|
-
setActionMessage({ type: 'info', text: 'No conversations selected.' });
|
|
108
|
-
return;
|
|
109
|
-
}
|
|
110
|
-
const ids = Array.from(selectedSessions);
|
|
111
|
-
const deletionResults = await Promise.all(ids.map(async (id) => ({
|
|
112
|
-
id,
|
|
113
|
-
success: await sessionManager.deleteSession(id),
|
|
114
|
-
})));
|
|
115
|
-
const succeededIds = deletionResults
|
|
116
|
-
.filter(result => result.success)
|
|
117
|
-
.map(result => result.id);
|
|
118
|
-
const failedIds = deletionResults
|
|
119
|
-
.filter(result => !result.success)
|
|
120
|
-
.map(result => result.id);
|
|
121
|
-
if (succeededIds.length > 0) {
|
|
122
|
-
setSessions(previous => previous.filter(session => !succeededIds.includes(session.id)));
|
|
123
|
-
setSelectedSessions(previous => {
|
|
124
|
-
const next = new Set(previous);
|
|
125
|
-
for (const id of succeededIds) {
|
|
126
|
-
next.delete(id);
|
|
127
|
-
}
|
|
128
|
-
return next;
|
|
129
|
-
});
|
|
130
|
-
}
|
|
131
|
-
if (failedIds.length > 0) {
|
|
132
|
-
setActionMessage({
|
|
133
|
-
type: 'error',
|
|
134
|
-
text: `Failed to delete ${failedIds.length} conversation${failedIds.length > 1 ? 's' : ''}.`,
|
|
135
|
-
});
|
|
136
|
-
}
|
|
137
|
-
else if (succeededIds.length > 0) {
|
|
138
|
-
setActionMessage({
|
|
139
|
-
type: 'info',
|
|
140
|
-
text: `Deleted ${succeededIds.length} conversation${succeededIds.length > 1 ? 's' : ''}.`,
|
|
141
|
-
});
|
|
142
|
-
}
|
|
143
|
-
else {
|
|
144
|
-
setActionMessage({ type: 'info', text: 'No conversations deleted.' });
|
|
145
|
-
}
|
|
146
|
-
}, [selectedSessions]);
|
|
147
|
-
const handleInput = useCallback((input, key) => {
|
|
148
|
-
if (key.escape) {
|
|
149
|
-
onBack();
|
|
150
|
-
return;
|
|
151
|
-
}
|
|
152
|
-
if (input === 'r' || input === 'R') {
|
|
153
|
-
void loadSessions();
|
|
154
|
-
}
|
|
155
|
-
}, [loadSessions, onBack]);
|
|
156
|
-
useInput(handleInput);
|
|
157
|
-
if (loading) {
|
|
158
|
-
return (React.createElement(Box, { flexDirection: "column", padding: 1 },
|
|
159
|
-
React.createElement(Box, { marginBottom: 2, borderStyle: "double", borderColor: "cyan", paddingX: 2, paddingY: 1 },
|
|
160
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
161
|
-
React.createElement(Gradient, { name: "rainbow" }, "Resume Conversation"),
|
|
162
|
-
React.createElement(Text, { color: "gray", dimColor: true }, "Loading your conversation history...")))));
|
|
163
|
-
}
|
|
164
|
-
if (error) {
|
|
165
|
-
return (React.createElement(Box, { flexDirection: "column", padding: 1 },
|
|
166
|
-
React.createElement(Box, { marginBottom: 2, borderStyle: "double", borderColor: "red", paddingX: 2, paddingY: 1 },
|
|
167
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
168
|
-
React.createElement(Gradient, { name: "rainbow" }, "Resume Conversation"),
|
|
169
|
-
React.createElement(Text, { color: "gray", dimColor: true }, "Select a conversation to resume"))),
|
|
170
|
-
React.createElement(Box, { marginBottom: 2 },
|
|
171
|
-
React.createElement(Alert, { variant: "error" }, error)),
|
|
172
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
173
|
-
React.createElement(Alert, { variant: "info" }, "Press Esc to return, or R to retry"))));
|
|
174
|
-
}
|
|
175
|
-
if (sessions.length === 0) {
|
|
176
|
-
return (React.createElement(Box, { flexDirection: "column", padding: 1 },
|
|
177
|
-
React.createElement(Box, { marginBottom: 2, borderStyle: "double", borderColor: "cyan", paddingX: 2, paddingY: 1 },
|
|
178
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
179
|
-
React.createElement(Gradient, { name: "rainbow" }, "Resume Conversation"),
|
|
180
|
-
React.createElement(Text, { color: "gray", dimColor: true }, "No conversations found"))),
|
|
181
|
-
React.createElement(Box, { marginBottom: 2 },
|
|
182
|
-
React.createElement(Alert, { variant: "info" }, "No previous conversations found. Start a new conversation to create your first session.")),
|
|
183
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
184
|
-
React.createElement(Alert, { variant: "info" }, "Press Esc to return to chat"))));
|
|
185
|
-
}
|
|
186
|
-
// Calculate available height for the list with safer bounds
|
|
187
|
-
const terminalHeight = stdout?.rows || 24;
|
|
188
|
-
const headerHeight = 7; // Header box height (including borders and padding)
|
|
189
|
-
const footerHeight = 4; // Footer info height (including margins)
|
|
190
|
-
const availableHeight = Math.max(5, terminalHeight - headerHeight - footerHeight);
|
|
191
|
-
const maxVisibleSessions = 10;
|
|
192
|
-
const desiredListSize = Math.max(3, Math.min(maxVisibleSessions, availableHeight));
|
|
193
|
-
const listLimit = Math.min(selectItems.length, desiredListSize);
|
|
194
|
-
const containerHeight = terminalHeight > 2 ? terminalHeight - 2 : undefined;
|
|
195
|
-
return (React.createElement(Box, { flexDirection: "column", padding: 1, height: containerHeight },
|
|
196
|
-
React.createElement(Box, { marginBottom: 1, borderStyle: "double", borderColor: "cyan", paddingX: 2, paddingY: 1 },
|
|
197
|
-
React.createElement(Box, { flexDirection: "column" },
|
|
198
|
-
React.createElement(Gradient, { name: "rainbow" }, "Resume Conversation"),
|
|
199
|
-
React.createElement(Text, { color: "gray", dimColor: true },
|
|
200
|
-
sessions.length,
|
|
201
|
-
" conversation",
|
|
202
|
-
sessions.length !== 1 ? 's' : '',
|
|
203
|
-
' ',
|
|
204
|
-
"available"))),
|
|
205
|
-
React.createElement(Box, { marginBottom: 1, flexShrink: 0 },
|
|
206
|
-
React.createElement(ScrollableSelectInput, { items: selectItems, limit: listLimit, onSelect: handleSelect, onToggleItem: handleToggleItem, onDeleteSelection: handleDeleteSelected, selectedValues: selectedSessions, indicator: ({ isSelected }) => (React.createElement(Text, { color: isSelected ? 'green' : 'gray' }, isSelected ? '❯ ' : ' ')), renderItem: ({ isSelected, isMarked, label }) => (React.createElement(Text, null,
|
|
207
|
-
React.createElement(Text, { color: isMarked ? 'green' : 'gray' }, isMarked ? '✔ ' : ' '),
|
|
208
|
-
React.createElement(Text, { color: isMarked ? 'green' : isSelected ? 'cyan' : 'white' }, label))) })),
|
|
209
|
-
React.createElement(Box, { flexDirection: "column", flexShrink: 0 },
|
|
210
|
-
actionMessage ? (React.createElement(Box, { marginBottom: 1 },
|
|
211
|
-
React.createElement(Alert, { variant: actionMessage.type }, actionMessage.text))) : null,
|
|
212
|
-
React.createElement(Alert, { variant: "info" },
|
|
213
|
-
"\u2191\u2193 navigate \u2022 Space mark \u2022 D delete \u2022 Enter select \u2022 Esc return \u2022 R refresh",
|
|
214
|
-
selectedSessions.size > 0
|
|
215
|
-
? ` • ${selectedSessions.size} selected`
|
|
216
|
-
: ''))));
|
|
217
|
-
}
|