Skip to content

fix: RAF-batch Vue node ResizeObserver writes to layoutStore#12299

Draft
christian-byrne wants to merge 3 commits into
mainfrom
glary/raf-batch-vue-node-resize-tracking
Draft

fix: RAF-batch Vue node ResizeObserver writes to layoutStore#12299
christian-byrne wants to merge 3 commits into
mainfrom
glary/raf-batch-vue-node-resize-tracking

Conversation

@christian-byrne
Copy link
Copy Markdown
Contributor

@christian-byrne christian-byrne commented May 15, 2026

PR Created by the Glary-Bot Agent


Summary

When the bottom panel opens (or any container animates resizing the canvas), Vue workflow nodes can briefly flicker / appear displaced before the view settles. Repro: open the console bottom panel after a job completes — the canvas goes blank/dark for ~1-2s while the splitter animates.

Root cause: the shared ResizeObserver in useVueNodeResizeTracking.ts runs its callback synchronously on every fire. During an animated splitter drag the RO fires multiple times per second, each time computing positions (sometimes via clientPosToCanvasPos, which reads lgCanvas.ds.offset/scale) and immediately writing layoutStore.batchUpdateNodeBounds(...). If the canvas RO hasn't yet updated ds for the new container size, the conversion is stale.

Change

Mirror the pattern already used by the sibling useSlotElementTracking.ts:

  • ResizeObserver callback only stashes the latest borderBoxSize per element into a pendingMeasurements map and schedules a createRafBatch flush.
  • The RAF flush does all layoutStore reads, position computation, and writes once per frame.
  • Multiple resizes of the same element within a frame coalesce into a single write at the most recent size.
  • widgets-grid signal entries continue to route directly to scheduleSlotLayoutSync (which is already RAF-batched), preserving single-RAF latency and existing linearMode / hidden-tab gating.

Tests

Existing 6 tests pass (with a controllable rafBatch mock so each test flushes synchronously). Added two regression tests:

  • defers layoutStore writes until the next animation frame — RO callback alone does not write; flush triggers the write.
  • coalesces successive resizes for the same node into one write per frame — two RO callbacks for the same element, only one write with the most recent size.

Verification

  • pnpm test:unit -- src/renderer/extensions/vueNodes/composables/ → 16/16 pass (this file + sibling slot tracking)
  • oxfmt, eslint, pnpm typecheck (via pre-commit) clean

Note

This is one of three planned PRs that work together to fix the bottom-panel/node-displacement bug. The other two will: (2) RAF-batch the canvas-element useResizeObserver in app.ts to stop link redraw flicker during splitter drags, and (3) add an explicit bottomPanelVisible watcher as belt-and-braces for the splitter transition.

┆Issue is synchronized with this Notion page by Unito

Glary-Bot added 2 commits May 15, 2026 20:55
Coalesces multiple ResizeObserver callbacks fired during the same frame
into a single layoutStore write, mirroring the pattern already used by
useSlotElementTracking. Defers measurement/write until next RAF so the
canvas ResizeObserver has had a chance to update lgCanvas.ds before any
DOM->canvas conversion fallback runs.

Fixes the case where opening the bottom panel during an animated splitter
resize causes Vue workflow nodes to flicker / appear displaced for ~1-2s
before the view settles, because the splitter's per-frame size changes
each triggered a synchronous layoutStore write with stale viewport state.
… hop

Address review: widgets-grid ResizeObserver signals were being staged
in this file's RAF batch and then forwarded to scheduleSlotLayoutSync(),
which RAF-batches a second time. Route them directly inside the RO
callback, restoring single-RAF latency for slot drift correction and
preserving the prior linearMode/hidden-tab gating behavior that the
slot tracker already handles.
@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai Bot commented May 15, 2026

Review Change Stack

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: 572cc89e-6014-4a98-a1cd-1cd37b7b6504

📥 Commits

Reviewing files that changed from the base of the PR and between 48a3269 and 63d62bc.

📒 Files selected for processing (2)
  • src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.test.ts
  • src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts
🚧 Files skipped from review as they are similar to previous changes (1)
  • src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts

📝 Walkthrough

Walkthrough

ResizeObserver measurements are queued into pendingMeasurements and flushed on the next animation frame via createRafBatch; the observer callback now enqueues measurements (with a widgets-grid signal path) and tests mock a controllable RAF batch and document visibility to validate deferred writes, coalescing, and hidden-tab behavior.

Changes

ResizeObserver RAF batching

Layer / File(s) Summary
Test mock and reset infrastructure
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.test.ts
Hoisted visibilityState ref for useDocumentVisibility, mock createRafBatch with rafBatchState (pending/flush/isScheduled/schedule/cancel), and beforeEach resets RAF pending and visibility.
Pending measurements and RAF batch wiring
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts
Import createRafBatch, add PendingMeasurement shape, pendingMeasurements Map, instantiate RAF batch, and update visibility watcher to defer/cancel queued measurements and re-observe deferred elements on visibility change.
Flush routine and processing loop
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts
Add flushPendingMeasurements with early exits for empty queue and linearMode, hidden-tab handling that defers pending elements, iterate queued (element, measurement) pairs to compute width/height from the queue, perform layout-store updates, and clear the queue after flush.
ResizeObserver callback refactor
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts
Rewrite RO callback to exit in linearMode, route widgets-grid signal-only entries to slot-layout sync, enqueue measured width/height into pendingMeasurements, and schedule RAF batch when queue becomes non-empty.
Test updates and new verification cases
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.test.ts
Update tests to call rafBatchState.flush() after RO callbacks; add tests for deferred layout-store writes until RAF flush, coalescing successive resizes into one bounds write + one slot-layout sync per frame, and deferral/unobserve when tab hides before flush.

Estimated code review effort

🎯 4 (Complex) | ⏱️ ~45 minutes

Possibly related PRs

  • Comfy-Org/ComfyUI_frontend#12214: Both PRs modify the widgets-grid signal-only path in useVueNodeResizeTracking.ts, coordinating RAF-batched resize handling with widget-link-position fixes.

Suggested labels

size:M

Suggested reviewers

  • ltdrdata

Poem

🐰 I queue the hops, then wait for light,
Frames gather measurements snug and tight,
One flush, one write, no frantic race,
Coalesced bounds find their proper place,
A rabbit's rhyme for tidy layout grace.


Caution

Pre-merge checks failed

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

  • Ignore (reviewers only)

❌ Failed checks (1 error, 1 warning)

Check name Status Explanation Resolution
End-To-End Regression Coverage For Fixes ❌ Error PR title uses "fix:" language and changes src/ code without browser_tests/ coverage or explanation of why E2E test was not added. Add Playwright regression test under browser_tests/ for node flicker/displacement during animated splitter drags, OR explain in PR description why E2E test is not practical.
Docstring Coverage ⚠️ Warning Docstring coverage is 0.00% which is insufficient. The required threshold is 80.00%. Write docstrings for the functions missing them to satisfy the coverage threshold.
✅ Passed checks (5 passed)
Check name Status Explanation
Title check ✅ Passed The title accurately summarizes the main change: RAF-batching Vue node ResizeObserver writes to layoutStore, which is the core refactoring described in the PR.
Description check ✅ Passed The description is comprehensive, covering summary, root cause, implementation details, test coverage, verification steps, and future PRs. It follows the template structure with clear sections.
Linked Issues check ✅ Passed Check skipped because no linked issues were found for this pull request.
Out of Scope Changes check ✅ Passed Check skipped because no linked issues were found for this pull request.
Adr Compliance For Entity/Litegraph Changes ✅ Passed ADR compliance check does not apply. PR modifies only renderer-layer Vue composables under src/renderer/extensions/vueNodes/composables/, not src/lib/litegraph/, src/ecs/, or graph entities.
✨ Finishing Touches
📝 Generate docstrings
  • Create stacked PR
  • Commit on current branch
🧪 Generate unit tests (beta)
  • Create PR with unit tests
  • Commit unit tests in branch glary/raf-batch-vue-node-resize-tracking

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

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 15, 2026

🎨 Storybook: ✅ Built — View Storybook

Details

⏰ Completed at: 05/15/2026, 09:22:01 PM UTC

Links

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 15, 2026

🎭 Playwright: ✅ 1604 passed, 0 failed · 2 flaky

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

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 15, 2026

📦 Bundle: 5.36 MB gzip 🔴 +243 B

Details

Summary

  • Raw size: 24.7 MB baseline 24.7 MB — 🔴 +799 B
  • Gzip: 5.36 MB baseline 5.36 MB — 🔴 +243 B
  • Brotli: 4.15 MB baseline 4.14 MB — 🔴 +372 B
  • Bundles: 268 current • 268 baseline • 118 added / 118 removed

Category Glance
Data & Services 🔴 +799 B (3.16 MB) · Vendor & Third-Party ⚪ 0 B (9.94 MB) · Other ⚪ 0 B (9.16 MB) · Graph Workspace ⚪ 0 B (1.24 MB) · Panels & Settings ⚪ 0 B (527 kB) · Utilities & Hooks ⚪ 0 B (366 kB) · + 5 more

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

Main entry bundles and manifests

File Before After Δ Raw Δ Gzip Δ Brotli
assets/index-CkgkNRXf.js (removed) 26.1 kB 🟢 -26.1 kB 🟢 -8.76 kB 🟢 -7.53 kB
assets/index-D2SUhgDX.js (new) 26.1 kB 🔴 +26.1 kB 🔴 +8.76 kB 🔴 +7.52 kB

Status: 1 added / 1 removed

Graph Workspace — 1.24 MB (baseline 1.24 MB) • ⚪ 0 B

Graph editor runtime, canvas, workflow orchestration

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

Status: 1 added / 1 removed

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

Top-level views, pages, and routed surfaces

File Before After Δ Raw Δ Gzip Δ Brotli
assets/CloudSurveyView-Bo9__MAY.js (removed) 19.6 kB 🟢 -19.6 kB 🟢 -5.14 kB 🟢 -4.56 kB
assets/CloudSurveyView-CzK3DG85.js (new) 19.6 kB 🔴 +19.6 kB 🔴 +5.14 kB 🔴 +4.57 kB
assets/CloudLoginView-Bi9I6YXz.js (removed) 12.5 kB 🟢 -12.5 kB 🟢 -3.55 kB 🟢 -3.12 kB
assets/CloudLoginView-D7nQp6wd.js (new) 12.5 kB 🔴 +12.5 kB 🔴 +3.55 kB 🔴 +3.12 kB
assets/CloudSignupView-CPX5BFNj.js (new) 10.4 kB 🔴 +10.4 kB 🔴 +3.06 kB 🔴 +2.7 kB
assets/CloudSignupView-ZR5csnP4.js (removed) 10.4 kB 🟢 -10.4 kB 🟢 -3.05 kB 🟢 -2.7 kB
assets/UserCheckView-DNRlpq7t.js (new) 9.07 kB 🔴 +9.07 kB 🔴 +2.34 kB 🔴 +2.05 kB
assets/UserCheckView-DQbAr4mn.js (removed) 9.07 kB 🟢 -9.07 kB 🟢 -2.34 kB 🟢 -2.05 kB
assets/CloudLayoutView-DijcZx5U.js (new) 7.81 kB 🔴 +7.81 kB 🔴 +2.49 kB 🔴 +2.17 kB
assets/CloudLayoutView-QGe6cCAk.js (removed) 7.81 kB 🟢 -7.81 kB 🟢 -2.48 kB 🟢 -2.18 kB
assets/CloudForgotPasswordView-B_FISXlv.js (new) 6.22 kB 🔴 +6.22 kB 🔴 +2.22 kB 🔴 +1.96 kB
assets/CloudForgotPasswordView-CBaei9xW.js (removed) 6.22 kB 🟢 -6.22 kB 🟢 -2.21 kB 🟢 -1.95 kB
assets/CloudAuthTimeoutView-ChMoGWeu.js (new) 5.58 kB 🔴 +5.58 kB 🔴 +2.05 kB 🔴 +1.8 kB
assets/CloudAuthTimeoutView-DywCyPRm.js (removed) 5.58 kB 🟢 -5.58 kB 🟢 -2.05 kB 🟢 -1.8 kB
assets/CloudSubscriptionRedirectView-BtWYkg5n.js (removed) 5.36 kB 🟢 -5.36 kB 🟢 -2.03 kB 🟢 -1.81 kB
assets/CloudSubscriptionRedirectView-BWI_pvBH.js (new) 5.36 kB 🔴 +5.36 kB 🔴 +2.03 kB 🔴 +1.79 kB
assets/UserSelectView-BBMPjVRD.js (new) 4.7 kB 🔴 +4.7 kB 🔴 +1.75 kB 🔴 +1.54 kB
assets/UserSelectView-CkNUY6A_.js (removed) 4.7 kB 🟢 -4.7 kB 🟢 -1.75 kB 🟢 -1.55 kB

