@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,267 +0,0 @@
1
- /**
2
- * QuickActions Component
3
- *
4
- * Renders action buttons based on CYCLIST markers detected in assistant messages.
5
- * Supports handoff buttons, yes/no questions, choices, and continue.
6
- *
7
- * For open questions with suggestions (<!-- CYCLIST:QUESTION:open:suggested text -->),
8
- * pre-fills the editor via 'cyclist:suggest-prompt' event instead of showing buttons.
9
- *
10
- * Story: MSSCI-12787 - Implement CYCLIST Marker Parsing and Action Buttons
11
- * Story: MSSCI-12771 - Accessibility Compliance (ARIA labels)
12
- */
13
-
14
- import React, { useState, useEffect, useCallback } from 'react';
15
- import { Button } from '@/components/ui/button';
16
- import { useMarkerActions } from '../hooks/useMarkerActions';
17
- import { useClaudeContext } from '../contexts/ClaudeContext';
18
- import type { MessageData } from '../types/message';
19
-
20
- interface ActionItem {
21
- label: string;
22
- command: string;
23
- }
24
-
25
- interface QuickActionsPropsWithMessage {
26
- message: MessageData;
27
- actions?: never;
28
- onAction?: (response: string) => void;
29
- }
30
-
31
- interface QuickActionsPropsWithActions {
32
- message?: never;
33
- actions: ActionItem[];
34
- onAction?: (response: string) => void;
35
- }
36
-
37
- type QuickActionsProps = QuickActionsPropsWithMessage | QuickActionsPropsWithActions;
38
-
39
- /**
40
- * Get relay mode setting from REST API
41
- */
42
- async function getRelayMode(): Promise<boolean> {
43
- try {
44
- const response = await fetch('/api/settings');
45
- if (response.ok) {
46
- const settings = await response.json();
47
- return settings?.workflow?.relay_mode ?? false;
48
- }
49
- } catch {
50
- // Ignore errors
51
- }
52
- return false;
53
- }
54
-
55
- export default function QuickActions(props: QuickActionsProps): React.ReactElement | null {
56
- // Get send function from Claude context
57
- const { send: claudeSend } = useClaudeContext();
58
-
59
- /**
60
- * Send a message to Claude via WebSocket context
61
- */
62
- const sendMessage = useCallback((text: string): void => {
63
- console.log('[QuickActions] sendMessage called:', text);
64
- claudeSend(text, []);
65
- }, [claudeSend]);
66
- const { onAction } = props;
67
- const [isDisabled, setIsDisabled] = useState(false);
68
- const [relayMode, setRelayMode] = useState(false);
69
-
70
- // Support both message-based and actions-based props
71
- const message = 'message' in props ? props.message : undefined;
72
- const directActions = 'actions' in props ? props.actions : undefined;
73
-
74
- const markerActions = useMarkerActions(message?.content);
75
-
76
- // Reset disabled state when message changes (new assistant response)
77
- useEffect(() => {
78
- setIsDisabled(false);
79
- }, [message?.timestamp]);
80
-
81
- // Check relay mode on mount
82
- useEffect(() => {
83
- getRelayMode().then(setRelayMode);
84
- }, []);
85
-
86
- // Clear QuickActions when user submits any message
87
- useEffect(() => {
88
- const handleUserSubmit = () => {
89
- setIsDisabled(true);
90
- };
91
- window.addEventListener('cyclist:user-submit', handleUserSubmit);
92
- return () => {
93
- window.removeEventListener('cyclist:user-submit', handleUserSubmit);
94
- };
95
- }, []);
96
-
97
- // Auto-execute handoff when relay mode is ON
98
- useEffect(() => {
99
- if (markerActions?.type === 'handoff' && relayMode && markerActions.value) {
100
- // Auto-execute after short delay
101
- const timer = setTimeout(() => {
102
- sendMessage(markerActions.value!);
103
- setIsDisabled(true);
104
- onAction?.(markerActions.value!);
105
- }, 100);
106
- return () => clearTimeout(timer);
107
- }
108
-
109
- if (markerActions?.type === 'invoke' && markerActions.value) {
110
- // INVOKE always auto-executes
111
- const timer = setTimeout(() => {
112
- sendMessage(markerActions.value!);
113
- setIsDisabled(true);
114
- onAction?.(markerActions.value!);
115
- }, 100);
116
- return () => clearTimeout(timer);
117
- }
118
- }, [markerActions, relayMode, onAction]);
119
-
120
- // Pre-fill editor with suggested prompt for open questions
121
- useEffect(() => {
122
- if (markerActions?.type === 'open' && markerActions.responses && markerActions.responses.length > 0) {
123
- const suggestion = markerActions.responses[0];
124
- window.dispatchEvent(new CustomEvent('cyclist:suggest-prompt', {
125
- detail: { prompt: suggestion }
126
- }));
127
- }
128
- }, [markerActions]);
129
-
130
- const handleButtonClick = useCallback((response: string) => {
131
- console.log('[QuickActions] Button clicked:', response);
132
- setIsDisabled(true);
133
- sendMessage(response);
134
- onAction?.(response);
135
- }, [onAction]);
136
-
137
- // If using direct actions prop (for accessibility testing), render those
138
- if (directActions && directActions.length > 0) {
139
- return (
140
- <div className="quick-actions">
141
- <div className="quick-actions-buttons">
142
- {directActions.map((action) => (
143
- <Button
144
- variant="secondary"
145
- size="sm"
146
- type="button"
147
- key={action.command}
148
- className="quick-action-btn"
149
- onClick={() => handleButtonClick(action.command)}
150
- disabled={isDisabled}
151
- aria-label={`${action.label}: ${action.command}`}
152
- >
153
- {action.label}
154
- </Button>
155
- ))}
156
- </div>
157
- </div>
158
- );
159
- }
160
-
161
- // No marker actions to render
162
- if (!markerActions) {
163
- return null;
164
- }
165
-
166
- // Open questions - no buttons needed, editor is pre-filled via useEffect
167
- if (markerActions.type === 'open') {
168
- return null;
169
- }
170
-
171
- // Continue marker - suppress the generic Continue button entirely
172
- // Users can always type to continue; showing a disabled button is confusing
173
- if (markerActions.type === 'continue') {
174
- return null;
175
- }
176
-
177
- // Auto-execute types don't render buttons
178
- if (markerActions.type === 'invoke') {
179
- return (
180
- <div className="quick-actions">
181
- <span className="auto-invoke-status">Invoking {markerActions.value}...</span>
182
- </div>
183
- );
184
- }
185
-
186
- // Auto-executing handoff in relay mode
187
- if (markerActions.type === 'handoff' && relayMode) {
188
- return (
189
- <div className="quick-actions">
190
- <span className="auto-invoke-status">Handing off to {markerActions.value}...</span>
191
- </div>
192
- );
193
- }
194
-
195
- return (
196
- <div className="quick-actions">
197
- {/* Handoff buttons */}
198
- {markerActions.type === 'handoff' && markerActions.responses && (
199
- <div className="quick-actions-buttons">
200
- {markerActions.responses.map((response) => (
201
- <Button
202
- variant="secondary"
203
- size="sm"
204
- type="button"
205
- key={response}
206
- className="quick-action-btn"
207
- onClick={() => handleButtonClick(response)}
208
- disabled={isDisabled}
209
- aria-label={`Continue with ${response}`}
210
- >
211
- {response}
212
- </Button>
213
- ))}
214
- </div>
215
- )}
216
-
217
- {/* Yes/No buttons */}
218
- {markerActions.type === 'yesno' && (
219
- <div className="quick-actions-buttons">
220
- <Button
221
- variant="secondary"
222
- size="sm"
223
- type="button"
224
- className="quick-action-btn"
225
- onClick={() => handleButtonClick('Yes')}
226
- disabled={isDisabled}
227
- aria-label="Answer Yes"
228
- >
229
- Yes
230
- </Button>
231
- <Button
232
- variant="secondary"
233
- size="sm"
234
- type="button"
235
- className="quick-action-btn"
236
- onClick={() => handleButtonClick('No')}
237
- disabled={isDisabled}
238
- aria-label="Answer No"
239
- >
240
- No
241
- </Button>
242
- </div>
243
- )}
244
-
245
- {/* Choice buttons */}
246
- {markerActions.type === 'choices' && markerActions.choices && (
247
- <div className="quick-actions-buttons">
248
- {markerActions.choices.map((choice) => (
249
- <Button
250
- variant="secondary"
251
- size="sm"
252
- type="button"
253
- key={choice.number}
254
- className="quick-action-btn"
255
- onClick={() => handleButtonClick(choice.text)}
256
- disabled={isDisabled}
257
- aria-label={`Choose option ${choice.number}: ${choice.text}`}
258
- >
259
- {choice.text}
260
- </Button>
261
- ))}
262
- </div>
263
- )}
264
- </div>
265
- );
266
- }
267
-
@@ -1,352 +0,0 @@
1
- /**
2
- * SpanTimeline Component
3
- *
4
- * Visual timeline of OTEL spans showing tool execution over time.
5
- *
6
- * Features:
7
- * - Horizontal timeline with spans as bars
8
- * - Color-coded by tool type
9
- * - Click to expand span details
10
- * - Real-time updates via WebSocket
11
- * - Zoom and scroll controls
12
- */
13
-
14
- import React, { useState, useEffect, useCallback, useRef, useMemo } from 'react';
15
- import { Button } from '@/components/ui/button';
16
- import { Skeleton } from '@/components/ui/skeleton';
17
- import { Badge } from '@/components/ui/badge';
18
- import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
19
-
20
- // =============================================================================
21
- // Types
22
- // =============================================================================
23
-
24
- interface EnrichedSpan {
25
- spanId: string;
26
- traceId: string;
27
- toolName: string;
28
- startTime: number;
29
- endTime?: number;
30
- durationMs: number;
31
- status: 'running' | 'completed' | 'error';
32
- success: boolean;
33
- error?: string;
34
- enrichment: {
35
- filePath?: string;
36
- command?: string;
37
- pattern?: string;
38
- summary?: string;
39
- };
40
- }
41
-
42
- interface SpanTimelineProps {
43
- /** Height of the timeline in pixels */
44
- height?: number;
45
- /** Whether to auto-scroll to newest spans */
46
- autoScroll?: boolean;
47
- }
48
-
49
- // Tool colors for visual distinction
50
- const TOOL_COLORS: Record<string, string> = {
51
- Read: '#4ade80', // Green
52
- Write: '#f87171', // Red
53
- Edit: '#fb923c', // Orange
54
- Bash: '#60a5fa', // Blue
55
- Grep: '#a78bfa', // Purple
56
- Glob: '#2dd4bf', // Teal
57
- Task: '#e879f9', // Magenta
58
- WebFetch: '#22d3ee', // Cyan
59
- WebSearch: '#fbbf24', // Yellow
60
- };
61
-
62
- // =============================================================================
63
- // Helper Functions
64
- // =============================================================================
65
-
66
- function formatTime(timestamp: number): string {
67
- return new Date(timestamp).toLocaleTimeString();
68
- }
69
-
70
- function formatDuration(ms: number): string {
71
- if (ms < 1000) return `${ms}ms`;
72
- if (ms < 60000) return `${(ms / 1000).toFixed(1)}s`;
73
- return `${(ms / 60000).toFixed(1)}m`;
74
- }
75
-
76
- function getToolColor(toolName: string): string {
77
- return TOOL_COLORS[toolName] || '#888888';
78
- }
79
-
80
- // =============================================================================
81
- // Component
82
- // =============================================================================
83
-
84
- export function SpanTimeline({ height = 200, autoScroll = true }: SpanTimelineProps): React.ReactElement {
85
- const [spans, setSpans] = useState<EnrichedSpan[]>([]);
86
- const [selectedSpan, setSelectedSpan] = useState<EnrichedSpan | null>(null);
87
- const [loading, setLoading] = useState(true);
88
- const [zoomLevel, setZoomLevel] = useState(1); // 1 = 1px per 100ms
89
- const timelineRef = useRef<HTMLDivElement>(null);
90
- const containerRef = useRef<HTMLDivElement>(null);
91
-
92
- // Fetch initial spans
93
- useEffect(() => {
94
- fetch('/api/spans?limit=100')
95
- .then(res => res.ok ? res.json() : { spans: [] })
96
- .then(data => {
97
- setSpans(data.spans || []);
98
- setLoading(false);
99
- })
100
- .catch(() => setLoading(false));
101
- }, []);
102
-
103
- // Subscribe to real-time updates
104
- useEffect(() => {
105
- const ws = new WebSocket(`ws://${window.location.host}/ws/spans`);
106
-
107
- ws.onmessage = (event) => {
108
- try {
109
- const data = JSON.parse(event.data);
110
- if (data.type === 'span' && data.span) {
111
- setSpans(prev => {
112
- // Update existing span or add new one
113
- const existing = prev.findIndex(s => s.spanId === data.span.spanId);
114
- if (existing >= 0) {
115
- const updated = [...prev];
116
- updated[existing] = data.span;
117
- return updated;
118
- }
119
- return [...prev, data.span].slice(-100); // Keep last 100
120
- });
121
- }
122
- } catch (err) {
123
- console.error('[SpanTimeline] WebSocket parse error:', err);
124
- }
125
- };
126
-
127
- return () => ws.close();
128
- }, []);
129
-
130
- // Auto-scroll to right (newest spans)
131
- useEffect(() => {
132
- if (autoScroll && containerRef.current) {
133
- containerRef.current.scrollLeft = containerRef.current.scrollWidth;
134
- }
135
- }, [spans, autoScroll]);
136
-
137
- // Calculate timeline bounds
138
- const { minTime, maxTime, timeRange } = useMemo(() => {
139
- if (spans.length === 0) {
140
- const now = Date.now();
141
- return { minTime: now - 60000, maxTime: now, timeRange: 60000 };
142
- }
143
-
144
- const times = spans.flatMap(s => [s.startTime, s.endTime || s.startTime + s.durationMs]);
145
- const min = Math.min(...times);
146
- const max = Math.max(...times);
147
- const range = Math.max(max - min, 10000); // At least 10 seconds
148
-
149
- return { minTime: min, maxTime: max, timeRange: range };
150
- }, [spans]);
151
-
152
- // Calculate timeline width based on zoom
153
- const timelineWidth = useMemo(() => {
154
- return Math.max(800, (timeRange / 100) * zoomLevel);
155
- }, [timeRange, zoomLevel]);
156
-
157
- // Handle span click
158
- const handleSpanClick = useCallback((span: EnrichedSpan) => {
159
- setSelectedSpan(selectedSpan?.spanId === span.spanId ? null : span);
160
- }, [selectedSpan]);
161
-
162
- // Zoom controls
163
- const handleZoomIn = useCallback(() => {
164
- setZoomLevel(prev => Math.min(prev * 1.5, 10));
165
- }, []);
166
-
167
- const handleZoomOut = useCallback(() => {
168
- setZoomLevel(prev => Math.max(prev / 1.5, 0.1));
169
- }, []);
170
-
171
- const handleZoomReset = useCallback(() => {
172
- setZoomLevel(1);
173
- }, []);
174
-
175
- if (loading) {
176
- return (
177
- <div className="span-timeline-loading p-2 space-y-2">
178
- <div className="flex gap-2 items-center">
179
- <Skeleton className="h-4 w-20" />
180
- <div className="flex-1" />
181
- <Skeleton className="h-6 w-6" />
182
- <Skeleton className="h-6 w-6" />
183
- <Skeleton className="h-6 w-6" />
184
- </div>
185
- <Skeleton className="h-[120px] w-full" />
186
- </div>
187
- );
188
- }
189
-
190
- return (
191
- <TooltipProvider delayDuration={300}>
192
- <div className="span-timeline" style={{ height }}>
193
- {/* Toolbar */}
194
- <div className="span-timeline-toolbar">
195
- <span className="span-count">{spans.length} spans</span>
196
- <div className="zoom-controls">
197
- <Tooltip>
198
- <TooltipTrigger asChild>
199
- <Button variant="ghost" size="icon" onClick={handleZoomOut}>−</Button>
200
- </TooltipTrigger>
201
- <TooltipContent>Zoom out</TooltipContent>
202
- </Tooltip>
203
- <Tooltip>
204
- <TooltipTrigger asChild>
205
- <Button variant="ghost" size="icon" onClick={handleZoomReset}>⟳</Button>
206
- </TooltipTrigger>
207
- <TooltipContent>Reset zoom</TooltipContent>
208
- </Tooltip>
209
- <Tooltip>
210
- <TooltipTrigger asChild>
211
- <Button variant="ghost" size="icon" onClick={handleZoomIn}>+</Button>
212
- </TooltipTrigger>
213
- <TooltipContent>Zoom in</TooltipContent>
214
- </Tooltip>
215
- </div>
216
- </div>
217
-
218
- {/* Timeline container */}
219
- <div className="span-timeline-container" ref={containerRef}>
220
- <div
221
- className="span-timeline-track"
222
- ref={timelineRef}
223
- style={{ width: timelineWidth }}
224
- >
225
- {/* Time axis */}
226
- <div className="span-timeline-axis">
227
- {Array.from({ length: Math.ceil(timeRange / 10000) + 1 }, (_, i) => {
228
- const time = minTime + i * 10000;
229
- const left = ((time - minTime) / timeRange) * timelineWidth;
230
- return (
231
- <div
232
- key={i}
233
- className="axis-tick"
234
- style={{ left }}
235
- >
236
- <span className="tick-label">{formatTime(time)}</span>
237
- </div>
238
- );
239
- })}
240
- </div>
241
-
242
- {/* Span bars */}
243
- <div className="span-timeline-bars">
244
- {spans.map((span, index) => {
245
- const left = ((span.startTime - minTime) / timeRange) * timelineWidth;
246
- const width = Math.max(4, (span.durationMs / timeRange) * timelineWidth);
247
- const top = (index % 5) * 28 + 4; // Stack in 5 rows
248
-
249
- return (
250
- <Tooltip key={span.spanId}>
251
- <TooltipTrigger asChild>
252
- <div
253
- className={`span-bar ${span.success ? '' : 'error'} ${
254
- selectedSpan?.spanId === span.spanId ? 'selected' : ''
255
- }`}
256
- style={{
257
- left,
258
- width,
259
- top,
260
- backgroundColor: getToolColor(span.toolName),
261
- }}
262
- onClick={() => handleSpanClick(span)}
263
- >
264
- <span className="span-bar-label">{span.toolName}</span>
265
- </div>
266
- </TooltipTrigger>
267
- <TooltipContent>{`${span.toolName}: ${formatDuration(span.durationMs)}`}</TooltipContent>
268
- </Tooltip>
269
- );
270
- })}
271
- </div>
272
- </div>
273
- </div>
274
-
275
- {/* Selected span details */}
276
- {selectedSpan && (
277
- <div className="span-timeline-details">
278
- <div className="detail-header">
279
- <Badge
280
- variant="default"
281
- className="tool-badge"
282
- style={{ backgroundColor: getToolColor(selectedSpan.toolName) }}
283
- >
284
- {selectedSpan.toolName}
285
- </Badge>
286
- <span className="duration">{formatDuration(selectedSpan.durationMs)}</span>
287
- <span className={`status ${selectedSpan.success ? 'success' : 'error'}`}>
288
- {selectedSpan.success ? '✓' : '✗'}
289
- </span>
290
- <Button
291
- variant="ghost"
292
- size="icon"
293
- className="close-details"
294
- onClick={() => setSelectedSpan(null)}
295
- >
296
- ×
297
- </Button>
298
- </div>
299
- <div className="detail-body">
300
- <div className="detail-row">
301
- <label>Time:</label>
302
- <span>{formatTime(selectedSpan.startTime)}</span>
303
- </div>
304
- {selectedSpan.enrichment.filePath && (
305
- <div className="detail-row">
306
- <label>File:</label>
307
- <span className="mono">{selectedSpan.enrichment.filePath}</span>
308
- </div>
309
- )}
310
- {selectedSpan.enrichment.command && (
311
- <div className="detail-row">
312
- <label>Command:</label>
313
- <span className="mono">{selectedSpan.enrichment.command}</span>
314
- </div>
315
- )}
316
- {selectedSpan.enrichment.pattern && (
317
- <div className="detail-row">
318
- <label>Pattern:</label>
319
- <span className="mono">{selectedSpan.enrichment.pattern}</span>
320
- </div>
321
- )}
322
- {selectedSpan.enrichment.summary && (
323
- <div className="detail-row">
324
- <label>Summary:</label>
325
- <span>{selectedSpan.enrichment.summary}</span>
326
- </div>
327
- )}
328
- {selectedSpan.error && (
329
- <div className="detail-row error">
330
- <label>Error:</label>
331
- <span>{selectedSpan.error}</span>
332
- </div>
333
- )}
334
- </div>
335
- </div>
336
- )}
337
-
338
- {/* Legend */}
339
- <div className="span-timeline-legend">
340
- {Object.entries(TOOL_COLORS).slice(0, 6).map(([tool, color]) => (
341
- <div key={tool} className="legend-item">
342
- <span className="legend-color" style={{ backgroundColor: color }} />
343
- <span className="legend-label">{tool}</span>
344
- </div>
345
- ))}
346
- </div>
347
- </div>
348
- </TooltipProvider>
349
- );
350
- }
351
-
352
- export default SpanTimeline;
@@ -1,84 +0,0 @@
1
- /**
2
- * StandalonePanel - Full-screen panel wrapper for BikeRack mode
3
- *
4
- * Story MSSCI-14821: StandalonePanel wrapper and ?panel=X client routing
5
- * Epic: 101 (BikeRack Mode)
6
- *
7
- * Renders a single panel full-screen based on ?panel=X URL parameter.
8
- * PANEL_REGISTRY is the single source of truth for routing (CE-2).
9
- *
10
- * Rules:
11
- * - No dockview-react imports (Rule 7)
12
- * - No BikeRack-specific props to panels (Rule 2)
13
- * - URL-based detection only (Rule 10)
14
- */
15
-
16
- import React from 'react';
17
- import {
18
- EnhancedSprintPanel,
19
- GitPanel,
20
- DiffsPanel,
21
- TodoPanel,
22
- WorkflowPanel,
23
- BackgroundPanel,
24
- AuditLogPanel,
25
- ChangedPanel,
26
- ACPanel,
27
- TTYPanel,
28
- DebugPanel,
29
- BikeLanePanel,
30
- SettingsPanel,
31
- } from './panels';
32
-
33
- /**
34
- * Registry mapping panel URL names to their components.
35
- * Single source of truth for standalone panel routing (CE-2).
36
- */
37
- export const PANEL_REGISTRY: Record<string, React.ComponentType> = {
38
- sprint: EnhancedSprintPanel,
39
- git: GitPanel,
40
- diffs: DiffsPanel,
41
- todos: TodoPanel,
42
- workflow: WorkflowPanel,
43
- background: BackgroundPanel,
44
- audit: AuditLogPanel,
45
- changed: ChangedPanel,
46
- ac: ACPanel,
47
- tty: TTYPanel,
48
- debug: DebugPanel,
49
- bikelane: BikeLanePanel,
50
- settings: SettingsPanel,
51
- };
52
-
53
- /**
54
- * Detect standalone panel mode from URL parameters (Rule 10).
55
- */
56
- export function getStandalonePanelName(): string | null {
57
- const params = new URLSearchParams(window.location.search);
58
- return params.get('panel');
59
- }
60
-
61
- /**
62
- * StandalonePanel wrapper - renders a single panel full-screen.
63
- */
64
- export function StandalonePanel(): React.ReactElement {
65
- const panelName = getStandalonePanelName();
66
- const PanelComponent = panelName ? PANEL_REGISTRY[panelName] : null;
67
-
68
- if (!PanelComponent) {
69
- return (
70
- <div style={{ display: 'flex', flexDirection: 'column', alignItems: 'center', justifyContent: 'center', height: '100vh', width: '100vw', backgroundColor: 'var(--bg-primary, #1a1a2e)', color: 'var(--text-primary, #e4e4e7)' }}>
71
- <h1>Panel not found</h1>
72
- <p>
73
- <a href="/bikerack" style={{ color: 'var(--accent, #818cf8)' }}>Back to BikeRack</a>
74
- </p>
75
- </div>
76
- );
77
- }
78
-
79
- return (
80
- <div style={{ height: '100vh', width: '100vw', overflow: 'auto', backgroundColor: 'var(--bg-primary, #1a1a2e)', color: 'var(--text-primary, #e4e4e7)' }}>
81
- <PanelComponent />
82
- </div>
83
- );
84
- }