@pennyfarthing/core 11.3.8 → 11.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (304) hide show
  1. package/README.md +1 -1
  2. package/package.json +4 -1
  3. package/packages/core/dist/public/css/react.css +1 -1
  4. package/packages/core/dist/public/js/react/react.js +24 -24
  5. package/packages/core/src/public/App.tsx +356 -0
  6. package/packages/core/src/public/components/AgentLoadDialog.tsx +202 -0
  7. package/packages/core/src/public/components/AgentPopup.tsx +308 -0
  8. package/packages/core/src/public/components/ApprovalModal/ApprovalModal.css +35 -0
  9. package/packages/core/src/public/components/ApprovalModal/index.tsx +632 -0
  10. package/packages/core/src/public/components/BikeRackIndex.tsx +53 -0
  11. package/packages/core/src/public/components/BikeRackWorkspace.tsx +217 -0
  12. package/packages/core/src/public/components/CommandPalette.tsx +554 -0
  13. package/packages/core/src/public/components/ConfirmDialog.tsx +168 -0
  14. package/packages/core/src/public/components/ContextIndicator/ContextIndicator.css +85 -0
  15. package/packages/core/src/public/components/ContextIndicator/index.tsx +330 -0
  16. package/packages/core/src/public/components/ContextSparkline.tsx +56 -0
  17. package/packages/core/src/public/components/ControlBar.tsx +636 -0
  18. package/packages/core/src/public/components/DeadCodeDialog.tsx +169 -0
  19. package/packages/core/src/public/components/DiffViewer.tsx +585 -0
  20. package/packages/core/src/public/components/DockviewWorkspace.tsx +749 -0
  21. package/packages/core/src/public/components/Editor.tsx +630 -0
  22. package/packages/core/src/public/components/ErrorBoundary.tsx +67 -0
  23. package/packages/core/src/public/components/FileTree.tsx +379 -0
  24. package/packages/core/src/public/components/FontPicker/FontPicker.css +276 -0
  25. package/packages/core/src/public/components/FontPicker/index.tsx +430 -0
  26. package/packages/core/src/public/components/FullFileTree.tsx +237 -0
  27. package/packages/core/src/public/components/HealthGauge.tsx +181 -0
  28. package/packages/core/src/public/components/Message.tsx +225 -0
  29. package/packages/core/src/public/components/MessageList.tsx +98 -0
  30. package/packages/core/src/public/components/MessageView.tsx +400 -0
  31. package/packages/core/src/public/components/ModeSwitch/ModeSwitch.css +165 -0
  32. package/packages/core/src/public/components/ModeSwitch/index.tsx +372 -0
  33. package/packages/core/src/public/components/PersonaHeader.tsx +242 -0
  34. package/packages/core/src/public/components/ProjectInfoBar.tsx +45 -0
  35. package/packages/core/src/public/components/QuickActions.tsx +267 -0
  36. package/packages/core/src/public/components/SpanTimeline.tsx +352 -0
  37. package/packages/core/src/public/components/StandalonePanel.tsx +82 -0
  38. package/packages/core/src/public/components/StatsStrip.tsx +162 -0
  39. package/packages/core/src/public/components/StreamingContent.tsx +77 -0
  40. package/packages/core/src/public/components/SubagentSpan.tsx +180 -0
  41. package/packages/core/src/public/components/TandemPortrait.tsx +72 -0
  42. package/packages/core/src/public/components/ThemePalette/ThemePalette.css +179 -0
  43. package/packages/core/src/public/components/ThemePalette/index.tsx +326 -0
  44. package/packages/core/src/public/components/ToolCallBlock.tsx +252 -0
  45. package/packages/core/src/public/components/ToolStack.tsx +209 -0
  46. package/packages/core/src/public/components/ToolStatus.tsx +57 -0
  47. package/packages/core/src/public/components/dialogs/CodeMarkersDialog.tsx +169 -0
  48. package/packages/core/src/public/components/dialogs/ComplexityDialog.tsx +163 -0
  49. package/packages/core/src/public/components/dialogs/DependenciesDialog.tsx +120 -0
  50. package/packages/core/src/public/components/dialogs/HotspotsDialog.tsx +451 -0
  51. package/packages/core/src/public/components/dialogs/ToolDialog.tsx +43 -0
  52. package/packages/core/src/public/components/panel-registry.ts +13 -0
  53. package/packages/core/src/public/components/panels/ACPanel.tsx +93 -0
  54. package/packages/core/src/public/components/panels/AcceptanceCriteriaPanel.tsx +104 -0
  55. package/packages/core/src/public/components/panels/AuditLogPanel.tsx +489 -0
  56. package/packages/core/src/public/components/panels/BackgroundPanel.tsx +115 -0
  57. package/packages/core/src/public/components/panels/BikeLanePanel.tsx +214 -0
  58. package/packages/core/src/public/components/panels/DebugPanel.tsx +344 -0
  59. package/packages/core/src/public/components/panels/DiffView.tsx +109 -0
  60. package/packages/core/src/public/components/panels/DiffsPanel.tsx +56 -0
  61. package/packages/core/src/public/components/panels/GitPanel.tsx +260 -0
  62. package/packages/core/src/public/components/panels/HotspotsPanel.tsx +365 -0
  63. package/packages/core/src/public/components/panels/MessageFeed.tsx +39 -0
  64. package/packages/core/src/public/components/panels/MessagePanel.tsx +497 -0
  65. package/packages/core/src/public/components/panels/ProgressPanel.tsx +189 -0
  66. package/packages/core/src/public/components/panels/SettingsPanel.tsx +361 -0
  67. package/packages/core/src/public/components/panels/SprintPanel.tsx +723 -0
  68. package/packages/core/src/public/components/panels/TandemPanel.tsx +104 -0
  69. package/packages/core/src/public/components/panels/TaskTracker.tsx +48 -0
  70. package/packages/core/src/public/components/panels/TeamPanel.tsx +64 -0
  71. package/packages/core/src/public/components/panels/TeamRoster.tsx +67 -0
  72. package/packages/core/src/public/components/panels/TodoPanel.tsx +142 -0
  73. package/packages/core/src/public/components/panels/WorkflowPanel.tsx +224 -0
  74. package/packages/core/src/public/components/panels/index.ts +24 -0
  75. package/packages/core/src/public/components/ui/alert-dialog.tsx +139 -0
  76. package/packages/core/src/public/components/ui/badge.tsx +36 -0
  77. package/packages/core/src/public/components/ui/button.tsx +57 -0
  78. package/packages/core/src/public/components/ui/checkbox.tsx +28 -0
  79. package/packages/core/src/public/components/ui/collapsible.tsx +9 -0
  80. package/packages/core/src/public/components/ui/command.tsx +151 -0
  81. package/packages/core/src/public/components/ui/dialog.tsx +120 -0
  82. package/packages/core/src/public/components/ui/popover.tsx +31 -0
  83. package/packages/core/src/public/components/ui/progress.tsx +28 -0
  84. package/packages/core/src/public/components/ui/scroll-area.tsx +46 -0
  85. package/packages/core/src/public/components/ui/select.tsx +157 -0
  86. package/packages/core/src/public/components/ui/separator.tsx +29 -0
  87. package/packages/core/src/public/components/ui/skeleton.tsx +15 -0
  88. package/packages/core/src/public/components/ui/switch.tsx +27 -0
  89. package/packages/core/src/public/components/ui/toggle-group.tsx +59 -0
  90. package/packages/core/src/public/components/ui/toggle.tsx +43 -0
  91. package/packages/core/src/public/components/ui/tooltip.tsx +30 -0
  92. package/packages/core/src/public/contexts/ClaudeContext.tsx +311 -0
  93. package/packages/core/src/public/contexts/MessageQueueContext.tsx +143 -0
  94. package/packages/core/src/public/css/theme-browser.css +550 -0
  95. package/packages/core/src/public/css/theme-system.css +630 -0
  96. package/packages/core/src/public/hooks/index.ts +49 -0
  97. package/packages/core/src/public/hooks/useAgentLoad.ts +105 -0
  98. package/packages/core/src/public/hooks/useBackgroundTasks.ts +131 -0
  99. package/packages/core/src/public/hooks/useClaude.ts +234 -0
  100. package/packages/core/src/public/hooks/useCodeMarkers.ts +101 -0
  101. package/packages/core/src/public/hooks/useColorScheme.ts +42 -0
  102. package/packages/core/src/public/hooks/useCommandHistory.ts +99 -0
  103. package/packages/core/src/public/hooks/useComplexity.ts +80 -0
  104. package/packages/core/src/public/hooks/useDeadCode.ts +99 -0
  105. package/packages/core/src/public/hooks/useDependencies.ts +82 -0
  106. package/packages/core/src/public/hooks/useDiffs.ts +143 -0
  107. package/packages/core/src/public/hooks/useFileBrowser.ts +73 -0
  108. package/packages/core/src/public/hooks/useFocusPanel.ts +137 -0
  109. package/packages/core/src/public/hooks/useGitStatus.ts +233 -0
  110. package/packages/core/src/public/hooks/useHealthScore.ts +71 -0
  111. package/packages/core/src/public/hooks/useHotspots.ts +123 -0
  112. package/packages/core/src/public/hooks/useLayoutPersistence.ts +141 -0
  113. package/packages/core/src/public/hooks/useMarkdownParser.ts +36 -0
  114. package/packages/core/src/public/hooks/useMarkerActions.ts +234 -0
  115. package/packages/core/src/public/hooks/useMessageQueue.ts +380 -0
  116. package/packages/core/src/public/hooks/useMessageStream.ts +131 -0
  117. package/packages/core/src/public/hooks/usePersona.ts +112 -0
  118. package/packages/core/src/public/hooks/usePlanModeExit.ts +105 -0
  119. package/packages/core/src/public/hooks/useResponsiveLayout.ts +173 -0
  120. package/packages/core/src/public/hooks/useSprint.ts +157 -0
  121. package/packages/core/src/public/hooks/useStatsStrip.ts +204 -0
  122. package/packages/core/src/public/hooks/useStory.ts +135 -0
  123. package/packages/core/src/public/hooks/useSubagentHelper.ts +64 -0
  124. package/packages/core/src/public/hooks/useSyntaxHighlighter.ts +52 -0
  125. package/packages/core/src/public/hooks/useTabCompletion.ts +124 -0
  126. package/packages/core/src/public/hooks/useTandemObservations.ts +165 -0
  127. package/packages/core/src/public/hooks/useTeamMembers.ts +273 -0
  128. package/packages/core/src/public/hooks/useTodos.ts +93 -0
  129. package/packages/core/src/public/hooks/useUserAvatar.ts +54 -0
  130. package/packages/core/src/public/images/cyclist-dark.png +0 -0
  131. package/packages/core/src/public/images/cyclist-light.png +0 -0
  132. package/packages/core/src/public/index.html +14 -0
  133. package/packages/core/src/public/index.tsx +10 -0
  134. package/packages/core/src/public/lib/utils.ts +6 -0
  135. package/packages/core/src/public/styles/dockview-theme.css +376 -0
  136. package/packages/core/src/public/styles/tailwind.css +4454 -0
  137. package/packages/core/src/public/types/message.ts +51 -0
  138. package/packages/core/src/public/utils/avatar-service.ts +73 -0
  139. package/packages/core/src/public/utils/color-presets.ts +940 -0
  140. package/packages/core/src/public/utils/font-presets.ts +362 -0
  141. package/packages/core/src/public/utils/formatDuration.ts +14 -0
  142. package/packages/core/src/public/utils/markdown.ts +249 -0
  143. package/packages/core/src/public/utils/messageFilters.ts +128 -0
  144. package/packages/core/src/public/utils/slash-commands.ts +341 -0
  145. package/packages/core/src/public/utils/subagent-display.ts +146 -0
  146. package/packages/core/src/public/utils/syntax.ts +219 -0
  147. package/packages/core/src/public/utils/toolIntentSummarizer.ts +199 -0
  148. package/packages/core/src/public/utils/toolStackGrouper.ts +106 -0
  149. package/packages/core/src/public/utils/toolTypeColors.ts +45 -0
  150. package/pennyfarthing-dist/pf/__pycache__/__init__.cpython-314.pyc +0 -0
  151. package/pennyfarthing-dist/pf/__pycache__/cli.cpython-314.pyc +0 -0
  152. package/pennyfarthing-dist/pf/__pycache__/context.cpython-314.pyc +0 -0
  153. package/pennyfarthing-dist/pf/bc/__pycache__/__init__.cpython-314.pyc +0 -0
  154. package/pennyfarthing-dist/pf/bc/__pycache__/cli.cpython-314.pyc +0 -0
  155. package/pennyfarthing-dist/pf/bc/__pycache__/focus.cpython-314.pyc +0 -0
  156. package/pennyfarthing-dist/pf/bc/__pycache__/split.cpython-314.pyc +0 -0
  157. package/pennyfarthing-dist/pf/bc/cli.py +0 -1
  158. package/pennyfarthing-dist/pf/bc/focus.py +0 -1
  159. package/pennyfarthing-dist/pf/bikerack/__pycache__/__init__.cpython-314.pyc +0 -0
  160. package/pennyfarthing-dist/pf/bikerack/__pycache__/base_panel.cpython-314.pyc +0 -0
  161. package/pennyfarthing-dist/pf/bikerack/__pycache__/cli.cpython-314.pyc +0 -0
  162. package/pennyfarthing-dist/pf/bikerack/__pycache__/git_panel.cpython-314.pyc +0 -0
  163. package/pennyfarthing-dist/pf/bikerack/__pycache__/launcher.cpython-314.pyc +0 -0
  164. package/pennyfarthing-dist/pf/bikerack/__pycache__/portrait_resolver.cpython-314.pyc +0 -0
  165. package/pennyfarthing-dist/pf/bikerack/__pycache__/sprint_panel.cpython-314.pyc +0 -0
  166. package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_data.cpython-314.pyc +0 -0
  167. package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_screen.cpython-314.pyc +0 -0
  168. package/pennyfarthing-dist/pf/bikerack/__pycache__/tui.cpython-314.pyc +0 -0
  169. package/pennyfarthing-dist/pf/bikerack/base_panel.py +0 -1
  170. package/pennyfarthing-dist/pf/bikerack/events.py +1 -7
  171. package/pennyfarthing-dist/pf/bikerack/git_panel.py +273 -10
  172. package/pennyfarthing-dist/pf/bikerack/portrait_resolver.py +21 -0
  173. package/pennyfarthing-dist/pf/bikerack/sprint_panel.py +58 -1
  174. package/pennyfarthing-dist/pf/bikerack/tui.py +5 -20
  175. package/pennyfarthing-dist/pf/bmad/__pycache__/__init__.cpython-314.pyc +0 -0
  176. package/pennyfarthing-dist/pf/bmad/__pycache__/cli.cpython-314.pyc +0 -0
  177. package/pennyfarthing-dist/pf/bmad/__pycache__/parser.cpython-314.pyc +0 -0
  178. package/pennyfarthing-dist/pf/bmad/parser.py +15 -9
  179. package/pennyfarthing-dist/pf/codemarkers/__pycache__/__init__.cpython-314.pyc +0 -0
  180. package/pennyfarthing-dist/pf/codemarkers/__pycache__/analyze.cpython-314.pyc +0 -0
  181. package/pennyfarthing-dist/pf/codemarkers/__pycache__/models.cpython-314.pyc +0 -0
  182. package/pennyfarthing-dist/pf/common/__pycache__/__init__.cpython-314.pyc +0 -0
  183. package/pennyfarthing-dist/pf/common/__pycache__/config.cpython-314.pyc +0 -0
  184. package/pennyfarthing-dist/pf/common/__pycache__/output.cpython-314.pyc +0 -0
  185. package/pennyfarthing-dist/pf/common/__pycache__/pr_config.cpython-314.pyc +0 -0
  186. package/pennyfarthing-dist/pf/common/__pycache__/themes.cpython-314.pyc +0 -0
  187. package/pennyfarthing-dist/pf/common/pr_config.py +27 -2
  188. package/pennyfarthing-dist/pf/complexity/__pycache__/__init__.cpython-314.pyc +0 -0
  189. package/pennyfarthing-dist/pf/complexity/__pycache__/analyze.cpython-314.pyc +0 -0
  190. package/pennyfarthing-dist/pf/complexity/__pycache__/models.cpython-314.pyc +0 -0
  191. package/pennyfarthing-dist/pf/consultation/__pycache__/__init__.cpython-314.pyc +0 -0
  192. package/pennyfarthing-dist/pf/consultation/__pycache__/cli.cpython-314.pyc +0 -0
  193. package/pennyfarthing-dist/pf/deadcode/__pycache__/__init__.cpython-314.pyc +0 -0
  194. package/pennyfarthing-dist/pf/deadcode/__pycache__/analyze.cpython-314.pyc +0 -0
  195. package/pennyfarthing-dist/pf/deadcode/__pycache__/cli.cpython-314.pyc +0 -0
  196. package/pennyfarthing-dist/pf/deadcode/__pycache__/models.cpython-314.pyc +0 -0
  197. package/pennyfarthing-dist/pf/dependencies/__pycache__/__init__.cpython-314.pyc +0 -0
  198. package/pennyfarthing-dist/pf/dependencies/__pycache__/analyze.cpython-314.pyc +0 -0
  199. package/pennyfarthing-dist/pf/dependencies/__pycache__/models.cpython-314.pyc +0 -0
  200. package/pennyfarthing-dist/pf/epic/__pycache__/__init__.cpython-314.pyc +0 -0
  201. package/pennyfarthing-dist/pf/epic/__pycache__/cli.cpython-314.pyc +0 -0
  202. package/pennyfarthing-dist/pf/git_group/__pycache__/__init__.cpython-314.pyc +0 -0
  203. package/pennyfarthing-dist/pf/git_group/__pycache__/cli.cpython-314.pyc +0 -0
  204. package/pennyfarthing-dist/pf/handoff/__pycache__/__init__.cpython-314.pyc +0 -0
  205. package/pennyfarthing-dist/pf/handoff/__pycache__/cli.cpython-314.pyc +0 -0
  206. package/pennyfarthing-dist/pf/handoff/__pycache__/complete_phase.cpython-314.pyc +0 -0
  207. package/pennyfarthing-dist/pf/handoff/__pycache__/marker.cpython-314.pyc +0 -0
  208. package/pennyfarthing-dist/pf/handoff/__pycache__/phase_check.cpython-314.pyc +0 -0
  209. package/pennyfarthing-dist/pf/handoff/__pycache__/resolve_gate.cpython-314.pyc +0 -0
  210. package/pennyfarthing-dist/pf/healthscore/__pycache__/__init__.cpython-314.pyc +0 -0
  211. package/pennyfarthing-dist/pf/healthscore/__pycache__/__main__.cpython-314.pyc +0 -0
  212. package/pennyfarthing-dist/pf/healthscore/__pycache__/analyze.cpython-314.pyc +0 -0
  213. package/pennyfarthing-dist/pf/healthscore/__pycache__/cli.cpython-314.pyc +0 -0
  214. package/pennyfarthing-dist/pf/healthscore/__pycache__/formatters.cpython-314.pyc +0 -0
  215. package/pennyfarthing-dist/pf/healthscore/__pycache__/models.cpython-314.pyc +0 -0
  216. package/pennyfarthing-dist/pf/hooks/__pycache__/__init__.cpython-314.pyc +0 -0
  217. package/pennyfarthing-dist/pf/hooks/__pycache__/bell_mode.cpython-314.pyc +0 -0
  218. package/pennyfarthing-dist/pf/hooks/__pycache__/cli.cpython-314.pyc +0 -0
  219. package/pennyfarthing-dist/pf/hooks/__pycache__/context_breaker.cpython-314.pyc +0 -0
  220. package/pennyfarthing-dist/pf/hooks/__pycache__/context_warning.cpython-314.pyc +0 -0
  221. package/pennyfarthing-dist/pf/hooks/__pycache__/cyclist_pretooluse.cpython-314.pyc +0 -0
  222. package/pennyfarthing-dist/pf/hooks/__pycache__/pre_edit_check.cpython-314.pyc +0 -0
  223. package/pennyfarthing-dist/pf/hooks/__pycache__/reflector_check.cpython-314.pyc +0 -0
  224. package/pennyfarthing-dist/pf/hooks/__pycache__/schema_validation.cpython-314.pyc +0 -0
  225. package/pennyfarthing-dist/pf/hooks/__pycache__/session_start.cpython-314.pyc +0 -0
  226. package/pennyfarthing-dist/pf/hooks/__pycache__/session_stop.cpython-314.pyc +0 -0
  227. package/pennyfarthing-dist/pf/hooks/__pycache__/sprint_yaml_validation.cpython-314.pyc +0 -0
  228. package/pennyfarthing-dist/pf/hooks/__pycache__/statusline.cpython-314.pyc +0 -0
  229. package/pennyfarthing-dist/pf/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
  230. package/pennyfarthing-dist/pf/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
  231. package/pennyfarthing-dist/pf/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
  232. package/pennyfarthing-dist/pf/hotspots/__pycache__/models.cpython-314.pyc +0 -0
  233. package/pennyfarthing-dist/pf/jira/__pycache__/__init__.cpython-314.pyc +0 -0
  234. package/pennyfarthing-dist/pf/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
  235. package/pennyfarthing-dist/pf/jira/__pycache__/claim.cpython-314.pyc +0 -0
  236. package/pennyfarthing-dist/pf/jira/__pycache__/cli.cpython-314.pyc +0 -0
  237. package/pennyfarthing-dist/pf/jira/__pycache__/client.cpython-314.pyc +0 -0
  238. package/pennyfarthing-dist/pf/jira/__pycache__/create.cpython-314.pyc +0 -0
  239. package/pennyfarthing-dist/pf/jira/__pycache__/epic.cpython-314.pyc +0 -0
  240. package/pennyfarthing-dist/pf/jira/__pycache__/operations.cpython-314.pyc +0 -0
  241. package/pennyfarthing-dist/pf/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
  242. package/pennyfarthing-dist/pf/jira/__pycache__/story.cpython-314.pyc +0 -0
  243. package/pennyfarthing-dist/pf/jira/__pycache__/sync.cpython-314.pyc +0 -0
  244. package/pennyfarthing-dist/pf/launch/__pycache__/__init__.cpython-314.pyc +0 -0
  245. package/pennyfarthing-dist/pf/launch/__pycache__/cli.cpython-314.pyc +0 -0
  246. package/pennyfarthing-dist/pf/prime/__pycache__/__init__.cpython-314.pyc +0 -0
  247. package/pennyfarthing-dist/pf/prime/__pycache__/cli.cpython-314.pyc +0 -0
  248. package/pennyfarthing-dist/pf/prime/__pycache__/loader.cpython-314.pyc +0 -0
  249. package/pennyfarthing-dist/pf/prime/__pycache__/models.cpython-314.pyc +0 -0
  250. package/pennyfarthing-dist/pf/prime/__pycache__/persona.cpython-314.pyc +0 -0
  251. package/pennyfarthing-dist/pf/prime/__pycache__/session.cpython-314.pyc +0 -0
  252. package/pennyfarthing-dist/pf/prime/__pycache__/tiers.cpython-314.pyc +0 -0
  253. package/pennyfarthing-dist/pf/prime/__pycache__/workflow.cpython-314.pyc +0 -0
  254. package/pennyfarthing-dist/pf/session/__pycache__/__init__.cpython-314.pyc +0 -0
  255. package/pennyfarthing-dist/pf/session/__pycache__/cli.cpython-314.pyc +0 -0
  256. package/pennyfarthing-dist/pf/settings/__pycache__/__init__.cpython-314.pyc +0 -0
  257. package/pennyfarthing-dist/pf/settings/__pycache__/cli.cpython-314.pyc +0 -0
  258. package/pennyfarthing-dist/pf/settings/__pycache__/settings.cpython-314.pyc +0 -0
  259. package/pennyfarthing-dist/pf/settings/settings.py +44 -8
  260. package/pennyfarthing-dist/pf/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
  261. package/pennyfarthing-dist/pf/sprint/__pycache__/archive.cpython-314.pyc +0 -0
  262. package/pennyfarthing-dist/pf/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
  263. package/pennyfarthing-dist/pf/sprint/__pycache__/cli.cpython-314.pyc +0 -0
  264. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
  265. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
  266. package/pennyfarthing-dist/pf/sprint/__pycache__/loader.cpython-314.pyc +0 -0
  267. package/pennyfarthing-dist/pf/sprint/__pycache__/status.cpython-314.pyc +0 -0
  268. package/pennyfarthing-dist/pf/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
  269. package/pennyfarthing-dist/pf/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
  270. package/pennyfarthing-dist/pf/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
  271. package/pennyfarthing-dist/pf/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
  272. package/pennyfarthing-dist/pf/sprint/__pycache__/validator.cpython-314.pyc +0 -0
  273. package/pennyfarthing-dist/pf/sprint/__pycache__/work.cpython-314.pyc +0 -0
  274. package/pennyfarthing-dist/pf/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
  275. package/pennyfarthing-dist/pf/sprint/story_finish.py +14 -2
  276. package/pennyfarthing-dist/pf/sprint/validator.py +7 -7
  277. package/pennyfarthing-dist/pf/tests/__pycache__/__init__.cpython-314.pyc +0 -0
  278. package/pennyfarthing-dist/pf/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
  279. package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
  280. package/pennyfarthing-dist/pf/tests/test_sprint_validator.py +44 -0
  281. package/pennyfarthing-dist/pf/theme/__pycache__/__init__.cpython-314.pyc +0 -0
  282. package/pennyfarthing-dist/pf/theme/__pycache__/cli.cpython-314.pyc +0 -0
  283. package/pennyfarthing-dist/pf/validate/__pycache__/__init__.cpython-314.pyc +0 -0
  284. package/pennyfarthing-dist/pf/validate/__pycache__/cli.cpython-314.pyc +0 -0
  285. package/pennyfarthing-dist/pf/workflow/__pycache__/__init__.cpython-314.pyc +0 -0
  286. package/pennyfarthing-dist/pf/workflow/__pycache__/cli.cpython-314.pyc +0 -0
  287. package/pennyfarthing-dist/pf/workflow/__pycache__/helpers.cpython-314.pyc +0 -0
  288. package/pennyfarthing-dist/pf/workflow/__pycache__/scale.cpython-314.pyc +0 -0
  289. package/pennyfarthing-dist/pf/workflow/__pycache__/state.cpython-314.pyc +0 -0
  290. package/pennyfarthing-dist/scripts/lib/find-root.sh +1 -1
  291. package/pennyfarthing-dist/scripts/misc/migrate_bmad_workflow.py +0 -1
  292. package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +13 -13
  293. package/pennyfarthing-dist/scripts/workflow/check.py +4 -6
  294. package/pennyfarthing-dist/scripts/workflow/complete-step.py +2 -2
  295. package/pennyfarthing-dist/skills/skill-registry.yaml +19 -0
  296. package/pennyfarthing-dist/workflows/tdd-tandem.yaml +15 -2
  297. package/packages/core/dist/workflow/__test_context_watch__/.session/.tandem-turn-counter +0 -1
  298. package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-session.md +0 -3
  299. package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-tandem-architect.md +0 -6
  300. package/packages/core/dist/workflow/__test_file_watch__/.session/95-4-tandem-architect.md +0 -6
  301. package/packages/core/dist/workflow/__test_file_watch__/workdir/trigger.ts +0 -1
  302. package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-architect.md +0 -6
  303. package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-toolcalls.jsonl +0 -1
  304. package/pennyfarthing-dist/pf/bikerack/changed_panel.py +0 -201
@@ -0,0 +1,115 @@
1
+ /**
2
+ * BackgroundPanel - Display background tasks
3
+ *
4
+ * Story MSSCI-12717 - React Migration
5
+ * Story MSSCI-12784 - Timer updates in real-time with accurate completion times
6
+ *
7
+ * Timer strategy:
8
+ * - On mount: fetch all tasks from backend (accurate snapshot)
9
+ * - Pending tasks: estimate elapsed from startedAt, update via interval
10
+ * - Completed tasks: use authoritative durationMs from backend
11
+ * - IPC events provide real-time updates with accurate timing
12
+ */
13
+
14
+ import React, { useState, useEffect } from 'react';
15
+ import { Button } from '@/components/ui/button';
16
+ import { useBackgroundTasks, BackgroundTask } from '../../hooks/useBackgroundTasks';
17
+
18
+ /**
19
+ * Format milliseconds into human-readable duration
20
+ */
21
+ function formatDuration(ms: number): string {
22
+ if (ms < 1000) return `${ms}ms`;
23
+ if (ms < 60000) return `${Math.floor(ms / 1000)}s`;
24
+ return `${Math.floor(ms / 60000)}m`;
25
+ }
26
+
27
+ function TaskItem({ task, tick }: { task: BackgroundTask; tick: number }): React.ReactElement {
28
+ const isPending = task.status === 'pending';
29
+ const statusIcon = isPending ? '...' : (task.success ? 'v' : 'x');
30
+ const statusClass = `task-item task-${task.status}${task.success === false ? ' task-error' : ''}`;
31
+
32
+ // For completed tasks, use authoritative durationMs from backend
33
+ // For pending tasks, estimate from startedAt (tick triggers re-render)
34
+ let elapsedStr: string;
35
+ if (task.status === 'completed' && task.durationMs !== undefined) {
36
+ // Completed: use accurate duration from backend
37
+ elapsedStr = formatDuration(task.durationMs);
38
+ } else {
39
+ // Pending: estimate elapsed time (tick forces recalculation)
40
+ void tick; // Intentionally used to trigger re-render
41
+ elapsedStr = formatDuration(Date.now() - task.startedAt);
42
+ }
43
+
44
+ return (
45
+ <div className={statusClass} data-testid={`task-${task.taskId}`}>
46
+ <span className="task-status">{statusIcon}</span>
47
+ <div className="task-content">
48
+ <span className="task-description">{task.description}</span>
49
+ <span className="task-type">{task.subagentType}</span>
50
+ </div>
51
+ <span className="task-elapsed">{elapsedStr}</span>
52
+ </div>
53
+ );
54
+ }
55
+
56
+ export function BackgroundPanel(): React.ReactElement {
57
+ const { tasks, pendingCount, completedCount, clearCompleted } = useBackgroundTasks();
58
+ const [tick, setTick] = useState(0);
59
+
60
+ // Timer for live updates - only runs when there are pending tasks and panel is visible
61
+ useEffect(() => {
62
+ if (pendingCount === 0) return;
63
+
64
+ const interval = setInterval(() => {
65
+ setTick(t => t + 1);
66
+ }, 1000);
67
+
68
+ return () => clearInterval(interval);
69
+ }, [pendingCount]);
70
+
71
+ if (tasks.length === 0) {
72
+ return (
73
+ <div className="background-panel empty" data-testid="background-panel">
74
+ <div className="placeholder">No subagent tasks</div>
75
+ </div>
76
+ );
77
+ }
78
+
79
+ // Sort: pending first, then by start time (newest first)
80
+ const sortedTasks = [...tasks].sort((a, b) => {
81
+ if (a.status === 'pending' && b.status !== 'pending') return -1;
82
+ if (a.status !== 'pending' && b.status === 'pending') return 1;
83
+ return b.startedAt - a.startedAt;
84
+ });
85
+
86
+ return (
87
+ <div className="background-panel" data-testid="background-panel">
88
+ <div className="panel-header">
89
+ <span className="task-counts">
90
+ {pendingCount > 0 && <span className="pending-count">{pendingCount} running</span>}
91
+ {completedCount > 0 && <span className="completed-count">{completedCount} done</span>}
92
+ </span>
93
+ {completedCount > 0 && (
94
+ <Button
95
+ variant="ghost"
96
+ size="sm"
97
+ type="button"
98
+ className="clear-button"
99
+ onClick={clearCompleted}
100
+ >
101
+ Clear
102
+ </Button>
103
+ )}
104
+ </div>
105
+
106
+ <div className="task-list">
107
+ {sortedTasks.map(task => (
108
+ <TaskItem key={task.taskId} task={task} tick={tick} />
109
+ ))}
110
+ </div>
111
+ </div>
112
+ );
113
+ }
114
+
115
+ export default BackgroundPanel;
@@ -0,0 +1,214 @@
1
+ /**
2
+ * BikeLanePanel - Display workflow visualization
3
+ *
4
+ * Story MSSCI-12849 - Missing AC & BikeLane panels in Progress tab
5
+ *
6
+ * Reference: Deleted vanilla JS in commit 9aea4f371
7
+ * - js/sidebar/bikelane.js
8
+ */
9
+
10
+ import React from 'react';
11
+ import { Badge } from '@/components/ui/badge';
12
+ import { Skeleton } from '@/components/ui/skeleton';
13
+ import type { WorkflowPhase } from '../../../story-parser.js';
14
+ import { useStory } from '../../hooks/useStory.js';
15
+
16
+ export interface PhaseHistoryEntry {
17
+ phase: string;
18
+ agent: string;
19
+ status: 'done' | 'current' | 'pending';
20
+ duration?: string;
21
+ }
22
+
23
+ export interface BikeLanePanelProps {
24
+ workflowType: string | null;
25
+ phases: WorkflowPhase[] | null;
26
+ phaseHistory?: PhaseHistoryEntry[] | null;
27
+ collapsed?: boolean;
28
+ onToggle?: () => void;
29
+ }
30
+
31
+ /**
32
+ * Format workflow type for display
33
+ * TDD, BDD -> uppercase
34
+ * trivial, others -> Title case
35
+ */
36
+ function formatWorkflowType(type: string | null): string {
37
+ if (!type) return '—';
38
+
39
+ const upperTypes = ['tdd', 'bdd'];
40
+ if (upperTypes.includes(type.toLowerCase())) {
41
+ return type.toUpperCase();
42
+ }
43
+
44
+ return type.charAt(0).toUpperCase() + type.slice(1);
45
+ }
46
+
47
+ /**
48
+ * Get icon for phase status
49
+ */
50
+ function getPhaseIcon(status: 'done' | 'current' | 'pending'): string {
51
+ switch (status) {
52
+ case 'done':
53
+ return '✓';
54
+ case 'current':
55
+ return '●';
56
+ case 'pending':
57
+ default:
58
+ return '○';
59
+ }
60
+ }
61
+
62
+ /**
63
+ * Phase step in progress visualization
64
+ */
65
+ function PhaseStep({ phase, isLast }: { phase: WorkflowPhase; isLast: boolean }): React.ReactElement {
66
+ const icon = getPhaseIcon(phase.status);
67
+ const statusClass = `phase-step ${phase.status}`;
68
+
69
+ return (
70
+ <>
71
+ <div className={statusClass}>
72
+ <span className="phase-icon">{icon}</span>
73
+ <span className="phase-label">{phase.label}</span>
74
+ </div>
75
+ {!isLast && <span className="phase-arrow">→</span>}
76
+ </>
77
+ );
78
+ }
79
+
80
+ /**
81
+ * Phase history item in timeline
82
+ */
83
+ function PhaseHistoryItem({ entry }: { entry: PhaseHistoryEntry }): React.ReactElement {
84
+ const icon = getPhaseIcon(entry.status);
85
+ const statusClass = `phase-history-item ${entry.status}`;
86
+
87
+ const durationText = entry.status === 'current'
88
+ ? 'in progress'
89
+ : entry.status === 'pending'
90
+ ? 'pending'
91
+ : entry.duration || '—';
92
+
93
+ return (
94
+ <div className={statusClass}>
95
+ <span className="history-icon">{icon}</span>
96
+ <span className="history-phase">{entry.phase.toUpperCase()}</span>
97
+ <span className="history-agent">{entry.agent}</span>
98
+ <span className="history-duration">{durationText}</span>
99
+ </div>
100
+ );
101
+ }
102
+
103
+ /**
104
+ * BikeLanePanel - Displays workflow progress and phase history
105
+ */
106
+ export function BikeLanePanel({
107
+ workflowType,
108
+ phases,
109
+ phaseHistory,
110
+ collapsed = false,
111
+ onToggle,
112
+ }: BikeLanePanelProps): React.ReactElement {
113
+ // Handle empty state
114
+ if (!workflowType && (!phases || phases.length === 0)) {
115
+ return (
116
+ <div className="bikelane-panel empty hidden" data-testid="bikelane-panel">
117
+ <div className="placeholder">No active workflow</div>
118
+ </div>
119
+ );
120
+ }
121
+
122
+ const formattedType = formatWorkflowType(workflowType);
123
+
124
+ // Handle collapsed state
125
+ if (collapsed) {
126
+ return (
127
+ <div className="bikelane-panel collapsed" data-testid="bikelane-panel">
128
+ <div className="bikelane-header" onClick={onToggle}>
129
+ <Badge variant="secondary" className="workflow-type-badge" data-workflow-type={workflowType || ''}>
130
+ {formattedType}
131
+ </Badge>
132
+ <span className="bikelane-expand">▶</span>
133
+ </div>
134
+ </div>
135
+ );
136
+ }
137
+
138
+ return (
139
+ <div className="bikelane-panel" data-testid="bikelane-panel">
140
+ <div className="bikelane-header" onClick={onToggle}>
141
+ <Badge variant="secondary" className="workflow-type-badge" data-workflow-type={workflowType || ''}>
142
+ {formattedType}
143
+ </Badge>
144
+ {onToggle && <span className="bikelane-expand">▼</span>}
145
+ </div>
146
+
147
+ {/* Phase progress visualization */}
148
+ {phases && phases.length > 0 && (
149
+ <div className="phase-progress">
150
+ {phases.map((phase, index) => (
151
+ <PhaseStep
152
+ key={phase.name}
153
+ phase={phase}
154
+ isLast={index === phases.length - 1}
155
+ />
156
+ ))}
157
+ </div>
158
+ )}
159
+
160
+ {/* Phase history timeline */}
161
+ {phaseHistory && phaseHistory.length > 0 && (
162
+ <div className="phase-history">
163
+ <div className="phase-history-list">
164
+ {phaseHistory.map((entry, index) => (
165
+ <PhaseHistoryItem key={index} entry={entry} />
166
+ ))}
167
+ </div>
168
+ </div>
169
+ )}
170
+ </div>
171
+ );
172
+ }
173
+
174
+ /**
175
+ * ConnectedBikeLanePanel - Self-contained panel that fetches its own data
176
+ *
177
+ * Used by DockingWorkspace via registerPanelComponent.
178
+ * Subscribes to story updates so it re-renders when session file changes.
179
+ */
180
+ export function ConnectedBikeLanePanel(): React.ReactElement {
181
+ const { story, isLoading, error } = useStory();
182
+
183
+ if (isLoading) {
184
+ return (
185
+ <div className="bikelane-panel loading" data-testid="bikelane-panel">
186
+ <div className="space-y-2 p-2">
187
+ <Skeleton className="h-6 w-16 rounded-full" />
188
+ <div className="flex gap-2 items-center">
189
+ <Skeleton className="h-4 w-12" />
190
+ <Skeleton className="h-4 w-12" />
191
+ <Skeleton className="h-4 w-12" />
192
+ </div>
193
+ </div>
194
+ </div>
195
+ );
196
+ }
197
+
198
+ if (error) {
199
+ return (
200
+ <div className="bikelane-panel error" data-testid="bikelane-panel">
201
+ <div className="error-message">{error.message}</div>
202
+ </div>
203
+ );
204
+ }
205
+
206
+ return (
207
+ <BikeLanePanel
208
+ workflowType={story?.workflow ?? null}
209
+ phases={story?.workflowPhases ?? null}
210
+ />
211
+ );
212
+ }
213
+
214
+ export default BikeLanePanel;
@@ -0,0 +1,344 @@
1
+ /**
2
+ * DebugPanel - Context usage and token stats display
3
+ *
4
+ * Story MSSCI-12717 - React Migration
5
+ * Story MSSCI-12799 - Tier display
6
+ * Story MSSCI-12782 - Token stats formatting
7
+ *
8
+ * Note: Tool call display moved to AuditLogPanel for comprehensive view.
9
+ */
10
+
11
+ import React, { useState, useEffect, useRef, useCallback } from 'react';
12
+ import { Button } from '@/components/ui/button';
13
+ import { Badge } from '@/components/ui/badge';
14
+ import { Separator } from '@/components/ui/separator';
15
+ import { HotspotsDialog } from '../dialogs/HotspotsDialog';
16
+ import { CodeMarkersDialog } from '../dialogs/CodeMarkersDialog';
17
+ import { ComplexityDialog } from '../dialogs/ComplexityDialog';
18
+ import { DependenciesDialog } from '../dialogs/DependenciesDialog';
19
+ import { AgentLoadDialog } from '../AgentLoadDialog';
20
+ import { DeadCodeDialog } from '../DeadCodeDialog';
21
+ import { HealthGauge } from '../HealthGauge';
22
+ import { ContextSparkline, SparklinePoint } from '../ContextSparkline';
23
+ import { useHealthScore } from '../../hooks/useHealthScore';
24
+
25
+ /** Context tier type */
26
+ type ContextTier = 'FULL' | 'REFRESH' | 'HANDOFF' | 'MINIMAL';
27
+
28
+ /** Matches ContextInfo shape from api/context.ts */
29
+ interface ContextData {
30
+ /** Used tokens (from check-context.sh CONTEXT_TOKENS) */
31
+ tokens?: number | null;
32
+ percent?: number | null;
33
+ status?: string | null;
34
+ error?: string | null;
35
+ /** System prompt overhead (first turn tokens) */
36
+ baseline?: number | null;
37
+ /** Tokens used by conversation (total - baseline) */
38
+ usableTokens?: number | null;
39
+ /** Conversation usage as % of available capacity */
40
+ usablePercent?: number | null;
41
+ /** Available capacity (max - baseline) */
42
+ available?: number | null;
43
+ /** Current context tier */
44
+ tier?: ContextTier;
45
+ /** Per-component token counts (MSSCI-12800) */
46
+ tokenCounts?: Record<string, number>;
47
+ /** Total tokens across all injected components (MSSCI-12800) */
48
+ totalTokens?: number;
49
+ }
50
+
51
+ /**
52
+ * Format a component name from snake_case to Title Case
53
+ *
54
+ * Examples:
55
+ * - agent_definition → Agent Definition
56
+ * - behavior_guide → Behavior Guide
57
+ * - persona_compressed → Persona (Compressed)
58
+ * - session_header → Session Header
59
+ *
60
+ * @param name - Component name in snake_case
61
+ * @returns Formatted component name in Title Case
62
+ */
63
+ export function formatComponentName(name: string): string {
64
+ // Handle special case for compressed persona
65
+ if (name === 'persona_compressed') {
66
+ return 'Persona (Compressed)';
67
+ }
68
+
69
+ // Convert snake_case to Title Case
70
+ return name
71
+ .split('_')
72
+ .map(word => word.charAt(0).toUpperCase() + word.slice(1))
73
+ .join(' ');
74
+ }
75
+
76
+ /**
77
+ * Calculate token savings percentage for a given tier vs FULL
78
+ *
79
+ * Token estimates:
80
+ * - FULL: ~4000 tokens (baseline)
81
+ * - REFRESH: ~600 tokens (85% savings)
82
+ * - HANDOFF: ~700 tokens (82% savings)
83
+ * - MINIMAL: ~200 tokens (95% savings)
84
+ */
85
+ export function calculateTierSavings(tier: ContextTier | undefined): number {
86
+ if (!tier) return 0;
87
+
88
+ switch (tier) {
89
+ case 'FULL':
90
+ return 0;
91
+ case 'REFRESH':
92
+ return 85;
93
+ case 'HANDOFF':
94
+ return 82;
95
+ case 'MINIMAL':
96
+ return 95;
97
+ default:
98
+ return 0;
99
+ }
100
+ }
101
+
102
+ export function DebugPanel(): React.ReactElement {
103
+ const [context, setContext] = useState<ContextData | null>(null);
104
+ const [tokenStats, setTokenStats] = useState<Record<string, unknown> | null>(null);
105
+ const [breakdownExpanded, setBreakdownExpanded] = useState(false);
106
+ const [hotspotsOpen, setHotspotsOpen] = useState(false);
107
+ const [codeMarkersOpen, setCodeMarkersOpen] = useState(false);
108
+ const [complexityOpen, setComplexityOpen] = useState(false);
109
+ const [dependenciesOpen, setDependenciesOpen] = useState(false);
110
+ const [agentLoadOpen, setAgentLoadOpen] = useState(false);
111
+ const [deadCodeOpen, setDeadCodeOpen] = useState(false);
112
+ const healthScore = useHealthScore();
113
+ const sparklineRef = useRef<SparklinePoint[]>([]);
114
+ const [sparklineVersion, setSparklineVersion] = useState(0);
115
+
116
+ const pushSparklinePoint = useCallback((percent: number, tokens: number) => {
117
+ const buf = sparklineRef.current;
118
+ buf.push({ percent, tokens, timestamp: Date.now() });
119
+ if (buf.length > 50) buf.shift();
120
+ setSparklineVersion(v => v + 1);
121
+ }, []);
122
+
123
+ const handleDimensionClick = (dimensionName: string) => {
124
+ switch (dimensionName) {
125
+ case 'churn':
126
+ setHotspotsOpen(true);
127
+ break;
128
+ case 'test_gaps':
129
+ // TODO: TestGapsDialog — for now no drill-down
130
+ break;
131
+ case 'todo_density':
132
+ case 'deprecation_debt':
133
+ setCodeMarkersOpen(true);
134
+ break;
135
+ case 'complexity':
136
+ setComplexityOpen(true);
137
+ break;
138
+ case 'dead_code':
139
+ setDeadCodeOpen(true);
140
+ break;
141
+ case 'dependency_freshness':
142
+ setDependenciesOpen(true);
143
+ break;
144
+ case 'agent_context_efficiency':
145
+ setAgentLoadOpen(true);
146
+ break;
147
+ }
148
+ };
149
+
150
+ useEffect(() => {
151
+ const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
152
+
153
+ // Connect to context WebSocket
154
+ const contextWs = new WebSocket(`${protocol}//${window.location.host}/ws/context`);
155
+ contextWs.onmessage = (event) => {
156
+ try {
157
+ const data = JSON.parse(event.data);
158
+ if (data.type === 'init' || data.type === 'update') {
159
+ const ctx = data.context as ContextData;
160
+ setContext(ctx);
161
+ if (ctx.percent != null) {
162
+ pushSparklinePoint(ctx.percent, ctx.tokens ?? 0);
163
+ }
164
+ }
165
+ } catch {
166
+ // Ignore parse errors
167
+ }
168
+ };
169
+
170
+ // Connect to token-stats WebSocket
171
+ const tokenWs = new WebSocket(`${protocol}//${window.location.host}/ws/token-stats`);
172
+ tokenWs.onmessage = (event) => {
173
+ try {
174
+ const data = JSON.parse(event.data);
175
+ setTokenStats(data as Record<string, unknown>);
176
+ } catch {
177
+ // Ignore parse errors
178
+ }
179
+ };
180
+
181
+ return () => {
182
+ contextWs.close();
183
+ tokenWs.close();
184
+ };
185
+ }, [pushSparklinePoint]);
186
+
187
+ // Compute tier-specific CSS class
188
+ const tierClass = context?.tier ? `tier-${context.tier.toLowerCase()}` : '';
189
+ const tierSavings = calculateTierSavings(context?.tier);
190
+
191
+ return (
192
+ <div className="debug-panel" data-testid="debug-panel">
193
+ <HealthGauge
194
+ score={healthScore.data?.composite_score ?? null}
195
+ dimensions={healthScore.data?.dimensions ?? []}
196
+ totalDimensions={8}
197
+ onDimensionClick={handleDimensionClick}
198
+ isLoading={healthScore.isLoading}
199
+ lastFetchedAt={healthScore.lastFetchedAt}
200
+ onRefresh={healthScore.refresh}
201
+ error={healthScore.error}
202
+ />
203
+
204
+ <Separator className="my-3" />
205
+
206
+ <h4>Context Usage</h4>
207
+ {context ? (
208
+ <div className="context-info">
209
+ {context.tier && (
210
+ <div className="tier-display">
211
+ <Badge
212
+ variant="outline"
213
+ className={`tier-badge ${tierClass}`}
214
+ data-testid="tier-badge"
215
+ >
216
+ {context.tier}
217
+ </Badge>
218
+ <span className="tier-savings" data-testid="tier-savings">
219
+ {tierSavings}% savings
220
+ </span>
221
+ </div>
222
+ )}
223
+ {context.tokenCounts && Object.keys(context.tokenCounts).length > 0 && (
224
+ <div className="component-breakdown" data-testid="component-breakdown">
225
+ <div className="breakdown-header">
226
+ <Button
227
+ variant="ghost"
228
+ size="sm"
229
+ className="breakdown-toggle"
230
+ data-testid="breakdown-toggle"
231
+ onClick={() => setBreakdownExpanded(!breakdownExpanded)}
232
+ aria-expanded={breakdownExpanded}
233
+ >
234
+ {breakdownExpanded ? '▼' : '▶'} Injected Context
235
+ </Button>
236
+ <span className="total-tokens" data-testid="total-tokens">
237
+ {context.totalTokens?.toLocaleString()} tokens
238
+ </span>
239
+ </div>
240
+ <div
241
+ className={`component-list ${breakdownExpanded ? 'expanded' : 'collapsed'}`}
242
+ data-testid="component-list"
243
+ aria-expanded={breakdownExpanded}
244
+ >
245
+ {/* Sort components by token count descending, filter out zero values */}
246
+ {Object.entries(context.tokenCounts)
247
+ .filter(([, count]) => count > 0)
248
+ .sort(([, a], [, b]) => b - a)
249
+ .map(([name, count]) => (
250
+ <div
251
+ key={name}
252
+ className="component-item"
253
+ data-testid={`component-${name}`}
254
+ >
255
+ <span className="component-name">
256
+ {formatComponentName(name)}
257
+ </span>
258
+ <span className="component-tokens">
259
+ {count.toLocaleString()}
260
+ </span>
261
+ </div>
262
+ ))}
263
+ </div>
264
+ </div>
265
+ )}
266
+ <div className="context-bar">
267
+ <div
268
+ className="context-fill"
269
+ style={{ width: `${context.percent || 0}%` }}
270
+ />
271
+ </div>
272
+ <ContextSparkline history={sparklineRef.current} key={sparklineVersion} />
273
+ <span className="context-text">
274
+ {(context.tokens ?? 0).toLocaleString()} / {context.baseline != null && context.available != null
275
+ ? (context.baseline + context.available).toLocaleString()
276
+ : '—'} tokens
277
+ ({context.percent || 0}%)
278
+ </span>
279
+ {context.baseline != null && (
280
+ <dl className="context-breakdown">
281
+ <dt>System Prompt</dt>
282
+ <dd>{context.baseline.toLocaleString()} tokens</dd>
283
+ <dt>Conversation</dt>
284
+ <dd>{context.usableTokens?.toLocaleString() ?? '—'} tokens</dd>
285
+ <dt>Available</dt>
286
+ <dd>{context.available?.toLocaleString() ?? '—'} tokens</dd>
287
+ </dl>
288
+ )}
289
+ </div>
290
+ ) : (
291
+ <div className="placeholder">No context data</div>
292
+ )}
293
+
294
+ <Separator className="my-3" />
295
+
296
+ <h4>Token Stats</h4>
297
+ {tokenStats ? (
298
+ <dl className="token-stats">
299
+ {tokenStats.inputTokens !== undefined && (
300
+ <div className="token-stat-card" data-testid="token-stat-input">
301
+ <dt>Input</dt>
302
+ <dd>{Number(tokenStats.inputTokens).toLocaleString()}</dd>
303
+ </div>
304
+ )}
305
+ {tokenStats.outputTokens !== undefined && (
306
+ <div className="token-stat-card" data-testid="token-stat-output">
307
+ <dt>Output</dt>
308
+ <dd>{Number(tokenStats.outputTokens).toLocaleString()}</dd>
309
+ </div>
310
+ )}
311
+ {tokenStats.cacheReadTokens !== undefined && (
312
+ <div className="token-stat-card" data-testid="token-stat-cache-read">
313
+ <dt>Cache Read</dt>
314
+ <dd>{Number(tokenStats.cacheReadTokens).toLocaleString()}</dd>
315
+ </div>
316
+ )}
317
+ {tokenStats.cacheCreationTokens !== undefined && (
318
+ <div className="token-stat-card" data-testid="token-stat-cache-write">
319
+ <dt>Cache Write</dt>
320
+ <dd>{Number(tokenStats.cacheCreationTokens).toLocaleString()}</dd>
321
+ </div>
322
+ )}
323
+ {tokenStats.totalCostUsd !== undefined && Number(tokenStats.totalCostUsd) > 0 && (
324
+ <div className="token-stat-card" data-testid="token-stat-cost">
325
+ <dt>Cost</dt>
326
+ <dd>${Number(tokenStats.totalCostUsd).toFixed(4)}</dd>
327
+ </div>
328
+ )}
329
+ </dl>
330
+ ) : (
331
+ <div className="placeholder">No token stats</div>
332
+ )}
333
+
334
+ <HotspotsDialog open={hotspotsOpen} onOpenChange={setHotspotsOpen} />
335
+ <CodeMarkersDialog open={codeMarkersOpen} onOpenChange={setCodeMarkersOpen} />
336
+ <ComplexityDialog open={complexityOpen} onOpenChange={setComplexityOpen} />
337
+ <DependenciesDialog open={dependenciesOpen} onOpenChange={setDependenciesOpen} />
338
+ <AgentLoadDialog isOpen={agentLoadOpen} onClose={() => setAgentLoadOpen(false)} />
339
+ <DeadCodeDialog isOpen={deadCodeOpen} onClose={() => setDeadCodeOpen(false)} />
340
+ </div>
341
+ );
342
+ }
343
+
344
+ export default DebugPanel;