@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,120 @@
1
+ import React, { useEffect } from 'react';
2
+ import { Badge } from '@/components/ui/badge';
3
+ import { Button } from '@/components/ui/button';
4
+ import { Skeleton } from '@/components/ui/skeleton';
5
+ import { ToolDialog } from './ToolDialog';
6
+ import { useDependencies } from '../../hooks/useDependencies';
7
+
8
+ export interface DependenciesDialogProps {
9
+ open: boolean;
10
+ onOpenChange: (open: boolean) => void;
11
+ }
12
+
13
+ function severityVariant(severity: string): 'destructive' | 'outline' | 'secondary' {
14
+ if (severity === 'high' || severity === 'critical') return 'destructive';
15
+ if (severity === 'moderate') return 'outline';
16
+ return 'secondary';
17
+ }
18
+
19
+ export function DependenciesDialog({ open, onOpenChange }: DependenciesDialogProps): React.ReactElement {
20
+ const { data, isLoading, error, refresh } = useDependencies({});
21
+
22
+ useEffect(() => {
23
+ if (open) refresh();
24
+ }, [open, refresh]);
25
+
26
+ const renderContent = () => {
27
+ if (isLoading) {
28
+ return (
29
+ <div className="dependencies-panel loading" data-testid="dependencies-panel">
30
+ <div className="space-y-3 p-2">
31
+ <Skeleton className="h-4 w-40" />
32
+ <Skeleton className="h-3 w-full" />
33
+ <Skeleton className="h-3 w-full" />
34
+ <Skeleton className="h-3 w-3/4" />
35
+ </div>
36
+ </div>
37
+ );
38
+ }
39
+
40
+ if (error) {
41
+ return (
42
+ <div className="dependencies-panel error" data-testid="dependencies-panel">
43
+ <div className="error-message">Error: {error.message}</div>
44
+ <Button variant="outline" size="sm" onClick={refresh}>Retry</Button>
45
+ </div>
46
+ );
47
+ }
48
+
49
+ if (!data) {
50
+ return (
51
+ <div className="dependencies-panel" data-testid="dependencies-panel">
52
+ <p>Click <strong>Analyze</strong> to check dependencies</p>
53
+ </div>
54
+ );
55
+ }
56
+
57
+ return (
58
+ <div className="dependencies-panel" data-testid="dependencies-panel">
59
+ {data.outdated.length > 0 && (
60
+ <div className="outdated-section" style={{ marginBottom: '16px' }}>
61
+ <h4>Outdated Packages ({data.outdated.length})</h4>
62
+ <table role="table" style={{ width: '100%', borderCollapse: 'collapse' }}>
63
+ <thead>
64
+ <tr>
65
+ <th className="text-left" style={{ padding: '4px 8px' }}>Package</th>
66
+ <th className="text-left" style={{ padding: '4px 8px' }}>Current</th>
67
+ <th className="text-left" style={{ padding: '4px 8px' }}>Wanted</th>
68
+ <th className="text-left" style={{ padding: '4px 8px' }}>Latest</th>
69
+ <th className="text-left" style={{ padding: '4px 8px' }}>Type</th>
70
+ </tr>
71
+ </thead>
72
+ <tbody>
73
+ {data.outdated.map((pkg) => (
74
+ <tr key={pkg.name}>
75
+ <td style={{ padding: '4px 8px' }}>{pkg.name}</td>
76
+ <td style={{ padding: '4px 8px' }}>{pkg.current}</td>
77
+ <td style={{ padding: '4px 8px' }}>{pkg.wanted}</td>
78
+ <td style={{ padding: '4px 8px' }}>{pkg.latest}</td>
79
+ <td style={{ padding: '4px 8px' }}>{pkg.type}</td>
80
+ </tr>
81
+ ))}
82
+ </tbody>
83
+ </table>
84
+ </div>
85
+ )}
86
+
87
+ {data.advisories.length > 0 && (
88
+ <div className="security-section">
89
+ <h4>Security Advisories</h4>
90
+ <div style={{ display: 'flex', gap: '8px', flexWrap: 'wrap' }}>
91
+ {data.advisories.map((adv) => (
92
+ <div key={adv.severity} style={{ display: 'flex', alignItems: 'center', gap: '4px' }}>
93
+ <Badge variant={severityVariant(adv.severity)}>
94
+ {adv.severity}
95
+ </Badge>
96
+ <span>{adv.count}</span>
97
+ </div>
98
+ ))}
99
+ </div>
100
+ </div>
101
+ )}
102
+
103
+ {data.outdated.length === 0 && data.advisories.length === 0 && (
104
+ <p>All dependencies are up to date with no known vulnerabilities.</p>
105
+ )}
106
+ </div>
107
+ );
108
+ };
109
+
110
+ return (
111
+ <ToolDialog
112
+ open={open}
113
+ onOpenChange={onOpenChange}
114
+ title="Dependencies"
115
+ description="Package staleness and security analysis"
116
+ >
117
+ {renderContent()}
118
+ </ToolDialog>
119
+ );
120
+ }
@@ -0,0 +1,451 @@
1
+ /**
2
+ * HotspotsDialog - Git history hotspot detector in a dialog
3
+ *
4
+ * Story: MSSCI-14442 - Migrate HotspotsPanel into HotspotsDialog
5
+ * Epic: epic-79 (Dialog Infrastructure + Hotspot Refactor)
6
+ *
7
+ * Migrated from HotspotsPanel — uses ToolDialog wrapper from 79-1.
8
+ * Shows files and directories with highest change frequency, bug fix
9
+ * concentration, and multi-author churn. Sortable table with time window controls.
10
+ */
11
+
12
+ import React, { useState, useMemo, useCallback } from 'react';
13
+ import { Button } from '@/components/ui/button';
14
+ import { Badge } from '@/components/ui/badge';
15
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
16
+ import { Skeleton } from '@/components/ui/skeleton';
17
+ import { ToolDialog } from './ToolDialog';
18
+ import { useHotspots, FileHotspot, DirectoryHotspot, HotspotRepoResult } from '../../hooks/useHotspots';
19
+
20
+ export interface HotspotsDialogProps {
21
+ open: boolean;
22
+ onOpenChange: (open: boolean) => void;
23
+ }
24
+
25
+ type SortField = 'hotspot_score' | 'change_count' | 'bug_fix_count' | 'author_count' | 'churn' | 'path';
26
+ type SortDirection = 'asc' | 'desc';
27
+ type ViewMode = 'files' | 'dirs';
28
+
29
+ const TIME_WINDOWS = [30, 60, 90] as const;
30
+
31
+ const SOURCE_EXTENSIONS = new Set([
32
+ '.ts', '.tsx', '.js', '.jsx', '.py', '.md', '.css', '.scss', '.html',
33
+ ]);
34
+
35
+ const CONFIG_EXTENSIONS = new Set([
36
+ '.json', '.yaml', '.yml', '.toml', '.env',
37
+ ]);
38
+
39
+ function getExtension(path: string): string {
40
+ const basename = path.split('/').pop() || '';
41
+ // Handle dotfiles like .env
42
+ if (basename.startsWith('.') && !basename.includes('.', 1)) {
43
+ return '.' + basename.slice(1);
44
+ }
45
+ const dotIndex = basename.lastIndexOf('.');
46
+ return dotIndex > 0 ? basename.slice(dotIndex) : '';
47
+ }
48
+
49
+ function matchesFilter(path: string, codeOnly: boolean, includeConfig: boolean): boolean {
50
+ if (!codeOnly) return true;
51
+ const ext = getExtension(path);
52
+ if (SOURCE_EXTENSIONS.has(ext)) return true;
53
+ if (includeConfig && CONFIG_EXTENSIONS.has(ext)) return true;
54
+ return false;
55
+ }
56
+
57
+ function SortableHeader({
58
+ label,
59
+ field,
60
+ currentSort,
61
+ currentDirection,
62
+ onSort,
63
+ align = 'right',
64
+ }: {
65
+ label: string;
66
+ field: SortField;
67
+ currentSort: SortField;
68
+ currentDirection: SortDirection;
69
+ onSort: (field: SortField) => void;
70
+ align?: 'left' | 'right';
71
+ }) {
72
+ const isActive = currentSort === field;
73
+ const arrow = isActive ? (currentDirection === 'desc' ? ' v' : ' ^') : '';
74
+
75
+ return (
76
+ <th
77
+ className={`text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2 cursor-pointer select-none ${align === 'left' ? 'text-left' : 'text-right'} ${isActive ? 'text-[var(--text-primary)]' : ''}`}
78
+ onClick={() => onSort(field)}
79
+ role="columnheader"
80
+ aria-sort={isActive ? (currentDirection === 'desc' ? 'descending' : 'ascending') : 'none'}
81
+ >
82
+ {label}{arrow}
83
+ </th>
84
+ );
85
+ }
86
+
87
+ function FileTable({
88
+ hotspots,
89
+ sortField,
90
+ sortDirection,
91
+ onSort,
92
+ }: {
93
+ hotspots: FileHotspot[];
94
+ sortField: SortField;
95
+ sortDirection: SortDirection;
96
+ onSort: (field: SortField) => void;
97
+ }) {
98
+ const sorted = useMemo(() => {
99
+ const items = [...hotspots];
100
+ items.sort((a, b) => {
101
+ const aVal = a[sortField as keyof FileHotspot];
102
+ const bVal = b[sortField as keyof FileHotspot];
103
+ if (typeof aVal === 'number' && typeof bVal === 'number') {
104
+ return sortDirection === 'desc' ? bVal - aVal : aVal - bVal;
105
+ }
106
+ const aStr = String(aVal);
107
+ const bStr = String(bVal);
108
+ return sortDirection === 'desc' ? bStr.localeCompare(aStr) : aStr.localeCompare(bStr);
109
+ });
110
+ return items;
111
+ }, [hotspots, sortField, sortDirection]);
112
+
113
+ return (
114
+ <table className="w-full text-sm" role="table">
115
+ <thead>
116
+ <tr className="border-b border-[var(--border)]">
117
+ <SortableHeader label="Score" field="hotspot_score" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
118
+ <SortableHeader label="Changes" field="change_count" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
119
+ <SortableHeader label="Fixes" field="bug_fix_count" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
120
+ <SortableHeader label="Authors" field="author_count" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
121
+ <SortableHeader label="Churn" field="churn" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
122
+ <SortableHeader label="File" field="path" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} align="left" />
123
+ </tr>
124
+ </thead>
125
+ <tbody>
126
+ {sorted.map((h) => (
127
+ <tr key={h.path} className="text-[var(--text-primary)]">
128
+ <td className="text-right py-1.5">
129
+ <Badge variant={h.hotspot_score >= 50 ? 'destructive' : h.hotspot_score >= 25 ? 'outline' : 'secondary'}>
130
+ {h.hotspot_score.toFixed(1)}
131
+ </Badge>
132
+ </td>
133
+ <td className="text-right py-1.5 tabular-nums font-mono">{h.change_count}</td>
134
+ <td className="text-right py-1.5 tabular-nums font-mono">{h.bug_fix_count}</td>
135
+ <td className="text-right py-1.5 tabular-nums font-mono">{h.author_count}</td>
136
+ <td className="text-right py-1.5 tabular-nums font-mono">{h.churn}</td>
137
+ <td className="text-left py-1.5">
138
+ <Tooltip>
139
+ <TooltipTrigger asChild>
140
+ <span className="truncate max-w-xs inline-block align-bottom font-mono text-xs">{h.path}</span>
141
+ </TooltipTrigger>
142
+ <TooltipContent>{h.path}</TooltipContent>
143
+ </Tooltip>
144
+ </td>
145
+ </tr>
146
+ ))}
147
+ </tbody>
148
+ </table>
149
+ );
150
+ }
151
+
152
+ function DirTable({
153
+ hotspots,
154
+ sortField,
155
+ sortDirection,
156
+ onSort,
157
+ }: {
158
+ hotspots: DirectoryHotspot[];
159
+ sortField: SortField;
160
+ sortDirection: SortDirection;
161
+ onSort: (field: SortField) => void;
162
+ }) {
163
+ const sorted = useMemo(() => {
164
+ const items = [...hotspots];
165
+ items.sort((a, b) => {
166
+ const fieldMap: Record<string, keyof DirectoryHotspot> = {
167
+ change_count: 'total_changes',
168
+ bug_fix_count: 'total_bug_fixes',
169
+ author_count: 'avg_author_count',
170
+ churn: 'file_count',
171
+ };
172
+ const key = (fieldMap[sortField] || sortField) as keyof DirectoryHotspot;
173
+ const aVal = a[key];
174
+ const bVal = b[key];
175
+ if (typeof aVal === 'number' && typeof bVal === 'number') {
176
+ return sortDirection === 'desc' ? bVal - aVal : aVal - bVal;
177
+ }
178
+ return sortDirection === 'desc'
179
+ ? String(bVal).localeCompare(String(aVal))
180
+ : String(aVal).localeCompare(String(bVal));
181
+ });
182
+ return items;
183
+ }, [hotspots, sortField, sortDirection]);
184
+
185
+ return (
186
+ <table className="w-full text-sm" role="table">
187
+ <thead>
188
+ <tr className="border-b border-[var(--border)]">
189
+ <SortableHeader label="Score" field="hotspot_score" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
190
+ <SortableHeader label="Changes" field="change_count" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
191
+ <SortableHeader label="Fixes" field="bug_fix_count" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
192
+ <SortableHeader label="Authors" field="author_count" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
193
+ <SortableHeader label="Files" field="churn" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} />
194
+ <SortableHeader label="Directory" field="path" currentSort={sortField} currentDirection={sortDirection} onSort={onSort} align="left" />
195
+ </tr>
196
+ </thead>
197
+ <tbody>
198
+ {sorted.map((d) => (
199
+ <tr key={d.path} className="text-[var(--text-primary)]">
200
+ <td className="text-right py-1.5">
201
+ <Badge variant={d.hotspot_score >= 50 ? 'destructive' : d.hotspot_score >= 25 ? 'outline' : 'secondary'}>
202
+ {d.hotspot_score.toFixed(1)}
203
+ </Badge>
204
+ </td>
205
+ <td className="text-right py-1.5 tabular-nums font-mono">{d.total_changes}</td>
206
+ <td className="text-right py-1.5 tabular-nums font-mono">{d.total_bug_fixes}</td>
207
+ <td className="text-right py-1.5 tabular-nums font-mono">{d.avg_author_count.toFixed(1)}</td>
208
+ <td className="text-right py-1.5 tabular-nums font-mono">{d.file_count}</td>
209
+ <td className="text-left py-1.5">
210
+ <Tooltip>
211
+ <TooltipTrigger asChild>
212
+ <span className="truncate max-w-xs inline-block align-bottom font-mono text-xs">{d.path}</span>
213
+ </TooltipTrigger>
214
+ <TooltipContent>{d.path}</TooltipContent>
215
+ </Tooltip>
216
+ </td>
217
+ </tr>
218
+ ))}
219
+ </tbody>
220
+ </table>
221
+ );
222
+ }
223
+
224
+ export function HotspotsDialog({ open, onOpenChange }: HotspotsDialogProps): React.ReactElement {
225
+ const [days, setDays] = useState<number>(90);
226
+ const [viewMode, setViewMode] = useState<ViewMode>('files');
227
+ const [sortField, setSortField] = useState<SortField>('hotspot_score');
228
+ const [sortDirection, setSortDirection] = useState<SortDirection>('desc');
229
+ const [includeOrchestrator, setIncludeOrchestrator] = useState(false);
230
+ const [codeOnly, setCodeOnly] = useState(false);
231
+ const [includeConfig, setIncludeConfig] = useState(false);
232
+
233
+ const { data, isLoading, error, refresh } = useHotspots({ days, includeOrchestrator });
234
+
235
+ const handleSort = useCallback((field: SortField) => {
236
+ setSortField((prev) => {
237
+ if (prev === field) {
238
+ setSortDirection((d) => (d === 'desc' ? 'asc' : 'desc'));
239
+ return prev;
240
+ }
241
+ setSortDirection('desc');
242
+ return field;
243
+ });
244
+ }, []);
245
+
246
+ const repoResults: HotspotRepoResult[] = useMemo(() => {
247
+ if (!data) return [];
248
+ if (data.repo_results) return data.repo_results;
249
+ if (data.file_hotspots) {
250
+ return [{
251
+ success: data.success,
252
+ repo_name: data.repo_name || '',
253
+ repo_path: data.repo_path || '',
254
+ time_window_days: data.time_window_days || days,
255
+ commit_count: data.commit_count || 0,
256
+ file_hotspots: data.file_hotspots || [],
257
+ directory_hotspots: data.directory_hotspots || [],
258
+ }];
259
+ }
260
+ return [];
261
+ }, [data, days]);
262
+
263
+ const allFiles = useMemo(() => {
264
+ const files: FileHotspot[] = [];
265
+ for (const r of repoResults) {
266
+ if (r.success) files.push(...r.file_hotspots);
267
+ }
268
+ return files;
269
+ }, [repoResults]);
270
+
271
+ const allDirs = useMemo(() => {
272
+ const dirs: DirectoryHotspot[] = [];
273
+ for (const r of repoResults) {
274
+ if (r.success) dirs.push(...r.directory_hotspots);
275
+ }
276
+ return dirs;
277
+ }, [repoResults]);
278
+
279
+ const filteredFiles = useMemo(
280
+ () => allFiles.filter((f) => matchesFilter(f.path, codeOnly, includeConfig)),
281
+ [allFiles, codeOnly, includeConfig],
282
+ );
283
+
284
+ const filteredDirs = useMemo(
285
+ () => allDirs.filter((d) => matchesFilter(d.path, codeOnly, includeConfig)),
286
+ [allDirs, codeOnly, includeConfig],
287
+ );
288
+
289
+ const totalCommits = repoResults.reduce((sum, r) => sum + (r.commit_count || 0), 0);
290
+
291
+ const renderContent = () => {
292
+ if (isLoading) {
293
+ return (
294
+ <div className="hotspots-panel loading" data-testid="hotspots-panel">
295
+ <div className="space-y-3 p-2">
296
+ <Skeleton className="h-4 w-40" />
297
+ <Skeleton className="h-3 w-full" />
298
+ <Skeleton className="h-3 w-full" />
299
+ <Skeleton className="h-3 w-3/4" />
300
+ <Skeleton className="h-3 w-full" />
301
+ <Skeleton className="h-3 w-5/6" />
302
+ </div>
303
+ </div>
304
+ );
305
+ }
306
+
307
+ if (error) {
308
+ return (
309
+ <div className="hotspots-panel error" data-testid="hotspots-panel">
310
+ <div className="error-message">{error.message}</div>
311
+ <Button variant="outline" size="sm" onClick={refresh}>Retry</Button>
312
+ </div>
313
+ );
314
+ }
315
+
316
+ return (
317
+ <TooltipProvider delayDuration={300}>
318
+ <div className="hotspots-panel" data-testid="hotspots-panel">
319
+ <div className="space-y-3">
320
+ <div className="flex flex-wrap items-center gap-x-4 gap-y-2">
321
+ <div className="flex gap-1">
322
+ {TIME_WINDOWS.map((w) => (
323
+ <Button
324
+ key={w}
325
+ variant={days === w ? 'default' : 'outline'}
326
+ size="sm"
327
+ onClick={() => setDays(w)}
328
+ >
329
+ {w}d
330
+ </Button>
331
+ ))}
332
+ </div>
333
+
334
+ <div className="flex gap-1">
335
+ <Button
336
+ variant={viewMode === 'files' ? 'default' : 'outline'}
337
+ size="sm"
338
+ onClick={() => setViewMode('files')}
339
+ >
340
+ Files
341
+ </Button>
342
+ <Button
343
+ variant={viewMode === 'dirs' ? 'default' : 'outline'}
344
+ size="sm"
345
+ onClick={() => setViewMode('dirs')}
346
+ >
347
+ Dirs
348
+ </Button>
349
+ </div>
350
+
351
+ <div className="ml-auto">
352
+ <Tooltip>
353
+ <TooltipTrigger asChild>
354
+ <Button variant="outline" size="sm" onClick={refresh}>
355
+ Analyze
356
+ </Button>
357
+ </TooltipTrigger>
358
+ <TooltipContent>Run hotspot analysis</TooltipContent>
359
+ </Tooltip>
360
+ </div>
361
+ </div>
362
+
363
+ <div className="flex flex-wrap items-center gap-x-6 gap-y-2">
364
+ <label className="flex items-center gap-2 text-sm text-[var(--text-secondary)] cursor-pointer">
365
+ <input
366
+ type="checkbox"
367
+ checked={includeOrchestrator}
368
+ onChange={(e) => setIncludeOrchestrator(e.target.checked)}
369
+ />
370
+ Include orchestrator
371
+ </label>
372
+
373
+ <label className="flex items-center gap-2 text-sm text-[var(--text-secondary)] cursor-pointer">
374
+ <input
375
+ type="checkbox"
376
+ aria-label="Code only"
377
+ checked={codeOnly}
378
+ onChange={(e) => setCodeOnly(e.target.checked)}
379
+ />
380
+ Code only
381
+ </label>
382
+
383
+ <label className="flex items-center gap-2 text-sm text-[var(--text-secondary)] cursor-pointer">
384
+ <input
385
+ type="checkbox"
386
+ aria-label="Include config"
387
+ checked={includeConfig}
388
+ disabled={!codeOnly}
389
+ onChange={(e) => setIncludeConfig(e.target.checked)}
390
+ className="disabled:opacity-40"
391
+ />
392
+ <span className={!codeOnly ? 'opacity-40' : ''}>Include config</span>
393
+ </label>
394
+ </div>
395
+ </div>
396
+
397
+ {data && (
398
+ <div className="flex gap-4 text-xs text-[var(--text-muted)]">
399
+ <span>{totalCommits} commits</span>
400
+ <span>{filteredFiles.length} files</span>
401
+ <span>{filteredDirs.length} dirs</span>
402
+ </div>
403
+ )}
404
+
405
+ {!data && (
406
+ <div className="text-center py-12 text-[var(--text-muted)]">
407
+ Click <strong>Analyze</strong> to detect code hotspots
408
+ </div>
409
+ )}
410
+
411
+ {data && viewMode === 'files' && (
412
+ filteredFiles.length > 0 ? (
413
+ <FileTable
414
+ hotspots={filteredFiles.slice(0, 50)}
415
+ sortField={sortField}
416
+ sortDirection={sortDirection}
417
+ onSort={handleSort}
418
+ />
419
+ ) : (
420
+ <div className="text-center py-12 text-[var(--text-muted)]">No file hotspots found</div>
421
+ )
422
+ )}
423
+
424
+ {data && viewMode === 'dirs' && (
425
+ filteredDirs.length > 0 ? (
426
+ <DirTable
427
+ hotspots={filteredDirs.slice(0, 50)}
428
+ sortField={sortField}
429
+ sortDirection={sortDirection}
430
+ onSort={handleSort}
431
+ />
432
+ ) : (
433
+ <div className="text-center py-12 text-[var(--text-muted)]">No directory hotspots found</div>
434
+ )
435
+ )}
436
+ </div>
437
+ </TooltipProvider>
438
+ );
439
+ };
440
+
441
+ return (
442
+ <ToolDialog
443
+ open={open}
444
+ onOpenChange={onOpenChange}
445
+ title="Hotspots"
446
+ description="Files and directories ranked by change frequency and complexity"
447
+ >
448
+ {renderContent()}
449
+ </ToolDialog>
450
+ );
451
+ }
@@ -0,0 +1,43 @@
1
+ import React from 'react';
2
+ import {
3
+ Dialog,
4
+ DialogContent,
5
+ DialogHeader,
6
+ DialogFooter,
7
+ DialogTitle,
8
+ DialogDescription,
9
+ } from '@/components/ui/dialog';
10
+ import { cn } from '@/lib/utils';
11
+
12
+ export interface ToolDialogProps {
13
+ open: boolean;
14
+ onOpenChange: (open: boolean) => void;
15
+ title: string;
16
+ description?: string;
17
+ footer?: React.ReactNode;
18
+ className?: string;
19
+ children: React.ReactNode;
20
+ }
21
+
22
+ export function ToolDialog({
23
+ open,
24
+ onOpenChange,
25
+ title,
26
+ description,
27
+ footer,
28
+ className,
29
+ children,
30
+ }: ToolDialogProps): React.ReactElement {
31
+ return (
32
+ <Dialog open={open} onOpenChange={onOpenChange}>
33
+ <DialogContent className={cn('max-w-5xl max-h-[80vh] overflow-y-auto', className)}>
34
+ <DialogHeader>
35
+ <DialogTitle>{title}</DialogTitle>
36
+ {description && <DialogDescription>{description}</DialogDescription>}
37
+ </DialogHeader>
38
+ {children}
39
+ {footer && <DialogFooter>{footer}</DialogFooter>}
40
+ </DialogContent>
41
+ </Dialog>
42
+ );
43
+ }
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Shared panel component registry
3
+ *
4
+ * Used by both DockviewWorkspace and BikeRackWorkspace to look up
5
+ * registered panel components by ID. Extracted to avoid importing
6
+ * DockviewWorkspace (which has shadcn/ui deps) from BikeRackWorkspace.
7
+ */
8
+
9
+ import type { ComponentType } from 'react';
10
+
11
+ export type PanelComponent = ComponentType;
12
+
13
+ export const panelRegistry: Map<string, PanelComponent> = new Map();