Status: 9 added / 9 removed / 2 unchanged

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

Configuration panels, inspectors, and settings screens

File Before After Δ Raw Δ Gzip Δ Brotli
assets/KeybindingPanel-67YmZ_Rz.js (new) 47.5 kB 🔴 +47.5 kB 🔴 +9.87 kB 🔴 +8.76 kB
assets/KeybindingPanel-CG2KY29x.js (removed) 47.5 kB 🟢 -47.5 kB 🟢 -9.87 kB 🟢 -8.75 kB
assets/SecretsPanel-1YqIEP4P.js (removed) 23.9 kB 🟢 -23.9 kB 🟢 -5.73 kB 🟢 -5.06 kB
assets/SecretsPanel-Cwu-mEhe.js (new) 23.9 kB 🔴 +23.9 kB 🔴 +5.73 kB 🔴 +5.05 kB
assets/LegacyCreditsPanel-CgYaWNiu.js (removed) 21.8 kB 🟢 -21.8 kB 🟢 -5.94 kB 🟢 -5.24 kB
assets/LegacyCreditsPanel-DNnyDVpY.js (new) 21.8 kB 🔴 +21.8 kB 🔴 +5.95 kB 🔴 +5.25 kB
assets/SubscriptionPanel-1eC1SxWj.js (new) 20.2 kB 🔴 +20.2 kB 🔴 +5.18 kB 🔴 +4.55 kB
assets/SubscriptionPanel-COrS3r9k.js (removed) 20.2 kB 🟢 -20.2 kB 🟢 -5.19 kB 🟢 -4.55 kB
assets/AboutPanel-_P4q24ls.js (new) 12 kB 🔴 +12 kB 🔴 +3.33 kB 🔴 +3 kB
assets/AboutPanel-CFYuZ6jR.js (removed) 12 kB 🟢 -12 kB 🟢 -3.33 kB 🟢 -2.99 kB
assets/ExtensionPanel-BHPNU4Wt.js (removed) 10.1 kB 🟢 -10.1 kB 🟢 -2.95 kB 🟢 -2.62 kB
assets/ExtensionPanel-D5KQ_eeL.js (new) 10.1 kB 🔴 +10.1 kB 🔴 +2.95 kB 🔴 +2.62 kB
assets/ServerConfigPanel-1tNY6qaE.js (new) 7.13 kB 🔴 +7.13 kB 🔴 +2.4 kB 🔴 +2.14 kB
assets/ServerConfigPanel-D8k3wcWi.js (removed) 7.13 kB 🟢 -7.13 kB 🟢 -2.39 kB 🟢 -2.14 kB
assets/UserPanel-AaiOY5fn.js (removed) 6.84 kB 🟢 -6.84 kB 🟢 -2.27 kB 🟢 -2.01 kB
assets/UserPanel-Dji1myb2.js (new) 6.84 kB 🔴 +6.84 kB 🔴 +2.28 kB 🔴 +2.01 kB
assets/cloudRemoteConfig-DpXColxS.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -1.02 kB 🟢 -881 B
assets/cloudRemoteConfig-PQiv_E3P.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +1.02 kB 🔴 +885 B
assets/refreshRemoteConfig-B-t3b7OM.js (removed) 1.45 kB 🟢 -1.45 kB 🟢 -650 B 🟢 -556 B
assets/refreshRemoteConfig-D63M2de0.js (new) 1.45 kB 🔴 +1.45 kB 🔴 +650 B 🔴 +553 B

Status: 10 added / 10 removed / 14 unchanged

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

Authentication, profile, and account management bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/auth-BcvfJM5S.js (new) 3.65 kB 🔴 +3.65 kB 🔴 +1.29 kB 🔴 +1.1 kB
assets/auth-D0BgFSJp.js (removed) 3.65 kB 🟢 -3.65 kB 🟢 -1.29 kB 🟢 -1.1 kB
assets/SignUpForm-BixR3kdR.js (removed) 3.19 kB 🟢 -3.19 kB 🟢 -1.29 kB 🟢 -1.15 kB
assets/SignUpForm-C0Zlq0IO.js (new) 3.19 kB 🔴 +3.19 kB 🔴 +1.29 kB 🔴 +1.15 kB
assets/UpdatePasswordContent-B1h8oAGb.js (new) 2.98 kB 🔴 +2.98 kB 🔴 +1.33 kB 🔴 +1.19 kB
assets/UpdatePasswordContent-DUC_Lm7U.js (removed) 2.98 kB 🟢 -2.98 kB 🟢 -1.33 kB 🟢 -1.18 kB
assets/authStore-CQAqKzRw.js (new) 1.27 kB 🔴 +1.27 kB 🔴 +599 B 🔴 +532 B
assets/authStore-DmmpFGKa.js (removed) 1.27 kB 🟢 -1.27 kB 🟢 -599 B 🟢 -531 B
assets/auth-BVFSvNLG.js (removed) 348 B 🟢 -348 B 🟢 -218 B 🟢 -209 B
assets/auth-MNeDW3h7.js (new) 348 B 🔴 +348 B 🔴 +219 B 🔴 +191 B

Status: 5 added / 5 removed / 2 unchanged

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

Modals, dialogs, drawers, and in-app editors

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyHubPublishDialog-BgZrvBn4.js (new) 85.8 kB 🔴 +85.8 kB 🔴 +18.6 kB 🔴 +15.9 kB
assets/ComfyHubPublishDialog-DX4-YeSt.js (removed) 85.8 kB 🟢 -85.8 kB 🟢 -18.6 kB 🟢 -15.9 kB
assets/useShareDialog-BN5aqIcK.js (removed) 23.9 kB 🟢 -23.9 kB 🟢 -5.81 kB 🟢 -5.14 kB
assets/useShareDialog-CIa_zXp5.js (new) 23.9 kB 🔴 +23.9 kB 🔴 +5.82 kB 🔴 +5.16 kB
assets/ComfyHubPublishDialog-CfCnFmln.js (new) 1.43 kB 🔴 +1.43 kB 🔴 +659 B 🔴 +586 B
assets/ComfyHubPublishDialog-XayroI31.js (removed) 1.43 kB 🟢 -1.43 kB 🟢 -659 B 🟢 -581 B
assets/useSubscriptionDialog-9c3D1tJI.js (removed) 1.25 kB 🟢 -1.25 kB 🟢 -592 B 🟢 -520 B
assets/useSubscriptionDialog-BMx5raaN.js (new) 1.25 kB 🔴 +1.25 kB 🔴 +592 B 🔴 +519 B

Status: 4 added / 4 removed

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

Reusable component library chunks

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyQueueButton-CWxy8cZ1.js (removed) 13.5 kB 🟢 -13.5 kB 🟢 -3.79 kB 🟢 -3.38 kB
assets/ComfyQueueButton-DYP3Ocng.js (new) 13.5 kB 🔴 +13.5 kB 🔴 +3.79 kB 🔴 +3.38 kB
assets/useTerminalTabs-2xpKEmNM.js (removed) 11.1 kB 🟢 -11.1 kB 🟢 -3.76 kB 🟢 -3.32 kB
assets/useTerminalTabs-DQMGkkWz.js (new) 11.1 kB 🔴 +11.1 kB 🔴 +3.77 kB 🔴 +3.31 kB
assets/SubscribeButton-BgPjogJQ.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +1.05 kB 🔴 +921 B
assets/SubscribeButton-CURjIwI0.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -1.05 kB 🟢 -918 B
assets/cloudFeedbackTopbarButton-CEvmaAKN.js (removed) 1.94 kB 🟢 -1.94 kB 🟢 -965 B 🟢 -862 B
assets/cloudFeedbackTopbarButton-CUjD1ZtT.js (new) 1.94 kB 🔴 +1.94 kB 🔴 +967 B 🔴 +862 B
assets/ComfyQueueButton-bxBlBhbk.js (removed) 1.35 kB 🟢 -1.35 kB 🟢 -627 B 🟢 -573 B
assets/ComfyQueueButton-pxs8Dork.js (new) 1.35 kB 🔴 +1.35 kB 🔴 +627 B 🔴 +574 B

Status: 5 added / 5 removed / 8 unchanged

Data & Services — 3.16 MB (baseline 3.16 MB) • 🔴 +799 B

Stores, services, APIs, and repositories

File Before After Δ Raw Δ Gzip Δ Brotli
assets/dialogService-Dm8WPViC.js (new) 2.09 MB 🔴 +2.09 MB 🔴 +477 kB 🔴 +361 kB
assets/dialogService-D-qgCbge.js (removed) 2.09 MB 🟢 -2.09 MB 🟢 -476 kB 🟢 -361 kB
assets/api-DE6z86IF.js (removed) 898 kB 🟢 -898 kB 🟢 -215 kB 🟢 -169 kB
assets/api-QE_CT7ew.js (new) 898 kB 🔴 +898 kB 🔴 +215 kB 🔴 +169 kB
assets/load3dService-DeezERbl.js (new) 116 kB 🔴 +116 kB 🔴 +25.4 kB 🔴 +21.6 kB
assets/load3dService-Do-bK5lA.js (removed) 116 kB 🟢 -116 kB 🟢 -25.4 kB 🟢 -21.6 kB
assets/workflowShareService-f86kyTZP.js (removed) 16.7 kB 🟢 -16.7 kB 🟢 -4.92 kB 🟢 -4.37 kB
assets/workflowShareService-XX_MtGuE.js (new) 16.7 kB 🔴 +16.7 kB 🔴 +4.92 kB 🔴 +4.36 kB
assets/keybindingService-04P_dWB4.js (new) 13.8 kB 🔴 +13.8 kB 🔴 +3.67 kB 🔴 +3.22 kB
assets/keybindingService-DvnXqk0F.js (removed) 13.8 kB 🟢 -13.8 kB 🟢 -3.67 kB 🟢 -3.22 kB
assets/releaseStore-C09Zn6V1.js (removed) 8.12 kB 🟢 -8.12 kB 🟢 -2.28 kB 🟢 -2 kB
assets/releaseStore-CWRJbd5O.js (new) 8.12 kB 🔴 +8.12 kB 🔴 +2.28 kB 🔴 +2 kB
assets/userStore-D9S0FuoV.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -934 B 🟢 -816 B
assets/userStore-dlg2Ifnw.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +931 B 🔴 +815 B
assets/audioService-Ctlx4yKk.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +880 B 🔴 +757 B
assets/audioService-Cx4kBWsh.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -881 B 🟢 -758 B
assets/releaseStore-BCFz0mBo.js (new) 1.27 kB 🔴 +1.27 kB 🔴 +596 B 🔴 +530 B
assets/releaseStore-BPKLkq3E.js (removed) 1.27 kB 🟢 -1.27 kB 🟢 -595 B 🟢 -528 B
assets/workflowDraftStore-C4xmOCPe.js (removed) 1.25 kB 🟢 -1.25 kB 🟢 -591 B 🟢 -522 B
assets/workflowDraftStore-iLKg50uz.js (new) 1.25 kB 🔴 +1.25 kB 🔴 +591 B 🔴 +524 B
assets/dialogService-Bchfly1Y.js (new) 1.24 kB 🔴 +1.24 kB 🔴 +584 B 🔴 +522 B
assets/dialogService-BRMDScWe.js (removed) 1.24 kB 🟢 -1.24 kB 🟢 -583 B 🟢 -521 B
assets/settingStore-BTR6ReAd.js (new) 1.23 kB 🔴 +1.23 kB 🔴 +587 B 🔴 +524 B
assets/settingStore-CYcv7Quz.js (removed) 1.23 kB 🟢 -1.23 kB 🟢 -586 B 🟢 -519 B
assets/assetsStore-D_h3DOdB.js (removed) 1.23 kB 🟢 -1.23 kB 🟢 -586 B 🟢 -516 B
assets/assetsStore-EDsrtlHr.js (new) 1.23 kB 🔴 +1.23 kB 🔴 +586 B 🔴 +522 B

Status: 13 added / 13 removed / 4 unchanged

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

