whale-code 6.4.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 (319) hide show
  1. package/README.md +95 -0
  2. package/bin/swag-agent.js +9 -0
  3. package/bin/swagmanager-mcp.js +321 -0
  4. package/dist/cli/app.d.ts +26 -0
  5. package/dist/cli/app.js +64 -0
  6. package/dist/cli/chat/AgentSelector.d.ts +14 -0
  7. package/dist/cli/chat/AgentSelector.js +14 -0
  8. package/dist/cli/chat/ChatApp.d.ts +9 -0
  9. package/dist/cli/chat/ChatApp.js +267 -0
  10. package/dist/cli/chat/ChatInput.d.ts +39 -0
  11. package/dist/cli/chat/ChatInput.js +509 -0
  12. package/dist/cli/chat/MarkdownText.d.ts +10 -0
  13. package/dist/cli/chat/MarkdownText.js +20 -0
  14. package/dist/cli/chat/MessageList.d.ts +37 -0
  15. package/dist/cli/chat/MessageList.js +80 -0
  16. package/dist/cli/chat/ModelSelector.d.ts +20 -0
  17. package/dist/cli/chat/ModelSelector.js +73 -0
  18. package/dist/cli/chat/RewindViewer.d.ts +26 -0
  19. package/dist/cli/chat/RewindViewer.js +185 -0
  20. package/dist/cli/chat/StoreSelector.d.ts +14 -0
  21. package/dist/cli/chat/StoreSelector.js +24 -0
  22. package/dist/cli/chat/StreamingText.d.ts +12 -0
  23. package/dist/cli/chat/StreamingText.js +12 -0
  24. package/dist/cli/chat/SubagentPanel.d.ts +45 -0
  25. package/dist/cli/chat/SubagentPanel.js +110 -0
  26. package/dist/cli/chat/TeamPanel.d.ts +21 -0
  27. package/dist/cli/chat/TeamPanel.js +42 -0
  28. package/dist/cli/chat/ToolIndicator.d.ts +25 -0
  29. package/dist/cli/chat/ToolIndicator.js +436 -0
  30. package/dist/cli/chat/hooks/useAgentLoop.d.ts +39 -0
  31. package/dist/cli/chat/hooks/useAgentLoop.js +382 -0
  32. package/dist/cli/chat/hooks/useSlashCommands.d.ts +37 -0
  33. package/dist/cli/chat/hooks/useSlashCommands.js +387 -0
  34. package/dist/cli/commands/config-cmd.d.ts +10 -0
  35. package/dist/cli/commands/config-cmd.js +99 -0
  36. package/dist/cli/commands/doctor.d.ts +14 -0
  37. package/dist/cli/commands/doctor.js +172 -0
  38. package/dist/cli/commands/init.d.ts +16 -0
  39. package/dist/cli/commands/init.js +278 -0
  40. package/dist/cli/commands/mcp.d.ts +12 -0
  41. package/dist/cli/commands/mcp.js +162 -0
  42. package/dist/cli/login/LoginApp.d.ts +7 -0
  43. package/dist/cli/login/LoginApp.js +157 -0
  44. package/dist/cli/print-mode.d.ts +31 -0
  45. package/dist/cli/print-mode.js +202 -0
  46. package/dist/cli/serve-mode.d.ts +37 -0
  47. package/dist/cli/serve-mode.js +636 -0
  48. package/dist/cli/services/agent-definitions.d.ts +25 -0
  49. package/dist/cli/services/agent-definitions.js +91 -0
  50. package/dist/cli/services/agent-events.d.ts +178 -0
  51. package/dist/cli/services/agent-events.js +175 -0
  52. package/dist/cli/services/agent-loop.d.ts +90 -0
  53. package/dist/cli/services/agent-loop.js +762 -0
  54. package/dist/cli/services/agent-worker-base.d.ts +97 -0
  55. package/dist/cli/services/agent-worker-base.js +220 -0
  56. package/dist/cli/services/auth-service.d.ts +30 -0
  57. package/dist/cli/services/auth-service.js +160 -0
  58. package/dist/cli/services/background-processes.d.ts +126 -0
  59. package/dist/cli/services/background-processes.js +318 -0
  60. package/dist/cli/services/browser-auth.d.ts +24 -0
  61. package/dist/cli/services/browser-auth.js +180 -0
  62. package/dist/cli/services/claude-md-loader.d.ts +16 -0
  63. package/dist/cli/services/claude-md-loader.js +58 -0
  64. package/dist/cli/services/config-store.d.ts +47 -0
  65. package/dist/cli/services/config-store.js +79 -0
  66. package/dist/cli/services/debug-log.d.ts +10 -0
  67. package/dist/cli/services/debug-log.js +52 -0
  68. package/dist/cli/services/error-logger.d.ts +58 -0
  69. package/dist/cli/services/error-logger.js +269 -0
  70. package/dist/cli/services/file-history.d.ts +21 -0
  71. package/dist/cli/services/file-history.js +83 -0
  72. package/dist/cli/services/format-server-response.d.ts +16 -0
  73. package/dist/cli/services/format-server-response.js +440 -0
  74. package/dist/cli/services/git-context.d.ts +11 -0
  75. package/dist/cli/services/git-context.js +66 -0
  76. package/dist/cli/services/hooks.d.ts +85 -0
  77. package/dist/cli/services/hooks.js +258 -0
  78. package/dist/cli/services/interactive-tools.d.ts +125 -0
  79. package/dist/cli/services/interactive-tools.js +260 -0
  80. package/dist/cli/services/keybinding-manager.d.ts +52 -0
  81. package/dist/cli/services/keybinding-manager.js +115 -0
  82. package/dist/cli/services/local-tools.d.ts +22 -0
  83. package/dist/cli/services/local-tools.js +697 -0
  84. package/dist/cli/services/lsp-manager.d.ts +18 -0
  85. package/dist/cli/services/lsp-manager.js +717 -0
  86. package/dist/cli/services/mcp-client.d.ts +48 -0
  87. package/dist/cli/services/mcp-client.js +157 -0
  88. package/dist/cli/services/memory-manager.d.ts +16 -0
  89. package/dist/cli/services/memory-manager.js +57 -0
  90. package/dist/cli/services/model-manager.d.ts +18 -0
  91. package/dist/cli/services/model-manager.js +71 -0
  92. package/dist/cli/services/model-router.d.ts +26 -0
  93. package/dist/cli/services/model-router.js +149 -0
  94. package/dist/cli/services/permission-modes.d.ts +13 -0
  95. package/dist/cli/services/permission-modes.js +43 -0
  96. package/dist/cli/services/rewind.d.ts +84 -0
  97. package/dist/cli/services/rewind.js +194 -0
  98. package/dist/cli/services/ripgrep.d.ts +28 -0
  99. package/dist/cli/services/ripgrep.js +138 -0
  100. package/dist/cli/services/sandbox.d.ts +29 -0
  101. package/dist/cli/services/sandbox.js +97 -0
  102. package/dist/cli/services/server-tools.d.ts +61 -0
  103. package/dist/cli/services/server-tools.js +543 -0
  104. package/dist/cli/services/session-persistence.d.ts +23 -0
  105. package/dist/cli/services/session-persistence.js +99 -0
  106. package/dist/cli/services/subagent-worker.d.ts +19 -0
  107. package/dist/cli/services/subagent-worker.js +41 -0
  108. package/dist/cli/services/subagent.d.ts +47 -0
  109. package/dist/cli/services/subagent.js +647 -0
  110. package/dist/cli/services/system-prompt.d.ts +7 -0
  111. package/dist/cli/services/system-prompt.js +198 -0
  112. package/dist/cli/services/team-lead.d.ts +73 -0
  113. package/dist/cli/services/team-lead.js +512 -0
  114. package/dist/cli/services/team-state.d.ts +77 -0
  115. package/dist/cli/services/team-state.js +398 -0
  116. package/dist/cli/services/teammate.d.ts +31 -0
  117. package/dist/cli/services/teammate.js +689 -0
  118. package/dist/cli/services/telemetry.d.ts +61 -0
  119. package/dist/cli/services/telemetry.js +209 -0
  120. package/dist/cli/services/tools/agent-tools.d.ts +14 -0
  121. package/dist/cli/services/tools/agent-tools.js +347 -0
  122. package/dist/cli/services/tools/file-ops.d.ts +15 -0
  123. package/dist/cli/services/tools/file-ops.js +487 -0
  124. package/dist/cli/services/tools/search-tools.d.ts +8 -0
  125. package/dist/cli/services/tools/search-tools.js +186 -0
  126. package/dist/cli/services/tools/shell-exec.d.ts +10 -0
  127. package/dist/cli/services/tools/shell-exec.js +168 -0
  128. package/dist/cli/services/tools/task-manager.d.ts +28 -0
  129. package/dist/cli/services/tools/task-manager.js +209 -0
  130. package/dist/cli/services/tools/web-tools.d.ts +11 -0
  131. package/dist/cli/services/tools/web-tools.js +395 -0
  132. package/dist/cli/setup/SetupApp.d.ts +9 -0
  133. package/dist/cli/setup/SetupApp.js +191 -0
  134. package/dist/cli/shared/MatrixIntro.d.ts +4 -0
  135. package/dist/cli/shared/MatrixIntro.js +83 -0
  136. package/dist/cli/shared/Theme.d.ts +74 -0
  137. package/dist/cli/shared/Theme.js +127 -0
  138. package/dist/cli/shared/WhaleBanner.d.ts +10 -0
  139. package/dist/cli/shared/WhaleBanner.js +12 -0
  140. package/dist/cli/shared/markdown.d.ts +21 -0
  141. package/dist/cli/shared/markdown.js +756 -0
  142. package/dist/cli/status/StatusApp.d.ts +4 -0
  143. package/dist/cli/status/StatusApp.js +105 -0
  144. package/dist/cli/stores/StoreApp.d.ts +7 -0
  145. package/dist/cli/stores/StoreApp.js +81 -0
  146. package/dist/index.d.ts +15 -0
  147. package/dist/index.js +538 -0
  148. package/dist/local-agent/connection.d.ts +48 -0
  149. package/dist/local-agent/connection.js +332 -0
  150. package/dist/local-agent/discovery.d.ts +18 -0
  151. package/dist/local-agent/discovery.js +146 -0
  152. package/dist/local-agent/executor.d.ts +34 -0
  153. package/dist/local-agent/executor.js +241 -0
  154. package/dist/local-agent/index.d.ts +14 -0
  155. package/dist/local-agent/index.js +198 -0
  156. package/dist/node/adapters/base.d.ts +35 -0
  157. package/dist/node/adapters/base.js +10 -0
  158. package/dist/node/adapters/discord.d.ts +29 -0
  159. package/dist/node/adapters/discord.js +299 -0
  160. package/dist/node/adapters/email.d.ts +23 -0
  161. package/dist/node/adapters/email.js +218 -0
  162. package/dist/node/adapters/imessage.d.ts +17 -0
  163. package/dist/node/adapters/imessage.js +118 -0
  164. package/dist/node/adapters/slack.d.ts +26 -0
  165. package/dist/node/adapters/slack.js +259 -0
  166. package/dist/node/adapters/sms.d.ts +23 -0
  167. package/dist/node/adapters/sms.js +161 -0
  168. package/dist/node/adapters/telegram.d.ts +17 -0
  169. package/dist/node/adapters/telegram.js +101 -0
  170. package/dist/node/adapters/webchat.d.ts +27 -0
  171. package/dist/node/adapters/webchat.js +160 -0
  172. package/dist/node/adapters/whatsapp.d.ts +28 -0
  173. package/dist/node/adapters/whatsapp.js +230 -0
  174. package/dist/node/cli.d.ts +2 -0
  175. package/dist/node/cli.js +325 -0
  176. package/dist/node/config.d.ts +17 -0
  177. package/dist/node/config.js +31 -0
  178. package/dist/node/runtime.d.ts +50 -0
  179. package/dist/node/runtime.js +351 -0
  180. package/dist/server/handlers/__test-utils__/mock-supabase.d.ts +11 -0
  181. package/dist/server/handlers/__test-utils__/mock-supabase.js +393 -0
  182. package/dist/server/handlers/analytics.d.ts +17 -0
  183. package/dist/server/handlers/analytics.js +266 -0
  184. package/dist/server/handlers/api-keys.d.ts +6 -0
  185. package/dist/server/handlers/api-keys.js +221 -0
  186. package/dist/server/handlers/billing.d.ts +33 -0
  187. package/dist/server/handlers/billing.js +272 -0
  188. package/dist/server/handlers/browser.d.ts +10 -0
  189. package/dist/server/handlers/browser.js +517 -0
  190. package/dist/server/handlers/catalog.d.ts +99 -0
  191. package/dist/server/handlers/catalog.js +976 -0
  192. package/dist/server/handlers/comms.d.ts +254 -0
  193. package/dist/server/handlers/comms.js +588 -0
  194. package/dist/server/handlers/creations.d.ts +6 -0
  195. package/dist/server/handlers/creations.js +479 -0
  196. package/dist/server/handlers/crm.d.ts +89 -0
  197. package/dist/server/handlers/crm.js +538 -0
  198. package/dist/server/handlers/discovery.d.ts +6 -0
  199. package/dist/server/handlers/discovery.js +288 -0
  200. package/dist/server/handlers/embeddings.d.ts +92 -0
  201. package/dist/server/handlers/embeddings.js +197 -0
  202. package/dist/server/handlers/enrichment.d.ts +8 -0
  203. package/dist/server/handlers/enrichment.js +768 -0
  204. package/dist/server/handlers/image-gen.d.ts +6 -0
  205. package/dist/server/handlers/image-gen.js +409 -0
  206. package/dist/server/handlers/inventory.d.ts +319 -0
  207. package/dist/server/handlers/inventory.js +447 -0
  208. package/dist/server/handlers/kali.d.ts +10 -0
  209. package/dist/server/handlers/kali.js +210 -0
  210. package/dist/server/handlers/llm-providers.d.ts +6 -0
  211. package/dist/server/handlers/llm-providers.js +673 -0
  212. package/dist/server/handlers/local-agent.d.ts +6 -0
  213. package/dist/server/handlers/local-agent.js +118 -0
  214. package/dist/server/handlers/meta-ads.d.ts +111 -0
  215. package/dist/server/handlers/meta-ads.js +2279 -0
  216. package/dist/server/handlers/nodes.d.ts +33 -0
  217. package/dist/server/handlers/nodes.js +699 -0
  218. package/dist/server/handlers/operations.d.ts +138 -0
  219. package/dist/server/handlers/operations.js +131 -0
  220. package/dist/server/handlers/platform.d.ts +23 -0
  221. package/dist/server/handlers/platform.js +227 -0
  222. package/dist/server/handlers/supply-chain.d.ts +19 -0
  223. package/dist/server/handlers/supply-chain.js +327 -0
  224. package/dist/server/handlers/transcription.d.ts +17 -0
  225. package/dist/server/handlers/transcription.js +121 -0
  226. package/dist/server/handlers/video-gen.d.ts +6 -0
  227. package/dist/server/handlers/video-gen.js +466 -0
  228. package/dist/server/handlers/voice.d.ts +8 -0
  229. package/dist/server/handlers/voice.js +1146 -0
  230. package/dist/server/handlers/workflow-steps.d.ts +86 -0
  231. package/dist/server/handlers/workflow-steps.js +2349 -0
  232. package/dist/server/handlers/workflows.d.ts +7 -0
  233. package/dist/server/handlers/workflows.js +989 -0
  234. package/dist/server/index.d.ts +1 -0
  235. package/dist/server/index.js +2427 -0
  236. package/dist/server/lib/batch-client.d.ts +80 -0
  237. package/dist/server/lib/batch-client.js +467 -0
  238. package/dist/server/lib/code-worker-pool.d.ts +31 -0
  239. package/dist/server/lib/code-worker-pool.js +224 -0
  240. package/dist/server/lib/code-worker.d.ts +1 -0
  241. package/dist/server/lib/code-worker.js +188 -0
  242. package/dist/server/lib/compaction-service.d.ts +32 -0
  243. package/dist/server/lib/compaction-service.js +162 -0
  244. package/dist/server/lib/logger.d.ts +19 -0
  245. package/dist/server/lib/logger.js +46 -0
  246. package/dist/server/lib/otel.d.ts +38 -0
  247. package/dist/server/lib/otel.js +126 -0
  248. package/dist/server/lib/pg-rate-limiter.d.ts +21 -0
  249. package/dist/server/lib/pg-rate-limiter.js +86 -0
  250. package/dist/server/lib/prompt-sanitizer.d.ts +37 -0
  251. package/dist/server/lib/prompt-sanitizer.js +177 -0
  252. package/dist/server/lib/provider-capabilities.d.ts +85 -0
  253. package/dist/server/lib/provider-capabilities.js +190 -0
  254. package/dist/server/lib/provider-failover.d.ts +74 -0
  255. package/dist/server/lib/provider-failover.js +210 -0
  256. package/dist/server/lib/rate-limiter.d.ts +39 -0
  257. package/dist/server/lib/rate-limiter.js +147 -0
  258. package/dist/server/lib/server-agent-loop.d.ts +107 -0
  259. package/dist/server/lib/server-agent-loop.js +667 -0
  260. package/dist/server/lib/server-subagent.d.ts +78 -0
  261. package/dist/server/lib/server-subagent.js +203 -0
  262. package/dist/server/lib/session-checkpoint.d.ts +51 -0
  263. package/dist/server/lib/session-checkpoint.js +145 -0
  264. package/dist/server/lib/ssrf-guard.d.ts +13 -0
  265. package/dist/server/lib/ssrf-guard.js +240 -0
  266. package/dist/server/lib/supabase-client.d.ts +7 -0
  267. package/dist/server/lib/supabase-client.js +78 -0
  268. package/dist/server/lib/template-resolver.d.ts +31 -0
  269. package/dist/server/lib/template-resolver.js +215 -0
  270. package/dist/server/lib/utils.d.ts +16 -0
  271. package/dist/server/lib/utils.js +147 -0
  272. package/dist/server/local-agent-gateway.d.ts +82 -0
  273. package/dist/server/local-agent-gateway.js +426 -0
  274. package/dist/server/providers/anthropic.d.ts +20 -0
  275. package/dist/server/providers/anthropic.js +199 -0
  276. package/dist/server/providers/bedrock.d.ts +20 -0
  277. package/dist/server/providers/bedrock.js +194 -0
  278. package/dist/server/providers/gemini.d.ts +24 -0
  279. package/dist/server/providers/gemini.js +486 -0
  280. package/dist/server/providers/openai.d.ts +24 -0
  281. package/dist/server/providers/openai.js +522 -0
  282. package/dist/server/providers/registry.d.ts +32 -0
  283. package/dist/server/providers/registry.js +58 -0
  284. package/dist/server/providers/shared.d.ts +32 -0
  285. package/dist/server/providers/shared.js +124 -0
  286. package/dist/server/providers/types.d.ts +92 -0
  287. package/dist/server/providers/types.js +12 -0
  288. package/dist/server/proxy-handlers.d.ts +6 -0
  289. package/dist/server/proxy-handlers.js +89 -0
  290. package/dist/server/tool-router.d.ts +149 -0
  291. package/dist/server/tool-router.js +803 -0
  292. package/dist/server/validation.d.ts +24 -0
  293. package/dist/server/validation.js +301 -0
  294. package/dist/server/worker.d.ts +19 -0
  295. package/dist/server/worker.js +201 -0
  296. package/dist/setup.d.ts +8 -0
  297. package/dist/setup.js +181 -0
  298. package/dist/shared/agent-core.d.ts +157 -0
  299. package/dist/shared/agent-core.js +534 -0
  300. package/dist/shared/anthropic-types.d.ts +105 -0
  301. package/dist/shared/anthropic-types.js +7 -0
  302. package/dist/shared/api-client.d.ts +90 -0
  303. package/dist/shared/api-client.js +379 -0
  304. package/dist/shared/constants.d.ts +33 -0
  305. package/dist/shared/constants.js +80 -0
  306. package/dist/shared/sse-parser.d.ts +26 -0
  307. package/dist/shared/sse-parser.js +259 -0
  308. package/dist/shared/tool-dispatch.d.ts +52 -0
  309. package/dist/shared/tool-dispatch.js +191 -0
  310. package/dist/shared/types.d.ts +72 -0
  311. package/dist/shared/types.js +7 -0
  312. package/dist/updater.d.ts +25 -0
  313. package/dist/updater.js +140 -0
  314. package/dist/webchat/widget.d.ts +0 -0
  315. package/dist/webchat/widget.js +397 -0
  316. package/package.json +95 -0
  317. package/src/cli/services/builtin-skills/commit.md +19 -0
  318. package/src/cli/services/builtin-skills/review-pr.md +21 -0
  319. package/src/cli/services/builtin-skills/review.md +18 -0
