Skip to content

fix: prevent subgraph node position corruption during graph transitions#10828

Open
jaeone94 wants to merge 5 commits intomainfrom
fix/subgraph-node-position-corruption
Open

fix: prevent subgraph node position corruption during graph transitions#10828
jaeone94 wants to merge 5 commits intomainfrom
fix/subgraph-node-position-corruption

Conversation

@jaeone94
Copy link
Copy Markdown
Collaborator

@jaeone94 jaeone94 commented Apr 2, 2026

Summary

Fix subgraph internal node positions being permanently corrupted when entering a subgraph after a draft workflow reload. The corruption accumulated across page refreshes, causing nodes to progressively drift apart or compress together.

Changes

  • What: In the ResizeObserver callback (useVueNodeResizeTracking.ts), node positions are now read from the Layout Store (source of truth, initialized from LiteGraph) instead of reverse-converting DOM screen coordinates via getBoundingClientRect() + clientPosToCanvasPos(). The fallback to DOM-based conversion is retained only for nodes not yet present in the Layout Store.

Root Cause

ResizeObserver was using getBoundingClientRect() to get DOM element positions, then converting them to canvas coordinates via clientPosToCanvasPos(). This conversion depends on the current canvas.ds.scale and canvas.ds.offset.

During graph transitions (e.g., entering a subgraph from a draft-loaded workflow), the canvas viewport was stale — it still had the parent graph's zoom level because fitView() hadn't run yet (it's scheduled via requestAnimationFrame). The ResizeObserver callback fired before fitView, converting DOM positions using the wrong scale/offset, and writing the corrupted positions to the Layout Store. The useLayoutSync writeback then permanently overwrote the LiteGraph node positions.

The corruption accumulated across sessions:

  1. Load workflow → enter subgraph → ResizeObserver writes corrupted positions
  2. Draft auto-saves the corrupted positions to localStorage
  3. Page refresh → draft loads with corrupted positions → enter subgraph → positions corrupted further
  4. Each cycle amplifies the drift based on the parent graph's zoom level

This is the same class of bug that PR #9121 fixed for slot positions — the DOM→canvas coordinate conversion is inherently fragile during viewport transitions. This PR applies the same principle to node positions.

Why This Only Affects main (No Backport Needed)

This bug requires two features that only exist on main, not on core/1.41 or core/1.42:

  1. PR fix: persist subgraph viewport across navigation and tab switches #10247 changed subgraphNavigationStore's watcher to flush: 'sync' and added requestAnimationFrame(fitView) on viewport cache miss. This creates the timing window where ResizeObserver fires before fitView corrects the canvas scale.
  2. PR Allow graph navigation by browser forward/backward #6811 added hash-based subgraph auto-entry on page load, which triggers graph transitions during the draft reload flow.

On 1.41/1.42, restoreViewport does nothing on cache miss (no fitView scheduling), and the watcher uses default async flush — so the ResizeObserver never runs with a stale viewport.

Review Focus

  • The core change is small: use nodeLayout.position (already in the Layout Store from initializeFromLiteGraph) instead of computing position from getBoundingClientRect(). This eliminates the dependency on canvas scale/offset being up-to-date during ResizeObserver callbacks.
  • The fallback path (getBoundingClientRectclientPosToCanvasPos) is retained for nodes not yet in the Layout Store (e.g., first render of a newly created node). At that point the canvas transform is stable, so the conversion is safe.
  • Unit tests updated to reflect that position is no longer overwritten from DOM when Layout Store already has the position.
  • E2E test added: load subgraph workflow → enter subgraph → reload (draft) → verify positions preserved.

E2E Test Fixes

  • subgraphDraftPositions.spec.ts: replaced comfyPage.setup({ clearStorage: false }) with page.reload() + explicit draft persistence polling. The setup() method performs a full navigation via goto() which bypassed the draft auto-load flow.
  • SubgraphHelper.packAllInteriorNodes: replaced canvas.click() with dispatchEvent('pointerdown'/'pointerup'). The position fix places subgraph nodes at their correct locations, which now overlap with DOM widget textareas that intercept pointer events.

Test Plan

  • Unit tests pass (useVueNodeResizeTracking.test.ts)
  • E2E test: subgraphDraftPositions.spec.ts — draft reload preserves subgraph node positions
  • Manual: load workflow with subgraph, zoom in/out on root graph, enter subgraph, verify no position drift
  • Manual: repeat with page refresh (draft reload) — positions should be stable across reloads
  • Manual: drag nodes inside subgraph — positions should update correctly
  • Manual: create new node inside subgraph — position should be set correctly (fallback path)

Screenshots

Before
스크린샷 2026-04-03 오전 3 56 48

After
스크린샷 2026-04-03 오전 3 58 24

┆Issue is synchronized with this Notion page by Unito

jaeone94 added 2 commits April 3, 2026 03:24
ResizeObserver was converting DOM positions back to canvas coordinates
using clientPosToCanvasPos, which depends on the current canvas
scale/offset. During graph transitions (e.g. entering a subgraph from
a draft-loaded workflow), the canvas viewport was stale (still had the
parent graph's zoom level), causing all subgraph node positions to be
permanently corrupted. The corruption accumulated across page refreshes
as drafts saved the corrupted positions.

Use the layout store's existing position (initialized from LiteGraph)
instead of reverse-converting DOM screen coordinates. Only fall back to
getBoundingClientRect conversion for nodes not yet in the layout store.

Fixes subgraph node positions drifting apart or compressing together
when entering a subgraph after draft workflow reload.
…t reload

Verifies that entering a subgraph, reloading the page (draft auto-loads),
and auto-entering the subgraph via hash navigation does not corrupt
internal node positions.
@jaeone94 jaeone94 requested a review from a team April 2, 2026 18:32
@dosubot dosubot bot added the size:L This PR changes 100-499 lines, ignoring generated files. label Apr 2, 2026
@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai bot commented Apr 2, 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

No actionable comments were generated in the recent review. 🎉

ℹ️ Recent review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 73d1e546-b745-474a-8d5c-7d5e32d1ee61

📥 Commits

Reviewing files that changed from the base of the PR and between 38d6a31 and f7b48c1.

📒 Files selected for processing (2)
  • browser_tests/fixtures/helpers/WorkflowHelper.ts
  • browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts

📝 Walkthrough

Walkthrough

Adds a Playwright spec that verifies subgraph internal node coordinates persist across draft-driven reloads; changes Vue node resize-tracking to prefer stored layout positions for x/y and updates its tests; replaces canvas click calls with explicit pointer events; adds a helper to wait for draft persistence.

Changes

Cohort / File(s) Summary
Subgraph Draft Position Testing
browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts
Adds a Playwright spec that enables draft persistence, loads subgraphs/basic-subgraph, captures internal subgraph node { id, x, y }, waits for draft persistence, reloads the page, waits for auto-entry, and asserts coordinates match pre-reload values using toBeCloseTo(..., 0).
Node Resize Tracking — Implementation & Tests
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts, src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.test.ts
Bounds computation now uses stored node layout position (nodeLayout?.position) for x/y when available; keeps getBoundingClientRect() fallback (which adjusts y by LiteGraph.NODE_TITLE_HEIGHT). Tests updated to expect no DOM reads or layout-store writes when stored position is present and removed obsolete assertions/variables.
Subgraph Helper — Canvas Interaction
browser_tests/fixtures/helpers/SubgraphHelper.ts
Replaces this.comfyPage.canvas.click() with explicit pointerdown and pointerup event dispatches (bubbles: true, cancelable: true, button: 0) when packing interior nodes to better simulate pointer interactions.
Workflow Helper — Draft Persistence Wait
browser_tests/fixtures/helpers/WorkflowHelper.ts
Adds waitForDraftPersisted({ timeout = 5000 } = {}), which waits until localStorage contains at least one key starting with Comfy.Workflow.Draft.v2: (uses given timeout).

Sequence Diagram(s)

(omitted)

Estimated code review effort

🎯 3 (Moderate) | ⏱️ ~20 minutes

Poem

🐰
I nudged each node and marked its place,
Stored every hop in a cozy space.
A reload came by, a blustery day,
I hopped back home — positions stayed.
Hooray for drafts that keep paths safe.

🚥 Pre-merge checks | ✅ 5
✅ Passed checks (5 passed)
Check name Status Explanation
Title check ✅ Passed The title 'fix: prevent subgraph node position corruption during graph transitions' clearly and concisely describes the main fix: preventing node position corruption in subgraphs during graph transitions.
Description check ✅ Passed The description comprehensively covers all required template sections: a clear summary of the fix, detailed changes explaining the root cause and solution, review focus areas, test plan with checkmarks, and supporting screenshots.
Docstring Coverage ✅ Passed No functions found in the changed files to evaluate docstring coverage. Skipping docstring coverage check.
End-To-End Regression Coverage For Fixes ✅ Passed PR uses bug-fix language ('fix:'), changes multiple browser_tests/ files including new E2E test, and documents the test purpose in PR description.
Adr Compliance For Entity/Litegraph Changes ✅ Passed PR modifies only renderer/extensions and test files; no changes to src/lib/litegraph/ or src/ecs/. Implementation reads positions from Layout Store without violating ADR 0003 or 0008.

✏️ Tip: You can configure your own custom pre-merge checks in the settings.

✨ Finishing Touches
📝 Generate docstrings
  • 🔄 Generating stacked PR...
  • ✅ Committed to branch successfully - (🔄 Check to regenerate)
🧪 Generate unit tests (beta)
  • Create PR with unit tests
  • Commit unit tests in branch fix/subgraph-node-position-corruption

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 Apr 2, 2026

🎨 Storybook: ✅ Built — View Storybook

Details

⏰ Completed at: 04/02/2026, 08:42:50 PM UTC

Links

@github-actions
Copy link
Copy Markdown

github-actions bot commented Apr 2, 2026

🎭 Playwright: ✅ 912 passed, 0 failed · 3 flaky

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

@github-actions
Copy link
Copy Markdown

github-actions bot commented Apr 2, 2026

📦 Bundle: 5.11 MB gzip 🟢 -95 B

Details

Summary

  • Raw size: 23.5 MB baseline 23.5 MB — 🔴 +112 B
  • Gzip: 5.11 MB baseline 5.11 MB — 🟢 -95 B
  • Brotli: 3.95 MB baseline 3.95 MB — 🔴 +352 B
  • Bundles: 250 current • 250 baseline • 114 added / 114 removed

Category Glance
Graph Workspace 🔴 +112 B (1.2 MB) · Vendor & Third-Party ⚪ 0 B (9.8 MB) · Other ⚪ 0 B (8.44 MB) · Data & Services ⚪ 0 B (2.97 MB) · Panels & Settings ⚪ 0 B (484 kB) · Utilities & Hooks ⚪ 0 B (338 kB) · + 5 more

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

Main entry bundles and manifests

File Before After Δ Raw Δ Gzip Δ Brotli
assets/index-BV5z4N-P.js (removed) 22.3 kB 🟢 -22.3 kB 🟢 -7.96 kB 🟢 -6.83 kB
assets/index-Ch0DIi-l.js (new) 22.3 kB 🔴 +22.3 kB 🔴 +7.95 kB 🔴 +6.83 kB

Status: 1 added / 1 removed

Graph Workspace — 1.2 MB (baseline 1.2 MB) • 🔴 +112 B

Graph editor runtime, canvas, workflow orchestration

File Before After Δ Raw Δ Gzip Δ Brotli
assets/GraphView-CzunRT3Z.js (new) 1.2 MB 🔴 +1.2 MB 🔴 +256 kB 🔴 +193 kB
assets/GraphView-DfWu0rwI.js (removed) 1.2 MB 🟢 -1.2 MB 🟢 -256 kB 🟢 -193 kB

Status: 1 added / 1 removed

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

Top-level views, pages, and routed surfaces

File Before After Δ Raw Δ Gzip Δ Brotli
assets/CloudSurveyView-D0Ijnp16.js (new) 15.7 kB 🔴 +15.7 kB 🔴 +3.4 kB 🔴 +2.91 kB
assets/CloudSurveyView-DN3agrkX.js (removed) 15.7 kB 🟢 -15.7 kB 🟢 -3.4 kB 🟢 -2.91 kB
assets/CloudLoginView--WD1T3-l.js (new) 12 kB 🔴 +12 kB 🔴 +3.36 kB 🔴 +2.96 kB
assets/CloudLoginView-PQ1uL-Ys.js (removed) 12 kB 🟢 -12 kB 🟢 -3.36 kB 🟢 -2.96 kB
assets/CloudSignupView-BWWsyWkM.js (removed) 9.78 kB 🟢 -9.78 kB 🟢 -2.85 kB 🟢 -2.51 kB
assets/CloudSignupView-DZyMo_eB.js (new) 9.78 kB 🔴 +9.78 kB 🔴 +2.85 kB 🔴 +2.51 kB
assets/UserCheckView-CCQrW0Rr.js (new) 9.04 kB 🔴 +9.04 kB 🔴 +2.33 kB 🔴 +2.03 kB
assets/UserCheckView-zyj6hHKC.js (removed) 9.04 kB 🟢 -9.04 kB 🟢 -2.33 kB 🟢 -2.04 kB
assets/CloudLayoutView-D39f1Z8g.js (removed) 7.54 kB 🟢 -7.54 kB 🟢 -2.36 kB 🟢 -2.04 kB
assets/CloudLayoutView-fWgBw-Tt.js (new) 7.54 kB 🔴 +7.54 kB 🔴 +2.36 kB 🔴 +2.04 kB
assets/CloudForgotPasswordView-BOjekjaE.js (removed) 5.94 kB 🟢 -5.94 kB 🟢 -2.09 kB 🟢 -1.83 kB
assets/CloudForgotPasswordView-JsOkRaXv.js (new) 5.94 kB 🔴 +5.94 kB 🔴 +2.09 kB 🔴 +1.84 kB
assets/CloudAuthTimeoutView-Ch8-gLSr.js (removed) 5.31 kB 🟢 -5.31 kB 🟢 -1.93 kB 🟢 -1.69 kB
assets/CloudAuthTimeoutView-GQcv4jWr.js (new) 5.31 kB 🔴 +5.31 kB 🔴 +1.93 kB 🔴 +1.69 kB
assets/CloudSubscriptionRedirectView-9nMzk_Cx.js (removed) 5.08 kB 🟢 -5.08 kB 🟢 -1.91 kB 🟢 -1.69 kB
assets/CloudSubscriptionRedirectView-f2nzomPm.js (new) 5.08 kB 🔴 +5.08 kB 🔴 +1.9 kB 🔴 +1.69 kB
assets/UserSelectView-BNC8Bj2q.js (removed) 4.71 kB 🟢 -4.71 kB 🟢 -1.74 kB 🟢 -1.54 kB
assets/UserSelectView-C-PSgKMr.js (new) 4.71 kB 🔴 +4.71 kB 🔴 +1.74 kB 🔴 +1.54 kB

Status: 9 added / 9 removed / 2 unchanged

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

Configuration panels, inspectors, and settings screens

File Before After Δ Raw Δ Gzip Δ Brotli
assets/KeybindingPanel-B2pdAUmK.js (removed) 46.6 kB 🟢 -46.6 kB 🟢 -9.52 kB 🟢 -8.47 kB
assets/KeybindingPanel-DRKC9-zq.js (new) 46.6 kB 🔴 +46.6 kB 🔴 +9.52 kB 🔴 +8.47 kB
assets/SecretsPanel-8fl4a9Rk.js (new) 22.4 kB 🔴 +22.4 kB 🔴 +5.42 kB 🔴 +4.78 kB
assets/SecretsPanel-BQejkTJL.js (removed) 22.4 kB 🟢 -22.4 kB 🟢 -5.43 kB 🟢 -4.77 kB
assets/LegacyCreditsPanel-DjuFJOb2.js (removed) 21.5 kB 🟢 -21.5 kB 🟢 -5.81 kB 🟢 -5.13 kB
assets/LegacyCreditsPanel-DpqV_E5C.js (new) 21.5 kB 🔴 +21.5 kB 🔴 +5.81 kB 🔴 +5.13 kB
assets/SubscriptionPanel-CjyaHAfZ.js (removed) 19.7 kB 🟢 -19.7 kB 🟢 -5.01 kB 🟢 -4.4 kB
assets/SubscriptionPanel-CrW-HCYL.js (new) 19.7 kB 🔴 +19.7 kB 🔴 +5.01 kB 🔴 +4.4 kB
assets/AboutPanel-BsA8AuqR.js (new) 12 kB 🔴 +12 kB 🔴 +3.33 kB 🔴 +3 kB
assets/AboutPanel-Cwr-1gCJ.js (removed) 12 kB 🟢 -12 kB 🟢 -3.33 kB 🟢 -2.98 kB
assets/ExtensionPanel-CeVZnIaJ.js (removed) 9.78 kB 🟢 -9.78 kB 🟢 -2.82 kB 🟢 -2.51 kB
assets/ExtensionPanel-D42akf2b.js (new) 9.78 kB 🔴 +9.78 kB 🔴 +2.82 kB 🔴 +2.51 kB
assets/ServerConfigPanel-DWmNi7En.js (removed) 6.85 kB 🟢 -6.85 kB 🟢 -2.27 kB 🟢 -2.03 kB
assets/ServerConfigPanel-Dz5yn_6E.js (new) 6.85 kB 🔴 +6.85 kB 🔴 +2.27 kB 🔴 +2.04 kB
assets/UserPanel-CpUIRg_5.js (removed) 6.56 kB 🟢 -6.56 kB 🟢 -2.15 kB 🟢 -1.89 kB
assets/UserPanel-IWMwzEh_.js (new) 6.56 kB 🔴 +6.56 kB 🔴 +2.15 kB 🔴 +1.89 kB
assets/cloudRemoteConfig-1ILNKqqj.js (removed) 1.85 kB 🟢 -1.85 kB 🟢 -904 B 🟢 -808 B
assets/cloudRemoteConfig-BBpKEQGN.js (new) 1.85 kB 🔴 +1.85 kB 🔴 +902 B 🔴 +792 B
assets/refreshRemoteConfig-B2hbMQcS.js (removed) 1.45 kB 🟢 -1.45 kB 🟢 -650 B 🟢 -558 B
assets/refreshRemoteConfig-DlLSkfv-.js (new) 1.45 kB 🔴 +1.45 kB 🔴 +648 B 🔴 +554 B

Status: 10 added / 10 removed / 12 unchanged

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

Authentication, profile, and account management bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/auth-BSl9EWKn.js (removed) 3.57 kB 🟢 -3.57 kB 🟢 -1.26 kB 🟢 -1.07 kB
assets/auth-BTDj5mu_.js (new) 3.57 kB 🔴 +3.57 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/SignUpForm-C39tyFg5.js (removed) 3.16 kB 🟢 -3.16 kB 🟢 -1.29 kB 🟢 -1.15 kB
assets/SignUpForm-D-hndYoK.js (new) 3.16 kB 🔴 +3.16 kB 🔴 +1.29 kB 🔴 +1.15 kB
assets/UpdatePasswordContent-BVmR_rAk.js (new) 2.66 kB 🔴 +2.66 kB 🔴 +1.19 kB 🔴 +1.05 kB
assets/UpdatePasswordContent-Dx43XTS-.js (removed) 2.66 kB 🟢 -2.66 kB 🟢 -1.19 kB 🟢 -1.06 kB
assets/authStore-BFSb_nTl.js (removed) 989 B 🟢 -989 B 🟢 -486 B 🟢 -436 B
assets/authStore-BzJbFnIj.js (new) 989 B 🔴 +989 B 🔴 +485 B 🔴 +434 B
assets/auth-BWTDyraX.js (new) 348 B 🔴 +348 B 🔴 +217 B 🔴 +186 B
assets/auth-CXMo2Dp7.js (removed) 348 B 🟢 -348 B 🟢 -218 B 🟢 -188 B

Status: 5 added / 5 removed / 2 unchanged

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

Modals, dialogs, drawers, and in-app editors

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useShareDialog-DLve5t7r.js (new) 108 kB 🔴 +108 kB 🔴 +22.4 kB 🔴 +18.9 kB
assets/useShareDialog-DvXHJUXQ.js (removed) 108 kB 🟢 -108 kB 🟢 -22.4 kB 🟢 -18.9 kB
assets/useSubscriptionDialog-C2fLN3wS.js (removed) 969 B 🟢 -969 B 🟢 -478 B 🟢 -424 B
assets/useSubscriptionDialog-OTACc_fU.js (new) 969 B 🔴 +969 B 🔴 +476 B 🔴 +419 B

Status: 2 added / 2 removed

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

Reusable component library chunks

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyQueueButton-BhvaFBJ6.js (removed) 13.5 kB 🟢 -13.5 kB 🟢 -3.79 kB 🟢 -3.38 kB
assets/ComfyQueueButton-CFUCbYNc.js (new) 13.5 kB 🔴 +13.5 kB 🔴 +3.79 kB 🔴 +3.38 kB
assets/useTerminalTabs-CHjmK-IB.js (new) 10.7 kB 🔴 +10.7 kB 🔴 +3.6 kB 🔴 +3.17 kB
assets/useTerminalTabs-Da7x-SH9.js (removed) 10.7 kB 🟢 -10.7 kB 🟢 -3.61 kB 🟢 -3.17 kB
assets/SubscribeButton-Buye1wd2.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -1.04 kB 🟢 -916 B
assets/SubscribeButton-DMq2Mpbd.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +1.05 kB 🔴 +914 B
assets/cloudFeedbackTopbarButton-C0EmvdxF.js (new) 1.66 kB 🔴 +1.66 kB 🔴 +843 B 🔴 +751 B
assets/cloudFeedbackTopbarButton-CVt_2Qjt.js (removed) 1.66 kB 🟢 -1.66 kB 🟢 -847 B 🟢 -753 B
assets/ComfyQueueButton-CinY-7pG.js (new) 1.03 kB 🔴 +1.03 kB 🔴 +490 B 🔴 +439 B
assets/ComfyQueueButton-DztZ2h7n.js (removed) 1.03 kB 🟢 -1.03 kB 🟢 -490 B 🟢 -444 B

Status: 5 added / 5 removed / 8 unchanged

Data & Services — 2.97 MB (baseline 2.97 MB) • ⚪ 0 B

Stores, services, APIs, and repositories

File Before After Δ Raw Δ Gzip Δ Brotli
assets/dialogService-CTkLAMhj.js (new) 1.94 MB 🔴 +1.94 MB 🔴 +445 kB 🔴 +338 kB
assets/dialogService-DL9b10ho.js (removed) 1.94 MB 🟢 -1.94 MB 🟢 -445 kB 🟢 -338 kB
assets/api-BXG1ih8w.js (removed) 885 kB 🟢 -885 kB 🟢 -211 kB 🟢 -166 kB
assets/api-CCbLi8Wm.js (new) 885 kB 🔴 +885 kB 🔴 +211 kB 🔴 +167 kB
assets/load3dService-bRz5iJN_.js (removed) 92.5 kB 🟢 -92.5 kB 🟢 -19.7 kB 🟢 -17 kB
assets/load3dService-CBZfpXii.js (new) 92.5 kB 🔴 +92.5 kB 🔴 +19.7 kB 🔴 +17 kB
assets/workflowShareService-D0VML-jN.js (new) 16.6 kB 🔴 +16.6 kB 🔴 +4.88 kB 🔴 +4.32 kB
assets/workflowShareService-DY_SMw_m.js (removed) 16.6 kB 🟢 -16.6 kB 🟢 -4.88 kB 🟢 -4.33 kB
assets/keybindingService-Dh8UVfhF.js (removed) 13.8 kB 🟢 -13.8 kB 🟢 -3.66 kB 🟢 -3.22 kB
assets/keybindingService-mFfEGdsA.js (new) 13.8 kB 🔴 +13.8 kB 🔴 +3.67 kB 🔴 +3.22 kB
assets/releaseStore-BcssGQkN.js (new) 8.12 kB 🔴 +8.12 kB 🔴 +2.28 kB 🔴 +2 kB
assets/releaseStore-Dx_k6Nwh.js (removed) 8.12 kB 🟢 -8.12 kB 🟢 -2.28 kB 🟢 -2 kB
assets/userStore-CITzgu1f.js (new) 2.24 kB 🔴 +2.24 kB 🔴 +869 B 🔴 +761 B
assets/userStore-Dn7NQxUX.js (removed) 2.24 kB 🟢 -2.24 kB 🟢 -871 B 🟢 -759 B
assets/audioService-ApIaKpyV.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -878 B 🟢 -762 B
assets/audioService-D4QbISSz.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +878 B 🔴 +757 B
assets/releaseStore-Cx9D-oBk.js (new) 993 B 🔴 +993 B 🔴 +482 B 🔴 +426 B
assets/releaseStore-Un5kaoPr.js (removed) 993 B 🟢 -993 B 🟢 -483 B 🟢 -429 B
assets/workflowDraftStore-DAJWrDdV.js (removed) 969 B 🟢 -969 B 🟢 -476 B 🟢 -426 B
assets/workflowDraftStore-wkduATNf.js (new) 969 B 🔴 +969 B 🔴 +474 B 🔴 +423 B
assets/dialogService-BMroT8KZ.js (new) 958 B 🔴 +958 B 🔴 +467 B 🔴 +419 B
assets/dialogService-BrIKg87L.js (removed) 958 B 🟢 -958 B 🟢 -470 B 🟢 -420 B
assets/settingStore-ClGwrQzi.js (removed) 956 B 🟢 -956 B 🟢 -472 B 🟢 -421 B
assets/settingStore-O63VKWed.js (new) 956 B 🔴 +956 B 🔴 +470 B 🔴 +417 B
assets/assetsStore-CyHgKzYc.js (new) 955 B 🔴 +955 B 🔴 +470 B 🔴 +419 B
assets/assetsStore-DAOHjKi5.js (removed) 955 B 🟢 -955 B 🟢 -472 B 🟢 -421 B

Status: 13 added / 13 removed / 4 unchanged

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

Helpers, composables, and utility bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useConflictDetection-DdrDyGtR.js (new) 232 kB 🔴 +232 kB 🔴 +51.3 kB 🔴 +41.8 kB
assets/useConflictDetection-nUW_zEQV.js (removed) 232 kB 🟢 -232 kB 🟢 -51.3 kB 🟢 -41.8 kB
assets/useLoad3dViewer-CGnLv-qj.js (removed) 18.7 kB 🟢 -18.7 kB 🟢 -4.44 kB 🟢 -3.88 kB
assets/useLoad3dViewer-D5NaP4PG.js (new) 18.7 kB 🔴 +18.7 kB 🔴 +4.43 kB 🔴 +3.87 kB
assets/useLoad3d-CAlWTvxt.js (removed) 15 kB 🟢 -15 kB 🟢 -3.79 kB 🟢 -3.36 kB
assets/useLoad3d-hivHc3oO.js (new) 15 kB 🔴 +15 kB 🔴 +3.79 kB 🔴 +3.35 kB
assets/useFeatureFlags-C04AK3nD.js (new) 5.78 kB 🔴 +5.78 kB 🔴 +1.75 kB 🔴 +1.48 kB
assets/useFeatureFlags-CCM59cHa.js (removed) 5.78 kB 🟢 -5.78 kB 🟢 -1.75 kB 🟢 -1.49 kB
assets/useCopyToClipboard-c9XJTbWU.js (removed) 5.29 kB 🟢 -5.29 kB 🟢 -1.86 kB 🟢 -1.57 kB
assets/useCopyToClipboard-CxIlW5UK.js (new) 5.29 kB 🔴 +5.29 kB 🔴 +1.86 kB 🔴 +1.57 kB
assets/useWorkspaceUI-CsHtvMBK.js (new) 3.34 kB 🔴 +3.34 kB 🔴 +979 B 🔴 +812 B
assets/useWorkspaceUI-CwmuDEF_.js (removed) 3.34 kB 🟢 -3.34 kB 🟢 -980 B 🟢 -813 B
assets/subscriptionCheckoutUtil-BgLX4THw.js (removed) 2.97 kB 🟢 -2.97 kB 🟢 -1.31 kB 🟢 -1.14 kB
assets/subscriptionCheckoutUtil-DbS6cD5P.js (new) 2.97 kB 🔴 +2.97 kB 🔴 +1.31 kB 🔴 +1.14 kB
assets/assetPreviewUtil-CVEJCYmX.js (removed) 2.27 kB 🟢 -2.27 kB 🟢 -961 B 🟢 -839 B
assets/assetPreviewUtil-DmDmsuot.js (new) 2.27 kB 🔴 +2.27 kB 🔴 +956 B 🔴 +833 B
assets/useUpstreamValue-Blcgw1HX.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -805 B 🟢 -719 B
assets/useUpstreamValue-BtMRZwqT.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +806 B 🔴 +713 B
assets/useLoad3d-DdJON6j9.js (new) 1.13 kB 🔴 +1.13 kB 🔴 +538 B 🔴 +481 B
assets/useLoad3d-DPO_MvYV.js (removed) 1.13 kB 🟢 -1.13 kB 🟢 -538 B 🟢 -481 B
assets/useLoad3dViewer-DR2oQZGO.js (new) 1.07 kB 🔴 +1.07 kB 🔴 +505 B 🔴 +455 B
assets/useLoad3dViewer-w0romBbw.js (removed) 1.07 kB 🟢 -1.07 kB 🟢 -507 B 🟢 -456 B
assets/useCurrentUser-B7zctOav.js (new) 955 B 🔴 +955 B 🔴 +471 B 🔴 +417 B
assets/useCurrentUser-PdfgoQSg.js (removed) 955 B 🟢 -955 B 🟢 -472 B 🟢 -421 B
assets/useWorkspaceSwitch-dLtVVp9x.js (new) 747 B 🔴 +747 B 🔴 +385 B 🔴 +332 B
assets/useWorkspaceSwitch-pZVEBoLx.js (removed) 747 B 🟢 -747 B 🟢 -385 B 🟢 -327 B

Status: 13 added / 13 removed / 13 unchanged

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

External libraries and shared vendor chunks

Status: 16 unchanged

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

Bundles that do not match a named category

File Before After Δ Raw Δ Gzip Δ Brotli
assets/core-BHlm_vyb.js (new) 76.8 kB 🔴 +76.8 kB 🔴 +19.9 kB 🔴 +17 kB
assets/core-DkY1yavs.js (removed) 76.8 kB 🟢 -76.8 kB 🟢 -19.9 kB 🟢 -17 kB
assets/groupNode-D23XCGlO.js (new) 74 kB 🔴 +74 kB 🔴 +18.5 kB 🔴 +16.3 kB
assets/groupNode-DWCVYI29.js (removed) 74 kB 🟢 -74 kB 🟢 -18.5 kB 🟢 -16.3 kB
assets/WidgetSelect-CpzzNLFQ.js (removed) 64.6 kB 🟢 -64.6 kB 🟢 -14.1 kB 🟢 -12.2 kB
assets/WidgetSelect-HhUWB_Dr.js (new) 64.6 kB 🔴 +64.6 kB 🔴 +14.1 kB 🔴 +12.2 kB
assets/SubscriptionRequiredDialogContentWorkspace-BI38VZNt.js (new) 48.9 kB 🔴 +48.9 kB 🔴 +9.29 kB 🔴 +7.97 kB
assets/SubscriptionRequiredDialogContentWorkspace-mrkB-sDu.js (removed) 48.9 kB 🟢 -48.9 kB 🟢 -9.29 kB 🟢 -7.96 kB
assets/WidgetPainter-DifUhHUw.js (removed) 33.3 kB 🟢 -33.3 kB 🟢 -8.12 kB 🟢 -7.19 kB
assets/WidgetPainter-DO1p6E9N.js (new) 33.3 kB 🔴 +33.3 kB 🔴 +8.11 kB 🔴 +7.21 kB
assets/Load3DControls-CIP153si.js (removed) 32.1 kB 🟢 -32.1 kB 🟢 -5.47 kB 🟢 -4.75 kB
assets/Load3DControls-Cy-DMlXf.js (new) 32.1 kB 🔴 +32.1 kB 🔴 +5.47 kB 🔴 +4.75 kB
assets/WorkspacePanelContent-C4RBYT8L.js (removed) 29.9 kB 🟢 -29.9 kB 🟢 -6.33 kB 🟢 -5.55 kB
assets/WorkspacePanelContent-d13xlOkj.js (new) 29.9 kB 🔴 +29.9 kB 🔴 +6.32 kB 🔴 +5.55 kB
assets/SubscriptionRequiredDialogContent-CfS_N51E.js (removed) 28.2 kB 🟢 -28.2 kB 🟢 -7.17 kB 🟢 -6.3 kB
assets/SubscriptionRequiredDialogContent-jZt34uSH.js (new) 28.2 kB 🔴 +28.2 kB 🔴 +7.17 kB 🔴 +6.3 kB
assets/Load3dViewerContent-B5RyVAXU.js (new) 24.5 kB 🔴 +24.5 kB 🔴 +5.33 kB 🔴 +4.64 kB
assets/Load3dViewerContent-BcMDGZux.js (removed) 24.5 kB 🟢 -24.5 kB 🟢 -5.33 kB 🟢 -4.65 kB
assets/WidgetImageCrop-BgNeb76W.js (new) 23.3 kB 🔴 +23.3 kB 🔴 +5.83 kB 🔴 +5.15 kB
assets/WidgetImageCrop-BVVFrmyp.js (removed) 23.3 kB 🟢 -23.3 kB 🟢 -5.83 kB 🟢 -5.14 kB
assets/SubscriptionPanelContentWorkspace-C80cg6jS.js (new) 22.2 kB 🔴 +22.2 kB 🔴 +5.18 kB 🔴 +4.56 kB
assets/SubscriptionPanelContentWorkspace-DW6iLuaj.js (removed) 22.2 kB 🟢 -22.2 kB 🟢 -5.18 kB 🟢 -4.56 kB
assets/SignInContent-BvQ1RNzW.js (new) 20.4 kB 🔴 +20.4 kB 🔴 +5.29 kB 🔴 +4.63 kB
assets/SignInContent-uEIZXh7P.js (removed) 20.4 kB 🟢 -20.4 kB 🟢 -5.29 kB 🟢 -4.62 kB
assets/CurrentUserPopoverWorkspace-Bc0IE3CV.js (new) 20.4 kB 🔴 +20.4 kB 🔴 +4.83 kB 🔴 +4.33 kB
assets/CurrentUserPopoverWorkspace-D7MiQTjj.js (removed) 20.4 kB 🟢 -20.4 kB 🟢 -4.83 kB 🟢 -4.33 kB
assets/WidgetInputNumber-B7_XpfHA.js (new) 19.1 kB 🔴 +19.1 kB 🔴 +4.84 kB 🔴 +4.3 kB
assets/WidgetInputNumber-C8AiTwcc.js (removed) 19.1 kB 🟢 -19.1 kB 🟢 -4.84 kB 🟢 -4.3 kB
assets/WidgetRecordAudio-BIksZ7tm.js (new) 18.1 kB 🔴 +18.1 kB 🔴 +5.18 kB 🔴 +4.64 kB
assets/WidgetRecordAudio-CDygsqZx.js (removed) 18.1 kB 🟢 -18.1 kB 🟢 -5.18 kB 🟢 -4.64 kB
assets/Load3D-CEE8Q-uV.js (removed) 16.9 kB 🟢 -16.9 kB 🟢 -4.12 kB 🟢 -3.6 kB
assets/Load3D-CSkwpyLb.js (new) 16.9 kB 🔴 +16.9 kB 🔴 +4.12 kB 🔴 +3.59 kB
assets/load3d-D00p-Qbx.js (removed) 15 kB 🟢 -15 kB 🟢 -4.32 kB 🟢 -3.73 kB
assets/load3d-PH2n50hL.js (new) 15 kB 🔴 +15 kB 🔴 +4.31 kB 🔴 +3.74 kB
assets/WaveAudioPlayer-BerjATsX.js (new) 13.4 kB 🔴 +13.4 kB 🔴 +3.69 kB 🔴 +3.22 kB
assets/WaveAudioPlayer-CmNzHmeS.js (removed) 13.4 kB 🟢 -13.4 kB 🟢 -3.69 kB 🟢 -3.22 kB
assets/WidgetCurve-B0rPpjyf.js (new) 12 kB 🔴 +12 kB 🔴 +3.85 kB 🔴 +3.48 kB
assets/WidgetCurve-DAcd1axu.js (removed) 12 kB 🟢 -12 kB 🟢 -3.85 kB 🟢 -3.49 kB
assets/TeamWorkspacesDialogContent-AKAsQ_SF.js (removed) 11.1 kB 🟢 -11.1 kB 🟢 -3.33 kB 🟢 -2.96 kB
assets/TeamWorkspacesDialogContent-CBL1WBc_.js (new) 11.1 kB 🔴 +11.1 kB 🔴 +3.33 kB 🔴 +2.98 kB
assets/nodeTemplates-C0T3UmTk.js (new) 9.58 kB 🔴 +9.58 kB 🔴 +3.37 kB 🔴 +2.97 kB
assets/nodeTemplates-DZU0cPVd.js (removed) 9.58 kB 🟢 -9.58 kB 🟢 -3.38 kB 🟢 -2.97 kB
assets/InviteMemberDialogContent-B6CmK5A1.js (removed) 7.77 kB 🟢 -7.77 kB 🟢 -2.45 kB 🟢 -2.14 kB
assets/InviteMemberDialogContent-BdxJO1oV.js (new) 7.77 kB 🔴 +7.77 kB 🔴 +2.45 kB 🔴 +2.13 kB
assets/Load3DConfiguration-BuGv4Tf3.js (removed) 6.6 kB 🟢 -6.6 kB 🟢 -2.04 kB 🟢 -1.78 kB
assets/Load3DConfiguration-DTT40bk6.js (new) 6.6 kB 🔴 +6.6 kB 🔴 +2.04 kB 🔴 +1.78 kB
assets/onboardingCloudRoutes-D5PdEj5H.js (new) 6.53 kB 🔴 +6.53 kB 🔴 +2.04 kB 🔴 +1.77 kB
assets/onboardingCloudRoutes-DLHjrPo6.js (removed) 6.53 kB 🟢 -6.53 kB 🟢 -2.04 kB 🟢 -1.75 kB
assets/WidgetWithControl-BGQkGsMQ.js (new) 5.99 kB 🔴 +5.99 kB 🔴 +2.38 kB 🔴 +2.12 kB
assets/WidgetWithControl-Btbw90ZJ.js (removed) 5.99 kB 🟢 -5.99 kB 🟢 -2.38 kB 🟢 -2.13 kB
assets/CreateWorkspaceDialogContent-C-0KpjdR.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +2.15 kB 🔴 +1.87 kB
assets/CreateWorkspaceDialogContent-CjXnPw8e.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -2.15 kB 🟢 -1.87 kB
assets/FreeTierDialogContent-B5B8_ag6.js (new) 5.82 kB 🔴 +5.82 kB 🔴 +2.04 kB 🔴 +1.81 kB
assets/FreeTierDialogContent-DJFsEUrF.js (removed) 5.82 kB 🟢 -5.82 kB 🟢 -2.05 kB 🟢 -1.81 kB
assets/EditWorkspaceDialogContent-BR3LVuX_.js (removed) 5.75 kB 🟢 -5.75 kB 🟢 -2.11 kB 🟢 -1.84 kB
assets/EditWorkspaceDialogContent-DSLQS_Jc.js (new) 5.75 kB 🔴 +5.75 kB 🔴 +2.11 kB 🔴 +1.84 kB
assets/WidgetTextarea-B2YT4kGA.js (removed) 5.53 kB 🟢 -5.53 kB 🟢 -2.17 kB 🟢 -1.92 kB
assets/WidgetTextarea-CY6XAy3-.js (new) 5.53 kB 🔴 +5.53 kB 🔴 +2.17 kB 🔴 +1.92 kB
assets/Preview3d-9Ga5pyow.js (removed) 5.36 kB 🟢 -5.36 kB 🟢 -1.79 kB 🟢 -1.56 kB
assets/Preview3d-IL92U91R.js (new) 5.36 kB 🔴 +5.36 kB 🔴 +1.79 kB 🔴 +1.56 kB
assets/ValueControlPopover-00LF_iHi.js (new) 5.33 kB 🔴 +5.33 kB 🔴 +1.93 kB 🔴 +1.72 kB
assets/ValueControlPopover-DkvXLwXi.js (removed) 5.33 kB 🟢 -5.33 kB 🟢 -1.93 kB 🟢 -1.72 kB
assets/CancelSubscriptionDialogContent-tmvwrHJx.js (new) 5.22 kB 🔴 +5.22 kB 🔴 +1.94 kB 🔴 +1.71 kB
assets/CancelSubscriptionDialogContent-v2uz99Wg.js (removed) 5.22 kB 🟢 -5.22 kB 🟢 -1.95 kB 🟢 -1.7 kB
assets/DeleteWorkspaceDialogContent-D0WhZcJB.js (removed) 4.65 kB 🟢 -4.65 kB 🟢 -1.79 kB 🟢 -1.55 kB
assets/DeleteWorkspaceDialogContent-HNm9PZ4x.js (new) 4.65 kB 🔴 +4.65 kB 🔴 +1.79 kB 🔴 +1.55 kB
assets/LeaveWorkspaceDialogContent-D4Rv8gza.js (new) 4.48 kB 🔴 +4.48 kB 🔴 +1.73 kB 🔴 +1.5 kB
assets/LeaveWorkspaceDialogContent-tuUgAsfr.js (removed) 4.48 kB 🟢 -4.48 kB 🟢 -1.74 kB 🟢 -1.5 kB
assets/RemoveMemberDialogContent-Bft2SXI7.js (removed) 4.46 kB 🟢 -4.46 kB 🟢 -1.69 kB 🟢 -1.47 kB
assets/RemoveMemberDialogContent-CBSPlJpb.js (new) 4.46 kB 🔴 +4.46 kB 🔴 +1.69 kB 🔴 +1.47 kB
assets/tierBenefits-CySL5Xdj.js (new) 4.45 kB 🔴 +4.45 kB 🔴 +1.58 kB 🔴 +1.36 kB
assets/tierBenefits-DyIC0b1p.js (removed) 4.45 kB 🟢 -4.45 kB 🟢 -1.58 kB 🟢 -1.36 kB
assets/RevokeInviteDialogContent-CstwBJ-b.js (new) 4.37 kB 🔴 +4.37 kB 🔴 +1.7 kB 🔴 +1.48 kB
assets/RevokeInviteDialogContent-DSLI2VJd.js (removed) 4.37 kB 🟢 -4.37 kB 🟢 -1.7 kB 🟢 -1.49 kB
assets/InviteMemberUpsellDialogContent-BYCrFUM-.js (removed) 4.27 kB 🟢 -4.27 kB 🟢 -1.56 kB 🟢 -1.37 kB
assets/InviteMemberUpsellDialogContent-xWAB07AP.js (new) 4.27 kB 🔴 +4.27 kB 🔴 +1.56 kB 🔴 +1.37 kB
assets/cloudSessionCookie-1JxWZUiw.js (new) 4.12 kB 🔴 +4.12 kB 🔴 +1.49 kB 🔴 +1.3 kB
assets/cloudSessionCookie-DLMEOb1r.js (removed) 4.12 kB 🟢 -4.12 kB 🟢 -1.49 kB 🟢 -1.3 kB
assets/saveMesh-eNWmfcfX.js (new) 3.92 kB 🔴 +3.92 kB 🔴 +1.68 kB 🔴 +1.48 kB
assets/saveMesh-stwkSCOT.js (removed) 3.92 kB 🟢 -3.92 kB 🟢 -1.68 kB 🟢 -1.48 kB
assets/Media3DTop-Ba6NJe7u.js (new) 3.85 kB 🔴 +3.85 kB 🔴 +1.62 kB 🔴 +1.43 kB
assets/Media3DTop-D2cFMOWJ.js (removed) 3.85 kB 🟢 -3.85 kB 🟢 -1.62 kB 🟢 -1.43 kB
assets/GlobalToast-DVqnGX6s.js (removed) 3.05 kB 🟢 -3.05 kB 🟢 -1.26 kB 🟢 -1.07 kB
assets/GlobalToast-xIxN8aqh.js (new) 3.05 kB 🔴 +3.05 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/SubscribeToRun-C5WZA9zG.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +985 B 🔴 +882 B
assets/SubscribeToRun-Dw0BVKPX.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -984 B 🟢 -876 B
assets/MediaAudioTop-DoABDgxp.js (removed) 2.02 kB 🟢 -2.02 kB 🟢 -982 B 🟢 -830 B
assets/MediaAudioTop-jdFp92fK.js (new) 2.02 kB 🔴 +2.02 kB 🔴 +983 B 🔴 +885 B
assets/CloudRunButtonWrapper-BdhPCaSO.js (removed) 1.99 kB 🟢 -1.99 kB 🟢 -910 B 🟢 -805 B
assets/CloudRunButtonWrapper-CY6BRml8.js (new) 1.99 kB 🔴 +1.99 kB 🔴 +910 B 🔴 +808 B
assets/graphHasMissingNodes-Ci-L_VOA.js (removed) 1.83 kB 🟢 -1.83 kB 🟢 -863 B 🟢 -751 B
assets/graphHasMissingNodes-CSyT7c7T.js (new) 1.83 kB 🔴 +1.83 kB 🔴 +864 B 🔴 +755 B
assets/cloudBadges-BhjwdvCg.js (removed) 1.77 kB 🟢 -1.77 kB 🟢 -889 B 🟢 -781 B
assets/cloudBadges-Dbh3mmjY.js (new) 1.77 kB 🔴 +1.77 kB 🔴 +888 B 🔴 +770 B
assets/cloudSubscription-Dj9m4bje.js (new) 1.68 kB 🔴 +1.68 kB 🔴 +811 B 🔴 +700 B
assets/cloudSubscription-OzZ0apuz.js (removed) 1.68 kB 🟢 -1.68 kB 🟢 -814 B 🟢 -708 B
assets/previousFullPath-BJx2aJIR.js (removed) 1.53 kB 🟢 -1.53 kB 🟢 -695 B 🟢 -602 B
assets/previousFullPath-DkNXqAZ3.js (new) 1.53 kB 🔴 +1.53 kB 🔴 +696 B 🔴 +600 B
assets/Load3D-BO1TJpUF.js (removed) 1.34 kB 🟢 -1.34 kB 🟢 -614 B 🟢 -546 B
assets/Load3D-DW3ItcFA.js (new) 1.34 kB 🔴 +1.34 kB 🔴 +610 B 🔴 +551 B
assets/nightlyBadges-8FX9dgn5.js (removed) 1.29 kB 🟢 -1.29 kB 🟢 -658 B 🟢 -587 B
assets/nightlyBadges-BwE-4440.js (new) 1.29 kB 🔴 +1.29 kB 🔴 +657 B 🔴 +582 B
assets/Load3dViewerContent--44XT8sY.js (removed) 1.23 kB 🟢 -1.23 kB 🟢 -567 B 🟢 -502 B
assets/Load3dViewerContent-DDMLDSQh.js (new) 1.23 kB 🔴 +1.23 kB 🔴 +565 B 🔴 +499 B
assets/SubscriptionPanelContentWorkspace-CefvsKfe.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -536 B 🟢 -470 B
assets/SubscriptionPanelContentWorkspace-ClgwMO9g.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +535 B 🔴 +465 B
assets/WidgetLegacy-_aZp7gNC.js (removed) 978 B 🟢 -978 B 🟢 -483 B 🟢 -427 B
assets/WidgetLegacy-72jeFNkE.js (new) 978 B 🔴 +978 B 🔴 +480 B 🔴 +424 B
assets/changeTracker-BYV1UtEg.js (removed) 952 B 🟢 -952 B 🟢 -471 B 🟢 -418 B
assets/changeTracker-CMHQVkRD.js (new) 952 B 🔴 +952 B 🔴 +470 B 🔴 +417 B

Status: 55 added / 55 removed / 79 unchanged

⚡ Performance Report

canvas-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 60.2 MB heap
canvas-mouse-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 56.1 MB heap
canvas-zoom-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 64.3 MB heap
dom-widget-clipping: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 46.7 MB heap
large-graph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 53.7 MB heap
large-graph-pan: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.0 MB heap
large-graph-zoom: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 61.5 MB heap
minimap-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 54.2 MB heap
subgraph-dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 47.4 MB heap
subgraph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 60.3 MB heap
subgraph-mouse-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 52.4 MB heap
viewport-pan-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 73.1 MB heap
vue-large-graph-idle: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 158.4 MB heap
vue-large-graph-pan: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 165.3 MB heap
workflow-execution: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 46.2 MB heap

⚠️ 4 regressions detected

Metric Baseline PR (median) Δ Sig
canvas-zoom-sweep: layout duration 1ms 1ms +28% ⚠️ z=2.9
subgraph-mouse-sweep: style recalc duration 44ms 50ms +14% ⚠️ z=2.3
subgraph-mouse-sweep: style recalc count 83 88 +6% ⚠️ z=4.1
subgraph-mouse-sweep: task duration 889ms 928ms +4% ⚠️ z=2.3
All metrics
Metric Baseline PR (median) Δ Sig
canvas-idle: avg frame time 17ms 17ms +0% z=-0.1
canvas-idle: p95 frame time 17ms 17ms +0%
canvas-idle: layout duration 0ms 0ms +0%
canvas-idle: style recalc duration 9ms 10ms +4% z=-1.4
canvas-idle: layout count 0 0 +0%
canvas-idle: style recalc count 10 11 +10% z=-0.3
canvas-idle: task duration 343ms 361ms +5% z=-1.1
canvas-idle: script duration 21ms 20ms -6% z=-2.5
canvas-idle: TBT 0ms 0ms +0%
canvas-idle: heap used 61.3 MB 60.2 MB -2%
canvas-idle: DOM nodes 20 22 +10% z=-0.5
canvas-idle: event listeners 6 6 +0% z=-1.2
canvas-mouse-sweep: avg frame time 17ms 17ms +0% z=-0.4
canvas-mouse-sweep: p95 frame time 17ms 17ms +0%
canvas-mouse-sweep: layout duration 3ms 4ms +5% z=0.1
canvas-mouse-sweep: style recalc duration 40ms 44ms +10% z=0.6
canvas-mouse-sweep: layout count 12 12 +0%
canvas-mouse-sweep: style recalc count 83 84 +1% z=2.3
canvas-mouse-sweep: task duration 944ms 968ms +3% z=1.8
canvas-mouse-sweep: script duration 132ms 137ms +4% z=0.2
canvas-mouse-sweep: TBT 0ms 0ms +0%
canvas-mouse-sweep: heap used 56.1 MB 56.1 MB +0%
canvas-mouse-sweep: DOM nodes 65 67 +3% z=1.8
canvas-mouse-sweep: event listeners 6 6 +0% z=-0.6
canvas-zoom-sweep: avg frame time 17ms 17ms -0% z=-1.1
canvas-zoom-sweep: p95 frame time 17ms 17ms +1%
canvas-zoom-sweep: layout duration 1ms 1ms +28% ⚠️ z=2.9
canvas-zoom-sweep: style recalc duration 18ms 19ms +4% z=-0.3
canvas-zoom-sweep: layout count 6 6 +0%
canvas-zoom-sweep: style recalc count 31 32 +3% z=1.6
canvas-zoom-sweep: task duration 295ms 308ms +5% z=-0.8
canvas-zoom-sweep: script duration 24ms 25ms +7% z=-0.6
canvas-zoom-sweep: TBT 0ms 0ms +0%
canvas-zoom-sweep: heap used 64.1 MB 64.3 MB +0%
canvas-zoom-sweep: DOM nodes 79 79 +0% z=-0.3
canvas-zoom-sweep: event listeners 19 19 +0% 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 11ms 10ms -6% z=-0.0
dom-widget-clipping: layout count 0 0 +0%
dom-widget-clipping: style recalc count 14 13 -7% z=-0.2
dom-widget-clipping: task duration 343ms 340ms -1% z=-1.5
dom-widget-clipping: script duration 66ms 69ms +5% z=0.4
dom-widget-clipping: TBT 0ms 0ms +0%
dom-widget-clipping: heap used 47.0 MB 46.7 MB -1%
dom-widget-clipping: DOM nodes 23 22 -4% z=-0.1
dom-widget-clipping: event listeners 2 2 +0% 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 11ms 12ms +3% z=-0.7
large-graph-idle: layout count 0 0 +0%
large-graph-idle: style recalc count 11 11 +0% z=-1.9
large-graph-idle: task duration 547ms 543ms -1% z=0.0
large-graph-idle: script duration 100ms 102ms +2% z=-0.0
large-graph-idle: TBT 0ms 0ms +0%
large-graph-idle: heap used 53.3 MB 53.7 MB +1%
large-graph-idle: DOM nodes -257 -255 -1% z=-309.4
large-graph-idle: event listeners -125 -125 +0% z=-24.8
large-graph-pan: avg frame time 17ms 17ms +0% z=0.3
large-graph-pan: p95 frame time 17ms 17ms +0%
large-graph-pan: layout duration 0ms 0ms +0%
large-graph-pan: style recalc duration 16ms 18ms +14% z=0.4
large-graph-pan: layout count 0 0 +0%
large-graph-pan: style recalc count 69 70 +1% z=0.7
large-graph-pan: task duration 1040ms 1093ms +5% z=0.2
large-graph-pan: script duration 382ms 405ms +6% z=-0.2
large-graph-pan: TBT 0ms 0ms +0%
large-graph-pan: heap used 64.4 MB 68.0 MB +6%
large-graph-pan: DOM nodes -261 -258 -1% z=-167.9
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 7ms 8ms +5%
large-graph-zoom: style recalc duration 17ms 18ms +6%
large-graph-zoom: layout count 60 60 +0%
large-graph-zoom: style recalc count 66 66 +0%
large-graph-zoom: task duration 1286ms 1337ms +4%
large-graph-zoom: script duration 483ms 491ms +2%
large-graph-zoom: TBT 0ms 0ms +0%
large-graph-zoom: heap used 59.3 MB 61.5 MB +4%
large-graph-zoom: DOM nodes -264 -263 -0%
large-graph-zoom: event listeners -125 -125 +0%
minimap-idle: avg frame time 17ms 17ms -0% z=-0.9
minimap-idle: p95 frame time 17ms 17ms +0%
minimap-idle: layout duration 0ms 0ms +0%
minimap-idle: style recalc duration 8ms 9ms +7% z=-0.7
minimap-idle: layout count 0 0 +0%
minimap-idle: style recalc count 10 9 -10% z=-0.8
minimap-idle: task duration 530ms 539ms +2% z=0.3
minimap-idle: script duration 93ms 98ms +5% z=0.0
minimap-idle: TBT 0ms 0ms +0%
minimap-idle: heap used 54.2 MB 54.2 MB +0%
minimap-idle: DOM nodes -259 -260 +0% z=-204.1
minimap-idle: event listeners -125 -127 +2% z=-199.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 12ms 12ms -4% z=-0.8
subgraph-dom-widget-clipping: layout count 0 0 +0%
subgraph-dom-widget-clipping: style recalc count 48 48 +0% z=0.1
subgraph-dom-widget-clipping: task duration 351ms 361ms +3% z=-0.9
subgraph-dom-widget-clipping: script duration 123ms 124ms +1% z=-0.6
subgraph-dom-widget-clipping: TBT 0ms 0ms +0%
subgraph-dom-widget-clipping: heap used 47.3 MB 47.4 MB +0%
subgraph-dom-widget-clipping: DOM nodes 22 22 +0% z=-0.2
subgraph-dom-widget-clipping: event listeners 8 8 +0% z=-1.4
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 10ms 10ms +6% z=-0.3
subgraph-idle: layout count 0 0 +0%
subgraph-idle: style recalc count 10 11 +10% z=0.2
subgraph-idle: task duration 339ms 343ms +1% z=-0.8
subgraph-idle: script duration 19ms 20ms +4% z=-0.3
subgraph-idle: TBT 0ms 0ms +0%
subgraph-idle: heap used 60.1 MB 60.3 MB +0%
subgraph-idle: DOM nodes 20 22 +10% z=0.1
subgraph-idle: event listeners 6 6 +0% variance too high
subgraph-mouse-sweep: avg frame time 17ms 17ms -0% z=-0.5
subgraph-mouse-sweep: p95 frame time 17ms 17ms -1%
subgraph-mouse-sweep: layout duration 5ms 4ms -8% z=-1.2
subgraph-mouse-sweep: style recalc duration 44ms 50ms +14% ⚠️ z=2.3
subgraph-mouse-sweep: layout count 16 16 +0%
subgraph-mouse-sweep: style recalc count 83 88 +6% ⚠️ z=4.1
subgraph-mouse-sweep: task duration 889ms 928ms +4% ⚠️ z=2.3
subgraph-mouse-sweep: script duration 101ms 106ms +5% z=0.8
subgraph-mouse-sweep: TBT 0ms 0ms +0%
subgraph-mouse-sweep: heap used 52.4 MB 52.4 MB +0%
subgraph-mouse-sweep: DOM nodes 72 73 +1% z=2.7
subgraph-mouse-sweep: event listeners 6 6 +0% variance too high
viewport-pan-sweep: avg frame time 17ms 17ms +0%
viewport-pan-sweep: p95 frame time 17ms 17ms +0%
viewport-pan-sweep: layout duration 0ms 0ms +0%
viewport-pan-sweep: style recalc duration 43ms 46ms +5%
viewport-pan-sweep: layout count 0 0 +0%
viewport-pan-sweep: style recalc count 251 252 +0%
viewport-pan-sweep: task duration 3858ms 3757ms -3%
viewport-pan-sweep: script duration 1418ms 1260ms -11%
viewport-pan-sweep: TBT 0ms 0ms +0%
viewport-pan-sweep: heap used 80.6 MB 73.1 MB -9%
viewport-pan-sweep: DOM nodes -256 -256 +0%
viewport-pan-sweep: event listeners -111 -111 +0%
vue-large-graph-idle: avg frame time 17ms 17ms +0%
vue-large-graph-idle: p95 frame time 17ms 17ms +0%
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 11961ms 12353ms +3%
vue-large-graph-idle: script duration 579ms 586ms +1%
vue-large-graph-idle: TBT 0ms 0ms +0%
vue-large-graph-idle: heap used 158.8 MB 158.4 MB -0%
vue-large-graph-idle: DOM nodes -8331 -8331 +0%
vue-large-graph-idle: event listeners -16462 -16464 +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 13ms 14ms +5%
vue-large-graph-pan: layout count 0 0 +0%
vue-large-graph-pan: style recalc count 67 65 -3%
vue-large-graph-pan: task duration 14383ms 14208ms -1%
vue-large-graph-pan: script duration 858ms 851ms -1%
vue-large-graph-pan: TBT 0ms 0ms +0%
vue-large-graph-pan: heap used 163.7 MB 165.3 MB +1%
vue-large-graph-pan: DOM nodes -8331 -8331 +0%
vue-large-graph-pan: event listeners -16464 -16460 -0%
workflow-execution: avg frame time 17ms 17ms +0% z=0.6
workflow-execution: p95 frame time 17ms 17ms +1%
workflow-execution: layout duration 1ms 2ms +8% z=-0.0
workflow-execution: style recalc duration 24ms 25ms +4% z=0.2
workflow-execution: layout count 5 5 +0% z=0.1
workflow-execution: style recalc count 17 21 +24% z=1.5
workflow-execution: task duration 119ms 133ms +12% z=1.0
workflow-execution: script duration 25ms 28ms +14% z=-0.3
workflow-execution: TBT 0ms 0ms +0%
workflow-execution: heap used 46.1 MB 46.2 MB +0%
workflow-execution: DOM nodes 156 159 +2% z=-0.3
workflow-execution: event listeners 71 71 +0% z=4.4
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-04-02T20:53:59.148Z",
  "gitSha": "97e6d7a590992e80c52fa7a4dffb7d6b05d934dd",
  "branch": "fix/subgraph-node-position-corruption",
  "measurements": [
    {
      "name": "canvas-idle",
      "durationMs": 2007.1170000000222,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 9.108000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 395.76900000000006,
      "heapDeltaBytes": -4420720,
      "heapUsedBytes": 43215800,
      "domNodes": 22,
      "jsHeapTotalBytes": 25165824,
      "scriptDurationMs": 20.357,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66999999999998,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "canvas-idle",
      "durationMs": 2034.7220000000448,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 9.549,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 342.638,
      "heapDeltaBytes": 20662844,
      "heapUsedBytes": 63868364,
      "domNodes": 18,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 16.181999999999995,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-idle",
      "durationMs": 1998.9520000000311,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 10.408000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 360.96999999999997,
      "heapDeltaBytes": 20148464,
      "heapUsedBytes": 63117628,
      "domNodes": 22,
      "jsHeapTotalBytes": 23330816,
      "scriptDurationMs": 19.668999999999993,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1793.4590000000128,
      "styleRecalcs": 75,
      "styleRecalcDurationMs": 39.641,
      "layouts": 12,
      "layoutDurationMs": 3.639,
      "taskDurationMs": 760.241,
      "heapDeltaBytes": 16019228,
      "heapUsedBytes": 58764536,
      "domNodes": 59,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 137.04,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 2016.2900000000263,
      "styleRecalcs": 85,
      "styleRecalcDurationMs": 47.34,
      "layouts": 12,
      "layoutDurationMs": 3.7799999999999994,
      "taskDurationMs": 982.8090000000001,
      "heapDeltaBytes": 15911652,
      "heapUsedBytes": 58953992,
      "domNodes": 67,
      "jsHeapTotalBytes": 23592960,
      "scriptDurationMs": 135.91,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 2017.898999999943,
      "styleRecalcs": 84,
      "styleRecalcDurationMs": 44.498999999999995,
      "layouts": 12,
      "layoutDurationMs": 3.417,
      "taskDurationMs": 968.215,
      "heapDeltaBytes": 16342236,
      "heapUsedBytes": 58819912,
      "domNodes": 67,
      "jsHeapTotalBytes": 23330816,
      "scriptDurationMs": 137.05100000000002,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1731.4830000000256,
      "styleRecalcs": 32,
      "styleRecalcDurationMs": 18.254999999999995,
      "layouts": 6,
      "layoutDurationMs": 0.773,
      "taskDurationMs": 296.71100000000007,
      "heapDeltaBytes": 24676680,
      "heapUsedBytes": 67472748,
      "domNodes": 78,
      "jsHeapTotalBytes": 20971520,
      "scriptDurationMs": 25.115999999999993,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1758.3889999999656,
      "styleRecalcs": 33,
      "styleRecalcDurationMs": 19.047,
      "layouts": 6,
      "layoutDurationMs": 0.6040000000000002,
      "taskDurationMs": 320.065,
      "heapDeltaBytes": 15432116,
      "heapUsedBytes": 67117448,
      "domNodes": 81,
      "jsHeapTotalBytes": 23330816,
      "scriptDurationMs": 25.273000000000003,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1722.5960000000669,
      "styleRecalcs": 30,
      "styleRecalcDurationMs": 18.648999999999997,
      "layouts": 6,
      "layoutDurationMs": 0.783,
      "taskDurationMs": 307.893,
      "heapDeltaBytes": 24699628,
      "heapUsedBytes": 67487272,
      "domNodes": 79,
      "jsHeapTotalBytes": 20447232,
      "scriptDurationMs": 26.038999999999998,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 542.412999999982,
      "styleRecalcs": 13,
      "styleRecalcDurationMs": 10.004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 338.824,
      "heapDeltaBytes": 6246748,
      "heapUsedBytes": 49009124,
      "domNodes": 22,
      "jsHeapTotalBytes": 13893632,
      "scriptDurationMs": 69.133,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 538.9700000000062,
      "styleRecalcs": 13,
      "styleRecalcDurationMs": 8.752999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 339.886,
      "heapDeltaBytes": 6428732,
      "heapUsedBytes": 49149324,
      "domNodes": 21,
      "jsHeapTotalBytes": 13107200,
      "scriptDurationMs": 66.44900000000001,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.669999999999998,
      "p95FrameDurationMs": 16.799999999999727
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 561.6679999999405,
      "styleRecalcs": 13,
      "styleRecalcDurationMs": 9.862,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 354.59,
      "heapDeltaBytes": 6172352,
      "heapUsedBytes": 49000568,
      "domNodes": 22,
      "jsHeapTotalBytes": 13631488,
      "scriptDurationMs": 70.15599999999999,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2046.96899999999,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 12.072,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 586.6210000000001,
      "heapDeltaBytes": 4964184,
      "heapUsedBytes": 55929080,
      "domNodes": -254,
      "jsHeapTotalBytes": 16183296,
      "scriptDurationMs": 109.16899999999998,
      "eventListeners": -121,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2057.643999999982,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 11.333,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 543.498,
      "heapDeltaBytes": 5218980,
      "heapUsedBytes": 56264024,
      "domNodes": -259,
      "jsHeapTotalBytes": 15921152,
      "scriptDurationMs": 102.08300000000001,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2033.2689999999047,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 11.501999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 543.1729999999999,
      "heapDeltaBytes": 17720492,
      "heapUsedBytes": 69694488,
      "domNodes": -255,
      "jsHeapTotalBytes": 14405632,
      "scriptDurationMs": 97.621,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2185.9329999999773,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 16.65,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1092.678,
      "heapDeltaBytes": 17475700,
      "heapUsedBytes": 71313748,
      "domNodes": -263,
      "jsHeapTotalBytes": 18747392,
      "scriptDurationMs": 401.49,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2129.542000000015,
      "styleRecalcs": 70,
      "styleRecalcDurationMs": 17.634000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1087.8609999999999,
      "heapDeltaBytes": 19280496,
      "heapUsedBytes": 71239420,
      "domNodes": -258,
      "jsHeapTotalBytes": 18485248,
      "scriptDurationMs": 404.58299999999997,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2195.363000000043,
      "styleRecalcs": 71,
      "styleRecalcDurationMs": 18.788,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1213.6670000000001,
      "heapDeltaBytes": 20719104,
      "heapUsedBytes": 72580268,
      "domNodes": -257,
      "jsHeapTotalBytes": 19009536,
      "scriptDurationMs": 467.697,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3145.339999999976,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 16.503999999999998,
      "layouts": 60,
      "layoutDurationMs": 7.454000000000001,
      "taskDurationMs": 1318.711,
      "heapDeltaBytes": -4401228,
      "heapUsedBytes": 59684100,
      "domNodes": -263,
      "jsHeapTotalBytes": 17346560,
      "scriptDurationMs": 487.713,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3211.3800000000197,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 17.768000000000004,
      "layouts": 60,
      "layoutDurationMs": 7.645,
      "taskDurationMs": 1405.152,
      "heapDeltaBytes": 9511724,
      "heapUsedBytes": 64521792,
      "domNodes": -262,
      "jsHeapTotalBytes": 18280448,
      "scriptDurationMs": 563.308,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3198.7589999999955,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 17.631999999999998,
      "layouts": 60,
      "layoutDurationMs": 7.543,
      "taskDurationMs": 1337.153,
      "heapDeltaBytes": 10760956,
      "heapUsedBytes": 66641512,
      "domNodes": -264,
      "jsHeapTotalBytes": 16240640,
      "scriptDurationMs": 490.83,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "minimap-idle",
      "durationMs": 2049.9829999999974,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 10.875,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 539.4300000000001,
      "heapDeltaBytes": 3990200,
      "heapUsedBytes": 56494980,
      "domNodes": -260,
      "jsHeapTotalBytes": 16183296,
      "scriptDurationMs": 98.258,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "minimap-idle",
      "durationMs": 2040.3929999999946,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.728,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 547.2680000000001,
      "heapDeltaBytes": 4565496,
      "heapUsedBytes": 56819296,
      "domNodes": -260,
      "jsHeapTotalBytes": 15921152,
      "scriptDurationMs": 101.19300000000001,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "minimap-idle",
      "durationMs": 2019.9620000000778,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.955000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 522.062,
      "heapDeltaBytes": 16793844,
      "heapUsedBytes": 70661556,
      "domNodes": -261,
      "jsHeapTotalBytes": 16183296,
      "scriptDurationMs": 90.532,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 555.5669999999964,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 11.96,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 358.302,
      "heapDeltaBytes": 6760256,
      "heapUsedBytes": 49718112,
      "domNodes": 22,
      "jsHeapTotalBytes": 13369344,
      "scriptDurationMs": 123.88499999999999,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 556.7020000000298,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 12.327,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 361.606,
      "heapDeltaBytes": 6427348,
      "heapUsedBytes": 49680588,
      "domNodes": 22,
      "jsHeapTotalBytes": 13107200,
      "scriptDurationMs": 124.16099999999999,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.669999999999998,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 576.7609999999195,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 11.678,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 360.90299999999996,
      "heapDeltaBytes": 6543960,
      "heapUsedBytes": 49583144,
      "domNodes": 21,
      "jsHeapTotalBytes": 13369344,
      "scriptDurationMs": 126.54099999999998,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2016.4219999999773,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 10.48,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 357.236,
      "heapDeltaBytes": 11235616,
      "heapUsedBytes": 63427788,
      "domNodes": 22,
      "jsHeapTotalBytes": 25952256,
      "scriptDurationMs": 20.319999999999997,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2024.815999999987,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 10.181000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 343.486,
      "heapDeltaBytes": 20139272,
      "heapUsedBytes": 63271356,
      "domNodes": 23,
      "jsHeapTotalBytes": 22544384,
      "scriptDurationMs": 19.551000000000005,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-idle",
      "durationMs": 1996.7599999999948,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 9.067,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 343.36899999999997,
      "heapDeltaBytes": 19894652,
      "heapUsedBytes": 62989088,
      "domNodes": 20,
      "jsHeapTotalBytes": 22544384,
      "scriptDurationMs": 18.322000000000003,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1983.3830000000034,
      "styleRecalcs": 83,
      "styleRecalcDurationMs": 50.367999999999995,
      "layouts": 16,
      "layoutDurationMs": 3.9880000000000004,
      "taskDurationMs": 905.3209999999999,
      "heapDeltaBytes": 11902664,
      "heapUsedBytes": 54961688,
      "domNodes": 71,
      "jsHeapTotalBytes": 22544384,
      "scriptDurationMs": 102.304,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 2027.3150000000442,
      "styleRecalcs": 88,
      "styleRecalcDurationMs": 50.127,
      "layouts": 16,
      "layoutDurationMs": 4.868,
      "taskDurationMs": 956.2300000000001,
      "heapDeltaBytes": 11913452,
      "heapUsedBytes": 54948032,
      "domNodes": 74,
      "jsHeapTotalBytes": 22544384,
      "scriptDurationMs": 105.818,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 2000.0609999999597,
      "styleRecalcs": 88,
      "styleRecalcDurationMs": 46.631,
      "layouts": 16,
      "layoutDurationMs": 4.3149999999999995,
      "taskDurationMs": 927.5730000000001,
      "heapDeltaBytes": 11869472,
      "heapUsedBytes": 54617292,
      "domNodes": 73,
      "jsHeapTotalBytes": 22544384,
      "scriptDurationMs": 107.723,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8230.79899999999,
      "styleRecalcs": 252,
      "styleRecalcDurationMs": 45.608000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3922.6330000000003,
      "heapDeltaBytes": 30588656,
      "heapUsedBytes": 81614436,
      "domNodes": -255,
      "jsHeapTotalBytes": 24776704,
      "scriptDurationMs": 1428.107,
      "eventListeners": -109,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8175.782999999967,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 43.976,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3706.33,
      "heapDeltaBytes": 25993236,
      "heapUsedBytes": 76670236,
      "domNodes": -257,
      "jsHeapTotalBytes": 21106688,
      "scriptDurationMs": 1257.706,
      "eventListeners": -111,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8219.03199999997,
      "styleRecalcs": 252,
      "styleRecalcDurationMs": 46.033,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3757.2699999999995,
      "heapDeltaBytes": 21747552,
      "heapUsedBytes": 72173172,
      "domNodes": -256,
      "jsHeapTotalBytes": 21368832,
      "scriptDurationMs": 1259.5,
      "eventListeners": -111,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12359.369000000015,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12346.63,
      "heapDeltaBytes": -28184716,
      "heapUsedBytes": 166078764,
      "domNodes": -8331,
      "jsHeapTotalBytes": 27090944,
      "scriptDurationMs": 582.427,
      "eventListeners": -16464,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333358,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12481.79799999997,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12470.949,
      "heapDeltaBytes": -34392600,
      "heapUsedBytes": 166322188,
      "domNodes": -8333,
      "jsHeapTotalBytes": 24207360,
      "scriptDurationMs": 585.538,
      "eventListeners": -16464,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333358,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12366.416999999956,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12353.012,
      "heapDeltaBytes": -31038508,
      "heapUsedBytes": 166120076,
      "domNodes": -8331,
      "jsHeapTotalBytes": 27090944,
      "scriptDurationMs": 602.3320000000001,
      "eventListeners": -16462,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.780000000000047,
      "p95FrameDurationMs": 16.80000000000291
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14231.569999999976,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 13.881000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14208.297999999999,
      "heapDeltaBytes": -6586648,
      "heapUsedBytes": 186909028,
      "domNodes": -8331,
      "jsHeapTotalBytes": 26742784,
      "scriptDurationMs": 846.461,
      "eventListeners": -16462,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.776666666666642,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14345.949999999959,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 14.037999999999995,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14324.529000000002,
      "heapDeltaBytes": -31645052,
      "heapUsedBytes": 160542976,
      "domNodes": -8331,
      "jsHeapTotalBytes": -2269184,
      "scriptDurationMs": 870.709,
      "eventListeners": -16456,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14135.51700000005,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 13.74900000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14113.362000000001,
      "heapDeltaBytes": -22291132,
      "heapUsedBytes": 173305580,
      "domNodes": -8333,
      "jsHeapTotalBytes": 24645632,
      "scriptDurationMs": 850.7130000000001,
      "eventListeners": -16460,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "workflow-execution",
      "durationMs": 471.78199999996195,
      "styleRecalcs": 23,
      "styleRecalcDurationMs": 24.657,
      "layouts": 5,
      "layoutDurationMs": 1.5469999999999997,
      "taskDurationMs": 133.22299999999998,
      "heapDeltaBytes": 4779332,
      "heapUsedBytes": 49875480,
      "domNodes": 192,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 28.220999999999997,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "workflow-execution",
      "durationMs": 459.68599999991966,
      "styleRecalcs": 21,
      "styleRecalcDurationMs": 28.880000000000003,
      "layouts": 6,
      "layoutDurationMs": 1.6489999999999998,
      "taskDurationMs": 137.07599999999996,
      "heapDeltaBytes": 4555392,
      "heapUsedBytes": 48163324,
      "domNodes": 159,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 33.717,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "workflow-execution",
      "durationMs": 444.753999999989,
      "styleRecalcs": 16,
      "styleRecalcDurationMs": 21.679000000000002,
      "layouts": 5,
      "layoutDurationMs": 1.2389999999999999,
      "taskDurationMs": 118.15900000000002,
      "heapDeltaBytes": 4516504,
      "heapUsedBytes": 48401148,
      "domNodes": 156,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 26.390000000000004,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    }
  ]
}

coderabbitai[bot]
coderabbitai bot previously approved these changes Apr 2, 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)
browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts (1)

