@pennyfarthing/core 11.3.8 → 11.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/package.json +4 -1
- package/packages/core/dist/public/css/react.css +1 -1
- package/packages/core/dist/public/js/react/react.js +24 -24
- package/packages/core/src/public/App.tsx +356 -0
- package/packages/core/src/public/components/AgentLoadDialog.tsx +202 -0
- package/packages/core/src/public/components/AgentPopup.tsx +308 -0
- package/packages/core/src/public/components/ApprovalModal/ApprovalModal.css +35 -0
- package/packages/core/src/public/components/ApprovalModal/index.tsx +632 -0
- package/packages/core/src/public/components/BikeRackIndex.tsx +53 -0
- package/packages/core/src/public/components/BikeRackWorkspace.tsx +217 -0
- package/packages/core/src/public/components/CommandPalette.tsx +554 -0
- package/packages/core/src/public/components/ConfirmDialog.tsx +168 -0
- package/packages/core/src/public/components/ContextIndicator/ContextIndicator.css +85 -0
- package/packages/core/src/public/components/ContextIndicator/index.tsx +330 -0
- package/packages/core/src/public/components/ContextSparkline.tsx +56 -0
- package/packages/core/src/public/components/ControlBar.tsx +636 -0
- package/packages/core/src/public/components/DeadCodeDialog.tsx +169 -0
- package/packages/core/src/public/components/DiffViewer.tsx +585 -0
- package/packages/core/src/public/components/DockviewWorkspace.tsx +749 -0
- package/packages/core/src/public/components/Editor.tsx +630 -0
- package/packages/core/src/public/components/ErrorBoundary.tsx +67 -0
- package/packages/core/src/public/components/FileTree.tsx +379 -0
- package/packages/core/src/public/components/FontPicker/FontPicker.css +276 -0
- package/packages/core/src/public/components/FontPicker/index.tsx +430 -0
- package/packages/core/src/public/components/FullFileTree.tsx +237 -0
- package/packages/core/src/public/components/HealthGauge.tsx +181 -0
- package/packages/core/src/public/components/Message.tsx +225 -0
- package/packages/core/src/public/components/MessageList.tsx +98 -0
- package/packages/core/src/public/components/MessageView.tsx +400 -0
- package/packages/core/src/public/components/ModeSwitch/ModeSwitch.css +165 -0
- package/packages/core/src/public/components/ModeSwitch/index.tsx +372 -0
- package/packages/core/src/public/components/PersonaHeader.tsx +242 -0
- package/packages/core/src/public/components/ProjectInfoBar.tsx +45 -0
- package/packages/core/src/public/components/QuickActions.tsx +267 -0
- package/packages/core/src/public/components/SpanTimeline.tsx +352 -0
- package/packages/core/src/public/components/StandalonePanel.tsx +82 -0
- package/packages/core/src/public/components/StatsStrip.tsx +162 -0
- package/packages/core/src/public/components/StreamingContent.tsx +77 -0
- package/packages/core/src/public/components/SubagentSpan.tsx +180 -0
- package/packages/core/src/public/components/TandemPortrait.tsx +72 -0
- package/packages/core/src/public/components/ThemePalette/ThemePalette.css +179 -0
- package/packages/core/src/public/components/ThemePalette/index.tsx +326 -0
- package/packages/core/src/public/components/ToolCallBlock.tsx +252 -0
- package/packages/core/src/public/components/ToolStack.tsx +209 -0
- package/packages/core/src/public/components/ToolStatus.tsx +57 -0
- package/packages/core/src/public/components/dialogs/CodeMarkersDialog.tsx +169 -0
- package/packages/core/src/public/components/dialogs/ComplexityDialog.tsx +163 -0
- package/packages/core/src/public/components/dialogs/DependenciesDialog.tsx +120 -0
- package/packages/core/src/public/components/dialogs/HotspotsDialog.tsx +451 -0
- package/packages/core/src/public/components/dialogs/ToolDialog.tsx +43 -0
- package/packages/core/src/public/components/panel-registry.ts +13 -0
- package/packages/core/src/public/components/panels/ACPanel.tsx +93 -0
- package/packages/core/src/public/components/panels/AcceptanceCriteriaPanel.tsx +104 -0
- package/packages/core/src/public/components/panels/AuditLogPanel.tsx +489 -0
- package/packages/core/src/public/components/panels/BackgroundPanel.tsx +115 -0
- package/packages/core/src/public/components/panels/BikeLanePanel.tsx +214 -0
- package/packages/core/src/public/components/panels/DebugPanel.tsx +344 -0
- package/packages/core/src/public/components/panels/DiffView.tsx +109 -0
- package/packages/core/src/public/components/panels/DiffsPanel.tsx +56 -0
- package/packages/core/src/public/components/panels/GitPanel.tsx +260 -0
- package/packages/core/src/public/components/panels/HotspotsPanel.tsx +365 -0
- package/packages/core/src/public/components/panels/MessageFeed.tsx +39 -0
- package/packages/core/src/public/components/panels/MessagePanel.tsx +497 -0
- package/packages/core/src/public/components/panels/ProgressPanel.tsx +189 -0
- package/packages/core/src/public/components/panels/SettingsPanel.tsx +361 -0
- package/packages/core/src/public/components/panels/SprintPanel.tsx +723 -0
- package/packages/core/src/public/components/panels/TandemPanel.tsx +104 -0
- package/packages/core/src/public/components/panels/TaskTracker.tsx +48 -0
- package/packages/core/src/public/components/panels/TeamPanel.tsx +64 -0
- package/packages/core/src/public/components/panels/TeamRoster.tsx +67 -0
- package/packages/core/src/public/components/panels/TodoPanel.tsx +142 -0
- package/packages/core/src/public/components/panels/WorkflowPanel.tsx +224 -0
- package/packages/core/src/public/components/panels/index.ts +24 -0
- package/packages/core/src/public/components/ui/alert-dialog.tsx +139 -0
- package/packages/core/src/public/components/ui/badge.tsx +36 -0
- package/packages/core/src/public/components/ui/button.tsx +57 -0
- package/packages/core/src/public/components/ui/checkbox.tsx +28 -0
- package/packages/core/src/public/components/ui/collapsible.tsx +9 -0
- package/packages/core/src/public/components/ui/command.tsx +151 -0
- package/packages/core/src/public/components/ui/dialog.tsx +120 -0
- package/packages/core/src/public/components/ui/popover.tsx +31 -0
- package/packages/core/src/public/components/ui/progress.tsx +28 -0
- package/packages/core/src/public/components/ui/scroll-area.tsx +46 -0
- package/packages/core/src/public/components/ui/select.tsx +157 -0
- package/packages/core/src/public/components/ui/separator.tsx +29 -0
- package/packages/core/src/public/components/ui/skeleton.tsx +15 -0
- package/packages/core/src/public/components/ui/switch.tsx +27 -0
- package/packages/core/src/public/components/ui/toggle-group.tsx +59 -0
- package/packages/core/src/public/components/ui/toggle.tsx +43 -0
- package/packages/core/src/public/components/ui/tooltip.tsx +30 -0
- package/packages/core/src/public/contexts/ClaudeContext.tsx +311 -0
- package/packages/core/src/public/contexts/MessageQueueContext.tsx +143 -0
- package/packages/core/src/public/css/theme-browser.css +550 -0
- package/packages/core/src/public/css/theme-system.css +630 -0
- package/packages/core/src/public/hooks/index.ts +49 -0
- package/packages/core/src/public/hooks/useAgentLoad.ts +105 -0
- package/packages/core/src/public/hooks/useBackgroundTasks.ts +131 -0
- package/packages/core/src/public/hooks/useClaude.ts +234 -0
- package/packages/core/src/public/hooks/useCodeMarkers.ts +101 -0
- package/packages/core/src/public/hooks/useColorScheme.ts +42 -0
- package/packages/core/src/public/hooks/useCommandHistory.ts +99 -0
- package/packages/core/src/public/hooks/useComplexity.ts +80 -0
- package/packages/core/src/public/hooks/useDeadCode.ts +99 -0
- package/packages/core/src/public/hooks/useDependencies.ts +82 -0
- package/packages/core/src/public/hooks/useDiffs.ts +143 -0
- package/packages/core/src/public/hooks/useFileBrowser.ts +73 -0
- package/packages/core/src/public/hooks/useFocusPanel.ts +137 -0
- package/packages/core/src/public/hooks/useGitStatus.ts +233 -0
- package/packages/core/src/public/hooks/useHealthScore.ts +71 -0
- package/packages/core/src/public/hooks/useHotspots.ts +123 -0
- package/packages/core/src/public/hooks/useLayoutPersistence.ts +141 -0
- package/packages/core/src/public/hooks/useMarkdownParser.ts +36 -0
- package/packages/core/src/public/hooks/useMarkerActions.ts +234 -0
- package/packages/core/src/public/hooks/useMessageQueue.ts +380 -0
- package/packages/core/src/public/hooks/useMessageStream.ts +131 -0
- package/packages/core/src/public/hooks/usePersona.ts +112 -0
- package/packages/core/src/public/hooks/usePlanModeExit.ts +105 -0
- package/packages/core/src/public/hooks/useResponsiveLayout.ts +173 -0
- package/packages/core/src/public/hooks/useSprint.ts +157 -0
- package/packages/core/src/public/hooks/useStatsStrip.ts +204 -0
- package/packages/core/src/public/hooks/useStory.ts +135 -0
- package/packages/core/src/public/hooks/useSubagentHelper.ts +64 -0
- package/packages/core/src/public/hooks/useSyntaxHighlighter.ts +52 -0
- package/packages/core/src/public/hooks/useTabCompletion.ts +124 -0
- package/packages/core/src/public/hooks/useTandemObservations.ts +165 -0
- package/packages/core/src/public/hooks/useTeamMembers.ts +273 -0
- package/packages/core/src/public/hooks/useTodos.ts +93 -0
- package/packages/core/src/public/hooks/useUserAvatar.ts +54 -0
- package/packages/core/src/public/images/cyclist-dark.png +0 -0
- package/packages/core/src/public/images/cyclist-light.png +0 -0
- package/packages/core/src/public/index.html +14 -0
- package/packages/core/src/public/index.tsx +10 -0
- package/packages/core/src/public/lib/utils.ts +6 -0
- package/packages/core/src/public/styles/dockview-theme.css +376 -0
- package/packages/core/src/public/styles/tailwind.css +4454 -0
- package/packages/core/src/public/types/message.ts +51 -0
- package/packages/core/src/public/utils/avatar-service.ts +73 -0
- package/packages/core/src/public/utils/color-presets.ts +940 -0
- package/packages/core/src/public/utils/font-presets.ts +362 -0
- package/packages/core/src/public/utils/formatDuration.ts +14 -0
- package/packages/core/src/public/utils/markdown.ts +249 -0
- package/packages/core/src/public/utils/messageFilters.ts +128 -0
- package/packages/core/src/public/utils/slash-commands.ts +341 -0
- package/packages/core/src/public/utils/subagent-display.ts +146 -0
- package/packages/core/src/public/utils/syntax.ts +219 -0
- package/packages/core/src/public/utils/toolIntentSummarizer.ts +199 -0
- package/packages/core/src/public/utils/toolStackGrouper.ts +106 -0
- package/packages/core/src/public/utils/toolTypeColors.ts +45 -0
- package/pennyfarthing-dist/pf/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/__pycache__/context.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/focus.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/__pycache__/split.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bc/cli.py +0 -1
- package/pennyfarthing-dist/pf/bc/focus.py +0 -1
- package/pennyfarthing-dist/pf/bikerack/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/base_panel.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/git_panel.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/launcher.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/portrait_resolver.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/sprint_panel.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_data.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_screen.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/__pycache__/tui.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bikerack/base_panel.py +0 -1
- package/pennyfarthing-dist/pf/bikerack/events.py +1 -7
- package/pennyfarthing-dist/pf/bikerack/git_panel.py +273 -10
- package/pennyfarthing-dist/pf/bikerack/portrait_resolver.py +21 -0
- package/pennyfarthing-dist/pf/bikerack/sprint_panel.py +58 -1
- package/pennyfarthing-dist/pf/bikerack/tui.py +5 -20
- package/pennyfarthing-dist/pf/bmad/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bmad/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bmad/__pycache__/parser.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/bmad/parser.py +15 -9
- package/pennyfarthing-dist/pf/codemarkers/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/codemarkers/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/codemarkers/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/config.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/output.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/pr_config.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/__pycache__/themes.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/common/pr_config.py +27 -2
- package/pennyfarthing-dist/pf/complexity/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/complexity/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/complexity/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/consultation/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/consultation/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/deadcode/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/dependencies/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/dependencies/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/dependencies/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/epic/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/epic/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/git_group/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/git_group/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/complete_phase.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/marker.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/phase_check.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/handoff/__pycache__/resolve_gate.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/__main__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/formatters.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/healthscore/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/bell_mode.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/context_breaker.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/context_warning.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/cyclist_pretooluse.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/pre_edit_check.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/reflector_check.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/schema_validation.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/session_start.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/session_stop.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/sprint_yaml_validation.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hooks/__pycache__/statusline.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/hotspots/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/claim.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/client.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/create.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/epic.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/operations.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/story.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/jira/__pycache__/sync.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/launch/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/launch/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/loader.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/models.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/persona.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/session.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/tiers.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/prime/__pycache__/workflow.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/session/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/session/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/__pycache__/settings.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/settings/settings.py +44 -8
- package/pennyfarthing-dist/pf/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/archive.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/loader.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/status.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/validator.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/work.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/sprint/story_finish.py +14 -2
- package/pennyfarthing-dist/pf/sprint/validator.py +7 -7
- package/pennyfarthing-dist/pf/tests/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
- package/pennyfarthing-dist/pf/tests/test_sprint_validator.py +44 -0
- package/pennyfarthing-dist/pf/theme/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/theme/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/validate/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/validate/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/__init__.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/cli.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/helpers.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/scale.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/pf/workflow/__pycache__/state.cpython-314.pyc +0 -0
- package/pennyfarthing-dist/scripts/lib/find-root.sh +1 -1
- package/pennyfarthing-dist/scripts/misc/migrate_bmad_workflow.py +0 -1
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +13 -13
- package/pennyfarthing-dist/scripts/workflow/check.py +4 -6
- package/pennyfarthing-dist/scripts/workflow/complete-step.py +2 -2
- package/pennyfarthing-dist/skills/skill-registry.yaml +19 -0
- package/pennyfarthing-dist/workflows/tdd-tandem.yaml +15 -2
- package/packages/core/dist/workflow/__test_context_watch__/.session/.tandem-turn-counter +0 -1
- package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-session.md +0 -3
- package/packages/core/dist/workflow/__test_context_watch__/.session/95-6-tandem-architect.md +0 -6
- package/packages/core/dist/workflow/__test_file_watch__/.session/95-4-tandem-architect.md +0 -6
- package/packages/core/dist/workflow/__test_file_watch__/workdir/trigger.ts +0 -1
- package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-architect.md +0 -6
- package/packages/core/dist/workflow/__test_tool_watch__/.session/95-5-tandem-toolcalls.jsonl +0 -1
- package/pennyfarthing-dist/pf/bikerack/changed_panel.py +0 -201
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* QuickActions Component
|
|
3
|
+
*
|
|
4
|
+
* Renders action buttons based on CYCLIST markers detected in assistant messages.
|
|
5
|
+
* Supports handoff buttons, yes/no questions, choices, and continue.
|
|
6
|
+
*
|
|
7
|
+
* For open questions with suggestions (<!-- CYCLIST:QUESTION:open:suggested text -->),
|
|
8
|
+
* pre-fills the editor via 'cyclist:suggest-prompt' event instead of showing buttons.
|
|
9
|
+
*
|
|
10
|
+
* Story: MSSCI-12787 - Implement CYCLIST Marker Parsing and Action Buttons
|
|
11
|
+
* Story: MSSCI-12771 - Accessibility Compliance (ARIA labels)
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import React, { useState, useEffect, useCallback } from 'react';
|
|
15
|
+
import { Button } from '@/components/ui/button';
|
|
16
|
+
import { useMarkerActions } from '../hooks/useMarkerActions';
|
|
17
|
+
import { useClaudeContext } from '../contexts/ClaudeContext';
|
|
18
|
+
import type { MessageData } from '../types/message';
|
|
19
|
+
|
|
20
|
+
interface ActionItem {
|
|
21
|
+
label: string;
|
|
22
|
+
command: string;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
interface QuickActionsPropsWithMessage {
|
|
26
|
+
message: MessageData;
|
|
27
|
+
actions?: never;
|
|
28
|
+
onAction?: (response: string) => void;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
interface QuickActionsPropsWithActions {
|
|
32
|
+
message?: never;
|
|
33
|
+
actions: ActionItem[];
|
|
34
|
+
onAction?: (response: string) => void;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
type QuickActionsProps = QuickActionsPropsWithMessage | QuickActionsPropsWithActions;
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Get relay mode setting from REST API
|
|
41
|
+
*/
|
|
42
|
+
async function getRelayMode(): Promise<boolean> {
|
|
43
|
+
try {
|
|
44
|
+
const response = await fetch('/api/settings');
|
|
45
|
+
if (response.ok) {
|
|
46
|
+
const settings = await response.json();
|
|
47
|
+
return settings?.workflow?.relay_mode ?? false;
|
|
48
|
+
}
|
|
49
|
+
} catch {
|
|
50
|
+
// Ignore errors
|
|
51
|
+
}
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export default function QuickActions(props: QuickActionsProps): React.ReactElement | null {
|
|
56
|
+
// Get send function from Claude context
|
|
57
|
+
const { send: claudeSend } = useClaudeContext();
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Send a message to Claude via WebSocket context
|
|
61
|
+
*/
|
|
62
|
+
const sendMessage = useCallback((text: string): void => {
|
|
63
|
+
console.log('[QuickActions] sendMessage called:', text);
|
|
64
|
+
claudeSend(text, []);
|
|
65
|
+
}, [claudeSend]);
|
|
66
|
+
const { onAction } = props;
|
|
67
|
+
const [isDisabled, setIsDisabled] = useState(false);
|
|
68
|
+
const [relayMode, setRelayMode] = useState(false);
|
|
69
|
+
|
|
70
|
+
// Support both message-based and actions-based props
|
|
71
|
+
const message = 'message' in props ? props.message : undefined;
|
|
72
|
+
const directActions = 'actions' in props ? props.actions : undefined;
|
|
73
|
+
|
|
74
|
+
const markerActions = useMarkerActions(message?.content);
|
|
75
|
+
|
|
76
|
+
// Reset disabled state when message changes (new assistant response)
|
|
77
|
+
useEffect(() => {
|
|
78
|
+
setIsDisabled(false);
|
|
79
|
+
}, [message?.timestamp]);
|
|
80
|
+
|
|
81
|
+
// Check relay mode on mount
|
|
82
|
+
useEffect(() => {
|
|
83
|
+
getRelayMode().then(setRelayMode);
|
|
84
|
+
}, []);
|
|
85
|
+
|
|
86
|
+
// Clear QuickActions when user submits any message
|
|
87
|
+
useEffect(() => {
|
|
88
|
+
const handleUserSubmit = () => {
|
|
89
|
+
setIsDisabled(true);
|
|
90
|
+
};
|
|
91
|
+
window.addEventListener('cyclist:user-submit', handleUserSubmit);
|
|
92
|
+
return () => {
|
|
93
|
+
window.removeEventListener('cyclist:user-submit', handleUserSubmit);
|
|
94
|
+
};
|
|
95
|
+
}, []);
|
|
96
|
+
|
|
97
|
+
// Auto-execute handoff when relay mode is ON
|
|
98
|
+
useEffect(() => {
|
|
99
|
+
if (markerActions?.type === 'handoff' && relayMode && markerActions.value) {
|
|
100
|
+
// Auto-execute after short delay
|
|
101
|
+
const timer = setTimeout(() => {
|
|
102
|
+
sendMessage(markerActions.value!);
|
|
103
|
+
setIsDisabled(true);
|
|
104
|
+
onAction?.(markerActions.value!);
|
|
105
|
+
}, 100);
|
|
106
|
+
return () => clearTimeout(timer);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
if (markerActions?.type === 'invoke' && markerActions.value) {
|
|
110
|
+
// INVOKE always auto-executes
|
|
111
|
+
const timer = setTimeout(() => {
|
|
112
|
+
sendMessage(markerActions.value!);
|
|
113
|
+
setIsDisabled(true);
|
|
114
|
+
onAction?.(markerActions.value!);
|
|
115
|
+
}, 100);
|
|
116
|
+
return () => clearTimeout(timer);
|
|
117
|
+
}
|
|
118
|
+
}, [markerActions, relayMode, onAction]);
|
|
119
|
+
|
|
120
|
+
// Pre-fill editor with suggested prompt for open questions
|
|
121
|
+
useEffect(() => {
|
|
122
|
+
if (markerActions?.type === 'open' && markerActions.responses && markerActions.responses.length > 0) {
|
|
123
|
+
const suggestion = markerActions.responses[0];
|
|
124
|
+
window.dispatchEvent(new CustomEvent('cyclist:suggest-prompt', {
|
|
125
|
+
detail: { prompt: suggestion }
|
|
126
|
+
}));
|
|
127
|
+
}
|
|
128
|
+
}, [markerActions]);
|
|
129
|
+
|
|
130
|
+
const handleButtonClick = useCallback((response: string) => {
|
|
131
|
+
console.log('[QuickActions] Button clicked:', response);
|
|
132
|
+
setIsDisabled(true);
|
|
133
|
+
sendMessage(response);
|
|
134
|
+
onAction?.(response);
|
|
135
|
+
}, [onAction]);
|
|
136
|
+
|
|
137
|
+
// If using direct actions prop (for accessibility testing), render those
|
|
138
|
+
if (directActions && directActions.length > 0) {
|
|
139
|
+
return (
|
|
140
|
+
<div className="quick-actions">
|
|
141
|
+
<div className="quick-actions-buttons">
|
|
142
|
+
{directActions.map((action) => (
|
|
143
|
+
<Button
|
|
144
|
+
variant="secondary"
|
|
145
|
+
size="sm"
|
|
146
|
+
type="button"
|
|
147
|
+
key={action.command}
|
|
148
|
+
className="quick-action-btn"
|
|
149
|
+
onClick={() => handleButtonClick(action.command)}
|
|
150
|
+
disabled={isDisabled}
|
|
151
|
+
aria-label={`${action.label}: ${action.command}`}
|
|
152
|
+
>
|
|
153
|
+
{action.label}
|
|
154
|
+
</Button>
|
|
155
|
+
))}
|
|
156
|
+
</div>
|
|
157
|
+
</div>
|
|
158
|
+
);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// No marker actions to render
|
|
162
|
+
if (!markerActions) {
|
|
163
|
+
return null;
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
// Open questions - no buttons needed, editor is pre-filled via useEffect
|
|
167
|
+
if (markerActions.type === 'open') {
|
|
168
|
+
return null;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// Continue marker - suppress the generic Continue button entirely
|
|
172
|
+
// Users can always type to continue; showing a disabled button is confusing
|
|
173
|
+
if (markerActions.type === 'continue') {
|
|
174
|
+
return null;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// Auto-execute types don't render buttons
|
|
178
|
+
if (markerActions.type === 'invoke') {
|
|
179
|
+
return (
|
|
180
|
+
<div className="quick-actions">
|
|
181
|
+
<span className="auto-invoke-status">Invoking {markerActions.value}...</span>
|
|
182
|
+
</div>
|
|
183
|
+
);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
// Auto-executing handoff in relay mode
|
|
187
|
+
if (markerActions.type === 'handoff' && relayMode) {
|
|
188
|
+
return (
|
|
189
|
+
<div className="quick-actions">
|
|
190
|
+
<span className="auto-invoke-status">Handing off to {markerActions.value}...</span>
|
|
191
|
+
</div>
|
|
192
|
+
);
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
return (
|
|
196
|
+
<div className="quick-actions">
|
|
197
|
+
{/* Handoff buttons */}
|
|
198
|
+
{markerActions.type === 'handoff' && markerActions.responses && (
|
|
199
|
+
<div className="quick-actions-buttons">
|
|
200
|
+
{markerActions.responses.map((response) => (
|
|
201
|
+
<Button
|
|
202
|
+
variant="secondary"
|
|
203
|
+
size="sm"
|
|
204
|
+
type="button"
|
|
205
|
+
key={response}
|
|
206
|
+
className="quick-action-btn"
|
|
207
|
+
onClick={() => handleButtonClick(response)}
|
|
208
|
+
disabled={isDisabled}
|
|
209
|
+
aria-label={`Continue with ${response}`}
|
|
210
|
+
>
|
|
211
|
+
{response}
|
|
212
|
+
</Button>
|
|
213
|
+
))}
|
|
214
|
+
</div>
|
|
215
|
+
)}
|
|
216
|
+
|
|
217
|
+
{/* Yes/No buttons */}
|
|
218
|
+
{markerActions.type === 'yesno' && (
|
|
219
|
+
<div className="quick-actions-buttons">
|
|
220
|
+
<Button
|
|
221
|
+
variant="secondary"
|
|
222
|
+
size="sm"
|
|
223
|
+
type="button"
|
|
224
|
+
className="quick-action-btn"
|
|
225
|
+
onClick={() => handleButtonClick('Yes')}
|
|
226
|
+
disabled={isDisabled}
|
|
227
|
+
aria-label="Answer Yes"
|
|
228
|
+
>
|
|
229
|
+
Yes
|
|
230
|
+
</Button>
|
|
231
|
+
<Button
|
|
232
|
+
variant="secondary"
|
|
233
|
+
size="sm"
|
|
234
|
+
type="button"
|
|
235
|
+
className="quick-action-btn"
|
|
236
|
+
onClick={() => handleButtonClick('No')}
|
|
237
|
+
disabled={isDisabled}
|
|
238
|
+
aria-label="Answer No"
|
|
239
|
+
>
|
|
240
|
+
No
|
|
241
|
+
</Button>
|
|
242
|
+
</div>
|
|
243
|
+
)}
|
|
244
|
+
|
|
245
|
+
{/* Choice buttons */}
|
|
246
|
+
{markerActions.type === 'choices' && markerActions.choices && (
|
|
247
|
+
<div className="quick-actions-buttons">
|
|
248
|
+
{markerActions.choices.map((choice) => (
|
|
249
|
+
<Button
|
|
250
|
+
variant="secondary"
|
|
251
|
+
size="sm"
|
|
252
|
+
type="button"
|
|
253
|
+
key={choice.number}
|
|
254
|
+
className="quick-action-btn"
|
|
255
|
+
onClick={() => handleButtonClick(choice.text)}
|
|
256
|
+
disabled={isDisabled}
|
|
257
|
+
aria-label={`Choose option ${choice.number}: ${choice.text}`}
|
|
258
|
+
>
|
|
259
|
+
{choice.text}
|
|
260
|
+
</Button>
|
|
261
|
+
))}
|
|
262
|
+
</div>
|
|
263
|
+
)}
|
|
264
|
+
</div>
|
|
265
|
+
);
|
|
266
|
+
}
|
|
267
|
+
|
|
@@ -0,0 +1,352 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SpanTimeline Component
|
|
3
|
+
*
|
|
4
|
+
* Visual timeline of OTEL spans showing tool execution over time.
|
|
5
|
+
*
|
|
6
|
+
* Features:
|
|
7
|
+
* - Horizontal timeline with spans as bars
|
|
8
|
+
* - Color-coded by tool type
|
|
9
|
+
* - Click to expand span details
|
|
10
|
+
* - Real-time updates via WebSocket
|
|
11
|
+
* - Zoom and scroll controls
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import React, { useState, useEffect, useCallback, useRef, useMemo } from 'react';
|
|
15
|
+
import { Button } from '@/components/ui/button';
|
|
16
|
+
import { Skeleton } from '@/components/ui/skeleton';
|
|
17
|
+
import { Badge } from '@/components/ui/badge';
|
|
18
|
+
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
|
19
|
+
|
|
20
|
+
// =============================================================================
|
|
21
|
+
// Types
|
|
22
|
+
// =============================================================================
|
|
23
|
+
|
|
24
|
+
interface EnrichedSpan {
|
|
25
|
+
spanId: string;
|
|
26
|
+
traceId: string;
|
|
27
|
+
toolName: string;
|
|
28
|
+
startTime: number;
|
|
29
|
+
endTime?: number;
|
|
30
|
+
durationMs: number;
|
|
31
|
+
status: 'running' | 'completed' | 'error';
|
|
32
|
+
success: boolean;
|
|
33
|
+
error?: string;
|
|
34
|
+
enrichment: {
|
|
35
|
+
filePath?: string;
|
|
36
|
+
command?: string;
|
|
37
|
+
pattern?: string;
|
|
38
|
+
summary?: string;
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
interface SpanTimelineProps {
|
|
43
|
+
/** Height of the timeline in pixels */
|
|
44
|
+
height?: number;
|
|
45
|
+
/** Whether to auto-scroll to newest spans */
|
|
46
|
+
autoScroll?: boolean;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
// Tool colors for visual distinction
|
|
50
|
+
const TOOL_COLORS: Record<string, string> = {
|
|
51
|
+
Read: '#4ade80', // Green
|
|
52
|
+
Write: '#f87171', // Red
|
|
53
|
+
Edit: '#fb923c', // Orange
|
|
54
|
+
Bash: '#60a5fa', // Blue
|
|
55
|
+
Grep: '#a78bfa', // Purple
|
|
56
|
+
Glob: '#2dd4bf', // Teal
|
|
57
|
+
Task: '#e879f9', // Magenta
|
|
58
|
+
WebFetch: '#22d3ee', // Cyan
|
|
59
|
+
WebSearch: '#fbbf24', // Yellow
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
// =============================================================================
|
|
63
|
+
// Helper Functions
|
|
64
|
+
// =============================================================================
|
|
65
|
+
|
|
66
|
+
function formatTime(timestamp: number): string {
|
|
67
|
+
return new Date(timestamp).toLocaleTimeString();
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
function formatDuration(ms: number): string {
|
|
71
|
+
if (ms < 1000) return `${ms}ms`;
|
|
72
|
+
if (ms < 60000) return `${(ms / 1000).toFixed(1)}s`;
|
|
73
|
+
return `${(ms / 60000).toFixed(1)}m`;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function getToolColor(toolName: string): string {
|
|
77
|
+
return TOOL_COLORS[toolName] || '#888888';
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// =============================================================================
|
|
81
|
+
// Component
|
|
82
|
+
// =============================================================================
|
|
83
|
+
|
|
84
|
+
export function SpanTimeline({ height = 200, autoScroll = true }: SpanTimelineProps): React.ReactElement {
|
|
85
|
+
const [spans, setSpans] = useState<EnrichedSpan[]>([]);
|
|
86
|
+
const [selectedSpan, setSelectedSpan] = useState<EnrichedSpan | null>(null);
|
|
87
|
+
const [loading, setLoading] = useState(true);
|
|
88
|
+
const [zoomLevel, setZoomLevel] = useState(1); // 1 = 1px per 100ms
|
|
89
|
+
const timelineRef = useRef<HTMLDivElement>(null);
|
|
90
|
+
const containerRef = useRef<HTMLDivElement>(null);
|
|
91
|
+
|
|
92
|
+
// Fetch initial spans
|
|
93
|
+
useEffect(() => {
|
|
94
|
+
fetch('/api/spans?limit=100')
|
|
95
|
+
.then(res => res.ok ? res.json() : { spans: [] })
|
|
96
|
+
.then(data => {
|
|
97
|
+
setSpans(data.spans || []);
|
|
98
|
+
setLoading(false);
|
|
99
|
+
})
|
|
100
|
+
.catch(() => setLoading(false));
|
|
101
|
+
}, []);
|
|
102
|
+
|
|
103
|
+
// Subscribe to real-time updates
|
|
104
|
+
useEffect(() => {
|
|
105
|
+
const ws = new WebSocket(`ws://${window.location.host}/ws/spans`);
|
|
106
|
+
|
|
107
|
+
ws.onmessage = (event) => {
|
|
108
|
+
try {
|
|
109
|
+
const data = JSON.parse(event.data);
|
|
110
|
+
if (data.type === 'span' && data.span) {
|
|
111
|
+
setSpans(prev => {
|
|
112
|
+
// Update existing span or add new one
|
|
113
|
+
const existing = prev.findIndex(s => s.spanId === data.span.spanId);
|
|
114
|
+
if (existing >= 0) {
|
|
115
|
+
const updated = [...prev];
|
|
116
|
+
updated[existing] = data.span;
|
|
117
|
+
return updated;
|
|
118
|
+
}
|
|
119
|
+
return [...prev, data.span].slice(-100); // Keep last 100
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
} catch (err) {
|
|
123
|
+
console.error('[SpanTimeline] WebSocket parse error:', err);
|
|
124
|
+
}
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
return () => ws.close();
|
|
128
|
+
}, []);
|
|
129
|
+
|
|
130
|
+
// Auto-scroll to right (newest spans)
|
|
131
|
+
useEffect(() => {
|
|
132
|
+
if (autoScroll && containerRef.current) {
|
|
133
|
+
containerRef.current.scrollLeft = containerRef.current.scrollWidth;
|
|
134
|
+
}
|
|
135
|
+
}, [spans, autoScroll]);
|
|
136
|
+
|
|
137
|
+
// Calculate timeline bounds
|
|
138
|
+
const { minTime, maxTime, timeRange } = useMemo(() => {
|
|
139
|
+
if (spans.length === 0) {
|
|
140
|
+
const now = Date.now();
|
|
141
|
+
return { minTime: now - 60000, maxTime: now, timeRange: 60000 };
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
const times = spans.flatMap(s => [s.startTime, s.endTime || s.startTime + s.durationMs]);
|
|
145
|
+
const min = Math.min(...times);
|
|
146
|
+
const max = Math.max(...times);
|
|
147
|
+
const range = Math.max(max - min, 10000); // At least 10 seconds
|
|
148
|
+
|
|
149
|
+
return { minTime: min, maxTime: max, timeRange: range };
|
|
150
|
+
}, [spans]);
|
|
151
|
+
|
|
152
|
+
// Calculate timeline width based on zoom
|
|
153
|
+
const timelineWidth = useMemo(() => {
|
|
154
|
+
return Math.max(800, (timeRange / 100) * zoomLevel);
|
|
155
|
+
}, [timeRange, zoomLevel]);
|
|
156
|
+
|
|
157
|
+
// Handle span click
|
|
158
|
+
const handleSpanClick = useCallback((span: EnrichedSpan) => {
|
|
159
|
+
setSelectedSpan(selectedSpan?.spanId === span.spanId ? null : span);
|
|
160
|
+
}, [selectedSpan]);
|
|
161
|
+
|
|
162
|
+
// Zoom controls
|
|
163
|
+
const handleZoomIn = useCallback(() => {
|
|
164
|
+
setZoomLevel(prev => Math.min(prev * 1.5, 10));
|
|
165
|
+
}, []);
|
|
166
|
+
|
|
167
|
+
const handleZoomOut = useCallback(() => {
|
|
168
|
+
setZoomLevel(prev => Math.max(prev / 1.5, 0.1));
|
|
169
|
+
}, []);
|
|
170
|
+
|
|
171
|
+
const handleZoomReset = useCallback(() => {
|
|
172
|
+
setZoomLevel(1);
|
|
173
|
+
}, []);
|
|
174
|
+
|
|
175
|
+
if (loading) {
|
|
176
|
+
return (
|
|
177
|
+
<div className="span-timeline-loading p-2 space-y-2">
|
|
178
|
+
<div className="flex gap-2 items-center">
|
|
179
|
+
<Skeleton className="h-4 w-20" />
|
|
180
|
+
<div className="flex-1" />
|
|
181
|
+
<Skeleton className="h-6 w-6" />
|
|
182
|
+
<Skeleton className="h-6 w-6" />
|
|
183
|
+
<Skeleton className="h-6 w-6" />
|
|
184
|
+
</div>
|
|
185
|
+
<Skeleton className="h-[120px] w-full" />
|
|
186
|
+
</div>
|
|
187
|
+
);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
return (
|
|
191
|
+
<TooltipProvider delayDuration={300}>
|
|
192
|
+
<div className="span-timeline" style={{ height }}>
|
|
193
|
+
{/* Toolbar */}
|
|
194
|
+
<div className="span-timeline-toolbar">
|
|
195
|
+
<span className="span-count">{spans.length} spans</span>
|
|
196
|
+
<div className="zoom-controls">
|
|
197
|
+
<Tooltip>
|
|
198
|
+
<TooltipTrigger asChild>
|
|
199
|
+
<Button variant="ghost" size="icon" onClick={handleZoomOut}>−</Button>
|
|
200
|
+
</TooltipTrigger>
|
|
201
|
+
<TooltipContent>Zoom out</TooltipContent>
|
|
202
|
+
</Tooltip>
|
|
203
|
+
<Tooltip>
|
|
204
|
+
<TooltipTrigger asChild>
|
|
205
|
+
<Button variant="ghost" size="icon" onClick={handleZoomReset}>⟳</Button>
|
|
206
|
+
</TooltipTrigger>
|
|
207
|
+
<TooltipContent>Reset zoom</TooltipContent>
|
|
208
|
+
</Tooltip>
|
|
209
|
+
<Tooltip>
|
|
210
|
+
<TooltipTrigger asChild>
|
|
211
|
+
<Button variant="ghost" size="icon" onClick={handleZoomIn}>+</Button>
|
|
212
|
+
</TooltipTrigger>
|
|
213
|
+
<TooltipContent>Zoom in</TooltipContent>
|
|
214
|
+
</Tooltip>
|
|
215
|
+
</div>
|
|
216
|
+
</div>
|
|
217
|
+
|
|
218
|
+
{/* Timeline container */}
|
|
219
|
+
<div className="span-timeline-container" ref={containerRef}>
|
|
220
|
+
<div
|
|
221
|
+
className="span-timeline-track"
|
|
222
|
+
ref={timelineRef}
|
|
223
|
+
style={{ width: timelineWidth }}
|
|
224
|
+
>
|
|
225
|
+
{/* Time axis */}
|
|
226
|
+
<div className="span-timeline-axis">
|
|
227
|
+
{Array.from({ length: Math.ceil(timeRange / 10000) + 1 }, (_, i) => {
|
|
228
|
+
const time = minTime + i * 10000;
|
|
229
|
+
const left = ((time - minTime) / timeRange) * timelineWidth;
|
|
230
|
+
return (
|
|
231
|
+
<div
|
|
232
|
+
key={i}
|
|
233
|
+
className="axis-tick"
|
|
234
|
+
style={{ left }}
|
|
235
|
+
>
|
|
236
|
+
<span className="tick-label">{formatTime(time)}</span>
|
|
237
|
+
</div>
|
|
238
|
+
);
|
|
239
|
+
})}
|
|
240
|
+
</div>
|
|
241
|
+
|
|
242
|
+
{/* Span bars */}
|
|
243
|
+
<div className="span-timeline-bars">
|
|
244
|
+
{spans.map((span, index) => {
|
|
245
|
+
const left = ((span.startTime - minTime) / timeRange) * timelineWidth;
|
|
246
|
+
const width = Math.max(4, (span.durationMs / timeRange) * timelineWidth);
|
|
247
|
+
const top = (index % 5) * 28 + 4; // Stack in 5 rows
|
|
248
|
+
|
|
249
|
+
return (
|
|
250
|
+
<Tooltip key={span.spanId}>
|
|
251
|
+
<TooltipTrigger asChild>
|
|
252
|
+
<div
|
|
253
|
+
className={`span-bar ${span.success ? '' : 'error'} ${
|
|
254
|
+
selectedSpan?.spanId === span.spanId ? 'selected' : ''
|
|
255
|
+
}`}
|
|
256
|
+
style={{
|
|
257
|
+
left,
|
|
258
|
+
width,
|
|
259
|
+
top,
|
|
260
|
+
backgroundColor: getToolColor(span.toolName),
|
|
261
|
+
}}
|
|
262
|
+
onClick={() => handleSpanClick(span)}
|
|
263
|
+
>
|
|
264
|
+
<span className="span-bar-label">{span.toolName}</span>
|
|
265
|
+
</div>
|
|
266
|
+
</TooltipTrigger>
|
|
267
|
+
<TooltipContent>{`${span.toolName}: ${formatDuration(span.durationMs)}`}</TooltipContent>
|
|
268
|
+
</Tooltip>
|
|
269
|
+
);
|
|
270
|
+
})}
|
|
271
|
+
</div>
|
|
272
|
+
</div>
|
|
273
|
+
</div>
|
|
274
|
+
|
|
275
|
+
{/* Selected span details */}
|
|
276
|
+
{selectedSpan && (
|
|
277
|
+
<div className="span-timeline-details">
|
|
278
|
+
<div className="detail-header">
|
|
279
|
+
<Badge
|
|
280
|
+
variant="default"
|
|
281
|
+
className="tool-badge"
|
|
282
|
+
style={{ backgroundColor: getToolColor(selectedSpan.toolName) }}
|
|
283
|
+
>
|
|
284
|
+
{selectedSpan.toolName}
|
|
285
|
+
</Badge>
|
|
286
|
+
<span className="duration">{formatDuration(selectedSpan.durationMs)}</span>
|
|
287
|
+
<span className={`status ${selectedSpan.success ? 'success' : 'error'}`}>
|
|
288
|
+
{selectedSpan.success ? '✓' : '✗'}
|
|
289
|
+
</span>
|
|
290
|
+
<Button
|
|
291
|
+
variant="ghost"
|
|
292
|
+
size="icon"
|
|
293
|
+
className="close-details"
|
|
294
|
+
onClick={() => setSelectedSpan(null)}
|
|
295
|
+
>
|
|
296
|
+
×
|
|
297
|
+
</Button>
|
|
298
|
+
</div>
|
|
299
|
+
<div className="detail-body">
|
|
300
|
+
<div className="detail-row">
|
|
301
|
+
<label>Time:</label>
|
|
302
|
+
<span>{formatTime(selectedSpan.startTime)}</span>
|
|
303
|
+
</div>
|
|
304
|
+
{selectedSpan.enrichment.filePath && (
|
|
305
|
+
<div className="detail-row">
|
|
306
|
+
<label>File:</label>
|
|
307
|
+
<span className="mono">{selectedSpan.enrichment.filePath}</span>
|
|
308
|
+
</div>
|
|
309
|
+
)}
|
|
310
|
+
{selectedSpan.enrichment.command && (
|
|
311
|
+
<div className="detail-row">
|
|
312
|
+
<label>Command:</label>
|
|
313
|
+
<span className="mono">{selectedSpan.enrichment.command}</span>
|
|
314
|
+
</div>
|
|
315
|
+
)}
|
|
316
|
+
{selectedSpan.enrichment.pattern && (
|
|
317
|
+
<div className="detail-row">
|
|
318
|
+
<label>Pattern:</label>
|
|
319
|
+
<span className="mono">{selectedSpan.enrichment.pattern}</span>
|
|
320
|
+
</div>
|
|
321
|
+
)}
|
|
322
|
+
{selectedSpan.enrichment.summary && (
|
|
323
|
+
<div className="detail-row">
|
|
324
|
+
<label>Summary:</label>
|
|
325
|
+
<span>{selectedSpan.enrichment.summary}</span>
|
|
326
|
+
</div>
|
|
327
|
+
)}
|
|
328
|
+
{selectedSpan.error && (
|
|
329
|
+
<div className="detail-row error">
|
|
330
|
+
<label>Error:</label>
|
|
331
|
+
<span>{selectedSpan.error}</span>
|
|
332
|
+
</div>
|
|
333
|
+
)}
|
|
334
|
+
</div>
|
|
335
|
+
</div>
|
|
336
|
+
)}
|
|
337
|
+
|
|
338
|
+
{/* Legend */}
|
|
339
|
+
<div className="span-timeline-legend">
|
|
340
|
+
{Object.entries(TOOL_COLORS).slice(0, 6).map(([tool, color]) => (
|
|
341
|
+
<div key={tool} className="legend-item">
|
|
342
|
+
<span className="legend-color" style={{ backgroundColor: color }} />
|
|
343
|
+
<span className="legend-label">{tool}</span>
|
|
344
|
+
</div>
|
|
345
|
+
))}
|
|
346
|
+
</div>
|
|
347
|
+
</div>
|
|
348
|
+
</TooltipProvider>
|
|
349
|
+
);
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
export default SpanTimeline;
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* StandalonePanel - Full-screen panel wrapper for BikeRack mode
|
|
3
|
+
*
|
|
4
|
+
* Story MSSCI-14821: StandalonePanel wrapper and ?panel=X client routing
|
|
5
|
+
* Epic: 101 (BikeRack Mode)
|
|
6
|
+
*
|
|
7
|
+
* Renders a single panel full-screen based on ?panel=X URL parameter.
|
|
8
|
+
* PANEL_REGISTRY is the single source of truth for routing (CE-2).
|
|
9
|
+
*
|
|
10
|
+
* Rules:
|
|
11
|
+
* - No dockview-react imports (Rule 7)
|
|
12
|
+
* - No BikeRack-specific props to panels (Rule 2)
|
|
13
|
+
* - URL-based detection only (Rule 10)
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import React from 'react';
|
|
17
|
+
import {
|
|
18
|
+
EnhancedSprintPanel,
|
|
19
|
+
GitPanel,
|
|
20
|
+
DiffsPanel,
|
|
21
|
+
TodoPanel,
|
|
22
|
+
WorkflowPanel,
|
|
23
|
+
BackgroundPanel,
|
|
24
|
+
AuditLogPanel,
|
|
25
|
+
ACPanel,
|
|
26
|
+
DebugPanel,
|
|
27
|
+
BikeLanePanel,
|
|
28
|
+
SettingsPanel,
|
|
29
|
+
ProgressPanel,
|
|
30
|
+
} from './panels';
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Registry mapping panel URL names to their components.
|
|
34
|
+
* Single source of truth for standalone panel routing (CE-2).
|
|
35
|
+
*/
|
|
36
|
+
export const PANEL_REGISTRY: Record<string, React.ComponentType> = {
|
|
37
|
+
sprint: EnhancedSprintPanel,
|
|
38
|
+
git: GitPanel,
|
|
39
|
+
diffs: DiffsPanel,
|
|
40
|
+
todos: TodoPanel,
|
|
41
|
+
workflow: WorkflowPanel,
|
|
42
|
+
background: BackgroundPanel,
|
|
43
|
+
audit: AuditLogPanel,
|
|
44
|
+
ac: ACPanel,
|
|
45
|
+
debug: DebugPanel,
|
|
46
|
+
bikelane: BikeLanePanel,
|
|
47
|
+
settings: SettingsPanel,
|
|
48
|
+
progress: ProgressPanel,
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Detect standalone panel mode from URL parameters (Rule 10).
|
|
53
|
+
*/
|
|
54
|
+
export function getStandalonePanelName(): string | null {
|
|
55
|
+
const params = new URLSearchParams(window.location.search);
|
|
56
|
+
return params.get('panel');
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* StandalonePanel wrapper - renders a single panel full-screen.
|
|
61
|
+
*/
|
|
62
|
+
export function StandalonePanel(): React.ReactElement {
|
|
63
|
+
const panelName = getStandalonePanelName();
|
|
64
|
+
const PanelComponent = panelName ? PANEL_REGISTRY[panelName] : null;
|
|
65
|
+
|
|
66
|
+
if (!PanelComponent) {
|
|
67
|
+
return (
|
|
68
|
+
<div style={{ display: 'flex', flexDirection: 'column', alignItems: 'center', justifyContent: 'center', height: '100vh', width: '100vw', backgroundColor: 'var(--bg-primary, #1a1a2e)', color: 'var(--text-primary, #e4e4e7)' }}>
|
|
69
|
+
<h1>Panel not found</h1>
|
|
70
|
+
<p>
|
|
71
|
+
<a href="/" style={{ color: 'var(--accent, #818cf8)' }}>Back to BikeRack</a>
|
|
72
|
+
</p>
|
|
73
|
+
</div>
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
return (
|
|
78
|
+
<div style={{ height: '100vh', width: '100vw', overflow: 'auto', backgroundColor: 'var(--bg-primary, #1a1a2e)', color: 'var(--text-primary, #e4e4e7)' }}>
|
|
79
|
+
<PanelComponent />
|
|
80
|
+
</div>
|
|
81
|
+
);
|
|
82
|
+
}
|