Skip to content

refactor(virtual-grid): migrate to @tanstack/vue-virtual (FE-535)#11885

Open
dante01yoon wants to merge 4 commits intomainfrom
jaewon/fe-535-virtual-grid-blank-on-scroll
Open

refactor(virtual-grid): migrate to @tanstack/vue-virtual (FE-535)#11885
dante01yoon wants to merge 4 commits intomainfrom
jaewon/fe-535-virtual-grid-blank-on-scroll

Conversation

@dante01yoon
Copy link
Copy Markdown
Collaborator

@dante01yoon dante01yoon commented May 4, 2026

Summary

Migrate VirtualGrid from a hand-rolled scroll/offset/clamp implementation to @tanstack/vue-virtual (already a project dependency, used by JobAssetsList). The FE-535 bug class is structurally eliminated rather than patched.

Fixes FE-535

Why migrate instead of clamp

Original PR (fix(virtual-grid): clamp offsetRows…) added maxOffsetRows = max(0, totalRows - viewRows) to prevent the empty-window symptom. That works, but the bug surfaces from a deeper class of issue: manual scrollY → offset translation drifts when the source of truth (DOM scroll position) changes outside our event-driven sampling window. Two confirmed triggers:

  1. PrimeVue Popover keeps the menu mounted on close → on reopen, useScroll's scrollY carries the previous deep value while items may have shrunk (WidgetSelectDropdown.vue:148-152 refreshes outputs on open).
  2. macOS rubberband over-scroll briefly moves scrollY past the natural max between throttled samples.

Adding clamps closes the two known holes; switching to a virtualizer that reads scrollOffset fresh from the DOM each computation removes the entire class — the browser has already auto-clamped scrollTop when content shrinks, and bound checks live inside the library.

What changes

  • src/components/common/VirtualGrid.vue — internals replaced with useVirtualizer (single row-virtualizer, count = ceil(items.length / cols)). Each virtual row is absolutely positioned via transform: translateY(start), with cols cells laid out by the consumer's gridStyle (CSS Grid) inside.
  • Public API unchanged: items, gridStyle, bufferRows, defaultItemHeight, defaultItemWidth, maxColumns props; #item slot with { item, index }; approach-end emit. bufferRows maps to tanstack overscan.
  • Drops scrollThrottle / resizeDebounce props (internal-only, no consumer passed them).
  • All 5 consumers (FormDropdownMenu, AssetsSidebarListView, AssetsSidebarGridView, AssetGrid, ManagerDialog) work without modification — verified by 613 passing tests across consumer suites.

Net diff: +110 / -236.

Test changes

VirtualGrid.test.ts:

  • Swap useScroll mock for a controllable useVirtualizer mock (same pattern as mockTanstackVirtualizer.ts used by JobAssetsList).
  • Drop the three FE-535 reproduction tests (they pinned the clamp implementation; that code path is gone). Replaced with one invariant test documenting the FE-535 motivation: "given items > 0, the user can see them — the grid never silently collapses".
  • Keep behavioral coverage: slot indices, maxColumns rendering, empty list, approach-end emission timing, last-page invariant, cols enforcement.

Review focus

  • Does the row-virtualizer model correctly preserve consumer gridStyle? Each visible row renders a sub-grid via the consumer-provided gridStyle plus the gridTemplateColumns override when maxColumns is finite. Vertical positioning is via transform; horizontal layout is unchanged from the previous implementation.
  • approach-end semantics: emits when last virtual row index ≥ rowCount - bufferRows - 1. Equivalent to the previous "≤ bufferRows rows remaining" trigger.

Video

before

before.mov

after

after.mov

Live verification of the migration in cloud env still pending (LoadImage form-dropdown widget repro).

Reproduce the dropdown-blank-on-scroll symptom at the VirtualGrid layer.
When scrollY drifts past the natural scroll maximum (items list shrinks
while popover scroll position is retained, or rubberband over-scroll
captured by useScroll), the naive offsetRows derivation collapses
state.start === state.end === items.length and renderedItems becomes
empty even though items > 0. Users then see a blank dropdown that only
recovers when another scroll event fires.

These two cases stay red on main and turn green once VirtualGrid clamps
offsetRows to the maximum that keeps the last page visible.
When scrollY drifts past the natural scroll maximum (items shrink while
the popover's scroll position is retained, or macOS rubberband
over-scroll briefly outruns the limit), the naive
`floor(scrollY / itemHeight)` makes state.start and state.end both
clamp to items.length and renderedItems becomes empty. Vue then renders
zero rows even though items > 0, and FormDropdownMenu does not show its
empty-state icon either since the items array itself is non-empty. The
user sees a blank dropdown until another scroll event re-syncs the
position, which can take a long pause if no further interaction
happens.

Clamp offsetRows to max(0, totalRows - viewRows) so the grid always
falls back to rendering the last page of items in this state. Once a
real scroll event fires, normal windowing resumes from a valid
position.

Fixes FE-535
@dante01yoon dante01yoon requested a review from a team May 4, 2026 05:12
@dosubot dosubot Bot added the size:S This PR changes 10-29 lines, ignoring generated files. label May 4, 2026
@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai Bot commented May 4, 2026

📝 Walkthrough

Walkthrough

Replaced custom scroll-based windowing in VirtualGrid with TanStack's useVirtualizer, updating props/emit behavior and rendering to virtual rows with absolute positioning. Tests were refactored to mock useVirtualizer via a new mockedVisibleEnd and include FE-535 cases validating end-of-list and approach-end behavior.

Changes

VirtualGrid virtualization migration

Layer / File(s) Summary
Props / Public Surface
src/components/common/VirtualGrid.vue
Removed scrollThrottle and resizeDebounce props; retained bufferRows, defaultItemHeight, defaultItemWidth, maxColumns. approach-end emit preserved.
Data / Derived Values
src/components/common/VirtualGrid.vue
Compute cols from container width and defaultItemWidth; compute rowCount from items.length and cols.
Core Implementation
src/components/common/VirtualGrid.vue
Introduce useVirtualizer with count = rowCount, estimateSize = defaultItemHeight, getScrollElement = container, and overscan = bufferRows. Replace spacer/slicing logic with virtualRows and virtualizer.getTotalSize() for container height.
Rendering / Template
src/components/common/VirtualGrid.vue
Render virtualRows absolutely positioned via rowStyle (translateY), compute slot index as virtualRow.index * cols + i, remove top/bottom spacer elements and prior renderedItems slicing.
End-of-list detection
src/components/common/VirtualGrid.vue
Add watch(virtualRows, ...) to emit approach-end when last virtual row index >= rowCount - bufferRows - 1. Removed previous scrollY/clamped-offset and whenever-based signaling.
Tests / Mocking
src/components/common/VirtualGrid.test.ts
Replace previous useScroll/mockedScrollY strategy by mocking @tanstack/vue-virtual's useVirtualizer driven by a new mockedVisibleEnd ref. Update existing tests to set mockedVisibleEnd; add FE-535 vitest cases: full-range rendering (mockedVisibleEnd = Infinity) and approach-end emission driven by virtual range.
Other
package.json
Manifest referenced in diffs (no API changes declared).

Estimated code review effort

🎯 4 (Complex) | ⏱️ ~45 minutes

Poem

I hopped among rows, counting each cell,
Swapped scroll for a virtualizer spell.
Now rows translate and render with grace,
No empty panels in that cozy place.
Happy hops — the end stays in sight! 🐇✨


Caution

Pre-merge checks failed

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

  • Ignore (reviewers only)

❌ Failed checks (1 error, 1 warning)

Check name Status Explanation Resolution
End-To-End Regression Coverage For Fixes ❌ Error PR uses bug-fix language but lacks end-to-end regression tests under browser_tests/ and provides no explanation for their absence. Add a Playwright regression test under browser_tests/ or provide a concrete explanation in the PR description of why E2E testing is not feasible for this fix.
Docstring Coverage ⚠️ Warning Docstring coverage is 0.00% which is insufficient. The required threshold is 80.00%. Write docstrings for the functions missing them to satisfy the coverage threshold.
✅ Passed checks (5 passed)
Check name Status Explanation
Title check ✅ Passed The title accurately describes the main change: migrating VirtualGrid to @tanstack/vue-virtual and references the FE-535 issue being addressed.
Description check ✅ Passed The description comprehensively covers all required template sections: Summary with context, detailed Changes explaining What and Dependencies, Review Focus with key design decisions, and issue reference (Fixes FE-535). Includes before/after videos and clear motivation for the migration approach.
Linked Issues check ✅ Passed Check skipped because no linked issues were found for this pull request.
Out of Scope Changes check ✅ Passed Check skipped because no linked issues were found for this pull request.
Adr Compliance For Entity/Litegraph Changes ✅ Passed Modified files are UI components in src/components/common/, not in scope (src/lib/litegraph/ or src/ecs/ or graph entity-related files).
✨ 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 jaewon/fe-535-virtual-grid-blank-on-scroll

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
Review rate limit: 7/8 reviews remaining, refill in 7 minutes and 30 seconds.

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

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 4, 2026

🎨 Storybook: ✅ Built — View Storybook

Details

⏰ Completed at: 05/04/2026, 10:52:37 AM UTC

Links

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 4, 2026

🎭 Playwright: ❌ 1463 passed, 1 failed · 1 flaky

❌ Failed Tests

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

@github-actions
Copy link
Copy Markdown

github-actions Bot commented May 4, 2026

📦 Bundle: 5.26 MB gzip 🟢 -399 B

Details

Summary

  • Raw size: 24.2 MB baseline 24.2 MB — 🟢 -1.1 kB
  • Gzip: 5.26 MB baseline 5.26 MB — 🟢 -399 B
  • Brotli: 4.07 MB baseline 4.07 MB — 🟢 -192 B
  • Bundles: 258 current • 258 baseline • 118 added / 118 removed

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

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

Main entry bundles and manifests

File Before After Δ Raw Δ Gzip Δ Brotli
assets/index-BMIxWvxX.js (new) 22.6 kB 🔴 +22.6 kB 🔴 +8 kB 🔴 +6.9 kB
assets/index-DnU2wpu3.js (removed) 22.6 kB 🟢 -22.6 kB 🟢 -8 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-B9MfIasJ.js (new) 1.24 MB 🔴 +1.24 MB 🔴 +265 kB 🔴 +199 kB
assets/GraphView-DQEchrsF.js (removed) 1.24 MB 🟢 -1.24 MB 🟢 -265 kB 🟢 -199 kB

Status: 1 added / 1 removed

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

Top-level views, pages, and routed surfaces

File Before After Δ Raw Δ Gzip Δ Brotli
assets/CloudSurveyView-B5B0DVat.js (new) 19.6 kB 🔴 +19.6 kB 🔴 +5.14 kB 🔴 +4.57 kB
assets/CloudSurveyView-CBu00oio.js (removed) 19.6 kB 🟢 -19.6 kB 🟢 -5.14 kB 🟢 -4.57 kB
assets/CloudLoginView--WKpLdkm.js (removed) 12.2 kB 🟢 -12.2 kB 🟢 -3.45 kB 🟢 -3.05 kB
assets/CloudLoginView-Dh8ZLGqW.js (new) 12.2 kB 🔴 +12.2 kB 🔴 +3.45 kB 🔴 +3.06 kB
assets/CloudSignupView-CrYUB7rl.js (removed) 9.97 kB 🟢 -9.97 kB 🟢 -2.95 kB 🟢 -2.6 kB
assets/CloudSignupView-DQCu-7Uw.js (new) 9.97 kB 🔴 +9.97 kB 🔴 +2.94 kB 🔴 +2.61 kB
assets/UserCheckView-C0Vqbw78.js (new) 9.07 kB 🔴 +9.07 kB 🔴 +2.33 kB 🔴 +2.05 kB
assets/UserCheckView-DMuWIRir.js (removed) 9.07 kB 🟢 -9.07 kB 🟢 -2.33 kB 🟢 -2.04 kB
assets/CloudLayoutView-BGXcMPpI.js (new) 7.73 kB 🔴 +7.73 kB 🔴 +2.45 kB 🔴 +2.15 kB
assets/CloudLayoutView-WWhAqj9t.js (removed) 7.73 kB 🟢 -7.73 kB 🟢 -2.46 kB 🟢 -2.15 kB
assets/CloudForgotPasswordView-D793OyMx.js (removed) 6.14 kB 🟢 -6.14 kB 🟢 -2.19 kB 🟢 -1.91 kB
assets/CloudForgotPasswordView-DrOUFjmY.js (new) 6.14 kB 🔴 +6.14 kB 🔴 +2.18 kB 🔴 +1.92 kB
assets/CloudAuthTimeoutView-DowCyEgH.js (new) 5.5 kB 🔴 +5.5 kB 🔴 +2.02 kB 🔴 +1.78 kB
assets/CloudAuthTimeoutView-DPfbTunS.js (removed) 5.5 kB 🟢 -5.5 kB 🟢 -2.02 kB 🟢 -1.79 kB
assets/CloudSubscriptionRedirectView-BwcfxoII.js (removed) 5.28 kB 🟢 -5.28 kB 🟢 -2 kB 🟢 -1.79 kB
assets/CloudSubscriptionRedirectView-D_S86b0I.js (new) 5.28 kB 🔴 +5.28 kB 🔴 +2 kB 🔴 +1.78 kB
assets/UserSelectView-BFXCJ0ra.js (removed) 4.73 kB 🟢 -4.73 kB 🟢 -1.76 kB 🟢 -1.55 kB
assets/UserSelectView-Efj7KFHI.js (new) 4.73 kB 🔴 +4.73 kB 🔴 +1.75 kB 🔴 +1.55 kB

Status: 9 added / 9 removed / 2 unchanged

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

Configuration panels, inspectors, and settings screens

