Skip to content

fix: stop event propagation on keybinding match to prevent dropdown expansion#11886

Open
kaili-yang wants to merge 3 commits intomainfrom
fix/dropdown-expands-on-run-keybind
Open

fix: stop event propagation on keybinding match to prevent dropdown expansion#11886
kaili-yang wants to merge 3 commits intomainfrom
fix/dropdown-expands-on-run-keybind

Conversation

@kaili-yang
Copy link
Copy Markdown
Collaborator

@kaili-yang kaili-yang commented May 4, 2026

Summary

When pressing Ctrl+Enter to run a job while a dropdown widget was focused, the dropdown would expand. This happened because:

  1. The keybinding handler was registered on window in bubble phase — firing after child element handlers
  2. PrimeVue's Select onKeyDown fires first and opens the dropdown on any Enter keypress regardless of modifier keys
  3. The keybinding handler only called preventDefault(), which doesn't stop other listeners from running

Fix:

  • Register the keybinding handler in capture phase ({ capture: true }) so it fires top-down, before any element-level handlers
  • Call event.stopPropagation() when a keybinding is matched, preventing the event from reaching the dropdown

Red-Green Verification

Commit Purpose
test: add failing test for keybinding stopPropagation when dropdown is focused CI red — proves test catches the bug
fix: stop event propagation on keybinding match to prevent dropdown expansion CI green — proves fix resolves it

Test Plan

focus a dropdown widget (e.g. KSampler sampler_name), press Ctrl+Enter → dropdown stays closed, job runs
before

before11.mov

after

after111.mov

Note

Medium Risk
Changes global keydown handling and propagation behavior, which can subtly affect other keyboard interactions across the app. Scope is small and covered by new targeted tests, reducing regression risk.

Overview
Prevents UI controls (e.g. dropdowns) from also handling keys when a global keybinding fires by stopping event propagation on matched shortcuts in keybindingService.keybindHandler.

Registers the window keydown listener in capture phase in GraphView.vue so keybindings run before component-level handlers. Adds a new Vitest suite verifying propagation is stopped for Ctrl+Enter when a non-input element is focused, and not stopped when no binding matches or when typing in text inputs.

Reviewed by Cursor Bugbot for commit 426d8a6. Bugbot is set up for automated code reviews on this repo. Configure here.

┆Issue is synchronized with this Notion page by Unito

@kaili-yang kaili-yang requested a review from a team May 4, 2026 05:31
@dosubot dosubot Bot added the size:XS This PR changes 0-9 lines, ignoring generated files. label May 4, 2026
@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai Bot commented May 4, 2026

Note

Reviews paused

It looks like this branch is under active development. To avoid overwhelming you with review comments due to an influx of new commits, CodeRabbit has automatically paused this review. You can configure this behavior by changing the reviews.auto_review.auto_pause_after_reviewed_commits setting.

Use the following commands to manage reviews:

  • @coderabbitai resume to resume automatic reviews.
  • @coderabbitai review to trigger a single review.

Use the checkboxes below for quick actions:

  • ▶️ Resume reviews
  • 🔍 Trigger review
📝 Walkthrough

Walkthrough

This PR implements keyboard event propagation control for keybindings. The keybinding service now calls stopPropagation() on matched keys, the global keydown listener switches to capture-phase handling, two canvas commands guard against execution during ghost node placement, and comprehensive propagation tests are added.

Changes

Keyboard Event Propagation Control

Layer / File(s) Summary
Core Event Handler
src/platform/keybindings/keybindingService.ts
When a keybinding matches and passes input/target checks, event.stopPropagation() is called immediately after event.preventDefault() to prevent further event bubbling.
Listener Registration
src/views/GraphView.vue
The global keydown event listener is registered with { capture: true } to use capture-phase event handling instead of the default bubbling phase.
Command Guards
src/composables/useCoreCommands.ts
Comfy.Canvas.DeleteSelectedItems and Comfy.Graph.ExitSubgraph now check for active ghost node placement; if set, they finalize the placement and return early, preventing command execution during this state.
Propagation Validation
src/platform/keybindings/keybindingService.propagation.test.ts
New Vitest suite verifies that matched keybindings call stopPropagation(), unbound keys and textinput targets do not, and guards correctly prevent command execution during ghost placement.

Sequence Diagram

sequenceDiagram
    participant User
    participant Window
    participant GraphView
    participant KeybindingService
    participant CommandStore
    participant CanvasState

    rect rgba(200, 150, 100, 0.5)
    Note over User,CanvasState: Before (default bubbling phase)
    User->>Window: Press Ctrl+Enter
    Window->>GraphView: keydown event (bubbling)
    GraphView->>KeybindingService: keybindHandler called
    KeybindingService->>KeybindingService: Match keybinding
    KeybindingService->>CommandStore: execute(QueuePrompt)
    CommandStore->>CanvasState: Check ghostNodeId
    Note over Window,CanvasState: Event continues bubbling
    end

    rect rgba(100, 150, 200, 0.5)
    Note over User,CanvasState: After (capture phase + stopPropagation)
    User->>Window: Press Ctrl+Enter
    Window->>GraphView: keydown event (capture phase)
    GraphView->>KeybindingService: keybindHandler called
    KeybindingService->>KeybindingService: Match keybinding
    KeybindingService->>Window: stopPropagation()
    KeybindingService->>CommandStore: execute(QueuePrompt)
    CommandStore->>CanvasState: Check ghostNodeId
    CanvasState-->>CommandStore: If ghostNodeId set, finalize & abort
    Note over Window,CanvasState: Event propagation halted
    end
Loading

Estimated code review effort

🎯 3 (Moderate) | ⏱️ ~23 minutes

Poem

🐰 A rabbit hops through key events swift,
With stopPropagation, no more drift!
Capture phase catches before bubbles rise,
Ghost placements safe—what a surprise! ✨
Events now flow in perfect alignment,
Commands respect the keyboard's quiet assignment.


Caution

Pre-merge checks failed

Please resolve all errors before merging. Addressing warnings is optional.

  • Ignore (reviewers only)

❌ Failed checks (1 error, 1 warning)

Check name Status Explanation Resolution
End-To-End Regression Coverage For Fixes ❌ Error PR uses bug-fix language and changes global keydown handling, but lacks browser_tests/ E2E coverage and concrete justification for omitting E2E regression tests. Add a Playwright regression test under browser_tests/ to verify the dropdown expansion fix, or provide concrete explanation of why E2E testing is impractical for this change.
Docstring Coverage ⚠️ Warning Docstring coverage is 0.00% which is insufficient. The required threshold is 80.00%. Write docstrings for the functions missing them to satisfy the coverage threshold.
✅ Passed checks (5 passed)
Check name Status Explanation
Title check ✅ Passed The title accurately describes the main change: preventing dropdown expansion by stopping event propagation on keybinding match.
Description check ✅ Passed The description follows the template structure with Summary, Changes, and Review Focus sections; includes detailed context, root cause analysis, and test verification.
Linked Issues check ✅ Passed Check skipped because no linked issues were found for this pull request.
Out of Scope Changes check ✅ Passed Check skipped because no linked issues were found for this pull request.
Adr Compliance For Entity/Litegraph Changes ✅ Passed PR does not modify files under src/lib/litegraph/, src/ecs/, or graph entity definition files; changes are limited to keybinding handling and command guards.
✨ Finishing Touches
🧪 Generate unit tests (beta)
  • Create PR with unit tests
  • Commit unit tests in branch fix/dropdown-expands-on-run-keybind

Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ Share

Comment @coderabbitai help to get the list of available commands and usage tips.

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 4, 2026

🎨 Storybook: ✅ Built — View Storybook

Details

⏰ Completed at: 05/04/2026, 10:55:27 PM UTC

Links

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 4, 2026

🎭 Playwright: ✅ 1483 passed, 0 failed · 1 flaky

📊 Browser Reports
  • chromium: View Report (✅ 1464 / ❌ 0 / ⚠️ 1 / ⏭️ 5)
  • chromium-2x: View Report (✅ 2 / ❌ 0 / ⚠️ 0 / ⏭️ 0)
  • chromium-0.5x: View Report (✅ 1 / ❌ 0 / ⚠️ 0 / ⏭️ 0)
  • mobile-chrome: View Report (✅ 16 / ❌ 0 / ⚠️ 0 / ⏭️ 0)

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 4, 2026

📦 Bundle: 5.26 MB gzip 🔴 +174 B

Details

Summary

  • Raw size: 24.2 MB baseline 24.2 MB — 🔴 +349 B
  • Gzip: 5.26 MB baseline 5.26 MB — 🔴 +174 B
  • Brotli: 4.07 MB baseline 4.07 MB — 🔴 +20 B
  • Bundles: 259 current • 259 baseline • 118 added / 118 removed

Category Glance
Graph Workspace 🔴 +237 B (1.24 MB) · Data & Services 🔴 +112 B (3.05 MB) · Vendor & Third-Party ⚪ 0 B (9.94 MB) · Other ⚪ 0 B (8.82 MB) · Panels & Settings ⚪ 0 B (489 kB) · Utilities & Hooks ⚪ 0 B (365 kB) · + 5 more

App Entry Points — 22.6 kB (baseline 22.6 kB) • ⚪ 0 B

Main entry bundles and manifests

File Before After Δ Raw Δ Gzip Δ Brotli
assets/index-B_cnUOBb.js (removed) 22.6 kB 🟢 -22.6 kB 🟢 -8 kB 🟢 -6.88 kB
assets/index-LTbVHlFI.js (new) 22.6 kB 🔴 +22.6 kB 🔴 +8.01 kB 🔴 +6.86 kB

Status: 1 added / 1 removed

Graph Workspace — 1.24 MB (baseline 1.24 MB) • 🔴 +237 B

Graph editor runtime, canvas, workflow orchestration

File Before After Δ Raw Δ Gzip Δ Brotli
assets/GraphView-CAjzLCrm.js (new) 1.24 MB 🔴 +1.24 MB 🔴 +265 kB 🔴 +199 kB
assets/GraphView-CwvBUNBN.js (removed) 1.24 MB 🟢 -1.24 MB 🟢 -265 kB 🟢 -199 kB

Status: 1 added / 1 removed

Views & Navigation — 82.3 kB (baseline 82.3 kB) • ⚪ 0 B

Top-level views, pages, and routed surfaces

File Before After Δ Raw Δ Gzip Δ Brotli
assets/CloudSurveyView-CcAgQpfa.js (new) 19.6 kB 🔴 +19.6 kB 🔴 +5.14 kB 🔴 +4.58 kB
assets/CloudSurveyView-D8r-edUT.js (removed) 19.6 kB 🟢 -19.6 kB 🟢 -5.14 kB 🟢 -4.57 kB
assets/CloudLoginView-D9k2ne_j.js (new) 12.4 kB 🔴 +12.4 kB 🔴 +3.51 kB 🔴 +3.1 kB
assets/CloudLoginView-jlIHz_aa.js (removed) 12.4 kB 🟢 -12.4 kB 🟢 -3.51 kB 🟢 -3.1 kB
assets/CloudSignupView-ByxroyOP.js (removed) 10.2 kB 🟢 -10.2 kB 🟢 -3 kB 🟢 -2.64 kB
assets/CloudSignupView-oV8kAwNZ.js (new) 10.2 kB 🔴 +10.2 kB 🔴 +3 kB 🔴 +2.64 kB
assets/UserCheckView-CpgnmB0b.js (new) 9.07 kB 🔴 +9.07 kB 🔴 +2.33 kB 🔴 +2.04 kB
assets/UserCheckView-CYdzkePl.js (removed) 9.07 kB 🟢 -9.07 kB 🟢 -2.33 kB 🟢 -2.04 kB
assets/CloudLayoutView-CicXDIk3.js (new) 7.73 kB 🔴 +7.73 kB 🔴 +2.45 kB 🔴 +2.15 kB
assets/CloudLayoutView-ClEp-lj-.js (removed) 7.73 kB 🟢 -7.73 kB 🟢 -2.45 kB 🟢 -2.14 kB
assets/CloudForgotPasswordView-r8xX_X2p.js (new) 6.14 kB 🔴 +6.14 kB 🔴 +2.19 kB 🔴 +1.92 kB
assets/CloudForgotPasswordView-Uai6LZM6.js (removed) 6.14 kB 🟢 -6.14 kB 🟢 -2.19 kB 🟢 -1.95 kB
assets/CloudAuthTimeoutView-DrBCQ_n_.js (new) 5.5 kB 🔴 +5.5 kB 🔴 +2.01 kB 🔴 +1.78 kB
assets/CloudAuthTimeoutView-DxRgqVGN.js (removed) 5.5 kB 🟢 -5.5 kB 🟢 -2.02 kB 🟢 -1.78 kB
assets/CloudSubscriptionRedirectView-BKIQYHU0.js (new) 5.28 kB 🔴 +5.28 kB 🔴 +2 kB 🔴 +1.77 kB
assets/CloudSubscriptionRedirectView-D6DHHKsD.js (removed) 5.28 kB 🟢 -5.28 kB 🟢 -2 kB 🟢 -1.77 kB
assets/UserSelectView-B4vCSuHa.js (removed) 4.73 kB 🟢 -4.73 kB 🟢 -1.75 kB 🟢 -1.56 kB
assets/UserSelectView-BE2aslwC.js (new) 4.73 kB 🔴 +4.73 kB 🔴 +1.75 kB 🔴 +1.55 kB

Status: 9 added / 9 removed / 2 unchanged

Panels & Settings — 489 kB (baseline 489 kB) • ⚪ 0 B

Configuration panels, inspectors, and settings screens

File Before After Δ Raw Δ Gzip Δ Brotli
assets/KeybindingPanel-DsbOcnrY.js (removed) 46.7 kB 🟢 -46.7 kB 🟢 -9.63 kB 🟢 -8.55 kB
assets/KeybindingPanel-hZQTyAZk.js (new) 46.7 kB 🔴 +46.7 kB 🔴 +9.62 kB 🔴 +8.55 kB
assets/SecretsPanel-D36pu8tN.js (removed) 22.9 kB 🟢 -22.9 kB 🟢 -5.54 kB 🟢 -4.87 kB
assets/SecretsPanel-DnqNF5x3.js (new) 22.9 kB 🔴 +22.9 kB 🔴 +5.55 kB 🔴 +4.87 kB
assets/LegacyCreditsPanel-C63lyNC_.js (removed) 21.7 kB 🟢 -21.7 kB 🟢 -5.91 kB 🟢 -5.21 kB
assets/LegacyCreditsPanel-CXCWs7oB.js (new) 21.7 kB 🔴 +21.7 kB 🔴 +5.91 kB 🔴 +5.22 kB
assets/SubscriptionPanel-5o7Bc9Vh.js (new) 20 kB 🔴 +20 kB 🔴 +5.11 kB 🔴 +4.49 kB
assets/SubscriptionPanel-BWoBhZby.js (removed) 20 kB 🟢 -20 kB 🟢 -5.11 kB 🟢 -4.5 kB
assets/AboutPanel-BHauq2gF.js (new) 12 kB 🔴 +12 kB 🔴 +3.32 kB 🔴 +2.98 kB
assets/AboutPanel-Fz9pqdc0.js (removed) 12 kB 🟢 -12 kB 🟢 -3.32 kB 🟢 -2.99 kB
assets/ExtensionPanel-BDWpsNZf.js (removed) 9.97 kB 🟢 -9.97 kB 🟢 -2.91 kB 🟢 -2.59 kB
assets/ExtensionPanel-Cvjv9tWf.js (new) 9.97 kB 🔴 +9.97 kB 🔴 +2.91 kB 🔴 +2.59 kB
assets/ServerConfigPanel-Cwe6yfeq.js (removed) 7.05 kB 🟢 -7.05 kB 🟢 -2.36 kB 🟢 -2.1 kB
assets/ServerConfigPanel-U8_hLeOg.js (new) 7.05 kB 🔴 +7.05 kB 🔴 +2.36 kB 🔴 +2.12 kB
assets/UserPanel-dg3QlVM4.js (new) 6.75 kB 🔴 +6.75 kB 🔴 +2.25 kB 🔴 +1.97 kB
assets/UserPanel-DsyoZ0m6.js (removed) 6.75 kB 🟢 -6.75 kB 🟢 -2.25 kB 🟢 -1.97 kB
assets/cloudRemoteConfig-Cn8pvjHh.js (new) 2.05 kB 🔴 +2.05 kB 🔴 +989 B 🔴 +850 B
assets/cloudRemoteConfig-Dp35QnRf.js (removed) 2.05 kB 🟢 -2.05 kB 🟢 -988 B 🟢 -851 B
assets/refreshRemoteConfig-CqYJ_tgS.js (removed) 1.45 kB 🟢 -1.45 kB 🟢 -648 B 🟢 -549 B
assets/refreshRemoteConfig-CtqiWTQ2.js (new) 1.45 kB 🔴 +1.45 kB 🔴 +648 B 🔴 +551 B

Status: 10 added / 10 removed / 11 unchanged

User & Accounts — 17.6 kB (baseline 17.6 kB) • ⚪ 0 B

Authentication, profile, and account management bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/auth-B6qyFq8M.js (new) 3.65 kB 🔴 +3.65 kB 🔴 +1.29 kB 🔴 +1.11 kB
assets/auth-Cv1WcNBO.js (removed) 3.65 kB 🟢 -3.65 kB 🟢 -1.29 kB 🟢 -1.11 kB
assets/SignUpForm-DVY_Sdw-.js (removed) 3.19 kB 🟢 -3.19 kB 🟢 -1.29 kB 🟢 -1.15 kB
assets/SignUpForm-rwCFa04a.js (new) 3.19 kB 🔴 +3.19 kB 🔴 +1.29 kB 🔴 +1.15 kB
assets/UpdatePasswordContent-BQshfmUF.js (removed) 2.9 kB 🟢 -2.9 kB 🟢 -1.3 kB 🟢 -1.16 kB
assets/UpdatePasswordContent-Dm7lIG9y.js (new) 2.9 kB 🔴 +2.9 kB 🔴 +1.3 kB 🔴 +1.16 kB
assets/authStore-CWzTsSqJ.js (new) 1.19 kB 🔴 +1.19 kB 🔴 +570 B 🔴 +505 B
assets/authStore-WoUv-ZJ2.js (removed) 1.19 kB 🟢 -1.19 kB 🟢 -566 B 🟢 -508 B
assets/auth-CZpFdkg5.js (removed) 348 B 🟢 -348 B 🟢 -214 B 🟢 -186 B
assets/auth-D7t6O0v9.js (new) 348 B 🔴 +348 B 🔴 +213 B 🔴 +204 B

Status: 5 added / 5 removed / 2 unchanged

Editors & Dialogs — 112 kB (baseline 112 kB) • ⚪ 0 B

Modals, dialogs, drawers, and in-app editors

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyHubPublishDialog-B0pp9aa8.js (removed) 85.8 kB 🟢 -85.8 kB 🟢 -18.6 kB 🟢 -15.9 kB
assets/ComfyHubPublishDialog-slAbu1Zy.js (new) 85.8 kB 🔴 +85.8 kB 🔴 +18.6 kB 🔴 +15.9 kB
assets/useShareDialog-D_vUeSiR.js (removed) 23.8 kB 🟢 -23.8 kB 🟢 -5.78 kB 🟢 -5.12 kB
assets/useShareDialog-DRFAEfQq.js (new) 23.8 kB 🔴 +23.8 kB 🔴 +5.79 kB 🔴 +5.12 kB
assets/ComfyHubPublishDialog-B3QGSDba.js (new) 1.35 kB 🔴 +1.35 kB 🔴 +629 B 🔴 +550 B
assets/ComfyHubPublishDialog-DHI2nrmG.js (removed) 1.35 kB 🟢 -1.35 kB 🟢 -626 B 🟢 -549 B
assets/useSubscriptionDialog-CfV_FQs_.js (new) 1.17 kB 🔴 +1.17 kB 🔴 +559 B 🔴 +488 B
assets/useSubscriptionDialog-Dyk3-jGD.js (removed) 1.17 kB 🟢 -1.17 kB 🟢 -556 B 🟢 -488 B

Status: 4 added / 4 removed

UI Components — 62.9 kB (baseline 62.9 kB) • ⚪ 0 B

Reusable component library chunks

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyQueueButton-BzjeQs8s.js (new) 13.5 kB 🔴 +13.5 kB 🔴 +3.79 kB 🔴 +3.38 kB
assets/ComfyQueueButton-jHH5ExYn.js (removed) 13.5 kB 🟢 -13.5 kB 🟢 -3.79 kB 🟢 -3.38 kB
assets/useTerminalTabs-BnjlPwik.js (removed) 11 kB 🟢 -11 kB 🟢 -3.73 kB 🟢 -3.28 kB
assets/useTerminalTabs-C5Z4oN7J.js (new) 11 kB 🔴 +11 kB 🔴 +3.73 kB 🔴 +3.29 kB
assets/SubscribeButton-55XVrWJH.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +1.05 kB 🔴 +935 B
assets/SubscribeButton-Dq9J1573.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -1.05 kB 🟢 -933 B
assets/cloudFeedbackTopbarButton-7iMA-rfP.js (new) 1.83 kB 🔴 +1.83 kB 🔴 +944 B 🔴 +827 B
assets/cloudFeedbackTopbarButton-CSCWiLmA.js (removed) 1.83 kB 🟢 -1.83 kB 🟢 -941 B 🟢 -828 B
assets/ComfyQueueButton-Cm6jxznb.js (removed) 1.27 kB 🟢 -1.27 kB 🟢 -593 B 🟢 -525 B
assets/ComfyQueueButton-u-A1wLQB.js (new) 1.27 kB 🔴 +1.27 kB 🔴 +595 B 🔴 +525 B

Status: 5 added / 5 removed / 9 unchanged

Data & Services — 3.05 MB (baseline 3.05 MB) • 🔴 +112 B

Stores, services, APIs, and repositories

File Before After Δ Raw Δ Gzip Δ Brotli
assets/dialogService-BJxIH8HC.js (removed) 1.99 MB 🟢 -1.99 MB 🟢 -458 kB 🟢 -347 kB
assets/dialogService-ekaAOQA7.js (new) 1.99 MB 🔴 +1.99 MB 🔴 +458 kB 🔴 +347 kB
assets/api-CbaLeFNz.js (new) 887 kB 🔴 +887 kB 🔴 +212 kB 🔴 +167 kB
assets/api-CPZ-DmUE.js (removed) 887 kB 🟢 -887 kB 🟢 -212 kB 🟢 -167 kB
assets/load3dService-CcU--m19.js (new) 115 kB 🔴 +115 kB 🔴 +25.1 kB 🔴 +21.4 kB
assets/load3dService-DjivRBJS.js (removed) 115 kB 🟢 -115 kB 🟢 -25.1 kB 🟢 -21.3 kB
assets/workflowShareService-DryWTkJ0.js (removed) 16.6 kB 🟢 -16.6 kB 🟢 -4.89 kB 🟢 -4.33 kB
assets/workflowShareService-gk8ZDH4N.js (new) 16.6 kB 🔴 +16.6 kB 🔴 +4.89 kB 🔴 +4.35 kB
assets/keybindingService-DA91r344.js (new) 13.9 kB 🔴 +13.9 kB 🔴 +3.69 kB 🔴 +3.23 kB
assets/keybindingService-DA31FAA4.js (removed) 13.8 kB 🟢 -13.8 kB 🟢 -3.67 kB 🟢 -3.21 kB
assets/releaseStore-BTwEsXvI.js (removed) 8.12 kB 🟢 -8.12 kB 🟢 -2.28 kB 🟢 -2 kB
assets/releaseStore-gULfA9Om.js (new) 8.12 kB 🔴 +8.12 kB 🔴 +2.28 kB 🔴 +2 kB
assets/userStore-DH1Xp8Xz.js (new) 2.24 kB 🔴 +2.24 kB 🔴 +870 B 🔴 +761 B
assets/userStore-DPYAC3CN.js (removed) 2.24 kB 🟢 -2.24 kB 🟢 -871 B 🟢 -763 B
assets/audioService-D7rkFw3S.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -876 B 🟢 -767 B
assets/audioService-gDG38irW.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +876 B 🔴 +763 B
assets/releaseStore-DqT2wPD8.js (new) 1.19 kB 🔴 +1.19 kB 🔴 +563 B 🔴 +496 B
assets/releaseStore-DvnagRsE.js (removed) 1.19 kB 🟢 -1.19 kB 🟢 -559 B 🟢 -496 B
assets/workflowDraftStore-Cyg-d2pQ.js (new) 1.17 kB 🔴 +1.17 kB 🔴 +558 B 🔴 +492 B
assets/workflowDraftStore-Ds9RP0Ub.js (removed) 1.17 kB 🟢 -1.17 kB 🟢 -554 B 🟢 -495 B
assets/dialogService-BPn6Pgkx.js (new) 1.16 kB 🔴 +1.16 kB 🔴 +550 B 🔴 +489 B
assets/dialogService-DoD7PRcw.js (removed) 1.16 kB 🟢 -1.16 kB 🟢 -547 B 🟢 -491 B
assets/settingStore-B_twTpCV.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +552 B 🔴 +487 B
assets/settingStore-CJv8KLJk.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -549 B 🟢 -489 B
assets/assetsStore-B3I4maeu.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +552 B 🔴 +487 B
assets/assetsStore-yTkQvJYr.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -550 B 🟢 -491 B

Status: 13 added / 13 removed / 4 unchanged

Utilities & Hooks — 365 kB (baseline 365 kB) • ⚪ 0 B

Helpers, composables, and utility bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useConflictDetection-Cb1MzuDu.js (removed) 233 kB 🟢 -233 kB 🟢 -51.8 kB 🟢 -42.3 kB
assets/useConflictDetection-DO-yBppo.js (new) 233 kB 🔴 +233 kB 🔴 +51.8 kB 🔴 +42.3 kB
assets/useLoad3d-CQKBN65U.js (new) 22.3 kB 🔴 +22.3 kB 🔴 +5.09 kB 🔴 +4.49 kB
assets/useLoad3d-DixGTEjJ.js (removed) 22.3 kB 🟢 -22.3 kB 🟢 -5.09 kB 🟢 -4.49 kB
assets/useLoad3dViewer-D_F37X3r.js (removed) 20.8 kB 🟢 -20.8 kB 🟢 -4.91 kB 🟢 -4.3 kB
assets/useLoad3dViewer-jBIpEEgY.js (new) 20.8 kB 🔴 +20.8 kB 🔴 +4.91 kB 🔴 +4.29 kB
assets/useFeatureFlags-BYJpjia0.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -1.79 kB 🟢 -1.53 kB
assets/useFeatureFlags-CjGVt4ZT.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +1.79 kB 🔴 +1.53 kB
assets/useCopyToClipboard-BYWIYDq4.js (new) 5.29 kB 🔴 +5.29 kB 🔴 +1.86 kB 🔴 +1.57 kB
assets/useCopyToClipboard-CxloohLV.js (removed) 5.29 kB 🟢 -5.29 kB 🟢 -1.86 kB 🟢 -1.57 kB
assets/useWorkspaceUI-CroGEgo2.js (removed) 3.34 kB 🟢 -3.34 kB 🟢 -982 B 🟢 -809 B
assets/useWorkspaceUI-iOj-QIIB.js (new) 3.34 kB 🔴 +3.34 kB 🔴 +981 B 🔴 +813 B
assets/subscriptionCheckoutUtil-Bel9XJis.js (removed) 3.31 kB 🟢 -3.31 kB 🟢 -1.36 kB 🟢 -1.18 kB
assets/subscriptionCheckoutUtil-DvFLfTRH.js (new) 3.31 kB 🔴 +3.31 kB 🔴 +1.36 kB 🔴 +1.19 kB
assets/assetPreviewUtil-C2fGtaCr.js (new) 2.27 kB 🔴 +2.27 kB 🔴 +957 B 🔴 +833 B
assets/assetPreviewUtil-DoKa_Ipp.js (removed) 2.27 kB 🟢 -2.27 kB 🟢 -958 B 🟢 -834 B
assets/useUpstreamValue-CD8OZZac.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -806 B 🟢 -714 B
assets/useUpstreamValue-DhI8Uf0F.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +805 B 🔴 +711 B
assets/useLoad3d-B4rfTi-P.js (new) 1.33 kB 🔴 +1.33 kB 🔴 +621 B 🔴 +563 B
assets/useLoad3d-BPv4PjiB.js (removed) 1.33 kB 🟢 -1.33 kB 🟢 -617 B 🟢 -561 B
assets/useLoad3dViewer-B3p-qegl.js (removed) 1.27 kB 🟢 -1.27 kB 🟢 -586 B 🟢 -527 B
assets/useLoad3dViewer-D9gsgvHU.js (new) 1.27 kB 🔴 +1.27 kB 🔴 +587 B 🔴 +526 B
assets/useCurrentUser-Dmfe6WvX.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -550 B 🟢 -490 B
assets/useCurrentUser-kBMQz2tw.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +553 B 🔴 +486 B
assets/useWorkspaceSwitch-Dh5ZvEln.js (new) 747 B 🔴 +747 B 🔴 +382 B 🔴 +330 B
assets/useWorkspaceSwitch-DJEW3JFg.js (removed) 747 B 🟢 -747 B 🟢 -385 B 🟢 -333 B

Status: 13 added / 13 removed / 18 unchanged

Vendor & Third-Party — 9.94 MB (baseline 9.94 MB) • ⚪ 0 B

External libraries and shared vendor chunks

Status: 16 unchanged

Other — 8.82 MB (baseline 8.82 MB) • ⚪ 0 B

Bundles that do not match a named category

File Before After Δ Raw Δ Gzip Δ Brotli
assets/core-BBUBT3wk.js (removed) 76.7 kB 🟢 -76.7 kB 🟢 -19.9 kB 🟢 -16.9 kB
assets/core-KM_lnLnV.js (new) 76.7 kB 🔴 +76.7 kB 🔴 +19.9 kB 🔴 +16.9 kB
assets/groupNode-CHnDROPZ.js (new) 74.9 kB 🔴 +74.9 kB 🔴 +18.7 kB 🔴 +16.5 kB
assets/groupNode-TOP5gTMQ.js (removed) 74.9 kB 🟢 -74.9 kB 🟢 -18.7 kB 🟢 -16.5 kB
assets/WidgetSelect-B7OcXL9c.js (removed) 67.4 kB 🟢 -67.4 kB 🟢 -14.6 kB 🟢 -12.7 kB
assets/WidgetSelect-CYekG-I-.js (new) 67.4 kB 🔴 +67.4 kB 🔴 +14.6 kB 🔴 +12.7 kB
assets/SubscriptionRequiredDialogContentWorkspace-DuqbQjNQ.js (new) 48.8 kB 🔴 +48.8 kB 🔴 +9.53 kB 🔴 +8.22 kB
assets/SubscriptionRequiredDialogContentWorkspace-Lp44g3Cn.js (removed) 48.8 kB 🟢 -48.8 kB 🟢 -9.52 kB 🟢 -8.24 kB
assets/Load3DControls-CNHLVO8p.js (new) 46.1 kB 🔴 +46.1 kB 🔴 +7.51 kB 🔴 +6.56 kB
assets/Load3DControls-T1kgRtz2.js (removed) 46.1 kB 🟢 -46.1 kB 🟢 -7.51 kB 🟢 -6.55 kB
assets/WidgetPainter-dMZosfkg.js (new) 34 kB 🔴 +34 kB 🔴 +8.3 kB 🔴 +7.36 kB
assets/WidgetPainter-DppuyjAq.js (removed) 34 kB 🟢 -34 kB 🟢 -8.3 kB 🟢 -7.37 kB
assets/WorkspacePanelContent-C9jFinhh.js (new) 32.8 kB 🔴 +32.8 kB 🔴 +7.01 kB 🔴 +6.2 kB
assets/WorkspacePanelContent-CSTz0zB3.js (removed) 32.8 kB 🟢 -32.8 kB 🟢 -7.01 kB 🟢 -6.2 kB
assets/Load3dViewerContent-DcVFhH-9.js (removed) 28 kB 🟢 -28 kB 🟢 -5.85 kB 🟢 -5.07 kB
assets/Load3dViewerContent-DUVLaEK5.js (new) 28 kB 🔴 +28 kB 🔴 +5.85 kB 🔴 +5.07 kB
assets/SubscriptionRequiredDialogContent-CUwNWjgc.js (removed) 27.5 kB 🟢 -27.5 kB 🟢 -6.98 kB 🟢 -6.16 kB
assets/SubscriptionRequiredDialogContent-oLfkQ1bM.js (new) 27.5 kB 🔴 +27.5 kB 🔴 +6.98 kB 🔴 +6.17 kB
assets/WidgetImageCrop-Cv65l7RI.js (removed) 24.3 kB 🟢 -24.3 kB 🟢 -6.2 kB 🟢 -5.46 kB
assets/WidgetImageCrop-ZeJn6jj3.js (new) 24.3 kB 🔴 +24.3 kB 🔴 +6.2 kB 🔴 +5.46 kB
assets/SubscriptionPanelContentWorkspace-CBVSsjY1.js (removed) 22.2 kB 🟢 -22.2 kB 🟢 -5.18 kB 🟢 -4.57 kB
assets/SubscriptionPanelContentWorkspace-CMZL6VKX.js (new) 22.2 kB 🔴 +22.2 kB 🔴 +5.18 kB 🔴 +4.56 kB
assets/SignInContent-Bj_tw-Rf.js (removed) 20.8 kB 🟢 -20.8 kB 🟢 -5.44 kB 🟢 -4.75 kB
assets/SignInContent-CeAwS7op.js (new) 20.8 kB 🔴 +20.8 kB 🔴 +5.44 kB 🔴 +4.75 kB
assets/CurrentUserPopoverWorkspace-BLH4XDZ-.js (new) 20.6 kB 🔴 +20.6 kB 🔴 +4.91 kB 🔴 +4.4 kB
assets/CurrentUserPopoverWorkspace-CohaMOw0.js (removed) 20.6 kB 🟢 -20.6 kB 🟢 -4.91 kB 🟢 -4.4 kB
assets/WidgetInputNumber-DrfdolWK.js (new) 19.1 kB 🔴 +19.1 kB 🔴 +4.84 kB 🔴 +4.29 kB
assets/WidgetInputNumber-hg8tYm88.js (removed) 19.1 kB 🟢 -19.1 kB 🟢 -4.84 kB 🟢 -4.29 kB
assets/Load3D-CmXi1R2Q.js (new) 18.5 kB 🔴 +18.5 kB 🔴 +4.39 kB 🔴 +3.83 kB
assets/Load3D-DsUkyFE-.js (removed) 18.5 kB 🟢 -18.5 kB 🟢 -4.39 kB 🟢 -3.83 kB
assets/WidgetRecordAudio-CAuI4t9T.js (new) 17.5 kB 🔴 +17.5 kB 🔴 +5.04 kB 🔴 +4.51 kB
assets/WidgetRecordAudio-D0O8kI4o.js (removed) 17.5 kB 🟢 -17.5 kB 🟢 -5.04 kB 🟢 -4.51 kB
assets/WidgetRange-Cnn4IgFL.js (new) 17.1 kB 🔴 +17.1 kB 🔴 +4.61 kB 🔴 +4.12 kB
assets/WidgetRange-DnceJoo5.js (removed) 17.1 kB 🟢 -17.1 kB 🟢 -4.61 kB 🟢 -4.11 kB
assets/load3d-BpTOBw5Y.js (new) 15.8 kB 🔴 +15.8 kB 🔴 +4.59 kB 🔴 +3.98 kB
assets/load3d-CT-G1RLj.js (removed) 15.8 kB 🟢 -15.8 kB 🟢 -4.59 kB 🟢 -3.98 kB
assets/WaveAudioPlayer-Bs4x3fMB.js (removed) 13.4 kB 🟢 -13.4 kB 🟢 -3.69 kB 🟢 -3.22 kB
assets/WaveAudioPlayer-D76T5CYB.js (new) 13.4 kB 🔴 +13.4 kB 🔴 +3.69 kB 🔴 +3.23 kB
assets/WidgetCurve-_f4v6DZk.js (new) 12.2 kB 🔴 +12.2 kB 🔴 +3.93 kB 🔴 +3.57 kB
assets/WidgetCurve-C4YtmR0O.js (removed) 12.2 kB 🟢 -12.2 kB 🟢 -3.93 kB 🟢 -3.56 kB
assets/TeamWorkspacesDialogContent-CldeEG6r.js (removed) 11.3 kB 🟢 -11.3 kB 🟢 -3.42 kB 🟢 -3.04 kB
assets/TeamWorkspacesDialogContent-CmPx_1-x.js (new) 11.3 kB 🔴 +11.3 kB 🔴 +3.42 kB 🔴 +3.07 kB
assets/nodeTemplates-_G5UxvnH.js (new) 9.84 kB 🔴 +9.84 kB 🔴 +3.48 kB 🔴 +3.08 kB
assets/nodeTemplates-Ru0JgpvX.js (removed) 9.84 kB 🟢 -9.84 kB 🟢 -3.48 kB 🟢 -3.06 kB
assets/NightlySurveyController-D363yqdD.js (removed) 8.97 kB 🟢 -8.97 kB 🟢 -3.15 kB 🟢 -2.79 kB
assets/NightlySurveyController-DQNJ8vAx.js (new) 8.97 kB 🔴 +8.97 kB 🔴 +3.16 kB 🔴 +2.78 kB
assets/Load3DConfiguration-DA1yOJ_g.js (removed) 8.61 kB 🟢 -8.61 kB 🟢 -2.54 kB 🟢 -2.23 kB
assets/Load3DConfiguration-DEFZid-x.js (new) 8.61 kB 🔴 +8.61 kB 🔴 +2.54 kB 🔴 +2.23 kB
assets/InviteMemberDialogContent-BtRwmFeQ.js (removed) 7.94 kB 🟢 -7.94 kB 🟢 -2.53 kB 🟢 -2.21 kB
assets/InviteMemberDialogContent-COXBA77S.js (new) 7.94 kB 🔴 +7.94 kB 🔴 +2.53 kB 🔴 +2.2 kB
assets/onboardingCloudRoutes-B6QsUBdb.js (removed) 6.73 kB 🟢 -6.73 kB 🟢 -2.12 kB 🟢 -1.82 kB
assets/onboardingCloudRoutes-UZFS9LpB.js (new) 6.73 kB 🔴 +6.73 kB 🔴 +2.13 kB 🔴 +1.83 kB
assets/CreateWorkspaceDialogContent-DBt-2kAX.js (new) 6.15 kB 🔴 +6.15 kB 🔴 +2.24 kB 🔴 +1.97 kB
assets/CreateWorkspaceDialogContent-PkENm0AC.js (removed) 6.15 kB 🟢 -6.15 kB 🟢 -2.24 kB 🟢 -1.96 kB
assets/WidgetWithControl-D6bReHFS.js (removed) 6.09 kB 🟢 -6.09 kB 🟢 -2.44 kB 🟢 -2.17 kB
assets/WidgetWithControl-ObVHm7ga.js (new) 6.09 kB 🔴 +6.09 kB 🔴 +2.45 kB 🔴 +2.18 kB
assets/FreeTierDialogContent-BDSZtXrK.js (new) 6.01 kB 🔴 +6.01 kB 🔴 +2.13 kB 🔴 +1.89 kB
assets/FreeTierDialogContent-DkSO1Mu2.js (removed) 6.01 kB 🟢 -6.01 kB 🟢 -2.13 kB 🟢 -1.9 kB
assets/EditWorkspaceDialogContent-CPRYyA8L.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -2.2 kB 🟢 -1.93 kB
assets/EditWorkspaceDialogContent-DhlQZkQf.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +2.2 kB 🔴 +1.93 kB
assets/WidgetTextarea-DVp4MLOk.js (new) 5.76 kB 🔴 +5.76 kB 🔴 +2.27 kB 🔴 +2.02 kB
assets/WidgetTextarea-mF8X__ll.js (removed) 5.76 kB 🟢 -5.76 kB 🟢 -2.27 kB 🟢 -2.01 kB
assets/Preview3d-CFbOSTPQ.js (removed) 5.73 kB 🟢 -5.73 kB 🟢 -1.92 kB 🟢 -1.68 kB
assets/Preview3d-DxAo21Mv.js (new) 5.73 kB 🔴 +5.73 kB 🔴 +1.92 kB 🔴 +1.67 kB
assets/ValueControlPopover-B1YwfznW.js (removed) 5.53 kB 🟢 -5.53 kB 🟢 -2.02 kB 🟢 -1.81 kB
assets/ValueControlPopover-UnqL6J0r.js (new) 5.53 kB 🔴 +5.53 kB 🔴 +2.02 kB 🔴 +1.81 kB
assets/CancelSubscriptionDialogContent-DagKPhLK.js (removed) 5.49 kB 🟢 -5.49 kB 🟢 -2.06 kB 🟢 -1.8 kB
assets/CancelSubscriptionDialogContent-kVF2WfSa.js (new) 5.49 kB 🔴 +5.49 kB 🔴 +2.06 kB 🔴 +1.81 kB
assets/DeleteWorkspaceDialogContent-D1i4PAk4.js (new) 4.85 kB 🔴 +4.85 kB 🔴 +1.88 kB 🔴 +1.64 kB
assets/DeleteWorkspaceDialogContent-DkSzXTET.js (removed) 4.85 kB 🟢 -4.85 kB 🟢 -1.88 kB 🟢 -1.63 kB
assets/LeaveWorkspaceDialogContent-46Fj-Kig.js (new) 4.68 kB 🔴 +4.68 kB 🔴 +1.82 kB 🔴 +1.6 kB
assets/LeaveWorkspaceDialogContent-UNVkZWnD.js (removed) 4.68 kB 🟢 -4.68 kB 🟢 -1.82 kB 🟢 -1.58 kB
assets/RemoveMemberDialogContent-6ykaTyap.js (new) 4.66 kB 🔴 +4.66 kB 🔴 +1.78 kB 🔴 +1.57 kB
assets/RemoveMemberDialogContent-BEiWfHk9.js (removed) 4.66 kB 🟢 -4.66 kB 🟢 -1.78 kB 🟢 -1.55 kB
assets/RevokeInviteDialogContent-B1wqUPub.js (new) 4.57 kB 🔴 +4.57 kB 🔴 +1.79 kB 🔴 +1.57 kB
assets/RevokeInviteDialogContent-C3-NYhPP.js (removed) 4.57 kB 🟢 -4.57 kB 🟢 -1.79 kB 🟢 -1.58 kB
assets/InviteMemberUpsellDialogContent-BRdGxLry.js (new) 4.47 kB 🔴 +4.47 kB 🔴 +1.65 kB 🔴 +1.45 kB
assets/InviteMemberUpsellDialogContent-DvAU6Tg1.js (removed) 4.47 kB 🟢 -4.47 kB 🟢 -1.65 kB 🟢 -1.45 kB
assets/tierBenefits-BB6zbxsp.js (new) 4.45 kB 🔴 +4.45 kB 🔴 +1.57 kB 🔴 +1.36 kB
assets/tierBenefits-CfctCR4j.js (removed) 4.45 kB 🟢 -4.45 kB 🟢 -1.58 kB 🟢 -1.36 kB
assets/cloudSessionCookie-DA8KD8SR.js (new) 4.31 kB 🔴 +4.31 kB 🔴 +1.58 kB 🔴 +1.38 kB
assets/cloudSessionCookie-FQU2jagD.js (removed) 4.31 kB 🟢 -4.31 kB 🟢 -1.57 kB 🟢 -1.38 kB
assets/Media3DTop-Bko9V5_v.js (removed) 4.04 kB 🟢 -4.04 kB 🟢 -1.71 kB 🟢 -1.52 kB
assets/Media3DTop-D2L3eVch.js (new) 4.04 kB 🔴 +4.04 kB 🔴 +1.71 kB 🔴 +1.52 kB
assets/saveMesh-CjlqFCbp.js (new) 4.03 kB 🔴 +4.03 kB 🔴 +1.76 kB 🔴 +1.56 kB
assets/saveMesh-DGU5unJK.js (removed) 4.03 kB 🟢 -4.03 kB 🟢 -1.76 kB 🟢 -1.55 kB
assets/GlobalToast-B2L_olvI.js (removed) 3.05 kB 🟢 -3.05 kB 🟢 -1.26 kB 🟢 -1.11 kB
assets/GlobalToast-D0PjYsTj.js (new) 3.05 kB 🔴 +3.05 kB 🔴 +1.26 kB 🔴 +1.11 kB
assets/CloudRunButtonWrapper-CWcRHiTh.js (removed) 2.23 kB 🟢 -2.23 kB 🟢 -1.02 kB 🟢 -910 B
assets/CloudRunButtonWrapper-Dr0oMRpp.js (new) 2.23 kB 🔴 +2.23 kB 🔴 +1.02 kB 🔴 +907 B
assets/SubscribeToRun-Bg6CwpEZ.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +980 B 🔴 +878 B
assets/SubscribeToRun-CZ08xP1C.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -983 B 🟢 -880 B
assets/MediaAudioTop-CUNVgeb2.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +1.01 kB 🔴 +862 B
assets/MediaAudioTop-vSG6ng07.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -1 kB 🟢 -864 B
assets/cloudBadges-BRFVrRnt.js (new) 1.96 kB 🔴 +1.96 kB 🔴 +978 B 🔴 +853 B
assets/cloudBadges-CaK-ms0p.js (removed) 1.96 kB 🟢 -1.96 kB 🟢 -975 B 🟢 -851 B
assets/cloudSubscription-B56Bg11w.js (new) 1.88 kB 🔴 +1.88 kB 🔴 +899 B 🔴 +780 B
assets/cloudSubscription-ghL9qC7Z.js (removed) 1.88 kB 🟢 -1.88 kB 🟢 -895 B 🟢 -779 B
assets/graphHasMissingNodes-BC39FcHX.js (removed) 1.83 kB 🟢 -1.83 kB 🟢 -862 B 🟢 -778 B
assets/graphHasMissingNodes-DdLfhfvo.js (new) 1.83 kB 🔴 +1.83 kB 🔴 +859 B 🔴 +758 B
assets/Load3D-DaGNnzxE.js (new) 1.58 kB 🔴 +1.58 kB 🔴 +708 B 🔴 +627 B
assets/Load3D-DklMXIRC.js (removed) 1.58 kB 🟢 -1.58 kB 🟢 -707 B 🟢 -626 B
assets/previousFullPath-Cc0nSFm1.js (removed) 1.53 kB 🟢 -1.53 kB 🟢 -695 B 🟢 -596 B
assets/previousFullPath-WWbVRMKq.js (new) 1.53 kB 🔴 +1.53 kB 🔴 +695 B 🔴 +623 B
assets/nightlyBadges-CzLkKgmB.js (new) 1.49 kB 🔴 +1.49 kB 🔴 +743 B 🔴 +655 B
assets/nightlyBadges-Dc_PEdn1.js (removed) 1.49 kB 🟢 -1.49 kB 🟢 -744 B 🟢 -654 B
assets/Load3dViewerContent-C9kXdTb2.js (new) 1.46 kB 🔴 +1.46 kB 🔴 +665 B 🔴 +588 B
assets/Load3dViewerContent-COjSh8EF.js (removed) 1.46 kB 🟢 -1.46 kB 🟢 -661 B 🟢 -590 B
assets/SubscriptionPanelContentWorkspace-CgWlUm9P.js (new) 1.35 kB 🔴 +1.35 kB 🔴 +618 B 🔴 +534 B
assets/SubscriptionPanelContentWorkspace-g3amgFsD.js (removed) 1.35 kB 🟢 -1.35 kB 🟢 -613 B 🟢 -533 B
assets/WidgetLegacy-DSA9glJ3.js (removed) 1.18 kB 🟢 -1.18 kB 🟢 -562 B 🟢 -499 B
assets/WidgetLegacy-UmwMtubq.js (new) 1.18 kB 🔴 +1.18 kB 🔴 +564 B 🔴 +495 B
assets/changeTracker--SBicO84.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -549 B 🟢 -485 B
assets/changeTracker-BGnuQqCn.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +552 B 🔴 +484 B

Status: 57 added / 57 removed / 79 unchanged

⚡ Performance Report

canvas-idle: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.9 MB heap
canvas-mouse-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 45.1 MB heap
canvas-zoom-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 46.3 MB heap
dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 54.8 MB heap
large-graph-idle: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 64.6 MB heap
large-graph-pan: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 60.8 MB heap
large-graph-zoom: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 66.5 MB heap
minimap-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 65.4 MB heap
subgraph-dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 55.1 MB heap
subgraph-idle: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 67.1 MB heap
subgraph-mouse-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 59.7 MB heap
viewport-pan-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 75.3 MB heap
vue-large-graph-idle: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 164.5 MB heap
vue-large-graph-pan: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 1ms TBT · 163.3 MB heap
workflow-execution: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 53.7 MB heap

No regressions detected.

All metrics
Metric Baseline PR (median) Δ Sig
canvas-idle: avg frame time 17ms 17ms -0% z=-0.9
canvas-idle: p95 frame time 17ms 17ms +0%
canvas-idle: layout duration 0ms 0ms +0%
canvas-idle: style recalc duration 9ms 9ms -2% z=-2.0
canvas-idle: layout count 0 0 +0%
canvas-idle: style recalc count 10 9 -10% z=-3.8
canvas-idle: task duration 384ms 369ms -4% z=-0.8
canvas-idle: script duration 20ms 15ms -24% z=-4.6
canvas-idle: TBT 0ms 0ms +0%
canvas-idle: heap used 68.1 MB 68.9 MB +1%
canvas-idle: DOM nodes 20 18 -10% z=-3.6
canvas-idle: event listeners 6 4 -33% z=-1.6
canvas-mouse-sweep: avg frame time 17ms 17ms +0% z=-0.4
canvas-mouse-sweep: p95 frame time 17ms 17ms +1%
canvas-mouse-sweep: layout duration 4ms 4ms +4% z=0.7
canvas-mouse-sweep: style recalc duration 39ms 39ms +1% z=-1.1
canvas-mouse-sweep: layout count 12 12 +0%
canvas-mouse-sweep: style recalc count 75 74 -1% z=-2.0
canvas-mouse-sweep: task duration 817ms 794ms -3% z=-1.2
canvas-mouse-sweep: script duration 120ms 126ms +5% z=-1.4
canvas-mouse-sweep: TBT 0ms 0ms +0%
canvas-mouse-sweep: heap used 52.9 MB 45.1 MB -15%
canvas-mouse-sweep: DOM nodes -264 -263 -0% z=-125.8
canvas-mouse-sweep: event listeners -131 -131 +0% z=-33.5
canvas-zoom-sweep: avg frame time 17ms 17ms +0% z=0.5
canvas-zoom-sweep: p95 frame time 17ms 17ms +1%
canvas-zoom-sweep: layout duration 1ms 1ms -4% z=-0.0
canvas-zoom-sweep: style recalc duration 18ms 17ms -8% z=-1.4
canvas-zoom-sweep: layout count 6 6 +0%
canvas-zoom-sweep: style recalc count 32 31 -3% z=-0.6
canvas-zoom-sweep: task duration 315ms 290ms -8% z=-1.6
canvas-zoom-sweep: script duration 25ms 18ms -30% z=-3.1
canvas-zoom-sweep: TBT 0ms 0ms +0%
canvas-zoom-sweep: heap used 47.2 MB 46.3 MB -2%
canvas-zoom-sweep: DOM nodes 80 76 -5% z=-4.1
canvas-zoom-sweep: event listeners 21 19 -10% z=-0.9
dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
dom-widget-clipping: p95 frame time 17ms 17ms +0%
dom-widget-clipping: layout duration 0ms 0ms +0%
dom-widget-clipping: style recalc duration 7ms 8ms +15% z=-2.2
dom-widget-clipping: layout count 0 0 +0%
dom-widget-clipping: style recalc count 10 12 +20% z=-2.2
dom-widget-clipping: task duration 332ms 327ms -1% z=-2.3
dom-widget-clipping: script duration 55ms 60ms +9% z=-2.5
dom-widget-clipping: TBT 0ms 0ms +0%
dom-widget-clipping: heap used 54.5 MB 54.8 MB +0%
dom-widget-clipping: DOM nodes 16 19 +19% z=-2.2
dom-widget-clipping: event listeners 2 0 -100% variance too high
large-graph-idle: avg frame time 17ms 17ms -0% z=-1.0
large-graph-idle: p95 frame time 17ms 17ms +0%
large-graph-idle: layout duration 0ms 0ms +0%
large-graph-idle: style recalc duration 8ms 8ms -1% z=-3.9
large-graph-idle: layout count 0 0 +0%
large-graph-idle: style recalc count 9 9 +0% z=-8.3
large-graph-idle: task duration 571ms 544ms -5% z=0.0
large-graph-idle: script duration 94ms 92ms -2% z=-1.0
large-graph-idle: TBT 0ms 0ms +0%
large-graph-idle: heap used 58.9 MB 64.6 MB +10%
large-graph-idle: DOM nodes -260 -262 +1% z=-317.2
large-graph-idle: event listeners -129 -131 +2% z=-25.8
large-graph-pan: avg frame time 17ms 17ms -0% z=-0.8
large-graph-pan: p95 frame time 17ms 17ms +1%
large-graph-pan: layout duration 0ms 0ms +0%
large-graph-pan: style recalc duration 18ms 17ms -7% z=-0.3
large-graph-pan: layout count 0 0 +0%
large-graph-pan: style recalc count 69 68 -1% z=-2.4
large-graph-pan: task duration 1144ms 1056ms -8% z=-0.6
large-graph-pan: script duration 400ms 401ms +0% z=-0.3
large-graph-pan: TBT 0ms 0ms +0%
large-graph-pan: heap used 59.4 MB 60.8 MB +2%
large-graph-pan: DOM nodes -266 -264 -1% z=-171.5
large-graph-pan: event listeners -127 -127 +0% z=-159.3
large-graph-zoom: avg frame time 17ms 17ms +0%
large-graph-zoom: p95 frame time 17ms 17ms +0%
large-graph-zoom: layout duration 8ms 7ms -8%
large-graph-zoom: style recalc duration 19ms 17ms -9%
large-graph-zoom: layout count 60 60 +0%
large-graph-zoom: style recalc count 65 65 +0%
large-graph-zoom: task duration 1314ms 1306ms -1%
large-graph-zoom: script duration 464ms 506ms +9%
large-graph-zoom: TBT 0ms 0ms +0%
large-graph-zoom: heap used 72.4 MB 66.5 MB -8%
large-graph-zoom: DOM nodes -266 -268 +1%
large-graph-zoom: event listeners -125 -125 +0%
minimap-idle: avg frame time 17ms 17ms +0% z=0.1
minimap-idle: p95 frame time 17ms 17ms -1%
minimap-idle: layout duration 0ms 0ms +0%
minimap-idle: style recalc duration 9ms 8ms -16% z=-2.4
minimap-idle: layout count 0 0 +0%
minimap-idle: style recalc count 10 8 -20% z=-2.3
minimap-idle: task duration 593ms 531ms -10% z=0.1
minimap-idle: script duration 96ms 93ms -3% z=-0.5
minimap-idle: TBT 0ms 0ms +0%
minimap-idle: heap used 62.4 MB 65.4 MB +5%
minimap-idle: DOM nodes -262 -265 +1% z=-207.7
minimap-idle: event listeners -129 -129 +0% z=-202.3
subgraph-dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
subgraph-dom-widget-clipping: p95 frame time 17ms 17ms -1%
subgraph-dom-widget-clipping: layout duration 0ms 0ms +0%
subgraph-dom-widget-clipping: style recalc duration 10ms 10ms +0% z=-2.4
subgraph-dom-widget-clipping: layout count 0 0 +0%
subgraph-dom-widget-clipping: style recalc count 46 46 +0% z=-3.3
subgraph-dom-widget-clipping: task duration 343ms 347ms +1% z=-1.7
subgraph-dom-widget-clipping: script duration 114ms 120ms +5% z=-1.4
subgraph-dom-widget-clipping: TBT 0ms 0ms +0%
subgraph-dom-widget-clipping: heap used 55.1 MB 55.1 MB -0%
subgraph-dom-widget-clipping: DOM nodes 18 18 +0% z=-3.7
subgraph-dom-widget-clipping: event listeners 8 6 -25% z=-1.7
subgraph-idle: avg frame time 17ms 17ms +0% z=0.4
subgraph-idle: p95 frame time 17ms 17ms +1%
subgraph-idle: layout duration 0ms 0ms +0%
subgraph-idle: style recalc duration 8ms 8ms +1% z=-2.8
subgraph-idle: layout count 0 0 +0%
subgraph-idle: style recalc count 9 9 +0% z=-2.9
subgraph-idle: task duration 382ms 351ms -8% z=-0.6
subgraph-idle: script duration 18ms 13ms -25% z=-2.6
subgraph-idle: TBT 0ms 0ms +0%
subgraph-idle: heap used 67.9 MB 67.1 MB -1%
subgraph-idle: DOM nodes 18 16 -11% z=-3.9
subgraph-idle: event listeners 6 4 -33% variance too high
subgraph-mouse-sweep: avg frame time 17ms 17ms -0% z=0.4
subgraph-mouse-sweep: p95 frame time 17ms 17ms -0%
subgraph-mouse-sweep: layout duration 5ms 4ms -9% z=-1.6
subgraph-mouse-sweep: style recalc duration 42ms 35ms -16% z=-2.2
subgraph-mouse-sweep: layout count 16 16 +0%
subgraph-mouse-sweep: style recalc count 77 75 -3% z=-2.7
subgraph-mouse-sweep: task duration 708ms 654ms -8% z=-1.6
subgraph-mouse-sweep: script duration 86ms 91ms +6% z=-1.5
subgraph-mouse-sweep: TBT 0ms 0ms +0%
subgraph-mouse-sweep: heap used 59.8 MB 59.7 MB -0%
subgraph-mouse-sweep: DOM nodes 61 60 -2% z=-3.1
subgraph-mouse-sweep: event listeners 6 4 -33% variance too high
viewport-pan-sweep: avg frame time 17ms 17ms +0%
viewport-pan-sweep: p95 frame time 17ms 17ms -1%
viewport-pan-sweep: layout duration 0ms 0ms +0%
viewport-pan-sweep: style recalc duration 54ms 51ms -6%
viewport-pan-sweep: layout count 0 0 +0%
viewport-pan-sweep: style recalc count 250 250 +0%
viewport-pan-sweep: task duration 3805ms 3612ms -5%
viewport-pan-sweep: script duration 1217ms 1276ms +5%
viewport-pan-sweep: TBT 0ms 0ms +0%
viewport-pan-sweep: heap used 65.2 MB 75.3 MB +16%
viewport-pan-sweep: DOM nodes -263 -260 -1%
viewport-pan-sweep: event listeners -113 -113 +0%
vue-large-graph-idle: avg frame time 17ms 17ms +3%
vue-large-graph-idle: p95 frame time 17ms 17ms +1%
vue-large-graph-idle: layout duration 0ms 0ms +0%
vue-large-graph-idle: style recalc duration 0ms 0ms +0%
vue-large-graph-idle: layout count 0 0 +0%
vue-large-graph-idle: style recalc count 0 0 +0%
vue-large-graph-idle: task duration 10817ms 11322ms +5%
vue-large-graph-idle: script duration 575ms 624ms +8%
vue-large-graph-idle: TBT 0ms 0ms +0%
vue-large-graph-idle: heap used 156.5 MB 164.5 MB +5%
vue-large-graph-idle: DOM nodes -8331 -8331 +0%
vue-large-graph-idle: event listeners -16466 -16468 +0%
vue-large-graph-pan: avg frame time 17ms 17ms +0%
vue-large-graph-pan: p95 frame time 17ms 17ms +0%
vue-large-graph-pan: layout duration 0ms 0ms +0%
vue-large-graph-pan: style recalc duration 18ms 16ms -12%
vue-large-graph-pan: layout count 0 0 +0%
vue-large-graph-pan: style recalc count 66 67 +2%
vue-large-graph-pan: task duration 12732ms 14219ms +12%
vue-large-graph-pan: script duration 816ms 907ms +11%
vue-large-graph-pan: TBT 0ms 1ms
vue-large-graph-pan: heap used 157.7 MB 163.3 MB +4%
vue-large-graph-pan: DOM nodes -8331 -8331 +0%
vue-large-graph-pan: event listeners -16462 -16488 +0%
workflow-execution: avg frame time 17ms 17ms -0% z=-0.4
workflow-execution: p95 frame time 17ms 17ms +0%
workflow-execution: layout duration 1ms 2ms +13% z=0.1
workflow-execution: style recalc duration 25ms 22ms -11% z=-0.8
workflow-execution: layout count 5 5 +0% z=0.1
workflow-execution: style recalc count 20 16 -20% z=-0.9
workflow-execution: task duration 126ms 116ms -7% z=-0.6
workflow-execution: script duration 26ms 23ms -10% z=-2.0
workflow-execution: TBT 0ms 0ms +0%
workflow-execution: heap used 52.8 MB 53.7 MB +2%
workflow-execution: DOM nodes 159 155 -3% z=-0.8
workflow-execution: event listeners 69 69 +0% z=3.9
Historical variance (last 15 runs)
Metric μ σ CV
canvas-idle: avg frame time 17ms 0ms 0.0%
canvas-idle: layout duration 0ms 0ms 0.0%
canvas-idle: style recalc duration 11ms 1ms 8.2%
canvas-idle: layout count 0 0 0.0%
canvas-idle: style recalc count 11 1 5.0%
canvas-idle: task duration 395ms 31ms 7.9%
canvas-idle: script duration 25ms 2ms 8.8%
canvas-idle: TBT 0ms 0ms 0.0%
canvas-idle: DOM nodes 23 1 5.6%
canvas-idle: event listeners 12 5 40.9%
canvas-mouse-sweep: avg frame time 17ms 0ms 0.0%
canvas-mouse-sweep: layout duration 4ms 0ms 5.4%
canvas-mouse-sweep: style recalc duration 43ms 3ms 7.4%
canvas-mouse-sweep: layout count 12 0 0.0%
canvas-mouse-sweep: style recalc count 79 2 3.0%
canvas-mouse-sweep: task duration 865ms 58ms 6.7%
canvas-mouse-sweep: script duration 136ms 6ms 4.8%
canvas-mouse-sweep: TBT 0ms 0ms 0.0%
canvas-mouse-sweep: DOM nodes 62 3 4.2%
canvas-mouse-sweep: event listeners 8 4 49.4%
canvas-zoom-sweep: avg frame time 17ms 0ms 0.0%
canvas-zoom-sweep: layout duration 1ms 0ms 7.0%
canvas-zoom-sweep: style recalc duration 19ms 2ms 8.0%
canvas-zoom-sweep: layout count 6 0 0.0%
canvas-zoom-sweep: style recalc count 31 0 1.5%
canvas-zoom-sweep: task duration 327ms 23ms 7.1%
canvas-zoom-sweep: script duration 27ms 3ms 11.1%
canvas-zoom-sweep: TBT 0ms 0ms 0.0%
canvas-zoom-sweep: DOM nodes 79 1 1.0%
canvas-zoom-sweep: event listeners 24 5 21.8%
dom-widget-clipping: avg frame time 17ms 0ms 0.0%
dom-widget-clipping: layout duration 0ms 0ms 0.0%
dom-widget-clipping: style recalc duration 10ms 1ms 8.0%
dom-widget-clipping: layout count 0 0 0.0%
dom-widget-clipping: style recalc count 13 0 3.8%
dom-widget-clipping: task duration 365ms 16ms 4.5%
dom-widget-clipping: script duration 68ms 3ms 4.8%
dom-widget-clipping: TBT 0ms 0ms 0.0%
dom-widget-clipping: DOM nodes 22 1 6.4%
dom-widget-clipping: event listeners 8 6 81.2%
large-graph-idle: avg frame time 17ms 0ms 0.0%
large-graph-idle: layout duration 0ms 0ms 0.0%
large-graph-idle: style recalc duration 12ms 1ms 8.6%
large-graph-idle: layout count 0 0 0.0%
large-graph-idle: style recalc count 12 0 2.7%
large-graph-idle: task duration 542ms 54ms 10.0%
large-graph-idle: script duration 102ms 11ms 10.3%
large-graph-idle: TBT 0ms 0ms 0.0%
large-graph-idle: DOM nodes 25 1 3.7%
large-graph-idle: event listeners 26 6 23.2%
large-graph-pan: avg frame time 17ms 0ms 0.0%
large-graph-pan: layout duration 0ms 0ms 0.0%
large-graph-pan: style recalc duration 17ms 1ms 4.6%
large-graph-pan: layout count 0 0 0.0%
large-graph-pan: style recalc count 70 1 0.9%
large-graph-pan: task duration 1082ms 43ms 4.0%
large-graph-pan: script duration 408ms 20ms 4.8%
large-graph-pan: TBT 0ms 0ms 0.0%
large-graph-pan: DOM nodes 19 2 8.7%
large-graph-pan: event listeners 5 1 16.8%
minimap-idle: avg frame time 17ms 0ms 0.0%
minimap-idle: layout duration 0ms 0ms 0.0%
minimap-idle: style recalc duration 10ms 1ms 8.6%
minimap-idle: layout count 0 0 0.0%
minimap-idle: style recalc count 10 1 7.1%
minimap-idle: task duration 527ms 47ms 9.0%
minimap-idle: script duration 98ms 10ms 10.1%
minimap-idle: TBT 0ms 0ms 0.0%
minimap-idle: DOM nodes 19 1 7.1%
minimap-idle: event listeners 5 1 14.4%
subgraph-dom-widget-clipping: avg frame time 17ms 0ms 0.0%
subgraph-dom-widget-clipping: layout duration 0ms 0ms 0.0%
subgraph-dom-widget-clipping: style recalc duration 13ms 1ms 7.4%
subgraph-dom-widget-clipping: layout count 0 0 0.0%
subgraph-dom-widget-clipping: style recalc count 48 1 1.2%
subgraph-dom-widget-clipping: task duration 378ms 18ms 4.9%
subgraph-dom-widget-clipping: script duration 128ms 6ms 4.9%
subgraph-dom-widget-clipping: TBT 0ms 0ms 0.0%
subgraph-dom-widget-clipping: DOM nodes 22 1 5.0%
subgraph-dom-widget-clipping: event listeners 16 6 36.0%
subgraph-idle: avg frame time 17ms 0ms 0.0%
subgraph-idle: layout duration 0ms 0ms 0.0%
subgraph-idle: style recalc duration 10ms 1ms 7.5%
subgraph-idle: layout count 0 0 0.0%
subgraph-idle: style recalc count 11 1 6.0%
subgraph-idle: task duration 370ms 31ms 8.5%
subgraph-idle: script duration 20ms 3ms 13.2%
subgraph-idle: TBT 0ms 0ms 0.0%
subgraph-idle: DOM nodes 22 1 6.9%
subgraph-idle: event listeners 10 7 64.5%
subgraph-mouse-sweep: avg frame time 17ms 0ms 0.0%
subgraph-mouse-sweep: layout duration 5ms 0ms 6.8%
subgraph-mouse-sweep: style recalc duration 42ms 3ms 7.8%
subgraph-mouse-sweep: layout count 16 0 0.0%
subgraph-mouse-sweep: style recalc count 80 2 2.4%
subgraph-mouse-sweep: task duration 766ms 69ms 9.0%
subgraph-mouse-sweep: script duration 101ms 7ms 6.5%
subgraph-mouse-sweep: TBT 0ms 0ms 0.0%
subgraph-mouse-sweep: DOM nodes 67 2 3.3%
subgraph-mouse-sweep: event listeners 8 4 52.6%
workflow-execution: avg frame time 17ms 0ms 0.0%
workflow-execution: layout duration 2ms 0ms 9.4%
workflow-execution: style recalc duration 24ms 2ms 9.1%
workflow-execution: layout count 5 1 11.0%
workflow-execution: style recalc count 18 2 11.5%
workflow-execution: task duration 123ms 11ms 8.8%
workflow-execution: script duration 29ms 3ms 10.2%
workflow-execution: TBT 0ms 0ms 0.0%
workflow-execution: DOM nodes 161 7 4.4%
workflow-execution: event listeners 52 4 8.4%
Trend (last 15 commits on main)
Metric Trend Dir Latest
canvas-idle: avg frame time ▆▃▆▁▆▃▆█▆▆▄▃▃▄▃ ➡️ 17ms
canvas-idle: p95 frame time ➡️ NaNms
canvas-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: style recalc duration ▇▇▆▆▃█▄▃▄▃▇▄▁▆▇ ➡️ 11ms
canvas-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
canvas-idle: style recalc count █▃▅▂▅▆▃▁▂▁▂▅▆▅▆ ➡️ 12
canvas-idle: task duration ▃▃▃▆▂▃▃▅▆▂█▃▁▃▃ ➡️ 391ms
canvas-idle: script duration ▄▃▅▇▂▅▃▆▇▅█▄▁▅▆ ➡️ 27ms
canvas-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: heap used ➡️ NaN MB
canvas-idle: DOM nodes █▇▆▅▃▇▃▁▂▂▅▆▆▆▇ ➡️ 24
canvas-idle: event listeners ▅█▅▄▁▅▁▁▁▄▅▅▁▅▄ 📉 11
canvas-mouse-sweep: avg frame time ▆█▆▃▁▃▁▆▆▁▃▆▆▃▃ ➡️ 17ms
canvas-mouse-sweep: p95 frame time ➡️ NaNms
canvas-mouse-sweep: layout duration ▁▃▂▄▁▂▁▃▆▂█▇▆▄▃ ➡️ 4ms
canvas-mouse-sweep: style recalc duration ▄▄▂▄▁▂▃▃▅▄█▆▂▄▄ ➡️ 43ms
canvas-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 12
canvas-mouse-sweep: style recalc count █▅▄▃▂▂▁▄▄▅▆▅▂▇▄ ➡️ 79
canvas-mouse-sweep: task duration █▆▄▂▂▃▂▄▄▅█▆▁▆▄ ➡️ 868ms
canvas-mouse-sweep: script duration ▄▅▄▆▄▆▆▆▅▅█▆▁▅▆ ➡️ 139ms
canvas-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-mouse-sweep: heap used ➡️ NaN MB
canvas-mouse-sweep: DOM nodes █▅▃▃▁▂▂▃▂▄▆▅▃▅▅ ➡️ 64
canvas-mouse-sweep: event listeners █▁▁▁▁▁▇▁▁▁██▇▁█ 📈 13
canvas-zoom-sweep: avg frame time ▅▅█▄▅▁▁▁▅▁▁▅▄▅▁ ➡️ 17ms
canvas-zoom-sweep: p95 frame time ➡️ NaNms
canvas-zoom-sweep: layout duration ▆▅▅▄▁▁█▅▃▅▇▆▁▂▆ ➡️ 1ms
canvas-zoom-sweep: style recalc duration ▆▅▄▆▅▃█▆▇▅▇▄▁▃▅ ➡️ 20ms
canvas-zoom-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 6
canvas-zoom-sweep: style recalc count ▁▁▃▄▆▃▆█▄▄▆▁▆▁▆ ➡️ 32
canvas-zoom-sweep: task duration ▄▂▁▇▂▂▄▅▆▃█▄▁▁▅ ➡️ 338ms
canvas-zoom-sweep: script duration ▃▃▂▇▂▂▅▇▆▅█▄▁▂▆ ➡️ 30ms
canvas-zoom-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-zoom-sweep: heap used ➡️ NaN MB
canvas-zoom-sweep: DOM nodes ▄▃▁▅█▁▃▆▄▅▅▃▃▄▃ ➡️ 79
canvas-zoom-sweep: event listeners ▁▁▂▅█▂▁▅▁▅▅▄▁▅▁ ➡️ 19
dom-widget-clipping: avg frame time ▂▄▅▅▂▄█▇▅▇▇▅▅▁▇ ➡️ 17ms
dom-widget-clipping: p95 frame time ➡️ NaNms
dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: style recalc duration ▆▆▂▆▄▃██▄▁▆▇▆▃▅ ➡️ 10ms
dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
dom-widget-clipping: style recalc count ▇█▅█▅▄█▇▇▁▇▄▇▂▅ ➡️ 13
dom-widget-clipping: task duration ▃▃▁▅▄▃▅▆▅▂▇█▁▅▅ ➡️ 371ms
dom-widget-clipping: script duration ▅▄▄▆▆▅▇▇▆▃█▇▁▇▇ ➡️ 71ms
dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: heap used ➡️ NaN MB
dom-widget-clipping: DOM nodes ▇▇▄▇▅▄█▇▅▁▅▄▇▃▄ ➡️ 21
dom-widget-clipping: event listeners ▅▅▅▅▁▅██▁▁▁▁█▁▁ 📉 2
large-graph-idle: avg frame time ▅▅▅▅▅▂▁▂▄▅▄▂▂▅█ ➡️ 17ms
large-graph-idle: p95 frame time ➡️ NaNms
large-graph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: style recalc duration ▅▅▅▆▄▅▃▄▅▅▆█▁▄▆ ➡️ 13ms
large-graph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-idle: style recalc count █▆█▃▃▁▃▆▃▆▆▃▆██ ➡️ 12
large-graph-idle: task duration ▂▃▂▆▂▃▃▇▅▃██▁▂▅ ➡️ 569ms
large-graph-idle: script duration ▄▅▄▆▄▅▅▇▆▅█▆▁▃▆ ➡️ 110ms
large-graph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: heap used ➡️ NaN MB
large-graph-idle: DOM nodes ▆█▅▂▅▃▁▂▃▅▅▆▂▆▅ ➡️ 25
large-graph-idle: event listeners ███▇██▄▁▄▇▇█▂█▇ ➡️ 29
large-graph-pan: avg frame time ▆▃▃▆█▃▁█▆▆▆▆█▁▆ ➡️ 17ms
large-graph-pan: p95 frame time ➡️ NaNms
large-graph-pan: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: style recalc duration ▃▂▄▄▁▅▂▂▁▄▄█▃▁▂ ➡️ 17ms
large-graph-pan: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-pan: style recalc count ▆▃█▂▃▂▂▂▁▇▅▃█▆▃ ➡️ 69
large-graph-pan: task duration ▄▃▄▆▄▄▄▆▄▄█▆▁▂▅ ➡️ 1100ms
large-graph-pan: script duration ▅▄▅▆▆▅▄▆▄▅█▄▁▄▅ ➡️ 413ms
large-graph-pan: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: heap used ➡️ NaN MB
large-graph-pan: DOM nodes ▅▃▆▂▄▁▃▁▁▅▁▂█▅▂ ➡️ 18
large-graph-pan: event listeners █▆█▁▁▆▁▁▃▆▁▃██▃ ➡️ 5
minimap-idle: avg frame time ▃▆▆▃█▁█▆▆▃▃▆█▆█ ➡️ 17ms
minimap-idle: p95 frame time ➡️ NaNms
minimap-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: style recalc duration ▄█▁█▅▅█▅▅▃▅▁▁▄▆ ➡️ 10ms
minimap-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
minimap-idle: style recalc count ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 9
minimap-idle: task duration ▃▄▁▅▁▃▄▅▇▃█▅▁▁▅ ➡️ 547ms
minimap-idle: script duration ▄▆▃▇▃▅▆▆▇▅█▅▁▃▆ ➡️ 106ms
minimap-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: heap used ➡️ NaN MB
minimap-idle: DOM nodes ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 19
minimap-idle: event listeners ▃▃▆▁▁▁▃▁▁▆▁▃█▆▁ ➡️ 4
subgraph-dom-widget-clipping: avg frame time ▅▄▄▄▄▄█▄▄▄▃▁▆▃▃ ➡️ 17ms
subgraph-dom-widget-clipping: p95 frame time ➡️ NaNms
subgraph-dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: style recalc duration ▂▄▃▅▅▃▂▅▇▃▄█▁▄▆ ➡️ 14ms
subgraph-dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-dom-widget-clipping: style recalc count ▇█▆▃▆▃▁▆█▇▃▆▇█▅ ➡️ 48
subgraph-dom-widget-clipping: task duration ▂▃▃▆▅▅▂▅█▂▆█▁▂▇ ➡️ 398ms
subgraph-dom-widget-clipping: script duration ▃▃▃▄▅▅▂▄█▂▅▇▁▂▅ ➡️ 131ms
subgraph-dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: heap used ➡️ NaN MB
subgraph-dom-widget-clipping: DOM nodes ▅▇▅▂▅▂▁▅▅▅▁▇▅█▄ ➡️ 22
subgraph-dom-widget-clipping: event listeners ▅▅▅▂▅▁▅██▁▁█▅█▅ 📈 16
subgraph-idle: avg frame time ▆▆█▁▆▃▆▆▆▃▆▁▃▆█ ➡️ 17ms
subgraph-idle: p95 frame time ➡️ NaNms
subgraph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: style recalc duration ▁▇▃▆▂▄▂▃▃▆▆▄▃▇█ ➡️ 12ms
subgraph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-idle: style recalc count ▃▆▃▃▂▅▁▂▁▆▃▃██▇ ➡️ 12
subgraph-idle: task duration ▁▃▁▇▁▁▃▆▅▂█▅▁▁▄ ➡️ 378ms
subgraph-idle: script duration ▁▃▂▇▁▂▃▇▆▂█▅▂▁▅ ➡️ 22ms
subgraph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: heap used ➡️ NaN MB
subgraph-idle: DOM nodes ▃▅▃▂▁▄▁▂▁▅▃▂▇█▇ ➡️ 24
subgraph-idle: event listeners ▁▅▁▁▁▁▁▁▁▅▄▁███ 📈 21
subgraph-mouse-sweep: avg frame time ▅▄▁▃▃▄▆▄▆▃▃█▁▃▃ ➡️ 17ms
subgraph-mouse-sweep: p95 frame time ➡️ NaNms
subgraph-mouse-sweep: layout duration ▁▄▄▄▃▃▅▅▅▂█▇▂▃▆ ➡️ 5ms
subgraph-mouse-sweep: style recalc duration ▃▂▄▅▂▃▄▅█▃█▆▁▂▅ ➡️ 43ms
subgraph-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 16
subgraph-mouse-sweep: style recalc count ▅▂▅▅▁▄▃▅█▅▆▄▂▄▅ ➡️ 81
subgraph-mouse-sweep: task duration ▃▂▄▅▂▄▄▅▇▄█▆▁▃▅ ➡️ 785ms
subgraph-mouse-sweep: script duration ▄▅▄▇▅▅▆▇▆▅██▁▄▆ ➡️ 105ms
subgraph-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-mouse-sweep: heap used ➡️ NaN MB
subgraph-mouse-sweep: DOM nodes ▅▁▄▅▁▄▃▃█▅▅▄▂▅▃ ➡️ 66
subgraph-mouse-sweep: event listeners ▇▁▂▇▁▂▂▂█▇▂▂▇▇▂ 📈 5
workflow-execution: avg frame time ▆▆▆▄▆▆▃▄▁▄█▆▅▄▆ ➡️ 17ms
workflow-execution: p95 frame time ➡️ NaNms
workflow-execution: layout duration ▁▆▁▃▂▄▃▂▃▃▅█▄▂▅ ➡️ 2ms
workflow-execution: style recalc duration ▃▇▅▇▁▅▆▇█▁██▂▄▆ ➡️ 25ms
workflow-execution: layout count ▁█▂▃▂▃▃▁▃▃▄▃▂▃▂ ➡️ 5
workflow-execution: style recalc count ▃█▅▇▁▄▅▆▅▅▅▅▄▄▂ ➡️ 15
workflow-execution: task duration ▂▅▄▅▁▄▆▆▆▁▇█▁▃▃ ➡️ 120ms
workflow-execution: script duration ▄▃▄▄▃▅▄▅▆▂▇█▁▃▄ ➡️ 29ms
workflow-execution: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
workflow-execution: heap used ➡️ NaN MB
workflow-execution: DOM nodes ▂█▃▆▁▄▃▅▃█▃▃▄▃▁ ➡️ 152
workflow-execution: event listeners ▅███▁▅███▁██▅█▅ ➡️ 49
Raw data
{
  "timestamp": "2026-05-04T23:07:35.051Z",
  "gitSha": "3acdd9f187e84ba6a05d7598e28325dd66d1b393",
  "branch": "fix/dropdown-expands-on-run-keybind",
  "measurements": [
    {
      "name": "canvas-idle",
      "durationMs": 2017.0739999999796,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 7.9319999999999995,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 369.468,
      "heapDeltaBytes": 22717124,
      "heapUsedBytes": 72220300,
      "domNodes": 16,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 13.824999999999998,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-idle",
      "durationMs": 2045.0250000000096,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 9.764,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 380.247,
      "heapDeltaBytes": 23885056,
      "heapUsedBytes": 72803848,
      "domNodes": 22,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 23.556,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "canvas-idle",
      "durationMs": 2001.0090000000673,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.996,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 347.433,
      "heapDeltaBytes": 22923152,
      "heapUsedBytes": 71633260,
      "domNodes": 18,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 14.927000000000001,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1927.3529999999823,
      "styleRecalcs": 75,
      "styleRecalcDurationMs": 43.142,
      "layouts": 12,
      "layoutDurationMs": 3.7409999999999997,
      "taskDurationMs": 821.033,
      "heapDeltaBytes": -3061064,
      "heapUsedBytes": 45782484,
      "domNodes": -263,
      "jsHeapTotalBytes": 15855616,
      "scriptDurationMs": 133.594,
      "eventListeners": -131,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1885.9939999999824,
      "styleRecalcs": 73,
      "styleRecalcDurationMs": 39.072,
      "layouts": 12,
      "layoutDurationMs": 4.08,
      "taskDurationMs": 793.667,
      "heapDeltaBytes": -694328,
      "heapUsedBytes": 47287564,
      "domNodes": -265,
      "jsHeapTotalBytes": 16117760,
      "scriptDurationMs": 124.299,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1790.5379999999695,
      "styleRecalcs": 74,
      "styleRecalcDurationMs": 36.418,
      "layouts": 12,
      "layoutDurationMs": 3.42,
      "taskDurationMs": 742.61,
      "heapDeltaBytes": -1200832,
      "heapUsedBytes": 66674896,
      "domNodes": 58,
      "jsHeapTotalBytes": 19660800,
      "scriptDurationMs": 126.25799999999998,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1711.7080000000442,
      "styleRecalcs": 30,
      "styleRecalcDurationMs": 16.994000000000003,
      "layouts": 6,
      "layoutDurationMs": 0.72,
      "taskDurationMs": 290.219,
      "heapDeltaBytes": 219556,
      "heapUsedBytes": 48482616,
      "domNodes": 74,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 17.804,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1720.6429999999955,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 17.927000000000003,
      "layouts": 6,
      "layoutDurationMs": 0.6410000000000001,
      "taskDurationMs": 290.873,
      "heapDeltaBytes": 218416,
      "heapUsedBytes": 48751984,
      "domNodes": 79,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 17.941999999999993,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1717.4989999999752,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 16.029999999999998,
      "layouts": 6,
      "layoutDurationMs": 0.518,
      "taskDurationMs": 286.39599999999996,
      "heapDeltaBytes": 170168,
      "heapUsedBytes": 48524424,
      "domNodes": 76,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 17.735999999999994,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 534.7939999999767,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 6.959999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 327.429,
      "heapDeltaBytes": 9141588,
      "heapUsedBytes": 57463032,
      "domNodes": 16,
      "jsHeapTotalBytes": 15728640,
      "scriptDurationMs": 59.81800000000001,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 539.7720000000845,
      "styleRecalcs": 13,
      "styleRecalcDurationMs": 9.628999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 332.61,
      "heapDeltaBytes": 9169824,
      "heapUsedBytes": 56929384,
      "domNodes": 21,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 63.237,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 530.9919999999693,
      "styleRecalcs": 12,
      "styleRecalcDurationMs": 8.126999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 320.689,
      "heapDeltaBytes": 9135924,
      "heapUsedBytes": 57426908,
      "domNodes": 19,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 58.455000000000005,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2013.2060000000251,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 7.870000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 527.4989999999999,
      "heapDeltaBytes": 8260064,
      "heapUsedBytes": 67729324,
      "domNodes": -262,
      "jsHeapTotalBytes": 3756032,
      "scriptDurationMs": 90.06900000000002,
      "eventListeners": -131,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2012.5860000000557,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.328999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 545.6650000000001,
      "heapDeltaBytes": 16027696,
      "heapUsedBytes": 75502152,
      "domNodes": -264,
      "jsHeapTotalBytes": 3756032,
      "scriptDurationMs": 91.93099999999998,
      "eventListeners": -131,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2054.2800000000625,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.183,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 544.366,
      "heapDeltaBytes": 9320348,
      "heapUsedBytes": 67517068,
      "domNodes": -262,
      "jsHeapTotalBytes": 1077248,
      "scriptDurationMs": 94.992,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2119.121000000007,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 17.407000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1066.9430000000002,
      "heapDeltaBytes": 9719396,
      "heapUsedBytes": 69170744,
      "domNodes": -260,
      "jsHeapTotalBytes": -815104,
      "scriptDurationMs": 401.448,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2129.6350000000075,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 16.868000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1055.9650000000001,
      "heapDeltaBytes": 2344568,
      "heapUsedBytes": 61473296,
      "domNodes": -264,
      "jsHeapTotalBytes": 1282048,
      "scriptDurationMs": 403.374,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2112.7219999999625,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 17.041,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1056.3890000000001,
      "heapDeltaBytes": 4409096,
      "heapUsedBytes": 63707112,
      "domNodes": -264,
      "jsHeapTotalBytes": 757760,
      "scriptDurationMs": 399.578,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3149.10100000003,
      "styleRecalcs": 64,
      "styleRecalcDurationMs": 16.199,
      "layouts": 60,
      "layoutDurationMs": 6.979000000000001,
      "taskDurationMs": 1310.2720000000002,
      "heapDeltaBytes": 8615640,
      "heapUsedBytes": 69689804,
      "domNodes": -272,
      "jsHeapTotalBytes": 4485120,
      "scriptDurationMs": 506.071,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3131.603000000041,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 18.784,
      "layouts": 60,
      "layoutDurationMs": 7.2620000000000005,
      "taskDurationMs": 1300.694,
      "heapDeltaBytes": -7973072,
      "heapUsedBytes": 60333296,
      "domNodes": 16,
      "jsHeapTotalBytes": 14299136,
      "scriptDurationMs": 527.1129999999999,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3181.6690000000563,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 17.139,
      "layouts": 60,
      "layoutDurationMs": 6.960000000000001,
      "taskDurationMs": 1306.076,
      "heapDeltaBytes": 8925052,
      "heapUsedBytes": 69786680,
      "domNodes": -268,
      "jsHeapTotalBytes": 815104,
      "scriptDurationMs": 497.58399999999995,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "minimap-idle",
      "durationMs": 2051.1960000000045,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 7.572000000000003,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 540.022,
      "heapDeltaBytes": 14050888,
      "heapUsedBytes": 73887840,
      "domNodes": -263,
      "jsHeapTotalBytes": 290816,
      "scriptDurationMs": 96.15499999999999,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2019.8030000000244,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.421000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 526.3739999999999,
      "heapDeltaBytes": 4295408,
      "heapUsedBytes": 64193588,
      "domNodes": -265,
      "jsHeapTotalBytes": 552960,
      "scriptDurationMs": 91.364,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2019.8560000000043,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 7.160000000000003,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 531.0989999999999,
      "heapDeltaBytes": 8285828,
      "heapUsedBytes": 68608004,
      "domNodes": -267,
      "jsHeapTotalBytes": -233472,
      "scriptDurationMs": 92.82400000000001,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 525.0169999999912,
      "styleRecalcs": 46,
      "styleRecalcDurationMs": 10.403,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 335.691,
      "heapDeltaBytes": 9124868,
      "heapUsedBytes": 57752784,
      "domNodes": 18,
      "jsHeapTotalBytes": 15728640,
      "scriptDurationMs": 113.68299999999998,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 554.4290000000274,
      "styleRecalcs": 47,
      "styleRecalcDurationMs": 11.669999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 372.07,
      "heapDeltaBytes": -11666880,
      "heapUsedBytes": 49763568,
      "domNodes": 20,
      "jsHeapTotalBytes": 25165824,
      "scriptDurationMs": 128.59,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 539.3309999999474,
      "styleRecalcs": 46,
      "styleRecalcDurationMs": 9.671000000000003,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 346.723,
      "heapDeltaBytes": 9438624,
      "heapUsedBytes": 57761212,
      "domNodes": 18,
      "jsHeapTotalBytes": 15990784,
      "scriptDurationMs": 119.60900000000001,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-idle",
      "durationMs": 1997.829999999965,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 10.236999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 350.626,
      "heapDeltaBytes": 22436956,
      "heapUsedBytes": 70391972,
      "domNodes": 22,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 16.656000000000006,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2003.9229999999861,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.209,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 384.994,
      "heapDeltaBytes": -3460040,
      "heapUsedBytes": 45188064,
      "domNodes": -262,
      "jsHeapTotalBytes": 15069184,
      "scriptDurationMs": 13.305000000000005,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2003.1149999999798,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 6.930000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 333.843,
      "heapDeltaBytes": 22495992,
      "heapUsedBytes": 71241844,
      "domNodes": 16,
      "jsHeapTotalBytes": 14417920,
      "scriptDurationMs": 12.313,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1706.165999999996,
      "styleRecalcs": 75,
      "styleRecalcDurationMs": 34.325,
      "layouts": 16,
      "layoutDurationMs": 4.412,
      "taskDurationMs": 684.745,
      "heapDeltaBytes": 645484,
      "heapUsedBytes": 49242024,
      "domNodes": -261,
      "jsHeapTotalBytes": 15593472,
      "scriptDurationMs": 90.72300000000001,
      "eventListeners": -131,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1705.885999999964,
      "styleRecalcs": 75,
      "styleRecalcDurationMs": 35.224,
      "layouts": 16,
      "layoutDurationMs": 4.1499999999999995,
      "taskDurationMs": 653.882,
      "heapDeltaBytes": 14034236,
      "heapUsedBytes": 62622132,
      "domNodes": 60,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 89.84899999999999,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66999999999998,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1701.4820000000555,
      "styleRecalcs": 75,
      "styleRecalcDurationMs": 35.943,
      "layouts": 16,
      "layoutDurationMs": 4.192,
      "taskDurationMs": 651.982,
      "heapDeltaBytes": 14272212,
      "heapUsedBytes": 62985084,
      "domNodes": 60,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 93.174,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8148.286999999982,
      "styleRecalcs": 249,
      "styleRecalcDurationMs": 50.50099999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3585.348,
      "heapDeltaBytes": 17221624,
      "heapUsedBytes": 76037724,
      "domNodes": -260,
      "jsHeapTotalBytes": 5505024,
      "scriptDurationMs": 1271.5159999999998,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8109.933999999953,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 51.50000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3611.8929999999996,
      "heapDeltaBytes": 20950524,
      "heapUsedBytes": 79255148,
      "domNodes": -255,
      "jsHeapTotalBytes": 3117056,
      "scriptDurationMs": 1275.607,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8172.759999999926,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 50.675,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3645.806,
      "heapDeltaBytes": 19480900,
      "heapUsedBytes": 78956592,
      "domNodes": -260,
      "jsHeapTotalBytes": 6029312,
      "scriptDurationMs": 1287.7669999999998,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 11382.724999999993,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 11371.48,
      "heapDeltaBytes": -36778608,
      "heapUsedBytes": 164343856,
      "domNodes": -8331,
      "jsHeapTotalBytes": 15527936,
      "scriptDurationMs": 632.83,
      "eventListeners": -16468,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333237,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 11106.992999999988,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 11085.293,
      "heapDeltaBytes": -25100032,
      "heapUsedBytes": 172477568,
      "domNodes": -8331,
      "jsHeapTotalBytes": 24965120,
      "scriptDurationMs": 593.2249999999999,
      "eventListeners": -16465,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333237,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 11334.98099999997,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 11322.097,
      "heapDeltaBytes": -25848508,
      "heapUsedBytes": 173021344,
      "domNodes": -8331,
      "jsHeapTotalBytes": 24702976,
      "scriptDurationMs": 623.779,
      "eventListeners": -16468,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.220000000000073,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14169.07699999996,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 15.307000000000015,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14145.868,
      "heapDeltaBytes": -42258024,
      "heapUsedBytes": 153200256,
      "domNodes": -8331,
      "jsHeapTotalBytes": -2297856,
      "scriptDurationMs": 907.2280000000001,
      "eventListeners": -16488,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333358,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14255.45999999997,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 15.52199999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14234.836,
      "heapDeltaBytes": -23851660,
      "heapUsedBytes": 174602228,
      "domNodes": -8329,
      "jsHeapTotalBytes": -2383872,
      "scriptDurationMs": 889.3100000000001,
      "eventListeners": -16490,
      "totalBlockingTimeMs": 28,
      "frameDurationMs": 17.220000000000073,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14238.238000000023,
      "styleRecalcs": 67,
      "styleRecalcDurationMs": 16.30799999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14218.717,
      "heapDeltaBytes": -33377272,
      "heapUsedBytes": 171207916,
      "domNodes": -8331,
      "jsHeapTotalBytes": -4743168,
      "scriptDurationMs": 961.2160000000001,
      "eventListeners": -16488,
      "totalBlockingTimeMs": 1,
      "frameDurationMs": 16.66333333333338,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "workflow-execution",
      "durationMs": 464.76400000005924,
      "styleRecalcs": 20,
      "styleRecalcDurationMs": 22.397999999999996,
      "layouts": 5,
      "layoutDurationMs": 1.6660000000000001,
      "taskDurationMs": 122.422,
      "heapDeltaBytes": 5385300,
      "heapUsedBytes": 55531260,
      "domNodes": 188,
      "jsHeapTotalBytes": 524288,
      "scriptDurationMs": 22.510999999999996,
      "eventListeners": 69,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "workflow-execution",
      "durationMs": 460.1639999999634,
      "styleRecalcs": 15,
      "styleRecalcDurationMs": 20.519999999999996,
      "layouts": 5,
      "layoutDurationMs": 1.2899999999999998,
      "taskDurationMs": 113.33100000000002,
      "heapDeltaBytes": 5028960,
      "heapUsedBytes": 56509796,
      "domNodes": 155,
      "jsHeapTotalBytes": 262144,
      "scriptDurationMs": 23.445,
      "eventListeners": 69,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "workflow-execution",
      "durationMs": 461.5329999999176,
      "styleRecalcs": 16,
      "styleRecalcDurationMs": 23.032000000000004,
      "layouts": 5,
      "layoutDurationMs": 1.5610000000000002,
      "taskDurationMs": 116.36899999999997,
      "heapDeltaBytes": 5034464,
      "heapUsedBytes": 56346004,
      "domNodes": 155,
      "jsHeapTotalBytes": 262144,
      "scriptDurationMs": 23.03,
      "eventListeners": 69,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.799999999999727
    }
  ]
}

Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick comments (2)
src/platform/keybindings/keybindingService.propagation.test.ts (2)

45-63: 💤 Low value

beforeEach re-mock of useDialogStore is redundant with the module-level mock.

vi.clearAllMocks() only resets call history — not mock return values — so the vi.mock('@/stores/dialogStore', ...) factory's return value already persists into each test. The vi.mocked(useDialogStore).mockReturnValue(...) block (lines 55–59) is therefore a no-op in the current setup.

🧹 Suggested simplification
  beforeEach(() => {
    vi.clearAllMocks()
    setActivePinia(createTestingPinia({ stubActions: false }))

    const commandStore = useCommandStore()
    commandStore.execute = vi.fn()

-   vi.mocked(useDialogStore).mockReturnValue({
-     dialogStack: []
-   } as Partial<ReturnType<typeof useDialogStore>> as ReturnType<
-     typeof useDialogStore
-   >)
-
    keybindingService = useKeybindingService()
    keybindingService.registerCoreKeybindings()
  })
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/platform/keybindings/keybindingService.propagation.test.ts` around lines
45 - 63, The test's beforeEach re-mocks useDialogStore unnecessarily: remove the
vi.mocked(useDialogStore).mockReturnValue(...) block inside beforeEach (the
lines setting dialogStack) because the module-level
vi.mock('@/stores/dialogStore', ...) already provides the mock return value and
persists across tests; keep vi.clearAllMocks(),
setActivePinia/createTestingPinia, commandStore.execute mock, and the
keybindingService initialization and registerCoreKeybindings calls, but delete
the redundant re-mock of useDialogStore.

65-83: Consider adding explicit documentation of keybinding assumptions in the test.

The test's correctness depends on three assumptions: (1) Ctrl+Enter is the core keybinding for Comfy.QueuePrompt, (2) the command has no targetElementId, and (3) F13 has no binding. These assumptions are valid (confirmed by src/platform/keybindings/defaults.ts), but documenting them explicitly within the test file—either as comments or assertion helpers—would make the test's dependencies clearer for future maintainers.

🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/platform/keybindings/keybindingService.propagation.test.ts` around lines
65 - 83, Add explicit assertions/comments in the test to document the three
implicit assumptions: that the command 'Comfy.QueuePrompt' is bound to
Ctrl+Enter, that the command returned by useCommandStore().execute (or the
command registry entry) has no targetElementId, and that 'F13' has no
keybinding; update the test around keybindingService.keybindHandler to either
(a) add short inline comments referencing these assumptions and file
defaults.ts, or (b) add quick assertions that look up the keybinding for
'Comfy.QueuePrompt' and assert it equals the expected Ctrl+Enter, assert the
command's targetElementId is undefined, and assert no binding exists for 'F13',
so future maintainers can see the prerequisites explicitly.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Nitpick comments:
In `@src/platform/keybindings/keybindingService.propagation.test.ts`:
- Around line 45-63: The test's beforeEach re-mocks useDialogStore
unnecessarily: remove the vi.mocked(useDialogStore).mockReturnValue(...) block
inside beforeEach (the lines setting dialogStack) because the module-level
vi.mock('@/stores/dialogStore', ...) already provides the mock return value and
persists across tests; keep vi.clearAllMocks(),
setActivePinia/createTestingPinia, commandStore.execute mock, and the
keybindingService initialization and registerCoreKeybindings calls, but delete
the redundant re-mock of useDialogStore.
- Around line 65-83: Add explicit assertions/comments in the test to document
the three implicit assumptions: that the command 'Comfy.QueuePrompt' is bound to
Ctrl+Enter, that the command returned by useCommandStore().execute (or the
command registry entry) has no targetElementId, and that 'F13' has no
keybinding; update the test around keybindingService.keybindHandler to either
(a) add short inline comments referencing these assumptions and file
defaults.ts, or (b) add quick assertions that look up the keybinding for
'Comfy.QueuePrompt' and assert it equals the expected Ctrl+Enter, assert the
command's targetElementId is undefined, and assert no binding exists for 'F13',
so future maintainers can see the prerequisites explicitly.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 702b1872-30f7-4e60-abbb-242f4b618ff1

📥 Commits

Reviewing files that changed from the base of the PR and between 0491836 and 426d8a6.

📒 Files selected for processing (3)
  • src/platform/keybindings/keybindingService.propagation.test.ts
  • src/platform/keybindings/keybindingService.ts
  • src/views/GraphView.vue

Comment thread src/platform/keybindings/keybindingService.ts Outdated
@codecov
Copy link
Copy Markdown

codecov Bot commented May 4, 2026

Codecov Report

❌ Patch coverage is 92.30769% with 1 line in your changes missing coverage. Please review.

Files with missing lines Patch % Lines
src/views/GraphView.vue 0.00% 1 Missing ⚠️
@@             Coverage Diff             @@
##             main   #11886       +/-   ##
===========================================
- Coverage   71.98%   56.32%   -15.67%     
===========================================
  Files        1493     1385      -108     
  Lines       89839    70906    -18933     
  Branches    25661    18861     -6800     
===========================================
- Hits        64672    39935    -24737     
- Misses      24255    30441     +6186     
+ Partials      912      530      -382     
Flag Coverage Δ
e2e ?
unit 56.32% <92.30%> (+0.23%) ⬆️

Flags with carried forward coverage won't be shown. Click here to find out more.

Files with missing lines Coverage Δ
src/composables/useCoreCommands.ts 26.46% <100.00%> (-27.06%) ⬇️
src/platform/keybindings/keybindingService.ts 78.20% <100.00%> (-13.69%) ⬇️
src/views/GraphView.vue 0.00% <0.00%> (-66.17%) ⬇️

... and 1001 files with indirect coverage changes

🚀 New features to boost your workflow:
  • ❄️ Test Analytics: Detect flaky tests, report on failures, and find test suite problems.
  • 📦 JS Bundle Analysis: Save yourself from yourself by tracking and limiting bundle sizes in JS merges.

Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick comments (2)
src/platform/keybindings/keybindingService.escape.test.ts (2)

103-120: ⚡ Quick win

Note: the two Escape guard paths are tested independently but there's no test that exercises them together via target mismatch.

The dialog-open guard (lines 110–120) blocks Comfy.Graph.ExitSubgraph even when the event originates from canvasContainer. That's one protection path. The other path — the targetElementId check blocking execution when the event originates from outside graph-canvas-container (e.g., a focused dialog element) — has no dedicated test case here. Consider adding a test where target is overridden to a non-canvas element and no dialog is open, asserting the command is not called. This would confirm the targetElementId mechanism on its own, independent of the dialog guard.

✅ Suggested additional test
+  it('should NOT execute Escape keybinding when event originates outside graph-canvas-container', async () => {
+    const outsideTarget = document.createElement('div')
+    outsideTarget.id = 'some-dialog-content'
+    document.body.appendChild(outsideTarget)
+
+    try {
+      const event = createKeyboardEvent('Escape', { target: outsideTarget })
+      await keybindingService.keybindHandler(event)
+
+      expect(mockCommandExecute).not.toHaveBeenCalled()
+    } finally {
+      outsideTarget.remove()
+    }
+  })
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/platform/keybindings/keybindingService.escape.test.ts` around lines 103 -
120, Add a new test that verifies the targetElementId guard independently:
create no dialog (do not push into useDialogStore().dialogStack), obtain
keybindingService via useKeybindingService(), create a keyboard event with
createKeyboardEvent('Escape') but override its target to an element whose id is
not 'graph-canvas-container' (simulate a focused dialog/control), call
keybindingService.keybindHandler(event) and assert mockCommandExecute was not
called for 'Comfy.Graph.ExitSubgraph'; this isolates the target-based blocking
path from the dialog-stack guard.

168-191: ⚡ Quick win

Implicit "no keybindings" assumption via fresh Pinia is fragile.

The test achieves "no keybinding matched" by calling setActivePinia(createPinia()) and then useKeybindingService() without registerCoreKeybindings(). If useKeybindingService() ever auto-registers core bindings internally, the Escape combo will now match canvasContainer (since targetElementId aligns), the command will fire, and the legacy modal code may not run — silently changing what the test proves. A more explicit approach would be to keep the existing pinia but not register any keybinding that matches a bare Escape key.

🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/platform/keybindings/keybindingService.escape.test.ts` around lines 168 -
191, The test currently relies on setActivePinia(createPinia()) to implicitly
avoid any Escape binding, which is fragile if useKeybindingService() later
auto-registers core bindings; instead, ensure no keybinding matches a bare
Escape explicitly: remove the createPinia call (keep the existing Pinia), call
useKeybindingService() to get keybindingService, and then either do not call
registerCoreKeybindings() or explicitly clear/unregister any
Escape/canvasContainer binding on keybindingService before invoking
keybindHandler(event) so the legacy modal-close branch runs deterministically.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Nitpick comments:
In `@src/platform/keybindings/keybindingService.escape.test.ts`:
- Around line 103-120: Add a new test that verifies the targetElementId guard
independently: create no dialog (do not push into useDialogStore().dialogStack),
obtain keybindingService via useKeybindingService(), create a keyboard event
with createKeyboardEvent('Escape') but override its target to an element whose
id is not 'graph-canvas-container' (simulate a focused dialog/control), call
keybindingService.keybindHandler(event) and assert mockCommandExecute was not
called for 'Comfy.Graph.ExitSubgraph'; this isolates the target-based blocking
path from the dialog-stack guard.
- Around line 168-191: The test currently relies on
setActivePinia(createPinia()) to implicitly avoid any Escape binding, which is
fragile if useKeybindingService() later auto-registers core bindings; instead,
ensure no keybinding matches a bare Escape explicitly: remove the createPinia
call (keep the existing Pinia), call useKeybindingService() to get
keybindingService, and then either do not call registerCoreKeybindings() or
explicitly clear/unregister any Escape/canvasContainer binding on
keybindingService before invoking keybindHandler(event) so the legacy
modal-close branch runs deterministically.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 3bc119f9-932f-4ce2-a3f6-43d8f3b6f330

📥 Commits

Reviewing files that changed from the base of the PR and between 426d8a6 and cbaaad7.

📒 Files selected for processing (2)
  • src/platform/keybindings/defaults.ts
  • src/platform/keybindings/keybindingService.escape.test.ts

Comment thread src/platform/keybindings/defaults.ts Outdated
@dosubot dosubot Bot added size:S This PR changes 10-29 lines, ignoring generated files. and removed size:XS This PR changes 0-9 lines, ignoring generated files. labels May 4, 2026
Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick comments (1)
src/views/GraphView.test.ts (1)

124-127: ⚡ Quick win

Share a single Pinia instance across setActivePinia and the render plugin.

createTestingPinia is called twice, producing two unrelated instances. The component's dependency injection resolves stores against the plugin-injected instance (line 141), while any direct store access in test code uses the setActivePinia instance (line 126). For the current test this is harmless, but any future test that pre-populates store state in the beforeEach and expects the component to observe it will fail silently.

♻️ Proposed refactor
+import type { TestingPinia } from '@pinia/testing'
 
+let pinia: TestingPinia
 
   beforeEach(() => {
     vi.clearAllMocks()
-    setActivePinia(createTestingPinia({ stubActions: true }))
+    pinia = createTestingPinia({ stubActions: true })
+    setActivePinia(pinia)
   })
     render(GraphView, {
       global: {
-        plugins: [createTestingPinia({ stubActions: true })],
+        plugins: [pinia],

Also applies to: 139-141

🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/views/GraphView.test.ts` around lines 124 - 127, Tests create two Pinia
instances causing stores to be different between setActivePinia and the render
plugin; instead call createTestingPinia once, assign it to a variable (e.g.,
const pinia = createTestingPinia({ stubActions: true })), pass that same pinia
to setActivePinia(pinia) in beforeEach and also pass it into the render call's
plugins option so the component and test code share the same Pinia instance
(update usages of setActivePinia, createTestingPinia, and the render invocation
accordingly).
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Nitpick comments:
In `@src/views/GraphView.test.ts`:
- Around line 124-127: Tests create two Pinia instances causing stores to be
different between setActivePinia and the render plugin; instead call
createTestingPinia once, assign it to a variable (e.g., const pinia =
createTestingPinia({ stubActions: true })), pass that same pinia to
setActivePinia(pinia) in beforeEach and also pass it into the render call's
plugins option so the component and test code share the same Pinia instance
(update usages of setActivePinia, createTestingPinia, and the render invocation
accordingly).

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: ca4b5cd7-1a52-43cd-80fc-9f189decd8bd

📥 Commits

Reviewing files that changed from the base of the PR and between cbaaad7 and 2cec81b.

📒 Files selected for processing (4)
  • src/composables/useCoreCommands.ts
  • src/platform/keybindings/keybindingService.propagation.test.ts
  • src/platform/keybindings/keybindingService.ts
  • src/views/GraphView.test.ts
🚧 Files skipped from review as they are similar to previous changes (1)
  • src/platform/keybindings/keybindingService.ts

Comment thread src/composables/useCoreCommands.ts Outdated
@kaili-yang kaili-yang force-pushed the fix/dropdown-expands-on-run-keybind branch from ca3eade to 35d171c Compare May 4, 2026 20:16
@dosubot dosubot Bot added size:XS This PR changes 0-9 lines, ignoring generated files. and removed size:S This PR changes 10-29 lines, ignoring generated files. labels May 4, 2026
Move keybindHandler to window capture phase and call stopPropagation()
on keybinding match so the event never reaches Reka UI dropdown triggers.
Add ghost-node guards in ExitSubgraph and DeleteSelectedItems so Escape
and Delete cancel ghost placement instead of bypassing the _ghostKeyHandler
that can no longer intercept in document capture.
@dosubot dosubot Bot added size:S This PR changes 10-29 lines, ignoring generated files. and removed size:XS This PR changes 0-9 lines, ignoring generated files. labels May 4, 2026
Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick comments (1)
src/composables/useCoreCommands.ts (1)

1134-1137: 💤 Low value

Guard is correct; consider extracting the repeated pattern.

The guard mirrors the one in Comfy.Canvas.DeleteSelectedItems for the same reason—Escape is now intercepted at the window-capture phase before _ghostKeyHandler can fire. The logic is sound.

That said, the same 3-line guard (check ghostNodeId, call finalizeGhostPlacement(true), early-return) now exists verbatim in two commands (differing only in whether the canvas is accessed via app.canvas or canvas). If this pattern spreads to additional commands, extracting it reduces drift risk.

♻️ Optional: extract a guard helper
+function cancelGhostPlacementIfActive(
+  canvas: ReturnType<typeof useCanvasStore>['getCanvas'] extends () => infer R ? R : never
+): boolean {
+  if (canvas.state.ghostNodeId == null) return false
+  canvas.finalizeGhostPlacement(true)
+  return true
+}

Then in each command:

-        if (canvas.state.ghostNodeId != null) {
-          canvas.finalizeGhostPlacement(true)
-          return
-        }
+        if (cancelGhostPlacementIfActive(canvas)) return
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@src/composables/useCoreCommands.ts` around lines 1134 - 1137, The repeated
3-line guard (checking canvas.state.ghostNodeId, calling
canvas.finalizeGhostPlacement(true), and early-return) appears in both the
current command and Comfy.Canvas.DeleteSelectedItems; extract this into a small
helper (e.g., a function like handleGhostEscape(canvas) or
finalizeGhostIfPresent(canvas)) and call it from each command instead of
duplicating the three lines; ensure the helper returns a boolean indicating that
it handled the ghost so callers can early-return, and update uses in the current
file (useCoreCommands.ts) and the Comfy.Canvas.DeleteSelectedItems command to
call the new helper.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Nitpick comments:
In `@src/composables/useCoreCommands.ts`:
- Around line 1134-1137: The repeated 3-line guard (checking
canvas.state.ghostNodeId, calling canvas.finalizeGhostPlacement(true), and
early-return) appears in both the current command and
Comfy.Canvas.DeleteSelectedItems; extract this into a small helper (e.g., a
function like handleGhostEscape(canvas) or finalizeGhostIfPresent(canvas)) and
call it from each command instead of duplicating the three lines; ensure the
helper returns a boolean indicating that it handled the ghost so callers can
early-return, and update uses in the current file (useCoreCommands.ts) and the
Comfy.Canvas.DeleteSelectedItems command to call the new helper.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 5fa8848f-0112-48c7-a7ea-78dfbc58cf14

📥 Commits

Reviewing files that changed from the base of the PR and between ca3eade and f7ca59d.

📒 Files selected for processing (3)
  • src/composables/useCoreCommands.ts
  • src/platform/keybindings/keybindingService.ts
  • src/views/GraphView.vue
✅ Files skipped from review due to trivial changes (1)
  • src/views/GraphView.vue
🚧 Files skipped from review as they are similar to previous changes (1)
  • src/platform/keybindings/keybindingService.ts

stopPropagation in capture phase for Escape was blocking BuilderFooterToolbar's
window-bubble listener and Reka UI / PrimeVue element-level handlers, causing
E2E tests to fail when pressing Escape to close menus, popovers, and dialogs.

Only suppress propagation for non-Escape keybindings; bare Escape must reach
all downstream handlers. Adds tests for the new propagation behaviour and for
the ghost-placement guards in DeleteSelectedItems and ExitSubgraph.
Copy link
Copy Markdown

@cursor cursor Bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cursor Bugbot has reviewed your changes and found 1 potential issue.

Fix All in Cursor

❌ Bugbot Autofix is OFF. To automatically fix reported issues with cloud agents, enable autofix in the Cursor dashboard.

Reviewed by Cursor Bugbot for commit 2e6ec6c. Configure here.

event.key === 'Escape' &&
!event.ctrlKey &&
!event.altKey &&
!event.metaKey
Copy link
Copy Markdown

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

isBareEscape missing shiftKey check allows unintended propagation

Low Severity

The isBareEscape condition checks !event.ctrlKey, !event.altKey, and !event.metaKey but omits !event.shiftKey. If a user registers a custom Shift+Escape keybinding, the check would still evaluate to true, skipping stopPropagation(). This is the exact same class of bug this PR fixes for Ctrl+Enter — a matched keybinding failing to prevent element-level handlers from firing — just for a hypothetical Shift+Escape binding instead.

Fix in Cursor Fix in Web

Reviewed by Cursor Bugbot for commit 2e6ec6c. Configure here.

Copy link
Copy Markdown
Collaborator

@dante01yoon dante01yoon left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Requesting changes for one Escape-priority regression I think is still present on the current head. Focused validation passed locally in an isolated PR worktree: pnpm exec vitest run src/platform/keybindings/keybindingService.propagation.test.ts src/platform/keybindings/keybindingService.escape.test.ts src/composables/useCoreCommands.test.ts (36 tests). GitHub checks are green.

!event.ctrlKey &&
!event.altKey &&
!event.metaKey
if (!isBareEscape) {
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

issue: Bare Escape still executes the global keybinding before element-level handlers get a chance to consume it. Because GraphView now registers keybindHandler on window capture, this branch skips stopPropagation() but still falls through to commandStore.execute("Comfy.Graph.ExitSubgraph") before the event reaches controls such as SingleSelect/MultiSelect, whose Escape handlers call stopEscapeToDocument() at the target. In a subgraph with an open select, Escape now exits the subgraph and then closes the select; before this PR the target handler could stop the bubble-phase keybinding and only close the select. Could we preserve the old priority for bare Escape, for example by not handling it in the capture listener and letting a bubble-phase path run after target handlers, or otherwise skipping the command when an Escape-consuming overlay is the target?

@dante01yoon dante01yoon removed their assignment May 5, 2026
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

size:S This PR changes 10-29 lines, ignoring generated files.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants