@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,115 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* BackgroundPanel - Display background tasks
|
|
3
|
-
*
|
|
4
|
-
* Story MSSCI-12717 - React Migration
|
|
5
|
-
* Story MSSCI-12784 - Timer updates in real-time with accurate completion times
|
|
6
|
-
*
|
|
7
|
-
* Timer strategy:
|
|
8
|
-
* - On mount: fetch all tasks from backend (accurate snapshot)
|
|
9
|
-
* - Pending tasks: estimate elapsed from startedAt, update via interval
|
|
10
|
-
* - Completed tasks: use authoritative durationMs from backend
|
|
11
|
-
* - IPC events provide real-time updates with accurate timing
|
|
12
|
-
*/
|
|
13
|
-
|
|
14
|
-
import React, { useState, useEffect } from 'react';
|
|
15
|
-
import { Button } from '@/components/ui/button';
|
|
16
|
-
import { useBackgroundTasks, BackgroundTask } from '../../hooks/useBackgroundTasks';
|
|
17
|
-
|
|
18
|
-
/**
|
|
19
|
-
* Format milliseconds into human-readable duration
|
|
20
|
-
*/
|
|
21
|
-
function formatDuration(ms: number): string {
|
|
22
|
-
if (ms < 1000) return `${ms}ms`;
|
|
23
|
-
if (ms < 60000) return `${Math.floor(ms / 1000)}s`;
|
|
24
|
-
return `${Math.floor(ms / 60000)}m`;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
function TaskItem({ task, tick }: { task: BackgroundTask; tick: number }): React.ReactElement {
|
|
28
|
-
const isPending = task.status === 'pending';
|
|
29
|
-
const statusIcon = isPending ? '...' : (task.success ? 'v' : 'x');
|
|
30
|
-
const statusClass = `task-item task-${task.status}${task.success === false ? ' task-error' : ''}`;
|
|
31
|
-
|
|
32
|
-
// For completed tasks, use authoritative durationMs from backend
|
|
33
|
-
// For pending tasks, estimate from startedAt (tick triggers re-render)
|
|
34
|
-
let elapsedStr: string;
|
|
35
|
-
if (task.status === 'completed' && task.durationMs !== undefined) {
|
|
36
|
-
// Completed: use accurate duration from backend
|
|
37
|
-
elapsedStr = formatDuration(task.durationMs);
|
|
38
|
-
} else {
|
|
39
|
-
// Pending: estimate elapsed time (tick forces recalculation)
|
|
40
|
-
void tick; // Intentionally used to trigger re-render
|
|
41
|
-
elapsedStr = formatDuration(Date.now() - task.startedAt);
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
return (
|
|
45
|
-
<div className={statusClass} data-testid={`task-${task.taskId}`}>
|
|
46
|
-
<span className="task-status">{statusIcon}</span>
|
|
47
|
-
<div className="task-content">
|
|
48
|
-
<span className="task-description">{task.description}</span>
|
|
49
|
-
<span className="task-type">{task.subagentType}</span>
|
|
50
|
-
</div>
|
|
51
|
-
<span className="task-elapsed">{elapsedStr}</span>
|
|
52
|
-
</div>
|
|
53
|
-
);
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
export function BackgroundPanel(): React.ReactElement {
|
|
57
|
-
const { tasks, pendingCount, completedCount, clearCompleted } = useBackgroundTasks();
|
|
58
|
-
const [tick, setTick] = useState(0);
|
|
59
|
-
|
|
60
|
-
// Timer for live updates - only runs when there are pending tasks and panel is visible
|
|
61
|
-
useEffect(() => {
|
|
62
|
-
if (pendingCount === 0) return;
|
|
63
|
-
|
|
64
|
-
const interval = setInterval(() => {
|
|
65
|
-
setTick(t => t + 1);
|
|
66
|
-
}, 1000);
|
|
67
|
-
|
|
68
|
-
return () => clearInterval(interval);
|
|
69
|
-
}, [pendingCount]);
|
|
70
|
-
|
|
71
|
-
if (tasks.length === 0) {
|
|
72
|
-
return (
|
|
73
|
-
<div className="background-panel empty" data-testid="background-panel">
|
|
74
|
-
<div className="placeholder">No subagent tasks</div>
|
|
75
|
-
</div>
|
|
76
|
-
);
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
// Sort: pending first, then by start time (newest first)
|
|
80
|
-
const sortedTasks = [...tasks].sort((a, b) => {
|
|
81
|
-
if (a.status === 'pending' && b.status !== 'pending') return -1;
|
|
82
|
-
if (a.status !== 'pending' && b.status === 'pending') return 1;
|
|
83
|
-
return b.startedAt - a.startedAt;
|
|
84
|
-
});
|
|
85
|
-
|
|
86
|
-
return (
|
|
87
|
-
<div className="background-panel" data-testid="background-panel">
|
|
88
|
-
<div className="panel-header">
|
|
89
|
-
<span className="task-counts">
|
|
90
|
-
{pendingCount > 0 && <span className="pending-count">{pendingCount} running</span>}
|
|
91
|
-
{completedCount > 0 && <span className="completed-count">{completedCount} done</span>}
|
|
92
|
-
</span>
|
|
93
|
-
{completedCount > 0 && (
|
|
94
|
-
<Button
|
|
95
|
-
variant="ghost"
|
|
96
|
-
size="sm"
|
|
97
|
-
type="button"
|
|
98
|
-
className="clear-button"
|
|
99
|
-
onClick={clearCompleted}
|
|
100
|
-
>
|
|
101
|
-
Clear
|
|
102
|
-
</Button>
|
|
103
|
-
)}
|
|
104
|
-
</div>
|
|
105
|
-
|
|
106
|
-
<div className="task-list">
|
|
107
|
-
{sortedTasks.map(task => (
|
|
108
|
-
<TaskItem key={task.taskId} task={task} tick={tick} />
|
|
109
|
-
))}
|
|
110
|
-
</div>
|
|
111
|
-
</div>
|
|
112
|
-
);
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
export default BackgroundPanel;
|
|
@@ -1,214 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* BikeLanePanel - Display workflow visualization
|
|
3
|
-
*
|
|
4
|
-
* Story MSSCI-12849 - Missing AC & BikeLane panels in Progress tab
|
|
5
|
-
*
|
|
6
|
-
* Reference: Deleted vanilla JS in commit 9aea4f371
|
|
7
|
-
* - js/sidebar/bikelane.js
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
import React from 'react';
|
|
11
|
-
import { Badge } from '@/components/ui/badge';
|
|
12
|
-
import { Skeleton } from '@/components/ui/skeleton';
|
|
13
|
-
import type { WorkflowPhase } from '../../../story-parser.js';
|
|
14
|
-
import { useStory } from '../../hooks/useStory.js';
|
|
15
|
-
|
|
16
|
-
export interface PhaseHistoryEntry {
|
|
17
|
-
phase: string;
|
|
18
|
-
agent: string;
|
|
19
|
-
status: 'done' | 'current' | 'pending';
|
|
20
|
-
duration?: string;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
export interface BikeLanePanelProps {
|
|
24
|
-
workflowType: string | null;
|
|
25
|
-
phases: WorkflowPhase[] | null;
|
|
26
|
-
phaseHistory?: PhaseHistoryEntry[] | null;
|
|
27
|
-
collapsed?: boolean;
|
|
28
|
-
onToggle?: () => void;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* Format workflow type for display
|
|
33
|
-
* TDD, BDD -> uppercase
|
|
34
|
-
* trivial, others -> Title case
|
|
35
|
-
*/
|
|
36
|
-
function formatWorkflowType(type: string | null): string {
|
|
37
|
-
if (!type) return '—';
|
|
38
|
-
|
|
39
|
-
const upperTypes = ['tdd', 'bdd'];
|
|
40
|
-
if (upperTypes.includes(type.toLowerCase())) {
|
|
41
|
-
return type.toUpperCase();
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
return type.charAt(0).toUpperCase() + type.slice(1);
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* Get icon for phase status
|
|
49
|
-
*/
|
|
50
|
-
function getPhaseIcon(status: 'done' | 'current' | 'pending'): string {
|
|
51
|
-
switch (status) {
|
|
52
|
-
case 'done':
|
|
53
|
-
return '✓';
|
|
54
|
-
case 'current':
|
|
55
|
-
return '●';
|
|
56
|
-
case 'pending':
|
|
57
|
-
default:
|
|
58
|
-
return '○';
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
/**
|
|
63
|
-
* Phase step in progress visualization
|
|
64
|
-
*/
|
|
65
|
-
function PhaseStep({ phase, isLast }: { phase: WorkflowPhase; isLast: boolean }): React.ReactElement {
|
|
66
|
-
const icon = getPhaseIcon(phase.status);
|
|
67
|
-
const statusClass = `phase-step ${phase.status}`;
|
|
68
|
-
|
|
69
|
-
return (
|
|
70
|
-
<>
|
|
71
|
-
<div className={statusClass}>
|
|
72
|
-
<span className="phase-icon">{icon}</span>
|
|
73
|
-
<span className="phase-label">{phase.label}</span>
|
|
74
|
-
</div>
|
|
75
|
-
{!isLast && <span className="phase-arrow">→</span>}
|
|
76
|
-
</>
|
|
77
|
-
);
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
/**
|
|
81
|
-
* Phase history item in timeline
|
|
82
|
-
*/
|
|
83
|
-
function PhaseHistoryItem({ entry }: { entry: PhaseHistoryEntry }): React.ReactElement {
|
|
84
|
-
const icon = getPhaseIcon(entry.status);
|
|
85
|
-
const statusClass = `phase-history-item ${entry.status}`;
|
|
86
|
-
|
|
87
|
-
const durationText = entry.status === 'current'
|
|
88
|
-
? 'in progress'
|
|
89
|
-
: entry.status === 'pending'
|
|
90
|
-
? 'pending'
|
|
91
|
-
: entry.duration || '—';
|
|
92
|
-
|
|
93
|
-
return (
|
|
94
|
-
<div className={statusClass}>
|
|
95
|
-
<span className="history-icon">{icon}</span>
|
|
96
|
-
<span className="history-phase">{entry.phase.toUpperCase()}</span>
|
|
97
|
-
<span className="history-agent">{entry.agent}</span>
|
|
98
|
-
<span className="history-duration">{durationText}</span>
|
|
99
|
-
</div>
|
|
100
|
-
);
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
/**
|
|
104
|
-
* BikeLanePanel - Displays workflow progress and phase history
|
|
105
|
-
*/
|
|
106
|
-
export function BikeLanePanel({
|
|
107
|
-
workflowType,
|
|
108
|
-
phases,
|
|
109
|
-
phaseHistory,
|
|
110
|
-
collapsed = false,
|
|
111
|
-
onToggle,
|
|
112
|
-
}: BikeLanePanelProps): React.ReactElement {
|
|
113
|
-
// Handle empty state
|
|
114
|
-
if (!workflowType && (!phases || phases.length === 0)) {
|
|
115
|
-
return (
|
|
116
|
-
<div className="bikelane-panel empty hidden" data-testid="bikelane-panel">
|
|
117
|
-
<div className="placeholder">No active workflow</div>
|
|
118
|
-
</div>
|
|
119
|
-
);
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
const formattedType = formatWorkflowType(workflowType);
|
|
123
|
-
|
|
124
|
-
// Handle collapsed state
|
|
125
|
-
if (collapsed) {
|
|
126
|
-
return (
|
|
127
|
-
<div className="bikelane-panel collapsed" data-testid="bikelane-panel">
|
|
128
|
-
<div className="bikelane-header" onClick={onToggle}>
|
|
129
|
-
<Badge variant="secondary" className="workflow-type-badge" data-workflow-type={workflowType || ''}>
|
|
130
|
-
{formattedType}
|
|
131
|
-
</Badge>
|
|
132
|
-
<span className="bikelane-expand">▶</span>
|
|
133
|
-
</div>
|
|
134
|
-
</div>
|
|
135
|
-
);
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
return (
|
|
139
|
-
<div className="bikelane-panel" data-testid="bikelane-panel">
|
|
140
|
-
<div className="bikelane-header" onClick={onToggle}>
|
|
141
|
-
<Badge variant="secondary" className="workflow-type-badge" data-workflow-type={workflowType || ''}>
|
|
142
|
-
{formattedType}
|
|
143
|
-
</Badge>
|
|
144
|
-
{onToggle && <span className="bikelane-expand">▼</span>}
|
|
145
|
-
</div>
|
|
146
|
-
|
|
147
|
-
{/* Phase progress visualization */}
|
|
148
|
-
{phases && phases.length > 0 && (
|
|
149
|
-
<div className="phase-progress">
|
|
150
|
-
{phases.map((phase, index) => (
|
|
151
|
-
<PhaseStep
|
|
152
|
-
key={phase.name}
|
|
153
|
-
phase={phase}
|
|
154
|
-
isLast={index === phases.length - 1}
|
|
155
|
-
/>
|
|
156
|
-
))}
|
|
157
|
-
</div>
|
|
158
|
-
)}
|
|
159
|
-
|
|
160
|
-
{/* Phase history timeline */}
|
|
161
|
-
{phaseHistory && phaseHistory.length > 0 && (
|
|
162
|
-
<div className="phase-history">
|
|
163
|
-
<div className="phase-history-list">
|
|
164
|
-
{phaseHistory.map((entry, index) => (
|
|
165
|
-
<PhaseHistoryItem key={index} entry={entry} />
|
|
166
|
-
))}
|
|
167
|
-
</div>
|
|
168
|
-
</div>
|
|
169
|
-
)}
|
|
170
|
-
</div>
|
|
171
|
-
);
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
/**
|
|
175
|
-
* ConnectedBikeLanePanel - Self-contained panel that fetches its own data
|
|
176
|
-
*
|
|
177
|
-
* Used by DockingWorkspace via registerPanelComponent.
|
|
178
|
-
* Subscribes to story updates so it re-renders when session file changes.
|
|
179
|
-
*/
|
|
180
|
-
export function ConnectedBikeLanePanel(): React.ReactElement {
|
|
181
|
-
const { story, isLoading, error } = useStory();
|
|
182
|
-
|
|
183
|
-
if (isLoading) {
|
|
184
|
-
return (
|
|
185
|
-
<div className="bikelane-panel loading" data-testid="bikelane-panel">
|
|
186
|
-
<div className="space-y-2 p-2">
|
|
187
|
-
<Skeleton className="h-6 w-16 rounded-full" />
|
|
188
|
-
<div className="flex gap-2 items-center">
|
|
189
|
-
<Skeleton className="h-4 w-12" />
|
|
190
|
-
<Skeleton className="h-4 w-12" />
|
|
191
|
-
<Skeleton className="h-4 w-12" />
|
|
192
|
-
</div>
|
|
193
|
-
</div>
|
|
194
|
-
</div>
|
|
195
|
-
);
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
if (error) {
|
|
199
|
-
return (
|
|
200
|
-
<div className="bikelane-panel error" data-testid="bikelane-panel">
|
|
201
|
-
<div className="error-message">{error.message}</div>
|
|
202
|
-
</div>
|
|
203
|
-
);
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
return (
|
|
207
|
-
<BikeLanePanel
|
|
208
|
-
workflowType={story?.workflow ?? null}
|
|
209
|
-
phases={story?.workflowPhases ?? null}
|
|
210
|
-
/>
|
|
211
|
-
);
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
export default BikeLanePanel;
|
|
@@ -1,65 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ChangedPanel - Full file tree with changed file highlighting
|
|
3
|
-
*
|
|
4
|
-
* Story MSSCI-12717 - React Migration
|
|
5
|
-
* ADR-0020 - Changed to use git as source of truth instead of /ws/diffs
|
|
6
|
-
*
|
|
7
|
-
* Shows full project directory tree with changed files highlighted.
|
|
8
|
-
* Uses /api/files for tree structure and /ws/git for change status.
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
import React, { useCallback, useMemo } from 'react';
|
|
12
|
-
import { FullFileTree } from '../FullFileTree';
|
|
13
|
-
import { useGitStatus } from '../../hooks/useGitStatus';
|
|
14
|
-
import type { FileStatus } from '../FileTree';
|
|
15
|
-
import type { DirectoryEntry } from '../../hooks/useFileBrowser';
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
* Map git status code to FileStatus
|
|
19
|
-
*/
|
|
20
|
-
function gitStatusToFileStatus(gitStatus: string): FileStatus {
|
|
21
|
-
const indexStatus = gitStatus[0] || ' ';
|
|
22
|
-
const workTreeStatus = gitStatus[1] || ' ';
|
|
23
|
-
|
|
24
|
-
if (indexStatus === 'D' || workTreeStatus === 'D') return 'deleted';
|
|
25
|
-
if (indexStatus === '?' || indexStatus === 'A') return 'created';
|
|
26
|
-
return 'modified';
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
export function ChangedPanel(): React.ReactElement {
|
|
30
|
-
const { repos } = useGitStatus();
|
|
31
|
-
|
|
32
|
-
// Build a Map<filePath, FileStatus> from all dirty files
|
|
33
|
-
const changedFiles = useMemo(() => {
|
|
34
|
-
const map = new Map<string, FileStatus>();
|
|
35
|
-
for (const repo of repos) {
|
|
36
|
-
for (const file of repo.files) {
|
|
37
|
-
// Use full path from repo for matching against /api/files paths
|
|
38
|
-
const fullPath = repos.length > 1
|
|
39
|
-
? `${repo.path}/${file.path}`
|
|
40
|
-
: `${repo.path}/${file.path}`;
|
|
41
|
-
map.set(fullPath, gitStatusToFileStatus(file.status));
|
|
42
|
-
// Also store relative path for fallback matching
|
|
43
|
-
map.set(file.path, gitStatusToFileStatus(file.status));
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
return map;
|
|
47
|
-
}, [repos]);
|
|
48
|
-
|
|
49
|
-
const handleFileClick = useCallback((entry: DirectoryEntry, status?: FileStatus) => {
|
|
50
|
-
// Open file in editor via API
|
|
51
|
-
fetch('/api/files/edit', {
|
|
52
|
-
method: 'POST',
|
|
53
|
-
headers: { 'Content-Type': 'application/json' },
|
|
54
|
-
body: JSON.stringify({ path: entry.path }),
|
|
55
|
-
}).catch(err => console.error('[ChangedPanel] Failed to open file:', err));
|
|
56
|
-
}, []);
|
|
57
|
-
|
|
58
|
-
return (
|
|
59
|
-
<div className="changed-panel" data-testid="changed-panel">
|
|
60
|
-
<FullFileTree changedFiles={changedFiles} onFileClick={handleFileClick} />
|
|
61
|
-
</div>
|
|
62
|
-
);
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
export default ChangedPanel;
|