@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.
Files changed (481) hide show
  1. package/cli.js +155 -37
  2. package/dist/Tool.js +38 -0
  3. package/dist/Tool.js.map +3 -3
  4. package/dist/commands/agents/AgentsCommand.js +52 -26
  5. package/dist/commands/agents/AgentsCommand.js.map +2 -2
  6. package/dist/commands/agents/constants.js +1 -1
  7. package/dist/commands/agents/constants.js.map +1 -1
  8. package/dist/commands/agents/index.js +1 -1
  9. package/dist/commands/bug.js +74 -7
  10. package/dist/commands/bug.js.map +3 -3
  11. package/dist/commands/clear.js +3 -0
  12. package/dist/commands/clear.js.map +2 -2
  13. package/dist/commands/compact.js +37 -0
  14. package/dist/commands/compact.js.map +2 -2
  15. package/dist/commands/context.js +84 -0
  16. package/dist/commands/context.js.map +7 -0
  17. package/dist/commands/ctx_viz.js +18 -10
  18. package/dist/commands/ctx_viz.js.map +2 -2
  19. package/dist/commands/doctor.js +158 -12
  20. package/dist/commands/doctor.js.map +2 -2
  21. package/dist/commands/export.js +156 -0
  22. package/dist/commands/export.js.map +7 -0
  23. package/dist/commands/mcp-interactive.js +21 -12
  24. package/dist/commands/mcp-interactive.js.map +2 -2
  25. package/dist/commands/model.js +6 -5
  26. package/dist/commands/model.js.map +2 -2
  27. package/dist/commands/permissions.js +86 -0
  28. package/dist/commands/permissions.js.map +7 -0
  29. package/dist/commands/quit.js +3 -1
  30. package/dist/commands/quit.js.map +2 -2
  31. package/dist/commands/sandbox.js +104 -0
  32. package/dist/commands/sandbox.js.map +7 -0
  33. package/dist/commands/status.js +58 -0
  34. package/dist/commands/status.js.map +7 -0
  35. package/dist/commands/tasks.js +108 -0
  36. package/dist/commands/tasks.js.map +7 -0
  37. package/dist/commands/todos.js +123 -0
  38. package/dist/commands/todos.js.map +7 -0
  39. package/dist/commands.js +20 -2
  40. package/dist/commands.js.map +2 -2
  41. package/dist/components/AgentThinkingBlock.js +10 -18
  42. package/dist/components/AgentThinkingBlock.js.map +2 -2
  43. package/dist/components/BackgroundTasksPanel.js +78 -29
  44. package/dist/components/BackgroundTasksPanel.js.map +2 -2
  45. package/dist/components/BashStreamingProgress.js +24 -0
  46. package/dist/components/BashStreamingProgress.js.map +7 -0
  47. package/dist/components/CollapsibleHint.js +14 -0
  48. package/dist/components/CollapsibleHint.js.map +7 -0
  49. package/dist/components/FileEditToolUpdatedMessage.js +1 -1
  50. package/dist/components/FileEditToolUpdatedMessage.js.map +2 -2
  51. package/dist/components/HotkeyHelpPanel.js +137 -0
  52. package/dist/components/HotkeyHelpPanel.js.map +7 -0
  53. package/dist/components/Logo.js +5 -5
  54. package/dist/components/Logo.js.map +2 -2
  55. package/dist/components/Message.js +23 -7
  56. package/dist/components/Message.js.map +3 -3
  57. package/dist/components/ModelConfig.js +16 -3
  58. package/dist/components/ModelConfig.js.map +2 -2
  59. package/dist/components/ModelListManager.js +3 -3
  60. package/dist/components/ModelListManager.js.map +2 -2
  61. package/dist/components/ModelSelector/ModelSelector.js +1 -1
  62. package/dist/components/Onboarding.js +19 -14
  63. package/dist/components/Onboarding.js.map +2 -2
  64. package/dist/components/ProgressBar.js +74 -0
  65. package/dist/components/ProgressBar.js.map +7 -0
  66. package/dist/components/PromptInput.js +156 -46
  67. package/dist/components/PromptInput.js.map +2 -2
  68. package/dist/components/RequestStatusIndicator.js +194 -0
  69. package/dist/components/RequestStatusIndicator.js.map +7 -0
  70. package/dist/components/Spinner.js +92 -27
  71. package/dist/components/Spinner.js.map +2 -2
  72. package/dist/components/SpinnerSymbol.js +21 -27
  73. package/dist/components/SpinnerSymbol.js.map +2 -2
  74. package/dist/components/StreamingBashOutput.js +9 -8
  75. package/dist/components/StreamingBashOutput.js.map +2 -2
  76. package/dist/components/SubagentBlock.js +1 -1
  77. package/dist/components/SubagentBlock.js.map +1 -1
  78. package/dist/components/SubagentProgress.js +10 -11
  79. package/dist/components/SubagentProgress.js.map +2 -2
  80. package/dist/components/TaskCard.js +16 -13
  81. package/dist/components/TaskCard.js.map +2 -2
  82. package/dist/components/TodoChangeBlock.js +1 -1
  83. package/dist/components/TodoChangeBlock.js.map +2 -2
  84. package/dist/components/TodoPanel.js +120 -29
  85. package/dist/components/TodoPanel.js.map +3 -3
  86. package/dist/components/TokenCounter.js +74 -0
  87. package/dist/components/TokenCounter.js.map +7 -0
  88. package/dist/components/TokenWarning.js +2 -1
  89. package/dist/components/TokenWarning.js.map +2 -2
  90. package/dist/components/TreeConnector.js +25 -0
  91. package/dist/components/TreeConnector.js.map +7 -0
  92. package/dist/components/TurnCompletionIndicator.js +18 -0
  93. package/dist/components/TurnCompletionIndicator.js.map +7 -0
  94. package/dist/components/messages/AssistantTextMessage.js +5 -2
  95. package/dist/components/messages/AssistantTextMessage.js.map +2 -2
  96. package/dist/components/messages/AssistantThinkingMessage.js +18 -3
  97. package/dist/components/messages/AssistantThinkingMessage.js.map +2 -2
  98. package/dist/components/messages/AssistantToolUseMessage.js +11 -8
  99. package/dist/components/messages/AssistantToolUseMessage.js.map +2 -2
  100. package/dist/components/messages/GroupRenderer.js +53 -0
  101. package/dist/components/messages/GroupRenderer.js.map +7 -0
  102. package/dist/components/messages/NestedTasksPreview.js +12 -0
  103. package/dist/components/messages/NestedTasksPreview.js.map +7 -0
  104. package/dist/components/messages/ParallelTasksGroupView.js +92 -0
  105. package/dist/components/messages/ParallelTasksGroupView.js.map +7 -0
  106. package/dist/components/messages/TaskInModuleView.js +198 -0
  107. package/dist/components/messages/TaskInModuleView.js.map +7 -0
  108. package/dist/components/messages/TaskOutputContent.js +53 -0
  109. package/dist/components/messages/TaskOutputContent.js.map +7 -0
  110. package/dist/components/messages/UserPromptMessage.js +1 -1
  111. package/dist/components/messages/UserPromptMessage.js.map +2 -2
  112. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js +2 -3
  113. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js.map +2 -2
  114. package/dist/components/permissions/FallbackPermissionRequest.js +4 -4
  115. package/dist/components/permissions/FallbackPermissionRequest.js.map +2 -2
  116. package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js +4 -4
  117. package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js.map +2 -2
  118. package/dist/constants/colors.js +48 -0
  119. package/dist/constants/colors.js.map +2 -2
  120. package/dist/constants/formatRules.js +102 -0
  121. package/dist/constants/formatRules.js.map +7 -0
  122. package/dist/constants/prompts.js +12 -34
  123. package/dist/constants/prompts.js.map +2 -2
  124. package/dist/constants/symbols.js +64 -6
  125. package/dist/constants/symbols.js.map +2 -2
  126. package/dist/constants/timing.js +5 -0
  127. package/dist/constants/timing.js.map +2 -2
  128. package/dist/core/config/defaults.js +84 -0
  129. package/dist/core/config/defaults.js.map +7 -0
  130. package/dist/core/config/index.js +111 -0
  131. package/dist/core/config/index.js.map +7 -0
  132. package/dist/core/config/loader.js +221 -0
  133. package/dist/core/config/loader.js.map +7 -0
  134. package/dist/core/config/migrations.js +128 -0
  135. package/dist/core/config/migrations.js.map +7 -0
  136. package/dist/core/config/schema.js +178 -0
  137. package/dist/core/config/schema.js.map +7 -0
  138. package/dist/core/costTracker.js +138 -0
  139. package/dist/core/costTracker.js.map +7 -0
  140. package/dist/core/index.js +5 -0
  141. package/dist/core/index.js.map +7 -0
  142. package/dist/core/permissions/auditLog.js +204 -0
  143. package/dist/core/permissions/auditLog.js.map +7 -0
  144. package/dist/core/permissions/engine/index.js +3 -0
  145. package/dist/core/permissions/engine/index.js.map +7 -0
  146. package/dist/core/permissions/engine/permissionEngine.js +106 -0
  147. package/dist/core/permissions/engine/permissionEngine.js.map +7 -0
  148. package/dist/core/permissions/engine/types.js +1 -0
  149. package/dist/core/permissions/engine/types.js.map +7 -0
  150. package/dist/core/permissions/index.js +84 -0
  151. package/dist/core/permissions/index.js.map +7 -0
  152. package/dist/core/permissions/ruleEngine.js +259 -0
  153. package/dist/core/permissions/ruleEngine.js.map +7 -0
  154. package/dist/core/permissions/rules/allowedToolsRule.js +62 -0
  155. package/dist/core/permissions/rules/allowedToolsRule.js.map +7 -0
  156. package/dist/core/permissions/rules/autoEscalationRule.js +291 -0
  157. package/dist/core/permissions/rules/autoEscalationRule.js.map +7 -0
  158. package/dist/core/permissions/rules/index.js +46 -0
  159. package/dist/core/permissions/rules/index.js.map +7 -0
  160. package/dist/core/permissions/rules/planModeRule.js +55 -0
  161. package/dist/core/permissions/rules/planModeRule.js.map +7 -0
  162. package/dist/core/permissions/rules/projectBoundaryRule.js +168 -0
  163. package/dist/core/permissions/rules/projectBoundaryRule.js.map +7 -0
  164. package/dist/core/permissions/rules/safeModeRule.js +65 -0
  165. package/dist/core/permissions/rules/safeModeRule.js.map +7 -0
  166. package/dist/core/permissions/rules/sensitivePathsRule.js +340 -0
  167. package/dist/core/permissions/rules/sensitivePathsRule.js.map +7 -0
  168. package/dist/core/permissions/types.js +127 -0
  169. package/dist/core/permissions/types.js.map +7 -0
  170. package/dist/core/tools/executor.js +143 -0
  171. package/dist/core/tools/executor.js.map +7 -0
  172. package/dist/core/tools/index.js +15 -0
  173. package/dist/core/tools/index.js.map +7 -0
  174. package/dist/core/tools/registry.js +183 -0
  175. package/dist/core/tools/registry.js.map +7 -0
  176. package/dist/core/tools/types.js +1 -0
  177. package/dist/core/tools/types.js.map +7 -0
  178. package/dist/cost-tracker.js +23 -15
  179. package/dist/cost-tracker.js.map +2 -2
  180. package/dist/entrypoints/cli.js +43 -43
  181. package/dist/entrypoints/cli.js.map +2 -2
  182. package/dist/entrypoints/mcp.js +12 -4
  183. package/dist/entrypoints/mcp.js.map +2 -2
  184. package/dist/history.js +14 -3
  185. package/dist/history.js.map +2 -2
  186. package/dist/hooks/useAgentTranscripts.js +116 -0
  187. package/dist/hooks/useAgentTranscripts.js.map +7 -0
  188. package/dist/hooks/useAnimationSync.js +53 -0
  189. package/dist/hooks/useAnimationSync.js.map +7 -0
  190. package/dist/hooks/useArrowKeyHistory.js +4 -2
  191. package/dist/hooks/useArrowKeyHistory.js.map +2 -2
  192. package/dist/hooks/useCanUseTool.js +3 -1
  193. package/dist/hooks/useCanUseTool.js.map +2 -2
  194. package/dist/hooks/useCancelRequest.js +4 -1
  195. package/dist/hooks/useCancelRequest.js.map +2 -2
  196. package/dist/hooks/useExitOnCtrlCD.js +9 -5
  197. package/dist/hooks/useExitOnCtrlCD.js.map +2 -2
  198. package/dist/hooks/useHookStatus.js +40 -0
  199. package/dist/hooks/useHookStatus.js.map +7 -0
  200. package/dist/hooks/useLogMessages.js +17 -1
  201. package/dist/hooks/useLogMessages.js.map +2 -2
  202. package/dist/hooks/useMessageGroups.js +43 -0
  203. package/dist/hooks/useMessageGroups.js.map +7 -0
  204. package/dist/hooks/useTerminalSize.js +62 -6
  205. package/dist/hooks/useTerminalSize.js.map +2 -2
  206. package/dist/hooks/useUnifiedCompletion.js +69 -0
  207. package/dist/hooks/useUnifiedCompletion.js.map +2 -2
  208. package/dist/i18n/index.js +109 -0
  209. package/dist/i18n/index.js.map +7 -0
  210. package/dist/i18n/locales/en.js +347 -0
  211. package/dist/i18n/locales/en.js.map +7 -0
  212. package/dist/i18n/locales/index.js +7 -0
  213. package/dist/i18n/locales/index.js.map +7 -0
  214. package/dist/i18n/locales/zh-CN.js +347 -0
  215. package/dist/i18n/locales/zh-CN.js.map +7 -0
  216. package/dist/i18n/types.js +8 -0
  217. package/dist/i18n/types.js.map +7 -0
  218. package/dist/query.js +175 -17
  219. package/dist/query.js.map +3 -3
  220. package/dist/screens/REPL.js +501 -192
  221. package/dist/screens/REPL.js.map +3 -3
  222. package/dist/services/adapters/chatCompletions.js +3 -1
  223. package/dist/services/adapters/chatCompletions.js.map +2 -2
  224. package/dist/services/adapters/messageNormalizer.js +354 -0
  225. package/dist/services/adapters/messageNormalizer.js.map +7 -0
  226. package/dist/services/adapters/responsesAPI.js +6 -3
  227. package/dist/services/adapters/responsesAPI.js.map +2 -2
  228. package/dist/services/checkpointManager.js +386 -0
  229. package/dist/services/checkpointManager.js.map +7 -0
  230. package/dist/services/claude.js +138 -11
  231. package/dist/services/claude.js.map +3 -3
  232. package/dist/services/compressionService.js +50 -1
  233. package/dist/services/compressionService.js.map +2 -2
  234. package/dist/services/contextMonitor.js +162 -0
  235. package/dist/services/contextMonitor.js.map +7 -0
  236. package/dist/services/customCommands.js +60 -41
  237. package/dist/services/customCommands.js.map +2 -2
  238. package/dist/services/hookExecutor.js +173 -1
  239. package/dist/services/hookExecutor.js.map +2 -2
  240. package/dist/services/intelligentCompactor.js +281 -0
  241. package/dist/services/intelligentCompactor.js.map +7 -0
  242. package/dist/services/lspConfig.js +109 -0
  243. package/dist/services/lspConfig.js.map +7 -0
  244. package/dist/services/mcpClient.js +273 -34
  245. package/dist/services/mcpClient.js.map +2 -2
  246. package/dist/services/modelOrchestrator.js +310 -0
  247. package/dist/services/modelOrchestrator.js.map +7 -0
  248. package/dist/services/openai.js +8 -1
  249. package/dist/services/openai.js.map +2 -2
  250. package/dist/services/outputStyles.js +138 -0
  251. package/dist/services/outputStyles.js.map +7 -0
  252. package/dist/services/plugins/index.js +5 -0
  253. package/dist/services/plugins/index.js.map +7 -0
  254. package/dist/services/plugins/lspServers.js +188 -0
  255. package/dist/services/plugins/lspServers.js.map +7 -0
  256. package/dist/services/plugins/pluginRuntime.js +229 -0
  257. package/dist/services/plugins/pluginRuntime.js.map +7 -0
  258. package/dist/services/plugins/pluginValidation.js +219 -0
  259. package/dist/services/plugins/pluginValidation.js.map +7 -0
  260. package/dist/services/plugins/skillMarketplace.js +556 -0
  261. package/dist/services/plugins/skillMarketplace.js.map +7 -0
  262. package/dist/services/responseStateManager.js +37 -3
  263. package/dist/services/responseStateManager.js.map +2 -2
  264. package/dist/services/sandbox/filesystemBoundary.js +300 -0
  265. package/dist/services/sandbox/filesystemBoundary.js.map +7 -0
  266. package/dist/services/sandbox/index.js +14 -0
  267. package/dist/services/sandbox/index.js.map +7 -0
  268. package/dist/services/sandbox/networkProxy.js +293 -0
  269. package/dist/services/sandbox/networkProxy.js.map +7 -0
  270. package/dist/services/sandbox/sandboxController.js +574 -0
  271. package/dist/services/sandbox/sandboxController.js.map +7 -0
  272. package/dist/services/sandbox/types.js +50 -0
  273. package/dist/services/sandbox/types.js.map +7 -0
  274. package/dist/services/sessionMemory.js +266 -0
  275. package/dist/services/sessionMemory.js.map +7 -0
  276. package/dist/services/taskRouter.js +324 -0
  277. package/dist/services/taskRouter.js.map +7 -0
  278. package/dist/tools/ArchitectTool/ArchitectTool.js +7 -1
  279. package/dist/tools/ArchitectTool/ArchitectTool.js.map +2 -2
  280. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js +3 -0
  281. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js.map +2 -2
  282. package/dist/tools/AskUserQuestionTool/AskUserQuestionTool.js.map +2 -2
  283. package/dist/tools/BaseTool.js +72 -0
  284. package/dist/tools/BaseTool.js.map +7 -0
  285. package/dist/tools/BashOutputTool/BashOutputToolResultMessage.js +3 -0
  286. package/dist/tools/BashOutputTool/BashOutputToolResultMessage.js.map +2 -2
  287. package/dist/tools/BashTool/BashTool.js +60 -3
  288. package/dist/tools/BashTool/BashTool.js.map +2 -2
  289. package/dist/tools/BashTool/BashToolResultMessage.js +3 -0
  290. package/dist/tools/BashTool/BashToolResultMessage.js.map +2 -2
  291. package/dist/tools/BashTool/OutputLine.js +54 -0
  292. package/dist/tools/BashTool/OutputLine.js.map +2 -2
  293. package/dist/tools/BashTool/prompt.js +192 -3
  294. package/dist/tools/BashTool/prompt.js.map +2 -2
  295. package/dist/tools/FileEditTool/FileEditTool.js +29 -4
  296. package/dist/tools/FileEditTool/FileEditTool.js.map +2 -2
  297. package/dist/tools/FileWriteTool/FileWriteTool.js +5 -5
  298. package/dist/tools/FileWriteTool/FileWriteTool.js.map +2 -2
  299. package/dist/tools/GlobTool/GlobTool.js +4 -2
  300. package/dist/tools/GlobTool/GlobTool.js.map +2 -2
  301. package/dist/tools/GrepTool/GrepTool.js +36 -7
  302. package/dist/tools/GrepTool/GrepTool.js.map +2 -2
  303. package/dist/tools/KillShellTool/KillShellToolResultMessage.js +3 -0
  304. package/dist/tools/KillShellTool/KillShellToolResultMessage.js.map +2 -2
  305. package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js +109 -0
  306. package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js.map +7 -0
  307. package/dist/tools/ListMcpResourcesTool/prompt.js +19 -0
  308. package/dist/tools/ListMcpResourcesTool/prompt.js.map +7 -0
  309. package/dist/tools/LspTool/LspTool.js +664 -0
  310. package/dist/tools/LspTool/LspTool.js.map +7 -0
  311. package/dist/tools/LspTool/prompt.js +27 -0
  312. package/dist/tools/LspTool/prompt.js.map +7 -0
  313. package/dist/tools/MCPTool/MCPTool.js +9 -1
  314. package/dist/tools/MCPTool/MCPTool.js.map +2 -2
  315. package/dist/tools/MemoryReadTool/MemoryReadTool.js +19 -6
  316. package/dist/tools/MemoryReadTool/MemoryReadTool.js.map +2 -2
  317. package/dist/tools/MemoryWriteTool/MemoryWriteTool.js +6 -6
  318. package/dist/tools/MemoryWriteTool/MemoryWriteTool.js.map +2 -2
  319. package/dist/tools/MultiEditTool/MultiEditTool.js +19 -2
  320. package/dist/tools/MultiEditTool/MultiEditTool.js.map +2 -2
  321. package/dist/tools/NotebookEditTool/NotebookEditTool.js +5 -1
  322. package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +2 -2
  323. package/dist/tools/NotebookReadTool/NotebookReadTool.js.map +2 -2
  324. package/dist/tools/PlanModeTool/EnterPlanModeTool.js +74 -0
  325. package/dist/tools/PlanModeTool/EnterPlanModeTool.js.map +7 -0
  326. package/dist/tools/PlanModeTool/ExitPlanModeTool.js +108 -0
  327. package/dist/tools/PlanModeTool/ExitPlanModeTool.js.map +7 -0
  328. package/dist/tools/PlanModeTool/prompt.js +94 -0
  329. package/dist/tools/PlanModeTool/prompt.js.map +7 -0
  330. package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js +130 -0
  331. package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js.map +7 -0
  332. package/dist/tools/ReadMcpResourceTool/prompt.js +17 -0
  333. package/dist/tools/ReadMcpResourceTool/prompt.js.map +7 -0
  334. package/dist/tools/SkillTool/SkillTool.js +6 -1
  335. package/dist/tools/SkillTool/SkillTool.js.map +2 -2
  336. package/dist/tools/SlashCommandTool/SlashCommandTool.js +260 -0
  337. package/dist/tools/SlashCommandTool/SlashCommandTool.js.map +7 -0
  338. package/dist/tools/SlashCommandTool/prompt.js +35 -0
  339. package/dist/tools/SlashCommandTool/prompt.js.map +7 -0
  340. package/dist/tools/TaskOutputTool/TaskOutputTool.js +189 -0
  341. package/dist/tools/TaskOutputTool/TaskOutputTool.js.map +7 -0
  342. package/dist/tools/TaskOutputTool/prompt.js +15 -0
  343. package/dist/tools/TaskOutputTool/prompt.js.map +7 -0
  344. package/dist/tools/TaskTool/TaskTool.js +302 -104
  345. package/dist/tools/TaskTool/TaskTool.js.map +2 -2
  346. package/dist/tools/TaskTool/prompt.js.map +2 -2
  347. package/dist/tools/TodoWriteTool/TodoWriteTool.js +42 -77
  348. package/dist/tools/TodoWriteTool/TodoWriteTool.js.map +2 -2
  349. package/dist/tools/URLFetcherTool/URLFetcherTool.js +4 -1
  350. package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +2 -2
  351. package/dist/tools/URLFetcherTool/cache.js +55 -8
  352. package/dist/tools/URLFetcherTool/cache.js.map +2 -2
  353. package/dist/tools.js +31 -2
  354. package/dist/tools.js.map +2 -2
  355. package/dist/types/hooks.js +4 -0
  356. package/dist/types/hooks.js.map +2 -2
  357. package/dist/types/marketplace.js.map +2 -2
  358. package/dist/types/messageGroup.js +36 -0
  359. package/dist/types/messageGroup.js.map +7 -0
  360. package/dist/types/plugin.js.map +2 -2
  361. package/dist/types/thinking.js +1 -0
  362. package/dist/types/thinking.js.map +7 -0
  363. package/dist/utils/BackgroundShellManager.js +136 -39
  364. package/dist/utils/BackgroundShellManager.js.map +2 -2
  365. package/dist/utils/MessageBatchBuffer.js +102 -0
  366. package/dist/utils/MessageBatchBuffer.js.map +7 -0
  367. package/dist/utils/PersistentShell.js +151 -1
  368. package/dist/utils/PersistentShell.js.map +2 -2
  369. package/dist/utils/agentLoader.js +1 -23
  370. package/dist/utils/agentLoader.js.map +2 -2
  371. package/dist/utils/agentTranscripts.js +641 -0
  372. package/dist/utils/agentTranscripts.js.map +7 -0
  373. package/dist/utils/animationManager.js +213 -0
  374. package/dist/utils/animationManager.js.map +7 -0
  375. package/dist/utils/animationSync.js +110 -0
  376. package/dist/utils/animationSync.js.map +7 -0
  377. package/dist/utils/asyncFile.js +215 -0
  378. package/dist/utils/asyncFile.js.map +7 -0
  379. package/dist/utils/backgroundAgentManager.js +231 -0
  380. package/dist/utils/backgroundAgentManager.js.map +7 -0
  381. package/dist/utils/config.js +63 -7
  382. package/dist/utils/config.js.map +2 -2
  383. package/dist/utils/conversationRecovery.js +19 -0
  384. package/dist/utils/conversationRecovery.js.map +2 -2
  385. package/dist/utils/exit.js +73 -0
  386. package/dist/utils/exit.js.map +7 -0
  387. package/dist/utils/format.js +73 -5
  388. package/dist/utils/format.js.map +2 -2
  389. package/dist/utils/generators.js +76 -6
  390. package/dist/utils/generators.js.map +2 -2
  391. package/dist/utils/globalErrorHandler.js +149 -0
  392. package/dist/utils/globalErrorHandler.js.map +7 -0
  393. package/dist/utils/groupHandlers/index.js +8 -0
  394. package/dist/utils/groupHandlers/index.js.map +7 -0
  395. package/dist/utils/groupHandlers/parallelTasksHandler.js +140 -0
  396. package/dist/utils/groupHandlers/parallelTasksHandler.js.map +7 -0
  397. package/dist/utils/groupHandlers/taskHandler.js +104 -0
  398. package/dist/utils/groupHandlers/taskHandler.js.map +7 -0
  399. package/dist/utils/groupHandlers/types.js +1 -0
  400. package/dist/utils/groupHandlers/types.js.map +7 -0
  401. package/dist/utils/logRotation.js +224 -0
  402. package/dist/utils/logRotation.js.map +7 -0
  403. package/dist/utils/marketplaceManager.js +3 -5
  404. package/dist/utils/marketplaceManager.js.map +2 -2
  405. package/dist/utils/memSafety.js +264 -0
  406. package/dist/utils/memSafety.js.map +7 -0
  407. package/dist/utils/messageGroupManager.js +274 -0
  408. package/dist/utils/messageGroupManager.js.map +7 -0
  409. package/dist/utils/messages.js +13 -4
  410. package/dist/utils/messages.js.map +2 -2
  411. package/dist/utils/model.js +119 -15
  412. package/dist/utils/model.js.map +3 -3
  413. package/dist/utils/permissions/filesystem.js +157 -5
  414. package/dist/utils/permissions/filesystem.js.map +2 -2
  415. package/dist/utils/plan/planMode.js +143 -0
  416. package/dist/utils/plan/planMode.js.map +7 -0
  417. package/dist/utils/pluginLoader.js +17 -21
  418. package/dist/utils/pluginLoader.js.map +2 -2
  419. package/dist/utils/ripgrep.js +55 -2
  420. package/dist/utils/ripgrep.js.map +2 -2
  421. package/dist/utils/sanitizeInput.js +32 -0
  422. package/dist/utils/sanitizeInput.js.map +7 -0
  423. package/dist/utils/secureKeyStorage.js +312 -0
  424. package/dist/utils/secureKeyStorage.js.map +7 -0
  425. package/dist/utils/session/sessionPlugins.js +67 -0
  426. package/dist/utils/session/sessionPlugins.js.map +7 -0
  427. package/dist/utils/taskDisplayUtils.js +257 -0
  428. package/dist/utils/taskDisplayUtils.js.map +7 -0
  429. package/dist/utils/teamConfig.js +2 -1
  430. package/dist/utils/teamConfig.js.map +2 -2
  431. package/dist/utils/todoStorage.js +92 -2
  432. package/dist/utils/todoStorage.js.map +2 -2
  433. package/dist/utils/toolTimeout.js +136 -0
  434. package/dist/utils/toolTimeout.js.map +7 -0
  435. package/dist/utils/tooling/safeRender.js +115 -0
  436. package/dist/utils/tooling/safeRender.js.map +7 -0
  437. package/dist/utils/userFriendlyError.js +346 -0
  438. package/dist/utils/userFriendlyError.js.map +7 -0
  439. package/dist/utils/vendor/ripgrep/arm64-darwin/rg +0 -0
  440. package/dist/version.js +2 -2
  441. package/dist/version.js.map +1 -1
  442. package/package.json +14 -4
  443. package/scripts/postinstall.js +128 -38
  444. package/dist/commands/agents.js +0 -2086
  445. package/dist/commands/agents.js.map +0 -7
  446. package/dist/commands/build.js +0 -74
  447. package/dist/commands/build.js.map +0 -7
  448. package/dist/commands/compression.js +0 -57
  449. package/dist/commands/compression.js.map +0 -7
  450. package/dist/commands/listen.js +0 -37
  451. package/dist/commands/listen.js.map +0 -7
  452. package/dist/commands/login.js +0 -37
  453. package/dist/commands/login.js.map +0 -7
  454. package/dist/commands/logout.js +0 -33
  455. package/dist/commands/logout.js.map +0 -7
  456. package/dist/commands/mcp.js +0 -40
  457. package/dist/commands/mcp.js.map +0 -7
  458. package/dist/commands/mcp_refresh.js +0 -40
  459. package/dist/commands/mcp_refresh.js.map +0 -7
  460. package/dist/commands/modelstatus.js +0 -21
  461. package/dist/commands/modelstatus.js.map +0 -7
  462. package/dist/commands/onboarding.js +0 -36
  463. package/dist/commands/onboarding.js.map +0 -7
  464. package/dist/commands/plugin-interactive.js +0 -446
  465. package/dist/commands/plugin-interactive.js.map +0 -7
  466. package/dist/commands/pr_comments.js +0 -61
  467. package/dist/commands/pr_comments.js.map +0 -7
  468. package/dist/commands/release-notes.js +0 -30
  469. package/dist/commands/release-notes.js.map +0 -7
  470. package/dist/commands/review.js +0 -51
  471. package/dist/commands/review.js.map +0 -7
  472. package/dist/components/Bug.js +0 -147
  473. package/dist/components/Bug.js.map +0 -7
  474. package/dist/components/ModelSelector.js +0 -2062
  475. package/dist/components/ModelSelector.js.map +0 -7
  476. package/dist/components/ModelStatusDisplay.js +0 -87
  477. package/dist/components/ModelStatusDisplay.js.map +0 -7
  478. package/dist/entrypoints/cli-wrapper.js +0 -61
  479. package/dist/entrypoints/cli-wrapper.js.map +0 -7
  480. package/dist/screens/Doctor.js +0 -22
  481. package/dist/screens/Doctor.js.map +0 -7
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/agentTranscripts.ts"],
4
+ "sourcesContent": ["/**\n * Agent Transcripts System\n *\n * Provides persistent storage for agent execution history, enabling:\n * - Resume interrupted agent executions\n * - View agent execution history\n * - Debug and audit agent behavior\n *\n * Storage location: ~/.minto/transcripts/{agentId}.json\n */\n\nimport {\n existsSync,\n readFileSync,\n writeFileSync,\n mkdirSync,\n readdirSync,\n unlinkSync,\n} from 'fs'\nimport { join } from 'path'\nimport { homedir } from 'os'\nimport { EventEmitter } from 'events'\nimport type { Message as MessageType } from '@query'\nimport { debug as debugLogger } from './debugLogger'\n\n/**\n * Event types for transcript updates\n */\nexport type TranscriptEventType =\n | 'created'\n | 'updated'\n | 'message_appended'\n | 'status_changed'\n | 'completed'\n | 'failed'\n | 'interrupted'\n\n/**\n * Event payload for transcript events\n */\nexport interface TranscriptEvent {\n type: TranscriptEventType\n agentId: string\n toolUseId?: string\n transcript: AgentTranscript\n previousStatus?: AgentStatus\n}\n\n/**\n * AgentTranscript Event Emitter\n *\n * Provides real-time event notification for transcript changes,\n * enabling UI components to react immediately without polling.\n *\n * Performance optimizations:\n * - Throttled message_appended events to reduce UI re-renders\n * - Immediate emission for status changes and critical events\n * - Batched updates for high-frequency events\n *\n * Usage:\n * ```typescript\n * // Subscribe to all updates\n * transcriptEmitter.on('update', (event) => { ... })\n *\n * // Subscribe to specific agent updates\n * transcriptEmitter.on('update:agent-123', (transcript) => { ... })\n *\n * // Subscribe to specific event types\n * transcriptEmitter.on('status_changed', (event) => { ... })\n * ```\n */\nclass AgentTranscriptEmitter extends EventEmitter {\n private static instance: AgentTranscriptEmitter\n\n // Throttle configuration\n private static readonly THROTTLE_INTERVAL = 100 // ms between message_appended events\n private static readonly HIGH_PRIORITY_EVENTS: TranscriptEventType[] = [\n 'created',\n 'completed',\n 'failed',\n 'interrupted',\n 'status_changed',\n ]\n\n // Throttle state per agent\n private throttleState: Map<\n string,\n {\n lastEmitTime: number\n pendingEvent: TranscriptEvent | null\n timeoutId: ReturnType<typeof setTimeout> | null\n }\n > = new Map()\n\n private constructor() {\n super()\n // Increase max listeners for concurrent agent scenarios\n this.setMaxListeners(100)\n }\n\n static getInstance(): AgentTranscriptEmitter {\n if (!AgentTranscriptEmitter.instance) {\n AgentTranscriptEmitter.instance = new AgentTranscriptEmitter()\n }\n return AgentTranscriptEmitter.instance\n }\n\n /**\n * Get or create throttle state for an agent\n */\n private getThrottleState(agentId: string) {\n let state = this.throttleState.get(agentId)\n if (!state) {\n state = {\n lastEmitTime: 0,\n pendingEvent: null,\n timeoutId: null,\n }\n this.throttleState.set(agentId, state)\n }\n return state\n }\n\n /**\n * Clean up throttle state for an agent (call when agent completes)\n */\n private cleanupThrottleState(agentId: string): void {\n const state = this.throttleState.get(agentId)\n if (state?.timeoutId) {\n clearTimeout(state.timeoutId)\n }\n this.throttleState.delete(agentId)\n }\n\n /**\n * Internal emit that actually sends the event\n */\n private doEmit(event: TranscriptEvent): void {\n // Emit general update event\n this.emit('update', event)\n\n // Emit agent-specific event\n this.emit(`update:${event.agentId}`, event.transcript)\n\n // Emit toolUseId-specific event (for Task tool UI updates)\n if (event.toolUseId) {\n this.emit(`update:toolUse:${event.toolUseId}`, event.transcript)\n }\n\n // Emit event-type specific event\n this.emit(event.type, event)\n\n debugLogger.trace('TRANSCRIPT_EVENT_EMITTED', {\n type: event.type,\n agentId: event.agentId,\n status: event.transcript.status,\n })\n }\n\n /**\n * Emit a transcript update event\n *\n * High-priority events (status changes, completion) are emitted immediately.\n * Low-priority events (message_appended) are throttled to reduce UI flickering.\n */\n emitUpdate(event: TranscriptEvent): void {\n // High-priority events: emit immediately\n if (AgentTranscriptEmitter.HIGH_PRIORITY_EVENTS.includes(event.type)) {\n // Clean up throttle state on completion events\n if (\n event.type === 'completed' ||\n event.type === 'failed' ||\n event.type === 'interrupted'\n ) {\n // Flush any pending event first\n const state = this.throttleState.get(event.agentId)\n if (state?.pendingEvent) {\n this.doEmit(state.pendingEvent)\n }\n this.cleanupThrottleState(event.agentId)\n }\n\n this.doEmit(event)\n return\n }\n\n // Low-priority events: apply throttling\n const state = this.getThrottleState(event.agentId)\n const now = Date.now()\n const timeSinceLastEmit = now - state.lastEmitTime\n\n if (timeSinceLastEmit >= AgentTranscriptEmitter.THROTTLE_INTERVAL) {\n // Enough time has passed, emit immediately\n state.lastEmitTime = now\n state.pendingEvent = null\n if (state.timeoutId) {\n clearTimeout(state.timeoutId)\n state.timeoutId = null\n }\n this.doEmit(event)\n } else {\n // Store as pending and schedule delayed emit\n state.pendingEvent = event\n\n if (!state.timeoutId) {\n const delay =\n AgentTranscriptEmitter.THROTTLE_INTERVAL - timeSinceLastEmit\n state.timeoutId = setTimeout(() => {\n const currentState = this.throttleState.get(event.agentId)\n if (currentState?.pendingEvent) {\n currentState.lastEmitTime = Date.now()\n const pendingEvent = currentState.pendingEvent\n currentState.pendingEvent = null\n currentState.timeoutId = null\n this.doEmit(pendingEvent)\n }\n }, delay)\n }\n }\n }\n\n /**\n * Subscribe to updates for a specific agent\n */\n onAgentUpdate(\n agentId: string,\n callback: (transcript: AgentTranscript) => void,\n ): () => void {\n const eventName = `update:${agentId}`\n this.on(eventName, callback)\n return () => this.off(eventName, callback)\n }\n\n /**\n * Subscribe to updates for a specific toolUseId\n */\n onToolUseUpdate(\n toolUseId: string,\n callback: (transcript: AgentTranscript) => void,\n ): () => void {\n const eventName = `update:toolUse:${toolUseId}`\n this.on(eventName, callback)\n return () => this.off(eventName, callback)\n }\n\n /**\n * Subscribe to all updates (any agent)\n */\n onAnyUpdate(callback: (event: TranscriptEvent) => void): () => void {\n this.on('update', callback)\n return () => this.off('update', callback)\n }\n\n /**\n * Subscribe to status change events\n */\n onStatusChange(callback: (event: TranscriptEvent) => void): () => void {\n this.on('status_changed', callback)\n return () => this.off('status_changed', callback)\n }\n}\n\n// Export singleton instance\nexport const transcriptEmitter = AgentTranscriptEmitter.getInstance()\n\n/**\n * Agent execution status\n */\nexport type AgentStatus = 'running' | 'completed' | 'failed' | 'interrupted'\n\n/**\n * Token usage statistics\n */\nexport interface TokenUsage {\n inputTokens: number\n outputTokens: number\n cacheReadTokens?: number\n cacheCreationTokens?: number\n}\n\n/**\n * Agent Transcript - Complete record of an agent's execution\n */\nexport interface AgentTranscript {\n /** Unique identifier for this agent execution */\n agentId: string\n\n /** Type of agent (e.g., \"general-purpose\", \"strategic-market-analyzer\") */\n agentType: string\n\n /** Parent agent ID if this is a nested agent call */\n parentAgentId?: string\n\n /** The tool_use ID that triggered this agent (for UI association) */\n toolUseId?: string\n\n /** Short description of the task (3-5 words) */\n description: string\n\n /** Full prompt given to the agent */\n prompt: string\n\n /** Unix timestamp when execution started */\n startTime: number\n\n /** Unix timestamp when execution ended (undefined if still running) */\n endTime?: number\n\n /** Current execution status */\n status: AgentStatus\n\n /** All messages in the agent's conversation */\n messages: MessageType[]\n\n /** Number of tool uses during execution */\n toolUseCount: number\n\n /** Token usage statistics */\n tokenUsage: TokenUsage\n\n /** Model used for this agent */\n model: string\n\n /** Fork number for log correlation */\n forkNumber?: number\n\n /** Message log name for log correlation */\n messageLogName?: string\n\n /** Additional metadata */\n metadata?: Record<string, unknown>\n\n /** API retry events (for display in UI) */\n retryEvents?: Array<{\n timestamp: number\n attempt: number\n maxRetries: number\n errorMessage: string\n delayMs: number\n }>\n}\n\n/**\n * Summary of an agent transcript (for listing)\n */\nexport interface AgentTranscriptSummary {\n agentId: string\n agentType: string\n description: string\n status: AgentStatus\n startTime: number\n endTime?: number\n messageCount: number\n toolUseCount: number\n model: string\n}\n\n/**\n * Options for listing transcripts\n */\nexport interface ListTranscriptsOptions {\n /** Filter by status */\n status?: AgentStatus | AgentStatus[]\n /** Filter by agent type */\n agentType?: string\n /** Limit number of results */\n limit?: number\n /** Sort order (newest first by default) */\n sortOrder?: 'asc' | 'desc'\n /** Only include transcripts after this timestamp */\n since?: number\n}\n\n// In-memory cache for active transcripts (faster access during execution)\nconst activeTranscripts = new Map<string, AgentTranscript>()\n\n/**\n * Get the transcripts directory path\n */\nfunction getTranscriptsDirectory(): string {\n const configDir =\n process.env.MINTO_CONFIG_DIR ??\n process.env.CLAUDE_CONFIG_DIR ??\n join(homedir(), '.minto')\n\n return join(configDir, 'transcripts')\n}\n\n/**\n * Ensure the transcripts directory exists\n */\nfunction ensureTranscriptsDirectory(): void {\n const dir = getTranscriptsDirectory()\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true })\n }\n}\n\n/**\n * Get the file path for a transcript\n */\nfunction getTranscriptPath(agentId: string): string {\n return join(getTranscriptsDirectory(), `${agentId}.json`)\n}\n\n/**\n * Create a new agent transcript\n */\nexport function createAgentTranscript(\n options: Omit<\n AgentTranscript,\n 'startTime' | 'status' | 'messages' | 'toolUseCount' | 'tokenUsage'\n > & {\n messages?: MessageType[]\n toolUseCount?: number\n tokenUsage?: Partial<TokenUsage>\n },\n): AgentTranscript {\n const transcript: AgentTranscript = {\n ...options,\n startTime: Date.now(),\n status: 'running',\n messages: options.messages || [],\n toolUseCount: options.toolUseCount || 0,\n tokenUsage: {\n inputTokens: options.tokenUsage?.inputTokens || 0,\n outputTokens: options.tokenUsage?.outputTokens || 0,\n cacheReadTokens: options.tokenUsage?.cacheReadTokens,\n cacheCreationTokens: options.tokenUsage?.cacheCreationTokens,\n },\n }\n\n // Store in memory cache\n activeTranscripts.set(options.agentId, transcript)\n\n // Persist to disk\n saveAgentTranscript(transcript, false) // Don't emit here, we'll emit 'created' specifically\n\n debugLogger.info('AGENT_TRANSCRIPT_CREATED', {\n agentId: options.agentId,\n agentType: options.agentType,\n description: options.description,\n })\n\n // Emit creation event\n transcriptEmitter.emitUpdate({\n type: 'created',\n agentId: options.agentId,\n toolUseId: options.toolUseId,\n transcript,\n })\n\n return transcript\n}\n\n/**\n * Save an agent transcript to disk\n * @param transcript The transcript to save\n * @param emitEvent Whether to emit an update event (default: true)\n */\nexport function saveAgentTranscript(\n transcript: AgentTranscript,\n emitEvent: boolean = true,\n): void {\n ensureTranscriptsDirectory()\n\n const filePath = getTranscriptPath(transcript.agentId)\n\n try {\n writeFileSync(filePath, JSON.stringify(transcript, null, 2), 'utf-8')\n\n // Update memory cache\n activeTranscripts.set(transcript.agentId, transcript)\n\n debugLogger.trace('AGENT_TRANSCRIPT_SAVED', {\n agentId: transcript.agentId,\n status: transcript.status,\n messageCount: transcript.messages.length,\n })\n\n // Emit update event if requested\n if (emitEvent) {\n transcriptEmitter.emitUpdate({\n type: 'updated',\n agentId: transcript.agentId,\n toolUseId: transcript.toolUseId,\n transcript,\n })\n }\n } catch (error) {\n debugLogger.error('AGENT_TRANSCRIPT_SAVE_FAILED', {\n agentId: transcript.agentId,\n error: error instanceof Error ? error.message : String(error),\n })\n throw error\n }\n}\n\n/**\n * Get an agent transcript by ID\n */\nexport function getAgentTranscript(agentId: string): AgentTranscript | null {\n // Check memory cache first\n const cached = activeTranscripts.get(agentId)\n if (cached) {\n return cached\n }\n\n // Try to load from disk\n const filePath = getTranscriptPath(agentId)\n\n if (!existsSync(filePath)) {\n return null\n }\n\n try {\n const content = readFileSync(filePath, 'utf-8')\n const transcript = JSON.parse(content) as AgentTranscript\n\n // Cache for future access\n activeTranscripts.set(agentId, transcript)\n\n return transcript\n } catch (error) {\n debugLogger.error('AGENT_TRANSCRIPT_READ_FAILED', {\n agentId,\n error: error instanceof Error ? error.message : String(error),\n })\n return null\n }\n}\n\n/**\n * Update an existing agent transcript\n * @param agentId The agent ID to update\n * @param updates The fields to update\n * @param emitEvent Whether to emit an update event (default: true)\n */\nexport function updateAgentTranscript(\n agentId: string,\n updates: Partial<Omit<AgentTranscript, 'agentId' | 'startTime'>>,\n emitEvent: boolean = true,\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n debugLogger.warn('AGENT_TRANSCRIPT_UPDATE_NOT_FOUND', { agentId })\n return null\n }\n\n const updated: AgentTranscript = {\n ...transcript,\n ...updates,\n // Merge token usage if provided\n tokenUsage: updates.tokenUsage\n ? { ...transcript.tokenUsage, ...updates.tokenUsage }\n : transcript.tokenUsage,\n }\n\n saveAgentTranscript(updated, emitEvent)\n return updated\n}\n\n/**\n * Append a message to an agent transcript\n */\nexport function appendMessageToTranscript(\n agentId: string,\n message: MessageType,\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n debugLogger.warn('AGENT_TRANSCRIPT_APPEND_NOT_FOUND', { agentId })\n return null\n }\n\n // \u2705 CRITICAL FIX: Create new array reference to trigger React re-renders\n // Instead of mutating transcript.messages.push(message), we create a new array\n const updatedMessages = [...transcript.messages, message]\n\n // Count tool uses\n let newToolUseCount = transcript.toolUseCount\n if (message.type === 'assistant') {\n const content = message.message.content\n if (Array.isArray(content)) {\n const toolUses = content.filter(block => block.type === 'tool_use')\n newToolUseCount += toolUses.length\n }\n }\n\n // Create updated transcript with new references\n const updated: AgentTranscript = {\n ...transcript,\n messages: updatedMessages,\n toolUseCount: newToolUseCount,\n }\n\n // Update memory cache with new reference\n activeTranscripts.set(agentId, updated)\n\n saveAgentTranscript(updated, false) // Don't emit generic update\n\n // Emit specific message_appended event\n transcriptEmitter.emitUpdate({\n type: 'message_appended',\n agentId,\n toolUseId: transcript.toolUseId,\n transcript: updated,\n })\n\n return updated\n}\n\n/**\n * Mark an agent transcript as completed\n */\nexport function completeAgentTranscript(\n agentId: string,\n tokenUsage?: Partial<TokenUsage>,\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n return null\n }\n\n const previousStatus = transcript.status\n\n // Merge partial token usage with existing\n const mergedTokenUsage: TokenUsage = tokenUsage\n ? { ...transcript.tokenUsage, ...tokenUsage }\n : transcript.tokenUsage\n\n const updated = updateAgentTranscript(\n agentId,\n {\n status: 'completed',\n endTime: Date.now(),\n tokenUsage: mergedTokenUsage,\n },\n false,\n ) // Don't emit generic update\n\n if (updated) {\n // Emit specific completed event\n transcriptEmitter.emitUpdate({\n type: 'completed',\n agentId,\n toolUseId: updated.toolUseId,\n transcript: updated,\n previousStatus,\n })\n\n // Also emit status_changed for subscribers\n transcriptEmitter.emitUpdate({\n type: 'status_changed',\n agentId,\n toolUseId: updated.toolUseId,\n transcript: updated,\n previousStatus,\n })\n }\n\n return updated\n}\n\n/**\n * Mark an agent transcript as failed\n */\nexport function failAgentTranscript(\n agentId: string,\n error?: string,\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n return null\n }\n\n const previousStatus = transcript.status\n\n const updated = updateAgentTranscript(\n agentId,\n {\n status: 'failed',\n endTime: Date.now(),\n metadata: error ? { error } : undefined,\n },\n false,\n ) // Don't emit generic update\n\n if (updated) {\n // Emit specific failed event\n transcriptEmitter.emitUpdate({\n type: 'failed',\n agentId,\n toolUseId: updated.toolUseId,\n transcript: updated,\n previousStatus,\n })\n\n // Also emit status_changed for subscribers\n transcriptEmitter.emitUpdate({\n type: 'status_changed',\n agentId,\n toolUseId: updated.toolUseId,\n transcript: updated,\n previousStatus,\n })\n }\n\n return updated\n}\n\n/**\n * Mark an agent transcript as interrupted\n */\nexport function interruptAgentTranscript(\n agentId: string,\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n return null\n }\n\n const previousStatus = transcript.status\n\n const updated = updateAgentTranscript(\n agentId,\n {\n status: 'interrupted',\n endTime: Date.now(),\n },\n false,\n ) // Don't emit generic update\n\n if (updated) {\n // Emit specific interrupted event\n transcriptEmitter.emitUpdate({\n type: 'interrupted',\n agentId,\n toolUseId: updated.toolUseId,\n transcript: updated,\n previousStatus,\n })\n\n // Also emit status_changed for subscribers\n transcriptEmitter.emitUpdate({\n type: 'status_changed',\n agentId,\n toolUseId: updated.toolUseId,\n transcript: updated,\n previousStatus,\n })\n }\n\n return updated\n}\n\n/**\n * Add a retry event to an agent transcript\n */\nexport function addRetryEventToTranscript(\n agentId: string,\n event: {\n attempt: number\n maxRetries: number\n errorMessage: string\n delayMs: number\n },\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n return null\n }\n\n const retryEvents = transcript.retryEvents || []\n retryEvents.push({\n ...event,\n timestamp: Date.now(),\n })\n\n return updateAgentTranscript(agentId, {\n retryEvents,\n })\n}\n\n/**\n * List agent transcripts with optional filtering\n */\nexport function listAgentTranscripts(\n options: ListTranscriptsOptions = {},\n): AgentTranscriptSummary[] {\n ensureTranscriptsDirectory()\n\n const dir = getTranscriptsDirectory()\n const files = readdirSync(dir).filter(f => f.endsWith('.json'))\n\n const summaries: AgentTranscriptSummary[] = []\n\n for (const file of files) {\n try {\n const content = readFileSync(join(dir, file), 'utf-8')\n const transcript = JSON.parse(content) as AgentTranscript\n\n // Apply filters\n if (options.status) {\n const statuses = Array.isArray(options.status)\n ? options.status\n : [options.status]\n if (!statuses.includes(transcript.status)) {\n continue\n }\n }\n\n if (options.agentType && transcript.agentType !== options.agentType) {\n continue\n }\n\n if (options.since && transcript.startTime < options.since) {\n continue\n }\n\n summaries.push({\n agentId: transcript.agentId,\n agentType: transcript.agentType,\n description: transcript.description,\n status: transcript.status,\n startTime: transcript.startTime,\n endTime: transcript.endTime,\n messageCount: transcript.messages.length,\n toolUseCount: transcript.toolUseCount,\n model: transcript.model,\n })\n } catch {\n // Skip invalid files\n continue\n }\n }\n\n // Sort by start time\n summaries.sort((a, b) => {\n const order = options.sortOrder === 'asc' ? 1 : -1\n return order * (b.startTime - a.startTime)\n })\n\n // Apply limit\n if (options.limit && options.limit > 0) {\n return summaries.slice(0, options.limit)\n }\n\n return summaries\n}\n\n/**\n * Delete an agent transcript\n */\nexport function deleteAgentTranscript(agentId: string): boolean {\n const filePath = getTranscriptPath(agentId)\n\n // Remove from cache\n activeTranscripts.delete(agentId)\n\n if (!existsSync(filePath)) {\n return false\n }\n\n try {\n unlinkSync(filePath)\n debugLogger.info('AGENT_TRANSCRIPT_DELETED', { agentId })\n return true\n } catch (error) {\n debugLogger.error('AGENT_TRANSCRIPT_DELETE_FAILED', {\n agentId,\n error: error instanceof Error ? error.message : String(error),\n })\n return false\n }\n}\n\n/**\n * Clean up old transcripts\n * @param maxAge Maximum age in milliseconds (default: 7 days)\n */\nexport function cleanupOldTranscripts(\n maxAge: number = 7 * 24 * 60 * 60 * 1000,\n): number {\n const cutoff = Date.now() - maxAge\n const transcripts = listAgentTranscripts()\n let deleted = 0\n\n for (const summary of transcripts) {\n // Only delete completed or failed transcripts that are old\n if (\n (summary.status === 'completed' || summary.status === 'failed') &&\n summary.startTime < cutoff\n ) {\n if (deleteAgentTranscript(summary.agentId)) {\n deleted++\n }\n }\n }\n\n debugLogger.info('AGENT_TRANSCRIPTS_CLEANUP', { deleted, maxAge })\n return deleted\n}\n\n/**\n * Get a transcript that can be resumed (running or interrupted)\n */\nexport function getResumableTranscript(\n agentId: string,\n): AgentTranscript | null {\n const transcript = getAgentTranscript(agentId)\n if (!transcript) {\n return null\n }\n\n // Only running or interrupted transcripts can be resumed\n if (transcript.status !== 'running' && transcript.status !== 'interrupted') {\n debugLogger.warn('AGENT_TRANSCRIPT_NOT_RESUMABLE', {\n agentId,\n status: transcript.status,\n })\n return null\n }\n\n return transcript\n}\n\n/**\n * Check if a transcript exists and is resumable\n */\nexport function canResumeTranscript(agentId: string): boolean {\n return getResumableTranscript(agentId) !== null\n}\n\n/**\n * Get all running agent transcripts\n */\nexport function getRunningTranscripts(): AgentTranscriptSummary[] {\n return listAgentTranscripts({ status: 'running' })\n}\n\n/**\n * Clear the in-memory transcript cache\n */\nexport function clearTranscriptCache(): void {\n activeTranscripts.clear()\n}\n\n// In-memory map from toolUseId to agentId for fast lookup\nconst toolUseToAgentMap = new Map<string, string>()\n\n// Reverse map from agentId to toolUseId for TaskOutput association\nconst agentToToolUseMap = new Map<string, string>()\n\n/**\n * Register a toolUseId -> agentId mapping (bidirectional)\n */\nexport function registerToolUseAgent(toolUseId: string, agentId: string): void {\n toolUseToAgentMap.set(toolUseId, agentId)\n agentToToolUseMap.set(agentId, toolUseId)\n}\n\n/**\n * Get agent ID from a tool use ID\n */\nexport function getAgentIdByToolUseId(toolUseId: string): string | null {\n return toolUseToAgentMap.get(toolUseId) || null\n}\n\n/**\n * Get tool use ID from an agent ID (reverse lookup for TaskOutput association)\n * Enhanced: Falls back to disk lookup when memory mapping is empty (for history restore)\n */\nexport function getToolUseIdByAgentId(agentId: string): string | null {\n // 1. Try memory mapping first (fast path)\n const cached = agentToToolUseMap.get(agentId)\n if (cached) {\n return cached\n }\n\n // 2. Try lazy bulk restore (first miss triggers full scan)\n if (!diskMappingsRestored) {\n restoreToolUseAgentMappings()\n\n // Check again after restore\n const afterRestore = agentToToolUseMap.get(agentId)\n if (afterRestore) {\n return afterRestore\n }\n }\n\n // 3. Last resort: load specific transcript by agentId and extract toolUseId\n const transcript = getAgentTranscript(agentId)\n if (transcript?.toolUseId) {\n // Cache for future lookups\n registerToolUseAgent(transcript.toolUseId, agentId)\n return transcript.toolUseId\n }\n\n return null\n}\n\n// Flag to track if we've already scanned disk for history restore\nlet diskMappingsRestored = false\n\n/**\n * Scan transcript files to find one matching the toolUseId\n * Used for history restore when memory mappings are empty\n */\nfunction findTranscriptByToolUseIdOnDisk(\n toolUseId: string,\n): AgentTranscript | null {\n const dir = getTranscriptsDirectory()\n if (!existsSync(dir)) return null\n\n try {\n const files = readdirSync(dir).filter(f => f.endsWith('.json'))\n\n for (const file of files) {\n try {\n const content = readFileSync(join(dir, file), 'utf-8')\n const transcript = JSON.parse(content) as AgentTranscript\n\n if (transcript.toolUseId === toolUseId) {\n // Cache to memory for subsequent access\n activeTranscripts.set(transcript.agentId, transcript)\n registerToolUseAgent(toolUseId, transcript.agentId)\n\n debugLogger.trace('TRANSCRIPT_FOUND_ON_DISK', {\n toolUseId,\n agentId: transcript.agentId,\n })\n\n return transcript\n }\n } catch {\n // Skip invalid files\n continue\n }\n }\n } catch (error) {\n debugLogger.error('TRANSCRIPT_DISK_SCAN_FAILED', {\n error: error instanceof Error ? error.message : String(error),\n })\n }\n\n return null\n}\n\n/**\n * Restore all toolUseId -> agentId mappings from disk\n * Called lazily on first lookup miss to support history restore\n */\nexport function restoreToolUseAgentMappings(): void {\n if (diskMappingsRestored) return\n\n const dir = getTranscriptsDirectory()\n if (!existsSync(dir)) {\n diskMappingsRestored = true\n return\n }\n\n try {\n const files = readdirSync(dir).filter(f => f.endsWith('.json'))\n let restoredCount = 0\n\n for (const file of files) {\n try {\n const content = readFileSync(join(dir, file), 'utf-8')\n const transcript = JSON.parse(content) as AgentTranscript\n\n if (transcript.toolUseId && transcript.agentId) {\n registerToolUseAgent(transcript.toolUseId, transcript.agentId)\n activeTranscripts.set(transcript.agentId, transcript)\n restoredCount++\n }\n } catch {\n // Skip invalid files\n continue\n }\n }\n\n debugLogger.info('TRANSCRIPT_MAPPINGS_RESTORED', {\n count: restoredCount,\n totalFiles: files.length,\n })\n } catch (error) {\n debugLogger.error('TRANSCRIPT_RESTORE_FAILED', {\n error: error instanceof Error ? error.message : String(error),\n })\n }\n\n diskMappingsRestored = true\n}\n\n/**\n * Get agent transcript by tool use ID\n * Enhanced: Falls back to disk scan when memory mapping is empty (for history restore)\n */\nexport function getAgentTranscriptByToolUseId(\n toolUseId: string,\n): AgentTranscript | null {\n // 1. Try memory mapping first (fast path for live sessions)\n const agentId = getAgentIdByToolUseId(toolUseId)\n if (agentId) {\n return getAgentTranscript(agentId)\n }\n\n // 2. Try lazy bulk restore (first miss triggers full scan)\n if (!diskMappingsRestored) {\n restoreToolUseAgentMappings()\n\n // Check again after restore\n const agentIdAfterRestore = getAgentIdByToolUseId(toolUseId)\n if (agentIdAfterRestore) {\n return getAgentTranscript(agentIdAfterRestore)\n }\n }\n\n // 3. Last resort: direct disk lookup (handles late-written transcripts)\n return findTranscriptByToolUseIdOnDisk(toolUseId)\n}\n\n/**\n * Clear the toolUseId -> agentId mapping cache\n */\nexport function clearToolUseAgentMap(): void {\n toolUseToAgentMap.clear()\n agentToToolUseMap.clear()\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["state"]
7
+ }
@@ -0,0 +1,213 @@
1
+ class AnimationManager {
2
+ static instance;
3
+ subscriptions = /* @__PURE__ */ new Map();
4
+ rafId = null;
5
+ lastFrameTime = 0;
6
+ isRunning = false;
7
+ // Use 30fps as base rate to reduce terminal render load
8
+ // Terminal rendering is expensive, unlike browser canvas rendering
9
+ FRAME_INTERVAL = 33;
10
+ // ~30fps (was 16ms/60fps)
11
+ constructor() {
12
+ }
13
+ static getInstance() {
14
+ if (!AnimationManager.instance) {
15
+ AnimationManager.instance = new AnimationManager();
16
+ }
17
+ return AnimationManager.instance;
18
+ }
19
+ /**
20
+ * Subscribe to animation updates
21
+ * @param id Unique identifier for this subscription
22
+ * @param callback Function to call on each animation frame
23
+ * @param interval Minimum interval between callback invocations (ms)
24
+ * @returns Unsubscribe function
25
+ */
26
+ subscribe(id, callback, interval) {
27
+ this.subscriptions.set(id, {
28
+ id,
29
+ callback,
30
+ interval,
31
+ lastUpdate: 0
32
+ });
33
+ if (!this.isRunning) {
34
+ this.start();
35
+ }
36
+ return () => this.unsubscribe(id);
37
+ }
38
+ /**
39
+ * Unsubscribe from animation updates
40
+ */
41
+ unsubscribe(id) {
42
+ this.subscriptions.delete(id);
43
+ if (this.subscriptions.size === 0) {
44
+ this.stop();
45
+ }
46
+ }
47
+ /**
48
+ * Start the animation loop
49
+ */
50
+ start() {
51
+ if (this.isRunning) return;
52
+ this.isRunning = true;
53
+ this.lastFrameTime = Date.now();
54
+ this.tick();
55
+ }
56
+ /**
57
+ * Stop the animation loop
58
+ */
59
+ stop() {
60
+ if (!this.isRunning) return;
61
+ this.isRunning = false;
62
+ if (this.rafId !== null) {
63
+ clearTimeout(this.rafId);
64
+ this.rafId = null;
65
+ }
66
+ }
67
+ /**
68
+ * Main animation tick
69
+ */
70
+ tick = () => {
71
+ if (!this.isRunning) return;
72
+ const now = Date.now();
73
+ const deltaTime = now - this.lastFrameTime;
74
+ this.lastFrameTime = now;
75
+ const callbacksToRun = [];
76
+ for (const sub of this.subscriptions.values()) {
77
+ const timeSinceLastUpdate = now - sub.lastUpdate;
78
+ if (timeSinceLastUpdate >= sub.interval) {
79
+ sub.lastUpdate = now;
80
+ callbacksToRun.push(sub.callback);
81
+ }
82
+ }
83
+ if (callbacksToRun.length > 0) {
84
+ for (const callback of callbacksToRun) {
85
+ try {
86
+ callback(deltaTime);
87
+ } catch (error) {
88
+ }
89
+ }
90
+ }
91
+ this.rafId = setTimeout(this.tick, this.FRAME_INTERVAL);
92
+ };
93
+ /**
94
+ * Get the number of active subscriptions
95
+ */
96
+ getSubscriptionCount() {
97
+ return this.subscriptions.size;
98
+ }
99
+ /**
100
+ * Check if animation loop is running
101
+ */
102
+ isAnimating() {
103
+ return this.isRunning;
104
+ }
105
+ }
106
+ const animationManager = AnimationManager.getInstance();
107
+ import { useState, useEffect, useRef } from "react";
108
+ function useUnifiedAnimation({
109
+ enabled,
110
+ startTime,
111
+ spinnerFrameCount,
112
+ componentId
113
+ }) {
114
+ const [animState, setAnimState] = useState(() => ({
115
+ spinnerFrame: 0,
116
+ dotVisible: true,
117
+ elapsedTime: Math.floor((Date.now() - startTime) / 1e3)
118
+ }));
119
+ const startTimeRef = useRef(startTime);
120
+ startTimeRef.current = startTime;
121
+ const timingRef = useRef({
122
+ lastSpinnerUpdate: 0,
123
+ lastDotUpdate: 0,
124
+ lastElapsedUpdate: 0
125
+ });
126
+ const SPINNER_INTERVAL = 200;
127
+ const DOT_INTERVAL = 800;
128
+ const ELAPSED_INTERVAL = 1e3;
129
+ useEffect(() => {
130
+ if (!enabled) {
131
+ return;
132
+ }
133
+ const uniqueId = `${componentId}-unified-animation`;
134
+ const unsubscribe = animationManager.subscribe(
135
+ uniqueId,
136
+ () => {
137
+ const now = Date.now();
138
+ const timing = timingRef.current;
139
+ setAnimState((prevState) => {
140
+ let needsUpdate = false;
141
+ let newSpinnerFrame = prevState.spinnerFrame;
142
+ let newDotVisible = prevState.dotVisible;
143
+ let newElapsedTime = prevState.elapsedTime;
144
+ if (now - timing.lastSpinnerUpdate >= SPINNER_INTERVAL) {
145
+ timing.lastSpinnerUpdate = now;
146
+ newSpinnerFrame = (newSpinnerFrame + 1) % spinnerFrameCount;
147
+ needsUpdate = true;
148
+ }
149
+ if (now - timing.lastDotUpdate >= DOT_INTERVAL) {
150
+ timing.lastDotUpdate = now;
151
+ newDotVisible = !newDotVisible;
152
+ needsUpdate = true;
153
+ }
154
+ if (now - timing.lastElapsedUpdate >= ELAPSED_INTERVAL) {
155
+ timing.lastElapsedUpdate = now;
156
+ newElapsedTime = Math.floor((now - startTimeRef.current) / 1e3);
157
+ needsUpdate = true;
158
+ }
159
+ if (!needsUpdate) {
160
+ return prevState;
161
+ }
162
+ return {
163
+ spinnerFrame: newSpinnerFrame,
164
+ dotVisible: newDotVisible,
165
+ elapsedTime: newElapsedTime
166
+ };
167
+ });
168
+ },
169
+ // Use fastest needed interval for checking
170
+ Math.min(SPINNER_INTERVAL, DOT_INTERVAL, ELAPSED_INTERVAL)
171
+ );
172
+ return unsubscribe;
173
+ }, [enabled, spinnerFrameCount, componentId]);
174
+ return animState;
175
+ }
176
+ function throttle(func, limit) {
177
+ let inThrottle = false;
178
+ let lastArgs = null;
179
+ return function(...args) {
180
+ if (!inThrottle) {
181
+ func.apply(this, args);
182
+ inThrottle = true;
183
+ setTimeout(() => {
184
+ inThrottle = false;
185
+ if (lastArgs) {
186
+ func.apply(this, lastArgs);
187
+ lastArgs = null;
188
+ }
189
+ }, limit);
190
+ } else {
191
+ lastArgs = args;
192
+ }
193
+ };
194
+ }
195
+ function debounce(func, wait) {
196
+ let timeoutId = null;
197
+ return function(...args) {
198
+ if (timeoutId !== null) {
199
+ clearTimeout(timeoutId);
200
+ }
201
+ timeoutId = setTimeout(() => {
202
+ func.apply(this, args);
203
+ timeoutId = null;
204
+ }, wait);
205
+ };
206
+ }
207
+ export {
208
+ animationManager,
209
+ debounce,
210
+ throttle,
211
+ useUnifiedAnimation
212
+ };
213
+ //# sourceMappingURL=animationManager.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/animationManager.ts"],
4
+ "sourcesContent": ["/**\n * Unified Animation Manager\n *\n * Consolidates multiple animation timers into a single requestAnimationFrame loop\n * to prevent render thrashing and screen flickering.\n *\n * Benefits:\n * - Single animation frame callback instead of multiple setInterval timers\n * - Batched state updates reduce React re-renders\n * - Configurable frame rates for different animation types\n * - Automatic cleanup on unmount\n */\n\ntype AnimationCallback = (deltaTime: number) => void\n\ninterface AnimationSubscription {\n id: string\n callback: AnimationCallback\n interval: number // Minimum interval between updates in ms\n lastUpdate: number\n}\n\nclass AnimationManager {\n private static instance: AnimationManager\n private subscriptions: Map<string, AnimationSubscription> = new Map()\n private rafId: ReturnType<typeof setTimeout> | null = null\n private lastFrameTime: number = 0\n private isRunning: boolean = false\n\n // Use 30fps as base rate to reduce terminal render load\n // Terminal rendering is expensive, unlike browser canvas rendering\n private readonly FRAME_INTERVAL = 33 // ~30fps (was 16ms/60fps)\n\n private constructor() {}\n\n static getInstance(): AnimationManager {\n if (!AnimationManager.instance) {\n AnimationManager.instance = new AnimationManager()\n }\n return AnimationManager.instance\n }\n\n /**\n * Subscribe to animation updates\n * @param id Unique identifier for this subscription\n * @param callback Function to call on each animation frame\n * @param interval Minimum interval between callback invocations (ms)\n * @returns Unsubscribe function\n */\n subscribe(\n id: string,\n callback: AnimationCallback,\n interval: number,\n ): () => void {\n this.subscriptions.set(id, {\n id,\n callback,\n interval,\n lastUpdate: 0,\n })\n\n if (!this.isRunning) {\n this.start()\n }\n\n return () => this.unsubscribe(id)\n }\n\n /**\n * Unsubscribe from animation updates\n */\n unsubscribe(id: string): void {\n this.subscriptions.delete(id)\n\n if (this.subscriptions.size === 0) {\n this.stop()\n }\n }\n\n /**\n * Start the animation loop\n */\n private start(): void {\n if (this.isRunning) return\n\n this.isRunning = true\n this.lastFrameTime = Date.now()\n this.tick()\n }\n\n /**\n * Stop the animation loop\n */\n private stop(): void {\n if (!this.isRunning) return\n\n this.isRunning = false\n if (this.rafId !== null) {\n clearTimeout(this.rafId)\n this.rafId = null\n }\n }\n\n /**\n * Main animation tick\n */\n private tick = (): void => {\n if (!this.isRunning) return\n\n const now = Date.now()\n const deltaTime = now - this.lastFrameTime\n this.lastFrameTime = now\n\n // Batch all callbacks that need to run this frame\n const callbacksToRun: AnimationCallback[] = []\n\n for (const sub of this.subscriptions.values()) {\n const timeSinceLastUpdate = now - sub.lastUpdate\n\n if (timeSinceLastUpdate >= sub.interval) {\n sub.lastUpdate = now\n callbacksToRun.push(sub.callback)\n }\n }\n\n // Execute all callbacks in a single batch\n // This allows React to batch the state updates\n if (callbacksToRun.length > 0) {\n for (const callback of callbacksToRun) {\n try {\n callback(deltaTime)\n } catch (error) {\n // Silently ignore callback errors to prevent animation loop from breaking\n }\n }\n }\n\n // Schedule next frame using setTimeout (works in Node.js/Bun)\n this.rafId = setTimeout(this.tick, this.FRAME_INTERVAL)\n }\n\n /**\n * Get the number of active subscriptions\n */\n getSubscriptionCount(): number {\n return this.subscriptions.size\n }\n\n /**\n * Check if animation loop is running\n */\n isAnimating(): boolean {\n return this.isRunning\n }\n}\n\n// Export singleton instance\nexport const animationManager = AnimationManager.getInstance()\n\n/**\n * React hook for unified animation updates\n *\n * Usage:\n * ```typescript\n * const { spinnerFrame, dotVisible, elapsedTime } = useUnifiedAnimation({\n * enabled: hasRunningTasks,\n * startTime: Date.now(),\n * spinnerFrameCount: 12,\n * })\n * ```\n */\nexport interface UnifiedAnimationState {\n spinnerFrame: number\n dotVisible: boolean\n elapsedTime: number\n}\n\nexport interface UseUnifiedAnimationOptions {\n enabled: boolean\n startTime: number\n spinnerFrameCount: number\n componentId: string\n}\n\nimport { useState, useEffect, useRef, useCallback } from 'react'\n\nexport function useUnifiedAnimation({\n enabled,\n startTime,\n spinnerFrameCount,\n componentId,\n}: UseUnifiedAnimationOptions): UnifiedAnimationState {\n // Use a single state object to batch updates\n const [animState, setAnimState] = useState<UnifiedAnimationState>(() => ({\n spinnerFrame: 0,\n dotVisible: true,\n elapsedTime: Math.floor((Date.now() - startTime) / 1000),\n }))\n\n // Store startTime in ref to avoid closure issues\n const startTimeRef = useRef(startTime)\n startTimeRef.current = startTime\n\n // Track timing internally to reduce state updates\n const timingRef = useRef({\n lastSpinnerUpdate: 0,\n lastDotUpdate: 0,\n lastElapsedUpdate: 0,\n })\n\n // Animation intervals - optimized for less flicker\n // Slower spinner reduces terminal re-render frequency\n const SPINNER_INTERVAL = 200 // 200ms (was 120ms) - reduces flicker by ~40%\n const DOT_INTERVAL = 800 // 800ms (was 600ms) - less visual noise\n const ELAPSED_INTERVAL = 1000 // 1s - unchanged\n\n useEffect(() => {\n if (!enabled) {\n return\n }\n\n const uniqueId = `${componentId}-unified-animation`\n\n const unsubscribe = animationManager.subscribe(\n uniqueId,\n () => {\n const now = Date.now()\n const timing = timingRef.current\n\n // Use functional state update to avoid closure issues\n setAnimState(prevState => {\n let needsUpdate = false\n let newSpinnerFrame = prevState.spinnerFrame\n let newDotVisible = prevState.dotVisible\n let newElapsedTime = prevState.elapsedTime\n\n // Check spinner update\n if (now - timing.lastSpinnerUpdate >= SPINNER_INTERVAL) {\n timing.lastSpinnerUpdate = now\n newSpinnerFrame = (newSpinnerFrame + 1) % spinnerFrameCount\n needsUpdate = true\n }\n\n // Check dot update\n if (now - timing.lastDotUpdate >= DOT_INTERVAL) {\n timing.lastDotUpdate = now\n newDotVisible = !newDotVisible\n needsUpdate = true\n }\n\n // Check elapsed time update\n if (now - timing.lastElapsedUpdate >= ELAPSED_INTERVAL) {\n timing.lastElapsedUpdate = now\n newElapsedTime = Math.floor((now - startTimeRef.current) / 1000)\n needsUpdate = true\n }\n\n // Return same object if no update needed (prevents unnecessary re-render)\n if (!needsUpdate) {\n return prevState\n }\n\n return {\n spinnerFrame: newSpinnerFrame,\n dotVisible: newDotVisible,\n elapsedTime: newElapsedTime,\n }\n })\n },\n // Use fastest needed interval for checking\n Math.min(SPINNER_INTERVAL, DOT_INTERVAL, ELAPSED_INTERVAL),\n )\n\n return unsubscribe\n }, [enabled, spinnerFrameCount, componentId])\n\n return animState\n}\n\n/**\n * Throttle utility for event handlers\n */\nexport function throttle<T extends (...args: any[]) => any>(\n func: T,\n limit: number,\n): (...args: Parameters<T>) => void {\n let inThrottle = false\n let lastArgs: Parameters<T> | null = null\n\n return function (this: any, ...args: Parameters<T>) {\n if (!inThrottle) {\n func.apply(this, args)\n inThrottle = true\n setTimeout(() => {\n inThrottle = false\n if (lastArgs) {\n func.apply(this, lastArgs)\n lastArgs = null\n }\n }, limit)\n } else {\n lastArgs = args\n }\n }\n}\n\n/**\n * Debounce utility for event handlers\n */\nexport function debounce<T extends (...args: any[]) => any>(\n func: T,\n wait: number,\n): (...args: Parameters<T>) => void {\n let timeoutId: ReturnType<typeof setTimeout> | null = null\n\n return function (this: any, ...args: Parameters<T>) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId)\n }\n timeoutId = setTimeout(() => {\n func.apply(this, args)\n timeoutId = null\n }, wait)\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }
@@ -0,0 +1,110 @@
1
+ import { EventEmitter } from "events";
2
+ const ANIMATION_INTERVALS = {
3
+ SPINNER: 120,
4
+ // Spinner frame rotation
5
+ BLINK: 600,
6
+ // Blinking dot visibility toggle
7
+ ELAPSED_TIME: 1e3
8
+ // Elapsed time counter
9
+ };
10
+ class AnimationSyncManager extends EventEmitter {
11
+ static instance = null;
12
+ timers = /* @__PURE__ */ new Map();
13
+ frame = {
14
+ spinnerFrame: 0,
15
+ blinkVisible: true,
16
+ elapsedSeconds: 0
17
+ };
18
+ subscriberCount = 0;
19
+ startTime = 0;
20
+ constructor() {
21
+ super();
22
+ this.setMaxListeners(100);
23
+ }
24
+ static getInstance() {
25
+ if (!AnimationSyncManager.instance) {
26
+ AnimationSyncManager.instance = new AnimationSyncManager();
27
+ }
28
+ return AnimationSyncManager.instance;
29
+ }
30
+ /**
31
+ * Subscribe to animation frame updates
32
+ * @param callback Called on every animation frame update
33
+ * @returns Unsubscribe function
34
+ */
35
+ subscribe(callback) {
36
+ this.subscriberCount++;
37
+ if (this.subscriberCount === 1) {
38
+ this.startTimers();
39
+ }
40
+ callback({ ...this.frame });
41
+ this.on("frame", callback);
42
+ return () => {
43
+ this.off("frame", callback);
44
+ this.subscriberCount--;
45
+ if (this.subscriberCount === 0) {
46
+ this.stopTimers();
47
+ }
48
+ };
49
+ }
50
+ /**
51
+ * Get current animation frame state
52
+ */
53
+ getFrame() {
54
+ return { ...this.frame };
55
+ }
56
+ /**
57
+ * Reset elapsed time counter (e.g., when starting a new operation)
58
+ */
59
+ resetElapsedTime() {
60
+ this.frame.elapsedSeconds = 0;
61
+ this.startTime = Date.now();
62
+ }
63
+ startTimers() {
64
+ this.startTime = Date.now();
65
+ const spinnerTimer = setInterval(() => {
66
+ this.frame.spinnerFrame = (this.frame.spinnerFrame + 1) % 12;
67
+ this.emit("frame", { ...this.frame });
68
+ }, ANIMATION_INTERVALS.SPINNER);
69
+ this.timers.set("SPINNER", spinnerTimer);
70
+ const blinkTimer = setInterval(() => {
71
+ this.frame.blinkVisible = !this.frame.blinkVisible;
72
+ }, ANIMATION_INTERVALS.BLINK);
73
+ this.timers.set("BLINK", blinkTimer);
74
+ const elapsedTimer = setInterval(() => {
75
+ this.frame.elapsedSeconds = Math.floor(
76
+ (Date.now() - this.startTime) / 1e3
77
+ );
78
+ }, ANIMATION_INTERVALS.ELAPSED_TIME);
79
+ this.timers.set("ELAPSED_TIME", elapsedTimer);
80
+ }
81
+ stopTimers() {
82
+ for (const timer of this.timers.values()) {
83
+ clearInterval(timer);
84
+ }
85
+ this.timers.clear();
86
+ this.frame = {
87
+ spinnerFrame: 0,
88
+ blinkVisible: true,
89
+ elapsedSeconds: 0
90
+ };
91
+ }
92
+ /**
93
+ * Cleanup - should be called on application shutdown
94
+ */
95
+ destroy() {
96
+ this.stopTimers();
97
+ this.removeAllListeners();
98
+ AnimationSyncManager.instance = null;
99
+ }
100
+ }
101
+ const animationSync = AnimationSyncManager.getInstance();
102
+ function useAnimationSync() {
103
+ return animationSync.getFrame();
104
+ }
105
+ export {
106
+ ANIMATION_INTERVALS,
107
+ animationSync,
108
+ useAnimationSync
109
+ };
110
+ //# sourceMappingURL=animationSync.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/animationSync.ts"],
4
+ "sourcesContent": ["/**\n * Animation Synchronization Manager\n *\n * Provides synchronized animation timing across all UI components.\n * This ensures consistent visual experience by keeping all animations in sync.\n *\n * Animation intervals (Claude Code spec compliant):\n * - Spinner: 120ms\n * - Blinking Dot: 600ms\n * - Elapsed Time: 1000ms\n */\n\nimport { EventEmitter } from 'events'\n\n// Animation intervals per Claude Code specification\nexport const ANIMATION_INTERVALS = {\n SPINNER: 120, // Spinner frame rotation\n BLINK: 600, // Blinking dot visibility toggle\n ELAPSED_TIME: 1000, // Elapsed time counter\n} as const\n\ntype AnimationType = keyof typeof ANIMATION_INTERVALS\n\ninterface AnimationFrame {\n spinnerFrame: number\n blinkVisible: boolean\n elapsedSeconds: number\n}\n\nclass AnimationSyncManager extends EventEmitter {\n private static instance: AnimationSyncManager | null = null\n private timers: Map<AnimationType, NodeJS.Timeout> = new Map()\n private frame: AnimationFrame = {\n spinnerFrame: 0,\n blinkVisible: true,\n elapsedSeconds: 0,\n }\n private subscriberCount: number = 0\n private startTime: number = 0\n\n private constructor() {\n super()\n this.setMaxListeners(100) // Allow many components to subscribe\n }\n\n static getInstance(): AnimationSyncManager {\n if (!AnimationSyncManager.instance) {\n AnimationSyncManager.instance = new AnimationSyncManager()\n }\n return AnimationSyncManager.instance\n }\n\n /**\n * Subscribe to animation frame updates\n * @param callback Called on every animation frame update\n * @returns Unsubscribe function\n */\n subscribe(callback: (frame: AnimationFrame) => void): () => void {\n this.subscriberCount++\n\n // Start timers if this is the first subscriber\n if (this.subscriberCount === 1) {\n this.startTimers()\n }\n\n // Immediately call with current frame\n callback({ ...this.frame })\n\n // Subscribe to updates\n this.on('frame', callback)\n\n return () => {\n this.off('frame', callback)\n this.subscriberCount--\n\n // Stop timers if no more subscribers\n if (this.subscriberCount === 0) {\n this.stopTimers()\n }\n }\n }\n\n /**\n * Get current animation frame state\n */\n getFrame(): AnimationFrame {\n return { ...this.frame }\n }\n\n /**\n * Reset elapsed time counter (e.g., when starting a new operation)\n */\n resetElapsedTime(): void {\n this.frame.elapsedSeconds = 0\n this.startTime = Date.now()\n }\n\n private startTimers(): void {\n this.startTime = Date.now()\n\n // Spinner timer (120ms)\n const spinnerTimer = setInterval(() => {\n this.frame.spinnerFrame = (this.frame.spinnerFrame + 1) % 12 // 12 frames for smooth animation\n this.emit('frame', { ...this.frame })\n }, ANIMATION_INTERVALS.SPINNER)\n this.timers.set('SPINNER', spinnerTimer)\n\n // Blink timer (600ms)\n const blinkTimer = setInterval(() => {\n this.frame.blinkVisible = !this.frame.blinkVisible\n // Note: We don't emit here to avoid double emissions\n // The blink state is included in the spinner frame update\n }, ANIMATION_INTERVALS.BLINK)\n this.timers.set('BLINK', blinkTimer)\n\n // Elapsed time timer (1000ms)\n const elapsedTimer = setInterval(() => {\n this.frame.elapsedSeconds = Math.floor(\n (Date.now() - this.startTime) / 1000,\n )\n // Note: We don't emit here to avoid double emissions\n }, ANIMATION_INTERVALS.ELAPSED_TIME)\n this.timers.set('ELAPSED_TIME', elapsedTimer)\n }\n\n private stopTimers(): void {\n for (const timer of this.timers.values()) {\n clearInterval(timer)\n }\n this.timers.clear()\n\n // Reset frame state\n this.frame = {\n spinnerFrame: 0,\n blinkVisible: true,\n elapsedSeconds: 0,\n }\n }\n\n /**\n * Cleanup - should be called on application shutdown\n */\n destroy(): void {\n this.stopTimers()\n this.removeAllListeners()\n AnimationSyncManager.instance = null\n }\n}\n\n/**\n * Global animation sync manager instance\n */\nexport const animationSync = AnimationSyncManager.getInstance()\n\n/**\n * React hook for synchronized animations\n * Usage:\n * ```tsx\n * function MyComponent() {\n * const frame = useAnimationSync()\n * return <Text>{spinnerChars[frame.spinnerFrame % spinnerChars.length]}</Text>\n * }\n * ```\n */\nexport function useAnimationSync(): AnimationFrame {\n // This is a placeholder - actual React hook implementation\n // should be in a separate file to avoid importing React here\n return animationSync.getFrame()\n}\n"],
5
+ "mappings": "AAYA,SAAS,oBAAoB;AAGtB,MAAM,sBAAsB;AAAA,EACjC,SAAS;AAAA;AAAA,EACT,OAAO;AAAA;AAAA,EACP,cAAc;AAAA;AAChB;AAUA,MAAM,6BAA6B,aAAa;AAAA,EAC9C,OAAe,WAAwC;AAAA,EAC/C,SAA6C,oBAAI,IAAI;AAAA,EACrD,QAAwB;AAAA,IAC9B,cAAc;AAAA,IACd,cAAc;AAAA,IACd,gBAAgB;AAAA,EAClB;AAAA,EACQ,kBAA0B;AAAA,EAC1B,YAAoB;AAAA,EAEpB,cAAc;AACpB,UAAM;AACN,SAAK,gBAAgB,GAAG;AAAA,EAC1B;AAAA,EAEA,OAAO,cAAoC;AACzC,QAAI,CAAC,qBAAqB,UAAU;AAClC,2BAAqB,WAAW,IAAI,qBAAqB;AAAA,IAC3D;AACA,WAAO,qBAAqB;AAAA,EAC9B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,UAAU,UAAuD;AAC/D,SAAK;AAGL,QAAI,KAAK,oBAAoB,GAAG;AAC9B,WAAK,YAAY;AAAA,IACnB;AAGA,aAAS,EAAE,GAAG,KAAK,MAAM,CAAC;AAG1B,SAAK,GAAG,SAAS,QAAQ;AAEzB,WAAO,MAAM;AACX,WAAK,IAAI,SAAS,QAAQ;AAC1B,WAAK;AAGL,UAAI,KAAK,oBAAoB,GAAG;AAC9B,aAAK,WAAW;AAAA,MAClB;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,WAA2B;AACzB,WAAO,EAAE,GAAG,KAAK,MAAM;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,mBAAyB;AACvB,SAAK,MAAM,iBAAiB;AAC5B,SAAK,YAAY,KAAK,IAAI;AAAA,EAC5B;AAAA,EAEQ,cAAoB;AAC1B,SAAK,YAAY,KAAK,IAAI;AAG1B,UAAM,eAAe,YAAY,MAAM;AACrC,WAAK,MAAM,gBAAgB,KAAK,MAAM,eAAe,KAAK;AAC1D,WAAK,KAAK,SAAS,EAAE,GAAG,KAAK,MAAM,CAAC;AAAA,IACtC,GAAG,oBAAoB,OAAO;AAC9B,SAAK,OAAO,IAAI,WAAW,YAAY;AAGvC,UAAM,aAAa,YAAY,MAAM;AACnC,WAAK,MAAM,eAAe,CAAC,KAAK,MAAM;AAAA,IAGxC,GAAG,oBAAoB,KAAK;AAC5B,SAAK,OAAO,IAAI,SAAS,UAAU;AAGnC,UAAM,eAAe,YAAY,MAAM;AACrC,WAAK,MAAM,iBAAiB,KAAK;AAAA,SAC9B,KAAK,IAAI,IAAI,KAAK,aAAa;AAAA,MAClC;AAAA,IAEF,GAAG,oBAAoB,YAAY;AACnC,SAAK,OAAO,IAAI,gBAAgB,YAAY;AAAA,EAC9C;AAAA,EAEQ,aAAmB;AACzB,eAAW,SAAS,KAAK,OAAO,OAAO,GAAG;AACxC,oBAAc,KAAK;AAAA,IACrB;AACA,SAAK,OAAO,MAAM;AAGlB,SAAK,QAAQ;AAAA,MACX,cAAc;AAAA,MACd,cAAc;AAAA,MACd,gBAAgB;AAAA,IAClB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,UAAgB;AACd,SAAK,WAAW;AAChB,SAAK,mBAAmB;AACxB,yBAAqB,WAAW;AAAA,EAClC;AACF;AAKO,MAAM,gBAAgB,qBAAqB,YAAY;AAYvD,SAAS,mBAAmC;AAGjD,SAAO,cAAc,SAAS;AAChC;",
6
+ "names": []
7
+ }