@pennyfarthing/core 11.3.7 → 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 (405) hide show
  1. package/README.md +1 -1
  2. package/package.json +17 -16
  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/hooks/cyclist-pretooluse-hook.sh +0 -0
  230. package/pennyfarthing-dist/pf/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
  231. package/pennyfarthing-dist/pf/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
  232. package/pennyfarthing-dist/pf/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
  233. package/pennyfarthing-dist/pf/hotspots/__pycache__/models.cpython-314.pyc +0 -0
  234. package/pennyfarthing-dist/pf/jira/__pycache__/__init__.cpython-314.pyc +0 -0
  235. package/pennyfarthing-dist/pf/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
  236. package/pennyfarthing-dist/pf/jira/__pycache__/claim.cpython-314.pyc +0 -0
  237. package/pennyfarthing-dist/pf/jira/__pycache__/cli.cpython-314.pyc +0 -0
  238. package/pennyfarthing-dist/pf/jira/__pycache__/client.cpython-314.pyc +0 -0
  239. package/pennyfarthing-dist/pf/jira/__pycache__/create.cpython-314.pyc +0 -0
  240. package/pennyfarthing-dist/pf/jira/__pycache__/epic.cpython-314.pyc +0 -0
  241. package/pennyfarthing-dist/pf/jira/__pycache__/operations.cpython-314.pyc +0 -0
  242. package/pennyfarthing-dist/pf/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
  243. package/pennyfarthing-dist/pf/jira/__pycache__/story.cpython-314.pyc +0 -0
  244. package/pennyfarthing-dist/pf/jira/__pycache__/sync.cpython-314.pyc +0 -0
  245. package/pennyfarthing-dist/pf/launch/__pycache__/__init__.cpython-314.pyc +0 -0
  246. package/pennyfarthing-dist/pf/launch/__pycache__/cli.cpython-314.pyc +0 -0
  247. package/pennyfarthing-dist/pf/prime/__pycache__/__init__.cpython-314.pyc +0 -0
  248. package/pennyfarthing-dist/pf/prime/__pycache__/cli.cpython-314.pyc +0 -0
  249. package/pennyfarthing-dist/pf/prime/__pycache__/loader.cpython-314.pyc +0 -0
  250. package/pennyfarthing-dist/pf/prime/__pycache__/models.cpython-314.pyc +0 -0
  251. package/pennyfarthing-dist/pf/prime/__pycache__/persona.cpython-314.pyc +0 -0
  252. package/pennyfarthing-dist/pf/prime/__pycache__/session.cpython-314.pyc +0 -0
  253. package/pennyfarthing-dist/pf/prime/__pycache__/tiers.cpython-314.pyc +0 -0
  254. package/pennyfarthing-dist/pf/prime/__pycache__/workflow.cpython-314.pyc +0 -0
  255. package/pennyfarthing-dist/pf/session/__pycache__/__init__.cpython-314.pyc +0 -0
  256. package/pennyfarthing-dist/pf/session/__pycache__/cli.cpython-314.pyc +0 -0
  257. package/pennyfarthing-dist/pf/settings/__pycache__/__init__.cpython-314.pyc +0 -0
  258. package/pennyfarthing-dist/pf/settings/__pycache__/cli.cpython-314.pyc +0 -0
  259. package/pennyfarthing-dist/pf/settings/__pycache__/settings.cpython-314.pyc +0 -0
  260. package/pennyfarthing-dist/pf/settings/settings.py +44 -8
  261. package/pennyfarthing-dist/pf/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
  262. package/pennyfarthing-dist/pf/sprint/__pycache__/archive.cpython-314.pyc +0 -0
  263. package/pennyfarthing-dist/pf/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
  264. package/pennyfarthing-dist/pf/sprint/__pycache__/cli.cpython-314.pyc +0 -0
  265. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
  266. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
  267. package/pennyfarthing-dist/pf/sprint/__pycache__/loader.cpython-314.pyc +0 -0
  268. package/pennyfarthing-dist/pf/sprint/__pycache__/status.cpython-314.pyc +0 -0
  269. package/pennyfarthing-dist/pf/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
  270. package/pennyfarthing-dist/pf/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
  271. package/pennyfarthing-dist/pf/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
  272. package/pennyfarthing-dist/pf/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
  273. package/pennyfarthing-dist/pf/sprint/__pycache__/validator.cpython-314.pyc +0 -0
  274. package/pennyfarthing-dist/pf/sprint/__pycache__/work.cpython-314.pyc +0 -0
  275. package/pennyfarthing-dist/pf/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
  276. package/pennyfarthing-dist/pf/sprint/story_finish.py +14 -2
  277. package/pennyfarthing-dist/pf/sprint/validator.py +7 -7
  278. package/pennyfarthing-dist/pf/tests/__pycache__/__init__.cpython-314.pyc +0 -0
  279. package/pennyfarthing-dist/pf/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
  280. package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
  281. package/pennyfarthing-dist/pf/tests/test_sprint_validator.py +44 -0
  282. package/pennyfarthing-dist/pf/theme/__pycache__/__init__.cpython-314.pyc +0 -0
  283. package/pennyfarthing-dist/pf/theme/__pycache__/cli.cpython-314.pyc +0 -0
  284. package/pennyfarthing-dist/pf/validate/__pycache__/__init__.cpython-314.pyc +0 -0
  285. package/pennyfarthing-dist/pf/validate/__pycache__/cli.cpython-314.pyc +0 -0
  286. package/pennyfarthing-dist/pf/workflow/__pycache__/__init__.cpython-314.pyc +0 -0
  287. package/pennyfarthing-dist/pf/workflow/__pycache__/cli.cpython-314.pyc +0 -0
  288. package/pennyfarthing-dist/pf/workflow/__pycache__/helpers.cpython-314.pyc +0 -0
  289. package/pennyfarthing-dist/pf/workflow/__pycache__/scale.cpython-314.pyc +0 -0
  290. package/pennyfarthing-dist/pf/workflow/__pycache__/state.cpython-314.pyc +0 -0
  291. package/pennyfarthing-dist/scripts/core/agent-session.sh +0 -0
  292. package/pennyfarthing-dist/scripts/core/check-context.sh +0 -0
  293. package/pennyfarthing-dist/scripts/core/dialogue-manager.sh +0 -0
  294. package/pennyfarthing-dist/scripts/core/pf.sh +0 -0
  295. package/pennyfarthing-dist/scripts/core/phase-check-start.sh +0 -0
  296. package/pennyfarthing-dist/scripts/core/prime.sh +0 -0
  297. package/pennyfarthing-dist/scripts/cyclist/is-cyclist.sh +0 -0
  298. package/pennyfarthing-dist/scripts/git/create-feature-branches.sh +0 -0
  299. package/pennyfarthing-dist/scripts/git/git-status-all.sh +0 -0
  300. package/pennyfarthing-dist/scripts/git/install-git-hooks.sh +0 -0
  301. package/pennyfarthing-dist/scripts/git/release.sh +0 -0
  302. package/pennyfarthing-dist/scripts/git/worktree-manager.sh +0 -0
  303. package/pennyfarthing-dist/scripts/health/drift-detection.sh +0 -0
  304. package/pennyfarthing-dist/scripts/hooks/bell-mode-hook.sh +0 -0
  305. package/pennyfarthing-dist/scripts/hooks/context-circuit-breaker.sh +0 -0
  306. package/pennyfarthing-dist/scripts/hooks/context-warning.sh +0 -0
  307. package/pennyfarthing-dist/scripts/hooks/cyclist-pretooluse-hook.sh +0 -0
  308. package/pennyfarthing-dist/scripts/hooks/dispatcher-template.sh +0 -0
  309. package/pennyfarthing-dist/scripts/hooks/otel-auto-config.sh +0 -0
  310. package/pennyfarthing-dist/scripts/hooks/post-merge.sh +0 -0
  311. package/pennyfarthing-dist/scripts/hooks/pre-commit.sh +0 -0
  312. package/pennyfarthing-dist/scripts/hooks/pre-edit-check.sh +0 -0
  313. package/pennyfarthing-dist/scripts/hooks/pre-push.sh +0 -0
  314. package/pennyfarthing-dist/scripts/hooks/question-reflector-check.sh +0 -0
  315. package/pennyfarthing-dist/scripts/hooks/question_reflector_check.py +0 -0
  316. package/pennyfarthing-dist/scripts/hooks/schema-validation.sh +0 -0
  317. package/pennyfarthing-dist/scripts/hooks/session-start.sh +0 -0
  318. package/pennyfarthing-dist/scripts/hooks/session-stop.sh +0 -0
  319. package/pennyfarthing-dist/scripts/hooks/sprint-yaml-validation.sh +0 -0
  320. package/pennyfarthing-dist/scripts/hooks/welcome-hook.sh +0 -0
  321. package/pennyfarthing-dist/scripts/jira/create-jira-epic.sh +0 -0
  322. package/pennyfarthing-dist/scripts/jira/create-jira-story.sh +0 -0
  323. package/pennyfarthing-dist/scripts/jira/jira-claim-story.sh +0 -0
  324. package/pennyfarthing-dist/scripts/jira/jira-reconcile.sh +0 -0
  325. package/pennyfarthing-dist/scripts/jira/jira-sync-story.sh +0 -0
  326. package/pennyfarthing-dist/scripts/jira/sync-epic-jira.sh +0 -0
  327. package/pennyfarthing-dist/scripts/lib/background-tasks.sh +0 -0
  328. package/pennyfarthing-dist/scripts/lib/checkpoint.sh +0 -0
  329. package/pennyfarthing-dist/scripts/lib/common.sh +0 -0
  330. package/pennyfarthing-dist/scripts/lib/env.sh +0 -0
  331. package/pennyfarthing-dist/scripts/lib/file-lock.sh +0 -0
  332. package/pennyfarthing-dist/scripts/lib/find-root.sh +1 -1
  333. package/pennyfarthing-dist/scripts/lib/logging.sh +0 -0
  334. package/pennyfarthing-dist/scripts/lib/retry.sh +0 -0
  335. package/pennyfarthing-dist/scripts/lib/run-pf.sh +0 -0
  336. package/pennyfarthing-dist/scripts/maintenance/migrate-theme-schema.mjs +0 -0
  337. package/pennyfarthing-dist/scripts/maintenance/sidecar-health.sh +0 -0
  338. package/pennyfarthing-dist/scripts/misc/add-short-names.sh +0 -0
  339. package/pennyfarthing-dist/scripts/misc/add_short_names.py +0 -0
  340. package/pennyfarthing-dist/scripts/misc/backlog.sh +0 -0
  341. package/pennyfarthing-dist/scripts/misc/check-status.sh +0 -0
  342. package/pennyfarthing-dist/scripts/misc/find-related-work.sh +0 -0
  343. package/pennyfarthing-dist/scripts/misc/generate-skill-docs.sh +0 -0
  344. package/pennyfarthing-dist/scripts/misc/log-skill-usage.sh +0 -0
  345. package/pennyfarthing-dist/scripts/misc/migrate-bmad-workflow.sh +0 -0
  346. package/pennyfarthing-dist/scripts/misc/migrate_bmad_workflow.py +0 -1
  347. package/pennyfarthing-dist/scripts/misc/repo-scan.sh +0 -0
  348. package/pennyfarthing-dist/scripts/misc/repo-utils.sh +0 -0
  349. package/pennyfarthing-dist/scripts/misc/run-ci.sh +0 -0
  350. package/pennyfarthing-dist/scripts/misc/run-timestamp.sh +0 -0
  351. package/pennyfarthing-dist/scripts/misc/session-cleanup.sh +0 -0
  352. package/pennyfarthing-dist/scripts/misc/skill-usage-report.sh +0 -0
  353. package/pennyfarthing-dist/scripts/misc/statusline.sh +0 -0
  354. package/pennyfarthing-dist/scripts/misc/uninstall.sh +0 -0
  355. package/pennyfarthing-dist/scripts/misc/validate-subagent-frontmatter.sh +0 -0
  356. package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +13 -13
  357. package/pennyfarthing-dist/scripts/portraits/generate-portraits.sh +0 -0
  358. package/pennyfarthing-dist/scripts/portraits/generate-tandem-portraits.sh +0 -0
  359. package/pennyfarthing-dist/scripts/story/create-story.sh +0 -0
  360. package/pennyfarthing-dist/scripts/story/size-story.sh +0 -0
  361. package/pennyfarthing-dist/scripts/story/story-template.sh +0 -0
  362. package/pennyfarthing-dist/scripts/tests/check.test.sh +0 -0
  363. package/pennyfarthing-dist/scripts/tests/dev-story-workflow-import.test.sh +0 -0
  364. package/pennyfarthing-dist/scripts/tests/epics-and-stories-workflow-import.test.sh +0 -0
  365. package/pennyfarthing-dist/scripts/tests/handoff-phase-update.test.sh +0 -0
  366. package/pennyfarthing-dist/scripts/tests/implementation-readiness-workflow-import.test.sh +0 -0
  367. package/pennyfarthing-dist/scripts/tests/migrate-bmad-workflow.test.sh +0 -0
  368. package/pennyfarthing-dist/scripts/tests/prd-workflow-import.test.sh +0 -0
  369. package/pennyfarthing-dist/scripts/tests/project-context-workflow-import.test.sh +0 -0
  370. package/pennyfarthing-dist/scripts/tests/test-character-voice.sh +0 -0
  371. package/pennyfarthing-dist/scripts/tests/test-drift-detection.sh +0 -0
  372. package/pennyfarthing-dist/scripts/tests/test-post-merge-hook.sh +0 -0
  373. package/pennyfarthing-dist/scripts/tests/test-session-checkpoint.sh +0 -0
  374. package/pennyfarthing-dist/scripts/tests/test-solo-command.sh +0 -0
  375. package/pennyfarthing-dist/scripts/tests/ux-design-workflow-import.test.sh +0 -0
  376. package/pennyfarthing-dist/scripts/theme/list-themes.sh +0 -0
  377. package/pennyfarthing-dist/scripts/validation/validate-agent-schema.sh +0 -0
  378. package/pennyfarthing-dist/scripts/workflow/check.py +4 -6
  379. package/pennyfarthing-dist/scripts/workflow/check.sh +0 -0
  380. package/pennyfarthing-dist/scripts/workflow/complete-step.py +2 -2
  381. package/pennyfarthing-dist/scripts/workflow/finish-story.sh +0 -0
  382. package/pennyfarthing-dist/scripts/workflow/fix-session-phase.sh +0 -0
  383. package/pennyfarthing-dist/scripts/workflow/get-workflow-type.py +0 -0
  384. package/pennyfarthing-dist/scripts/workflow/get-workflow-type.sh +0 -0
  385. package/pennyfarthing-dist/scripts/workflow/list-workflows.sh +0 -0
  386. package/pennyfarthing-dist/scripts/workflow/phase-owner.sh +0 -0
  387. package/pennyfarthing-dist/scripts/workflow/resume-workflow.sh +0 -0
  388. package/pennyfarthing-dist/scripts/workflow/show-workflow.sh +0 -0
  389. package/pennyfarthing-dist/scripts/workflow/start-workflow.sh +0 -0
  390. package/pennyfarthing-dist/scripts/workflow/workflow-status.sh +0 -0
  391. package/pennyfarthing-dist/skills/pf-story/scripts/create-story.sh +0 -0
  392. package/pennyfarthing-dist/skills/pf-story/scripts/size-story.sh +0 -0
  393. package/pennyfarthing-dist/skills/pf-story/scripts/story-template.sh +0 -0
  394. package/pennyfarthing-dist/skills/skill-registry.yaml +19 -0
  395. package/pennyfarthing-dist/workflows/release/steps/step-10-publish.md +41 -9
  396. package/pennyfarthing-dist/workflows/tdd-tandem.yaml +15 -2
  397. package/packages/core/dist/workflow/__test_context_watch__/.session/.tandem-turn-counter +0 -1
  398. package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-session.md +0 -3
  399. package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-tandem-architect.md +0 -6
  400. package/packages/core/dist/workflow/__test_file_watch__/.session/95-4-tandem-architect.md +0 -6
  401. package/packages/core/dist/workflow/__test_file_watch__/workdir/trigger.ts +0 -1
  402. package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-architect.md +0 -6
  403. package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-toolcalls.jsonl +0 -1
  404. package/pennyfarthing-dist/pf/bikerack/changed_panel.py +0 -201
  405. package/scripts/README.md +0 -41
@@ -0,0 +1,131 @@
1
+ /**
2
+ * useMessageStream Hook
3
+ *
4
+ * React hook for subscribing to Claude message stream via WebSocket.
5
+ * Migrated from IPC to WebSocket for unified communication.
6
+ *
7
+ * Story MSSCI-12698 - MessageView Component with Streaming
8
+ */
9
+
10
+ import { useState, useEffect, useCallback, useRef } from 'react';
11
+ import type { MessageData } from '../types/message';
12
+
13
+ interface UseMessageStreamResult {
14
+ messages: MessageData[];
15
+ isStreaming: boolean;
16
+ error: Error | null;
17
+ isConnected: boolean;
18
+ }
19
+
20
+ interface WebSocketMessage {
21
+ type: 'message' | 'complete' | 'error' | 'init' | 'mode';
22
+ message?: {
23
+ type: string;
24
+ content?: string | Array<{ type: string; text?: string }>;
25
+ [key: string]: unknown;
26
+ };
27
+ error?: string;
28
+ }
29
+
30
+ export function useMessageStream(): UseMessageStreamResult {
31
+ const [messages, setMessages] = useState<MessageData[]>([]);
32
+ const [isStreaming, setIsStreaming] = useState(false);
33
+ const [error, setError] = useState<Error | null>(null);
34
+ const [isConnected, setIsConnected] = useState(false);
35
+ const wsRef = useRef<WebSocket | null>(null);
36
+ const reconnectTimeoutRef = useRef<ReturnType<typeof setTimeout>>();
37
+
38
+ const handleWebSocketMessage = useCallback((data: WebSocketMessage) => {
39
+ if (data.type === 'message' && data.message) {
40
+ const msg = data.message;
41
+
42
+ // Extract content string
43
+ let contentStr = '';
44
+ if (typeof msg.content === 'string') {
45
+ contentStr = msg.content;
46
+ } else if (Array.isArray(msg.content)) {
47
+ contentStr = msg.content
48
+ .filter((c) => c.type === 'text' && c.text)
49
+ .map((c) => c.text)
50
+ .join('');
51
+ }
52
+
53
+ // Map SDK 'assistant' type to our internal 'agent' type
54
+ const messageType = msg.type === 'assistant' ? 'agent' : msg.type as MessageData['type'];
55
+
56
+ const message: MessageData = {
57
+ type: messageType,
58
+ content: contentStr,
59
+ timestamp: Date.now(),
60
+ isStreaming: true,
61
+ };
62
+
63
+ setMessages((prev) => [...prev, message]);
64
+
65
+ // Update streaming state for agent messages
66
+ if (msg.type === 'assistant') {
67
+ setIsStreaming(true);
68
+ }
69
+ } else if (data.type === 'complete') {
70
+ setIsStreaming(false);
71
+ } else if (data.type === 'error' && data.error) {
72
+ setError(new Error(data.error));
73
+ setIsStreaming(false);
74
+ }
75
+ }, []);
76
+
77
+ const connect = useCallback(() => {
78
+ const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
79
+ const wsUrl = `${protocol}//${window.location.host}/ws/claude`;
80
+
81
+ const ws = new WebSocket(wsUrl);
82
+ wsRef.current = ws;
83
+
84
+ ws.onopen = () => {
85
+ console.log('[useMessageStream] Connected');
86
+ setIsConnected(true);
87
+ setError(null);
88
+ };
89
+
90
+ ws.onmessage = (event) => {
91
+ try {
92
+ const data = JSON.parse(event.data) as WebSocketMessage;
93
+ handleWebSocketMessage(data);
94
+ } catch (err) {
95
+ console.error('[useMessageStream] Failed to parse message:', err);
96
+ }
97
+ };
98
+
99
+ ws.onclose = () => {
100
+ console.log('[useMessageStream] Disconnected');
101
+ setIsConnected(false);
102
+ wsRef.current = null;
103
+
104
+ // Attempt reconnection after delay
105
+ reconnectTimeoutRef.current = setTimeout(() => {
106
+ console.log('[useMessageStream] Attempting reconnect...');
107
+ connect();
108
+ }, 2000);
109
+ };
110
+
111
+ ws.onerror = () => {
112
+ setError(new Error('WebSocket connection failed'));
113
+ };
114
+ }, [handleWebSocketMessage]);
115
+
116
+ useEffect(() => {
117
+ connect();
118
+
119
+ return () => {
120
+ if (reconnectTimeoutRef.current) {
121
+ clearTimeout(reconnectTimeoutRef.current);
122
+ }
123
+ if (wsRef.current) {
124
+ wsRef.current.close();
125
+ wsRef.current = null;
126
+ }
127
+ };
128
+ }, [connect]);
129
+
130
+ return { messages, isStreaming, error, isConnected };
131
+ }
@@ -0,0 +1,112 @@
1
+ /**
2
+ * usePersona Hook
3
+ *
4
+ * React hook for fetching and subscribing to persona data.
5
+ * Story MSSCI-12700 - PersonaHeader Component
6
+ * Story MSSCI-12860 - IPC to WebSocket Migration (Phase 1)
7
+ *
8
+ * Provides:
9
+ * - character: Current agent character name
10
+ * - theme: Current theme name
11
+ * - role: Agent role/title
12
+ */
13
+
14
+ import { useState, useEffect, useRef } from 'react';
15
+
16
+ export interface TandemAgentData {
17
+ character: string;
18
+ role: string;
19
+ slug: string;
20
+ theme: string;
21
+ isThinking: boolean;
22
+ }
23
+
24
+ export interface PersonaData {
25
+ character: string | null;
26
+ theme: string | null;
27
+ role: string | null;
28
+ slug: string | null;
29
+ quote: string | null; // Random catchphrase from theme
30
+ tandemAgent?: TandemAgentData | null;
31
+ }
32
+
33
+ interface UsePersonaResult {
34
+ persona: PersonaData | null;
35
+ isStreaming: boolean;
36
+ isLoading: boolean;
37
+ error: Error | null;
38
+ }
39
+
40
+ export function usePersona(): UsePersonaResult {
41
+ const [persona, setPersona] = useState<PersonaData | null>(null);
42
+ const [isStreaming, setIsStreaming] = useState(false);
43
+ const [isLoading, setIsLoading] = useState(true);
44
+ const [error, setError] = useState<Error | null>(null);
45
+ const wsRef = useRef<WebSocket | null>(null);
46
+ const reconnectTimeoutRef = useRef<ReturnType<typeof setTimeout>>();
47
+
48
+ useEffect(() => {
49
+ const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
50
+ const wsUrl = `${protocol}//${window.location.host}/ws/persona`;
51
+
52
+ const connect = () => {
53
+ try {
54
+ wsRef.current = new WebSocket(wsUrl);
55
+
56
+ wsRef.current.onopen = () => {
57
+ console.debug('[usePersona] WebSocket connected');
58
+ };
59
+
60
+ wsRef.current.onmessage = (event) => {
61
+ try {
62
+ const data = JSON.parse(event.data);
63
+
64
+ // Story 94-1: Handle streaming state updates
65
+ if (data.type === 'streaming') {
66
+ setIsStreaming(data.isStreaming ?? false);
67
+ return;
68
+ }
69
+
70
+ // Persona data (initial or agent change) — extract isStreaming if present
71
+ if (data.isStreaming !== undefined) {
72
+ setIsStreaming(data.isStreaming);
73
+ }
74
+ setPersona(data as PersonaData);
75
+ setIsLoading(false);
76
+ setError(null);
77
+ } catch (err) {
78
+ console.error('[usePersona] Failed to parse message:', err);
79
+ }
80
+ };
81
+
82
+ wsRef.current.onclose = () => {
83
+ console.debug('[usePersona] WebSocket closed, reconnecting...');
84
+ setIsStreaming(false);
85
+ reconnectTimeoutRef.current = setTimeout(connect, 2000);
86
+ };
87
+
88
+ wsRef.current.onerror = (err) => {
89
+ console.error('[usePersona] WebSocket error:', err);
90
+ setError(new Error('WebSocket connection failed'));
91
+ };
92
+ } catch (err) {
93
+ console.error('[usePersona] WebSocket init failed:', err);
94
+ setError(err instanceof Error ? err : new Error('Failed to connect'));
95
+ setIsLoading(false);
96
+ }
97
+ };
98
+
99
+ connect();
100
+
101
+ return () => {
102
+ if (reconnectTimeoutRef.current) {
103
+ clearTimeout(reconnectTimeoutRef.current);
104
+ }
105
+ if (wsRef.current) {
106
+ wsRef.current.close();
107
+ }
108
+ };
109
+ }, []);
110
+
111
+ return { persona, isStreaming, isLoading, error };
112
+ }
@@ -0,0 +1,105 @@
1
+ /**
2
+ * usePlanModeExit - Smooth plan mode exit with tirepump choice
3
+ *
4
+ * Story: MSSCI-14327
5
+ *
6
+ * Handles the transition from plan mode to accept mode after plan approval,
7
+ * and offers the user a choice to tirepump (commit/push) changes.
8
+ */
9
+
10
+ import { useState, useCallback } from 'react';
11
+ import type { Mode } from '../components/ModeSwitch/index';
12
+
13
+ /** The Claude CLI mode to transition to after plan exit */
14
+ export const PLAN_EXIT_MODE = 'acceptEdits';
15
+
16
+ export interface PlanModeExitOptions {
17
+ currentMode: Mode;
18
+ setMode: (mode: Mode) => void;
19
+ approved: boolean;
20
+ wsConnected?: boolean;
21
+ }
22
+
23
+ export interface TirepumpOption {
24
+ action: 'tirepump' | 'continue';
25
+ label: string;
26
+ }
27
+
28
+ export interface PlanModeExitResult {
29
+ showTirepumpChoice: boolean;
30
+ options?: TirepumpOption[];
31
+ localOnly?: boolean;
32
+ }
33
+
34
+ export interface TirepumpChoiceOptions {
35
+ choice: 'tirepump' | 'continue';
36
+ currentAgent?: string;
37
+ setMode?: (mode: Mode) => void;
38
+ onContextClear?: (data: { agent: string }) => void;
39
+ }
40
+
41
+ export interface TirepumpChoiceResult {
42
+ contextCleared: boolean;
43
+ action: 'tirepump' | 'continue';
44
+ }
45
+
46
+ const TIREPUMP_OPTIONS: TirepumpOption[] = [
47
+ { action: 'tirepump', label: 'Commit & push changes' },
48
+ { action: 'continue', label: 'Continue without committing' },
49
+ ];
50
+
51
+ export async function handlePlanModeExit(
52
+ options: PlanModeExitOptions
53
+ ): Promise<PlanModeExitResult> {
54
+ const { currentMode, setMode, approved, wsConnected } = options;
55
+
56
+ if (!approved) {
57
+ return { showTirepumpChoice: false };
58
+ }
59
+
60
+ // Transition from plan to accept if not already there
61
+ if (currentMode !== 'accept') {
62
+ setMode('accept');
63
+ }
64
+
65
+ const localOnly = wsConnected === false;
66
+
67
+ return {
68
+ showTirepumpChoice: true,
69
+ options: TIREPUMP_OPTIONS,
70
+ ...(localOnly && { localOnly: true }),
71
+ };
72
+ }
73
+
74
+ export async function handleTirepumpChoice(
75
+ options: TirepumpChoiceOptions
76
+ ): Promise<TirepumpChoiceResult> {
77
+ const { choice, currentAgent, onContextClear } = options;
78
+
79
+ if (choice === 'tirepump') {
80
+ onContextClear?.({ agent: currentAgent ?? '' });
81
+ return { contextCleared: true, action: 'tirepump' };
82
+ }
83
+
84
+ return { contextCleared: false, action: 'continue' };
85
+ }
86
+
87
+ export function usePlanModeExit() {
88
+ const [showChoice, setShowChoice] = useState(false);
89
+ const [exitResult, setExitResult] = useState<PlanModeExitResult | null>(null);
90
+
91
+ const exitPlanMode = useCallback(async (options: PlanModeExitOptions) => {
92
+ const result = await handlePlanModeExit(options);
93
+ setExitResult(result);
94
+ setShowChoice(result.showTirepumpChoice);
95
+ return result;
96
+ }, []);
97
+
98
+ const chooseTirepump = useCallback(async (options: TirepumpChoiceOptions) => {
99
+ const result = await handleTirepumpChoice(options);
100
+ setShowChoice(false);
101
+ return result;
102
+ }, []);
103
+
104
+ return { showChoice, exitResult, exitPlanMode, chooseTirepump };
105
+ }
@@ -0,0 +1,173 @@
1
+ /**
2
+ * useResponsiveLayout - Responsive breakpoint detection hook
3
+ *
4
+ * Story MSSCI-12770: Responsive Breakpoints
5
+ *
6
+ * Provides:
7
+ * - Breakpoint detection (small <1024, medium 1024-1440, large >1440)
8
+ * - Sidebar width recommendations
9
+ * - Minimum dimension violation detection
10
+ * - Resize event handling
11
+ */
12
+
13
+ import { useState, useEffect, useCallback } from 'react';
14
+
15
+ // =============================================================================
16
+ // Constants
17
+ // =============================================================================
18
+
19
+ /**
20
+ * Breakpoint thresholds in pixels
21
+ */
22
+ export const BREAKPOINTS = {
23
+ small: 1024, // < 1024px
24
+ large: 1440, // >= 1440px
25
+ } as const;
26
+
27
+ /**
28
+ * Sidebar widths for each breakpoint
29
+ */
30
+ export const SIDEBAR_WIDTHS = {
31
+ medium: 300, // Default width
32
+ large: 400, // Expanded width for large screens
33
+ } as const;
34
+
35
+ /**
36
+ * Minimum supported dimensions
37
+ */
38
+ export const MIN_DIMENSIONS = {
39
+ width: 800,
40
+ height: 600,
41
+ } as const;
42
+
43
+ /**
44
+ * CSS custom property names for breakpoints
45
+ */
46
+ export const CSS_BREAKPOINT_VARS = '--breakpoint-small: 1024px; --breakpoint-large: 1440px;';
47
+
48
+ /**
49
+ * Media query strings for CSS-in-JS usage
50
+ */
51
+ export const MEDIA_QUERIES = {
52
+ small: '(max-width: 1023px)',
53
+ medium: '(min-width: 1024px) and (max-width: 1439px)',
54
+ large: '(min-width: 1440px)',
55
+ } as const;
56
+
57
+ // =============================================================================
58
+ // Types
59
+ // =============================================================================
60
+
61
+ export type Breakpoint = 'small' | 'medium' | 'large';
62
+
63
+ export interface MinimumViolation {
64
+ width: boolean;
65
+ height: boolean;
66
+ }
67
+
68
+ export interface ResponsiveLayoutState {
69
+ /** Current breakpoint based on window width */
70
+ breakpoint: Breakpoint;
71
+ /** True if width < 1024px */
72
+ isSmall: boolean;
73
+ /** True if width >= 1024px and < 1440px */
74
+ isMedium: boolean;
75
+ /** True if width >= 1440px */
76
+ isLarge: boolean;
77
+ /** Current window width */
78
+ width: number;
79
+ /** Current window height */
80
+ height: number;
81
+ /** Recommended sidebar width for current breakpoint */
82
+ sidebarWidth: number;
83
+ /** True if either dimension is below minimum */
84
+ isBelowMinimum: boolean;
85
+ /** Which dimensions violate minimums */
86
+ minimumViolation: MinimumViolation;
87
+ }
88
+
89
+ // =============================================================================
90
+ // Helper Functions
91
+ // =============================================================================
92
+
93
+ /**
94
+ * Determine breakpoint from width
95
+ */
96
+ function getBreakpoint(width: number): Breakpoint {
97
+ if (width < BREAKPOINTS.small) {
98
+ return 'small';
99
+ }
100
+ if (width >= BREAKPOINTS.large) {
101
+ return 'large';
102
+ }
103
+ return 'medium';
104
+ }
105
+
106
+ /**
107
+ * Get sidebar width for breakpoint
108
+ */
109
+ function getSidebarWidth(breakpoint: Breakpoint): number {
110
+ if (breakpoint === 'large') {
111
+ return SIDEBAR_WIDTHS.large;
112
+ }
113
+ return SIDEBAR_WIDTHS.medium;
114
+ }
115
+
116
+ /**
117
+ * Check minimum dimension violations
118
+ */
119
+ function checkMinimumViolation(width: number, height: number): MinimumViolation {
120
+ return {
121
+ width: width < MIN_DIMENSIONS.width,
122
+ height: height < MIN_DIMENSIONS.height,
123
+ };
124
+ }
125
+
126
+ // =============================================================================
127
+ // Hook
128
+ // =============================================================================
129
+
130
+ /**
131
+ * React hook for responsive layout detection
132
+ *
133
+ * Detects current breakpoint based on window dimensions and provides
134
+ * utilities for responsive behavior.
135
+ */
136
+ export function useResponsiveLayout(): ResponsiveLayoutState {
137
+ const getWindowDimensions = useCallback(() => {
138
+ return {
139
+ width: typeof window !== 'undefined' ? window.innerWidth : 1200,
140
+ height: typeof window !== 'undefined' ? window.innerHeight : 800,
141
+ };
142
+ }, []);
143
+
144
+ const [dimensions, setDimensions] = useState(getWindowDimensions);
145
+
146
+ useEffect(() => {
147
+ const handleResize = () => {
148
+ setDimensions(getWindowDimensions());
149
+ };
150
+
151
+ window.addEventListener('resize', handleResize);
152
+ return () => {
153
+ window.removeEventListener('resize', handleResize);
154
+ };
155
+ }, [getWindowDimensions]);
156
+
157
+ const breakpoint = getBreakpoint(dimensions.width);
158
+ const minimumViolation = checkMinimumViolation(dimensions.width, dimensions.height);
159
+
160
+ return {
161
+ breakpoint,
162
+ isSmall: breakpoint === 'small',
163
+ isMedium: breakpoint === 'medium',
164
+ isLarge: breakpoint === 'large',
165
+ width: dimensions.width,
166
+ height: dimensions.height,
167
+ sidebarWidth: getSidebarWidth(breakpoint),
168
+ isBelowMinimum: minimumViolation.width || minimumViolation.height,
169
+ minimumViolation,
170
+ };
171
+ }
172
+
173
+ export default useResponsiveLayout;
@@ -0,0 +1,157 @@
1
+ /**
2
+ * useSprint Hook
3
+ *
4
+ * React hook for subscribing to sprint data.
5
+ * Story MSSCI-14189 - Enhanced Sprint Panel
6
+ *
7
+ * Uses WebSocket /ws/sprint for real-time updates.
8
+ */
9
+
10
+ import { useState, useEffect, useRef } from 'react';
11
+
12
+ // =============================================================================
13
+ // Types matching EnhancedSprintPanel expectations
14
+ // =============================================================================
15
+
16
+ export interface SprintStory {
17
+ id: string;
18
+ title: string;
19
+ points: number;
20
+ status: 'backlog' | 'in_progress' | 'done' | 'cancelled' | 'blocked';
21
+ jiraKey: string | null;
22
+ hasContext?: boolean;
23
+ assignedTo?: string | null;
24
+ completed?: string | null;
25
+ started?: string | null;
26
+ workflow?: string | null;
27
+ priority?: string | null;
28
+ description?: string | null;
29
+ }
30
+
31
+ export interface SprintEpic {
32
+ id: string;
33
+ title: string;
34
+ jiraKey: string | null;
35
+ stories: SprintStory[];
36
+ hasContext?: boolean;
37
+ }
38
+
39
+ export interface FutureEpicChild {
40
+ id: string;
41
+ title: string;
42
+ estimatedPoints: number;
43
+ status: 'ready' | 'blocked' | 'planning';
44
+ jiraKey: string | null;
45
+ storyCount: number;
46
+ }
47
+
48
+ export interface FutureEpic {
49
+ id: string;
50
+ title: string;
51
+ description: string;
52
+ estimatedPoints: number;
53
+ status: 'ready' | 'blocked' | 'planning';
54
+ children: FutureEpicChild[];
55
+ }
56
+
57
+ export interface SprintData {
58
+ currentStory: SprintStory | null;
59
+ nextStory: SprintStory | null;
60
+ epics: SprintEpic[];
61
+ futureEpics: FutureEpic[];
62
+ sprint: {
63
+ number: number;
64
+ name: string;
65
+ done: number;
66
+ remaining: number;
67
+ inProgress: number;
68
+ endDate: string;
69
+ };
70
+ }
71
+
72
+ interface UseSprintResult {
73
+ data: SprintData | null;
74
+ isLoading: boolean;
75
+ error: Error | null;
76
+ }
77
+
78
+ /** WebSocket message format from /ws/sprint */
79
+ interface SprintMessage extends Partial<SprintData> {
80
+ type: 'init' | 'update';
81
+ }
82
+
83
+ export function useSprint(): UseSprintResult {
84
+ const [data, setData] = useState<SprintData | null>(null);
85
+ const [isLoading, setIsLoading] = useState(true);
86
+ const [error, setError] = useState<Error | null>(null);
87
+ const wsRef = useRef<WebSocket | null>(null);
88
+ const reconnectTimeoutRef = useRef<ReturnType<typeof setTimeout>>();
89
+ const isMountedRef = useRef(true);
90
+
91
+ useEffect(() => {
92
+ const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
93
+ const wsUrl = `${protocol}//${window.location.host}/ws/sprint`;
94
+
95
+ const connect = () => {
96
+ // Don't reconnect if component has unmounted
97
+ if (!isMountedRef.current) {
98
+ return;
99
+ }
100
+
101
+ try {
102
+ wsRef.current = new WebSocket(wsUrl);
103
+
104
+ wsRef.current.onopen = () => {
105
+ console.debug('[useSprint] WebSocket connected');
106
+ };
107
+
108
+ wsRef.current.onmessage = (event) => {
109
+ try {
110
+ const msg = JSON.parse(event.data) as SprintMessage;
111
+ if (msg.type === 'init' || msg.type === 'update') {
112
+ // Extract data, excluding type field
113
+ const { type: _type, ...sprintData } = msg;
114
+ setData((prev) => {
115
+ if (!prev) return sprintData as SprintData;
116
+ // Merge partial updates
117
+ return { ...prev, ...sprintData } as SprintData;
118
+ });
119
+ setIsLoading(false);
120
+ setError(null);
121
+ }
122
+ } catch (err) {
123
+ console.error('[useSprint] Failed to parse message:', err);
124
+ }
125
+ };
126
+
127
+ wsRef.current.onclose = () => {
128
+ console.debug('[useSprint] WebSocket closed, reconnecting...');
129
+ reconnectTimeoutRef.current = setTimeout(connect, 2000);
130
+ };
131
+
132
+ wsRef.current.onerror = (err) => {
133
+ console.error('[useSprint] WebSocket error:', err);
134
+ setError(new Error('WebSocket connection failed'));
135
+ };
136
+ } catch (err) {
137
+ console.error('[useSprint] WebSocket init failed:', err);
138
+ setError(err instanceof Error ? err : new Error('Failed to connect'));
139
+ setIsLoading(false);
140
+ }
141
+ };
142
+
143
+ connect();
144
+
145
+ return () => {
146
+ isMountedRef.current = false;
147
+ if (reconnectTimeoutRef.current) {
148
+ clearTimeout(reconnectTimeoutRef.current);
149
+ }
150
+ if (wsRef.current) {
151
+ wsRef.current.close();
152
+ }
153
+ };
154
+ }, []);
155
+
156
+ return { data, isLoading, error };
157
+ }