Skip to content

fix: separate surfaced outputs from inspectable 3D media#10698

Open
benceruleanlu wants to merge 4 commits intomainfrom
bl/fix-ply-asset-previewability
Open

fix: separate surfaced outputs from inspectable 3D media#10698
benceruleanlu wants to merge 4 commits intomainfrom
bl/fix-ply-asset-previewability

Conversation

@benceruleanlu
Copy link
Copy Markdown
Member

@benceruleanlu benceruleanlu commented Mar 29, 2026

Summary

Split 3D output handling so shared media classification stays broad while UI inspectability only includes formats the browser viewer can actually load.

Changes

  • What: moved 3D extension lists into @comfyorg/shared-frontend-utils/mediaExtensions, kept previewOutput/previewableOutputs as surfaced outputs, and added inspectableOutput/inspectableOutputs for viewer and lightbox entry points.
  • Breaking: none.
  • Dependencies: none.

Review Focus

  • Shared 3D registry stays in sync with the load3d viewer.
  • USDZ and other non-loadable 3D formats remain downloadable and visible, but do not get Inspect affordances.
  • Inspect affordances now key off inspectable outputs instead of broad 3D classification.

Validation

  • pnpm exec vitest run packages/shared-frontend-utils/src/formatUtil.test.ts src/stores/queueStore.test.ts src/components/queue/job/JobAssetsList.test.ts src/composables/queue/useResultGallery.test.ts src/composables/queue/useJobMenu.test.ts src/platform/assets/components/MediaAssetContextMenu.test.ts
  • pnpm typecheck
  • pnpm lint

┆Issue is synchronized with this Notion page by Unito

@coderabbitai
Copy link
Copy Markdown
Contributor

coderabbitai bot commented Mar 29, 2026

📝 Walkthrough

Walkthrough

A centralized 3D media-extensions module and a filename/extension-based preview/inspection API were added; codebase updated to use extension helpers and "inspectable" output semantics across utilities, components, stores, services, and tests.

Changes

Cohort / File(s) Summary
Media Extensions Module
packages/shared-frontend-utils/package.json, packages/shared-frontend-utils/src/mediaExtensions.ts
Adds ./mediaExtensions export and a new module exporting THREE_D_LOADABLE_EXTENSIONS, THREE_D_MEDIA_EXTENSIONS, union types, and type-guard helpers isThreeDMediaExtension / isThreeDLoadableExtension.
Format Utilities & Tests
packages/shared-frontend-utils/src/formatUtil.ts, packages/shared-frontend-utils/src/formatUtil.test.ts
Removes isPreviewableMediaType; adds getFileExtension and isPreviewableMediaFilename; getMediaTypeFromFilename now uses extension helpers; tests updated for filename-based preview checks and additional 3D extensions.
3D Loading Constants
src/extensions/core/load3d/constants.ts
Replaces hardcoded SUPPORTED_EXTENSIONS with a Set derived from THREE_D_LOADABLE_EXTENSIONS (dot-prefixed); SUPPORTED_EXTENSIONS_ACCEPT preserved.
Queue Store & Output Caching
src/stores/queueStore.ts, src/stores/queueStore.test.ts, src/services/jobOutputCache.ts, src/services/jobOutputCache.test.ts
Introduces supportsInspection, filterInspectable, inspectableOutputs / inspectableOutput on Result/Task models; renames/introduces getInspectableOutputsForTask with a backward-compatible getOutputsForTask wrapper; tests extended for .ply vs .usdz, lazy loading, caching, and inspection filtering.
Gallery & Composables
src/composables/queue/useResultGallery.ts, src/composables/queue/useResultGallery.test.ts, src/composables/queue/useJobMenu.ts, src/composables/queue/useJobMenu.test.ts
Gallery and composables now source and match against inspectableOutput / getInspectableOutputsForTask; job menu gating for inspect-asset uses inspectableOutput; tests updated accordingly.
UI Components & Tests (Assets / Jobs / Sidebar)
src/platform/assets/components/MediaAssetCard.vue, src/platform/assets/components/MediaAssetContextMenu.vue, src/platform/assets/components/MediaAssetContextMenu.test.ts, src/components/sidebar/tabs/AssetsSidebarTab.vue, src/components/queue/job/JobAssetsList.vue, src/components/queue/job/JobAssetsList.test.ts, src/components/sidebar/tabs/JobHistorySidebarTab.vue, src/components/queue/QueueProgressOverlay.vue
Components switched preview/inspect checks to isPreviewableMediaFilename(asset.name); removed fileKind prop and MediaKind usage; view/inspect actions now depend on inspectableOutput presence, with early returns when absent; tests adjusted for new inspectable semantics.
Misc / Tests Updated
src/components/queue/QueueProgressOverlay.vue, src/stores/..., src/.../useJobMenu.test.ts, src/services/.../jobOutputCache.test.ts
Widespread test updates to include/observe inspectableOutput vs previewOutput, ensuring non-inspectable 3D formats (e.g., usdz) are previewable but not inspectable; compatibility wrapper added for getOutputsForTask.

Sequence Diagram(s)

sequenceDiagram
  participant User
  participant UI as Asset UI / Menu
  participant Store as Queue Store
  participant Cache as JobOutputCache
  participant Gallery

  User->>UI: click "Inspect" / view asset
  UI->>Store: request taskRef.inspectableOutput
  Store-->>UI: returns inspectableOutput (or undefined)
  alt inspectableOutput present
    UI->>Cache: getInspectableOutputsForTask(task)
    Cache-->>UI: returns inspectable outputs
    UI->>Gallery: open gallery with inspectable output URL
    Gallery-->>User: shows 3D viewer
  else missing inspectableOutput
    UI-->>User: short-circuit (no gallery)
  end
Loading

Estimated code review effort

🎯 3 (Moderate) | ⏱️ ~30 minutes

Poem

🐰 I hopped through extensions, tidy and bright,
I sniffed each filename in morning light.
From .ply to .glb I cheerfully bind,
Filenames whispered what loaders would find.
A rabbit's small clap for logic aligned. 🥕


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 ('fix:'), makes no changes to browser_tests/, and lacks explanation for missing e2e tests. Add or update Playwright test under browser_tests/ for 3D format handling, or add concrete explanation in PR description for why e2e testing is impractical.
Docstring Coverage ⚠️ Warning Docstring coverage is 40.00% which is insufficient. The required threshold is 80.00%. Write docstrings for the functions missing them to satisfy the coverage threshold.
✅ Passed checks (3 passed)
Check name Status Explanation
Adr Compliance For Entity/Litegraph Changes ✅ Passed PR contains no changes to src/lib/litegraph/, src/ecs/, or graph entity files; check scope conditions are not met.
Title check ✅ Passed The title accurately summarizes the main change: separating surfaced outputs from inspectable 3D media, which is the core objective of this PR.
Description check ✅ Passed The description follows the template structure with Summary, Changes, Breaking/Dependencies sections, Review Focus, and Validation steps. It clearly explains the changes and rationale.
✨ 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 bl/fix-ply-asset-previewability

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

❤️ Share

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

@github-actions
Copy link
Copy Markdown

github-actions bot commented Mar 29, 2026

🎨 Storybook: ✅ Built — View Storybook

Details

⏰ Completed at: 04/07/2026, 10:37:07 PM UTC

Links

@github-actions
Copy link
Copy Markdown

github-actions bot commented Mar 29, 2026

🎭 Playwright: ✅ 980 passed, 0 failed · 6 flaky

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

@socket-security
Copy link
Copy Markdown

socket-security bot commented Mar 29, 2026

No dependency changes detected. Learn more about Socket for GitHub.

👍 No dependency changes detected in pull request

@github-actions
Copy link
Copy Markdown

github-actions bot commented Mar 29, 2026

📦 Bundle: 5.12 MB gzip 🔴 +707 B

Details

Summary

  • Raw size: 23.6 MB baseline 23.6 MB — 🔴 +2.29 kB
  • Gzip: 5.12 MB baseline 5.12 MB — 🔴 +707 B
  • Brotli: 3.96 MB baseline 3.96 MB — 🔴 +432 B
  • Bundles: 250 current • 250 baseline • 119 added / 119 removed

Category Glance
Utilities & Hooks 🔴 +1.52 kB (340 kB) · Data & Services 🔴 +636 B (2.98 MB) · Other 🔴 +92 B (8.53 MB) · Graph Workspace 🔴 +49 B (1.2 MB) · Vendor & Third-Party ⚪ 0 B (9.8 MB) · Panels & Settings ⚪ 0 B (484 kB) · + 5 more

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

Main entry bundles and manifests

File Before After Δ Raw Δ Gzip Δ Brotli
assets/index-DYBvG8AS.js (removed) 22.3 kB 🟢 -22.3 kB 🟢 -7.94 kB 🟢 -6.79 kB
assets/index-fJ6Qcx8h.js (new) 22.3 kB 🔴 +22.3 kB 🔴 +7.95 kB 🔴 +6.8 kB

Status: 1 added / 1 removed

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

Graph editor runtime, canvas, workflow orchestration

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

Status: 1 added / 1 removed

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

Top-level views, pages, and routed surfaces

File Before After Δ Raw Δ Gzip Δ Brotli
assets/CloudSurveyView-BQrTNajE.js (new) 15.7 kB 🔴 +15.7 kB 🔴 +3.4 kB 🔴 +2.9 kB
assets/CloudSurveyView-WdCyXJII.js (removed) 15.7 kB 🟢 -15.7 kB 🟢 -3.39 kB 🟢 -2.89 kB
assets/CloudLoginView-Beo_QApD.js (removed) 12 kB 🟢 -12 kB 🟢 -3.36 kB 🟢 -2.97 kB
assets/CloudLoginView-BzbC2P4A.js (new) 12 kB 🔴 +12 kB 🔴 +3.36 kB 🔴 +2.96 kB
assets/CloudSignupView-fE-T7SLA.js (new) 9.78 kB 🔴 +9.78 kB 🔴 +2.85 kB 🔴 +2.52 kB
assets/CloudSignupView-ltEL1U-e.js (removed) 9.78 kB 🟢 -9.78 kB 🟢 -2.85 kB 🟢 -2.5 kB
assets/UserCheckView-BC2KLyDn.js (removed) 9.04 kB 🟢 -9.04 kB 🟢 -2.33 kB 🟢 -2.03 kB
assets/UserCheckView-BxC06c2g.js (new) 9.04 kB 🔴 +9.04 kB 🔴 +2.33 kB 🔴 +2.03 kB
assets/CloudLayoutView-lOBiFVZJ.js (removed) 7.54 kB 🟢 -7.54 kB 🟢 -2.36 kB 🟢 -2.05 kB
assets/CloudLayoutView-UndcfmTa.js (new) 7.54 kB 🔴 +7.54 kB 🔴 +2.36 kB 🔴 +2.05 kB
assets/CloudForgotPasswordView-BtMoAdNe.js (new) 5.94 kB 🔴 +5.94 kB 🔴 +2.09 kB 🔴 +1.85 kB
assets/CloudForgotPasswordView-WBBN5Rer.js (removed) 5.94 kB 🟢 -5.94 kB 🟢 -2.09 kB 🟢 -1.84 kB
assets/CloudAuthTimeoutView-B1Jp6CPg.js (new) 5.31 kB 🔴 +5.31 kB 🔴 +1.93 kB 🔴 +1.68 kB
assets/CloudAuthTimeoutView-CEMOgpVi.js (removed) 5.31 kB 🟢 -5.31 kB 🟢 -1.93 kB 🟢 -1.7 kB
assets/CloudSubscriptionRedirectView-BCzBivFe.js (new) 5.08 kB 🔴 +5.08 kB 🔴 +1.91 kB 🔴 +1.68 kB
assets/CloudSubscriptionRedirectView-DXy_21pF.js (removed) 5.08 kB 🟢 -5.08 kB 🟢 -1.91 kB 🟢 -1.69 kB
assets/UserSelectView-D9czzpAc.js (removed) 4.71 kB 🟢 -4.71 kB 🟢 -1.74 kB 🟢 -1.54 kB
assets/UserSelectView-Dlfs6x_K.js (new) 4.71 kB 🔴 +4.71 kB 🔴 +1.74 kB 🔴 +1.54 kB

Status: 9 added / 9 removed / 2 unchanged

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

Configuration panels, inspectors, and settings screens

File Before After Δ Raw Δ Gzip Δ Brotli
assets/KeybindingPanel-Czgq6dtJ.js (new) 46.6 kB 🔴 +46.6 kB 🔴 +9.52 kB 🔴 +8.47 kB
assets/KeybindingPanel-D7uudKHL.js (removed) 46.6 kB 🟢 -46.6 kB 🟢 -9.52 kB 🟢 -8.47 kB
assets/SecretsPanel-BzagGUJM.js (removed) 22.4 kB 🟢 -22.4 kB 🟢 -5.42 kB 🟢 -4.76 kB
assets/SecretsPanel-NvwnCqrG.js (new) 22.4 kB 🔴 +22.4 kB 🔴 +5.42 kB 🔴 +4.76 kB
assets/LegacyCreditsPanel-Cphnzexc.js (new) 21.5 kB 🔴 +21.5 kB 🔴 +5.81 kB 🔴 +5.13 kB
assets/LegacyCreditsPanel-DyENPsmj.js (removed) 21.5 kB 🟢 -21.5 kB 🟢 -5.81 kB 🟢 -5.13 kB
assets/SubscriptionPanel-B2u1Tsjo.js (new) 19.7 kB 🔴 +19.7 kB 🔴 +5.01 kB 🔴 +4.4 kB
assets/SubscriptionPanel-DYHiAd5B.js (removed) 19.7 kB 🟢 -19.7 kB 🟢 -5.01 kB 🟢 -4.4 kB
assets/AboutPanel-BcTOf9bO.js (new) 12 kB 🔴 +12 kB 🔴 +3.33 kB 🔴 +2.99 kB
assets/AboutPanel-BqFdF0r-.js (removed) 12 kB 🟢 -12 kB 🟢 -3.33 kB 🟢 -2.98 kB
assets/ExtensionPanel-BFDWl0aO.js (removed) 9.78 kB 🟢 -9.78 kB 🟢 -2.82 kB 🟢 -2.51 kB
assets/ExtensionPanel-BYyclzbW.js (new) 9.78 kB 🔴 +9.78 kB 🔴 +2.82 kB 🔴 +2.5 kB
assets/ServerConfigPanel-DTXBlOJQ.js (new) 6.85 kB 🔴 +6.85 kB 🔴 +2.28 kB 🔴 +2.04 kB
assets/ServerConfigPanel-nCJ1u92F.js (removed) 6.85 kB 🟢 -6.85 kB 🟢 -2.27 kB 🟢 -2.05 kB
assets/UserPanel-BVmg-YnI.js (new) 6.56 kB 🔴 +6.56 kB 🔴 +2.15 kB 🔴 +1.88 kB
assets/UserPanel-l1A3Sb1M.js (removed) 6.56 kB 🟢 -6.56 kB 🟢 -2.15 kB 🟢 -1.88 kB
assets/cloudRemoteConfig-DQ5igz17.js (removed) 1.85 kB 🟢 -1.85 kB 🟢 -901 B 🟢 -786 B
assets/cloudRemoteConfig-LGeyEgQU.js (new) 1.85 kB 🔴 +1.85 kB 🔴 +903 B 🔴 +771 B
assets/refreshRemoteConfig-DKyFbwYW.js (removed) 1.45 kB 🟢 -1.45 kB 🟢 -647 B 🟢 -555 B
assets/refreshRemoteConfig-IxH5XxYd.js (new) 1.45 kB 🔴 +1.45 kB 🔴 +651 B 🔴 +551 B

Status: 10 added / 10 removed / 12 unchanged

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

Authentication, profile, and account management bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/auth-BDb6WWph.js (removed) 3.57 kB 🟢 -3.57 kB 🟢 -1.26 kB 🟢 -1.08 kB
assets/auth-CLHD4E_V.js (new) 3.57 kB 🔴 +3.57 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/SignUpForm-B_LruTkw.js (new) 3.16 kB 🔴 +3.16 kB 🔴 +1.29 kB 🔴 +1.14 kB
assets/SignUpForm-BjCsHljN.js (removed) 3.16 kB 🟢 -3.16 kB 🟢 -1.29 kB 🟢 -1.15 kB
assets/UpdatePasswordContent-D46Qp-7t.js (removed) 2.66 kB 🟢 -2.66 kB 🟢 -1.19 kB 🟢 -1.05 kB
assets/UpdatePasswordContent-DCyZSPZt.js (new) 2.66 kB 🔴 +2.66 kB 🔴 +1.19 kB 🔴 +1.05 kB
assets/authStore-BPw0O93m.js (removed) 989 B 🟢 -989 B 🟢 -483 B 🟢 -435 B
assets/authStore-DviePNQx.js (new) 989 B 🔴 +989 B 🔴 +483 B 🔴 +429 B
assets/auth-B3e-D-DG.js (removed) 348 B 🟢 -348 B 🟢 -215 B 🟢 -190 B
assets/auth-C_sIWQTm.js (new) 348 B 🔴 +348 B 🔴 +216 B 🔴 +192 B

Status: 5 added / 5 removed / 2 unchanged

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

Modals, dialogs, drawers, and in-app editors

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useShareDialog-BGDNY3be.js (removed) 108 kB 🟢 -108 kB 🟢 -22.4 kB 🟢 -18.9 kB
assets/useShareDialog-BYe0ckC1.js (new) 108 kB 🔴 +108 kB 🔴 +22.4 kB 🔴 +18.9 kB
assets/useSubscriptionDialog-DYh6Oc0N.js (new) 969 B 🔴 +969 B 🔴 +475 B 🔴 +415 B
assets/useSubscriptionDialog-Z54pW66L.js (removed) 969 B 🟢 -969 B 🟢 -476 B 🟢 -416 B

Status: 2 added / 2 removed

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

Reusable component library chunks

File Before After Δ Raw Δ Gzip Δ Brotli
assets/ComfyQueueButton-B6gP4yps.js (new) 13.5 kB 🔴 +13.5 kB 🔴 +3.79 kB 🔴 +3.37 kB
assets/ComfyQueueButton-BnUvJUlw.js (removed) 13.5 kB 🟢 -13.5 kB 🟢 -3.79 kB 🟢 -3.38 kB
assets/useTerminalTabs-BpJxrcA6.js (removed) 10.7 kB 🟢 -10.7 kB 🟢 -3.6 kB 🟢 -3.16 kB
assets/useTerminalTabs-Ex86QhqD.js (new) 10.7 kB 🔴 +10.7 kB 🔴 +3.6 kB 🔴 +3.17 kB
assets/SubscribeButton-DWkPFUQM.js (removed) 2.42 kB 🟢 -2.42 kB 🟢 -1.05 kB 🟢 -916 B
assets/SubscribeButton-K-ahZvta.js (new) 2.42 kB 🔴 +2.42 kB 🔴 +1.04 kB 🔴 +915 B
assets/cloudFeedbackTopbarButton-Deiptvoy.js (new) 1.66 kB 🔴 +1.66 kB 🔴 +840 B 🔴 +742 B
assets/cloudFeedbackTopbarButton-DYMuk9zh.js (removed) 1.66 kB 🟢 -1.66 kB 🟢 -844 B 🟢 -764 B
assets/ComfyQueueButton-BLCVQIeQ.js (new) 1.03 kB 🔴 +1.03 kB 🔴 +489 B 🔴 +437 B
assets/ComfyQueueButton-Bz-UzbpU.js (removed) 1.03 kB 🟢 -1.03 kB 🟢 -488 B 🟢 -440 B

Status: 5 added / 5 removed / 8 unchanged

Data & Services — 2.98 MB (baseline 2.97 MB) • 🔴 +636 B

Stores, services, APIs, and repositories

File Before After Δ Raw Δ Gzip Δ Brotli
assets/dialogService-BANfvj-z.js (new) 1.94 MB 🔴 +1.94 MB 🔴 +447 kB 🔴 +338 kB
assets/dialogService-DGV9jgn6.js (removed) 1.94 MB 🟢 -1.94 MB 🟢 -446 kB 🟢 -338 kB
assets/api-ChR1RkMk.js (removed) 885 kB 🟢 -885 kB 🟢 -211 kB 🟢 -167 kB
assets/api-CYAg8J9s.js (new) 885 kB 🔴 +885 kB 🔴 +211 kB 🔴 +166 kB
assets/load3dService-Be-0rRrG.js (removed) 92.5 kB 🟢 -92.5 kB 🟢 -19.7 kB 🟢 -16.9 kB
assets/load3dService-Lq3Vyw_w.js (new) 92.5 kB 🔴 +92.5 kB 🔴 +19.7 kB 🔴 +16.9 kB
assets/workflowShareService-CitObsvb.js (removed) 16.6 kB 🟢 -16.6 kB 🟢 -4.88 kB 🟢 -4.33 kB
assets/workflowShareService-DJW8j-X8.js (new) 16.6 kB 🔴 +16.6 kB 🔴 +4.88 kB 🔴 +4.32 kB
assets/keybindingService-B728lUoo.js (removed) 13.8 kB 🟢 -13.8 kB 🟢 -3.67 kB 🟢 -3.21 kB
assets/keybindingService-BQNeQIQb.js (new) 13.8 kB 🔴 +13.8 kB 🔴 +3.66 kB 🔴 +3.22 kB
assets/releaseStore-CuctZOOv.js (removed) 8.12 kB 🟢 -8.12 kB 🟢 -2.28 kB 🟢 -2 kB
assets/releaseStore-DWKbQd0B.js (new) 8.12 kB 🔴 +8.12 kB 🔴 +2.28 kB 🔴 +2 kB
assets/userStore-BvBtZonM.js (new) 2.24 kB 🔴 +2.24 kB 🔴 +870 B 🔴 +768 B
assets/userStore-CkBrfn2I.js (removed) 2.24 kB 🟢 -2.24 kB 🟢 -868 B 🟢 -768 B
assets/audioService-Cq4-NDt2.js (new) 1.8 kB 🔴 +1.8 kB 🔴 +878 B 🔴 +761 B
assets/audioService-PPBMtEgl.js (removed) 1.8 kB 🟢 -1.8 kB 🟢 -876 B 🟢 -761 B
assets/releaseStore-AVYR5sxM.js (removed) 993 B 🟢 -993 B 🟢 -480 B 🟢 -423 B
assets/releaseStore-WmBvVn6O.js (new) 993 B 🔴 +993 B 🔴 +480 B 🔴 +425 B
assets/workflowDraftStore-gjra0hus.js (removed) 969 B 🟢 -969 B 🟢 -474 B 🟢 -421 B
assets/workflowDraftStore-GsIvvCCV.js (new) 969 B 🔴 +969 B 🔴 +473 B 🔴 +422 B
assets/dialogService-CFeCuvJF.js (removed) 958 B 🟢 -958 B 🟢 -466 B 🟢 -415 B
assets/dialogService-ClFYrz95.js (new) 958 B 🔴 +958 B 🔴 +465 B 🔴 +415 B
assets/settingStore-BysVBmGP.js (removed) 956 B 🟢 -956 B 🟢 -470 B 🟢 -415 B
assets/settingStore-DrS6Uv1t.js (new) 956 B 🔴 +956 B 🔴 +469 B 🔴 +413 B
assets/assetsStore-C-mmbsup.js (removed) 955 B 🟢 -955 B 🟢 -470 B 🟢 -415 B
assets/assetsStore-CsqaY6Ly.js (new) 955 B 🔴 +955 B 🔴 +469 B 🔴 +416 B

Status: 13 added / 13 removed / 4 unchanged

Utilities & Hooks — 340 kB (baseline 338 kB) • 🔴 +1.52 kB

Helpers, composables, and utility bundles

File Before After Δ Raw Δ Gzip Δ Brotli
assets/useConflictDetection-BQhX2d3_.js (removed) 232 kB 🟢 -232 kB 🟢 -51.3 kB 🟢 -41.8 kB
assets/useConflictDetection-CAAt3Hvj.js (new) 232 kB 🔴 +232 kB 🔴 +51.3 kB 🔴 +41.7 kB
assets/useLoad3dViewer-CdUw1VOC.js (new) 18.7 kB 🔴 +18.7 kB 🔴 +4.44 kB 🔴 +3.9 kB
assets/useLoad3dViewer-CKc1_RRM.js (removed) 18.7 kB 🟢 -18.7 kB 🟢 -4.43 kB 🟢 -3.88 kB
assets/formatUtil-bOE7x3-T.js (new) 16.8 kB 🔴 +16.8 kB 🔴 +5.75 kB 🔴 +5.01 kB
assets/formatUtil-CbZjUsYR.js (removed) 15.3 kB 🟢 -15.3 kB 🟢 -5.32 kB 🟢 -4.65 kB
assets/useLoad3d-1fjrqADZ.js (removed) 15 kB 🟢 -15 kB 🟢 -3.79 kB 🟢 -3.36 kB
assets/useLoad3d-m7MBRMhZ.js (new) 15 kB 🔴 +15 kB 🔴 +3.79 kB 🔴 +3.36 kB
assets/useFeatureFlags-CLotajMI.js (removed) 5.78 kB 🟢 -5.78 kB 🟢 -1.75 kB 🟢 -1.48 kB
assets/useFeatureFlags-sKDM2miC.js (new) 5.78 kB 🔴 +5.78 kB 🔴 +1.75 kB 🔴 +1.48 kB
assets/useCopyToClipboard-A71N-sz4.js (new) 5.29 kB 🔴 +5.29 kB 🔴 +1.86 kB 🔴 +1.57 kB
assets/useCopyToClipboard-DzOWMekN.js (removed) 5.29 kB 🟢 -5.29 kB 🟢 -1.86 kB 🟢 -1.57 kB
assets/useWorkspaceUI-B9V_DzsU.js (removed) 3.34 kB 🟢 -3.34 kB 🟢 -980 B 🟢 -809 B
assets/useWorkspaceUI-swMKRfjU.js (new) 3.34 kB 🔴 +3.34 kB 🔴 +979 B 🔴 +807 B
assets/subscriptionCheckoutUtil-CZ_CnSj7.js (removed) 2.97 kB 🟢 -2.97 kB 🟢 -1.3 kB 🟢 -1.14 kB
assets/subscriptionCheckoutUtil-DOx_tE92.js (new) 2.97 kB 🔴 +2.97 kB 🔴 +1.31 kB 🔴 +1.14 kB
assets/assetPreviewUtil-DcD2jLDL.js (new) 2.27 kB 🔴 +2.27 kB 🔴 +960 B 🔴 +834 B
assets/assetPreviewUtil-DxmZxecX.js (removed) 2.27 kB 🟢 -2.27 kB 🟢 -959 B 🟢 -834 B
assets/useUpstreamValue-CBfXYjhw.js (new) 2.08 kB 🔴 +2.08 kB 🔴 +807 B 🔴 +712 B
assets/useUpstreamValue-CxVQFEJY.js (removed) 2.08 kB 🟢 -2.08 kB 🟢 -806 B 🟢 -713 B
assets/useLoad3d-DusasZj-.js (new) 1.13 kB 🔴 +1.13 kB 🔴 +539 B 🔴 +479 B
assets/useLoad3d-DY6tOy7d.js (removed) 1.13 kB 🟢 -1.13 kB 🟢 -538 B 🟢 -482 B
assets/useLoad3dViewer-BPwjHgo7.js (removed) 1.07 kB 🟢 -1.07 kB 🟢 -504 B 🟢 -454 B
assets/useLoad3dViewer-DETNLKxE.js (new) 1.07 kB 🔴 +1.07 kB 🔴 +504 B 🔴 +452 B
assets/useCurrentUser-BnUZ17tO.js (removed) 955 B 🟢 -955 B 🟢 -470 B 🟢 -417 B
assets/useCurrentUser-DjRSCxyw.js (new) 955 B 🔴 +955 B 🔴 +470 B 🔴 +416 B
assets/useWorkspaceSwitch-BY-hK2hZ.js (removed) 747 B 🟢 -747 B 🟢 -382 B 🟢 -332 B
assets/useWorkspaceSwitch-Zh3Wa06O.js (new) 747 B 🔴 +747 B 🔴 +386 B 🔴 +333 B

Status: 14 added / 14 removed / 12 unchanged

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

External libraries and shared vendor chunks

Status: 16 unchanged

Other — 8.53 MB (baseline 8.53 MB) • 🔴 +92 B

Bundles that do not match a named category

File Before After Δ Raw Δ Gzip Δ Brotli
assets/core-CIf5sUJN.js (removed) 76.8 kB 🟢 -76.8 kB 🟢 -19.9 kB 🟢 -16.9 kB
assets/core-DAIMqzJN.js (new) 76.8 kB 🔴 +76.8 kB 🔴 +19.9 kB 🔴 +17 kB
assets/groupNode-BeyPmsdY.js (new) 74 kB 🔴 +74 kB 🔴 +18.5 kB 🔴 +16.3 kB
assets/groupNode-ClxLqidy.js (removed) 74 kB 🟢 -74 kB 🟢 -18.5 kB 🟢 -16.3 kB
assets/WidgetSelect-C2KgnRDS.js (removed) 64.6 kB 🟢 -64.6 kB 🟢 -14.1 kB 🟢 -12.2 kB
assets/WidgetSelect-D5LSzzdJ.js (new) 64.6 kB 🔴 +64.6 kB 🔴 +14.1 kB 🔴 +12.2 kB
assets/SubscriptionRequiredDialogContentWorkspace-31o9RRTu.js (removed) 48.9 kB 🟢 -48.9 kB 🟢 -9.3 kB 🟢 -7.93 kB
assets/SubscriptionRequiredDialogContentWorkspace-DDGU41G6.js (new) 48.9 kB 🔴 +48.9 kB 🔴 +9.3 kB 🔴 +7.95 kB
assets/WidgetPainter-CHZ0quwM.js (new) 33.3 kB 🔴 +33.3 kB 🔴 +8.11 kB 🔴 +7.18 kB
assets/WidgetPainter-gNpjugpQ.js (removed) 33.3 kB 🟢 -33.3 kB 🟢 -8.11 kB 🟢 -7.21 kB
assets/Load3DControls-CIo5_HmQ.js (new) 32.1 kB 🔴 +32.1 kB 🔴 +5.47 kB 🔴 +4.75 kB
assets/Load3DControls-Cwle13RJ.js (removed) 32.1 kB 🟢 -32.1 kB 🟢 -5.47 kB 🟢 -4.76 kB
assets/WorkspacePanelContent-3eqtCOmX.js (new) 29.9 kB 🔴 +29.9 kB 🔴 +6.33 kB 🔴 +5.56 kB
assets/WorkspacePanelContent-BipKZORA.js (removed) 29.9 kB 🟢 -29.9 kB 🟢 -6.33 kB 🟢 -5.54 kB
assets/SubscriptionRequiredDialogContent-Bc2NAIEe.js (new) 28.2 kB 🔴 +28.2 kB 🔴 +7.17 kB 🔴 +6.29 kB
assets/SubscriptionRequiredDialogContent-DWnkFftU.js (removed) 28.2 kB 🟢 -28.2 kB 🟢 -7.17 kB 🟢 -6.34 kB
assets/Load3dViewerContent-DU9d5hqy.js (removed) 24.5 kB 🟢 -24.5 kB 🟢 -5.33 kB 🟢 -4.64 kB
assets/Load3dViewerContent-JlGiyF9p.js (new) 24.5 kB 🔴 +24.5 kB 🔴 +5.33 kB 🔴 +4.63 kB
assets/WidgetImageCrop-DE4Q0omF.js (removed) 23.3 kB 🟢 -23.3 kB 🟢 -5.82 kB 🟢 -5.13 kB
assets/WidgetImageCrop-IdHlNd9A.js (new) 23.3 kB 🔴 +23.3 kB 🔴 +5.83 kB 🔴 +5.13 kB
assets/SubscriptionPanelContentWorkspace-BFUXAuLR.js (new) 22.2 kB 🔴 +22.2 kB 🔴 +5.18 kB 🔴 +4.55 kB
assets/SubscriptionPanelContentWorkspace-DwjiCTAz.js (removed) 22.2 kB 🟢 -22.2 kB 🟢 -5.18 kB 🟢 -4.56 kB
assets/SignInContent-BpB7u-qm.js (removed) 20.4 kB 🟢 -20.4 kB 🟢 -5.29 kB 🟢 -4.62 kB
assets/SignInContent-C6fZ5Lms.js (new) 20.4 kB 🔴 +20.4 kB 🔴 +5.29 kB 🔴 +4.62 kB
assets/CurrentUserPopoverWorkspace-BaVdb1NR.js (removed) 20.4 kB 🟢 -20.4 kB 🟢 -4.83 kB 🟢 -4.33 kB
assets/CurrentUserPopoverWorkspace-cAirQv_W.js (new) 20.4 kB 🔴 +20.4 kB 🔴 +4.83 kB 🔴 +4.32 kB
assets/WidgetInputNumber-C_BU2MtO.js (new) 19.1 kB 🔴 +19.1 kB 🔴 +4.84 kB 🔴 +4.29 kB
assets/WidgetInputNumber-Dd2Z6fCr.js (removed) 19.1 kB 🟢 -19.1 kB 🟢 -4.84 kB 🟢 -4.3 kB
assets/WidgetRecordAudio-1tMKpEzv.js (removed) 18.1 kB 🟢 -18.1 kB 🟢 -5.18 kB 🟢 -4.64 kB
assets/WidgetRecordAudio-wbrmkXX2.js (new) 18.1 kB 🔴 +18.1 kB 🔴 +5.18 kB 🔴 +4.64 kB
assets/Load3D-CMKxkLIM.js (new) 16.9 kB 🔴 +16.9 kB 🔴 +4.12 kB 🔴 +3.59 kB
assets/Load3D-DNovywgu.js (removed) 16.9 kB 🟢 -16.9 kB 🟢 -4.11 kB 🟢 -3.59 kB
assets/load3d-CAJuRyu0.js (removed) 15 kB 🟢 -15 kB 🟢 -4.31 kB 🟢 -3.75 kB
assets/load3d-CZO3_hUv.js (new) 15 kB 🔴 +15 kB 🔴 +4.32 kB 🔴 +3.74 kB
assets/WaveAudioPlayer-B_PE-auH.js (removed) 13.4 kB 🟢 -13.4 kB 🟢 -3.69 kB 🟢 -3.22 kB
assets/WaveAudioPlayer-BdlJIxTn.js (new) 13.4 kB 🔴 +13.4 kB 🔴 +3.69 kB 🔴 +3.22 kB
assets/WidgetCurve-B5iuGO2p.js (new) 12 kB 🔴 +12 kB 🔴 +3.85 kB 🔴 +3.49 kB
assets/WidgetCurve-jDTnX1se.js (removed) 12 kB 🟢 -12 kB 🟢 -3.85 kB 🟢 -3.48 kB
assets/TeamWorkspacesDialogContent-BkztxMTz.js (new) 11.1 kB 🔴 +11.1 kB 🔴 +3.33 kB 🔴 +2.97 kB
assets/TeamWorkspacesDialogContent-DReGzroe.js (removed) 11.1 kB 🟢 -11.1 kB 🟢 -3.33 kB 🟢 -2.96 kB
assets/AudioPreviewPlayer-CjIq2mEy.js (new) 10.8 kB 🔴 +10.8 kB 🔴 +3.13 kB 🔴 +2.81 kB
assets/AudioPreviewPlayer-CQTQuFDu.js (removed) 10.8 kB 🟢 -10.8 kB 🟢 -3.13 kB 🟢 -2.8 kB
assets/nodeTemplates-BhKy2FSQ.js (removed) 9.58 kB 🟢 -9.58 kB 🟢 -3.37 kB 🟢 -2.97 kB
assets/nodeTemplates-ClaMiFTH.js (new) 9.58 kB 🔴 +9.58 kB 🔴 +3.37 kB 🔴 +2.96 kB
assets/InviteMemberDialogContent-CxD0MKMz.js (new) 7.77 kB 🔴 +7.77 kB 🔴 +2.45 kB 🔴 +2.15 kB
assets/InviteMemberDialogContent-CxXZu3yT.js (removed) 7.77 kB 🟢 -7.77 kB 🟢 -2.45 kB 🟢 -2.14 kB
assets/Load3DConfiguration-BRiPutYB.js (new) 6.6 kB 🔴 +6.6 kB 🔴 +2.04 kB 🔴 +1.78 kB
assets/Load3DConfiguration-CBSKWcQA.js (removed) 6.6 kB 🟢 -6.6 kB 🟢 -2.04 kB 🟢 -1.78 kB
assets/onboardingCloudRoutes-0-GOxmyk.js (new) 6.53 kB 🔴 +6.53 kB 🔴 +2.04 kB 🔴 +1.75 kB
assets/onboardingCloudRoutes-DRuNBtKE.js (removed) 6.53 kB 🟢 -6.53 kB 🟢 -2.04 kB 🟢 -1.74 kB
assets/WidgetWithControl-CW5oqilr.js (removed) 5.99 kB 🟢 -5.99 kB 🟢 -2.38 kB 🟢 -2.12 kB
assets/WidgetWithControl-DHXAVi_3.js (new) 5.99 kB 🔴 +5.99 kB 🔴 +2.38 kB 🔴 +2.13 kB
assets/CreateWorkspaceDialogContent-BODZomol.js (new) 5.95 kB 🔴 +5.95 kB 🔴 +2.15 kB 🔴 +1.87 kB
assets/CreateWorkspaceDialogContent-DJ5P8kAN.js (removed) 5.95 kB 🟢 -5.95 kB 🟢 -2.15 kB 🟢 -1.87 kB
assets/FreeTierDialogContent-5ijIVzzv.js (removed) 5.82 kB 🟢 -5.82 kB 🟢 -2.05 kB 🟢 -1.81 kB
assets/FreeTierDialogContent-KA31ncgy.js (new) 5.82 kB 🔴 +5.82 kB 🔴 +2.04 kB 🔴 +1.8 kB
assets/EditWorkspaceDialogContent-JlJPm9hP.js (removed) 5.75 kB 🟢 -5.75 kB 🟢 -2.11 kB 🟢 -1.84 kB
assets/EditWorkspaceDialogContent-OIgHq8JR.js (new) 5.75 kB 🔴 +5.75 kB 🔴 +2.11 kB 🔴 +1.84 kB
assets/WidgetTextarea-Buh57Dwf.js (new) 5.53 kB 🔴 +5.53 kB 🔴 +2.17 kB 🔴 +1.93 kB
assets/WidgetTextarea-NSsjo9Sq.js (removed) 5.53 kB 🟢 -5.53 kB 🟢 -2.17 kB 🟢 -1.93 kB
assets/Preview3d-DJL686US.js (removed) 5.36 kB 🟢 -5.36 kB 🟢 -1.79 kB 🟢 -1.56 kB
assets/Preview3d-xDGM4q7W.js (new) 5.36 kB 🔴 +5.36 kB 🔴 +1.79 kB 🔴 +1.56 kB
assets/ValueControlPopover-CFvLybZC.js (removed) 5.33 kB 🟢 -5.33 kB 🟢 -1.93 kB 🟢 -1.72 kB
assets/ValueControlPopover-DpCJqx5-.js (new) 5.33 kB 🔴 +5.33 kB 🔴 +1.93 kB 🔴 +1.72 kB
assets/CancelSubscriptionDialogContent-D5DBGRfZ.js (new) 5.22 kB 🔴 +5.22 kB 🔴 +1.95 kB 🔴 +1.7 kB
assets/CancelSubscriptionDialogContent-LsWJ4C2h.js (removed) 5.22 kB 🟢 -5.22 kB 🟢 -1.95 kB 🟢 -1.7 kB
assets/DeleteWorkspaceDialogContent-By3D9vDY.js (new) 4.65 kB 🔴 +4.65 kB 🔴 +1.79 kB 🔴 +1.55 kB
assets/DeleteWorkspaceDialogContent-t-IuJ9WQ.js (removed) 4.65 kB 🟢 -4.65 kB 🟢 -1.79 kB 🟢 -1.55 kB
assets/missingModelDownload-C_WqyjKg.js (removed) 4.5 kB 🟢 -4.5 kB 🟢 -1.76 kB 🟢 -1.55 kB
assets/missingModelDownload-D_eRpsKQ.js (new) 4.5 kB 🔴 +4.5 kB 🔴 +1.76 kB 🔴 +1.55 kB
assets/LeaveWorkspaceDialogContent-DzpUHj_o.js (removed) 4.48 kB 🟢 -4.48 kB 🟢 -1.74 kB 🟢 -1.51 kB
assets/LeaveWorkspaceDialogContent-tkXMiUOJ.js (new) 4.48 kB 🔴 +4.48 kB 🔴 +1.74 kB 🔴 +1.5 kB
assets/RemoveMemberDialogContent-BR5Zm8mv.js (removed) 4.46 kB 🟢 -4.46 kB 🟢 -1.69 kB 🟢 -1.47 kB
assets/RemoveMemberDialogContent-CguqVEnQ.js (new) 4.46 kB 🔴 +4.46 kB 🔴 +1.69 kB 🔴 +1.47 kB
assets/tierBenefits-BTS5CrLR.js (new) 4.45 kB 🔴 +4.45 kB 🔴 +1.58 kB 🔴 +1.37 kB
assets/tierBenefits-CpZ8iK5O.js (removed) 4.45 kB 🟢 -4.45 kB 🟢 -1.58 kB 🟢 -1.36 kB
assets/RevokeInviteDialogContent-C1NtJKWA.js (new) 4.37 kB 🔴 +4.37 kB 🔴 +1.7 kB 🔴 +1.48 kB
assets/RevokeInviteDialogContent-Du_Il-Ir.js (removed) 4.37 kB 🟢 -4.37 kB 🟢 -1.7 kB 🟢 -1.48 kB
assets/InviteMemberUpsellDialogContent-CulkQDIp.js (removed) 4.27 kB 🟢 -4.27 kB 🟢 -1.56 kB 🟢 -1.37 kB
assets/InviteMemberUpsellDialogContent-D_elUXQw.js (new) 4.27 kB 🔴 +4.27 kB 🔴 +1.56 kB 🔴 +1.37 kB
assets/cloudSessionCookie-BEunmftL.js (removed) 4.12 kB 🟢 -4.12 kB 🟢 -1.48 kB 🟢 -1.29 kB
assets/cloudSessionCookie-CS-aAKII.js (new) 4.12 kB 🔴 +4.12 kB 🔴 +1.49 kB 🔴 +1.29 kB
assets/saveMesh-nmoJkXQ5.js (new) 3.92 kB 🔴 +3.92 kB 🔴 +1.68 kB 🔴 +1.49 kB
assets/saveMesh-t5CFrXgD.js (removed) 3.92 kB 🟢 -3.92 kB 🟢 -1.68 kB 🟢 -1.48 kB
assets/Media3DTop-BpqGAjrx.js (new) 3.85 kB 🔴 +3.85 kB 🔴 +1.62 kB 🔴 +1.42 kB
assets/Media3DTop-CCW5rMiz.js (removed) 3.85 kB 🟢 -3.85 kB 🟢 -1.62 kB 🟢 -1.43 kB
assets/GlobalToast-BUe-3gDl.js (new) 3.05 kB 🔴 +3.05 kB 🔴 +1.26 kB 🔴 +1.07 kB
assets/GlobalToast-SIDrUzfL.js (removed) 3.05 kB 🟢 -3.05 kB 🟢 -1.26 kB 🟢 -1.11 kB
assets/SubscribeToRun-DA7zpIaa.js (removed) 2.13 kB 🟢 -2.13 kB 🟢 -982 B 🟢 -879 B
assets/SubscribeToRun-DNxv7AZT.js (new) 2.13 kB 🔴 +2.13 kB 🔴 +983 B 🔴 +882 B
assets/MediaAudioTop-B1GpHXCM.js (new) 2.02 kB 🔴 +2.02 kB 🔴 +981 B 🔴 +878 B
assets/MediaAudioTop-zEvM3Eps.js (removed) 2.02 kB 🟢 -2.02 kB 🟢 -979 B 🟢 -836 B
assets/CloudRunButtonWrapper-BG6blDTM.js (removed) 1.99 kB 🟢 -1.99 kB 🟢 -910 B 🟢 -804 B
assets/CloudRunButtonWrapper-Bu0gU9dO.js (new) 1.99 kB 🔴 +1.99 kB 🔴 +912 B 🔴 +806 B
assets/graphHasMissingNodes-B3KdtH1L.js (new) 1.83 kB 🔴 +1.83 kB 🔴 +862 B 🔴 +753 B
assets/graphHasMissingNodes-CylitUC9.js (removed) 1.83 kB 🟢 -1.83 kB 🟢 -860 B 🟢 -753 B
assets/cloudBadges-BtaedYf-.js (new) 1.77 kB 🔴 +1.77 kB 🔴 +890 B 🔴 +770 B
assets/cloudBadges-CujbKotG.js (removed) 1.77 kB 🟢 -1.77 kB 🟢 -889 B 🟢 -785 B
assets/cloudSubscription-4cVU0nUw.js (removed) 1.68 kB 🟢 -1.68 kB 🟢 -812 B 🟢 -708 B
assets/cloudSubscription-DnqxH8nQ.js (new) 1.68 kB 🔴 +1.68 kB 🔴 +814 B 🔴 +714 B
assets/previousFullPath-BM8xFuo1.js (new) 1.53 kB 🔴 +1.53 kB 🔴 +695 B 🔴 +601 B
assets/previousFullPath-nBr5sQIT.js (removed) 1.53 kB 🟢 -1.53 kB 🟢 -693 B 🟢 -599 B
assets/Load3D-Cr3bC7JX.js (removed) 1.34 kB 🟢 -1.34 kB 🟢 -610 B 🟢 -542 B
assets/Load3D-CUe4hg3O.js (new) 1.34 kB 🔴 +1.34 kB 🔴 +611 B 🔴 +559 B
assets/nightlyBadges-DV_mhvYO.js (new) 1.29 kB 🔴 +1.29 kB 🔴 +657 B 🔴 +580 B
assets/nightlyBadges-lAMrBrGl.js (removed) 1.29 kB 🟢 -1.29 kB 🟢 -655 B 🟢 -581 B
assets/Load3dViewerContent-BpBdjri_.js (removed) 1.23 kB 🟢 -1.23 kB 🟢 -562 B 🟢 -498 B
assets/Load3dViewerContent-u3Rn5_I1.js (new) 1.23 kB 🔴 +1.23 kB 🔴 +564 B 🔴 +500 B
assets/SubscriptionPanelContentWorkspace-DaT9T1sk.js (new) 1.15 kB 🔴 +1.15 kB 🔴 +532 B 🔴 +463 B
assets/SubscriptionPanelContentWorkspace-DdMrUSUt.js (removed) 1.15 kB 🟢 -1.15 kB 🟢 -533 B 🟢 -468 B
assets/WidgetLegacy-B_lDrsLg.js (removed) 978 B 🟢 -978 B 🟢 -481 B 🟢 -423 B
assets/WidgetLegacy-DOPHlmcx.js (new) 978 B 🔴 +978 B 🔴 +479 B 🔴 +422 B
assets/changeTracker-BanPDNWD.js (new) 952 B 🔴 +952 B 🔴 +470 B 🔴 +412 B
assets/changeTracker-Dg1ELn7L.js (removed) 952 B 🟢 -952 B 🟢 -470 B 🟢 -412 B
assets/constants-De49Wq6b.js (new) 858 B 🔴 +858 B 🔴 +424 B 🔴 +341 B
assets/constants-CWGvtpeq.js (removed) 766 B 🟢 -766 B 🟢 -374 B 🟢 -304 B
assets/missingModelDownload-DnfEZUBE.js (removed) 267 B 🟢 -267 B 🟢 -184 B 🟢 -177 B
assets/missingModelDownload-RrLrwC3n.js (new) 267 B 🔴 +267 B 🔴 +184 B 🔴 +171 B

