@pennyfarthing/cyclist 10.4.0 → 11.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/agent-load.d.ts +1 -2
- package/dist/api/agent-load.d.ts.map +1 -1
- package/dist/api/agent-load.js +2 -123
- package/dist/api/agent-load.js.map +1 -1
- package/dist/api/audit-log.d.ts +1 -17
- package/dist/api/audit-log.d.ts.map +1 -1
- package/dist/api/audit-log.js +2 -162
- package/dist/api/audit-log.js.map +1 -1
- package/dist/api/background-tasks.d.ts +1 -26
- package/dist/api/background-tasks.d.ts.map +1 -1
- package/dist/api/background-tasks.js +2 -55
- package/dist/api/background-tasks.js.map +1 -1
- package/dist/api/bell.d.ts +1 -18
- package/dist/api/bell.d.ts.map +1 -1
- package/dist/api/bell.js +2 -33
- package/dist/api/bell.js.map +1 -1
- package/dist/api/code-markers.d.ts +1 -8
- package/dist/api/code-markers.d.ts.map +1 -1
- package/dist/api/code-markers.js +2 -61
- package/dist/api/code-markers.js.map +1 -1
- package/dist/api/complexity.d.ts +1 -2
- package/dist/api/complexity.d.ts.map +1 -1
- package/dist/api/complexity.js +2 -46
- package/dist/api/complexity.js.map +1 -1
- package/dist/api/context.d.ts +1 -37
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +2 -143
- package/dist/api/context.js.map +1 -1
- package/dist/api/dead-code.d.ts +1 -2
- package/dist/api/dead-code.d.ts.map +1 -1
- package/dist/api/dead-code.js +2 -69
- package/dist/api/dead-code.js.map +1 -1
- package/dist/api/dependencies.d.ts +1 -2
- package/dist/api/dependencies.d.ts.map +1 -1
- package/dist/api/dependencies.js +2 -42
- package/dist/api/dependencies.js.map +1 -1
- package/dist/api/evaluation.d.ts +1 -19
- package/dist/api/evaluation.d.ts.map +1 -1
- package/dist/api/evaluation.js +2 -127
- package/dist/api/evaluation.js.map +1 -1
- package/dist/api/file-browser.d.ts +1 -8
- package/dist/api/file-browser.d.ts.map +1 -1
- package/dist/api/file-browser.js +2 -114
- package/dist/api/file-browser.js.map +1 -1
- package/dist/api/git.d.ts +1 -46
- package/dist/api/git.d.ts.map +1 -1
- package/dist/api/git.js +2 -354
- package/dist/api/git.js.map +1 -1
- package/dist/api/health-score.d.ts +1 -2
- package/dist/api/health-score.d.ts.map +1 -1
- package/dist/api/health-score.js +2 -46
- package/dist/api/health-score.js.map +1 -1
- package/dist/api/hook-request.d.ts +1 -40
- package/dist/api/hook-request.d.ts.map +1 -1
- package/dist/api/hook-request.js +2 -277
- package/dist/api/hook-request.js.map +1 -1
- package/dist/api/hotspots.d.ts +1 -2
- package/dist/api/hotspots.d.ts.map +1 -1
- package/dist/api/hotspots.js +2 -61
- package/dist/api/hotspots.js.map +1 -1
- package/dist/api/identity.d.ts +1 -16
- package/dist/api/identity.d.ts.map +1 -1
- package/dist/api/identity.js +2 -78
- package/dist/api/identity.js.map +1 -1
- package/dist/api/index.d.ts +1 -34
- package/dist/api/index.d.ts.map +1 -1
- package/dist/api/index.js +2 -44
- package/dist/api/index.js.map +1 -1
- package/dist/api/mode.d.ts +1 -22
- package/dist/api/mode.d.ts.map +1 -1
- package/dist/api/mode.js +2 -37
- package/dist/api/mode.js.map +1 -1
- package/dist/api/otlp.d.ts +1 -2
- package/dist/api/otlp.d.ts.map +1 -1
- package/dist/api/otlp.js +2 -46
- package/dist/api/otlp.js.map +1 -1
- package/dist/api/permissions.d.ts +1 -15
- package/dist/api/permissions.d.ts.map +1 -1
- package/dist/api/permissions.js +2 -66
- package/dist/api/permissions.js.map +1 -1
- package/dist/api/persona.d.ts +1 -8
- package/dist/api/persona.d.ts.map +1 -1
- package/dist/api/persona.js +2 -67
- package/dist/api/persona.js.map +1 -1
- package/dist/api/portrait.d.ts +1 -5
- package/dist/api/portrait.d.ts.map +1 -1
- package/dist/api/portrait.js +2 -27
- package/dist/api/portrait.js.map +1 -1
- package/dist/api/settings.d.ts +1 -53
- package/dist/api/settings.d.ts.map +1 -1
- package/dist/api/settings.js +2 -464
- package/dist/api/settings.js.map +1 -1
- package/dist/api/spans.d.ts +1 -16
- package/dist/api/spans.d.ts.map +1 -1
- package/dist/api/spans.js +2 -244
- package/dist/api/spans.js.map +1 -1
- package/dist/api/stats.d.ts +1 -12
- package/dist/api/stats.d.ts.map +1 -1
- package/dist/api/stats.js +2 -84
- package/dist/api/stats.js.map +1 -1
- package/dist/api/story.d.ts +1 -2
- package/dist/api/story.d.ts.map +1 -1
- package/dist/api/story.js +2 -14
- package/dist/api/story.js.map +1 -1
- package/dist/api/telemetry.d.ts +1 -18
- package/dist/api/telemetry.d.ts.map +1 -1
- package/dist/api/telemetry.js +2 -164
- package/dist/api/telemetry.js.map +1 -1
- package/dist/api/theme-agents.d.ts +1 -60
- package/dist/api/theme-agents.d.ts.map +1 -1
- package/dist/api/theme-agents.js +2 -213
- package/dist/api/theme-agents.js.map +1 -1
- package/dist/api/todos.d.ts +1 -32
- package/dist/api/todos.d.ts.map +1 -1
- package/dist/api/todos.js +2 -43
- package/dist/api/todos.js.map +1 -1
- package/dist/api/token-stats.d.ts +1 -7
- package/dist/api/token-stats.d.ts.map +1 -1
- package/dist/api/token-stats.js +2 -35
- package/dist/api/token-stats.js.map +1 -1
- package/dist/api/welcome.d.ts +1 -21
- package/dist/api/welcome.d.ts.map +1 -1
- package/dist/api/welcome.js +2 -34
- package/dist/api/welcome.js.map +1 -1
- package/dist/bikerack.js +2 -2
- package/dist/bikerack.js.map +1 -1
- package/dist/env.d.ts +6 -0
- package/dist/env.d.ts.map +1 -0
- package/dist/env.js +10 -0
- package/dist/env.js.map +1 -0
- package/dist/focus.d.ts +53 -0
- package/dist/focus.d.ts.map +1 -0
- package/dist/focus.js +122 -0
- package/dist/focus.js.map +1 -0
- package/dist/git-cache.d.ts +1 -0
- package/dist/git-cache.d.ts.map +1 -1
- package/dist/git-cache.js +3 -1
- package/dist/git-cache.js.map +1 -1
- package/dist/menu-builder.d.ts.map +1 -1
- package/dist/menu-builder.js +0 -1
- package/dist/menu-builder.js.map +1 -1
- package/dist/prime.d.ts +3 -3
- package/dist/prime.d.ts.map +1 -1
- package/dist/prime.js +38 -14
- package/dist/prime.js.map +1 -1
- package/dist/public/css/react.css +1 -1
- package/dist/public/js/react/react.js +53 -61
- package/dist/server.d.ts +18 -85
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +105 -405
- package/dist/server.js.map +1 -1
- package/dist/sprint-data.d.ts +1 -1
- package/dist/sprint-data.d.ts.map +1 -1
- package/dist/sprint-data.js +2 -2
- package/dist/sprint-data.js.map +1 -1
- package/dist/theme-metadata.d.ts +3 -3
- package/dist/theme-metadata.d.ts.map +1 -1
- package/dist/theme-metadata.js +4 -4
- package/dist/theme-metadata.js.map +1 -1
- package/dist/websocket.d.ts +2 -0
- package/dist/websocket.d.ts.map +1 -1
- package/dist/websocket.js +53 -75
- package/dist/websocket.js.map +1 -1
- package/package.json +2 -6
- package/portraits/hogans-heroes/large/burkhalter-35312.png +0 -0
- package/portraits/hogans-heroes/large/carter-34352.png +0 -0
- package/portraits/hogans-heroes/large/hochstetter-45314.png +0 -0
- package/portraits/hogans-heroes/large/hogan-44541.png +0 -0
- package/portraits/hogans-heroes/large/kinch-35241.png +0 -0
- package/portraits/hogans-heroes/large/klink-23434.png +0 -0
- package/portraits/hogans-heroes/large/lebeau-45443.png +0 -0
- package/portraits/hogans-heroes/large/marya-53543.png +0 -0
- package/portraits/hogans-heroes/large/newkirk-54432.png +0 -0
- package/portraits/hogans-heroes/large/schultz-42453.png +0 -0
- package/portraits/hogans-heroes/large/underground-55131.png +0 -0
- package/portraits/hogans-heroes/medium/burkhalter-35312.png +0 -0
- package/portraits/hogans-heroes/medium/carter-34352.png +0 -0
- package/portraits/hogans-heroes/medium/hochstetter-45314.png +0 -0
- package/portraits/hogans-heroes/medium/hogan-44541.png +0 -0
- package/portraits/hogans-heroes/medium/kinch-35241.png +0 -0
- package/portraits/hogans-heroes/medium/klink-23434.png +0 -0
- package/portraits/hogans-heroes/medium/lebeau-45443.png +0 -0
- package/portraits/hogans-heroes/medium/marya-53543.png +0 -0
- package/portraits/hogans-heroes/medium/newkirk-54432.png +0 -0
- package/portraits/hogans-heroes/medium/schultz-42453.png +0 -0
- package/portraits/hogans-heroes/medium/underground-55131.png +0 -0
- package/portraits/monty-python/large/announcer-44441.png +0 -0
- package/portraits/monty-python/large/arguer-35412.png +0 -0
- package/portraits/monty-python/large/bicycle-repair-man-35241.png +0 -0
- package/portraits/monty-python/large/colonel-35423.png +0 -0
- package/portraits/monty-python/large/counsellor-45341.png +0 -0
- package/portraits/monty-python/large/gumbys-23524.png +0 -0
- package/portraits/monty-python/large/nudge-43533.png +0 -0
- package/portraits/monty-python/large/praline-45413.png +0 -0
- package/portraits/monty-python/large/silly-walks-55322.png +0 -0
- package/portraits/monty-python/large/wensleydale-54451.png +0 -0
- package/portraits/monty-python/large/xim-nez-43534.png +0 -0
- package/portraits/monty-python/medium/announcer-44441.png +0 -0
- package/portraits/monty-python/medium/arguer-35412.png +0 -0
- package/portraits/monty-python/medium/bicycle-repair-man-35241.png +0 -0
- package/portraits/monty-python/medium/colonel-35423.png +0 -0
- package/portraits/monty-python/medium/counsellor-45341.png +0 -0
- package/portraits/monty-python/medium/gumbys-23524.png +0 -0
- package/portraits/monty-python/medium/nudge-43533.png +0 -0
- package/portraits/monty-python/medium/praline-45413.png +0 -0
- package/portraits/monty-python/medium/silly-walks-55322.png +0 -0
- package/portraits/monty-python/medium/wensleydale-54451.png +0 -0
- package/portraits/monty-python/medium/xim-nez-43534.png +0 -0
- package/portraits/stephen-king/large/andy-55231.png +0 -0
- package/portraits/stephen-king/large/christine-25112.png +0 -0
- package/portraits/stephen-king/large/danny-53243.png +0 -0
- package/portraits/stephen-king/large/flagg-55311.png +0 -0
- package/portraits/stephen-king/large/gaunt-54421.png +0 -0
- package/portraits/stephen-king/large/jack-44224.png +0 -0
- package/portraits/stephen-king/large/johnny-44353.png +0 -0
- package/portraits/stephen-king/large/margaret-15415.png +0 -0
- package/portraits/stephen-king/large/paul-45233.png +0 -0
- package/portraits/stephen-king/large/pennywise-54411.png +0 -0
- package/portraits/stephen-king/large/roland-35121.png +0 -0
- package/portraits/stephen-king/medium/andy-55231.png +0 -0
- package/portraits/stephen-king/medium/christine-25112.png +0 -0
- package/portraits/stephen-king/medium/danny-53243.png +0 -0
- package/portraits/stephen-king/medium/flagg-55311.png +0 -0
- package/portraits/stephen-king/medium/gaunt-54421.png +0 -0
- package/portraits/stephen-king/medium/jack-44224.png +0 -0
- package/portraits/stephen-king/medium/johnny-44353.png +0 -0
- package/portraits/stephen-king/medium/margaret-15415.png +0 -0
- package/portraits/stephen-king/medium/paul-45233.png +0 -0
- package/portraits/stephen-king/medium/pennywise-54411.png +0 -0
- package/portraits/stephen-king/medium/roland-35121.png +0 -0
- package/portraits/star-trek-tng/large/beverly-44352.png +0 -0
- package/portraits/star-trek-tng/large/data-55241.png +0 -0
- package/portraits/star-trek-tng/large/deanna-43353.png +0 -0
- package/portraits/star-trek-tng/large/geordi-54342.png +0 -0
- package/portraits/star-trek-tng/large/jean-luc-45342.png +0 -0
- package/portraits/star-trek-tng/large/kathryn-45332.png +0 -0
- package/portraits/star-trek-tng/large/miles-35342.png +0 -0
- package/portraits/star-trek-tng/large/q-53521.png +0 -0
- package/portraits/star-trek-tng/large/spock-45231.png +0 -0
- package/portraits/star-trek-tng/large/troi-44352.png +0 -0
- package/portraits/star-trek-tng/medium/beverly-44352.png +0 -0
- package/portraits/star-trek-tng/medium/data-55241.png +0 -0
- package/portraits/star-trek-tng/medium/deanna-43353.png +0 -0
- package/portraits/star-trek-tng/medium/geordi-54342.png +0 -0
- package/portraits/star-trek-tng/medium/jean-luc-45342.png +0 -0
- package/portraits/star-trek-tng/medium/kathryn-45332.png +0 -0
- package/portraits/star-trek-tng/medium/miles-35342.png +0 -0
- package/portraits/star-trek-tng/medium/q-53521.png +0 -0
- package/portraits/star-trek-tng/medium/spock-45231.png +0 -0
- package/portraits/star-trek-tng/medium/troi-44352.png +0 -0
- package/src/public/App.tsx +0 -340
- package/src/public/components/AgentLoadDialog.tsx +0 -202
- package/src/public/components/AgentPopup.tsx +0 -308
- package/src/public/components/ApprovalModal/ApprovalModal.css +0 -35
- package/src/public/components/ApprovalModal/index.tsx +0 -632
- package/src/public/components/BikeRackIndex.tsx +0 -54
- package/src/public/components/BikeRackWorkspace.tsx +0 -142
- package/src/public/components/CommandPalette.tsx +0 -555
- package/src/public/components/ConfirmDialog.tsx +0 -168
- package/src/public/components/ContextIndicator/ContextIndicator.css +0 -85
- package/src/public/components/ContextIndicator/index.tsx +0 -330
- package/src/public/components/ContextSparkline.tsx +0 -56
- package/src/public/components/ControlBar.tsx +0 -636
- package/src/public/components/DeadCodeDialog.tsx +0 -169
- package/src/public/components/DiffViewer.tsx +0 -585
- package/src/public/components/DockviewWorkspace.tsx +0 -737
- package/src/public/components/Editor.tsx +0 -630
- package/src/public/components/ErrorBoundary.tsx +0 -67
- package/src/public/components/FileTree.tsx +0 -379
- package/src/public/components/FontPicker/FontPicker.css +0 -276
- package/src/public/components/FontPicker/index.tsx +0 -430
- package/src/public/components/FullFileTree.tsx +0 -237
- package/src/public/components/HealthGauge.tsx +0 -181
- package/src/public/components/Message.tsx +0 -225
- package/src/public/components/MessageList.tsx +0 -98
- package/src/public/components/MessageView.tsx +0 -400
- package/src/public/components/ModeSwitch/ModeSwitch.css +0 -165
- package/src/public/components/ModeSwitch/index.tsx +0 -372
- package/src/public/components/PersonaHeader.tsx +0 -240
- package/src/public/components/QuickActions.tsx +0 -267
- package/src/public/components/SpanTimeline.tsx +0 -352
- package/src/public/components/StandalonePanel.tsx +0 -84
- package/src/public/components/StatsStrip.tsx +0 -162
- package/src/public/components/StreamingContent.tsx +0 -77
- package/src/public/components/SubagentSpan.tsx +0 -180
- package/src/public/components/TandemPortrait.tsx +0 -72
- package/src/public/components/ThemePalette/ThemePalette.css +0 -179
- package/src/public/components/ThemePalette/index.tsx +0 -326
- package/src/public/components/ToolCallBlock.tsx +0 -252
- package/src/public/components/ToolStack.tsx +0 -209
- package/src/public/components/ToolStatus.tsx +0 -57
- package/src/public/components/dialogs/CodeMarkersDialog.tsx +0 -169
- package/src/public/components/dialogs/ComplexityDialog.tsx +0 -163
- package/src/public/components/dialogs/DependenciesDialog.tsx +0 -120
- package/src/public/components/dialogs/HotspotsDialog.tsx +0 -451
- package/src/public/components/dialogs/ToolDialog.tsx +0 -43
- package/src/public/components/panel-registry.ts +0 -11
- package/src/public/components/panels/ACPanel.tsx +0 -93
- package/src/public/components/panels/AcceptanceCriteriaPanel.tsx +0 -104
- package/src/public/components/panels/AuditLogPanel.tsx +0 -465
- package/src/public/components/panels/BackgroundPanel.tsx +0 -115
- package/src/public/components/panels/BikeLanePanel.tsx +0 -214
- package/src/public/components/panels/ChangedPanel.tsx +0 -65
- package/src/public/components/panels/DebugPanel.tsx +0 -344
- package/src/public/components/panels/DiffsPanel.tsx +0 -155
- package/src/public/components/panels/GitPanel.tsx +0 -216
- package/src/public/components/panels/HotspotsPanel.tsx +0 -365
- package/src/public/components/panels/MessagePanel.tsx +0 -497
- package/src/public/components/panels/SettingsPanel.tsx +0 -453
- package/src/public/components/panels/SprintPanel.tsx +0 -670
- package/src/public/components/panels/TTYPanel.tsx +0 -299
- package/src/public/components/panels/TodoPanel.tsx +0 -142
- package/src/public/components/panels/WorkflowPanel.tsx +0 -224
- package/src/public/components/panels/index.ts +0 -24
- package/src/public/components/ui/alert-dialog.tsx +0 -139
- package/src/public/components/ui/badge.tsx +0 -36
- package/src/public/components/ui/button.tsx +0 -57
- package/src/public/components/ui/checkbox.tsx +0 -28
- package/src/public/components/ui/collapsible.tsx +0 -9
- package/src/public/components/ui/command.tsx +0 -151
- package/src/public/components/ui/dialog.tsx +0 -120
- package/src/public/components/ui/popover.tsx +0 -31
- package/src/public/components/ui/progress.tsx +0 -28
- package/src/public/components/ui/scroll-area.tsx +0 -46
- package/src/public/components/ui/select.tsx +0 -157
- package/src/public/components/ui/separator.tsx +0 -29
- package/src/public/components/ui/skeleton.tsx +0 -15
- package/src/public/components/ui/switch.tsx +0 -27
- package/src/public/components/ui/toggle-group.tsx +0 -59
- package/src/public/components/ui/toggle.tsx +0 -43
- package/src/public/components/ui/tooltip.tsx +0 -30
- package/src/public/contexts/ClaudeContext.tsx +0 -311
- package/src/public/contexts/MessageQueueContext.tsx +0 -143
- package/src/public/css/theme-browser.css +0 -550
- package/src/public/css/theme-system.css +0 -630
- package/src/public/hooks/index.ts +0 -49
- package/src/public/hooks/useAgentLoad.ts +0 -105
- package/src/public/hooks/useBackgroundTasks.ts +0 -131
- package/src/public/hooks/useClaude.ts +0 -234
- package/src/public/hooks/useCodeMarkers.ts +0 -101
- package/src/public/hooks/useColorScheme.ts +0 -42
- package/src/public/hooks/useCommandHistory.ts +0 -99
- package/src/public/hooks/useComplexity.ts +0 -80
- package/src/public/hooks/useDeadCode.ts +0 -99
- package/src/public/hooks/useDependencies.ts +0 -82
- package/src/public/hooks/useDiffs.ts +0 -143
- package/src/public/hooks/useFileBrowser.ts +0 -71
- package/src/public/hooks/useGitStatus.ts +0 -233
- package/src/public/hooks/useHealthScore.ts +0 -69
- package/src/public/hooks/useHotspots.ts +0 -123
- package/src/public/hooks/useLayoutPersistence.ts +0 -138
- package/src/public/hooks/useMarkdownParser.ts +0 -36
- package/src/public/hooks/useMarkerActions.ts +0 -234
- package/src/public/hooks/useMessageQueue.ts +0 -380
- package/src/public/hooks/useMessageStream.ts +0 -131
- package/src/public/hooks/usePersona.ts +0 -112
- package/src/public/hooks/usePlanModeExit.ts +0 -105
- package/src/public/hooks/useResponsiveLayout.ts +0 -173
- package/src/public/hooks/useSprint.ts +0 -147
- package/src/public/hooks/useStatsStrip.ts +0 -204
- package/src/public/hooks/useStory.ts +0 -135
- package/src/public/hooks/useSubagentHelper.ts +0 -64
- package/src/public/hooks/useSyntaxHighlighter.ts +0 -52
- package/src/public/hooks/useTabCompletion.ts +0 -124
- package/src/public/hooks/useTodos.ts +0 -93
- package/src/public/hooks/useUserAvatar.ts +0 -54
- package/src/public/index.tsx +0 -10
- package/src/public/lib/utils.ts +0 -6
- package/src/public/styles/dockview-theme.css +0 -459
- package/src/public/styles/tailwind.css +0 -4396
- package/src/public/types/electron.d.ts +0 -18
- package/src/public/types/message.ts +0 -51
- package/src/public/utils/avatar-service.ts +0 -73
- package/src/public/utils/color-presets.ts +0 -940
- package/src/public/utils/font-presets.ts +0 -362
- package/src/public/utils/formatDuration.ts +0 -14
- package/src/public/utils/markdown.ts +0 -249
- package/src/public/utils/messageFilters.ts +0 -128
- package/src/public/utils/slash-commands.ts +0 -353
- package/src/public/utils/subagent-display.ts +0 -146
- package/src/public/utils/syntax.ts +0 -219
- package/src/public/utils/toolIntentSummarizer.ts +0 -199
- package/src/public/utils/toolStackGrouper.ts +0 -106
- package/src/public/utils/toolTypeColors.ts +0 -45
|
@@ -1,162 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* StatsStrip Component
|
|
3
|
-
*
|
|
4
|
-
* Displays context percentage, model badge, PWD, and identity info.
|
|
5
|
-
* Story MSSCI-12699 - StatsStrip Component
|
|
6
|
-
*
|
|
7
|
-
* Features:
|
|
8
|
-
* - Context percentage with color thresholds (safe < 70%, warning 70-85%, danger >= 85%)
|
|
9
|
-
* - Model badge showing current model name
|
|
10
|
-
* - PWD showing current working directory (folder name)
|
|
11
|
-
* - Identity section showing Jira email and GitHub username
|
|
12
|
-
* - Real-time updates via IPC subscriptions
|
|
13
|
-
*/
|
|
14
|
-
|
|
15
|
-
import React from 'react';
|
|
16
|
-
import { Badge } from '@/components/ui/badge';
|
|
17
|
-
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
|
18
|
-
import { useStatsStrip } from '../hooks/useStatsStrip';
|
|
19
|
-
|
|
20
|
-
/**
|
|
21
|
-
* Extract short model name from full model ID
|
|
22
|
-
* e.g., 'claude-opus-4-5-20251101' -> 'opus'
|
|
23
|
-
* 'claude-3-5-sonnet-20241022' -> 'sonnet'
|
|
24
|
-
*/
|
|
25
|
-
function formatModelName(model: string | null): string {
|
|
26
|
-
if (!model) return '\u2014'; // em dash placeholder
|
|
27
|
-
|
|
28
|
-
// Extract model family name
|
|
29
|
-
const lowerModel = model.toLowerCase();
|
|
30
|
-
if (lowerModel.includes('opus')) return 'opus';
|
|
31
|
-
if (lowerModel.includes('sonnet')) return 'sonnet';
|
|
32
|
-
if (lowerModel.includes('haiku')) return 'haiku';
|
|
33
|
-
|
|
34
|
-
// Fallback: return last segment before date
|
|
35
|
-
const parts = model.split('-');
|
|
36
|
-
if (parts.length >= 2) {
|
|
37
|
-
// Find the part that's not a date (YYYYMMDD)
|
|
38
|
-
for (let i = parts.length - 2; i >= 0; i--) {
|
|
39
|
-
if (!/^\d{8}$/.test(parts[i])) {
|
|
40
|
-
return parts[i];
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
return model;
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
/**
|
|
49
|
-
* Determine context level class based on percentage
|
|
50
|
-
*/
|
|
51
|
-
function getContextLevel(percent: number): string {
|
|
52
|
-
if (percent >= 85) return 'level-danger';
|
|
53
|
-
if (percent >= 70) return 'level-warning';
|
|
54
|
-
return 'level-safe';
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
/**
|
|
58
|
-
* Extract folder name from full path
|
|
59
|
-
*/
|
|
60
|
-
function getFolderName(pwd: string): string {
|
|
61
|
-
if (!pwd) return '';
|
|
62
|
-
const parts = pwd.split('/');
|
|
63
|
-
return parts[parts.length - 1] || parts[parts.length - 2] || pwd;
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
export default function StatsStrip(): React.ReactElement {
|
|
67
|
-
const { context, stats, projectInfo } = useStatsStrip();
|
|
68
|
-
|
|
69
|
-
const percent = context?.percent ?? 0;
|
|
70
|
-
const model = stats?.model ?? null;
|
|
71
|
-
const pwd = projectInfo?.pwd ?? '';
|
|
72
|
-
const jiraEmail = projectInfo?.jiraEmail ?? null;
|
|
73
|
-
const githubUsername = projectInfo?.githubUsername ?? null;
|
|
74
|
-
|
|
75
|
-
const contextLevel = getContextLevel(percent);
|
|
76
|
-
const folderName = getFolderName(pwd);
|
|
77
|
-
|
|
78
|
-
return (
|
|
79
|
-
<TooltipProvider delayDuration={300}>
|
|
80
|
-
<div className="stats-strip" data-testid="stats-strip">
|
|
81
|
-
{/* Left side: identity info */}
|
|
82
|
-
<div className="stats-left" data-testid="stats-left">
|
|
83
|
-
{/* PWD */}
|
|
84
|
-
<Tooltip>
|
|
85
|
-
<TooltipTrigger asChild>
|
|
86
|
-
<span
|
|
87
|
-
className="stats-pwd"
|
|
88
|
-
data-testid="stats-pwd"
|
|
89
|
-
data-full-path={pwd}
|
|
90
|
-
>
|
|
91
|
-
{folderName}
|
|
92
|
-
</span>
|
|
93
|
-
</TooltipTrigger>
|
|
94
|
-
<TooltipContent>{pwd}</TooltipContent>
|
|
95
|
-
</Tooltip>
|
|
96
|
-
|
|
97
|
-
{/* Jira email - only show if configured */}
|
|
98
|
-
{jiraEmail && (
|
|
99
|
-
<Tooltip>
|
|
100
|
-
<TooltipTrigger asChild>
|
|
101
|
-
<span
|
|
102
|
-
className="stats-jira-email"
|
|
103
|
-
data-testid="jira-email"
|
|
104
|
-
>
|
|
105
|
-
{jiraEmail}
|
|
106
|
-
</span>
|
|
107
|
-
</TooltipTrigger>
|
|
108
|
-
<TooltipContent>{`Jira: ${jiraEmail}`}</TooltipContent>
|
|
109
|
-
</Tooltip>
|
|
110
|
-
)}
|
|
111
|
-
|
|
112
|
-
{/* GitHub user - only show if configured */}
|
|
113
|
-
{githubUsername && (
|
|
114
|
-
<Tooltip>
|
|
115
|
-
<TooltipTrigger asChild>
|
|
116
|
-
<span
|
|
117
|
-
className="stats-github-user"
|
|
118
|
-
data-testid="github-user"
|
|
119
|
-
>
|
|
120
|
-
@{githubUsername}
|
|
121
|
-
</span>
|
|
122
|
-
</TooltipTrigger>
|
|
123
|
-
<TooltipContent>{`GitHub: ${githubUsername}`}</TooltipContent>
|
|
124
|
-
</Tooltip>
|
|
125
|
-
)}
|
|
126
|
-
</div>
|
|
127
|
-
|
|
128
|
-
{/* Right side: metrics */}
|
|
129
|
-
<div className="stats-right" data-testid="stats-right">
|
|
130
|
-
{/* Model badge */}
|
|
131
|
-
<Tooltip>
|
|
132
|
-
<TooltipTrigger asChild>
|
|
133
|
-
<Badge
|
|
134
|
-
variant="secondary"
|
|
135
|
-
className="stats-model-badge"
|
|
136
|
-
data-testid="model-badge"
|
|
137
|
-
>
|
|
138
|
-
{formatModelName(model)}
|
|
139
|
-
</Badge>
|
|
140
|
-
</TooltipTrigger>
|
|
141
|
-
{model && <TooltipContent>{model}</TooltipContent>}
|
|
142
|
-
</Tooltip>
|
|
143
|
-
|
|
144
|
-
{/* Context meter */}
|
|
145
|
-
<div
|
|
146
|
-
className={`stats-context-meter ${contextLevel}`}
|
|
147
|
-
data-testid="context-meter"
|
|
148
|
-
>
|
|
149
|
-
<span className="context-percent" data-testid="context-percent">
|
|
150
|
-
{percent}%
|
|
151
|
-
</span>
|
|
152
|
-
<div
|
|
153
|
-
className="context-fill"
|
|
154
|
-
data-testid="context-fill"
|
|
155
|
-
style={{ width: `${percent}%` }}
|
|
156
|
-
/>
|
|
157
|
-
</div>
|
|
158
|
-
</div>
|
|
159
|
-
</div>
|
|
160
|
-
</TooltipProvider>
|
|
161
|
-
);
|
|
162
|
-
}
|
|
@@ -1,77 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* StreamingContent Component
|
|
3
|
-
*
|
|
4
|
-
* Displays progressively streaming text with cursor indicator.
|
|
5
|
-
* Story MSSCI-12698 - MessageView Component with Streaming
|
|
6
|
-
* Story MSSCI-12771 - Accessibility Compliance (screen reader announcements)
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import React, { useEffect, useRef, useState } from 'react';
|
|
10
|
-
import { parseMarkdown } from '../utils/markdown';
|
|
11
|
-
|
|
12
|
-
interface StreamingContentProps {
|
|
13
|
-
content: string;
|
|
14
|
-
isStreaming: boolean;
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
export default function StreamingContent({ content, isStreaming }: StreamingContentProps): React.ReactElement {
|
|
18
|
-
const html = parseMarkdown(content);
|
|
19
|
-
const [statusMessage, setStatusMessage] = useState<string>('');
|
|
20
|
-
const lastAnnouncedLength = useRef(0);
|
|
21
|
-
const announceThrottleRef = useRef<NodeJS.Timeout | null>(null);
|
|
22
|
-
|
|
23
|
-
// Announce streaming state changes
|
|
24
|
-
useEffect(() => {
|
|
25
|
-
if (isStreaming) {
|
|
26
|
-
setStatusMessage('Claude is thinking...');
|
|
27
|
-
} else if (content && lastAnnouncedLength.current > 0) {
|
|
28
|
-
setStatusMessage('Response complete');
|
|
29
|
-
}
|
|
30
|
-
}, [isStreaming, content]);
|
|
31
|
-
|
|
32
|
-
// Throttled content announcements during streaming
|
|
33
|
-
useEffect(() => {
|
|
34
|
-
if (!isStreaming) {
|
|
35
|
-
lastAnnouncedLength.current = content.length;
|
|
36
|
-
return;
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
// Clear any pending announcement
|
|
40
|
-
if (announceThrottleRef.current) {
|
|
41
|
-
clearTimeout(announceThrottleRef.current);
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
// Throttle announcements to every 2 seconds to avoid screen reader overload
|
|
45
|
-
announceThrottleRef.current = setTimeout(() => {
|
|
46
|
-
if (content.length > lastAnnouncedLength.current) {
|
|
47
|
-
lastAnnouncedLength.current = content.length;
|
|
48
|
-
}
|
|
49
|
-
}, 2000);
|
|
50
|
-
|
|
51
|
-
return () => {
|
|
52
|
-
if (announceThrottleRef.current) {
|
|
53
|
-
clearTimeout(announceThrottleRef.current);
|
|
54
|
-
}
|
|
55
|
-
};
|
|
56
|
-
}, [content, isStreaming]);
|
|
57
|
-
|
|
58
|
-
return (
|
|
59
|
-
<>
|
|
60
|
-
{/* Status region for streaming state changes */}
|
|
61
|
-
<div role="status" aria-live="polite" className="visually-hidden">
|
|
62
|
-
{statusMessage}
|
|
63
|
-
</div>
|
|
64
|
-
|
|
65
|
-
{/* Main streaming content */}
|
|
66
|
-
<div
|
|
67
|
-
data-testid="streaming-content"
|
|
68
|
-
className="streaming-content"
|
|
69
|
-
aria-busy={isStreaming}
|
|
70
|
-
aria-atomic="false"
|
|
71
|
-
aria-live="polite"
|
|
72
|
-
>
|
|
73
|
-
<div dangerouslySetInnerHTML={{ __html: html }} />
|
|
74
|
-
</div>
|
|
75
|
-
</>
|
|
76
|
-
);
|
|
77
|
-
}
|
|
@@ -1,180 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* SubagentSpan Component
|
|
3
|
-
*
|
|
4
|
-
* Collapsible container for subagent message groups.
|
|
5
|
-
* Story MSSCI-12698 - MessageView Component with Streaming
|
|
6
|
-
* Story MSSCI-12776 - Theme-Aware Subagent Display Messages
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import React, { useState, useMemo } from 'react';
|
|
10
|
-
import { Badge } from '@/components/ui/badge';
|
|
11
|
-
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
|
12
|
-
import Message from './Message';
|
|
13
|
-
import ToolCallBlock from './ToolCallBlock';
|
|
14
|
-
import { useSubagentHelper } from '../hooks/useSubagentHelper';
|
|
15
|
-
import { generateFriendlyMessage } from '../utils/subagent-display';
|
|
16
|
-
import type { SubagentMessage } from '../types/message';
|
|
17
|
-
|
|
18
|
-
interface SubagentSpanProps {
|
|
19
|
-
type: string;
|
|
20
|
-
name: string;
|
|
21
|
-
messages: SubagentMessage[];
|
|
22
|
-
defaultCollapsed?: boolean;
|
|
23
|
-
onCollapseChange?: (collapsed: boolean) => void;
|
|
24
|
-
// MSSCI-12776: Theme-aware helper display (optional - can be overridden by props for testing)
|
|
25
|
-
helperName?: string | null;
|
|
26
|
-
helperStyle?: string | null;
|
|
27
|
-
friendlyMessage?: string | null;
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
export default function SubagentSpan({
|
|
31
|
-
type,
|
|
32
|
-
name,
|
|
33
|
-
messages,
|
|
34
|
-
defaultCollapsed = true,
|
|
35
|
-
onCollapseChange,
|
|
36
|
-
helperName: propHelperName,
|
|
37
|
-
helperStyle: propHelperStyle,
|
|
38
|
-
friendlyMessage: propFriendlyMessage,
|
|
39
|
-
}: SubagentSpanProps): React.ReactElement {
|
|
40
|
-
const [isCollapsed, setIsCollapsed] = useState(defaultCollapsed);
|
|
41
|
-
|
|
42
|
-
// MSSCI-12776: Fetch themed helper from current persona (AC2)
|
|
43
|
-
const { helper } = useSubagentHelper();
|
|
44
|
-
|
|
45
|
-
// Generate friendly message from subagent context (AC3)
|
|
46
|
-
const generatedFriendlyMessage = useMemo(() => {
|
|
47
|
-
return generateFriendlyMessage({ subagent_type: type, description: name }, { plural: helper?.plural });
|
|
48
|
-
}, [type, name, helper?.plural]);
|
|
49
|
-
|
|
50
|
-
// Use truthy props if provided (for testing), otherwise use hook/generated values (AC4, AC5)
|
|
51
|
-
// Treating null same as undefined - both mean "use fallback"
|
|
52
|
-
const helperName = propHelperName ?? helper?.name;
|
|
53
|
-
const helperStyle = propHelperStyle ?? helper?.style;
|
|
54
|
-
const friendlyMessage = propFriendlyMessage ?? generatedFriendlyMessage;
|
|
55
|
-
|
|
56
|
-
// Determine display values with fallbacks (AC5)
|
|
57
|
-
const displayName = helperName || type;
|
|
58
|
-
const displayMessage = friendlyMessage || name;
|
|
59
|
-
|
|
60
|
-
// Count non-result messages for the collapsed summary
|
|
61
|
-
const messageCount = messages.filter(m => m.type !== 'tool_result').length;
|
|
62
|
-
|
|
63
|
-
// Group tool_use and tool_result by tool_id
|
|
64
|
-
const toolResults = new Map<string, SubagentMessage>();
|
|
65
|
-
messages.forEach(msg => {
|
|
66
|
-
if (msg.type === 'tool_result' && msg.tool_id) {
|
|
67
|
-
toolResults.set(msg.tool_id, msg);
|
|
68
|
-
}
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
const renderMessage = (msg: SubagentMessage, index: number) => {
|
|
72
|
-
if (msg.type === 'tool_use' && msg.tool_name && msg.tool_id) {
|
|
73
|
-
const result = toolResults.get(msg.tool_id);
|
|
74
|
-
return (
|
|
75
|
-
<div key={`tool-wrapper-${msg.tool_id}`} data-testid="message-tool_use" className="message message-tool_use">
|
|
76
|
-
<ToolCallBlock
|
|
77
|
-
toolUse={{
|
|
78
|
-
type: 'tool_use',
|
|
79
|
-
tool_name: msg.tool_name,
|
|
80
|
-
tool_id: msg.tool_id,
|
|
81
|
-
input: msg.input || {},
|
|
82
|
-
timestamp: msg.timestamp,
|
|
83
|
-
}}
|
|
84
|
-
result={result ? {
|
|
85
|
-
type: 'tool_result',
|
|
86
|
-
tool_id: result.tool_id!,
|
|
87
|
-
content: result.content || '',
|
|
88
|
-
timestamp: result.timestamp,
|
|
89
|
-
} : undefined}
|
|
90
|
-
/>
|
|
91
|
-
</div>
|
|
92
|
-
);
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
if (msg.type === 'tool_result') {
|
|
96
|
-
// Skip tool_result - it's rendered with tool_use
|
|
97
|
-
return null;
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
// Subagent prompts (user messages within subagent) — truncated single line
|
|
101
|
-
if (msg.type === 'user') {
|
|
102
|
-
const truncated = (msg.content || '').slice(0, 120).replace(/\n/g, ' ');
|
|
103
|
-
return (
|
|
104
|
-
<div
|
|
105
|
-
key={`subagent-prompt-${index}`}
|
|
106
|
-
data-testid="subagent-prompt"
|
|
107
|
-
className="message message-subagent-prompt"
|
|
108
|
-
>
|
|
109
|
-
<div className="message-content">{truncated}{(msg.content || '').length > 120 ? '...' : ''}</div>
|
|
110
|
-
</div>
|
|
111
|
-
);
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
return (
|
|
115
|
-
<Message
|
|
116
|
-
key={`msg-${index}`}
|
|
117
|
-
message={{
|
|
118
|
-
type: msg.type,
|
|
119
|
-
content: msg.content,
|
|
120
|
-
timestamp: msg.timestamp,
|
|
121
|
-
}}
|
|
122
|
-
/>
|
|
123
|
-
);
|
|
124
|
-
};
|
|
125
|
-
|
|
126
|
-
return (
|
|
127
|
-
<div
|
|
128
|
-
data-testid="subagent-span"
|
|
129
|
-
data-collapsible="true"
|
|
130
|
-
className={`subagent-span subagent-${type} ${isCollapsed ? 'collapsed' : ''}`}
|
|
131
|
-
>
|
|
132
|
-
<div
|
|
133
|
-
data-testid="subagent-span-header"
|
|
134
|
-
className="subagent-header"
|
|
135
|
-
onClick={() => {
|
|
136
|
-
const next = !isCollapsed;
|
|
137
|
-
setIsCollapsed(next);
|
|
138
|
-
onCollapseChange?.(next);
|
|
139
|
-
}}
|
|
140
|
-
>
|
|
141
|
-
<span className="subagent-toggle">{isCollapsed ? '▶' : '▼'}</span>
|
|
142
|
-
|
|
143
|
-
{/* Helper name or fallback to type (AC4, AC5) */}
|
|
144
|
-
<TooltipProvider delayDuration={300}>
|
|
145
|
-
{helperStyle ? (
|
|
146
|
-
<Tooltip>
|
|
147
|
-
<TooltipTrigger asChild>
|
|
148
|
-
<span className="subagent-helper-name">
|
|
149
|
-
{displayName}
|
|
150
|
-
</span>
|
|
151
|
-
</TooltipTrigger>
|
|
152
|
-
<TooltipContent>{helperStyle}</TooltipContent>
|
|
153
|
-
</Tooltip>
|
|
154
|
-
) : (
|
|
155
|
-
<span className="subagent-helper-name">
|
|
156
|
-
{displayName}
|
|
157
|
-
</span>
|
|
158
|
-
)}
|
|
159
|
-
</TooltipProvider>
|
|
160
|
-
|
|
161
|
-
{/* Friendly message or fallback to name (AC4, AC5) */}
|
|
162
|
-
<span className="subagent-friendly-message">
|
|
163
|
-
{displayMessage}
|
|
164
|
-
</span>
|
|
165
|
-
|
|
166
|
-
{/* Type badge for debugging context (AC4) */}
|
|
167
|
-
<Badge variant="outline" data-testid="subagent-type-badge" className="subagent-type-badge">
|
|
168
|
-
{type}
|
|
169
|
-
</Badge>
|
|
170
|
-
|
|
171
|
-
<span className="subagent-count">{messageCount}</span>
|
|
172
|
-
</div>
|
|
173
|
-
{!isCollapsed && (
|
|
174
|
-
<div className="subagent-content">
|
|
175
|
-
{messages.map((msg, index) => renderMessage(msg, index))}
|
|
176
|
-
</div>
|
|
177
|
-
)}
|
|
178
|
-
</div>
|
|
179
|
-
);
|
|
180
|
-
}
|
|
@@ -1,72 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* TandemPortrait Component
|
|
3
|
-
*
|
|
4
|
-
* Renders backseat agent portrait below primary in PersonaHeader.
|
|
5
|
-
* Story: MSSCI-14674 (96-1) - TandemPortrait Component
|
|
6
|
-
* Epic: MSSCI-14673 (Cyclist Tandem UI)
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import React, { useState } from 'react';
|
|
10
|
-
|
|
11
|
-
export interface TandemPortraitProps {
|
|
12
|
-
character: string;
|
|
13
|
-
role: string;
|
|
14
|
-
slug: string;
|
|
15
|
-
theme: string;
|
|
16
|
-
isActive: boolean;
|
|
17
|
-
isThinking: boolean;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
const AGENT_ABBREV: Record<string, string> = {
|
|
21
|
-
pm: 'PM',
|
|
22
|
-
sm: 'SM',
|
|
23
|
-
dev: 'DEV',
|
|
24
|
-
tea: 'TEA',
|
|
25
|
-
reviewer: 'REV',
|
|
26
|
-
architect: 'ARC',
|
|
27
|
-
devops: 'OPS',
|
|
28
|
-
'ux-designer': 'UX',
|
|
29
|
-
'tech-writer': 'TW',
|
|
30
|
-
orchestrator: 'ORC',
|
|
31
|
-
ba: 'BA',
|
|
32
|
-
};
|
|
33
|
-
|
|
34
|
-
export default function TandemPortrait({
|
|
35
|
-
character,
|
|
36
|
-
role,
|
|
37
|
-
slug,
|
|
38
|
-
theme,
|
|
39
|
-
isActive,
|
|
40
|
-
isThinking,
|
|
41
|
-
}: TandemPortraitProps): React.ReactElement | null {
|
|
42
|
-
const [portraitError, setPortraitError] = useState(false);
|
|
43
|
-
|
|
44
|
-
if (!isActive) return null;
|
|
45
|
-
|
|
46
|
-
const abbrev = AGENT_ABBREV[role] || role.toUpperCase();
|
|
47
|
-
|
|
48
|
-
return (
|
|
49
|
-
<div
|
|
50
|
-
className={`persona-tandem-portrait${isThinking ? ' avatar-tandem-thinking' : ''}`}
|
|
51
|
-
data-testid="tandem-portrait"
|
|
52
|
-
role="img"
|
|
53
|
-
aria-label={`${character} (${role}) - observing`}
|
|
54
|
-
tabIndex={-1}
|
|
55
|
-
>
|
|
56
|
-
{!portraitError ? (
|
|
57
|
-
<img
|
|
58
|
-
src={`/portraits/${theme}/medium/${slug}.png`}
|
|
59
|
-
alt=""
|
|
60
|
-
aria-hidden="true"
|
|
61
|
-
className="tandem-portrait-image"
|
|
62
|
-
onError={() => setPortraitError(true)}
|
|
63
|
-
/>
|
|
64
|
-
) : (
|
|
65
|
-
<span className="tandem-portrait-fallback" aria-hidden="true">🤖</span>
|
|
66
|
-
)}
|
|
67
|
-
<span className="tandem-role-badge" data-testid="tandem-role-badge" aria-hidden="true">
|
|
68
|
-
{abbrev}
|
|
69
|
-
</span>
|
|
70
|
-
</div>
|
|
71
|
-
);
|
|
72
|
-
}
|
|
@@ -1,179 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ThemePalette Component Styles
|
|
3
|
-
* Story MSSCI-12768 - Color Palette System
|
|
4
|
-
*
|
|
5
|
-
* The React component uses shadcn Popover + Command primitives for the
|
|
6
|
-
* dropdown, search, keyboard nav, and click-outside handling.
|
|
7
|
-
* These styles cover the trigger button, swatch rendering, and the
|
|
8
|
-
* vanilla JS renderThemePalette() fallback which still uses custom DOM.
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
/* ==========================================================================
|
|
12
|
-
Trigger button (shared between React + vanilla)
|
|
13
|
-
========================================================================== */
|
|
14
|
-
|
|
15
|
-
.theme-palette-button {
|
|
16
|
-
display: flex;
|
|
17
|
-
align-items: center;
|
|
18
|
-
gap: 8px;
|
|
19
|
-
padding: 6px 12px;
|
|
20
|
-
background: var(--bg-secondary);
|
|
21
|
-
border: 1px solid var(--border);
|
|
22
|
-
border-radius: 4px;
|
|
23
|
-
color: var(--text-primary);
|
|
24
|
-
font-size: 13px;
|
|
25
|
-
cursor: pointer;
|
|
26
|
-
transition: border-color 0.15s ease, background-color 0.15s ease;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
.theme-palette-button:hover {
|
|
30
|
-
border-color: var(--border-focus);
|
|
31
|
-
background: var(--bg-tertiary);
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
.theme-palette-button:focus {
|
|
35
|
-
outline: none;
|
|
36
|
-
border-color: var(--accent);
|
|
37
|
-
box-shadow: 0 0 0 2px rgba(79, 70, 229, 0.2);
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
.theme-palette-current {
|
|
41
|
-
display: flex;
|
|
42
|
-
align-items: center;
|
|
43
|
-
gap: 6px;
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
.theme-palette-name {
|
|
47
|
-
font-weight: 500;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
.theme-palette-chevron {
|
|
51
|
-
width: 14px;
|
|
52
|
-
height: 14px;
|
|
53
|
-
opacity: 0.6;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
/* ==========================================================================
|
|
57
|
-
Popover content (React component)
|
|
58
|
-
========================================================================== */
|
|
59
|
-
|
|
60
|
-
.theme-palette-popover {
|
|
61
|
-
width: 260px;
|
|
62
|
-
background: var(--bg-secondary);
|
|
63
|
-
border: 1px solid var(--border);
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
/* ==========================================================================
|
|
67
|
-
Swatches (shared between React + vanilla)
|
|
68
|
-
========================================================================== */
|
|
69
|
-
|
|
70
|
-
.preset-swatches {
|
|
71
|
-
display: flex;
|
|
72
|
-
gap: 2px;
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
.preset-swatch {
|
|
76
|
-
width: 14px;
|
|
77
|
-
height: 14px;
|
|
78
|
-
border-radius: 3px;
|
|
79
|
-
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
80
|
-
flex-shrink: 0;
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
.preset-name {
|
|
84
|
-
flex: 1;
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
.preset-check-icon {
|
|
88
|
-
width: 16px;
|
|
89
|
-
height: 16px;
|
|
90
|
-
flex-shrink: 0;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
/* ==========================================================================
|
|
94
|
-
Vanilla JS fallback (renderThemePalette)
|
|
95
|
-
These styles are only used by the imperative DOM-based renderer.
|
|
96
|
-
========================================================================== */
|
|
97
|
-
|
|
98
|
-
.theme-palette {
|
|
99
|
-
position: relative;
|
|
100
|
-
display: inline-block;
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
.theme-palette-menu {
|
|
104
|
-
position: absolute;
|
|
105
|
-
top: 100%;
|
|
106
|
-
left: 0;
|
|
107
|
-
z-index: 100;
|
|
108
|
-
min-width: 220px;
|
|
109
|
-
max-height: 360px;
|
|
110
|
-
overflow-y: auto;
|
|
111
|
-
margin-top: 4px;
|
|
112
|
-
padding: 4px;
|
|
113
|
-
background: var(--bg-secondary);
|
|
114
|
-
border: 1px solid var(--border);
|
|
115
|
-
border-radius: 6px;
|
|
116
|
-
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
|
|
117
|
-
opacity: 0;
|
|
118
|
-
visibility: hidden;
|
|
119
|
-
transform: translateY(-8px);
|
|
120
|
-
transition: opacity 0.15s ease, transform 0.15s ease, visibility 0.15s ease;
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
.theme-palette-menu.open {
|
|
124
|
-
opacity: 1;
|
|
125
|
-
visibility: visible;
|
|
126
|
-
transform: translateY(0);
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
.theme-palette-option {
|
|
130
|
-
display: flex;
|
|
131
|
-
align-items: center;
|
|
132
|
-
gap: 10px;
|
|
133
|
-
width: 100%;
|
|
134
|
-
padding: 8px 10px;
|
|
135
|
-
background: transparent;
|
|
136
|
-
border: none;
|
|
137
|
-
border-radius: 4px;
|
|
138
|
-
color: var(--text-primary);
|
|
139
|
-
font-size: 13px;
|
|
140
|
-
text-align: left;
|
|
141
|
-
cursor: pointer;
|
|
142
|
-
transition: background-color 0.1s ease;
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
.theme-palette-option:hover,
|
|
146
|
-
.theme-palette-option.focused {
|
|
147
|
-
background: var(--bg-tertiary);
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
.theme-palette-option.active {
|
|
151
|
-
background: rgba(79, 70, 229, 0.15);
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
.theme-palette-option:focus {
|
|
155
|
-
outline: none;
|
|
156
|
-
background: var(--bg-tertiary);
|
|
157
|
-
box-shadow: inset 0 0 0 1px var(--accent);
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
.preset-check {
|
|
161
|
-
color: var(--accent);
|
|
162
|
-
font-size: 12px;
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
.preset-group-header {
|
|
166
|
-
padding: 6px 10px 4px;
|
|
167
|
-
font-size: 10px;
|
|
168
|
-
font-weight: 600;
|
|
169
|
-
text-transform: uppercase;
|
|
170
|
-
letter-spacing: 0.05em;
|
|
171
|
-
color: var(--text-muted);
|
|
172
|
-
user-select: none;
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
.preset-group-header:not(:first-child) {
|
|
176
|
-
margin-top: 4px;
|
|
177
|
-
border-top: 1px solid var(--border);
|
|
178
|
-
padding-top: 8px;
|
|
179
|
-
}
|