Helpers, composables, and utility bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useConflictDetection-BUYhtKVz.js (removed) 234 kB 🟢 -234 kB 🟢 -52.1 kB 🟢 -42.5 kB
assets/useConflictDetection-DspUcRM8.js (new) 234 kB 🔴 +234 kB 🔴 +52.1 kB 🔴 +42.4 kB
assets/useLoad3d-CLMb1xI3.js (removed) 22.7 kB 🟢 -22.7 kB 🟢 -5.22 kB 🟢 -4.61 kB
assets/useLoad3d-rg6ifhs1.js (new) 22.7 kB 🔴 +22.7 kB 🔴 +5.22 kB 🔴 +4.61 kB
assets/useLoad3dViewer-B37ZK13e.js (removed) 21 kB 🟢 -21 kB 🟢 -4.93 kB 🟢 -4.3 kB
assets/useLoad3dViewer-DNQ30tlA.js (new) 21 kB 🔴 +21 kB 🔴 +4.92 kB 🔴 +4.31 kB
assets/useFeatureFlags-DmSqgvTX.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +1.8 kB 🔴 +1.52 kB
assets/useFeatureFlags-DRimVAyX.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -1.79 kB 🟢 -1.52 kB
assets/useCopyToClipboard-3A6IqCoM.js (removed) 5.29 kB 🟢 -5.29 kB 🟢 -1.86 kB 🟢 -1.58 kB
assets/useCopyToClipboard-nB0zQMAK.js (new) 5.29 kB 🔴 +5.29 kB 🔴 +1.86 kB 🔴 +1.58 kB
assets/useWorkspaceUI-DxQILF7P.js (removed) 3.34 kB 🟢 -3.34 kB 🟢 -982 B 🟢 -815 B
assets/useWorkspaceUI-jMR-iEax.js (new) 3.34 kB 🔴 +3.34 kB 🔴 +981 B 🔴 +811 B
assets/subscriptionCheckoutUtil-4JMhFJOt.js (removed) 3.31 kB 🟢 -3.31 kB 🟢 -1.36 kB 🟢 -1.18 kB
assets/subscriptionCheckoutUtil-Br9NHifi.js (new) 3.31 kB 🔴 +3.31 kB 🔴 +1.36 kB 🔴 +1.18 kB
assets/assetPreviewUtil-CV5vE5xm.js (new) 2.43 kB 🔴 +2.43 kB 🔴 +1.01 kB 🔴 +878 B
assets/assetPreviewUtil-ItOejEht.js (removed) 2.43 kB 🟢 -2.43 kB 🟢 -1.01 kB 🟢 -878 B
assets/useUpstreamValue-666lvRLz.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +804 B 🔴 +717 B
assets/useUpstreamValue-C1nOQRc_.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -803 B 🟢 -714 B
assets/useLoad3d-CIRlaXAf.js (new) 1.41 kB 🔴 +1.41 kB 🔴 +651 B 🔴 +585 B
assets/useLoad3d-mzDkAMC4.js (removed) 1.41 kB 🟢 -1.41 kB 🟢 -652 B 🟢 -583 B
assets/useLoad3dViewer-8CH0_8Kn.js (new) 1.35 kB 🔴 +1.35 kB 🔴 +620 B 🔴 +566 B
assets/useLoad3dViewer-D9F1JGKp.js (removed) 1.35 kB 🟢 -1.35 kB 🟢 -620 B 🟢 -569 B
assets/useCurrentUser-B530UFWy.js (new) 1.23 kB 🔴 +1.23 kB 🔴 +587 B 🔴 +522 B
assets/useCurrentUser-KCgYM_4q.js (removed) 1.23 kB 🟢 -1.23 kB 🟢 -587 B 🟢 -520 B
assets/useWorkspaceSwitch-B-lk74yo.js (new) 747 B 🔴 +747 B 🔴 +384 B 🔴 +327 B
assets/useWorkspaceSwitch-K7frXquj.js (removed) 747 B 🟢 -747 B 🟢 -382 B 🟢 -330 B

Status: 13 added / 13 removed / 18 unchanged

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

External libraries and shared vendor chunks

Status: 16 unchanged

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

Bundles that do not match a named category

File Before After Δ Raw Δ Gzip Δ Brotli
assets/core-BczHY4dn.js (removed) 78 kB 🟢 -78 kB 🟢 -20.2 kB 🟢 -17.2 kB
assets/core-IhDKHfbC.js (new) 78 kB 🔴 +78 kB 🔴 +20.2 kB 🔴 +17.2 kB
assets/groupNode-BKBpssQ4.js (new) 74.9 kB 🔴 +74.9 kB 🔴 +18.7 kB 🔴 +16.5 kB
assets/groupNode-CLtsfd4a.js (removed) 74.9 kB 🟢 -74.9 kB 🟢 -18.7 kB 🟢 -16.5 kB
assets/WidgetSelect-C2xEv6_d.js (removed) 68.4 kB 🟢 -68.4 kB 🟢 -15.1 kB 🟢 -13 kB
assets/WidgetSelect-CpEkDORw.js (new) 68.4 kB 🔴 +68.4 kB 🔴 +15.1 kB 🔴 +13 kB
assets/SubscriptionRequiredDialogContentWorkspace-CCHdvF9I.js (new) 48.9 kB 🔴 +48.9 kB 🔴 +9.56 kB 🔴 +8.25 kB
assets/SubscriptionRequiredDialogContentWorkspace-D3NX1li6.js (removed) 48.9 kB 🟢 -48.9 kB 🟢 -9.55 kB 🟢 -8.27 kB
assets/Load3DControls-BODIA-pl.js (removed) 46.1 kB 🟢 -46.1 kB 🟢 -7.5 kB 🟢 -6.54 kB
assets/Load3DControls-whNqqz48.js (new) 46.1 kB 🔴 +46.1 kB 🔴 +7.51 kB 🔴 +6.56 kB
assets/WorkspacePanelContent-BlHJxuGp.js (removed) 34.3 kB 🟢 -34.3 kB 🟢 -7.44 kB 🟢 -6.58 kB
assets/WorkspacePanelContent-C_DdYIFj.js (new) 34.3 kB 🔴 +34.3 kB 🔴 +7.44 kB 🔴 +6.59 kB
assets/WidgetPainter-B3wxVmyz.js (new) 33.5 kB 🔴 +33.5 kB 🔴 +8.29 kB 🔴 +7.34 kB
assets/WidgetPainter-DrFDIyYp.js (removed) 33.5 kB 🟢 -33.5 kB 🟢 -8.29 kB 🟢 -7.33 kB
assets/Load3dViewerContent-BQWU9CqR.js (removed) 30.6 kB 🟢 -30.6 kB 🟢 -6.2 kB 🟢 -5.37 kB
assets/Load3dViewerContent-BvWTXhzo.js (new) 30.6 kB 🔴 +30.6 kB 🔴 +6.19 kB 🔴 +5.36 kB
assets/SubscriptionRequiredDialogContent-D8ZnEtEc.js (removed) 27.6 kB 🟢 -27.6 kB 🟢 -7.02 kB 🟢 -6.2 kB
assets/SubscriptionRequiredDialogContent-SR3GAkRS.js (new) 27.6 kB 🔴 +27.6 kB 🔴 +7.02 kB 🔴 +6.19 kB
assets/WidgetImageCrop--zGMlmFd.js (removed) 24.4 kB 🟢 -24.4 kB 🟢 -6.23 kB 🟢 -5.48 kB
assets/WidgetImageCrop-w0EiIDd7.js (new) 24.4 kB 🔴 +24.4 kB 🔴 +6.24 kB 🔴 +5.49 kB
assets/SubscriptionPanelContentWorkspace-CgXHS2xs.js (removed) 22.2 kB 🟢 -22.2 kB 🟢 -5.17 kB 🟢 -4.56 kB
assets/SubscriptionPanelContentWorkspace-YcDlUPLc.js (new) 22.2 kB 🔴 +22.2 kB 🔴 +5.17 kB 🔴 +4.55 kB
assets/SignInContent-B3P2X2gl.js (new) 20.9 kB 🔴 +20.9 kB 🔴 +5.48 kB 🔴 +4.8 kB
assets/SignInContent-BFpcCMwb.js (removed) 20.9 kB 🟢 -20.9 kB 🟢 -5.47 kB 🟢 -4.8 kB
assets/CurrentUserPopoverWorkspace-C9pYG17u.js (new) 20.9 kB 🔴 +20.9 kB 🔴 +4.99 kB 🔴 +4.46 kB
assets/CurrentUserPopoverWorkspace-CVnFpAhu.js (removed) 20.9 kB 🟢 -20.9 kB 🟢 -4.99 kB 🟢 -4.47 kB
assets/WidgetInputNumber-3jOCW-x2.js (new) 19.1 kB 🔴 +19.1 kB 🔴 +4.84 kB 🔴 +4.29 kB
assets/WidgetInputNumber-BJFff-GY.js (removed) 19.1 kB 🟢 -19.1 kB 🟢 -4.84 kB 🟢 -4.3 kB
assets/Load3D-BgG9qflK.js (new) 18.5 kB 🔴 +18.5 kB 🔴 +4.39 kB 🔴 +3.84 kB
assets/Load3D-DZoTZ22M.js (removed) 18.5 kB 🟢 -18.5 kB 🟢 -4.39 kB 🟢 -3.83 kB
assets/WidgetRecordAudio-BWbc7zjH.js (removed) 17.6 kB 🟢 -17.6 kB 🟢 -5.07 kB 🟢 -4.53 kB
assets/WidgetRecordAudio-D_RbmZgZ.js (new) 17.6 kB 🔴 +17.6 kB 🔴 +5.07 kB 🔴 +4.52 kB
assets/WidgetRange-kJe8t5xG.js (new) 17.1 kB 🔴 +17.1 kB 🔴 +4.65 kB 🔴 +4.13 kB
assets/WidgetRange-qfukD9ME.js (removed) 17.1 kB 🟢 -17.1 kB 🟢 -4.64 kB 🟢 -4.14 kB
assets/load3d-Cx7_WNpn.js (removed) 15.9 kB 🟢 -15.9 kB 🟢 -4.62 kB 🟢 -4.01 kB
assets/load3d-D_XVBUtR.js (new) 15.9 kB 🔴 +15.9 kB 🔴 +4.62 kB 🔴 +4.01 kB
assets/WaveAudioPlayer-D8IAqTRX.js (removed) 13.4 kB 🟢 -13.4 kB 🟢 -3.68 kB 🟢 -3.23 kB
assets/WaveAudioPlayer-DMxDxYK2.js (new) 13.4 kB 🔴 +13.4 kB 🔴 +3.68 kB 🔴 +3.23 kB
assets/WidgetCurve-3OodTVm5.js (new) 12.3 kB 🔴 +12.3 kB 🔴 +3.96 kB 🔴 +3.58 kB
assets/WidgetCurve-Cn6iTXOx.js (removed) 12.3 kB 🟢 -12.3 kB 🟢 -3.96 kB 🟢 -3.59 kB
assets/TeamWorkspacesDialogContent-Dd1IhQUk.js (removed) 11.4 kB 🟢 -11.4 kB 🟢 -3.45 kB 🟢 -3.07 kB
assets/TeamWorkspacesDialogContent-DJ5WMiNq.js (new) 11.4 kB 🔴 +11.4 kB 🔴 +3.46 kB 🔴 +3.08 kB
assets/nodeTemplates-4WmnnZj3.js (removed) 9.92 kB 🟢 -9.92 kB 🟢 -3.52 kB 🟢 -3.11 kB
assets/nodeTemplates-B_47RqlQ.js (new) 9.92 kB 🔴 +9.92 kB 🔴 +3.52 kB 🔴 +3.12 kB
assets/NightlySurveyController-B4AsMEkX.js (removed) 9.05 kB 🟢 -9.05 kB 🟢 -3.18 kB 🟢 -2.81 kB
assets/NightlySurveyController-CF5xwsxV.js (new) 9.05 kB 🔴 +9.05 kB 🔴 +3.18 kB 🔴 +2.81 kB
assets/Load3DConfiguration-D0l8bdln.js (new) 8.77 kB 🔴 +8.77 kB 🔴 +2.61 kB 🔴 +2.3 kB
assets/Load3DConfiguration-DFjz4Y4U.js (removed) 8.77 kB 🟢 -8.77 kB 🟢 -2.61 kB 🟢 -2.3 kB
assets/InviteMemberDialogContent-CM5F6vDg.js (removed) 8.02 kB 🟢 -8.02 kB 🟢 -2.56 kB 🟢 -2.24 kB
assets/InviteMemberDialogContent-DcqNgShP.js (new) 8.02 kB 🔴 +8.02 kB 🔴 +2.56 kB 🔴 +2.25 kB
assets/onboardingCloudRoutes-4X6X2J-P.js (new) 6.94 kB 🔴 +6.94 kB 🔴 +2.18 kB 🔴 +1.87 kB
assets/onboardingCloudRoutes-CW2voh_4.js (removed) 6.94 kB 🟢 -6.94 kB 🟢 -2.17 kB 🟢 -1.86 kB
assets/CreateWorkspaceDialogContent-CJ6XcwGg.js (new) 6.23 kB 🔴 +6.23 kB 🔴 +2.27 kB 🔴 +1.99 kB
assets/CreateWorkspaceDialogContent-DpyliyX6.js (removed) 6.23 kB 🟢 -6.23 kB 🟢 -2.27 kB 🟢 -1.98 kB
assets/WidgetWithControl-CmpxgbYv.js (new) 6.2 kB 🔴 +6.2 kB 🔴 +2.5 kB 🔴 +2.22 kB
assets/WidgetWithControl-IqVtrX2_.js (removed) 6.2 kB 🟢 -6.2 kB 🟢 -2.5 kB 🟢 -2.19 kB
assets/FreeTierDialogContent-BQvl9nJg.js (new) 6.09 kB 🔴 +6.09 kB 🔴 +2.17 kB 🔴 +1.92 kB
assets/FreeTierDialogContent-D7FX5z40.js (removed) 6.09 kB 🟢 -6.09 kB 🟢 -2.17 kB 🟢 -1.93 kB
assets/EditWorkspaceDialogContent-BHAw2k5q.js (new) 6.03 kB 🔴 +6.03 kB 🔴 +2.23 kB 🔴 +1.97 kB
assets/EditWorkspaceDialogContent-Cf0JODsu.js (removed) 6.03 kB 🟢 -6.03 kB 🟢 -2.23 kB 🟢 -1.95 kB
assets/WidgetTextarea-BHYAfFIc.js (removed) 5.84 kB 🟢 -5.84 kB 🟢 -2.3 kB 🟢 -2.03 kB
assets/WidgetTextarea-DZUIHYnQ.js (new) 5.84 kB 🔴 +5.84 kB 🔴 +2.3 kB 🔴 +2.03 kB
assets/Preview3d-BZ9mOmtp.js (removed) 5.81 kB 🟢 -5.81 kB 🟢 -1.96 kB 🟢 -1.71 kB
assets/Preview3d-utb-R1Hl.js (new) 5.81 kB 🔴 +5.81 kB 🔴 +1.96 kB 🔴 +1.71 kB
assets/ValueControlPopover-CXwgYKg9.js (removed) 5.61 kB 🟢 -5.61 kB 🟢 -2.05 kB 🟢 -1.85 kB
assets/ValueControlPopover-Dxs4Fqbv.js (new) 5.61 kB 🔴 +5.61 kB 🔴 +2.05 kB 🔴 +1.84 kB
assets/CancelSubscriptionDialogContent-CNoOj1X6.js (new) 5.53 kB 🔴 +5.53 kB 🔴 +2.08 kB 🔴 +1.83 kB
assets/CancelSubscriptionDialogContent-CX6cjsTD.js (removed) 5.53 kB 🟢 -5.53 kB 🟢 -2.08 kB 🟢 -1.82 kB
assets/DeleteWorkspaceDialogContent-CATBjziJ.js (removed) 4.93 kB 🟢 -4.93 kB 🟢 -1.91 kB 🟢 -1.66 kB
assets/DeleteWorkspaceDialogContent-DMfbMHYu.js (new) 4.93 kB 🔴 +4.93 kB 🔴 +1.91 kB 🔴 +1.66 kB
assets/saveMesh-DqY7ZQ48.js (removed) 4.78 kB 🟢 -4.78 kB 🟢 -1.9 kB 🟢 -1.69 kB
assets/saveMesh-FWgZCXRX.js (new) 4.78 kB 🔴 +4.78 kB 🔴 +1.9 kB 🔴 +1.7 kB
assets/LeaveWorkspaceDialogContent-CCvJoWOu.js (new) 4.76 kB 🔴 +4.76 kB 🔴 +1.86 kB 🔴 +1.61 kB
assets/LeaveWorkspaceDialogContent-DpLlk3MX.js (removed) 4.76 kB 🟢 -4.76 kB 🟢 -1.86 kB 🟢 -1.61 kB
assets/RemoveMemberDialogContent-8v9Lm6hn.js (removed) 4.74 kB 🟢 -4.74 kB 🟢 -1.81 kB 🟢 -1.58 kB
assets/RemoveMemberDialogContent-aBt7E-Fv.js (new) 4.74 kB 🔴 +4.74 kB 🔴 +1.81 kB 🔴 +1.58 kB
assets/RevokeInviteDialogContent-dXoteZtX.js (removed) 4.65 kB 🟢 -4.65 kB 🟢 -1.82 kB 🟢 -1.59 kB
assets/RevokeInviteDialogContent-ya5w2mBN.js (new) 4.65 kB 🔴 +4.65 kB 🔴 +1.82 kB 🔴 +1.59 kB
assets/InviteMemberUpsellDialogContent-AdxKwPbs.js (removed) 4.55 kB 🟢 -4.55 kB 🟢 -1.68 kB 🟢 -1.48 kB
assets/InviteMemberUpsellDialogContent-DUvuVMuq.js (new) 4.55 kB 🔴 +4.55 kB 🔴 +1.69 kB 🔴 +1.48 kB
assets/tierBenefits-D3gnvJ4f.js (new) 4.45 kB 🔴 +4.45 kB 🔴 +1.58 kB 🔴 +1.36 kB
assets/tierBenefits-DfZ4C_PY.js (removed) 4.45 kB 🟢 -4.45 kB 🟢 -1.58 kB 🟢 -1.36 kB
assets/Media3DTop-CT0chRzb.js (removed) 4.43 kB 🟢 -4.43 kB 🟢 -1.81 kB 🟢 -1.61 kB
assets/Media3DTop-gdVtycYw.js (new) 4.43 kB 🔴 +4.43 kB 🔴 +1.81 kB 🔴 +1.61 kB
assets/cloudSessionCookie-CvDnv07M.js (removed) 4.39 kB 🟢 -4.39 kB 🟢 -1.61 kB 🟢 -1.4 kB
assets/cloudSessionCookie-DCNzMvx8.js (new) 4.39 kB 🔴 +4.39 kB 🔴 +1.61 kB 🔴 +1.4 kB
assets/GlobalToast-DopyndQ8.js (new) 3.05 kB 🔴 +3.05 kB 🔴 +1.26 kB 🔴 +1.12 kB
assets/GlobalToast-DQdAD4cw.js (removed) 3.05 kB 🟢 -3.05 kB 🟢 -1.26 kB 🟢 -1.12 kB
assets/CloudRunButtonWrapper-B2Hi1fEe.js (new) 2.31 kB 🔴 +2.31 kB 🔴 +1.05 kB 🔴 +937 B
assets/CloudRunButtonWrapper-DMPNp0hJ.js (removed) 2.31 kB 🟢 -2.31 kB 🟢 -1.05 kB 🟢 -944 B
assets/SubscribeToRun-Du3NDlxz.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -982 B 🟢 -874 B
assets/SubscribeToRun-HDX82do1.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +984 B 🔴 +873 B
assets/MediaAudioTop-BxB2i1XN.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +1 kB 🔴 +889 B
assets/MediaAudioTop-YRN6sxWo.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -1.01 kB 🟢 -862 B
assets/cloudBadges-Bi-DSXD6.js (removed) 2.04 kB 🟢 -2.04 kB 🟢 -1.01 kB 🟢 -876 B
assets/cloudBadges-KHG5JgBp.js (new) 2.04 kB 🔴 +2.04 kB 🔴 +1.01 kB 🔴 +878 B
assets/cloudSubscription-8YoBSykU.js (removed) 1.96 kB 🟢 -1.96 kB 🟢 -929 B 🟢 -811 B
assets/cloudSubscription-CSY673vA.js (new) 1.96 kB 🔴 +1.96 kB 🔴 +934 B 🔴 +812 B
assets/graphHasMissingNodes-DIfH8PON.js (removed) 1.84 kB 🟢 -1.84 kB 🟢 -861 B 🟢 -764 B
assets/graphHasMissingNodes-F5oDPA5s.js (new) 1.84 kB 🔴 +1.84 kB 🔴 +860 B 🔴 +756 B
assets/Load3D-b2uLKk6M.js (removed) 1.66 kB 🟢 -1.66 kB 🟢 -739 B 🟢 -660 B
assets/Load3D-CNpx_7sw.js (new) 1.66 kB 🔴 +1.66 kB 🔴 +740 B 🔴 +660 B
assets/nightlyBadges-2QolLUea.js (removed) 1.57 kB 🟢 -1.57 kB 🟢 -779 B 🟢 -692 B
assets/nightlyBadges-C443lYbI.js (new) 1.57 kB 🔴 +1.57 kB 🔴 +780 B 🔴 +690 B
assets/Load3dViewerContent-CvBmH7cT.js (new) 1.54 kB 🔴 +1.54 kB 🔴 +695 B 🔴 +619 B
assets/Load3dViewerContent-DHGnsm-2.js (removed) 1.54 kB 🟢 -1.54 kB 🟢 -695 B 🟢 -620 B
assets/previousFullPath-B3alJmB1.js (removed) 1.53 kB 🟢 -1.53 kB 🟢 -693 B 🟢 -601 B
assets/previousFullPath-CbMSQD9s.js (new) 1.53 kB 🔴 +1.53 kB 🔴 +695 B 🔴 +623 B
assets/SubscriptionPanelContentWorkspace-IEFmfMB0.js (removed) 1.43 kB 🟢 -1.43 kB 🟢 -651 B 🟢 -581 B
assets/SubscriptionPanelContentWorkspace-ZTrcfCsB.js (new) 1.43 kB 🔴 +1.43 kB 🔴 +651 B 🔴 +582 B
assets/WidgetLegacy-C1v7mtKa.js (removed) 1.25 kB 🟢 -1.25 kB 🟢 -597 B 🟢 -532 B
assets/WidgetLegacy-DaoIB_QE.js (new) 1.25 kB 🔴 +1.25 kB 🔴 +598 B 🔴 +530 B
assets/changeTracker-BY6yUM_Y.js (new) 1.23 kB 🔴 +1.23 kB 🔴 +587 B 🔴 +521 B
assets/changeTracker-C0w8oupP.js (removed) 1.23 kB 🟢 -1.23 kB 🟢 -587 B 🟢 -519 B

Status: 57 added / 57 removed / 86 unchanged

⚡ Performance Report

canvas-idle: · 60.0 avg FPS · 59.7 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.1 MB heap
canvas-mouse-sweep: · 60.0 avg FPS · 59.7 P5 FPS ✅ (target: ≥52) · 0ms TBT · 49.1 MB heap
canvas-zoom-sweep: · 60.0 avg FPS · 59.7 P5 FPS ✅ (target: ≥52) · 0ms TBT · 58.3 MB heap
dom-widget-clipping: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 55.4 MB heap
large-graph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 69.6 MB heap
large-graph-pan: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 86.6 MB heap
large-graph-zoom: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 69.4 MB heap
minimap-idle: · 60.0 avg FPS · 59.7 P5 FPS ✅ (target: ≥52) · 0ms TBT · 99.8 MB heap
subgraph-dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 57.4 MB heap
subgraph-idle: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 66.8 MB heap
subgraph-mouse-sweep: · 60.0 avg FPS · 59.7 P5 FPS ✅ (target: ≥52) · 0ms TBT · 45.7 MB heap
subgraph-transition-enter: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 156ms TBT · 213.9 MB heap
viewport-pan-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 70.4 MB heap
vue-large-graph-idle: · 57.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 223.8 MB heap
vue-large-graph-pan: · 56.2 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 175.9 MB heap
workflow-execution: · 60.0 avg FPS · 59.7 P5 FPS ✅ (target: ≥52) · 0ms TBT · 53.3 MB heap

⚠️ 8 regressions detected

Metric Baseline PR (median) Δ Sig
canvas-mouse-sweep: layout duration 3ms 4ms +28% ⚠️ z=3.6
canvas-zoom-sweep: layout duration 1ms 1ms +42% ⚠️ z=4.4
large-graph-pan: style recalc duration 17ms 21ms +23% ⚠️ z=5.0
large-graph-pan: task duration 1036ms 1208ms +17% ⚠️ z=2.9
minimap-idle: task duration 513ms 698ms +36% ⚠️ z=3.6
subgraph-dom-widget-clipping: task duration 346ms 473ms +37% ⚠️ z=5.1
subgraph-dom-widget-clipping: script duration 121ms 150ms +24% ⚠️ z=3.4
subgraph-mouse-sweep: layout duration 4ms 6ms +42% ⚠️ z=4.4
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 9ms -7% z=-2.3
canvas-idle: layout count 0 0 +0%
canvas-idle: style recalc count 10 9 -10% z=-3.8
canvas-idle: task duration 428ms 408ms -5% z=0.4
canvas-idle: script duration 21ms 25ms +18% z=-0.2
canvas-idle: TBT 0ms 0ms +0%
canvas-idle: heap used 66.2 MB 68.1 MB +3%
canvas-idle: DOM nodes -259 18 -107% z=-3.6
canvas-idle: event listeners -129 4 -103% z=-1.6
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 +28% ⚠️ z=3.6
canvas-mouse-sweep: style recalc duration 39ms 44ms +14% z=0.4
canvas-mouse-sweep: layout count 12 12 +0%
canvas-mouse-sweep: style recalc count 82 75 -9% z=-1.8
canvas-mouse-sweep: task duration 956ms 888ms -7% z=0.4
canvas-mouse-sweep: script duration 120ms 131ms +9% z=-0.8
canvas-mouse-sweep: TBT 0ms 0ms +0%
canvas-mouse-sweep: heap used 66.0 MB 49.1 MB -26%
canvas-mouse-sweep: DOM nodes -262 -263 +0% z=-125.8
canvas-mouse-sweep: event listeners -129 -133 +3% z=-33.9
canvas-zoom-sweep: avg frame time 17ms 17ms +0% z=1.4
canvas-zoom-sweep: p95 frame time 17ms 17ms +0%
canvas-zoom-sweep: layout duration 1ms 1ms +42% ⚠️ z=4.4
canvas-zoom-sweep: style recalc duration 15ms 20ms +33% z=0.8
canvas-zoom-sweep: layout count 6 6 +0%
canvas-zoom-sweep: style recalc count 32 31 -3% z=-0.6
canvas-zoom-sweep: task duration 278ms 372ms +34% z=2.0
canvas-zoom-sweep: script duration 22ms 30ms +33% z=0.9
canvas-zoom-sweep: TBT 0ms 0ms +0%
canvas-zoom-sweep: heap used 47.8 MB 58.3 MB +22%
canvas-zoom-sweep: DOM nodes 79 77 -3% z=-2.8
canvas-zoom-sweep: event listeners 21 20 -5% z=-0.7
dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
dom-widget-clipping: p95 frame time 17ms 17ms +1%
dom-widget-clipping: layout duration 0ms 0ms +0%
dom-widget-clipping: style recalc duration 8ms 8ms -7% z=-2.8
dom-widget-clipping: layout count 0 0 +0%
dom-widget-clipping: style recalc count 12 11 -13% z=-5.2
dom-widget-clipping: task duration 321ms 365ms +14% z=0.1
dom-widget-clipping: script duration 63ms 62ms -1% z=-1.8
dom-widget-clipping: TBT 0ms 0ms +0%
dom-widget-clipping: heap used 54.8 MB 55.4 MB +1%
dom-widget-clipping: DOM nodes 20 17 -15% z=-3.6
dom-widget-clipping: event listeners 2 1 -50% variance too high
large-graph-idle: avg frame time 17ms 17ms +0% z=-0.2
large-graph-idle: p95 frame time 17ms 17ms +0%
large-graph-idle: layout duration 0ms 0ms +0%
large-graph-idle: style recalc duration 8ms 9ms +11% z=-3.0
large-graph-idle: layout count 0 0 +0%
large-graph-idle: style recalc count 9 10 +6% z=-6.7
large-graph-idle: task duration 511ms 599ms +17% z=1.1
large-graph-idle: script duration 85ms 102ms +21% z=-0.0
large-graph-idle: TBT 0ms 0ms +0%
large-graph-idle: heap used 58.0 MB 69.6 MB +20%
large-graph-idle: DOM nodes -261 -122 -53% z=-161.7
large-graph-idle: event listeners -129 -63 -52% z=-14.5
large-graph-pan: avg frame time 17ms 17ms +0% z=0.3
large-graph-pan: p95 frame time 17ms 17ms +1%
large-graph-pan: layout duration 0ms 0ms +0%
large-graph-pan: style recalc duration 17ms 21ms +23% ⚠️ z=5.0
large-graph-pan: layout count 0 0 +0%
large-graph-pan: style recalc count 68 69 +1% z=-0.9
large-graph-pan: task duration 1036ms 1208ms +17% ⚠️ z=2.9
large-graph-pan: script duration 388ms 399ms +3% z=-0.4
large-graph-pan: TBT 0ms 0ms +0%
large-graph-pan: heap used 62.8 MB 86.6 MB +38%
large-graph-pan: DOM nodes -261 -261 +0% z=-169.7
large-graph-pan: event listeners -129 -142 +10% z=-177.4
large-graph-zoom: avg frame time 17ms 17ms +0%
large-graph-zoom: p95 frame time 17ms 17ms +1%
large-graph-zoom: layout duration 7ms 9ms +29%
large-graph-zoom: style recalc duration 17ms 22ms +26%
large-graph-zoom: layout count 60 60 +0%
large-graph-zoom: style recalc count 66 65 -2%
large-graph-zoom: task duration 1305ms 1443ms +11%
large-graph-zoom: script duration 474ms 518ms +9%
large-graph-zoom: TBT 0ms 0ms +0%
large-graph-zoom: heap used 102.1 MB 69.4 MB -32%
large-graph-zoom: DOM nodes -265 -266 +0%
large-graph-zoom: event listeners -125 -127 +2%
minimap-idle: avg frame time 17ms 17ms +0% z=0.1
minimap-idle: p95 frame time 17ms 17ms +0%
minimap-idle: layout duration 0ms 0ms +0%
minimap-idle: style recalc duration 7ms 8ms +11% z=-1.8
minimap-idle: layout count 0 0 +0%
minimap-idle: style recalc count 8 9 +6% z=-1.6
minimap-idle: task duration 513ms 698ms +36% ⚠️ z=3.6
minimap-idle: script duration 86ms 111ms +30% z=1.3
minimap-idle: TBT 0ms 0ms +0%
minimap-idle: heap used 96.4 MB 99.8 MB +4%
minimap-idle: DOM nodes -258 -264 +2% z=-206.6
minimap-idle: event listeners -129 -129 +0% z=-202.3
subgraph-dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
subgraph-dom-widget-clipping: p95 frame time 17ms 17ms -1%
subgraph-dom-widget-clipping: layout duration 0ms 0ms +0%
subgraph-dom-widget-clipping: style recalc duration 11ms 14ms +31% z=1.3
subgraph-dom-widget-clipping: layout count 0 0 +0%
subgraph-dom-widget-clipping: style recalc count 47 48 +1% z=-0.8
subgraph-dom-widget-clipping: task duration 346ms 473ms +37% ⚠️ z=5.1
subgraph-dom-widget-clipping: script duration 121ms 150ms +24% ⚠️ z=3.4
subgraph-dom-widget-clipping: TBT 0ms 0ms +0%
subgraph-dom-widget-clipping: heap used 56.1 MB 57.4 MB +2%
subgraph-dom-widget-clipping: DOM nodes 20 20 +0% z=-1.9
subgraph-dom-widget-clipping: event listeners 8 7 -13% z=-1.6
subgraph-idle: avg frame time 17ms 17ms +0% z=0.9
subgraph-idle: p95 frame time 17ms 17ms +0%
subgraph-idle: layout duration 0ms 0ms +0%
subgraph-idle: style recalc duration 8ms 10ms +32% z=-0.3
subgraph-idle: layout count 0 0 +0%
subgraph-idle: style recalc count 9 10 +11% z=-1.4
subgraph-idle: task duration 335ms 420ms +26% z=1.6
subgraph-idle: script duration 12ms 23ms +94% z=1.2
subgraph-idle: TBT 0ms 0ms +0%
subgraph-idle: heap used 68.1 MB 66.8 MB -2%
subgraph-idle: DOM nodes 18 20 +8% z=-1.5
subgraph-idle: event listeners 6 5 -17% variance too high
subgraph-mouse-sweep: avg frame time 17ms 17ms +0% z=0.8
subgraph-mouse-sweep: p95 frame time 17ms 17ms +0%
subgraph-mouse-sweep: layout duration 4ms 6ms +42% ⚠️ z=4.4
subgraph-mouse-sweep: style recalc duration 34ms 46ms +34% z=0.9
subgraph-mouse-sweep: layout count 16 16 +0%
subgraph-mouse-sweep: style recalc count 75 81 +7% z=0.2
subgraph-mouse-sweep: task duration 674ms 892ms +32% z=1.8
subgraph-mouse-sweep: script duration 85ms 102ms +20% z=0.2
subgraph-mouse-sweep: TBT 0ms 0ms +0%
subgraph-mouse-sweep: heap used 60.8 MB 45.7 MB -25%
subgraph-mouse-sweep: DOM nodes 60 -261 -534% z=-146.5
subgraph-mouse-sweep: event listeners 4 -133 -3425% variance too high
subgraph-transition-enter: avg frame time 17ms 17ms +0%
subgraph-transition-enter: p95 frame time 17ms 17ms -0%
subgraph-transition-enter: layout duration 14ms 12ms -17%
subgraph-transition-enter: style recalc duration 27ms 28ms +6%
subgraph-transition-enter: layout count 4 4 +0%
subgraph-transition-enter: style recalc count 16 15 -6%
subgraph-transition-enter: task duration 846ms 900ms +6%
subgraph-transition-enter: script duration 29ms 34ms +16%
subgraph-transition-enter: TBT 153ms 156ms +2%
subgraph-transition-enter: heap used 111.4 MB 213.9 MB +92%
subgraph-transition-enter: DOM nodes 12627 13513 +7%
subgraph-transition-enter: event listeners 1639 2527 +54%
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 48ms 62ms +29%
viewport-pan-sweep: layout count 0 0 +0%
viewport-pan-sweep: style recalc count 250 250 -0%
viewport-pan-sweep: task duration 3463ms 3989ms +15%
viewport-pan-sweep: script duration 1214ms 1271ms +5%
viewport-pan-sweep: TBT 0ms 0ms +0%
viewport-pan-sweep: heap used 71.6 MB 70.4 MB -2%
viewport-pan-sweep: DOM nodes -260 -261 +0%
viewport-pan-sweep: event listeners -113 -113 +0%
vue-large-graph-idle: avg frame time 17ms 17ms +2%
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 10851ms 15263ms +41%
vue-large-graph-idle: script duration 543ms 652ms +20%
vue-large-graph-idle: TBT 0ms 0ms +0%
vue-large-graph-idle: heap used 164.5 MB 223.8 MB +36%
vue-large-graph-idle: DOM nodes -8331 -8332 +0%
vue-large-graph-idle: event listeners -16470 -16473 +0%
vue-large-graph-pan: avg frame time 18ms 18ms +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 17ms 21ms +24%
vue-large-graph-pan: layout count 0 0 +0%
vue-large-graph-pan: style recalc count 101 81 -20%
vue-large-graph-pan: task duration 16961ms 15833ms -7%
vue-large-graph-pan: script duration 1012ms 923ms -9%
vue-large-graph-pan: TBT 0ms 0ms +0%
vue-large-graph-pan: heap used 279.4 MB 175.9 MB -37%
vue-large-graph-pan: DOM nodes -8331 -8331 +0%
vue-large-graph-pan: event listeners -16486 -16459 -0%
workflow-execution: avg frame time 17ms 17ms +0% z=0.6
workflow-execution: p95 frame time 17ms 17ms -0%
workflow-execution: layout duration 2ms 2ms -2% z=0.2
workflow-execution: style recalc duration 25ms 23ms -10% z=-0.8
workflow-execution: layout count 6 4 -33% z=-1.7
workflow-execution: style recalc count 19 17 -11% z=-0.4
workflow-execution: task duration 130ms 111ms -14% z=-1.1
workflow-execution: script duration 30ms 21ms -29% z=-2.6
workflow-execution: TBT 0ms 0ms +0%
workflow-execution: heap used 56.6 MB 53.3 MB -6%
workflow-execution: DOM nodes 159 169 +6% z=1.1
workflow-execution: event listeners 69 54 -22% z=0.5
Historical variance (last 15 runs)
Metric μ σ CV
canvas-idle: avg frame time 17ms 0ms 0.0%
canvas-idle: layout duration 0ms 0ms 0.0%
canvas-idle: style recalc duration 11ms 1ms 8.2%
canvas-idle: layout count 0 0 0.0%
canvas-idle: style recalc count 11 1 5.0%
canvas-idle: task duration 395ms 31ms 7.9%
canvas-idle: script duration 25ms 2ms 8.8%
canvas-idle: TBT 0ms 0ms 0.0%
canvas-idle: DOM nodes 23 1 5.6%
canvas-idle: event listeners 12 5 40.9%
canvas-mouse-sweep: avg frame time 17ms 0ms 0.0%
canvas-mouse-sweep: layout duration 4ms 0ms 5.4%
canvas-mouse-sweep: style recalc duration 43ms 3ms 7.4%
canvas-mouse-sweep: layout count 12 0 0.0%
canvas-mouse-sweep: style recalc count 79 2 3.0%
canvas-mouse-sweep: task duration 865ms 58ms 6.7%
canvas-mouse-sweep: script duration 136ms 6ms 4.8%
canvas-mouse-sweep: TBT 0ms 0ms 0.0%
canvas-mouse-sweep: DOM nodes 62 3 4.2%
canvas-mouse-sweep: event listeners 8 4 49.4%
canvas-zoom-sweep: avg frame time 17ms 0ms 0.0%
canvas-zoom-sweep: layout duration 1ms 0ms 7.0%
canvas-zoom-sweep: style recalc duration 19ms 2ms 8.0%
canvas-zoom-sweep: layout count 6 0 0.0%
canvas-zoom-sweep: style recalc count 31 0 1.5%
canvas-zoom-sweep: task duration 327ms 23ms 7.1%
canvas-zoom-sweep: script duration 27ms 3ms 11.1%
canvas-zoom-sweep: TBT 0ms 0ms 0.0%
canvas-zoom-sweep: DOM nodes 79 1 1.0%
canvas-zoom-sweep: event listeners 24 5 21.8%
dom-widget-clipping: avg frame time 17ms 0ms 0.0%
dom-widget-clipping: layout duration 0ms 0ms 0.0%
dom-widget-clipping: style recalc duration 10ms 1ms 8.0%
dom-widget-clipping: layout count 0 0 0.0%
dom-widget-clipping: style recalc count 13 0 3.8%
dom-widget-clipping: task duration 365ms 16ms 4.5%
dom-widget-clipping: script duration 68ms 3ms 4.8%
dom-widget-clipping: TBT 0ms 0ms 0.0%
dom-widget-clipping: DOM nodes 22 1 6.4%
dom-widget-clipping: event listeners 8 6 81.2%
large-graph-idle: avg frame time 17ms 0ms 0.0%
large-graph-idle: layout duration 0ms 0ms 0.0%
large-graph-idle: style recalc duration 12ms 1ms 8.6%
large-graph-idle: layout count 0 0 0.0%
large-graph-idle: style recalc count 12 0 2.7%
large-graph-idle: task duration 542ms 54ms 10.0%
large-graph-idle: script duration 102ms 11ms 10.3%
large-graph-idle: TBT 0ms 0ms 0.0%
large-graph-idle: DOM nodes 25 1 3.7%
large-graph-idle: event listeners 26 6 23.2%
large-graph-pan: avg frame time 17ms 0ms 0.0%
large-graph-pan: layout duration 0ms 0ms 0.0%
large-graph-pan: style recalc duration 17ms 1ms 4.6%
large-graph-pan: layout count 0 0 0.0%
large-graph-pan: style recalc count 70 1 0.9%
large-graph-pan: task duration 1082ms 43ms 4.0%
large-graph-pan: script duration 408ms 20ms 4.8%
large-graph-pan: TBT 0ms 0ms 0.0%
large-graph-pan: DOM nodes 19 2 8.7%
large-graph-pan: event listeners 5 1 16.8%
minimap-idle: avg frame time 17ms 0ms 0.0%
minimap-idle: layout duration 0ms 0ms 0.0%
minimap-idle: style recalc duration 10ms 1ms 8.6%
minimap-idle: layout count 0 0 0.0%
minimap-idle: style recalc count 10 1 7.1%
minimap-idle: task duration 527ms 47ms 9.0%
minimap-idle: script duration 98ms 10ms 10.1%
minimap-idle: TBT 0ms 0ms 0.0%
minimap-idle: DOM nodes 19 1 7.1%
minimap-idle: event listeners 5 1 14.4%
subgraph-dom-widget-clipping: avg frame time 17ms 0ms 0.0%
subgraph-dom-widget-clipping: layout duration 0ms 0ms 0.0%
subgraph-dom-widget-clipping: style recalc duration 13ms 1ms 7.4%
subgraph-dom-widget-clipping: layout count 0 0 0.0%
subgraph-dom-widget-clipping: style recalc count 48 1 1.2%
subgraph-dom-widget-clipping: task duration 378ms 18ms 4.9%
subgraph-dom-widget-clipping: script duration 128ms 6ms 4.9%
subgraph-dom-widget-clipping: TBT 0ms 0ms 0.0%
subgraph-dom-widget-clipping: DOM nodes 22 1 5.0%
subgraph-dom-widget-clipping: event listeners 16 6 36.0%
subgraph-idle: avg frame time 17ms 0ms 0.0%
subgraph-idle: layout duration 0ms 0ms 0.0%
subgraph-idle: style recalc duration 10ms 1ms 7.5%
subgraph-idle: layout count 0 0 0.0%
subgraph-idle: style recalc count 11 1 6.0%
subgraph-idle: task duration 370ms 31ms 8.5%
subgraph-idle: script duration 20ms 3ms 13.2%
subgraph-idle: TBT 0ms 0ms 0.0%
subgraph-idle: DOM nodes 22 1 6.9%
subgraph-idle: event listeners 10 7 64.5%
subgraph-mouse-sweep: avg frame time 17ms 0ms 0.0%
subgraph-mouse-sweep: layout duration 5ms 0ms 6.8%
subgraph-mouse-sweep: style recalc duration 42ms 3ms 7.8%
subgraph-mouse-sweep: layout count 16 0 0.0%
subgraph-mouse-sweep: style recalc count 80 2 2.4%
subgraph-mouse-sweep: task duration 766ms 69ms 9.0%
subgraph-mouse-sweep: script duration 101ms 7ms 6.5%
subgraph-mouse-sweep: TBT 0ms 0ms 0.0%
subgraph-mouse-sweep: DOM nodes 67 2 3.3%
subgraph-mouse-sweep: event listeners 8 4 52.6%
workflow-execution: avg frame time 17ms 0ms 0.0%
workflow-execution: layout duration 2ms 0ms 9.4%
workflow-execution: style recalc duration 24ms 2ms 9.1%
workflow-execution: layout count 5 1 11.0%
workflow-execution: style recalc count 18 2 11.5%
workflow-execution: task duration 123ms 11ms 8.8%
workflow-execution: script duration 29ms 3ms 10.2%
workflow-execution: TBT 0ms 0ms 0.0%
workflow-execution: DOM nodes 161 7 4.4%
workflow-execution: event listeners 52 4 8.4%
Trend (last 15 commits on main)
Metric Trend Dir Latest
canvas-idle: avg frame time ▆▃▆▁▆▃▆█▆▆▄▃▃▄▃ ➡️ 17ms
canvas-idle: p95 frame time ➡️ NaNms
canvas-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: style recalc duration ▇▇▆▆▃█▄▃▄▃▇▄▁▆▇ ➡️ 11ms
canvas-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
canvas-idle: style recalc count █▃▅▂▅▆▃▁▂▁▂▅▆▅▆ ➡️ 12
canvas-idle: task duration ▃▃▃▆▂▃▃▅▆▂█▃▁▃▃ ➡️ 391ms
canvas-idle: script duration ▄▃▅▇▂▅▃▆▇▅█▄▁▅▆ ➡️ 27ms
canvas-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: heap used ➡️ NaN MB
canvas-idle: DOM nodes █▇▆▅▃▇▃▁▂▂▅▆▆▆▇ ➡️ 24
canvas-idle: event listeners ▅█▅▄▁▅▁▁▁▄▅▅▁▅▄ 📉 11
canvas-mouse-sweep: avg frame time ▆█▆▃▁▃▁▆▆▁▃▆▆▃▃ ➡️ 17ms
canvas-mouse-sweep: p95 frame time ➡️ NaNms
canvas-mouse-sweep: layout duration ▁▃▂▄▁▂▁▃▆▂█▇▆▄▃ ➡️ 4ms
canvas-mouse-sweep: style recalc duration ▄▄▂▄▁▂▃▃▅▄█▆▂▄▄ ➡️ 43ms
canvas-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 12
canvas-mouse-sweep: style recalc count █▅▄▃▂▂▁▄▄▅▆▅▂▇▄ ➡️ 79
canvas-mouse-sweep: task duration █▆▄▂▂▃▂▄▄▅█▆▁▆▄ ➡️ 868ms
canvas-mouse-sweep: script duration ▄▅▄▆▄▆▆▆▅▅█▆▁▅▆ ➡️ 139ms
canvas-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-mouse-sweep: heap used ➡️ NaN MB
canvas-mouse-sweep: DOM nodes █▅▃▃▁▂▂▃▂▄▆▅▃▅▅ ➡️ 64
canvas-mouse-sweep: event listeners █▁▁▁▁▁▇▁▁▁██▇▁█ 📈 13
canvas-zoom-sweep: avg frame time ▅▅█▄▅▁▁▁▅▁▁▅▄▅▁ ➡️ 17ms
canvas-zoom-sweep: p95 frame time ➡️ NaNms
canvas-zoom-sweep: layout duration ▆▅▅▄▁▁█▅▃▅▇▆▁▂▆ ➡️ 1ms
canvas-zoom-sweep: style recalc duration ▆▅▄▆▅▃█▆▇▅▇▄▁▃▅ ➡️ 20ms
canvas-zoom-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 6
canvas-zoom-sweep: style recalc count ▁▁▃▄▆▃▆█▄▄▆▁▆▁▆ ➡️ 32
canvas-zoom-sweep: task duration ▄▂▁▇▂▂▄▅▆▃█▄▁▁▅ ➡️ 338ms
canvas-zoom-sweep: script duration ▃▃▂▇▂▂▅▇▆▅█▄▁▂▆ ➡️ 30ms
canvas-zoom-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-zoom-sweep: heap used ➡️ NaN MB
canvas-zoom-sweep: DOM nodes ▄▃▁▅█▁▃▆▄▅▅▃▃▄▃ ➡️ 79
canvas-zoom-sweep: event listeners ▁▁▂▅█▂▁▅▁▅▅▄▁▅▁ ➡️ 19
dom-widget-clipping: avg frame time ▂▄▅▅▂▄█▇▅▇▇▅▅▁▇ ➡️ 17ms
dom-widget-clipping: p95 frame time ➡️ NaNms
dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: style recalc duration ▆▆▂▆▄▃██▄▁▆▇▆▃▅ ➡️ 10ms
dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
dom-widget-clipping: style recalc count ▇█▅█▅▄█▇▇▁▇▄▇▂▅ ➡️ 13
dom-widget-clipping: task duration ▃▃▁▅▄▃▅▆▅▂▇█▁▅▅ ➡️ 371ms
dom-widget-clipping: script duration ▅▄▄▆▆▅▇▇▆▃█▇▁▇▇ ➡️ 71ms
dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: heap used ➡️ NaN MB
dom-widget-clipping: DOM nodes ▇▇▄▇▅▄█▇▅▁▅▄▇▃▄ ➡️ 21
dom-widget-clipping: event listeners ▅▅▅▅▁▅██▁▁▁▁█▁▁ 📉 2
large-graph-idle: avg frame time ▅▅▅▅▅▂▁▂▄▅▄▂▂▅█ ➡️ 17ms
large-graph-idle: p95 frame time ➡️ NaNms
large-graph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: style recalc duration ▅▅▅▆▄▅▃▄▅▅▆█▁▄▆ ➡️ 13ms
large-graph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-idle: style recalc count █▆█▃▃▁▃▆▃▆▆▃▆██ ➡️ 12
large-graph-idle: task duration ▂▃▂▆▂▃▃▇▅▃██▁▂▅ ➡️ 569ms
large-graph-idle: script duration ▄▅▄▆▄▅▅▇▆▅█▆▁▃▆ ➡️ 110ms
large-graph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: heap used ➡️ NaN MB
large-graph-idle: DOM nodes ▆█▅▂▅▃▁▂▃▅▅▆▂▆▅ ➡️ 25
large-graph-idle: event listeners ███▇██▄▁▄▇▇█▂█▇ ➡️ 29
large-graph-pan: avg frame time ▆▃▃▆█▃▁█▆▆▆▆█▁▆ ➡️ 17ms
large-graph-pan: p95 frame time ➡️ NaNms
large-graph-pan: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: style recalc duration ▃▂▄▄▁▅▂▂▁▄▄█▃▁▂ ➡️ 17ms
large-graph-pan: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-pan: style recalc count ▆▃█▂▃▂▂▂▁▇▅▃█▆▃ ➡️ 69
large-graph-pan: task duration ▄▃▄▆▄▄▄▆▄▄█▆▁▂▅ ➡️ 1100ms
large-graph-pan: script duration ▅▄▅▆▆▅▄▆▄▅█▄▁▄▅ ➡️ 413ms
large-graph-pan: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: heap used ➡️ NaN MB
large-graph-pan: DOM nodes ▅▃▆▂▄▁▃▁▁▅▁▂█▅▂ ➡️ 18
large-graph-pan: event listeners █▆█▁▁▆▁▁▃▆▁▃██▃ ➡️ 5
minimap-idle: avg frame time ▃▆▆▃█▁█▆▆▃▃▆█▆█ ➡️ 17ms
minimap-idle: p95 frame time ➡️ NaNms
minimap-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: style recalc duration ▄█▁█▅▅█▅▅▃▅▁▁▄▆ ➡️ 10ms
minimap-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
minimap-idle: style recalc count ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 9
minimap-idle: task duration ▃▄▁▅▁▃▄▅▇▃█▅▁▁▅ ➡️ 547ms
minimap-idle: script duration ▄▆▃▇▃▅▆▆▇▅█▅▁▃▆ ➡️ 106ms
minimap-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: heap used ➡️ NaN MB
minimap-idle: DOM nodes ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 19
minimap-idle: event listeners ▃▃▆▁▁▁▃▁▁▆▁▃█▆▁ ➡️ 4
subgraph-dom-widget-clipping: avg frame time ▅▄▄▄▄▄█▄▄▄▃▁▆▃▃ ➡️ 17ms
subgraph-dom-widget-clipping: p95 frame time ➡️ NaNms
subgraph-dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: style recalc duration ▂▄▃▅▅▃▂▅▇▃▄█▁▄▆ ➡️ 14ms
subgraph-dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-dom-widget-clipping: style recalc count ▇█▆▃▆▃▁▆█▇▃▆▇█▅ ➡️ 48
subgraph-dom-widget-clipping: task duration ▂▃▃▆▅▅▂▅█▂▆█▁▂▇ ➡️ 398ms
subgraph-dom-widget-clipping: script duration ▃▃▃▄▅▅▂▄█▂▅▇▁▂▅ ➡️ 131ms
subgraph-dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: heap used ➡️ NaN MB
subgraph-dom-widget-clipping: DOM nodes ▅▇▅▂▅▂▁▅▅▅▁▇▅█▄ ➡️ 22
subgraph-dom-widget-clipping: event listeners ▅▅▅▂▅▁▅██▁▁█▅█▅ 📈 16
subgraph-idle: avg frame time ▆▆█▁▆▃▆▆▆▃▆▁▃▆█ ➡️ 17ms
subgraph-idle: p95 frame time ➡️ NaNms
subgraph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: style recalc duration ▁▇▃▆▂▄▂▃▃▆▆▄▃▇█ ➡️ 12ms
subgraph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-idle: style recalc count ▃▆▃▃▂▅▁▂▁▆▃▃██▇ ➡️ 12
subgraph-idle: task duration ▁▃▁▇▁▁▃▆▅▂█▅▁▁▄ ➡️ 378ms
subgraph-idle: script duration ▁▃▂▇▁▂▃▇▆▂█▅▂▁▅ ➡️ 22ms
subgraph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: heap used ➡️ NaN MB
subgraph-idle: DOM nodes ▃▅▃▂▁▄▁▂▁▅▃▂▇█▇ ➡️ 24
subgraph-idle: event listeners ▁▅▁▁▁▁▁▁▁▅▄▁███ 📈 21
subgraph-mouse-sweep: avg frame time ▅▄▁▃▃▄▆▄▆▃▃█▁▃▃ ➡️ 17ms
subgraph-mouse-sweep: p95 frame time ➡️ NaNms
subgraph-mouse-sweep: layout duration ▁▄▄▄▃▃▅▅▅▂█▇▂▃▆ ➡️ 5ms
subgraph-mouse-sweep: style recalc duration ▃▂▄▅▂▃▄▅█▃█▆▁▂▅ ➡️ 43ms
subgraph-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 16
subgraph-mouse-sweep: style recalc count ▅▂▅▅▁▄▃▅█▅▆▄▂▄▅ ➡️ 81
subgraph-mouse-sweep: task duration ▃▂▄▅▂▄▄▅▇▄█▆▁▃▅ ➡️ 785ms
subgraph-mouse-sweep: script duration ▄▅▄▇▅▅▆▇▆▅██▁▄▆ ➡️ 105ms
subgraph-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-mouse-sweep: heap used ➡️ NaN MB
subgraph-mouse-sweep: DOM nodes ▅▁▄▅▁▄▃▃█▅▅▄▂▅▃ ➡️ 66
subgraph-mouse-sweep: event listeners ▇▁▂▇▁▂▂▂█▇▂▂▇▇▂ 📈 5
workflow-execution: avg frame time ▆▆▆▄▆▆▃▄▁▄█▆▅▄▆ ➡️ 17ms
workflow-execution: p95 frame time ➡️ NaNms
workflow-execution: layout duration ▁▆▁▃▂▄▃▂▃▃▅█▄▂▅ ➡️ 2ms
workflow-execution: style recalc duration ▃▇▅▇▁▅▆▇█▁██▂▄▆ ➡️ 25ms
workflow-execution: layout count ▁█▂▃▂▃▃▁▃▃▄▃▂▃▂ ➡️ 5
workflow-execution: style recalc count ▃█▅▇▁▄▅▆▅▅▅▅▄▄▂ ➡️ 15
workflow-execution: task duration ▂▅▄▅▁▄▆▆▆▁▇█▁▃▃ ➡️ 120ms
workflow-execution: script duration ▄▃▄▄▃▅▄▅▆▂▇█▁▃▄ ➡️ 29ms
workflow-execution: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
workflow-execution: heap used ➡️ NaN MB
workflow-execution: DOM nodes ▂█▃▆▁▄▃▅▃█▃▃▄▃▁ ➡️ 152
workflow-execution: event listeners ▅███▁▅███▁██▅█▅ ➡️ 49
Raw data
{
  "timestamp": "2026-05-15T21:31:32.722Z",
  "gitSha": "4cddcb316a47cf12d3355b0a019b784e43c15acd",
  "branch": "glary/raf-batch-vue-node-resize-tracking",
  "measurements": [
    {
      "name": "canvas-idle",
      "durationMs": 2028.1359999999893,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 9.442000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 415.91999999999996,
      "heapDeltaBytes": 23257564,
      "heapUsedBytes": 71374336,
      "domNodes": 20,
      "jsHeapTotalBytes": 14417920,
      "scriptDurationMs": 28.026,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-idle",
      "durationMs": 2044.685999999956,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 8.122,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 400.822,
      "heapDeltaBytes": 22901216,
      "heapUsedBytes": 71430300,
      "domNodes": 16,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 21.47600000000001,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1970.8540000000028,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 45.617000000000004,
      "layouts": 12,
      "layoutDurationMs": 4.347,
      "taskDurationMs": 944.768,
      "heapDeltaBytes": 4265476,
      "heapUsedBytes": 52915196,
      "domNodes": -262,
      "jsHeapTotalBytes": 15593472,
      "scriptDurationMs": 137.03600000000003,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1852.0379999999932,
      "styleRecalcs": 73,
      "styleRecalcDurationMs": 42.278999999999996,
      "layouts": 12,
      "layoutDurationMs": 4.272,
      "taskDurationMs": 830.926,
      "heapDeltaBytes": 1303912,
      "heapUsedBytes": 49958004,
      "domNodes": -264,
      "jsHeapTotalBytes": 15855616,
      "scriptDurationMs": 124.20799999999998,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1733.9910000000032,
      "styleRecalcs": 30,
      "styleRecalcDurationMs": 18.819999999999997,
      "layouts": 6,
      "layoutDurationMs": 0.77,
      "taskDurationMs": 348.18000000000006,
      "heapDeltaBytes": 511628,
      "heapUsedBytes": 49268004,
      "domNodes": 75,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 25.805000000000003,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1762.1919999999704,
      "styleRecalcs": 32,
      "styleRecalcDurationMs": 21.775000000000002,
      "layouts": 6,
      "layoutDurationMs": 0.915,
      "taskDurationMs": 396.09700000000004,
      "heapDeltaBytes": 23339200,
      "heapUsedBytes": 72975508,
      "domNodes": 79,
      "jsHeapTotalBytes": 24641536,
      "scriptDurationMs": 33.84000000000001,
      "eventListeners": 21,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 594.1919999999925,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 8.459000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 388.82699999999994,
      "heapDeltaBytes": 9569384,
      "heapUsedBytes": 58654520,
      "domNodes": 18,
      "jsHeapTotalBytes": 14417920,
      "scriptDurationMs": 65.845,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 551.1419999999703,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 6.898999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 342.057,
      "heapDeltaBytes": 8862468,
      "heapUsedBytes": 57465820,
      "domNodes": 16,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 58.03000000000001,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999727
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2040.2270000000158,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.989999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 599.89,
      "heapDeltaBytes": 7971092,
      "heapUsedBytes": 66553316,
      "domNodes": -263,
      "jsHeapTotalBytes": 290816,
      "scriptDurationMs": 99.06400000000001,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2023.2789999999454,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 9.139999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 598.843,
      "heapDeltaBytes": 14353128,
      "heapUsedBytes": 79472232,
      "domNodes": 20,
      "jsHeapTotalBytes": 14036992,
      "scriptDurationMs": 105.47600000000001,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2169.4840000000113,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 21.344999999999995,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1162.4279999999999,
      "heapDeltaBytes": 11220904,
      "heapUsedBytes": 70878632,
      "domNodes": -263,
      "jsHeapTotalBytes": 5476352,
      "scriptDurationMs": 399.419,
      "eventListeners": -157,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2250.2420000000143,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 21.272999999999996,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1253.8819999999998,
      "heapDeltaBytes": 50765568,
      "heapUsedBytes": 110804348,
      "domNodes": -259,
      "jsHeapTotalBytes": 41127936,
      "scriptDurationMs": 399.545,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3232.3560000000384,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 21.543,
      "layouts": 60,
      "layoutDurationMs": 8.864,
      "taskDurationMs": 1448.7259999999999,
      "heapDeltaBytes": 8191336,
      "heapUsedBytes": 69673716,
      "domNodes": -265,
      "jsHeapTotalBytes": 3756032,
      "scriptDurationMs": 517.3530000000001,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3196.388000000013,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 21.735,
      "layouts": 60,
      "layoutDurationMs": 8.940999999999999,
      "taskDurationMs": 1438.002,
      "heapDeltaBytes": 14058312,
      "heapUsedBytes": 75767824,
      "domNodes": -267,
      "jsHeapTotalBytes": 3493888,
      "scriptDurationMs": 519.574,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "minimap-idle",
      "durationMs": 2103.6040000000185,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 7.715000000000003,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 695.451,
      "heapDeltaBytes": 44657252,
      "heapUsedBytes": 106069924,
      "domNodes": -265,
      "jsHeapTotalBytes": 32796672,
      "scriptDurationMs": 108.561,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "minimap-idle",
      "durationMs": 2037.3710000000074,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.491999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 700.6680000000001,
      "heapDeltaBytes": 43379456,
      "heapUsedBytes": 103190240,
      "domNodes": -262,
      "jsHeapTotalBytes": 32796672,
      "scriptDurationMs": 114.182,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 588.4759999999574,
      "styleRecalcs": 47,
      "styleRecalcDurationMs": 11.745999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 386.31499999999994,
      "heapDeltaBytes": 9146564,
      "heapUsedBytes": 58097076,
      "domNodes": 19,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 136.383,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 850.166999999999,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 16.136,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 559.7959999999999,
      "heapDeltaBytes": -4403208,
      "heapUsedBytes": 62191824,
      "domNodes": 21,
      "jsHeapTotalBytes": 23425024,
      "scriptDurationMs": 162.71999999999997,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666682,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2020.8519999999908,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 11.449999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 432.84399999999994,
      "heapDeltaBytes": 1551688,
      "heapUsedBytes": 68223900,
      "domNodes": 22,
      "jsHeapTotalBytes": 18706432,
      "scriptDurationMs": 27.166000000000004,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2006.621999999993,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.873,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 407.903,
      "heapDeltaBytes": 23116272,
      "heapUsedBytes": 71807400,
      "domNodes": 17,
      "jsHeapTotalBytes": 14942208,
      "scriptDurationMs": 19.813,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1740.554000000003,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 41.909,
      "layouts": 16,
      "layoutDurationMs": 5.084,
      "taskDurationMs": 809.452,
      "heapDeltaBytes": -21303100,
      "heapUsedBytes": 47235776,
      "domNodes": -258,
      "jsHeapTotalBytes": 26173440,
      "scriptDurationMs": 104.70700000000001,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 2022.8270000000066,
      "styleRecalcs": 85,
      "styleRecalcDurationMs": 49.272999999999996,
      "layouts": 16,
      "layoutDurationMs": 7.175,
      "taskDurationMs": 974.713,
      "heapDeltaBytes": -360912,
      "heapUsedBytes": 48499844,
      "domNodes": -263,
      "jsHeapTotalBytes": 15593472,
      "scriptDurationMs": 99.74699999999999,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-transition-enter",
      "durationMs": 1150.811000000033,
      "styleRecalcs": 15,
      "styleRecalcDurationMs": 28.343999999999994,
      "layouts": 4,
      "layoutDurationMs": 11.741999999999999,
      "taskDurationMs": 899.9710000000001,
      "heapDeltaBytes": 81823660,
      "heapUsedBytes": 224268912,
      "domNodes": 13513,
      "jsHeapTotalBytes": 80478208,
      "scriptDurationMs": 34.063999999999986,
      "eventListeners": 2527,
      "totalBlockingTimeMs": 156,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8192.68999999997,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 61.098000000000006,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3949.1100000000006,
      "heapDeltaBytes": 15958232,
      "heapUsedBytes": 75757448,
      "domNodes": -260,
      "jsHeapTotalBytes": 4718592,
      "scriptDurationMs": 1258.775,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8206.04000000003,
      "styleRecalcs": 249,
      "styleRecalcDurationMs": 62.324000000000005,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 4029.863,
      "heapDeltaBytes": 13183852,
      "heapUsedBytes": 71962224,
      "domNodes": -262,
      "jsHeapTotalBytes": 7573504,
      "scriptDurationMs": 1282.2469999999998,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.80000000000109
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 13731.302999999969,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 13719.362,
      "heapDeltaBytes": -39939984,
      "heapUsedBytes": 169736876,
      "domNodes": -8331,
      "jsHeapTotalBytes": 26275840,
      "scriptDurationMs": 593.6680000000001,
      "eventListeners": -16460,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.80000000000291
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 16835.53400000005,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 16806.846,
      "heapDeltaBytes": 55263144,
      "heapUsedBytes": 299600620,
      "domNodes": -8333,
      "jsHeapTotalBytes": 39645184,
      "scriptDurationMs": 710.272,
      "eventListeners": -16486,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.776666666666642,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 16046.425999999996,
      "styleRecalcs": 81,
      "styleRecalcDurationMs": 21.16699999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 16009.577,
      "heapDeltaBytes": -42471156,
      "heapUsedBytes": 177388296,
      "domNodes": -8331,
      "jsHeapTotalBytes": -11296768,
      "scriptDurationMs": 914.5509999999999,
      "eventListeners": -16458,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.779999999999927,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 15680.050999999934,
      "styleRecalcs": 80,
      "styleRecalcDurationMs": 21.631999999999984,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 15657.008000000002,
      "heapDeltaBytes": -13891636,
      "heapUsedBytes": 191543332,
      "domNodes": -8331,
      "jsHeapTotalBytes": 23830528,
      "scriptDurationMs": 930.858,
      "eventListeners": -16460,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.776666666666642,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "workflow-execution",
      "durationMs": 464.5269999999755,
      "styleRecalcs": 22,
      "styleRecalcDurationMs": 25.564999999999998,
      "layouts": 4,
      "layoutDurationMs": 1.58,
      "taskDurationMs": 125.05199999999999,
      "heapDeltaBytes": 5364144,
      "heapUsedBytes": 56470088,
      "domNodes": 192,
      "jsHeapTotalBytes": 262144,
      "scriptDurationMs": 23.461,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999727
    },
    {
      "name": "workflow-execution",
      "durationMs": 128.2730000000356,
      "styleRecalcs": 12,
      "styleRecalcDurationMs": 19.5,
      "layouts": 4,
      "layoutDurationMs": 1.5739999999999998,
      "taskDurationMs": 97.524,
      "heapDeltaBytes": 3366744,
      "heapUsedBytes": 55268612,
      "domNodes": 145,
      "jsHeapTotalBytes": 262144,
      "scriptDurationMs": 19.377999999999993,
      "eventListeners": 37,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    }
  ]
}

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.

Caution

Some comments are outside the diff and can’t be posted inline due to platform limitations.

⚠️ Outside diff range comments (1)
src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts (1)

91-102: ⚠️ Potential issue | 🟠 Major | ⚡ Quick win

Move hidden-tab deferral into the visibility watch callback.

When a ResizeObserver fires while the tab is visible, its measurement is queued on RAF via rafBatch.schedule(). If the tab becomes hidden before that RAF callback executes, the browser will suspend the callback indefinitely. When the tab becomes visible again and RAF resumes, flushPendingMeasurements() will execute with visibility.value === 'visible', causing the hidden-state check at line 129 to be skipped and the stale measurements to be applied instead of being re-deferred.

Move the hidden-state re-deferral logic from lines 127–136 into the visibility watch at lines 91–102 so that pending measurements are immediately shunted to deferredElements the moment the tab is hidden, before RAF suspension takes effect.

Suggested fix
 watch(visibility, (state) => {
-  if (state !== 'visible' || deferredElements.size === 0) return
+  if (state === 'hidden') {
+    for (const element of pendingMeasurements.keys()) {
+      deferredElements.add(element)
+      markElementForFreshMeasurement(element)
+      resizeObserver.unobserve(element)
+    }
+    pendingMeasurements.clear()
+    return
+  }
+
+  if (deferredElements.size === 0) return
 
   // Re-observe deferred elements to trigger fresh measurements
   for (const element of deferredElements) {
     if (element.isConnected) {
       markElementForFreshMeasurement(element)
@@
-  // Skip measurements when tab is hidden — bounding rects are unreliable.
-  // Re-defer the elements so they get a fresh measurement on revisit.
-  if (visibility.value === 'hidden') {
-    for (const element of pendingMeasurements.keys()) {
-      deferredElements.add(element)
-      markElementForFreshMeasurement(element)
-      resizeObserver.unobserve(element)
-    }
-    pendingMeasurements.clear()
-    return
-  }
+  if (visibility.value === 'hidden') return
🤖 Prompt for AI Agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

In `@src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts`
around lines 91 - 102, The visibility watch must immediately re-defer any
pending measurements when the tab becomes hidden to avoid RAF suspension
applying stale measurements; modify the callback on visibility (the watch for
visibility) so that when state !== 'visible' you iterate any elements currently
queued for RAF (the pending/queued set used by
rafBatch.schedule()/flushPendingMeasurements()), move them into
deferredElements, call markElementForFreshMeasurement(element) for each, and
stop/clear the RAF batch/scheduled callback (so flushPendingMeasurements() will
not run later), and when visibility becomes 'visible' re-observe
deferredElements with resizeObserver and clear deferredElements as the existing
code does. Ensure you reference and update the same pending-measurement
structures and rafBatch scheduling used by rafBatch.schedule() /
flushPendingMeasurements() so there is no race between RAF and visibility
changes.
🤖 Prompt for all review comments with AI agents
Verify each finding against current code. Fix only still-valid issues, skip the
rest with a brief reason, keep changes minimal, and validate.

Outside diff comments:
In `@src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts`:
- Around line 91-102: The visibility watch must immediately re-defer any pending
measurements when the tab becomes hidden to avoid RAF suspension applying stale
measurements; modify the callback on visibility (the watch for visibility) so
that when state !== 'visible' you iterate any elements currently queued for RAF
(the pending/queued set used by rafBatch.schedule()/flushPendingMeasurements()),
move them into deferredElements, call markElementForFreshMeasurement(element)
for each, and stop/clear the RAF batch/scheduled callback (so
flushPendingMeasurements() will not run later), and when visibility becomes
'visible' re-observe deferredElements with resizeObserver and clear
deferredElements as the existing code does. Ensure you reference and update the
same pending-measurement structures and rafBatch scheduling used by
rafBatch.schedule() / flushPendingMeasurements() so there is no race between RAF
and visibility changes.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 064ce991-3777-4739-8bb5-55ef07d46caa

📥 Commits

Reviewing files that changed from the base of the PR and between 7160a9e and 48a3269.

📒 Files selected for processing (2)
  • src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.test.ts
  • src/renderer/extensions/vueNodes/composables/useVueNodeResizeTracking.ts

@codecov
Copy link
Copy Markdown

codecov Bot commented May 15, 2026

Codecov Report

❌ Patch coverage is 83.33333% with 7 lines in your changes missing coverage. Please review.

Files with missing lines Patch % Lines
...s/vueNodes/composables/useVueNodeResizeTracking.ts 83.33% 7 Missing ⚠️
@@             Coverage Diff             @@
##             main   #12299       +/-   ##
===========================================
- Coverage   74.67%   59.58%   -15.10%     
===========================================
  Files        1526     1412      -114     
  Lines       95359    71900    -23459     
  Branches    27134    19029     -8105     
===========================================
- Hits        71212    42839    -28373     
- Misses      23285    28588     +5303     
+ Partials      862      473      -389     
Flag Coverage Δ
e2e ?
unit 59.58% <83.33%> (+<0.01%) ⬆️

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

Files with missing lines Coverage Δ
...s/vueNodes/composables/useVueNodeResizeTracking.ts 89.78% <83.33%> (+1.48%) ⬆️

... and 1022 files with indirect coverage changes

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

Address CodeRabbit review: between an RO callback and its scheduled RAF
flush, the tab can be hidden. The browser then suspends RAF until the
tab is shown again, at which point flushPendingMeasurements() runs with
visibility === 'visible' and writes stale bounds.

Move the hidden-state re-deferral into the visibility watch so pending
measurements are immediately shunted to deferredElements / unobserved
the moment the tab is hidden, and cancel the scheduled RAF. The flush
path keeps a defensive hidden check for any RO fire that races a flip
back to visible.
@glary-bot
Copy link
Copy Markdown

glary-bot Bot commented May 15, 2026

@coderabbitai Addressed in 63d62bc. Moved the hidden-state re-deferral into the visibility watch so pending measurements are immediately shunted to deferredElements (and the scheduled RAF cancelled) the moment the tab is hidden — no longer relying on the suspended RAF callback to defer them later.

Kept a defensive visibility.value === 'hidden' check at the top of flushPendingMeasurements() to handle the inverse race: an RO callback that fired while visible can still land in the flush after a hidden→visible flip during the same task. Pulled the defer body into a shared deferElementsForHiddenTab() helper so both call sites stay in sync.

Added a regression test (re-defers pending measurements when the tab becomes hidden before flush) that verifies the full hidden→visible cycle: RO callback queued → tab hidden → flush is a no-op + element unobserved → tab visible → element re-observed. 9/9 tests passing.

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

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants