@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,209 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ToolStack Component
|
|
3
|
-
*
|
|
4
|
-
* Collapsible container for consecutive tool use groups.
|
|
5
|
-
* Story MSSCI-13400 - Tool use stack between messages
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import React, { useState, useCallback, useEffect, useRef, useMemo } from 'react';
|
|
9
|
-
import { Badge } from '@/components/ui/badge';
|
|
10
|
-
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
|
11
|
-
import ToolCallBlock, { getToolBadgeLabel } from './ToolCallBlock';
|
|
12
|
-
import type { ToolStackData } from '../utils/toolStackGrouper';
|
|
13
|
-
import { generateToolIntentSummary } from '../utils/toolIntentSummarizer';
|
|
14
|
-
import { formatDuration } from '../utils/formatDuration';
|
|
15
|
-
|
|
16
|
-
interface ToolResultMessage {
|
|
17
|
-
type: 'tool_result';
|
|
18
|
-
tool_id: string;
|
|
19
|
-
content: string;
|
|
20
|
-
timestamp: number;
|
|
21
|
-
durationMs?: number;
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
interface ToolStackProps {
|
|
25
|
-
stack: ToolStackData;
|
|
26
|
-
toolResults: Map<string, ToolResultMessage>;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
export default function ToolStack({ stack, toolResults }: ToolStackProps): React.ReactElement {
|
|
30
|
-
// AC4: Expand automatically when stack has active (pending) tool
|
|
31
|
-
const [isCollapsed, setIsCollapsed] = useState(!stack.isActive);
|
|
32
|
-
|
|
33
|
-
// Track previous isActive to detect transitions
|
|
34
|
-
const prevIsActiveRef = useRef(stack.isActive);
|
|
35
|
-
|
|
36
|
-
// When stack transitions from active to inactive, keep it expanded
|
|
37
|
-
// This ensures completed tools remain visible for user review
|
|
38
|
-
useEffect(() => {
|
|
39
|
-
if (prevIsActiveRef.current && !stack.isActive) {
|
|
40
|
-
// Stack just completed - keep expanded for review
|
|
41
|
-
setIsCollapsed(false);
|
|
42
|
-
}
|
|
43
|
-
prevIsActiveRef.current = stack.isActive;
|
|
44
|
-
}, [stack.isActive]);
|
|
45
|
-
|
|
46
|
-
// Count display - just the number
|
|
47
|
-
const countText = `${stack.count}`;
|
|
48
|
-
|
|
49
|
-
// Active tool summary - show what's happening RIGHT NOW
|
|
50
|
-
const activeSummary = useMemo(() => {
|
|
51
|
-
if (stack.isActive && stack.tools.length > 0) {
|
|
52
|
-
// Show the most recent (last) tool's summary
|
|
53
|
-
const lastTool = stack.tools[stack.tools.length - 1];
|
|
54
|
-
return generateToolIntentSummary(lastTool.tool_name, lastTool.input);
|
|
55
|
-
}
|
|
56
|
-
return null;
|
|
57
|
-
}, [stack.tools, stack.isActive]);
|
|
58
|
-
|
|
59
|
-
// Generate summary of tool intents for collapsed view
|
|
60
|
-
const collapsedSummary = useMemo(() => {
|
|
61
|
-
// Show last tool's summary as the main description
|
|
62
|
-
if (stack.tools.length === 0) return '';
|
|
63
|
-
const lastTool = stack.tools[stack.tools.length - 1];
|
|
64
|
-
return generateToolIntentSummary(lastTool.tool_name, lastTool.input);
|
|
65
|
-
}, [stack.tools]);
|
|
66
|
-
|
|
67
|
-
// Compute tool type counts for mini badges
|
|
68
|
-
const toolTypeCounts = useMemo(() => {
|
|
69
|
-
const counts = new Map<string, number>();
|
|
70
|
-
stack.tools.forEach(tool => {
|
|
71
|
-
const count = counts.get(tool.tool_name) || 0;
|
|
72
|
-
counts.set(tool.tool_name, count + 1);
|
|
73
|
-
});
|
|
74
|
-
return counts;
|
|
75
|
-
}, [stack.tools]);
|
|
76
|
-
|
|
77
|
-
// Total duration across all tools in stack
|
|
78
|
-
const totalDuration = useMemo(() => {
|
|
79
|
-
return stack.tools.reduce((sum, tool) => {
|
|
80
|
-
const result = toolResults.get(tool.tool_id);
|
|
81
|
-
return sum + (result?.durationMs || 0);
|
|
82
|
-
}, 0);
|
|
83
|
-
}, [stack.tools, toolResults]);
|
|
84
|
-
|
|
85
|
-
// AC4: Determine which tool is current (pending)
|
|
86
|
-
const getToolClass = (toolIndex: number): string => {
|
|
87
|
-
const tool = stack.tools[toolIndex];
|
|
88
|
-
const result = toolResults.get(tool.tool_id);
|
|
89
|
-
|
|
90
|
-
// If stack is active, last tool without result is current
|
|
91
|
-
if (stack.isActive) {
|
|
92
|
-
const isLastTool = toolIndex === stack.tools.length - 1;
|
|
93
|
-
const hasPendingResult = !result;
|
|
94
|
-
|
|
95
|
-
if (isLastTool && hasPendingResult) {
|
|
96
|
-
return 'tool-current';
|
|
97
|
-
}
|
|
98
|
-
return 'tool-historical';
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
// If stack is not active, all tools are historical
|
|
102
|
-
return 'tool-historical';
|
|
103
|
-
};
|
|
104
|
-
|
|
105
|
-
// Toggle collapse state
|
|
106
|
-
const handleToggle = useCallback(() => {
|
|
107
|
-
setIsCollapsed(prev => !prev);
|
|
108
|
-
}, []);
|
|
109
|
-
|
|
110
|
-
// Keyboard navigation
|
|
111
|
-
const handleKeyDown = useCallback((e: React.KeyboardEvent) => {
|
|
112
|
-
if (e.key === 'Enter' || e.key === ' ') {
|
|
113
|
-
e.preventDefault();
|
|
114
|
-
handleToggle();
|
|
115
|
-
}
|
|
116
|
-
}, [handleToggle]);
|
|
117
|
-
|
|
118
|
-
// AC4: When active and user tries to collapse, still show current tool
|
|
119
|
-
const shouldShowTools = !isCollapsed || stack.isActive;
|
|
120
|
-
|
|
121
|
-
return (
|
|
122
|
-
<div
|
|
123
|
-
data-testid="tool-stack"
|
|
124
|
-
data-collapsible="true"
|
|
125
|
-
className={`tool-stack ${isCollapsed && !stack.isActive ? 'collapsed' : ''}`}
|
|
126
|
-
>
|
|
127
|
-
<div
|
|
128
|
-
data-testid="tool-stack-header"
|
|
129
|
-
className="tool-stack-header"
|
|
130
|
-
onClick={handleToggle}
|
|
131
|
-
onKeyDown={handleKeyDown}
|
|
132
|
-
role="button"
|
|
133
|
-
tabIndex={0}
|
|
134
|
-
aria-expanded={!isCollapsed || stack.isActive}
|
|
135
|
-
aria-label={`Tool stack with ${countText}: ${collapsedSummary}`}
|
|
136
|
-
>
|
|
137
|
-
<span className="tool-stack-toggle">
|
|
138
|
-
{isCollapsed && !stack.isActive ? '▶' : '▼'}
|
|
139
|
-
</span>
|
|
140
|
-
<span
|
|
141
|
-
data-testid="tool-stack-count"
|
|
142
|
-
className="tool-stack-count-badge"
|
|
143
|
-
>
|
|
144
|
-
{countText}
|
|
145
|
-
</span>
|
|
146
|
-
{stack.isActive && activeSummary ? (
|
|
147
|
-
<span className="tool-stack-active-summary">
|
|
148
|
-
{activeSummary}
|
|
149
|
-
</span>
|
|
150
|
-
) : (
|
|
151
|
-
<>
|
|
152
|
-
<span className="tool-stack-badges">
|
|
153
|
-
<TooltipProvider delayDuration={300}>
|
|
154
|
-
{Array.from(toolTypeCounts.entries()).slice(0, 4).map(([type, count]) => (
|
|
155
|
-
<Tooltip key={type}>
|
|
156
|
-
<TooltipTrigger asChild>
|
|
157
|
-
<Badge
|
|
158
|
-
variant="outline"
|
|
159
|
-
className={`tool-mini-badge badge-${type.toLowerCase()}`}
|
|
160
|
-
>
|
|
161
|
-
{getToolBadgeLabel(type)}
|
|
162
|
-
</Badge>
|
|
163
|
-
</TooltipTrigger>
|
|
164
|
-
<TooltipContent>{`${count} ${type} call${count > 1 ? 's' : ''}`}</TooltipContent>
|
|
165
|
-
</Tooltip>
|
|
166
|
-
))}
|
|
167
|
-
</TooltipProvider>
|
|
168
|
-
</span>
|
|
169
|
-
<span className="tool-stack-summary">
|
|
170
|
-
{collapsedSummary}
|
|
171
|
-
</span>
|
|
172
|
-
</>
|
|
173
|
-
)}
|
|
174
|
-
<span className="tool-stack-duration">
|
|
175
|
-
{totalDuration > 0 ? formatDuration(totalDuration) : ''}
|
|
176
|
-
</span>
|
|
177
|
-
</div>
|
|
178
|
-
|
|
179
|
-
{shouldShowTools && (
|
|
180
|
-
<div className="tool-stack-content">
|
|
181
|
-
{stack.tools.map((tool, index) => {
|
|
182
|
-
const result = toolResults.get(tool.tool_id);
|
|
183
|
-
const toolClass = getToolClass(index);
|
|
184
|
-
|
|
185
|
-
return (
|
|
186
|
-
<ToolCallBlock
|
|
187
|
-
key={tool.tool_id || `tool-${index}`}
|
|
188
|
-
toolUse={{
|
|
189
|
-
type: 'tool_use',
|
|
190
|
-
tool_name: tool.tool_name,
|
|
191
|
-
tool_id: tool.tool_id,
|
|
192
|
-
input: tool.input,
|
|
193
|
-
timestamp: tool.timestamp,
|
|
194
|
-
}}
|
|
195
|
-
result={result ? {
|
|
196
|
-
type: 'tool_result',
|
|
197
|
-
tool_id: result.tool_id,
|
|
198
|
-
content: result.content,
|
|
199
|
-
timestamp: result.timestamp,
|
|
200
|
-
} : undefined}
|
|
201
|
-
className={toolClass}
|
|
202
|
-
/>
|
|
203
|
-
);
|
|
204
|
-
})}
|
|
205
|
-
</div>
|
|
206
|
-
)}
|
|
207
|
-
</div>
|
|
208
|
-
);
|
|
209
|
-
}
|
|
@@ -1,57 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* ToolStatus Component
|
|
3
|
-
*
|
|
4
|
-
* Displays status indicator for tool execution state.
|
|
5
|
-
* Story MSSCI-13402 - Tool use visual design polish
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import React from 'react';
|
|
9
|
-
|
|
10
|
-
export type ToolStatusType = 'pending' | 'success' | 'error';
|
|
11
|
-
|
|
12
|
-
export interface ToolStatusProps {
|
|
13
|
-
status: ToolStatusType;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Status indicator component showing pending/success/error state
|
|
18
|
-
*/
|
|
19
|
-
export function ToolStatus({ status }: ToolStatusProps): React.ReactElement {
|
|
20
|
-
const getAriaLabel = (): string => {
|
|
21
|
-
switch (status) {
|
|
22
|
-
case 'pending':
|
|
23
|
-
return 'Tool running';
|
|
24
|
-
case 'success':
|
|
25
|
-
return 'Tool completed successfully';
|
|
26
|
-
case 'error':
|
|
27
|
-
return 'Tool failed with error';
|
|
28
|
-
default:
|
|
29
|
-
return 'Tool status unknown';
|
|
30
|
-
}
|
|
31
|
-
};
|
|
32
|
-
|
|
33
|
-
const getContent = (): React.ReactNode => {
|
|
34
|
-
switch (status) {
|
|
35
|
-
case 'pending':
|
|
36
|
-
return <span className="spinner" data-loading="true" />;
|
|
37
|
-
case 'success':
|
|
38
|
-
return '✓';
|
|
39
|
-
case 'error':
|
|
40
|
-
return '✗';
|
|
41
|
-
default:
|
|
42
|
-
return '?';
|
|
43
|
-
}
|
|
44
|
-
};
|
|
45
|
-
|
|
46
|
-
return (
|
|
47
|
-
<span
|
|
48
|
-
data-testid="tool-status-indicator"
|
|
49
|
-
className={`tool-status-indicator status-${status}`}
|
|
50
|
-
aria-label={getAriaLabel()}
|
|
51
|
-
>
|
|
52
|
-
{getContent()}
|
|
53
|
-
</span>
|
|
54
|
-
);
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
export default ToolStatus;
|
|
@@ -1,169 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* CodeMarkersDialog — Story 80-3 (MSSCI-14456)
|
|
3
|
-
*
|
|
4
|
-
* Dialog displaying code markers (TODO, FIXME, HACK, XXX) with tabs,
|
|
5
|
-
* sortable table, staleness badges, and summary stats.
|
|
6
|
-
*/
|
|
7
|
-
import React, { useState, useMemo, useEffect } from 'react';
|
|
8
|
-
import { Badge } from '@/components/ui/badge';
|
|
9
|
-
import { Skeleton } from '@/components/ui/skeleton';
|
|
10
|
-
import { ToolDialog } from './ToolDialog';
|
|
11
|
-
import { useCodeMarkers } from '../../hooks/useCodeMarkers';
|
|
12
|
-
|
|
13
|
-
export interface CodeMarkersDialogProps {
|
|
14
|
-
open: boolean;
|
|
15
|
-
onOpenChange: (open: boolean) => void;
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
type TabId = 'all' | 'stale' | 'deprecated';
|
|
19
|
-
type SortField = 'marker_type' | 'path' | 'line' | 'author' | 'age_days';
|
|
20
|
-
type SortDirection = 'asc' | 'desc';
|
|
21
|
-
|
|
22
|
-
export function CodeMarkersDialog({ open, onOpenChange }: CodeMarkersDialogProps): React.ReactElement {
|
|
23
|
-
const { data, isLoading, error, refresh } = useCodeMarkers({ days: 90, repo: 'pennyfarthing' });
|
|
24
|
-
|
|
25
|
-
useEffect(() => {
|
|
26
|
-
if (open) refresh();
|
|
27
|
-
}, [open, refresh]);
|
|
28
|
-
|
|
29
|
-
const [activeTab, setActiveTab] = useState<TabId>('all');
|
|
30
|
-
const [sortField, setSortField] = useState<SortField>('age_days');
|
|
31
|
-
const [sortDirection, setSortDirection] = useState<SortDirection>('desc');
|
|
32
|
-
|
|
33
|
-
const filteredMarkers = useMemo(() => {
|
|
34
|
-
if (!data?.markers) return [];
|
|
35
|
-
switch (activeTab) {
|
|
36
|
-
case 'stale':
|
|
37
|
-
return data.markers.filter((m) => m.is_stale);
|
|
38
|
-
case 'deprecated':
|
|
39
|
-
return [];
|
|
40
|
-
default:
|
|
41
|
-
return data.markers;
|
|
42
|
-
}
|
|
43
|
-
}, [data, activeTab]);
|
|
44
|
-
|
|
45
|
-
const sortedMarkers = useMemo(() => {
|
|
46
|
-
const sorted = [...filteredMarkers];
|
|
47
|
-
sorted.sort((a, b) => {
|
|
48
|
-
const aVal = a[sortField];
|
|
49
|
-
const bVal = b[sortField];
|
|
50
|
-
if (typeof aVal === 'string' && typeof bVal === 'string') {
|
|
51
|
-
return sortDirection === 'asc' ? aVal.localeCompare(bVal) : bVal.localeCompare(aVal);
|
|
52
|
-
}
|
|
53
|
-
const aNum = Number(aVal);
|
|
54
|
-
const bNum = Number(bVal);
|
|
55
|
-
return sortDirection === 'asc' ? aNum - bNum : bNum - aNum;
|
|
56
|
-
});
|
|
57
|
-
return sorted;
|
|
58
|
-
}, [filteredMarkers, sortField, sortDirection]);
|
|
59
|
-
|
|
60
|
-
const handleSort = (field: SortField) => {
|
|
61
|
-
if (sortField === field) {
|
|
62
|
-
setSortDirection((d) => (d === 'asc' ? 'desc' : 'asc'));
|
|
63
|
-
} else {
|
|
64
|
-
setSortField(field);
|
|
65
|
-
setSortDirection('desc');
|
|
66
|
-
}
|
|
67
|
-
};
|
|
68
|
-
|
|
69
|
-
const sortArrow = (field: SortField) => {
|
|
70
|
-
if (sortField !== field) return '';
|
|
71
|
-
return sortDirection === 'desc' ? ' v' : ' ^';
|
|
72
|
-
};
|
|
73
|
-
|
|
74
|
-
return (
|
|
75
|
-
<ToolDialog
|
|
76
|
-
open={open}
|
|
77
|
-
onOpenChange={onOpenChange}
|
|
78
|
-
title="Code Markers"
|
|
79
|
-
description="TODO, FIXME, HACK, and XXX markers across the codebase"
|
|
80
|
-
>
|
|
81
|
-
{isLoading && (
|
|
82
|
-
<div className="space-y-2">
|
|
83
|
-
<Skeleton className="h-4 w-full" />
|
|
84
|
-
<Skeleton className="h-4 w-3/4" />
|
|
85
|
-
<Skeleton className="h-4 w-1/2" />
|
|
86
|
-
</div>
|
|
87
|
-
)}
|
|
88
|
-
|
|
89
|
-
{error && (
|
|
90
|
-
<div className="p-4 rounded border border-[var(--status-error)]/20 bg-[var(--status-error)]/5 text-[var(--status-error)] text-sm">
|
|
91
|
-
{error.message}
|
|
92
|
-
</div>
|
|
93
|
-
)}
|
|
94
|
-
|
|
95
|
-
{!isLoading && !error && data && (
|
|
96
|
-
<>
|
|
97
|
-
{/* Summary stats */}
|
|
98
|
-
<div className="flex gap-4 text-xs text-[var(--text-muted)]" data-testid="code-markers-summary">
|
|
99
|
-
<span data-testid="summary-total">Total: <span className="tabular-nums font-mono">{data.summary.total_markers}</span></span>
|
|
100
|
-
<span data-testid="summary-stale">Stale: <span className="tabular-nums font-mono">{data.summary.stale_markers}</span></span>
|
|
101
|
-
{Object.entries(data.summary.by_type).map(([type, count]) => (
|
|
102
|
-
<span key={type} data-testid={`summary-type-${type.toLowerCase()}`}>{type}: <span className="tabular-nums font-mono">{count}</span></span>
|
|
103
|
-
))}
|
|
104
|
-
</div>
|
|
105
|
-
|
|
106
|
-
{/* Tabs */}
|
|
107
|
-
<div role="tablist" className="flex gap-4 border-b border-[var(--border)] mb-3">
|
|
108
|
-
{(['all', 'stale', 'deprecated'] as const).map((tab) => (
|
|
109
|
-
<button
|
|
110
|
-
key={tab}
|
|
111
|
-
role="tab"
|
|
112
|
-
aria-selected={activeTab === tab}
|
|
113
|
-
onClick={() => setActiveTab(tab)}
|
|
114
|
-
className={`pb-2 text-sm capitalize ${activeTab === tab ? 'border-b-2 border-[var(--accent)] text-[var(--text-primary)] font-medium' : 'text-[var(--text-muted)]'}`}
|
|
115
|
-
>
|
|
116
|
-
{tab}
|
|
117
|
-
</button>
|
|
118
|
-
))}
|
|
119
|
-
</div>
|
|
120
|
-
|
|
121
|
-
{/* Table or empty state */}
|
|
122
|
-
{sortedMarkers.length === 0 ? (
|
|
123
|
-
<div className="text-center py-12 text-[var(--text-muted)]">No markers found</div>
|
|
124
|
-
) : (
|
|
125
|
-
<table role="table" className="w-full text-sm">
|
|
126
|
-
<thead>
|
|
127
|
-
<tr className="border-b border-[var(--border)]">
|
|
128
|
-
<th className="cursor-pointer select-none text-left text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2" onClick={() => handleSort('marker_type')}>
|
|
129
|
-
Type{sortArrow('marker_type')}
|
|
130
|
-
</th>
|
|
131
|
-
<th className="cursor-pointer select-none text-left text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2" onClick={() => handleSort('path')}>
|
|
132
|
-
File{sortArrow('path')}
|
|
133
|
-
</th>
|
|
134
|
-
<th className="cursor-pointer select-none text-right text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2" onClick={() => handleSort('line')}>
|
|
135
|
-
Line{sortArrow('line')}
|
|
136
|
-
</th>
|
|
137
|
-
<th className="text-left text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2">Text</th>
|
|
138
|
-
<th className="cursor-pointer select-none text-left text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2" onClick={() => handleSort('author')}>
|
|
139
|
-
Author{sortArrow('author')}
|
|
140
|
-
</th>
|
|
141
|
-
<th className="cursor-pointer select-none text-right text-xs font-medium uppercase tracking-wider text-[var(--text-muted)] pb-2" onClick={() => handleSort('age_days')}>
|
|
142
|
-
Age{sortArrow('age_days')}
|
|
143
|
-
</th>
|
|
144
|
-
</tr>
|
|
145
|
-
</thead>
|
|
146
|
-
<tbody>
|
|
147
|
-
{sortedMarkers.map((marker, i) => (
|
|
148
|
-
<tr key={`${marker.path}:${marker.line}:${i}`} className="text-[var(--text-primary)]">
|
|
149
|
-
<td className="py-1.5">
|
|
150
|
-
<Badge variant={marker.is_stale ? 'destructive' : 'secondary'}>
|
|
151
|
-
{marker.marker_type}
|
|
152
|
-
</Badge>
|
|
153
|
-
</td>
|
|
154
|
-
<td className="py-1.5 font-mono text-xs">{marker.path}</td>
|
|
155
|
-
<td className="text-right py-1.5 tabular-nums font-mono">{marker.line}</td>
|
|
156
|
-
<td className="py-1.5 truncate max-w-xs">{marker.text}</td>
|
|
157
|
-
<td className="py-1.5">{marker.author}</td>
|
|
158
|
-
<td className="text-right py-1.5 tabular-nums font-mono">{marker.age_days}d</td>
|
|
159
|
-
</tr>
|
|
160
|
-
))}
|
|
161
|
-
</tbody>
|
|
162
|
-
</table>
|
|
163
|
-
)}
|
|
164
|
-
</>
|
|
165
|
-
)}
|
|
166
|
-
|
|
167
|
-
</ToolDialog>
|
|
168
|
-
);
|
|
169
|
-
}
|
|
@@ -1,163 +0,0 @@
|
|
|
1
|
-
import React, { useState, useMemo, useCallback, useEffect } from 'react';
|
|
2
|
-
import { Badge } from '@/components/ui/badge';
|
|
3
|
-
import { Button } from '@/components/ui/button';
|
|
4
|
-
import { Skeleton } from '@/components/ui/skeleton';
|
|
5
|
-
import { ToolDialog } from './ToolDialog';
|
|
6
|
-
import { useComplexity, FileComplexity } from '../../hooks/useComplexity';
|
|
7
|
-
|
|
8
|
-
export interface ComplexityDialogProps {
|
|
9
|
-
open: boolean;
|
|
10
|
-
onOpenChange: (open: boolean) => void;
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
type SortField = 'avg_cyclomatic_complexity' | 'max_nesting_depth' | 'longest_function' | 'total_lines' | 'function_count' | 'path';
|
|
14
|
-
type SortDirection = 'asc' | 'desc';
|
|
15
|
-
|
|
16
|
-
function SortableHeader({
|
|
17
|
-
label,
|
|
18
|
-
field,
|
|
19
|
-
currentSort,
|
|
20
|
-
currentDirection,
|
|
21
|
-
onSort,
|
|
22
|
-
align = 'right',
|
|
23
|
-
}: {
|
|
24
|
-
label: string;
|
|
25
|
-
field: SortField;
|
|
26
|
-
currentSort: SortField;
|
|
27
|
-
currentDirection: SortDirection;
|
|
28
|
-
onSort: (field: SortField) => void;
|
|
29
|
-
align?: 'left' | 'right';
|
|
30
|
-
}) {
|
|
31
|
-
const isActive = currentSort === field;
|
|
32
|
-
const arrow = isActive ? (currentDirection === 'desc' ? ' v' : ' ^') : '';
|
|
33
|
-
|
|
34
|
-
return (
|
|
35
|
-
<th
|
|
36
|
-
className={`text-${align} ${isActive ? 'active' : ''}`}
|
|
37
|
-
onClick={() => onSort(field)}
|
|
38
|
-
role="columnheader"
|
|
39
|
-
aria-sort={isActive ? (currentDirection === 'desc' ? 'descending' : 'ascending') : 'none'}
|
|
40
|
-
style={{ cursor: 'pointer', userSelect: 'none', padding: '4px 8px' }}
|
|
41
|
-
>
|
|
42
|
-
{label}{arrow}
|
|
43
|
-
</th>
|
|
44
|
-
);
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
export function ComplexityDialog({ open, onOpenChange }: ComplexityDialogProps): React.ReactElement {
|
|
48
|
-
const [sortField, setSortField] = useState<SortField>('avg_cyclomatic_complexity');
|
|
49
|
-
const [sortDirection, setSortDirection] = useState<SortDirection>('desc');
|
|
50
|
-
|
|
51
|
-
const { data, isLoading, error, refresh } = useComplexity({});
|
|
52
|
-
|
|
53
|
-
useEffect(() => {
|
|
54
|
-
if (open) refresh();
|
|
55
|
-
}, [open, refresh]);
|
|
56
|
-
|
|
57
|
-
const handleSort = useCallback((field: SortField) => {
|
|
58
|
-
setSortField((prev) => {
|
|
59
|
-
if (prev === field) {
|
|
60
|
-
setSortDirection((d) => (d === 'desc' ? 'asc' : 'desc'));
|
|
61
|
-
return prev;
|
|
62
|
-
}
|
|
63
|
-
setSortDirection('desc');
|
|
64
|
-
return field;
|
|
65
|
-
});
|
|
66
|
-
}, []);
|
|
67
|
-
|
|
68
|
-
const sorted = useMemo(() => {
|
|
69
|
-
if (!data?.files) return [];
|
|
70
|
-
const items = [...data.files];
|
|
71
|
-
items.sort((a, b) => {
|
|
72
|
-
const aVal = a[sortField as keyof FileComplexity];
|
|
73
|
-
const bVal = b[sortField as keyof FileComplexity];
|
|
74
|
-
if (typeof aVal === 'number' && typeof bVal === 'number') {
|
|
75
|
-
return sortDirection === 'desc' ? bVal - aVal : aVal - bVal;
|
|
76
|
-
}
|
|
77
|
-
return sortDirection === 'desc'
|
|
78
|
-
? String(bVal).localeCompare(String(aVal))
|
|
79
|
-
: String(aVal).localeCompare(String(bVal));
|
|
80
|
-
});
|
|
81
|
-
return items;
|
|
82
|
-
}, [data, sortField, sortDirection]);
|
|
83
|
-
|
|
84
|
-
const renderContent = () => {
|
|
85
|
-
if (isLoading) {
|
|
86
|
-
return (
|
|
87
|
-
<div className="complexity-panel loading" data-testid="complexity-panel">
|
|
88
|
-
<div className="space-y-3 p-2">
|
|
89
|
-
<Skeleton className="h-4 w-40" />
|
|
90
|
-
<Skeleton className="h-3 w-full" />
|
|
91
|
-
<Skeleton className="h-3 w-full" />
|
|
92
|
-
<Skeleton className="h-3 w-3/4" />
|
|
93
|
-
</div>
|
|
94
|
-
</div>
|
|
95
|
-
);
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
if (error) {
|
|
99
|
-
return (
|
|
100
|
-
<div className="complexity-panel error" data-testid="complexity-panel">
|
|
101
|
-
<div className="error-message">Error: {error.message}</div>
|
|
102
|
-
<Button variant="outline" size="sm" onClick={refresh}>Retry</Button>
|
|
103
|
-
</div>
|
|
104
|
-
);
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
if (!data) {
|
|
108
|
-
return (
|
|
109
|
-
<div className="complexity-panel" data-testid="complexity-panel">
|
|
110
|
-
<p>Click <strong>Analyze</strong> to run complexity analysis</p>
|
|
111
|
-
</div>
|
|
112
|
-
);
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
return (
|
|
116
|
-
<div className="complexity-panel" data-testid="complexity-panel">
|
|
117
|
-
<div className="complexity-summary" style={{ marginBottom: '8px' }}>
|
|
118
|
-
<span>{data.file_count} files analyzed</span>
|
|
119
|
-
</div>
|
|
120
|
-
|
|
121
|
-
<table className="complexity-table" role="table" style={{ width: '100%', borderCollapse: 'collapse' }}>
|
|
122
|
-
<thead>
|
|
123
|
-
<tr>
|
|
124
|
-
<SortableHeader label="Complexity" field="avg_cyclomatic_complexity" currentSort={sortField} currentDirection={sortDirection} onSort={handleSort} />
|
|
125
|
-
<SortableHeader label="Nesting" field="max_nesting_depth" currentSort={sortField} currentDirection={sortDirection} onSort={handleSort} />
|
|
126
|
-
<SortableHeader label="Longest Fn" field="longest_function" currentSort={sortField} currentDirection={sortDirection} onSort={handleSort} />
|
|
127
|
-
<SortableHeader label="Lines" field="total_lines" currentSort={sortField} currentDirection={sortDirection} onSort={handleSort} />
|
|
128
|
-
<SortableHeader label="Functions" field="function_count" currentSort={sortField} currentDirection={sortDirection} onSort={handleSort} />
|
|
129
|
-
<SortableHeader label="File" field="path" currentSort={sortField} currentDirection={sortDirection} onSort={handleSort} align="left" />
|
|
130
|
-
</tr>
|
|
131
|
-
</thead>
|
|
132
|
-
<tbody>
|
|
133
|
-
{sorted.map((f) => (
|
|
134
|
-
<tr key={f.path}>
|
|
135
|
-
<td className="text-right" style={{ padding: '4px 8px' }}>
|
|
136
|
-
<Badge variant={f.avg_cyclomatic_complexity >= 7 ? 'destructive' : f.avg_cyclomatic_complexity >= 4 ? 'outline' : 'secondary'}>
|
|
137
|
-
{Number(f.avg_cyclomatic_complexity).toFixed(1)}
|
|
138
|
-
</Badge>
|
|
139
|
-
</td>
|
|
140
|
-
<td className="text-right" style={{ padding: '4px 8px' }}>{f.max_nesting_depth}</td>
|
|
141
|
-
<td className="text-right" style={{ padding: '4px 8px' }}>{f.longest_function}</td>
|
|
142
|
-
<td className="text-right" style={{ padding: '4px 8px' }}>{f.total_lines}</td>
|
|
143
|
-
<td className="text-right" style={{ padding: '4px 8px' }}>{f.function_count}</td>
|
|
144
|
-
<td className="text-left" style={{ padding: '4px 8px' }}>{f.path}</td>
|
|
145
|
-
</tr>
|
|
146
|
-
))}
|
|
147
|
-
</tbody>
|
|
148
|
-
</table>
|
|
149
|
-
</div>
|
|
150
|
-
);
|
|
151
|
-
};
|
|
152
|
-
|
|
153
|
-
return (
|
|
154
|
-
<ToolDialog
|
|
155
|
-
open={open}
|
|
156
|
-
onOpenChange={onOpenChange}
|
|
157
|
-
title="Complexity"
|
|
158
|
-
description="Cyclomatic complexity analysis"
|
|
159
|
-
>
|
|
160
|
-
{renderContent()}
|
|
161
|
-
</ToolDialog>
|
|
162
|
-
);
|
|
163
|
-
}
|