35-46: Draft persistence poll could be more specific.

The poll checks for any draft key starting with 'Comfy.Workflow.Draft.v2:'. If test isolation is imperfect (e.g., stale localStorage from a prior run), this could pass prematurely before the current workflow's draft is written. Consider polling for the draft content to contain expected node IDs, or verifying key count increased.

That said, if the fixture guarantees a clean browser context, this is acceptable.

🔧 Optional: more robust draft detection
       // Wait for the debounced draft persistence to flush to localStorage
       await expect
         .poll(
           () =>
-            comfyPage.page.evaluate(() =>
-              Object.keys(localStorage).some((k) =>
-                k.startsWith('Comfy.Workflow.Draft.v2:')
-              )
-            ),
+            comfyPage.page.evaluate(() => {
+              const draftKeys = Object.keys(localStorage).filter((k) =>
+                k.startsWith('Comfy.Workflow.Draft.v2:')
+              )
+              // Verify at least one draft exists and contains subgraph data
+              return draftKeys.some((k) => {
+                const raw = localStorage.getItem(k)
+                return raw && raw.includes('subgraphs')
+              })
+            }),
           { timeout: 3000 }
         )
         .toBe(true)
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts` around lines 35
- 46, The poll currently checks any localStorage key starting with
'Comfy.Workflow.Draft.v2:' which can pass due to stale data; update the poll
(the comfyPage.page.evaluate call) to locate the specific draft key for the
current workflow (or read all matching keys and verify the stored draft JSON
contains the expected node IDs or that the matching-key count increased), and
assert that the draft value includes those expected node IDs (or matches the
current workflowId) before returning true so the test only proceeds once the
correct draft was persisted.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Nitpick comments:
In `@browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts`:
- Around line 35-46: The poll currently checks any localStorage key starting
with 'Comfy.Workflow.Draft.v2:' which can pass due to stale data; update the
poll (the comfyPage.page.evaluate call) to locate the specific draft key for the
current workflow (or read all matching keys and verify the stored draft JSON
contains the expected node IDs or that the matching-key count increased), and
assert that the draft value includes those expected node IDs (or matches the
current workflowId) before returning true so the test only proceeds once the
correct draft was persisted.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 380ae82c-e80e-48f5-94ec-4b64b57935cc

📥 Commits

Reviewing files that changed from the base of the PR and between fa0239a and 8b9d2d0.

📒 Files selected for processing (1)
  • browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts

coderabbitai[bot]
coderabbitai bot previously approved these changes Apr 2, 2026
Use page.reload() with explicit draft persistence polling instead of
comfyPage.setup(), which triggered a full navigation that bypassed
the draft auto-load flow.
@jaeone94 jaeone94 force-pushed the fix/subgraph-node-position-corruption branch from 8b9d2d0 to 1c429d2 Compare April 2, 2026 19:15
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)
browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts (1)

39-50: Make draft-flush waiting assert a real write, not just key presence.

This poll can pass on a pre-existing Comfy.Workflow.Draft.v2:* key, so reload may happen before this test’s latest state is actually persisted. Consider waiting for the draft payload to change.

Proposed hardening
+      const draftSnapshotBefore = await comfyPage.page.evaluate(() => {
+        const key = Object.keys(localStorage).find((k) =>
+          k.startsWith('Comfy.Workflow.Draft.v2:')
+        )
+        return key
+          ? { key, value: localStorage.getItem(key) }
+          : { key: null, value: null }
+      })
+
       // Wait for the debounced draft persistence to flush to localStorage
       await expect
         .poll(
           () =>
-            comfyPage.page.evaluate(() =>
-              Object.keys(localStorage).some((k) =>
-                k.startsWith('Comfy.Workflow.Draft.v2:')
-              )
-            ),
+            comfyPage.page.evaluate((before) => {
+              const key =
+                before.key ??
+                Object.keys(localStorage).find((k) =>
+                  k.startsWith('Comfy.Workflow.Draft.v2:')
+                )
+              if (!key) return false
+              const value = localStorage.getItem(key)
+              return value !== null && value !== before.value
+            }, draftSnapshotBefore),
           { timeout: 3000 }
         )
         .toBe(true)
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts` around lines 39
- 50, The current poll only checks for existence of a Comfy.Workflow.Draft.v2:*
key and can succeed on a stale pre-existing key; update the wait to assert a
real write by first snapshotting the current draft payload via
comfyPage.page.evaluate (reading value(s) for keys that
startWith('Comfy.Workflow.Draft.v2:')) and then poll until the payload for that
key(s) changes (or contains expected content from the test) instead of just
toBe(true); modify the poll caller around comfyPage.page.evaluate and the
localStorage check so it compares previousValue !== currentValue (or verifies
the expected serialized draft fields) before proceeding.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Nitpick comments:
In `@browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts`:
- Around line 39-50: The current poll only checks for existence of a
Comfy.Workflow.Draft.v2:* key and can succeed on a stale pre-existing key;
update the wait to assert a real write by first snapshotting the current draft
payload via comfyPage.page.evaluate (reading value(s) for keys that
startWith('Comfy.Workflow.Draft.v2:')) and then poll until the payload for that
key(s) changes (or contains expected content from the test) instead of just
toBe(true); modify the poll caller around comfyPage.page.evaluate and the
localStorage check so it compares previousValue !== currentValue (or verifies
the expected serialized draft fields) before proceeding.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 7df9fbe3-840d-4874-8eb9-adcb1a7e5e56

📥 Commits

Reviewing files that changed from the base of the PR and between 8b9d2d0 and 1c429d2.

📒 Files selected for processing (1)
  • browser_tests/tests/subgraph/subgraphDraftPositions.spec.ts

coderabbitai[bot]
coderabbitai bot previously approved these changes Apr 2, 2026
@jaeone94 jaeone94 requested a review from DrJKL April 2, 2026 19:48
k.startsWith('Comfy.Workflow.Draft.v2:')
)
),
{ timeout: 3000 }
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

I would keep the default 5s here.
It's rare that you want tighter timing constraints for things like this.

Copy link
Copy Markdown
Collaborator Author

Choose a reason for hiding this comment

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

Good call, changed to default 5s

await expect
.poll(
() =>
comfyPage.page.evaluate(() =>
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

Optional: this seems like it would make sense to put into the page object, if you can find a good name.

Copy link
Copy Markdown
Collaborator Author

Choose a reason for hiding this comment

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

Extracted to WorkflowHelper.waitForDraftPersisted().

…overlay

canvas.click() gets intercepted by DOM widget textarea after the
position fix places nodes at correct locations. dispatchEvent bypasses
the pointer event interception check.
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)
browser_tests/fixtures/helpers/SubgraphHelper.ts (1)

487-496: Extract repeated synthetic canvas click dispatch into a private helper.

packAllInteriorNodes duplicates the same pointerdown/pointerup sequence twice. A small helper keeps this deterministic event payload in one place and avoids drift later.

Proposed refactor
 export class SubgraphHelper {
   constructor(private readonly comfyPage: ComfyPage) {}
+
+  private async dispatchPrimaryCanvasPointerClick(): Promise<void> {
+    const eventInit = {
+      bubbles: true,
+      cancelable: true,
+      button: 0
+    }
+    await this.comfyPage.canvas.dispatchEvent('pointerdown', eventInit)
+    await this.comfyPage.canvas.dispatchEvent('pointerup', eventInit)
+  }
@@
   async packAllInteriorNodes(hostNodeId: string): Promise<void> {
@@
-    await this.comfyPage.canvas.dispatchEvent('pointerdown', {
-      bubbles: true,
-      cancelable: true,
-      button: 0
-    })
-    await this.comfyPage.canvas.dispatchEvent('pointerup', {
-      bubbles: true,
-      cancelable: true,
-      button: 0
-    })
+    await this.dispatchPrimaryCanvasPointerClick()
@@
-    await this.comfyPage.canvas.dispatchEvent('pointerdown', {
-      bubbles: true,
-      cancelable: true,
-      button: 0
-    })
-    await this.comfyPage.canvas.dispatchEvent('pointerup', {
-      bubbles: true,
-      cancelable: true,
-      button: 0
-    })
+    await this.dispatchPrimaryCanvasPointerClick()

As per coding guidelines, “Watch out for Code Smells and refactor to avoid them.”

Also applies to: 505-514

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

In `@browser_tests/fixtures/helpers/SubgraphHelper.ts` around lines 487 - 496,
packAllInteriorNodes currently repeats the same synthetic pointerdown/pointerup
sequence (this.comfyPage.canvas.dispatchEvent with bubbles/cancelable/button) in
two places (around the shown block and again at lines 505-514); extract that
logic into a private helper method (e.g., private dispatchCanvasClick or private
clickCanvas) on SubgraphHelper that accepts the target element
(this.comfyPage.canvas) and dispatches pointerdown then pointerup with the
deterministic payload, then replace both duplicated blocks in
packAllInteriorNodes with calls to that helper to centralize the event payload
and avoid duplication.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Nitpick comments:
In `@browser_tests/fixtures/helpers/SubgraphHelper.ts`:
- Around line 487-496: packAllInteriorNodes currently repeats the same synthetic
pointerdown/pointerup sequence (this.comfyPage.canvas.dispatchEvent with
bubbles/cancelable/button) in two places (around the shown block and again at
lines 505-514); extract that logic into a private helper method (e.g., private
dispatchCanvasClick or private clickCanvas) on SubgraphHelper that accepts the
target element (this.comfyPage.canvas) and dispatches pointerdown then pointerup
with the deterministic payload, then replace both duplicated blocks in
packAllInteriorNodes with calls to that helper to centralize the event payload
and avoid duplication.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 309f14f1-4433-4444-8f90-0e51496c75f0

📥 Commits

Reviewing files that changed from the base of the PR and between 1c429d2 and 38d6a31.

📒 Files selected for processing (1)
  • browser_tests/fixtures/helpers/SubgraphHelper.ts

coderabbitai[bot]
coderabbitai bot previously approved these changes Apr 2, 2026
Move draft localStorage polling into WorkflowHelper for reuse.
Use default 5s timeout instead of 3s per review feedback.
@jaeone94 jaeone94 requested a review from DrJKL April 2, 2026 20:54
@artokun
Copy link
Copy Markdown
Contributor

artokun commented Apr 2, 2026

Nice fix! We have a complementary one-file fix in #10810 that addresses a related but separate issue — restoreViewport() calling fitView() on cache miss, which races with loadGraphData's own viewport restore on template load.

Zero file overlap between our PRs (yours touches useVueNodeResizeTracking.ts, ours touches subgraphNavigationStore.ts), so no conflicts.

If you want to bundle them, cherry-pick commands:

git fetch https://github.com/artokun/ComfyUI_frontend.git fix/viewport-template-override
git cherry-pick df7c4c5ad  # fix: use public nodes getter, remove redundant variable
git cherry-pick 650834d0d  # fix: don't override loadGraphData viewport on cache miss

Or we can rebase #10810 onto your branch as the base. Either way works — happy to help land both fixes together.

@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai bot commented Apr 2, 2026

Caution

Docstrings generation - FAILED

No docstrings were generated.

@jaeone94
Copy link
Copy Markdown
Collaborator Author

jaeone94 commented Apr 3, 2026

@artokun

Thanks so much for looking into this and offering to bundle them together — really appreciate the collaboration! 😍

Since both PRs have zero file overlap and address the same viewport transition problem from different angles (yours fixes restoreViewport racing with loadGraphData, ours fixes ResizeObserver using stale viewport state), I think both fixes are independently necessary and complement each other well.

I'd prefer to keep them as separate PRs if that's okay — keeping the scope narrow makes it easier to reason about each fix in isolation, and safer to revert independently if either one causes unexpected issues down the road.

Looking forward to seeing #10810 land as well — together they should make viewport transitions much more robust!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

size:L This PR changes 100-499 lines, ignoring generated files.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants