Skip to content

fix(load3d): update renderer pixel ratio on canvas zoom to fix LOD resolution#11734

Open
kaili-yang wants to merge 3 commits intomainfrom
fix/load3d-lod-zoom-resolution
Open

fix(load3d): update renderer pixel ratio on canvas zoom to fix LOD resolution#11734
kaili-yang wants to merge 3 commits intomainfrom
fix/load3d-lod-zoom-resolution

Conversation

@kaili-yang
Copy link
Copy Markdown
Collaborator

@kaili-yang kaili-yang commented Apr 29, 2026

Summary

Preview 3D and Animation nodes were stuck at the LOD from initial page load because CSS scale3d transforms don't affect clientWidth/clientHeighthandleResize() always received layout-space dimensions regardless of zoom level. This fix passes ds.scale as the renderer pixel ratio so the 3D scene renders at the correct visual resolution when the graph is zoomed in or out.

Changes

  • What: In Load3d.handleResize(), call renderer.setPixelRatio(ds.scale) before setSize so pixel density scales with canvas zoom. A getZoomScale callback is threaded through Load3DOptionsLoad3d constructor → handleResize. In useLoad3d, a watcher on canvasStore.appScalePercentage triggers handleResize whenever the zoom level changes.
  • What: Fix SceneManager.captureScene() to save and restore the renderer's logical size and pixel ratio around capture, so exact-pixel output is unaffected by the current zoom state.

Review Focus

  • handleResize now calls setPixelRatio before setSize. Three.js renders at logicalWidth × pixelRatio physical pixels while CSS displays it at logicalWidth CSS pixels — this is the standard pattern for HiDPI but here used to match the visual zoom level.
  • captureScene must reset pixelRatio to 1 so setSize(w, h) produces exactly w×h pixel output. It saves and restores both logical size and pixel ratio via renderer.getSize() / renderer.getPixelRatio().
  • The zoom watcher is guarded with getActivePinia() to avoid errors in unit tests and non-Pinia contexts.

Test

after
1
2

before


Note

Cursor Bugbot is generating a summary for commit aa36082. Configure here.

┆Issue is synchronized with this Notion page by Unito

…solution

Preview 3D and Animation nodes were stuck at the LOD from initial page load
because CSS scale3d transforms don't affect clientWidth/clientHeight, so
handleResize() always used layout-space dimensions regardless of zoom level.

Fix: pass ds.scale as pixelRatio to the renderer so the 3D scene renders at
the correct visual resolution when the graph is zoomed in or out. Also fixes
captureScene() to save/restore logical size and pixelRatio so exact-pixel
captures are unaffected by zoom state.
Cover the three new behaviors introduced by the LOD zoom fix:
- handleResize uses getZoomScaleCallback value for setPixelRatio
- handleResize falls back to pixelRatio 1 when no callback is set
- useLoad3d passes getZoomScale option to createLoad3d
- zoom watcher calls handleResize when canvas appScalePercentage changes
@kaili-yang kaili-yang requested a review from a team April 29, 2026 01:22
@kaili-yang kaili-yang requested a review from jtydhr88 as a code owner April 29, 2026 01:22
@dosubot dosubot Bot added the size:M This PR changes 30-99 lines, ignoring generated files. label Apr 29, 2026
@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai Bot commented Apr 29, 2026

📝 Walkthrough

Walkthrough

Adds zoom-aware rendering: introduces a getZoomScale callback on Load3D options, uses it to set renderer pixel ratio during resize, updates scene capture to preserve/restore renderer size and pixel ratio, and watches canvas scale in the composable to trigger load3d.handleResize().

Changes

Cohort / File(s) Summary
Composable Integration
src/composables/useLoad3d.ts, src/composables/useLoad3d.test.ts
Composable now watches useCanvasStore().appScalePercentage and calls load3d.handleResize() on change; injects getZoomScale callback into createLoad3d. Tests mock Pinia/canvas store and assert watcher and callback behavior.
Load3d Core
src/extensions/core/load3d/Load3d.ts, src/extensions/core/load3d/Load3d.test.ts
Load3d accepts optional getZoomScale in options, stores it, and uses its return (default 1) to call renderer.setPixelRatio() inside handleResize(). Tests added for pixel-ratio handling and fallback.
Scene Capture
src/extensions/core/load3d/SceneManager.ts
captureScene now saves/restores renderer size and pixel ratio via getSize()/getPixelRatio(), forces setPixelRatio(1) for capture, then restores original pixel ratio and size.
Interface Definition
src/extensions/core/load3d/interfaces.ts
Load3DOptions extended with optional getZoomScale?: () => number to expose zoom scale to Load3d.

Sequence Diagram

sequenceDiagram
    participant CanvasStore as Canvas Store
    participant Composable as useLoad3d
    participant Load3d as Load3d
    participant Renderer as THREE.WebGLRenderer

    CanvasStore->>Composable: appScalePercentage change (watcher)
    Composable->>Load3d: handleResize()
    Load3d->>Load3d: call getZoomScale() (if provided)
    Load3d-->>Renderer: setPixelRatio(zoomScale or 1)
    Renderer->>Renderer: adjust internal resolution / render size
Loading

Estimated code review effort

🎯 3 (Moderate) | ⏱️ ~20 minutes

Poem

🐰 I hopped through pixels, counted each scale,
A watcher nudged, the renderer set sail,
Capture paused, then restored just right,
Zoom and resize — a snug little light! ✨

🚥 Pre-merge checks | ✅ 7
✅ Passed checks (7 passed)
Check name Status Explanation
Title check ✅ Passed The title clearly and concisely describes the main change: updating the renderer pixel ratio on canvas zoom to fix LOD resolution, which is directly supported by the changeset across Load3d, useLoad3d, and related components.
Docstring Coverage ✅ Passed No functions found in the changed files to evaluate docstring coverage. Skipping docstring coverage check.
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.
End-To-End Regression Coverage For Fixes ✅ Passed PR introduces browser_tests/tests/load3d/load3dLod.spec.ts, a new Playwright regression test suite that verifies the LOD resolution fix by asserting canvas width increases on zoom.
Adr Compliance For Entity/Litegraph Changes ✅ Passed Check applies only to changes in src/lib/litegraph/, src/ecs/, or graph entity files. PR modifies only 3D loader extension and composable code, none under applicable paths.
Description check ✅ Passed The PR description is comprehensive, covering the issue, changes made, review focus areas, and test evidence with screenshots.

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

✨ 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 fix/load3d-lod-zoom-resolution

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 29, 2026

🎨 Storybook: ✅ Built — View Storybook

Details

⏰ Completed at: 04/29/2026, 03:25:06 AM UTC

Links

@github-actions
Copy link
Copy Markdown

github-actions Bot commented Apr 29, 2026

🎭 Playwright: ❌ 1417 passed, 1 failed · 4 flaky

❌ Failed Tests

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

@github-actions
Copy link
Copy Markdown

github-actions Bot commented Apr 29, 2026

📦 Bundle: 5.23 MB gzip 🔴 +231 B

Details

Summary

  • Raw size: 24.1 MB baseline 24.1 MB — 🔴 +474 B
  • Gzip: 5.23 MB baseline 5.23 MB — 🔴 +231 B
  • Brotli: 4.05 MB baseline 4.05 MB — 🔴 +315 B
  • Bundles: 257 current • 257 baseline • 118 added / 118 removed

Category Glance
Data & Services 🔴 +289 B (3.04 MB) · Utilities & Hooks 🔴 +185 B (364 kB) · Vendor & Third-Party ⚪ 0 B (9.88 MB) · Other ⚪ 0 B (8.83 MB) · Graph Workspace ⚪ 0 B (1.24 MB) · Panels & Settings ⚪ 0 B (488 kB) · + 5 more

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

Main entry bundles and manifests

File Before After Δ Raw Δ Gzip Δ Brotli
assets/index-DDcrOjQJ.js (new) 22.5 kB 🔴 +22.5 kB 🔴 +7.99 kB 🔴 +6.85 kB
assets/index-ks1u9d_N.js (removed) 22.5 kB 🟢 -22.5 kB 🟢 -7.98 kB 🟢 -6.87 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-BO7WhBaY.js (new) 1.24 MB 🔴 +1.24 MB 🔴 +264 kB 🔴 +199 kB
assets/GraphView-DekpTE6f.js (removed) 1.24 MB 🟢 -1.24 MB 🟢 -264 kB 🟢 -199 kB

Status: 1 added / 1 removed

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

Top-level views, pages, and routed surfaces

File Before After Δ Raw Δ Gzip Δ Brotli
assets/CloudSurveyView-Ds1M-fhm.js (removed) 15.7 kB 🟢 -15.7 kB 🟢 -3.41 kB 🟢 -2.92 kB
assets/CloudSurveyView-PDtU2D4Y.js (new) 15.7 kB 🔴 +15.7 kB 🔴 +3.41 kB 🔴 +2.92 kB
assets/CloudLoginView-Bfrb1S7C.js (new) 12.2 kB 🔴 +12.2 kB 🔴 +3.43 kB 🔴 +3.04 kB
assets/CloudLoginView-DyEmkJS9.js (removed) 12.2 kB 🟢 -12.2 kB 🟢 -3.43 kB 🟢 -3.04 kB
assets/CloudSignupView-B8OvUF3U.js (removed) 9.94 kB 🟢 -9.94 kB 🟢 -2.93 kB 🟢 -2.57 kB
assets/CloudSignupView-DxFvp05N.js (new) 9.94 kB 🔴 +9.94 kB 🔴 +2.93 kB 🔴 +2.58 kB
assets/UserCheckView-B-Mc91zH.js (new) 9.07 kB 🔴 +9.07 kB 🔴 +2.33 kB 🔴 +2.04 kB
assets/UserCheckView-kTmRVvmQ.js (removed) 9.07 kB 🟢 -9.07 kB 🟢 -2.33 kB 🟢 -2.04 kB
assets/CloudLayoutView-B2Sz9yB0.js (removed) 7.7 kB 🟢 -7.7 kB 🟢 -2.43 kB 🟢 -2.11 kB
assets/CloudLayoutView-C-K49jvZ.js (new) 7.7 kB 🔴 +7.7 kB 🔴 +2.44 kB 🔴 +2.12 kB
assets/CloudForgotPasswordView-B7h3bJLE.js (new) 6.11 kB 🔴 +6.11 kB 🔴 +2.17 kB 🔴 +1.9 kB
assets/CloudForgotPasswordView-M-8iM71v.js (removed) 6.11 kB 🟢 -6.11 kB 🟢 -2.17 kB 🟢 -1.92 kB
assets/CloudAuthTimeoutView-C4T8tf5s.js (removed) 5.47 kB 🟢 -5.47 kB 🟢 -2 kB 🟢 -1.76 kB
assets/CloudAuthTimeoutView-CuZFo4s6.js (new) 5.47 kB 🔴 +5.47 kB 🔴 +2 kB 🔴 +1.75 kB
assets/CloudSubscriptionRedirectView-CMAMHJYN.js (new) 5.25 kB 🔴 +5.25 kB 🔴 +1.98 kB 🔴 +1.76 kB
assets/CloudSubscriptionRedirectView-Dlid9K9w.js (removed) 5.25 kB 🟢 -5.25 kB 🟢 -1.98 kB 🟢 -1.76 kB
assets/UserSelectView-702nrljO.js (new) 4.73 kB 🔴 +4.73 kB 🔴 +1.76 kB 🔴 +1.55 kB
assets/UserSelectView-DmepKyIL.js (removed) 4.73 kB 🟢 -4.73 kB 🟢 -1.75 kB 🟢 -1.56 kB

Status: 9 added / 9 removed / 2 unchanged

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

Configuration panels, inspectors, and settings screens

File Before After Δ Raw Δ Gzip Δ Brotli
assets/KeybindingPanel-jmtQVtRP.js (removed) 46.7 kB 🟢 -46.7 kB 🟢 -9.6 kB 🟢 -8.53 kB
assets/KeybindingPanel-xdr2Wsw7.js (new) 46.7 kB 🔴 +46.7 kB 🔴 +9.6 kB 🔴 +8.52 kB
assets/SecretsPanel-BZLyOXTY.js (new) 22.9 kB 🔴 +22.9 kB 🔴 +5.54 kB 🔴 +4.87 kB
assets/SecretsPanel-XzXSmQyy.js (removed) 22.9 kB 🟢 -22.9 kB 🟢 -5.54 kB 🟢 -4.88 kB
assets/LegacyCreditsPanel-BGvGgSUB.js (removed) 21.7 kB 🟢 -21.7 kB 🟢 -5.89 kB 🟢 -5.19 kB
assets/LegacyCreditsPanel-Ctz9DbuT.js (new) 21.7 kB 🔴 +21.7 kB 🔴 +5.89 kB 🔴 +5.19 kB
assets/SubscriptionPanel-Ctbogbon.js (new) 19.9 kB 🔴 +19.9 kB 🔴 +5.06 kB 🔴 +4.46 kB
assets/SubscriptionPanel-DP-k36c4.js (removed) 19.9 kB 🟢 -19.9 kB 🟢 -5.06 kB 🟢 -4.45 kB
assets/AboutPanel-4fd8zLMY.js (new) 12 kB 🔴 +12 kB 🔴 +3.32 kB 🔴 +2.98 kB
assets/AboutPanel-BW0QUmQ4.js (removed) 12 kB 🟢 -12 kB 🟢 -3.33 kB 🟢 -2.98 kB
assets/ExtensionPanel-B5isT6lO.js (removed) 9.94 kB 🟢 -9.94 kB 🟢 -2.9 kB 🟢 -2.57 kB
assets/ExtensionPanel-g7daIIV7.js (new) 9.94 kB 🔴 +9.94 kB 🔴 +2.9 kB 🔴 +2.57 kB
assets/ServerConfigPanel-BxtR_LGN.js (new) 7.02 kB 🔴 +7.02 kB 🔴 +2.35 kB 🔴 +2.08 kB
assets/ServerConfigPanel-CPw57ZVK.js (removed) 7.02 kB 🟢 -7.02 kB 🟢 -2.35 kB 🟢 -2.1 kB
assets/UserPanel-DHJb0QB8.js (new) 6.73 kB 🔴 +6.73 kB 🔴 +2.23 kB 🔴 +1.96 kB
assets/UserPanel-DqW-oaJi.js (removed) 6.73 kB 🟢 -6.73 kB 🟢 -2.23 kB 🟢 -1.95 kB
assets/cloudRemoteConfig-Ck-uFs_Z.js (removed) 2.02 kB 🟢 -2.02 kB 🟢 -974 B 🟢 -834 B
assets/cloudRemoteConfig-DmmCacEr.js (new) 2.02 kB 🔴 +2.02 kB 🔴 +975 B 🔴 +837 B
assets/refreshRemoteConfig-CAUVCZRq.js (new) 1.45 kB 🔴 +1.45 kB 🔴 +648 B 🔴 +552 B
assets/refreshRemoteConfig-Dx1yJ66s.js (removed) 1.45 kB 🟢 -1.45 kB 🟢 -649 B 🟢 -555 B

Status: 10 added / 10 removed / 11 unchanged

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

Authentication, profile, and account management bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/auth-DGCtDBdr.js (new) 3.57 kB 🔴 +3.57 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/auth-dHFnwlHw.js (removed) 3.57 kB 🟢 -3.57 kB 🟢 -1.26 kB 🟢 -1.07 kB
assets/SignUpForm-BAdsKg4E.js (removed) 3.16 kB 🟢 -3.16 kB 🟢 -1.29 kB 🟢 -1.15 kB
assets/SignUpForm-Umkjs0PB.js (new) 3.16 kB 🔴 +3.16 kB 🔴 +1.29 kB 🔴 +1.15 kB
assets/UpdatePasswordContent-CHRV4JUR.js (new) 2.87 kB 🔴 +2.87 kB 🔴 +1.29 kB 🔴 +1.14 kB
assets/UpdatePasswordContent-u6memggz.js (removed) 2.87 kB 🟢 -2.87 kB 🟢 -1.29 kB 🟢 -1.14 kB
assets/authStore-BySyHFHH.js (new) 1.16 kB 🔴 +1.16 kB 🔴 +556 B 🔴 +499 B
assets/authStore-DLNgoBCR.js (removed) 1.16 kB 🟢 -1.16 kB 🟢 -555 B 🟢 -495 B
assets/auth-Bg9JObAP.js (new) 348 B 🔴 +348 B 🔴 +217 B 🔴 +193 B
assets/auth-Cg2PusUL.js (removed) 348 B 🟢 -348 B 🟢 -216 B 🟢 -189 B

Status: 5 added / 5 removed / 2 unchanged

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

Modals, dialogs, drawers, and in-app editors

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyHubPublishDialog-CcLoqaRJ.js (new) 86.9 kB 🔴 +86.9 kB 🔴 +18.8 kB 🔴 +16.1 kB
assets/ComfyHubPublishDialog-DpwIIN1q.js (removed) 86.9 kB 🟢 -86.9 kB 🟢 -18.8 kB 🟢 -16.1 kB
assets/useShareDialog-CJQSdiXA.js (removed) 23.7 kB 🟢 -23.7 kB 🟢 -5.76 kB 🟢 -5.13 kB
assets/useShareDialog-CSQ0fl6h.js (new) 23.7 kB 🔴 +23.7 kB 🔴 +5.77 kB 🔴 +5.11 kB
assets/ComfyHubPublishDialog-BwHN07Wd.js (removed) 1.32 kB 🟢 -1.32 kB 🟢 -616 B 🟢 -543 B
assets/ComfyHubPublishDialog-DC8IyMQZ.js (new) 1.32 kB 🔴 +1.32 kB 🔴 +616 B 🔴 +546 B
assets/useSubscriptionDialog-BkAzZZK6.js (new) 1.14 kB 🔴 +1.14 kB 🔴 +547 B 🔴 +482 B
assets/useSubscriptionDialog-DC7NDEux.js (removed) 1.14 kB 🟢 -1.14 kB 🟢 -545 B 🟢 -479 B

Status: 4 added / 4 removed

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

Reusable component library chunks

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyQueueButton-DesodOzO.js (new) 13.5 kB 🔴 +13.5 kB 🔴 +3.79 kB 🔴 +3.38 kB
assets/ComfyQueueButton-W-tJDLIM.js (removed) 13.5 kB 🟢 -13.5 kB 🟢 -3.79 kB 🟢 -3.38 kB
assets/useTerminalTabs-BGYLSrpR.js (new) 11 kB 🔴 +11 kB 🔴 +3.71 kB 🔴 +3.27 kB
assets/useTerminalTabs-D2P4v3wp.js (removed) 11 kB 🟢 -11 kB 🟢 -3.71 kB 🟢 -3.26 kB
assets/SubscribeButton-Baf7_Oc9.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -1.05 kB 🟢 -946 B
assets/SubscribeButton-BAydIX9T.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +1.05 kB 🔴 +938 B
assets/cloudFeedbackTopbarButton-CBUno-Lr.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +930 B 🔴 +818 B
assets/cloudFeedbackTopbarButton-DOZIfqBk.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -931 B 🟢 -816 B
assets/ComfyQueueButton-Czn-sTWx.js (new) 1.24 kB 🔴 +1.24 kB 🔴 +581 B 🔴 +516 B
assets/ComfyQueueButton-DpQGVZ0v.js (removed) 1.24 kB 🟢 -1.24 kB 🟢 -582 B 🟢 -515 B

Status: 5 added / 5 removed / 8 unchanged

Data & Services — 3.04 MB (baseline 3.04 MB) • 🔴 +289 B

Stores, services, APIs, and repositories

File Before After Δ Raw Δ Gzip Δ Brotli
assets/dialogService-B02vE4cj.js (removed) 1.97 MB 🟢 -1.97 MB 🟢 -454 kB 🟢 -344 kB
assets/dialogService-CSk8grre.js (new) 1.97 MB 🔴 +1.97 MB 🔴 +454 kB 🔴 +344 kB
assets/api-D2EXVTek.js (removed) 894 kB 🟢 -894 kB 🟢 -214 kB 🟢 -168 kB
assets/api-ztTjRc_g.js (new) 894 kB 🔴 +894 kB 🔴 +214 kB 🔴 +168 kB
assets/load3dService-I3mfS7F7.js (new) 114 kB 🔴 +114 kB 🔴 +24.9 kB 🔴 +21.1 kB
assets/load3dService-BtXannqB.js (removed) 114 kB 🟢 -114 kB 🟢 -24.8 kB 🟢 -21 kB
assets/workflowShareService-D5Nl6aVh.js (removed) 16.6 kB 🟢 -16.6 kB 🟢 -4.89 kB 🟢 -4.33 kB
assets/workflowShareService-Du353eO3.js (new) 16.6 kB 🔴 +16.6 kB 🔴 +4.89 kB 🔴 +4.33 kB
assets/keybindingService-BpCGgZzU.js (removed) 13.8 kB 🟢 -13.8 kB 🟢 -3.67 kB 🟢 -3.22 kB
assets/keybindingService-tm-Q0UFc.js (new) 13.8 kB 🔴 +13.8 kB 🔴 +3.67 kB 🔴 +3.22 kB
assets/releaseStore-CxLkuepY.js (removed) 8.12 kB 🟢 -8.12 kB 🟢 -2.28 kB 🟢 -2 kB
assets/releaseStore-ZU6VMsEw.js (new) 8.12 kB 🔴 +8.12 kB 🔴 +2.28 kB 🔴 +2 kB
assets/userStore-BkL2UL75.js (new) 2.24 kB 🔴 +2.24 kB 🔴 +870 B 🔴 +761 B
assets/userStore-CB6wkb-h.js (removed) 2.24 kB 🟢 -2.24 kB 🟢 -870 B 🟢 -762 B
assets/audioService-DbM0a4qo.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -876 B 🟢 -759 B
assets/audioService-STetBzw4.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +875 B 🔴 +762 B
assets/releaseStore-CY-1EG3L.js (removed) 1.16 kB 🟢 -1.16 kB 🟢 -548 B 🟢 -485 B
assets/releaseStore-m5Ktd2WW.js (new) 1.16 kB 🔴 +1.16 kB 🔴 +551 B 🔴 +491 B
assets/workflowDraftStore-Cam93wgZ.js (removed) 1.14 kB 🟢 -1.14 kB 🟢 -544 B 🟢 -480 B
assets/workflowDraftStore-Dc_IsTTP.js (new) 1.14 kB 🔴 +1.14 kB 🔴 +546 B 🔴 +483 B
assets/dialogService-DVvcb-Ew.js (removed) 1.13 kB 🟢 -1.13 kB 🟢 -537 B 🟢 -476 B
assets/dialogService-qUc23gqN.js (new) 1.13 kB 🔴 +1.13 kB 🔴 +538 B 🔴 +480 B
assets/settingStore-B5pP7XHy.js (removed) 1.12 kB 🟢 -1.12 kB 🟢 -540 B 🟢 -476 B
assets/settingStore-CXJGVmXC.js (new) 1.12 kB 🔴 +1.12 kB 🔴 +541 B 🔴 +478 B
assets/assetsStore-BVcmuoSI.js (new) 1.12 kB 🔴 +1.12 kB 🔴 +541 B 🔴 +481 B
assets/assetsStore-CVWRn6E8.js (removed) 1.12 kB 🟢 -1.12 kB 🟢 -540 B 🟢 -481 B

Status: 13 added / 13 removed / 4 unchanged

Utilities & Hooks — 364 kB (baseline 364 kB) • 🔴 +185 B

Helpers, composables, and utility bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useConflictDetection-CtRBuJ1d.js (removed) 233 kB 🟢 -233 kB 🟢 -51.8 kB 🟢 -42.2 kB
assets/useConflictDetection-Ii3MHy_H.js (new) 233 kB 🔴 +233 kB 🔴 +51.8 kB 🔴 +42.2 kB
assets/useLoad3d-Cd-Wq8oj.js (new) 22.4 kB 🔴 +22.4 kB 🔴 +5.14 kB 🔴 +4.54 kB
assets/useLoad3d-BOY8lcCT.js (removed) 22.3 kB 🟢 -22.3 kB 🟢 -5.09 kB 🟢 -4.5 kB
assets/useLoad3dViewer-D62Q3j5Y.js (new) 20.8 kB 🔴 +20.8 kB 🔴 +4.91 kB 🔴 +4.29 kB
assets/useLoad3dViewer-uL4bHv8M.js (removed) 20.8 kB 🟢 -20.8 kB 🟢 -4.91 kB 🟢 -4.29 kB
assets/useFeatureFlags-BDNPFCw3.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +1.79 kB 🔴 +1.52 kB
assets/useFeatureFlags-CO-Mxouh.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -1.79 kB 🟢 -1.53 kB
assets/useCopyToClipboard-CTRdF4e-.js (new) 5.29 kB 🔴 +5.29 kB 🔴 +1.86 kB 🔴 +1.57 kB
assets/useCopyToClipboard-k9ngG5fS.js (removed) 5.29 kB 🟢 -5.29 kB 🟢 -1.86 kB 🟢 -1.57 kB
assets/useWorkspaceUI-CSQtyNkG.js (removed) 3.34 kB 🟢 -3.34 kB 🟢 -982 B 🟢 -808 B
assets/useWorkspaceUI-JWHuOFsn.js (new) 3.34 kB 🔴 +3.34 kB 🔴 +980 B 🔴 +807 B
assets/subscriptionCheckoutUtil-Cru_dAcu.js (removed) 3.31 kB 🟢 -3.31 kB 🟢 -1.36 kB 🟢 -1.19 kB
assets/subscriptionCheckoutUtil-DQZE__aF.js (new) 3.31 kB 🔴 +3.31 kB 🔴 +1.36 kB 🔴 +1.19 kB
assets/assetPreviewUtil-14SjDLb3.js (new) 2.27 kB 🔴 +2.27 kB 🔴 +957 B 🔴 +833 B
assets/assetPreviewUtil-BdoXncYl.js (removed) 2.27 kB 🟢 -2.27 kB 🟢 -958 B 🟢 -835 B
assets/useUpstreamValue-C35NSacZ.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +805 B 🔴 +712 B
assets/useUpstreamValue-C3ZBzV91.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -807 B 🟢 -709 B
assets/useLoad3d-23VVpggL.js (removed) 1.3 kB 🟢 -1.3 kB 🟢 -605 B 🟢 -551 B
assets/useLoad3d-Da44aP2M.js (new) 1.3 kB 🔴 +1.3 kB 🔴 +608 B 🔴 +558 B
assets/useLoad3dViewer-Byq_maXY.js (new) 1.24 kB 🔴 +1.24 kB 🔴 +574 B 🔴 +516 B
assets/useLoad3dViewer-D_sQgsq4.js (removed) 1.24 kB 🟢 -1.24 kB 🟢 -573 B 🟢 -512 B
assets/useCurrentUser-BfPwbbSa.js (removed) 1.12 kB 🟢 -1.12 kB 🟢 -541 B 🟢 -477 B
assets/useCurrentUser-G4qTprVu.js (new) 1.12 kB 🔴 +1.12 kB 🔴 +541 B 🔴 +481 B
assets/useWorkspaceSwitch-CGCBi6mf.js (removed) 747 B 🟢 -747 B 🟢 -383 B 🟢 -330 B
assets/useWorkspaceSwitch-CSONuKHb.js (new) 747 B 🔴 +747 B 🔴 +382 B 🔴 +331 B

Status: 13 added / 13 removed / 18 unchanged

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

External libraries and shared vendor chunks

Status: 16 unchanged

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

Bundles that do not match a named category

File Before After Δ Raw Δ Gzip Δ Brotli
assets/core-ByrYhzHy.js (new) 76.6 kB 🔴 +76.6 kB 🔴 +19.8 kB 🔴 +16.9 kB
assets/core-yRVazdd8.js (removed) 76.6 kB 🟢 -76.6 kB 🟢 -19.8 kB 🟢 -17 kB
assets/groupNode-DDm7DRkC.js (new) 74.9 kB 🔴 +74.9 kB 🔴 +18.7 kB 🔴 +16.5 kB
assets/groupNode-WLG3aNiU.js (removed) 74.9 kB 🟢 -74.9 kB 🟢 -18.7 kB 🟢 -16.5 kB
assets/WidgetSelect-CquCTG9K.js (removed) 67.2 kB 🟢 -67.2 kB 🟢 -14.6 kB 🟢 -12.6 kB
assets/WidgetSelect-khLTcgH1.js (new) 67.2 kB 🔴 +67.2 kB 🔴 +14.6 kB 🔴 +12.6 kB
assets/SubscriptionRequiredDialogContentWorkspace-OUqJfFQM.js (removed) 49.3 kB 🟢 -49.3 kB 🟢 -9.41 kB 🟢 -8.04 kB
assets/SubscriptionRequiredDialogContentWorkspace-yAwqMWwU.js (new) 49.3 kB 🔴 +49.3 kB 🔴 +9.42 kB 🔴 +8.05 kB
assets/Load3DControls-CeYDEryZ.js (new) 46.1 kB 🔴 +46.1 kB 🔴 +7.51 kB 🔴 +6.54 kB
assets/Load3DControls-ik7wuk6Z.js (removed) 46.1 kB 🟢 -46.1 kB 🟢 -7.51 kB 🟢 -6.55 kB
assets/WidgetPainter-CX9O5XsO.js (removed) 34 kB 🟢 -34 kB 🟢 -8.28 kB 🟢 -7.35 kB
assets/WidgetPainter-KZS2ovIl.js (new) 34 kB 🔴 +34 kB 🔴 +8.28 kB 🔴 +7.37 kB
assets/WorkspacePanelContent-3B9OoFYN.js (removed) 32.8 kB 🟢 -32.8 kB 🟢 -6.99 kB 🟢 -6.2 kB
assets/WorkspacePanelContent-C8pT_QiU.js (new) 32.8 kB 🔴 +32.8 kB 🔴 +7 kB 🔴 +6.18 kB
assets/Load3dViewerContent-B3PtB8Fn.js (new) 28 kB 🔴 +28 kB 🔴 +5.85 kB 🔴 +5.07 kB
assets/Load3dViewerContent-BCUI3nbw.js (removed) 28 kB 🟢 -28 kB 🟢 -5.85 kB 🟢 -5.07 kB
assets/SubscriptionRequiredDialogContent-CyQ8AkJF.js (new) 27.5 kB 🔴 +27.5 kB 🔴 +6.96 kB 🔴 +6.16 kB
assets/SubscriptionRequiredDialogContent-DJ252VHL.js (removed) 27.5 kB 🟢 -27.5 kB 🟢 -6.96 kB 🟢 -6.15 kB
assets/WidgetImageCrop-auJRl4Ly.js (removed) 24.3 kB 🟢 -24.3 kB 🟢 -6.18 kB 🟢 -5.44 kB
assets/WidgetImageCrop-CHQchPZ7.js (new) 24.3 kB 🔴 +24.3 kB 🔴 +6.18 kB 🔴 +5.44 kB
assets/SubscriptionPanelContentWorkspace-76Zk87Tr.js (removed) 22.2 kB 🟢 -22.2 kB 🟢 -5.18 kB 🟢 -4.57 kB
assets/SubscriptionPanelContentWorkspace-BJ7ILw5U.js (new) 22.2 kB 🔴 +22.2 kB 🔴 +5.18 kB 🔴 +4.57 kB
assets/SignInContent-DH-dhsnN.js (removed) 20.6 kB 🟢 -20.6 kB 🟢 -5.35 kB 🟢 -4.67 kB
assets/SignInContent-OB8tb6V-.js (new) 20.6 kB 🔴 +20.6 kB 🔴 +5.35 kB 🔴 +4.68 kB
assets/CurrentUserPopoverWorkspace-CnwEdpr9.js (removed) 20.5 kB 🟢 -20.5 kB 🟢 -4.89 kB 🟢 -4.38 kB
assets/CurrentUserPopoverWorkspace-DR5UMMeu.js (new) 20.5 kB 🔴 +20.5 kB 🔴 +4.9 kB 🔴 +4.38 kB
assets/WidgetInputNumber-DVIqRumG.js (new) 19.1 kB 🔴 +19.1 kB 🔴 +4.84 kB 🔴 +4.29 kB
assets/WidgetInputNumber-DWiaVJgD.js (removed) 19.1 kB 🟢 -19.1 kB 🟢 -4.84 kB 🟢 -4.29 kB
assets/Load3D-DeUUP6WJ.js (new) 18.5 kB 🔴 +18.5 kB 🔴 +4.39 kB 🔴 +3.84 kB
assets/Load3D-eEfsOv9j.js (removed) 18.5 kB 🟢 -18.5 kB 🟢 -4.39 kB 🟢 -3.84 kB
assets/WidgetRecordAudio-DrhImzb-.js (new) 17.4 kB 🔴 +17.4 kB 🔴 +5 kB 🔴 +4.48 kB
assets/WidgetRecordAudio-dT8v_Hm0.js (removed) 17.4 kB 🟢 -17.4 kB 🟢 -5 kB 🟢 -4.47 kB
assets/WidgetRange-BYezXz1Y.js (removed) 17 kB 🟢 -17 kB 🟢 -4.59 kB 🟢 -4.1 kB
assets/WidgetRange-DVkVizgc.js (new) 17 kB 🔴 +17 kB 🔴 +4.59 kB 🔴 +4.12 kB
assets/load3d-CxfYpEjs.js (new) 15.7 kB 🔴 +15.7 kB 🔴 +4.56 kB 🔴 +3.95 kB
assets/load3d-D2piqf6P.js (removed) 15.7 kB 🟢 -15.7 kB 🟢 -4.56 kB 🟢 -3.94 kB
assets/WaveAudioPlayer-DEpeFbE9.js (removed) 13.4 kB 🟢 -13.4 kB 🟢 -3.69 kB 🟢 -3.23 kB
assets/WaveAudioPlayer-FgqHdAgH.js (new) 13.4 kB 🔴 +13.4 kB 🔴 +3.69 kB 🔴 +3.23 kB
assets/WidgetCurve-DgQZ6j6K.js (new) 12.1 kB 🔴 +12.1 kB 🔴 +3.92 kB 🔴 +3.55 kB
assets/WidgetCurve-DPvLsSSE.js (removed) 12.1 kB 🟢 -12.1 kB 🟢 -3.91 kB 🟢 -3.55 kB
assets/TeamWorkspacesDialogContent-D42FUVF4.js (new) 11.3 kB 🔴 +11.3 kB 🔴 +3.41 kB 🔴 +3.04 kB
assets/TeamWorkspacesDialogContent-DskZx_25.js (removed) 11.3 kB 🟢 -11.3 kB 🟢 -3.41 kB 🟢 -3.03 kB
assets/nodeTemplates-B5z6CA_E.js (new) 9.81 kB 🔴 +9.81 kB 🔴 +3.47 kB 🔴 +3.06 kB
assets/nodeTemplates-CrIZyg77.js (removed) 9.81 kB 🟢 -9.81 kB 🟢 -3.47 kB 🟢 -3.06 kB
assets/NightlySurveyController-DNgvc36D.js (new) 8.99 kB 🔴 +8.99 kB 🔴 +3.15 kB 🔴 +2.78 kB
assets/NightlySurveyController-Nczg4y41.js (removed) 8.99 kB 🟢 -8.99 kB 🟢 -3.15 kB 🟢 -2.77 kB
assets/InviteMemberDialogContent-Bv3cH4l6.js (new) 7.91 kB 🔴 +7.91 kB 🔴 +2.51 kB 🔴 +2.21 kB
assets/InviteMemberDialogContent-qszF30Kk.js (removed) 7.91 kB 🟢 -7.91 kB 🟢 -2.51 kB 🟢 -2.21 kB
assets/Load3DConfiguration-BlhCcDcI.js (removed) 7.82 kB 🟢 -7.82 kB 🟢 -2.31 kB 🟢 -2.02 kB
assets/Load3DConfiguration-CDDvaoQf.js (new) 7.82 kB 🔴 +7.82 kB 🔴 +2.31 kB 🔴 +2.02 kB
assets/onboardingCloudRoutes-D-zNnrNG.js (new) 6.58 kB 🔴 +6.58 kB 🔴 +2.05 kB 🔴 +1.76 kB
assets/onboardingCloudRoutes-Dsgi91gS.js (removed) 6.58 kB 🟢 -6.58 kB 🟢 -2.05 kB 🟢 -1.76 kB
assets/CreateWorkspaceDialogContent-CzsYyorc.js (new) 6.12 kB 🔴 +6.12 kB 🔴 +2.22 kB 🔴 +1.94 kB
assets/CreateWorkspaceDialogContent-T_SU_IOG.js (removed) 6.12 kB 🟢 -6.12 kB 🟢 -2.22 kB 🟢 -1.94 kB
assets/WidgetWithControl-DOEL_pmp.js (removed) 6.02 kB 🟢 -6.02 kB 🟢 -2.4 kB 🟢 -2.15 kB
assets/WidgetWithControl-Dw4IuZnn.js (new) 6.02 kB 🔴 +6.02 kB 🔴 +2.4 kB 🔴 +2.14 kB
assets/FreeTierDialogContent-Cv9ofQK8.js (new) 5.99 kB 🔴 +5.99 kB 🔴 +2.12 kB 🔴 +1.87 kB
assets/FreeTierDialogContent-CZycHeZ0.js (removed) 5.99 kB 🟢 -5.99 kB 🟢 -2.12 kB 🟢 -1.87 kB
assets/EditWorkspaceDialogContent-BIgFN8Ab.js (removed) 5.92 kB 🟢 -5.92 kB 🟢 -2.18 kB 🟢 -1.92 kB
assets/EditWorkspaceDialogContent-DmppvAJj.js (new) 5.92 kB 🔴 +5.92 kB 🔴 +2.19 kB 🔴 +1.92 kB
assets/WidgetTextarea-BSGkhM9_.js (removed) 5.73 kB 🟢 -5.73 kB 🟢 -2.26 kB 🟢 -1.99 kB
assets/WidgetTextarea-dPT3kaLR.js (new) 5.73 kB 🔴 +5.73 kB 🔴 +2.26 kB 🔴 +1.99 kB
assets/Preview3d-BRZzE4it.js (new) 5.7 kB 🔴 +5.7 kB 🔴 +1.91 kB 🔴 +1.67 kB
assets/Preview3d-WDBfz3FX.js (removed) 5.7 kB 🟢 -5.7 kB 🟢 -1.91 kB 🟢 -1.68 kB
assets/ValueControlPopover-D-ThhN6P.js (removed) 5.5 kB 🟢 -5.5 kB 🟢 -2 kB 🟢 -1.8 kB
assets/ValueControlPopover-D9aMCRkJ.js (new) 5.5 kB 🔴 +5.5 kB 🔴 +2 kB 🔴 +1.8 kB
assets/CancelSubscriptionDialogContent-B1RyrW56.js (removed) 5.46 kB 🟢 -5.46 kB 🟢 -2.04 kB 🟢 -1.79 kB
assets/CancelSubscriptionDialogContent-BeQ995OS.js (new) 5.46 kB 🔴 +5.46 kB 🔴 +2.05 kB 🔴 +1.79 kB
assets/DeleteWorkspaceDialogContent-BPIeHZpA.js (removed) 4.82 kB 🟢 -4.82 kB 🟢 -1.86 kB 🟢 -1.62 kB
assets/DeleteWorkspaceDialogContent-CjYwZ5E9.js (new) 4.82 kB 🔴 +4.82 kB 🔴 +1.86 kB 🔴 +1.62 kB
assets/LeaveWorkspaceDialogContent-BFTswo1p.js (removed) 4.65 kB 🟢 -4.65 kB 🟢 -1.81 kB 🟢 -1.57 kB
assets/LeaveWorkspaceDialogContent-DOdhaogp.js (new) 4.65 kB 🔴 +4.65 kB 🔴 +1.81 kB 🔴 +1.57 kB
assets/RemoveMemberDialogContent-BUcrYHvq.js (new) 4.63 kB 🔴 +4.63 kB 🔴 +1.76 kB 🔴 +1.54 kB
assets/RemoveMemberDialogContent-C7VR141X.js (removed) 4.63 kB 🟢 -4.63 kB 🟢 -1.76 kB 🟢 -1.54 kB
assets/RevokeInviteDialogContent-Cph_YLpO.js (removed) 4.54 kB 🟢 -4.54 kB 🟢 -1.77 kB 🟢 -1.55 kB
assets/RevokeInviteDialogContent-njTkdBpY.js (new) 4.54 kB 🔴 +4.54 kB 🔴 +1.77 kB 🔴 +1.56 kB
assets/tierBenefits-B8sY-jrX.js (new) 4.45 kB 🔴 +4.45 kB 🔴 +1.57 kB 🔴 +1.36 kB
assets/tierBenefits-BMg3wnaY.js (removed) 4.45 kB 🟢 -4.45 kB 🟢 -1.57 kB 🟢 -1.36 kB
assets/InviteMemberUpsellDialogContent-CdzOiZuo.js (removed) 4.44 kB 🟢 -4.44 kB 🟢 -1.64 kB 🟢 -1.43 kB
assets/InviteMemberUpsellDialogContent-D7POiPbh.js (new) 4.44 kB 🔴 +4.44 kB 🔴 +1.64 kB 🔴 +1.44 kB
assets/cloudSessionCookie-D_7sE1zr.js (removed) 4.28 kB 🟢 -4.28 kB 🟢 -1.56 kB 🟢 -1.36 kB
assets/cloudSessionCookie-JZY5KqR9.js (new) 4.28 kB 🔴 +4.28 kB 🔴 +1.56 kB 🔴 +1.36 kB
assets/Media3DTop-HXkX4ArI.js (removed) 4.01 kB 🟢 -4.01 kB 🟢 -1.69 kB 🟢 -1.5 kB
assets/Media3DTop-tEKIeloW.js (new) 4.01 kB 🔴 +4.01 kB 🔴 +1.7 kB 🔴 +1.5 kB
assets/saveMesh-6q_bLxki.js (new) 3.98 kB 🔴 +3.98 kB 🔴 +1.73 kB 🔴 +1.54 kB
assets/saveMesh-BB1wbJFw.js (removed) 3.98 kB 🟢 -3.98 kB 🟢 -1.73 kB 🟢 -1.52 kB
assets/GlobalToast--RimdOph.js (removed) 3.05 kB 🟢 -3.05 kB 🟢 -1.26 kB 🟢 -1.08 kB
assets/GlobalToast-8lUlDEix.js (new) 3.05 kB 🔴 +3.05 kB 🔴 +1.26 kB 🔴 +1.11 kB
assets/CloudRunButtonWrapper-B8WM8rWP.js (removed) 2.2 kB 🟢 -2.2 kB 🟢 -1.01 kB 🟢 -897 B
assets/CloudRunButtonWrapper-Bl9x_q9S.js (new) 2.2 kB 🔴 +2.2 kB 🔴 +1.01 kB 🔴 +897 B
assets/SubscribeToRun-DFglo6qz.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -983 B 🟢 -864 B
assets/SubscribeToRun-sqKcb09U.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +982 B 🔴 +887 B
assets/MediaAudioTop-C1SGvGDr.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -1.01 kB 🟢 -869 B
assets/MediaAudioTop-CyCDil0h.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +1 kB 🔴 +861 B
assets/cloudBadges-CWhTfiX8.js (new) 1.93 kB 🔴 +1.93 kB 🔴 +963 B 🔴 +838 B
assets/cloudBadges-D26R8K1q.js (removed) 1.93 kB 🟢 -1.93 kB 🟢 -963 B 🟢 -835 B
assets/cloudSubscription-CAuZ6IT3.js (removed) 1.85 kB 🟢 -1.85 kB 🟢 -884 B 🟢 -776 B
assets/cloudSubscription-D3MgRR2u.js (new) 1.85 kB 🔴 +1.85 kB 🔴 +886 B 🔴 +770 B
assets/graphHasMissingNodes-BL2Gc1wZ.js (removed) 1.84 kB 🟢 -1.84 kB 🟢 -862 B 🟢 -759 B
assets/graphHasMissingNodes-LMo2kvgg.js (new) 1.84 kB 🔴 +1.84 kB 🔴 +860 B 🔴 +757 B
assets/Load3D-Ch51dr86.js (removed) 1.55 kB 🟢 -1.55 kB 🟢 -698 B 🟢 -621 B
assets/Load3D-Db2ImLWr.js (new) 1.55 kB 🔴 +1.55 kB 🔴 +698 B 🔴 +622 B
assets/previousFullPath-B1_REdPa.js (removed) 1.53 kB 🟢 -1.53 kB 🟢 -693 B 🟢 -595 B
assets/previousFullPath-CSAcRniL.js (new) 1.53 kB 🔴 +1.53 kB 🔴 +692 B 🔴 +599 B
assets/nightlyBadges-Bbd6UDTQ.js (new) 1.46 kB 🔴 +1.46 kB 🔴 +732 B 🔴 +648 B
assets/nightlyBadges-BGED9NA7.js (removed) 1.46 kB 🟢 -1.46 kB 🟢 -729 B 🟢 -666 B
assets/Load3dViewerContent-BLRabA18.js (new) 1.43 kB 🔴 +1.43 kB 🔴 +650 B 🔴 +590 B
assets/Load3dViewerContent-BWOl05Yk.js (removed) 1.43 kB 🟢 -1.43 kB 🟢 -648 B 🟢 -577 B
assets/SubscriptionPanelContentWorkspace-BuV10Vsu.js (removed) 1.32 kB 🟢 -1.32 kB 🟢 -606 B 🟢 -524 B
assets/SubscriptionPanelContentWorkspace-utTT3baA.js (new) 1.32 kB 🔴 +1.32 kB 🔴 +606 B 🔴 +526 B
assets/WidgetLegacy-CF55QuNb.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -552 B 🟢 -486 B
assets/WidgetLegacy-Utrf8Iam.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +552 B 🔴 +488 B
assets/changeTracker-CTskWvtJ.js (new) 1.12 kB 🔴 +1.12 kB 🔴 +541 B 🔴 +474 B
assets/changeTracker-MCUGfD7w.js (removed) 1.12 kB 🟢 -1.12 kB 🟢 -540 B 🟢 -470 B

Status: 57 added / 57 removed / 78 unchanged

⚡ Performance Report

canvas-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 63.8 MB heap
canvas-mouse-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 49.9 MB heap
canvas-zoom-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.7 MB heap
dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 51.2 MB heap
large-graph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 62.6 MB heap
large-graph-pan: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 56.5 MB heap
large-graph-zoom: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 65.7 MB heap
minimap-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 60.8 MB heap
subgraph-dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 56.1 MB heap
subgraph-idle: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 65.1 MB heap
subgraph-mouse-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 49.3 MB heap
viewport-pan-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 69.6 MB heap
vue-large-graph-idle: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 166.5 MB heap
vue-large-graph-pan: · 56.3 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 158.9 MB heap
workflow-execution: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 50.4 MB heap

No regressions detected.

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 10ms 6ms -34% z=-5.0
canvas-idle: layout count 0 0 +0%
canvas-idle: style recalc count 10 10 +0% z=-2.1
canvas-idle: task duration 399ms 301ms -25% z=-3.0
canvas-idle: script duration 25ms 16ms -33% z=-4.0
canvas-idle: TBT 0ms 0ms +0%
canvas-idle: heap used 65.1 MB 63.8 MB -2%
canvas-idle: DOM nodes 20 18 -10% z=-3.6
canvas-idle: event listeners 4 6 +50% z=-1.2
canvas-mouse-sweep: avg frame time 17ms 17ms +0% z=-0.4
canvas-mouse-sweep: p95 frame time 17ms 17ms +1%
canvas-mouse-sweep: layout duration 3ms 3ms -18% z=-4.5
canvas-mouse-sweep: style recalc duration 40ms 27ms -32% z=-4.9
canvas-mouse-sweep: layout count 12 12 +0%
canvas-mouse-sweep: style recalc count 75 72 -4% z=-2.8
canvas-mouse-sweep: task duration 861ms 623ms -28% z=-4.2
canvas-mouse-sweep: script duration 138ms 96ms -31% z=-6.1
canvas-mouse-sweep: TBT 0ms 0ms +0%
canvas-mouse-sweep: heap used 45.1 MB 49.9 MB +11%
canvas-mouse-sweep: DOM nodes -260 -258 -1% z=-123.8
canvas-mouse-sweep: event listeners -131 -129 -2% z=-33.0
canvas-zoom-sweep: avg frame time 17ms 17ms +0% z=0.5
canvas-zoom-sweep: p95 frame time 17ms 17ms -1%
canvas-zoom-sweep: layout duration 1ms 1ms -21% z=-2.5
canvas-zoom-sweep: style recalc duration 17ms 14ms -19% z=-3.5
canvas-zoom-sweep: layout count 6 6 +0%
canvas-zoom-sweep: style recalc count 31 31 +0% z=-0.6
canvas-zoom-sweep: task duration 308ms 236ms -23% z=-3.9
canvas-zoom-sweep: script duration 25ms 19ms -21% z=-2.6
canvas-zoom-sweep: TBT 0ms 0ms +0%
canvas-zoom-sweep: heap used 68.6 MB 68.7 MB +0%
canvas-zoom-sweep: DOM nodes 78 79 +1% 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.6
dom-widget-clipping: p95 frame time 17ms 17ms +0%
dom-widget-clipping: layout duration 0ms 0ms +0%
dom-widget-clipping: style recalc duration 10ms 7ms -33% z=-4.1
dom-widget-clipping: layout count 0 0 +0%
dom-widget-clipping: style recalc count 12 12 +0% z=-2.2
dom-widget-clipping: task duration 346ms 271ms -22% z=-5.7
dom-widget-clipping: script duration 65ms 47ms -28% z=-6.4
dom-widget-clipping: TBT 0ms 0ms +0%
dom-widget-clipping: heap used 52.7 MB 51.2 MB -3%
dom-widget-clipping: DOM nodes 20 18 -10% z=-2.9
dom-widget-clipping: event listeners 0 2 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 9ms 7ms -22% z=-4.8
large-graph-idle: layout count 0 0 +0%
large-graph-idle: style recalc count 9 10 +11% z=-5.1
large-graph-idle: task duration 540ms 442ms -18% z=-1.8
large-graph-idle: script duration 102ms 67ms -34% z=-3.3
large-graph-idle: TBT 0ms 0ms +0%
large-graph-idle: heap used 61.3 MB 62.6 MB +2%
large-graph-idle: DOM nodes -263 -258 -2% z=-312.8
large-graph-idle: event listeners -127 -125 -2% z=-24.8
large-graph-pan: avg frame time 17ms 17ms -0% z=-0.8
large-graph-pan: p95 frame time 17ms 17ms +0%
large-graph-pan: layout duration 0ms 0ms +0%
large-graph-pan: style recalc duration 18ms 16ms -9% z=-1.4
large-graph-pan: layout count 0 0 +0%
large-graph-pan: style recalc count 69 68 -1% z=-2.4
large-graph-pan: task duration 1171ms 807ms -31% z=-6.4
large-graph-pan: script duration 457ms 280ms -39% z=-6.5
large-graph-pan: TBT 0ms 0ms +0%
large-graph-pan: heap used 62.5 MB 56.5 MB -10%
large-graph-pan: DOM nodes -264 -260 -2% z=-169.1
large-graph-pan: event listeners -125 -127 +2% 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 7ms -3%
large-graph-zoom: style recalc duration 18ms 17ms -8%
large-graph-zoom: layout count 60 60 +0%
large-graph-zoom: style recalc count 65 66 +2%
large-graph-zoom: task duration 1330ms 1049ms -21%
large-graph-zoom: script duration 488ms 382ms -22%
large-graph-zoom: TBT 0ms 0ms +0%
large-graph-zoom: heap used 66.0 MB 65.7 MB -1%
large-graph-zoom: DOM nodes -269 -262 -3%
large-graph-zoom: event listeners -125 -123 -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 9ms 6ms -34% z=-4.5
minimap-idle: layout count 0 0 +0%
minimap-idle: style recalc count 8 9 +13% z=-0.8
minimap-idle: task duration 566ms 420ms -26% z=-2.3
minimap-idle: script duration 99ms 63ms -37% z=-3.6
minimap-idle: TBT 0ms 0ms +0%
minimap-idle: heap used 62.4 MB 60.8 MB -3%
minimap-idle: DOM nodes -264 -260 -2% z=-204.1
minimap-idle: event listeners -127 -125 -2% z=-196.2
subgraph-dom-widget-clipping: avg frame time 17ms 17ms -0% z=-0.9
subgraph-dom-widget-clipping: p95 frame time 17ms 17ms +0%
subgraph-dom-widget-clipping: layout duration 0ms 0ms +0%
subgraph-dom-widget-clipping: style recalc duration 13ms 10ms -21% z=-2.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 376ms 291ms -23% z=-4.8
subgraph-dom-widget-clipping: script duration 127ms 98ms -23% z=-4.7
subgraph-dom-widget-clipping: TBT 0ms 0ms +0%
subgraph-dom-widget-clipping: heap used 51.9 MB 56.1 MB +8%
subgraph-dom-widget-clipping: DOM nodes 22 22 +0% z=-0.2
subgraph-dom-widget-clipping: event listeners 6 8 +33% 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 7ms -27% z=-4.5
subgraph-idle: layout count 0 0 +0%
subgraph-idle: style recalc count 10 11 +10% z=0.2
subgraph-idle: task duration 375ms 278ms -26% z=-2.9
subgraph-idle: script duration 20ms 15ms -25% z=-1.8
subgraph-idle: TBT 0ms 0ms +0%
subgraph-idle: heap used 65.2 MB 65.1 MB -0%
subgraph-idle: DOM nodes 20 22 +10% z=0.1
subgraph-idle: event listeners 4 6 +50% variance too high
subgraph-mouse-sweep: avg frame time 17ms 17ms -0% z=0.4
subgraph-mouse-sweep: p95 frame time 17ms 17ms -1%
subgraph-mouse-sweep: layout duration 5ms 4ms -20% z=-3.3
subgraph-mouse-sweep: style recalc duration 38ms 30ms -23% z=-3.9
subgraph-mouse-sweep: layout count 16 16 +0%
subgraph-mouse-sweep: style recalc count 76 76 +0% z=-2.1
subgraph-mouse-sweep: task duration 711ms 558ms -22% z=-3.0
subgraph-mouse-sweep: script duration 102ms 72ms -29% z=-4.3
subgraph-mouse-sweep: TBT 0ms 0ms +0%
subgraph-mouse-sweep: heap used 56.9 MB 49.3 MB -13%
subgraph-mouse-sweep: DOM nodes 62 62 +0% z=-2.2
subgraph-mouse-sweep: event listeners 4 4 +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 51ms 49ms -3%
viewport-pan-sweep: layout count 0 0 +0%
viewport-pan-sweep: style recalc count 249 250 +0%
viewport-pan-sweep: task duration 3919ms 2801ms -29%
viewport-pan-sweep: script duration 1336ms 944ms -29%
viewport-pan-sweep: TBT 0ms 0ms +0%
viewport-pan-sweep: heap used 67.3 MB 69.6 MB +3%
viewport-pan-sweep: DOM nodes -263 -256 -3%
viewport-pan-sweep: event listeners -111 -109 -2%
vue-large-graph-idle: avg frame time 17ms 17ms +0%
vue-large-graph-idle: p95 frame time 17ms 17ms +1%
vue-large-graph-idle: layout duration 0ms 0ms +0%
vue-large-graph-idle: style recalc duration 0ms 0ms +0%
vue-large-graph-idle: layout count 0 0 +0%
vue-large-graph-idle: style recalc count 0 0 +0%
vue-large-graph-idle: task duration 12804ms 10310ms -19%
vue-large-graph-idle: script duration 645ms 498ms -23%
vue-large-graph-idle: TBT 0ms 0ms +0%
vue-large-graph-idle: heap used 154.0 MB 166.5 MB +8%
vue-large-graph-idle: DOM nodes -9850 -9850 +0%
vue-large-graph-idle: event listeners -23973 -23963 -0%
vue-large-graph-pan: avg frame time 17ms 18ms +3%
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 16ms -8%
vue-large-graph-pan: layout count 0 0 +0%
vue-large-graph-pan: style recalc count 68 65 -4%
vue-large-graph-pan: task duration 14375ms 12510ms -13%
vue-large-graph-pan: script duration 883ms 737ms -17%
vue-large-graph-pan: TBT 0ms 0ms +0%
vue-large-graph-pan: heap used 157.8 MB 158.9 MB +1%
vue-large-graph-pan: DOM nodes -9850 -9850 +0%
vue-large-graph-pan: event listeners -23983 -23957 -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 1ms +2% z=-1.2
workflow-execution: style recalc duration 23ms 21ms -10% z=-1.5
workflow-execution: layout count 5 5 +0% z=0.1
workflow-execution: style recalc count 18 17 -6% z=-0.4
workflow-execution: task duration 140ms 112ms -20% z=-1.0
workflow-execution: script duration 29ms 23ms -21% z=-2.1
workflow-execution: TBT 0ms 0ms +0%
workflow-execution: heap used 51.1 MB 50.4 MB -2%
workflow-execution: DOM nodes 158 159 +1% z=-0.3
workflow-execution: event listeners 69 71 +3% 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-29T03:36:34.162Z",
  "gitSha": "9a196b02cca75ed9dc617d8a5abe6a8a6ee6614d",
  "branch": "fix/load3d-lod-zoom-resolution",
  "measurements": [
    {
      "name": "canvas-idle",
      "durationMs": 2044.6949999999902,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 5.916999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 296.89200000000005,
      "heapDeltaBytes": 904680,
      "heapUsedBytes": 66848148,
      "domNodes": 18,
      "jsHeapTotalBytes": 19496960,
      "scriptDurationMs": 12.732000000000001,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "canvas-idle",
      "durationMs": 2007.6510000000098,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 7.409999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 335.296,
      "heapDeltaBytes": -22889912,
      "heapUsedBytes": 43336416,
      "domNodes": -257,
      "jsHeapTotalBytes": 18837504,
      "scriptDurationMs": 19.249,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-idle",
      "durationMs": 1994.0110000000004,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 6.360000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 301.088,
      "heapDeltaBytes": 1671312,
      "heapUsedBytes": 67905756,
      "domNodes": 20,
      "jsHeapTotalBytes": 19234816,
      "scriptDurationMs": 16.462,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1694.0069999999992,
      "styleRecalcs": 72,
      "styleRecalcDurationMs": 25.077999999999996,
      "layouts": 12,
      "layoutDurationMs": 2.722,
      "taskDurationMs": 597.1099999999999,
      "heapDeltaBytes": 4397228,
      "heapUsedBytes": 52314096,
      "domNodes": -259,
      "jsHeapTotalBytes": 16379904,
      "scriptDurationMs": 93.25,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1730.517999999961,
      "styleRecalcs": 72,
      "styleRecalcDurationMs": 27.048,
      "layouts": 12,
      "layoutDurationMs": 2.723,
      "taskDurationMs": 622.81,
      "heapDeltaBytes": 10928944,
      "heapUsedBytes": 58623632,
      "domNodes": -257,
      "jsHeapTotalBytes": 15331328,
      "scriptDurationMs": 101.67800000000001,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66666666666665,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1707.8599999999824,
      "styleRecalcs": 74,
      "styleRecalcDurationMs": 31.777,
      "layouts": 12,
      "layoutDurationMs": 3.111,
      "taskDurationMs": 636.7120000000001,
      "heapDeltaBytes": -3503764,
      "heapUsedBytes": 44880888,
      "domNodes": -258,
      "jsHeapTotalBytes": 15593472,
      "scriptDurationMs": 96.123,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1713.4209999999825,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 13.460999999999999,
      "layouts": 6,
      "layoutDurationMs": 0.5320000000000001,
      "taskDurationMs": 236.02,
      "heapDeltaBytes": 301512,
      "heapUsedBytes": 47699320,
      "domNodes": 79,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 19.400000000000002,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666682,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1718.939999999975,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 13.820999999999998,
      "layouts": 6,
      "layoutDurationMs": 0.51,
      "taskDurationMs": 234.106,
      "heapDeltaBytes": 5648404,
      "heapUsedBytes": 72064212,
      "domNodes": 79,
      "jsHeapTotalBytes": 18972672,
      "scriptDurationMs": 14.838000000000001,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1719.6919999998954,
      "styleRecalcs": 32,
      "styleRecalcDurationMs": 14.490000000000002,
      "layouts": 6,
      "layoutDurationMs": 0.5790000000000001,
      "taskDurationMs": 248.78500000000003,
      "heapDeltaBytes": 5887628,
      "heapUsedBytes": 72096120,
      "domNodes": 80,
      "jsHeapTotalBytes": 18710528,
      "scriptDurationMs": 21.238,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.669999999999998,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 463.2780000000025,
      "styleRecalcs": 12,
      "styleRecalcDurationMs": 6.657999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 275.849,
      "heapDeltaBytes": -12315756,
      "heapUsedBytes": 53705592,
      "domNodes": 18,
      "jsHeapTotalBytes": 20283392,
      "scriptDurationMs": 47.023999999999994,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 446.9360000000506,
      "styleRecalcs": 12,
      "styleRecalcDurationMs": 6.671999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 268.512,
      "heapDeltaBytes": -12461512,
      "heapUsedBytes": 53678328,
      "domNodes": 19,
      "jsHeapTotalBytes": 20021248,
      "scriptDurationMs": 48.135,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 461.6159999999354,
      "styleRecalcs": 12,
      "styleRecalcDurationMs": 5.777000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 271.32,
      "heapDeltaBytes": -12684520,
      "heapUsedBytes": 52761284,
      "domNodes": 18,
      "jsHeapTotalBytes": 21331968,
      "scriptDurationMs": 44.948,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2015.9800000000132,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 7.195,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 422.76199999999994,
      "heapDeltaBytes": 15552844,
      "heapUsedBytes": 72167100,
      "domNodes": -252,
      "jsHeapTotalBytes": -229376,
      "scriptDurationMs": 67.368,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2023.631000000023,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 7.746000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 442.331,
      "heapDeltaBytes": 7655704,
      "heapUsedBytes": 65684320,
      "domNodes": -259,
      "jsHeapTotalBytes": -229376,
      "scriptDurationMs": 66.50099999999999,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2037.319000000025,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 7.002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 480.36500000000007,
      "heapDeltaBytes": 6560712,
      "heapUsedBytes": 62787900,
      "domNodes": -258,
      "jsHeapTotalBytes": -1081344,
      "scriptDurationMs": 83.903,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2071.1149999999634,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 15.878999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 807.345,
      "heapDeltaBytes": -1627328,
      "heapUsedBytes": 57251336,
      "domNodes": -260,
      "jsHeapTotalBytes": 6529024,
      "scriptDurationMs": 279.45799999999997,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2070.5259999999726,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 16.177000000000003,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 803.0509999999999,
      "heapDeltaBytes": 490368,
      "heapUsedBytes": 59509136,
      "domNodes": -260,
      "jsHeapTotalBytes": 1286144,
      "scriptDurationMs": 280.208,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2100.168999999937,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 16.365000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 837.646,
      "heapDeltaBytes": 413664,
      "heapUsedBytes": 59280772,
      "domNodes": -260,
      "jsHeapTotalBytes": 958464,
      "scriptDurationMs": 288.06000000000006,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3087.184000000036,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 16.82,
      "layouts": 60,
      "layoutDurationMs": 6.968,
      "taskDurationMs": 1048.729,
      "heapDeltaBytes": 8772716,
      "heapUsedBytes": 69167680,
      "domNodes": -264,
      "jsHeapTotalBytes": 4489216,
      "scriptDurationMs": 382.493,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3037.8299999999854,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 16.637,
      "layouts": 60,
      "layoutDurationMs": 6.739999999999999,
      "taskDurationMs": 1012.6409999999998,
      "heapDeltaBytes": 8512412,
      "heapUsedBytes": 68852596,
      "domNodes": -262,
      "jsHeapTotalBytes": 5537792,
      "scriptDurationMs": 373.159,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3071.364000000017,
      "styleRecalcs": 67,
      "styleRecalcDurationMs": 17.535,
      "layouts": 60,
      "layoutDurationMs": 6.976999999999999,
      "taskDurationMs": 1079.491,
      "heapDeltaBytes": -7768124,
      "heapUsedBytes": 54191160,
      "domNodes": -261,
      "jsHeapTotalBytes": 3497984,
      "scriptDurationMs": 406.81800000000004,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2011.1709999999903,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 5.558999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 419.57500000000005,
      "heapDeltaBytes": 16780324,
      "heapUsedBytes": 75855212,
      "domNodes": -251,
      "jsHeapTotalBytes": -32768,
      "scriptDurationMs": 62.46499999999999,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2025.3010000000131,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 5.861999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 408.586,
      "heapDeltaBytes": 4965976,
      "heapUsedBytes": 63781324,
      "domNodes": -260,
      "jsHeapTotalBytes": 557056,
      "scriptDurationMs": 62.682999999999986,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "minimap-idle",
      "durationMs": 1995.712000000026,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 5.935000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 430.51599999999996,
      "heapDeltaBytes": 4019928,
      "heapUsedBytes": 63051464,
      "domNodes": -260,
      "jsHeapTotalBytes": 229376,
      "scriptDurationMs": 66.82000000000001,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 455.0490000000309,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 9.850000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 282.10699999999997,
      "heapDeltaBytes": 9903108,
      "heapUsedBytes": 58986268,
      "domNodes": 22,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 98.449,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 448.3539999999948,
      "styleRecalcs": 49,
      "styleRecalcDurationMs": 10.033,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 291.37600000000003,
      "heapDeltaBytes": -11799260,
      "heapUsedBytes": 54495000,
      "domNodes": 23,
      "jsHeapTotalBytes": 19759104,
      "scriptDurationMs": 97.535,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 470.49600000002556,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 10.195,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 290.53200000000004,
      "heapDeltaBytes": 9551496,
      "heapUsedBytes": 58834388,
      "domNodes": 22,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 102.91300000000001,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-idle",
      "durationMs": 1990.9119999999803,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 6.9220000000000015,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 279.00199999999995,
      "heapDeltaBytes": 1219904,
      "heapUsedBytes": 67784528,
      "domNodes": 22,
      "jsHeapTotalBytes": 18710528,
      "scriptDurationMs": 15.35,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2001.7149999999901,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 7.364000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 268.95799999999997,
      "heapDeltaBytes": 22985832,
      "heapUsedBytes": 70891136,
      "domNodes": 22,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 15.698999999999998,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2003.67899999992,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 6.222,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 278.444,
      "heapDeltaBytes": 1982524,
      "heapUsedBytes": 68274456,
      "domNodes": 19,
      "jsHeapTotalBytes": 20709376,
      "scriptDurationMs": 11.88,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1673.64299999997,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 28.798,
      "layouts": 16,
      "layoutDurationMs": 3.645,
      "taskDurationMs": 512.3870000000001,
      "heapDeltaBytes": -7063000,
      "heapUsedBytes": 59331636,
      "domNodes": 62,
      "jsHeapTotalBytes": 19234816,
      "scriptDurationMs": 72.39500000000001,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66666666666665,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1690.1890000000321,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 29.614,
      "layouts": 16,
      "layoutDurationMs": 3.557,
      "taskDurationMs": 557.532,
      "heapDeltaBytes": -23391360,
      "heapUsedBytes": 43224348,
      "domNodes": 62,
      "jsHeapTotalBytes": 19361792,
      "scriptDurationMs": 72.13,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1675.9319999999889,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 31.249,
      "layouts": 16,
      "layoutDurationMs": 3.993,
      "taskDurationMs": 582.1129999999999,
      "heapDeltaBytes": -14973100,
      "heapUsedBytes": 51700656,
      "domNodes": -256,
      "jsHeapTotalBytes": 18575360,
      "scriptDurationMs": 75.712,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8096.8210000000345,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 49.434,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 2775.592,
      "heapDeltaBytes": 16246440,
      "heapUsedBytes": 72929964,
      "domNodes": -256,
      "jsHeapTotalBytes": 2334720,
      "scriptDurationMs": 943.683,
      "eventListeners": -109,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8100.706999999943,
      "styleRecalcs": 249,
      "styleRecalcDurationMs": 48.333,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 2800.8450000000003,
      "heapDeltaBytes": 10384532,
      "heapUsedBytes": 68323600,
      "domNodes": -261,
      "jsHeapTotalBytes": 6201344,
      "scriptDurationMs": 919.5400000000001,
      "eventListeners": -111,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.80000000000109
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8114.850000000047,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 49.543000000000006,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 2844.718,
      "heapDeltaBytes": 20900160,
      "heapUsedBytes": 78199268,
      "domNodes": -252,
      "jsHeapTotalBytes": 2007040,
      "scriptDurationMs": 954.7410000000001,
      "eventListeners": -109,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 10303.762000000006,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 10258.838,
      "heapDeltaBytes": -46803168,
      "heapUsedBytes": 174605540,
      "domNodes": -9850,
      "jsHeapTotalBytes": 28372992,
      "scriptDurationMs": 517.231,
      "eventListeners": -23963,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333358,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 10664.013999999952,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 10654.411,
      "heapDeltaBytes": -54155012,
      "heapUsedBytes": 174873284,
      "domNodes": -9850,
      "jsHeapTotalBytes": 27848704,
      "scriptDurationMs": 497.802,
      "eventListeners": -23963,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.80000000000291
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 10337.23299999997,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 10310.01,
      "heapDeltaBytes": -46469668,
      "heapUsedBytes": 174313180,
      "domNodes": -9858,
      "jsHeapTotalBytes": 25751552,
      "scriptDurationMs": 473.14199999999994,
      "eventListeners": -23963,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333358,
      "p95FrameDurationMs": 16.80000000000109
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 12130.983999999955,
      "styleRecalcs": 64,
      "styleRecalcDurationMs": 15.372999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12114.259,
      "heapDeltaBytes": -34524900,
      "heapUsedBytes": 180541404,
      "domNodes": -9850,
      "jsHeapTotalBytes": 25530368,
      "scriptDurationMs": 690.348,
      "eventListeners": -23961,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333338,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 12535.652000000027,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 16.410000000000007,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12510.493999999997,
      "heapDeltaBytes": -47580348,
      "heapUsedBytes": 165996820,
      "domNodes": -9854,
      "jsHeapTotalBytes": -10424320,
      "scriptDurationMs": 744.311,
      "eventListeners": -23957,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.776666666666642,
      "p95FrameDurationMs": 16.80000000000291
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 12606.820999999967,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 15.601000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12584.615,
      "heapDeltaBytes": -49859136,
      "heapUsedBytes": 166648280,
      "domNodes": -9850,
      "jsHeapTotalBytes": -10162176,
      "scriptDurationMs": 737.084,
      "eventListeners": -23955,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.776666666666642,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "workflow-execution",
      "durationMs": 442.3979999999119,
      "styleRecalcs": 17,
      "styleRecalcDurationMs": 20.984999999999996,
      "layouts": 5,
      "layoutDurationMs": 1.368,
      "taskDurationMs": 99.76700000000001,
      "heapDeltaBytes": 5272248,
      "heapUsedBytes": 54718280,
      "domNodes": 167,
      "jsHeapTotalBytes": 262144,
      "scriptDurationMs": 19.349999999999998,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999727
    },
    {
      "name": "workflow-execution",
      "durationMs": 459.41000000004806,
      "styleRecalcs": 16,
      "styleRecalcDurationMs": 19.312999999999995,
      "layouts": 5,
      "layoutDurationMs": 1.145,
      "taskDurationMs": 112.119,
      "heapDeltaBytes": -14823208,
      "heapUsedBytes": 52797492,
      "domNodes": 156,
      "jsHeapTotalBytes": 4030464,
      "scriptDurationMs": 23.482999999999997,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "workflow-execution",
      "durationMs": 437.4670000000833,
      "styleRecalcs": 20,
      "styleRecalcDurationMs": 21.223,
      "layouts": 6,
      "layoutDurationMs": 1.391,
      "taskDurationMs": 117.19399999999999,
      "heapDeltaBytes": -14958420,
      "heapUsedBytes": 52736276,
      "domNodes": 159,
      "jsHeapTotalBytes": 4030464,
      "scriptDurationMs": 22.808000000000003,
      "eventListeners": 71,
      "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/extensions/core/load3d/SceneManager.ts (1)

335-453: ⚠️ Potential issue | 🟠 Major

Ensure renderer state is always restored on capture failure.

captureScene() mutates renderer size/pixel ratio/clear state, but restoration only happens on the success path. If any step throws, the renderer can stay in a corrupted state for later renders/captures.

Suggested fix (move restoration to finally)
-  captureScene(
-    width: number,
-    height: number
-  ): Promise<{ scene: string; mask: string; normal: string }> {
-    return new Promise(async (resolve, reject) => {
-      try {
+  async captureScene(
+    width: number,
+    height: number
+  ): Promise<{ scene: string; mask: string; normal: string }> {
+    const originalSize = new THREE.Vector2()
+    this.renderer.getSize(originalSize)
+    const originalPixelRatio = this.renderer.getPixelRatio()
+    const originalClearColor = this.renderer.getClearColor(new THREE.Color())
+    const originalClearAlpha = this.renderer.getClearAlpha()
+    const originalOutputColorSpace = this.renderer.outputColorSpace
+    const originalMaterials = new Map<
+      THREE.Mesh,
+      THREE.Material | THREE.Material[]
+    >()
+    const gridVisible = this.gridHelper.visible
+    try {
         // ... existing capture logic ...
-        resolve({
+        return {
           scene: sceneData,
           mask: maskData,
           normal: normalData
-        })
-      } catch (error) {
-        reject(error)
+        }
+      } finally {
+        this.scene.traverse((child) => {
+          if (child instanceof THREE.Mesh) {
+            const originalMaterial = originalMaterials.get(child)
+            if (originalMaterial) child.material = originalMaterial
+          }
+        })
+        this.gridHelper.visible = gridVisible
+        this.renderer.setClearColor(originalClearColor, originalClearAlpha)
+        this.renderer.setPixelRatio(originalPixelRatio)
+        this.renderer.setSize(originalSize.x, originalSize.y)
+        this.renderer.outputColorSpace = originalOutputColorSpace
+        this.handleResize(originalSize.x, originalSize.y)
       }
-    })
   }
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/extensions/core/load3d/SceneManager.ts` around lines 335 - 453,
captureScene currently mutates renderer state (pixelRatio, size, clear color,
outputColorSpace), swaps mesh materials and gridHelper visibility but only
restores these on the success path; move all restoration logic (reset renderer
pixel ratio/size/outputColorSpace/clearColor/clearAlpha, restore
originalMaterials for meshes, restore gridHelper.visible and call
handleResize(originalSize.x, originalSize.y)) into a finally block so it always
runs on success or error, keeping the try for rendering steps and catch to
reject; ensure the finally runs before resolve/reject by either using
async/await (avoid mixing new Promise with async) or calling the restore code
inside a finally in the existing Promise wrapper; reference symbols:
captureScene, this.renderer, this.getActiveCamera(), this.backgroundTexture,
this.backgroundMesh, this.updateBackgroundSize(), originalMaterials map,
this.scene.traverse(), this.gridHelper, and this.handleResize().
🧹 Nitpick comments (1)
src/extensions/core/load3d/Load3d.test.ts (1)

387-417: Assert call order (setPixelRatio before setSize) to prevent regressions.

These tests validate the pixel-ratio value, but not the ordering invariant that this fix depends on.

Suggested test hardening
-      Object.assign(ctx.load3d, {
-        renderer: { domElement: canvas, setSize: vi.fn(), setPixelRatio },
+      const setSize = vi.fn()
+      Object.assign(ctx.load3d, {
+        renderer: { domElement: canvas, setSize, setPixelRatio },
         getZoomScaleCallback: () => 2.5,
         targetWidth: 0,
         targetHeight: 0,
         isViewerMode: false,
         cameraManager: { ...ctx.cameraManager, handleResize: vi.fn() },
         sceneManager: { ...ctx.sceneManager, handleResize: vi.fn() }
       })

       ctx.load3d.handleResize()

       expect(setPixelRatio).toHaveBeenCalledWith(2.5)
+      expect(setPixelRatio.mock.invocationCallOrder[0]).toBeLessThan(
+        setSize.mock.invocationCallOrder[0]
+      )

Also applies to: 419-449

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

In `@src/extensions/core/load3d/Load3d.test.ts` around lines 387 - 417, Update the
test for ctx.load3d.handleResize to assert call order: ensure
renderer.setPixelRatio is called with the value from getZoomScaleCallback before
renderer.setSize is invoked; locate the test that assigns ctx.load3d with
renderer: { domElement: canvas, setSize: vi.fn(), setPixelRatio } and
replace/add an assertion that verifies setPixelRatio was called prior to setSize
(i.e., check call order on setPixelRatio and setSize mocks) — apply the same
change to the similar test covering lines ~419-449.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Outside diff comments:
In `@src/extensions/core/load3d/SceneManager.ts`:
- Around line 335-453: captureScene currently mutates renderer state
(pixelRatio, size, clear color, outputColorSpace), swaps mesh materials and
gridHelper visibility but only restores these on the success path; move all
restoration logic (reset renderer pixel
ratio/size/outputColorSpace/clearColor/clearAlpha, restore originalMaterials for
meshes, restore gridHelper.visible and call handleResize(originalSize.x,
originalSize.y)) into a finally block so it always runs on success or error,
keeping the try for rendering steps and catch to reject; ensure the finally runs
before resolve/reject by either using async/await (avoid mixing new Promise with
async) or calling the restore code inside a finally in the existing Promise
wrapper; reference symbols: captureScene, this.renderer, this.getActiveCamera(),
this.backgroundTexture, this.backgroundMesh, this.updateBackgroundSize(),
originalMaterials map, this.scene.traverse(), this.gridHelper, and
this.handleResize().

---

Nitpick comments:
In `@src/extensions/core/load3d/Load3d.test.ts`:
- Around line 387-417: Update the test for ctx.load3d.handleResize to assert
call order: ensure renderer.setPixelRatio is called with the value from
getZoomScaleCallback before renderer.setSize is invoked; locate the test that
assigns ctx.load3d with renderer: { domElement: canvas, setSize: vi.fn(),
setPixelRatio } and replace/add an assertion that verifies setPixelRatio was
called prior to setSize (i.e., check call order on setPixelRatio and setSize
mocks) — apply the same change to the similar test covering lines ~419-449.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 154e7f87-65b8-4551-8947-8d39801d5b0e

📥 Commits

Reviewing files that changed from the base of the PR and between bb74ec9 and aa36082.

📒 Files selected for processing (6)
  • src/composables/useLoad3d.test.ts
  • src/composables/useLoad3d.ts
  • src/extensions/core/load3d/Load3d.test.ts
  • src/extensions/core/load3d/Load3d.ts
  • src/extensions/core/load3d/SceneManager.ts
  • src/extensions/core/load3d/interfaces.ts

@codecov
Copy link
Copy Markdown

codecov Bot commented Apr 29, 2026

Codecov Report

❌ Patch coverage is 40.00000% with 9 lines in your changes missing coverage. Please review.

Files with missing lines Patch % Lines
src/extensions/core/load3d/SceneManager.ts 0.00% 7 Missing ⚠️
src/composables/useLoad3d.ts 80.00% 1 Missing ⚠️
src/extensions/core/load3d/Load3d.ts 66.66% 1 Missing ⚠️
@@            Coverage Diff             @@
##             main   #11734      +/-   ##
==========================================
+ Coverage   51.90%   52.37%   +0.47%     
==========================================
  Files        1376     1376              
  Lines       70445    70460      +15     
  Branches    18716    19631     +915     
==========================================
+ Hits        36562    36902     +340     
+ Misses      33336    33011     -325     
  Partials      547      547              
Flag Coverage Δ
unit 52.37% <40.00%> (+0.47%) ⬆️

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

Files with missing lines Coverage Δ
src/composables/useLoad3d.ts 80.49% <80.00%> (-0.01%) ⬇️
src/extensions/core/load3d/Load3d.ts 34.08% <66.66%> (+1.12%) ⬆️
src/extensions/core/load3d/SceneManager.ts 0.00% <0.00%> (ø)

... and 23 files with indirect coverage changes

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

Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

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

🧹 Nitpick comments (1)
browser_tests/tests/load3d/load3dLod.spec.ts (1)

20-22: Remove redundant inline comments in the test body

These comments restate what the code already expresses and should be removed to match repo guidance.

♻️ Suggested cleanup
-      // Zoom in — ds.scale increases, triggering appScalePercentage to update,
-      // which fires the watch in useLoad3d that calls handleResize with the
-      // new zoom-based pixel ratio.
       await comfyPage.canvasOps.zoom(-120, 5)

-      // Physical pixel count must grow: canvas.width = clientWidth × ds.scale
       await expect
         .poll(() => load3d.canvas.evaluate((el: HTMLCanvasElement) => el.width))
         .toBeGreaterThan(initialWidth)

As per coding guidelines: "Avoid new usage of code comments; do not add or retain redundant comments."

Also applies to: 25-26

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

In `@browser_tests/tests/load3d/load3dLod.spec.ts` around lines 20 - 22, Remove
the redundant inline comments in the test body that merely restate the code
behavior (the blocks describing ds.scale increasing, appScalePercentage
updating, and the useLoad3d watch calling handleResize), and also remove the
similar comment pair later in the test; leave the code and assertions unchanged
(references to ds.scale, appScalePercentage, useLoad3d, and handleResize help
you locate the exact comment blocks to delete).
🤖 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/load3d/load3dLod.spec.ts`:
- Around line 20-22: Remove the redundant inline comments in the test body that
merely restate the code behavior (the blocks describing ds.scale increasing,
appScalePercentage updating, and the useLoad3d watch calling handleResize), and
also remove the similar comment pair later in the test; leave the code and
assertions unchanged (references to ds.scale, appScalePercentage, useLoad3d, and
handleResize help you locate the exact comment blocks to delete).

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 5148e3cd-cdaf-416a-9814-f7fa41ea2b37

📥 Commits

Reviewing files that changed from the base of the PR and between aa36082 and 8fdb3d6.

📒 Files selected for processing (1)
  • browser_tests/tests/load3d/load3dLod.spec.ts

Copy link
Copy Markdown
Collaborator

@dante01yoon dante01yoon left a comment

Choose a reason for hiding this comment

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

Thanks for the focused Load3D fix. The unit-level pieces look reasonable, and I verified the focused unit slices locally: pnpm exec vitest run src/composables/useLoad3d.test.ts src/extensions/core/load3d/Load3d.test.ts passes with 99 tests.

issue: I need to request changes because the PR’s own browser regression test is failing on the current head. The failing check is playwright-tests-chromium-sharded (3, 8): after zooming in, the Load3D canvas backing width remains 374 instead of increasing, which means the user-visible zoom-resolution path is not yet proven to work. codecov/patch is also red.

The separate title-editor test in that shard is reported flaky, but the Load3D LOD test itself is the hard failure.

await comfyPage.canvasOps.zoom(-120, 5)

// Physical pixel count must grow: canvas.width = clientWidth × ds.scale
await expect
Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

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

issue: This assertion is currently failing in CI on this PR head. The log shows initialWidth is 374, and after canvasOps.zoom(-120, 5) the polled canvas.width is still 374 through all retries. Since this test is the regression coverage for the feature, we should fix the actual zoom-to-resize path (or the test setup if it is not exercising the real zoom path) before merging.

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

Labels

size:M This PR changes 30-99 lines, ignoring generated files.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants