@pennyfarthing/core 11.3.7 → 11.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/package.json +17 -16
- 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/hooks/cyclist-pretooluse-hook.sh +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/core/agent-session.sh +0 -0
- package/pennyfarthing-dist/scripts/core/check-context.sh +0 -0
- package/pennyfarthing-dist/scripts/core/dialogue-manager.sh +0 -0
- package/pennyfarthing-dist/scripts/core/pf.sh +0 -0
- package/pennyfarthing-dist/scripts/core/phase-check-start.sh +0 -0
- package/pennyfarthing-dist/scripts/core/prime.sh +0 -0
- package/pennyfarthing-dist/scripts/cyclist/is-cyclist.sh +0 -0
- package/pennyfarthing-dist/scripts/git/create-feature-branches.sh +0 -0
- package/pennyfarthing-dist/scripts/git/git-status-all.sh +0 -0
- package/pennyfarthing-dist/scripts/git/install-git-hooks.sh +0 -0
- package/pennyfarthing-dist/scripts/git/release.sh +0 -0
- package/pennyfarthing-dist/scripts/git/worktree-manager.sh +0 -0
- package/pennyfarthing-dist/scripts/health/drift-detection.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/bell-mode-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/context-circuit-breaker.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/context-warning.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/cyclist-pretooluse-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/dispatcher-template.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/otel-auto-config.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/post-merge.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/pre-commit.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/pre-edit-check.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/pre-push.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/question-reflector-check.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/question_reflector_check.py +0 -0
- package/pennyfarthing-dist/scripts/hooks/schema-validation.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/session-start.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/session-stop.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/sprint-yaml-validation.sh +0 -0
- package/pennyfarthing-dist/scripts/hooks/welcome-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/create-jira-epic.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/create-jira-story.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/jira-claim-story.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/jira-reconcile.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/jira-sync-story.sh +0 -0
- package/pennyfarthing-dist/scripts/jira/sync-epic-jira.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/background-tasks.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/checkpoint.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/common.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/env.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/file-lock.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/find-root.sh +1 -1
- package/pennyfarthing-dist/scripts/lib/logging.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/retry.sh +0 -0
- package/pennyfarthing-dist/scripts/lib/run-pf.sh +0 -0
- package/pennyfarthing-dist/scripts/maintenance/migrate-theme-schema.mjs +0 -0
- package/pennyfarthing-dist/scripts/maintenance/sidecar-health.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/add-short-names.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/add_short_names.py +0 -0
- package/pennyfarthing-dist/scripts/misc/backlog.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/check-status.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/find-related-work.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/generate-skill-docs.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/log-skill-usage.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/migrate-bmad-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/migrate_bmad_workflow.py +0 -1
- package/pennyfarthing-dist/scripts/misc/repo-scan.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/repo-utils.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/run-ci.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/run-timestamp.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/session-cleanup.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/skill-usage-report.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/statusline.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/uninstall.sh +0 -0
- package/pennyfarthing-dist/scripts/misc/validate-subagent-frontmatter.sh +0 -0
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.py +13 -13
- package/pennyfarthing-dist/scripts/portraits/generate-portraits.sh +0 -0
- package/pennyfarthing-dist/scripts/portraits/generate-tandem-portraits.sh +0 -0
- package/pennyfarthing-dist/scripts/story/create-story.sh +0 -0
- package/pennyfarthing-dist/scripts/story/size-story.sh +0 -0
- package/pennyfarthing-dist/scripts/story/story-template.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/check.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/dev-story-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/epics-and-stories-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/handoff-phase-update.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/implementation-readiness-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/migrate-bmad-workflow.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/prd-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/project-context-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-character-voice.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-drift-detection.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-post-merge-hook.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-session-checkpoint.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/test-solo-command.sh +0 -0
- package/pennyfarthing-dist/scripts/tests/ux-design-workflow-import.test.sh +0 -0
- package/pennyfarthing-dist/scripts/theme/list-themes.sh +0 -0
- package/pennyfarthing-dist/scripts/validation/validate-agent-schema.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/check.py +4 -6
- package/pennyfarthing-dist/scripts/workflow/check.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/complete-step.py +2 -2
- package/pennyfarthing-dist/scripts/workflow/finish-story.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/fix-session-phase.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/get-workflow-type.py +0 -0
- package/pennyfarthing-dist/scripts/workflow/get-workflow-type.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/list-workflows.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/phase-owner.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/resume-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/show-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/start-workflow.sh +0 -0
- package/pennyfarthing-dist/scripts/workflow/workflow-status.sh +0 -0
- package/pennyfarthing-dist/skills/pf-story/scripts/create-story.sh +0 -0
- package/pennyfarthing-dist/skills/pf-story/scripts/size-story.sh +0 -0
- package/pennyfarthing-dist/skills/pf-story/scripts/story-template.sh +0 -0
- package/pennyfarthing-dist/skills/skill-registry.yaml +19 -0
- package/pennyfarthing-dist/workflows/release/steps/step-10-publish.md +41 -9
- 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
- package/scripts/README.md +0 -41
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ConfirmDialog Component
|
|
3
|
+
*
|
|
4
|
+
* Generic confirmation modal for destructive or important actions,
|
|
5
|
+
* built on shadcn AlertDialog (Radix UI primitives).
|
|
6
|
+
*
|
|
7
|
+
* Features:
|
|
8
|
+
* - Title and message customization
|
|
9
|
+
* - Confirm/Cancel button labels
|
|
10
|
+
* - Danger mode styling for destructive actions (uses destructive button variant)
|
|
11
|
+
* - Escape key to cancel (handled by Radix)
|
|
12
|
+
* - Click outside to cancel (handled by Radix)
|
|
13
|
+
* - Focus trap for accessibility (handled by Radix)
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import React, { useRef, useCallback } from 'react';
|
|
17
|
+
import {
|
|
18
|
+
AlertDialog,
|
|
19
|
+
AlertDialogContent,
|
|
20
|
+
AlertDialogHeader,
|
|
21
|
+
AlertDialogTitle,
|
|
22
|
+
AlertDialogDescription,
|
|
23
|
+
AlertDialogFooter,
|
|
24
|
+
AlertDialogAction,
|
|
25
|
+
AlertDialogCancel,
|
|
26
|
+
} from '@/components/ui/alert-dialog';
|
|
27
|
+
import { buttonVariants } from '@/components/ui/button';
|
|
28
|
+
import { cn } from '@/lib/utils';
|
|
29
|
+
|
|
30
|
+
// =============================================================================
|
|
31
|
+
// Types
|
|
32
|
+
// =============================================================================
|
|
33
|
+
|
|
34
|
+
export interface ConfirmDialogProps {
|
|
35
|
+
/** Whether the dialog is open */
|
|
36
|
+
isOpen: boolean;
|
|
37
|
+
/** Dialog title */
|
|
38
|
+
title: string;
|
|
39
|
+
/** Dialog message/description */
|
|
40
|
+
message: string;
|
|
41
|
+
/** Confirm button label (default: "Confirm") */
|
|
42
|
+
confirmLabel?: string;
|
|
43
|
+
/** Cancel button label (default: "Cancel") */
|
|
44
|
+
cancelLabel?: string;
|
|
45
|
+
/** Whether this is a dangerous/destructive action */
|
|
46
|
+
isDanger?: boolean;
|
|
47
|
+
/** Called when user confirms */
|
|
48
|
+
onConfirm: () => void;
|
|
49
|
+
/** Called when user cancels */
|
|
50
|
+
onCancel: () => void;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// =============================================================================
|
|
54
|
+
// Component
|
|
55
|
+
// =============================================================================
|
|
56
|
+
|
|
57
|
+
export function ConfirmDialog({
|
|
58
|
+
isOpen,
|
|
59
|
+
title,
|
|
60
|
+
message,
|
|
61
|
+
confirmLabel = 'Confirm',
|
|
62
|
+
cancelLabel = 'Cancel',
|
|
63
|
+
isDanger = false,
|
|
64
|
+
onConfirm,
|
|
65
|
+
onCancel,
|
|
66
|
+
}: ConfirmDialogProps): React.ReactElement {
|
|
67
|
+
return (
|
|
68
|
+
<AlertDialog open={isOpen} onOpenChange={(open) => { if (!open) onCancel(); }}>
|
|
69
|
+
<AlertDialogContent>
|
|
70
|
+
<AlertDialogHeader>
|
|
71
|
+
<AlertDialogTitle>{title}</AlertDialogTitle>
|
|
72
|
+
<AlertDialogDescription>{message}</AlertDialogDescription>
|
|
73
|
+
</AlertDialogHeader>
|
|
74
|
+
<AlertDialogFooter>
|
|
75
|
+
<AlertDialogCancel onClick={onCancel}>{cancelLabel}</AlertDialogCancel>
|
|
76
|
+
<AlertDialogAction
|
|
77
|
+
className={cn(isDanger && buttonVariants({ variant: 'destructive' }))}
|
|
78
|
+
onClick={onConfirm}
|
|
79
|
+
>
|
|
80
|
+
{confirmLabel}
|
|
81
|
+
</AlertDialogAction>
|
|
82
|
+
</AlertDialogFooter>
|
|
83
|
+
</AlertDialogContent>
|
|
84
|
+
</AlertDialog>
|
|
85
|
+
);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
// =============================================================================
|
|
89
|
+
// Hook for easy usage
|
|
90
|
+
// =============================================================================
|
|
91
|
+
|
|
92
|
+
interface UseConfirmDialogOptions {
|
|
93
|
+
title: string;
|
|
94
|
+
message: string;
|
|
95
|
+
confirmLabel?: string;
|
|
96
|
+
cancelLabel?: string;
|
|
97
|
+
isDanger?: boolean;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
interface UseConfirmDialogReturn {
|
|
101
|
+
isOpen: boolean;
|
|
102
|
+
confirm: () => Promise<boolean>;
|
|
103
|
+
dialogProps: ConfirmDialogProps;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
/**
|
|
107
|
+
* Hook for programmatic confirmation dialogs
|
|
108
|
+
*
|
|
109
|
+
* Usage:
|
|
110
|
+
* ```tsx
|
|
111
|
+
* const { confirm, dialogProps } = useConfirmDialog({
|
|
112
|
+
* title: 'Delete item?',
|
|
113
|
+
* message: 'This cannot be undone.',
|
|
114
|
+
* isDanger: true,
|
|
115
|
+
* });
|
|
116
|
+
*
|
|
117
|
+
* const handleDelete = async () => {
|
|
118
|
+
* if (await confirm()) {
|
|
119
|
+
* // User confirmed, proceed with deletion
|
|
120
|
+
* }
|
|
121
|
+
* };
|
|
122
|
+
*
|
|
123
|
+
* return (
|
|
124
|
+
* <>
|
|
125
|
+
* <button onClick={handleDelete}>Delete</button>
|
|
126
|
+
* <ConfirmDialog {...dialogProps} />
|
|
127
|
+
* </>
|
|
128
|
+
* );
|
|
129
|
+
* ```
|
|
130
|
+
*/
|
|
131
|
+
export function useConfirmDialog(options: UseConfirmDialogOptions): UseConfirmDialogReturn {
|
|
132
|
+
const [isOpen, setIsOpen] = React.useState(false);
|
|
133
|
+
const resolveRef = useRef<((value: boolean) => void) | null>(null);
|
|
134
|
+
|
|
135
|
+
const confirm = useCallback(() => {
|
|
136
|
+
return new Promise<boolean>((resolve) => {
|
|
137
|
+
resolveRef.current = resolve;
|
|
138
|
+
setIsOpen(true);
|
|
139
|
+
});
|
|
140
|
+
}, []);
|
|
141
|
+
|
|
142
|
+
const handleConfirm = useCallback(() => {
|
|
143
|
+
setIsOpen(false);
|
|
144
|
+
resolveRef.current?.(true);
|
|
145
|
+
resolveRef.current = null;
|
|
146
|
+
}, []);
|
|
147
|
+
|
|
148
|
+
const handleCancel = useCallback(() => {
|
|
149
|
+
setIsOpen(false);
|
|
150
|
+
resolveRef.current?.(false);
|
|
151
|
+
resolveRef.current = null;
|
|
152
|
+
}, []);
|
|
153
|
+
|
|
154
|
+
const dialogProps: ConfirmDialogProps = {
|
|
155
|
+
isOpen,
|
|
156
|
+
title: options.title,
|
|
157
|
+
message: options.message,
|
|
158
|
+
confirmLabel: options.confirmLabel,
|
|
159
|
+
cancelLabel: options.cancelLabel,
|
|
160
|
+
isDanger: options.isDanger,
|
|
161
|
+
onConfirm: handleConfirm,
|
|
162
|
+
onCancel: handleCancel,
|
|
163
|
+
};
|
|
164
|
+
|
|
165
|
+
return { isOpen, confirm, dialogProps };
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
export default ConfirmDialog;
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ContextIndicator Component Styles
|
|
3
|
+
*
|
|
4
|
+
* Story MSSCI-12712 - ContextIndicator Component
|
|
5
|
+
*
|
|
6
|
+
* Layout, compact mode, level text colors, and warning animation.
|
|
7
|
+
* The progress bar itself is rendered by shadcn Progress primitive.
|
|
8
|
+
*
|
|
9
|
+
* Color scheme:
|
|
10
|
+
* - normal (green): Safe context usage (<70%)
|
|
11
|
+
* - elevated (amber): Warning context usage (70-89%)
|
|
12
|
+
* - high (red): Danger context usage (90%+)
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
.context-indicator {
|
|
16
|
+
display: flex;
|
|
17
|
+
align-items: center;
|
|
18
|
+
gap: 8px;
|
|
19
|
+
padding: 4px 8px;
|
|
20
|
+
border-radius: var(--border-radius, 4px);
|
|
21
|
+
background: var(--bg-tertiary, rgba(255, 255, 255, 0.05));
|
|
22
|
+
font-size: 12px;
|
|
23
|
+
font-family: var(--font-mono, 'SF Mono', Monaco, monospace);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/* Compact mode for smaller displays */
|
|
27
|
+
.context-indicator--compact {
|
|
28
|
+
padding: 2px 6px;
|
|
29
|
+
gap: 4px;
|
|
30
|
+
font-size: 10px;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/* Percentage text */
|
|
34
|
+
.context-percent {
|
|
35
|
+
min-width: 32px;
|
|
36
|
+
text-align: right;
|
|
37
|
+
color: var(--text-secondary, rgba(255, 255, 255, 0.7));
|
|
38
|
+
transition: color 0.3s ease-out;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
.context-indicator--compact .context-percent {
|
|
42
|
+
min-width: 28px;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/* Level text colors - Normal (green) */
|
|
46
|
+
.context-indicator.level-normal .context-percent {
|
|
47
|
+
color: var(--color-success, #22c55e);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
/* Level text colors - Elevated (amber) */
|
|
51
|
+
.context-indicator.level-elevated .context-percent {
|
|
52
|
+
color: var(--color-warning, #f59e0b);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/* Level text colors - High (red) */
|
|
56
|
+
.context-indicator.level-high .context-percent {
|
|
57
|
+
color: var(--color-danger, #ef4444);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/* Warning state (data-warning attribute) */
|
|
61
|
+
.context-indicator[data-warning] {
|
|
62
|
+
animation: pulse-warning 2s ease-in-out infinite;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
@keyframes pulse-warning {
|
|
66
|
+
0%, 100% {
|
|
67
|
+
opacity: 1;
|
|
68
|
+
}
|
|
69
|
+
50% {
|
|
70
|
+
opacity: 0.8;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/* Screen reader only warning text */
|
|
75
|
+
.visually-hidden {
|
|
76
|
+
position: absolute;
|
|
77
|
+
width: 1px;
|
|
78
|
+
height: 1px;
|
|
79
|
+
padding: 0;
|
|
80
|
+
margin: -1px;
|
|
81
|
+
overflow: hidden;
|
|
82
|
+
clip: rect(0, 0, 0, 0);
|
|
83
|
+
white-space: nowrap;
|
|
84
|
+
border: 0;
|
|
85
|
+
}
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ContextIndicator Component
|
|
3
|
+
*
|
|
4
|
+
* Displays context window usage as a visual progress bar with threshold colors.
|
|
5
|
+
* Story MSSCI-12712 - ContextIndicator Component
|
|
6
|
+
*
|
|
7
|
+
* Features:
|
|
8
|
+
* - Percentage display with clamping (0-100%)
|
|
9
|
+
* - Color thresholds: normal (<70%), elevated (70-89%), high (90%+)
|
|
10
|
+
* - Real-time updates via IPC subscriptions
|
|
11
|
+
* - Warning display at 90% threshold
|
|
12
|
+
* - Tooltip showing exact token count (shadcn Tooltip)
|
|
13
|
+
* - Accessible with ARIA attributes
|
|
14
|
+
* - Uses shadcn Progress primitive for the progress bar
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import React, { useState, useEffect } from 'react';
|
|
18
|
+
import { Progress } from '@/components/ui/progress';
|
|
19
|
+
import {
|
|
20
|
+
Tooltip,
|
|
21
|
+
TooltipTrigger,
|
|
22
|
+
TooltipContent,
|
|
23
|
+
TooltipProvider,
|
|
24
|
+
} from '@/components/ui/tooltip';
|
|
25
|
+
import { cn } from '@/lib/utils';
|
|
26
|
+
import './ContextIndicator.css';
|
|
27
|
+
|
|
28
|
+
// ============================================================================
|
|
29
|
+
// Constants
|
|
30
|
+
// ============================================================================
|
|
31
|
+
|
|
32
|
+
/** Threshold values for context level colors */
|
|
33
|
+
export const CONTEXT_THRESHOLDS = {
|
|
34
|
+
WARNING: 70,
|
|
35
|
+
DANGER: 90,
|
|
36
|
+
} as const;
|
|
37
|
+
|
|
38
|
+
/** Test IDs for DOM querying */
|
|
39
|
+
export const CONTEXT_INDICATOR_TESTID = 'context-indicator';
|
|
40
|
+
export const CONTEXT_PERCENT_TESTID = 'context-percent';
|
|
41
|
+
export const CONTEXT_BAR_TESTID = 'context-bar';
|
|
42
|
+
export const CONTEXT_FILL_TESTID = 'context-fill';
|
|
43
|
+
|
|
44
|
+
/** CSS class names */
|
|
45
|
+
export const COMPONENT_CLASSNAME = 'context-indicator';
|
|
46
|
+
export const COMPACT_MODE_CLASSNAME = 'context-indicator--compact';
|
|
47
|
+
|
|
48
|
+
/** Accessibility constants */
|
|
49
|
+
export const PROGRESS_BAR_ROLE = 'progressbar';
|
|
50
|
+
export const CONTEXT_INDICATOR_ARIA_LIVE = 'polite';
|
|
51
|
+
|
|
52
|
+
/** Warning message displayed at high context usage */
|
|
53
|
+
export const CONTEXT_WARNING_MESSAGE = 'Context usage is high. Consider starting a new conversation.';
|
|
54
|
+
|
|
55
|
+
// ============================================================================
|
|
56
|
+
// Types
|
|
57
|
+
// ============================================================================
|
|
58
|
+
|
|
59
|
+
export type ContextLevel = 'normal' | 'elevated' | 'high';
|
|
60
|
+
|
|
61
|
+
export interface ContextData {
|
|
62
|
+
percent: number;
|
|
63
|
+
used?: number;
|
|
64
|
+
total?: number;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export interface ContextIndicatorProps {
|
|
68
|
+
/** Override percentage (0-100). If not provided, uses hook data */
|
|
69
|
+
percent?: number;
|
|
70
|
+
/** Used tokens for tooltip */
|
|
71
|
+
used?: number;
|
|
72
|
+
/** Total tokens for tooltip */
|
|
73
|
+
total?: number;
|
|
74
|
+
/** Compact mode for smaller displays */
|
|
75
|
+
compact?: boolean;
|
|
76
|
+
/** Additional CSS class name */
|
|
77
|
+
className?: string;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
interface UseContextIndicatorResult {
|
|
81
|
+
context: ContextData | null;
|
|
82
|
+
isLoading: boolean;
|
|
83
|
+
error: Error | null;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// ============================================================================
|
|
87
|
+
// Utility Functions
|
|
88
|
+
// ============================================================================
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Format a number as a percentage string with clamping.
|
|
92
|
+
* @param value - The percentage value (may be outside 0-100)
|
|
93
|
+
* @returns Formatted string like "45%"
|
|
94
|
+
*/
|
|
95
|
+
export function formatPercentage(value: number | null | undefined): string {
|
|
96
|
+
if (value === null || value === undefined || isNaN(value)) {
|
|
97
|
+
return '0%';
|
|
98
|
+
}
|
|
99
|
+
const clamped = Math.min(100, Math.max(0, Math.round(value)));
|
|
100
|
+
return `${clamped}%`;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
/**
|
|
104
|
+
* Format a token count with thousands separator.
|
|
105
|
+
* @param count - The token count
|
|
106
|
+
* @returns Formatted string like "50,000"
|
|
107
|
+
*/
|
|
108
|
+
export function formatTokenCount(count: number | null | undefined): string {
|
|
109
|
+
if (count === null || count === undefined || isNaN(count)) {
|
|
110
|
+
return '—';
|
|
111
|
+
}
|
|
112
|
+
return count.toLocaleString('en-US');
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Format tooltip text showing used/total tokens.
|
|
117
|
+
* @param used - Used tokens
|
|
118
|
+
* @param total - Total tokens
|
|
119
|
+
* @returns Formatted string like "50,000 / 200,000 tokens"
|
|
120
|
+
*/
|
|
121
|
+
export function formatTooltip(
|
|
122
|
+
used: number | null | undefined,
|
|
123
|
+
total: number | null | undefined
|
|
124
|
+
): string {
|
|
125
|
+
const usedStr = formatTokenCount(used);
|
|
126
|
+
const totalStr = formatTokenCount(total);
|
|
127
|
+
return `${usedStr} / ${totalStr} tokens`;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Determine the context level based on percentage.
|
|
132
|
+
* @param percent - The percentage value (0-100)
|
|
133
|
+
* @returns The context level: 'normal', 'elevated', or 'high'
|
|
134
|
+
*/
|
|
135
|
+
export function getContextLevel(percent: number): ContextLevel {
|
|
136
|
+
if (percent >= CONTEXT_THRESHOLDS.DANGER) {
|
|
137
|
+
return 'high';
|
|
138
|
+
}
|
|
139
|
+
if (percent >= CONTEXT_THRESHOLDS.WARNING) {
|
|
140
|
+
return 'elevated';
|
|
141
|
+
}
|
|
142
|
+
return 'normal';
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
/**
|
|
146
|
+
* Get the CSS class name for a context level.
|
|
147
|
+
* @param level - The context level
|
|
148
|
+
* @returns CSS class name like "level-normal"
|
|
149
|
+
*/
|
|
150
|
+
export function getLevelClassName(level: ContextLevel): string {
|
|
151
|
+
return `level-${level}`;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
/**
|
|
155
|
+
* Determine if warning should be shown.
|
|
156
|
+
* @param percent - The percentage value
|
|
157
|
+
* @returns True if at or above danger threshold
|
|
158
|
+
*/
|
|
159
|
+
export function shouldShowWarning(percent: number): boolean {
|
|
160
|
+
return percent >= CONTEXT_THRESHOLDS.DANGER;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Get ARIA attributes for the progress bar.
|
|
165
|
+
* @param valuenow - Current value
|
|
166
|
+
* @param valuemax - Maximum value
|
|
167
|
+
* @returns Object with aria attributes
|
|
168
|
+
*/
|
|
169
|
+
export function getAriaAttributes(
|
|
170
|
+
valuenow: number,
|
|
171
|
+
valuemax: number
|
|
172
|
+
): {
|
|
173
|
+
'aria-valuenow': number;
|
|
174
|
+
'aria-valuemin': number;
|
|
175
|
+
'aria-valuemax': number;
|
|
176
|
+
'aria-label': string;
|
|
177
|
+
} {
|
|
178
|
+
return {
|
|
179
|
+
'aria-valuenow': valuenow,
|
|
180
|
+
'aria-valuemin': 0,
|
|
181
|
+
'aria-valuemax': valuemax,
|
|
182
|
+
'aria-label': `Context usage: ${valuenow}%`,
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
// ============================================================================
|
|
187
|
+
// Progress bar color mapping
|
|
188
|
+
// ============================================================================
|
|
189
|
+
|
|
190
|
+
/** Tailwind indicator color classes keyed by context level */
|
|
191
|
+
const LEVEL_INDICATOR_COLORS: Record<ContextLevel, string> = {
|
|
192
|
+
normal: 'bg-green-500',
|
|
193
|
+
elevated: 'bg-amber-500',
|
|
194
|
+
high: 'bg-red-500',
|
|
195
|
+
};
|
|
196
|
+
|
|
197
|
+
// ============================================================================
|
|
198
|
+
// Hook
|
|
199
|
+
// ============================================================================
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* React hook for fetching and subscribing to context data.
|
|
203
|
+
* Uses WebSocket /ws/context for real-time updates.
|
|
204
|
+
*/
|
|
205
|
+
export function useContextIndicator(): UseContextIndicatorResult {
|
|
206
|
+
const [context, setContext] = useState<ContextData | null>(null);
|
|
207
|
+
const [isLoading, setIsLoading] = useState(true);
|
|
208
|
+
const [error, setError] = useState<Error | null>(null);
|
|
209
|
+
|
|
210
|
+
useEffect(() => {
|
|
211
|
+
// Connect to context WebSocket
|
|
212
|
+
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
|
213
|
+
const ws = new WebSocket(`${protocol}//${window.location.host}/ws/context`);
|
|
214
|
+
|
|
215
|
+
ws.onopen = () => {
|
|
216
|
+
console.log('[ContextIndicator] WebSocket connected');
|
|
217
|
+
};
|
|
218
|
+
|
|
219
|
+
ws.onmessage = (event) => {
|
|
220
|
+
try {
|
|
221
|
+
const data = JSON.parse(event.data);
|
|
222
|
+
if (data.type === 'init' || data.type === 'update') {
|
|
223
|
+
setContext(data.context as ContextData | null);
|
|
224
|
+
setIsLoading(false);
|
|
225
|
+
}
|
|
226
|
+
} catch (err) {
|
|
227
|
+
console.error('[ContextIndicator] Failed to parse message:', err);
|
|
228
|
+
setError(err instanceof Error ? err : new Error('Failed to parse context'));
|
|
229
|
+
}
|
|
230
|
+
};
|
|
231
|
+
|
|
232
|
+
ws.onerror = () => {
|
|
233
|
+
setError(new Error('WebSocket connection failed'));
|
|
234
|
+
setIsLoading(false);
|
|
235
|
+
};
|
|
236
|
+
|
|
237
|
+
ws.onclose = () => {
|
|
238
|
+
console.log('[ContextIndicator] WebSocket disconnected');
|
|
239
|
+
};
|
|
240
|
+
|
|
241
|
+
return () => ws.close();
|
|
242
|
+
}, []);
|
|
243
|
+
|
|
244
|
+
return { context, isLoading, error };
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// Re-export as useContext for test compatibility
|
|
248
|
+
export const useContext = useContextIndicator;
|
|
249
|
+
|
|
250
|
+
// ============================================================================
|
|
251
|
+
// Component
|
|
252
|
+
// ============================================================================
|
|
253
|
+
|
|
254
|
+
/**
|
|
255
|
+
* ContextIndicator Component
|
|
256
|
+
*
|
|
257
|
+
* Displays context window usage as a visual progress bar using shadcn
|
|
258
|
+
* Progress and Tooltip primitives.
|
|
259
|
+
*/
|
|
260
|
+
export default function ContextIndicator({
|
|
261
|
+
percent: propPercent,
|
|
262
|
+
used: propUsed,
|
|
263
|
+
total: propTotal,
|
|
264
|
+
compact = false,
|
|
265
|
+
className = '',
|
|
266
|
+
}: ContextIndicatorProps): React.ReactElement {
|
|
267
|
+
const { context } = useContextIndicator();
|
|
268
|
+
|
|
269
|
+
// Use props if provided, otherwise use hook data
|
|
270
|
+
const percent = propPercent ?? context?.percent ?? 0;
|
|
271
|
+
const used = propUsed ?? context?.used;
|
|
272
|
+
const total = propTotal ?? context?.total;
|
|
273
|
+
|
|
274
|
+
const clampedPercent = Math.min(100, Math.max(0, Math.round(percent)));
|
|
275
|
+
const level = getContextLevel(percent);
|
|
276
|
+
const levelClass = getLevelClassName(level);
|
|
277
|
+
const showWarning = shouldShowWarning(percent);
|
|
278
|
+
const tooltip = formatTooltip(used, total);
|
|
279
|
+
const ariaAttrs = getAriaAttributes(percent, 100);
|
|
280
|
+
|
|
281
|
+
const rootClassNames = cn(
|
|
282
|
+
COMPONENT_CLASSNAME,
|
|
283
|
+
levelClass,
|
|
284
|
+
compact && COMPACT_MODE_CLASSNAME,
|
|
285
|
+
className,
|
|
286
|
+
);
|
|
287
|
+
|
|
288
|
+
return (
|
|
289
|
+
<TooltipProvider delayDuration={300}>
|
|
290
|
+
<Tooltip>
|
|
291
|
+
<TooltipTrigger asChild>
|
|
292
|
+
<div
|
|
293
|
+
className={rootClassNames}
|
|
294
|
+
data-testid={CONTEXT_INDICATOR_TESTID}
|
|
295
|
+
data-warning={showWarning || undefined}
|
|
296
|
+
aria-live={CONTEXT_INDICATOR_ARIA_LIVE}
|
|
297
|
+
>
|
|
298
|
+
<Progress
|
|
299
|
+
value={clampedPercent}
|
|
300
|
+
className={cn(
|
|
301
|
+
'context-bar',
|
|
302
|
+
compact ? 'h-1 min-w-[40px] max-w-[60px]' : 'h-1.5 min-w-[60px] max-w-[100px]',
|
|
303
|
+
)}
|
|
304
|
+
data-testid={CONTEXT_BAR_TESTID}
|
|
305
|
+
{...ariaAttrs}
|
|
306
|
+
indicatorClassName={cn(
|
|
307
|
+
'transition-all duration-300 ease-out',
|
|
308
|
+
LEVEL_INDICATOR_COLORS[level],
|
|
309
|
+
)}
|
|
310
|
+
/>
|
|
311
|
+
<span
|
|
312
|
+
className={cn('context-percent', `text-${level}`)}
|
|
313
|
+
data-testid={CONTEXT_PERCENT_TESTID}
|
|
314
|
+
>
|
|
315
|
+
{formatPercentage(percent)}
|
|
316
|
+
</span>
|
|
317
|
+
{showWarning && (
|
|
318
|
+
<span className="context-warning visually-hidden">
|
|
319
|
+
{CONTEXT_WARNING_MESSAGE}
|
|
320
|
+
</span>
|
|
321
|
+
)}
|
|
322
|
+
</div>
|
|
323
|
+
</TooltipTrigger>
|
|
324
|
+
<TooltipContent side="bottom">
|
|
325
|
+
<p>{tooltip}</p>
|
|
326
|
+
</TooltipContent>
|
|
327
|
+
</Tooltip>
|
|
328
|
+
</TooltipProvider>
|
|
329
|
+
);
|
|
330
|
+
}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
|
|
3
|
+
export interface SparklinePoint {
|
|
4
|
+
percent: number;
|
|
5
|
+
tokens: number;
|
|
6
|
+
timestamp: number;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export interface ContextSparklineProps {
|
|
10
|
+
history: SparklinePoint[];
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
function getBarColor(tokens: number): string {
|
|
14
|
+
if (tokens >= 50000) return 'var(--color-danger, #ef4444)';
|
|
15
|
+
if (tokens >= 5000) return 'var(--color-warning, #f59e0b)';
|
|
16
|
+
return 'var(--color-success, #22c55e)';
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function ContextSparkline({ history }: ContextSparklineProps): React.ReactElement | null {
|
|
20
|
+
if (history.length < 2) return null;
|
|
21
|
+
|
|
22
|
+
const W = 200;
|
|
23
|
+
const H = 32;
|
|
24
|
+
const PAD = 2;
|
|
25
|
+
const usable = H - PAD * 2;
|
|
26
|
+
const latest = history[history.length - 1];
|
|
27
|
+
const barWidth = W / history.length;
|
|
28
|
+
|
|
29
|
+
return (
|
|
30
|
+
<div className="context-sparkline" data-testid="context-sparkline">
|
|
31
|
+
<svg
|
|
32
|
+
viewBox={`0 0 ${W} ${H}`}
|
|
33
|
+
role="img"
|
|
34
|
+
aria-label={`Context usage trend: currently ${latest.percent}%`}
|
|
35
|
+
>
|
|
36
|
+
{history.map((p, i) => {
|
|
37
|
+
const barHeight = Math.max(1, (p.percent / 100) * usable);
|
|
38
|
+
const x = i * barWidth;
|
|
39
|
+
const y = PAD + usable - barHeight;
|
|
40
|
+
return (
|
|
41
|
+
<rect
|
|
42
|
+
key={i}
|
|
43
|
+
x={x}
|
|
44
|
+
y={y}
|
|
45
|
+
width={barWidth}
|
|
46
|
+
height={barHeight}
|
|
47
|
+
fill={getBarColor(p.tokens)}
|
|
48
|
+
opacity={0.8}
|
|
49
|
+
rx={0.5}
|
|
50
|
+
/>
|
|
51
|
+
);
|
|
52
|
+
})}
|
|
53
|
+
</svg>
|
|
54
|
+
</div>
|
|
55
|
+
);
|
|
56
|
+
}
|