@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,299 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* TTYPanel - Terminal panel using xterm.js over WebSocket
|
|
3
|
-
*
|
|
4
|
-
* Story MSSCI-14211 - TTY Panel with xterm.js terminal emulator
|
|
5
|
-
* Epic: Epic 76 - Dockview Panel Migration
|
|
6
|
-
*
|
|
7
|
-
* Features:
|
|
8
|
-
* - Embeds xterm.js terminal in a Dockview panel
|
|
9
|
-
* - Communicates with server via /ws/pty WebSocket (works in both Electron and web mode)
|
|
10
|
-
* - Loads user's shell with environment (bash/zsh profile)
|
|
11
|
-
* - Opens in project root directory
|
|
12
|
-
* - Proper resize handling via FitAddon
|
|
13
|
-
* - Standard terminal features (colors, cursor, scrollback)
|
|
14
|
-
*/
|
|
15
|
-
|
|
16
|
-
import React, { useRef, useEffect, useState, useCallback } from 'react';
|
|
17
|
-
import { Button } from '@/components/ui/button';
|
|
18
|
-
import { Terminal } from 'xterm';
|
|
19
|
-
import { FitAddon } from 'xterm-addon-fit';
|
|
20
|
-
import 'xterm/css/xterm.css';
|
|
21
|
-
|
|
22
|
-
/** Props for TTYPanel component */
|
|
23
|
-
export interface TTYPanelProps {
|
|
24
|
-
/** Project root directory for the terminal. Defaults to current project. */
|
|
25
|
-
projectRoot?: string;
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
/** Terminal status states */
|
|
29
|
-
type TerminalStatus = 'connecting' | 'connected' | 'error' | 'exited';
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* TTYPanel - Terminal emulator panel for Cyclist
|
|
33
|
-
*/
|
|
34
|
-
export function TTYPanel({ projectRoot }: TTYPanelProps): React.ReactElement {
|
|
35
|
-
const terminalRef = useRef<HTMLDivElement>(null);
|
|
36
|
-
const xtermRef = useRef<Terminal | null>(null);
|
|
37
|
-
const fitAddonRef = useRef<FitAddon | null>(null);
|
|
38
|
-
const wsRef = useRef<WebSocket | null>(null);
|
|
39
|
-
const resizeObserverRef = useRef<ResizeObserver | null>(null);
|
|
40
|
-
const resizeTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
|
41
|
-
|
|
42
|
-
const [status, setStatus] = useState<TerminalStatus>('connecting');
|
|
43
|
-
const [errorMessage, setErrorMessage] = useState<string | null>(null);
|
|
44
|
-
|
|
45
|
-
/**
|
|
46
|
-
* Build the WebSocket URL for /ws/pty
|
|
47
|
-
*/
|
|
48
|
-
const getWsUrl = useCallback(() => {
|
|
49
|
-
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
50
|
-
return `${protocol}//${window.location.host}/ws/pty`;
|
|
51
|
-
}, []);
|
|
52
|
-
|
|
53
|
-
/**
|
|
54
|
-
* Spawn a new PTY session via WebSocket
|
|
55
|
-
*/
|
|
56
|
-
const spawnPty = useCallback(() => {
|
|
57
|
-
const ws = wsRef.current;
|
|
58
|
-
if (!ws || ws.readyState !== WebSocket.OPEN) return;
|
|
59
|
-
|
|
60
|
-
setStatus('connecting');
|
|
61
|
-
setErrorMessage(null);
|
|
62
|
-
|
|
63
|
-
const fitAddon = fitAddonRef.current;
|
|
64
|
-
const cols = fitAddon ? Math.max(fitAddon.proposeDimensions()?.cols ?? 80, 1) : 80;
|
|
65
|
-
const rows = fitAddon ? Math.max(fitAddon.proposeDimensions()?.rows ?? 24, 1) : 24;
|
|
66
|
-
|
|
67
|
-
ws.send(JSON.stringify({
|
|
68
|
-
type: 'spawn',
|
|
69
|
-
cwd: projectRoot,
|
|
70
|
-
cols,
|
|
71
|
-
rows,
|
|
72
|
-
}));
|
|
73
|
-
}, [projectRoot]);
|
|
74
|
-
|
|
75
|
-
/**
|
|
76
|
-
* Handle restart button click
|
|
77
|
-
*/
|
|
78
|
-
const handleRestart = useCallback(() => {
|
|
79
|
-
xtermRef.current?.clear();
|
|
80
|
-
xtermRef.current?.reset();
|
|
81
|
-
spawnPty();
|
|
82
|
-
}, [spawnPty]);
|
|
83
|
-
|
|
84
|
-
/**
|
|
85
|
-
* Initialize terminal and WebSocket
|
|
86
|
-
*/
|
|
87
|
-
useEffect(() => {
|
|
88
|
-
if (!terminalRef.current) return;
|
|
89
|
-
|
|
90
|
-
// Create xterm Terminal
|
|
91
|
-
const terminal = new Terminal({
|
|
92
|
-
allowProposedApi: true,
|
|
93
|
-
allowTransparency: false,
|
|
94
|
-
fontFamily: 'Menlo, Monaco, "Courier New", monospace',
|
|
95
|
-
fontSize: 14,
|
|
96
|
-
scrollback: 5000,
|
|
97
|
-
cursorBlink: true,
|
|
98
|
-
theme: {
|
|
99
|
-
background: '#1e1e1e',
|
|
100
|
-
foreground: '#d4d4d4',
|
|
101
|
-
cursor: '#d4d4d4',
|
|
102
|
-
cursorAccent: '#1e1e1e',
|
|
103
|
-
},
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
const fitAddon = new FitAddon();
|
|
107
|
-
terminal.loadAddon(fitAddon);
|
|
108
|
-
|
|
109
|
-
xtermRef.current = terminal;
|
|
110
|
-
fitAddonRef.current = fitAddon;
|
|
111
|
-
|
|
112
|
-
terminal.open(terminalRef.current);
|
|
113
|
-
|
|
114
|
-
// Initial fit after open
|
|
115
|
-
setTimeout(() => {
|
|
116
|
-
fitAddon.fit();
|
|
117
|
-
terminal.focus();
|
|
118
|
-
}, 0);
|
|
119
|
-
|
|
120
|
-
// Connect WebSocket to /ws/pty
|
|
121
|
-
const wsUrl = getWsUrl();
|
|
122
|
-
const ws = new WebSocket(wsUrl);
|
|
123
|
-
wsRef.current = ws;
|
|
124
|
-
|
|
125
|
-
ws.onopen = () => {
|
|
126
|
-
// Spawn PTY once connected
|
|
127
|
-
const cols = Math.max(fitAddon.proposeDimensions()?.cols ?? 80, 1);
|
|
128
|
-
const rows = Math.max(fitAddon.proposeDimensions()?.rows ?? 24, 1);
|
|
129
|
-
|
|
130
|
-
ws.send(JSON.stringify({
|
|
131
|
-
type: 'spawn',
|
|
132
|
-
cwd: projectRoot,
|
|
133
|
-
cols,
|
|
134
|
-
rows,
|
|
135
|
-
}));
|
|
136
|
-
};
|
|
137
|
-
|
|
138
|
-
ws.onmessage = (event) => {
|
|
139
|
-
try {
|
|
140
|
-
const msg = JSON.parse(event.data);
|
|
141
|
-
if (msg.type === 'data') {
|
|
142
|
-
terminal.write(msg.data);
|
|
143
|
-
} else if (msg.type === 'spawn') {
|
|
144
|
-
setStatus('connected');
|
|
145
|
-
} else if (msg.type === 'exit') {
|
|
146
|
-
setStatus('exited');
|
|
147
|
-
} else if (msg.type === 'error') {
|
|
148
|
-
setStatus('error');
|
|
149
|
-
setErrorMessage(msg.error);
|
|
150
|
-
}
|
|
151
|
-
} catch {
|
|
152
|
-
// Non-JSON data, write directly
|
|
153
|
-
terminal.write(event.data);
|
|
154
|
-
}
|
|
155
|
-
};
|
|
156
|
-
|
|
157
|
-
ws.onerror = () => {
|
|
158
|
-
setStatus('error');
|
|
159
|
-
setErrorMessage('WebSocket connection failed');
|
|
160
|
-
};
|
|
161
|
-
|
|
162
|
-
ws.onclose = () => {
|
|
163
|
-
if (status === 'connected') {
|
|
164
|
-
setStatus('exited');
|
|
165
|
-
}
|
|
166
|
-
};
|
|
167
|
-
|
|
168
|
-
// Send terminal input to PTY via WebSocket
|
|
169
|
-
const dataDisposable = terminal.onData((data) => {
|
|
170
|
-
if (ws.readyState === WebSocket.OPEN) {
|
|
171
|
-
ws.send(JSON.stringify({ type: 'data', data }));
|
|
172
|
-
}
|
|
173
|
-
});
|
|
174
|
-
|
|
175
|
-
// Handle terminal resize - notify PTY via WebSocket
|
|
176
|
-
const resizeDisposable = terminal.onResize(({ cols, rows }) => {
|
|
177
|
-
if (ws.readyState === WebSocket.OPEN) {
|
|
178
|
-
ws.send(JSON.stringify({ type: 'resize', cols, rows }));
|
|
179
|
-
}
|
|
180
|
-
});
|
|
181
|
-
|
|
182
|
-
// ResizeObserver for container resize handling with debounce
|
|
183
|
-
const resizeObserver = new ResizeObserver(() => {
|
|
184
|
-
if (resizeTimeoutRef.current) {
|
|
185
|
-
clearTimeout(resizeTimeoutRef.current);
|
|
186
|
-
}
|
|
187
|
-
resizeTimeoutRef.current = setTimeout(() => {
|
|
188
|
-
fitAddonRef.current?.fit();
|
|
189
|
-
}, 100);
|
|
190
|
-
});
|
|
191
|
-
|
|
192
|
-
resizeObserver.observe(terminalRef.current);
|
|
193
|
-
resizeObserverRef.current = resizeObserver;
|
|
194
|
-
|
|
195
|
-
// Cleanup on unmount
|
|
196
|
-
return () => {
|
|
197
|
-
if (resizeTimeoutRef.current) {
|
|
198
|
-
clearTimeout(resizeTimeoutRef.current);
|
|
199
|
-
}
|
|
200
|
-
resizeObserverRef.current?.disconnect();
|
|
201
|
-
|
|
202
|
-
// Kill PTY and close WebSocket
|
|
203
|
-
if (ws.readyState === WebSocket.OPEN) {
|
|
204
|
-
ws.send(JSON.stringify({ type: 'kill' }));
|
|
205
|
-
ws.close();
|
|
206
|
-
}
|
|
207
|
-
wsRef.current = null;
|
|
208
|
-
|
|
209
|
-
dataDisposable.dispose();
|
|
210
|
-
resizeDisposable.dispose();
|
|
211
|
-
terminal.dispose();
|
|
212
|
-
};
|
|
213
|
-
}, [getWsUrl, projectRoot]);
|
|
214
|
-
|
|
215
|
-
// Render status message for error/exit states
|
|
216
|
-
const renderStatusOverlay = () => {
|
|
217
|
-
if (status === 'error') {
|
|
218
|
-
return (
|
|
219
|
-
<div className="tty-overlay" data-testid="tty-error">
|
|
220
|
-
<div className="tty-overlay-content">
|
|
221
|
-
<span className="tty-overlay-icon">⚠️</span>
|
|
222
|
-
<span className="tty-overlay-message">Failed to spawn shell: {errorMessage}</span>
|
|
223
|
-
<Button
|
|
224
|
-
variant="outline"
|
|
225
|
-
size="sm"
|
|
226
|
-
className="tty-restart-button"
|
|
227
|
-
onClick={handleRestart}
|
|
228
|
-
aria-label="Restart terminal"
|
|
229
|
-
>
|
|
230
|
-
Restart
|
|
231
|
-
</Button>
|
|
232
|
-
</div>
|
|
233
|
-
</div>
|
|
234
|
-
);
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
if (status === 'exited') {
|
|
238
|
-
return (
|
|
239
|
-
<div className="tty-overlay" data-testid="tty-exited">
|
|
240
|
-
<div className="tty-overlay-content">
|
|
241
|
-
<span className="tty-overlay-icon">✓</span>
|
|
242
|
-
<span className="tty-overlay-message">Process ended</span>
|
|
243
|
-
<Button
|
|
244
|
-
variant="outline"
|
|
245
|
-
size="sm"
|
|
246
|
-
className="tty-restart-button"
|
|
247
|
-
onClick={handleRestart}
|
|
248
|
-
aria-label="New session"
|
|
249
|
-
>
|
|
250
|
-
New Session
|
|
251
|
-
</Button>
|
|
252
|
-
</div>
|
|
253
|
-
</div>
|
|
254
|
-
);
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
return null;
|
|
258
|
-
};
|
|
259
|
-
|
|
260
|
-
return (
|
|
261
|
-
<div
|
|
262
|
-
className="tty-panel"
|
|
263
|
-
data-testid="tty-panel"
|
|
264
|
-
role="region"
|
|
265
|
-
aria-label="Terminal"
|
|
266
|
-
aria-describedby="tty-keyboard-help"
|
|
267
|
-
>
|
|
268
|
-
{/* Hidden keyboard help for screen readers */}
|
|
269
|
-
<span id="tty-keyboard-help" className="sr-only">
|
|
270
|
-
Terminal panel. Use keyboard to interact with the shell.
|
|
271
|
-
</span>
|
|
272
|
-
|
|
273
|
-
{/* Status announcer for screen readers */}
|
|
274
|
-
<div
|
|
275
|
-
data-testid="tty-status"
|
|
276
|
-
aria-live="polite"
|
|
277
|
-
className="sr-only"
|
|
278
|
-
>
|
|
279
|
-
{status === 'connecting' && 'Connecting to terminal...'}
|
|
280
|
-
{status === 'connected' && 'Terminal connected'}
|
|
281
|
-
{status === 'error' && `Terminal error: ${errorMessage}`}
|
|
282
|
-
{status === 'exited' && 'Terminal session ended'}
|
|
283
|
-
</div>
|
|
284
|
-
|
|
285
|
-
{/* Terminal container */}
|
|
286
|
-
<div
|
|
287
|
-
ref={terminalRef}
|
|
288
|
-
className="tty-terminal-container"
|
|
289
|
-
data-testid="tty-terminal-container"
|
|
290
|
-
style={{ height: '100%', width: '100%' }}
|
|
291
|
-
/>
|
|
292
|
-
|
|
293
|
-
{/* Status overlay for error/exit states */}
|
|
294
|
-
{renderStatusOverlay()}
|
|
295
|
-
</div>
|
|
296
|
-
);
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
export default TTYPanel;
|
|
@@ -1,142 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* TodoPanel - Task list with progress panel
|
|
3
|
-
*
|
|
4
|
-
* Extracted from ProgressPanel as part of MSSCI-14188.
|
|
5
|
-
* Shows todos grouped by status with progress bar.
|
|
6
|
-
*
|
|
7
|
-
* Story: MSSCI-14188 - Split Progress panel into Workflow, AC, and Todo panels
|
|
8
|
-
* Epic: epic-76 (Dockview Panel Migration)
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
import React from 'react';
|
|
12
|
-
import { Skeleton } from '@/components/ui/skeleton';
|
|
13
|
-
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
|
14
|
-
import { useTodos, TodoItem } from '../../hooks/useTodos';
|
|
15
|
-
|
|
16
|
-
// =============================================================================
|
|
17
|
-
// Todo Item Component
|
|
18
|
-
// =============================================================================
|
|
19
|
-
|
|
20
|
-
function TodoItemView({ todo }: { todo: TodoItem }): React.ReactElement {
|
|
21
|
-
const statusIcon = {
|
|
22
|
-
pending: '\u25CB',
|
|
23
|
-
in_progress: '\u25CF',
|
|
24
|
-
completed: '\u2713',
|
|
25
|
-
}[todo.status];
|
|
26
|
-
|
|
27
|
-
const statusClass = `todo-item todo-${todo.status}`;
|
|
28
|
-
|
|
29
|
-
// Use activeForm for in_progress (more descriptive), content for others
|
|
30
|
-
const displayText = todo.status === 'in_progress' && todo.activeForm
|
|
31
|
-
? todo.activeForm
|
|
32
|
-
: todo.content;
|
|
33
|
-
|
|
34
|
-
return (
|
|
35
|
-
<div className={statusClass} data-testid={`todo-${todo.id}`}>
|
|
36
|
-
<span className="todo-status">{statusIcon}</span>
|
|
37
|
-
<span className="todo-subject">{displayText}</span>
|
|
38
|
-
{todo.blockedBy && todo.blockedBy.length > 0 && (
|
|
39
|
-
<TooltipProvider delayDuration={300}>
|
|
40
|
-
<Tooltip>
|
|
41
|
-
<TooltipTrigger asChild>
|
|
42
|
-
<span className="todo-blocked">
|
|
43
|
-
(blocked)
|
|
44
|
-
</span>
|
|
45
|
-
</TooltipTrigger>
|
|
46
|
-
<TooltipContent>{`Blocked by: ${todo.blockedBy.join(', ')}`}</TooltipContent>
|
|
47
|
-
</Tooltip>
|
|
48
|
-
</TooltipProvider>
|
|
49
|
-
)}
|
|
50
|
-
</div>
|
|
51
|
-
);
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
// =============================================================================
|
|
55
|
-
// TodoPanel Component
|
|
56
|
-
// =============================================================================
|
|
57
|
-
|
|
58
|
-
export function TodoPanel(): React.ReactElement {
|
|
59
|
-
const { todos, isLoading, error } = useTodos();
|
|
60
|
-
|
|
61
|
-
if (isLoading) {
|
|
62
|
-
return (
|
|
63
|
-
<div className="todo-panel loading" data-testid="todo-panel">
|
|
64
|
-
<div className="space-y-2 p-2">
|
|
65
|
-
<Skeleton className="h-3 w-full" />
|
|
66
|
-
<Skeleton className="h-4 w-5/6" />
|
|
67
|
-
<Skeleton className="h-4 w-3/4" />
|
|
68
|
-
<Skeleton className="h-4 w-4/5" />
|
|
69
|
-
</div>
|
|
70
|
-
</div>
|
|
71
|
-
);
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
if (error) {
|
|
75
|
-
return (
|
|
76
|
-
<div className="todo-panel error" data-testid="todo-panel">
|
|
77
|
-
<div className="error-message">{error.message}</div>
|
|
78
|
-
</div>
|
|
79
|
-
);
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
if (todos.length === 0) {
|
|
83
|
-
return (
|
|
84
|
-
<div className="todo-panel" data-testid="todo-panel">
|
|
85
|
-
<div className="placeholder">No active tasks</div>
|
|
86
|
-
</div>
|
|
87
|
-
);
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
// Group by status
|
|
91
|
-
const inProgress = todos.filter(t => t.status === 'in_progress');
|
|
92
|
-
const pending = todos.filter(t => t.status === 'pending');
|
|
93
|
-
const completed = todos.filter(t => t.status === 'completed');
|
|
94
|
-
|
|
95
|
-
// Calculate progress
|
|
96
|
-
const totalCount = todos.length;
|
|
97
|
-
const completedCount = completed.length;
|
|
98
|
-
const progressPercent = totalCount > 0 ? Math.round((completedCount / totalCount) * 100) : 0;
|
|
99
|
-
|
|
100
|
-
return (
|
|
101
|
-
<div className="todo-panel" data-testid="todo-panel">
|
|
102
|
-
<div className="todo-content">
|
|
103
|
-
<div className="progress-bar-container">
|
|
104
|
-
<div
|
|
105
|
-
className="progress-bar"
|
|
106
|
-
style={{ width: `${progressPercent}%` }}
|
|
107
|
-
/>
|
|
108
|
-
<span className="progress-text">{completedCount}/{totalCount}</span>
|
|
109
|
-
</div>
|
|
110
|
-
|
|
111
|
-
{inProgress.length > 0 && (
|
|
112
|
-
<div className="todo-section">
|
|
113
|
-
<h4>In Progress</h4>
|
|
114
|
-
{inProgress.map(todo => (
|
|
115
|
-
<TodoItemView key={todo.id} todo={todo} />
|
|
116
|
-
))}
|
|
117
|
-
</div>
|
|
118
|
-
)}
|
|
119
|
-
|
|
120
|
-
{pending.length > 0 && (
|
|
121
|
-
<div className="todo-section">
|
|
122
|
-
<h4>Pending</h4>
|
|
123
|
-
{pending.map(todo => (
|
|
124
|
-
<TodoItemView key={todo.id} todo={todo} />
|
|
125
|
-
))}
|
|
126
|
-
</div>
|
|
127
|
-
)}
|
|
128
|
-
|
|
129
|
-
{completed.length > 0 && (
|
|
130
|
-
<div className="todo-section todo-completed">
|
|
131
|
-
<h4>Completed ({completed.length})</h4>
|
|
132
|
-
{completed.map(todo => (
|
|
133
|
-
<TodoItemView key={todo.id} todo={todo} />
|
|
134
|
-
))}
|
|
135
|
-
</div>
|
|
136
|
-
)}
|
|
137
|
-
</div>
|
|
138
|
-
</div>
|
|
139
|
-
);
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
export default TodoPanel;
|
|
@@ -1,224 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* WorkflowPanel - BikeLane phase visualization panel
|
|
3
|
-
*
|
|
4
|
-
* Extracted from ProgressPanel as part of MSSCI-14188.
|
|
5
|
-
* Shows workflow type badge (TDD/BDD/Trivial) and phase progress.
|
|
6
|
-
* MSSCI-14300: Added stepped workflow "Step N of M" display.
|
|
7
|
-
* MSSCI-14301: Added available workflows discovery list.
|
|
8
|
-
*
|
|
9
|
-
* Story: MSSCI-14188 - Split Progress panel into Workflow, AC, and Todo panels
|
|
10
|
-
* Epic: epic-76 (Dockview Panel Migration)
|
|
11
|
-
*/
|
|
12
|
-
|
|
13
|
-
import React, { useCallback } from 'react';
|
|
14
|
-
import { Badge } from '@/components/ui/badge';
|
|
15
|
-
import { Button } from '@/components/ui/button';
|
|
16
|
-
import { Skeleton } from '@/components/ui/skeleton';
|
|
17
|
-
import { useClaudeContext } from '../../contexts/ClaudeContext';
|
|
18
|
-
import { useStory } from '../../hooks/useStory';
|
|
19
|
-
import type { WorkflowPhase, AvailableWorkflow } from '../../../story-parser.js';
|
|
20
|
-
|
|
21
|
-
// =============================================================================
|
|
22
|
-
// Helper Functions
|
|
23
|
-
// =============================================================================
|
|
24
|
-
|
|
25
|
-
function getPhaseIcon(status: 'done' | 'current' | 'pending'): string {
|
|
26
|
-
switch (status) {
|
|
27
|
-
case 'done':
|
|
28
|
-
return '\u2713';
|
|
29
|
-
case 'current':
|
|
30
|
-
return '\u25CF';
|
|
31
|
-
case 'pending':
|
|
32
|
-
default:
|
|
33
|
-
return '\u25CB';
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
function formatWorkflowType(type: string | null): string {
|
|
38
|
-
if (!type) return '\u2014';
|
|
39
|
-
|
|
40
|
-
const upperTypes = ['tdd', 'bdd'];
|
|
41
|
-
if (upperTypes.includes(type.toLowerCase())) {
|
|
42
|
-
return type.toUpperCase();
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
return type.charAt(0).toUpperCase() + type.slice(1);
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
// =============================================================================
|
|
49
|
-
// Phase Step Component (for phased workflows)
|
|
50
|
-
// =============================================================================
|
|
51
|
-
|
|
52
|
-
function PhaseStep({ phase, isLast }: { phase: WorkflowPhase; isLast: boolean }): React.ReactElement {
|
|
53
|
-
const icon = getPhaseIcon(phase.status);
|
|
54
|
-
const statusClass = `phase-step ${phase.status}`;
|
|
55
|
-
|
|
56
|
-
return (
|
|
57
|
-
<>
|
|
58
|
-
<div className={statusClass}>
|
|
59
|
-
<span className="phase-icon">{icon}</span>
|
|
60
|
-
<span className="phase-label">{phase.label}</span>
|
|
61
|
-
</div>
|
|
62
|
-
{!isLast && <span className="phase-arrow">{'\u2192'}</span>}
|
|
63
|
-
</>
|
|
64
|
-
);
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
// =============================================================================
|
|
68
|
-
// Stepped Progress Component (for stepped workflows)
|
|
69
|
-
// =============================================================================
|
|
70
|
-
|
|
71
|
-
function SteppedProgress({ phases }: { phases: WorkflowPhase[] }): React.ReactElement {
|
|
72
|
-
const total = phases.length;
|
|
73
|
-
const currentIndex = phases.findIndex(p => p.status === 'current');
|
|
74
|
-
const doneCount = phases.filter(p => p.status === 'done').length;
|
|
75
|
-
|
|
76
|
-
// If all done, current step = total; otherwise use 1-based index of current
|
|
77
|
-
const currentStep = currentIndex >= 0 ? currentIndex + 1 : (doneCount === total ? total : 1);
|
|
78
|
-
const currentPhase = currentIndex >= 0 ? phases[currentIndex] : null;
|
|
79
|
-
|
|
80
|
-
return (
|
|
81
|
-
<div className="stepped-progress">
|
|
82
|
-
<span className="stepped-counter">Step {currentStep} of {total}</span>
|
|
83
|
-
{currentPhase && (
|
|
84
|
-
<span className="stepped-current-label">{currentPhase.label}</span>
|
|
85
|
-
)}
|
|
86
|
-
</div>
|
|
87
|
-
);
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
// =============================================================================
|
|
91
|
-
// Available Workflows List (MSSCI-14301)
|
|
92
|
-
// =============================================================================
|
|
93
|
-
|
|
94
|
-
function AvailableWorkflowsList({ workflows, onStart }: {
|
|
95
|
-
workflows: AvailableWorkflow[];
|
|
96
|
-
onStart?: (workflow: AvailableWorkflow) => void;
|
|
97
|
-
}): React.ReactElement {
|
|
98
|
-
return (
|
|
99
|
-
<div className="available-workflows">
|
|
100
|
-
<div className="available-workflows-header">
|
|
101
|
-
<span className="available-workflows-title">Available Workflows ({workflows.length})</span>
|
|
102
|
-
</div>
|
|
103
|
-
<div className="available-workflows-list">
|
|
104
|
-
{workflows.map((wf) => (
|
|
105
|
-
<div
|
|
106
|
-
key={wf.name}
|
|
107
|
-
className="workflow-entry"
|
|
108
|
-
data-testid="workflow-entry"
|
|
109
|
-
data-workflow-entry-type={wf.type}
|
|
110
|
-
>
|
|
111
|
-
<div className="workflow-entry-header">
|
|
112
|
-
<span className="workflow-entry-name">{wf.name}</span>
|
|
113
|
-
<Badge variant="outline" className="workflow-entry-type-badge">
|
|
114
|
-
{wf.type}
|
|
115
|
-
</Badge>
|
|
116
|
-
</div>
|
|
117
|
-
<div className="workflow-entry-description">{wf.description}</div>
|
|
118
|
-
{wf.type === 'stepped' && (
|
|
119
|
-
<div className="workflow-entry-hint">/workflow start {wf.name}</div>
|
|
120
|
-
)}
|
|
121
|
-
<div className="workflow-entry-footer">
|
|
122
|
-
<Button
|
|
123
|
-
variant="ghost"
|
|
124
|
-
size="sm"
|
|
125
|
-
className="workflow-start-button"
|
|
126
|
-
data-testid="workflow-start-button"
|
|
127
|
-
onClick={() => onStart?.(wf)}
|
|
128
|
-
>
|
|
129
|
-
Start
|
|
130
|
-
</Button>
|
|
131
|
-
</div>
|
|
132
|
-
</div>
|
|
133
|
-
))}
|
|
134
|
-
</div>
|
|
135
|
-
</div>
|
|
136
|
-
);
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
// =============================================================================
|
|
140
|
-
// WorkflowPanel Component
|
|
141
|
-
// =============================================================================
|
|
142
|
-
|
|
143
|
-
export function WorkflowPanel(): React.ReactElement {
|
|
144
|
-
const { story, isLoading, error, availableWorkflows } = useStory();
|
|
145
|
-
const { send, isConnected } = useClaudeContext();
|
|
146
|
-
|
|
147
|
-
const handleStartWorkflow = useCallback((wf: AvailableWorkflow) => {
|
|
148
|
-
if (!isConnected) return;
|
|
149
|
-
send(`/workflow start ${wf.name}`);
|
|
150
|
-
}, [send, isConnected]);
|
|
151
|
-
|
|
152
|
-
if (isLoading) {
|
|
153
|
-
return (
|
|
154
|
-
<div className="workflow-panel loading" data-testid="workflow-panel">
|
|
155
|
-
<div className="space-y-2 p-2">
|
|
156
|
-
<Skeleton className="h-6 w-16 rounded-full" />
|
|
157
|
-
<div className="flex gap-2 items-center">
|
|
158
|
-
<Skeleton className="h-4 w-12" />
|
|
159
|
-
<Skeleton className="h-4 w-12" />
|
|
160
|
-
<Skeleton className="h-4 w-12" />
|
|
161
|
-
</div>
|
|
162
|
-
</div>
|
|
163
|
-
</div>
|
|
164
|
-
);
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
if (error) {
|
|
168
|
-
return (
|
|
169
|
-
<div className="workflow-panel error" data-testid="workflow-panel">
|
|
170
|
-
<div className="error-message">{error.message}</div>
|
|
171
|
-
</div>
|
|
172
|
-
);
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
const workflowType = story?.workflow ?? null;
|
|
176
|
-
const phases = story?.workflowPhases ?? null;
|
|
177
|
-
const isStepped = story?.workflowType === 'stepped';
|
|
178
|
-
|
|
179
|
-
if (!workflowType && (!phases || phases.length === 0)) {
|
|
180
|
-
// MSSCI-14301: Show available workflows when no active workflow
|
|
181
|
-
if (availableWorkflows && availableWorkflows.length > 0) {
|
|
182
|
-
return (
|
|
183
|
-
<div className="workflow-panel" data-testid="workflow-panel">
|
|
184
|
-
<AvailableWorkflowsList workflows={availableWorkflows} onStart={handleStartWorkflow} />
|
|
185
|
-
</div>
|
|
186
|
-
);
|
|
187
|
-
}
|
|
188
|
-
return (
|
|
189
|
-
<div className="workflow-panel" data-testid="workflow-panel">
|
|
190
|
-
<div className="placeholder">No active workflow</div>
|
|
191
|
-
</div>
|
|
192
|
-
);
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
const formattedType = formatWorkflowType(workflowType);
|
|
196
|
-
|
|
197
|
-
return (
|
|
198
|
-
<div className="workflow-panel" data-testid="workflow-panel">
|
|
199
|
-
<div className="workflow-content">
|
|
200
|
-
<Badge variant="secondary" className="workflow-type-badge" data-workflow-type={workflowType || ''}>
|
|
201
|
-
{formattedType}
|
|
202
|
-
</Badge>
|
|
203
|
-
|
|
204
|
-
{phases && phases.length > 0 && (
|
|
205
|
-
isStepped ? (
|
|
206
|
-
<SteppedProgress phases={phases} />
|
|
207
|
-
) : (
|
|
208
|
-
<div className="phase-progress">
|
|
209
|
-
{phases.map((phase, index) => (
|
|
210
|
-
<PhaseStep
|
|
211
|
-
key={phase.name}
|
|
212
|
-
phase={phase}
|
|
213
|
-
isLast={index === phases.length - 1}
|
|
214
|
-
/>
|
|
215
|
-
))}
|
|
216
|
-
</div>
|
|
217
|
-
)
|
|
218
|
-
)}
|
|
219
|
-
</div>
|
|
220
|
-
</div>
|
|
221
|
-
);
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
export default WorkflowPanel;
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Panel Components Index
|
|
3
|
-
*
|
|
4
|
-
* Story MSSCI-12717 - React Migration
|
|
5
|
-
* Story MSSCI-14188 - Split Progress panel into Workflow, AC, and Todo panels
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
export { MessagePanel } from './MessagePanel';
|
|
9
|
-
export { SprintPanel, EnhancedSprintPanel } from './SprintPanel';
|
|
10
|
-
export { GitPanel } from './GitPanel';
|
|
11
|
-
export { WorkflowPanel } from './WorkflowPanel';
|
|
12
|
-
export { ACPanel } from './ACPanel';
|
|
13
|
-
export { TodoPanel } from './TodoPanel';
|
|
14
|
-
export { BackgroundPanel } from './BackgroundPanel';
|
|
15
|
-
export { ChangedPanel } from './ChangedPanel';
|
|
16
|
-
export { DiffsPanel } from './DiffsPanel';
|
|
17
|
-
export { DebugPanel } from './DebugPanel';
|
|
18
|
-
export { SettingsPanel } from './SettingsPanel';
|
|
19
|
-
export { AuditLogPanel } from './AuditLogPanel';
|
|
20
|
-
export { TTYPanel } from './TTYPanel';
|
|
21
|
-
|
|
22
|
-
// Legacy exports - kept for backwards compatibility and tests
|
|
23
|
-
export { AcceptanceCriteriaPanel, ConnectedAcceptanceCriteriaPanel } from './AcceptanceCriteriaPanel';
|
|
24
|
-
export { BikeLanePanel, ConnectedBikeLanePanel } from './BikeLanePanel';
|