@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,181 @@
1
+ import React, { useState, useEffect } from 'react';
2
+ import { Button } from '@/components/ui/button';
3
+
4
+ export interface HealthGaugeDimension {
5
+ name: string;
6
+ score: number | null;
7
+ weight: number;
8
+ }
9
+
10
+ export interface HealthGaugeProps {
11
+ score: number | null;
12
+ dimensions: HealthGaugeDimension[];
13
+ totalDimensions?: number;
14
+ onDimensionClick?: (dimensionName: string) => void;
15
+ isLoading?: boolean;
16
+ lastFetchedAt?: number | null;
17
+ onRefresh?: () => void;
18
+ error?: Error | null;
19
+ }
20
+
21
+ const GRADE_BANDS: { min: number; grade: string; color: string }[] = [
22
+ { min: 90, grade: 'A', color: '#22c55e' },
23
+ { min: 75, grade: 'B', color: '#84cc16' },
24
+ { min: 60, grade: 'C', color: '#eab308' },
25
+ { min: 40, grade: 'D', color: '#f97316' },
26
+ { min: 0, grade: 'F', color: '#ef4444' },
27
+ ];
28
+
29
+ const DIMENSION_LABELS: Record<string, string> = {
30
+ churn: 'Churn',
31
+ todo_density: 'TODO Density',
32
+ complexity: 'Complexity',
33
+ test_gaps: 'Test Gaps',
34
+ dead_code: 'Dead Code',
35
+ deprecation_debt: 'Deprecation Debt',
36
+ dependency_freshness: 'Dependency Freshness',
37
+ agent_context_efficiency: 'Agent Context Efficiency',
38
+ };
39
+
40
+ function getGrade(score: number): { grade: string; color: string } {
41
+ for (const band of GRADE_BANDS) {
42
+ if (score >= band.min) {
43
+ return { grade: band.grade, color: band.color };
44
+ }
45
+ }
46
+ return { grade: 'F', color: '#ef4444' };
47
+ }
48
+
49
+ // SVG arc helper for a semicircle gauge
50
+ function describeArc(cx: number, cy: number, r: number, startAngle: number, endAngle: number): string {
51
+ const start = polarToCartesian(cx, cy, r, endAngle);
52
+ const end = polarToCartesian(cx, cy, r, startAngle);
53
+ const largeArc = endAngle - startAngle <= 180 ? '0' : '1';
54
+ return `M ${start.x} ${start.y} A ${r} ${r} 0 ${largeArc} 0 ${end.x} ${end.y}`;
55
+ }
56
+
57
+ function polarToCartesian(cx: number, cy: number, r: number, angleDeg: number) {
58
+ const rad = ((angleDeg - 90) * Math.PI) / 180;
59
+ return {
60
+ x: cx + r * Math.cos(rad),
61
+ y: cy + r * Math.sin(rad),
62
+ };
63
+ }
64
+
65
+ function formatAge(ms: number): string {
66
+ const seconds = Math.floor(ms / 1000);
67
+ if (seconds < 60) return `${seconds}s ago`;
68
+ const minutes = Math.floor(seconds / 60);
69
+ if (minutes < 60) return `${minutes}m ago`;
70
+ const hours = Math.floor(minutes / 60);
71
+ return `${hours}h ago`;
72
+ }
73
+
74
+ export function HealthGauge({ score, dimensions, totalDimensions, onDimensionClick, isLoading, lastFetchedAt, onRefresh, error }: HealthGaugeProps): React.ReactElement {
75
+ const hasData = score !== null && score !== undefined;
76
+ const gradeInfo = hasData ? getGrade(score) : null;
77
+ const fillAngle = hasData ? (score / 100) * 180 : 0;
78
+
79
+ // Live-updating age display
80
+ const [ageText, setAgeText] = useState<string | null>(null);
81
+ useEffect(() => {
82
+ if (!lastFetchedAt) {
83
+ setAgeText(null);
84
+ return;
85
+ }
86
+ const tick = () => setAgeText(formatAge(Date.now() - lastFetchedAt));
87
+ tick();
88
+ const id = setInterval(tick, 10_000);
89
+ return () => clearInterval(id);
90
+ }, [lastFetchedAt]);
91
+
92
+ // Use all 8 dimension keys so rows always render (even before data arrives)
93
+ const allDimKeys = Object.keys(DIMENSION_LABELS);
94
+ const dimMap = new Map(dimensions.map((d) => [d.name, d]));
95
+
96
+ return (
97
+ <div
98
+ data-testid="health-gauge"
99
+ data-grade={gradeInfo?.grade ?? null}
100
+ >
101
+ <div className="health-gauge-header">
102
+ <div className="health-gauge-status">
103
+ {ageText && <span className="health-gauge-age" data-testid="health-gauge-age">{ageText}</span>}
104
+ {error && <span className="health-gauge-error" data-testid="health-gauge-error">Failed</span>}
105
+ </div>
106
+ {onRefresh && (
107
+ <Button
108
+ variant="outline"
109
+ size="sm"
110
+ className="health-gauge-refresh"
111
+ data-testid="health-gauge-refresh"
112
+ onClick={onRefresh}
113
+ disabled={isLoading}
114
+ >
115
+ {isLoading ? 'Analyzing...' : hasData ? 'Refresh' : 'Analyze'}
116
+ </Button>
117
+ )}
118
+ </div>
119
+
120
+ <svg viewBox="0 0 200 120" width="200" height="120" className={isLoading ? 'opacity-50' : ''}>
121
+ {/* Background arc (grey) */}
122
+ <path
123
+ d={describeArc(100, 100, 80, 0, 180)}
124
+ fill="none"
125
+ stroke="#333"
126
+ strokeWidth="12"
127
+ strokeLinecap="round"
128
+ />
129
+ {/* Fill arc (colored by grade) */}
130
+ {hasData && fillAngle > 0 && (
131
+ <path
132
+ d={describeArc(100, 100, 80, 0, fillAngle)}
133
+ fill="none"
134
+ stroke={gradeInfo!.color}
135
+ strokeWidth="12"
136
+ strokeLinecap="round"
137
+ />
138
+ )}
139
+ {/* Score text */}
140
+ <text x="100" y="85" textAnchor="middle" fontSize="28" fill="currentColor">
141
+ {hasData ? String(Math.round(score)) : '--'}
142
+ </text>
143
+ {/* Grade letter */}
144
+ {gradeInfo && (
145
+ <text x="100" y="108" textAnchor="middle" fontSize="16" fill={gradeInfo.color}>
146
+ {gradeInfo.grade}
147
+ </text>
148
+ )}
149
+ </svg>
150
+
151
+ {/* Dimension count for partial data */}
152
+ {hasData && totalDimensions && dimensions.length < totalDimensions && (
153
+ <div className="health-gauge-partial">
154
+ {dimensions.length} of {totalDimensions} dimensions
155
+ </div>
156
+ )}
157
+
158
+ {/* Dimension breakdown — always visible, each row opens its dialog */}
159
+ <div data-testid="dimension-breakdown" className="health-gauge-breakdown">
160
+ {allDimKeys.map((dimName) => {
161
+ const dim = dimMap.get(dimName);
162
+ return (
163
+ <div
164
+ key={dimName}
165
+ data-testid={`dimension-${dimName}`}
166
+ className="health-gauge-dimension"
167
+ onClick={() => onDimensionClick?.(dimName)}
168
+ >
169
+ <span className="dimension-label">
170
+ {DIMENSION_LABELS[dimName] || dimName}
171
+ </span>
172
+ <span className="dimension-score">
173
+ {dim?.score !== null && dim?.score !== undefined ? dim.score.toFixed(1) : '--'}
174
+ </span>
175
+ </div>
176
+ );
177
+ })}
178
+ </div>
179
+ </div>
180
+ );
181
+ }
@@ -0,0 +1,225 @@
1
+ /**
2
+ * Message Component
3
+ *
4
+ * Renders a single message with avatar and content.
5
+ * Story MSSCI-12698 - MessageView Component with Streaming
6
+ * Story MSSCI-12777 - User Avatar from GitHub
7
+ */
8
+
9
+ import React, { useState, useEffect, useRef, useCallback } from 'react';
10
+ import { Badge } from '@/components/ui/badge';
11
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
12
+ import { parseMarkdown } from '../utils/markdown';
13
+ import StreamingContent from './StreamingContent';
14
+ import { usePersona } from '../hooks/usePersona';
15
+ import { useUserAvatar } from '../hooks/useUserAvatar';
16
+ import type { MessageData } from '../types/message';
17
+
18
+ interface MessageProps {
19
+ message: MessageData;
20
+ isLastAgentMessage?: boolean;
21
+ /** Whether this is the first message in a turn (shows avatar) */
22
+ isFirstInTurn?: boolean;
23
+ }
24
+
25
+ interface AssistantAvatarProps {
26
+ isStreaming?: boolean;
27
+ agentSlug?: string;
28
+ agentTheme?: string;
29
+ agentCharacter?: string;
30
+ }
31
+
32
+ function AssistantAvatar({ isStreaming, agentSlug, agentTheme, agentCharacter }: AssistantAvatarProps): React.ReactElement {
33
+ const { persona } = usePersona();
34
+ const [imageError, setImageError] = useState(false);
35
+
36
+ // Use per-message persona if available, fall back to current global persona
37
+ const slug = agentSlug || persona?.slug;
38
+ const theme = agentTheme || persona?.theme;
39
+ const character = agentCharacter || persona?.character;
40
+ const avatarClass = isStreaming ? 'avatar-portrait avatar-thinking' : 'avatar-portrait';
41
+
42
+ if (slug && theme && !imageError) {
43
+ return (
44
+ <img
45
+ src={`/portraits/${theme}/small/${slug}.png`}
46
+ alt={character || 'Agent'}
47
+ className={avatarClass}
48
+ onError={() => setImageError(true)}
49
+ />
50
+ );
51
+ }
52
+
53
+ return <span className={isStreaming ? 'avatar-emoji avatar-thinking' : 'avatar-emoji'}>🤖</span>;
54
+ }
55
+
56
+ function UserAvatar(): React.ReactElement {
57
+ const { avatarUrl, isLoading } = useUserAvatar();
58
+ const [imageError, setImageError] = useState(false);
59
+
60
+ if (isLoading) {
61
+ return <span className="avatar-emoji">👤</span>;
62
+ }
63
+
64
+ if (avatarUrl && !imageError) {
65
+ return (
66
+ <img
67
+ src={avatarUrl}
68
+ alt="User"
69
+ className="avatar-portrait"
70
+ onError={() => setImageError(true)}
71
+ />
72
+ );
73
+ }
74
+
75
+ return <span className="avatar-emoji">👤</span>;
76
+ }
77
+
78
+ function useSortableTables(contentRef: React.RefObject<HTMLDivElement | null>) {
79
+ const attachSort = useCallback(() => {
80
+ const el = contentRef.current;
81
+ if (!el) return;
82
+ const headers = el.querySelectorAll<HTMLTableCellElement>('th.sortable-th');
83
+ headers.forEach((th) => {
84
+ if (th.dataset.sortBound) return;
85
+ th.dataset.sortBound = '1';
86
+ th.style.cursor = 'pointer';
87
+ th.addEventListener('click', () => {
88
+ const colIdx = parseInt(th.dataset.col || '0', 10);
89
+ const table = th.closest('table');
90
+ if (!table) return;
91
+ const tbody = table.querySelector('tbody');
92
+ if (!tbody) return;
93
+ const rows = Array.from(tbody.querySelectorAll('tr'));
94
+ const currentDir = th.dataset.sortDir === 'asc' ? 'desc' : 'asc';
95
+
96
+ // Clear all indicators in this table
97
+ table.querySelectorAll<HTMLTableCellElement>('th.sortable-th').forEach((h) => {
98
+ h.dataset.sortDir = '';
99
+ const ind = h.querySelector('.sort-indicator');
100
+ if (ind) ind.textContent = '';
101
+ });
102
+
103
+ th.dataset.sortDir = currentDir;
104
+ const indicator = th.querySelector('.sort-indicator');
105
+ if (indicator) indicator.textContent = currentDir === 'asc' ? ' \u25B2' : ' \u25BC';
106
+
107
+ rows.sort((a, b) => {
108
+ const aText = (a.children[colIdx]?.textContent || '').trim();
109
+ const bText = (b.children[colIdx]?.textContent || '').trim();
110
+ const aNum = parseFloat(aText);
111
+ const bNum = parseFloat(bText);
112
+ // Numeric sort if both parse as numbers
113
+ if (!isNaN(aNum) && !isNaN(bNum)) {
114
+ return currentDir === 'asc' ? aNum - bNum : bNum - aNum;
115
+ }
116
+ const cmp = aText.localeCompare(bText, undefined, { sensitivity: 'base' });
117
+ return currentDir === 'asc' ? cmp : -cmp;
118
+ });
119
+
120
+ for (const row of rows) {
121
+ tbody.appendChild(row);
122
+ }
123
+ });
124
+ });
125
+ }, [contentRef]);
126
+
127
+ useEffect(() => {
128
+ attachSort();
129
+ });
130
+ }
131
+
132
+ export default function Message({ message, isLastAgentMessage, isFirstInTurn = true }: MessageProps): React.ReactElement {
133
+ const contentRef = useRef<HTMLDivElement>(null);
134
+ useSortableTables(contentRef);
135
+ const roleClass = `message-${message.type}`;
136
+ const testId = `message-${message.type}`;
137
+ const continuationClass = !isFirstInTurn ? ' continuation' : '';
138
+
139
+ // For bell-injected messages (queued messages injected via PostToolUse hook)
140
+ // Show with 🔔 indicator so user knows it was sent mid-turn
141
+ if (message.type === 'bell_injected') {
142
+ const html = message.content ? parseMarkdown(message.content) : '';
143
+ return (
144
+ <TooltipProvider delayDuration={300}>
145
+ <div data-testid="message-bell-injected" className={`message message-user message-bell-injected${continuationClass}`}>
146
+ <div data-testid="avatar" className="message-avatar">
147
+ <UserAvatar />
148
+ </div>
149
+ <div className="message-content">
150
+ <Tooltip>
151
+ <TooltipTrigger asChild>
152
+ <Badge variant="secondary" className="bell-indicator">🔔</Badge>
153
+ </TooltipTrigger>
154
+ <TooltipContent>Injected via Bell Mode</TooltipContent>
155
+ </Tooltip>
156
+ <div dangerouslySetInnerHTML={{ __html: html }} />
157
+ {message.imageCount && message.imageCount > 0 && (
158
+ <Tooltip>
159
+ <TooltipTrigger asChild>
160
+ <Badge variant="outline" className="message-attachment-indicator">
161
+ 📎 {message.imageCount}
162
+ </Badge>
163
+ </TooltipTrigger>
164
+ <TooltipContent>{`${message.imageCount} image${message.imageCount > 1 ? 's' : ''} attached`}</TooltipContent>
165
+ </Tooltip>
166
+ )}
167
+ </div>
168
+ </div>
169
+ </TooltipProvider>
170
+ );
171
+ }
172
+
173
+ // For streaming agent messages, use StreamingContent with throbbing avatar
174
+ // Only throb the last agent message's avatar
175
+ if (message.type === 'agent' && message.isStreaming) {
176
+ const showThrob = isLastAgentMessage !== false;
177
+ return (
178
+ <div data-testid={testId} className={`message ${roleClass}${continuationClass}`}>
179
+ <div data-testid="avatar" className="message-avatar">
180
+ <AssistantAvatar
181
+ isStreaming={showThrob}
182
+ agentSlug={message.agentSlug}
183
+ agentTheme={message.agentTheme}
184
+ agentCharacter={message.agentCharacter}
185
+ />
186
+ </div>
187
+ <div className="message-content">
188
+ <StreamingContent content={message.content || ''} isStreaming={message.isStreaming ?? false} />
189
+ </div>
190
+ </div>
191
+ );
192
+ }
193
+
194
+ // For regular messages, render markdown
195
+ const html = message.content ? parseMarkdown(message.content) : '';
196
+
197
+ return (
198
+ <div data-testid={testId} className={`message ${roleClass}${continuationClass}`} ref={contentRef}>
199
+ <div data-testid="avatar" className="message-avatar">
200
+ {message.type === 'user' ? <UserAvatar /> : (
201
+ <AssistantAvatar
202
+ agentSlug={message.agentSlug}
203
+ agentTheme={message.agentTheme}
204
+ agentCharacter={message.agentCharacter}
205
+ />
206
+ )}
207
+ </div>
208
+ <div className="message-content">
209
+ <div dangerouslySetInnerHTML={{ __html: html }} />
210
+ {message.type === 'user' && message.imageCount && message.imageCount > 0 && (
211
+ <TooltipProvider delayDuration={300}>
212
+ <Tooltip>
213
+ <TooltipTrigger asChild>
214
+ <Badge variant="outline" className="message-attachment-indicator">
215
+ 📎 {message.imageCount}
216
+ </Badge>
217
+ </TooltipTrigger>
218
+ <TooltipContent>{`${message.imageCount} image${message.imageCount > 1 ? 's' : ''} attached`}</TooltipContent>
219
+ </Tooltip>
220
+ </TooltipProvider>
221
+ )}
222
+ </div>
223
+ </div>
224
+ );
225
+ }
@@ -0,0 +1,98 @@
1
+ /**
2
+ * MessageList Component
3
+ *
4
+ * Scrolling container for message list with auto-scroll support.
5
+ * Story MSSCI-12698 - MessageView Component with Streaming
6
+ */
7
+
8
+ import React, { useRef, useEffect, useCallback, forwardRef, useImperativeHandle } from 'react';
9
+
10
+ interface MessageListProps {
11
+ children: React.ReactNode;
12
+ onScrollChange?: (isAtBottom: boolean) => void;
13
+ autoScroll?: boolean;
14
+ }
15
+
16
+ export interface MessageListHandle {
17
+ scrollToBottom: (behavior?: ScrollBehavior) => void;
18
+ }
19
+
20
+ const MessageList = forwardRef<MessageListHandle, MessageListProps>(
21
+ ({ children, onScrollChange, autoScroll = true }, ref) => {
22
+ const containerRef = useRef<HTMLDivElement>(null);
23
+ const isAtBottomRef = useRef(true);
24
+
25
+ const checkIfAtBottom = useCallback((scrollTopOverride?: number) => {
26
+ const container = containerRef.current;
27
+ if (!container) return true;
28
+
29
+ const scrollTop = scrollTopOverride ?? container.scrollTop;
30
+ const scrollHeight = container.scrollHeight || 0;
31
+ const clientHeight = container.clientHeight || 0;
32
+
33
+ // Test environment: scrollHeight may be 0 but we have children
34
+ // If scrollTop is explicitly 0 and we have children, assume not at bottom
35
+ if (scrollTop === 0 && scrollHeight === 0 && container.children.length > 0) {
36
+ return false;
37
+ }
38
+
39
+ // If scrollTop is 0 and there's scrollable content, we're at top (not bottom)
40
+ if (scrollTop === 0 && scrollHeight > clientHeight) {
41
+ return false;
42
+ }
43
+
44
+ // If there's no scrollable area, consider it at bottom
45
+ if (scrollHeight <= clientHeight) {
46
+ return true;
47
+ }
48
+
49
+ const threshold = 50; // pixels from bottom
50
+ const isAtBottom = scrollHeight - scrollTop - clientHeight < threshold;
51
+ return isAtBottom;
52
+ }, []);
53
+
54
+ const scrollToBottom = useCallback((behavior: ScrollBehavior = 'auto') => {
55
+ const container = containerRef.current;
56
+ if (!container) return;
57
+
58
+ container.scrollTo({
59
+ top: container.scrollHeight,
60
+ behavior,
61
+ });
62
+ }, []);
63
+
64
+ // Expose scrollToBottom to parent via ref
65
+ useImperativeHandle(ref, () => ({
66
+ scrollToBottom,
67
+ }));
68
+
69
+ const handleScroll = useCallback((event: React.UIEvent<HTMLDivElement>) => {
70
+ const target = event.target as HTMLDivElement;
71
+ const isAtBottom = checkIfAtBottom(target.scrollTop);
72
+ isAtBottomRef.current = isAtBottom;
73
+ onScrollChange?.(isAtBottom);
74
+ }, [checkIfAtBottom, onScrollChange]);
75
+
76
+ // Auto-scroll when children change (new messages)
77
+ useEffect(() => {
78
+ if (autoScroll && isAtBottomRef.current) {
79
+ scrollToBottom();
80
+ }
81
+ }, [children, autoScroll, scrollToBottom]);
82
+
83
+ return (
84
+ <div
85
+ ref={containerRef}
86
+ data-testid="message-list"
87
+ className="message-list"
88
+ onScroll={handleScroll}
89
+ >
90
+ {children}
91
+ </div>
92
+ );
93
+ }
94
+ );
95
+
96
+ MessageList.displayName = 'MessageList';
97
+
98
+ export default MessageList;