Status: 59 added / 59 removed / 75 unchanged

⚡ Performance Report

canvas-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 60.2 MB heap
canvas-mouse-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 56.4 MB heap
canvas-zoom-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 64.4 MB heap
dom-widget-clipping: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 46.8 MB heap
large-graph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 53.0 MB heap
large-graph-pan: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 68.6 MB heap
large-graph-zoom: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 59.1 MB heap
minimap-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 53.9 MB heap
subgraph-dom-widget-clipping: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 47.3 MB heap
subgraph-idle: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 60.1 MB heap
subgraph-mouse-sweep: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 52.5 MB heap
viewport-pan-sweep: · 60.0 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 71.8 MB heap
vue-large-graph-idle: · 58.1 avg FPS · 59.5 P5 FPS ✅ (target: ≥52) · 0ms TBT · 158.5 MB heap
vue-large-graph-pan: · 56.3 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 166.4 MB heap
workflow-execution: · 60.0 avg FPS · 59.9 P5 FPS ✅ (target: ≥52) · 0ms TBT · 46.8 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 10ms 9ms -13% z=-2.6
canvas-idle: layout count 0 0 +0%
canvas-idle: style recalc count 10 10 +0% z=-2.1
canvas-idle: task duration 406ms 356ms -12% z=-1.3
canvas-idle: script duration 29ms 20ms -29% z=-2.2
canvas-idle: TBT 0ms 0ms +0%
canvas-idle: heap used 59.9 MB 60.2 MB +0%
canvas-idle: DOM nodes 20 20 +0% z=-2.0
canvas-idle: event listeners 6 6 +0% z=-1.2
canvas-mouse-sweep: avg frame time 17ms 17ms +0% z=0.7
canvas-mouse-sweep: p95 frame time 17ms 17ms +0%
canvas-mouse-sweep: layout duration 4ms 4ms -5% z=0.9
canvas-mouse-sweep: style recalc duration 50ms 41ms -18% z=-0.5
canvas-mouse-sweep: layout count 12 12 +0%
canvas-mouse-sweep: style recalc count 84 81 -4% z=1.0
canvas-mouse-sweep: task duration 1082ms 921ms -15% z=1.0
canvas-mouse-sweep: script duration 145ms 121ms -16% z=-2.2
canvas-mouse-sweep: TBT 0ms 0ms +0%
canvas-mouse-sweep: heap used 57.0 MB 56.4 MB -1%
canvas-mouse-sweep: DOM nodes 67 65 -3% z=1.0
canvas-mouse-sweep: event listeners 6 6 +0% z=-0.6
canvas-zoom-sweep: avg frame time 17ms 17ms +0% z=0.5
canvas-zoom-sweep: p95 frame time 17ms 17ms +0%
canvas-zoom-sweep: layout duration 1ms 1ms -12% z=-1.3
canvas-zoom-sweep: style recalc duration 20ms 17ms -15% z=-1.4
canvas-zoom-sweep: layout count 6 6 +0%
canvas-zoom-sweep: style recalc count 33 31 -6% z=-0.6
canvas-zoom-sweep: task duration 378ms 303ms -20% z=-1.0
canvas-zoom-sweep: script duration 30ms 24ms -22% z=-1.2
canvas-zoom-sweep: TBT 0ms 0ms +0%
canvas-zoom-sweep: heap used 64.1 MB 64.4 MB +0%
canvas-zoom-sweep: DOM nodes 79 79 +0% z=-0.3
canvas-zoom-sweep: event listeners 19 19 +0% z=-0.9
dom-widget-clipping: avg frame time 17ms 17ms +0% z=0.1
dom-widget-clipping: p95 frame time 17ms 17ms +0%
dom-widget-clipping: layout duration 0ms 0ms +0%
dom-widget-clipping: style recalc duration 10ms 9ms -1% z=-0.5
dom-widget-clipping: layout count 0 0 +0%
dom-widget-clipping: style recalc count 13 13 +0% z=-0.2
dom-widget-clipping: task duration 373ms 335ms -10% z=-1.8
dom-widget-clipping: script duration 70ms 64ms -8% z=-1.1
dom-widget-clipping: TBT 0ms 0ms +0%
dom-widget-clipping: heap used 46.8 MB 46.8 MB -0%
dom-widget-clipping: DOM nodes 22 22 +0% z=-0.1
dom-widget-clipping: event listeners 2 2 +0% variance too high
large-graph-idle: avg frame time 17ms 17ms +0% z=-0.2
large-graph-idle: p95 frame time 17ms 17ms -1%
large-graph-idle: layout duration 0ms 0ms +0%
large-graph-idle: style recalc duration 12ms 9ms -24% z=-3.0
large-graph-idle: layout count 0 0 +0%
large-graph-idle: style recalc count 11 11 +0% z=-1.9
large-graph-idle: task duration 697ms 526ms -25% z=-0.3
large-graph-idle: script duration 124ms 87ms -30% z=-1.5
large-graph-idle: TBT 0ms 0ms +0%
large-graph-idle: heap used 51.9 MB 53.0 MB +2%
large-graph-idle: DOM nodes -256 -258 +1% z=-312.8
large-graph-idle: event listeners -123 -125 +2% z=-24.8
large-graph-pan: avg frame time 17ms 17ms +0% z=1.3
large-graph-pan: p95 frame time 17ms 17ms -0%
large-graph-pan: layout duration 0ms 0ms +0%
large-graph-pan: style recalc duration 17ms 18ms +8% z=0.7
large-graph-pan: layout count 0 0 +0%
large-graph-pan: style recalc count 69 70 +1% z=0.7
large-graph-pan: task duration 1261ms 1041ms -17% z=-0.9
large-graph-pan: script duration 456ms 372ms -18% z=-1.8
large-graph-pan: TBT 0ms 0ms +0%
large-graph-pan: heap used 69.6 MB 68.6 MB -1%
large-graph-pan: DOM nodes -262 -260 -1% z=-169.1
large-graph-pan: event listeners -127 -125 -2% z=-156.9
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 +1%
large-graph-zoom: style recalc duration 17ms 16ms -6%
large-graph-zoom: layout count 60 60 +0%
large-graph-zoom: style recalc count 66 66 +0%
large-graph-zoom: task duration 1408ms 1283ms -9%
large-graph-zoom: script duration 522ms 477ms -9%
large-graph-zoom: TBT 0ms 0ms +0%
large-graph-zoom: heap used 61.8 MB 59.1 MB -4%
large-graph-zoom: DOM nodes -267 -265 -1%
large-graph-zoom: event listeners -123 -123 +0%
minimap-idle: avg frame time 17ms 17ms +0% z=-0.9
minimap-idle: p95 frame time 17ms 17ms -1%
minimap-idle: layout duration 0ms 0ms +0%
minimap-idle: style recalc duration 9ms 7ms -19% z=-2.5
minimap-idle: layout count 0 0 +0%
minimap-idle: style recalc count 9 9 +0% z=-0.8
minimap-idle: task duration 704ms 521ms -26% z=-0.1
minimap-idle: script duration 119ms 81ms -31% z=-1.7
minimap-idle: TBT 0ms 0ms +0%
minimap-idle: heap used 54.0 MB 53.9 MB -0%
minimap-idle: DOM nodes -261 -260 -0% z=-204.1
minimap-idle: event listeners -127 -125 -2% z=-196.2
subgraph-dom-widget-clipping: avg frame time 17ms 17ms -0% z=-0.9
subgraph-dom-widget-clipping: p95 frame time 17ms 17ms +0%
subgraph-dom-widget-clipping: layout duration 0ms 0ms +0%
subgraph-dom-widget-clipping: style recalc duration 13ms 13ms -6% z=-0.2
subgraph-dom-widget-clipping: layout count 0 0 +0%
subgraph-dom-widget-clipping: style recalc count 48 49 +2% z=1.8
subgraph-dom-widget-clipping: task duration 386ms 363ms -6% z=-0.8
subgraph-dom-widget-clipping: script duration 131ms 130ms -0% z=0.3
subgraph-dom-widget-clipping: TBT 0ms 0ms +0%
subgraph-dom-widget-clipping: heap used 47.0 MB 47.3 MB +1%
subgraph-dom-widget-clipping: DOM nodes 22 23 +5% z=0.7
subgraph-dom-widget-clipping: event listeners 8 8 +0% z=-1.4
subgraph-idle: avg frame time 17ms 17ms +0% z=0.4
subgraph-idle: p95 frame time 17ms 17ms -1%
subgraph-idle: layout duration 0ms 0ms +0%
subgraph-idle: style recalc duration 10ms 9ms -14% z=-1.8
subgraph-idle: layout count 0 0 +0%
subgraph-idle: style recalc count 11 11 +0% z=0.2
subgraph-idle: task duration 418ms 334ms -20% z=-1.1
subgraph-idle: script duration 24ms 18ms -24% z=-0.7
subgraph-idle: TBT 0ms 0ms +0%
subgraph-idle: heap used 60.1 MB 60.1 MB +0%
subgraph-idle: DOM nodes 22 22 +0% z=0.1
subgraph-idle: event listeners 6 6 +0% variance too high
subgraph-mouse-sweep: avg frame time 17ms 17ms -0% z=-0.5
subgraph-mouse-sweep: p95 frame time 17ms 17ms +0%
subgraph-mouse-sweep: layout duration 4ms 5ms +9% z=0.3
subgraph-mouse-sweep: style recalc duration 46ms 43ms -7% z=0.2
subgraph-mouse-sweep: layout count 16 16 +0%
subgraph-mouse-sweep: style recalc count 84 83 -1% z=1.5
subgraph-mouse-sweep: task duration 944ms 857ms -9% z=1.3
subgraph-mouse-sweep: script duration 107ms 96ms -11% z=-0.8
subgraph-mouse-sweep: TBT 0ms 0ms +0%
subgraph-mouse-sweep: heap used 52.4 MB 52.5 MB +0%
subgraph-mouse-sweep: DOM nodes 72 72 +0% z=2.3
subgraph-mouse-sweep: event listeners 6 6 +0% variance too high
viewport-pan-sweep: avg frame time 17ms 17ms -0%
viewport-pan-sweep: p95 frame time 17ms 17ms +0%
viewport-pan-sweep: layout duration 0ms 0ms +0%
viewport-pan-sweep: style recalc duration 50ms 46ms -7%
viewport-pan-sweep: layout count 0 0 +0%
viewport-pan-sweep: style recalc count 252 250 -1%
viewport-pan-sweep: task duration 4020ms 3437ms -15%
viewport-pan-sweep: script duration 1333ms 1158ms -13%
viewport-pan-sweep: TBT 0ms 0ms +0%
viewport-pan-sweep: heap used 73.6 MB 71.8 MB -2%
viewport-pan-sweep: DOM nodes -257 -257 +0%
viewport-pan-sweep: event listeners -111 -111 +0%
vue-large-graph-idle: avg frame time 17ms 17ms -0%
vue-large-graph-idle: p95 frame time 17ms 17ms +0%
vue-large-graph-idle: layout duration 0ms 0ms +0%
vue-large-graph-idle: style recalc duration 0ms 0ms +0%
vue-large-graph-idle: layout count 0 0 +0%
vue-large-graph-idle: style recalc count 0 0 +0%
vue-large-graph-idle: task duration 12596ms 12688ms +1%
vue-large-graph-idle: script duration 639ms 559ms -13%
vue-large-graph-idle: TBT 0ms 0ms +0%
vue-large-graph-idle: heap used 157.9 MB 158.5 MB +0%
vue-large-graph-idle: DOM nodes -8331 -8331 +0%
vue-large-graph-idle: event listeners -16464 -16464 +0%
vue-large-graph-pan: avg frame time 17ms 18ms +3%
vue-large-graph-pan: p95 frame time 17ms 17ms -1%
vue-large-graph-pan: layout duration 0ms 0ms +0%
vue-large-graph-pan: style recalc duration 14ms 14ms +3%
vue-large-graph-pan: layout count 0 0 +0%
vue-large-graph-pan: style recalc count 65 67 +3%
vue-large-graph-pan: task duration 14325ms 14680ms +2%
vue-large-graph-pan: script duration 879ms 825ms -6%
vue-large-graph-pan: TBT 0ms 0ms +0%
vue-large-graph-pan: heap used 165.7 MB 166.4 MB +0%
vue-large-graph-pan: DOM nodes -8333 -8331 -0%
vue-large-graph-pan: event listeners -16458 -16458 +0%
workflow-execution: avg frame time 17ms 17ms +0% z=0.6
workflow-execution: p95 frame time 17ms 17ms -1%
workflow-execution: layout duration 1ms 1ms +29% z=-0.6
workflow-execution: style recalc duration 24ms 24ms +0% z=-0.3
workflow-execution: layout count 4 5 +25% z=0.1
workflow-execution: style recalc count 18 18 +0% z=0.1
workflow-execution: task duration 123ms 118ms -4% z=-0.4
workflow-execution: script duration 26ms 24ms -9% z=-1.7
workflow-execution: TBT 0ms 0ms +0%
workflow-execution: heap used 46.5 MB 46.8 MB +1%
workflow-execution: DOM nodes 159 158 -1% z=-0.4
workflow-execution: event listeners 71 71 +0% z=4.4
Historical variance (last 15 runs)
Metric μ σ CV
canvas-idle: avg frame time 17ms 0ms 0.0%
canvas-idle: layout duration 0ms 0ms 0.0%
canvas-idle: style recalc duration 11ms 1ms 8.2%
canvas-idle: layout count 0 0 0.0%
canvas-idle: style recalc count 11 1 5.0%
canvas-idle: task duration 395ms 31ms 7.9%
canvas-idle: script duration 25ms 2ms 8.8%
canvas-idle: TBT 0ms 0ms 0.0%
canvas-idle: DOM nodes 23 1 5.6%
canvas-idle: event listeners 12 5 40.9%
canvas-mouse-sweep: avg frame time 17ms 0ms 0.0%
canvas-mouse-sweep: layout duration 4ms 0ms 5.4%
canvas-mouse-sweep: style recalc duration 43ms 3ms 7.4%
canvas-mouse-sweep: layout count 12 0 0.0%
canvas-mouse-sweep: style recalc count 79 2 3.0%
canvas-mouse-sweep: task duration 865ms 58ms 6.7%
canvas-mouse-sweep: script duration 136ms 6ms 4.8%
canvas-mouse-sweep: TBT 0ms 0ms 0.0%
canvas-mouse-sweep: DOM nodes 62 3 4.2%
canvas-mouse-sweep: event listeners 8 4 49.4%
canvas-zoom-sweep: avg frame time 17ms 0ms 0.0%
canvas-zoom-sweep: layout duration 1ms 0ms 7.0%
canvas-zoom-sweep: style recalc duration 19ms 2ms 8.0%
canvas-zoom-sweep: layout count 6 0 0.0%
canvas-zoom-sweep: style recalc count 31 0 1.5%
canvas-zoom-sweep: task duration 327ms 23ms 7.1%
canvas-zoom-sweep: script duration 27ms 3ms 11.1%
canvas-zoom-sweep: TBT 0ms 0ms 0.0%
canvas-zoom-sweep: DOM nodes 79 1 1.0%
canvas-zoom-sweep: event listeners 24 5 21.8%
dom-widget-clipping: avg frame time 17ms 0ms 0.0%
dom-widget-clipping: layout duration 0ms 0ms 0.0%
dom-widget-clipping: style recalc duration 10ms 1ms 8.0%
dom-widget-clipping: layout count 0 0 0.0%
dom-widget-clipping: style recalc count 13 0 3.8%
dom-widget-clipping: task duration 365ms 16ms 4.5%
dom-widget-clipping: script duration 68ms 3ms 4.8%
dom-widget-clipping: TBT 0ms 0ms 0.0%
dom-widget-clipping: DOM nodes 22 1 6.4%
dom-widget-clipping: event listeners 8 6 81.2%
large-graph-idle: avg frame time 17ms 0ms 0.0%
large-graph-idle: layout duration 0ms 0ms 0.0%
large-graph-idle: style recalc duration 12ms 1ms 8.6%
large-graph-idle: layout count 0 0 0.0%
large-graph-idle: style recalc count 12 0 2.7%
large-graph-idle: task duration 542ms 54ms 10.0%
large-graph-idle: script duration 102ms 11ms 10.3%
large-graph-idle: TBT 0ms 0ms 0.0%
large-graph-idle: DOM nodes 25 1 3.7%
large-graph-idle: event listeners 26 6 23.2%
large-graph-pan: avg frame time 17ms 0ms 0.0%
large-graph-pan: layout duration 0ms 0ms 0.0%
large-graph-pan: style recalc duration 17ms 1ms 4.6%
large-graph-pan: layout count 0 0 0.0%
large-graph-pan: style recalc count 70 1 0.9%
large-graph-pan: task duration 1082ms 43ms 4.0%
large-graph-pan: script duration 408ms 20ms 4.8%
large-graph-pan: TBT 0ms 0ms 0.0%
large-graph-pan: DOM nodes 19 2 8.7%
large-graph-pan: event listeners 5 1 16.8%
minimap-idle: avg frame time 17ms 0ms 0.0%
minimap-idle: layout duration 0ms 0ms 0.0%
minimap-idle: style recalc duration 10ms 1ms 8.6%
minimap-idle: layout count 0 0 0.0%
minimap-idle: style recalc count 10 1 7.1%
minimap-idle: task duration 527ms 47ms 9.0%
minimap-idle: script duration 98ms 10ms 10.1%
minimap-idle: TBT 0ms 0ms 0.0%
minimap-idle: DOM nodes 19 1 7.1%
minimap-idle: event listeners 5 1 14.4%
subgraph-dom-widget-clipping: avg frame time 17ms 0ms 0.0%
subgraph-dom-widget-clipping: layout duration 0ms 0ms 0.0%
subgraph-dom-widget-clipping: style recalc duration 13ms 1ms 7.4%
subgraph-dom-widget-clipping: layout count 0 0 0.0%
subgraph-dom-widget-clipping: style recalc count 48 1 1.2%
subgraph-dom-widget-clipping: task duration 378ms 18ms 4.9%
subgraph-dom-widget-clipping: script duration 128ms 6ms 4.9%
subgraph-dom-widget-clipping: TBT 0ms 0ms 0.0%
subgraph-dom-widget-clipping: DOM nodes 22 1 5.0%
subgraph-dom-widget-clipping: event listeners 16 6 36.0%
subgraph-idle: avg frame time 17ms 0ms 0.0%
subgraph-idle: layout duration 0ms 0ms 0.0%
subgraph-idle: style recalc duration 10ms 1ms 7.5%
subgraph-idle: layout count 0 0 0.0%
subgraph-idle: style recalc count 11 1 6.0%
subgraph-idle: task duration 370ms 31ms 8.5%
subgraph-idle: script duration 20ms 3ms 13.2%
subgraph-idle: TBT 0ms 0ms 0.0%
subgraph-idle: DOM nodes 22 1 6.9%
subgraph-idle: event listeners 10 7 64.5%
subgraph-mouse-sweep: avg frame time 17ms 0ms 0.0%
subgraph-mouse-sweep: layout duration 5ms 0ms 6.8%
subgraph-mouse-sweep: style recalc duration 42ms 3ms 7.8%
subgraph-mouse-sweep: layout count 16 0 0.0%
subgraph-mouse-sweep: style recalc count 80 2 2.4%
subgraph-mouse-sweep: task duration 766ms 69ms 9.0%
subgraph-mouse-sweep: script duration 101ms 7ms 6.5%
subgraph-mouse-sweep: TBT 0ms 0ms 0.0%
subgraph-mouse-sweep: DOM nodes 67 2 3.3%
subgraph-mouse-sweep: event listeners 8 4 52.6%
workflow-execution: avg frame time 17ms 0ms 0.0%
workflow-execution: layout duration 2ms 0ms 9.4%
workflow-execution: style recalc duration 24ms 2ms 9.1%
workflow-execution: layout count 5 1 11.0%
workflow-execution: style recalc count 18 2 11.5%
workflow-execution: task duration 123ms 11ms 8.8%
workflow-execution: script duration 29ms 3ms 10.2%
workflow-execution: TBT 0ms 0ms 0.0%
workflow-execution: DOM nodes 161 7 4.4%
workflow-execution: event listeners 52 4 8.4%
Trend (last 15 commits on main)
Metric Trend Dir Latest
canvas-idle: avg frame time ▆▃▆▁▆▃▆█▆▆▄▃▃▄▃ ➡️ 17ms
canvas-idle: p95 frame time ➡️ NaNms
canvas-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: style recalc duration ▇▇▆▆▃█▄▃▄▃▇▄▁▆▇ ➡️ 11ms
canvas-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
canvas-idle: style recalc count █▃▅▂▅▆▃▁▂▁▂▅▆▅▆ ➡️ 12
canvas-idle: task duration ▃▃▃▆▂▃▃▅▆▂█▃▁▃▃ ➡️ 391ms
canvas-idle: script duration ▄▃▅▇▂▅▃▆▇▅█▄▁▅▆ ➡️ 27ms
canvas-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-idle: heap used ➡️ NaN MB
canvas-idle: DOM nodes █▇▆▅▃▇▃▁▂▂▅▆▆▆▇ ➡️ 24
canvas-idle: event listeners ▅█▅▄▁▅▁▁▁▄▅▅▁▅▄ 📉 11
canvas-mouse-sweep: avg frame time ▆█▆▃▁▃▁▆▆▁▃▆▆▃▃ ➡️ 17ms
canvas-mouse-sweep: p95 frame time ➡️ NaNms
canvas-mouse-sweep: layout duration ▁▃▂▄▁▂▁▃▆▂█▇▆▄▃ ➡️ 4ms
canvas-mouse-sweep: style recalc duration ▄▄▂▄▁▂▃▃▅▄█▆▂▄▄ ➡️ 43ms
canvas-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 12
canvas-mouse-sweep: style recalc count █▅▄▃▂▂▁▄▄▅▆▅▂▇▄ ➡️ 79
canvas-mouse-sweep: task duration █▆▄▂▂▃▂▄▄▅█▆▁▆▄ ➡️ 868ms
canvas-mouse-sweep: script duration ▄▅▄▆▄▆▆▆▅▅█▆▁▅▆ ➡️ 139ms
canvas-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-mouse-sweep: heap used ➡️ NaN MB
canvas-mouse-sweep: DOM nodes █▅▃▃▁▂▂▃▂▄▆▅▃▅▅ ➡️ 64
canvas-mouse-sweep: event listeners █▁▁▁▁▁▇▁▁▁██▇▁█ 📈 13
canvas-zoom-sweep: avg frame time ▅▅█▄▅▁▁▁▅▁▁▅▄▅▁ ➡️ 17ms
canvas-zoom-sweep: p95 frame time ➡️ NaNms
canvas-zoom-sweep: layout duration ▆▅▅▄▁▁█▅▃▅▇▆▁▂▆ ➡️ 1ms
canvas-zoom-sweep: style recalc duration ▆▅▄▆▅▃█▆▇▅▇▄▁▃▅ ➡️ 20ms
canvas-zoom-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 6
canvas-zoom-sweep: style recalc count ▁▁▃▄▆▃▆█▄▄▆▁▆▁▆ ➡️ 32
canvas-zoom-sweep: task duration ▄▂▁▇▂▂▄▅▆▃█▄▁▁▅ ➡️ 338ms
canvas-zoom-sweep: script duration ▃▃▂▇▂▂▅▇▆▅█▄▁▂▆ ➡️ 30ms
canvas-zoom-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
canvas-zoom-sweep: heap used ➡️ NaN MB
canvas-zoom-sweep: DOM nodes ▄▃▁▅█▁▃▆▄▅▅▃▃▄▃ ➡️ 79
canvas-zoom-sweep: event listeners ▁▁▂▅█▂▁▅▁▅▅▄▁▅▁ ➡️ 19
dom-widget-clipping: avg frame time ▂▄▅▅▂▄█▇▅▇▇▅▅▁▇ ➡️ 17ms
dom-widget-clipping: p95 frame time ➡️ NaNms
dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: style recalc duration ▆▆▂▆▄▃██▄▁▆▇▆▃▅ ➡️ 10ms
dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
dom-widget-clipping: style recalc count ▇█▅█▅▄█▇▇▁▇▄▇▂▅ ➡️ 13
dom-widget-clipping: task duration ▃▃▁▅▄▃▅▆▅▂▇█▁▅▅ ➡️ 371ms
dom-widget-clipping: script duration ▅▄▄▆▆▅▇▇▆▃█▇▁▇▇ ➡️ 71ms
dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
dom-widget-clipping: heap used ➡️ NaN MB
dom-widget-clipping: DOM nodes ▇▇▄▇▅▄█▇▅▁▅▄▇▃▄ ➡️ 21
dom-widget-clipping: event listeners ▅▅▅▅▁▅██▁▁▁▁█▁▁ 📉 2
large-graph-idle: avg frame time ▅▅▅▅▅▂▁▂▄▅▄▂▂▅█ ➡️ 17ms
large-graph-idle: p95 frame time ➡️ NaNms
large-graph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: style recalc duration ▅▅▅▆▄▅▃▄▅▅▆█▁▄▆ ➡️ 13ms
large-graph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-idle: style recalc count █▆█▃▃▁▃▆▃▆▆▃▆██ ➡️ 12
large-graph-idle: task duration ▂▃▂▆▂▃▃▇▅▃██▁▂▅ ➡️ 569ms
large-graph-idle: script duration ▄▅▄▆▄▅▅▇▆▅█▆▁▃▆ ➡️ 110ms
large-graph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-idle: heap used ➡️ NaN MB
large-graph-idle: DOM nodes ▆█▅▂▅▃▁▂▃▅▅▆▂▆▅ ➡️ 25
large-graph-idle: event listeners ███▇██▄▁▄▇▇█▂█▇ ➡️ 29
large-graph-pan: avg frame time ▆▃▃▆█▃▁█▆▆▆▆█▁▆ ➡️ 17ms
large-graph-pan: p95 frame time ➡️ NaNms
large-graph-pan: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: style recalc duration ▃▂▄▄▁▅▂▂▁▄▄█▃▁▂ ➡️ 17ms
large-graph-pan: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
large-graph-pan: style recalc count ▆▃█▂▃▂▂▂▁▇▅▃█▆▃ ➡️ 69
large-graph-pan: task duration ▄▃▄▆▄▄▄▆▄▄█▆▁▂▅ ➡️ 1100ms
large-graph-pan: script duration ▅▄▅▆▆▅▄▆▄▅█▄▁▄▅ ➡️ 413ms
large-graph-pan: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
large-graph-pan: heap used ➡️ NaN MB
large-graph-pan: DOM nodes ▅▃▆▂▄▁▃▁▁▅▁▂█▅▂ ➡️ 18
large-graph-pan: event listeners █▆█▁▁▆▁▁▃▆▁▃██▃ ➡️ 5
minimap-idle: avg frame time ▃▆▆▃█▁█▆▆▃▃▆█▆█ ➡️ 17ms
minimap-idle: p95 frame time ➡️ NaNms
minimap-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: style recalc duration ▄█▁█▅▅█▅▅▃▅▁▁▄▆ ➡️ 10ms
minimap-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
minimap-idle: style recalc count ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 9
minimap-idle: task duration ▃▄▁▅▁▃▄▅▇▃█▅▁▁▅ ➡️ 547ms
minimap-idle: script duration ▄▆▃▇▃▅▆▆▇▅█▅▁▃▆ ➡️ 106ms
minimap-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
minimap-idle: heap used ➡️ NaN MB
minimap-idle: DOM nodes ▃▅▂▄█▃▆▁▂▅▂▁▅▆▃ ➡️ 19
minimap-idle: event listeners ▃▃▆▁▁▁▃▁▁▆▁▃█▆▁ ➡️ 4
subgraph-dom-widget-clipping: avg frame time ▅▄▄▄▄▄█▄▄▄▃▁▆▃▃ ➡️ 17ms
subgraph-dom-widget-clipping: p95 frame time ➡️ NaNms
subgraph-dom-widget-clipping: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: style recalc duration ▂▄▃▅▅▃▂▅▇▃▄█▁▄▆ ➡️ 14ms
subgraph-dom-widget-clipping: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-dom-widget-clipping: style recalc count ▇█▆▃▆▃▁▆█▇▃▆▇█▅ ➡️ 48
subgraph-dom-widget-clipping: task duration ▂▃▃▆▅▅▂▅█▂▆█▁▂▇ ➡️ 398ms
subgraph-dom-widget-clipping: script duration ▃▃▃▄▅▅▂▄█▂▅▇▁▂▅ ➡️ 131ms
subgraph-dom-widget-clipping: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-dom-widget-clipping: heap used ➡️ NaN MB
subgraph-dom-widget-clipping: DOM nodes ▅▇▅▂▅▂▁▅▅▅▁▇▅█▄ ➡️ 22
subgraph-dom-widget-clipping: event listeners ▅▅▅▂▅▁▅██▁▁█▅█▅ 📈 16
subgraph-idle: avg frame time ▆▆█▁▆▃▆▆▆▃▆▁▃▆█ ➡️ 17ms
subgraph-idle: p95 frame time ➡️ NaNms
subgraph-idle: layout duration ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: style recalc duration ▁▇▃▆▂▄▂▃▃▆▆▄▃▇█ ➡️ 12ms
subgraph-idle: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0
subgraph-idle: style recalc count ▃▆▃▃▂▅▁▂▁▆▃▃██▇ ➡️ 12
subgraph-idle: task duration ▁▃▁▇▁▁▃▆▅▂█▅▁▁▄ ➡️ 378ms
subgraph-idle: script duration ▁▃▂▇▁▂▃▇▆▂█▅▂▁▅ ➡️ 22ms
subgraph-idle: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-idle: heap used ➡️ NaN MB
subgraph-idle: DOM nodes ▃▅▃▂▁▄▁▂▁▅▃▂▇█▇ ➡️ 24
subgraph-idle: event listeners ▁▅▁▁▁▁▁▁▁▅▄▁███ 📈 21
subgraph-mouse-sweep: avg frame time ▅▄▁▃▃▄▆▄▆▃▃█▁▃▃ ➡️ 17ms
subgraph-mouse-sweep: p95 frame time ➡️ NaNms
subgraph-mouse-sweep: layout duration ▁▄▄▄▃▃▅▅▅▂█▇▂▃▆ ➡️ 5ms
subgraph-mouse-sweep: style recalc duration ▃▂▄▅▂▃▄▅█▃█▆▁▂▅ ➡️ 43ms
subgraph-mouse-sweep: layout count ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 16
subgraph-mouse-sweep: style recalc count ▅▂▅▅▁▄▃▅█▅▆▄▂▄▅ ➡️ 81
subgraph-mouse-sweep: task duration ▃▂▄▅▂▄▄▅▇▄█▆▁▃▅ ➡️ 785ms
subgraph-mouse-sweep: script duration ▄▅▄▇▅▅▆▇▆▅██▁▄▆ ➡️ 105ms
subgraph-mouse-sweep: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
subgraph-mouse-sweep: heap used ➡️ NaN MB
subgraph-mouse-sweep: DOM nodes ▅▁▄▅▁▄▃▃█▅▅▄▂▅▃ ➡️ 66
subgraph-mouse-sweep: event listeners ▇▁▂▇▁▂▂▂█▇▂▂▇▇▂ 📈 5
workflow-execution: avg frame time ▆▆▆▄▆▆▃▄▁▄█▆▅▄▆ ➡️ 17ms
workflow-execution: p95 frame time ➡️ NaNms
workflow-execution: layout duration ▁▆▁▃▂▄▃▂▃▃▅█▄▂▅ ➡️ 2ms
workflow-execution: style recalc duration ▃▇▅▇▁▅▆▇█▁██▂▄▆ ➡️ 25ms
workflow-execution: layout count ▁█▂▃▂▃▃▁▃▃▄▃▂▃▂ ➡️ 5
workflow-execution: style recalc count ▃█▅▇▁▄▅▆▅▅▅▅▄▄▂ ➡️ 15
workflow-execution: task duration ▂▅▄▅▁▄▆▆▆▁▇█▁▃▃ ➡️ 120ms
workflow-execution: script duration ▄▃▄▄▃▅▄▅▆▂▇█▁▃▄ ➡️ 29ms
workflow-execution: TBT ▄▄▄▄▄▄▄▄▄▄▄▄▄▄▄ ➡️ 0ms
workflow-execution: heap used ➡️ NaN MB
workflow-execution: DOM nodes ▂█▃▆▁▄▃▅▃█▃▃▄▃▁ ➡️ 152
workflow-execution: event listeners ▅███▁▅███▁██▅█▅ ➡️ 49
Raw data
{
  "timestamp": "2026-04-07T22:47:55.750Z",
  "gitSha": "5fe6bb1ec9ce530b2c373b44701dbb389a4c1588",
  "branch": "bl/fix-ply-asset-previewability",
  "measurements": [
    {
      "name": "canvas-idle",
      "durationMs": 2039.705999999967,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 8.528,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 362.28299999999996,
      "heapDeltaBytes": 20054724,
      "heapUsedBytes": 62721164,
      "domNodes": 17,
      "jsHeapTotalBytes": 22806528,
      "scriptDurationMs": 16.349999999999998,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "canvas-idle",
      "durationMs": 2021.9000000000165,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 8.806000000000001,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 347.66300000000007,
      "heapDeltaBytes": 21035976,
      "heapUsedBytes": 63822172,
      "domNodes": 22,
      "jsHeapTotalBytes": 22806528,
      "scriptDurationMs": 20.369999999999997,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "canvas-idle",
      "durationMs": 2044.8579999999765,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 8.39,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 355.59200000000004,
      "heapDeltaBytes": 20345796,
      "heapUsedBytes": 63140112,
      "domNodes": 20,
      "jsHeapTotalBytes": 22806528,
      "scriptDurationMs": 21.87,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 2045.803000000035,
      "styleRecalcs": 81,
      "styleRecalcDurationMs": 40.923,
      "layouts": 12,
      "layoutDurationMs": 3.4420000000000006,
      "taskDurationMs": 921.3299999999999,
      "heapDeltaBytes": 15917432,
      "heapUsedBytes": 59001796,
      "domNodes": 66,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 119.80799999999999,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66999999999998,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 1998.6599999999726,
      "styleRecalcs": 80,
      "styleRecalcDurationMs": 40.38399999999999,
      "layouts": 12,
      "layoutDurationMs": 3.797,
      "taskDurationMs": 900.257,
      "heapDeltaBytes": 16275564,
      "heapUsedBytes": 59148952,
      "domNodes": 65,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 121.33800000000001,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-mouse-sweep",
      "durationMs": 2063.1480000000693,
      "styleRecalcs": 84,
      "styleRecalcDurationMs": 47.017,
      "layouts": 12,
      "layoutDurationMs": 4.167000000000001,
      "taskDurationMs": 977.873,
      "heapDeltaBytes": 15987636,
      "heapUsedBytes": 59139204,
      "domNodes": 64,
      "jsHeapTotalBytes": 23330816,
      "scriptDurationMs": 125.70399999999998,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1739.2239999999788,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 16.924000000000003,
      "layouts": 6,
      "layoutDurationMs": 0.709,
      "taskDurationMs": 304.02699999999993,
      "heapDeltaBytes": 15431568,
      "heapUsedBytes": 67341080,
      "domNodes": 79,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 25.273999999999997,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1751.2879999999882,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 17.247000000000003,
      "layouts": 6,
      "layoutDurationMs": 0.5840000000000001,
      "taskDurationMs": 303.427,
      "heapDeltaBytes": 24723288,
      "heapUsedBytes": 67522548,
      "domNodes": 80,
      "jsHeapTotalBytes": 20709376,
      "scriptDurationMs": 23.003000000000004,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "canvas-zoom-sweep",
      "durationMs": 1735.7509999999365,
      "styleRecalcs": 31,
      "styleRecalcDurationMs": 16.604,
      "layouts": 6,
      "layoutDurationMs": 0.5740000000000001,
      "taskDurationMs": 300.315,
      "heapDeltaBytes": 24543240,
      "heapUsedBytes": 67520336,
      "domNodes": 79,
      "jsHeapTotalBytes": 20709376,
      "scriptDurationMs": 23.592000000000002,
      "eventListeners": 19,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66999999999998,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 564.0419999999722,
      "styleRecalcs": 12,
      "styleRecalcDurationMs": 9.468,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 334.53,
      "heapDeltaBytes": 6181840,
      "heapUsedBytes": 49040544,
      "domNodes": 19,
      "jsHeapTotalBytes": 13369344,
      "scriptDurationMs": 72.019,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.65999999999999,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 569.0950000000043,
      "styleRecalcs": 15,
      "styleRecalcDurationMs": 10.870999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 334.644,
      "heapDeltaBytes": 6792844,
      "heapUsedBytes": 49442076,
      "domNodes": 24,
      "jsHeapTotalBytes": 13107200,
      "scriptDurationMs": 64.165,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.669999999999998,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "dom-widget-clipping",
      "durationMs": 541.5269999999737,
      "styleRecalcs": 13,
      "styleRecalcDurationMs": 9.023,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 323.724,
      "heapDeltaBytes": 6234876,
      "heapUsedBytes": 49038316,
      "domNodes": 22,
      "jsHeapTotalBytes": 13893632,
      "scriptDurationMs": 60.85099999999999,
      "eventListeners": 2,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2023.4700000000316,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 8.263,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 525.5179999999999,
      "heapDeltaBytes": 18368828,
      "heapUsedBytes": 70317976,
      "domNodes": -257,
      "jsHeapTotalBytes": 15454208,
      "scriptDurationMs": 86.729,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2072.862999999984,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 11.678999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 534.398,
      "heapDeltaBytes": 4689180,
      "heapUsedBytes": 55600788,
      "domNodes": -258,
      "jsHeapTotalBytes": 16969728,
      "scriptDurationMs": 86.738,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.699999999999818
    },
    {
      "name": "large-graph-idle",
      "durationMs": 2040.345000000002,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 9.106,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 524.8539999999999,
      "heapDeltaBytes": 4546048,
      "heapUsedBytes": 54998152,
      "domNodes": -258,
      "jsHeapTotalBytes": 16707584,
      "scriptDurationMs": 85.482,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2145.7960000000185,
      "styleRecalcs": 71,
      "styleRecalcDurationMs": 18.972999999999995,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1041.241,
      "heapDeltaBytes": 20633452,
      "heapUsedBytes": 72438652,
      "domNodes": -256,
      "jsHeapTotalBytes": 16912384,
      "scriptDurationMs": 371.719,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2134.730999999988,
      "styleRecalcs": 70,
      "styleRecalcDurationMs": 17.858000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1002.5999999999999,
      "heapDeltaBytes": 12920412,
      "heapUsedBytes": 66004440,
      "domNodes": -260,
      "jsHeapTotalBytes": 18485248,
      "scriptDurationMs": 356.15799999999996,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66999999999998,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "large-graph-pan",
      "durationMs": 2120.0309999999263,
      "styleRecalcs": 69,
      "styleRecalcDurationMs": 16.349000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 1071.4700000000003,
      "heapDeltaBytes": 19795100,
      "heapUsedBytes": 71929532,
      "domNodes": -260,
      "jsHeapTotalBytes": 18223104,
      "scriptDurationMs": 416.571,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3163.5180000000105,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 18.641000000000005,
      "layouts": 60,
      "layoutDurationMs": 8.325,
      "taskDurationMs": 1308.184,
      "heapDeltaBytes": 8440312,
      "heapUsedBytes": 64075032,
      "domNodes": -264,
      "jsHeapTotalBytes": 17813504,
      "scriptDurationMs": 484.01300000000003,
      "eventListeners": -127,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3118.0229999999938,
      "styleRecalcs": 65,
      "styleRecalcDurationMs": 16.279000000000003,
      "layouts": 60,
      "layoutDurationMs": 7.866999999999999,
      "taskDurationMs": 1261.4650000000001,
      "heapDeltaBytes": 1308392,
      "heapUsedBytes": 55850268,
      "domNodes": -265,
      "jsHeapTotalBytes": 18280448,
      "scriptDurationMs": 460.48,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "large-graph-zoom",
      "durationMs": 3169.583999999986,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 16.427999999999997,
      "layouts": 60,
      "layoutDurationMs": 7.917000000000001,
      "taskDurationMs": 1282.504,
      "heapDeltaBytes": 7322476,
      "heapUsedBytes": 61950356,
      "domNodes": -265,
      "jsHeapTotalBytes": 17231872,
      "scriptDurationMs": 476.62699999999995,
      "eventListeners": -123,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2020.5030000000193,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 7.298000000000003,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 522.9019999999999,
      "heapDeltaBytes": 2737124,
      "heapUsedBytes": 56844604,
      "domNodes": -260,
      "jsHeapTotalBytes": 16707584,
      "scriptDurationMs": 80.59899999999999,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2017.973999999981,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 8.021999999999998,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 511.5109999999999,
      "heapDeltaBytes": 1782488,
      "heapUsedBytes": 55025992,
      "domNodes": -259,
      "jsHeapTotalBytes": 15134720,
      "scriptDurationMs": 81.49300000000001,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "minimap-idle",
      "durationMs": 2008.181000000036,
      "styleRecalcs": 9,
      "styleRecalcDurationMs": 7.486,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 521.4909999999999,
      "heapDeltaBytes": 4296100,
      "heapUsedBytes": 56520840,
      "domNodes": -260,
      "jsHeapTotalBytes": 16445440,
      "scriptDurationMs": 82.05600000000001,
      "eventListeners": -125,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 556.4379999999574,
      "styleRecalcs": 49,
      "styleRecalcDurationMs": 13.096000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 368.757,
      "heapDeltaBytes": 6534036,
      "heapUsedBytes": 49612736,
      "domNodes": 23,
      "jsHeapTotalBytes": 12845056,
      "scriptDurationMs": 130.302,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.799999999999727
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 563.2699999999886,
      "styleRecalcs": 48,
      "styleRecalcDurationMs": 11.297,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 354.514,
      "heapDeltaBytes": 6487996,
      "heapUsedBytes": 49585468,
      "domNodes": 21,
      "jsHeapTotalBytes": 13369344,
      "scriptDurationMs": 124.891,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-dom-widget-clipping",
      "durationMs": 537.4089999999114,
      "styleRecalcs": 49,
      "styleRecalcDurationMs": 12.541000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 362.759,
      "heapDeltaBytes": 7069820,
      "heapUsedBytes": 49873364,
      "domNodes": 24,
      "jsHeapTotalBytes": 13369344,
      "scriptDurationMs": 132.468,
      "eventListeners": 8,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.663333333333338,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "subgraph-idle",
      "durationMs": 1998.221000000001,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 9.026000000000002,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 334.306,
      "heapDeltaBytes": 19902056,
      "heapUsedBytes": 63043552,
      "domNodes": 22,
      "jsHeapTotalBytes": 22806528,
      "scriptDurationMs": 16.538999999999998,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "subgraph-idle",
      "durationMs": 2025.6840000000125,
      "styleRecalcs": 11,
      "styleRecalcDurationMs": 9.155999999999999,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 338.12600000000003,
      "heapDeltaBytes": 19922640,
      "heapUsedBytes": 63127208,
      "domNodes": 22,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 18.86,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-idle",
      "durationMs": 1998.7199999999348,
      "styleRecalcs": 10,
      "styleRecalcDurationMs": 8.476,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 332.241,
      "heapDeltaBytes": 19954492,
      "heapUsedBytes": 63036736,
      "domNodes": 20,
      "jsHeapTotalBytes": 22806528,
      "scriptDurationMs": 18.472,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1756.2179999999898,
      "styleRecalcs": 76,
      "styleRecalcDurationMs": 43.067,
      "layouts": 16,
      "layoutDurationMs": 4.997,
      "taskDurationMs": 693.797,
      "heapDeltaBytes": 11576844,
      "heapUsedBytes": 54636580,
      "domNodes": 63,
      "jsHeapTotalBytes": 22806528,
      "scriptDurationMs": 99.429,
      "eventListeners": 4,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1984.9179999999933,
      "styleRecalcs": 85,
      "styleRecalcDurationMs": 43.907000000000004,
      "layouts": 16,
      "layoutDurationMs": 4.785,
      "taskDurationMs": 875.319,
      "heapDeltaBytes": 11828984,
      "heapUsedBytes": 55116872,
      "domNodes": 75,
      "jsHeapTotalBytes": 23068672,
      "scriptDurationMs": 95.54300000000002,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "subgraph-mouse-sweep",
      "durationMs": 1962.4529999999822,
      "styleRecalcs": 83,
      "styleRecalcDurationMs": 42.219,
      "layouts": 16,
      "layoutDurationMs": 4.5729999999999995,
      "taskDurationMs": 856.644,
      "heapDeltaBytes": 11953916,
      "heapUsedBytes": 55017432,
      "domNodes": 72,
      "jsHeapTotalBytes": 22544384,
      "scriptDurationMs": 95.37899999999999,
      "eventListeners": 6,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333335,
      "p95FrameDurationMs": 16.800000000000182
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8185.084000000017,
      "styleRecalcs": 252,
      "styleRecalcDurationMs": 47.794000000000004,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3530.763,
      "heapDeltaBytes": 18240412,
      "heapUsedBytes": 68939716,
      "domNodes": -257,
      "jsHeapTotalBytes": 20320256,
      "scriptDurationMs": 1170.4460000000001,
      "eventListeners": -109,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.80000000000109
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8168.885999999986,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 46.486,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3389.513,
      "heapDeltaBytes": 24230148,
      "heapUsedBytes": 75305580,
      "domNodes": -257,
      "jsHeapTotalBytes": 18485248,
      "scriptDurationMs": 1139.4959999999999,
      "eventListeners": -111,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.670000000000012,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "viewport-pan-sweep",
      "durationMs": 8180.964000000017,
      "styleRecalcs": 250,
      "styleRecalcDurationMs": 45.615,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 3436.773,
      "heapDeltaBytes": 24629580,
      "heapUsedBytes": 76068968,
      "domNodes": -258,
      "jsHeapTotalBytes": 20058112,
      "scriptDurationMs": 1158.112,
      "eventListeners": -111,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12812.846999999976,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12799.349,
      "heapDeltaBytes": -35730084,
      "heapUsedBytes": 166173212,
      "domNodes": -8331,
      "jsHeapTotalBytes": 27353088,
      "scriptDurationMs": 598.833,
      "eventListeners": -16462,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.220000000000073,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12702.46499999996,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12688.461,
      "heapDeltaBytes": -30524216,
      "heapUsedBytes": 165287476,
      "domNodes": -8331,
      "jsHeapTotalBytes": 26566656,
      "scriptDurationMs": 538.952,
      "eventListeners": -16464,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.219999999999953,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-idle",
      "durationMs": 12630.691999999954,
      "styleRecalcs": 0,
      "styleRecalcDurationMs": 0,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 12617.874,
      "heapDeltaBytes": -32105100,
      "heapUsedBytes": 166513728,
      "domNodes": -8331,
      "jsHeapTotalBytes": 27615232,
      "scriptDurationMs": 558.702,
      "eventListeners": -16468,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.223333333333358,
      "p95FrameDurationMs": 16.799999999999272
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14629.512000000033,
      "styleRecalcs": 67,
      "styleRecalcDurationMs": 14.275999999999984,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14607.264000000001,
      "heapDeltaBytes": -7675592,
      "heapUsedBytes": 186268896,
      "domNodes": -8331,
      "jsHeapTotalBytes": 25694208,
      "scriptDurationMs": 808.04,
      "eventListeners": -16460,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.77333333333336,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14703.540999999972,
      "styleRecalcs": 68,
      "styleRecalcDurationMs": 14.771000000000006,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14680.226,
      "heapDeltaBytes": -32096808,
      "heapUsedBytes": 174024576,
      "domNodes": -8331,
      "jsHeapTotalBytes": 25956352,
      "scriptDurationMs": 824.896,
      "eventListeners": -16458,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 17.220000000000073,
      "p95FrameDurationMs": 16.700000000000728
    },
    {
      "name": "vue-large-graph-pan",
      "durationMs": 14708.50900000005,
      "styleRecalcs": 66,
      "styleRecalcDurationMs": 14.023999999999981,
      "layouts": 0,
      "layoutDurationMs": 0,
      "taskDurationMs": 14688.142,
      "heapDeltaBytes": -19380736,
      "heapUsedBytes": 174476692,
      "domNodes": -8331,
      "jsHeapTotalBytes": 24907776,
      "scriptDurationMs": 834.43,
      "eventListeners": -16458,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 18.333333333333332,
      "p95FrameDurationMs": 33.29999999999927
    },
    {
      "name": "workflow-execution",
      "durationMs": 442.63999999998305,
      "styleRecalcs": 17,
      "styleRecalcDurationMs": 23.575,
      "layouts": 5,
      "layoutDurationMs": 1.5,
      "taskDurationMs": 123.05799999999998,
      "heapDeltaBytes": 4741400,
      "heapUsedBytes": 49359576,
      "domNodes": 168,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 27.550000000000004,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.666666666666668,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "workflow-execution",
      "durationMs": 445.1569999999947,
      "styleRecalcs": 18,
      "styleRecalcDurationMs": 23.477999999999998,
      "layouts": 4,
      "layoutDurationMs": 1.1260000000000001,
      "taskDurationMs": 112.10699999999999,
      "heapDeltaBytes": 4385192,
      "heapUsedBytes": 49091252,
      "domNodes": 158,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 23.798,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.66333333333332,
      "p95FrameDurationMs": 16.700000000000273
    },
    {
      "name": "workflow-execution",
      "durationMs": 438.7530000000197,
      "styleRecalcs": 20,
      "styleRecalcDurationMs": 25.119000000000003,
      "layouts": 6,
      "layoutDurationMs": 1.469,
      "taskDurationMs": 118.226,
      "heapDeltaBytes": 4455796,
      "heapUsedBytes": 48419340,
      "domNodes": 157,
      "jsHeapTotalBytes": 0,
      "scriptDurationMs": 24.07,
      "eventListeners": 71,
      "totalBlockingTimeMs": 0,
      "frameDurationMs": 16.669999999999998,
      "p95FrameDurationMs": 16.700000000000273
    }
  ]
}