@@ -0,0 +1,267 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ /**
3
+ * ChatApp — whale code CLI
4
+ *
5
+ * Uses Ink's <Static> for completed messages — written to stdout once,
6
+ * never re-rendered. Only the active area (streaming, tools, input)
7
+ * is managed by Ink's render loop. This prevents scroll bounce when
8
+ * content exceeds the terminal height.
9
+ */
10
+ import { useState, useEffect, useRef, useMemo, useCallback } from "react";
11
+ import { Box, Text, Static, useApp, useInput } from "ink";
12
+ import Spinner from "ink-spinner";
13
+ import { canUseAgent, getServerToolCount, getModelShortName, setModel, getPermissionMode, mcpClientManager, } from "../services/agent-loop.js";
14
+ import { CompletedMessage } from "./MessageList.js";
15
+ import { ToolIndicator } from "./ToolIndicator.js";
16
+ import { SubagentPanel } from "./SubagentPanel.js";
17
+ import { TeamPanel } from "./TeamPanel.js";
18
+ import { StreamingText } from "./StreamingText.js";
19
+ import { ChatInput } from "./ChatInput.js";
20
+ import { StoreSelector } from "./StoreSelector.js";
21
+ import { ModelSelector } from "./ModelSelector.js";
22
+ import { RewindViewer, RewindOutcome } from "./RewindViewer.js";
23
+ import { RewindManager } from "../services/rewind.js";
24
+ import { colors, symbols } from "../shared/Theme.js";
25
+ import { loadKeybindings, matchesBinding } from "../services/keybinding-manager.js";
26
+ import { loadConfig, updateConfig } from "../services/config-store.js";
27
+ import { createRequire } from "module";
28
+ import { fileURLToPath } from "url";
29
+ import { dirname, join } from "path";
30
+ // Extracted hooks
31
+ import { useSlashCommands } from "./hooks/useSlashCommands.js";
32
+ import { useAgentLoop } from "./hooks/useAgentLoop.js";
33
+ // Thinking verbs — rotate randomly each render (Claude Code parity)
34
+ const THINKING_VERBS = [
35
+ "thinking…",
36
+ "reasoning…",
37
+ "considering…",
38
+ "analyzing…",
39
+ "evaluating…",
40
+ "pondering…",
41
+ "processing…",
42
+ "reflecting…",
43
+ "examining…",
44
+ "working…",
45
+ ];
46
+ function randomVerb() {
47
+ return THINKING_VERBS[Math.floor(Math.random() * THINKING_VERBS.length)];
48
+ }
49
+ const PKG_NAME = "whale-code";
50
+ const __filename = fileURLToPath(import.meta.url);
51
+ const __dirname = dirname(__filename);
52
+ const PKG_VERSION = createRequire(import.meta.url)(join(__dirname, "..", "..", "..", "package.json")).version;
53
+ // ── Component ──
54
+ export function ChatApp() {
55
+ const { exit } = useApp();
56
+ // Core state
57
+ const [ready, setReady] = useState(false);
58
+ const [error, setError] = useState("");
59
+ const [messages, setMessages] = useState([]);
60
+ const [streamingText, setStreamingText] = useState("");
61
+ const [isStreaming, setIsStreaming] = useState(false);
62
+ const [activeTools, setActiveTools] = useState([]);
63
+ const [subagentActivity, setSubagentActivity] = useState(new Map());
64
+ const [completedSubagents, setCompletedSubagents] = useState([]);
65
+ const [teamState, setTeamState] = useState(null);
66
+ // UI state
67
+ const [userLabel, setUserLabel] = useState("");
68
+ const [toolsExpanded, setToolsExpanded] = useState(false);
69
+ const [serverToolsAvailable, setServerToolsAvailable] = useState(0);
70
+ const [storeSelectMode, setStoreSelectMode] = useState(false);
71
+ const [storeList, setStoreList] = useState([]);
72
+ const [modelSelectMode, setModelSelectMode] = useState(false);
73
+ const [currentModel, setCurrentModel] = useState(getModelShortName());
74
+ const [sessionId, setSessionId] = useState(null);
75
+ const [thinkingEnabled, setThinkingEnabled] = useState(() => {
76
+ try {
77
+ return loadConfig().thinking_enabled ?? true;
78
+ }
79
+ catch {
80
+ return true;
81
+ }
82
+ });
83
+ // Stream activity indicator — show spinner when no new content for 800ms+
84
+ const [streamStale, setStreamStale] = useState(false);
85
+ const lastContentUpdateRef = useRef(Date.now());
86
+ // Poll-based staleness check — avoids re-renders on every content change
87
+ useEffect(() => {
88
+ if (!isStreaming) {
89
+ setStreamStale(false);
90
+ return;
91
+ }
92
+ lastContentUpdateRef.current = Date.now();
93
+ const interval = setInterval(() => {
94
+ setStreamStale(Date.now() - lastContentUpdateRef.current > 800);
95
+ }, 400);
96
+ return () => clearInterval(interval);
97
+ }, [isStreaming]);
98
+ // Rewind state
99
+ const [showRewind, setShowRewind] = useState(false);
100
+ const rewindManagerRef = useRef(new RewindManager());
101
+ const turnIndexRef = useRef(0);
102
+ // Refs
103
+ const conversationRef = useRef([]);
104
+ const abortRef = useRef(null);
105
+ const accTextRef = useRef("");
106
+ const thinkingVerbRef = useRef(randomVerb());
107
+ // No timer cleanup needed — unified flush timer is managed inside useAgentLoop,
108
+ // stale polling interval is cleaned up by its own effect.
109
+ // ── Init ──
110
+ useEffect(() => {
111
+ const check = canUseAgent();
112
+ if (!check.ready) {
113
+ setError(check.reason || "Run 'whale login' to authenticate.");
114
+ }
115
+ else {
116
+ const config = loadConfig();
117
+ if (config.email)
118
+ setUserLabel(config.store_name || config.email);
119
+ setReady(true);
120
+ getServerToolCount().then((count) => setServerToolsAvailable(count));
121
+ mcpClientManager.connectAll().catch(() => { });
122
+ }
123
+ return () => {
124
+ mcpClientManager.disconnectAll().catch(() => { });
125
+ };
126
+ }, []);
127
+ // ── Keys (configurable via ~/.swagmanager/keybindings.json) ──
128
+ const kb = useMemo(() => loadKeybindings(), []);
129
+ useInput((input, key) => {
130
+ if (matchesBinding(kb.exit, input, key)) {
131
+ if (abortRef.current)
132
+ abortRef.current.abort();
133
+ exit();
134
+ }
135
+ if (matchesBinding(kb.cancel_stream, input, key) && isStreaming && abortRef.current) {
136
+ abortRef.current.abort();
137
+ }
138
+ if (matchesBinding(kb.toggle_expand, input, key)) {
139
+ setToolsExpanded((prev) => !prev);
140
+ }
141
+ if (matchesBinding(kb.toggle_thinking, input, key)) {
142
+ setThinkingEnabled((prev) => {
143
+ const next = !prev;
144
+ try {
145
+ updateConfig({ thinking_enabled: next });
146
+ }
147
+ catch { /* best effort */ }
148
+ setMessages((msgs) => [...msgs, {
149
+ role: "assistant",
150
+ text: ` Thinking: ${next ? "on" : "off"}`,
151
+ }]);
152
+ return next;
153
+ });
154
+ }
155
+ });
156
+ // ── Extracted hooks ──
157
+ const { handleCommand, handleStoreSelect, handleStoreCancel } = useSlashCommands({
158
+ exit, toolsExpanded, serverToolsAvailable, sessionId, thinkingEnabled,
159
+ conversationRef,
160
+ setMessages, setStreamingText, setActiveTools, setTeamState,
161
+ setStoreList, setStoreSelectMode, setModelSelectMode, setCurrentModel,
162
+ setSessionId, setThinkingEnabled, setUserLabel, setServerToolsAvailable,
163
+ setShowRewind,
164
+ rewindCheckpointCount: rewindManagerRef.current.getCheckpointCount(),
165
+ PKG_NAME, PKG_VERSION,
166
+ });
167
+ const { handleSend } = useAgentLoop({
168
+ isStreaming, thinkingEnabled, conversationRef, abortRef,
169
+ accTextRef, lastContentUpdateRef, thinkingVerbRef,
170
+ rewindManagerRef, turnIndexRef,
171
+ setMessages, setStreamingText, setIsStreaming, setActiveTools,
172
+ setSubagentActivity, setCompletedSubagents, setTeamState,
173
+ });
174
+ // ── Rewind handler ──
175
+ const handleRewind = useCallback((checkpointIndex, outcome) => {
176
+ const rm = rewindManagerRef.current;
177
+ if (outcome === RewindOutcome.Cancel) {
178
+ setShowRewind(false);
179
+ return;
180
+ }
181
+ if (outcome === RewindOutcome.RevertOnly) {
182
+ // Revert files only, keep conversation
183
+ const result = rm.revertFilesFrom(checkpointIndex);
184
+ setShowRewind(false);
185
+ const total = result.filesReverted.length + result.filesDeleted.length;
186
+ const parts = [];
187
+ if (result.filesReverted.length > 0)
188
+ parts.push(`${result.filesReverted.length} reverted`);
189
+ if (result.filesDeleted.length > 0)
190
+ parts.push(`${result.filesDeleted.length} deleted`);
191
+ if (result.errors.length > 0)
192
+ parts.push(`${result.errors.length} errors`);
193
+ setMessages(prev => [...prev, {
194
+ role: "assistant",
195
+ text: total > 0
196
+ ? ` ${symbols.check} File changes reverted (${parts.join(", ")})`
197
+ : ` ${symbols.check} No file changes to revert.`,
198
+ }]);
199
+ return;
200
+ }
201
+ // RewindOnly or RewindAndRevert
202
+ const shouldRevertFiles = outcome === RewindOutcome.RewindAndRevert;
203
+ const result = rm.rewindTo(checkpointIndex);
204
+ // Truncate messages
205
+ setMessages(prev => prev.slice(0, result.messageCount));
206
+ // Truncate conversation history to match
207
+ // We need to figure out how many Anthropic messages correspond to the checkpoint.
208
+ // The checkpoint stores messageCount which is the UI messages count at that point.
209
+ // Rebuild conversation from remaining messages by keeping only the conversation
210
+ // entries up to the corresponding turn.
211
+ const targetTurnIndex = rm.getCheckpoints()[checkpointIndex]?.turnIndex ?? 0;
212
+ // Each user/assistant exchange is roughly 2 conversation entries (user + assistant).
213
+ // But the actual conversation is maintained independently.
214
+ // We slice it to 2*(turnIndex+1) entries (user+assistant pairs)
215
+ const conversationSlicePoint = Math.min((targetTurnIndex + 1) * 2, conversationRef.current.length);
216
+ conversationRef.current = conversationRef.current.slice(0, conversationSlicePoint);
217
+ // Reset turn index to match
218
+ turnIndexRef.current = targetTurnIndex + 1;
219
+ setShowRewind(false);
220
+ const parts = [];
221
+ if (shouldRevertFiles) {
222
+ if (result.filesReverted.length > 0)
223
+ parts.push(`${result.filesReverted.length} files reverted`);
224
+ if (result.filesDeleted.length > 0)
225
+ parts.push(`${result.filesDeleted.length} files deleted`);
226
+ }
227
+ if (result.errors.length > 0)
228
+ parts.push(`${result.errors.length} revert errors`);
229
+ const detail = parts.length > 0 ? ` (${parts.join(", ")})` : "";
230
+ setMessages(prev => [...prev, {
231
+ role: "assistant",
232
+ text: ` ${symbols.check} Rewound to turn ${targetTurnIndex + 1}${detail}`,
233
+ }]);
234
+ }, []);
235
+ // ── Render ──
236
+ const termWidth = process.stdout.columns || 80;
237
+ const contentWidth = Math.max(20, termWidth - 2);
238
+ const { staticItems, dynamicMessages } = useMemo(() => {
239
+ const items = [{ id: "header", type: "header" }];
240
+ const cutoff = Math.max(0, messages.length - 1);
241
+ for (let i = 0; i < cutoff; i++) {
242
+ items.push({ id: `msg-${i}`, type: "message", msg: messages[i], index: i });
243
+ }
244
+ const tail = messages.length > 0 ? [messages[messages.length - 1]] : [];
245
+ return { staticItems: items, dynamicMessages: tail };
246
+ }, [messages]);
247
+ if (error) {
248
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, paddingRight: 1, children: [_jsx(Text, { children: " " }), _jsx(Text, { color: colors.brand, bold: true, children: "\u25C6 whale code" }), _jsx(Text, { children: " " }), _jsx(Text, { color: colors.error, children: error })] }));
249
+ }
250
+ if (!ready) {
251
+ return (_jsx(Box, { flexDirection: "column", paddingLeft: 1, paddingRight: 1, children: _jsx(Text, { color: colors.tertiary, children: "loading..." }) }));
252
+ }
253
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, paddingRight: 1, children: [_jsx(Static, { items: staticItems, children: (item) => {
254
+ if (item.type === "header") {
255
+ return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Text, { children: " " }), _jsxs(Text, { children: [_jsx(Text, { color: colors.brand, bold: true, children: "\u25C6 whale code" }), userLabel ? _jsxs(Text, { color: colors.dim, children: [" ", userLabel] }) : null, _jsxs(Text, { color: colors.dim, children: [" ", currentModel] }), thinkingEnabled ? _jsx(Text, { color: colors.warning, children: " thinking" }) : null, getPermissionMode() !== "default" && (_jsxs(Text, { color: getPermissionMode() === "yolo" ? colors.error : colors.info, children: [" ", getPermissionMode()] })), serverToolsAvailable > 0 ? (_jsxs(Text, { color: colors.tertiary, children: [" ", symbols.dot, " ", serverToolsAvailable, " server tools"] })) : null] }), _jsx(Text, { color: colors.separator, children: "─".repeat(contentWidth) })] }, item.id));
256
+ }
257
+ return _jsx(CompletedMessage, { msg: item.msg, index: item.index, toolsExpanded: toolsExpanded }, item.id);
258
+ } }), dynamicMessages.map((msg) => (_jsx(CompletedMessage, { msg: msg, index: messages.length - 1, toolsExpanded: toolsExpanded }, `dynamic-${messages.length - 1}`))), teamState ? (_jsx(TeamPanel, { team: teamState })) : (_jsxs(_Fragment, { children: [isStreaming && !streamingText && activeTools.length === 0 && (_jsxs(Box, { marginLeft: 2, marginY: 1, children: [_jsx(Text, { color: colors.brand, children: _jsx(Spinner, { type: "dots" }) }), _jsxs(Text, { color: colors.dim, children: [" ", thinkingVerbRef.current] })] })), activeTools.length > 0 && (_jsx(Box, { flexDirection: "column", marginLeft: 2, children: activeTools.map((tc, i) => (_jsxs(Box, { flexDirection: "column", children: [_jsx(ToolIndicator, { id: `live-${tc.name}-${i}`, name: tc.name, status: tc.status, input: tc.input, expanded: toolsExpanded }), tc.name === "task" && tc.status === "running" && (subagentActivity.size > 0 || completedSubagents.length > 0) && (_jsx(SubagentPanel, { running: subagentActivity, completed: completedSubagents }))] }, `live-${tc.name}-${i}`))) })), streamingText && (_jsxs(Box, { marginLeft: 2, flexDirection: "column", children: [_jsx(StreamingText, { text: streamingText }), streamStale && (_jsx(Box, { marginTop: 0, children: _jsx(Text, { color: colors.brand, children: _jsx(Spinner, { type: "dots" }) }) }))] }))] })), showRewind ? (_jsx(RewindViewer, { checkpoints: rewindManagerRef.current.getCheckpoints(), onRewind: handleRewind, onCancel: () => setShowRewind(false) })) : storeSelectMode ? (_jsx(StoreSelector, { stores: storeList, currentStoreId: loadConfig().store_id || "", onSelect: handleStoreSelect, onCancel: handleStoreCancel })) : modelSelectMode ? (_jsx(ModelSelector, { currentModel: currentModel, onSelect: (model) => {
259
+ setModelSelectMode(false);
260
+ setModel(model.value);
261
+ setCurrentModel(model.value);
262
+ setMessages((prev) => [...prev, {
263
+ role: "assistant",
264
+ text: ` ${symbols.check} Model: ${model.label} (${model.modelId})`,
265
+ }]);
266
+ }, onCancel: () => setModelSelectMode(false) })) : (_jsx(ChatInput, { onSubmit: handleSend, onCommand: handleCommand, disabled: isStreaming, agentName: "whale code" }))] }));
267
+ }
@@ -0,0 +1,39 @@
1
+ /**
2
+ * ChatInput — Custom input with bracketed paste + image attachments
3
+ *
4
+ * Replaces ink-text-input with raw stdin handling to fix:
5
+ * - Paste losing content on enter (newlines in paste triggered submit)
6
+ * - Multi-line paste mangled/unformatted
7
+ * - No drag-drop image support
8
+ *
9
+ * Features:
10
+ * - Bracketed paste mode — clean multi-line paste
11
+ * - Drag-drop images — detects image file paths, attaches as chips
12
+ * - Image chips above input like Claude Code: [image1.png] [image2.jpg]
13
+ * - Backspace on empty input removes last image
14
+ * - Slash command menu preserved
15
+ * - Multi-line input with ⎸ continuation markers
16
+ */
17
+ export interface ImageAttachment {
18
+ path: string;
19
+ name: string;
20
+ base64: string;
21
+ mediaType: string;
22
+ }
23
+ export interface FileAttachment {
24
+ path: string;
25
+ name: string;
26
+ }
27
+ export interface SlashCommand {
28
+ name: string;
29
+ description: string;
30
+ }
31
+ export declare const SLASH_COMMANDS: SlashCommand[];
32
+ interface ChatInputProps {
33
+ onSubmit: (message: string, images?: ImageAttachment[]) => void;
34
+ onCommand: (command: string) => void;
35
+ disabled: boolean;
36
+ agentName?: string;
37
+ }
38
+ export declare function ChatInput({ onSubmit, onCommand, disabled, agentName }: ChatInputProps): import("react/jsx-runtime").JSX.Element;
39
+ export {};