@butlerw/vellum 0.1.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.
Files changed (446) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +411 -0
  3. package/__fixtures__/responses/code-generation.json +42 -0
  4. package/__fixtures__/responses/error-response.json +20 -0
  5. package/__fixtures__/responses/hello-world.json +32 -0
  6. package/dist/auth-6MCXESOH.js +26 -0
  7. package/dist/chunk-SECXJGWA.js +597 -0
  8. package/dist/index.js +34023 -0
  9. package/package.json +67 -0
  10. package/src/__tests__/commands.e2e.test.ts +728 -0
  11. package/src/__tests__/credentials.test.ts +713 -0
  12. package/src/__tests__/mode-e2e.test.ts +391 -0
  13. package/src/__tests__/tui-integration.test.tsx +1271 -0
  14. package/src/agents/__tests__/task-persistence.test.ts +235 -0
  15. package/src/agents/commands/delegate.ts +240 -0
  16. package/src/agents/commands/index.ts +10 -0
  17. package/src/agents/commands/resume.ts +335 -0
  18. package/src/agents/index.ts +29 -0
  19. package/src/agents/task-persistence.ts +272 -0
  20. package/src/agents/task-resumption.ts +242 -0
  21. package/src/app.tsx +4737 -0
  22. package/src/commands/__tests__/.gitkeep +1 -0
  23. package/src/commands/__tests__/agents.test.ts +606 -0
  24. package/src/commands/__tests__/auth.test.ts +626 -0
  25. package/src/commands/__tests__/autocomplete.test.ts +683 -0
  26. package/src/commands/__tests__/batch.test.ts +287 -0
  27. package/src/commands/__tests__/chain-pipe-parser.test.ts +654 -0
  28. package/src/commands/__tests__/completion.test.ts +238 -0
  29. package/src/commands/__tests__/core.test.ts +363 -0
  30. package/src/commands/__tests__/executor.test.ts +496 -0
  31. package/src/commands/__tests__/exit-codes.test.ts +220 -0
  32. package/src/commands/__tests__/init.test.ts +243 -0
  33. package/src/commands/__tests__/language.test.ts +353 -0
  34. package/src/commands/__tests__/mode-cli.test.ts +667 -0
  35. package/src/commands/__tests__/model.test.ts +277 -0
  36. package/src/commands/__tests__/parser.test.ts +493 -0
  37. package/src/commands/__tests__/performance.bench.ts +380 -0
  38. package/src/commands/__tests__/registry.test.ts +534 -0
  39. package/src/commands/__tests__/resume.test.ts +449 -0
  40. package/src/commands/__tests__/security.test.ts +845 -0
  41. package/src/commands/__tests__/stream-json.test.ts +372 -0
  42. package/src/commands/__tests__/user-commands.test.ts +597 -0
  43. package/src/commands/adapters.ts +267 -0
  44. package/src/commands/agent.ts +395 -0
  45. package/src/commands/agents/generate.ts +506 -0
  46. package/src/commands/agents/index.ts +272 -0
  47. package/src/commands/agents/show.ts +271 -0
  48. package/src/commands/agents/validate.ts +387 -0
  49. package/src/commands/auth.ts +883 -0
  50. package/src/commands/autocomplete.ts +480 -0
  51. package/src/commands/batch/command.ts +388 -0
  52. package/src/commands/batch/executor.ts +361 -0
  53. package/src/commands/batch/index.ts +12 -0
  54. package/src/commands/commit.ts +235 -0
  55. package/src/commands/completion/index.ts +371 -0
  56. package/src/commands/condense.ts +191 -0
  57. package/src/commands/config.ts +344 -0
  58. package/src/commands/context-provider.ts +173 -0
  59. package/src/commands/copy.ts +329 -0
  60. package/src/commands/core/clear.ts +38 -0
  61. package/src/commands/core/exit.ts +43 -0
  62. package/src/commands/core/help.ts +354 -0
  63. package/src/commands/core/index.ts +15 -0
  64. package/src/commands/cost.ts +179 -0
  65. package/src/commands/credentials.tsx +618 -0
  66. package/src/commands/custom-agents/__tests__/custom-agents.test.ts +709 -0
  67. package/src/commands/custom-agents/create.ts +377 -0
  68. package/src/commands/custom-agents/export.ts +135 -0
  69. package/src/commands/custom-agents/import.ts +199 -0
  70. package/src/commands/custom-agents/index.ts +372 -0
  71. package/src/commands/custom-agents/info.ts +318 -0
  72. package/src/commands/custom-agents/list.ts +267 -0
  73. package/src/commands/custom-agents/validate.ts +388 -0
  74. package/src/commands/diff-mode.ts +241 -0
  75. package/src/commands/env.ts +53 -0
  76. package/src/commands/executor.ts +579 -0
  77. package/src/commands/exit-codes.ts +202 -0
  78. package/src/commands/index.ts +701 -0
  79. package/src/commands/init/index.ts +15 -0
  80. package/src/commands/init/prompts.ts +366 -0
  81. package/src/commands/init/templates/commands-readme.md +80 -0
  82. package/src/commands/init/templates/example-command.md +79 -0
  83. package/src/commands/init/templates/example-skill.md +168 -0
  84. package/src/commands/init/templates/example-workflow.md +101 -0
  85. package/src/commands/init/templates/prompts-readme.md +52 -0
  86. package/src/commands/init/templates/rules-readme.md +63 -0
  87. package/src/commands/init/templates/skills-readme.md +83 -0
  88. package/src/commands/init/templates/workflows-readme.md +94 -0
  89. package/src/commands/init.ts +391 -0
  90. package/src/commands/install.ts +90 -0
  91. package/src/commands/language.ts +191 -0
  92. package/src/commands/loaders/.gitkeep +1 -0
  93. package/src/commands/lsp.ts +199 -0
  94. package/src/commands/markdown-commands.ts +253 -0
  95. package/src/commands/mcp.ts +588 -0
  96. package/src/commands/memory/export.ts +341 -0
  97. package/src/commands/memory/index.ts +148 -0
  98. package/src/commands/memory/list.ts +261 -0
  99. package/src/commands/memory/search.ts +346 -0
  100. package/src/commands/memory/utils.ts +15 -0
  101. package/src/commands/metrics.ts +75 -0
  102. package/src/commands/migrate/index.ts +16 -0
  103. package/src/commands/migrate/prompts.ts +477 -0
  104. package/src/commands/mode.ts +331 -0
  105. package/src/commands/model.ts +298 -0
  106. package/src/commands/onboard.ts +205 -0
  107. package/src/commands/open.ts +169 -0
  108. package/src/commands/output/stream-json.ts +373 -0
  109. package/src/commands/parser/chain-parser.ts +370 -0
  110. package/src/commands/parser/index.ts +29 -0
  111. package/src/commands/parser/pipe-parser.ts +480 -0
  112. package/src/commands/parser.ts +588 -0
  113. package/src/commands/persistence.ts +355 -0
  114. package/src/commands/progress.ts +18 -0
  115. package/src/commands/prompt/index.ts +17 -0
  116. package/src/commands/prompt/validate.ts +621 -0
  117. package/src/commands/prompt-priority.ts +401 -0
  118. package/src/commands/registry.ts +374 -0
  119. package/src/commands/sandbox/index.ts +131 -0
  120. package/src/commands/security/index.ts +21 -0
  121. package/src/commands/security/input-sanitizer.ts +168 -0
  122. package/src/commands/security/permission-checker.ts +456 -0
  123. package/src/commands/security/sensitive-data.ts +350 -0
  124. package/src/commands/session/delete.ts +38 -0
  125. package/src/commands/session/export.ts +39 -0
  126. package/src/commands/session/index.ts +26 -0
  127. package/src/commands/session/list.ts +26 -0
  128. package/src/commands/session/resume.ts +562 -0
  129. package/src/commands/session/search.ts +434 -0
  130. package/src/commands/session/show.ts +26 -0
  131. package/src/commands/settings.ts +368 -0
  132. package/src/commands/setup.ts +23 -0
  133. package/src/commands/shell/index.ts +16 -0
  134. package/src/commands/shell/setup.ts +422 -0
  135. package/src/commands/shell-init.ts +50 -0
  136. package/src/commands/shell-integration/index.ts +194 -0
  137. package/src/commands/skill.ts +1220 -0
  138. package/src/commands/spec.ts +558 -0
  139. package/src/commands/status.ts +246 -0
  140. package/src/commands/theme.ts +211 -0
  141. package/src/commands/think.ts +551 -0
  142. package/src/commands/trust.ts +211 -0
  143. package/src/commands/tutorial.ts +522 -0
  144. package/src/commands/types.ts +512 -0
  145. package/src/commands/update.ts +274 -0
  146. package/src/commands/usage.ts +213 -0
  147. package/src/commands/user-commands.ts +630 -0
  148. package/src/commands/utils.ts +142 -0
  149. package/src/commands/vim.ts +152 -0
  150. package/src/commands/workflow.ts +257 -0
  151. package/src/components/header.tsx +25 -0
  152. package/src/components/input.tsx +25 -0
  153. package/src/components/message-list.tsx +32 -0
  154. package/src/components/status-bar.tsx +23 -0
  155. package/src/index.tsx +614 -0
  156. package/src/onboarding/__tests__/tutorial.test.ts +740 -0
  157. package/src/onboarding/index.ts +69 -0
  158. package/src/onboarding/tips/index.ts +9 -0
  159. package/src/onboarding/tips/tip-engine.ts +459 -0
  160. package/src/onboarding/tutorial/index.ts +88 -0
  161. package/src/onboarding/tutorial/lessons/basics.ts +151 -0
  162. package/src/onboarding/tutorial/lessons/index.ts +151 -0
  163. package/src/onboarding/tutorial/lessons/modes.ts +230 -0
  164. package/src/onboarding/tutorial/lessons/tools.ts +172 -0
  165. package/src/onboarding/tutorial/progress-tracker.ts +350 -0
  166. package/src/onboarding/tutorial/storage.ts +249 -0
  167. package/src/onboarding/tutorial/tutorial-system.ts +462 -0
  168. package/src/onboarding/tutorial/types.ts +310 -0
  169. package/src/orchestrator-singleton.ts +129 -0
  170. package/src/shutdown.ts +33 -0
  171. package/src/test/e2e/assertions.ts +267 -0
  172. package/src/test/e2e/fixtures.ts +204 -0
  173. package/src/test/e2e/harness.ts +575 -0
  174. package/src/test/e2e/index.ts +57 -0
  175. package/src/test/e2e/types.ts +228 -0
  176. package/src/test/fixtures/__tests__/fake-response-loader.test.ts +314 -0
  177. package/src/test/fixtures/fake-response-loader.ts +314 -0
  178. package/src/test/fixtures/index.ts +20 -0
  179. package/src/tui/__tests__/mcp-panel.test.tsx +82 -0
  180. package/src/tui/__tests__/mcp-wiring.test.tsx +78 -0
  181. package/src/tui/__tests__/mode-components.test.tsx +395 -0
  182. package/src/tui/__tests__/permission-ask-flow.test.tsx +138 -0
  183. package/src/tui/__tests__/sidebar-panel-data.test.tsx +148 -0
  184. package/src/tui/__tests__/tools-panel-hotkeys.test.tsx +41 -0
  185. package/src/tui/adapters/agent-adapter.ts +1008 -0
  186. package/src/tui/adapters/index.ts +48 -0
  187. package/src/tui/adapters/message-adapter.ts +315 -0
  188. package/src/tui/adapters/persistence-bridge.ts +331 -0
  189. package/src/tui/adapters/session-adapter.ts +419 -0
  190. package/src/tui/buffered-stdout.ts +223 -0
  191. package/src/tui/components/AgentProgress.tsx +424 -0
  192. package/src/tui/components/Banner/AsciiArt.ts +160 -0
  193. package/src/tui/components/Banner/Banner.tsx +355 -0
  194. package/src/tui/components/Banner/ShimmerContext.tsx +131 -0
  195. package/src/tui/components/Banner/ShimmerText.tsx +193 -0
  196. package/src/tui/components/Banner/TypeWriterGradient.tsx +321 -0
  197. package/src/tui/components/Banner/index.ts +61 -0
  198. package/src/tui/components/Banner/useShimmer.ts +241 -0
  199. package/src/tui/components/ChatView.tsx +11 -0
  200. package/src/tui/components/Checkpoint/CheckpointDiffView.tsx +371 -0
  201. package/src/tui/components/Checkpoint/SnapshotCheckpointPanel.tsx +440 -0
  202. package/src/tui/components/Checkpoint/index.ts +19 -0
  203. package/src/tui/components/CostDisplay.tsx +226 -0
  204. package/src/tui/components/InitErrorBanner.tsx +122 -0
  205. package/src/tui/components/Input/Autocomplete.tsx +603 -0
  206. package/src/tui/components/Input/EnhancedCommandInput.tsx +471 -0
  207. package/src/tui/components/Input/HighlightedText.tsx +236 -0
  208. package/src/tui/components/Input/MentionAutocomplete.tsx +375 -0
  209. package/src/tui/components/Input/TextInput.tsx +1002 -0
  210. package/src/tui/components/Input/__tests__/Autocomplete.test.tsx +374 -0
  211. package/src/tui/components/Input/__tests__/TextInput.test.tsx +241 -0
  212. package/src/tui/components/Input/__tests__/highlight.test.ts +219 -0
  213. package/src/tui/components/Input/__tests__/slash-command-utils.test.ts +104 -0
  214. package/src/tui/components/Input/highlight.ts +362 -0
  215. package/src/tui/components/Input/index.ts +36 -0
  216. package/src/tui/components/Input/slash-command-utils.ts +135 -0
  217. package/src/tui/components/Layout.tsx +432 -0
  218. package/src/tui/components/McpPanel.tsx +137 -0
  219. package/src/tui/components/MemoryPanel.tsx +448 -0
  220. package/src/tui/components/Messages/CodeBlock.tsx +527 -0
  221. package/src/tui/components/Messages/DiffView.tsx +679 -0
  222. package/src/tui/components/Messages/ImageReference.tsx +89 -0
  223. package/src/tui/components/Messages/MarkdownBlock.tsx +228 -0
  224. package/src/tui/components/Messages/MarkdownRenderer.tsx +498 -0
  225. package/src/tui/components/Messages/MessageBubble.tsx +270 -0
  226. package/src/tui/components/Messages/MessageList.tsx +1719 -0
  227. package/src/tui/components/Messages/StreamingText.tsx +216 -0
  228. package/src/tui/components/Messages/ThinkingBlock.tsx +408 -0
  229. package/src/tui/components/Messages/ToolResultPreview.tsx +243 -0
  230. package/src/tui/components/Messages/__tests__/CodeBlock.test.tsx +296 -0
  231. package/src/tui/components/Messages/__tests__/DiffView.test.tsx +239 -0
  232. package/src/tui/components/Messages/__tests__/MarkdownRenderer.test.tsx +303 -0
  233. package/src/tui/components/Messages/__tests__/MessageBubble.test.tsx +268 -0
  234. package/src/tui/components/Messages/__tests__/MessageList.test.tsx +324 -0
  235. package/src/tui/components/Messages/__tests__/StreamingText.test.tsx +215 -0
  236. package/src/tui/components/Messages/index.ts +25 -0
  237. package/src/tui/components/ModeIndicator.tsx +177 -0
  238. package/src/tui/components/ModeSelector.tsx +216 -0
  239. package/src/tui/components/ModelSelector.tsx +339 -0
  240. package/src/tui/components/OnboardingWizard.tsx +670 -0
  241. package/src/tui/components/PhaseProgressIndicator.tsx +270 -0
  242. package/src/tui/components/RateLimitIndicator.tsx +82 -0
  243. package/src/tui/components/ScreenReaderLayout.tsx +295 -0
  244. package/src/tui/components/SettingsPanel.tsx +643 -0
  245. package/src/tui/components/Sidebar/SystemStatusPanel.tsx +284 -0
  246. package/src/tui/components/Sidebar/index.ts +9 -0
  247. package/src/tui/components/Status/ModelStatusBar.tsx +270 -0
  248. package/src/tui/components/Status/index.ts +12 -0
  249. package/src/tui/components/StatusBar/AgentModeIndicator.tsx +257 -0
  250. package/src/tui/components/StatusBar/ContextProgress.tsx +167 -0
  251. package/src/tui/components/StatusBar/FileChangesIndicator.tsx +62 -0
  252. package/src/tui/components/StatusBar/GitIndicator.tsx +89 -0
  253. package/src/tui/components/StatusBar/HeaderBar.tsx +126 -0
  254. package/src/tui/components/StatusBar/ModelIndicator.tsx +157 -0
  255. package/src/tui/components/StatusBar/PersistenceStatusIndicator.tsx +210 -0
  256. package/src/tui/components/StatusBar/ResilienceIndicator.tsx +106 -0
  257. package/src/tui/components/StatusBar/SandboxIndicator.tsx +167 -0
  258. package/src/tui/components/StatusBar/StatusBar.tsx +368 -0
  259. package/src/tui/components/StatusBar/ThinkingModeIndicator.tsx +170 -0
  260. package/src/tui/components/StatusBar/TokenBreakdown.tsx +246 -0
  261. package/src/tui/components/StatusBar/TokenCounter.tsx +135 -0
  262. package/src/tui/components/StatusBar/TrustModeIndicator.tsx +130 -0
  263. package/src/tui/components/StatusBar/WorkspaceIndicator.tsx +86 -0
  264. package/src/tui/components/StatusBar/__tests__/AgentModeIndicator.test.tsx +193 -0
  265. package/src/tui/components/StatusBar/__tests__/StatusBar.test.tsx +729 -0
  266. package/src/tui/components/StatusBar/index.ts +60 -0
  267. package/src/tui/components/TipBanner.tsx +115 -0
  268. package/src/tui/components/TodoItem.tsx +208 -0
  269. package/src/tui/components/TodoPanel.tsx +455 -0
  270. package/src/tui/components/Tools/ApprovalQueue.tsx +407 -0
  271. package/src/tui/components/Tools/OptionSelector.tsx +160 -0
  272. package/src/tui/components/Tools/PermissionDialog.tsx +286 -0
  273. package/src/tui/components/Tools/ToolParams.tsx +483 -0
  274. package/src/tui/components/Tools/ToolsPanel.tsx +178 -0
  275. package/src/tui/components/Tools/__tests__/PermissionDialog.test.tsx +510 -0
  276. package/src/tui/components/Tools/__tests__/ToolParams.test.tsx +432 -0
  277. package/src/tui/components/Tools/index.ts +21 -0
  278. package/src/tui/components/TrustPrompt.tsx +279 -0
  279. package/src/tui/components/UpdateBanner.tsx +166 -0
  280. package/src/tui/components/VimModeIndicator.tsx +112 -0
  281. package/src/tui/components/backtrack/BacktrackControls.tsx +402 -0
  282. package/src/tui/components/backtrack/index.ts +13 -0
  283. package/src/tui/components/common/AutoApprovalStatus.tsx +251 -0
  284. package/src/tui/components/common/CostWarning.tsx +294 -0
  285. package/src/tui/components/common/DynamicShortcutHints.tsx +209 -0
  286. package/src/tui/components/common/EnhancedLoadingIndicator.tsx +305 -0
  287. package/src/tui/components/common/ErrorBoundary.tsx +140 -0
  288. package/src/tui/components/common/GradientText.tsx +224 -0
  289. package/src/tui/components/common/HotkeyHelpModal.tsx +193 -0
  290. package/src/tui/components/common/HotkeyHints.tsx +70 -0
  291. package/src/tui/components/common/MaxSizedBox.tsx +354 -0
  292. package/src/tui/components/common/NewMessagesBadge.tsx +65 -0
  293. package/src/tui/components/common/ProtectedFileLegend.tsx +89 -0
  294. package/src/tui/components/common/ScrollIndicator.tsx +160 -0
  295. package/src/tui/components/common/Spinner.tsx +342 -0
  296. package/src/tui/components/common/StreamingIndicator.tsx +316 -0
  297. package/src/tui/components/common/VirtualizedList/VirtualizedList.tsx +428 -0
  298. package/src/tui/components/common/VirtualizedList/hooks/index.ts +19 -0
  299. package/src/tui/components/common/VirtualizedList/hooks/useBatchedScroll.ts +64 -0
  300. package/src/tui/components/common/VirtualizedList/hooks/useScrollAnchor.ts +290 -0
  301. package/src/tui/components/common/VirtualizedList/hooks/useVirtualization.ts +340 -0
  302. package/src/tui/components/common/VirtualizedList/index.ts +30 -0
  303. package/src/tui/components/common/VirtualizedList/types.ts +107 -0
  304. package/src/tui/components/common/__tests__/NewMessagesBadge.test.tsx +74 -0
  305. package/src/tui/components/common/__tests__/ScrollIndicator.test.tsx +193 -0
  306. package/src/tui/components/common/index.ts +110 -0
  307. package/src/tui/components/index.ts +79 -0
  308. package/src/tui/components/session/CheckpointPanel.tsx +323 -0
  309. package/src/tui/components/session/RollbackDialog.tsx +169 -0
  310. package/src/tui/components/session/SessionItem.tsx +136 -0
  311. package/src/tui/components/session/SessionListPanel.tsx +252 -0
  312. package/src/tui/components/session/SessionPicker.tsx +449 -0
  313. package/src/tui/components/session/SessionPreview.tsx +240 -0
  314. package/src/tui/components/session/__tests__/session.test.tsx +408 -0
  315. package/src/tui/components/session/index.ts +28 -0
  316. package/src/tui/components/session/types.ts +116 -0
  317. package/src/tui/components/theme/__tests__/tokens.test.ts +471 -0
  318. package/src/tui/components/theme/index.ts +227 -0
  319. package/src/tui/components/theme/tokens.ts +484 -0
  320. package/src/tui/config/defaults.ts +134 -0
  321. package/src/tui/config/index.ts +17 -0
  322. package/src/tui/context/AnimationContext.tsx +284 -0
  323. package/src/tui/context/AppContext.tsx +349 -0
  324. package/src/tui/context/BracketedPasteContext.tsx +372 -0
  325. package/src/tui/context/LspContext.tsx +192 -0
  326. package/src/tui/context/McpContext.tsx +325 -0
  327. package/src/tui/context/MessagesContext.tsx +870 -0
  328. package/src/tui/context/OverflowContext.tsx +213 -0
  329. package/src/tui/context/RateLimitContext.tsx +108 -0
  330. package/src/tui/context/ResilienceContext.tsx +275 -0
  331. package/src/tui/context/RootProvider.tsx +136 -0
  332. package/src/tui/context/ScrollContext.tsx +331 -0
  333. package/src/tui/context/ToolsContext.tsx +702 -0
  334. package/src/tui/context/__tests__/BracketedPasteContext.test.tsx +416 -0
  335. package/src/tui/context/index.ts +140 -0
  336. package/src/tui/enterprise-integration.ts +282 -0
  337. package/src/tui/hooks/__tests__/useBacktrack.test.tsx +138 -0
  338. package/src/tui/hooks/__tests__/useBracketedPaste.test.tsx +222 -0
  339. package/src/tui/hooks/__tests__/useCopyMode.test.tsx +336 -0
  340. package/src/tui/hooks/__tests__/useHotkeys.ctrl-input.test.tsx +96 -0
  341. package/src/tui/hooks/__tests__/useHotkeys.test.tsx +454 -0
  342. package/src/tui/hooks/__tests__/useInputHistory.test.tsx +660 -0
  343. package/src/tui/hooks/__tests__/useLineBuffer.test.ts +295 -0
  344. package/src/tui/hooks/__tests__/useModeController.test.ts +137 -0
  345. package/src/tui/hooks/__tests__/useModeShortcuts.test.tsx +142 -0
  346. package/src/tui/hooks/__tests__/useScrollController.test.ts +464 -0
  347. package/src/tui/hooks/__tests__/useVim.test.tsx +531 -0
  348. package/src/tui/hooks/index.ts +252 -0
  349. package/src/tui/hooks/useAgentLoop.ts +712 -0
  350. package/src/tui/hooks/useAlternateBuffer.ts +398 -0
  351. package/src/tui/hooks/useAnimatedScrollbar.ts +241 -0
  352. package/src/tui/hooks/useBacktrack.ts +443 -0
  353. package/src/tui/hooks/useBracketedPaste.ts +104 -0
  354. package/src/tui/hooks/useCollapsible.ts +240 -0
  355. package/src/tui/hooks/useCopyMode.ts +382 -0
  356. package/src/tui/hooks/useCostSummary.ts +75 -0
  357. package/src/tui/hooks/useDesktopNotification.ts +414 -0
  358. package/src/tui/hooks/useDiffMode.ts +44 -0
  359. package/src/tui/hooks/useFileChangeStats.ts +110 -0
  360. package/src/tui/hooks/useFileSuggestions.ts +284 -0
  361. package/src/tui/hooks/useFlickerDetector.ts +250 -0
  362. package/src/tui/hooks/useGitStatus.ts +200 -0
  363. package/src/tui/hooks/useHotkeys.ts +579 -0
  364. package/src/tui/hooks/useImagePaste.ts +114 -0
  365. package/src/tui/hooks/useInputHighlight.ts +145 -0
  366. package/src/tui/hooks/useInputHistory.ts +246 -0
  367. package/src/tui/hooks/useKeyboardScroll.ts +209 -0
  368. package/src/tui/hooks/useLineBuffer.ts +356 -0
  369. package/src/tui/hooks/useMentionAutocomplete.ts +235 -0
  370. package/src/tui/hooks/useModeController.ts +167 -0
  371. package/src/tui/hooks/useModeShortcuts.ts +196 -0
  372. package/src/tui/hooks/usePermissionHandler.ts +146 -0
  373. package/src/tui/hooks/usePersistence.ts +480 -0
  374. package/src/tui/hooks/usePersistenceShortcuts.ts +225 -0
  375. package/src/tui/hooks/usePlaceholderRotation.ts +143 -0
  376. package/src/tui/hooks/useProviderStatus.ts +270 -0
  377. package/src/tui/hooks/useRateLimitStatus.ts +90 -0
  378. package/src/tui/hooks/useScreenReader.ts +315 -0
  379. package/src/tui/hooks/useScrollController.ts +450 -0
  380. package/src/tui/hooks/useScrollEventBatcher.ts +185 -0
  381. package/src/tui/hooks/useSidebarPanelData.ts +115 -0
  382. package/src/tui/hooks/useSmoothScroll.ts +202 -0
  383. package/src/tui/hooks/useSnapshots.ts +300 -0
  384. package/src/tui/hooks/useStateAndRef.ts +50 -0
  385. package/src/tui/hooks/useTerminalSize.ts +206 -0
  386. package/src/tui/hooks/useToolApprovalController.ts +91 -0
  387. package/src/tui/hooks/useVim.ts +334 -0
  388. package/src/tui/hooks/useWorkspace.ts +56 -0
  389. package/src/tui/i18n/__tests__/init.test.ts +278 -0
  390. package/src/tui/i18n/__tests__/language-config.test.ts +199 -0
  391. package/src/tui/i18n/__tests__/locale-detection.test.ts +250 -0
  392. package/src/tui/i18n/__tests__/settings-integration.test.ts +262 -0
  393. package/src/tui/i18n/index.ts +72 -0
  394. package/src/tui/i18n/init.ts +131 -0
  395. package/src/tui/i18n/language-config.ts +106 -0
  396. package/src/tui/i18n/locale-detection.ts +173 -0
  397. package/src/tui/i18n/settings-integration.ts +557 -0
  398. package/src/tui/i18n/tui-namespace.ts +538 -0
  399. package/src/tui/i18n/types.ts +312 -0
  400. package/src/tui/index.ts +43 -0
  401. package/src/tui/lsp-integration.ts +409 -0
  402. package/src/tui/metrics-integration.ts +366 -0
  403. package/src/tui/plugins.ts +383 -0
  404. package/src/tui/resilience.ts +342 -0
  405. package/src/tui/sandbox-integration.ts +317 -0
  406. package/src/tui/services/clipboard.ts +348 -0
  407. package/src/tui/services/fuzzy-search.ts +441 -0
  408. package/src/tui/services/index.ts +72 -0
  409. package/src/tui/services/markdown-renderer.ts +565 -0
  410. package/src/tui/services/open-external.ts +247 -0
  411. package/src/tui/services/syntax-highlighter.ts +483 -0
  412. package/src/tui/slash-commands.ts +12 -0
  413. package/src/tui/theme/index.ts +15 -0
  414. package/src/tui/theme/provider.tsx +206 -0
  415. package/src/tui/tip-integration.ts +300 -0
  416. package/src/tui/types/__tests__/ink-extended.test.ts +121 -0
  417. package/src/tui/types/ink-extended.ts +87 -0
  418. package/src/tui/utils/__tests__/bracketedPaste.test.ts +231 -0
  419. package/src/tui/utils/__tests__/heightEstimator.test.ts +157 -0
  420. package/src/tui/utils/__tests__/text-width.test.ts +158 -0
  421. package/src/tui/utils/__tests__/textSanitizer.test.ts +266 -0
  422. package/src/tui/utils/__tests__/ui-sizing.test.ts +169 -0
  423. package/src/tui/utils/bracketedPaste.ts +107 -0
  424. package/src/tui/utils/cursor-manager.ts +131 -0
  425. package/src/tui/utils/detectTerminal.ts +596 -0
  426. package/src/tui/utils/findLastSafeSplitPoint.ts +92 -0
  427. package/src/tui/utils/heightEstimator.ts +198 -0
  428. package/src/tui/utils/index.ts +91 -0
  429. package/src/tui/utils/isNarrowWidth.ts +52 -0
  430. package/src/tui/utils/stdoutGuard.ts +90 -0
  431. package/src/tui/utils/synchronized-update.ts +70 -0
  432. package/src/tui/utils/text-width.ts +225 -0
  433. package/src/tui/utils/textSanitizer.ts +225 -0
  434. package/src/tui/utils/textUtils.ts +114 -0
  435. package/src/tui/utils/ui-sizing.ts +192 -0
  436. package/src/tui-blessed/app.ts +160 -0
  437. package/src/tui-blessed/index.ts +2 -0
  438. package/src/tui-blessed/neo-blessed.d.ts +6 -0
  439. package/src/tui-blessed/test.ts +21 -0
  440. package/src/tui-blessed/types.ts +14 -0
  441. package/src/utils/icons.ts +130 -0
  442. package/src/utils/index.ts +33 -0
  443. package/src/utils/resume-hint.ts +86 -0
  444. package/src/version.ts +1 -0
  445. package/tsconfig.json +8 -0
  446. package/vitest.config.ts +35 -0
@@ -0,0 +1,670 @@
1
+ /**
2
+ * OnboardingWizard TUI Component (Phase 38)
3
+ *
4
+ * React Ink component that renders the onboarding wizard UI.
5
+ * Provides step-by-step navigation with progress indicator.
6
+ *
7
+ * @module tui/components/OnboardingWizard
8
+ */
9
+
10
+ import {
11
+ type CredentialManager,
12
+ createCompleteStep,
13
+ createModeSelectStep,
14
+ createProviderSelectStep,
15
+ createWelcomeStep,
16
+ formatCompletionMessage,
17
+ formatCredentialPrompt,
18
+ formatQuickStart,
19
+ formatWelcomeContent,
20
+ getRecommendedSource,
21
+ ONBOARDING_STEP_CONFIG,
22
+ ONBOARDING_STEPS,
23
+ type OnboardingStep,
24
+ type PROVIDER_INFO,
25
+ OnboardingWizard as WizardCore,
26
+ } from "@vellum/core";
27
+ import { Box, Text, useApp, useInput } from "ink";
28
+ import type React from "react";
29
+ import { useCallback, useEffect, useState } from "react";
30
+ import { useTUITranslation } from "../i18n/index.js";
31
+ import { useTheme } from "../theme/index.js";
32
+ import { TextInput } from "./Input/TextInput.js";
33
+
34
+ // =============================================================================
35
+ // Types
36
+ // =============================================================================
37
+
38
+ /** Onboarding provider type */
39
+ type OnboardingProvider = keyof typeof PROVIDER_INFO;
40
+
41
+ /**
42
+ * Props for OnboardingWizard component
43
+ */
44
+ export interface OnboardingWizardProps {
45
+ /** Pre-initialized wizard instance */
46
+ wizard?: WizardCore;
47
+ /** Initial step to show */
48
+ initialStep?: OnboardingStep;
49
+ /** Callback when onboarding completes */
50
+ onComplete?: (result: { provider: string; mode: string; credentialsConfigured: boolean }) => void;
51
+ /** Callback when user cancels */
52
+ onCancel?: () => void;
53
+ /** Credential manager for secure API key storage */
54
+ credentialManager?: CredentialManager;
55
+ }
56
+
57
+ /**
58
+ * Internal state
59
+ */
60
+ interface WizardState {
61
+ step: OnboardingStep;
62
+ input: string;
63
+ error: string | null;
64
+ isLoading: boolean;
65
+ selectedProvider: OnboardingProvider | null;
66
+ selectedMode: string | null;
67
+ credentialsConfigured: boolean;
68
+ }
69
+
70
+ // =============================================================================
71
+ // Sub-components
72
+ // =============================================================================
73
+
74
+ /**
75
+ * Progress indicator showing current step
76
+ */
77
+ function ProgressBar({ currentStep }: { currentStep: OnboardingStep }): React.ReactElement {
78
+ const { theme } = useTheme();
79
+ const currentIndex = ONBOARDING_STEPS.indexOf(currentStep);
80
+ const total = ONBOARDING_STEPS.length;
81
+
82
+ return (
83
+ <Box marginBottom={1}>
84
+ <Text color={theme.colors.muted}>
85
+ Step {currentIndex + 1} of {total}:{" "}
86
+ </Text>
87
+ <Text bold color={theme.colors.primary}>
88
+ {ONBOARDING_STEP_CONFIG[currentStep].icon} {ONBOARDING_STEP_CONFIG[currentStep].title}
89
+ </Text>
90
+ <Text color={theme.colors.muted}>
91
+ {" "}
92
+ [
93
+ {ONBOARDING_STEPS.map((_s: OnboardingStep, i: number) =>
94
+ i <= currentIndex ? "●" : "○"
95
+ ).join("")}
96
+ ]
97
+ </Text>
98
+ </Box>
99
+ );
100
+ }
101
+
102
+ /**
103
+ * Welcome step content
104
+ */
105
+ function WelcomeContent(): React.ReactElement {
106
+ const { theme } = useTheme();
107
+ const welcomeStep = createWelcomeStep();
108
+ const content = welcomeStep.getContent();
109
+
110
+ return (
111
+ <Box flexDirection="column">
112
+ <Text>{formatWelcomeContent(content)}</Text>
113
+ <Box marginTop={1}>
114
+ <Text color={theme.colors.muted}>Press Enter to continue, Esc to skip setup</Text>
115
+ </Box>
116
+ </Box>
117
+ );
118
+ }
119
+
120
+ /**
121
+ * Provider selection content with arrow-key navigation
122
+ */
123
+ function ProviderSelectContent({
124
+ onSelect,
125
+ error,
126
+ }: {
127
+ onSelect: (provider: OnboardingProvider) => void;
128
+ error: string | null;
129
+ }): React.ReactElement {
130
+ const { theme } = useTheme();
131
+ const { t } = useTUITranslation();
132
+ const providerStep = createProviderSelectStep();
133
+ const providers = providerStep.getProviders();
134
+
135
+ // Track focused index for keyboard navigation
136
+ const [focusedIndex, setFocusedIndex] = useState(0);
137
+
138
+ // Handle keyboard input
139
+ useInput(
140
+ useCallback(
141
+ (input: string, key) => {
142
+ // Arrow navigation
143
+ if (key.upArrow || input === "k") {
144
+ setFocusedIndex((prev) => (prev > 0 ? prev - 1 : providers.length - 1));
145
+ return;
146
+ }
147
+
148
+ if (key.downArrow || input === "j") {
149
+ setFocusedIndex((prev) => (prev < providers.length - 1 ? prev + 1 : 0));
150
+ return;
151
+ }
152
+
153
+ // Confirm selection with Enter
154
+ if (key.return) {
155
+ const selected = providers[focusedIndex];
156
+ if (selected) {
157
+ onSelect(selected.id);
158
+ }
159
+ }
160
+ },
161
+ [focusedIndex, providers, onSelect]
162
+ )
163
+ );
164
+
165
+ return (
166
+ <Box flexDirection="column">
167
+ <Text bold>{t("onboarding.selectProvider")}</Text>
168
+ <Box flexDirection="column" marginTop={1}>
169
+ {providers.map((provider, index) => {
170
+ const isFocused = index === focusedIndex;
171
+ const indicator = isFocused ? ">" : " ";
172
+ const providerName = t(`providers.${provider.id}.name`);
173
+ const providerDescription = t(`providers.${provider.id}.description`);
174
+ const providerShortcut = t(`providers.${provider.id}.shortcut`);
175
+ const apiNote = provider.requiresApiKey
176
+ ? t("onboarding.apiKeyRequired")
177
+ : t("onboarding.noApiKeyNeeded");
178
+
179
+ return (
180
+ <Box key={provider.id} flexDirection="column">
181
+ <Text color={isFocused ? theme.colors.primary : undefined} bold={isFocused}>
182
+ {indicator} {providerShortcut} {providerName}
183
+ </Text>
184
+ <Text
185
+ color={isFocused ? theme.colors.primary : theme.colors.muted}
186
+ dimColor={!isFocused}
187
+ >
188
+ {" "}
189
+ {providerDescription} {apiNote}
190
+ </Text>
191
+ </Box>
192
+ );
193
+ })}
194
+ </Box>
195
+ {error && (
196
+ <Box marginTop={1}>
197
+ <Text color={theme.colors.error}>! {error}</Text>
198
+ </Box>
199
+ )}
200
+ <Box marginTop={1}>
201
+ <Text color={theme.colors.muted}>{t("onboarding.providerNav")}</Text>
202
+ </Box>
203
+ </Box>
204
+ );
205
+ }
206
+
207
+ /**
208
+ * Credential setup content
209
+ */
210
+ function CredentialSetupContent({
211
+ provider,
212
+ onInputChange,
213
+ inputValue,
214
+ error,
215
+ }: {
216
+ provider: OnboardingProvider;
217
+ onInputChange: (value: string) => void;
218
+ inputValue: string;
219
+ error: string | null;
220
+ }): React.ReactElement {
221
+ const { theme } = useTheme();
222
+ return (
223
+ <Box flexDirection="column">
224
+ <Text>{formatCredentialPrompt(provider)}</Text>
225
+ <Box marginTop={1}>
226
+ <Text color={theme.colors.info}>API Key: </Text>
227
+ <TextInput value={inputValue} onChange={onInputChange} mask="*" />
228
+ </Box>
229
+ {error && (
230
+ <Box marginTop={1}>
231
+ <Text color={theme.colors.error}>! {error}</Text>
232
+ </Box>
233
+ )}
234
+ <Box marginTop={1}>
235
+ <Text color={theme.colors.muted}>
236
+ Press Backspace to go back, 'skip' to configure later
237
+ </Text>
238
+ </Box>
239
+ </Box>
240
+ );
241
+ }
242
+
243
+ /**
244
+ * Mode selection content with arrow-key navigation
245
+ */
246
+ function ModeSelectContent({
247
+ onSelect,
248
+ error,
249
+ }: {
250
+ onSelect: (mode: string) => void;
251
+ error: string | null;
252
+ }): React.ReactElement {
253
+ const { theme } = useTheme();
254
+ const { t } = useTUITranslation();
255
+ const modeStep = createModeSelectStep();
256
+ const modes = modeStep.getModes();
257
+
258
+ // Track focused index for keyboard navigation
259
+ const [focusedIndex, setFocusedIndex] = useState(0);
260
+
261
+ // Handle keyboard input
262
+ useInput(
263
+ useCallback(
264
+ (input: string, key) => {
265
+ // Arrow navigation
266
+ if (key.upArrow || input === "k") {
267
+ setFocusedIndex((prev) => (prev > 0 ? prev - 1 : modes.length - 1));
268
+ return;
269
+ }
270
+
271
+ if (key.downArrow || input === "j") {
272
+ setFocusedIndex((prev) => (prev < modes.length - 1 ? prev + 1 : 0));
273
+ return;
274
+ }
275
+
276
+ // Confirm selection with Enter
277
+ if (key.return) {
278
+ const selected = modes[focusedIndex];
279
+ if (selected) {
280
+ onSelect(selected.id);
281
+ }
282
+ }
283
+ },
284
+ [focusedIndex, modes, onSelect]
285
+ )
286
+ );
287
+
288
+ return (
289
+ <Box flexDirection="column">
290
+ <Text bold>{t("onboarding.selectMode")}</Text>
291
+ <Box flexDirection="column" marginTop={1}>
292
+ {modes.map((mode, index) => {
293
+ const isFocused = index === focusedIndex;
294
+ const indicator = isFocused ? ">" : " ";
295
+
296
+ return (
297
+ <Box key={mode.id} flexDirection="column">
298
+ <Text color={isFocused ? theme.colors.primary : undefined} bold={isFocused}>
299
+ {indicator} {mode.icon} {mode.name}
300
+ </Text>
301
+ <Text
302
+ color={isFocused ? theme.colors.primary : theme.colors.muted}
303
+ dimColor={!isFocused}
304
+ >
305
+ {" "}
306
+ {mode.description}
307
+ </Text>
308
+ <Text
309
+ color={isFocused ? theme.colors.primary : theme.colors.muted}
310
+ dimColor={!isFocused}
311
+ >
312
+ {" "}Best for: {mode.useCase}
313
+ </Text>
314
+ </Box>
315
+ );
316
+ })}
317
+ </Box>
318
+ {error && (
319
+ <Box marginTop={1}>
320
+ <Text color={theme.colors.error}>! {error}</Text>
321
+ </Box>
322
+ )}
323
+ <Box marginTop={1}>
324
+ <Text color={theme.colors.muted}>{t("onboarding.modeNav")}</Text>
325
+ </Box>
326
+ <Box>
327
+ <Text color={theme.colors.info}>{t("onboarding.modeSwitchHint")}</Text>
328
+ </Box>
329
+ </Box>
330
+ );
331
+ }
332
+
333
+ /**
334
+ * Complete step content
335
+ */
336
+ function CompleteContent({
337
+ provider,
338
+ mode,
339
+ credentialsConfigured,
340
+ }: {
341
+ provider: OnboardingProvider;
342
+ mode: string;
343
+ credentialsConfigured: boolean;
344
+ }): React.ReactElement {
345
+ const { theme } = useTheme();
346
+ const completeStep = createCompleteStep();
347
+ const summary = {
348
+ provider,
349
+ mode: mode as "vibe" | "plan" | "spec",
350
+ credentialsConfigured,
351
+ warnings: [],
352
+ };
353
+
354
+ return (
355
+ <Box flexDirection="column">
356
+ <Text>{formatCompletionMessage(summary)}</Text>
357
+ <Text>
358
+ {formatQuickStart(completeStep.getQuickStartTips(), completeStep.getNextSteps(summary))}
359
+ </Text>
360
+ <Box marginTop={1}>
361
+ <Text color={theme.colors.success}>Press Enter to start using Vellum!</Text>
362
+ </Box>
363
+ </Box>
364
+ );
365
+ }
366
+
367
+ // =============================================================================
368
+ // Main Component
369
+ // =============================================================================
370
+
371
+ /**
372
+ * OnboardingWizard - Interactive setup wizard component
373
+ */
374
+ export function OnboardingWizard({
375
+ wizard: providedWizard,
376
+ initialStep = "welcome",
377
+ onComplete,
378
+ onCancel,
379
+ credentialManager,
380
+ }: OnboardingWizardProps): React.ReactElement {
381
+ const { exit } = useApp();
382
+ const { theme } = useTheme();
383
+
384
+ // Initialize wizard
385
+ const [wizard] = useState(() => providedWizard || new WizardCore());
386
+
387
+ // Connect credential manager to wizard when provided
388
+ useEffect(() => {
389
+ if (credentialManager) {
390
+ wizard.setCredentialManager(credentialManager);
391
+ }
392
+ }, [wizard, credentialManager]);
393
+
394
+ // Component state
395
+ const [state, setState] = useState<WizardState>({
396
+ step: initialStep,
397
+ input: "",
398
+ error: null,
399
+ isLoading: false,
400
+ selectedProvider: null,
401
+ selectedMode: null,
402
+ credentialsConfigured: false,
403
+ });
404
+
405
+ // Load wizard state on mount
406
+ useEffect(() => {
407
+ wizard.loadState().then(() => {
408
+ const wizardState = wizard.getState();
409
+ setState((s) => ({
410
+ ...s,
411
+ step: wizardState.currentStep,
412
+ selectedProvider: wizardState.selectedProvider as OnboardingProvider | null,
413
+ selectedMode: wizardState.selectedMode || null,
414
+ }));
415
+ });
416
+ }, [wizard]);
417
+
418
+ // Handle input change
419
+ const handleInputChange = useCallback((value: string) => {
420
+ setState((s) => ({ ...s, input: value, error: null }));
421
+ }, []);
422
+
423
+ // Handle provider selection (from arrow-key selector)
424
+ const handleProviderSelect = useCallback(
425
+ async (provider: OnboardingProvider) => {
426
+ setState((s) => ({ ...s, isLoading: true, error: null }));
427
+
428
+ try {
429
+ const result = await wizard.executeProviderSelect(provider);
430
+ if (result.success && result.next) {
431
+ setState((s) => ({
432
+ ...s,
433
+ step: "credential-setup",
434
+ input: "",
435
+ isLoading: false,
436
+ selectedProvider: (result.data?.provider as OnboardingProvider) || provider,
437
+ }));
438
+ } else if (result.error) {
439
+ setState((s) => ({ ...s, error: result.error || null, isLoading: false }));
440
+ }
441
+ } catch (err) {
442
+ setState((s) => ({
443
+ ...s,
444
+ error: err instanceof Error ? err.message : "An error occurred",
445
+ isLoading: false,
446
+ }));
447
+ }
448
+ },
449
+ [wizard]
450
+ );
451
+
452
+ // Handle mode selection (from arrow-key selector)
453
+ const handleModeSelect = useCallback(
454
+ async (mode: string) => {
455
+ setState((s) => ({ ...s, isLoading: true, error: null }));
456
+
457
+ try {
458
+ const result = await wizard.executeModeSelect(mode);
459
+ if (result.success && result.next) {
460
+ setState((s) => ({
461
+ ...s,
462
+ step: "complete",
463
+ input: "",
464
+ isLoading: false,
465
+ selectedMode: (result.data?.mode as string) || mode,
466
+ }));
467
+ } else if (result.back) {
468
+ setState((s) => ({ ...s, step: "credential-setup", input: "", isLoading: false }));
469
+ } else if (result.error) {
470
+ setState((s) => ({ ...s, error: result.error || null, isLoading: false }));
471
+ }
472
+ } catch (err) {
473
+ setState((s) => ({
474
+ ...s,
475
+ error: err instanceof Error ? err.message : "An error occurred",
476
+ isLoading: false,
477
+ }));
478
+ }
479
+ },
480
+ [wizard]
481
+ );
482
+
483
+ // Handle step submission
484
+ // biome-ignore lint/complexity/noExcessiveCognitiveComplexity: Wizard step handling requires processing multiple step types with different logic
485
+ const handleSubmit = useCallback(async () => {
486
+ setState((s) => ({ ...s, isLoading: true, error: null }));
487
+
488
+ try {
489
+ switch (state.step) {
490
+ case "welcome": {
491
+ const result = await wizard.executeWelcome();
492
+ if (result.next) {
493
+ setState((s) => ({ ...s, step: "provider-select", input: "", isLoading: false }));
494
+ }
495
+ break;
496
+ }
497
+
498
+ case "provider-select": {
499
+ // Provider selection is now handled by handleProviderSelect
500
+ // This case handles Enter when no selection made (use default)
501
+ const result = await wizard.executeProviderSelect("anthropic");
502
+ if (result.success && result.next) {
503
+ setState((s) => ({
504
+ ...s,
505
+ step: "credential-setup",
506
+ input: "",
507
+ isLoading: false,
508
+ selectedProvider: (result.data?.provider as OnboardingProvider) || "anthropic",
509
+ }));
510
+ } else if (result.error) {
511
+ setState((s) => ({ ...s, error: result.error || null, isLoading: false }));
512
+ }
513
+ break;
514
+ }
515
+
516
+ case "credential-setup": {
517
+ const source = getRecommendedSource();
518
+ const result = await wizard.executeCredentialSetup(state.input, source);
519
+ if (result.success && result.next) {
520
+ setState((s) => ({
521
+ ...s,
522
+ step: "mode-select",
523
+ input: "",
524
+ isLoading: false,
525
+ credentialsConfigured: (result.data?.saved as boolean) ?? false,
526
+ }));
527
+ } else if (result.back) {
528
+ setState((s) => ({ ...s, step: "provider-select", input: "", isLoading: false }));
529
+ } else if (result.error) {
530
+ setState((s) => ({ ...s, error: result.error || null, isLoading: false }));
531
+ }
532
+ break;
533
+ }
534
+
535
+ case "mode-select": {
536
+ const result = await wizard.executeModeSelect(state.input || "1");
537
+ if (result.success && result.next) {
538
+ setState((s) => ({
539
+ ...s,
540
+ step: "complete",
541
+ input: "",
542
+ isLoading: false,
543
+ selectedMode: (result.data?.mode as string) || "vibe",
544
+ }));
545
+ } else if (result.back) {
546
+ setState((s) => ({ ...s, step: "credential-setup", input: "", isLoading: false }));
547
+ } else if (result.error) {
548
+ setState((s) => ({ ...s, error: result.error || null, isLoading: false }));
549
+ }
550
+ break;
551
+ }
552
+
553
+ case "complete": {
554
+ await wizard.executeComplete();
555
+ await wizard.saveConfig();
556
+
557
+ // Fire completion callback - parent component handles transition
558
+ onComplete?.({
559
+ provider: state.selectedProvider || "anthropic",
560
+ mode: state.selectedMode || "vibe",
561
+ credentialsConfigured: state.credentialsConfigured,
562
+ });
563
+ break;
564
+ }
565
+ }
566
+ } catch (err) {
567
+ setState((s) => ({
568
+ ...s,
569
+ error: err instanceof Error ? err.message : "An error occurred",
570
+ isLoading: false,
571
+ }));
572
+ }
573
+ }, [state, wizard, onComplete]);
574
+
575
+ // Handle keyboard input
576
+ useInput(
577
+ useCallback(
578
+ (_input: string, key) => {
579
+ // Escape to cancel
580
+ if (key.escape) {
581
+ onCancel?.();
582
+ exit();
583
+ return;
584
+ }
585
+
586
+ // Enter to submit
587
+ if (key.return) {
588
+ handleSubmit();
589
+ return;
590
+ }
591
+
592
+ // Backspace on empty input = back
593
+ if (key.backspace && state.input === "" && state.step !== "welcome") {
594
+ const steps = ONBOARDING_STEPS;
595
+ const currentIndex = steps.indexOf(state.step);
596
+ if (currentIndex > 0) {
597
+ const prevStep = steps[currentIndex - 1];
598
+ if (prevStep) {
599
+ setState((s) => ({ ...s, step: prevStep, error: null }));
600
+ }
601
+ }
602
+ }
603
+ },
604
+ [state, handleSubmit, onCancel, exit]
605
+ )
606
+ );
607
+
608
+ // Render step content
609
+ const renderStepContent = () => {
610
+ switch (state.step) {
611
+ case "welcome":
612
+ return <WelcomeContent />;
613
+
614
+ case "provider-select":
615
+ return <ProviderSelectContent onSelect={handleProviderSelect} error={state.error} />;
616
+
617
+ case "credential-setup":
618
+ return (
619
+ <CredentialSetupContent
620
+ provider={state.selectedProvider || "anthropic"}
621
+ onInputChange={handleInputChange}
622
+ inputValue={state.input}
623
+ error={state.error}
624
+ />
625
+ );
626
+
627
+ case "mode-select":
628
+ return <ModeSelectContent onSelect={handleModeSelect} error={state.error} />;
629
+
630
+ case "complete":
631
+ return (
632
+ <CompleteContent
633
+ provider={state.selectedProvider || "anthropic"}
634
+ mode={state.selectedMode || "vibe"}
635
+ credentialsConfigured={state.credentialsConfigured}
636
+ />
637
+ );
638
+ }
639
+ };
640
+
641
+ return (
642
+ <Box flexDirection="column" padding={1}>
643
+ {/* Header */}
644
+ <Box borderStyle="round" borderColor="#DAA520" paddingX={2} paddingY={1} marginBottom={1}>
645
+ <Text bold color="#DAA520">
646
+ ✦ Vellum Setup Wizard
647
+ </Text>
648
+ </Box>
649
+
650
+ {/* Progress */}
651
+ <ProgressBar currentStep={state.step} />
652
+
653
+ {/* Content */}
654
+ <Box flexDirection="column" marginY={1}>
655
+ {state.isLoading ? (
656
+ <Text color={theme.colors.info}>Processing...</Text>
657
+ ) : (
658
+ renderStepContent()
659
+ )}
660
+ </Box>
661
+
662
+ {/* Footer */}
663
+ <Box marginTop={1} borderStyle="single" borderColor={theme.colors.muted} paddingX={1}>
664
+ <Text color={theme.colors.muted}>Esc: Cancel | Enter: Continue | Backspace: Back</Text>
665
+ </Box>
666
+ </Box>
667
+ );
668
+ }
669
+
670
+ export default OnboardingWizard;