Copy link
Copy Markdown
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🧹 Nitpick comments (1)
packages/shared-frontend-utils/src/formatUtil.test.ts (1)

179-183: Consider adding an explicit null case for getFileExtension parity.

You already assert undefined and empty input; adding null would fully mirror the “nullable input” contract tested elsewhere in this suite.

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

In `@packages/shared-frontend-utils/src/formatUtil.test.ts` around lines 179 -
183, Add a test case asserting getFileExtension(null) returns null to mirror
existing checks for undefined and empty string; update the test block in
formatUtil.test.ts (the it(...) containing getFileExtension('README'),
getFileExtension(''), getFileExtension(undefined)) to include
expect(getFileExtension(null)).toBe(null) so null input is explicitly covered
for the getFileExtension function.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Inline comments:
In `@packages/shared-frontend-utils/src/formatUtil.ts`:
- Around line 629-637: The getFileExtension function incorrectly only splits on
'/' so Windows paths and URLs with query/hash can return wrong extensions;
update the extraction of fullFilename in getFileExtension to split on both '/'
and backslash (e.g., using a regex like /[\/\\]/), then strip any query or
fragment by removing anything after '?' or '#' (e.g., split on /[?#]/ and take
[0]) before computing dotIndex; keep the existing dotIndex check (dotIndex <= 0)
and the final slice + toLowerCase logic to return the extension.

---

Nitpick comments:
In `@packages/shared-frontend-utils/src/formatUtil.test.ts`:
- Around line 179-183: Add a test case asserting getFileExtension(null) returns
null to mirror existing checks for undefined and empty string; update the test
block in formatUtil.test.ts (the it(...) containing getFileExtension('README'),
getFileExtension(''), getFileExtension(undefined)) to include
expect(getFileExtension(null)).toBe(null) so null input is explicitly covered
for the getFileExtension function.
🪄 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: 3874b422-b71d-42c8-a5c9-e6d32b4074a2

📥 Commits

Reviewing files that changed from the base of the PR and between cc8ef09 and a1b46d7.

📒 Files selected for processing (12)
  • packages/shared-frontend-utils/package.json
  • packages/shared-frontend-utils/src/formatUtil.test.ts
  • packages/shared-frontend-utils/src/formatUtil.ts
  • packages/shared-frontend-utils/src/mediaExtensions.ts
  • src/components/queue/job/JobAssetsList.test.ts
  • src/components/sidebar/tabs/AssetsSidebarTab.vue
  • src/extensions/core/load3d/constants.ts
  • src/platform/assets/components/MediaAssetCard.vue
  • src/platform/assets/components/MediaAssetContextMenu.test.ts
  • src/platform/assets/components/MediaAssetContextMenu.vue
  • src/stores/queueStore.test.ts
  • src/stores/queueStore.ts

Comment on lines +629 to +637
export function getFileExtension(
filename: string | null | undefined
): string | null {
if (!filename) return null
const fullFilename = filename.split('/').pop() ?? filename
const dotIndex = fullFilename.lastIndexOf('.')
if (dotIndex <= 0) return null
return fullFilename.slice(dotIndex + 1).toLowerCase()
}
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

Harden extension parsing for backslash/query/hash inputs.

On Line 633, only / is treated as a separator. Inputs like Windows-style paths or URL-like strings can produce incorrect extensions and false “not previewable” results.

🔧 Proposed fix
 export function getFileExtension(
   filename: string | null | undefined
 ): string | null {
   if (!filename) return null
-  const fullFilename = filename.split('/').pop() ?? filename
-  const dotIndex = fullFilename.lastIndexOf('.')
+  const fullFilename = filename.split(/[\\/]/).pop() ?? filename
+  const cleanFilename = fullFilename.split(/[?#]/)[0]
+  const dotIndex = cleanFilename.lastIndexOf('.')
   if (dotIndex <= 0) return null
-  return fullFilename.slice(dotIndex + 1).toLowerCase()
+  return cleanFilename.slice(dotIndex + 1).toLowerCase()
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
export function getFileExtension(
filename: string | null | undefined
): string | null {
if (!filename) return null
const fullFilename = filename.split('/').pop() ?? filename
const dotIndex = fullFilename.lastIndexOf('.')
if (dotIndex <= 0) return null
return fullFilename.slice(dotIndex + 1).toLowerCase()
}
export function getFileExtension(
filename: string | null | undefined
): string | null {
if (!filename) return null
const fullFilename = filename.split(/[\\/]/).pop() ?? filename
const cleanFilename = fullFilename.split(/[?#]/)[0]
const dotIndex = cleanFilename.lastIndexOf('.')
if (dotIndex <= 0) return null
return cleanFilename.slice(dotIndex + 1).toLowerCase()
}
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@packages/shared-frontend-utils/src/formatUtil.ts` around lines 629 - 637, The
getFileExtension function incorrectly only splits on '/' so Windows paths and
URLs with query/hash can return wrong extensions; update the extraction of
fullFilename in getFileExtension to split on both '/' and backslash (e.g., using
a regex like /[\/\\]/), then strip any query or fragment by removing anything
after '?' or '#' (e.g., split on /[?#]/ and take [0]) before computing dotIndex;
keep the existing dotIndex check (dotIndex <= 0) and the final slice +
toLowerCase logic to return the extension.

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/queue/job/JobAssetsList.test.ts`:
- Around line 290-293: The test uses an undefined helper createResultItem and
the renderJobAssetsList call uses incorrect positional args; replace
createResultItem with createPreviewOutput when calling
createTaskRef(createPreviewOutput('job-1.ply', 'model')) and change the
renderJobAssetsList invocation to use the object form (e.g.
renderJobAssetsList({ jobs: [job], onViewItem })) so the helper names and call
signature match the test helpers.
🪄 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: 2bdf213f-eef7-4bb1-9c73-f69d9234f6ba

📥 Commits

Reviewing files that changed from the base of the PR and between a1b46d7 and 5699d18.

📒 Files selected for processing (2)
  • src/components/queue/job/JobAssetsList.test.ts
  • src/components/sidebar/tabs/AssetsSidebarTab.vue
✅ Files skipped from review due to trivial changes (1)
  • src/components/sidebar/tabs/AssetsSidebarTab.vue

Comment thread src/components/queue/job/JobAssetsList.test.ts Outdated
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

Caution

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

⚠️ Outside diff range comments (2)
src/components/queue/job/JobAssetsList.test.ts (1)

299-355: ⚠️ Potential issue | 🟠 Major

Please add a browser regression for this fix.

These component tests help, but the PR metadata still describes a user-visible fix: across queue/history/sidebar flows without any browser_tests/ coverage or a concrete note explaining why Playwright coverage is not practical. Please add one browser-level regression, or document the constraint in the PR description.

Based on learnings, "End-to-end regression coverage for fixes: Use only PR metadata ... Fail otherwise and request author to add Playwright regression test under browser_tests/ or explain why test is not practical."

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

In `@src/components/queue/job/JobAssetsList.test.ts` around lines 299 - 355, Add a
browser-level regression: create a Playwright test under browser_tests that
exercises the JobAssetsList UI (use the same scenarios covered in
JobAssetsList.test.ts such as clicking the icon for a completed PLY job without
a preview tile and verifying the "view" action is triggered, or hovering a
completed job without inspectable output and asserting the View button is not
shown) by rendering the component/page and interacting with the icon/row to
assert the expected behavior; if adding a browser test is impractical, update
the PR metadata to include a concrete note explaining why Playwright coverage
cannot be added for this fix and reference the unit tests (e.g.,
renderJobAssetsList, onViewItem behavior) so reviewers can accept the exception.
src/services/jobOutputCache.test.ts (1)

33-55: ⚠️ Potential issue | 🟡 Minor

Use the real filename-based inspection logic in this regression case.

createResultItem() now hard-codes supportsInspection, so the .usdz path here still passes even if ResultItemImpl.supportsInspection or isPreviewableMediaFilename() regresses. Please add at least one case that derives inspectability from the real filename + mediaType path instead of injecting the boolean.

Based on learnings, "Aim for behavioral coverage of critical and new features in unit/component tests".

Also applies to: 127-139

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

In `@src/services/jobOutputCache.test.ts` around lines 33 - 55, The test helper
createResultItem currently forces supportsInspection via Object.defineProperty
which masks regressions in ResultItemImpl.supportsInspection and
isPreviewableMediaFilename; change the helper so when supportsInspection is not
explicitly provided it is not overridden (i.e., remove or skip the
Object.defineProperty for 'supportsInspection' unless the caller passes a value)
and add at least one test case that creates an item with a .usdz filename (or
other inspectable extension) without passing supportsInspection so the code path
uses the real filename+mediaType logic in ResultItemImpl and
isPreviewableMediaFilename to determine inspectability.
🤖 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/stores/queueStore.ts`:
- Around line 233-240: The supportsInspection getter currently checks
isImage/isVideo/isAudio before the 3D filename gate, causing stale mediaType to
incorrectly enable inspection for non-previewable 3D files; change the
evaluation order in the supportsInspection getter to short-circuit the 3D case
first by returning (this.is3D && isPreviewableMediaFilename(this.filename))
before checking this.isImage, this.isVideo, or this.isAudio so non-previewable
3D files cannot fall through to the image/video/audio branches (update the
supportsInspection getter that references is3D, isImage, isVideo, isAudio,
isPreviewableMediaFilename, and filename).

---

Outside diff comments:
In `@src/components/queue/job/JobAssetsList.test.ts`:
- Around line 299-355: Add a browser-level regression: create a Playwright test
under browser_tests that exercises the JobAssetsList UI (use the same scenarios
covered in JobAssetsList.test.ts such as clicking the icon for a completed PLY
job without a preview tile and verifying the "view" action is triggered, or
hovering a completed job without inspectable output and asserting the View
button is not shown) by rendering the component/page and interacting with the
icon/row to assert the expected behavior; if adding a browser test is
impractical, update the PR metadata to include a concrete note explaining why
Playwright coverage cannot be added for this fix and reference the unit tests
(e.g., renderJobAssetsList, onViewItem behavior) so reviewers can accept the
exception.

In `@src/services/jobOutputCache.test.ts`:
- Around line 33-55: The test helper createResultItem currently forces
supportsInspection via Object.defineProperty which masks regressions in
ResultItemImpl.supportsInspection and isPreviewableMediaFilename; change the
helper so when supportsInspection is not explicitly provided it is not
overridden (i.e., remove or skip the Object.defineProperty for
'supportsInspection' unless the caller passes a value) and add at least one test
case that creates an item with a .usdz filename (or other inspectable extension)
without passing supportsInspection so the code path uses the real
filename+mediaType logic in ResultItemImpl and isPreviewableMediaFilename to
determine inspectability.
🪄 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: 7025a030-98c1-4759-8e6f-8813af3dd78c

📥 Commits

Reviewing files that changed from the base of the PR and between 5699d18 and f78b42a.

📒 Files selected for processing (12)
  • src/components/queue/QueueProgressOverlay.vue
  • src/components/queue/job/JobAssetsList.test.ts
  • src/components/queue/job/JobAssetsList.vue
  • src/components/sidebar/tabs/JobHistorySidebarTab.vue
  • src/composables/queue/useJobMenu.test.ts
  • src/composables/queue/useJobMenu.ts
  • src/composables/queue/useResultGallery.test.ts
  • src/composables/queue/useResultGallery.ts
  • src/services/jobOutputCache.test.ts
  • src/services/jobOutputCache.ts
  • src/stores/queueStore.test.ts
  • src/stores/queueStore.ts
🚧 Files skipped from review as they are similar to previous changes (1)
  • src/stores/queueStore.test.ts

Comment thread src/stores/queueStore.ts
Comment on lines +233 to +240
get supportsInspection(): boolean {
return (
this.isImage ||
this.isVideo ||
this.isAudio ||
(this.is3D && isPreviewableMediaFilename(this.filename))
)
}
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

Short-circuit 3D files before the media-type fallbacks.

A non-loadable 3D file can still return true here when the upstream mediaType is stale ('images', 'video', etc.), because isImage / isVideo / isAudio are checked before the 3D filename gate. That would bring back View affordances for formats like USDZ.

Suggested fix
  get supportsInspection(): boolean {
-    return (
-      this.isImage ||
-      this.isVideo ||
-      this.isAudio ||
-      (this.is3D && isPreviewableMediaFilename(this.filename))
-    )
+    if (this.is3D) {
+      return isPreviewableMediaFilename(this.filename)
+    }
+
+    return this.isImage || this.isVideo || this.isAudio
  }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
get supportsInspection(): boolean {
return (
this.isImage ||
this.isVideo ||
this.isAudio ||
(this.is3D && isPreviewableMediaFilename(this.filename))
)
}
get supportsInspection(): boolean {
if (this.is3D) {
return isPreviewableMediaFilename(this.filename)
}
return this.isImage || this.isVideo || this.isAudio
}
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@src/stores/queueStore.ts` around lines 233 - 240, The supportsInspection
getter currently checks isImage/isVideo/isAudio before the 3D filename gate,
causing stale mediaType to incorrectly enable inspection for non-previewable 3D
files; change the evaluation order in the supportsInspection getter to
short-circuit the 3D case first by returning (this.is3D &&
isPreviewableMediaFilename(this.filename)) before checking this.isImage,
this.isVideo, or this.isAudio so non-previewable 3D files cannot fall through to
the image/video/audio branches (update the supportsInspection getter that
references is3D, isImage, isVideo, isAudio, isPreviewableMediaFilename, and
filename).

@benceruleanlu benceruleanlu changed the title fix: distinguish 3D media from loadable previews fix: separate surfaced outputs from inspectable 3D media Apr 7, 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.

♻️ Duplicate comments (1)
packages/shared-frontend-utils/src/formatUtil.ts (1)

629-637: ⚠️ Potential issue | 🟡 Minor

Strip query/fragment suffixes before parsing the extension.

Line 633 handles Windows separators now, but model.glb?download=1 and model.glb#view still produce glb?download=1 / glb#view. That makes both getMediaTypeFromFilename() and isPreviewableMediaFilename() misclassify URL-like inputs as non-media.

🔧 Proposed fix
 export function getFileExtension(
   filename: string | null | undefined
 ): string | null {
   if (!filename) return null
   const fullFilename = filename.split(/[/\\]/).pop() ?? filename
-  const dotIndex = fullFilename.lastIndexOf('.')
+  const cleanFilename = fullFilename.split(/[?#]/)[0]
+  const dotIndex = cleanFilename.lastIndexOf('.')
   if (dotIndex <= 0) return null
-  return fullFilename.slice(dotIndex + 1).toLowerCase()
+  return cleanFilename.slice(dotIndex + 1).toLowerCase()
 }
🤖 Prompt for AI Agents
Verify each finding against the current code and only fix it if needed.

In `@packages/shared-frontend-utils/src/formatUtil.ts` around lines 629 - 637,
getFileExtension currently extracts the last path segment but doesn't remove URL
query or fragment suffixes, so inputs like "model.glb?download=1" return
"glb?download=1"; update getFileExtension to strip any query ("?") and fragment
("#") parts from the final segment before finding the last dot. Locate the
function getFileExtension and after computing fullFilename, trim anything after
the first '?' or '#' (e.g., split on /[?#]/ and take the first part) then
continue with lastIndexOf('.') and returning the lower-cased extension; this
ensures getMediaTypeFromFilename and isPreviewableMediaFilename receive a clean
filename.
🧹 Nitpick comments (1)
packages/shared-frontend-utils/src/formatUtil.ts (1)

639-654: Consider renaming this to isInspectableMediaFilename.

The implementation is stricter than the PR's retained previewable/surfaced concept because non-loadable 3D files return false here. Keeping that behavior under a previewable export blurs the new previewable-vs-inspectable split and makes future misuse easy.

As per coding guidelines, "Code must conform to style guides and use clear names for everything."

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

In `@packages/shared-frontend-utils/src/formatUtil.ts` around lines 639 - 654, The
function isPreviewableMediaFilename has a misleading name because it excludes
non-loadable 3D files; rename the exported symbol to isInspectableMediaFilename
and update its JSDoc to reflect "inspectable" semantics (strictly
loadable/inspectable assets), then update all internal references/imports
(calls, tests, and re-exports) to use isInspectableMediaFilename instead of
isPreviewableMediaFilename while preserving the existing implementation
(getFileExtension, IMAGE_EXTENSIONS, VIDEO_EXTENSIONS, AUDIO_EXTENSIONS,
isThreeDLoadableExtension) and export shape.
🤖 Prompt for all review comments with AI agents
Verify each finding against the current code and only fix it if needed.

Duplicate comments:
In `@packages/shared-frontend-utils/src/formatUtil.ts`:
- Around line 629-637: getFileExtension currently extracts the last path segment
but doesn't remove URL query or fragment suffixes, so inputs like
"model.glb?download=1" return "glb?download=1"; update getFileExtension to strip
any query ("?") and fragment ("#") parts from the final segment before finding
the last dot. Locate the function getFileExtension and after computing
fullFilename, trim anything after the first '?' or '#' (e.g., split on /[?#]/
and take the first part) then continue with lastIndexOf('.') and returning the
lower-cased extension; this ensures getMediaTypeFromFilename and
isPreviewableMediaFilename receive a clean filename.

---

Nitpick comments:
In `@packages/shared-frontend-utils/src/formatUtil.ts`:
- Around line 639-654: The function isPreviewableMediaFilename has a misleading
name because it excludes non-loadable 3D files; rename the exported symbol to
isInspectableMediaFilename and update its JSDoc to reflect "inspectable"
semantics (strictly loadable/inspectable assets), then update all internal
references/imports (calls, tests, and re-exports) to use
isInspectableMediaFilename instead of isPreviewableMediaFilename while
preserving the existing implementation (getFileExtension, IMAGE_EXTENSIONS,
VIDEO_EXTENSIONS, AUDIO_EXTENSIONS, isThreeDLoadableExtension) and export shape.

ℹ️ Review info
⚙️ Run configuration

Configuration used: Path: .coderabbit.yaml

Review profile: CHILL

Plan: Pro

Run ID: 0ae1e2da-fbc7-4ea5-9c84-59c65da1ceb1

📥 Commits

Reviewing files that changed from the base of the PR and between f78b42a and 759032c.

📒 Files selected for processing (2)
  • packages/shared-frontend-utils/src/formatUtil.test.ts
  • packages/shared-frontend-utils/src/formatUtil.ts
🚧 Files skipped from review as they are similar to previous changes (1)
  • packages/shared-frontend-utils/src/formatUtil.test.ts

@benceruleanlu benceruleanlu marked this pull request as ready for review April 7, 2026 23:06
@benceruleanlu benceruleanlu requested a review from a team April 7, 2026 23:06
@benceruleanlu benceruleanlu requested a review from jtydhr88 as a code owner April 7, 2026 23:06
@dosubot dosubot bot added the size:XL This PR changes 500-999 lines, ignoring generated files. label Apr 7, 2026
Copy link
Copy Markdown

@chatgpt-codex-connector chatgpt-codex-connector bot left a comment

Choose a reason for hiding this comment

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

💡 Codex Review

Here are some automated review suggestions for this pull request.

Reviewed commit: 759032ced2

ℹ️ About Codex in GitHub

Your team has set up Codex to review pull requests in this repo. Reviews are triggered when you

  • Open a pull request for review
  • Mark a draft as ready
  • Comment "@codex review".

If Codex has suggestions, it will comment; otherwise it will react with 👍.

Codex can also answer questions or update the PR. Try commenting "@codex address that feedback".

Comment on lines +181 to +183
if (!inspectableOutput) {
return
}
Copy link
Copy Markdown

Choose a reason for hiding this comment

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

P1 Badge Defer inspectability check until after lazy output loading

This early return prevents inspection for completed jobs whose preview_output is non-loadable (for example .usdz) but whose full outputs include inspectable media. In that case taskRef.inspectableOutput is initially empty, yet openResultGallery can still discover inspectable outputs via getInspectableOutputsForTask lazy loading when outputs_count > 1; returning here makes that path unreachable and leaves valid inspectable results inaccessible.

Useful? React with 👍 / 👎.

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

Labels

size:XL This PR changes 500-999 lines, ignoring generated files.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

1 participant