File Before After Δ Raw Δ Gzip Δ Brotli
assets/KeybindingPanel-CfelqgdV.js (new) 46.7 kB 🔴 +46.7 kB 🔴 +9.62 kB 🔴 +8.54 kB
assets/KeybindingPanel-ChG02WZk.js (removed) 46.7 kB 🟢 -46.7 kB 🟢 -9.62 kB 🟢 -8.56 kB
assets/SecretsPanel-BoKQRaTb.js (removed) 22.9 kB 🟢 -22.9 kB 🟢 -5.55 kB 🟢 -4.87 kB
assets/SecretsPanel-DwR4f2kC.js (new) 22.9 kB 🔴 +22.9 kB 🔴 +5.55 kB 🔴 +4.89 kB
assets/LegacyCreditsPanel-CiQcvvGZ.js (removed) 21.7 kB 🟢 -21.7 kB 🟢 -5.91 kB 🟢 -5.22 kB
assets/LegacyCreditsPanel-CsckZMyE.js (new) 21.7 kB 🔴 +21.7 kB 🔴 +5.9 kB 🔴 +5.21 kB
assets/SubscriptionPanel-BEO_s2rU.js (removed) 19.9 kB 🟢 -19.9 kB 🟢 -5.09 kB 🟢 -4.46 kB
assets/SubscriptionPanel-BirvJGvP.js (new) 19.9 kB 🔴 +19.9 kB 🔴 +5.08 kB 🔴 +4.47 kB
assets/AboutPanel-DAR-_Zzn.js (new) 12 kB 🔴 +12 kB 🔴 +3.33 kB 🔴 +2.98 kB
assets/AboutPanel-DikWJ3ak.js (removed) 12 kB 🟢 -12 kB 🟢 -3.33 kB 🟢 -2.99 kB
assets/ExtensionPanel-CrnOuq8o.js (new) 9.97 kB 🔴 +9.97 kB 🔴 +2.91 kB 🔴 +2.58 kB
assets/ExtensionPanel-D6119gu7.js (removed) 9.97 kB 🟢 -9.97 kB 🟢 -2.91 kB 🟢 -2.59 kB
assets/ServerConfigPanel-Cop0ljDW.js (new) 7.05 kB 🔴 +7.05 kB 🔴 +2.36 kB 🔴 +2.11 kB
assets/ServerConfigPanel-kCGAET3H.js (removed) 7.05 kB 🟢 -7.05 kB 🟢 -2.37 kB 🟢 -2.11 kB
assets/UserPanel-B7g9yi07.js (new) 6.75 kB 🔴 +6.75 kB 🔴 +2.24 kB 🔴 +1.97 kB
assets/UserPanel-BBdFQ0zM.js (removed) 6.75 kB 🟢 -6.75 kB 🟢 -2.25 kB 🟢 -1.97 kB
assets/cloudRemoteConfig-3oR1G8oi.js (new) 2.05 kB 🔴 +2.05 kB 🔴 +989 B 🔴 +852 B
assets/cloudRemoteConfig-D3t5iJ4K.js (removed) 2.05 kB 🟢 -2.05 kB 🟢 -989 B 🟢 -848 B
assets/refreshRemoteConfig-Bb0wROBB.js (new) 1.45 kB 🔴 +1.45 kB 🔴 +645 B 🔴 +555 B
assets/refreshRemoteConfig-n4GGQnbx.js (removed) 1.45 kB 🟢 -1.45 kB 🟢 -648 B 🟢 -552 B

Status: 10 added / 10 removed / 11 unchanged

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

Authentication, profile, and account management bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/auth-BLkwS5iF.js (removed) 3.57 kB 🟢 -3.57 kB 🟢 -1.26 kB 🟢 -1.08 kB
assets/auth-BSjYgvRg.js (new) 3.57 kB 🔴 +3.57 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/SignUpForm-C2l0xX1b.js (new) 3.19 kB 🔴 +3.19 kB 🔴 +1.29 kB 🔴 +1.15 kB
assets/SignUpForm-CXqflNEb.js (removed) 3.19 kB 🟢 -3.19 kB 🟢 -1.29 kB 🟢 -1.15 kB
assets/UpdatePasswordContent-B85sHUW7.js (new) 2.9 kB 🔴 +2.9 kB 🔴 +1.3 kB 🔴 +1.15 kB
assets/UpdatePasswordContent-BYFTLbUy.js (removed) 2.9 kB 🟢 -2.9 kB 🟢 -1.3 kB 🟢 -1.15 kB
assets/authStore-BaBPKCK-.js (removed) 1.19 kB 🟢 -1.19 kB 🟢 -568 B 🟢 -505 B
assets/authStore-HfkCtXFL.js (new) 1.19 kB 🔴 +1.19 kB 🔴 +569 B 🔴 +508 B
assets/auth-B1ThXWZp.js (new) 348 B 🔴 +348 B 🔴 +218 B 🔴 +190 B
assets/auth-TddgQqHg.js (removed) 348 B 🟢 -348 B 🟢 -216 B 🟢 -188 B

Status: 5 added / 5 removed / 2 unchanged

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

Modals, dialogs, drawers, and in-app editors

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyHubPublishDialog-CxhK_MiT.js (removed) 85.8 kB 🟢 -85.8 kB 🟢 -18.6 kB 🟢 -15.9 kB
assets/ComfyHubPublishDialog-OwsPkKGA.js (new) 85.8 kB 🔴 +85.8 kB 🔴 +18.6 kB 🔴 +15.9 kB
assets/useShareDialog-CHdZ1MaM.js (removed) 23.8 kB 🟢 -23.8 kB 🟢 -5.78 kB 🟢 -5.13 kB
assets/useShareDialog-ZuxWc1tu.js (new) 23.8 kB 🔴 +23.8 kB 🔴 +5.78 kB 🔴 +5.13 kB
assets/ComfyHubPublishDialog-CT9MOFIQ.js (new) 1.35 kB 🔴 +1.35 kB 🔴 +628 B 🔴 +553 B
assets/ComfyHubPublishDialog-De4n05Ys.js (removed) 1.35 kB 🟢 -1.35 kB 🟢 -629 B 🟢 -554 B
assets/useSubscriptionDialog-C3y83My3.js (new) 1.17 kB 🔴 +1.17 kB 🔴 +560 B 🔴 +486 B
assets/useSubscriptionDialog-D0Cq5Ep2.js (removed) 1.17 kB 🟢 -1.17 kB 🟢 -560 B 🟢 -484 B

Status: 4 added / 4 removed

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

Reusable component library chunks

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyQueueButton-sdYZUKtI.js (new) 13.5 kB 🔴 +13.5 kB 🔴 +3.79 kB 🔴 +3.38 kB
assets/ComfyQueueButton-ZEFbKQ3o.js (removed) 13.5 kB 🟢 -13.5 kB 🟢 -3.79 kB 🟢 -3.38 kB
assets/useTerminalTabs-D0BI6Wjj.js (new) 11 kB 🔴 +11 kB 🔴 +3.73 kB 🔴 +3.28 kB
assets/useTerminalTabs-DqeG1zN_.js (removed) 11 kB 🟢 -11 kB 🟢 -3.73 kB 🟢 -3.29 kB
assets/SubscribeButton-C7J0UTMz.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +1.05 kB 🔴 +948 B
assets/SubscribeButton-De1ccFhv.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -1.05 kB 🟢 -936 B
assets/cloudFeedbackTopbarButton-CP0ISEgd.js (new) 1.83 kB 🔴 +1.83 kB 🔴 +945 B 🔴 +827 B
assets/cloudFeedbackTopbarButton-DgQgW03d.js (removed) 1.83 kB 🟢 -1.83 kB 🟢 -947 B 🟢 -830 B
assets/ComfyQueueButton-DmhcfNVg.js (new) 1.27 kB 🔴 +1.27 kB 🔴 +596 B 🔴 +528 B
assets/ComfyQueueButton-z6l_DW9C.js (removed) 1.27 kB 🟢 -1.27 kB 🟢 -597 B 🟢 -527 B

Status: 5 added / 5 removed / 9 unchanged

Data & Services — 3.05 MB (baseline 3.05 MB) • 🟢 -1.1 kB

Stores, services, APIs, and repositories

File Before After Δ Raw Δ Gzip Δ Brotli
assets/dialogService-DAwHmAWz.js (removed) 1.99 MB 🟢 -1.99 MB 🟢 -458 kB 🟢 -347 kB
assets/dialogService-BWbPDLhq.js (new) 1.99 MB 🔴 +1.99 MB 🔴 +457 kB 🔴 +346 kB
assets/api-C_noAr5i.js (new) 887 kB 🔴 +887 kB 🔴 +212 kB 🔴 +167 kB
assets/api-DPOly-Nk.js (removed) 887 kB 🟢 -887 kB 🟢 -212 kB 🟢 -167 kB
assets/load3dService-BrzNFck3.js (new) 115 kB 🔴 +115 kB 🔴 +25 kB 🔴 +21.3 kB
assets/load3dService-DtbjDCh9.js (removed) 115 kB 🟢 -115 kB 🟢 -25 kB 🟢 -21.3 kB
assets/workflowShareService-BbHMaYMf.js (removed) 16.6 kB 🟢 -16.6 kB 🟢 -4.89 kB 🟢 -4.34 kB
assets/workflowShareService-CsYou3MV.js (new) 16.6 kB 🔴 +16.6 kB 🔴 +4.89 kB 🔴 +4.33 kB
assets/keybindingService-BLAyOa_9.js (new) 13.8 kB 🔴 +13.8 kB 🔴 +3.67 kB 🔴 +3.22 kB
assets/keybindingService-C1hen6Wx.js (removed) 13.8 kB 🟢 -13.8 kB 🟢 -3.67 kB 🟢 -3.22 kB
assets/releaseStore-D1o4_Pwf.js (removed) 8.12 kB 🟢 -8.12 kB 🟢 -2.28 kB 🟢 -2 kB
assets/releaseStore-eQEuzEGV.js (new) 8.12 kB 🔴 +8.12 kB 🔴 +2.28 kB 🔴 +2 kB
assets/userStore-DEAIq4Ln.js (removed) 2.24 kB 🟢 -2.24 kB 🟢 -870 B 🟢 -759 B
assets/userStore-nb4eG9Px.js (new) 2.24 kB 🔴 +2.24 kB 🔴 +869 B 🔴 +761 B
assets/audioService-0FqyXq1A.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +878 B 🔴 +764 B
assets/audioService-D0yN6GcT.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -878 B 🟢 -764 B
assets/releaseStore-BEwnF-7V.js (new) 1.19 kB 🔴 +1.19 kB 🔴 +565 B 🔴 +502 B
assets/releaseStore-pOcHZ3up.js (removed) 1.19 kB 🟢 -1.19 kB 🟢 -563 B 🟢 -495 B
assets/workflowDraftStore-BxsGrJTU.js (removed) 1.17 kB 🟢 -1.17 kB 🟢 -559 B 🟢 -494 B
assets/workflowDraftStore-C1JmiG13.js (new) 1.17 kB 🔴 +1.17 kB 🔴 +559 B 🔴 +492 B
assets/dialogService-BcVM47GC.js (removed) 1.16 kB 🟢 -1.16 kB 🟢 -551 B 🟢 -488 B
assets/dialogService-CV4CrMWu.js (new) 1.16 kB 🔴 +1.16 kB 🔴 +551 B 🔴 +488 B
assets/settingStore-BvI5g_Eu.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -554 B 🟢 -485 B
assets/settingStore-Dnas2Zv0.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +554 B 🔴 +487 B
assets/assetsStore-2_qCeB7Q.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +555 B 🔴 +486 B
assets/assetsStore-CXG6dh-u.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -554 B 🟢 -486 B

Status: 13 added / 13 removed / 4 unchanged

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

Helpers, composables, and utility bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useConflictDetection-CoU0HxaF.js (new) 233 kB 🔴 +233 kB 🔴 +51.8 kB 🔴 +42.3 kB
assets/useConflictDetection-D70JTENY.js (removed) 233 kB 🟢 -233 kB 🟢 -51.8 kB 🟢 -42.2 kB
assets/useLoad3d-BJwlEo96.js (removed) 22.3 kB 🟢 -22.3 kB 🟢 -5.09 kB 🟢 -4.51 kB
assets/useLoad3d-CO06PpsH.js (new) 22.3 kB 🔴 +22.3 kB 🔴 +5.09 kB 🔴 +4.51 kB
assets/useLoad3dViewer-Co3ZY_lI.js (removed) 20.8 kB 🟢 -20.8 kB 🟢 -4.91 kB 🟢 -4.3 kB
assets/useLoad3dViewer-DHiRANeI.js (new) 20.8 kB 🔴 +20.8 kB 🔴 +4.91 kB 🔴 +4.31 kB
assets/useFeatureFlags-Dg-Zgt5d.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +1.79 kB 🔴 +1.52 kB
assets/useFeatureFlags-EksjwOWC.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -1.79 kB 🟢 -1.53 kB
assets/useCopyToClipboard-BZWSSh-c.js (removed) 5.29 kB 🟢 -5.29 kB 🟢 -1.86 kB 🟢 -1.57 kB
assets/useCopyToClipboard-CFUvKDBf.js (new) 5.29 kB 🔴 +5.29 kB 🔴 +1.86 kB 🔴 +1.57 kB
assets/useWorkspaceUI-BzGiMbk5.js (removed) 3.34 kB 🟢 -3.34 kB 🟢 -982 B 🟢 -811 B
assets/useWorkspaceUI-jPBAYwF6.js (new) 3.34 kB 🔴 +3.34 kB 🔴 +983 B 🔴 +811 B
assets/subscriptionCheckoutUtil-DzIFGzhc.js (removed) 3.31 kB 🟢 -3.31 kB 🟢 -1.36 kB 🟢 -1.19 kB
assets/subscriptionCheckoutUtil-kzCuhJOG.js (new) 3.31 kB 🔴 +3.31 kB 🔴 +1.36 kB 🔴 +1.19 kB
assets/assetPreviewUtil-BoujyNaf.js (new) 2.27 kB 🔴 +2.27 kB 🔴 +956 B 🔴 +832 B
assets/assetPreviewUtil-DJ2fA5-l.js (removed) 2.27 kB 🟢 -2.27 kB 🟢 -959 B 🟢 -835 B
assets/useUpstreamValue-B1YIrzgh.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -805 B 🟢 -711 B
assets/useUpstreamValue-BVBZUK8t.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +805 B 🔴 +714 B
assets/useLoad3d-B3w9SURh.js (removed) 1.33 kB 🟢 -1.33 kB 🟢 -619 B 🟢 -559 B
assets/useLoad3d-DcNSsikc.js (new) 1.33 kB 🔴 +1.33 kB 🔴 +620 B 🔴 +554 B
assets/useLoad3dViewer-BEso-Qgk.js (new) 1.27 kB 🔴 +1.27 kB 🔴 +587 B 🔴 +527 B
assets/useLoad3dViewer-BWWO-5R7.js (removed) 1.27 kB 🟢 -1.27 kB 🟢 -587 B 🟢 -525 B
assets/useCurrentUser-DWmr0N4m.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +555 B 🔴 +486 B
assets/useCurrentUser-r0tAGfGt.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -554 B 🟢 -485 B
assets/useWorkspaceSwitch-2Mp9GNbL.js (new) 747 B 🔴 +747 B 🔴 +386 B 🔴 +330 B
assets/useWorkspaceSwitch-CvGh3EVu.js (removed) 747 B 🟢 -747 B 🟢 -383 B 🟢 -332 B

Status: 13 added / 13 removed / 18 unchanged

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

External libraries and shared vendor chunks

Status: 16 unchanged

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

Bundles that do not match a named category

File Before After Δ Raw Δ Gzip Δ Brotli
assets/core-ChdaTG5z.js (new) 76.6 kB 🔴 +76.6 kB 🔴 +19.8 kB 🔴 +16.9 kB
assets/core-CHh805tI.js (removed) 76.6 kB 🟢 -76.6 kB 🟢 -19.8 kB 🟢 -16.9 kB
assets/groupNode-Cu19JNo4.js (new) 74.9 kB 🔴 +74.9 kB 🔴 +18.7 kB 🔴 +16.5 kB
assets/groupNode-fX92vpLu.js (removed) 74.9 kB 🟢 -74.9 kB 🟢 -18.7 kB 🟢 -16.5 kB
assets/WidgetSelect-Dn-KeIJz.js (removed) 67.2 kB 🟢 -67.2 kB 🟢 -14.6 kB 🟢 -12.7 kB
assets/WidgetSelect-nOfaK9vA.js (new) 67.2 kB 🔴 +67.2 kB 🔴 +14.6 kB 🔴 +12.6 kB
assets/SubscriptionRequiredDialogContentWorkspace-88YSD_uk.js (new) 48.8 kB 🔴 +48.8 kB 🔴 +9.52 kB 🔴 +8.22 kB
assets/SubscriptionRequiredDialogContentWorkspace-CMHyolGN.js (removed) 48.8 kB 🟢 -48.8 kB 🟢 -9.53 kB 🟢 -8.23 kB
assets/Load3DControls-CfrV9Ohj.js (new) 46.1 kB 🔴 +46.1 kB 🔴 +7.51 kB 🔴 +6.55 kB
assets/Load3DControls-CipL9U-X.js (removed) 46.1 kB 🟢 -46.1 kB 🟢 -7.51 kB 🟢 -6.54 kB
assets/WidgetPainter-CAJE2JVa.js (removed) 34 kB 🟢 -34 kB 🟢 -8.3 kB 🟢 -7.37 kB
assets/WidgetPainter-DM3kvp7m.js (new) 34 kB 🔴 +34 kB 🔴 +8.3 kB 🔴 +7.37 kB
assets/WorkspacePanelContent-C06gTyiz.js (new) 32.8 kB 🔴 +32.8 kB 🔴 +7.01 kB 🔴 +6.19 kB
assets/WorkspacePanelContent-DLfDZ6h_.js (removed) 32.8 kB 🟢 -32.8 kB 🟢 -7.01 kB 🟢 -6.19 kB
assets/Load3dViewerContent-CTXMb-ol.js (removed) 28 kB 🟢 -28 kB 🟢 -5.85 kB 🟢 -5.07 kB
assets/Load3dViewerContent-o-TF48mR.js (new) 28 kB 🔴 +28 kB 🔴 +5.85 kB 🔴 +5.07 kB
assets/SubscriptionRequiredDialogContent-BgW5Pw_c.js (new) 27.5 kB 🔴 +27.5 kB 🔴 +6.98 kB 🔴 +6.16 kB
assets/SubscriptionRequiredDialogContent-Df7gcUBO.js (removed) 27.5 kB 🟢 -27.5 kB 🟢 -6.98 kB 🟢 -6.17 kB
assets/WidgetImageCrop--RlrfzJb.js (removed) 24.3 kB 🟢 -24.3 kB 🟢 -6.2 kB 🟢 -5.46 kB
assets/WidgetImageCrop-Dcub4SU4.js (new) 24.3 kB 🔴 +24.3 kB 🔴 +6.2 kB 🔴 +5.46 kB
assets/SubscriptionPanelContentWorkspace-84XIc62L.js (new) 22.2 kB 🔴 +22.2 kB 🔴 +5.17 kB 🔴 +4.56 kB
assets/SubscriptionPanelContentWorkspace-C-y3ka1i.js (removed) 22.2 kB 🟢 -22.2 kB 🟢 -5.17 kB 🟢 -4.55 kB
assets/SignInContent-BiEbpWKx.js (new) 20.6 kB 🔴 +20.6 kB 🔴 +5.37 kB 🔴 +4.69 kB
assets/SignInContent-D_R-mega.js (removed) 20.6 kB 🟢 -20.6 kB 🟢 -5.38 kB 🟢 -4.71 kB
assets/CurrentUserPopoverWorkspace-BfHHqjPU.js (removed) 20.6 kB 🟢 -20.6 kB 🟢 -4.92 kB 🟢 -4.4 kB
assets/CurrentUserPopoverWorkspace-CW7ZnI6w.js (new) 20.6 kB 🔴 +20.6 kB 🔴 +4.91 kB 🔴 +4.4 kB
assets/WidgetInputNumber-CJtb2McF.js (new) 19.1 kB 🔴 +19.1 kB 🔴 +4.84 kB 🔴 +4.29 kB
assets/WidgetInputNumber-cPql4ybc.js (removed) 19.1 kB 🟢 -19.1 kB 🟢 -4.84 kB 🟢 -4.3 kB
assets/Load3D-CdiRySaY.js (removed) 18.5 kB 🟢 -18.5 kB 🟢 -4.39 kB 🟢 -3.83 kB
assets/Load3D-Q2M65-Al.js (new) 18.5 kB 🔴 +18.5 kB 🔴 +4.39 kB 🔴 +3.83 kB
assets/WidgetRecordAudio-Bl9JjERr.js (new) 17.4 kB 🔴 +17.4 kB 🔴 +5.02 kB 🔴 +4.49 kB
assets/WidgetRecordAudio-CNMi9ayM.js (removed) 17.4 kB 🟢 -17.4 kB 🟢 -5.02 kB 🟢 -4.48 kB
assets/WidgetRange-BbdFq8aF.js (new) 17.1 kB 🔴 +17.1 kB 🔴 +4.61 kB 🔴 +4.13 kB
assets/WidgetRange-prwRL6Qj.js (removed) 17.1 kB 🟢 -17.1 kB 🟢 -4.61 kB 🟢 -4.12 kB
assets/load3d-CKhrzBhg.js (new) 15.8 kB 🔴 +15.8 kB 🔴 +4.59 kB 🔴 +3.97 kB
assets/load3d-DU60tfer.js (removed) 15.8 kB 🟢 -15.8 kB 🟢 -4.59 kB 🟢 -3.97 kB
assets/WaveAudioPlayer-BKdOvQf5.js (removed) 13.4 kB 🟢 -13.4 kB 🟢 -3.69 kB 🟢 -3.23 kB
assets/WaveAudioPlayer-D2SarO3X.js (new) 13.4 kB 🔴 +13.4 kB 🔴 +3.69 kB 🔴 +3.23 kB
assets/WidgetCurve-DHT1QUBd.js (removed) 12.2 kB 🟢 -12.2 kB 🟢 -3.94 kB 🟢 -3.56 kB
assets/WidgetCurve-Dq2mhadv.js (new) 12.2 kB 🔴 +12.2 kB 🔴 +3.93 kB 🔴 +3.56 kB
assets/TeamWorkspacesDialogContent-2NzN0oLk.js (removed) 11.3 kB 🟢 -11.3 kB 🟢 -3.43 kB 🟢 -3.05 kB
assets/TeamWorkspacesDialogContent-DIbkXkCv.js (new) 11.3 kB 🔴 +11.3 kB 🔴 +3.42 kB 🔴 +3.05 kB
assets/nodeTemplates-D354Wnyn.js (removed) 9.84 kB 🟢 -9.84 kB 🟢 -3.48 kB 🟢 -3.06 kB
assets/nodeTemplates-l3KA-hxu.js (new) 9.84 kB 🔴 +9.84 kB 🔴 +3.48 kB 🔴 +3.08 kB
assets/NightlySurveyController-C7MYOhkq.js (removed) 9.02 kB 🟢 -9.02 kB 🟢 -3.17 kB 🟢 -2.79 kB
assets/NightlySurveyController-CBRgcT5L.js (new) 9.02 kB 🔴 +9.02 kB 🔴 +3.16 kB 🔴 +2.79 kB
assets/Load3DConfiguration-BNtgda6F.js (removed) 8.03 kB 🟢 -8.03 kB 🟢 -2.36 kB 🟢 -2.06 kB
assets/Load3DConfiguration-DkKHzQzH.js (new) 8.03 kB 🔴 +8.03 kB 🔴 +2.36 kB 🔴 +2.06 kB
assets/InviteMemberDialogContent-Bn95nDaC.js (removed) 7.94 kB 🟢 -7.94 kB 🟢 -2.53 kB 🟢 -2.21 kB
assets/InviteMemberDialogContent-WmG5rvU9.js (new) 7.94 kB 🔴 +7.94 kB 🔴 +2.53 kB 🔴 +2.21 kB
assets/onboardingCloudRoutes-BE8oeA7F.js (removed) 6.59 kB 🟢 -6.59 kB 🟢 -2.06 kB 🟢 -1.78 kB
assets/onboardingCloudRoutes-ipXXr0EE.js (new) 6.59 kB 🔴 +6.59 kB 🔴 +2.06 kB 🔴 +1.77 kB
assets/CreateWorkspaceDialogContent-DxkgmItf.js (removed) 6.15 kB 🟢 -6.15 kB 🟢 -2.24 kB 🟢 -1.94 kB
assets/CreateWorkspaceDialogContent-H0ShC6m6.js (new) 6.15 kB 🔴 +6.15 kB 🔴 +2.24 kB 🔴 +1.95 kB
assets/WidgetWithControl-CizZaj-L.js (new) 6.05 kB 🔴 +6.05 kB 🔴 +2.42 kB 🔴 +2.14 kB
assets/WidgetWithControl-CSVPcb46.js (removed) 6.05 kB 🟢 -6.05 kB 🟢 -2.42 kB 🟢 -2.15 kB
assets/FreeTierDialogContent-CohUR-Vw.js (new) 6.01 kB 🔴 +6.01 kB 🔴 +2.13 kB 🔴 +1.88 kB
assets/FreeTierDialogContent-mXTK0rUl.js (removed) 6.01 kB 🟢 -6.01 kB 🟢 -2.14 kB 🟢 -1.88 kB
assets/EditWorkspaceDialogContent-B_gQ-b8f.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -2.2 kB 🟢 -1.93 kB
assets/EditWorkspaceDialogContent-Cy1855xP.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +2.2 kB 🔴 +1.93 kB
assets/WidgetTextarea-J0I_8ivJ.js (removed) 5.76 kB 🟢 -5.76 kB 🟢 -2.27 kB 🟢 -2.02 kB
assets/WidgetTextarea-nGiV6N9u.js (new) 5.76 kB 🔴 +5.76 kB 🔴 +2.27 kB 🔴 +2.02 kB
assets/Preview3d-B3JfVtz9.js (new) 5.73 kB 🔴 +5.73 kB 🔴 +1.92 kB 🔴 +1.69 kB
assets/Preview3d-FIrc1VzH.js (removed) 5.73 kB 🟢 -5.73 kB 🟢 -1.93 kB 🟢 -1.68 kB
assets/ValueControlPopover-Bt3mTEyE.js (new) 5.53 kB 🔴 +5.53 kB 🔴 +2.02 kB 🔴 +1.81 kB
assets/ValueControlPopover-DQypFrHJ.js (removed) 5.53 kB 🟢 -5.53 kB 🟢 -2.02 kB 🟢 -1.79 kB
assets/CancelSubscriptionDialogContent-DcmIi9d7.js (removed) 5.49 kB 🟢 -5.49 kB 🟢 -2.06 kB 🟢 -1.8 kB
assets/CancelSubscriptionDialogContent-fRyhXzU2.js (new) 5.49 kB 🔴 +5.49 kB 🔴 +2.06 kB 🔴 +1.81 kB
assets/DeleteWorkspaceDialogContent-DeLhy7mk.js (removed) 4.85 kB 🟢 -4.85 kB 🟢 -1.88 kB 🟢 -1.63 kB
assets/DeleteWorkspaceDialogContent-S2APRzSe.js (new) 4.85 kB 🔴 +4.85 kB 🔴 +1.88 kB 🔴 +1.63 kB
assets/LeaveWorkspaceDialogContent-BNF71JS6.js (new) 4.68 kB 🔴 +4.68 kB 🔴 +1.82 kB 🔴 +1.58 kB
assets/LeaveWorkspaceDialogContent-eqeDDdDg.js (removed) 4.68 kB 🟢 -4.68 kB 🟢 -1.82 kB 🟢 -1.59 kB
assets/RemoveMemberDialogContent-CWZsBKYw.js (new) 4.66 kB 🔴 +4.66 kB 🔴 +1.78 kB 🔴 +1.55 kB
assets/RemoveMemberDialogContent-Dd5Dm378.js (removed) 4.66 kB 🟢 -4.66 kB 🟢 -1.78 kB 🟢 -1.55 kB
assets/RevokeInviteDialogContent-CATB0g9S.js (removed) 4.57 kB 🟢 -4.57 kB 🟢 -1.79 kB 🟢 -1.57 kB
assets/RevokeInviteDialogContent-DZ7q8MwI.js (new) 4.57 kB 🔴 +4.57 kB 🔴 +1.79 kB 🔴 +1.57 kB
assets/InviteMemberUpsellDialogContent-Cky_mK2Z.js (removed) 4.47 kB 🟢 -4.47 kB 🟢 -1.65 kB 🟢 -1.44 kB
assets/InviteMemberUpsellDialogContent-D-RhNvUv.js (new) 4.47 kB 🔴 +4.47 kB 🔴 +1.65 kB 🔴 +1.46 kB
assets/tierBenefits-CuD-YlaS.js (removed) 4.45 kB 🟢 -4.45 kB 🟢 -1.57 kB 🟢 -1.36 kB
assets/tierBenefits-I4irept7.js (new) 4.45 kB 🔴 +4.45 kB 🔴 +1.57 kB 🔴 +1.36 kB
assets/cloudSessionCookie-DORtVZY4.js (new) 4.31 kB 🔴 +4.31 kB 🔴 +1.58 kB 🔴 +1.37 kB
assets/cloudSessionCookie-lkrpLThU.js (removed) 4.31 kB 🟢 -4.31 kB 🟢 -1.58 kB 🟢 -1.38 kB
assets/Media3DTop-BFOrF-Re.js (removed) 4.04 kB 🟢 -4.04 kB 🟢 -1.71 kB 🟢 -1.51 kB
assets/Media3DTop-Dqbg61jp.js (new) 4.04 kB 🔴 +4.04 kB 🔴 +1.71 kB 🔴 +1.51 kB
assets/saveMesh-CdsYcgtH.js (removed) 4.03 kB 🟢 -4.03 kB 🟢 -1.76 kB 🟢 -1.55 kB
assets/saveMesh-iOClQSvw.js (new) 4.03 kB 🔴 +4.03 kB 🔴 +1.76 kB 🔴 +1.56 kB
assets/GlobalToast-DIVUnBPN.js (removed) 3.05 kB 🟢 -3.05 kB 🟢 -1.26 kB 🟢 -1.08 kB
assets/GlobalToast-DKwDQnEq.js (new) 3.05 kB 🔴 +3.05 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/CloudRunButtonWrapper-CkpruMpK.js (new) 2.23 kB 🔴 +2.23 kB 🔴 +1.02 kB 🔴 +905 B
assets/CloudRunButtonWrapper-DGVpuUpb.js (removed) 2.23 kB 🟢 -2.23 kB 🟢 -1.02 kB 🟢 -906 B
assets/SubscribeToRun-Dkc9_b2T.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +982 B 🔴 +878 B
assets/SubscribeToRun-DtCmdbZi.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -980 B 🟢 -864 B
assets/MediaAudioTop-bqM5O2_n.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +1 kB 🔴 +862 B
assets/MediaAudioTop-E8ykZ8ad.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -1.01 kB 🟢 -868 B
assets/cloudBadges-B8YMbDgn.js (removed) 1.96 kB 🟢 -1.96 kB 🟢 -979 B 🟢 -845 B
assets/cloudBadges-LwIr_t_1.js (new) 1.96 kB 🔴 +1.96 kB 🔴 +977 B 🔴 +845 B
assets/cloudSubscription-B2L8VD4p.js (removed) 1.88 kB 🟢 -1.88 kB 🟢 -901 B 🟢 -776 B
assets/cloudSubscription-fQU9WHuL.js (new) 1.88 kB 🔴 +1.88 kB 🔴 +899 B 🔴 +778 B
assets/graphHasMissingNodes-C2DUBs9i.js (removed) 1.83 kB 🟢 -1.83 kB 🟢 -859 B 🟢 -754 B
assets/graphHasMissingNodes-DJ9kg9q0.js (new) 1.83 kB 🔴 +1.83 kB 🔴 +860 B 🔴 +768 B
assets/Load3D-BCWoBnIk.js (new) 1.58 kB 🔴 +1.58 kB 🔴 +711 B 🔴 +637 B
assets/Load3D-UV30UtvI.js (removed) 1.58 kB 🟢 -1.58 kB 🟢 -712 B 🟢 -629 B
assets/previousFullPath-B5O2iRzV.js (removed) 1.53 kB 🟢 -1.53 kB 🟢 -695 B 🟢 -619 B
assets/previousFullPath-bosEQbbv.js (new) 1.53 kB 🔴 +1.53 kB 🔴 +693 B 🔴 +593 B
assets/nightlyBadges-Dv7npNfq.js (removed) 1.49 kB 🟢 -1.49 kB 🟢 -744 B 🟢 -666 B
assets/nightlyBadges-WJTYNaNy.js (new) 1.49 kB 🔴 +1.49 kB 🔴 +746 B 🔴 +683 B
assets/Load3dViewerContent-BiC2eKdG.js (new) 1.46 kB 🔴 +1.46 kB 🔴 +662 B 🔴 +590 B
assets/Load3dViewerContent-DsSBrd7w.js (removed) 1.46 kB 🟢 -1.46 kB 🟢 -663 B 🟢 -603 B
assets/SubscriptionPanelContentWorkspace-DCxXY6JF.js (removed) 1.35 kB 🟢 -1.35 kB 🟢 -618 B 🟢 -537 B
assets/SubscriptionPanelContentWorkspace-v-OLvNVw.js (new) 1.35 kB 🔴 +1.35 kB 🔴 +618 B 🔴 +540 B
assets/WidgetLegacy--w46VLa-.js (new) 1.18 kB 🔴 +1.18 kB 🔴 +566 B 🔴 +494 B
assets/WidgetLegacy-CbwXgg1-.js (removed) 1.18 kB 🟢 -1.18 kB 🟢 -566 B 🟢 -494 B
assets/changeTracker-a443QjPu.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +554 B 🔴 +484 B
assets/changeTracker-Cdp6jrcr.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -554 B 🟢 -484 B

Status: 57 added / 57 removed / 78 unchanged

⚡ Performance Report

canvas-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.4 MB heap
canvas-mouse-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 59.5 MB heap
canvas-zoom-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 46.3 MB heap
dom-widget-clipping: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 54.8 MB heap
large-graph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 59.8 MB heap
large-graph-pan: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 65.6 MB heap
large-graph-zoom: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 69.3 MB heap
minimap-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 65.4 MB heap
subgraph-dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 55.1 MB heap
subgraph-idle: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.5 MB heap
subgraph-mouse-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 59.8 MB heap
viewport-pan-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 65.8 MB heap
vue-large-graph-idle: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 163.7 MB heap
vue-large-graph-pan: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 51ms TBT · 147.2 MB heap
workflow-execution: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 52.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 -1%
canvas-idle: layout duration 0ms 0ms +0%
canvas-idle: style recalc duration 8ms 8ms +4% z=-3.0
canvas-idle: layout count 0 0 +0%
canvas-idle: style recalc count 8 8 +0% z=-5.6
canvas-idle: task duration 378ms 335ms -11% z=-1.9
canvas-idle: script duration 19ms 15ms -23% z=-4.7
canvas-idle: TBT 0ms 0ms +0%
canvas-idle: heap used 68.4 MB 68.4 MB +0%
canvas-idle: DOM nodes 16 16 +0% z=-5.2
canvas-idle: event listeners 4 4 +0% z=-1.6
canvas-mouse-sweep: avg frame time 17ms 17ms +0% z=-0.4
canvas-mouse-sweep: p95 frame time 17ms 17ms -1%
canvas-mouse-sweep: layout duration 4ms 4ms +5% z=1.5
canvas-mouse-sweep: style recalc duration 42ms 43ms +2% z=0.0
canvas-mouse-sweep: layout count 12 12 +0%
canvas-mouse-sweep: style recalc count 72 76 +6% z=-1.1
canvas-mouse-sweep: task duration 821ms 779ms -5% z=-1.5
canvas-mouse-sweep: script duration 128ms 122ms -5% z=-2.1
canvas-mouse-sweep: TBT 0ms 0ms +0%
canvas-mouse-sweep: heap used 46.5 MB 59.5 MB +28%
canvas-mouse-sweep: DOM nodes -265 58 -122% z=-1.7
canvas-mouse-sweep: event listeners -133 4 -103% z=-1.1
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 +5% z=1.3
canvas-zoom-sweep: style recalc duration 18ms 18ms -3% z=-1.1
canvas-zoom-sweep: layout count 6 6 +0%
canvas-zoom-sweep: style recalc count 32 31 -3% z=-0.6
canvas-zoom-sweep: task duration 310ms 296ms -4% z=-1.3
canvas-zoom-sweep: script duration 22ms 22ms -0% z=-1.9
canvas-zoom-sweep: TBT 0ms 0ms +0%
canvas-zoom-sweep: heap used 46.3 MB 46.3 MB +0%
canvas-zoom-sweep: DOM nodes 76 79 +4% z=-0.3
canvas-zoom-sweep: event listeners 19 19 +0% z=-0.9
dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
dom-widget-clipping: p95 frame time 17ms 17ms +1%
dom-widget-clipping: layout duration 0ms 0ms +0%
dom-widget-clipping: style recalc duration 7ms 7ms +0% z=-3.7
dom-widget-clipping: layout count 0 0 +0%
dom-widget-clipping: style recalc count 10 11 +10% z=-4.2
dom-widget-clipping: task duration 338ms 313ms -8% z=-3.1
dom-widget-clipping: script duration 59ms 55ms -7% z=-4.0
dom-widget-clipping: TBT 0ms 0ms +0%
dom-widget-clipping: heap used 54.8 MB 54.8 MB -0%
dom-widget-clipping: DOM nodes 16 17 +6% z=-3.6
dom-widget-clipping: event listeners 0 0 +0% variance too high
large-graph-idle: avg frame time 17ms 17ms +0% z=-0.2
large-graph-idle: p95 frame time 17ms 17ms +0%
large-graph-idle: layout duration 0ms 0ms +0%
large-graph-idle: style recalc duration 8ms 8ms -10% z=-4.4
large-graph-idle: layout count 0 0 +0%
large-graph-idle: style recalc count 9 9 +0% z=-8.3
large-graph-idle: task duration 583ms 518ms -11% z=-0.4
large-graph-idle: script duration 94ms 79ms -17% z=-2.3
large-graph-idle: TBT 0ms 0ms +0%
large-graph-idle: heap used 58.0 MB 59.8 MB +3%
large-graph-idle: DOM nodes -261 -262 +0% z=-317.2
large-graph-idle: event listeners -129 -129 +0% z=-25.4
large-graph-pan: avg frame time 17ms 17ms +0% z=0.3
large-graph-pan: p95 frame time 17ms 17ms -1%
large-graph-pan: layout duration 0ms 0ms +0%
large-graph-pan: style recalc duration 19ms 18ms -1% z=1.4
large-graph-pan: layout count 0 0 +0%
large-graph-pan: style recalc count 68 68 +0% z=-2.4
large-graph-pan: task duration 1094ms 1093ms -0% z=0.2
large-graph-pan: script duration 378ms 412ms +9% z=0.2
large-graph-pan: TBT 0ms 0ms +0%
large-graph-pan: heap used 61.7 MB 65.6 MB +6%
large-graph-pan: DOM nodes -265 -262 -1% z=-170.3
large-graph-pan: event listeners -129 -129 +0% z=-161.7
large-graph-zoom: avg frame time 17ms 17ms +0%
large-graph-zoom: p95 frame time 17ms 17ms -1%
large-graph-zoom: layout duration 8ms 8ms -2%
large-graph-zoom: style recalc duration 19ms 18ms -4%
large-graph-zoom: layout count 60 60 +0%
large-graph-zoom: style recalc count 65 65 +0%
large-graph-zoom: task duration 1340ms 1290ms -4%
large-graph-zoom: script duration 494ms 470ms -5%
large-graph-zoom: TBT 0ms 0ms +0%
large-graph-zoom: heap used 69.3 MB 69.3 MB +0%
large-graph-zoom: DOM nodes -270 -269 -0%
large-graph-zoom: event listeners -125 -125 +0%
minimap-idle: avg frame time 17ms 17ms +0% z=0.1
minimap-idle: p95 frame time 17ms 17ms +0%
minimap-idle: layout duration 0ms 0ms +0%
minimap-idle: style recalc duration 8ms 6ms -19% z=-4.2
minimap-idle: layout count 0 0 +0%
minimap-idle: style recalc count 8 7 -13% z=-3.8
minimap-idle: task duration 598ms 500ms -16% z=-0.6
minimap-idle: script duration 97ms 75ms -22% z=-2.3
minimap-idle: TBT 0ms 0ms +0%
minimap-idle: heap used 62.0 MB 65.4 MB +5%
minimap-idle: DOM nodes -265 -266 +0% z=-208.5
minimap-idle: event listeners -129 -129 +0% z=-202.3
subgraph-dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
subgraph-dom-widget-clipping: p95 frame time 17ms 17ms -1%
subgraph-dom-widget-clipping: layout duration 0ms 0ms +0%
subgraph-dom-widget-clipping: style recalc duration 11ms 10ms -5% z=-2.6
subgraph-dom-widget-clipping: layout count 0 0 +0%
subgraph-dom-widget-clipping: style recalc count 46 46 +0% z=-3.3
subgraph-dom-widget-clipping: task duration 367ms 349ms -5% z=-1.6
subgraph-dom-widget-clipping: script duration 130ms 125ms -4% z=-0.4
subgraph-dom-widget-clipping: TBT 0ms 0ms +0%
subgraph-dom-widget-clipping: heap used 54.8 MB 55.1 MB +0%
subgraph-dom-widget-clipping: DOM nodes 18 18 +0% z=-3.7
subgraph-dom-widget-clipping: event listeners 6 6 +0% z=-1.7
subgraph-idle: avg frame time 17ms 17ms -0% z=-0.7
subgraph-idle: p95 frame time 17ms 17ms +0%
subgraph-idle: layout duration 0ms 0ms +0%
subgraph-idle: style recalc duration 9ms 8ms -12% z=-3.2
subgraph-idle: layout count 0 0 +0%
subgraph-idle: style recalc count 9 9 +0% z=-2.9
subgraph-idle: task duration 362ms 336ms -7% z=-1.1
subgraph-idle: script duration 16ms 13ms -16% z=-2.6
subgraph-idle: TBT 0ms 0ms +0%
subgraph-idle: heap used 68.5 MB 68.5 MB +0%
subgraph-idle: DOM nodes 18 18 +0% z=-2.5
subgraph-idle: event listeners 4 4 +0% 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 4ms 5ms +16% z=1.2
subgraph-mouse-sweep: style recalc duration 38ms 40ms +7% z=-0.6
subgraph-mouse-sweep: layout count 16 16 +0%
subgraph-mouse-sweep: style recalc count 75 75 +0% z=-2.7
subgraph-mouse-sweep: task duration 723ms 683ms -6% z=-1.2
subgraph-mouse-sweep: script duration 95ms 95ms +1% z=-0.8
subgraph-mouse-sweep: TBT 0ms 0ms +0%
subgraph-mouse-sweep: heap used 48.2 MB 59.8 MB +24%
subgraph-mouse-sweep: DOM nodes -265 60 -123% z=-3.1
subgraph-mouse-sweep: event listeners -131 4 -103% variance too high
viewport-pan-sweep: avg frame time 17ms 17ms +0%
viewport-pan-sweep: p95 frame time 17ms 17ms -1%
viewport-pan-sweep: layout duration 0ms 0ms +0%
viewport-pan-sweep: style recalc duration 59ms 58ms -1%
viewport-pan-sweep: layout count 0 0 +0%
viewport-pan-sweep: style recalc count 250 250 +0%
viewport-pan-sweep: task duration 3880ms 3442ms -11%
viewport-pan-sweep: script duration 1329ms 1146ms -14%
viewport-pan-sweep: TBT 0ms 0ms +0%
viewport-pan-sweep: heap used 73.3 MB 65.8 MB -10%
viewport-pan-sweep: DOM nodes -264 -262 -1%
viewport-pan-sweep: event listeners -113 -113 +0%
vue-large-graph-idle: avg frame time 18ms 17ms -3%
vue-large-graph-idle: p95 frame time 17ms 17ms +0%
vue-large-graph-idle: layout duration 0ms 0ms +0%
vue-large-graph-idle: style recalc duration 0ms 0ms +0%
vue-large-graph-idle: layout count 0 0 +0%
vue-large-graph-idle: style recalc count 0 0 +0%
vue-large-graph-idle: task duration 13290ms 12937ms -3%
vue-large-graph-idle: script duration 650ms 595ms -8%
vue-large-graph-idle: TBT 0ms 0ms +0%
vue-large-graph-idle: heap used 162.7 MB 163.7 MB +1%
vue-large-graph-idle: DOM nodes -8331 -8331 +0%
vue-large-graph-idle: event listeners -16461 -16462 +0%
vue-large-graph-pan: avg frame time 17ms 17ms +0%
vue-large-graph-pan: p95 frame time 17ms 17ms +0%
vue-large-graph-pan: layout duration 0ms 0ms +0%
vue-large-graph-pan: style recalc duration 20ms 19ms -8%
vue-large-graph-pan: layout count 0 0 +0%
vue-large-graph-pan: style recalc count 77 76 -1%
vue-large-graph-pan: task duration 15451ms 15430ms -0%
vue-large-graph-pan: script duration 890ms 873ms -2%
vue-large-graph-pan: TBT 0ms 51ms
vue-large-graph-pan: heap used 166.2 MB 147.2 MB -11%
vue-large-graph-pan: DOM nodes -8331 -8329 -0%
vue-large-graph-pan: event listeners -16488 -16490 +0%
workflow-execution: avg frame time 17ms 17ms -0% z=-0.4
workflow-execution: p95 frame time 17ms 17ms +1%
workflow-execution: layout duration 2ms 2ms -2% z=0.8
workflow-execution: style recalc duration 24ms 26ms +8% z=0.6
workflow-execution: layout count 5 5 +0% z=0.1
workflow-execution: style recalc count 18 18 +0% z=0.1
workflow-execution: task duration 126ms 126ms +0% z=0.3
workflow-execution: script duration 25ms 25ms +2% z=-1.3
workflow-execution: TBT 0ms 0ms +0%
workflow-execution: heap used 52.4 MB 52.4 MB +0%
workflow-execution: DOM nodes 166 157 -5% z=-0.5
workflow-execution: event listeners 69 69 +0% z=3.9
Historical variance (last 15 runs)
Metric μ σ CV
canvas-idle: avg frame time 17ms 0ms 0.0%
canvas-idle: layout duration 0ms 0ms 0.0%
canvas-idle: style recalc duration 11ms 1ms 8.2%
canvas-idle: layout count 0 0 0.0%
canvas-idle: style recalc count 11 1 5.0%
canvas-idle: task duration 395ms 31ms 7.9%
canvas-idle: script duration 25ms 2ms 8.8%
canvas-idle: TBT 0ms 0ms 0.0%
canvas-idle: DOM nodes 23 1 5.6%
canvas-idle: event listeners 12 5 40.9%
canvas-mouse-sweep: avg frame time 17ms 0ms 0.0%
canvas-mouse-sweep: layout duration 4ms 0ms 5.4%
canvas-mouse-sweep: style recalc duration 43ms 3ms 7.4%
canvas-mouse-sweep: layout count 12 0 0.0%
canvas-mouse-sweep: style recalc count 79 2 3.0%
canvas-mouse-sweep: task duration 865ms 58ms 6.7%
canvas-mouse-sweep: script duration 136ms 6ms 4.8%
canvas-mouse-sweep: TBT 0ms 0ms 0.0%
canvas-mouse-sweep: DOM nodes 62 3 4.2%
canvas-mouse-sweep: event listeners 8 4 49.4%
canvas-zoom-sweep: avg frame time 17ms 0ms 0.0%
canvas-zoom-sweep: layout duration 1ms 0ms 7.0%
canvas-zoom-sweep: style recalc duration 19ms 2ms 8.0%
canvas-zoom-sweep: layout count 6 0 0.0%
canvas-zoom-sweep: style recalc count 31 0 1.5%
canvas-zoom-sweep: task duration 327ms 23ms 7.1%
canvas-zoom-sweep: script duration 27ms 3ms 11.1%
canvas-zoom-sweep: TBT 0ms 0ms 0.0%
canvas-zoom-sweep: DOM nodes 79 1 1.0%
canvas-zoom-sweep: event listeners 24 5 21.8%
dom-widget-clipping: avg frame time 17ms 0ms 0.0%
dom-widget-clipping: layout duration 0ms 0ms 0.0%
dom-widget-clipping: style recalc duration 10ms 1ms 8.0%
dom-widget-clipping: layout count 0 0 0.0%
dom-widget-clipping: style recalc count 13 0 3.8%
dom-widget-clipping: task duration 365ms 16ms 4.5%
dom-widget-clipping: script duration 68ms 3ms 4.8%
dom-widget-clipping: TBT 0ms 0ms 0.0%
dom-widget-clipping: DOM nodes 22 1 6.4%
dom-widget-clipping: event listeners 8 6 81.2%
large-graph-idle: avg frame time 17ms 0ms 0.0%
large-graph-idle: layout duration 0ms 0ms 0.0%
large-graph-idle: style recalc duration 12ms 1ms 8.6%
large-graph-idle: layout count 0 0 0.0%
large-graph-idle: style recalc count 12 0 2.7%
large-graph-idle: task duration 542ms 54ms 10.0%
large-graph-idle: script duration 102ms 11ms 10.3%
large-graph-idle: TBT 0ms 0ms 0.0%
large-graph-idle: DOM nodes 25 1 3.7%
large-graph-idle: event listeners 26 6 23.2%
large-graph-pan: avg frame time 17ms 0ms 0.0%
large-graph-pan: layout duration 0ms 0ms 0.0%
large-graph-pan: style recalc duration 17ms 1ms 4.6%
large-graph-pan: layout count 0 0 0.0%
large-graph-pan: style recalc count 70 1 0.9%
large-graph-pan: task duration 1082ms 43ms 4.0%
large-graph-pan: script duration 408ms 20ms 4.8%
large-graph-pan: TBT 0ms 0ms 0.0%
large-graph-pan: DOM nodes 19 2 8.7%
large-graph-pan: event listeners 5 1 16.8%
minimap-idle: avg frame time 17ms 0ms 0.0%
minimap-idle: layout duration 0ms 0ms 0.0%
minimap-idle: style recalc duration 10ms 1ms 8.6%
minimap-idle: layout count 0 0 0.0%
minimap-idle: style recalc count 10 1 7.1%
minimap-idle: task duration 527ms 47ms 9.0%
minimap-idle: script duration 98ms 10ms 10.1%
minimap-idle: TBT 0ms 0ms 0.0%
minimap-idle: DOM nodes 19 1 7.1%
minimap-idle: event listeners 5 1 14.4%
subgraph-dom-widget-clipping: avg frame time 17ms 0ms 0.0%
subgraph-dom-widget-clipping: layout duration 0ms 0ms 0.0%
subgraph-dom-widget-clipping: style recalc duration 13ms 1ms 7.4%
subgraph-dom-widget-clipping: layout count 0 0 0.0%
subgraph-dom-widget-clipping: style recalc count 48 1 1.2%
subgraph-dom-widget-clipping: task duration 378ms 18ms 4.9%
subgraph-dom-widget-clipping: script duration 128ms 6ms 4.9%
subgraph-dom-widget-clipping: TBT 0ms 0ms 0.0%
subgraph-dom-widget-clipping: DOM nodes 22 1 5.0%
subgraph-dom-widget-clipping: event listeners 16 6 36.0%
subgraph-idle: avg frame time 17ms 0ms 0.0%
subgraph-idle: layout duration 0ms 0ms 0.0%
subgraph-idle: style recalc duration 10ms 1ms 7.5%
subgraph-idle: layout count 0 0 0.0%
subgraph-idle: style recalc count 11 1 6.0%
subgraph-idle: task duration 370ms 31ms 8.5%
subgraph-idle: script duration 20ms 3ms 13.2%
subgraph-idle: TBT 0ms 0ms 0.0%
subgraph-idle: DOM nodes 22 1 6.9%
subgraph-idle: event listeners 10 7 64.5%
subgraph-mouse-sweep: avg frame time 17ms 0ms 0.0%
subgraph-mouse-sweep: layout duration 5ms 0ms 6.8%
subgraph-mouse-sweep: style recalc duration 42ms 3ms 7.8%
subgraph-mouse-sweep: layout count 16 0 0.0%
subgraph-mouse-sweep: style recalc count 80 2 2.4%
subgraph-mouse-sweep: task duration 766ms 69ms 9.0%
subgraph-mouse-sweep: script duration 101ms 7ms 6.5%
subgraph-mouse-sweep: TBT 0ms 0ms 0.0%
subgraph-mouse-sweep: DOM nodes 67 2 3.3%
subgraph-mouse-sweep: event listeners 8 4 52.6%
workflow-execution: avg frame time 17ms 0ms 0.0%
workflow-execution: layout duration 2ms 0ms 9.4%
workflow-execution: style recalc duration 24ms 2ms 9.1%
workflow-execution: layout count 5 1 11.0%
workflow-execution: style recalc count 18 2 11.5%
workflow-execution: task duration 123ms 11ms 8.8%
workflow-execution: script duration 29ms 3ms 10.2%
workflow-execution: TBT 0ms 0ms 0.0%
workflow-execution: DOM nodes 161 7 4.4%
workflow-execution: event listeners 52 4 8.4%
Trend (last 15 commits on main)
Metric Trend Dir Latest
canvas-idle: avg frame time ▆▃▆▁▆▃▆█▆▆▄▃▃▄▃ ➡️ 17ms
canvas-idle: p95 frame time ➡️ NaNms
canvas-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: style recalc duration ▇▇▆▆▃█▄▃▄▃▇▄▁▆▇ ➡️ 11ms
canvas-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
canvas-idle: style recalc count █▃▅▂▅▆▃▁▂▁▂▅▆▅▆ ➡️ 12
canvas-idle: task duration ▃▃▃▆▂▃▃▅▆▂█▃▁▃▃ ➡️ 391ms
canvas-idle: script duration ▄▃▅▇▂▅▃▆▇▅█▄▁▅▆ ➡️ 27ms
canvas-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: heap used ➡️ NaN MB
canvas-idle: DOM nodes █▇▆▅▃▇▃▁▂▂▅▆▆▆▇ ➡️ 24
canvas-idle: event listeners ▅█▅▄▁▅▁▁▁▄▅▅▁▅▄ 📉 11
canvas-mouse-sweep: avg frame time ▆█▆▃▁▃▁▆▆▁▃▆▆▃▃ ➡️ 17ms
canvas-mouse-sweep: p95 frame time ➡️ NaNms
canvas-mouse-sweep: layout duration ▁▃▂▄▁▂▁▃▆▂█▇▆▄▃ ➡️ 4ms
canvas-mouse-sweep: style recalc duration ▄▄▂▄▁▂▃▃▅▄█▆▂▄▄ ➡️ 43ms
canvas-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 12
canvas-mouse-sweep: style recalc count █▅▄▃▂▂▁▄▄▅▆▅▂▇▄ ➡️ 79
canvas-mouse-sweep: task duration █▆▄▂▂▃▂▄▄▅█▆▁▆▄ ➡️ 868ms
canvas-mouse-sweep: script duration ▄▅▄▆▄▆▆▆▅▅█▆▁▅▆ ➡️ 139ms
canvas-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-mouse-sweep: heap used ➡️ NaN MB
canvas-mouse-sweep: DOM nodes █▅▃▃▁▂▂▃▂▄▆▅▃▅▅ ➡️ 64
canvas-mouse-sweep: event listeners █▁▁▁▁▁▇▁▁▁██▇▁█ 📈 13
canvas-zoom-sweep: avg frame time ▅▅█▄▅▁▁▁▅▁▁▅▄▅▁ ➡️ 17ms
canvas-zoom-sweep: p95 frame time ➡️ NaNms
canvas-zoom-sweep: layout duration ▆▅▅▄▁▁█▅▃▅▇▆▁▂▆ ➡️ 1ms
canvas-zoom-sweep: style recalc duration ▆▅▄▆▅▃█▆▇▅▇▄▁▃▅ ➡️ 20ms
canvas-zoom-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 6
canvas-zoom-sweep: style recalc count ▁▁▃▄▆▃▆█▄▄▆▁▆▁▆ ➡️ 32
canvas-zoom-sweep: task duration ▄▂▁▇▂▂▄▅▆▃█▄▁▁▅ ➡️ 338ms
canvas-zoom-sweep: script duration ▃▃▂▇▂▂▅▇▆▅█▄▁▂▆ ➡️ 30ms
canvas-zoom-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-zoom-sweep: heap used ➡️ NaN MB
canvas-zoom-sweep: DOM nodes ▄▃▁▅█▁▃▆▄▅▅▃▃▄▃ ➡️ 79
canvas-zoom-sweep: event listeners ▁▁▂▅█▂▁▅▁▅▅▄▁▅▁ ➡️ 19
dom-widget-clipping: avg frame time ▂▄▅▅▂▄█▇▅▇▇▅▅▁▇ ➡️ 17ms
dom-widget-clipping: p95 frame time ➡️ NaNms
dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: style recalc duration ▆▆▂▆▄▃██▄▁▆▇▆▃▅ ➡️ 10ms
dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
dom-widget-clipping: style recalc count ▇█▅█▅▄█▇▇▁▇▄▇▂▅ ➡️ 13
dom-widget-clipping: task duration ▃▃▁▅▄▃▅▆▅▂▇█▁▅▅ ➡️ 371ms
dom-widget-clipping: script duration ▅▄▄▆▆▅▇▇▆▃█▇▁▇▇ ➡️ 71ms
dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: heap used ➡️ NaN MB
dom-widget-clipping: DOM nodes ▇▇▄▇▅▄█▇▅▁▅▄▇▃▄ ➡️ 21
dom-widget-clipping: event listeners ▅▅▅▅▁▅██▁▁▁▁█▁▁ 📉 2
large-graph-idle: avg frame time ▅▅▅▅▅▂▁▂▄▅▄▂▂▅█ ➡️ 17ms
large-graph-idle: p95 frame time ➡️ NaNms
large-graph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: style recalc duration ▅▅▅▆▄▅▃▄▅▅▆█▁▄▆ ➡️ 13ms
large-graph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-idle: style recalc count █▆█▃▃▁▃▆▃▆▆▃▆██ ➡️ 12
large-graph-idle: task duration ▂▃▂▆▂▃▃▇▅▃██▁▂▅ ➡️ 569ms
large-graph-idle: script duration ▄▅▄▆▄▅▅▇▆▅█▆▁▃▆ ➡️ 110ms
large-graph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: heap used ➡️ NaN MB
large-graph-idle: DOM nodes ▆█▅▂▅▃▁▂▃▅▅▆▂▆▅ ➡️ 25
large-graph-idle: event listeners ███▇██▄▁▄▇▇█▂█▇ ➡️ 29
large-graph-pan: avg frame time ▆▃▃▆█▃▁█▆▆▆▆█▁▆ ➡️ 17ms
large-graph-pan: p95 frame time ➡️ NaNms
large-graph-pan: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: style recalc duration ▃▂▄▄▁▅▂▂▁▄▄█▃▁▂ ➡️ 17ms
large-graph-pan: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-pan: style recalc count ▆▃█▂▃▂▂▂▁▇▅▃█▆▃ ➡️ 69
large-graph-pan: task duration ▄▃▄▆▄▄▄▆▄▄█▆▁▂▅ ➡️ 1100ms
large-graph-pan: script duration ▅▄▅▆▆▅▄▆▄▅█▄▁▄▅ ➡️ 413ms
large-graph-pan: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: heap used ➡️ NaN MB
large-graph-pan: DOM nodes ▅▃▆▂▄▁▃▁▁▅▁▂█▅▂ ➡️ 18
large-graph-pan: event listeners █▆█▁▁▆▁▁▃▆▁▃██▃ ➡️ 5
minimap-idle: avg frame time ▃▆▆▃█▁█▆▆▃▃▆█▆█ ➡️ 17ms
minimap-idle: p95 frame time ➡️ NaNms
minimap-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: style recalc duration ▄█▁█▅▅█▅▅▃▅▁▁▄▆ ➡️ 10ms
minimap-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
minimap-idle: style recalc count ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 9
minimap-idle: task duration ▃▄▁▅▁▃▄▅▇▃█▅▁▁▅ ➡️ 547ms
minimap-idle: script duration ▄▆▃▇▃▅▆▆▇▅█▅▁▃▆ ➡️ 106ms
minimap-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: heap used ➡️ NaN MB
minimap-idle: DOM nodes ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 19
minimap-idle: event listeners ▃▃▆▁▁▁▃▁▁▆▁▃█▆▁ ➡️ 4
subgraph-dom-widget-clipping: avg frame time ▅▄▄▄▄▄█▄▄▄▃▁▆▃▃ ➡️ 17ms
subgraph-dom-widget-clipping: p95 frame time ➡️ NaNms
subgraph-dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: style recalc duration ▂▄▃▅▅▃▂▅▇▃▄█▁▄▆ ➡️ 14ms
subgraph-dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-dom-widget-clipping: style recalc count ▇█▆▃▆▃▁▆█▇▃▆▇█▅ ➡️ 48
subgraph-dom-widget-clipping: task duration ▂▃▃▆▅▅▂▅█▂▆█▁▂▇ ➡️ 398ms
subgraph-dom-widget-clipping: script duration ▃▃▃▄▅▅▂▄█▂▅▇▁▂▅ ➡️ 131ms
subgraph-dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: heap used ➡️ NaN MB
subgraph-dom-widget-clipping: DOM nodes ▅▇▅▂▅▂▁▅▅▅▁▇▅█▄ ➡️ 22
subgraph-dom-widget-clipping: event listeners ▅▅▅▂▅▁▅██▁▁█▅█▅ 📈 16
subgraph-idle: avg frame time ▆▆█▁▆▃▆▆▆▃▆▁▃▆█ ➡️ 17ms
subgraph-idle: p95 frame time ➡️ NaNms
subgraph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: style recalc duration ▁▇▃▆▂▄▂▃▃▆▆▄▃▇█ ➡️ 12ms
subgraph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-idle: style recalc count ▃▆▃▃▂▅▁▂▁▆▃▃██▇ ➡️ 12
subgraph-idle: task duration ▁▃▁▇▁▁▃▆▅▂█▅▁▁▄ ➡️ 378ms
subgraph-idle: script duration ▁▃▂▇▁▂▃▇▆▂█▅▂▁▅ ➡️ 22ms
subgraph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: heap used ➡️ NaN MB
subgraph-idle: DOM nodes ▃▅▃▂▁▄▁▂▁▅▃▂▇█▇ ➡️ 24
subgraph-idle: event listeners ▁▅▁▁▁▁▁▁▁▅▄▁███ 📈 21
subgraph-mouse-sweep: avg frame time ▅▄▁▃▃▄▆▄▆▃▃█▁▃▃ ➡️ 17ms
subgraph-mouse-sweep: p95 frame time ➡️ NaNms
subgraph-mouse-sweep: layout duration ▁▄▄▄▃▃▅▅▅▂█▇▂▃▆ ➡️ 5ms
subgraph-mouse-sweep: style recalc duration ▃▂▄▅▂▃▄▅█▃█▆▁▂▅ ➡️ 43ms
subgraph-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 16
subgraph-mouse-sweep: style recalc count ▅▂▅▅▁▄▃▅█▅▆▄▂▄▅ ➡️ 81
subgraph-mouse-sweep: task duration ▃▂▄▅▂▄▄▅▇▄█▆▁▃▅ ➡️ 785ms
subgraph-mouse-sweep: script duration ▄▅▄▇▅▅▆▇▆▅██▁▄▆ ➡️ 105ms
subgraph-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-mouse-sweep: heap used ➡️ NaN MB
subgraph-mouse-sweep: DOM nodes ▅▁▄▅▁▄▃▃█▅▅▄▂▅▃ ➡️ 66
subgraph-mouse-sweep: event listeners ▇▁▂▇▁▂▂▂█▇▂▂▇▇▂ 📈 5
workflow-execution: avg frame time ▆▆▆▄▆▆▃▄▁▄█▆▅▄▆ ➡️ 17ms
workflow-execution: p95 frame time ➡️ NaNms
workflow-execution: layout duration ▁▆▁▃▂▄▃▂▃▃▅█▄▂▅ ➡️ 2ms
workflow-execution: style recalc duration ▃▇▅▇▁▅▆▇█▁██▂▄▆ ➡️ 25ms
workflow-execution: layout count ▁█▂▃▂▃▃▁▃▃▄▃▂▃▂ ➡️ 5
workflow-execution: style recalc count ▃█▅▇▁▄▅▆▅▅▅▅▄▄▂ ➡️ 15
workflow-execution: task duration ▂▅▄▅▁▄▆▆▆▁▇█▁▃▃ ➡️ 120ms
workflow-execution: script duration ▄▃▄▄▃▅▄▅▆▂▇█▁▃▄ ➡️ 29ms
workflow-execution: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
workflow-execution: heap used ➡️ NaN MB
workflow-execution: DOM nodes ▂█▃▆▁▄▃▅▃█▃▃▄▃▁ ➡️ 152
workflow-execution: event listeners ▅███▁▅███▁██▅█▅ ➡️ 49
Raw data
{
  "timestamp": "2026-05-04T11:05:16.413Z",
  "gitSha": "840727954e6479062dcc1413bff9d37cf419b2e4",
  "branch": "jaewon/fe-535-virtual-grid-blank-on-scroll",
  "measurements": [
    {
      "name": "canvas-idle",
      "durationMs": 2010.8470000000125,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 8.275999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 326.43199999999996,
      "heapDeltaBytes": 22830492,
      "heapUsedBytes": 72250460,
      "domNodes": 16,
      "jsHeapTotalBytes": 14942208,
      "scriptDurationMs": 14.601000000000003,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-idle",
      "durationMs": 2016.8710000000374,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.12,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 335.04200000000003,
      "heapDeltaBytes": 22967336,
      "heapUsedBytes": 71737024,
      "domNodes": 18,
      "jsHeapTotalBytes": 14417920,
      "scriptDurationMs": 15.056,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-idle",
      "durationMs": 2033.0250000001797,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 6.185000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 336.20399999999995,
      "heapDeltaBytes": 22925800,
      "heapUsedBytes": 71673836,
      "domNodes": 16,
      "jsHeapTotalBytes": 14942208,
      "scriptDurationMs": 14.697999999999999,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1815.979999999854,
      "styleRecalcs": 71,
      "styleRecalcDurationMs": 33.907,
      "layouts": 12,
      "layoutDurationMs": 3.224,
      "taskDurationMs": 779.337,
      "heapDeltaBytes": -2068724,
      "heapUsedBytes": 46648188,
      "domNodes": -264,
      "jsHeapTotalBytes": 15593472,
      "scriptDurationMs": 122.215,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1844.3620000000465,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 44.28099999999999,
      "layouts": 12,
      "layoutDurationMs": 4.022,
      "taskDurationMs": 775.1400000000001,
      "heapDeltaBytes": -3547200,
      "heapUsedBytes": 62348440,
      "domNodes": 58,
      "jsHeapTotalBytes": 19755008,
      "scriptDurationMs": 128.749,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 2002.8959999999643,
      "styleRecalcs": 80,
      "styleRecalcDurationMs": 42.792,
      "layouts": 12,
      "layoutDurationMs": 3.91,
      "taskDurationMs": 919.1,
      "heapDeltaBytes": 18887204,
      "heapUsedBytes": 68230004,
      "domNodes": 62,
      "jsHeapTotalBytes": 15728640,
      "scriptDurationMs": 118.186,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1731.443999999783,
      "styleRecalcs": 30,
      "styleRecalcDurationMs": 17.185000000000002,
      "layouts": 6,
      "layoutDurationMs": 0.7009999999999998,
      "taskDurationMs": 311.87399999999997,
      "heapDeltaBytes": 164796,
      "heapUsedBytes": 48524900,
      "domNodes": 75,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 21.544000000000008,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1728.5460000000512,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 17.808999999999997,
      "layouts": 6,
      "layoutDurationMs": 0.737,
      "taskDurationMs": 295.87,
      "heapDeltaBytes": 253768,
      "heapUsedBytes": 47877880,
      "domNodes": 80,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 27.723,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1719.9839999998403,
      "styleRecalcs": 32,
      "styleRecalcDurationMs": 17.521,
      "layouts": 6,
      "layoutDurationMs": 0.65,
      "taskDurationMs": 282.192,
      "heapDeltaBytes": 115040,
      "heapUsedBytes": 48855756,
      "domNodes": 79,
      "jsHeapTotalBytes": 14942208,
      "scriptDurationMs": 17.695000000000004,
      "eventListeners": 21,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 491.8870000001334,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 5.958999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 306.76099999999997,
      "heapDeltaBytes": 8844960,
      "heapUsedBytes": 57417828,
      "domNodes": 14,
      "jsHeapTotalBytes": 15728640,
      "scriptDurationMs": 53.12299999999999,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 594.1309999998339,
      "styleRecalcs": 13,
      "styleRecalcDurationMs": 9.045000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 360.761,
      "heapDeltaBytes": 9798396,
      "heapUsedBytes": 57409060,
      "domNodes": 22,
      "jsHeapTotalBytes": 15466496,
      "scriptDurationMs": 65.50399999999999,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 532.1570000000975,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 6.905999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 312.933,
      "heapDeltaBytes": 9114736,
      "heapUsedBytes": 57443920,
      "domNodes": 17,
      "jsHeapTotalBytes": 15990784,
      "scriptDurationMs": 54.754999999999995,
      "eventListeners": 0,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2021.4710000000196,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 6.919999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 517.8439999999999,
      "heapDeltaBytes": 4284304,
      "heapUsedBytes": 62751016,
      "domNodes": -262,
      "jsHeapTotalBytes": 552960,
      "scriptDurationMs": 76.92399999999999,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2008.5340000000542,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 7.615,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 517.232,
      "heapDeltaBytes": 2961812,
      "heapUsedBytes": 59862028,
      "domNodes": -262,
      "jsHeapTotalBytes": 5533696,
      "scriptDurationMs": 79.78200000000002,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2017.7619999999479,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 8.258000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 522.654,
      "heapDeltaBytes": 9680268,
      "heapUsedBytes": 68277804,
      "domNodes": -262,
      "jsHeapTotalBytes": 28672,
      "scriptDurationMs": 78.688,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2097.328999999945,
      "styleRecalcs": 67,
      "styleRecalcDurationMs": 18.427,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1092.7720000000002,
      "heapDeltaBytes": 9027424,
      "heapUsedBytes": 68802040,
      "domNodes": -259,
      "jsHeapTotalBytes": 2854912,
      "scriptDurationMs": 412.441,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2102.0750000000135,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 19.185000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1025.08,
      "heapDeltaBytes": 10286744,
      "heapUsedBytes": 68950072,
      "domNodes": -262,
      "jsHeapTotalBytes": 5214208,
      "scriptDurationMs": 366.457,
      "eventListeners": -159,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2128.3089999999447,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 18.386,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1102.79,
      "heapDeltaBytes": 3902148,
      "heapUsedBytes": 63348840,
      "domNodes": -267,
      "jsHeapTotalBytes": 2330624,
      "scriptDurationMs": 437.875,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3133.9210000001003,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 19.214000000000002,
      "layouts": 60,
      "layoutDurationMs": 8.362000000000002,
      "taskDurationMs": 1290.4979999999998,
      "heapDeltaBytes": 7946356,
      "heapUsedBytes": 69228508,
      "domNodes": -269,
      "jsHeapTotalBytes": 4485120,
      "scriptDurationMs": 469.86000000000007,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3134.405999999899,
      "styleRecalcs": 64,
      "styleRecalcDurationMs": 18.158,
      "layouts": 60,
      "layoutDurationMs": 8.016,
      "taskDurationMs": 1279.2250000000001,
      "heapDeltaBytes": 11963984,
      "heapUsedBytes": 72713812,
      "domNodes": -270,
      "jsHeapTotalBytes": -233472,
      "scriptDurationMs": 466.701,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3148.290000000088,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 18.322999999999997,
      "layouts": 60,
      "layoutDurationMs": 8.232999999999999,
      "taskDurationMs": 1335.21,
      "heapDeltaBytes": 12982784,
      "heapUsedBytes": 74013252,
      "domNodes": -268,
      "jsHeapTotalBytes": 815104,
      "scriptDurationMs": 511.51299999999986,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2021.6239999999743,
      "styleRecalcs": 8,
      "styleRecalcDurationMs": 7.213000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 558.997,
      "heapDeltaBytes": 8306464,
      "heapUsedBytes": 68573116,
      "domNodes": -264,
      "jsHeapTotalBytes": -1019904,
      "scriptDurationMs": 90.766,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66999999999998,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "minimap-idle",
      "durationMs": 2028.150000000096,
      "styleRecalcs": 7,
      "styleRecalcDurationMs": 5.885999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 497.641,
      "heapDeltaBytes": 3674888,
      "heapUsedBytes": 63892076,
      "domNodes": -266,
      "jsHeapTotalBytes": 28672,
      "scriptDurationMs": 75.039,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2033.3679999998822,
      "styleRecalcs": 7,
      "styleRecalcDurationMs": 6.072000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 499.64099999999996,
      "heapDeltaBytes": 9789044,
      "heapUsedBytes": 70182396,
      "domNodes": -266,
      "jsHeapTotalBytes": -233472,
      "scriptDurationMs": 75.46,
      "eventListeners": -129,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 587.0549999999639,
      "styleRecalcs": 46,
      "styleRecalcDurationMs": 11.252999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 370.606,
      "heapDeltaBytes": 9110856,
      "heapUsedBytes": 57773736,
      "domNodes": 18,
      "jsHeapTotalBytes": 16252928,
      "scriptDurationMs": 134.41500000000002,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 571.9100000001163,
      "styleRecalcs": 46,
      "styleRecalcDurationMs": 10.237,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 349.243,
      "heapDeltaBytes": 9123052,
      "heapUsedBytes": 57757452,
      "domNodes": 18,
      "jsHeapTotalBytes": 15990784,
      "scriptDurationMs": 125.493,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 538.0669999999554,
      "styleRecalcs": 46,
      "styleRecalcDurationMs": 9.995000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 335.241,
      "heapDeltaBytes": 9264884,
      "heapUsedBytes": 57829564,
      "domNodes": 18,
      "jsHeapTotalBytes": 15728640,
      "scriptDurationMs": 116.59999999999998,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-idle",
      "durationMs": 1988.7099999998554,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.082,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 385.75800000000004,
      "heapDeltaBytes": 22241132,
      "heapUsedBytes": 71902364,
      "domNodes": 18,
      "jsHeapTotalBytes": 14942208,
      "scriptDurationMs": 14.167,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2005.3159999999934,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 7.925,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 335.946,
      "heapDeltaBytes": 23215688,
      "heapUsedBytes": 71852340,
      "domNodes": 18,
      "jsHeapTotalBytes": 15728640,
      "scriptDurationMs": 13.395000000000001,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2003.6239999999452,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 7.023,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 322.379,
      "heapDeltaBytes": 22507556,
      "heapUsedBytes": 71190712,
      "domNodes": 18,
      "jsHeapTotalBytes": 14680064,
      "scriptDurationMs": 12.786,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1725.945000000138,
      "styleRecalcs": 77,
      "styleRecalcDurationMs": 40.8,
      "layouts": 16,
      "layoutDurationMs": 4.755,
      "taskDurationMs": 671.0290000000001,
      "heapDeltaBytes": 14855448,
      "heapUsedBytes": 62968556,
      "domNodes": 65,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 95.367,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1677.9059999998935,
      "styleRecalcs": 75,
      "styleRecalcDurationMs": 40.366,
      "layouts": 16,
      "layoutDurationMs": 5.069000000000001,
      "taskDurationMs": 698.778,
      "heapDeltaBytes": 14042452,
      "heapUsedBytes": 62706204,
      "domNodes": 60,
      "jsHeapTotalBytes": 15204352,
      "scriptDurationMs": 97.938,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1686.1039999998866,
      "styleRecalcs": 74,
      "styleRecalcDurationMs": 36.33,
      "layouts": 16,
      "layoutDurationMs": 7.110000000000001,
      "taskDurationMs": 682.8349999999999,
      "heapDeltaBytes": -1730840,
      "heapUsedBytes": 46900360,
      "domNodes": -264,
      "jsHeapTotalBytes": 15593472,
      "scriptDurationMs": 87.27400000000002,
      "eventListeners": -133,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8144.048000000112,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 59.022999999999996,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3607.417,
      "heapDeltaBytes": 10562892,
      "heapUsedBytes": 69003888,
      "domNodes": -263,
      "jsHeapTotalBytes": 6524928,
      "scriptDurationMs": 1190.064,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8152.800999999954,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 56.582,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3425.198,
      "heapDeltaBytes": 19838344,
      "heapUsedBytes": 79234220,
      "domNodes": -262,
      "jsHeapTotalBytes": 6029312,
      "scriptDurationMs": 1131.51,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333338,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8140.1599999999235,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 58.137,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3441.5330000000004,
      "heapDeltaBytes": 9922912,
      "heapUsedBytes": 67034520,
      "domNodes": -259,
      "jsHeapTotalBytes": 6262784,
      "scriptDurationMs": 1146.304,
      "eventListeners": -113,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 13516.605000000027,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 13500.02,
      "heapDeltaBytes": -43029236,
      "heapUsedBytes": 160343196,
      "domNodes": -8331,
      "jsHeapTotalBytes": 26013696,
      "scriptDurationMs": 594.882,
      "eventListeners": -16460,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.220000000000073,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12685.647000000017,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12674.314000000002,
      "heapDeltaBytes": -32031812,
      "heapUsedBytes": 171878532,
      "domNodes": -8331,
      "jsHeapTotalBytes": 24965120,
      "scriptDurationMs": 569.175,
      "eventListeners": -16466,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12975.12299999994,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12936.989999999998,
      "heapDeltaBytes": -34757212,
      "heapUsedBytes": 171695548,
      "domNodes": -8331,
      "jsHeapTotalBytes": 24440832,
      "scriptDurationMs": 609.366,
      "eventListeners": -16462,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 15801.235000000133,
      "styleRecalcs": 84,
      "styleRecalcDurationMs": 19.411000000000012,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 15776.167000000001,
      "heapDeltaBytes": -36869456,
      "heapUsedBytes": 159580956,
      "domNodes": -8329,
      "jsHeapTotalBytes": -2035712,
      "scriptDurationMs": 873.0849999999999,
      "eventListeners": -16488,
      "totalBlockingTimeMs": 51,
      "frameDurationMs": 17.223333333333237,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 15432.823999999982,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 18.093999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 15410.740999999998,
      "heapDeltaBytes": -44234128,
      "heapUsedBytes": 154393256,
      "domNodes": -8329,
      "jsHeapTotalBytes": -4657152,
      "scriptDurationMs": 837.747,
      "eventListeners": -16490,
      "totalBlockingTimeMs": 8,
      "frameDurationMs": 17.220000000000073,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 15470.589000000018,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 18.502000000000017,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 15430.011999999997,
      "heapDeltaBytes": -58944580,
      "heapUsedBytes": 153459432,
      "domNodes": -8331,
      "jsHeapTotalBytes": -2560000,
      "scriptDurationMs": 898.544,
      "eventListeners": -16490,
      "totalBlockingTimeMs": 54,
      "frameDurationMs": 17.216666666666665,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "workflow-execution",
      "durationMs": 476.52900000002774,
      "styleRecalcs": 26,
      "styleRecalcDurationMs": 30.331000000000003,
      "layouts": 5,
      "layoutDurationMs": 1.766,
      "taskDurationMs": 143.22199999999998,
      "heapDeltaBytes": 5442564,
      "heapUsedBytes": 55285040,
      "domNodes": 196,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 25.21,
      "eventListeners": 69,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "workflow-execution",
      "durationMs": 460.02199999998084,
      "styleRecalcs": 18,
      "styleRecalcDurationMs": 25.521,
      "layouts": 5,
      "layoutDurationMs": 1.4329999999999998,
      "taskDurationMs": 118.698,
      "heapDeltaBytes": 5014312,
      "heapUsedBytes": 54958208,
      "domNodes": 157,
      "jsHeapTotalBytes": 524288,
      "scriptDurationMs": 25.014000000000003,
      "eventListeners": 69,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.799999999999727
    },
    {
      "name": "workflow-execution",
      "durationMs": 456.26599999991413,
      "styleRecalcs": 15,
      "styleRecalcDurationMs": 24.676999999999996,
      "layouts": 5,
      "layoutDurationMs": 1.6580000000000001,
      "taskDurationMs": 125.83899999999998,
      "heapDeltaBytes": 5023864,
      "heapUsedBytes": 54996684,
      "domNodes": 155,
      "jsHeapTotalBytes": 524288,
      "scriptDurationMs": 26.403,
      "eventListeners": 69,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999727
    }
  ]
}

Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Inline comments:
In `@src/components/common/VirtualGrid.test.ts`:
- Around line 271-298: The test "keeps last page visible when items shrink below
current scroll (FE-535)" never actually shrinks the list; after mounting
VirtualGrid with items and a high mockedScrollY, update the items array to a
smaller set (reuse the items variable, e.g. replace with createItems(n) where n
< original count) after the initial render, trigger a reactive update (await
nextTick() or call the component's props update mechanism) and then assert that
renderedItems (queryAllByText(/^Item \d+$/)) still has length > 0; locate the
mounts and assertions around the items variable, mockedScrollY, VirtualGrid
render call and the final expect to implement this change.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: f7a1a1b0-3f67-4377-96cb-1681abe2ac68

📥 Commits

Reviewing files that changed from the base of the PR and between 0491836 and 96d0d3b.

📒 Files selected for processing (2)
  • src/components/common/VirtualGrid.test.ts
  • src/components/common/VirtualGrid.vue

Comment on lines +271 to +298
it('keeps last page visible when items shrink below current scroll (FE-535)', async () => {
const items = createItems(8)
mockedWidth.value = 400
mockedHeight.value = 240
mockedScrollY.value = 3000

render(VirtualGrid, {
props: {
items,
gridStyle: defaultGridStyle,
defaultItemHeight: 100,
defaultItemWidth: 100,
maxColumns: 4,
bufferRows: 1
},
slots: {
item: `<template #item="{ item }">
<div class="test-item">{{ item.name }}</div>
</template>`
},
container: document.body.appendChild(document.createElement('div'))
})

await nextTick()

const renderedItems = screen.queryAllByText(/^Item \d+$/)
expect(renderedItems.length).toBeGreaterThan(0)
})
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue | 🟡 Minor | ⚡ Quick win

Line 271 test name says “items shrink”, but the test never shrinks items.

Right now it mounts once with a small list and high scrollY, which mostly
duplicates the previous case. To cover the retained-scroll + list-shrink path,
update items after initial render and assert rendering still remains non-blank.

🔧 Proposed test adjustment
 it('keeps last page visible when items shrink below current scroll (FE-535)', async () => {
-  const items = createItems(8)
+  const initialItems = createItems(40)
   mockedWidth.value = 400
   mockedHeight.value = 240
   mockedScrollY.value = 3000

-  render(VirtualGrid, {
+  const { rerender } = render(VirtualGrid, {
     props: {
-      items,
+      items: initialItems,
       gridStyle: defaultGridStyle,
       defaultItemHeight: 100,
       defaultItemWidth: 100,
       maxColumns: 4,
       bufferRows: 1
@@
   })

   await nextTick()

+  await rerender({
+    items: createItems(8),
+    gridStyle: defaultGridStyle,
+    defaultItemHeight: 100,
+    defaultItemWidth: 100,
+    maxColumns: 4,
+    bufferRows: 1
+  })
+  await nextTick()
+
   const renderedItems = screen.queryAllByText(/^Item \d+$/)
   expect(renderedItems.length).toBeGreaterThan(0)
 })
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/components/common/VirtualGrid.test.ts` around lines 271 - 298, The test
"keeps last page visible when items shrink below current scroll (FE-535)" never
actually shrinks the list; after mounting VirtualGrid with items and a high
mockedScrollY, update the items array to a smaller set (reuse the items
variable, e.g. replace with createItems(n) where n < original count) after the
initial render, trigger a reactive update (await nextTick() or call the
component's props update mechanism) and then assert that renderedItems
(queryAllByText(/^Item \d+$/)) still has length > 0; locate the mounts and
assertions around the items variable, mockedScrollY, VirtualGrid render call and
the final expect to implement this change.

@codecov
Copy link
Copy Markdown

codecov Bot commented May 4, 2026

Codecov Report

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

Files with missing lines Patch % Lines
src/components/common/VirtualGrid.vue 95.65% 1 Missing ⚠️
@@             Coverage Diff             @@
##             main   #11885       +/-   ##
===========================================
- Coverage   72.01%   56.08%   -15.93%     
===========================================
  Files        1493     1383      -110     
  Lines       89607    70778    -18829     
  Branches    25580    18810     -6770     
===========================================
- Hits        64531    39697    -24834     
- Misses      24167    30554     +6387     
+ Partials      909      527      -382     
Flag Coverage Δ
e2e ?
unit 56.08% <95.65%> (-0.01%) ⬇️

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

Files with missing lines Coverage Δ
src/components/common/VirtualGrid.vue 96.96% <95.65%> (-3.04%) ⬇️

... and 993 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.

Existing FE-535 cases both fall into the maxOffsetRows=0 branch (small
list, viewRows >= totalRows). Add a third case that exercises the other
branch — a 100-item list scrolled far past the natural maximum, where
maxOffsetRows must clamp the raw offsetRows (21 vs 100000/120) and the
last page (including the final item) must remain visible.

Asserting Item 99 is present also guards against a regression where
the clamp accidentally pushes offsetRows toward the top of the list.
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.

♻️ Duplicate comments (1)
src/components/common/VirtualGrid.test.ts (1)

271-298: ⚠️ Potential issue | 🟡 Minor | ⚡ Quick win

Exercise the actual shrink-after-scroll scenario.

This still mounts once with 8 items and a high scrollY; it never reproduces the retained-scroll + list-shrink case described by FE-535. Please rerender with a smaller items array after the initial mount and assert the grid stays non-blank.

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

In `@src/components/common/VirtualGrid.test.ts` around lines 271 - 298, The test
mounts VirtualGrid with createItems(8) and a high mockedScrollY but never
simulates the shrink-after-scroll scenario; after the initial render + await
nextTick(), rerender the component (or call the testing-library rerender) with a
smaller items array (e.g. createItems(3)) while keeping mockedScrollY the same
to reproduce retained-scroll + list-shrink, then assert via
screen.queryAllByText(/^Item \d+$/) that renderedItems.length remains > 0 so the
grid is not blank; update the test that starts with render(VirtualGrid, ...) and
the assertion block accordingly.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Duplicate comments:
In `@src/components/common/VirtualGrid.test.ts`:
- Around line 271-298: The test mounts VirtualGrid with createItems(8) and a
high mockedScrollY but never simulates the shrink-after-scroll scenario; after
the initial render + await nextTick(), rerender the component (or call the
testing-library rerender) with a smaller items array (e.g. createItems(3)) while
keeping mockedScrollY the same to reproduce retained-scroll + list-shrink, then
assert via screen.queryAllByText(/^Item \d+$/) that renderedItems.length remains
> 0 so the grid is not blank; update the test that starts with
render(VirtualGrid, ...) and the assertion block accordingly.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 6d7814f8-99d2-4c50-946f-b75d4fb453ff

📥 Commits

Reviewing files that changed from the base of the PR and between 96d0d3b and 4658379.

📒 Files selected for processing (1)
  • src/components/common/VirtualGrid.test.ts

Replace the hand-rolled scroll/offset/clamp logic with a single
useVirtualizer driving rows of `cols` cells. The FE-535 bug class
(offsetRows desync when scrollY drifts past the natural max — popover
remount with retained scroll, macOS rubberband over-scroll) is
structurally eliminated: tanstack reads scrollOffset fresh from the
DOM each computation, the browser auto-clamps scrollTop when content
shrinks, and bounds are handled inside the library.

Public API (props, slots, emits) unchanged. All 5 consumers
(FormDropdownMenu, AssetsSidebarListView, AssetsSidebarGridView,
AssetGrid, ManagerDialog) continue to work without modification.

- Drops `scrollThrottle` / `resizeDebounce` props (internal-only,
  no consumers passed them)
- Drops manual top/bottom spacers in favor of getTotalSize() +
  absolutely positioned virtual rows
- Drops first-child measurement loop (tanstack measures via
  ResizeObserver)
- bufferRows -> overscan (semantic equivalent)
- approach-end fires when the last virtual row index is within
  bufferRows of rowCount

Tests: keep behavioral coverage (slot indices, maxColumns rendering,
empty list, approach-end emission, last-page invariant). Drop the
clamp-implementation-specific FE-535 reproductions since the bug
class is gone — replaced with a single invariant test documenting
the FE-535 motivation.
@dosubot dosubot Bot added size:L This PR changes 100-499 lines, ignoring generated files. and removed size:S This PR changes 10-29 lines, ignoring generated files. labels May 4, 2026
@dante01yoon dante01yoon changed the title fix(virtual-grid): clamp offsetRows so last page stays visible (FE-535) refactor(virtual-grid): migrate to @tanstack/vue-virtual (FE-535) May 4, 2026
Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai Bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 2

🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Inline comments:
In `@src/components/common/VirtualGrid.vue`:
- Around line 60-63: Clamp and normalize maxColumns before using it in layout
math: inside the computed cols() (and any similar usage for rowCount), if
maxColumns is finite convert it to a positive integer via Math.max(1,
Math.floor(maxColumns)) and use that value instead of returning maxColumns raw;
otherwise fall back to Math.floor(width.value / defaultItemWidth) || 1. Update
any other places that read maxColumns (e.g., rowCount calculation) to use the
same normalized/clamped value to prevent zero, negative or fractional column
counts.
- Around line 107-117: The watcher on virtualRows currently calls
emit('approach-end') repeatedly while the viewport stays near the end; modify
the watcher to only emit once per crossing by adding a small state guard (e.g.,
a ref like lastApproachIndex or hasEmittedApproachEnd) and only calling
emit('approach-end') when the last.index crosses the threshold (last.index >=
rowCount.value - bufferRows - 1) and is greater than the stored
lastApproachIndex (or if hasEmittedApproachEnd is false), then update that state
when you emit; reference the virtualRows watcher, last (rows.at(-1)), rowCount,
bufferRows, and the emit('approach-end') call when implementing this guard.
🪄 Autofix (Beta)

Fix all unresolved CodeRabbit comments on this PR:

  • Push a commit to this branch (recommended)
  • Create a new PR with the fixes

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: b542f1c9-bd91-4969-a9d6-1f0f40392c82

📥 Commits

Reviewing files that changed from the base of the PR and between 4658379 and 442fe7e.

📒 Files selected for processing (2)
  • src/components/common/VirtualGrid.test.ts
  • src/components/common/VirtualGrid.vue

Comment on lines 60 to 63
const cols = computed(() => {
if (maxColumns !== Infinity) return maxColumns
return Math.floor(width.value / itemWidth.value) || 1
return Math.floor(width.value / defaultItemWidth) || 1
})
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue | 🟠 Major | ⚡ Quick win

Clamp maxColumns to a positive integer before using it in layout math.

Line 61 currently returns maxColumns as-is. If callers pass 0, negative, or fractional values, cols/rowCount become invalid and can destabilize virtualization state.

🔧 Proposed fix
+const normalizedMaxColumns = computed(() =>
+  maxColumns === Infinity ? Infinity : Math.max(1, Math.floor(maxColumns))
+)
+
 const cols = computed(() => {
-  if (maxColumns !== Infinity) return maxColumns
+  if (normalizedMaxColumns.value !== Infinity) {
+    return normalizedMaxColumns.value
+  }
   return Math.floor(width.value / defaultItemWidth) || 1
 })
 
 const rowCount = computed(() => Math.ceil(items.length / cols.value))

Also applies to: 65-65

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

In `@src/components/common/VirtualGrid.vue` around lines 60 - 63, Clamp and
normalize maxColumns before using it in layout math: inside the computed cols()
(and any similar usage for rowCount), if maxColumns is finite convert it to a
positive integer via Math.max(1, Math.floor(maxColumns)) and use that value
instead of returning maxColumns raw; otherwise fall back to
Math.floor(width.value / defaultItemWidth) || 1. Update any other places that
read maxColumns (e.g., rowCount calculation) to use the same normalized/clamped
value to prevent zero, negative or fractional column counts.

Comment on lines +107 to +117
watch(
virtualRows,
(rows) => {
const last = rows.at(-1)
if (!last) return
if (last.index >= rowCount.value - bufferRows - 1) {
emit('approach-end')
}
}
}
const onResize = debounce(updateItemSize, resizeDebounce)
watch([width, height], onResize, { flush: 'post' })
whenever(() => items, updateItemSize, { flush: 'post' })
onBeforeUnmount(() => {
onResize.cancel()
})
},
{ flush: 'post' }
)
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue | 🟠 Major | ⚡ Quick win

approach-end currently re-emits on every near-end row update.

Because Line 112 is evaluated inside a watcher of virtualRows, the event can fire repeatedly while the viewport remains near the end, which can trigger duplicate downstream loads.

🔧 Proposed fix
-watch(
-  virtualRows,
-  (rows) => {
-    const last = rows.at(-1)
-    if (!last) return
-    if (last.index >= rowCount.value - bufferRows - 1) {
-      emit('approach-end')
-    }
-  },
-  { flush: 'post' }
-)
+const isNearEnd = computed(() => {
+  const last = virtualRows.value.at(-1)
+  if (!last) return false
+  return last.index >= rowCount.value - bufferRows - 1
+})
+
+watch(
+  isNearEnd,
+  (nearEnd, wasNearEnd) => {
+    if (nearEnd && !wasNearEnd) emit('approach-end')
+  },
+  { flush: 'post' }
+)
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/components/common/VirtualGrid.vue` around lines 107 - 117, The watcher on
virtualRows currently calls emit('approach-end') repeatedly while the viewport
stays near the end; modify the watcher to only emit once per crossing by adding
a small state guard (e.g., a ref like lastApproachIndex or
hasEmittedApproachEnd) and only calling emit('approach-end') when the last.index
crosses the threshold (last.index >= rowCount.value - bufferRows - 1) and is
greater than the stored lastApproachIndex (or if hasEmittedApproachEnd is
false), then update that state when you emit; reference the virtualRows watcher,
last (rows.at(-1)), rowCount, bufferRows, and the emit('approach-end') call when
implementing this guard.

Copy link
Copy Markdown
Contributor

@christian-byrne christian-byrne left a comment

Choose a reason for hiding this comment

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

Did you have any luck reproducing the original bug from the bug report video?

Some CI checks are also failing.

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

WDYT about migratintg the form dropdown widget to tanstack first, rather than mass-migrating all consumers of VirtualGrid like this?

Copy link
Copy Markdown
Contributor

@christian-byrne christian-byrne left a comment

Choose a reason for hiding this comment

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

question: The PR title and description describe a one-line clamp on offsetRows, but commit 442fe7e replaces the hand-rolled virtualization with @tanstack/vue-virtual. The "Changes" and "Review Focus" sections still describe the old approach. Could the description be updated so reviewers know what they're actually evaluating (a refactor + adoption of a new dependency, not a localized clamp)?

Comment on lines +107 to +117
watch(
virtualRows,
(rows) => {
const last = rows.at(-1)
if (!last) return
if (last.index >= rowCount.value - bufferRows - 1) {
emit('approach-end')
}
}
}
const onResize = debounce(updateItemSize, resizeDebounce)
watch([width, height], onResize, { flush: 'post' })
whenever(() => items, updateItemSize, { flush: 'post' })
onBeforeUnmount(() => {
onResize.cancel()
})
},
{ flush: 'post' }
)
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

issue (non-blocking): the previous implementation used whenever(() => state.value.isNearEnd, ...) which is edge-triggered (only fires on false→true transition). The new watch(virtualRows, ...) runs every time virtualRows recomputes, and getVirtualItems() returns a fresh array on every read, so this can re-emit approach-end many times while the user lingers near the end of the list, and again immediately after newly-loaded items arrive (because the new last row is still within bufferRows of the new end).

Consumers happen to mitigate this today (AssetsSidebarTab.handleApproachEnd is useDebounceFn(..., 300) with an isLoadingMore guard), so it won't cause a runaway loop in this codebase, but the contract has silently changed. Consider tracking the previous "near end" boolean and emitting only on transitions to preserve the old semantics.

},
estimateSize: () => defaultItemHeight,
getScrollElement: () => container.value,
overscan: bufferRows
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

nitpick (non-blocking): count uses a getter, but overscan: bufferRows and estimateSize: () => defaultItemHeight capture the prop values at construction. With Vue 3.5 reactive props destructuring, prop changes won't propagate to the virtualizer. No current consumer mutates bufferRows or defaultItemHeight, so this is theoretical, but mirroring the count getter pattern keeps the API consistent.

Comment on lines 60 to 63
const cols = computed(() => {
if (maxColumns !== Infinity) return maxColumns
return Math.floor(width.value / itemWidth.value) || 1
return Math.floor(width.value / defaultItemWidth) || 1
})
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

question: the previous implementation re-derived cols from the first rendered item's measured clientWidth via updateItemSize() on resize. The new code uses only defaultItemWidth as the divisor. If any consumer renders items at a width different from the prop default and does not pass maxColumns, column count will now be wrong (and item indices emitted via the slot will desync from the visual grid). Did you verify all current callers either pass maxColumns or have defaultItemWidth exactly equal to the rendered item width?

mockedScrollY.value = 0

const onApproachEnd = vi.fn()
it('renders the last item when the entire range is visible (FE-535 invariant)', async () => {
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

suggestion (non-blocking): the new test mocks useVirtualizer and asserts that all items render when mockedVisibleEnd = Infinity, which verifies wiring rather than the original FE-535 triggers (PrimeVue Popover retaining mount + retained scrollY across reopen, macOS rubberband over-scroll between throttled samples). The previous tests exercised the actual scroll-state-desync bug. Two options to close this:

  1. Drive a real useVirtualizer against a JSDOM scroll container in the unit test, instead of mocking it out.
  2. Add a Playwright spec that reopens the LoadImage form-dropdown from the original bug report and asserts thumbnails are visible after a scroll → close → reopen cycle.

Without one of these, FE-535 could regress on a future @TanStack upgrade and we wouldn't catch it.

Comment on lines +92 to 100
function rowStyle(virtualRow: VirtualItem): CSSProperties {
return {
start: clamp(fromCol, 0, items?.length),
end: clamp(toCol, fromCol, items?.length),
isNearEnd: hasMoreToRender && remainingCol <= cols.value * bufferRows
position: 'absolute',
top: 0,
left: 0,
width: '100%',
transform: `translateY(${virtualRow.start}px)`
}
})
const renderedItems = computed(() =>
isValidGrid.value ? items.slice(state.value.start, state.value.end) : []
)

function rowsToHeight(itemsCount: number): string {
const rows = Math.ceil(itemsCount / cols.value)
return `${rows * itemHeight.value}px`
}
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

nitpick (non-blocking): rowStyle() is a plain function so a fresh style object is constructed for every virtual row on every render. Inlining the style literal in the template (or at least caching the static fields) would let Vue skip work when only virtualRow.start changes. Negligible at ~20 visible rows, mention only because this loop is the hot path during scroll.

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

Labels

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

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants