@within-7/minto 0.1.7 → 0.2.0
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 +155 -37
- package/dist/Tool.js +38 -0
- package/dist/Tool.js.map +3 -3
- package/dist/commands/agents/AgentsCommand.js +52 -26
- package/dist/commands/agents/AgentsCommand.js.map +2 -2
- package/dist/commands/agents/constants.js +1 -1
- package/dist/commands/agents/constants.js.map +1 -1
- package/dist/commands/agents/index.js +1 -1
- package/dist/commands/bug.js +74 -7
- package/dist/commands/bug.js.map +3 -3
- package/dist/commands/clear.js +3 -0
- package/dist/commands/clear.js.map +2 -2
- package/dist/commands/compact.js +37 -0
- package/dist/commands/compact.js.map +2 -2
- package/dist/commands/context.js +84 -0
- package/dist/commands/context.js.map +7 -0
- package/dist/commands/ctx_viz.js +18 -10
- package/dist/commands/ctx_viz.js.map +2 -2
- package/dist/commands/doctor.js +158 -12
- package/dist/commands/doctor.js.map +2 -2
- package/dist/commands/export.js +156 -0
- package/dist/commands/export.js.map +7 -0
- package/dist/commands/mcp-interactive.js +21 -12
- package/dist/commands/mcp-interactive.js.map +2 -2
- package/dist/commands/model.js +6 -5
- package/dist/commands/model.js.map +2 -2
- package/dist/commands/permissions.js +86 -0
- package/dist/commands/permissions.js.map +7 -0
- package/dist/commands/quit.js +3 -1
- package/dist/commands/quit.js.map +2 -2
- package/dist/commands/sandbox.js +104 -0
- package/dist/commands/sandbox.js.map +7 -0
- package/dist/commands/status.js +58 -0
- package/dist/commands/status.js.map +7 -0
- package/dist/commands/tasks.js +108 -0
- package/dist/commands/tasks.js.map +7 -0
- package/dist/commands/todos.js +123 -0
- package/dist/commands/todos.js.map +7 -0
- package/dist/commands.js +20 -2
- package/dist/commands.js.map +2 -2
- package/dist/components/AgentThinkingBlock.js +10 -18
- package/dist/components/AgentThinkingBlock.js.map +2 -2
- package/dist/components/BackgroundTasksPanel.js +78 -29
- package/dist/components/BackgroundTasksPanel.js.map +2 -2
- package/dist/components/BashStreamingProgress.js +24 -0
- package/dist/components/BashStreamingProgress.js.map +7 -0
- package/dist/components/CollapsibleHint.js +14 -0
- package/dist/components/CollapsibleHint.js.map +7 -0
- package/dist/components/FileEditToolUpdatedMessage.js +1 -1
- package/dist/components/FileEditToolUpdatedMessage.js.map +2 -2
- package/dist/components/HotkeyHelpPanel.js +137 -0
- package/dist/components/HotkeyHelpPanel.js.map +7 -0
- package/dist/components/Logo.js +5 -5
- package/dist/components/Logo.js.map +2 -2
- package/dist/components/Message.js +23 -7
- package/dist/components/Message.js.map +3 -3
- package/dist/components/ModelConfig.js +16 -3
- package/dist/components/ModelConfig.js.map +2 -2
- package/dist/components/ModelListManager.js +3 -3
- package/dist/components/ModelListManager.js.map +2 -2
- package/dist/components/ModelSelector/ModelSelector.js +1 -1
- package/dist/components/Onboarding.js +19 -14
- package/dist/components/Onboarding.js.map +2 -2
- package/dist/components/ProgressBar.js +74 -0
- package/dist/components/ProgressBar.js.map +7 -0
- package/dist/components/PromptInput.js +156 -46
- package/dist/components/PromptInput.js.map +2 -2
- package/dist/components/RequestStatusIndicator.js +194 -0
- package/dist/components/RequestStatusIndicator.js.map +7 -0
- package/dist/components/Spinner.js +92 -27
- package/dist/components/Spinner.js.map +2 -2
- package/dist/components/SpinnerSymbol.js +21 -27
- package/dist/components/SpinnerSymbol.js.map +2 -2
- package/dist/components/StreamingBashOutput.js +9 -8
- package/dist/components/StreamingBashOutput.js.map +2 -2
- package/dist/components/SubagentBlock.js +1 -1
- package/dist/components/SubagentBlock.js.map +1 -1
- package/dist/components/SubagentProgress.js +10 -11
- package/dist/components/SubagentProgress.js.map +2 -2
- package/dist/components/TaskCard.js +16 -13
- package/dist/components/TaskCard.js.map +2 -2
- package/dist/components/TodoChangeBlock.js +1 -1
- package/dist/components/TodoChangeBlock.js.map +2 -2
- package/dist/components/TodoPanel.js +120 -29
- package/dist/components/TodoPanel.js.map +3 -3
- package/dist/components/TokenCounter.js +74 -0
- package/dist/components/TokenCounter.js.map +7 -0
- package/dist/components/TokenWarning.js +2 -1
- package/dist/components/TokenWarning.js.map +2 -2
- package/dist/components/TreeConnector.js +25 -0
- package/dist/components/TreeConnector.js.map +7 -0
- package/dist/components/TurnCompletionIndicator.js +18 -0
- package/dist/components/TurnCompletionIndicator.js.map +7 -0
- package/dist/components/messages/AssistantTextMessage.js +5 -2
- package/dist/components/messages/AssistantTextMessage.js.map +2 -2
- package/dist/components/messages/AssistantThinkingMessage.js +18 -3
- package/dist/components/messages/AssistantThinkingMessage.js.map +2 -2
- package/dist/components/messages/AssistantToolUseMessage.js +11 -8
- package/dist/components/messages/AssistantToolUseMessage.js.map +2 -2
- package/dist/components/messages/GroupRenderer.js +53 -0
- package/dist/components/messages/GroupRenderer.js.map +7 -0
- package/dist/components/messages/NestedTasksPreview.js +12 -0
- package/dist/components/messages/NestedTasksPreview.js.map +7 -0
- package/dist/components/messages/ParallelTasksGroupView.js +92 -0
- package/dist/components/messages/ParallelTasksGroupView.js.map +7 -0
- package/dist/components/messages/TaskInModuleView.js +198 -0
- package/dist/components/messages/TaskInModuleView.js.map +7 -0
- package/dist/components/messages/TaskOutputContent.js +53 -0
- package/dist/components/messages/TaskOutputContent.js.map +7 -0
- package/dist/components/messages/UserPromptMessage.js +1 -1
- package/dist/components/messages/UserPromptMessage.js.map +2 -2
- package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js +2 -3
- package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js.map +2 -2
- package/dist/components/permissions/FallbackPermissionRequest.js +4 -4
- package/dist/components/permissions/FallbackPermissionRequest.js.map +2 -2
- package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js +4 -4
- package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js.map +2 -2
- package/dist/constants/colors.js +48 -0
- package/dist/constants/colors.js.map +2 -2
- package/dist/constants/formatRules.js +102 -0
- package/dist/constants/formatRules.js.map +7 -0
- package/dist/constants/prompts.js +12 -34
- package/dist/constants/prompts.js.map +2 -2
- package/dist/constants/symbols.js +64 -6
- package/dist/constants/symbols.js.map +2 -2
- package/dist/constants/timing.js +5 -0
- package/dist/constants/timing.js.map +2 -2
- package/dist/core/config/defaults.js +84 -0
- package/dist/core/config/defaults.js.map +7 -0
- package/dist/core/config/index.js +111 -0
- package/dist/core/config/index.js.map +7 -0
- package/dist/core/config/loader.js +221 -0
- package/dist/core/config/loader.js.map +7 -0
- package/dist/core/config/migrations.js +128 -0
- package/dist/core/config/migrations.js.map +7 -0
- package/dist/core/config/schema.js +178 -0
- package/dist/core/config/schema.js.map +7 -0
- package/dist/core/costTracker.js +138 -0
- package/dist/core/costTracker.js.map +7 -0
- package/dist/core/index.js +5 -0
- package/dist/core/index.js.map +7 -0
- package/dist/core/permissions/auditLog.js +204 -0
- package/dist/core/permissions/auditLog.js.map +7 -0
- package/dist/core/permissions/engine/index.js +3 -0
- package/dist/core/permissions/engine/index.js.map +7 -0
- package/dist/core/permissions/engine/permissionEngine.js +106 -0
- package/dist/core/permissions/engine/permissionEngine.js.map +7 -0
- package/dist/core/permissions/engine/types.js +1 -0
- package/dist/core/permissions/engine/types.js.map +7 -0
- package/dist/core/permissions/index.js +84 -0
- package/dist/core/permissions/index.js.map +7 -0
- package/dist/core/permissions/ruleEngine.js +259 -0
- package/dist/core/permissions/ruleEngine.js.map +7 -0
- package/dist/core/permissions/rules/allowedToolsRule.js +62 -0
- package/dist/core/permissions/rules/allowedToolsRule.js.map +7 -0
- package/dist/core/permissions/rules/autoEscalationRule.js +291 -0
- package/dist/core/permissions/rules/autoEscalationRule.js.map +7 -0
- package/dist/core/permissions/rules/index.js +46 -0
- package/dist/core/permissions/rules/index.js.map +7 -0
- package/dist/core/permissions/rules/planModeRule.js +55 -0
- package/dist/core/permissions/rules/planModeRule.js.map +7 -0
- package/dist/core/permissions/rules/projectBoundaryRule.js +168 -0
- package/dist/core/permissions/rules/projectBoundaryRule.js.map +7 -0
- package/dist/core/permissions/rules/safeModeRule.js +65 -0
- package/dist/core/permissions/rules/safeModeRule.js.map +7 -0
- package/dist/core/permissions/rules/sensitivePathsRule.js +340 -0
- package/dist/core/permissions/rules/sensitivePathsRule.js.map +7 -0
- package/dist/core/permissions/types.js +127 -0
- package/dist/core/permissions/types.js.map +7 -0
- package/dist/core/tools/executor.js +143 -0
- package/dist/core/tools/executor.js.map +7 -0
- package/dist/core/tools/index.js +15 -0
- package/dist/core/tools/index.js.map +7 -0
- package/dist/core/tools/registry.js +183 -0
- package/dist/core/tools/registry.js.map +7 -0
- package/dist/core/tools/types.js +1 -0
- package/dist/core/tools/types.js.map +7 -0
- package/dist/cost-tracker.js +23 -15
- package/dist/cost-tracker.js.map +2 -2
- package/dist/entrypoints/cli.js +43 -43
- package/dist/entrypoints/cli.js.map +2 -2
- package/dist/entrypoints/mcp.js +12 -4
- package/dist/entrypoints/mcp.js.map +2 -2
- package/dist/history.js +14 -3
- package/dist/history.js.map +2 -2
- package/dist/hooks/useAgentTranscripts.js +116 -0
- package/dist/hooks/useAgentTranscripts.js.map +7 -0
- package/dist/hooks/useAnimationSync.js +53 -0
- package/dist/hooks/useAnimationSync.js.map +7 -0
- package/dist/hooks/useArrowKeyHistory.js +4 -2
- package/dist/hooks/useArrowKeyHistory.js.map +2 -2
- package/dist/hooks/useCanUseTool.js +3 -1
- package/dist/hooks/useCanUseTool.js.map +2 -2
- package/dist/hooks/useCancelRequest.js +4 -1
- package/dist/hooks/useCancelRequest.js.map +2 -2
- package/dist/hooks/useExitOnCtrlCD.js +9 -5
- package/dist/hooks/useExitOnCtrlCD.js.map +2 -2
- package/dist/hooks/useHookStatus.js +40 -0
- package/dist/hooks/useHookStatus.js.map +7 -0
- package/dist/hooks/useLogMessages.js +17 -1
- package/dist/hooks/useLogMessages.js.map +2 -2
- package/dist/hooks/useMessageGroups.js +43 -0
- package/dist/hooks/useMessageGroups.js.map +7 -0
- package/dist/hooks/useTerminalSize.js +62 -6
- package/dist/hooks/useTerminalSize.js.map +2 -2
- package/dist/hooks/useUnifiedCompletion.js +69 -0
- package/dist/hooks/useUnifiedCompletion.js.map +2 -2
- package/dist/i18n/index.js +109 -0
- package/dist/i18n/index.js.map +7 -0
- package/dist/i18n/locales/en.js +347 -0
- package/dist/i18n/locales/en.js.map +7 -0
- package/dist/i18n/locales/index.js +7 -0
- package/dist/i18n/locales/index.js.map +7 -0
- package/dist/i18n/locales/zh-CN.js +347 -0
- package/dist/i18n/locales/zh-CN.js.map +7 -0
- package/dist/i18n/types.js +8 -0
- package/dist/i18n/types.js.map +7 -0
- package/dist/query.js +175 -17
- package/dist/query.js.map +3 -3
- package/dist/screens/REPL.js +501 -192
- package/dist/screens/REPL.js.map +3 -3
- package/dist/services/adapters/chatCompletions.js +3 -1
- package/dist/services/adapters/chatCompletions.js.map +2 -2
- package/dist/services/adapters/messageNormalizer.js +354 -0
- package/dist/services/adapters/messageNormalizer.js.map +7 -0
- package/dist/services/adapters/responsesAPI.js +6 -3
- package/dist/services/adapters/responsesAPI.js.map +2 -2
- package/dist/services/checkpointManager.js +386 -0
- package/dist/services/checkpointManager.js.map +7 -0
- package/dist/services/claude.js +138 -11
- package/dist/services/claude.js.map +3 -3
- package/dist/services/compressionService.js +50 -1
- package/dist/services/compressionService.js.map +2 -2
- package/dist/services/contextMonitor.js +162 -0
- package/dist/services/contextMonitor.js.map +7 -0
- package/dist/services/customCommands.js +60 -41
- package/dist/services/customCommands.js.map +2 -2
- package/dist/services/hookExecutor.js +173 -1
- package/dist/services/hookExecutor.js.map +2 -2
- package/dist/services/intelligentCompactor.js +281 -0
- package/dist/services/intelligentCompactor.js.map +7 -0
- package/dist/services/lspConfig.js +109 -0
- package/dist/services/lspConfig.js.map +7 -0
- package/dist/services/mcpClient.js +273 -34
- package/dist/services/mcpClient.js.map +2 -2
- package/dist/services/modelOrchestrator.js +310 -0
- package/dist/services/modelOrchestrator.js.map +7 -0
- package/dist/services/openai.js +8 -1
- package/dist/services/openai.js.map +2 -2
- package/dist/services/outputStyles.js +138 -0
- package/dist/services/outputStyles.js.map +7 -0
- package/dist/services/plugins/index.js +5 -0
- package/dist/services/plugins/index.js.map +7 -0
- package/dist/services/plugins/lspServers.js +188 -0
- package/dist/services/plugins/lspServers.js.map +7 -0
- package/dist/services/plugins/pluginRuntime.js +229 -0
- package/dist/services/plugins/pluginRuntime.js.map +7 -0
- package/dist/services/plugins/pluginValidation.js +219 -0
- package/dist/services/plugins/pluginValidation.js.map +7 -0
- package/dist/services/plugins/skillMarketplace.js +556 -0
- package/dist/services/plugins/skillMarketplace.js.map +7 -0
- package/dist/services/responseStateManager.js +37 -3
- package/dist/services/responseStateManager.js.map +2 -2
- package/dist/services/sandbox/filesystemBoundary.js +300 -0
- package/dist/services/sandbox/filesystemBoundary.js.map +7 -0
- package/dist/services/sandbox/index.js +14 -0
- package/dist/services/sandbox/index.js.map +7 -0
- package/dist/services/sandbox/networkProxy.js +293 -0
- package/dist/services/sandbox/networkProxy.js.map +7 -0
- package/dist/services/sandbox/sandboxController.js +574 -0
- package/dist/services/sandbox/sandboxController.js.map +7 -0
- package/dist/services/sandbox/types.js +50 -0
- package/dist/services/sandbox/types.js.map +7 -0
- package/dist/services/sessionMemory.js +266 -0
- package/dist/services/sessionMemory.js.map +7 -0
- package/dist/services/taskRouter.js +324 -0
- package/dist/services/taskRouter.js.map +7 -0
- package/dist/tools/ArchitectTool/ArchitectTool.js +7 -1
- package/dist/tools/ArchitectTool/ArchitectTool.js.map +2 -2
- package/dist/tools/AskExpertModelTool/AskExpertModelTool.js +3 -0
- package/dist/tools/AskExpertModelTool/AskExpertModelTool.js.map +2 -2
- package/dist/tools/AskUserQuestionTool/AskUserQuestionTool.js.map +2 -2
- package/dist/tools/BaseTool.js +72 -0
- package/dist/tools/BaseTool.js.map +7 -0
- package/dist/tools/BashOutputTool/BashOutputToolResultMessage.js +3 -0
- package/dist/tools/BashOutputTool/BashOutputToolResultMessage.js.map +2 -2
- package/dist/tools/BashTool/BashTool.js +60 -3
- package/dist/tools/BashTool/BashTool.js.map +2 -2
- package/dist/tools/BashTool/BashToolResultMessage.js +3 -0
- package/dist/tools/BashTool/BashToolResultMessage.js.map +2 -2
- package/dist/tools/BashTool/OutputLine.js +54 -0
- package/dist/tools/BashTool/OutputLine.js.map +2 -2
- package/dist/tools/BashTool/prompt.js +192 -3
- package/dist/tools/BashTool/prompt.js.map +2 -2
- package/dist/tools/FileEditTool/FileEditTool.js +29 -4
- package/dist/tools/FileEditTool/FileEditTool.js.map +2 -2
- package/dist/tools/FileWriteTool/FileWriteTool.js +5 -5
- package/dist/tools/FileWriteTool/FileWriteTool.js.map +2 -2
- package/dist/tools/GlobTool/GlobTool.js +4 -2
- package/dist/tools/GlobTool/GlobTool.js.map +2 -2
- package/dist/tools/GrepTool/GrepTool.js +36 -7
- package/dist/tools/GrepTool/GrepTool.js.map +2 -2
- package/dist/tools/KillShellTool/KillShellToolResultMessage.js +3 -0
- package/dist/tools/KillShellTool/KillShellToolResultMessage.js.map +2 -2
- package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js +109 -0
- package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js.map +7 -0
- package/dist/tools/ListMcpResourcesTool/prompt.js +19 -0
- package/dist/tools/ListMcpResourcesTool/prompt.js.map +7 -0
- package/dist/tools/LspTool/LspTool.js +664 -0
- package/dist/tools/LspTool/LspTool.js.map +7 -0
- package/dist/tools/LspTool/prompt.js +27 -0
- package/dist/tools/LspTool/prompt.js.map +7 -0
- package/dist/tools/MCPTool/MCPTool.js +9 -1
- package/dist/tools/MCPTool/MCPTool.js.map +2 -2
- package/dist/tools/MemoryReadTool/MemoryReadTool.js +19 -6
- package/dist/tools/MemoryReadTool/MemoryReadTool.js.map +2 -2
- package/dist/tools/MemoryWriteTool/MemoryWriteTool.js +6 -6
- package/dist/tools/MemoryWriteTool/MemoryWriteTool.js.map +2 -2
- package/dist/tools/MultiEditTool/MultiEditTool.js +19 -2
- package/dist/tools/MultiEditTool/MultiEditTool.js.map +2 -2
- package/dist/tools/NotebookEditTool/NotebookEditTool.js +5 -1
- package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +2 -2
- package/dist/tools/NotebookReadTool/NotebookReadTool.js.map +2 -2
- package/dist/tools/PlanModeTool/EnterPlanModeTool.js +74 -0
- package/dist/tools/PlanModeTool/EnterPlanModeTool.js.map +7 -0
- package/dist/tools/PlanModeTool/ExitPlanModeTool.js +108 -0
- package/dist/tools/PlanModeTool/ExitPlanModeTool.js.map +7 -0
- package/dist/tools/PlanModeTool/prompt.js +94 -0
- package/dist/tools/PlanModeTool/prompt.js.map +7 -0
- package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js +130 -0
- package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js.map +7 -0
- package/dist/tools/ReadMcpResourceTool/prompt.js +17 -0
- package/dist/tools/ReadMcpResourceTool/prompt.js.map +7 -0
- package/dist/tools/SkillTool/SkillTool.js +6 -1
- package/dist/tools/SkillTool/SkillTool.js.map +2 -2
- package/dist/tools/SlashCommandTool/SlashCommandTool.js +260 -0
- package/dist/tools/SlashCommandTool/SlashCommandTool.js.map +7 -0
- package/dist/tools/SlashCommandTool/prompt.js +35 -0
- package/dist/tools/SlashCommandTool/prompt.js.map +7 -0
- package/dist/tools/TaskOutputTool/TaskOutputTool.js +189 -0
- package/dist/tools/TaskOutputTool/TaskOutputTool.js.map +7 -0
- package/dist/tools/TaskOutputTool/prompt.js +15 -0
- package/dist/tools/TaskOutputTool/prompt.js.map +7 -0
- package/dist/tools/TaskTool/TaskTool.js +302 -104
- package/dist/tools/TaskTool/TaskTool.js.map +2 -2
- package/dist/tools/TaskTool/prompt.js.map +2 -2
- package/dist/tools/TodoWriteTool/TodoWriteTool.js +42 -77
- package/dist/tools/TodoWriteTool/TodoWriteTool.js.map +2 -2
- package/dist/tools/URLFetcherTool/URLFetcherTool.js +4 -1
- package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +2 -2
- package/dist/tools/URLFetcherTool/cache.js +55 -8
- package/dist/tools/URLFetcherTool/cache.js.map +2 -2
- package/dist/tools.js +31 -2
- package/dist/tools.js.map +2 -2
- package/dist/types/hooks.js +4 -0
- package/dist/types/hooks.js.map +2 -2
- package/dist/types/marketplace.js.map +2 -2
- package/dist/types/messageGroup.js +36 -0
- package/dist/types/messageGroup.js.map +7 -0
- package/dist/types/plugin.js.map +2 -2
- package/dist/types/thinking.js +1 -0
- package/dist/types/thinking.js.map +7 -0
- package/dist/utils/BackgroundShellManager.js +136 -39
- package/dist/utils/BackgroundShellManager.js.map +2 -2
- package/dist/utils/MessageBatchBuffer.js +102 -0
- package/dist/utils/MessageBatchBuffer.js.map +7 -0
- package/dist/utils/PersistentShell.js +151 -1
- package/dist/utils/PersistentShell.js.map +2 -2
- package/dist/utils/agentLoader.js +1 -23
- package/dist/utils/agentLoader.js.map +2 -2
- package/dist/utils/agentTranscripts.js +641 -0
- package/dist/utils/agentTranscripts.js.map +7 -0
- package/dist/utils/animationManager.js +213 -0
- package/dist/utils/animationManager.js.map +7 -0
- package/dist/utils/animationSync.js +110 -0
- package/dist/utils/animationSync.js.map +7 -0
- package/dist/utils/asyncFile.js +215 -0
- package/dist/utils/asyncFile.js.map +7 -0
- package/dist/utils/backgroundAgentManager.js +231 -0
- package/dist/utils/backgroundAgentManager.js.map +7 -0
- package/dist/utils/config.js +63 -7
- package/dist/utils/config.js.map +2 -2
- package/dist/utils/conversationRecovery.js +19 -0
- package/dist/utils/conversationRecovery.js.map +2 -2
- package/dist/utils/exit.js +73 -0
- package/dist/utils/exit.js.map +7 -0
- package/dist/utils/format.js +73 -5
- package/dist/utils/format.js.map +2 -2
- package/dist/utils/generators.js +76 -6
- package/dist/utils/generators.js.map +2 -2
- package/dist/utils/globalErrorHandler.js +149 -0
- package/dist/utils/globalErrorHandler.js.map +7 -0
- package/dist/utils/groupHandlers/index.js +8 -0
- package/dist/utils/groupHandlers/index.js.map +7 -0
- package/dist/utils/groupHandlers/parallelTasksHandler.js +140 -0
- package/dist/utils/groupHandlers/parallelTasksHandler.js.map +7 -0
- package/dist/utils/groupHandlers/taskHandler.js +104 -0
- package/dist/utils/groupHandlers/taskHandler.js.map +7 -0
- package/dist/utils/groupHandlers/types.js +1 -0
- package/dist/utils/groupHandlers/types.js.map +7 -0
- package/dist/utils/logRotation.js +224 -0
- package/dist/utils/logRotation.js.map +7 -0
- package/dist/utils/marketplaceManager.js +3 -5
- package/dist/utils/marketplaceManager.js.map +2 -2
- package/dist/utils/memSafety.js +264 -0
- package/dist/utils/memSafety.js.map +7 -0
- package/dist/utils/messageGroupManager.js +274 -0
- package/dist/utils/messageGroupManager.js.map +7 -0
- package/dist/utils/messages.js +13 -4
- package/dist/utils/messages.js.map +2 -2
- package/dist/utils/model.js +119 -15
- package/dist/utils/model.js.map +3 -3
- package/dist/utils/permissions/filesystem.js +157 -5
- package/dist/utils/permissions/filesystem.js.map +2 -2
- package/dist/utils/plan/planMode.js +143 -0
- package/dist/utils/plan/planMode.js.map +7 -0
- package/dist/utils/pluginLoader.js +17 -21
- package/dist/utils/pluginLoader.js.map +2 -2
- package/dist/utils/ripgrep.js +55 -2
- package/dist/utils/ripgrep.js.map +2 -2
- package/dist/utils/sanitizeInput.js +32 -0
- package/dist/utils/sanitizeInput.js.map +7 -0
- package/dist/utils/secureKeyStorage.js +312 -0
- package/dist/utils/secureKeyStorage.js.map +7 -0
- package/dist/utils/session/sessionPlugins.js +67 -0
- package/dist/utils/session/sessionPlugins.js.map +7 -0
- package/dist/utils/taskDisplayUtils.js +257 -0
- package/dist/utils/taskDisplayUtils.js.map +7 -0
- package/dist/utils/teamConfig.js +2 -1
- package/dist/utils/teamConfig.js.map +2 -2
- package/dist/utils/todoStorage.js +92 -2
- package/dist/utils/todoStorage.js.map +2 -2
- package/dist/utils/toolTimeout.js +136 -0
- package/dist/utils/toolTimeout.js.map +7 -0
- package/dist/utils/tooling/safeRender.js +115 -0
- package/dist/utils/tooling/safeRender.js.map +7 -0
- package/dist/utils/userFriendlyError.js +346 -0
- package/dist/utils/userFriendlyError.js.map +7 -0
- package/dist/utils/vendor/ripgrep/arm64-darwin/rg +0 -0
- package/dist/version.js +2 -2
- package/dist/version.js.map +1 -1
- package/package.json +14 -4
- package/scripts/postinstall.js +128 -38
- package/dist/commands/agents.js +0 -2086
- package/dist/commands/agents.js.map +0 -7
- package/dist/commands/build.js +0 -74
- package/dist/commands/build.js.map +0 -7
- package/dist/commands/compression.js +0 -57
- package/dist/commands/compression.js.map +0 -7
- package/dist/commands/listen.js +0 -37
- package/dist/commands/listen.js.map +0 -7
- package/dist/commands/login.js +0 -37
- package/dist/commands/login.js.map +0 -7
- package/dist/commands/logout.js +0 -33
- package/dist/commands/logout.js.map +0 -7
- package/dist/commands/mcp.js +0 -40
- package/dist/commands/mcp.js.map +0 -7
- package/dist/commands/mcp_refresh.js +0 -40
- package/dist/commands/mcp_refresh.js.map +0 -7
- package/dist/commands/modelstatus.js +0 -21
- package/dist/commands/modelstatus.js.map +0 -7
- package/dist/commands/onboarding.js +0 -36
- package/dist/commands/onboarding.js.map +0 -7
- package/dist/commands/plugin-interactive.js +0 -446
- package/dist/commands/plugin-interactive.js.map +0 -7
- package/dist/commands/pr_comments.js +0 -61
- package/dist/commands/pr_comments.js.map +0 -7
- package/dist/commands/release-notes.js +0 -30
- package/dist/commands/release-notes.js.map +0 -7
- package/dist/commands/review.js +0 -51
- package/dist/commands/review.js.map +0 -7
- package/dist/components/Bug.js +0 -147
- package/dist/components/Bug.js.map +0 -7
- package/dist/components/ModelSelector.js +0 -2062
- package/dist/components/ModelSelector.js.map +0 -7
- package/dist/components/ModelStatusDisplay.js +0 -87
- package/dist/components/ModelStatusDisplay.js.map +0 -7
- package/dist/entrypoints/cli-wrapper.js +0 -61
- package/dist/entrypoints/cli-wrapper.js.map +0 -7
- package/dist/screens/Doctor.js +0 -22
- package/dist/screens/Doctor.js.map +0 -7
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/hooks/useTerminalSize.ts"],
|
|
4
|
-
"sourcesContent": ["import { useEffect, useState } from 'react'\n\n// Global state to share across all hook instances\nlet globalSize = {\n columns: process.stdout.columns || 80,\n rows: process.stdout.rows || 24,\n}\n\nconst listeners = new Set<() => void>()\nlet isListenerAttached = false\n\nfunction updateAllListeners() {\n
|
|
5
|
-
"mappings": "AAAA,SAAS,WAAW,
|
|
4
|
+
"sourcesContent": ["import { useEffect, useState, useRef, useCallback } from 'react'\n\n// Global state to share across all hook instances\nlet globalSize = {\n columns: process.stdout.columns || 80,\n rows: process.stdout.rows || 24,\n}\n\nconst listeners = new Set<() => void>()\nlet isListenerAttached = false\nlet debounceTimer: NodeJS.Timeout | null = null\n\n// Callbacks to trigger full re-render when significant resize occurs\nconst resizeCallbacks = new Set<() => void>()\n\n// Debounce delay in ms - prevents rapid-fire re-renders during resize\n// Increased from 150ms to 300ms to better handle window focus transitions\nconst RESIZE_DEBOUNCE_MS = 300\n\n// Threshold for detecting any significant size change (either direction)\n// Any resize > 5 columns should trigger a re-render to prevent ghost artifacts\nconst SIZE_CHANGE_THRESHOLD = 5\n\n// Track if we're currently in a fullscreen exit clearing cycle to prevent re-triggers\nlet isInClearingCycle = false\n\n// Track last significant resize time to prevent rapid re-triggers (e.g., from window focus changes)\nlet lastSignificantResizeTime = 0\nconst RESIZE_COOLDOWN_MS = 2000 // 2 second cooldown between significant resize triggers\n\nfunction updateAllListeners() {\n // Clear any pending debounce\n if (debounceTimer) {\n clearTimeout(debounceTimer)\n }\n\n // Debounce the resize event to prevent multiple rapid re-renders\n debounceTimer = setTimeout(() => {\n const newColumns = process.stdout.columns || 80\n const newRows = process.stdout.rows || 24\n\n // Only update if dimensions actually changed\n if (globalSize.columns !== newColumns || globalSize.rows !== newRows) {\n const oldColumns = globalSize.columns\n\n // Detect if screen size changed significantly (in either direction)\n // This catches both entering and exiting fullscreen, as well as manual window resizing\n const columnDiff = Math.abs(oldColumns - newColumns)\n const isSignificantResize = columnDiff > SIZE_CHANGE_THRESHOLD\n\n globalSize = {\n columns: newColumns,\n rows: newRows,\n }\n listeners.forEach(listener => listener())\n\n // If significant resize and not already in clearing cycle, trigger callbacks\n // The clearing cycle lock prevents multiple triggers during the clear/re-render process\n // Also enforce a cooldown period to prevent rapid re-triggers from window focus changes\n const now = Date.now()\n const timeSinceLastResize = now - lastSignificantResizeTime\n const isCooldownExpired = timeSinceLastResize > RESIZE_COOLDOWN_MS\n\n if (isSignificantResize && !isInClearingCycle && isCooldownExpired) {\n isInClearingCycle = true\n lastSignificantResizeTime = now\n resizeCallbacks.forEach(callback => callback())\n // Release the lock after a delay to allow re-render to complete\n setTimeout(() => {\n isInClearingCycle = false\n }, 500)\n }\n }\n debounceTimer = null\n }, RESIZE_DEBOUNCE_MS)\n}\n\nexport function useTerminalSize() {\n const [size, setSize] = useState(globalSize)\n\n useEffect(() => {\n // Add this component's listener to the set\n const updateSize = () => setSize({ ...globalSize })\n listeners.add(updateSize)\n\n // Only attach the global resize listener once\n if (!isListenerAttached) {\n // Increase max listeners to prevent warnings\n process.stdout.setMaxListeners(20)\n process.stdout.on('resize', updateAllListeners)\n isListenerAttached = true\n }\n\n return () => {\n // Remove this component's listener\n listeners.delete(updateSize)\n\n // If no more listeners, remove the global listener\n if (listeners.size === 0 && isListenerAttached) {\n process.stdout.off('resize', updateAllListeners)\n isListenerAttached = false\n // Clear any pending debounce\n if (debounceTimer) {\n clearTimeout(debounceTimer)\n debounceTimer = null\n }\n }\n }\n }, [])\n\n return size\n}\n\n/**\n * Hook to register a callback that fires when a significant resize occurs.\n * This is detected when the terminal width changes by more than 5 columns.\n * Useful for triggering a full UI re-render to clear ghost artifacts.\n *\n * @deprecated Use useSignificantResizeCallback instead (same function, clearer name)\n */\nexport function useFullscreenExitCallback(callback: () => void) {\n return useSignificantResizeCallback(callback)\n}\n\n/**\n * Hook to register a callback that fires when a significant terminal resize occurs.\n * This is detected when the terminal width changes by more than SIZE_CHANGE_THRESHOLD columns.\n * Useful for triggering a full UI re-render to clear ghost artifacts caused by Ink's\n * differential rendering.\n */\nexport function useSignificantResizeCallback(callback: () => void) {\n const callbackRef = useRef(callback)\n callbackRef.current = callback\n\n useEffect(() => {\n // Ensure the global resize listener is attached\n // This is needed because this hook might be called\n // before any component calls useTerminalSize()\n if (!isListenerAttached) {\n process.stdout.setMaxListeners(20)\n process.stdout.on('resize', updateAllListeners)\n isListenerAttached = true\n }\n\n const wrappedCallback = () => callbackRef.current()\n resizeCallbacks.add(wrappedCallback)\n\n return () => {\n resizeCallbacks.delete(wrappedCallback)\n }\n }, [])\n}\n"],
|
|
5
|
+
"mappings": "AAAA,SAAS,WAAW,UAAU,cAA2B;AAGzD,IAAI,aAAa;AAAA,EACf,SAAS,QAAQ,OAAO,WAAW;AAAA,EACnC,MAAM,QAAQ,OAAO,QAAQ;AAC/B;AAEA,MAAM,YAAY,oBAAI,IAAgB;AACtC,IAAI,qBAAqB;AACzB,IAAI,gBAAuC;AAG3C,MAAM,kBAAkB,oBAAI,IAAgB;AAI5C,MAAM,qBAAqB;AAI3B,MAAM,wBAAwB;AAG9B,IAAI,oBAAoB;AAGxB,IAAI,4BAA4B;AAChC,MAAM,qBAAqB;AAE3B,SAAS,qBAAqB;AAE5B,MAAI,eAAe;AACjB,iBAAa,aAAa;AAAA,EAC5B;AAGA,kBAAgB,WAAW,MAAM;AAC/B,UAAM,aAAa,QAAQ,OAAO,WAAW;AAC7C,UAAM,UAAU,QAAQ,OAAO,QAAQ;AAGvC,QAAI,WAAW,YAAY,cAAc,WAAW,SAAS,SAAS;AACpE,YAAM,aAAa,WAAW;AAI9B,YAAM,aAAa,KAAK,IAAI,aAAa,UAAU;AACnD,YAAM,sBAAsB,aAAa;AAEzC,mBAAa;AAAA,QACX,SAAS;AAAA,QACT,MAAM;AAAA,MACR;AACA,gBAAU,QAAQ,cAAY,SAAS,CAAC;AAKxC,YAAM,MAAM,KAAK,IAAI;AACrB,YAAM,sBAAsB,MAAM;AAClC,YAAM,oBAAoB,sBAAsB;AAEhD,UAAI,uBAAuB,CAAC,qBAAqB,mBAAmB;AAClE,4BAAoB;AACpB,oCAA4B;AAC5B,wBAAgB,QAAQ,cAAY,SAAS,CAAC;AAE9C,mBAAW,MAAM;AACf,8BAAoB;AAAA,QACtB,GAAG,GAAG;AAAA,MACR;AAAA,IACF;AACA,oBAAgB;AAAA,EAClB,GAAG,kBAAkB;AACvB;AAEO,SAAS,kBAAkB;AAChC,QAAM,CAAC,MAAM,OAAO,IAAI,SAAS,UAAU;AAE3C,YAAU,MAAM;AAEd,UAAM,aAAa,MAAM,QAAQ,EAAE,GAAG,WAAW,CAAC;AAClD,cAAU,IAAI,UAAU;AAGxB,QAAI,CAAC,oBAAoB;AAEvB,cAAQ,OAAO,gBAAgB,EAAE;AACjC,cAAQ,OAAO,GAAG,UAAU,kBAAkB;AAC9C,2BAAqB;AAAA,IACvB;AAEA,WAAO,MAAM;AAEX,gBAAU,OAAO,UAAU;AAG3B,UAAI,UAAU,SAAS,KAAK,oBAAoB;AAC9C,gBAAQ,OAAO,IAAI,UAAU,kBAAkB;AAC/C,6BAAqB;AAErB,YAAI,eAAe;AACjB,uBAAa,aAAa;AAC1B,0BAAgB;AAAA,QAClB;AAAA,MACF;AAAA,IACF;AAAA,EACF,GAAG,CAAC,CAAC;AAEL,SAAO;AACT;AASO,SAAS,0BAA0B,UAAsB;AAC9D,SAAO,6BAA6B,QAAQ;AAC9C;AAQO,SAAS,6BAA6B,UAAsB;AACjE,QAAM,cAAc,OAAO,QAAQ;AACnC,cAAY,UAAU;AAEtB,YAAU,MAAM;AAId,QAAI,CAAC,oBAAoB;AACvB,cAAQ,OAAO,gBAAgB,EAAE;AACjC,cAAQ,OAAO,GAAG,UAAU,kBAAkB;AAC9C,2BAAqB;AAAA,IACvB;AAEA,UAAM,kBAAkB,MAAM,YAAY,QAAQ;AAClD,oBAAgB,IAAI,eAAe;AAEnC,WAAO,MAAM;AACX,sBAAgB,OAAO,eAAe;AAAA,IACxC;AAAA,EACF,GAAG,CAAC,CAAC;AACP;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -12,6 +12,7 @@ import {
|
|
|
12
12
|
getEssentialCommands,
|
|
13
13
|
getMinimalFallbackCommands
|
|
14
14
|
} from "../utils/commonUnixCommands.js";
|
|
15
|
+
import { listMCPResources } from "../services/mcpClient.js";
|
|
15
16
|
const INITIAL_STATE = {
|
|
16
17
|
suggestions: [],
|
|
17
18
|
selectedIndex: 0,
|
|
@@ -143,6 +144,15 @@ function useUnifiedCompletion({
|
|
|
143
144
|
// Use cursor position as end
|
|
144
145
|
};
|
|
145
146
|
}
|
|
147
|
+
if (word.startsWith("mcp:") || word.includes("mcp:")) {
|
|
148
|
+
const mcpPrefix = word.includes("mcp:") ? word.slice(word.indexOf("mcp:") + 4) : "";
|
|
149
|
+
return {
|
|
150
|
+
type: "mcp",
|
|
151
|
+
prefix: mcpPrefix,
|
|
152
|
+
startPos: start,
|
|
153
|
+
endPos: cursorOffset
|
|
154
|
+
};
|
|
155
|
+
}
|
|
146
156
|
return {
|
|
147
157
|
type: "file",
|
|
148
158
|
prefix: word,
|
|
@@ -356,6 +366,29 @@ function useUnifiedCompletion({
|
|
|
356
366
|
setModelSuggestions([]);
|
|
357
367
|
}
|
|
358
368
|
}, []);
|
|
369
|
+
const [mcpResourceSuggestions, setMcpResourceSuggestions] = useState([]);
|
|
370
|
+
useEffect(() => {
|
|
371
|
+
listMCPResources().then((resources) => {
|
|
372
|
+
const suggestions2 = resources.map((resource) => ({
|
|
373
|
+
value: resource.uri,
|
|
374
|
+
displayValue: `\u{1F50C} ${resource.name} :: ${resource.description || resource.uri}`,
|
|
375
|
+
type: "mcp-resource",
|
|
376
|
+
score: 80,
|
|
377
|
+
metadata: {
|
|
378
|
+
uri: resource.uri,
|
|
379
|
+
serverName: resource.serverName,
|
|
380
|
+
mimeType: resource.mimeType
|
|
381
|
+
}
|
|
382
|
+
}));
|
|
383
|
+
setMcpResourceSuggestions(suggestions2);
|
|
384
|
+
}).catch((error) => {
|
|
385
|
+
console.warn(
|
|
386
|
+
"[useUnifiedCompletion] MCP resources not available:",
|
|
387
|
+
error
|
|
388
|
+
);
|
|
389
|
+
setMcpResourceSuggestions([]);
|
|
390
|
+
});
|
|
391
|
+
}, []);
|
|
359
392
|
useEffect(() => {
|
|
360
393
|
getActiveAgents().then((agents) => {
|
|
361
394
|
const suggestions2 = agents.map((config) => {
|
|
@@ -583,11 +616,42 @@ function useUnifiedCompletion({
|
|
|
583
616
|
},
|
|
584
617
|
[agentSuggestions, modelSuggestions, calculateMatchScore]
|
|
585
618
|
);
|
|
619
|
+
const generateMcpResourceSuggestions = useCallback(
|
|
620
|
+
(prefix) => {
|
|
621
|
+
if (mcpResourceSuggestions.length === 0) {
|
|
622
|
+
return [
|
|
623
|
+
{
|
|
624
|
+
value: "",
|
|
625
|
+
displayValue: "\u26A0\uFE0F No MCP resources available",
|
|
626
|
+
type: "mcp-resource",
|
|
627
|
+
score: 0,
|
|
628
|
+
metadata: { isPlaceholder: true }
|
|
629
|
+
}
|
|
630
|
+
];
|
|
631
|
+
}
|
|
632
|
+
if (!prefix) {
|
|
633
|
+
return mcpResourceSuggestions;
|
|
634
|
+
}
|
|
635
|
+
const lowerPrefix = prefix.toLowerCase();
|
|
636
|
+
return mcpResourceSuggestions.filter((s) => {
|
|
637
|
+
const uri = s.value.toLowerCase();
|
|
638
|
+
const name = s.metadata?.uri?.toLowerCase() || "";
|
|
639
|
+
const server = s.metadata?.serverName?.toLowerCase() || "";
|
|
640
|
+
return uri.includes(lowerPrefix) || name.includes(lowerPrefix) || server.includes(lowerPrefix);
|
|
641
|
+
}).map((s) => ({
|
|
642
|
+
...s,
|
|
643
|
+
score: s.value.toLowerCase().startsWith(lowerPrefix) ? 100 : 80
|
|
644
|
+
})).sort((a, b) => b.score - a.score);
|
|
645
|
+
},
|
|
646
|
+
[mcpResourceSuggestions]
|
|
647
|
+
);
|
|
586
648
|
const generateSuggestions = useCallback(
|
|
587
649
|
(context) => {
|
|
588
650
|
switch (context.type) {
|
|
589
651
|
case "command":
|
|
590
652
|
return generateCommandSuggestions(context.prefix);
|
|
653
|
+
case "mcp":
|
|
654
|
+
return generateMcpResourceSuggestions(context.prefix);
|
|
591
655
|
case "agent": {
|
|
592
656
|
const mentionSuggestions = generateMentionSuggestions(context.prefix);
|
|
593
657
|
const fileSuggestions = generateFileSuggestions(context.prefix, true);
|
|
@@ -653,6 +717,7 @@ function useUnifiedCompletion({
|
|
|
653
717
|
},
|
|
654
718
|
[
|
|
655
719
|
generateCommandSuggestions,
|
|
720
|
+
generateMcpResourceSuggestions,
|
|
656
721
|
generateMentionSuggestions,
|
|
657
722
|
generateFileSuggestions,
|
|
658
723
|
generateUnixCommandSuggestions,
|
|
@@ -664,6 +729,8 @@ function useUnifiedCompletion({
|
|
|
664
729
|
let completion;
|
|
665
730
|
if (context.type === "command") {
|
|
666
731
|
completion = `/${suggestion.value} `;
|
|
732
|
+
} else if (context.type === "mcp") {
|
|
733
|
+
completion = `mcp:${suggestion.value} `;
|
|
667
734
|
} else if (context.type === "agent") {
|
|
668
735
|
if (suggestion.type === "agent") {
|
|
669
736
|
completion = `@${suggestion.value} `;
|
|
@@ -971,6 +1038,8 @@ function useUnifiedCompletion({
|
|
|
971
1038
|
return true;
|
|
972
1039
|
case "agent":
|
|
973
1040
|
return true;
|
|
1041
|
+
case "mcp":
|
|
1042
|
+
return true;
|
|
974
1043
|
case "file":
|
|
975
1044
|
const prefix = context.prefix;
|
|
976
1045
|
if (prefix.startsWith("./") || prefix.startsWith("../") || prefix.startsWith("/") || prefix.startsWith("~") || prefix.includes("/")) {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/hooks/useUnifiedCompletion.ts"],
|
|
4
|
-
"sourcesContent": ["import { useState, useCallback, useEffect, useRef } from 'react'\nimport { useInput } from 'ink'\nimport { existsSync, statSync, readdirSync } from 'fs'\nimport { join, dirname, basename, resolve } from 'path'\nimport { getCwd } from '@utils/state'\nimport { getCommand } from '@commands'\nimport { getActiveAgents } from '@utils/agentLoader'\nimport { getModelManager } from '@utils/model'\nimport { glob } from 'glob'\nimport { matchCommands } from '@utils/fuzzyMatcher'\nimport {\n getCommonSystemCommands,\n getCommandPriority,\n getEssentialCommands,\n getMinimalFallbackCommands,\n} from '@utils/commonUnixCommands'\nimport type { Command } from '@commands'\n\n// Unified suggestion type for all completion types\nexport interface UnifiedSuggestion {\n value: string\n displayValue: string\n type: 'command' | 'agent' | 'file' | 'ask'\n icon?: string\n score: number\n metadata?: any\n // Clean type system for smart matching\n isSmartMatch?: boolean // Instead of magic string checking\n originalContext?: 'mention' | 'file' | 'command' // Track source context\n}\n\ninterface CompletionContext {\n type: 'command' | 'agent' | 'file' | null\n prefix: string\n startPos: number\n endPos: number\n}\n\n// Terminal behavior state for preview and cycling\ninterface TerminalState {\n originalWord: string\n wordContext: { start: number; end: number } | null\n isPreviewMode: boolean\n}\n\ninterface Props {\n input: string\n cursorOffset: number\n onInputChange: (value: string) => void\n setCursorOffset: (offset: number) => void\n commands: Command[]\n onSubmit?: (value: string, isSubmittingSlashCommand?: boolean) => void\n}\n\n/**\n * Unified completion system - Linus approved\n * One hook to rule them all, no bullshit, no complexity\n */\n// Unified completion state - single source of truth\ninterface CompletionState {\n suggestions: UnifiedSuggestion[]\n selectedIndex: number\n isActive: boolean\n context: CompletionContext | null\n preview: {\n isActive: boolean\n originalInput: string\n wordRange: [number, number]\n } | null\n emptyDirMessage: string\n suppressUntil: number // timestamp for suppression\n}\n\nconst INITIAL_STATE: CompletionState = {\n suggestions: [],\n selectedIndex: 0,\n isActive: false,\n context: null,\n preview: null,\n emptyDirMessage: '',\n suppressUntil: 0,\n}\n\nexport function useUnifiedCompletion({\n input,\n cursorOffset,\n onInputChange,\n setCursorOffset,\n commands,\n onSubmit,\n}: Props) {\n // Single state for entire completion system - Linus approved\n const [state, setState] = useState<CompletionState>(INITIAL_STATE)\n\n // State update helpers - clean and simple\n const updateState = useCallback((updates: Partial<CompletionState>) => {\n setState(prev => ({ ...prev, ...updates }))\n }, [])\n\n const resetCompletion = useCallback(() => {\n setState(prev => ({\n ...prev,\n suggestions: [],\n selectedIndex: 0,\n isActive: false,\n context: null,\n preview: null,\n emptyDirMessage: '',\n }))\n }, [])\n\n const activateCompletion = useCallback(\n (suggestions: UnifiedSuggestion[], context: CompletionContext) => {\n setState(prev => ({\n ...prev,\n suggestions: suggestions, // Keep the order from generateSuggestions (already sorted with weights)\n selectedIndex: 0,\n isActive: true,\n context,\n preview: null,\n }))\n },\n [],\n )\n\n // Direct state access - no legacy wrappers needed\n const { suggestions, selectedIndex, isActive, emptyDirMessage } = state\n\n // Find common prefix among suggestions (terminal behavior)\n const findCommonPrefix = useCallback(\n (suggestions: UnifiedSuggestion[]): string => {\n if (suggestions.length === 0) return ''\n if (suggestions.length === 1) return suggestions[0].value\n\n let prefix = suggestions[0].value\n\n for (let i = 1; i < suggestions.length; i++) {\n const str = suggestions[i].value\n let j = 0\n while (j < prefix.length && j < str.length && prefix[j] === str[j]) {\n j++\n }\n prefix = prefix.slice(0, j)\n\n if (prefix.length === 0) return ''\n }\n\n return prefix\n },\n [],\n )\n\n // Clean word detection - Linus approved simplicity\n const getWordAtCursor = useCallback((): CompletionContext | null => {\n if (!input) return null\n\n // IMPORTANT: Only match the word/prefix BEFORE the cursor\n // Don't include text after cursor to avoid confusion\n let start = cursorOffset\n\n // Move start backwards to find word beginning\n // Stop at whitespace or special boundaries\n while (start > 0) {\n const char = input[start - 1]\n // Stop at whitespace\n if (/\\s/.test(char)) break\n\n // For @mentions, include @ and stop\n if (char === '@' && start < cursorOffset) {\n start--\n break\n }\n\n // For paths, be smarter about / handling\n if (char === '/') {\n // Look ahead to see what we've collected so far\n const collectedSoFar = input.slice(start, cursorOffset)\n\n // If we already have a path component, this / is part of the path\n if (collectedSoFar.includes('/') || collectedSoFar.includes('.')) {\n start--\n continue\n }\n\n // Check if this is part of a path pattern like ./ or ../ or ~/\n if (start > 1) {\n const prevChar = input[start - 2]\n if (prevChar === '.' || prevChar === '~') {\n // It's part of ./ or ../ or ~/ - keep going\n start--\n continue\n }\n }\n\n // Check if this is a standalone / at the beginning (command)\n if (start === 1 || (start > 1 && /\\s/.test(input[start - 2]))) {\n start--\n break // It's a command slash\n }\n\n // Otherwise treat as path separator\n start--\n continue\n }\n\n // Special handling for dots in paths\n if (char === '.' && start > 0) {\n // Check if this might be start of ./ or ../\n const nextChar = start < input.length ? input[start] : ''\n if (nextChar === '/' || nextChar === '.') {\n start--\n continue // Part of a path pattern\n }\n }\n\n start--\n }\n\n // The word is from start to cursor position (not beyond)\n const word = input.slice(start, cursorOffset)\n if (!word) return null\n\n // Priority-based type detection - no special cases needed\n if (word.startsWith('/')) {\n const beforeWord = input.slice(0, start).trim()\n const isCommand = beforeWord === '' && !word.includes('/', 1)\n return {\n type: isCommand ? 'command' : 'file',\n prefix: isCommand ? word.slice(1) : word,\n startPos: start,\n endPos: cursorOffset, // Use cursor position as end\n }\n }\n\n if (word.startsWith('@')) {\n const content = word.slice(1) // Remove @\n\n // Check if this looks like an email (contains @ in the middle)\n if (word.includes('@', 1)) {\n // This looks like an email, treat as regular text\n return null\n }\n\n // Trigger completion for @mentions (agents, ask-models, files)\n return {\n type: 'agent', // This will trigger mixed agent+file completion\n prefix: content,\n startPos: start,\n endPos: cursorOffset, // Use cursor position as end\n }\n }\n\n // Everything else defaults to file completion\n return {\n type: 'file',\n prefix: word,\n startPos: start,\n endPos: cursorOffset, // Use cursor position as end\n }\n }, [input, cursorOffset])\n\n // System commands cache - populated dynamically from $PATH\n const [systemCommands, setSystemCommands] = useState<string[]>([])\n const [isLoadingCommands, setIsLoadingCommands] = useState(false)\n\n // Dynamic command classification based on intrinsic features\n const classifyCommand = useCallback(\n (cmd: string): 'core' | 'common' | 'dev' | 'system' => {\n const lowerCmd = cmd.toLowerCase()\n let score = 0\n\n // === FEATURE 1: Name Length & Complexity ===\n // Short, simple names are usually core commands\n if (cmd.length <= 4) score += 40\n else if (cmd.length <= 6) score += 20\n else if (cmd.length <= 8) score += 10\n else if (cmd.length > 15) score -= 30 // Very long names are specialized\n\n // === FEATURE 2: Character Patterns ===\n // Simple alphabetic names are more likely core\n if (/^[a-z]+$/.test(lowerCmd)) score += 30\n\n // Mixed case, numbers, dots suggest specialized tools\n if (/[A-Z]/.test(cmd)) score -= 15\n if (/\\d/.test(cmd)) score -= 20\n if (cmd.includes('.')) score -= 25\n if (cmd.includes('-')) score -= 10\n if (cmd.includes('_')) score -= 15\n\n // === FEATURE 3: Linguistic Patterns ===\n // Single, common English words\n const commonWords = [\n 'list',\n 'copy',\n 'move',\n 'find',\n 'print',\n 'show',\n 'edit',\n 'view',\n ]\n if (commonWords.some(word => lowerCmd.includes(word.slice(0, 3))))\n score += 25\n\n // Domain-specific prefixes/suffixes\n const devPrefixes = ['git', 'npm', 'node', 'py', 'docker', 'kubectl']\n if (devPrefixes.some(prefix => lowerCmd.startsWith(prefix))) score += 15\n\n // System/daemon indicators\n const systemIndicators = [\n 'daemon',\n 'helper',\n 'responder',\n 'service',\n 'd$',\n 'ctl$',\n ]\n if (\n systemIndicators.some(indicator =>\n indicator.endsWith('$')\n ? lowerCmd.endsWith(indicator.slice(0, -1))\n : lowerCmd.includes(indicator),\n )\n )\n score -= 40\n\n // === FEATURE 4: File Extension Indicators ===\n // Commands with extensions are usually scripts/specialized tools\n if (/\\.(pl|py|sh|rb|js)$/.test(lowerCmd)) score -= 35\n\n // === FEATURE 5: Path Location Heuristics ===\n // Note: We don't have path info here, but can infer from name patterns\n // Commands that look like they belong in /usr/local/bin or specialized dirs\n const buildToolPatterns = [\n 'bindep',\n 'render',\n 'mako',\n 'webpack',\n 'babel',\n 'eslint',\n ]\n if (buildToolPatterns.some(pattern => lowerCmd.includes(pattern)))\n score -= 25\n\n // === FEATURE 6: Vowel/Consonant Patterns ===\n // Unix commands often have abbreviated names with few vowels\n const vowelRatio =\n (lowerCmd.match(/[aeiou]/g) || []).length / lowerCmd.length\n if (vowelRatio < 0.2) score += 15 // Very few vowels (like 'ls', 'cp', 'mv')\n if (vowelRatio > 0.5) score -= 10 // Too many vowels (usually full words)\n\n // === CLASSIFICATION BASED ON SCORE ===\n if (score >= 50) return 'core' // 50+: Core unix commands\n if (score >= 20) return 'common' // 20-49: Common dev tools\n if (score >= -10) return 'dev' // -10-19: Specialized dev tools\n return 'system' // <-10: System/edge commands\n },\n [],\n )\n\n // Load system commands from PATH (like real terminal)\n const loadSystemCommands = useCallback(async () => {\n if (systemCommands.length > 0 || isLoadingCommands) return // Already loaded or loading\n\n setIsLoadingCommands(true)\n try {\n const { readdirSync, statSync } = await import('fs')\n const pathDirs = (process.env.PATH || '').split(':').filter(Boolean)\n const commandSet = new Set<string>()\n\n // Get essential commands from utils\n const essentialCommands = getEssentialCommands()\n\n // Add essential commands first\n essentialCommands.forEach(cmd => commandSet.add(cmd))\n\n // Scan PATH directories for executables\n for (const dir of pathDirs) {\n try {\n if (readdirSync && statSync) {\n const entries = readdirSync(dir)\n for (const entry of entries) {\n try {\n const fullPath = `${dir}/${entry}`\n const stats = statSync(fullPath)\n // Check if it's executable (rough check)\n if (stats.isFile() && (stats.mode & 0o111) !== 0) {\n commandSet.add(entry)\n }\n } catch {\n // Skip files we can't stat\n }\n }\n }\n } catch {\n // Skip directories we can't read\n }\n }\n\n const commands = Array.from(commandSet).sort()\n setSystemCommands(commands)\n } catch (error) {\n console.warn('Failed to load system commands, using fallback:', error)\n // Use minimal fallback commands from utils if system scan fails\n setSystemCommands(getMinimalFallbackCommands())\n } finally {\n setIsLoadingCommands(false)\n }\n }, [systemCommands.length, isLoadingCommands])\n\n // Load commands on first use\n useEffect(() => {\n loadSystemCommands()\n }, [loadSystemCommands])\n\n // Generate command suggestions (slash commands)\n const generateCommandSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n const filteredCommands = commands.filter(cmd => !cmd.isHidden)\n\n if (!prefix) {\n // Show all commands when prefix is empty (for single /)\n return filteredCommands.map(cmd => ({\n value: cmd.userFacingName(),\n displayValue: `/${cmd.userFacingName()}`,\n type: 'command' as const,\n score: 100,\n }))\n }\n\n return filteredCommands\n .filter(cmd => {\n const names = [cmd.userFacingName(), ...(cmd.aliases || [])]\n return names.some(name =>\n name.toLowerCase().startsWith(prefix.toLowerCase()),\n )\n })\n .map(cmd => ({\n value: cmd.userFacingName(),\n displayValue: `/${cmd.userFacingName()}`,\n type: 'command' as const,\n score:\n 100 -\n prefix.length +\n (cmd.userFacingName().startsWith(prefix) ? 10 : 0),\n }))\n },\n [commands],\n )\n\n // Clean Unix command scoring using fuzzy matcher\n const calculateUnixCommandScore = useCallback(\n (cmd: string, prefix: string): number => {\n const result = matchCommands([cmd], prefix)\n return result.length > 0 ? result[0].score : 0\n },\n [],\n )\n\n // Clean Unix command suggestions using fuzzy matcher with common commands boost\n const generateUnixCommandSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n if (!prefix) return []\n\n // Loading state\n if (isLoadingCommands) {\n return [\n {\n value: 'loading...',\n displayValue: `\u23F3 Loading system commands...`,\n type: 'file' as const,\n score: 0,\n metadata: { isLoading: true },\n },\n ]\n }\n\n // IMPORTANT: Only use commands that exist on the system (intersection)\n const commonCommands = getCommonSystemCommands(systemCommands)\n\n // Deduplicate commands (in case of any duplicates)\n const uniqueCommands = Array.from(new Set(commonCommands))\n\n // Use fuzzy matcher ONLY on the unique intersection\n const matches = matchCommands(uniqueCommands, prefix)\n\n // Boost common commands\n const boostedMatches = matches\n .map(match => {\n const priority = getCommandPriority(match.command)\n return {\n ...match,\n score: match.score + priority * 0.5, // Add priority boost\n }\n })\n .sort((a, b) => b.score - a.score)\n\n // Limit results intelligently\n let results = boostedMatches.slice(0, 8)\n\n // If we have very high scores (900+), show fewer\n const perfectMatches = boostedMatches.filter(m => m.score >= 900)\n if (perfectMatches.length > 0 && perfectMatches.length <= 3) {\n results = perfectMatches\n }\n // If we have good scores (100+), prefer them\n else if (boostedMatches.length > 8) {\n const goodMatches = boostedMatches.filter(m => m.score >= 100)\n if (goodMatches.length <= 5) {\n results = goodMatches\n }\n }\n\n return results.map(item => ({\n value: item.command,\n displayValue: `$ ${item.command}`,\n type: 'command' as const,\n score: item.score,\n metadata: { isUnixCommand: true },\n }))\n },\n [systemCommands, isLoadingCommands],\n )\n\n // Agent suggestions cache\n const [agentSuggestions, setAgentSuggestions] = useState<UnifiedSuggestion[]>(\n [],\n )\n\n // Model suggestions cache\n const [modelSuggestions, setModelSuggestions] = useState<UnifiedSuggestion[]>(\n [],\n )\n\n // Load model suggestions\n useEffect(() => {\n try {\n const modelManager = getModelManager()\n const allModels = modelManager.getAllAvailableModelNames()\n\n const suggestions = allModels.map(modelId => {\n // Professional and clear description for expert model consultation\n return {\n value: `ask-${modelId}`,\n displayValue: `\uD83E\uDD9C ask-${modelId} :: Consult ${modelId} for expert opinion and specialized analysis`,\n type: 'ask' as const,\n score: 90, // Higher than agents - put ask-models on top\n metadata: { modelId },\n }\n })\n\n setModelSuggestions(suggestions)\n } catch (error) {\n console.warn('[useUnifiedCompletion] Failed to load models:', error)\n // No fallback - rely on dynamic loading only\n setModelSuggestions([])\n }\n }, [])\n\n // Load agent suggestions on mount\n useEffect(() => {\n getActiveAgents()\n .then(agents => {\n // agents is an array of AgentConfig, not an object\n const suggestions = agents.map(config => {\n // \uD83E\uDDE0 \u667A\u80FD\u63CF\u8FF0\u7B97\u6CD5 - \u9002\u5E94\u6027\u957F\u5EA6\u63A7\u5236\n let shortDesc = config.whenToUse\n\n // \u79FB\u9664\u5E38\u89C1\u7684\u5197\u4F59\u524D\u7F00\uFF0C\u4F46\u4FDD\u7559\u6838\u5FC3\u5185\u5BB9\n const prefixPatterns = [\n /^Use this agent when you need (assistance with: )?/i,\n /^Use PROACTIVELY (when|to) /i,\n /^Specialized in /i,\n /^Implementation specialist for /i,\n /^Design validation specialist\\.? Use PROACTIVELY to /i,\n /^Task validation specialist\\.? Use PROACTIVELY to /i,\n /^Requirements validation specialist\\.? Use PROACTIVELY to /i,\n ]\n\n for (const pattern of prefixPatterns) {\n shortDesc = shortDesc.replace(pattern, '')\n }\n\n // \uD83C\uDFAF \u7CBE\u51C6\u65AD\u53E5\u7B97\u6CD5\uFF1A\u4E2D\u82F1\u6587\u53E5\u53F7\u611F\u53F9\u53F7\u4F18\u5148 \u2192 \u9017\u53F7 \u2192 \u7701\u7565\n const findSmartBreak = (text: string, maxLength: number) => {\n if (text.length <= maxLength) return text\n\n // \u7B2C\u4E00\u4F18\u5148\u7EA7\uFF1A\u4E2D\u82F1\u6587\u53E5\u53F7\u3001\u611F\u53F9\u53F7\n const sentenceEndings = /[.!\u3002\uFF01]/\n const firstSentenceMatch = text.search(sentenceEndings)\n if (firstSentenceMatch !== -1) {\n const firstSentence = text.slice(0, firstSentenceMatch).trim()\n if (firstSentence.length >= 5) {\n return firstSentence\n }\n }\n\n // \u5982\u679C\u7B2C\u4E00\u53E5\u8FC7\u957F\uFF0C\u627E\u9017\u53F7\u65AD\u53E5\n if (text.length > maxLength) {\n const commaEndings = /[,\uFF0C]/\n const commas = []\n let match\n const regex = new RegExp(commaEndings, 'g')\n while ((match = regex.exec(text)) !== null) {\n commas.push(match.index)\n }\n\n // \u627E\u6700\u540E\u4E00\u4E2A\u5728maxLength\u5185\u7684\u9017\u53F7\n for (let i = commas.length - 1; i >= 0; i--) {\n const commaPos = commas[i]\n if (commaPos < maxLength) {\n const clause = text.slice(0, commaPos).trim()\n if (clause.length >= 5) {\n return clause\n }\n }\n }\n }\n\n // \u6700\u540E\u9009\u62E9\uFF1A\u76F4\u63A5\u7701\u7565\n return text.slice(0, maxLength) + '...'\n }\n\n shortDesc = findSmartBreak(shortDesc.trim(), 80) // \u589E\u52A0\u523080\u5B57\u7B26\u9650\u5236\n\n // \u5982\u679C\u5904\u7406\u540E\u4E3A\u7A7A\u6216\u592A\u77ED\uFF0C\u4F7F\u7528\u539F\u59CB\u63CF\u8FF0\n if (!shortDesc || shortDesc.length < 5) {\n shortDesc = findSmartBreak(config.whenToUse, 80)\n }\n\n return {\n value: `run-agent-${config.agentType}`,\n displayValue: `\uD83D\uDC64 run-agent-${config.agentType} :: ${shortDesc}`, // \u4EBA\u7C7B\u56FE\u6807 + run-agent\u524D\u7F00 + \u7B80\u6D01\u63CF\u8FF0\n type: 'agent' as const,\n score: 85, // Lower than ask-models\n metadata: config,\n }\n })\n // Agents loaded successfully\n setAgentSuggestions(suggestions)\n })\n .catch(error => {\n console.warn('[useUnifiedCompletion] Failed to load agents:', error)\n // No fallback - rely on dynamic loading only\n setAgentSuggestions([])\n })\n }, [])\n\n // Generate agent and model suggestions using fuzzy matching\n const generateMentionSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n // Combine agent and model suggestions\n const allSuggestions = [...agentSuggestions, ...modelSuggestions]\n\n if (!prefix) {\n // Show all suggestions when prefix is empty (for single @)\n return allSuggestions.sort((a, b) => {\n // Ask models first (higher score), then agents\n if (a.type === 'ask' && b.type === 'agent') return -1\n if (a.type === 'agent' && b.type === 'ask') return 1\n return b.score - a.score\n })\n }\n\n // Use fuzzy matching for intelligent completion\n const candidates = allSuggestions.map(s => s.value)\n const matches = matchCommands(candidates, prefix)\n\n // Create result mapping with fuzzy scores\n const fuzzyResults = matches\n .map(match => {\n const suggestion = allSuggestions.find(\n s => s.value === match.command,\n )!\n return {\n ...suggestion,\n score: match.score, // Use fuzzy match score instead of simple scoring\n }\n })\n .sort((a, b) => {\n // Ask models first (for equal scores), then agents\n if (a.type === 'ask' && b.type === 'agent') return -1\n if (a.type === 'agent' && b.type === 'ask') return 1\n return b.score - a.score\n })\n\n return fuzzyResults\n },\n [agentSuggestions, modelSuggestions],\n )\n\n // Unix-style path completion - preserves user input semantics\n const generateFileSuggestions = useCallback(\n (prefix: string, isAtReference: boolean = false): UnifiedSuggestion[] => {\n try {\n const cwd = getCwd()\n\n // Parse user input preserving original format\n const userPath = prefix || '.'\n const isAbsolutePath = userPath.startsWith('/')\n const isHomePath = userPath.startsWith('~')\n\n // Resolve search directory - but keep user's path format for output\n let searchPath: string\n if (isHomePath) {\n searchPath = userPath.replace('~', process.env.HOME || '')\n } else if (isAbsolutePath) {\n searchPath = userPath\n } else {\n searchPath = resolve(cwd, userPath)\n }\n\n // Determine search directory and filename filter\n // If path ends with '/', treat it as directory navigation\n const endsWithSlash = userPath.endsWith('/')\n const searchStat = existsSync(searchPath) ? statSync(searchPath) : null\n\n let searchDir: string\n let nameFilter: string\n\n if (endsWithSlash || searchStat?.isDirectory()) {\n // User is navigating into a directory or path ends with /\n searchDir = searchPath\n nameFilter = ''\n } else {\n // User might be typing a partial filename\n searchDir = dirname(searchPath)\n nameFilter = basename(searchPath)\n }\n\n if (!existsSync(searchDir)) return []\n\n // Get directory entries with filter\n const showHidden = nameFilter.startsWith('.') || userPath.includes('/.')\n const entries = readdirSync(searchDir)\n .filter(entry => {\n // Filter hidden files unless user explicitly wants them\n if (!showHidden && entry.startsWith('.')) return false\n // Filter by name if there's a filter\n if (\n nameFilter &&\n !entry.toLowerCase().startsWith(nameFilter.toLowerCase())\n )\n return false\n return true\n })\n .sort((a, b) => {\n // Sort directories first, then files\n const aPath = join(searchDir, a)\n const bPath = join(searchDir, b)\n const aIsDir = statSync(aPath).isDirectory()\n const bIsDir = statSync(bPath).isDirectory()\n\n if (aIsDir && !bIsDir) return -1\n if (!aIsDir && bIsDir) return 1\n\n // Within same type, sort alphabetically\n return a.toLowerCase().localeCompare(b.toLowerCase())\n })\n .slice(0, 25) // Show more entries for better visibility\n\n return entries.map(entry => {\n const entryPath = join(searchDir, entry)\n const isDir = statSync(entryPath).isDirectory()\n const icon = isDir ? '\uD83D\uDCC1' : '\uD83D\uDCC4'\n\n // Unix-style path building - preserve user's original path format\n let value: string\n\n if (userPath.includes('/')) {\n // User typed path with separators - maintain structure\n if (endsWithSlash) {\n // User explicitly ended with / - they're inside the directory\n value = userPath + entry + (isDir ? '/' : '')\n } else if (searchStat?.isDirectory()) {\n // Path is a directory but doesn't end with / - add separator\n value = userPath + '/' + entry + (isDir ? '/' : '')\n } else {\n // User is completing a filename - replace basename\n const userDir = userPath.includes('/')\n ? userPath.substring(0, userPath.lastIndexOf('/'))\n : ''\n value = userDir\n ? userDir + '/' + entry + (isDir ? '/' : '')\n : entry + (isDir ? '/' : '')\n }\n } else {\n // User typed simple name - check if it's an existing directory\n if (searchStat?.isDirectory()) {\n // Existing directory - navigate into it\n value = userPath + '/' + entry + (isDir ? '/' : '')\n } else {\n // Simple completion at current level\n value = entry + (isDir ? '/' : '')\n }\n }\n\n return {\n value,\n displayValue: `${icon} ${entry}${isDir ? '/' : ''}`,\n type: 'file' as const,\n score: isDir ? 80 : 70,\n }\n })\n } catch {\n return []\n }\n },\n [],\n )\n\n // Unified smart matching - single algorithm with different weights\n const calculateMatchScore = useCallback(\n (suggestion: UnifiedSuggestion, prefix: string): number => {\n const lowerPrefix = prefix.toLowerCase()\n const value = suggestion.value.toLowerCase()\n const displayValue = suggestion.displayValue.toLowerCase()\n\n let matchFound = false\n let score = 0\n\n // Check for actual matches first\n if (value.startsWith(lowerPrefix)) {\n matchFound = true\n score = 100 // Highest priority\n } else if (value.includes(lowerPrefix)) {\n matchFound = true\n score = 95\n } else if (displayValue.includes(lowerPrefix)) {\n matchFound = true\n score = 90\n } else {\n // Word boundary matching for compound names like \"general\" -> \"run-agent-general-purpose\"\n const words = value.split(/[-_]/)\n if (words.some(word => word.startsWith(lowerPrefix))) {\n matchFound = true\n score = 93\n } else {\n // Acronym matching (last resort)\n const acronym = words.map(word => word[0]).join('')\n if (acronym.startsWith(lowerPrefix)) {\n matchFound = true\n score = 88\n }\n }\n }\n\n // Only return score if we found a match\n if (!matchFound) return 0\n\n // Type preferences (small bonus)\n if (suggestion.type === 'ask') score += 2\n if (suggestion.type === 'agent') score += 1\n\n return score\n },\n [],\n )\n\n // Generate smart mention suggestions without data pollution\n const generateSmartMentionSuggestions = useCallback(\n (\n prefix: string,\n sourceContext: 'file' | 'agent' = 'file',\n ): UnifiedSuggestion[] => {\n if (!prefix || prefix.length < 2) return []\n\n const allSuggestions = [...agentSuggestions, ...modelSuggestions]\n\n return allSuggestions\n .map(suggestion => {\n const matchScore = calculateMatchScore(suggestion, prefix)\n if (matchScore === 0) return null\n\n // Clean transformation without data pollution\n return {\n ...suggestion,\n score: matchScore,\n isSmartMatch: true,\n originalContext: sourceContext,\n // Only modify display for clarity, keep value clean\n displayValue: `\uD83C\uDFAF ${suggestion.displayValue}`,\n }\n })\n .filter(Boolean)\n .sort((a, b) => b.score - a.score)\n .slice(0, 5)\n },\n [agentSuggestions, modelSuggestions, calculateMatchScore],\n )\n\n // Generate all suggestions based on context\n const generateSuggestions = useCallback(\n (context: CompletionContext): UnifiedSuggestion[] => {\n switch (context.type) {\n case 'command':\n return generateCommandSuggestions(context.prefix)\n case 'agent': {\n // @ reference: combine mentions and files with clean priority\n const mentionSuggestions = generateMentionSuggestions(context.prefix)\n const fileSuggestions = generateFileSuggestions(context.prefix, true) // isAtReference=true\n\n // Apply weights for @ context (agents/models should be prioritized but files visible)\n const weightedSuggestions = [\n ...mentionSuggestions.map(s => ({\n ...s,\n // In @ context, agents/models get high priority\n weightedScore: s.score + 150,\n })),\n ...fileSuggestions.map(s => ({\n ...s,\n // Files get lower priority but still visible\n weightedScore: s.score + 10, // Small boost to ensure visibility\n })),\n ]\n\n // Sort by weighted score - no artificial limits\n return weightedSuggestions\n .sort((a, b) => b.weightedScore - a.weightedScore)\n .map(({ weightedScore, ...suggestion }) => suggestion)\n // No limit or very generous limit (e.g., 30 items)\n }\n case 'file': {\n // For normal input, try to match everything intelligently\n const fileSuggestions = generateFileSuggestions(context.prefix, false)\n const unixSuggestions = generateUnixCommandSuggestions(context.prefix)\n\n // IMPORTANT: Also try to match agents and models WITHOUT requiring @\n // This enables smart matching for inputs like \"gp5\", \"daoqi\", etc.\n const mentionMatches = generateMentionSuggestions(context.prefix).map(\n s => ({\n ...s,\n isSmartMatch: true,\n // Show that @ will be added when selected\n displayValue: `\\u2192 ${s.displayValue}`, // Arrow to indicate it will transform\n }),\n )\n\n // Apply source-based priority weights with special handling for exact matches\n // Priority order: Exact Unix > Unix commands > agents/models > files\n const weightedSuggestions = [\n ...unixSuggestions.map(s => ({\n ...s,\n // Unix commands get boost, but exact matches get huge boost\n sourceWeight: s.score >= 10000 ? 5000 : 200, // Exact match gets massive boost\n weightedScore: s.score >= 10000 ? s.score + 5000 : s.score + 200,\n })),\n ...mentionMatches.map(s => ({\n ...s,\n // Agents/models get medium priority boost (but less to avoid overriding exact Unix)\n sourceWeight: 50,\n weightedScore: s.score + 50,\n })),\n ...fileSuggestions.map(s => ({\n ...s,\n // Files get no boost (baseline)\n sourceWeight: 0,\n weightedScore: s.score,\n })),\n ]\n\n // Sort by weighted score and deduplicate\n const seen = new Set<string>()\n const deduplicatedResults = weightedSuggestions\n .sort((a, b) => b.weightedScore - a.weightedScore)\n .filter(item => {\n // Filter out duplicates based on value\n if (seen.has(item.value)) return false\n seen.add(item.value)\n return true\n })\n .map(({ weightedScore, sourceWeight, ...suggestion }) => suggestion) // Remove weight fields\n // No limit - show all relevant matches\n\n return deduplicatedResults\n }\n default:\n return []\n }\n },\n [\n generateCommandSuggestions,\n generateMentionSuggestions,\n generateFileSuggestions,\n generateUnixCommandSuggestions,\n generateSmartMentionSuggestions,\n ],\n )\n\n // Complete with a suggestion - \u652F\u6301\u4E07\u80FD@\u5F15\u7528 + slash\u547D\u4EE4\u81EA\u52A8\u6267\u884C\n const completeWith = useCallback(\n (suggestion: UnifiedSuggestion, context: CompletionContext) => {\n let completion: string\n\n if (context.type === 'command') {\n completion = `/${suggestion.value} `\n } else if (context.type === 'agent') {\n // \uD83D\uDE80 \u4E07\u80FD@\u5F15\u7528\uFF1A\u6839\u636E\u5EFA\u8BAE\u7C7B\u578B\u51B3\u5B9A\u8865\u5168\u683C\u5F0F\n if (suggestion.type === 'agent') {\n completion = `@${suggestion.value} ` // \u4EE3\u7406\u8865\u5168\n } else if (suggestion.type === 'ask') {\n completion = `@${suggestion.value} ` // Ask\u6A21\u578B\u8865\u5168\n } else {\n // File reference in @mention context - no space for directories to allow expansion\n const isDirectory = suggestion.value.endsWith('/')\n completion = `@${suggestion.value}${isDirectory ? '' : ' '}` // \u6587\u4EF6\u5939\u4E0D\u52A0\u7A7A\u683C\uFF0C\u6587\u4EF6\u52A0\u7A7A\u683C\n }\n } else {\n // Regular file completion OR smart mention matching\n if (suggestion.isSmartMatch) {\n // Smart mention - add @ prefix and space\n completion = `@${suggestion.value} `\n } else {\n // Regular file completion - no space for directories to allow expansion\n const isDirectory = suggestion.value.endsWith('/')\n completion = suggestion.value + (isDirectory ? '' : ' ')\n }\n }\n\n // Special handling for absolute paths in file completion\n // When completing an absolute path, we should replace the entire current word/path\n let actualEndPos: number\n\n if (\n context.type === 'file' &&\n suggestion.value.startsWith('/') &&\n !suggestion.isSmartMatch\n ) {\n // For absolute paths, find the end of the current path/word\n let end = context.startPos\n while (\n end < input.length &&\n input[end] !== ' ' &&\n input[end] !== '\\n'\n ) {\n end++\n }\n actualEndPos = end\n } else {\n // Original logic for other cases\n const currentWord = input.slice(context.startPos)\n const nextSpaceIndex = currentWord.indexOf(' ')\n actualEndPos =\n nextSpaceIndex === -1\n ? input.length\n : context.startPos + nextSpaceIndex\n }\n\n const newInput =\n input.slice(0, context.startPos) +\n completion +\n input.slice(actualEndPos)\n onInputChange(newInput)\n setCursorOffset(context.startPos + completion.length)\n\n // Don't auto-execute slash commands - let user press Enter to submit\n // This gives users a chance to add arguments or modify the command\n\n // Completion applied\n },\n [input, onInputChange, setCursorOffset, onSubmit, commands],\n )\n\n // Partial complete to common prefix\n const partialComplete = useCallback(\n (prefix: string, context: CompletionContext) => {\n const completion =\n context.type === 'command'\n ? `/${prefix}`\n : context.type === 'agent'\n ? `@${prefix}`\n : prefix\n\n const newInput =\n input.slice(0, context.startPos) +\n completion +\n input.slice(context.endPos)\n onInputChange(newInput)\n setCursorOffset(context.startPos + completion.length)\n },\n [input, onInputChange, setCursorOffset],\n )\n\n // Handle Tab key - simplified and unified\n useInput((input_str, key) => {\n if (!key.tab) return false\n if (key.shift) return false\n\n const context = getWordAtCursor()\n if (!context) return false\n\n // If menu is already showing, cycle through suggestions\n if (state.isActive && state.suggestions.length > 0) {\n const nextIndex = (state.selectedIndex + 1) % state.suggestions.length\n const nextSuggestion = state.suggestions[nextIndex]\n\n if (state.context) {\n // Calculate proper word boundaries\n const currentWord = input.slice(state.context.startPos)\n const wordEnd = currentWord.search(/\\s/)\n const actualEndPos =\n wordEnd === -1 ? input.length : state.context.startPos + wordEnd\n\n // Apply appropriate prefix based on context type and suggestion type\n let preview: string\n if (state.context.type === 'command') {\n preview = `/${nextSuggestion.value}`\n } else if (state.context.type === 'agent') {\n // For @mentions, always add @ prefix\n preview = `@${nextSuggestion.value}`\n } else if (nextSuggestion.isSmartMatch) {\n // Smart match from normal input - add @ prefix\n preview = `@${nextSuggestion.value}`\n } else {\n preview = nextSuggestion.value\n }\n\n // Apply preview\n const newInput =\n input.slice(0, state.context.startPos) +\n preview +\n input.slice(actualEndPos)\n\n onInputChange(newInput)\n setCursorOffset(state.context.startPos + preview.length)\n\n // Update state\n updateState({\n selectedIndex: nextIndex,\n preview: {\n isActive: true,\n originalInput: input,\n wordRange: [\n state.context.startPos,\n state.context.startPos + preview.length,\n ],\n },\n })\n }\n return true\n }\n\n // Generate new suggestions\n const currentSuggestions = generateSuggestions(context)\n\n if (currentSuggestions.length === 0) {\n return false // Let Tab pass through\n } else if (currentSuggestions.length === 1) {\n // Single match: complete immediately\n completeWith(currentSuggestions[0], context)\n return true\n } else {\n // Show menu and apply first suggestion\n activateCompletion(currentSuggestions, context)\n\n // Immediately apply first suggestion as preview\n const firstSuggestion = currentSuggestions[0]\n const currentWord = input.slice(context.startPos)\n const wordEnd = currentWord.search(/\\s/)\n const actualEndPos =\n wordEnd === -1 ? input.length : context.startPos + wordEnd\n\n let preview: string\n if (context.type === 'command') {\n preview = `/${firstSuggestion.value}`\n } else if (context.type === 'agent') {\n preview = `@${firstSuggestion.value}`\n } else if (firstSuggestion.isSmartMatch) {\n // Smart match from normal input - add @ prefix\n preview = `@${firstSuggestion.value}`\n } else {\n preview = firstSuggestion.value\n }\n\n const newInput =\n input.slice(0, context.startPos) + preview + input.slice(actualEndPos)\n\n onInputChange(newInput)\n setCursorOffset(context.startPos + preview.length)\n\n updateState({\n preview: {\n isActive: true,\n originalInput: input,\n wordRange: [context.startPos, context.startPos + preview.length],\n },\n })\n\n return true\n }\n })\n\n // Handle navigation keys - simplified and unified\n useInput((inputChar, key) => {\n // Enter key - confirm selection and end completion (always add space)\n if (key.return && state.isActive && state.suggestions.length > 0) {\n const selectedSuggestion = state.suggestions[state.selectedIndex]\n if (selectedSuggestion && state.context) {\n // For Enter key, always add space even for directories to indicate completion end\n let completion: string\n\n if (state.context.type === 'command') {\n completion = `/${selectedSuggestion.value} `\n } else if (state.context.type === 'agent') {\n if (selectedSuggestion.type === 'agent') {\n completion = `@${selectedSuggestion.value} `\n } else if (selectedSuggestion.type === 'ask') {\n completion = `@${selectedSuggestion.value} `\n } else {\n // File reference in @mention context - always add space on Enter\n completion = `@${selectedSuggestion.value} `\n }\n } else if (selectedSuggestion.isSmartMatch) {\n // Smart match from normal input - add @ prefix\n completion = `@${selectedSuggestion.value} `\n } else {\n // Regular file completion - always add space on Enter\n completion = selectedSuggestion.value + ' '\n }\n\n // Apply completion with forced space\n const currentWord = input.slice(state.context.startPos)\n const nextSpaceIndex = currentWord.indexOf(' ')\n const actualEndPos =\n nextSpaceIndex === -1\n ? input.length\n : state.context.startPos + nextSpaceIndex\n\n const newInput =\n input.slice(0, state.context.startPos) +\n completion +\n input.slice(actualEndPos)\n onInputChange(newInput)\n setCursorOffset(state.context.startPos + completion.length)\n }\n resetCompletion()\n return true\n }\n\n if (!state.isActive || state.suggestions.length === 0) return false\n\n // Arrow key navigation with preview\n const handleNavigation = (newIndex: number) => {\n const preview = state.suggestions[newIndex].value\n\n if (state.preview?.isActive && state.context) {\n const newInput =\n input.slice(0, state.context.startPos) +\n preview +\n input.slice(state.preview.wordRange[1])\n\n onInputChange(newInput)\n setCursorOffset(state.context.startPos + preview.length)\n\n updateState({\n selectedIndex: newIndex,\n preview: {\n ...state.preview,\n wordRange: [\n state.context.startPos,\n state.context.startPos + preview.length,\n ],\n },\n })\n } else {\n updateState({ selectedIndex: newIndex })\n }\n }\n\n if (key.downArrow) {\n const nextIndex = (state.selectedIndex + 1) % state.suggestions.length\n handleNavigation(nextIndex)\n return true\n }\n\n if (key.upArrow) {\n const nextIndex =\n state.selectedIndex === 0\n ? state.suggestions.length - 1\n : state.selectedIndex - 1\n handleNavigation(nextIndex)\n return true\n }\n\n // Space key - complete and potentially continue for directories\n if (inputChar === ' ' && state.isActive && state.suggestions.length > 0) {\n const selectedSuggestion = state.suggestions[state.selectedIndex]\n const isDirectory = selectedSuggestion.value.endsWith('/')\n\n if (!state.context) return false\n\n // Apply completion if needed\n const currentWordAtContext = input.slice(\n state.context.startPos,\n state.context.startPos + selectedSuggestion.value.length,\n )\n\n if (currentWordAtContext !== selectedSuggestion.value) {\n completeWith(selectedSuggestion, state.context)\n }\n\n resetCompletion()\n\n if (isDirectory) {\n // Continue completion for directories\n setTimeout(() => {\n const newContext = {\n ...state.context,\n prefix: selectedSuggestion.value,\n endPos: state.context.startPos + selectedSuggestion.value.length,\n }\n\n const newSuggestions = generateSuggestions(newContext)\n\n if (newSuggestions.length > 0) {\n activateCompletion(newSuggestions, newContext)\n } else {\n updateState({\n emptyDirMessage: `Directory is empty: ${selectedSuggestion.value}`,\n })\n setTimeout(() => updateState({ emptyDirMessage: '' }), 3000)\n }\n }, 50)\n }\n\n return true\n }\n\n // Right arrow key - same as space but different semantics\n if (key.rightArrow) {\n const selectedSuggestion = state.suggestions[state.selectedIndex]\n const isDirectory = selectedSuggestion.value.endsWith('/')\n\n if (!state.context) return false\n\n // Apply completion\n const currentWordAtContext = input.slice(\n state.context.startPos,\n state.context.startPos + selectedSuggestion.value.length,\n )\n\n if (currentWordAtContext !== selectedSuggestion.value) {\n completeWith(selectedSuggestion, state.context)\n }\n\n resetCompletion()\n\n if (isDirectory) {\n // Continue for directories\n setTimeout(() => {\n const newContext = {\n ...state.context,\n prefix: selectedSuggestion.value,\n endPos: state.context.startPos + selectedSuggestion.value.length,\n }\n\n const newSuggestions = generateSuggestions(newContext)\n\n if (newSuggestions.length > 0) {\n activateCompletion(newSuggestions, newContext)\n } else {\n updateState({\n emptyDirMessage: `Directory is empty: ${selectedSuggestion.value}`,\n })\n setTimeout(() => updateState({ emptyDirMessage: '' }), 3000)\n }\n }, 50)\n }\n\n return true\n }\n\n if (key.escape) {\n // Restore original text if in preview mode\n if (state.preview?.isActive && state.context) {\n onInputChange(state.preview.originalInput)\n setCursorOffset(state.context.startPos + state.context.prefix.length)\n }\n\n resetCompletion()\n return true\n }\n\n return false\n })\n\n // Handle delete/backspace keys - unified state management\n useInput((input_str, key) => {\n if (key.backspace || key.delete) {\n if (state.isActive) {\n resetCompletion()\n // Smart suppression based on input complexity\n const suppressionTime = input.length > 10 ? 200 : 100\n updateState({\n suppressUntil: Date.now() + suppressionTime,\n })\n return true\n }\n }\n return false\n })\n\n // Input tracking with ref to avoid infinite loops\n const lastInputRef = useRef('')\n\n // Smart auto-triggering with cycle prevention\n useEffect(() => {\n // Prevent infinite loops by using ref\n if (lastInputRef.current === input) return\n\n const inputLengthChange = Math.abs(\n input.length - lastInputRef.current.length,\n )\n const isHistoryNavigation =\n (inputLengthChange > 10 || // Large content change\n (inputLengthChange > 5 &&\n !input.includes(lastInputRef.current.slice(-5)))) && // Different content\n input !== lastInputRef.current\n\n // Update ref (no state update)\n lastInputRef.current = input\n\n // Skip if in preview mode or suppressed\n if (state.preview?.isActive || Date.now() < state.suppressUntil) {\n return\n }\n\n // Clear suggestions on history navigation\n if (isHistoryNavigation && state.isActive) {\n resetCompletion()\n return\n }\n\n const context = getWordAtCursor()\n\n if (context && shouldAutoTrigger(context)) {\n const newSuggestions = generateSuggestions(context)\n\n if (newSuggestions.length === 0) {\n resetCompletion()\n } else if (\n newSuggestions.length === 1 &&\n shouldAutoHideSingleMatch(newSuggestions[0], context)\n ) {\n resetCompletion() // Perfect match - hide\n } else {\n activateCompletion(newSuggestions, context)\n }\n } else if (state.context) {\n // Check if context changed significantly\n const contextChanged =\n !context ||\n state.context.type !== context.type ||\n state.context.startPos !== context.startPos ||\n !context.prefix.startsWith(state.context.prefix)\n\n if (contextChanged) {\n resetCompletion()\n }\n }\n }, [input, cursorOffset])\n\n // Smart triggering - only when it makes sense\n const shouldAutoTrigger = useCallback(\n (context: CompletionContext): boolean => {\n switch (context.type) {\n case 'command':\n // Trigger immediately for slash commands\n return true\n case 'agent':\n // Trigger immediately for agent references\n return true\n case 'file':\n // Be selective about file completion - avoid noise\n const prefix = context.prefix\n\n // Always trigger for clear path patterns\n if (\n prefix.startsWith('./') ||\n prefix.startsWith('../') ||\n prefix.startsWith('/') ||\n prefix.startsWith('~') ||\n prefix.includes('/')\n ) {\n return true\n }\n\n // Trigger for single dot followed by something (like .g for .gitignore)\n if (prefix.startsWith('.') && prefix.length >= 2) {\n return true\n }\n\n // Skip very short prefixes that are likely code\n return false\n default:\n return false\n }\n },\n [],\n )\n\n // Helper function to determine if single suggestion should be auto-hidden\n const shouldAutoHideSingleMatch = useCallback(\n (suggestion: UnifiedSuggestion, context: CompletionContext): boolean => {\n // Extract the actual typed input from context\n const currentInput = input.slice(context.startPos, context.endPos)\n // Check if should auto-hide single match\n\n // For files: more intelligent matching\n if (context.type === 'file') {\n // Special case: if suggestion is a directory (ends with /), don't auto-hide\n // because user might want to continue navigating into it\n if (suggestion.value.endsWith('/')) {\n // Directory suggestion, keeping visible\n return false\n }\n\n // Check exact match\n if (currentInput === suggestion.value) {\n // Exact match, hiding\n return true\n }\n\n // Check if current input is a complete file path and suggestion is just the filename\n // e.g., currentInput: \"src/tools/ThinkTool/ThinkTool.tsx\", suggestion: \"ThinkTool.tsx\"\n if (\n currentInput.endsWith('/' + suggestion.value) ||\n currentInput.endsWith(suggestion.value)\n ) {\n // Path ends with suggestion, hiding\n return true\n }\n\n return false\n }\n\n // For commands: check if /prefix exactly matches /command\n if (context.type === 'command') {\n const fullCommand = `/${suggestion.value}`\n const matches = currentInput === fullCommand\n // Check command match\n return matches\n }\n\n // For agents: check if @prefix exactly matches @agent-name\n if (context.type === 'agent') {\n const fullAgent = `@${suggestion.value}`\n const matches = currentInput === fullAgent\n // Check agent match\n return matches\n }\n\n return false\n },\n [input],\n )\n\n return {\n suggestions,\n selectedIndex,\n isActive,\n emptyDirMessage,\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["import { useState, useCallback, useEffect, useRef } from 'react'\nimport { useInput } from 'ink'\nimport { existsSync, statSync, readdirSync } from 'fs'\nimport { join, dirname, basename, resolve } from 'path'\nimport { getCwd } from '@utils/state'\nimport { getCommand } from '@commands'\nimport { getActiveAgents } from '@utils/agentLoader'\nimport { getModelManager } from '@utils/model'\nimport { glob } from 'glob'\nimport { matchCommands } from '@utils/fuzzyMatcher'\nimport {\n getCommonSystemCommands,\n getCommandPriority,\n getEssentialCommands,\n getMinimalFallbackCommands,\n} from '@utils/commonUnixCommands'\nimport { listMCPResources, type McpResource } from '@services/mcpClient'\nimport type { Command } from '@commands'\n\n// Unified suggestion type for all completion types\nexport interface UnifiedSuggestion {\n value: string\n displayValue: string\n type: 'command' | 'agent' | 'file' | 'ask' | 'mcp-resource'\n icon?: string\n score: number\n metadata?: any\n // Clean type system for smart matching\n isSmartMatch?: boolean // Instead of magic string checking\n originalContext?: 'mention' | 'file' | 'command' | 'mcp' // Track source context\n}\n\ninterface CompletionContext {\n type: 'command' | 'agent' | 'file' | 'mcp' | null\n prefix: string\n startPos: number\n endPos: number\n}\n\n// Terminal behavior state for preview and cycling\ninterface TerminalState {\n originalWord: string\n wordContext: { start: number; end: number } | null\n isPreviewMode: boolean\n}\n\ninterface Props {\n input: string\n cursorOffset: number\n onInputChange: (value: string) => void\n setCursorOffset: (offset: number) => void\n commands: Command[]\n onSubmit?: (value: string, isSubmittingSlashCommand?: boolean) => void\n}\n\n/**\n * Unified completion system - Linus approved\n * One hook to rule them all, no bullshit, no complexity\n */\n// Unified completion state - single source of truth\ninterface CompletionState {\n suggestions: UnifiedSuggestion[]\n selectedIndex: number\n isActive: boolean\n context: CompletionContext | null\n preview: {\n isActive: boolean\n originalInput: string\n wordRange: [number, number]\n } | null\n emptyDirMessage: string\n suppressUntil: number // timestamp for suppression\n}\n\nconst INITIAL_STATE: CompletionState = {\n suggestions: [],\n selectedIndex: 0,\n isActive: false,\n context: null,\n preview: null,\n emptyDirMessage: '',\n suppressUntil: 0,\n}\n\nexport function useUnifiedCompletion({\n input,\n cursorOffset,\n onInputChange,\n setCursorOffset,\n commands,\n onSubmit,\n}: Props) {\n // Single state for entire completion system - Linus approved\n const [state, setState] = useState<CompletionState>(INITIAL_STATE)\n\n // State update helpers - clean and simple\n const updateState = useCallback((updates: Partial<CompletionState>) => {\n setState(prev => ({ ...prev, ...updates }))\n }, [])\n\n const resetCompletion = useCallback(() => {\n setState(prev => ({\n ...prev,\n suggestions: [],\n selectedIndex: 0,\n isActive: false,\n context: null,\n preview: null,\n emptyDirMessage: '',\n }))\n }, [])\n\n const activateCompletion = useCallback(\n (suggestions: UnifiedSuggestion[], context: CompletionContext) => {\n setState(prev => ({\n ...prev,\n suggestions: suggestions, // Keep the order from generateSuggestions (already sorted with weights)\n selectedIndex: 0,\n isActive: true,\n context,\n preview: null,\n }))\n },\n [],\n )\n\n // Direct state access - no legacy wrappers needed\n const { suggestions, selectedIndex, isActive, emptyDirMessage } = state\n\n // Find common prefix among suggestions (terminal behavior)\n const findCommonPrefix = useCallback(\n (suggestions: UnifiedSuggestion[]): string => {\n if (suggestions.length === 0) return ''\n if (suggestions.length === 1) return suggestions[0].value\n\n let prefix = suggestions[0].value\n\n for (let i = 1; i < suggestions.length; i++) {\n const str = suggestions[i].value\n let j = 0\n while (j < prefix.length && j < str.length && prefix[j] === str[j]) {\n j++\n }\n prefix = prefix.slice(0, j)\n\n if (prefix.length === 0) return ''\n }\n\n return prefix\n },\n [],\n )\n\n // Clean word detection - Linus approved simplicity\n const getWordAtCursor = useCallback((): CompletionContext | null => {\n if (!input) return null\n\n // IMPORTANT: Only match the word/prefix BEFORE the cursor\n // Don't include text after cursor to avoid confusion\n let start = cursorOffset\n\n // Move start backwards to find word beginning\n // Stop at whitespace or special boundaries\n while (start > 0) {\n const char = input[start - 1]\n // Stop at whitespace\n if (/\\s/.test(char)) break\n\n // For @mentions, include @ and stop\n if (char === '@' && start < cursorOffset) {\n start--\n break\n }\n\n // For paths, be smarter about / handling\n if (char === '/') {\n // Look ahead to see what we've collected so far\n const collectedSoFar = input.slice(start, cursorOffset)\n\n // If we already have a path component, this / is part of the path\n if (collectedSoFar.includes('/') || collectedSoFar.includes('.')) {\n start--\n continue\n }\n\n // Check if this is part of a path pattern like ./ or ../ or ~/\n if (start > 1) {\n const prevChar = input[start - 2]\n if (prevChar === '.' || prevChar === '~') {\n // It's part of ./ or ../ or ~/ - keep going\n start--\n continue\n }\n }\n\n // Check if this is a standalone / at the beginning (command)\n if (start === 1 || (start > 1 && /\\s/.test(input[start - 2]))) {\n start--\n break // It's a command slash\n }\n\n // Otherwise treat as path separator\n start--\n continue\n }\n\n // Special handling for dots in paths\n if (char === '.' && start > 0) {\n // Check if this might be start of ./ or ../\n const nextChar = start < input.length ? input[start] : ''\n if (nextChar === '/' || nextChar === '.') {\n start--\n continue // Part of a path pattern\n }\n }\n\n start--\n }\n\n // The word is from start to cursor position (not beyond)\n const word = input.slice(start, cursorOffset)\n if (!word) return null\n\n // Priority-based type detection - no special cases needed\n if (word.startsWith('/')) {\n const beforeWord = input.slice(0, start).trim()\n const isCommand = beforeWord === '' && !word.includes('/', 1)\n return {\n type: isCommand ? 'command' : 'file',\n prefix: isCommand ? word.slice(1) : word,\n startPos: start,\n endPos: cursorOffset, // Use cursor position as end\n }\n }\n\n if (word.startsWith('@')) {\n const content = word.slice(1) // Remove @\n\n // Check if this looks like an email (contains @ in the middle)\n if (word.includes('@', 1)) {\n // This looks like an email, treat as regular text\n return null\n }\n\n // Trigger completion for @mentions (agents, ask-models, files)\n return {\n type: 'agent', // This will trigger mixed agent+file completion\n prefix: content,\n startPos: start,\n endPos: cursorOffset, // Use cursor position as end\n }\n }\n\n // Check for MCP resource references (mcp: prefix)\n if (word.startsWith('mcp:') || word.includes('mcp:')) {\n const mcpPrefix = word.includes('mcp:')\n ? word.slice(word.indexOf('mcp:') + 4)\n : ''\n return {\n type: 'mcp',\n prefix: mcpPrefix,\n startPos: start,\n endPos: cursorOffset,\n }\n }\n\n // Everything else defaults to file completion\n return {\n type: 'file',\n prefix: word,\n startPos: start,\n endPos: cursorOffset, // Use cursor position as end\n }\n }, [input, cursorOffset])\n\n // System commands cache - populated dynamically from $PATH\n const [systemCommands, setSystemCommands] = useState<string[]>([])\n const [isLoadingCommands, setIsLoadingCommands] = useState(false)\n\n // Dynamic command classification based on intrinsic features\n const classifyCommand = useCallback(\n (cmd: string): 'core' | 'common' | 'dev' | 'system' => {\n const lowerCmd = cmd.toLowerCase()\n let score = 0\n\n // === FEATURE 1: Name Length & Complexity ===\n // Short, simple names are usually core commands\n if (cmd.length <= 4) score += 40\n else if (cmd.length <= 6) score += 20\n else if (cmd.length <= 8) score += 10\n else if (cmd.length > 15) score -= 30 // Very long names are specialized\n\n // === FEATURE 2: Character Patterns ===\n // Simple alphabetic names are more likely core\n if (/^[a-z]+$/.test(lowerCmd)) score += 30\n\n // Mixed case, numbers, dots suggest specialized tools\n if (/[A-Z]/.test(cmd)) score -= 15\n if (/\\d/.test(cmd)) score -= 20\n if (cmd.includes('.')) score -= 25\n if (cmd.includes('-')) score -= 10\n if (cmd.includes('_')) score -= 15\n\n // === FEATURE 3: Linguistic Patterns ===\n // Single, common English words\n const commonWords = [\n 'list',\n 'copy',\n 'move',\n 'find',\n 'print',\n 'show',\n 'edit',\n 'view',\n ]\n if (commonWords.some(word => lowerCmd.includes(word.slice(0, 3))))\n score += 25\n\n // Domain-specific prefixes/suffixes\n const devPrefixes = ['git', 'npm', 'node', 'py', 'docker', 'kubectl']\n if (devPrefixes.some(prefix => lowerCmd.startsWith(prefix))) score += 15\n\n // System/daemon indicators\n const systemIndicators = [\n 'daemon',\n 'helper',\n 'responder',\n 'service',\n 'd$',\n 'ctl$',\n ]\n if (\n systemIndicators.some(indicator =>\n indicator.endsWith('$')\n ? lowerCmd.endsWith(indicator.slice(0, -1))\n : lowerCmd.includes(indicator),\n )\n )\n score -= 40\n\n // === FEATURE 4: File Extension Indicators ===\n // Commands with extensions are usually scripts/specialized tools\n if (/\\.(pl|py|sh|rb|js)$/.test(lowerCmd)) score -= 35\n\n // === FEATURE 5: Path Location Heuristics ===\n // Note: We don't have path info here, but can infer from name patterns\n // Commands that look like they belong in /usr/local/bin or specialized dirs\n const buildToolPatterns = [\n 'bindep',\n 'render',\n 'mako',\n 'webpack',\n 'babel',\n 'eslint',\n ]\n if (buildToolPatterns.some(pattern => lowerCmd.includes(pattern)))\n score -= 25\n\n // === FEATURE 6: Vowel/Consonant Patterns ===\n // Unix commands often have abbreviated names with few vowels\n const vowelRatio =\n (lowerCmd.match(/[aeiou]/g) || []).length / lowerCmd.length\n if (vowelRatio < 0.2) score += 15 // Very few vowels (like 'ls', 'cp', 'mv')\n if (vowelRatio > 0.5) score -= 10 // Too many vowels (usually full words)\n\n // === CLASSIFICATION BASED ON SCORE ===\n if (score >= 50) return 'core' // 50+: Core unix commands\n if (score >= 20) return 'common' // 20-49: Common dev tools\n if (score >= -10) return 'dev' // -10-19: Specialized dev tools\n return 'system' // <-10: System/edge commands\n },\n [],\n )\n\n // Load system commands from PATH (like real terminal)\n const loadSystemCommands = useCallback(async () => {\n if (systemCommands.length > 0 || isLoadingCommands) return // Already loaded or loading\n\n setIsLoadingCommands(true)\n try {\n const { readdirSync, statSync } = await import('fs')\n const pathDirs = (process.env.PATH || '').split(':').filter(Boolean)\n const commandSet = new Set<string>()\n\n // Get essential commands from utils\n const essentialCommands = getEssentialCommands()\n\n // Add essential commands first\n essentialCommands.forEach(cmd => commandSet.add(cmd))\n\n // Scan PATH directories for executables\n for (const dir of pathDirs) {\n try {\n if (readdirSync && statSync) {\n const entries = readdirSync(dir)\n for (const entry of entries) {\n try {\n const fullPath = `${dir}/${entry}`\n const stats = statSync(fullPath)\n // Check if it's executable (rough check)\n if (stats.isFile() && (stats.mode & 0o111) !== 0) {\n commandSet.add(entry)\n }\n } catch {\n // Skip files we can't stat\n }\n }\n }\n } catch {\n // Skip directories we can't read\n }\n }\n\n const commands = Array.from(commandSet).sort()\n setSystemCommands(commands)\n } catch (error) {\n console.warn('Failed to load system commands, using fallback:', error)\n // Use minimal fallback commands from utils if system scan fails\n setSystemCommands(getMinimalFallbackCommands())\n } finally {\n setIsLoadingCommands(false)\n }\n }, [systemCommands.length, isLoadingCommands])\n\n // Load commands on first use\n useEffect(() => {\n loadSystemCommands()\n }, [loadSystemCommands])\n\n // Generate command suggestions (slash commands)\n const generateCommandSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n const filteredCommands = commands.filter(cmd => !cmd.isHidden)\n\n if (!prefix) {\n // Show all commands when prefix is empty (for single /)\n return filteredCommands.map(cmd => ({\n value: cmd.userFacingName(),\n displayValue: `/${cmd.userFacingName()}`,\n type: 'command' as const,\n score: 100,\n }))\n }\n\n return filteredCommands\n .filter(cmd => {\n const names = [cmd.userFacingName(), ...(cmd.aliases || [])]\n return names.some(name =>\n name.toLowerCase().startsWith(prefix.toLowerCase()),\n )\n })\n .map(cmd => ({\n value: cmd.userFacingName(),\n displayValue: `/${cmd.userFacingName()}`,\n type: 'command' as const,\n score:\n 100 -\n prefix.length +\n (cmd.userFacingName().startsWith(prefix) ? 10 : 0),\n }))\n },\n [commands],\n )\n\n // Clean Unix command scoring using fuzzy matcher\n const calculateUnixCommandScore = useCallback(\n (cmd: string, prefix: string): number => {\n const result = matchCommands([cmd], prefix)\n return result.length > 0 ? result[0].score : 0\n },\n [],\n )\n\n // Clean Unix command suggestions using fuzzy matcher with common commands boost\n const generateUnixCommandSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n if (!prefix) return []\n\n // Loading state\n if (isLoadingCommands) {\n return [\n {\n value: 'loading...',\n displayValue: `\u23F3 Loading system commands...`,\n type: 'file' as const,\n score: 0,\n metadata: { isLoading: true },\n },\n ]\n }\n\n // IMPORTANT: Only use commands that exist on the system (intersection)\n const commonCommands = getCommonSystemCommands(systemCommands)\n\n // Deduplicate commands (in case of any duplicates)\n const uniqueCommands = Array.from(new Set(commonCommands))\n\n // Use fuzzy matcher ONLY on the unique intersection\n const matches = matchCommands(uniqueCommands, prefix)\n\n // Boost common commands\n const boostedMatches = matches\n .map(match => {\n const priority = getCommandPriority(match.command)\n return {\n ...match,\n score: match.score + priority * 0.5, // Add priority boost\n }\n })\n .sort((a, b) => b.score - a.score)\n\n // Limit results intelligently\n let results = boostedMatches.slice(0, 8)\n\n // If we have very high scores (900+), show fewer\n const perfectMatches = boostedMatches.filter(m => m.score >= 900)\n if (perfectMatches.length > 0 && perfectMatches.length <= 3) {\n results = perfectMatches\n }\n // If we have good scores (100+), prefer them\n else if (boostedMatches.length > 8) {\n const goodMatches = boostedMatches.filter(m => m.score >= 100)\n if (goodMatches.length <= 5) {\n results = goodMatches\n }\n }\n\n return results.map(item => ({\n value: item.command,\n displayValue: `$ ${item.command}`,\n type: 'command' as const,\n score: item.score,\n metadata: { isUnixCommand: true },\n }))\n },\n [systemCommands, isLoadingCommands],\n )\n\n // Agent suggestions cache\n const [agentSuggestions, setAgentSuggestions] = useState<UnifiedSuggestion[]>(\n [],\n )\n\n // Model suggestions cache\n const [modelSuggestions, setModelSuggestions] = useState<UnifiedSuggestion[]>(\n [],\n )\n\n // Load model suggestions\n useEffect(() => {\n try {\n const modelManager = getModelManager()\n const allModels = modelManager.getAllAvailableModelNames()\n\n const suggestions = allModels.map(modelId => {\n // Professional and clear description for expert model consultation\n return {\n value: `ask-${modelId}`,\n displayValue: `\uD83E\uDD9C ask-${modelId} :: Consult ${modelId} for expert opinion and specialized analysis`,\n type: 'ask' as const,\n score: 90, // Higher than agents - put ask-models on top\n metadata: { modelId },\n }\n })\n\n setModelSuggestions(suggestions)\n } catch (error) {\n console.warn('[useUnifiedCompletion] Failed to load models:', error)\n // No fallback - rely on dynamic loading only\n setModelSuggestions([])\n }\n }, [])\n\n // MCP Resource suggestions cache\n const [mcpResourceSuggestions, setMcpResourceSuggestions] = useState<\n UnifiedSuggestion[]\n >([])\n\n // Load MCP resource suggestions\n useEffect(() => {\n listMCPResources()\n .then(resources => {\n const suggestions = resources.map(resource => ({\n value: resource.uri,\n displayValue: `\uD83D\uDD0C ${resource.name} :: ${resource.description || resource.uri}`,\n type: 'mcp-resource' as const,\n score: 80,\n metadata: {\n uri: resource.uri,\n serverName: resource.serverName,\n mimeType: resource.mimeType,\n },\n }))\n setMcpResourceSuggestions(suggestions)\n })\n .catch(error => {\n // MCP resources not available - this is fine\n console.warn(\n '[useUnifiedCompletion] MCP resources not available:',\n error,\n )\n setMcpResourceSuggestions([])\n })\n }, [])\n\n // Load agent suggestions on mount\n useEffect(() => {\n getActiveAgents()\n .then(agents => {\n // agents is an array of AgentConfig, not an object\n const suggestions = agents.map(config => {\n // \uD83E\uDDE0 \u667A\u80FD\u63CF\u8FF0\u7B97\u6CD5 - \u9002\u5E94\u6027\u957F\u5EA6\u63A7\u5236\n let shortDesc = config.whenToUse\n\n // \u79FB\u9664\u5E38\u89C1\u7684\u5197\u4F59\u524D\u7F00\uFF0C\u4F46\u4FDD\u7559\u6838\u5FC3\u5185\u5BB9\n const prefixPatterns = [\n /^Use this agent when you need (assistance with: )?/i,\n /^Use PROACTIVELY (when|to) /i,\n /^Specialized in /i,\n /^Implementation specialist for /i,\n /^Design validation specialist\\.? Use PROACTIVELY to /i,\n /^Task validation specialist\\.? Use PROACTIVELY to /i,\n /^Requirements validation specialist\\.? Use PROACTIVELY to /i,\n ]\n\n for (const pattern of prefixPatterns) {\n shortDesc = shortDesc.replace(pattern, '')\n }\n\n // \uD83C\uDFAF \u7CBE\u51C6\u65AD\u53E5\u7B97\u6CD5\uFF1A\u4E2D\u82F1\u6587\u53E5\u53F7\u611F\u53F9\u53F7\u4F18\u5148 \u2192 \u9017\u53F7 \u2192 \u7701\u7565\n const findSmartBreak = (text: string, maxLength: number) => {\n if (text.length <= maxLength) return text\n\n // \u7B2C\u4E00\u4F18\u5148\u7EA7\uFF1A\u4E2D\u82F1\u6587\u53E5\u53F7\u3001\u611F\u53F9\u53F7\n const sentenceEndings = /[.!\u3002\uFF01]/\n const firstSentenceMatch = text.search(sentenceEndings)\n if (firstSentenceMatch !== -1) {\n const firstSentence = text.slice(0, firstSentenceMatch).trim()\n if (firstSentence.length >= 5) {\n return firstSentence\n }\n }\n\n // \u5982\u679C\u7B2C\u4E00\u53E5\u8FC7\u957F\uFF0C\u627E\u9017\u53F7\u65AD\u53E5\n if (text.length > maxLength) {\n const commaEndings = /[,\uFF0C]/\n const commas = []\n let match\n const regex = new RegExp(commaEndings, 'g')\n while ((match = regex.exec(text)) !== null) {\n commas.push(match.index)\n }\n\n // \u627E\u6700\u540E\u4E00\u4E2A\u5728maxLength\u5185\u7684\u9017\u53F7\n for (let i = commas.length - 1; i >= 0; i--) {\n const commaPos = commas[i]\n if (commaPos < maxLength) {\n const clause = text.slice(0, commaPos).trim()\n if (clause.length >= 5) {\n return clause\n }\n }\n }\n }\n\n // \u6700\u540E\u9009\u62E9\uFF1A\u76F4\u63A5\u7701\u7565\n return text.slice(0, maxLength) + '...'\n }\n\n shortDesc = findSmartBreak(shortDesc.trim(), 80) // \u589E\u52A0\u523080\u5B57\u7B26\u9650\u5236\n\n // \u5982\u679C\u5904\u7406\u540E\u4E3A\u7A7A\u6216\u592A\u77ED\uFF0C\u4F7F\u7528\u539F\u59CB\u63CF\u8FF0\n if (!shortDesc || shortDesc.length < 5) {\n shortDesc = findSmartBreak(config.whenToUse, 80)\n }\n\n return {\n value: `run-agent-${config.agentType}`,\n displayValue: `\uD83D\uDC64 run-agent-${config.agentType} :: ${shortDesc}`, // \u4EBA\u7C7B\u56FE\u6807 + run-agent\u524D\u7F00 + \u7B80\u6D01\u63CF\u8FF0\n type: 'agent' as const,\n score: 85, // Lower than ask-models\n metadata: config,\n }\n })\n // Agents loaded successfully\n setAgentSuggestions(suggestions)\n })\n .catch(error => {\n console.warn('[useUnifiedCompletion] Failed to load agents:', error)\n // No fallback - rely on dynamic loading only\n setAgentSuggestions([])\n })\n }, [])\n\n // Generate agent and model suggestions using fuzzy matching\n const generateMentionSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n // Combine agent and model suggestions\n const allSuggestions = [...agentSuggestions, ...modelSuggestions]\n\n if (!prefix) {\n // Show all suggestions when prefix is empty (for single @)\n return allSuggestions.sort((a, b) => {\n // Ask models first (higher score), then agents\n if (a.type === 'ask' && b.type === 'agent') return -1\n if (a.type === 'agent' && b.type === 'ask') return 1\n return b.score - a.score\n })\n }\n\n // Use fuzzy matching for intelligent completion\n const candidates = allSuggestions.map(s => s.value)\n const matches = matchCommands(candidates, prefix)\n\n // Create result mapping with fuzzy scores\n const fuzzyResults = matches\n .map(match => {\n const suggestion = allSuggestions.find(\n s => s.value === match.command,\n )!\n return {\n ...suggestion,\n score: match.score, // Use fuzzy match score instead of simple scoring\n }\n })\n .sort((a, b) => {\n // Ask models first (for equal scores), then agents\n if (a.type === 'ask' && b.type === 'agent') return -1\n if (a.type === 'agent' && b.type === 'ask') return 1\n return b.score - a.score\n })\n\n return fuzzyResults\n },\n [agentSuggestions, modelSuggestions],\n )\n\n // Unix-style path completion - preserves user input semantics\n const generateFileSuggestions = useCallback(\n (prefix: string, isAtReference: boolean = false): UnifiedSuggestion[] => {\n try {\n const cwd = getCwd()\n\n // Parse user input preserving original format\n const userPath = prefix || '.'\n const isAbsolutePath = userPath.startsWith('/')\n const isHomePath = userPath.startsWith('~')\n\n // Resolve search directory - but keep user's path format for output\n let searchPath: string\n if (isHomePath) {\n searchPath = userPath.replace('~', process.env.HOME || '')\n } else if (isAbsolutePath) {\n searchPath = userPath\n } else {\n searchPath = resolve(cwd, userPath)\n }\n\n // Determine search directory and filename filter\n // If path ends with '/', treat it as directory navigation\n const endsWithSlash = userPath.endsWith('/')\n const searchStat = existsSync(searchPath) ? statSync(searchPath) : null\n\n let searchDir: string\n let nameFilter: string\n\n if (endsWithSlash || searchStat?.isDirectory()) {\n // User is navigating into a directory or path ends with /\n searchDir = searchPath\n nameFilter = ''\n } else {\n // User might be typing a partial filename\n searchDir = dirname(searchPath)\n nameFilter = basename(searchPath)\n }\n\n if (!existsSync(searchDir)) return []\n\n // Get directory entries with filter\n const showHidden = nameFilter.startsWith('.') || userPath.includes('/.')\n const entries = readdirSync(searchDir)\n .filter(entry => {\n // Filter hidden files unless user explicitly wants them\n if (!showHidden && entry.startsWith('.')) return false\n // Filter by name if there's a filter\n if (\n nameFilter &&\n !entry.toLowerCase().startsWith(nameFilter.toLowerCase())\n )\n return false\n return true\n })\n .sort((a, b) => {\n // Sort directories first, then files\n const aPath = join(searchDir, a)\n const bPath = join(searchDir, b)\n const aIsDir = statSync(aPath).isDirectory()\n const bIsDir = statSync(bPath).isDirectory()\n\n if (aIsDir && !bIsDir) return -1\n if (!aIsDir && bIsDir) return 1\n\n // Within same type, sort alphabetically\n return a.toLowerCase().localeCompare(b.toLowerCase())\n })\n .slice(0, 25) // Show more entries for better visibility\n\n return entries.map(entry => {\n const entryPath = join(searchDir, entry)\n const isDir = statSync(entryPath).isDirectory()\n const icon = isDir ? '\uD83D\uDCC1' : '\uD83D\uDCC4'\n\n // Unix-style path building - preserve user's original path format\n let value: string\n\n if (userPath.includes('/')) {\n // User typed path with separators - maintain structure\n if (endsWithSlash) {\n // User explicitly ended with / - they're inside the directory\n value = userPath + entry + (isDir ? '/' : '')\n } else if (searchStat?.isDirectory()) {\n // Path is a directory but doesn't end with / - add separator\n value = userPath + '/' + entry + (isDir ? '/' : '')\n } else {\n // User is completing a filename - replace basename\n const userDir = userPath.includes('/')\n ? userPath.substring(0, userPath.lastIndexOf('/'))\n : ''\n value = userDir\n ? userDir + '/' + entry + (isDir ? '/' : '')\n : entry + (isDir ? '/' : '')\n }\n } else {\n // User typed simple name - check if it's an existing directory\n if (searchStat?.isDirectory()) {\n // Existing directory - navigate into it\n value = userPath + '/' + entry + (isDir ? '/' : '')\n } else {\n // Simple completion at current level\n value = entry + (isDir ? '/' : '')\n }\n }\n\n return {\n value,\n displayValue: `${icon} ${entry}${isDir ? '/' : ''}`,\n type: 'file' as const,\n score: isDir ? 80 : 70,\n }\n })\n } catch {\n return []\n }\n },\n [],\n )\n\n // Unified smart matching - single algorithm with different weights\n const calculateMatchScore = useCallback(\n (suggestion: UnifiedSuggestion, prefix: string): number => {\n const lowerPrefix = prefix.toLowerCase()\n const value = suggestion.value.toLowerCase()\n const displayValue = suggestion.displayValue.toLowerCase()\n\n let matchFound = false\n let score = 0\n\n // Check for actual matches first\n if (value.startsWith(lowerPrefix)) {\n matchFound = true\n score = 100 // Highest priority\n } else if (value.includes(lowerPrefix)) {\n matchFound = true\n score = 95\n } else if (displayValue.includes(lowerPrefix)) {\n matchFound = true\n score = 90\n } else {\n // Word boundary matching for compound names like \"general\" -> \"run-agent-general-purpose\"\n const words = value.split(/[-_]/)\n if (words.some(word => word.startsWith(lowerPrefix))) {\n matchFound = true\n score = 93\n } else {\n // Acronym matching (last resort)\n const acronym = words.map(word => word[0]).join('')\n if (acronym.startsWith(lowerPrefix)) {\n matchFound = true\n score = 88\n }\n }\n }\n\n // Only return score if we found a match\n if (!matchFound) return 0\n\n // Type preferences (small bonus)\n if (suggestion.type === 'ask') score += 2\n if (suggestion.type === 'agent') score += 1\n\n return score\n },\n [],\n )\n\n // Generate smart mention suggestions without data pollution\n const generateSmartMentionSuggestions = useCallback(\n (\n prefix: string,\n sourceContext: 'file' | 'agent' = 'file',\n ): UnifiedSuggestion[] => {\n if (!prefix || prefix.length < 2) return []\n\n const allSuggestions = [...agentSuggestions, ...modelSuggestions]\n\n return allSuggestions\n .map(suggestion => {\n const matchScore = calculateMatchScore(suggestion, prefix)\n if (matchScore === 0) return null\n\n // Clean transformation without data pollution\n return {\n ...suggestion,\n score: matchScore,\n isSmartMatch: true,\n originalContext: sourceContext,\n // Only modify display for clarity, keep value clean\n displayValue: `\uD83C\uDFAF ${suggestion.displayValue}`,\n }\n })\n .filter(Boolean)\n .sort((a, b) => b.score - a.score)\n .slice(0, 5)\n },\n [agentSuggestions, modelSuggestions, calculateMatchScore],\n )\n\n // Generate MCP resource suggestions\n const generateMcpResourceSuggestions = useCallback(\n (prefix: string): UnifiedSuggestion[] => {\n if (mcpResourceSuggestions.length === 0) {\n // No MCP resources available\n return [\n {\n value: '',\n displayValue: '\u26A0\uFE0F No MCP resources available',\n type: 'mcp-resource' as const,\n score: 0,\n metadata: { isPlaceholder: true },\n },\n ]\n }\n\n if (!prefix) {\n // Show all MCP resources when no prefix\n return mcpResourceSuggestions\n }\n\n // Filter by prefix (match on uri, name, or server)\n const lowerPrefix = prefix.toLowerCase()\n return mcpResourceSuggestions\n .filter(s => {\n const uri = s.value.toLowerCase()\n const name = s.metadata?.uri?.toLowerCase() || ''\n const server = s.metadata?.serverName?.toLowerCase() || ''\n return (\n uri.includes(lowerPrefix) ||\n name.includes(lowerPrefix) ||\n server.includes(lowerPrefix)\n )\n })\n .map(s => ({\n ...s,\n score: s.value.toLowerCase().startsWith(lowerPrefix) ? 100 : 80,\n }))\n .sort((a, b) => b.score - a.score)\n },\n [mcpResourceSuggestions],\n )\n\n // Generate all suggestions based on context\n const generateSuggestions = useCallback(\n (context: CompletionContext): UnifiedSuggestion[] => {\n switch (context.type) {\n case 'command':\n return generateCommandSuggestions(context.prefix)\n case 'mcp':\n return generateMcpResourceSuggestions(context.prefix)\n case 'agent': {\n // @ reference: combine mentions and files with clean priority\n const mentionSuggestions = generateMentionSuggestions(context.prefix)\n const fileSuggestions = generateFileSuggestions(context.prefix, true) // isAtReference=true\n\n // Apply weights for @ context (agents/models should be prioritized but files visible)\n const weightedSuggestions = [\n ...mentionSuggestions.map(s => ({\n ...s,\n // In @ context, agents/models get high priority\n weightedScore: s.score + 150,\n })),\n ...fileSuggestions.map(s => ({\n ...s,\n // Files get lower priority but still visible\n weightedScore: s.score + 10, // Small boost to ensure visibility\n })),\n ]\n\n // Sort by weighted score - no artificial limits\n return weightedSuggestions\n .sort((a, b) => b.weightedScore - a.weightedScore)\n .map(({ weightedScore, ...suggestion }) => suggestion)\n // No limit or very generous limit (e.g., 30 items)\n }\n case 'file': {\n // For normal input, try to match everything intelligently\n const fileSuggestions = generateFileSuggestions(context.prefix, false)\n const unixSuggestions = generateUnixCommandSuggestions(context.prefix)\n\n // IMPORTANT: Also try to match agents and models WITHOUT requiring @\n // This enables smart matching for inputs like \"gp5\", \"daoqi\", etc.\n const mentionMatches = generateMentionSuggestions(context.prefix).map(\n s => ({\n ...s,\n isSmartMatch: true,\n // Show that @ will be added when selected\n displayValue: `\\u2192 ${s.displayValue}`, // Arrow to indicate it will transform\n }),\n )\n\n // Apply source-based priority weights with special handling for exact matches\n // Priority order: Exact Unix > Unix commands > agents/models > files\n const weightedSuggestions = [\n ...unixSuggestions.map(s => ({\n ...s,\n // Unix commands get boost, but exact matches get huge boost\n sourceWeight: s.score >= 10000 ? 5000 : 200, // Exact match gets massive boost\n weightedScore: s.score >= 10000 ? s.score + 5000 : s.score + 200,\n })),\n ...mentionMatches.map(s => ({\n ...s,\n // Agents/models get medium priority boost (but less to avoid overriding exact Unix)\n sourceWeight: 50,\n weightedScore: s.score + 50,\n })),\n ...fileSuggestions.map(s => ({\n ...s,\n // Files get no boost (baseline)\n sourceWeight: 0,\n weightedScore: s.score,\n })),\n ]\n\n // Sort by weighted score and deduplicate\n const seen = new Set<string>()\n const deduplicatedResults = weightedSuggestions\n .sort((a, b) => b.weightedScore - a.weightedScore)\n .filter(item => {\n // Filter out duplicates based on value\n if (seen.has(item.value)) return false\n seen.add(item.value)\n return true\n })\n .map(({ weightedScore, sourceWeight, ...suggestion }) => suggestion) // Remove weight fields\n // No limit - show all relevant matches\n\n return deduplicatedResults\n }\n default:\n return []\n }\n },\n [\n generateCommandSuggestions,\n generateMcpResourceSuggestions,\n generateMentionSuggestions,\n generateFileSuggestions,\n generateUnixCommandSuggestions,\n generateSmartMentionSuggestions,\n ],\n )\n\n // Complete with a suggestion - \u652F\u6301\u4E07\u80FD@\u5F15\u7528 + slash\u547D\u4EE4\u81EA\u52A8\u6267\u884C\n const completeWith = useCallback(\n (suggestion: UnifiedSuggestion, context: CompletionContext) => {\n let completion: string\n\n if (context.type === 'command') {\n completion = `/${suggestion.value} `\n } else if (context.type === 'mcp') {\n // MCP resource completion - add mcp: prefix and space\n completion = `mcp:${suggestion.value} `\n } else if (context.type === 'agent') {\n // \uD83D\uDE80 \u4E07\u80FD@\u5F15\u7528\uFF1A\u6839\u636E\u5EFA\u8BAE\u7C7B\u578B\u51B3\u5B9A\u8865\u5168\u683C\u5F0F\n if (suggestion.type === 'agent') {\n completion = `@${suggestion.value} ` // \u4EE3\u7406\u8865\u5168\n } else if (suggestion.type === 'ask') {\n completion = `@${suggestion.value} ` // Ask\u6A21\u578B\u8865\u5168\n } else {\n // File reference in @mention context - no space for directories to allow expansion\n const isDirectory = suggestion.value.endsWith('/')\n completion = `@${suggestion.value}${isDirectory ? '' : ' '}` // \u6587\u4EF6\u5939\u4E0D\u52A0\u7A7A\u683C\uFF0C\u6587\u4EF6\u52A0\u7A7A\u683C\n }\n } else {\n // Regular file completion OR smart mention matching\n if (suggestion.isSmartMatch) {\n // Smart mention - add @ prefix and space\n completion = `@${suggestion.value} `\n } else {\n // Regular file completion - no space for directories to allow expansion\n const isDirectory = suggestion.value.endsWith('/')\n completion = suggestion.value + (isDirectory ? '' : ' ')\n }\n }\n\n // Special handling for absolute paths in file completion\n // When completing an absolute path, we should replace the entire current word/path\n let actualEndPos: number\n\n if (\n context.type === 'file' &&\n suggestion.value.startsWith('/') &&\n !suggestion.isSmartMatch\n ) {\n // For absolute paths, find the end of the current path/word\n let end = context.startPos\n while (\n end < input.length &&\n input[end] !== ' ' &&\n input[end] !== '\\n'\n ) {\n end++\n }\n actualEndPos = end\n } else {\n // Original logic for other cases\n const currentWord = input.slice(context.startPos)\n const nextSpaceIndex = currentWord.indexOf(' ')\n actualEndPos =\n nextSpaceIndex === -1\n ? input.length\n : context.startPos + nextSpaceIndex\n }\n\n const newInput =\n input.slice(0, context.startPos) +\n completion +\n input.slice(actualEndPos)\n onInputChange(newInput)\n setCursorOffset(context.startPos + completion.length)\n\n // Don't auto-execute slash commands - let user press Enter to submit\n // This gives users a chance to add arguments or modify the command\n\n // Completion applied\n },\n [input, onInputChange, setCursorOffset, onSubmit, commands],\n )\n\n // Partial complete to common prefix\n const partialComplete = useCallback(\n (prefix: string, context: CompletionContext) => {\n const completion =\n context.type === 'command'\n ? `/${prefix}`\n : context.type === 'agent'\n ? `@${prefix}`\n : prefix\n\n const newInput =\n input.slice(0, context.startPos) +\n completion +\n input.slice(context.endPos)\n onInputChange(newInput)\n setCursorOffset(context.startPos + completion.length)\n },\n [input, onInputChange, setCursorOffset],\n )\n\n // Handle Tab key - simplified and unified\n useInput((input_str, key) => {\n if (!key.tab) return false\n if (key.shift) return false\n\n const context = getWordAtCursor()\n if (!context) return false\n\n // If menu is already showing, cycle through suggestions\n if (state.isActive && state.suggestions.length > 0) {\n const nextIndex = (state.selectedIndex + 1) % state.suggestions.length\n const nextSuggestion = state.suggestions[nextIndex]\n\n if (state.context) {\n // Calculate proper word boundaries\n const currentWord = input.slice(state.context.startPos)\n const wordEnd = currentWord.search(/\\s/)\n const actualEndPos =\n wordEnd === -1 ? input.length : state.context.startPos + wordEnd\n\n // Apply appropriate prefix based on context type and suggestion type\n let preview: string\n if (state.context.type === 'command') {\n preview = `/${nextSuggestion.value}`\n } else if (state.context.type === 'agent') {\n // For @mentions, always add @ prefix\n preview = `@${nextSuggestion.value}`\n } else if (nextSuggestion.isSmartMatch) {\n // Smart match from normal input - add @ prefix\n preview = `@${nextSuggestion.value}`\n } else {\n preview = nextSuggestion.value\n }\n\n // Apply preview\n const newInput =\n input.slice(0, state.context.startPos) +\n preview +\n input.slice(actualEndPos)\n\n onInputChange(newInput)\n setCursorOffset(state.context.startPos + preview.length)\n\n // Update state\n updateState({\n selectedIndex: nextIndex,\n preview: {\n isActive: true,\n originalInput: input,\n wordRange: [\n state.context.startPos,\n state.context.startPos + preview.length,\n ],\n },\n })\n }\n return true\n }\n\n // Generate new suggestions\n const currentSuggestions = generateSuggestions(context)\n\n if (currentSuggestions.length === 0) {\n return false // Let Tab pass through\n } else if (currentSuggestions.length === 1) {\n // Single match: complete immediately\n completeWith(currentSuggestions[0], context)\n return true\n } else {\n // Show menu and apply first suggestion\n activateCompletion(currentSuggestions, context)\n\n // Immediately apply first suggestion as preview\n const firstSuggestion = currentSuggestions[0]\n const currentWord = input.slice(context.startPos)\n const wordEnd = currentWord.search(/\\s/)\n const actualEndPos =\n wordEnd === -1 ? input.length : context.startPos + wordEnd\n\n let preview: string\n if (context.type === 'command') {\n preview = `/${firstSuggestion.value}`\n } else if (context.type === 'agent') {\n preview = `@${firstSuggestion.value}`\n } else if (firstSuggestion.isSmartMatch) {\n // Smart match from normal input - add @ prefix\n preview = `@${firstSuggestion.value}`\n } else {\n preview = firstSuggestion.value\n }\n\n const newInput =\n input.slice(0, context.startPos) + preview + input.slice(actualEndPos)\n\n onInputChange(newInput)\n setCursorOffset(context.startPos + preview.length)\n\n updateState({\n preview: {\n isActive: true,\n originalInput: input,\n wordRange: [context.startPos, context.startPos + preview.length],\n },\n })\n\n return true\n }\n })\n\n // Handle navigation keys - simplified and unified\n useInput((inputChar, key) => {\n // Enter key - confirm selection and end completion (always add space)\n if (key.return && state.isActive && state.suggestions.length > 0) {\n const selectedSuggestion = state.suggestions[state.selectedIndex]\n if (selectedSuggestion && state.context) {\n // For Enter key, always add space even for directories to indicate completion end\n let completion: string\n\n if (state.context.type === 'command') {\n completion = `/${selectedSuggestion.value} `\n } else if (state.context.type === 'agent') {\n if (selectedSuggestion.type === 'agent') {\n completion = `@${selectedSuggestion.value} `\n } else if (selectedSuggestion.type === 'ask') {\n completion = `@${selectedSuggestion.value} `\n } else {\n // File reference in @mention context - always add space on Enter\n completion = `@${selectedSuggestion.value} `\n }\n } else if (selectedSuggestion.isSmartMatch) {\n // Smart match from normal input - add @ prefix\n completion = `@${selectedSuggestion.value} `\n } else {\n // Regular file completion - always add space on Enter\n completion = selectedSuggestion.value + ' '\n }\n\n // Apply completion with forced space\n const currentWord = input.slice(state.context.startPos)\n const nextSpaceIndex = currentWord.indexOf(' ')\n const actualEndPos =\n nextSpaceIndex === -1\n ? input.length\n : state.context.startPos + nextSpaceIndex\n\n const newInput =\n input.slice(0, state.context.startPos) +\n completion +\n input.slice(actualEndPos)\n onInputChange(newInput)\n setCursorOffset(state.context.startPos + completion.length)\n }\n resetCompletion()\n return true\n }\n\n if (!state.isActive || state.suggestions.length === 0) return false\n\n // Arrow key navigation with preview\n const handleNavigation = (newIndex: number) => {\n const preview = state.suggestions[newIndex].value\n\n if (state.preview?.isActive && state.context) {\n const newInput =\n input.slice(0, state.context.startPos) +\n preview +\n input.slice(state.preview.wordRange[1])\n\n onInputChange(newInput)\n setCursorOffset(state.context.startPos + preview.length)\n\n updateState({\n selectedIndex: newIndex,\n preview: {\n ...state.preview,\n wordRange: [\n state.context.startPos,\n state.context.startPos + preview.length,\n ],\n },\n })\n } else {\n updateState({ selectedIndex: newIndex })\n }\n }\n\n if (key.downArrow) {\n const nextIndex = (state.selectedIndex + 1) % state.suggestions.length\n handleNavigation(nextIndex)\n return true\n }\n\n if (key.upArrow) {\n const nextIndex =\n state.selectedIndex === 0\n ? state.suggestions.length - 1\n : state.selectedIndex - 1\n handleNavigation(nextIndex)\n return true\n }\n\n // Space key - complete and potentially continue for directories\n if (inputChar === ' ' && state.isActive && state.suggestions.length > 0) {\n const selectedSuggestion = state.suggestions[state.selectedIndex]\n const isDirectory = selectedSuggestion.value.endsWith('/')\n\n if (!state.context) return false\n\n // Apply completion if needed\n const currentWordAtContext = input.slice(\n state.context.startPos,\n state.context.startPos + selectedSuggestion.value.length,\n )\n\n if (currentWordAtContext !== selectedSuggestion.value) {\n completeWith(selectedSuggestion, state.context)\n }\n\n resetCompletion()\n\n if (isDirectory) {\n // Continue completion for directories\n setTimeout(() => {\n const newContext = {\n ...state.context,\n prefix: selectedSuggestion.value,\n endPos: state.context.startPos + selectedSuggestion.value.length,\n }\n\n const newSuggestions = generateSuggestions(newContext)\n\n if (newSuggestions.length > 0) {\n activateCompletion(newSuggestions, newContext)\n } else {\n updateState({\n emptyDirMessage: `Directory is empty: ${selectedSuggestion.value}`,\n })\n setTimeout(() => updateState({ emptyDirMessage: '' }), 3000)\n }\n }, 50)\n }\n\n return true\n }\n\n // Right arrow key - same as space but different semantics\n if (key.rightArrow) {\n const selectedSuggestion = state.suggestions[state.selectedIndex]\n const isDirectory = selectedSuggestion.value.endsWith('/')\n\n if (!state.context) return false\n\n // Apply completion\n const currentWordAtContext = input.slice(\n state.context.startPos,\n state.context.startPos + selectedSuggestion.value.length,\n )\n\n if (currentWordAtContext !== selectedSuggestion.value) {\n completeWith(selectedSuggestion, state.context)\n }\n\n resetCompletion()\n\n if (isDirectory) {\n // Continue for directories\n setTimeout(() => {\n const newContext = {\n ...state.context,\n prefix: selectedSuggestion.value,\n endPos: state.context.startPos + selectedSuggestion.value.length,\n }\n\n const newSuggestions = generateSuggestions(newContext)\n\n if (newSuggestions.length > 0) {\n activateCompletion(newSuggestions, newContext)\n } else {\n updateState({\n emptyDirMessage: `Directory is empty: ${selectedSuggestion.value}`,\n })\n setTimeout(() => updateState({ emptyDirMessage: '' }), 3000)\n }\n }, 50)\n }\n\n return true\n }\n\n if (key.escape) {\n // Restore original text if in preview mode\n if (state.preview?.isActive && state.context) {\n onInputChange(state.preview.originalInput)\n setCursorOffset(state.context.startPos + state.context.prefix.length)\n }\n\n resetCompletion()\n return true\n }\n\n return false\n })\n\n // Handle delete/backspace keys - unified state management\n useInput((input_str, key) => {\n if (key.backspace || key.delete) {\n if (state.isActive) {\n resetCompletion()\n // Smart suppression based on input complexity\n const suppressionTime = input.length > 10 ? 200 : 100\n updateState({\n suppressUntil: Date.now() + suppressionTime,\n })\n return true\n }\n }\n return false\n })\n\n // Input tracking with ref to avoid infinite loops\n const lastInputRef = useRef('')\n\n // Smart auto-triggering with cycle prevention\n useEffect(() => {\n // Prevent infinite loops by using ref\n if (lastInputRef.current === input) return\n\n const inputLengthChange = Math.abs(\n input.length - lastInputRef.current.length,\n )\n const isHistoryNavigation =\n (inputLengthChange > 10 || // Large content change\n (inputLengthChange > 5 &&\n !input.includes(lastInputRef.current.slice(-5)))) && // Different content\n input !== lastInputRef.current\n\n // Update ref (no state update)\n lastInputRef.current = input\n\n // Skip if in preview mode or suppressed\n if (state.preview?.isActive || Date.now() < state.suppressUntil) {\n return\n }\n\n // Clear suggestions on history navigation\n if (isHistoryNavigation && state.isActive) {\n resetCompletion()\n return\n }\n\n const context = getWordAtCursor()\n\n if (context && shouldAutoTrigger(context)) {\n const newSuggestions = generateSuggestions(context)\n\n if (newSuggestions.length === 0) {\n resetCompletion()\n } else if (\n newSuggestions.length === 1 &&\n shouldAutoHideSingleMatch(newSuggestions[0], context)\n ) {\n resetCompletion() // Perfect match - hide\n } else {\n activateCompletion(newSuggestions, context)\n }\n } else if (state.context) {\n // Check if context changed significantly\n const contextChanged =\n !context ||\n state.context.type !== context.type ||\n state.context.startPos !== context.startPos ||\n !context.prefix.startsWith(state.context.prefix)\n\n if (contextChanged) {\n resetCompletion()\n }\n }\n }, [input, cursorOffset])\n\n // Smart triggering - only when it makes sense\n const shouldAutoTrigger = useCallback(\n (context: CompletionContext): boolean => {\n switch (context.type) {\n case 'command':\n // Trigger immediately for slash commands\n return true\n case 'agent':\n // Trigger immediately for agent references\n return true\n case 'mcp':\n // Trigger immediately for MCP resource references\n return true\n case 'file':\n // Be selective about file completion - avoid noise\n const prefix = context.prefix\n\n // Always trigger for clear path patterns\n if (\n prefix.startsWith('./') ||\n prefix.startsWith('../') ||\n prefix.startsWith('/') ||\n prefix.startsWith('~') ||\n prefix.includes('/')\n ) {\n return true\n }\n\n // Trigger for single dot followed by something (like .g for .gitignore)\n if (prefix.startsWith('.') && prefix.length >= 2) {\n return true\n }\n\n // Skip very short prefixes that are likely code\n return false\n default:\n return false\n }\n },\n [],\n )\n\n // Helper function to determine if single suggestion should be auto-hidden\n const shouldAutoHideSingleMatch = useCallback(\n (suggestion: UnifiedSuggestion, context: CompletionContext): boolean => {\n // Extract the actual typed input from context\n const currentInput = input.slice(context.startPos, context.endPos)\n // Check if should auto-hide single match\n\n // For files: more intelligent matching\n if (context.type === 'file') {\n // Special case: if suggestion is a directory (ends with /), don't auto-hide\n // because user might want to continue navigating into it\n if (suggestion.value.endsWith('/')) {\n // Directory suggestion, keeping visible\n return false\n }\n\n // Check exact match\n if (currentInput === suggestion.value) {\n // Exact match, hiding\n return true\n }\n\n // Check if current input is a complete file path and suggestion is just the filename\n // e.g., currentInput: \"src/tools/ThinkTool/ThinkTool.tsx\", suggestion: \"ThinkTool.tsx\"\n if (\n currentInput.endsWith('/' + suggestion.value) ||\n currentInput.endsWith(suggestion.value)\n ) {\n // Path ends with suggestion, hiding\n return true\n }\n\n return false\n }\n\n // For commands: check if /prefix exactly matches /command\n if (context.type === 'command') {\n const fullCommand = `/${suggestion.value}`\n const matches = currentInput === fullCommand\n // Check command match\n return matches\n }\n\n // For agents: check if @prefix exactly matches @agent-name\n if (context.type === 'agent') {\n const fullAgent = `@${suggestion.value}`\n const matches = currentInput === fullAgent\n // Check agent match\n return matches\n }\n\n return false\n },\n [input],\n )\n\n return {\n suggestions,\n selectedIndex,\n isActive,\n emptyDirMessage,\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["suggestions", "readdirSync", "statSync", "commands"]
|
|
7
7
|
}
|