@pennyfarthing/cyclist 10.4.0 → 11.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/agent-load.d.ts +1 -2
- package/dist/api/agent-load.d.ts.map +1 -1
- package/dist/api/agent-load.js +2 -123
- package/dist/api/agent-load.js.map +1 -1
- package/dist/api/audit-log.d.ts +1 -17
- package/dist/api/audit-log.d.ts.map +1 -1
- package/dist/api/audit-log.js +2 -162
- package/dist/api/audit-log.js.map +1 -1
- package/dist/api/background-tasks.d.ts +1 -26
- package/dist/api/background-tasks.d.ts.map +1 -1
- package/dist/api/background-tasks.js +2 -55
- package/dist/api/background-tasks.js.map +1 -1
- package/dist/api/bell.d.ts +1 -18
- package/dist/api/bell.d.ts.map +1 -1
- package/dist/api/bell.js +2 -33
- package/dist/api/bell.js.map +1 -1
- package/dist/api/code-markers.d.ts +1 -8
- package/dist/api/code-markers.d.ts.map +1 -1
- package/dist/api/code-markers.js +2 -61
- package/dist/api/code-markers.js.map +1 -1
- package/dist/api/complexity.d.ts +1 -2
- package/dist/api/complexity.d.ts.map +1 -1
- package/dist/api/complexity.js +2 -46
- package/dist/api/complexity.js.map +1 -1
- package/dist/api/context.d.ts +1 -37
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +2 -143
- package/dist/api/context.js.map +1 -1
- package/dist/api/dead-code.d.ts +1 -2
- package/dist/api/dead-code.d.ts.map +1 -1
- package/dist/api/dead-code.js +2 -69
- package/dist/api/dead-code.js.map +1 -1
- package/dist/api/dependencies.d.ts +1 -2
- package/dist/api/dependencies.d.ts.map +1 -1
- package/dist/api/dependencies.js +2 -42
- package/dist/api/dependencies.js.map +1 -1
- package/dist/api/evaluation.d.ts +1 -19
- package/dist/api/evaluation.d.ts.map +1 -1
- package/dist/api/evaluation.js +2 -127
- package/dist/api/evaluation.js.map +1 -1
- package/dist/api/file-browser.d.ts +1 -8
- package/dist/api/file-browser.d.ts.map +1 -1
- package/dist/api/file-browser.js +2 -114
- package/dist/api/file-browser.js.map +1 -1
- package/dist/api/git.d.ts +1 -46
- package/dist/api/git.d.ts.map +1 -1
- package/dist/api/git.js +2 -354
- package/dist/api/git.js.map +1 -1
- package/dist/api/health-score.d.ts +1 -2
- package/dist/api/health-score.d.ts.map +1 -1
- package/dist/api/health-score.js +2 -46
- package/dist/api/health-score.js.map +1 -1
- package/dist/api/hook-request.d.ts +1 -40
- package/dist/api/hook-request.d.ts.map +1 -1
- package/dist/api/hook-request.js +2 -277
- package/dist/api/hook-request.js.map +1 -1
- package/dist/api/hotspots.d.ts +1 -2
- package/dist/api/hotspots.d.ts.map +1 -1
- package/dist/api/hotspots.js +2 -61
- package/dist/api/hotspots.js.map +1 -1
- package/dist/api/identity.d.ts +1 -16
- package/dist/api/identity.d.ts.map +1 -1
- package/dist/api/identity.js +2 -78
- package/dist/api/identity.js.map +1 -1
- package/dist/api/index.d.ts +1 -34
- package/dist/api/index.d.ts.map +1 -1
- package/dist/api/index.js +2 -44
- package/dist/api/index.js.map +1 -1
- package/dist/api/mode.d.ts +1 -22
- package/dist/api/mode.d.ts.map +1 -1
- package/dist/api/mode.js +2 -37
- package/dist/api/mode.js.map +1 -1
- package/dist/api/otlp.d.ts +1 -2
- package/dist/api/otlp.d.ts.map +1 -1
- package/dist/api/otlp.js +2 -46
- package/dist/api/otlp.js.map +1 -1
- package/dist/api/permissions.d.ts +1 -15
- package/dist/api/permissions.d.ts.map +1 -1
- package/dist/api/permissions.js +2 -66
- package/dist/api/permissions.js.map +1 -1
- package/dist/api/persona.d.ts +1 -8
- package/dist/api/persona.d.ts.map +1 -1
- package/dist/api/persona.js +2 -67
- package/dist/api/persona.js.map +1 -1
- package/dist/api/portrait.d.ts +1 -5
- package/dist/api/portrait.d.ts.map +1 -1
- package/dist/api/portrait.js +2 -27
- package/dist/api/portrait.js.map +1 -1
- package/dist/api/settings.d.ts +1 -53
- package/dist/api/settings.d.ts.map +1 -1
- package/dist/api/settings.js +2 -464
- package/dist/api/settings.js.map +1 -1
- package/dist/api/spans.d.ts +1 -16
- package/dist/api/spans.d.ts.map +1 -1
- package/dist/api/spans.js +2 -244
- package/dist/api/spans.js.map +1 -1
- package/dist/api/stats.d.ts +1 -12
- package/dist/api/stats.d.ts.map +1 -1
- package/dist/api/stats.js +2 -84
- package/dist/api/stats.js.map +1 -1
- package/dist/api/story.d.ts +1 -2
- package/dist/api/story.d.ts.map +1 -1
- package/dist/api/story.js +2 -14
- package/dist/api/story.js.map +1 -1
- package/dist/api/telemetry.d.ts +1 -18
- package/dist/api/telemetry.d.ts.map +1 -1
- package/dist/api/telemetry.js +2 -164
- package/dist/api/telemetry.js.map +1 -1
- package/dist/api/theme-agents.d.ts +1 -60
- package/dist/api/theme-agents.d.ts.map +1 -1
- package/dist/api/theme-agents.js +2 -213
- package/dist/api/theme-agents.js.map +1 -1
- package/dist/api/todos.d.ts +1 -32
- package/dist/api/todos.d.ts.map +1 -1
- package/dist/api/todos.js +2 -43
- package/dist/api/todos.js.map +1 -1
- package/dist/api/token-stats.d.ts +1 -7
- package/dist/api/token-stats.d.ts.map +1 -1
- package/dist/api/token-stats.js +2 -35
- package/dist/api/token-stats.js.map +1 -1
- package/dist/api/welcome.d.ts +1 -21
- package/dist/api/welcome.d.ts.map +1 -1
- package/dist/api/welcome.js +2 -34
- package/dist/api/welcome.js.map +1 -1
- package/dist/bikerack.js +2 -2
- package/dist/bikerack.js.map +1 -1
- package/dist/env.d.ts +6 -0
- package/dist/env.d.ts.map +1 -0
- package/dist/env.js +10 -0
- package/dist/env.js.map +1 -0
- package/dist/focus.d.ts +53 -0
- package/dist/focus.d.ts.map +1 -0
- package/dist/focus.js +122 -0
- package/dist/focus.js.map +1 -0
- package/dist/git-cache.d.ts +1 -0
- package/dist/git-cache.d.ts.map +1 -1
- package/dist/git-cache.js +3 -1
- package/dist/git-cache.js.map +1 -1
- package/dist/menu-builder.d.ts.map +1 -1
- package/dist/menu-builder.js +0 -1
- package/dist/menu-builder.js.map +1 -1
- package/dist/prime.d.ts +3 -3
- package/dist/prime.d.ts.map +1 -1
- package/dist/prime.js +38 -14
- package/dist/prime.js.map +1 -1
- package/dist/public/css/react.css +1 -1
- package/dist/public/js/react/react.js +53 -61
- package/dist/server.d.ts +18 -85
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +105 -405
- package/dist/server.js.map +1 -1
- package/dist/sprint-data.d.ts +1 -1
- package/dist/sprint-data.d.ts.map +1 -1
- package/dist/sprint-data.js +2 -2
- package/dist/sprint-data.js.map +1 -1
- package/dist/theme-metadata.d.ts +3 -3
- package/dist/theme-metadata.d.ts.map +1 -1
- package/dist/theme-metadata.js +4 -4
- package/dist/theme-metadata.js.map +1 -1
- package/dist/websocket.d.ts +2 -0
- package/dist/websocket.d.ts.map +1 -1
- package/dist/websocket.js +53 -75
- package/dist/websocket.js.map +1 -1
- package/package.json +2 -6
- package/portraits/hogans-heroes/large/burkhalter-35312.png +0 -0
- package/portraits/hogans-heroes/large/carter-34352.png +0 -0
- package/portraits/hogans-heroes/large/hochstetter-45314.png +0 -0
- package/portraits/hogans-heroes/large/hogan-44541.png +0 -0
- package/portraits/hogans-heroes/large/kinch-35241.png +0 -0
- package/portraits/hogans-heroes/large/klink-23434.png +0 -0
- package/portraits/hogans-heroes/large/lebeau-45443.png +0 -0
- package/portraits/hogans-heroes/large/marya-53543.png +0 -0
- package/portraits/hogans-heroes/large/newkirk-54432.png +0 -0
- package/portraits/hogans-heroes/large/schultz-42453.png +0 -0
- package/portraits/hogans-heroes/large/underground-55131.png +0 -0
- package/portraits/hogans-heroes/medium/burkhalter-35312.png +0 -0
- package/portraits/hogans-heroes/medium/carter-34352.png +0 -0
- package/portraits/hogans-heroes/medium/hochstetter-45314.png +0 -0
- package/portraits/hogans-heroes/medium/hogan-44541.png +0 -0
- package/portraits/hogans-heroes/medium/kinch-35241.png +0 -0
- package/portraits/hogans-heroes/medium/klink-23434.png +0 -0
- package/portraits/hogans-heroes/medium/lebeau-45443.png +0 -0
- package/portraits/hogans-heroes/medium/marya-53543.png +0 -0
- package/portraits/hogans-heroes/medium/newkirk-54432.png +0 -0
- package/portraits/hogans-heroes/medium/schultz-42453.png +0 -0
- package/portraits/hogans-heroes/medium/underground-55131.png +0 -0
- package/portraits/monty-python/large/announcer-44441.png +0 -0
- package/portraits/monty-python/large/arguer-35412.png +0 -0
- package/portraits/monty-python/large/bicycle-repair-man-35241.png +0 -0
- package/portraits/monty-python/large/colonel-35423.png +0 -0
- package/portraits/monty-python/large/counsellor-45341.png +0 -0
- package/portraits/monty-python/large/gumbys-23524.png +0 -0
- package/portraits/monty-python/large/nudge-43533.png +0 -0
- package/portraits/monty-python/large/praline-45413.png +0 -0
- package/portraits/monty-python/large/silly-walks-55322.png +0 -0
- package/portraits/monty-python/large/wensleydale-54451.png +0 -0
- package/portraits/monty-python/large/xim-nez-43534.png +0 -0
- package/portraits/monty-python/medium/announcer-44441.png +0 -0
- package/portraits/monty-python/medium/arguer-35412.png +0 -0
- package/portraits/monty-python/medium/bicycle-repair-man-35241.png +0 -0
- package/portraits/monty-python/medium/colonel-35423.png +0 -0
- package/portraits/monty-python/medium/counsellor-45341.png +0 -0
- package/portraits/monty-python/medium/gumbys-23524.png +0 -0
- package/portraits/monty-python/medium/nudge-43533.png +0 -0
- package/portraits/monty-python/medium/praline-45413.png +0 -0
- package/portraits/monty-python/medium/silly-walks-55322.png +0 -0
- package/portraits/monty-python/medium/wensleydale-54451.png +0 -0
- package/portraits/monty-python/medium/xim-nez-43534.png +0 -0
- package/portraits/stephen-king/large/andy-55231.png +0 -0
- package/portraits/stephen-king/large/christine-25112.png +0 -0
- package/portraits/stephen-king/large/danny-53243.png +0 -0
- package/portraits/stephen-king/large/flagg-55311.png +0 -0
- package/portraits/stephen-king/large/gaunt-54421.png +0 -0
- package/portraits/stephen-king/large/jack-44224.png +0 -0
- package/portraits/stephen-king/large/johnny-44353.png +0 -0
- package/portraits/stephen-king/large/margaret-15415.png +0 -0
- package/portraits/stephen-king/large/paul-45233.png +0 -0
- package/portraits/stephen-king/large/pennywise-54411.png +0 -0
- package/portraits/stephen-king/large/roland-35121.png +0 -0
- package/portraits/stephen-king/medium/andy-55231.png +0 -0
- package/portraits/stephen-king/medium/christine-25112.png +0 -0
- package/portraits/stephen-king/medium/danny-53243.png +0 -0
- package/portraits/stephen-king/medium/flagg-55311.png +0 -0
- package/portraits/stephen-king/medium/gaunt-54421.png +0 -0
- package/portraits/stephen-king/medium/jack-44224.png +0 -0
- package/portraits/stephen-king/medium/johnny-44353.png +0 -0
- package/portraits/stephen-king/medium/margaret-15415.png +0 -0
- package/portraits/stephen-king/medium/paul-45233.png +0 -0
- package/portraits/stephen-king/medium/pennywise-54411.png +0 -0
- package/portraits/stephen-king/medium/roland-35121.png +0 -0
- package/portraits/star-trek-tng/large/beverly-44352.png +0 -0
- package/portraits/star-trek-tng/large/data-55241.png +0 -0
- package/portraits/star-trek-tng/large/deanna-43353.png +0 -0
- package/portraits/star-trek-tng/large/geordi-54342.png +0 -0
- package/portraits/star-trek-tng/large/jean-luc-45342.png +0 -0
- package/portraits/star-trek-tng/large/kathryn-45332.png +0 -0
- package/portraits/star-trek-tng/large/miles-35342.png +0 -0
- package/portraits/star-trek-tng/large/q-53521.png +0 -0
- package/portraits/star-trek-tng/large/spock-45231.png +0 -0
- package/portraits/star-trek-tng/large/troi-44352.png +0 -0
- package/portraits/star-trek-tng/medium/beverly-44352.png +0 -0
- package/portraits/star-trek-tng/medium/data-55241.png +0 -0
- package/portraits/star-trek-tng/medium/deanna-43353.png +0 -0
- package/portraits/star-trek-tng/medium/geordi-54342.png +0 -0
- package/portraits/star-trek-tng/medium/jean-luc-45342.png +0 -0
- package/portraits/star-trek-tng/medium/kathryn-45332.png +0 -0
- package/portraits/star-trek-tng/medium/miles-35342.png +0 -0
- package/portraits/star-trek-tng/medium/q-53521.png +0 -0
- package/portraits/star-trek-tng/medium/spock-45231.png +0 -0
- package/portraits/star-trek-tng/medium/troi-44352.png +0 -0
- package/src/public/App.tsx +0 -340
- package/src/public/components/AgentLoadDialog.tsx +0 -202
- package/src/public/components/AgentPopup.tsx +0 -308
- package/src/public/components/ApprovalModal/ApprovalModal.css +0 -35
- package/src/public/components/ApprovalModal/index.tsx +0 -632
- package/src/public/components/BikeRackIndex.tsx +0 -54
- package/src/public/components/BikeRackWorkspace.tsx +0 -142
- package/src/public/components/CommandPalette.tsx +0 -555
- package/src/public/components/ConfirmDialog.tsx +0 -168
- package/src/public/components/ContextIndicator/ContextIndicator.css +0 -85
- package/src/public/components/ContextIndicator/index.tsx +0 -330
- package/src/public/components/ContextSparkline.tsx +0 -56
- package/src/public/components/ControlBar.tsx +0 -636
- package/src/public/components/DeadCodeDialog.tsx +0 -169
- package/src/public/components/DiffViewer.tsx +0 -585
- package/src/public/components/DockviewWorkspace.tsx +0 -737
- package/src/public/components/Editor.tsx +0 -630
- package/src/public/components/ErrorBoundary.tsx +0 -67
- package/src/public/components/FileTree.tsx +0 -379
- package/src/public/components/FontPicker/FontPicker.css +0 -276
- package/src/public/components/FontPicker/index.tsx +0 -430
- package/src/public/components/FullFileTree.tsx +0 -237
- package/src/public/components/HealthGauge.tsx +0 -181
- package/src/public/components/Message.tsx +0 -225
- package/src/public/components/MessageList.tsx +0 -98
- package/src/public/components/MessageView.tsx +0 -400
- package/src/public/components/ModeSwitch/ModeSwitch.css +0 -165
- package/src/public/components/ModeSwitch/index.tsx +0 -372
- package/src/public/components/PersonaHeader.tsx +0 -240
- package/src/public/components/QuickActions.tsx +0 -267
- package/src/public/components/SpanTimeline.tsx +0 -352
- package/src/public/components/StandalonePanel.tsx +0 -84
- package/src/public/components/StatsStrip.tsx +0 -162
- package/src/public/components/StreamingContent.tsx +0 -77
- package/src/public/components/SubagentSpan.tsx +0 -180
- package/src/public/components/TandemPortrait.tsx +0 -72
- package/src/public/components/ThemePalette/ThemePalette.css +0 -179
- package/src/public/components/ThemePalette/index.tsx +0 -326
- package/src/public/components/ToolCallBlock.tsx +0 -252
- package/src/public/components/ToolStack.tsx +0 -209
- package/src/public/components/ToolStatus.tsx +0 -57
- package/src/public/components/dialogs/CodeMarkersDialog.tsx +0 -169
- package/src/public/components/dialogs/ComplexityDialog.tsx +0 -163
- package/src/public/components/dialogs/DependenciesDialog.tsx +0 -120
- package/src/public/components/dialogs/HotspotsDialog.tsx +0 -451
- package/src/public/components/dialogs/ToolDialog.tsx +0 -43
- package/src/public/components/panel-registry.ts +0 -11
- package/src/public/components/panels/ACPanel.tsx +0 -93
- package/src/public/components/panels/AcceptanceCriteriaPanel.tsx +0 -104
- package/src/public/components/panels/AuditLogPanel.tsx +0 -465
- package/src/public/components/panels/BackgroundPanel.tsx +0 -115
- package/src/public/components/panels/BikeLanePanel.tsx +0 -214
- package/src/public/components/panels/ChangedPanel.tsx +0 -65
- package/src/public/components/panels/DebugPanel.tsx +0 -344
- package/src/public/components/panels/DiffsPanel.tsx +0 -155
- package/src/public/components/panels/GitPanel.tsx +0 -216
- package/src/public/components/panels/HotspotsPanel.tsx +0 -365
- package/src/public/components/panels/MessagePanel.tsx +0 -497
- package/src/public/components/panels/SettingsPanel.tsx +0 -453
- package/src/public/components/panels/SprintPanel.tsx +0 -670
- package/src/public/components/panels/TTYPanel.tsx +0 -299
- package/src/public/components/panels/TodoPanel.tsx +0 -142
- package/src/public/components/panels/WorkflowPanel.tsx +0 -224
- package/src/public/components/panels/index.ts +0 -24
- package/src/public/components/ui/alert-dialog.tsx +0 -139
- package/src/public/components/ui/badge.tsx +0 -36
- package/src/public/components/ui/button.tsx +0 -57
- package/src/public/components/ui/checkbox.tsx +0 -28
- package/src/public/components/ui/collapsible.tsx +0 -9
- package/src/public/components/ui/command.tsx +0 -151
- package/src/public/components/ui/dialog.tsx +0 -120
- package/src/public/components/ui/popover.tsx +0 -31
- package/src/public/components/ui/progress.tsx +0 -28
- package/src/public/components/ui/scroll-area.tsx +0 -46
- package/src/public/components/ui/select.tsx +0 -157
- package/src/public/components/ui/separator.tsx +0 -29
- package/src/public/components/ui/skeleton.tsx +0 -15
- package/src/public/components/ui/switch.tsx +0 -27
- package/src/public/components/ui/toggle-group.tsx +0 -59
- package/src/public/components/ui/toggle.tsx +0 -43
- package/src/public/components/ui/tooltip.tsx +0 -30
- package/src/public/contexts/ClaudeContext.tsx +0 -311
- package/src/public/contexts/MessageQueueContext.tsx +0 -143
- package/src/public/css/theme-browser.css +0 -550
- package/src/public/css/theme-system.css +0 -630
- package/src/public/hooks/index.ts +0 -49
- package/src/public/hooks/useAgentLoad.ts +0 -105
- package/src/public/hooks/useBackgroundTasks.ts +0 -131
- package/src/public/hooks/useClaude.ts +0 -234
- package/src/public/hooks/useCodeMarkers.ts +0 -101
- package/src/public/hooks/useColorScheme.ts +0 -42
- package/src/public/hooks/useCommandHistory.ts +0 -99
- package/src/public/hooks/useComplexity.ts +0 -80
- package/src/public/hooks/useDeadCode.ts +0 -99
- package/src/public/hooks/useDependencies.ts +0 -82
- package/src/public/hooks/useDiffs.ts +0 -143
- package/src/public/hooks/useFileBrowser.ts +0 -71
- package/src/public/hooks/useGitStatus.ts +0 -233
- package/src/public/hooks/useHealthScore.ts +0 -69
- package/src/public/hooks/useHotspots.ts +0 -123
- package/src/public/hooks/useLayoutPersistence.ts +0 -138
- package/src/public/hooks/useMarkdownParser.ts +0 -36
- package/src/public/hooks/useMarkerActions.ts +0 -234
- package/src/public/hooks/useMessageQueue.ts +0 -380
- package/src/public/hooks/useMessageStream.ts +0 -131
- package/src/public/hooks/usePersona.ts +0 -112
- package/src/public/hooks/usePlanModeExit.ts +0 -105
- package/src/public/hooks/useResponsiveLayout.ts +0 -173
- package/src/public/hooks/useSprint.ts +0 -147
- package/src/public/hooks/useStatsStrip.ts +0 -204
- package/src/public/hooks/useStory.ts +0 -135
- package/src/public/hooks/useSubagentHelper.ts +0 -64
- package/src/public/hooks/useSyntaxHighlighter.ts +0 -52
- package/src/public/hooks/useTabCompletion.ts +0 -124
- package/src/public/hooks/useTodos.ts +0 -93
- package/src/public/hooks/useUserAvatar.ts +0 -54
- package/src/public/index.tsx +0 -10
- package/src/public/lib/utils.ts +0 -6
- package/src/public/styles/dockview-theme.css +0 -459
- package/src/public/styles/tailwind.css +0 -4396
- package/src/public/types/electron.d.ts +0 -18
- package/src/public/types/message.ts +0 -51
- package/src/public/utils/avatar-service.ts +0 -73
- package/src/public/utils/color-presets.ts +0 -940
- package/src/public/utils/font-presets.ts +0 -362
- package/src/public/utils/formatDuration.ts +0 -14
- package/src/public/utils/markdown.ts +0 -249
- package/src/public/utils/messageFilters.ts +0 -128
- package/src/public/utils/slash-commands.ts +0 -353
- package/src/public/utils/subagent-display.ts +0 -146
- package/src/public/utils/syntax.ts +0 -219
- package/src/public/utils/toolIntentSummarizer.ts +0 -199
- package/src/public/utils/toolStackGrouper.ts +0 -106
- package/src/public/utils/toolTypeColors.ts +0 -45
|
@@ -1,105 +0,0 @@
|
|
|
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
|
-
}
|
|
@@ -1,173 +0,0 @@
|
|
|
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;
|
|
@@ -1,147 +0,0 @@
|
|
|
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 FutureEpic {
|
|
40
|
-
id: string;
|
|
41
|
-
title: string;
|
|
42
|
-
description: string;
|
|
43
|
-
estimatedPoints: number;
|
|
44
|
-
status: 'ready' | 'blocked' | 'planning';
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
export interface SprintData {
|
|
48
|
-
currentStory: SprintStory | null;
|
|
49
|
-
nextStory: SprintStory | null;
|
|
50
|
-
epics: SprintEpic[];
|
|
51
|
-
futureEpics: FutureEpic[];
|
|
52
|
-
sprint: {
|
|
53
|
-
number: number;
|
|
54
|
-
name: string;
|
|
55
|
-
done: number;
|
|
56
|
-
remaining: number;
|
|
57
|
-
inProgress: number;
|
|
58
|
-
endDate: string;
|
|
59
|
-
};
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
interface UseSprintResult {
|
|
63
|
-
data: SprintData | null;
|
|
64
|
-
isLoading: boolean;
|
|
65
|
-
error: Error | null;
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
/** WebSocket message format from /ws/sprint */
|
|
69
|
-
interface SprintMessage extends Partial<SprintData> {
|
|
70
|
-
type: 'init' | 'update';
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
export function useSprint(): UseSprintResult {
|
|
74
|
-
const [data, setData] = useState<SprintData | null>(null);
|
|
75
|
-
const [isLoading, setIsLoading] = useState(true);
|
|
76
|
-
const [error, setError] = useState<Error | null>(null);
|
|
77
|
-
const wsRef = useRef<WebSocket | null>(null);
|
|
78
|
-
const reconnectTimeoutRef = useRef<ReturnType<typeof setTimeout>>();
|
|
79
|
-
const isMountedRef = useRef(true);
|
|
80
|
-
|
|
81
|
-
useEffect(() => {
|
|
82
|
-
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
83
|
-
const wsUrl = `${protocol}//${window.location.host}/ws/sprint`;
|
|
84
|
-
|
|
85
|
-
const connect = () => {
|
|
86
|
-
// Don't reconnect if component has unmounted
|
|
87
|
-
if (!isMountedRef.current) {
|
|
88
|
-
return;
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
try {
|
|
92
|
-
wsRef.current = new WebSocket(wsUrl);
|
|
93
|
-
|
|
94
|
-
wsRef.current.onopen = () => {
|
|
95
|
-
console.debug('[useSprint] WebSocket connected');
|
|
96
|
-
};
|
|
97
|
-
|
|
98
|
-
wsRef.current.onmessage = (event) => {
|
|
99
|
-
try {
|
|
100
|
-
const msg = JSON.parse(event.data) as SprintMessage;
|
|
101
|
-
if (msg.type === 'init' || msg.type === 'update') {
|
|
102
|
-
// Extract data, excluding type field
|
|
103
|
-
const { type: _type, ...sprintData } = msg;
|
|
104
|
-
setData((prev) => {
|
|
105
|
-
if (!prev) return sprintData as SprintData;
|
|
106
|
-
// Merge partial updates
|
|
107
|
-
return { ...prev, ...sprintData } as SprintData;
|
|
108
|
-
});
|
|
109
|
-
setIsLoading(false);
|
|
110
|
-
setError(null);
|
|
111
|
-
}
|
|
112
|
-
} catch (err) {
|
|
113
|
-
console.error('[useSprint] Failed to parse message:', err);
|
|
114
|
-
}
|
|
115
|
-
};
|
|
116
|
-
|
|
117
|
-
wsRef.current.onclose = () => {
|
|
118
|
-
console.debug('[useSprint] WebSocket closed, reconnecting...');
|
|
119
|
-
reconnectTimeoutRef.current = setTimeout(connect, 2000);
|
|
120
|
-
};
|
|
121
|
-
|
|
122
|
-
wsRef.current.onerror = (err) => {
|
|
123
|
-
console.error('[useSprint] WebSocket error:', err);
|
|
124
|
-
setError(new Error('WebSocket connection failed'));
|
|
125
|
-
};
|
|
126
|
-
} catch (err) {
|
|
127
|
-
console.error('[useSprint] WebSocket init failed:', err);
|
|
128
|
-
setError(err instanceof Error ? err : new Error('Failed to connect'));
|
|
129
|
-
setIsLoading(false);
|
|
130
|
-
}
|
|
131
|
-
};
|
|
132
|
-
|
|
133
|
-
connect();
|
|
134
|
-
|
|
135
|
-
return () => {
|
|
136
|
-
isMountedRef.current = false;
|
|
137
|
-
if (reconnectTimeoutRef.current) {
|
|
138
|
-
clearTimeout(reconnectTimeoutRef.current);
|
|
139
|
-
}
|
|
140
|
-
if (wsRef.current) {
|
|
141
|
-
wsRef.current.close();
|
|
142
|
-
}
|
|
143
|
-
};
|
|
144
|
-
}, []);
|
|
145
|
-
|
|
146
|
-
return { data, isLoading, error };
|
|
147
|
-
}
|
|
@@ -1,204 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* useStatsStrip Hook
|
|
3
|
-
*
|
|
4
|
-
* React hook for fetching and subscribing to stats strip data.
|
|
5
|
-
* Story MSSCI-12699 - StatsStrip Component
|
|
6
|
-
*
|
|
7
|
-
* IPC DEPRECATED - Now uses WebSocket for all data:
|
|
8
|
-
* - /ws/context: Context percentage, used/total tokens
|
|
9
|
-
* - /ws/stats: Model name
|
|
10
|
-
* - /api/identity: PWD, Jira email, GitHub username (REST, fetched once)
|
|
11
|
-
*/
|
|
12
|
-
|
|
13
|
-
import { useState, useEffect, useCallback, useRef } from 'react';
|
|
14
|
-
|
|
15
|
-
export interface ContextData {
|
|
16
|
-
percent: number;
|
|
17
|
-
used?: number;
|
|
18
|
-
total?: number;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
export interface StatsData {
|
|
22
|
-
model: string | null;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
export interface ProjectInfoData {
|
|
26
|
-
pwd: string;
|
|
27
|
-
jiraEmail: string | null;
|
|
28
|
-
githubUsername: string | null;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
interface UseStatsStripResult {
|
|
32
|
-
context: ContextData | null;
|
|
33
|
-
stats: StatsData | null;
|
|
34
|
-
projectInfo: ProjectInfoData | null;
|
|
35
|
-
isLoading: boolean;
|
|
36
|
-
error: Error | null;
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
const WS_RECONNECT_DELAY = 2000;
|
|
40
|
-
|
|
41
|
-
export function useStatsStrip(): UseStatsStripResult {
|
|
42
|
-
const [context, setContext] = useState<ContextData | null>(null);
|
|
43
|
-
const [stats, setStats] = useState<StatsData | null>(null);
|
|
44
|
-
const [projectInfo, setProjectInfo] = useState<ProjectInfoData | null>(null);
|
|
45
|
-
const [isLoading, setIsLoading] = useState(true);
|
|
46
|
-
const [error, setError] = useState<Error | null>(null);
|
|
47
|
-
|
|
48
|
-
// Track load state for each source
|
|
49
|
-
const loadStateRef = useRef({
|
|
50
|
-
context: false,
|
|
51
|
-
stats: false,
|
|
52
|
-
projectInfo: false,
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
const checkLoadComplete = useCallback(() => {
|
|
56
|
-
const state = loadStateRef.current;
|
|
57
|
-
if (state.context && state.stats && state.projectInfo) {
|
|
58
|
-
setIsLoading(false);
|
|
59
|
-
}
|
|
60
|
-
}, []);
|
|
61
|
-
|
|
62
|
-
// WebSocket for /ws/context
|
|
63
|
-
useEffect(() => {
|
|
64
|
-
let ws: WebSocket | null = null;
|
|
65
|
-
let reconnectTimer: ReturnType<typeof setTimeout> | null = null;
|
|
66
|
-
let mounted = true;
|
|
67
|
-
|
|
68
|
-
const connect = () => {
|
|
69
|
-
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
70
|
-
ws = new WebSocket(`${protocol}//${window.location.host}/ws/context`);
|
|
71
|
-
|
|
72
|
-
ws.onopen = () => {
|
|
73
|
-
console.log('[useStatsStrip] Context WebSocket connected');
|
|
74
|
-
};
|
|
75
|
-
|
|
76
|
-
ws.onmessage = (event) => {
|
|
77
|
-
try {
|
|
78
|
-
const data = JSON.parse(event.data);
|
|
79
|
-
if (data.type === 'init' || data.type === 'update') {
|
|
80
|
-
const ctx = data.context;
|
|
81
|
-
if (ctx) {
|
|
82
|
-
setContext({
|
|
83
|
-
percent: ctx.percent ?? 0,
|
|
84
|
-
used: ctx.tokens ?? undefined,
|
|
85
|
-
total: ctx.available ? (ctx.tokens ?? 0) + ctx.available : undefined,
|
|
86
|
-
});
|
|
87
|
-
}
|
|
88
|
-
if (!loadStateRef.current.context) {
|
|
89
|
-
loadStateRef.current.context = true;
|
|
90
|
-
checkLoadComplete();
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
} catch (err) {
|
|
94
|
-
console.error('[useStatsStrip] Failed to parse context message:', err);
|
|
95
|
-
}
|
|
96
|
-
};
|
|
97
|
-
|
|
98
|
-
ws.onclose = () => {
|
|
99
|
-
if (mounted) {
|
|
100
|
-
reconnectTimer = setTimeout(connect, WS_RECONNECT_DELAY);
|
|
101
|
-
}
|
|
102
|
-
};
|
|
103
|
-
|
|
104
|
-
ws.onerror = (err) => {
|
|
105
|
-
console.error('[useStatsStrip] Context WebSocket error:', err);
|
|
106
|
-
ws?.close();
|
|
107
|
-
};
|
|
108
|
-
};
|
|
109
|
-
|
|
110
|
-
connect();
|
|
111
|
-
|
|
112
|
-
return () => {
|
|
113
|
-
mounted = false;
|
|
114
|
-
if (reconnectTimer) clearTimeout(reconnectTimer);
|
|
115
|
-
ws?.close();
|
|
116
|
-
};
|
|
117
|
-
}, [checkLoadComplete]);
|
|
118
|
-
|
|
119
|
-
// WebSocket for /ws/stats
|
|
120
|
-
useEffect(() => {
|
|
121
|
-
let ws: WebSocket | null = null;
|
|
122
|
-
let reconnectTimer: ReturnType<typeof setTimeout> | null = null;
|
|
123
|
-
let mounted = true;
|
|
124
|
-
|
|
125
|
-
const connect = () => {
|
|
126
|
-
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
127
|
-
ws = new WebSocket(`${protocol}//${window.location.host}/ws/stats`);
|
|
128
|
-
|
|
129
|
-
ws.onopen = () => {
|
|
130
|
-
console.log('[useStatsStrip] Stats WebSocket connected');
|
|
131
|
-
};
|
|
132
|
-
|
|
133
|
-
ws.onmessage = (event) => {
|
|
134
|
-
try {
|
|
135
|
-
const data = JSON.parse(event.data);
|
|
136
|
-
// Stats sends data directly (not wrapped in type)
|
|
137
|
-
setStats({ model: data.model ?? null });
|
|
138
|
-
// Also extract pwd from stats (updated on Bash tool completions)
|
|
139
|
-
if (data.pwd) {
|
|
140
|
-
setProjectInfo(prev => ({
|
|
141
|
-
pwd: data.pwd,
|
|
142
|
-
jiraEmail: prev?.jiraEmail ?? null,
|
|
143
|
-
githubUsername: prev?.githubUsername ?? null,
|
|
144
|
-
}));
|
|
145
|
-
}
|
|
146
|
-
if (!loadStateRef.current.stats) {
|
|
147
|
-
loadStateRef.current.stats = true;
|
|
148
|
-
checkLoadComplete();
|
|
149
|
-
}
|
|
150
|
-
} catch (err) {
|
|
151
|
-
console.error('[useStatsStrip] Failed to parse stats message:', err);
|
|
152
|
-
}
|
|
153
|
-
};
|
|
154
|
-
|
|
155
|
-
ws.onclose = () => {
|
|
156
|
-
if (mounted) {
|
|
157
|
-
reconnectTimer = setTimeout(connect, WS_RECONNECT_DELAY);
|
|
158
|
-
}
|
|
159
|
-
};
|
|
160
|
-
|
|
161
|
-
ws.onerror = (err) => {
|
|
162
|
-
console.error('[useStatsStrip] Stats WebSocket error:', err);
|
|
163
|
-
ws?.close();
|
|
164
|
-
};
|
|
165
|
-
};
|
|
166
|
-
|
|
167
|
-
connect();
|
|
168
|
-
|
|
169
|
-
return () => {
|
|
170
|
-
mounted = false;
|
|
171
|
-
if (reconnectTimer) clearTimeout(reconnectTimer);
|
|
172
|
-
ws?.close();
|
|
173
|
-
};
|
|
174
|
-
}, [checkLoadComplete]);
|
|
175
|
-
|
|
176
|
-
// REST for /api/identity (jiraEmail, githubUsername - fetched once)
|
|
177
|
-
// pwd comes from /ws/stats (updated on Bash tool completions)
|
|
178
|
-
useEffect(() => {
|
|
179
|
-
const fetchIdentity = async () => {
|
|
180
|
-
try {
|
|
181
|
-
const response = await fetch('/api/identity');
|
|
182
|
-
if (response.ok) {
|
|
183
|
-
const data = await response.json();
|
|
184
|
-
// Merge with existing projectInfo (preserve pwd from stats)
|
|
185
|
-
setProjectInfo(prev => ({
|
|
186
|
-
pwd: prev?.pwd ?? '',
|
|
187
|
-
jiraEmail: data.jiraEmail ?? null,
|
|
188
|
-
githubUsername: data.githubUsername ?? null,
|
|
189
|
-
}));
|
|
190
|
-
}
|
|
191
|
-
} catch (err) {
|
|
192
|
-
console.error('[useStatsStrip] Failed to fetch identity:', err);
|
|
193
|
-
setError(err instanceof Error ? err : new Error('Failed to fetch identity'));
|
|
194
|
-
} finally {
|
|
195
|
-
loadStateRef.current.projectInfo = true;
|
|
196
|
-
checkLoadComplete();
|
|
197
|
-
}
|
|
198
|
-
};
|
|
199
|
-
|
|
200
|
-
fetchIdentity();
|
|
201
|
-
}, [checkLoadComplete]);
|
|
202
|
-
|
|
203
|
-
return { context, stats, projectInfo, isLoading, error };
|
|
204
|
-
}
|