@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,585 +0,0 @@
1
- /**
2
- * DiffViewer - Display file diffs with syntax highlighting
3
- *
4
- * Story MSSCI-12711 - DiffViewer Component (Epic 71: Codebase Awareness)
5
- *
6
- * Features:
7
- * - Side-by-side or unified diff view with toggle
8
- * - Syntax highlighting per file type
9
- * - Green additions, red deletions
10
- * - Keyboard navigation between hunks
11
- * - Toggle partial/full file view
12
- */
13
-
14
- import React, { useState, useCallback, useMemo, useRef, useEffect, KeyboardEvent } from 'react';
15
- import { Button } from '@/components/ui/button';
16
-
17
- // =============================================================================
18
- // Types
19
- // =============================================================================
20
-
21
- export type DiffLineType = 'added' | 'removed' | 'unchanged';
22
-
23
- export type ViewMode = 'unified' | 'side-by-side';
24
-
25
- export type FileViewMode = 'partial' | 'full';
26
-
27
- export interface DiffLine {
28
- type: DiffLineType;
29
- content: string;
30
- oldLineNumber?: number;
31
- newLineNumber?: number;
32
- }
33
-
34
- export interface DiffHunk {
35
- oldStart: number;
36
- oldCount: number;
37
- newStart: number;
38
- newCount: number;
39
- lines: DiffLine[];
40
- }
41
-
42
- export interface DiffData {
43
- filePath: string;
44
- oldContent: string;
45
- newContent: string;
46
- hunks: DiffHunk[];
47
- }
48
-
49
- interface DiffViewerPropsBase {
50
- viewMode?: ViewMode;
51
- fileViewMode?: FileViewMode;
52
- onViewModeChange?: (mode: ViewMode) => void;
53
- onFileViewModeChange?: (mode: FileViewMode) => void;
54
- onHunkNavigate?: (hunkIndex: number) => void;
55
- }
56
-
57
- interface DiffViewerPropsWithData extends DiffViewerPropsBase {
58
- data: DiffData;
59
- filePath?: never;
60
- oldContent?: never;
61
- newContent?: never;
62
- }
63
-
64
- interface DiffViewerPropsWithContent extends DiffViewerPropsBase {
65
- data?: never;
66
- filePath: string;
67
- oldContent: string;
68
- newContent: string;
69
- }
70
-
71
- export type DiffViewerProps = DiffViewerPropsWithData | DiffViewerPropsWithContent;
72
-
73
- /**
74
- * Generate hunks from content differences for simple add/remove cases
75
- */
76
- function generateSimpleHunks(oldContent: string, newContent: string): DiffHunk[] {
77
- const isNew = oldContent === '' && newContent !== '';
78
- const isDeleted = oldContent !== '' && newContent === '';
79
-
80
- if (isNew) {
81
- const lines = newContent.split('\n');
82
- return [{
83
- oldStart: 0,
84
- oldCount: 0,
85
- newStart: 1,
86
- newCount: lines.length,
87
- lines: lines.map((content, i) => ({
88
- type: 'added' as DiffLineType,
89
- content,
90
- newLineNumber: i + 1,
91
- })),
92
- }];
93
- }
94
-
95
- if (isDeleted) {
96
- const lines = oldContent.split('\n');
97
- return [{
98
- oldStart: 1,
99
- oldCount: lines.length,
100
- newStart: 0,
101
- newCount: 0,
102
- lines: lines.map((content, i) => ({
103
- type: 'removed' as DiffLineType,
104
- content,
105
- oldLineNumber: i + 1,
106
- })),
107
- }];
108
- }
109
-
110
- // If both have content, treat as full modification
111
- return [];
112
- }
113
-
114
- // =============================================================================
115
- // Utility Functions
116
- // =============================================================================
117
-
118
- /**
119
- * Get file extension from path
120
- */
121
- function getFileExtension(filePath: string): string {
122
- const filename = filePath.split('/').pop() || '';
123
- if (filename.startsWith('.')) {
124
- return filename.slice(1); // .gitignore -> gitignore
125
- }
126
- const parts = filename.split('.');
127
- if (parts.length < 2) return '';
128
- return parts.pop() || '';
129
- }
130
-
131
- /**
132
- * Get CSS language class for syntax highlighting
133
- */
134
- function getLanguageClass(ext: string): string {
135
- const langMap: Record<string, string> = {
136
- ts: 'lang-typescript',
137
- tsx: 'lang-typescript',
138
- js: 'lang-javascript',
139
- jsx: 'lang-javascript',
140
- css: 'lang-css',
141
- html: 'lang-html',
142
- json: 'lang-json',
143
- md: 'lang-markdown',
144
- py: 'lang-python',
145
- gitignore: 'lang-gitignore',
146
- };
147
- return langMap[ext] || 'lang-plain';
148
- }
149
-
150
- /**
151
- * Generate hidden lines between two hunks
152
- */
153
- function generateHiddenLines(
154
- oldContent: string,
155
- startLine: number,
156
- endLine: number
157
- ): DiffLine[] {
158
- const lines = oldContent.split('\n');
159
- const result: DiffLine[] = [];
160
- for (let i = startLine; i < endLine && i < lines.length; i++) {
161
- result.push({
162
- type: 'unchanged',
163
- content: lines[i],
164
- oldLineNumber: i + 1,
165
- newLineNumber: i + 1,
166
- });
167
- }
168
- return result;
169
- }
170
-
171
- // =============================================================================
172
- // DiffLine Component
173
- // =============================================================================
174
-
175
- interface DiffLineProps {
176
- line: DiffLine;
177
- }
178
-
179
- function DiffLineComponent({ line }: DiffLineProps): React.ReactElement {
180
- // Use non-breaking space for unchanged lines to preserve whitespace in HTML
181
- const prefix = line.type === 'added' ? '+' : line.type === 'removed' ? '-' : '\u00A0';
182
- const ariaLabel = `Line ${line.oldLineNumber || line.newLineNumber}, ${line.type}: ${line.content}`;
183
-
184
- return (
185
- <div
186
- data-testid={`diff-line-${line.type}`}
187
- className={`diff-line diff-line-${line.type}`}
188
- aria-label={ariaLabel}
189
- >
190
- <span
191
- data-testid="diff-line-number-old"
192
- className="diff-line-number diff-line-number-old"
193
- >
194
- {line.oldLineNumber ?? ''}
195
- </span>
196
- <span
197
- data-testid="diff-line-number-new"
198
- className="diff-line-number diff-line-number-new"
199
- >
200
- {line.newLineNumber ?? ''}
201
- </span>
202
- <span
203
- data-testid={`diff-line-prefix-${line.type}`}
204
- className="diff-line-prefix"
205
- >
206
- {prefix}
207
- </span>
208
- <span className="diff-line-content">{line.content}</span>
209
- </div>
210
- );
211
- }
212
-
213
- // =============================================================================
214
- // DiffHunk Component
215
- // =============================================================================
216
-
217
- interface DiffHunkProps {
218
- hunk: DiffHunk;
219
- index: number;
220
- isFocused: boolean;
221
- tabIndex: number;
222
- }
223
-
224
- function DiffHunkComponent({ hunk, index, isFocused, tabIndex }: DiffHunkProps): React.ReactElement {
225
- const hunkRef = useRef<HTMLDivElement>(null);
226
- const header = `@@ -${hunk.oldStart},${hunk.oldCount} +${hunk.newStart},${hunk.newCount} @@`;
227
-
228
- useEffect(() => {
229
- if (isFocused && hunkRef.current) {
230
- hunkRef.current.scrollIntoView({ behavior: 'smooth', block: 'center' });
231
- }
232
- }, [isFocused]);
233
-
234
- return (
235
- <div
236
- ref={hunkRef}
237
- data-testid="diff-hunk"
238
- className={`diff-hunk${isFocused ? ' diff-hunk-focused' : ''}`}
239
- tabIndex={tabIndex}
240
- >
241
- <div data-testid="diff-hunk-header" className="diff-hunk-header">
242
- {header}
243
- </div>
244
- {hunk.lines.map((line, lineIndex) => (
245
- <DiffLineComponent key={`${index}-${lineIndex}`} line={line} />
246
- ))}
247
- </div>
248
- );
249
- }
250
-
251
- // =============================================================================
252
- // Unified View Component
253
- // =============================================================================
254
-
255
- interface UnifiedViewProps {
256
- hunks: DiffHunk[];
257
- focusedHunk: number;
258
- fileViewMode: FileViewMode;
259
- oldContent: string;
260
- expandedSections: Set<number>;
261
- onExpandSection: (index: number) => void;
262
- }
263
-
264
- function UnifiedView({
265
- hunks,
266
- focusedHunk,
267
- fileViewMode,
268
- oldContent,
269
- expandedSections,
270
- onExpandSection,
271
- }: UnifiedViewProps): React.ReactElement {
272
- if (fileViewMode === 'full') {
273
- // Show all lines, not just hunks
274
- const allLines = oldContent.split('\n').map((content, i) => ({
275
- type: 'unchanged' as DiffLineType,
276
- content,
277
- oldLineNumber: i + 1,
278
- newLineNumber: i + 1,
279
- }));
280
-
281
- // Merge hunks with unchanged lines
282
- const mergedLines: DiffLine[] = [];
283
- let lineIndex = 0;
284
-
285
- for (const hunk of hunks) {
286
- // Add unchanged lines before this hunk
287
- while (lineIndex < hunk.oldStart - 1) {
288
- mergedLines.push(allLines[lineIndex]);
289
- lineIndex++;
290
- }
291
- // Add hunk lines
292
- for (const line of hunk.lines) {
293
- mergedLines.push(line);
294
- if (line.type !== 'added') {
295
- lineIndex++;
296
- }
297
- }
298
- }
299
- // Add remaining unchanged lines
300
- while (lineIndex < allLines.length) {
301
- mergedLines.push(allLines[lineIndex]);
302
- lineIndex++;
303
- }
304
-
305
- return (
306
- <div data-testid="diff-full-view" className="diff-full-view">
307
- {mergedLines.map((line, i) => (
308
- <DiffLineComponent key={i} line={line} />
309
- ))}
310
- </div>
311
- );
312
- }
313
-
314
- // Partial view - show only hunks with expand buttons between
315
- const elements: React.ReactElement[] = [];
316
-
317
- for (let i = 0; i < hunks.length; i++) {
318
- const hunk = hunks[i];
319
-
320
- // Add expand button between hunks
321
- if (i > 0) {
322
- const prevHunk = hunks[i - 1];
323
- const prevEnd = prevHunk.oldStart + prevHunk.oldCount - 1;
324
- const currentStart = hunk.oldStart - 1;
325
- const hiddenCount = currentStart - prevEnd;
326
-
327
- if (hiddenCount > 0) {
328
- if (expandedSections.has(i)) {
329
- // Show the expanded lines
330
- const hiddenLines = generateHiddenLines(oldContent, prevEnd, currentStart);
331
- elements.push(
332
- <div key={`expanded-${i}`} className="diff-expanded-section">
333
- {hiddenLines.map((line, j) => (
334
- <DiffLineComponent key={`exp-${i}-${j}`} line={line} />
335
- ))}
336
- </div>
337
- );
338
- } else {
339
- elements.push(
340
- <Button
341
- variant="ghost"
342
- size="sm"
343
- key={`expand-${i}`}
344
- type="button"
345
- className="diff-expand-button"
346
- aria-label="Expand hidden lines"
347
- onClick={() => onExpandSection(i)}
348
- >
349
- {hiddenCount} lines hidden ···
350
- </Button>
351
- );
352
- }
353
- }
354
- }
355
-
356
- elements.push(
357
- <DiffHunkComponent
358
- key={`hunk-${i}`}
359
- hunk={hunk}
360
- index={i}
361
- isFocused={focusedHunk === i}
362
- tabIndex={i === 0 ? 0 : -1}
363
- />
364
- );
365
- }
366
-
367
- return (
368
- <div data-testid="diff-partial-view" className="diff-partial-view">
369
- {elements}
370
- </div>
371
- );
372
- }
373
-
374
- // =============================================================================
375
- // Side-by-Side View Component
376
- // =============================================================================
377
-
378
- interface SideBySideViewProps {
379
- hunks: DiffHunk[];
380
- focusedHunk: number;
381
- }
382
-
383
- function SideBySideView({ hunks, focusedHunk }: SideBySideViewProps): React.ReactElement {
384
- // Separate lines into old (removed/unchanged) and new (added/unchanged)
385
- const oldLines: DiffLine[] = [];
386
- const newLines: DiffLine[] = [];
387
-
388
- for (const hunk of hunks) {
389
- for (const line of hunk.lines) {
390
- if (line.type === 'removed') {
391
- oldLines.push(line);
392
- } else if (line.type === 'added') {
393
- newLines.push(line);
394
- } else {
395
- oldLines.push(line);
396
- newLines.push(line);
397
- }
398
- }
399
- }
400
-
401
- return (
402
- <div data-testid="diff-side-by-side" className="diff-side-by-side">
403
- <div data-testid="diff-panel-old" className="diff-panel diff-panel-old">
404
- {oldLines.map((line, i) => (
405
- <DiffLineComponent key={`old-${i}`} line={line} />
406
- ))}
407
- </div>
408
- <div data-testid="diff-panel-new" className="diff-panel diff-panel-new">
409
- {newLines.map((line, i) => (
410
- <DiffLineComponent key={`new-${i}`} line={line} />
411
- ))}
412
- </div>
413
- </div>
414
- );
415
- }
416
-
417
- // =============================================================================
418
- // DiffViewer Component
419
- // =============================================================================
420
-
421
- export function DiffViewer(props: DiffViewerProps): React.ReactElement {
422
- const {
423
- viewMode = 'unified',
424
- fileViewMode = 'partial',
425
- onViewModeChange,
426
- onFileViewModeChange,
427
- onHunkNavigate,
428
- } = props;
429
-
430
- // Support both data prop and direct props
431
- const data: DiffData = 'data' in props && props.data
432
- ? props.data
433
- : {
434
- filePath: (props as DiffViewerPropsWithContent).filePath,
435
- oldContent: (props as DiffViewerPropsWithContent).oldContent,
436
- newContent: (props as DiffViewerPropsWithContent).newContent,
437
- hunks: generateSimpleHunks(
438
- (props as DiffViewerPropsWithContent).oldContent,
439
- (props as DiffViewerPropsWithContent).newContent
440
- ),
441
- };
442
-
443
- const [focusedHunk, setFocusedHunk] = useState(0);
444
- const [expandedSections, setExpandedSections] = useState<Set<number>>(new Set());
445
- const viewerRef = useRef<HTMLDivElement>(null);
446
-
447
- const ext = getFileExtension(data.filePath);
448
- const langClass = getLanguageClass(ext);
449
-
450
- const isNewFile = data.oldContent === '' && data.newContent !== '';
451
- const isDeletedFile = data.oldContent !== '' && data.newContent === '';
452
- const hasNoChanges = data.hunks.length === 0;
453
-
454
- const handleKeyDown = useCallback(
455
- (e: KeyboardEvent<HTMLDivElement>) => {
456
- const nextKeys = ['j', 'ArrowDown', 'n'];
457
- const prevKeys = ['k', 'ArrowUp'];
458
-
459
- if (nextKeys.includes(e.key) || (e.key === 'n' && !e.shiftKey)) {
460
- const nextIndex = focusedHunk + 1;
461
- if (nextIndex < data.hunks.length) {
462
- setFocusedHunk(nextIndex);
463
- onHunkNavigate?.(nextIndex);
464
- }
465
- } else if (prevKeys.includes(e.key) || (e.key === 'N' && e.shiftKey)) {
466
- const prevIndex = focusedHunk - 1;
467
- if (prevIndex >= 0) {
468
- setFocusedHunk(prevIndex);
469
- onHunkNavigate?.(prevIndex);
470
- }
471
- }
472
- },
473
- [focusedHunk, data.hunks.length, onHunkNavigate]
474
- );
475
-
476
- const handleViewModeToggle = useCallback(() => {
477
- const newMode = viewMode === 'unified' ? 'side-by-side' : 'unified';
478
- onViewModeChange?.(newMode);
479
- }, [viewMode, onViewModeChange]);
480
-
481
- const handleFileViewModeToggle = useCallback(() => {
482
- const newMode = fileViewMode === 'partial' ? 'full' : 'partial';
483
- onFileViewModeChange?.(newMode);
484
- }, [fileViewMode, onFileViewModeChange]);
485
-
486
- const handleExpandSection = useCallback((index: number) => {
487
- setExpandedSections((prev) => {
488
- const next = new Set(prev);
489
- next.add(index);
490
- return next;
491
- });
492
- }, []);
493
-
494
- // Empty/special states
495
- if (hasNoChanges && !isNewFile && !isDeletedFile) {
496
- return (
497
- <div
498
- ref={viewerRef}
499
- role="region"
500
- aria-label="Diff viewer"
501
- data-testid="diff-viewer"
502
- className={`diff-viewer ${langClass}`}
503
- >
504
- <div className="diff-header">
505
- <span className="diff-file-path">{data.filePath}</span>
506
- </div>
507
- <div className="diff-empty">No changes</div>
508
- </div>
509
- );
510
- }
511
-
512
- return (
513
- <div
514
- ref={viewerRef}
515
- role="region"
516
- aria-label="Diff viewer"
517
- data-testid="diff-viewer"
518
- className={`diff-viewer ${langClass}`}
519
- onKeyDown={handleKeyDown}
520
- tabIndex={-1}
521
- >
522
- {/* Header */}
523
- <div className="diff-header">
524
- <span className="diff-file-path">{data.filePath}</span>
525
- <div className="diff-controls">
526
- <Button
527
- variant="outline"
528
- size="sm"
529
- type="button"
530
- aria-label="Toggle view mode"
531
- onClick={handleViewModeToggle}
532
- className="diff-view-toggle"
533
- >
534
- {viewMode === 'unified' ? 'Side-by-side' : 'Unified'}
535
- </Button>
536
- <Button
537
- variant="outline"
538
- size="sm"
539
- type="button"
540
- aria-label={fileViewMode === 'partial' ? 'Show full file' : 'Show changes only'}
541
- onClick={handleFileViewModeToggle}
542
- className="diff-file-view-toggle"
543
- >
544
- {fileViewMode === 'partial' ? 'Show full file' : 'Show changes only'}
545
- </Button>
546
- </div>
547
- </div>
548
-
549
- {/* New/Deleted file indicators */}
550
- {isNewFile && <div data-testid="diff-new-file-indicator" className="diff-new-file">New file</div>}
551
- {isDeletedFile && <div data-testid="diff-deleted-file-indicator" className="diff-deleted-file">File deleted</div>}
552
-
553
- {/* Hunk position indicator (for accessibility) */}
554
- {data.hunks.length > 1 && (
555
- <div className="diff-hunk-position" aria-live="polite">
556
- Hunk {focusedHunk + 1} of {data.hunks.length}
557
- </div>
558
- )}
559
-
560
- {/* Content */}
561
- {viewMode === 'unified' ? (
562
- <UnifiedView
563
- hunks={data.hunks}
564
- focusedHunk={focusedHunk}
565
- fileViewMode={fileViewMode}
566
- oldContent={data.oldContent}
567
- expandedSections={expandedSections}
568
- onExpandSection={handleExpandSection}
569
- />
570
- ) : (
571
- <SideBySideView hunks={data.hunks} focusedHunk={focusedHunk} />
572
- )}
573
-
574
- {/* Unified view marker */}
575
- {viewMode === 'unified' && fileViewMode === 'partial' && (
576
- <div data-testid="diff-unified" style={{ display: 'none' }} />
577
- )}
578
- {viewMode === 'unified' && fileViewMode === 'full' && (
579
- <div data-testid="diff-unified" style={{ display: 'none' }} />
580
- )}
581
- </div>
582
- );
583
- }
584
-
585
- export default DiffViewer;