@pennyfarthing/cyclist 10.4.0 → 11.0.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 (384) hide show
  1. package/dist/api/agent-load.d.ts +1 -2
  2. package/dist/api/agent-load.d.ts.map +1 -1
  3. package/dist/api/agent-load.js +2 -123
  4. package/dist/api/agent-load.js.map +1 -1
  5. package/dist/api/audit-log.d.ts +1 -17
  6. package/dist/api/audit-log.d.ts.map +1 -1
  7. package/dist/api/audit-log.js +2 -162
  8. package/dist/api/audit-log.js.map +1 -1
  9. package/dist/api/background-tasks.d.ts +1 -26
  10. package/dist/api/background-tasks.d.ts.map +1 -1
  11. package/dist/api/background-tasks.js +2 -55
  12. package/dist/api/background-tasks.js.map +1 -1
  13. package/dist/api/bell.d.ts +1 -18
  14. package/dist/api/bell.d.ts.map +1 -1
  15. package/dist/api/bell.js +2 -33
  16. package/dist/api/bell.js.map +1 -1
  17. package/dist/api/code-markers.d.ts +1 -8
  18. package/dist/api/code-markers.d.ts.map +1 -1
  19. package/dist/api/code-markers.js +2 -61
  20. package/dist/api/code-markers.js.map +1 -1
  21. package/dist/api/complexity.d.ts +1 -2
  22. package/dist/api/complexity.d.ts.map +1 -1
  23. package/dist/api/complexity.js +2 -46
  24. package/dist/api/complexity.js.map +1 -1
  25. package/dist/api/context.d.ts +1 -37
  26. package/dist/api/context.d.ts.map +1 -1
  27. package/dist/api/context.js +2 -143
  28. package/dist/api/context.js.map +1 -1
  29. package/dist/api/dead-code.d.ts +1 -2
  30. package/dist/api/dead-code.d.ts.map +1 -1
  31. package/dist/api/dead-code.js +2 -69
  32. package/dist/api/dead-code.js.map +1 -1
  33. package/dist/api/dependencies.d.ts +1 -2
  34. package/dist/api/dependencies.d.ts.map +1 -1
  35. package/dist/api/dependencies.js +2 -42
  36. package/dist/api/dependencies.js.map +1 -1
  37. package/dist/api/evaluation.d.ts +1 -19
  38. package/dist/api/evaluation.d.ts.map +1 -1
  39. package/dist/api/evaluation.js +2 -127
  40. package/dist/api/evaluation.js.map +1 -1
  41. package/dist/api/file-browser.d.ts +1 -8
  42. package/dist/api/file-browser.d.ts.map +1 -1
  43. package/dist/api/file-browser.js +2 -114
  44. package/dist/api/file-browser.js.map +1 -1
  45. package/dist/api/git.d.ts +1 -46
  46. package/dist/api/git.d.ts.map +1 -1
  47. package/dist/api/git.js +2 -354
  48. package/dist/api/git.js.map +1 -1
  49. package/dist/api/health-score.d.ts +1 -2
  50. package/dist/api/health-score.d.ts.map +1 -1
  51. package/dist/api/health-score.js +2 -46
  52. package/dist/api/health-score.js.map +1 -1
  53. package/dist/api/hook-request.d.ts +1 -40
  54. package/dist/api/hook-request.d.ts.map +1 -1
  55. package/dist/api/hook-request.js +2 -277
  56. package/dist/api/hook-request.js.map +1 -1
  57. package/dist/api/hotspots.d.ts +1 -2
  58. package/dist/api/hotspots.d.ts.map +1 -1
  59. package/dist/api/hotspots.js +2 -61
  60. package/dist/api/hotspots.js.map +1 -1
  61. package/dist/api/identity.d.ts +1 -16
  62. package/dist/api/identity.d.ts.map +1 -1
  63. package/dist/api/identity.js +2 -78
  64. package/dist/api/identity.js.map +1 -1
  65. package/dist/api/index.d.ts +1 -34
  66. package/dist/api/index.d.ts.map +1 -1
  67. package/dist/api/index.js +2 -44
  68. package/dist/api/index.js.map +1 -1
  69. package/dist/api/mode.d.ts +1 -22
  70. package/dist/api/mode.d.ts.map +1 -1
  71. package/dist/api/mode.js +2 -37
  72. package/dist/api/mode.js.map +1 -1
  73. package/dist/api/otlp.d.ts +1 -2
  74. package/dist/api/otlp.d.ts.map +1 -1
  75. package/dist/api/otlp.js +2 -46
  76. package/dist/api/otlp.js.map +1 -1
  77. package/dist/api/permissions.d.ts +1 -15
  78. package/dist/api/permissions.d.ts.map +1 -1
  79. package/dist/api/permissions.js +2 -66
  80. package/dist/api/permissions.js.map +1 -1
  81. package/dist/api/persona.d.ts +1 -8
  82. package/dist/api/persona.d.ts.map +1 -1
  83. package/dist/api/persona.js +2 -67
  84. package/dist/api/persona.js.map +1 -1
  85. package/dist/api/portrait.d.ts +1 -5
  86. package/dist/api/portrait.d.ts.map +1 -1
  87. package/dist/api/portrait.js +2 -27
  88. package/dist/api/portrait.js.map +1 -1
  89. package/dist/api/settings.d.ts +1 -53
  90. package/dist/api/settings.d.ts.map +1 -1
  91. package/dist/api/settings.js +2 -464
  92. package/dist/api/settings.js.map +1 -1
  93. package/dist/api/spans.d.ts +1 -16
  94. package/dist/api/spans.d.ts.map +1 -1
  95. package/dist/api/spans.js +2 -244
  96. package/dist/api/spans.js.map +1 -1
  97. package/dist/api/stats.d.ts +1 -12
  98. package/dist/api/stats.d.ts.map +1 -1
  99. package/dist/api/stats.js +2 -84
  100. package/dist/api/stats.js.map +1 -1
  101. package/dist/api/story.d.ts +1 -2
  102. package/dist/api/story.d.ts.map +1 -1
  103. package/dist/api/story.js +2 -14
  104. package/dist/api/story.js.map +1 -1
  105. package/dist/api/telemetry.d.ts +1 -18
  106. package/dist/api/telemetry.d.ts.map +1 -1
  107. package/dist/api/telemetry.js +2 -164
  108. package/dist/api/telemetry.js.map +1 -1
  109. package/dist/api/theme-agents.d.ts +1 -60
  110. package/dist/api/theme-agents.d.ts.map +1 -1
  111. package/dist/api/theme-agents.js +2 -213
  112. package/dist/api/theme-agents.js.map +1 -1
  113. package/dist/api/todos.d.ts +1 -32
  114. package/dist/api/todos.d.ts.map +1 -1
  115. package/dist/api/todos.js +2 -43
  116. package/dist/api/todos.js.map +1 -1
  117. package/dist/api/token-stats.d.ts +1 -7
  118. package/dist/api/token-stats.d.ts.map +1 -1
  119. package/dist/api/token-stats.js +2 -35
  120. package/dist/api/token-stats.js.map +1 -1
  121. package/dist/api/welcome.d.ts +1 -21
  122. package/dist/api/welcome.d.ts.map +1 -1
  123. package/dist/api/welcome.js +2 -34
  124. package/dist/api/welcome.js.map +1 -1
  125. package/dist/bikerack.js +2 -2
  126. package/dist/bikerack.js.map +1 -1
  127. package/dist/env.d.ts +6 -0
  128. package/dist/env.d.ts.map +1 -0
  129. package/dist/env.js +10 -0
  130. package/dist/env.js.map +1 -0
  131. package/dist/focus.d.ts +53 -0
  132. package/dist/focus.d.ts.map +1 -0
  133. package/dist/focus.js +122 -0
  134. package/dist/focus.js.map +1 -0
  135. package/dist/git-cache.d.ts +1 -0
  136. package/dist/git-cache.d.ts.map +1 -1
  137. package/dist/git-cache.js +3 -1
  138. package/dist/git-cache.js.map +1 -1
  139. package/dist/menu-builder.d.ts.map +1 -1
  140. package/dist/menu-builder.js +0 -1
  141. package/dist/menu-builder.js.map +1 -1
  142. package/dist/prime.d.ts +3 -3
  143. package/dist/prime.d.ts.map +1 -1
  144. package/dist/prime.js +38 -14
  145. package/dist/prime.js.map +1 -1
  146. package/dist/public/css/react.css +1 -1
  147. package/dist/public/js/react/react.js +53 -61
  148. package/dist/server.d.ts +18 -85
  149. package/dist/server.d.ts.map +1 -1
  150. package/dist/server.js +105 -405
  151. package/dist/server.js.map +1 -1
  152. package/dist/sprint-data.d.ts +1 -1
  153. package/dist/sprint-data.d.ts.map +1 -1
  154. package/dist/sprint-data.js +2 -2
  155. package/dist/sprint-data.js.map +1 -1
  156. package/dist/theme-metadata.d.ts +3 -3
  157. package/dist/theme-metadata.d.ts.map +1 -1
  158. package/dist/theme-metadata.js +4 -4
  159. package/dist/theme-metadata.js.map +1 -1
  160. package/dist/websocket.d.ts +2 -0
  161. package/dist/websocket.d.ts.map +1 -1
  162. package/dist/websocket.js +53 -75
  163. package/dist/websocket.js.map +1 -1
  164. package/package.json +2 -6
  165. package/portraits/hogans-heroes/large/burkhalter-35312.png +0 -0
  166. package/portraits/hogans-heroes/large/carter-34352.png +0 -0
  167. package/portraits/hogans-heroes/large/hochstetter-45314.png +0 -0
  168. package/portraits/hogans-heroes/large/hogan-44541.png +0 -0
  169. package/portraits/hogans-heroes/large/kinch-35241.png +0 -0
  170. package/portraits/hogans-heroes/large/klink-23434.png +0 -0
  171. package/portraits/hogans-heroes/large/lebeau-45443.png +0 -0
  172. package/portraits/hogans-heroes/large/marya-53543.png +0 -0
  173. package/portraits/hogans-heroes/large/newkirk-54432.png +0 -0
  174. package/portraits/hogans-heroes/large/schultz-42453.png +0 -0
  175. package/portraits/hogans-heroes/large/underground-55131.png +0 -0
  176. package/portraits/hogans-heroes/medium/burkhalter-35312.png +0 -0
  177. package/portraits/hogans-heroes/medium/carter-34352.png +0 -0
  178. package/portraits/hogans-heroes/medium/hochstetter-45314.png +0 -0
  179. package/portraits/hogans-heroes/medium/hogan-44541.png +0 -0
  180. package/portraits/hogans-heroes/medium/kinch-35241.png +0 -0
  181. package/portraits/hogans-heroes/medium/klink-23434.png +0 -0
  182. package/portraits/hogans-heroes/medium/lebeau-45443.png +0 -0
  183. package/portraits/hogans-heroes/medium/marya-53543.png +0 -0
  184. package/portraits/hogans-heroes/medium/newkirk-54432.png +0 -0
  185. package/portraits/hogans-heroes/medium/schultz-42453.png +0 -0
  186. package/portraits/hogans-heroes/medium/underground-55131.png +0 -0
  187. package/portraits/monty-python/large/announcer-44441.png +0 -0
  188. package/portraits/monty-python/large/arguer-35412.png +0 -0
  189. package/portraits/monty-python/large/bicycle-repair-man-35241.png +0 -0
  190. package/portraits/monty-python/large/colonel-35423.png +0 -0
  191. package/portraits/monty-python/large/counsellor-45341.png +0 -0
  192. package/portraits/monty-python/large/gumbys-23524.png +0 -0
  193. package/portraits/monty-python/large/nudge-43533.png +0 -0
  194. package/portraits/monty-python/large/praline-45413.png +0 -0
  195. package/portraits/monty-python/large/silly-walks-55322.png +0 -0
  196. package/portraits/monty-python/large/wensleydale-54451.png +0 -0
  197. package/portraits/monty-python/large/xim-nez-43534.png +0 -0
  198. package/portraits/monty-python/medium/announcer-44441.png +0 -0
  199. package/portraits/monty-python/medium/arguer-35412.png +0 -0
  200. package/portraits/monty-python/medium/bicycle-repair-man-35241.png +0 -0
  201. package/portraits/monty-python/medium/colonel-35423.png +0 -0
  202. package/portraits/monty-python/medium/counsellor-45341.png +0 -0
  203. package/portraits/monty-python/medium/gumbys-23524.png +0 -0
  204. package/portraits/monty-python/medium/nudge-43533.png +0 -0
  205. package/portraits/monty-python/medium/praline-45413.png +0 -0
  206. package/portraits/monty-python/medium/silly-walks-55322.png +0 -0
  207. package/portraits/monty-python/medium/wensleydale-54451.png +0 -0
  208. package/portraits/monty-python/medium/xim-nez-43534.png +0 -0
  209. package/portraits/stephen-king/large/andy-55231.png +0 -0
  210. package/portraits/stephen-king/large/christine-25112.png +0 -0
  211. package/portraits/stephen-king/large/danny-53243.png +0 -0
  212. package/portraits/stephen-king/large/flagg-55311.png +0 -0
  213. package/portraits/stephen-king/large/gaunt-54421.png +0 -0
  214. package/portraits/stephen-king/large/jack-44224.png +0 -0
  215. package/portraits/stephen-king/large/johnny-44353.png +0 -0
  216. package/portraits/stephen-king/large/margaret-15415.png +0 -0
  217. package/portraits/stephen-king/large/paul-45233.png +0 -0
  218. package/portraits/stephen-king/large/pennywise-54411.png +0 -0
  219. package/portraits/stephen-king/large/roland-35121.png +0 -0
  220. package/portraits/stephen-king/medium/andy-55231.png +0 -0
  221. package/portraits/stephen-king/medium/christine-25112.png +0 -0
  222. package/portraits/stephen-king/medium/danny-53243.png +0 -0
  223. package/portraits/stephen-king/medium/flagg-55311.png +0 -0
  224. package/portraits/stephen-king/medium/gaunt-54421.png +0 -0
  225. package/portraits/stephen-king/medium/jack-44224.png +0 -0
  226. package/portraits/stephen-king/medium/johnny-44353.png +0 -0
  227. package/portraits/stephen-king/medium/margaret-15415.png +0 -0
  228. package/portraits/stephen-king/medium/paul-45233.png +0 -0
  229. package/portraits/stephen-king/medium/pennywise-54411.png +0 -0
  230. package/portraits/stephen-king/medium/roland-35121.png +0 -0
  231. package/portraits/star-trek-tng/large/beverly-44352.png +0 -0
  232. package/portraits/star-trek-tng/large/data-55241.png +0 -0
  233. package/portraits/star-trek-tng/large/deanna-43353.png +0 -0
  234. package/portraits/star-trek-tng/large/geordi-54342.png +0 -0
  235. package/portraits/star-trek-tng/large/jean-luc-45342.png +0 -0
  236. package/portraits/star-trek-tng/large/kathryn-45332.png +0 -0
  237. package/portraits/star-trek-tng/large/miles-35342.png +0 -0
  238. package/portraits/star-trek-tng/large/q-53521.png +0 -0
  239. package/portraits/star-trek-tng/large/spock-45231.png +0 -0
  240. package/portraits/star-trek-tng/large/troi-44352.png +0 -0
  241. package/portraits/star-trek-tng/medium/beverly-44352.png +0 -0
  242. package/portraits/star-trek-tng/medium/data-55241.png +0 -0
  243. package/portraits/star-trek-tng/medium/deanna-43353.png +0 -0
  244. package/portraits/star-trek-tng/medium/geordi-54342.png +0 -0
  245. package/portraits/star-trek-tng/medium/jean-luc-45342.png +0 -0
  246. package/portraits/star-trek-tng/medium/kathryn-45332.png +0 -0
  247. package/portraits/star-trek-tng/medium/miles-35342.png +0 -0
  248. package/portraits/star-trek-tng/medium/q-53521.png +0 -0
  249. package/portraits/star-trek-tng/medium/spock-45231.png +0 -0
  250. package/portraits/star-trek-tng/medium/troi-44352.png +0 -0
  251. package/src/public/App.tsx +0 -340
  252. package/src/public/components/AgentLoadDialog.tsx +0 -202
  253. package/src/public/components/AgentPopup.tsx +0 -308
  254. package/src/public/components/ApprovalModal/ApprovalModal.css +0 -35
  255. package/src/public/components/ApprovalModal/index.tsx +0 -632
  256. package/src/public/components/BikeRackIndex.tsx +0 -54
  257. package/src/public/components/BikeRackWorkspace.tsx +0 -142
  258. package/src/public/components/CommandPalette.tsx +0 -555
  259. package/src/public/components/ConfirmDialog.tsx +0 -168
  260. package/src/public/components/ContextIndicator/ContextIndicator.css +0 -85
  261. package/src/public/components/ContextIndicator/index.tsx +0 -330
  262. package/src/public/components/ContextSparkline.tsx +0 -56
  263. package/src/public/components/ControlBar.tsx +0 -636
  264. package/src/public/components/DeadCodeDialog.tsx +0 -169
  265. package/src/public/components/DiffViewer.tsx +0 -585
  266. package/src/public/components/DockviewWorkspace.tsx +0 -737
  267. package/src/public/components/Editor.tsx +0 -630
  268. package/src/public/components/ErrorBoundary.tsx +0 -67
  269. package/src/public/components/FileTree.tsx +0 -379
  270. package/src/public/components/FontPicker/FontPicker.css +0 -276
  271. package/src/public/components/FontPicker/index.tsx +0 -430
  272. package/src/public/components/FullFileTree.tsx +0 -237
  273. package/src/public/components/HealthGauge.tsx +0 -181
  274. package/src/public/components/Message.tsx +0 -225
  275. package/src/public/components/MessageList.tsx +0 -98
  276. package/src/public/components/MessageView.tsx +0 -400
  277. package/src/public/components/ModeSwitch/ModeSwitch.css +0 -165
  278. package/src/public/components/ModeSwitch/index.tsx +0 -372
  279. package/src/public/components/PersonaHeader.tsx +0 -240
  280. package/src/public/components/QuickActions.tsx +0 -267
  281. package/src/public/components/SpanTimeline.tsx +0 -352
  282. package/src/public/components/StandalonePanel.tsx +0 -84
  283. package/src/public/components/StatsStrip.tsx +0 -162
  284. package/src/public/components/StreamingContent.tsx +0 -77
  285. package/src/public/components/SubagentSpan.tsx +0 -180
  286. package/src/public/components/TandemPortrait.tsx +0 -72
  287. package/src/public/components/ThemePalette/ThemePalette.css +0 -179
  288. package/src/public/components/ThemePalette/index.tsx +0 -326
  289. package/src/public/components/ToolCallBlock.tsx +0 -252
  290. package/src/public/components/ToolStack.tsx +0 -209
  291. package/src/public/components/ToolStatus.tsx +0 -57
  292. package/src/public/components/dialogs/CodeMarkersDialog.tsx +0 -169
  293. package/src/public/components/dialogs/ComplexityDialog.tsx +0 -163
  294. package/src/public/components/dialogs/DependenciesDialog.tsx +0 -120
  295. package/src/public/components/dialogs/HotspotsDialog.tsx +0 -451
  296. package/src/public/components/dialogs/ToolDialog.tsx +0 -43
  297. package/src/public/components/panel-registry.ts +0 -11
  298. package/src/public/components/panels/ACPanel.tsx +0 -93
  299. package/src/public/components/panels/AcceptanceCriteriaPanel.tsx +0 -104
  300. package/src/public/components/panels/AuditLogPanel.tsx +0 -465
  301. package/src/public/components/panels/BackgroundPanel.tsx +0 -115
  302. package/src/public/components/panels/BikeLanePanel.tsx +0 -214
  303. package/src/public/components/panels/ChangedPanel.tsx +0 -65
  304. package/src/public/components/panels/DebugPanel.tsx +0 -344
  305. package/src/public/components/panels/DiffsPanel.tsx +0 -155
  306. package/src/public/components/panels/GitPanel.tsx +0 -216
  307. package/src/public/components/panels/HotspotsPanel.tsx +0 -365
  308. package/src/public/components/panels/MessagePanel.tsx +0 -497
  309. package/src/public/components/panels/SettingsPanel.tsx +0 -453
  310. package/src/public/components/panels/SprintPanel.tsx +0 -670
  311. package/src/public/components/panels/TTYPanel.tsx +0 -299
  312. package/src/public/components/panels/TodoPanel.tsx +0 -142
  313. package/src/public/components/panels/WorkflowPanel.tsx +0 -224
  314. package/src/public/components/panels/index.ts +0 -24
  315. package/src/public/components/ui/alert-dialog.tsx +0 -139
  316. package/src/public/components/ui/badge.tsx +0 -36
  317. package/src/public/components/ui/button.tsx +0 -57
  318. package/src/public/components/ui/checkbox.tsx +0 -28
  319. package/src/public/components/ui/collapsible.tsx +0 -9
  320. package/src/public/components/ui/command.tsx +0 -151
  321. package/src/public/components/ui/dialog.tsx +0 -120
  322. package/src/public/components/ui/popover.tsx +0 -31
  323. package/src/public/components/ui/progress.tsx +0 -28
  324. package/src/public/components/ui/scroll-area.tsx +0 -46
  325. package/src/public/components/ui/select.tsx +0 -157
  326. package/src/public/components/ui/separator.tsx +0 -29
  327. package/src/public/components/ui/skeleton.tsx +0 -15
  328. package/src/public/components/ui/switch.tsx +0 -27
  329. package/src/public/components/ui/toggle-group.tsx +0 -59
  330. package/src/public/components/ui/toggle.tsx +0 -43
  331. package/src/public/components/ui/tooltip.tsx +0 -30
  332. package/src/public/contexts/ClaudeContext.tsx +0 -311
  333. package/src/public/contexts/MessageQueueContext.tsx +0 -143
  334. package/src/public/css/theme-browser.css +0 -550
  335. package/src/public/css/theme-system.css +0 -630
  336. package/src/public/hooks/index.ts +0 -49
  337. package/src/public/hooks/useAgentLoad.ts +0 -105
  338. package/src/public/hooks/useBackgroundTasks.ts +0 -131
  339. package/src/public/hooks/useClaude.ts +0 -234
  340. package/src/public/hooks/useCodeMarkers.ts +0 -101
  341. package/src/public/hooks/useColorScheme.ts +0 -42
  342. package/src/public/hooks/useCommandHistory.ts +0 -99
  343. package/src/public/hooks/useComplexity.ts +0 -80
  344. package/src/public/hooks/useDeadCode.ts +0 -99
  345. package/src/public/hooks/useDependencies.ts +0 -82
  346. package/src/public/hooks/useDiffs.ts +0 -143
  347. package/src/public/hooks/useFileBrowser.ts +0 -71
  348. package/src/public/hooks/useGitStatus.ts +0 -233
  349. package/src/public/hooks/useHealthScore.ts +0 -69
  350. package/src/public/hooks/useHotspots.ts +0 -123
  351. package/src/public/hooks/useLayoutPersistence.ts +0 -138
  352. package/src/public/hooks/useMarkdownParser.ts +0 -36
  353. package/src/public/hooks/useMarkerActions.ts +0 -234
  354. package/src/public/hooks/useMessageQueue.ts +0 -380
  355. package/src/public/hooks/useMessageStream.ts +0 -131
  356. package/src/public/hooks/usePersona.ts +0 -112
  357. package/src/public/hooks/usePlanModeExit.ts +0 -105
  358. package/src/public/hooks/useResponsiveLayout.ts +0 -173
  359. package/src/public/hooks/useSprint.ts +0 -147
  360. package/src/public/hooks/useStatsStrip.ts +0 -204
  361. package/src/public/hooks/useStory.ts +0 -135
  362. package/src/public/hooks/useSubagentHelper.ts +0 -64
  363. package/src/public/hooks/useSyntaxHighlighter.ts +0 -52
  364. package/src/public/hooks/useTabCompletion.ts +0 -124
  365. package/src/public/hooks/useTodos.ts +0 -93
  366. package/src/public/hooks/useUserAvatar.ts +0 -54
  367. package/src/public/index.tsx +0 -10
  368. package/src/public/lib/utils.ts +0 -6
  369. package/src/public/styles/dockview-theme.css +0 -459
  370. package/src/public/styles/tailwind.css +0 -4396
  371. package/src/public/types/electron.d.ts +0 -18
  372. package/src/public/types/message.ts +0 -51
  373. package/src/public/utils/avatar-service.ts +0 -73
  374. package/src/public/utils/color-presets.ts +0 -940
  375. package/src/public/utils/font-presets.ts +0 -362
  376. package/src/public/utils/formatDuration.ts +0 -14
  377. package/src/public/utils/markdown.ts +0 -249
  378. package/src/public/utils/messageFilters.ts +0 -128
  379. package/src/public/utils/slash-commands.ts +0 -353
  380. package/src/public/utils/subagent-display.ts +0 -146
  381. package/src/public/utils/syntax.ts +0 -219
  382. package/src/public/utils/toolIntentSummarizer.ts +0 -199
  383. package/src/public/utils/toolStackGrouper.ts +0 -106
  384. package/src/public/utils/toolTypeColors.ts +0 -45
@@ -1,630 +0,0 @@
1
- /**
2
- * Editor Component
3
- *
4
- * React textarea editor with command history, tab completion, and message queue.
5
- * Story MSSCI-12717 - React Migration
6
- *
7
- * Features:
8
- * - Textarea with Enter=submit, Shift+Enter=newline
9
- * - Command history (Up/Down arrows)
10
- * - Tab completion popup for /commands
11
- * - Message queue indicator
12
- * - Image paste handling
13
- * - Mode toolbar (Plan/Manual/Accept)
14
- */
15
-
16
- import React, {
17
- useState,
18
- useRef,
19
- useCallback,
20
- useEffect,
21
- KeyboardEvent,
22
- ClipboardEvent,
23
- ChangeEvent,
24
- } from 'react';
25
- import { Button } from '@/components/ui/button';
26
- import { Badge } from '@/components/ui/badge';
27
- import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
28
- import { useCommandHistory } from '../hooks/useCommandHistory';
29
- import { useTabCompletion } from '../hooks/useTabCompletion';
30
- import { useMessageQueueContext, QueuedMessage } from '../contexts/MessageQueueContext';
31
- import { ModeSwitch, Mode, useModeSync, useModeSwitchShortcuts } from './ModeSwitch';
32
- import { trackCommandUsage } from '../utils/slash-commands';
33
-
34
- // =============================================================================
35
- // Types
36
- // =============================================================================
37
-
38
- export interface PastedImage {
39
- dataUrl: string;
40
- mimeType: string;
41
- filename: string;
42
- }
43
-
44
- export interface EditorProps {
45
- onSubmit: (text: string, images: PastedImage[]) => void;
46
- isProcessing?: boolean;
47
- placeholder?: string;
48
- /** Callback to immediately inject a queued message (abort + send) */
49
- onInject?: (index: number) => Promise<boolean>;
50
- }
51
-
52
- // PermissionMode type moved to ModeSwitch component
53
-
54
- // Supported image types for paste
55
- const SUPPORTED_IMAGE_TYPES = ['image/png', 'image/jpeg', 'image/gif', 'image/webp'];
56
- const IMAGE_MAX_SIZE_BYTES = 20 * 1024 * 1024; // 20MB
57
-
58
- // =============================================================================
59
- // Completion Popup Component
60
- // =============================================================================
61
-
62
- interface CompletionPopupProps {
63
- commands: Array<{ name: string; description: string }>;
64
- selectedIndex: number;
65
- visible: boolean;
66
- onSelect: (index: number) => void;
67
- }
68
-
69
- function CompletionPopup({ commands, selectedIndex, visible, onSelect }: CompletionPopupProps) {
70
- if (!visible || commands.length === 0) return null;
71
-
72
- return (
73
- <div className="completion-popup" data-testid="completion-popup">
74
- {commands.map((cmd, index) => (
75
- <div
76
- key={cmd.name}
77
- className={`completion-item ${index === selectedIndex ? 'selected' : ''}`}
78
- onClick={() => onSelect(index)}
79
- data-testid={`completion-item-${index}`}
80
- >
81
- <span className="completion-name">{cmd.name}</span>
82
- <span className="completion-desc">{cmd.description}</span>
83
- </div>
84
- ))}
85
- </div>
86
- );
87
- }
88
-
89
- // =============================================================================
90
- // Queue Display Component (MSSCI-12275)
91
- // =============================================================================
92
-
93
- interface QueueDisplayProps {
94
- queue: QueuedMessage[];
95
- bellMode: boolean;
96
- onRemove: (index: number) => void;
97
- onClear: () => void;
98
- /** Callback to immediately inject a queued message (abort + send) */
99
- onInject?: (index: number) => Promise<boolean>;
100
- }
101
-
102
- /**
103
- * Escape HTML for safe rendering
104
- */
105
- function escapeHtml(str: string): string {
106
- if (!str) return '';
107
- return str
108
- .replace(/&/g, '&amp;')
109
- .replace(/</g, '&lt;')
110
- .replace(/>/g, '&gt;')
111
- .replace(/"/g, '&quot;');
112
- }
113
-
114
- function QueueDisplay({ queue, bellMode, onRemove, onClear, onInject }: QueueDisplayProps) {
115
- if (queue.length === 0) return null;
116
-
117
- return (
118
- <TooltipProvider delayDuration={300}>
119
- <div className="queue-display" data-testid="queue-display">
120
- <div className="queue-header">
121
- <span className="queue-count">{queue.length} queued</span>
122
- {bellMode && (
123
- <Tooltip>
124
- <TooltipTrigger asChild>
125
- <Badge variant="secondary" className="queue-mode-badge bell-mode">🔔</Badge>
126
- </TooltipTrigger>
127
- <TooltipContent>Bell mode active - messages inject via hook</TooltipContent>
128
- </Tooltip>
129
- )}
130
- <Tooltip>
131
- <TooltipTrigger asChild>
132
- <Button
133
- variant="ghost"
134
- size="sm"
135
- type="button"
136
- className="queue-clear-btn"
137
- onClick={onClear}
138
- >
139
- Clear
140
- </Button>
141
- </TooltipTrigger>
142
- <TooltipContent>Clear all queued messages</TooltipContent>
143
- </Tooltip>
144
- </div>
145
- <ul className="queue-list">
146
- {queue.map((msg, index) => {
147
- const truncated = msg.text.length > 60 ? msg.text.substring(0, 60) + '...' : msg.text;
148
- const hasImages = msg.images && msg.images.length > 0;
149
-
150
- return (
151
- <li key={index} className="queue-item" data-testid={`queue-item-${index}`}>
152
- <span className="queue-item-text">{escapeHtml(truncated)}</span>
153
- {hasImages && (
154
- <Tooltip>
155
- <TooltipTrigger asChild>
156
- <Badge variant="outline" className="queue-image-indicator">
157
- 📎{msg.images.length}
158
- </Badge>
159
- </TooltipTrigger>
160
- <TooltipContent>{`${msg.images.length} image(s) attached`}</TooltipContent>
161
- </Tooltip>
162
- )}
163
- <div className="queue-item-actions">
164
- {onInject && (
165
- <Tooltip>
166
- <TooltipTrigger asChild>
167
- <Button
168
- variant="ghost"
169
- size="icon"
170
- type="button"
171
- className="queue-item-inject"
172
- onClick={() => onInject(index)}
173
- >
174
-
175
- </Button>
176
- </TooltipTrigger>
177
- <TooltipContent>Send now (abort current and send this message)</TooltipContent>
178
- </Tooltip>
179
- )}
180
- <Tooltip>
181
- <TooltipTrigger asChild>
182
- <Button
183
- variant="ghost"
184
- size="icon"
185
- type="button"
186
- className="queue-item-remove"
187
- onClick={() => onRemove(index)}
188
- >
189
- ×
190
- </Button>
191
- </TooltipTrigger>
192
- <TooltipContent>Remove from queue</TooltipContent>
193
- </Tooltip>
194
- </div>
195
- </li>
196
- );
197
- })}
198
- </ul>
199
- </div>
200
- </TooltipProvider>
201
- );
202
- }
203
-
204
- // =============================================================================
205
- // Image Preview Component
206
- // =============================================================================
207
-
208
- interface ImagePreviewProps {
209
- images: PastedImage[];
210
- onRemove: (index: number) => void;
211
- }
212
-
213
- function ImagePreview({ images, onRemove }: ImagePreviewProps) {
214
- if (images.length === 0) return null;
215
-
216
- return (
217
- <TooltipProvider delayDuration={300}>
218
- <div className="image-preview" data-testid="image-preview">
219
- {images.map((img, index) => (
220
- <div key={index} className="image-preview-item">
221
- <img src={img.dataUrl} alt={img.filename} />
222
- <Tooltip>
223
- <TooltipTrigger asChild>
224
- <Button
225
- variant="ghost"
226
- size="icon"
227
- type="button"
228
- className="image-remove"
229
- onClick={() => onRemove(index)}
230
- >
231
- X
232
- </Button>
233
- </TooltipTrigger>
234
- <TooltipContent>Remove image</TooltipContent>
235
- </Tooltip>
236
- </div>
237
- ))}
238
- </div>
239
- </TooltipProvider>
240
- );
241
- }
242
-
243
- // =============================================================================
244
- // Editor Component
245
- // =============================================================================
246
-
247
- export function Editor({ onSubmit, isProcessing = false, placeholder, onInject }: EditorProps): React.ReactElement {
248
- const textareaRef = useRef<HTMLTextAreaElement>(null);
249
- const [value, setValue] = useState('');
250
- const [pendingImages, setPendingImages] = useState<PastedImage[]>([]);
251
-
252
- // Mode state synced with Claude backend
253
- const { mode, setMode } = useModeSync();
254
-
255
- // Register Cmd+1/2/3 shortcuts for mode switching
256
- useModeSwitchShortcuts(setMode);
257
-
258
- // Hooks
259
- const { addToHistory, navigateUp, navigateDown, resetNavigation } = useCommandHistory();
260
- const {
261
- state: completionState,
262
- showCompletion,
263
- hideCompletion,
264
- updateCompletion,
265
- navigateUp: completionUp,
266
- navigateDown: completionDown,
267
- selectCurrent,
268
- isVisible: isCompletionVisible,
269
- } = useTabCompletion();
270
- const {
271
- queue,
272
- queueCount,
273
- bellMode,
274
- queueMessage,
275
- removeFromQueue,
276
- clearQueue,
277
- setProcessing,
278
- resumeQueue,
279
- } = useMessageQueueContext();
280
-
281
- // Sync processing state
282
- useEffect(() => {
283
- setProcessing(isProcessing);
284
- }, [isProcessing, setProcessing]);
285
-
286
- // Mode initialization is now handled by useModeSync hook
287
-
288
- // Listen for suggested prompts from QuickActions
289
- useEffect(() => {
290
- const handleSuggestPrompt = (e: CustomEvent<{ prompt: string }>) => {
291
- setValue(e.detail.prompt);
292
- textareaRef.current?.focus();
293
- };
294
- window.addEventListener('cyclist:suggest-prompt', handleSuggestPrompt as EventListener);
295
- return () => {
296
- window.removeEventListener('cyclist:suggest-prompt', handleSuggestPrompt as EventListener);
297
- };
298
- }, []);
299
-
300
- // ==========================================================================
301
- // Image Handling
302
- // ==========================================================================
303
-
304
- const handleImagePaste = useCallback(async (clipboardData: DataTransfer): Promise<boolean> => {
305
- let imageFile: File | null = null;
306
-
307
- // Check clipboard items
308
- if (clipboardData.items) {
309
- for (const item of clipboardData.items) {
310
- if (item.type && SUPPORTED_IMAGE_TYPES.includes(item.type)) {
311
- imageFile = item.getAsFile();
312
- break;
313
- }
314
- }
315
- }
316
-
317
- // Fallback to files
318
- if (!imageFile && clipboardData.files?.length > 0) {
319
- for (const file of clipboardData.files) {
320
- if (file.type && SUPPORTED_IMAGE_TYPES.includes(file.type)) {
321
- imageFile = file;
322
- break;
323
- }
324
- }
325
- }
326
-
327
- if (!imageFile) return false;
328
-
329
- // Check size
330
- if (imageFile.size > IMAGE_MAX_SIZE_BYTES) {
331
- console.warn('Image too large:', imageFile.size);
332
- return false;
333
- }
334
-
335
- // Convert to data URL
336
- return new Promise((resolve) => {
337
- const reader = new FileReader();
338
- reader.onload = () => {
339
- const dataUrl = reader.result as string;
340
- const filename = imageFile!.name || `Pasted Image.${imageFile!.type.split('/')[1] || 'png'}`;
341
- setPendingImages(prev => [...prev, {
342
- dataUrl,
343
- mimeType: imageFile!.type,
344
- filename,
345
- }]);
346
- resolve(true);
347
- };
348
- reader.onerror = () => resolve(false);
349
- reader.readAsDataURL(imageFile);
350
- });
351
- }, []);
352
-
353
- const removeImage = useCallback((index: number) => {
354
- setPendingImages(prev => prev.filter((_, i) => i !== index));
355
- }, []);
356
-
357
- // ==========================================================================
358
- // Slash Prefix Detection
359
- // ==========================================================================
360
-
361
- const getSlashPrefix = useCallback((): { prefix: string; start: number; end: number } | null => {
362
- if (!textareaRef.current) return null;
363
- const cursorPos = textareaRef.current.selectionStart;
364
- const text = value;
365
-
366
- // Find word start
367
- let wordStart = cursorPos;
368
- while (wordStart > 0 && text[wordStart - 1] !== ' ' && text[wordStart - 1] !== '\n') {
369
- wordStart--;
370
- }
371
-
372
- const word = text.substring(wordStart, cursorPos);
373
- if (word.startsWith('/')) {
374
- return { prefix: word, start: wordStart, end: cursorPos };
375
- }
376
- return null;
377
- }, [value]);
378
-
379
- const replaceSlashPrefix = useCallback((commandName: string) => {
380
- const prefixInfo = getSlashPrefix();
381
- if (!prefixInfo) {
382
- setValue(prev => prev + commandName);
383
- return;
384
- }
385
-
386
- const { start, end } = prefixInfo;
387
- setValue(prev => prev.substring(0, start) + commandName + prev.substring(end));
388
-
389
- // Move cursor after command
390
- setTimeout(() => {
391
- if (textareaRef.current) {
392
- const newPos = start + commandName.length;
393
- textareaRef.current.selectionStart = textareaRef.current.selectionEnd = newPos;
394
- }
395
- }, 0);
396
- }, [getSlashPrefix]);
397
-
398
- // ==========================================================================
399
- // Submit Logic
400
- // ==========================================================================
401
-
402
- const handleSubmit = useCallback(() => {
403
- const trimmed = value.trim();
404
- if (!trimmed && pendingImages.length === 0) return;
405
-
406
- // If processing, queue the message
407
- if (isProcessing) {
408
- const queued = queueMessage({ text: trimmed, images: [...pendingImages] });
409
- if (queued) {
410
- setValue('');
411
- setPendingImages([]);
412
- textareaRef.current?.focus();
413
- }
414
- return;
415
- }
416
-
417
- // Add to history and submit
418
- addToHistory(trimmed);
419
- resetNavigation();
420
-
421
- // Track slash command usage for frequency sorting
422
- if (trimmed.startsWith('/')) {
423
- const command = trimmed.split(/\s/)[0]; // Extract "/command" from "/command args"
424
- trackCommandUsage(command);
425
- }
426
-
427
- // Resume queue if it was paused (e.g., after abort)
428
- resumeQueue();
429
-
430
- // Dispatch event to clear QuickActions (Reflector questions)
431
- window.dispatchEvent(new CustomEvent('cyclist:user-submit'));
432
-
433
- onSubmit(trimmed, pendingImages);
434
- setValue('');
435
- setPendingImages([]);
436
- textareaRef.current?.focus();
437
- }, [value, pendingImages, isProcessing, queueMessage, addToHistory, resetNavigation, resumeQueue, onSubmit]);
438
-
439
- // ==========================================================================
440
- // Event Handlers
441
- // ==========================================================================
442
-
443
- const handleKeyDown = useCallback((e: KeyboardEvent<HTMLTextAreaElement>) => {
444
- // Mode shortcuts are now handled globally by useModeSwitchShortcuts
445
-
446
- // Tab - trigger or select completion
447
- if (e.key === 'Tab' && !e.shiftKey && !e.ctrlKey && !e.altKey) {
448
- if (isCompletionVisible) {
449
- e.preventDefault();
450
- const selected = selectCurrent();
451
- if (selected) replaceSlashPrefix(selected);
452
- return;
453
- }
454
- const prefixInfo = getSlashPrefix();
455
- if (prefixInfo) {
456
- e.preventDefault();
457
- showCompletion(prefixInfo.prefix);
458
- return;
459
- }
460
- }
461
-
462
- // Escape - close completion
463
- if (e.key === 'Escape') {
464
- if (isCompletionVisible) {
465
- e.preventDefault();
466
- hideCompletion();
467
- return;
468
- }
469
- }
470
-
471
- // Enter - submit or select completion
472
- if (e.key === 'Enter' && !e.shiftKey) {
473
- if (isCompletionVisible) {
474
- e.preventDefault();
475
- const selected = selectCurrent();
476
- if (selected) replaceSlashPrefix(selected);
477
- return;
478
- }
479
- e.preventDefault();
480
- handleSubmit();
481
- return;
482
- }
483
-
484
- // Up arrow - completion or history
485
- if (e.key === 'ArrowUp' && !e.shiftKey && !e.ctrlKey && !e.altKey) {
486
- if (isCompletionVisible) {
487
- e.preventDefault();
488
- completionUp();
489
- return;
490
- }
491
- // Only navigate history if at start
492
- if (textareaRef.current?.selectionStart === 0) {
493
- const prev = navigateUp(value);
494
- if (prev !== null) {
495
- e.preventDefault();
496
- setValue(prev);
497
- return;
498
- }
499
- }
500
- }
501
-
502
- // Down arrow - completion or history
503
- if (e.key === 'ArrowDown' && !e.shiftKey && !e.ctrlKey && !e.altKey) {
504
- if (isCompletionVisible) {
505
- e.preventDefault();
506
- completionDown();
507
- return;
508
- }
509
- // Only navigate history if at end
510
- if (textareaRef.current?.selectionEnd === value.length) {
511
- const next = navigateDown();
512
- if (next !== null) {
513
- e.preventDefault();
514
- setValue(next);
515
- return;
516
- }
517
- }
518
- }
519
- }, [
520
- isCompletionVisible, selectCurrent, replaceSlashPrefix,
521
- getSlashPrefix, showCompletion, hideCompletion, handleSubmit,
522
- completionUp, completionDown, navigateUp, navigateDown, value
523
- ]);
524
-
525
- const handleChange = useCallback((e: ChangeEvent<HTMLTextAreaElement>) => {
526
- const newValue = e.target.value;
527
- setValue(newValue);
528
-
529
- // Auto-show completion when "/" is typed at start
530
- if (newValue === '/' && !isCompletionVisible) {
531
- showCompletion('/');
532
- return;
533
- }
534
-
535
- // Update completion if visible
536
- if (isCompletionVisible) {
537
- const prefixInfo = getSlashPrefix();
538
- if (prefixInfo) {
539
- updateCompletion(prefixInfo.prefix);
540
- } else {
541
- hideCompletion();
542
- }
543
- }
544
- }, [isCompletionVisible, showCompletion, getSlashPrefix, updateCompletion, hideCompletion]);
545
-
546
- const handlePaste = useCallback(async (e: ClipboardEvent<HTMLTextAreaElement>) => {
547
- const clipboardData = e.clipboardData;
548
- if (!clipboardData) return;
549
-
550
- // Check for image
551
- let hasImage = false;
552
- if (clipboardData.items) {
553
- for (const item of clipboardData.items) {
554
- if (item.type && SUPPORTED_IMAGE_TYPES.includes(item.type)) {
555
- hasImage = true;
556
- break;
557
- }
558
- }
559
- }
560
- if (!hasImage && clipboardData.files?.length > 0) {
561
- for (const file of clipboardData.files) {
562
- if (file.type && SUPPORTED_IMAGE_TYPES.includes(file.type)) {
563
- hasImage = true;
564
- break;
565
- }
566
- }
567
- }
568
-
569
- if (hasImage) {
570
- e.preventDefault();
571
- await handleImagePaste(clipboardData);
572
- }
573
- }, [handleImagePaste]);
574
-
575
- const handleCompletionSelect = useCallback((index: number) => {
576
- const cmd = completionState.commands[index];
577
- if (cmd) {
578
- replaceSlashPrefix(cmd.name);
579
- hideCompletion();
580
- }
581
- }, [completionState.commands, replaceSlashPrefix, hideCompletion]);
582
-
583
- // ==========================================================================
584
- // Render
585
- // ==========================================================================
586
-
587
- return (
588
- <div className="editor-container" data-testid="editor-container">
589
- <ModeSwitch
590
- mode={mode}
591
- onModeChange={setMode}
592
- className="editor-mode-switch"
593
- />
594
- <div className="editor-wrapper" id="editor-wrapper">
595
- <ImagePreview images={pendingImages} onRemove={removeImage} />
596
-
597
- <textarea
598
- ref={textareaRef}
599
- id="editor-textarea"
600
- className="editor-textarea"
601
- value={value}
602
- onChange={handleChange}
603
- onKeyDown={handleKeyDown}
604
- onPaste={handlePaste}
605
- placeholder={placeholder}
606
- spellCheck={false}
607
- autoFocus
608
- data-testid="editor-textarea"
609
- />
610
-
611
- <CompletionPopup
612
- commands={completionState.commands}
613
- selectedIndex={completionState.selectedIndex}
614
- visible={isCompletionVisible}
615
- onSelect={handleCompletionSelect}
616
- />
617
- </div>
618
-
619
- <QueueDisplay
620
- queue={queue}
621
- bellMode={bellMode}
622
- onRemove={removeFromQueue}
623
- onClear={clearQueue}
624
- onInject={onInject}
625
- />
626
- </div>
627
- );
628
- }
629
-
630
- export default Editor;
@@ -1,67 +0,0 @@
1
- /**
2
- * ErrorBoundary - Catches React errors to prevent entire app from crashing
3
- *
4
- * When a panel component throws (e.g., undefined.split()), this boundary
5
- * catches the error and displays a fallback UI instead of blanking the screen.
6
- */
7
-
8
- import React, { Component, ErrorInfo, ReactNode } from 'react';
9
-
10
- interface Props {
11
- children: ReactNode;
12
- fallback?: ReactNode;
13
- panelName?: string;
14
- }
15
-
16
- interface State {
17
- hasError: boolean;
18
- error?: Error;
19
- }
20
-
21
- export class ErrorBoundary extends Component<Props, State> {
22
- state: State = { hasError: false };
23
-
24
- static getDerivedStateFromError(error: Error): State {
25
- return { hasError: true, error };
26
- }
27
-
28
- componentDidCatch(error: Error, errorInfo: ErrorInfo) {
29
- const panelContext = this.props.panelName ? ` in ${this.props.panelName}` : '';
30
- console.error(`[ErrorBoundary] Caught error${panelContext}:`, error, errorInfo);
31
- }
32
-
33
- render() {
34
- if (this.state.hasError) {
35
- if (this.props.fallback) {
36
- return this.props.fallback;
37
- }
38
-
39
- return (
40
- <div className="error-boundary-fallback" style={{
41
- padding: '16px',
42
- color: 'var(--status-error, #ef4444)',
43
- backgroundColor: 'var(--bg-tertiary, #0f0f1a)',
44
- border: '1px solid var(--status-error, #ef4444)',
45
- borderRadius: '4px',
46
- margin: '8px',
47
- }}>
48
- <h4 style={{ margin: '0 0 8px 0' }}>
49
- {this.props.panelName ? `${this.props.panelName} Error` : 'Panel Error'}
50
- </h4>
51
- <pre style={{
52
- margin: 0,
53
- fontSize: '12px',
54
- whiteSpace: 'pre-wrap',
55
- wordBreak: 'break-word',
56
- }}>
57
- {this.state.error?.message}
58
- </pre>
59
- </div>
60
- );
61
- }
62
-
63
- return this.props.children;
64
- }
65
- }
66
-
67
- export default ErrorBoundary;