@dxos/plugin-deck 0.8.4-main.fd6878d → 0.8.4-staging.60fe92afc8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +102 -5
- package/PLUGIN.mdl +541 -0
- package/README.md +1 -1
- package/dist/lib/neutral/DeckLayout-DEURA3KR.mjs +291 -0
- package/dist/lib/neutral/DeckLayout-DEURA3KR.mjs.map +7 -0
- package/dist/lib/neutral/DeckPlugin.mjs +87 -0
- package/dist/lib/neutral/DeckPlugin.mjs.map +7 -0
- package/dist/lib/neutral/DeckPlugin.node.mjs +18 -0
- package/dist/lib/neutral/DeckPlugin.node.mjs.map +7 -0
- package/dist/lib/neutral/DeckPlugin.workerd.mjs +16 -0
- package/dist/lib/neutral/DeckPlugin.workerd.mjs.map +7 -0
- package/dist/lib/neutral/DeckSettings-W5I57OXM.mjs +27 -0
- package/dist/lib/neutral/DeckSettings-W5I57OXM.mjs.map +7 -0
- package/dist/lib/neutral/add-toast-TNB6DXWU.mjs +24 -0
- package/dist/lib/neutral/add-toast-TNB6DXWU.mjs.map +7 -0
- package/dist/lib/neutral/adjust-HNU5CCRO.mjs +93 -0
- package/dist/lib/neutral/adjust-HNU5CCRO.mjs.map +7 -0
- package/dist/lib/neutral/app-graph-builder-HMLT627T.mjs +129 -0
- package/dist/lib/neutral/app-graph-builder-HMLT627T.mjs.map +7 -0
- package/dist/lib/neutral/capabilities/index.mjs +25 -0
- package/dist/lib/neutral/capabilities/index.mjs.map +7 -0
- package/dist/lib/neutral/check-app-scheme-INSOF72J.mjs +10 -0
- package/dist/lib/neutral/chunk-324PPIZB.mjs +101 -0
- package/dist/lib/neutral/chunk-324PPIZB.mjs.map +7 -0
- package/dist/lib/neutral/chunk-BS4EOYMK.mjs +282 -0
- package/dist/lib/neutral/chunk-BS4EOYMK.mjs.map +7 -0
- package/dist/lib/neutral/chunk-GBIGQKYW.mjs +112 -0
- package/dist/lib/neutral/chunk-GBIGQKYW.mjs.map +7 -0
- package/dist/lib/neutral/chunk-GLB73Q6U.mjs +22 -0
- package/dist/lib/neutral/chunk-GLB73Q6U.mjs.map +7 -0
- package/dist/lib/neutral/chunk-J5LGTIGS.mjs +10 -0
- package/dist/lib/neutral/chunk-PYEY5SEC.mjs +37 -0
- package/dist/lib/neutral/chunk-PYEY5SEC.mjs.map +7 -0
- package/dist/lib/neutral/chunk-Q4W6B4IB.mjs +8 -0
- package/dist/lib/neutral/chunk-Q4W6B4IB.mjs.map +7 -0
- package/dist/lib/neutral/chunk-WAXJPQJI.mjs +1306 -0
- package/dist/lib/neutral/chunk-WAXJPQJI.mjs.map +7 -0
- package/dist/lib/neutral/chunk-ZYYOSX5I.mjs +69 -0
- package/dist/lib/neutral/chunk-ZYYOSX5I.mjs.map +7 -0
- package/dist/lib/neutral/close-ASKR22A6.mjs +44 -0
- package/dist/lib/neutral/close-ASKR22A6.mjs.map +7 -0
- package/dist/lib/neutral/components/index.mjs +126 -0
- package/dist/lib/neutral/components/index.mjs.map +7 -0
- package/dist/lib/neutral/containers/index.mjs +32 -0
- package/dist/lib/neutral/containers/index.mjs.map +7 -0
- package/dist/lib/neutral/hooks/index.mjs +159 -0
- package/dist/lib/neutral/hooks/index.mjs.map +7 -0
- package/dist/lib/neutral/index.mjs +38 -0
- package/dist/lib/neutral/index.mjs.map +7 -0
- package/dist/lib/neutral/meta.json +1 -0
- package/dist/lib/neutral/meta.mjs +8 -0
- package/dist/lib/neutral/meta.mjs.map +7 -0
- package/dist/lib/neutral/notification-tracker-P36322BH.mjs +182 -0
- package/dist/lib/neutral/notification-tracker-P36322BH.mjs.map +7 -0
- package/dist/lib/neutral/open-5OYNO3RT.mjs +159 -0
- package/dist/lib/neutral/open-5OYNO3RT.mjs.map +7 -0
- package/dist/lib/neutral/operation-handler-266CVMTW.mjs +13 -0
- package/dist/lib/neutral/operation-handler-266CVMTW.mjs.map +7 -0
- package/dist/lib/neutral/operations/index.mjs +8 -0
- package/dist/lib/neutral/operations/index.mjs.map +7 -0
- package/dist/lib/neutral/plugin.mjs +16 -0
- package/dist/lib/neutral/plugin.mjs.map +7 -0
- package/dist/lib/neutral/react-root-HH5DEUOG.mjs +44 -0
- package/dist/lib/neutral/react-root-HH5DEUOG.mjs.map +7 -0
- package/dist/lib/neutral/react-surface-3UVVCK3O.mjs +34 -0
- package/dist/lib/neutral/react-surface-3UVVCK3O.mjs.map +7 -0
- package/dist/lib/neutral/revert-workspace-B2QLT2C4.mjs +21 -0
- package/dist/lib/neutral/revert-workspace-B2QLT2C4.mjs.map +7 -0
- package/dist/lib/neutral/scroll-into-view-B52C3PJO.mjs +21 -0
- package/dist/lib/neutral/scroll-into-view-B52C3PJO.mjs.map +7 -0
- package/dist/lib/neutral/set-PA35ONXO.mjs +37 -0
- package/dist/lib/neutral/set-PA35ONXO.mjs.map +7 -0
- package/dist/lib/neutral/set-layout-mode-RPCCPQRB.mjs +85 -0
- package/dist/lib/neutral/set-layout-mode-RPCCPQRB.mjs.map +7 -0
- package/dist/lib/neutral/settings-EGNYUM4T.mjs +33 -0
- package/dist/lib/neutral/settings-EGNYUM4T.mjs.map +7 -0
- package/dist/lib/neutral/state-IIDXMQUO.mjs +86 -0
- package/dist/lib/neutral/state-IIDXMQUO.mjs.map +7 -0
- package/dist/lib/neutral/switch-workspace-LZF4KZXH.mjs +60 -0
- package/dist/lib/neutral/switch-workspace-LZF4KZXH.mjs.map +7 -0
- package/dist/lib/neutral/translations.mjs +63 -0
- package/dist/lib/neutral/translations.mjs.map +7 -0
- package/dist/lib/neutral/types/index.mjs +34 -0
- package/dist/lib/neutral/types/index.mjs.map +7 -0
- package/dist/lib/neutral/update-companion-YUCZZVGY.mjs +32 -0
- package/dist/lib/neutral/update-companion-YUCZZVGY.mjs.map +7 -0
- package/dist/lib/neutral/update-complementary-7FZNB55J.mjs +28 -0
- package/dist/lib/neutral/update-complementary-7FZNB55J.mjs.map +7 -0
- package/dist/lib/neutral/update-dialog-FNQTSSAP.mjs +29 -0
- package/dist/lib/neutral/update-dialog-FNQTSSAP.mjs.map +7 -0
- package/dist/lib/neutral/update-plank-size-3YW4NXEY.mjs +26 -0
- package/dist/lib/neutral/update-plank-size-3YW4NXEY.mjs.map +7 -0
- package/dist/lib/neutral/update-popover-G2VUD7E6.mjs +33 -0
- package/dist/lib/neutral/update-popover-G2VUD7E6.mjs.map +7 -0
- package/dist/lib/neutral/update-sidebar-KRHPUHUB.mjs +25 -0
- package/dist/lib/neutral/update-sidebar-KRHPUHUB.mjs.map +7 -0
- package/dist/lib/neutral/url-handler-A6HLW4RB.mjs +229 -0
- package/dist/lib/neutral/url-handler-A6HLW4RB.mjs.map +7 -0
- package/dist/types/src/DeckPlugin.d.ts +3 -1
- package/dist/types/src/DeckPlugin.d.ts.map +1 -1
- package/dist/types/src/DeckPlugin.node.d.ts +4 -0
- package/dist/types/src/DeckPlugin.node.d.ts.map +1 -0
- package/dist/types/src/DeckPlugin.test.d.ts +2 -0
- package/dist/types/src/DeckPlugin.test.d.ts.map +1 -0
- package/dist/types/src/DeckPlugin.workerd.d.ts +4 -0
- package/dist/types/src/DeckPlugin.workerd.d.ts.map +1 -0
- package/dist/types/src/capabilities/app-graph-builder.d.ts +4 -2
- package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -1
- package/dist/types/src/capabilities/check-app-scheme.d.ts +17 -2
- package/dist/types/src/capabilities/check-app-scheme.d.ts.map +1 -1
- package/dist/types/src/capabilities/index.d.ts +197 -12
- package/dist/types/src/capabilities/index.d.ts.map +1 -1
- package/dist/types/src/capabilities/notification-tracker.d.ts +13 -0
- package/dist/types/src/capabilities/notification-tracker.d.ts.map +1 -0
- package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
- package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
- package/dist/types/src/capabilities/react-root.d.ts +4 -2
- package/dist/types/src/capabilities/react-root.d.ts.map +1 -1
- package/dist/types/src/capabilities/react-surface.d.ts +3 -2
- package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
- package/dist/types/src/capabilities/settings.d.ts +5 -2
- package/dist/types/src/capabilities/settings.d.ts.map +1 -1
- package/dist/types/src/capabilities/state.d.ts +141 -51
- package/dist/types/src/capabilities/state.d.ts.map +1 -1
- package/dist/types/src/capabilities/tools.d.ts +4 -3
- package/dist/types/src/capabilities/tools.d.ts.map +1 -1
- package/dist/types/src/capabilities/url-handler.d.ts +3 -2
- package/dist/types/src/capabilities/url-handler.d.ts.map +1 -1
- package/dist/types/src/components/DeckSettings/DeckSettings.d.ts +4 -4
- package/dist/types/src/components/DeckSettings/DeckSettings.d.ts.map +1 -1
- package/dist/types/src/components/DeckSettings/DeckSettings.stories.d.ts +67 -0
- package/dist/types/src/components/DeckSettings/DeckSettings.stories.d.ts.map +1 -0
- package/dist/types/src/components/DeckSettings/index.d.ts +1 -1
- package/dist/types/src/components/DeckSettings/index.d.ts.map +1 -1
- package/dist/types/src/components/Matrix/Matrix.d.ts +41 -0
- package/dist/types/src/components/Matrix/Matrix.d.ts.map +1 -0
- package/dist/types/src/components/Matrix/Matrix.stories.d.ts +17 -0
- package/dist/types/src/components/Matrix/Matrix.stories.d.ts.map +1 -0
- package/dist/types/src/components/Matrix/index.d.ts +3 -0
- package/dist/types/src/components/Matrix/index.d.ts.map +1 -0
- package/dist/types/src/components/index.d.ts +3 -2
- package/dist/types/src/components/index.d.ts.map +1 -1
- package/dist/types/src/{components/DeckLayout → containers/Deck}/Banner.d.ts +1 -1
- package/dist/types/src/containers/Deck/Banner.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/Deck.d.ts +25 -0
- package/dist/types/src/containers/Deck/Deck.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/Deck.stories.d.ts +67 -0
- package/dist/types/src/containers/Deck/Deck.stories.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/DeckContent.d.ts +7 -0
- package/dist/types/src/containers/Deck/DeckContent.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/DeckRoot.d.ts +39 -0
- package/dist/types/src/containers/Deck/DeckRoot.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/DeckViewport.d.ts +16 -0
- package/dist/types/src/containers/Deck/DeckViewport.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/StatusBar.d.ts.map +1 -0
- package/dist/types/src/containers/Deck/index.d.ts +2 -0
- package/dist/types/src/containers/Deck/index.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/ActiveNode.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/DeckLayout.d.ts +5 -0
- package/dist/types/src/containers/DeckLayout/DeckLayout.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/DeckLayout.stories.d.ts +69 -0
- package/dist/types/src/containers/DeckLayout/DeckLayout.stories.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/Dialog.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/Fallback.d.ts +2 -0
- package/dist/types/src/containers/DeckLayout/Fallback.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/Popover.d.ts +5 -0
- package/dist/types/src/containers/DeckLayout/Popover.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/Toast.d.ts +10 -0
- package/dist/types/src/containers/DeckLayout/Toast.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/constants.d.ts.map +1 -0
- package/dist/types/src/containers/DeckLayout/index.d.ts +4 -0
- package/dist/types/src/containers/DeckLayout/index.d.ts.map +1 -0
- package/dist/types/src/containers/Plank/Plank.d.ts +17 -0
- package/dist/types/src/containers/Plank/Plank.d.ts.map +1 -0
- package/dist/types/src/containers/Plank/Plank.stories.d.ts +67 -0
- package/dist/types/src/containers/Plank/Plank.stories.d.ts.map +1 -0
- package/dist/types/src/containers/Plank/PlankComponent.d.ts +17 -0
- package/dist/types/src/containers/Plank/PlankComponent.d.ts.map +1 -0
- package/dist/types/src/containers/Plank/PlankContent.d.ts +10 -0
- package/dist/types/src/containers/Plank/PlankContent.d.ts.map +1 -0
- package/dist/types/src/{components → containers}/Plank/PlankControls.d.ts +5 -6
- package/dist/types/src/containers/Plank/PlankControls.d.ts.map +1 -0
- package/dist/types/src/containers/Plank/PlankError.d.ts +15 -0
- package/dist/types/src/containers/Plank/PlankError.d.ts.map +1 -0
- package/dist/types/src/{components → containers}/Plank/PlankHeading.d.ts +5 -4
- package/dist/types/src/containers/Plank/PlankHeading.d.ts.map +1 -0
- package/dist/types/src/{components → containers}/Plank/PlankLoading.d.ts.map +1 -1
- package/dist/types/src/containers/Plank/PlankRoot.d.ts +37 -0
- package/dist/types/src/containers/Plank/PlankRoot.d.ts.map +1 -0
- package/dist/types/src/containers/Plank/index.d.ts +4 -0
- package/dist/types/src/containers/Plank/index.d.ts.map +1 -0
- package/dist/types/src/containers/Sidebar/ComplementarySidebar.d.ts.map +1 -0
- package/dist/types/src/containers/Sidebar/Sidebar.d.ts.map +1 -0
- package/dist/types/src/{components → containers}/Sidebar/SidebarButton.d.ts +1 -1
- package/dist/types/src/containers/Sidebar/SidebarButton.d.ts.map +1 -0
- package/dist/types/src/{components → containers}/Sidebar/index.d.ts.map +1 -1
- package/dist/types/src/containers/index.d.ts +6 -0
- package/dist/types/src/containers/index.d.ts.map +1 -0
- package/dist/types/src/hooks/index.d.ts +2 -1
- package/dist/types/src/hooks/index.d.ts.map +1 -1
- package/dist/types/src/hooks/useBreakpoints.d.ts +1 -1
- package/dist/types/src/hooks/useCompanions.d.ts.map +1 -1
- package/dist/types/src/hooks/useDeckCompanions.d.ts +4 -4
- package/dist/types/src/hooks/useDeckCompanions.d.ts.map +1 -1
- package/dist/types/src/hooks/useDeckState.d.ts +17 -0
- package/dist/types/src/hooks/useDeckState.d.ts.map +1 -0
- package/dist/types/src/hooks/useMainSize.d.ts +2 -2
- package/dist/types/src/hooks/useMainSize.d.ts.map +1 -1
- package/dist/types/src/hooks/useNodeActionExpander.d.ts +1 -1
- package/dist/types/src/hooks/useNodeActionExpander.d.ts.map +1 -1
- package/dist/types/src/hooks/useSelectedCompanion.d.ts +13 -0
- package/dist/types/src/hooks/useSelectedCompanion.d.ts.map +1 -0
- package/dist/types/src/index.d.ts +1 -4
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/layout.d.ts +20 -9
- package/dist/types/src/layout.d.ts.map +1 -1
- package/dist/types/src/layout.test.d.ts +2 -0
- package/dist/types/src/layout.test.d.ts.map +1 -0
- package/dist/types/src/meta.d.ts +2 -3
- package/dist/types/src/meta.d.ts.map +1 -1
- package/dist/types/src/operations/add-toast.d.ts +5 -0
- package/dist/types/src/operations/add-toast.d.ts.map +1 -0
- package/dist/types/src/operations/adjust.d.ts +5 -0
- package/dist/types/src/operations/adjust.d.ts.map +1 -0
- package/dist/types/src/operations/close.d.ts +5 -0
- package/dist/types/src/operations/close.d.ts.map +1 -0
- package/dist/types/src/operations/helpers.d.ts +3 -0
- package/dist/types/src/operations/helpers.d.ts.map +1 -0
- package/dist/types/src/operations/index.d.ts +3 -0
- package/dist/types/src/operations/index.d.ts.map +1 -0
- package/dist/types/src/operations/open.d.ts +5 -0
- package/dist/types/src/operations/open.d.ts.map +1 -0
- package/dist/types/src/operations/revert-workspace.d.ts +5 -0
- package/dist/types/src/operations/revert-workspace.d.ts.map +1 -0
- package/dist/types/src/operations/scroll-into-view.d.ts +5 -0
- package/dist/types/src/operations/scroll-into-view.d.ts.map +1 -0
- package/dist/types/src/operations/set-layout-mode.d.ts +9 -0
- package/dist/types/src/operations/set-layout-mode.d.ts.map +1 -0
- package/dist/types/src/operations/set.d.ts +5 -0
- package/dist/types/src/operations/set.d.ts.map +1 -0
- package/dist/types/src/operations/switch-workspace.d.ts +5 -0
- package/dist/types/src/operations/switch-workspace.d.ts.map +1 -0
- package/dist/types/src/operations/update-companion.d.ts +5 -0
- package/dist/types/src/operations/update-companion.d.ts.map +1 -0
- package/dist/types/src/operations/update-complementary.d.ts +5 -0
- package/dist/types/src/operations/update-complementary.d.ts.map +1 -0
- package/dist/types/src/operations/update-dialog.d.ts +5 -0
- package/dist/types/src/operations/update-dialog.d.ts.map +1 -0
- package/dist/types/src/operations/update-plank-size.d.ts +5 -0
- package/dist/types/src/operations/update-plank-size.d.ts.map +1 -0
- package/dist/types/src/operations/update-popover.d.ts +5 -0
- package/dist/types/src/operations/update-popover.d.ts.map +1 -0
- package/dist/types/src/operations/update-sidebar.d.ts +5 -0
- package/dist/types/src/operations/update-sidebar.d.ts.map +1 -0
- package/dist/types/src/plugin.d.ts +4 -0
- package/dist/types/src/plugin.d.ts.map +1 -0
- package/dist/types/src/translations.d.ts +48 -53
- package/dist/types/src/translations.d.ts.map +1 -1
- package/dist/types/src/types/DeckCapabilities.d.ts +188 -0
- package/dist/types/src/types/DeckCapabilities.d.ts.map +1 -0
- package/dist/types/src/types/DeckEvents.d.ts +5 -0
- package/dist/types/src/types/DeckEvents.d.ts.map +1 -0
- package/dist/types/src/types/DeckOperation.d.ts +15 -0
- package/dist/types/src/types/DeckOperation.d.ts.map +1 -0
- package/dist/types/src/types/Settings.d.ts +10 -0
- package/dist/types/src/types/Settings.d.ts.map +1 -0
- package/dist/types/src/types/index.d.ts +4 -0
- package/dist/types/src/types/index.d.ts.map +1 -1
- package/dist/types/src/types/schema.d.ts +70 -69
- package/dist/types/src/types/schema.d.ts.map +1 -1
- package/dist/types/src/util/index.d.ts +2 -1
- package/dist/types/src/util/index.d.ts.map +1 -1
- package/dist/types/src/util/layoutAppliesTopbar.d.ts +1 -1
- package/dist/types/src/util/layoutAppliesTopbar.d.ts.map +1 -1
- package/dist/types/src/util/plank-url-params.d.ts +14 -0
- package/dist/types/src/util/plank-url-params.d.ts.map +1 -0
- package/dist/types/src/util/plank-url-params.test.d.ts +2 -0
- package/dist/types/src/util/plank-url-params.test.d.ts.map +1 -0
- package/dist/types/src/util/sanitize-persisted-state.d.ts +19 -0
- package/dist/types/src/util/sanitize-persisted-state.d.ts.map +1 -0
- package/dist/types/src/util/sanitize-persisted-state.test.d.ts +2 -0
- package/dist/types/src/util/sanitize-persisted-state.test.d.ts.map +1 -0
- package/dist/types/src/util/set-active.d.ts +19 -4
- package/dist/types/src/util/set-active.d.ts.map +1 -1
- package/dist/types/src/util/set-active.test.d.ts +2 -0
- package/dist/types/src/util/set-active.test.d.ts.map +1 -0
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +126 -62
- package/src/DeckPlugin.node.ts +17 -0
- package/src/DeckPlugin.test.ts +27 -0
- package/src/DeckPlugin.ts +57 -71
- package/src/DeckPlugin.workerd.ts +16 -0
- package/src/capabilities/app-graph-builder.ts +111 -126
- package/src/capabilities/check-app-scheme.ts +123 -24
- package/src/capabilities/index.ts +14 -14
- package/src/capabilities/notification-tracker.ts +170 -0
- package/src/capabilities/operation-handler.ts +16 -0
- package/src/capabilities/react-root.tsx +40 -27
- package/src/capabilities/react-surface.tsx +28 -22
- package/src/capabilities/settings.ts +29 -19
- package/src/capabilities/state.ts +74 -88
- package/src/capabilities/tools.ts +58 -51
- package/src/capabilities/url-handler.ts +290 -48
- package/src/components/DeckSettings/DeckSettings.stories.tsx +37 -0
- package/src/components/DeckSettings/DeckSettings.tsx +19 -73
- package/src/components/DeckSettings/index.ts +2 -2
- package/src/components/Matrix/Matrix.stories.tsx +220 -0
- package/src/components/Matrix/Matrix.tsx +205 -0
- package/src/components/Matrix/SPEC.md +219 -0
- package/src/components/Matrix/index.ts +6 -0
- package/src/components/index.ts +6 -3
- package/src/containers/Deck/Banner.tsx +41 -0
- package/src/containers/Deck/Deck.stories.tsx +81 -0
- package/src/containers/Deck/Deck.tsx +21 -0
- package/src/containers/Deck/DeckContent.tsx +102 -0
- package/src/containers/Deck/DeckRoot.tsx +50 -0
- package/src/containers/Deck/DeckViewport.tsx +450 -0
- package/src/{components/DeckLayout → containers/Deck}/StatusBar.tsx +4 -4
- package/src/containers/Deck/index.ts +5 -0
- package/src/{components → containers}/DeckLayout/ActiveNode.tsx +9 -4
- package/src/containers/DeckLayout/DeckLayout.stories.tsx +395 -0
- package/src/containers/DeckLayout/DeckLayout.tsx +62 -0
- package/src/containers/DeckLayout/Dialog.tsx +52 -0
- package/src/containers/DeckLayout/Fallback.tsx +24 -0
- package/src/containers/DeckLayout/Popover.tsx +166 -0
- package/src/containers/DeckLayout/Toast.tsx +76 -0
- package/src/{components → containers}/DeckLayout/constants.ts +1 -0
- package/src/{components → containers}/DeckLayout/index.ts +3 -2
- package/src/containers/Plank/Plank.stories.tsx +108 -0
- package/src/containers/Plank/Plank.tsx +22 -0
- package/src/containers/Plank/PlankComponent.tsx +223 -0
- package/src/containers/Plank/PlankContent.tsx +45 -0
- package/src/{components → containers}/Plank/PlankControls.tsx +45 -38
- package/src/containers/Plank/PlankError.tsx +82 -0
- package/src/{components → containers}/Plank/PlankHeading.tsx +69 -65
- package/src/{components → containers}/Plank/PlankLoading.tsx +1 -1
- package/src/containers/Plank/PlankRoot.tsx +49 -0
- package/src/{components → containers}/Plank/index.ts +0 -2
- package/src/containers/Sidebar/ComplementarySidebar.tsx +189 -0
- package/src/containers/Sidebar/Sidebar.tsx +39 -0
- package/src/containers/Sidebar/SidebarButton.tsx +100 -0
- package/src/containers/index.ts +11 -0
- package/src/hooks/index.ts +2 -1
- package/src/hooks/useCompanions.ts +3 -3
- package/src/hooks/useDeckCompanions.ts +8 -10
- package/src/hooks/useDeckState.ts +73 -0
- package/src/hooks/useMainSize.ts +2 -2
- package/src/hooks/useNodeActionExpander.ts +4 -4
- package/src/hooks/useSelectedCompanion.ts +32 -0
- package/src/index.ts +1 -4
- package/src/layout.test.ts +59 -0
- package/src/layout.ts +38 -40
- package/src/meta.ts +27 -6
- package/src/operations/add-toast.ts +24 -0
- package/src/operations/adjust.ts +82 -0
- package/src/operations/close.ts +35 -0
- package/src/operations/helpers.ts +22 -0
- package/src/operations/index.ts +23 -0
- package/src/operations/open.ts +201 -0
- package/src/operations/revert-workspace.ts +22 -0
- package/src/operations/scroll-into-view.ts +24 -0
- package/src/operations/set-layout-mode.ts +84 -0
- package/src/operations/set.ts +36 -0
- package/src/operations/switch-workspace.ts +66 -0
- package/src/operations/update-companion.ts +35 -0
- package/src/operations/update-complementary.ts +33 -0
- package/src/operations/update-dialog.ts +34 -0
- package/src/operations/update-plank-size.ts +28 -0
- package/src/operations/update-popover.ts +36 -0
- package/src/operations/update-sidebar.ts +28 -0
- package/src/plugin.ts +11 -0
- package/src/translations.ts +48 -52
- package/src/types/DeckCapabilities.ts +34 -0
- package/src/types/DeckEvents.ts +21 -0
- package/src/types/DeckOperation.ts +56 -0
- package/src/types/Settings.ts +36 -0
- package/src/types/index.ts +5 -0
- package/src/types/schema.ts +50 -57
- package/src/util/index.ts +2 -1
- package/src/util/layoutAppliesTopbar.ts +2 -2
- package/src/util/plank-url-params.test.ts +85 -0
- package/src/util/plank-url-params.ts +36 -0
- package/src/util/sanitize-persisted-state.test.ts +79 -0
- package/src/util/sanitize-persisted-state.ts +52 -0
- package/src/util/set-active.test.ts +106 -0
- package/src/util/set-active.ts +50 -30
- package/src/vite-env.d.ts +10 -0
- package/dist/lib/browser/app-graph-builder-DVEKLXB4.mjs +0 -152
- package/dist/lib/browser/app-graph-builder-DVEKLXB4.mjs.map +0 -7
- package/dist/lib/browser/check-app-scheme-3BQJXWEY.mjs +0 -32
- package/dist/lib/browser/check-app-scheme-3BQJXWEY.mjs.map +0 -7
- package/dist/lib/browser/chunk-6MQIYIPY.mjs +0 -1494
- package/dist/lib/browser/chunk-6MQIYIPY.mjs.map +0 -7
- package/dist/lib/browser/chunk-CNTGBCMK.mjs +0 -145
- package/dist/lib/browser/chunk-CNTGBCMK.mjs.map +0 -7
- package/dist/lib/browser/chunk-F5BQOOEG.mjs +0 -160
- package/dist/lib/browser/chunk-F5BQOOEG.mjs.map +0 -7
- package/dist/lib/browser/chunk-M57WD3V6.mjs +0 -16
- package/dist/lib/browser/chunk-M57WD3V6.mjs.map +0 -7
- package/dist/lib/browser/chunk-MX7WRVX3.mjs +0 -127
- package/dist/lib/browser/chunk-MX7WRVX3.mjs.map +0 -7
- package/dist/lib/browser/chunk-NRCPV6AV.mjs +0 -129
- package/dist/lib/browser/chunk-NRCPV6AV.mjs.map +0 -7
- package/dist/lib/browser/chunk-Z5KITAZW.mjs +0 -13
- package/dist/lib/browser/chunk-Z5KITAZW.mjs.map +0 -7
- package/dist/lib/browser/index.mjs +0 -174
- package/dist/lib/browser/index.mjs.map +0 -7
- package/dist/lib/browser/intent-resolver-2SUIIV6N.mjs +0 -521
- package/dist/lib/browser/intent-resolver-2SUIIV6N.mjs.map +0 -7
- package/dist/lib/browser/meta.json +0 -1
- package/dist/lib/browser/react-root-GVZANZX7.mjs +0 -43
- package/dist/lib/browser/react-root-GVZANZX7.mjs.map +0 -7
- package/dist/lib/browser/react-surface-NXSSD2GW.mjs +0 -40
- package/dist/lib/browser/react-surface-NXSSD2GW.mjs.map +0 -7
- package/dist/lib/browser/settings-LUPQPZ27.mjs +0 -29
- package/dist/lib/browser/settings-LUPQPZ27.mjs.map +0 -7
- package/dist/lib/browser/state-CRXR7X63.mjs +0 -12
- package/dist/lib/browser/toolkit-OBKFXX23.mjs +0 -47
- package/dist/lib/browser/toolkit-OBKFXX23.mjs.map +0 -7
- package/dist/lib/browser/types/index.mjs +0 -32
- package/dist/lib/browser/url-handler-LROZYQ26.mjs +0 -70
- package/dist/lib/browser/url-handler-LROZYQ26.mjs.map +0 -7
- package/dist/types/src/capabilities/capabilities.d.ts +0 -184
- package/dist/types/src/capabilities/capabilities.d.ts.map +0 -1
- package/dist/types/src/capabilities/intent-resolver.d.ts +0 -4
- package/dist/types/src/capabilities/intent-resolver.d.ts.map +0 -1
- package/dist/types/src/capabilities/toolkit.d.ts +0 -5
- package/dist/types/src/capabilities/toolkit.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/ActiveNode.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/Banner.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/ContentEmpty.d.ts +0 -3
- package/dist/types/src/components/DeckLayout/ContentEmpty.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/DeckLayout.d.ts +0 -6
- package/dist/types/src/components/DeckLayout/DeckLayout.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/Dialog.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/Fallback.d.ts +0 -3
- package/dist/types/src/components/DeckLayout/Fallback.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/Popover.d.ts +0 -5
- package/dist/types/src/components/DeckLayout/Popover.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/StatusBar.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/Toast.d.ts +0 -5
- package/dist/types/src/components/DeckLayout/Toast.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/Topbar.d.ts +0 -3
- package/dist/types/src/components/DeckLayout/Topbar.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/constants.d.ts.map +0 -1
- package/dist/types/src/components/DeckLayout/index.d.ts +0 -4
- package/dist/types/src/components/DeckLayout/index.d.ts.map +0 -1
- package/dist/types/src/components/Plank/Plank.d.ts +0 -27
- package/dist/types/src/components/Plank/Plank.d.ts.map +0 -1
- package/dist/types/src/components/Plank/Plank.stories.d.ts +0 -8
- package/dist/types/src/components/Plank/Plank.stories.d.ts.map +0 -1
- package/dist/types/src/components/Plank/PlankControls.d.ts.map +0 -1
- package/dist/types/src/components/Plank/PlankError.d.ts +0 -13
- package/dist/types/src/components/Plank/PlankError.d.ts.map +0 -1
- package/dist/types/src/components/Plank/PlankHeading.d.ts.map +0 -1
- package/dist/types/src/components/Plank/index.d.ts +0 -6
- package/dist/types/src/components/Plank/index.d.ts.map +0 -1
- package/dist/types/src/components/Sidebar/ComplementarySidebar.d.ts.map +0 -1
- package/dist/types/src/components/Sidebar/Sidebar.d.ts.map +0 -1
- package/dist/types/src/components/Sidebar/SidebarButton.d.ts.map +0 -1
- package/dist/types/src/components/fragments.d.ts +0 -4
- package/dist/types/src/components/fragments.d.ts.map +0 -1
- package/dist/types/src/events.d.ts +0 -4
- package/dist/types/src/events.d.ts.map +0 -1
- package/dist/types/src/hooks/useHoistStatusbar.d.ts +0 -3
- package/dist/types/src/hooks/useHoistStatusbar.d.ts.map +0 -1
- package/dist/types/src/util/overscroll.d.ts +0 -47
- package/dist/types/src/util/overscroll.d.ts.map +0 -1
- package/src/capabilities/capabilities.ts +0 -14
- package/src/capabilities/intent-resolver.ts +0 -469
- package/src/capabilities/toolkit.ts +0 -55
- package/src/components/DeckLayout/Banner.tsx +0 -39
- package/src/components/DeckLayout/ContentEmpty.tsx +0 -31
- package/src/components/DeckLayout/DeckLayout.tsx +0 -302
- package/src/components/DeckLayout/Dialog.tsx +0 -36
- package/src/components/DeckLayout/Fallback.tsx +0 -28
- package/src/components/DeckLayout/Popover.tsx +0 -104
- package/src/components/DeckLayout/Toast.tsx +0 -61
- package/src/components/DeckLayout/Topbar.tsx +0 -11
- package/src/components/Plank/Plank.stories.tsx +0 -56
- package/src/components/Plank/Plank.tsx +0 -267
- package/src/components/Plank/PlankError.tsx +0 -49
- package/src/components/Sidebar/ComplementarySidebar.tsx +0 -194
- package/src/components/Sidebar/Sidebar.tsx +0 -42
- package/src/components/Sidebar/SidebarButton.tsx +0 -87
- package/src/components/fragments.ts +0 -14
- package/src/events.ts +0 -11
- package/src/hooks/useHoistStatusbar.ts +0 -25
- package/src/util/overscroll.ts +0 -69
- /package/dist/lib/{browser/state-CRXR7X63.mjs.map → neutral/check-app-scheme-INSOF72J.mjs.map} +0 -0
- /package/dist/lib/{browser/types/index.mjs.map → neutral/chunk-J5LGTIGS.mjs.map} +0 -0
- /package/dist/types/src/{components/DeckLayout → containers/Deck}/StatusBar.d.ts +0 -0
- /package/dist/types/src/{components → containers}/DeckLayout/ActiveNode.d.ts +0 -0
- /package/dist/types/src/{components → containers}/DeckLayout/Dialog.d.ts +0 -0
- /package/dist/types/src/{components → containers}/DeckLayout/constants.d.ts +0 -0
- /package/dist/types/src/{components → containers}/Plank/PlankLoading.d.ts +0 -0
- /package/dist/types/src/{components → containers}/Sidebar/ComplementarySidebar.d.ts +0 -0
- /package/dist/types/src/{components → containers}/Sidebar/Sidebar.d.ts +0 -0
- /package/dist/types/src/{components → containers}/Sidebar/index.d.ts +0 -0
- /package/src/{components → containers}/Sidebar/index.ts +0 -0
|
@@ -0,0 +1,291 @@
|
|
|
1
|
+
import {
|
|
2
|
+
Deck,
|
|
3
|
+
PlankErrorFallback
|
|
4
|
+
} from "./chunk-WAXJPQJI.mjs";
|
|
5
|
+
import "./chunk-GBIGQKYW.mjs";
|
|
6
|
+
import "./chunk-J5LGTIGS.mjs";
|
|
7
|
+
|
|
8
|
+
// src/containers/DeckLayout/DeckLayout.tsx
|
|
9
|
+
import React5, { useCallback as useCallback3 } from "react";
|
|
10
|
+
import { useAtomCapability, useOperationInvoker, usePluginManager } from "@dxos/app-framework/ui";
|
|
11
|
+
import { LayoutOperation } from "@dxos/app-toolkit";
|
|
12
|
+
import { Mosaic } from "@dxos/react-ui-mosaic";
|
|
13
|
+
import { useDeckState as useDeckState3 } from "#hooks";
|
|
14
|
+
import { DeckCapabilities, getMode } from "#types";
|
|
15
|
+
|
|
16
|
+
// src/containers/DeckLayout/ActiveNode.tsx
|
|
17
|
+
import React from "react";
|
|
18
|
+
import { Surface } from "@dxos/app-framework/ui";
|
|
19
|
+
import { AppSurface, useAppGraph } from "@dxos/app-toolkit/ui";
|
|
20
|
+
import { useNode } from "@dxos/plugin-graph";
|
|
21
|
+
import { useAttended } from "@dxos/react-ui-attention";
|
|
22
|
+
import { useNodeActionExpander } from "#hooks";
|
|
23
|
+
var ActiveNode = () => {
|
|
24
|
+
const [id] = useAttended();
|
|
25
|
+
const { graph } = useAppGraph();
|
|
26
|
+
const activeNode = useNode(graph, id);
|
|
27
|
+
useNodeActionExpander(activeNode);
|
|
28
|
+
return /* @__PURE__ */ React.createElement("div", {
|
|
29
|
+
className: "sr-only"
|
|
30
|
+
}, /* @__PURE__ */ React.createElement(Surface.Surface, {
|
|
31
|
+
type: AppSurface.DocumentTitle,
|
|
32
|
+
data: {
|
|
33
|
+
subject: activeNode
|
|
34
|
+
},
|
|
35
|
+
limit: 1
|
|
36
|
+
}));
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
// src/containers/DeckLayout/Dialog.tsx
|
|
40
|
+
import React2, { useCallback } from "react";
|
|
41
|
+
import { Surface as Surface2 } from "@dxos/app-framework/ui";
|
|
42
|
+
import { AppSurface as AppSurface2 } from "@dxos/app-toolkit/ui";
|
|
43
|
+
import { AlertDialog, Dialog as NaturalDialog } from "@dxos/react-ui";
|
|
44
|
+
import { useDeckState } from "#hooks";
|
|
45
|
+
var Dialog = () => {
|
|
46
|
+
const { state, updateEphemeral } = useDeckState();
|
|
47
|
+
const { dialogOpen, dialogType, dialogBlockAlign, dialogOverlayClasses, dialogOverlayStyle, dialogContent } = state;
|
|
48
|
+
const Root = dialogType === "alert" ? AlertDialog.Root : NaturalDialog.Root;
|
|
49
|
+
const Overlay = dialogType === "alert" ? AlertDialog.Overlay : NaturalDialog.Overlay;
|
|
50
|
+
const handleOpenChange = useCallback((nextOpen) => {
|
|
51
|
+
updateEphemeral((s) => ({
|
|
52
|
+
...s,
|
|
53
|
+
dialogOpen: nextOpen
|
|
54
|
+
}));
|
|
55
|
+
}, [
|
|
56
|
+
updateEphemeral
|
|
57
|
+
]);
|
|
58
|
+
return /* @__PURE__ */ React2.createElement(Root, {
|
|
59
|
+
modal: dialogBlockAlign !== "end",
|
|
60
|
+
open: dialogOpen,
|
|
61
|
+
onOpenChange: handleOpenChange
|
|
62
|
+
}, dialogBlockAlign === "end" ? (
|
|
63
|
+
// TODO(burdon): Placeholder creates a suspense boundary; replace with defaults.
|
|
64
|
+
/* @__PURE__ */ React2.createElement(Surface2.Surface, {
|
|
65
|
+
type: AppSurface2.Dialog,
|
|
66
|
+
data: dialogContent ?? void 0,
|
|
67
|
+
limit: 1,
|
|
68
|
+
fallback: PlankErrorFallback,
|
|
69
|
+
placeholder: /* @__PURE__ */ React2.createElement("div", null)
|
|
70
|
+
})
|
|
71
|
+
) : /* @__PURE__ */ React2.createElement(Overlay, {
|
|
72
|
+
blockAlign: dialogBlockAlign,
|
|
73
|
+
classNames: dialogOverlayClasses,
|
|
74
|
+
style: dialogOverlayStyle
|
|
75
|
+
}, /* @__PURE__ */ React2.createElement(Surface2.Surface, {
|
|
76
|
+
type: AppSurface2.Dialog,
|
|
77
|
+
data: dialogContent ?? void 0,
|
|
78
|
+
limit: 1,
|
|
79
|
+
fallback: PlankErrorFallback
|
|
80
|
+
})));
|
|
81
|
+
};
|
|
82
|
+
|
|
83
|
+
// src/containers/DeckLayout/Popover.tsx
|
|
84
|
+
import { createContext } from "@radix-ui/react-context";
|
|
85
|
+
import React3, { useCallback as useCallback2, useEffect, useRef, useState } from "react";
|
|
86
|
+
import { Surface as Surface3 } from "@dxos/app-framework/ui";
|
|
87
|
+
import { AppSurface as AppSurface3, useObjectMenuItems } from "@dxos/app-toolkit/ui";
|
|
88
|
+
import { Obj } from "@dxos/echo";
|
|
89
|
+
import { Card, Popover, toLocalizedString, Toolbar, useTranslation } from "@dxos/react-ui";
|
|
90
|
+
import { Menu } from "@dxos/react-ui-menu";
|
|
91
|
+
import { useDeckState as useDeckState2 } from "#hooks";
|
|
92
|
+
import { meta } from "#meta";
|
|
93
|
+
var DEBOUNCE_DELAY = 40;
|
|
94
|
+
var [DeckPopoverProvider, useDeckPopoverContext] = createContext("DeckPopover");
|
|
95
|
+
var PopoverRoot = ({ children }) => {
|
|
96
|
+
const { state } = useDeckState2();
|
|
97
|
+
const virtualRef = useRef(null);
|
|
98
|
+
const [virtualIter, setVirtualIter] = useState(0);
|
|
99
|
+
const [open, setOpen] = useState(false);
|
|
100
|
+
const debounceRef = useRef(null);
|
|
101
|
+
useEffect(() => {
|
|
102
|
+
setOpen(false);
|
|
103
|
+
if (state.popoverOpen) {
|
|
104
|
+
if (debounceRef.current) {
|
|
105
|
+
clearTimeout(debounceRef.current);
|
|
106
|
+
}
|
|
107
|
+
if (state.popoverAnchor && virtualRef.current !== state.popoverAnchor) {
|
|
108
|
+
virtualRef.current = state.popoverAnchor ?? null;
|
|
109
|
+
setVirtualIter((iter) => iter + 1);
|
|
110
|
+
}
|
|
111
|
+
debounceRef.current = setTimeout(() => setOpen(true), DEBOUNCE_DELAY);
|
|
112
|
+
}
|
|
113
|
+
}, [
|
|
114
|
+
state.popoverOpen,
|
|
115
|
+
state.popoverAnchorId,
|
|
116
|
+
state.popoverAnchor,
|
|
117
|
+
state.popoverContent
|
|
118
|
+
]);
|
|
119
|
+
return /* @__PURE__ */ React3.createElement(DeckPopoverProvider, {
|
|
120
|
+
setOpen
|
|
121
|
+
}, /* @__PURE__ */ React3.createElement(Popover.Root, {
|
|
122
|
+
modal: false,
|
|
123
|
+
open
|
|
124
|
+
}, state.popoverAnchor && /* @__PURE__ */ React3.createElement(Popover.VirtualTrigger, {
|
|
125
|
+
key: virtualIter,
|
|
126
|
+
virtualRef
|
|
127
|
+
}), children));
|
|
128
|
+
};
|
|
129
|
+
var PopoverContent = () => {
|
|
130
|
+
const { t } = useTranslation(meta.id);
|
|
131
|
+
const { state, updateEphemeral } = useDeckState2();
|
|
132
|
+
const { setOpen } = useDeckPopoverContext("PopoverContent");
|
|
133
|
+
const popoverSubject = state.popoverContent && "subject" in state.popoverContent ? state.popoverContent.subject : void 0;
|
|
134
|
+
const isObjectPopover = Obj.isObject(popoverSubject);
|
|
135
|
+
const objectMenuItems = useObjectMenuItems(popoverSubject);
|
|
136
|
+
const title = state.popoverTitle ? toLocalizedString(state.popoverTitle, t) : "Unknown";
|
|
137
|
+
const icon = isObjectPopover ? Obj.getIcon(popoverSubject)?.icon ?? "ph--circle-dashed--regular" : void 0;
|
|
138
|
+
const content = state.popoverContent;
|
|
139
|
+
const isBasePopover = state.popoverKind === "base" && !!content && "component" in content;
|
|
140
|
+
const handleClose = useCallback2(() => {
|
|
141
|
+
setOpen(false);
|
|
142
|
+
updateEphemeral((state2) => ({
|
|
143
|
+
...state2,
|
|
144
|
+
popoverOpen: false,
|
|
145
|
+
popoverAnchor: void 0,
|
|
146
|
+
popoverAnchorId: void 0,
|
|
147
|
+
popoverSide: void 0
|
|
148
|
+
}));
|
|
149
|
+
}, [
|
|
150
|
+
updateEphemeral
|
|
151
|
+
]);
|
|
152
|
+
const handleInteractOutside = useCallback2((event) => {
|
|
153
|
+
if (
|
|
154
|
+
// TODO(thure): CodeMirror should not focus itself when it updates.
|
|
155
|
+
event.type === "dismissableLayer.focusOutside" && event.currentTarget?.classList.contains("cm-content")
|
|
156
|
+
) {
|
|
157
|
+
event.preventDefault();
|
|
158
|
+
} else {
|
|
159
|
+
handleClose();
|
|
160
|
+
}
|
|
161
|
+
}, [
|
|
162
|
+
handleClose
|
|
163
|
+
]);
|
|
164
|
+
return /* @__PURE__ */ React3.createElement(Popover.Portal, null, /* @__PURE__ */ React3.createElement(Popover.Content, {
|
|
165
|
+
side: state.popoverSide,
|
|
166
|
+
sticky: "always",
|
|
167
|
+
hideWhenDetached: true,
|
|
168
|
+
onOpenAutoFocus: (event) => event.preventDefault(),
|
|
169
|
+
onInteractOutside: handleInteractOutside,
|
|
170
|
+
onEscapeKeyDown: handleInteractOutside
|
|
171
|
+
}, /* @__PURE__ */ React3.createElement(Popover.Viewport, null, isBasePopover && content && "component" in content ? (
|
|
172
|
+
/* Base popover: a plugin-provided component (e.g. editor link preview). */
|
|
173
|
+
/* @__PURE__ */ React3.createElement(Surface3.Surface, {
|
|
174
|
+
type: AppSurface3.Popover,
|
|
175
|
+
data: content,
|
|
176
|
+
limit: 1
|
|
177
|
+
})
|
|
178
|
+
) : (
|
|
179
|
+
/*
|
|
180
|
+
* Card popover (default). Rendered for any open popover that isn't an explicit
|
|
181
|
+
* base-component popover so the popover can never collapse to a bare 1px frame: the
|
|
182
|
+
* header (icon + title + menu) always renders, and the body falls back to a fixed-
|
|
183
|
+
* height "no preview" row when no subject resolves a card Surface (e.g. system-type
|
|
184
|
+
* objects like a raw Feed that have no registered card and no renderable fields).
|
|
185
|
+
*/
|
|
186
|
+
/* @__PURE__ */ React3.createElement(Menu.Root, null, /* @__PURE__ */ React3.createElement(Card.Root, {
|
|
187
|
+
border: false,
|
|
188
|
+
classNames: "dx-card-popover"
|
|
189
|
+
}, /* @__PURE__ */ React3.createElement(Card.Header, null, /* @__PURE__ */ React3.createElement(Card.IconBlock, null, icon && /* @__PURE__ */ React3.createElement(Card.Icon, {
|
|
190
|
+
icon
|
|
191
|
+
})), /* @__PURE__ */ React3.createElement(Card.Title, null, title), /* @__PURE__ */ React3.createElement(Card.IconBlock, null, /* @__PURE__ */ React3.createElement(Menu.Trigger, {
|
|
192
|
+
asChild: true,
|
|
193
|
+
disabled: !objectMenuItems.length
|
|
194
|
+
}, /* @__PURE__ */ React3.createElement(Toolbar.IconButton, {
|
|
195
|
+
variant: "ghost",
|
|
196
|
+
density: "sm",
|
|
197
|
+
icon: "ph--dots-three-vertical--regular",
|
|
198
|
+
iconOnly: true,
|
|
199
|
+
label: "Actions"
|
|
200
|
+
})), /* @__PURE__ */ React3.createElement(Menu.Content, {
|
|
201
|
+
items: objectMenuItems
|
|
202
|
+
}))), content && "subject" in content ? /* @__PURE__ */ React3.createElement(Surface3.Surface, {
|
|
203
|
+
type: AppSurface3.Card,
|
|
204
|
+
data: content,
|
|
205
|
+
limit: 1
|
|
206
|
+
}) : /* @__PURE__ */ React3.createElement(Card.Body, {
|
|
207
|
+
classNames: "min-bs-8"
|
|
208
|
+
}, /* @__PURE__ */ React3.createElement(Card.Row, null, /* @__PURE__ */ React3.createElement(Card.Text, {
|
|
209
|
+
variant: "description"
|
|
210
|
+
}, t("popover-no-preview.message"))))))
|
|
211
|
+
)), /* @__PURE__ */ React3.createElement(Popover.Arrow, null)));
|
|
212
|
+
};
|
|
213
|
+
|
|
214
|
+
// src/containers/DeckLayout/Toast.tsx
|
|
215
|
+
import React4, { useState as useState2 } from "react";
|
|
216
|
+
import { Button, Toast as NaturalToast, toLocalizedString as toLocalizedString2, useTranslation as useTranslation2 } from "@dxos/react-ui";
|
|
217
|
+
import { meta as meta2 } from "#meta";
|
|
218
|
+
var Toast = ({ id, title, description, icon, duration, actionLabel, actionAlt, onAction, onOpenChange }) => {
|
|
219
|
+
const { t } = useTranslation2(meta2.id);
|
|
220
|
+
const [open, setOpen] = useState2(true);
|
|
221
|
+
const handleOpenChange = (next) => {
|
|
222
|
+
setOpen(next);
|
|
223
|
+
onOpenChange?.(next);
|
|
224
|
+
};
|
|
225
|
+
return /* @__PURE__ */ React4.createElement(NaturalToast.Root, {
|
|
226
|
+
"data-testid": id,
|
|
227
|
+
open,
|
|
228
|
+
duration,
|
|
229
|
+
onOpenChange: handleOpenChange
|
|
230
|
+
}, /* @__PURE__ */ React4.createElement(NaturalToast.Title, {
|
|
231
|
+
icon,
|
|
232
|
+
onClose: () => handleOpenChange(false)
|
|
233
|
+
}, title && /* @__PURE__ */ React4.createElement("span", null, toLocalizedString2(title, t))), description && /* @__PURE__ */ React4.createElement(NaturalToast.Description, null, toLocalizedString2(description, t)), onAction && actionAlt && actionLabel && /* @__PURE__ */ React4.createElement(NaturalToast.Actions, null, /* @__PURE__ */ React4.createElement(NaturalToast.Action, {
|
|
234
|
+
altText: toLocalizedString2(actionAlt, t),
|
|
235
|
+
asChild: true
|
|
236
|
+
}, /* @__PURE__ */ React4.createElement(Button, {
|
|
237
|
+
"data-testid": "toast.action",
|
|
238
|
+
variant: "primary",
|
|
239
|
+
onClick: () => onAction?.()
|
|
240
|
+
}, toLocalizedString2(actionLabel, t)))));
|
|
241
|
+
};
|
|
242
|
+
var Toaster = ({ toasts, onDismissToast }) => {
|
|
243
|
+
return /* @__PURE__ */ React4.createElement(React4.Fragment, null, toasts?.map((toast) => /* @__PURE__ */ React4.createElement(Toast, {
|
|
244
|
+
...toast,
|
|
245
|
+
key: toast.id,
|
|
246
|
+
onOpenChange: (open) => {
|
|
247
|
+
if (!open) {
|
|
248
|
+
onDismissToast?.(toast.id);
|
|
249
|
+
}
|
|
250
|
+
return open;
|
|
251
|
+
}
|
|
252
|
+
})));
|
|
253
|
+
};
|
|
254
|
+
|
|
255
|
+
// src/containers/DeckLayout/DeckLayout.tsx
|
|
256
|
+
var DeckLayout = ({ onDismissToast }) => {
|
|
257
|
+
const settings = useAtomCapability(DeckCapabilities.Settings);
|
|
258
|
+
const pluginManager = usePluginManager();
|
|
259
|
+
const { invokePromise } = useOperationInvoker();
|
|
260
|
+
const { deck, state, updateState } = useDeckState3();
|
|
261
|
+
const layoutMode = getMode(deck);
|
|
262
|
+
const { toasts } = state;
|
|
263
|
+
const handleLayoutChange = useCallback3((request) => {
|
|
264
|
+
void invokePromise(LayoutOperation.SetLayoutMode, request);
|
|
265
|
+
}, [
|
|
266
|
+
invokePromise
|
|
267
|
+
]);
|
|
268
|
+
return /* @__PURE__ */ React5.createElement(Mosaic.Root, null, /* @__PURE__ */ React5.createElement(PopoverRoot, null, /* @__PURE__ */ React5.createElement(ActiveNode, null), /* @__PURE__ */ React5.createElement(Deck.Root, {
|
|
269
|
+
settings,
|
|
270
|
+
pluginManager,
|
|
271
|
+
layoutMode,
|
|
272
|
+
deck,
|
|
273
|
+
state,
|
|
274
|
+
updateState,
|
|
275
|
+
onLayoutChange: handleLayoutChange
|
|
276
|
+
}, /* @__PURE__ */ React5.createElement(Deck.Content, null, /* @__PURE__ */ React5.createElement(Deck.Viewport, null, deck.solo ? /* @__PURE__ */ React5.createElement(Deck.SoloMode, null) : deck.active.length === 0 ? /* @__PURE__ */ React5.createElement(Deck.ContentEmpty, null) : /* @__PURE__ */ React5.createElement(Deck.MultiMode, null)))), /* @__PURE__ */ React5.createElement(PopoverContent, null), /* @__PURE__ */ React5.createElement(Dialog, null), /* @__PURE__ */ React5.createElement(Toaster, {
|
|
277
|
+
toasts,
|
|
278
|
+
onDismissToast
|
|
279
|
+
})));
|
|
280
|
+
};
|
|
281
|
+
|
|
282
|
+
// src/containers/DeckLayout/constants.ts
|
|
283
|
+
var NAV_ID = "NavTree";
|
|
284
|
+
|
|
285
|
+
// src/containers/DeckLayout/index.ts
|
|
286
|
+
var DeckLayout_default = DeckLayout;
|
|
287
|
+
export {
|
|
288
|
+
NAV_ID,
|
|
289
|
+
DeckLayout_default as default
|
|
290
|
+
};
|
|
291
|
+
//# sourceMappingURL=DeckLayout-DEURA3KR.mjs.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/containers/DeckLayout/DeckLayout.tsx", "../../../src/containers/DeckLayout/ActiveNode.tsx", "../../../src/containers/DeckLayout/Dialog.tsx", "../../../src/containers/DeckLayout/Popover.tsx", "../../../src/containers/DeckLayout/Toast.tsx", "../../../src/containers/DeckLayout/constants.ts", "../../../src/containers/DeckLayout/index.ts"],
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\nimport React, { useCallback } from 'react';\n\nimport { useAtomCapability, useOperationInvoker, usePluginManager } from '@dxos/app-framework/ui';\nimport { LayoutOperation } from '@dxos/app-toolkit';\nimport { Mosaic } from '@dxos/react-ui-mosaic';\n\nimport { useDeckState } from '#hooks';\nimport { DeckCapabilities, getMode } from '#types';\n\nimport { Deck, type DeckLayoutChangeRequest } from '../Deck';\nimport { ActiveNode } from './ActiveNode';\nimport { Dialog } from './Dialog';\nimport { PopoverContent, PopoverRoot } from './Popover';\nimport { Toaster, type ToasterProps } from './Toast';\n\nexport type DeckLayoutProps = Pick<ToasterProps, 'onDismissToast'>;\n\nexport const DeckLayout = ({ onDismissToast }: DeckLayoutProps) => {\n const settings = useAtomCapability(DeckCapabilities.Settings);\n const pluginManager = usePluginManager();\n const { invokePromise } = useOperationInvoker();\n const { deck, state, updateState } = useDeckState();\n const layoutMode = getMode(deck);\n const { toasts } = state;\n\n const handleLayoutChange = useCallback(\n (request: DeckLayoutChangeRequest) => {\n void invokePromise(LayoutOperation.SetLayoutMode, request);\n },\n [invokePromise],\n );\n\n return (\n <Mosaic.Root>\n <PopoverRoot>\n <ActiveNode />\n <Deck.Root\n settings={settings}\n pluginManager={pluginManager}\n layoutMode={layoutMode}\n deck={deck}\n state={state}\n updateState={updateState}\n onLayoutChange={handleLayoutChange}\n >\n <Deck.Content>\n <Deck.Viewport>\n {deck.solo ? <Deck.SoloMode /> : deck.active.length === 0 ? <Deck.ContentEmpty /> : <Deck.MultiMode />}\n </Deck.Viewport>\n </Deck.Content>\n </Deck.Root>\n <PopoverContent />\n <Dialog />\n <Toaster toasts={toasts} onDismissToast={onDismissToast} />\n </PopoverRoot>\n </Mosaic.Root>\n );\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport React from 'react';\n\nimport { Surface } from '@dxos/app-framework/ui';\nimport { AppSurface, useAppGraph } from '@dxos/app-toolkit/ui';\nimport { useNode } from '@dxos/plugin-graph';\nimport { useAttended } from '@dxos/react-ui-attention';\n\nimport { useNodeActionExpander } from '#hooks';\n\n// TODO(burdon): Factor out to effect in plugin set document title.\nexport const ActiveNode = () => {\n const [id] = useAttended();\n const { graph } = useAppGraph();\n const activeNode = useNode(graph, id);\n useNodeActionExpander(activeNode);\n\n return (\n <div className='sr-only'>\n {/* TODO(wittjosiah): Weird that this is a surface, feel like it's not really render logic.\n Probably this lives in React-land currently in order to access translations? */}\n <Surface.Surface\n type={AppSurface.DocumentTitle}\n data={{ subject: activeNode } satisfies AppSurface.DocumentTitleData}\n limit={1}\n />\n </div>\n );\n};\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport React, { useCallback } from 'react';\n\nimport { Surface } from '@dxos/app-framework/ui';\nimport { AppSurface } from '@dxos/app-toolkit/ui';\nimport { AlertDialog, Dialog as NaturalDialog } from '@dxos/react-ui';\n\nimport { useDeckState } from '#hooks';\n\nimport { PlankErrorFallback } from '../Plank';\n\nexport const Dialog = () => {\n const { state, updateEphemeral } = useDeckState();\n const { dialogOpen, dialogType, dialogBlockAlign, dialogOverlayClasses, dialogOverlayStyle, dialogContent } = state;\n const Root = dialogType === 'alert' ? AlertDialog.Root : NaturalDialog.Root;\n const Overlay = dialogType === 'alert' ? AlertDialog.Overlay : NaturalDialog.Overlay;\n\n const handleOpenChange = useCallback(\n (nextOpen: boolean) => {\n updateEphemeral((s) => ({ ...s, dialogOpen: nextOpen }));\n },\n [updateEphemeral],\n );\n\n // TODO(thure): End block alignment affecting `modal` and whether the surface renders in an overlay is tailored to the needs of the ambient chat dialog. As the feature matures, consider separating concerns.\n return (\n <Root modal={dialogBlockAlign !== 'end'} open={dialogOpen} onOpenChange={handleOpenChange}>\n {dialogBlockAlign === 'end' ? (\n // TODO(burdon): Placeholder creates a suspense boundary; replace with defaults.\n <Surface.Surface\n type={AppSurface.Dialog}\n data={dialogContent ?? undefined}\n limit={1}\n fallback={PlankErrorFallback}\n placeholder={<div />}\n />\n ) : (\n <Overlay blockAlign={dialogBlockAlign} classNames={dialogOverlayClasses} style={dialogOverlayStyle}>\n <Surface.Surface\n type={AppSurface.Dialog}\n data={dialogContent ?? undefined}\n limit={1}\n fallback={PlankErrorFallback}\n />\n </Overlay>\n )}\n </Root>\n );\n};\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport { createContext } from '@radix-ui/react-context';\nimport React, { type PropsWithChildren, useCallback, useEffect, useRef, useState } from 'react';\n\nimport { Surface } from '@dxos/app-framework/ui';\nimport { AppSurface, useObjectMenuItems } from '@dxos/app-toolkit/ui';\nimport { Obj } from '@dxos/echo';\nimport {\n Card,\n Popover,\n type PopoverContentInteractOutsideEvent,\n toLocalizedString,\n Toolbar,\n useTranslation,\n} from '@dxos/react-ui';\nimport { Menu } from '@dxos/react-ui-menu';\n\nimport { useDeckState } from '#hooks';\nimport { meta } from '#meta';\n\nconst DEBOUNCE_DELAY = 40;\n\ntype DeckPopoverContextValue = {\n setOpen: (open: boolean) => void;\n};\n\nconst [DeckPopoverProvider, useDeckPopoverContext] = createContext<DeckPopoverContextValue>('DeckPopover');\n\nexport type PopoverRootProps = PropsWithChildren;\n\nexport const PopoverRoot = ({ children }: PopoverRootProps) => {\n const { state } = useDeckState();\n const virtualRef = useRef<HTMLButtonElement | null>(null);\n const [virtualIter, setVirtualIter] = useState(0);\n const [open, setOpen] = useState(false);\n const debounceRef = useRef<NodeJS.Timeout | null>(null);\n\n // TODO(thure): This is a workaround for the race condition between displaying a Popover and either rendering\n // the anchor further down the tree or measuring the virtual trigger's client rect.\n useEffect(() => {\n setOpen(false);\n if (state.popoverOpen) {\n if (debounceRef.current) {\n clearTimeout(debounceRef.current);\n }\n if (state.popoverAnchor && virtualRef.current !== state.popoverAnchor) {\n virtualRef.current = state.popoverAnchor ?? null;\n setVirtualIter((iter) => iter + 1);\n }\n debounceRef.current = setTimeout(() => setOpen(true), DEBOUNCE_DELAY);\n }\n }, [state.popoverOpen, state.popoverAnchorId, state.popoverAnchor, state.popoverContent]);\n\n return (\n <DeckPopoverProvider setOpen={setOpen}>\n <Popover.Root modal={false} open={open}>\n {state.popoverAnchor && <Popover.VirtualTrigger key={virtualIter} virtualRef={virtualRef} />}\n {children}\n </Popover.Root>\n </DeckPopoverProvider>\n );\n};\n\nexport const PopoverContent = () => {\n const { t } = useTranslation(meta.id);\n const { state, updateEphemeral } = useDeckState();\n const { setOpen } = useDeckPopoverContext('PopoverContent');\n const popoverSubject =\n state.popoverContent && 'subject' in state.popoverContent ? state.popoverContent.subject : undefined;\n const isObjectPopover = Obj.isObject(popoverSubject);\n const objectMenuItems = useObjectMenuItems(popoverSubject);\n const title = state.popoverTitle ? toLocalizedString(state.popoverTitle, t) : 'Unknown';\n const icon = isObjectPopover ? (Obj.getIcon(popoverSubject)?.icon ?? 'ph--circle-dashed--regular') : undefined;\n const content = state.popoverContent;\n // A base popover renders a plugin-provided component; everything else falls through to the card.\n const isBasePopover = state.popoverKind === 'base' && !!content && 'component' in content;\n\n const handleClose = useCallback(() => {\n setOpen(false);\n updateEphemeral((state) => ({\n ...state,\n popoverOpen: false,\n popoverAnchor: undefined,\n popoverAnchorId: undefined,\n popoverSide: undefined,\n }));\n }, [updateEphemeral]);\n\n const handleInteractOutside = useCallback(\n (event: KeyboardEvent | PopoverContentInteractOutsideEvent) => {\n if (\n // TODO(thure): CodeMirror should not focus itself when it updates.\n event.type === 'dismissableLayer.focusOutside' &&\n (event.currentTarget as HTMLElement | undefined)?.classList.contains('cm-content')\n ) {\n event.preventDefault();\n } else {\n handleClose();\n }\n },\n [handleClose],\n );\n\n return (\n <Popover.Portal>\n <Popover.Content\n side={state.popoverSide}\n sticky='always'\n hideWhenDetached\n onOpenAutoFocus={(event) => event.preventDefault()}\n onInteractOutside={handleInteractOutside}\n onEscapeKeyDown={handleInteractOutside}\n >\n <Popover.Viewport>\n {isBasePopover && content && 'component' in content ? (\n /* Base popover: a plugin-provided component (e.g. editor link preview). */\n <Surface.Surface type={AppSurface.Popover} data={content} limit={1} />\n ) : (\n /*\n * Card popover (default). Rendered for any open popover that isn't an explicit\n * base-component popover so the popover can never collapse to a bare 1px frame: the\n * header (icon + title + menu) always renders, and the body falls back to a fixed-\n * height \"no preview\" row when no subject resolves a card Surface (e.g. system-type\n * objects like a raw Feed that have no registered card and no renderable fields).\n */\n <Menu.Root>\n <Card.Root border={false} classNames='dx-card-popover'>\n <Card.Header>\n <Card.IconBlock>{icon && <Card.Icon icon={icon} />}</Card.IconBlock>\n <Card.Title>{title}</Card.Title>\n {/* TODO(wittjosiah): Reconcile with Card.Menu. */}\n <Card.IconBlock>\n <Menu.Trigger asChild disabled={!objectMenuItems.length}>\n <Toolbar.IconButton\n variant='ghost'\n density='sm'\n icon='ph--dots-three-vertical--regular'\n iconOnly\n label='Actions'\n />\n </Menu.Trigger>\n <Menu.Content items={objectMenuItems} />\n </Card.IconBlock>\n </Card.Header>\n\n {content && 'subject' in content ? (\n <Surface.Surface type={AppSurface.Card} data={content} limit={1} />\n ) : (\n <Card.Body classNames='min-bs-8'>\n <Card.Row>\n <Card.Text variant='description'>{t('popover-no-preview.message')}</Card.Text>\n </Card.Row>\n </Card.Body>\n )}\n </Card.Root>\n </Menu.Root>\n )}\n </Popover.Viewport>\n <Popover.Arrow />\n </Popover.Content>\n </Popover.Portal>\n );\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport React, { useState } from 'react';\n\nimport { type LayoutOperation } from '@dxos/app-toolkit';\nimport { Button, Toast as NaturalToast, type ToastRootProps, toLocalizedString, useTranslation } from '@dxos/react-ui';\n\nimport { meta } from '#meta';\n\n// TODO(wittjosiah): Render remaining duration as a progress bar within the toast.\nexport const Toast = ({\n id,\n title,\n description,\n icon,\n duration,\n actionLabel,\n actionAlt,\n onAction,\n onOpenChange,\n}: LayoutOperation.Toast & Pick<ToastRootProps, 'onOpenChange'>) => {\n const { t } = useTranslation(meta.id);\n\n // Control the open state so closing flips Radix's `open` (playing the exit animation) rather than\n // unmounting abruptly. Both the close button and Radix's own timeout/swipe route through here.\n const [open, setOpen] = useState(true);\n const handleOpenChange = (next: boolean) => {\n setOpen(next);\n onOpenChange?.(next);\n };\n\n return (\n <NaturalToast.Root data-testid={id} open={open} duration={duration} onOpenChange={handleOpenChange}>\n <NaturalToast.Title icon={icon} onClose={() => handleOpenChange(false)}>\n {title && <span>{toLocalizedString(title, t)}</span>}\n </NaturalToast.Title>\n {description && <NaturalToast.Description>{toLocalizedString(description, t)}</NaturalToast.Description>}\n {onAction && actionAlt && actionLabel && (\n <NaturalToast.Actions>\n <NaturalToast.Action altText={toLocalizedString(actionAlt, t)} asChild>\n <Button data-testid='toast.action' variant='primary' onClick={() => onAction?.()}>\n {toLocalizedString(actionLabel, t)}\n </Button>\n </NaturalToast.Action>\n </NaturalToast.Actions>\n )}\n </NaturalToast.Root>\n );\n};\n\nexport type ToasterProps = {\n toasts?: LayoutOperation.Toast[];\n onDismissToast?: (id: string) => void;\n};\n\nexport const Toaster = ({ toasts, onDismissToast }: ToasterProps) => {\n return (\n <>\n {toasts?.map((toast) => (\n <Toast\n {...toast}\n key={toast.id}\n onOpenChange={(open: boolean) => {\n if (!open) {\n onDismissToast?.(toast.id);\n }\n\n return open;\n }}\n />\n ))}\n </>\n );\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nexport const NAV_ID = 'NavTree';\n\nexport const SURFACE_PREFIX = 'surface:';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { DeckLayout } from './DeckLayout';\n\nexport { NAV_ID } from './constants';\n\nexport default DeckLayout;\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["React", "useCallback", "useAtomCapability", "useOperationInvoker", "usePluginManager", "LayoutOperation", "Mosaic", "useDeckState", "DeckCapabilities", "getMode", "React", "Surface", "AppSurface", "useAppGraph", "useNode", "useAttended", "useNodeActionExpander", "ActiveNode", "id", "graph", "activeNode", "div", "className", "type", "DocumentTitle", "data", "subject", "limit", "React", "useCallback", "Surface", "AppSurface", "AlertDialog", "Dialog", "NaturalDialog", "useDeckState", "Dialog", "state", "updateEphemeral", "useDeckState", "dialogOpen", "dialogType", "dialogBlockAlign", "dialogOverlayClasses", "dialogOverlayStyle", "dialogContent", "Root", "AlertDialog", "NaturalDialog", "Overlay", "handleOpenChange", "useCallback", "nextOpen", "s", "React", "modal", "open", "onOpenChange", "Surface", "type", "AppSurface", "data", "undefined", "limit", "fallback", "PlankErrorFallback", "placeholder", "div", "blockAlign", "classNames", "style", "createContext", "React", "useCallback", "useEffect", "useRef", "useState", "Surface", "AppSurface", "useObjectMenuItems", "Obj", "Card", "Popover", "toLocalizedString", "Toolbar", "useTranslation", "Menu", "useDeckState", "meta", "DEBOUNCE_DELAY", "DeckPopoverProvider", "useDeckPopoverContext", "PopoverRoot", "children", "state", "virtualRef", "virtualIter", "setVirtualIter", "open", "setOpen", "debounceRef", "popoverOpen", "current", "clearTimeout", "popoverAnchor", "iter", "setTimeout", "popoverAnchorId", "popoverContent", "Root", "modal", "VirtualTrigger", "key", "PopoverContent", "t", "id", "updateEphemeral", "popoverSubject", "subject", "undefined", "isObjectPopover", "isObject", "objectMenuItems", "title", "popoverTitle", "icon", "getIcon", "content", "isBasePopover", "popoverKind", "handleClose", "popoverSide", "handleInteractOutside", "event", "type", "currentTarget", "classList", "contains", "preventDefault", "Portal", "Content", "side", "sticky", "hideWhenDetached", "onOpenAutoFocus", "onInteractOutside", "onEscapeKeyDown", "Viewport", "data", "limit", "border", "classNames", "Header", "IconBlock", "Icon", "Title", "Trigger", "asChild", "disabled", "length", "IconButton", "variant", "density", "iconOnly", "label", "items", "Body", "Row", "Text", "Arrow", "React", "useState", "Button", "Toast", "NaturalToast", "toLocalizedString", "useTranslation", "meta", "id", "title", "description", "icon", "duration", "actionLabel", "actionAlt", "onAction", "onOpenChange", "t", "open", "setOpen", "handleOpenChange", "next", "Root", "data-testid", "Title", "onClose", "span", "Description", "Actions", "Action", "altText", "asChild", "variant", "onClick", "Toaster", "toasts", "onDismissToast", "map", "toast", "key", "DeckLayout", "onDismissToast", "settings", "useAtomCapability", "DeckCapabilities", "Settings", "pluginManager", "usePluginManager", "invokePromise", "useOperationInvoker", "deck", "state", "updateState", "useDeckState", "layoutMode", "getMode", "toasts", "handleLayoutChange", "useCallback", "request", "LayoutOperation", "SetLayoutMode", "React", "Mosaic", "Root", "PopoverRoot", "ActiveNode", "Deck", "onLayoutChange", "Content", "Viewport", "solo", "SoloMode", "active", "length", "ContentEmpty", "MultiMode", "PopoverContent", "Dialog", "Toaster", "NAV_ID", "DeckLayout"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import "./chunk-J5LGTIGS.mjs";
|
|
2
|
+
|
|
3
|
+
// src/DeckPlugin.ts
|
|
4
|
+
import { setAutoFreeze } from "immer";
|
|
5
|
+
import { ActivationEvent, ActivationEvents, Plugin } from "@dxos/app-framework";
|
|
6
|
+
import { AppActivationEvents, AppPlugin } from "@dxos/app-toolkit";
|
|
7
|
+
import { translations as stackTranslations } from "@dxos/react-ui-stack/translations";
|
|
8
|
+
import { AppGraphBuilder, CheckAppScheme, DeckSettings, DeckState, NotificationTracker, OperationHandler, ReactRoot, ReactSurface, UrlHandler } from "#capabilities";
|
|
9
|
+
import { meta } from "#meta";
|
|
10
|
+
import { translations } from "#translations";
|
|
11
|
+
import { DeckEvents } from "#types";
|
|
12
|
+
|
|
13
|
+
// raw-loader:/__w/dxos/dxos/packages/plugins/plugin-deck/PLUGIN.mdl?raw
|
|
14
|
+
var PLUGIN_default = "---\nid: org.dxos.plugin.deck\nname: DeckPlugin\nversion: 0.1.0\n---\n\nThe Deck plugin is the core layout engine for DXOS Composer. It manages the multi-plank workspace\n(\"deck\"), the sidebar, the complementary sidebar, dialogs, popovers, and toast notifications.\nIt owns the persisted and ephemeral layout state, handles URL routing, and implements every\n`LayoutOperation` that changes what is visible on screen.\n\n## Extensions\n\nThe following extension dialects are used in this document.\nEach extension is defined in the Appendix or resolved via its URI.\n\n| Term | URI |\n|-------------|--------------------------------|\n| `type` | `org.dxos.mdl.type@1.0` |\n| `feat` | `org.dxos.mdl.feat@1.0` |\n| `test` | `org.dxos.mdl.test@1.0` |\n| `component` | `org.dxos.mdl.component@1.0` |\n| `op` | `org.dxos.mdl.op@1.0` |\n\n## Types\n\n```mdl\ntype LayoutMode\n literals: multi | solo | solo--fullscreen\n```\n\n```mdl\ntype PartAdjustment\n literals: close | companion | solo | solo--fullscreen | increment-start | increment-end\n```\n\n```mdl\ntype DeckState\n fields:\n initialized: boolean # false until the deck has left solo mode for the first time\n active: string[] # item IDs of planks displayed in multi mode\n inactive: string[] # item IDs that have been closed; persisted for reopening\n solo?: string # item ID of the single plank in solo or fullscreen mode\n fullscreen: boolean # true when the solo plank is displayed without any chrome\n plankSizing: Record<string, number> # persisted plank widths in rem, keyed by item ID\n companionOpen: boolean # whether the companion pane is visible\n companionVariant?: string # which companion variant to display\n companionFrameSizing: Record<string, number> # companion frame widths in rem\n```\n\n```mdl\ntype StoredDeckState\n desc: Persisted plugin state stored in KVS/localStorage.\n fields:\n sidebarState: closed | collapsed | expanded\n complementarySidebarState: closed | collapsed | expanded\n complementarySidebarPanel?: string\n activeDeck: string\n previousDeck: string\n decks: Record<string, DeckState>\n previousMode: Record<string, LayoutMode>\n```\n\n```mdl\ntype EphemeralDeckState\n desc: Transient plugin state that is NOT persisted across sessions.\n fields:\n dialogOpen: boolean\n dialogType?: default | alert\n dialogBlockAlign?: start | center | end\n dialogContent?: { component: string; props?: any }\n popoverOpen: boolean\n popoverSide?: top | right | bottom | left\n popoverKind?: base | card\n popoverTitle?: string\n popoverContentRef?: string\n popoverContent?: { component: string } | { subject: any }\n toasts: Toast[]\n currentUndoId?: string\n scrollIntoView?: string\n```\n\n```mdl\ntype DeckSettings\n fields:\n enableDeck?: boolean # display multiple panels side by side\n encapsulatedPlanks?: boolean # render each plank in an isolated container\n showHints?: boolean # show keyboard shortcut hints in the UI\n enableNativeRedirect?: boolean # redirect supported URLs to the native desktop app\n```\n\n## Components\n\n```mdl\ncomponent Matrix\n desc: |\n The primary workspace canvas. Renders the active planks side by side in multi mode,\n or a single plank in solo/fullscreen mode. Manages plank-resize handles and the\n companion pane alongside each plank.\n props:\n state: StoredDeckState # persisted layout state atom\n ephemeral: EphemeralDeckState # transient state atom\n state:\n mode: LayoutMode # current layout mode derived from state\n actions:\n openPlank(id: string)\n closePlank(id: string)\n adjustPlank(id: string, type: PartAdjustment)\n resizePlank(id: string, size: number)\n layout: |\n \u250C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u252C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u252C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2510\n \u2502 sidebar \u2502 plank A \u2502 plank B \u2502\n \u2502 \u2502 \u2502 \u2502\n \u2502 \u2502 [content] \u2502 [content] \u2502\n \u2502 \u2502 \u2502 \u2502\n \u2502 \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524 \u2502\n \u2502 \u2502 companion \u2502 \u2502\n \u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n```\n\n```mdl\ncomponent DeckSettings\n desc: Settings panel rendered in the Composer settings surface.\n props:\n settings: DeckSettings\n actions:\n updateSetting(key: string, value: any)\n```\n\n## Operations\n\n```mdl\nop Open\n desc: |\n Opens one or more subjects (item IDs or navigation paths) in the deck.\n In multi mode uses stack semantics: truncates after the pivot then appends.\n In solo or uninitialised mode, replaces the current subject entirely.\n Validates each target against the app graph and redirects to a 404 node if not found.\n Fires ScrollIntoView, Expose, and an observability event for each newly opened item.\n input:\n subject: string[] # item IDs or navigation paths to open\n pivotId?: string # truncate deck after this item before appending\n key?: string # replace an existing plank whose ID shares this key prefix\n workspace?: string # switch to this workspace before opening\n navigation?: immediate # skip validation; expand path only\n scrollIntoView?: boolean # default true; set false to suppress auto-scroll\n state?: any\n variant?: string\n output: string[]\n effects: [echo:read, layout:state]\n requires: [AppGraph, AttentionCapabilities, ClientCapabilities, DeckCapabilities]\n```\n\n```mdl\nop Close\n desc: |\n Removes one or more subjects from the active deck.\n Computes which plank (if any) should receive attention after removal and\n schedules ScrollIntoView for it.\n input:\n subject: string[]\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities, AttentionCapabilities]\n```\n\n```mdl\nop SetLayoutMode\n desc: |\n Transitions to a new layout mode (multi, solo, solo--fullscreen) or reverts\n to the previously recorded mode. Persists the previous mode so it can be\n restored later. Toggling solo--fullscreen flips the fullscreen flag rather\n than unconditionally setting it.\n input:\n mode?: LayoutMode # target mode; omit if reverting\n subject?: string # item to solo; used only for solo / solo--fullscreen\n revert?: boolean # revert to the previously persisted mode\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateSidebar\n desc: Update the sidebar visibility state (closed / collapsed / expanded).\n input:\n state?: closed | collapsed | expanded\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateComplementary\n desc: Update the complementary-sidebar visibility state and optional panel selection.\n input:\n state?: closed | collapsed | expanded\n panel?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateDialog\n desc: |\n Show or hide the global dialog overlay. Accepts a surface component identifier\n or a boolean state flag. Propagates to EphemeralDeckState.\n input:\n subject?: string # surface component identifier to display\n props?: any # props forwarded to the dialog surface\n state?: boolean # false to close the dialog\n type?: default | alert\n blockAlign?: start | center | end\n overlayClasses?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdatePopover\n desc: |\n Show or hide the global popover. Accepts anchor, content reference, or\n an explicit subject reference.\n input:\n subject?: any # subject reference to pass to the popover surface\n contentRef?: string # surface component identifier for popover content\n anchorId?: string # DOM element ID for positioning the popover anchor\n side?: top | right | bottom | left\n kind?: base | card\n title?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateCompanion\n desc: Update the companion-pane open state and active variant.\n input:\n open?: boolean\n variant?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop AdjustPlank\n desc: Apply a PartAdjustment to a specific plank (close, solo, move, companion).\n input:\n id: string\n type: PartAdjustment\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdatePlankSize\n desc: Persist the width of a plank in rem.\n input:\n id: string\n size: number\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop AddToast\n desc: Push a toast notification into the ephemeral queue.\n input:\n title?: string\n body?: string\n duration?: number\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop ScrollIntoView\n desc: Schedule a subject to be scrolled into view when its plank mounts.\n input:\n subject: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop SwitchWorkspace\n desc: Switch the active workspace deck, persisting the previous workspace ID.\n input:\n subject: string # workspace ID to switch to\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop RevertWorkspace\n desc: Revert to the previously active workspace.\n input: void\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n## Features\n\n```mdl\nfeat F-1: Multi-Plank Layout (Deck)\n\n req F-1.1:\n when: enableDeck setting is true\n then: deck renders active planks side by side in a scrollable row\n\n req F-1.2:\n when: user opens a new subject with a pivot\n then: deck is truncated after the pivot and the new subject is appended\n\n req F-1.3:\n when: subject is already present in the active deck\n then: deck is unchanged and the existing plank scrolls into view\n\n req F-1.4:\n when: user resizes a plank\n then: new width is persisted in plankSizing and restored on next load\n```\n\n```mdl\nfeat F-2: Solo / Fullscreen Mode\n\n req F-2.1:\n when: user triggers SetLayoutMode with mode=solo\n then: only the targeted plank is visible; other planks move to inactive\n\n req F-2.2:\n when: user triggers SetLayoutMode with mode=solo--fullscreen\n then: plank fills the viewport; sidebar and heading chrome are hidden\n\n req F-2.3:\n when: user triggers SetLayoutMode with revert=true\n then: layout returns to the previously persisted mode\n```\n\n```mdl\nfeat F-3: Sidebar & Complementary Sidebar\n\n req F-3.1:\n when: UpdateSidebar is dispatched with a new state\n then: sidebar transitions between closed / collapsed / expanded and state is persisted\n\n req F-3.2:\n when: UpdateComplementary is dispatched with a panel\n then: complementary sidebar opens to the specified panel\n```\n\n```mdl\nfeat F-4: Dialog & Popover Overlay\n\n req F-4.1:\n when: UpdateDialog is dispatched with a subject component\n then: the matching surface is rendered inside the global dialog overlay\n\n req F-4.2:\n when: UpdatePopover is dispatched with an anchorId\n then: popover is positioned relative to that DOM element\n\n req F-4.3:\n when: dialog or popover state is set to false\n then: overlay is closed and content is unmounted\n```\n\n```mdl\nfeat F-5: URL Navigation & Deep Linking\n\n req F-5.1:\n when: app loads with a URL referencing an item\n then: DeckPlugin resolves the path and opens the item in the deck\n\n req F-5.2:\n when: active deck changes\n then: browser URL is updated to reflect the current plank state\n\n req F-5.3:\n when: a navigation target is not found in the app graph\n then: Open operation redirects to a 404 node\n```\n\n```mdl\nfeat F-6: Toast Notifications\n\n req F-6.1:\n when: AddToast is dispatched\n then: a toast is appended to the queue and displayed for its configured duration\n\n req F-6.2:\n when: ShowUndo is dispatched\n then: a toast with an undo action is shown; invoking it executes the linked undo operation\n```\n\n## Acceptance\n\n```mdl\ntest T-1: Open subject in multi mode\n given: deck is in multi mode with planks A and B active\n when: Open({ subject: ['C'] }) is dispatched\n then:\n - plank C is appended to the right of plank B\n - browser URL reflects the new deck state\n - plank C is scrolled into view\n```\n\n```mdl\ntest T-2: Open with pivot truncation\n given: deck has planks [A, B, C] active\n when: Open({ subject: ['D'], pivotId: 'A' }) is dispatched\n then:\n - active deck becomes [A, D]\n - planks B and C are moved to inactive\n```\n\n```mdl\ntest T-3: Open already-visible subject\n given: deck has plank A active\n when: Open({ subject: ['A'] }) is dispatched\n then:\n - active deck is unchanged\n - plank A scrolls into view\n```\n\n```mdl\ntest T-4: Solo mode\n given: deck is in multi mode\n when: SetLayoutMode({ mode: 'solo', subject: 'A' }) is dispatched\n then:\n - only plank A is visible\n - mode is stored as solo in StoredDeckState\n - previousMode records 'multi' for the active deck\n```\n\n```mdl\ntest T-5: Revert from solo to multi\n given: layout mode is solo, previousMode is multi\n when: SetLayoutMode({ revert: true }) is dispatched\n then:\n - deck returns to multi mode\n - previously inactive planks are restored\n```\n\n```mdl\ntest T-6: Close plank\n given: deck has planks [A, B, C] active; B has attention\n when: Close({ subject: ['B'] }) is dispatched\n then:\n - active deck becomes [A, C]\n - attention moves to an adjacent plank\n - ScrollIntoView is scheduled for the newly attended plank\n```\n\n```mdl\ntest T-7: Toast is shown and dismissed\n given: no toasts are in the queue\n when: AddToast({ title: 'Saved', duration: 3000 }) is dispatched\n then:\n - toast appears in the UI\n - toast is removed after 3 seconds\n```\n\n---\n\n## Appendix: Extension Definitions\n\nExtension block types used in this document are defined below using\nthe core `ext` primitive \u2014 the only construct the base language provides.\n\n```mdl\next type\n uri: org.dxos.mdl.type@1.0\n desc: A named data structure with typed fields and optional literals.\n fields:\n desc?: Prose\n fields?: FieldMap # name[?]: TypeExpr (# inline comment)\n literals?: UnionList # a | b | c\n extends?: TypeRef[]\n```\n\n```mdl\next feat\n uri: org.dxos.mdl.feat@1.0\n desc: A named feature grouping one or more requirements.\n fields:\n desc?: Prose\n req: RequirementList\n nesting: self # feat blocks may contain feat blocks\n```\n\n```mdl\next test\n uri: org.dxos.mdl.test@1.0\n desc: An acceptance scenario expressed as given / when / then steps.\n fields:\n given?: Step | Step[]\n when?: Step | Step[]\n then: Step | Step[]\n tags?: TagList\n```\n\n```mdl\next component\n uri: org.dxos.mdl.component@1.0\n desc: A UI component with props, internal state, slots, actions, and events.\n fields:\n desc?: Prose\n props?: FieldMap # external inputs (immutable inside component)\n state?: FieldMap # internal reactive state\n slots?: FieldMap # named ReactNode injection points\n actions?: ActionMap # methods the component exposes or handles\n emits?: EventMap # events the component raises to its parent\n layout?: CodeBlock # ASCII sketch of visual structure (non-normative)\n```\n\n```mdl\next op\n uri: org.dxos.mdl.op@1.0\n desc: |\n A named operation with typed inputs, outputs, and declared errors.\n Pure ops have no effects or requires. Effectful ops declare both.\n fields:\n desc?: Prose\n input?: FieldMap # named input parameters\n output?: TypeExpr # return type\n errors?: ErrorMap # name: Prose (when this error occurs)\n effects?: EffectList # echo:read | echo:write | http | fs | ...\n requires?: ServiceList # injected service dependencies\n note?: Prose # implementation guidance (non-normative)\n```\n";
|
|
15
|
+
|
|
16
|
+
// src/DeckPlugin.ts
|
|
17
|
+
setAutoFreeze(false);
|
|
18
|
+
var DeckPlugin = Plugin.define(meta).pipe(
|
|
19
|
+
AppPlugin.addAppGraphModule({
|
|
20
|
+
activate: AppGraphBuilder
|
|
21
|
+
}),
|
|
22
|
+
AppPlugin.addOperationHandlerModule({
|
|
23
|
+
activate: OperationHandler
|
|
24
|
+
}),
|
|
25
|
+
AppPlugin.addSurfaceModule({
|
|
26
|
+
activate: ReactSurface
|
|
27
|
+
}),
|
|
28
|
+
AppPlugin.addTranslationsModule({
|
|
29
|
+
translations: [
|
|
30
|
+
...translations,
|
|
31
|
+
...stackTranslations
|
|
32
|
+
]
|
|
33
|
+
}),
|
|
34
|
+
Plugin.addModule({
|
|
35
|
+
activatesOn: AppActivationEvents.SetupSettings,
|
|
36
|
+
firesAfterActivation: [
|
|
37
|
+
DeckEvents.SettingsReady
|
|
38
|
+
],
|
|
39
|
+
activate: DeckSettings
|
|
40
|
+
}),
|
|
41
|
+
Plugin.addModule({
|
|
42
|
+
activatesOn: ActivationEvent.allOf(DeckEvents.SettingsReady, ActivationEvents.ProcessManagerReady),
|
|
43
|
+
activate: CheckAppScheme
|
|
44
|
+
}),
|
|
45
|
+
Plugin.addModule({
|
|
46
|
+
// TODO(wittjosiah): Does not integrate with settings store.
|
|
47
|
+
// Should this be a different event?
|
|
48
|
+
// Should settings store be renamed to be more generic?
|
|
49
|
+
activatesOn: ActivationEvent.oneOf(AppActivationEvents.SetupSettings, AppActivationEvents.SetupAppGraph),
|
|
50
|
+
firesAfterActivation: [
|
|
51
|
+
AppActivationEvents.LayoutReady,
|
|
52
|
+
DeckEvents.StateReady
|
|
53
|
+
],
|
|
54
|
+
activate: DeckState
|
|
55
|
+
}),
|
|
56
|
+
Plugin.addModule({
|
|
57
|
+
activatesOn: ActivationEvents.Startup,
|
|
58
|
+
activate: ReactRoot
|
|
59
|
+
}),
|
|
60
|
+
// Plugin.addModule({
|
|
61
|
+
// activatesOn: Events.SetupArtifactDefinition,
|
|
62
|
+
// activate: Tools,
|
|
63
|
+
// }),
|
|
64
|
+
Plugin.addModule({
|
|
65
|
+
activatesOn: ActivationEvent.allOf(ActivationEvents.ProcessManagerReady, DeckEvents.StateReady),
|
|
66
|
+
activate: UrlHandler
|
|
67
|
+
}),
|
|
68
|
+
Plugin.addModule({
|
|
69
|
+
activatesOn: ActivationEvent.allOf(ActivationEvents.ProcessManagerReady, DeckEvents.StateReady),
|
|
70
|
+
activate: NotificationTracker
|
|
71
|
+
}),
|
|
72
|
+
AppPlugin.addPluginAssetModule({
|
|
73
|
+
asset: {
|
|
74
|
+
pluginId: meta.id,
|
|
75
|
+
path: "PLUGIN.mdl",
|
|
76
|
+
content: PLUGIN_default,
|
|
77
|
+
mimeType: "application/x-mdl"
|
|
78
|
+
}
|
|
79
|
+
}),
|
|
80
|
+
Plugin.make
|
|
81
|
+
);
|
|
82
|
+
var DeckPlugin_default = DeckPlugin;
|
|
83
|
+
export {
|
|
84
|
+
DeckPlugin,
|
|
85
|
+
DeckPlugin_default as default
|
|
86
|
+
};
|
|
87
|
+
//# sourceMappingURL=DeckPlugin.mjs.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/DeckPlugin.ts", "raw-loader:/__w/dxos/dxos/packages/plugins/plugin-deck/PLUGIN.mdl?raw"],
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2025 DXOS.org\n//\n\nimport { setAutoFreeze } from 'immer';\n\nimport { ActivationEvent, ActivationEvents, Plugin } from '@dxos/app-framework';\nimport { AppActivationEvents, AppPlugin } from '@dxos/app-toolkit';\nimport { translations as stackTranslations } from '@dxos/react-ui-stack/translations';\n\nimport {\n AppGraphBuilder,\n CheckAppScheme,\n DeckSettings,\n DeckState,\n NotificationTracker,\n OperationHandler,\n ReactRoot,\n ReactSurface,\n UrlHandler,\n} from '#capabilities';\nimport { meta } from '#meta';\nimport { translations } from '#translations';\nimport { DeckEvents } from '#types';\n\n// eslint-disable-next-line import/no-relative-packages\nimport pluginSpec from '../PLUGIN.mdl?raw';\n\n// NOTE(Zan): When producing values with immer, we shouldn't auto-freeze them because\n// our signal implementation needs to add some hidden properties to the produced values.\n// TODO(Zan): Move this to a more global location if we use immer more broadly.\nsetAutoFreeze(false);\n\nexport const DeckPlugin = Plugin.define(meta).pipe(\n AppPlugin.addAppGraphModule({ activate: AppGraphBuilder }),\n AppPlugin.addOperationHandlerModule({ activate: OperationHandler }),\n AppPlugin.addSurfaceModule({ activate: ReactSurface }),\n AppPlugin.addTranslationsModule({ translations: [...translations, ...stackTranslations] }),\n Plugin.addModule({\n activatesOn: AppActivationEvents.SetupSettings,\n firesAfterActivation: [DeckEvents.SettingsReady],\n activate: DeckSettings,\n }),\n Plugin.addModule({\n activatesOn: ActivationEvent.allOf(DeckEvents.SettingsReady, ActivationEvents.ProcessManagerReady),\n activate: CheckAppScheme,\n }),\n Plugin.addModule({\n // TODO(wittjosiah): Does not integrate with settings store.\n // Should this be a different event?\n // Should settings store be renamed to be more generic?\n activatesOn: ActivationEvent.oneOf(AppActivationEvents.SetupSettings, AppActivationEvents.SetupAppGraph),\n firesAfterActivation: [AppActivationEvents.LayoutReady, DeckEvents.StateReady],\n activate: DeckState,\n }),\n Plugin.addModule({\n activatesOn: ActivationEvents.Startup,\n activate: ReactRoot,\n }),\n // Plugin.addModule({\n // activatesOn: Events.SetupArtifactDefinition,\n // activate: Tools,\n // }),\n Plugin.addModule({\n activatesOn: ActivationEvent.allOf(ActivationEvents.ProcessManagerReady, DeckEvents.StateReady),\n activate: UrlHandler,\n }),\n Plugin.addModule({\n activatesOn: ActivationEvent.allOf(ActivationEvents.ProcessManagerReady, DeckEvents.StateReady),\n activate: NotificationTracker,\n }),\n AppPlugin.addPluginAssetModule({\n asset: { pluginId: meta.id, path: 'PLUGIN.mdl', content: pluginSpec, mimeType: 'application/x-mdl' },\n }),\n Plugin.make,\n);\n\nexport default DeckPlugin;\n", "---\nid: org.dxos.plugin.deck\nname: DeckPlugin\nversion: 0.1.0\n---\n\nThe Deck plugin is the core layout engine for DXOS Composer. It manages the multi-plank workspace\n(\"deck\"), the sidebar, the complementary sidebar, dialogs, popovers, and toast notifications.\nIt owns the persisted and ephemeral layout state, handles URL routing, and implements every\n`LayoutOperation` that changes what is visible on screen.\n\n## Extensions\n\nThe following extension dialects are used in this document.\nEach extension is defined in the Appendix or resolved via its URI.\n\n| Term | URI |\n|-------------|--------------------------------|\n| `type` | `org.dxos.mdl.type@1.0` |\n| `feat` | `org.dxos.mdl.feat@1.0` |\n| `test` | `org.dxos.mdl.test@1.0` |\n| `component` | `org.dxos.mdl.component@1.0` |\n| `op` | `org.dxos.mdl.op@1.0` |\n\n## Types\n\n```mdl\ntype LayoutMode\n literals: multi | solo | solo--fullscreen\n```\n\n```mdl\ntype PartAdjustment\n literals: close | companion | solo | solo--fullscreen | increment-start | increment-end\n```\n\n```mdl\ntype DeckState\n fields:\n initialized: boolean # false until the deck has left solo mode for the first time\n active: string[] # item IDs of planks displayed in multi mode\n inactive: string[] # item IDs that have been closed; persisted for reopening\n solo?: string # item ID of the single plank in solo or fullscreen mode\n fullscreen: boolean # true when the solo plank is displayed without any chrome\n plankSizing: Record<string, number> # persisted plank widths in rem, keyed by item ID\n companionOpen: boolean # whether the companion pane is visible\n companionVariant?: string # which companion variant to display\n companionFrameSizing: Record<string, number> # companion frame widths in rem\n```\n\n```mdl\ntype StoredDeckState\n desc: Persisted plugin state stored in KVS/localStorage.\n fields:\n sidebarState: closed | collapsed | expanded\n complementarySidebarState: closed | collapsed | expanded\n complementarySidebarPanel?: string\n activeDeck: string\n previousDeck: string\n decks: Record<string, DeckState>\n previousMode: Record<string, LayoutMode>\n```\n\n```mdl\ntype EphemeralDeckState\n desc: Transient plugin state that is NOT persisted across sessions.\n fields:\n dialogOpen: boolean\n dialogType?: default | alert\n dialogBlockAlign?: start | center | end\n dialogContent?: { component: string; props?: any }\n popoverOpen: boolean\n popoverSide?: top | right | bottom | left\n popoverKind?: base | card\n popoverTitle?: string\n popoverContentRef?: string\n popoverContent?: { component: string } | { subject: any }\n toasts: Toast[]\n currentUndoId?: string\n scrollIntoView?: string\n```\n\n```mdl\ntype DeckSettings\n fields:\n enableDeck?: boolean # display multiple panels side by side\n encapsulatedPlanks?: boolean # render each plank in an isolated container\n showHints?: boolean # show keyboard shortcut hints in the UI\n enableNativeRedirect?: boolean # redirect supported URLs to the native desktop app\n```\n\n## Components\n\n```mdl\ncomponent Matrix\n desc: |\n The primary workspace canvas. Renders the active planks side by side in multi mode,\n or a single plank in solo/fullscreen mode. Manages plank-resize handles and the\n companion pane alongside each plank.\n props:\n state: StoredDeckState # persisted layout state atom\n ephemeral: EphemeralDeckState # transient state atom\n state:\n mode: LayoutMode # current layout mode derived from state\n actions:\n openPlank(id: string)\n closePlank(id: string)\n adjustPlank(id: string, type: PartAdjustment)\n resizePlank(id: string, size: number)\n layout: |\n \u250C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u252C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u252C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2510\n \u2502 sidebar \u2502 plank A \u2502 plank B \u2502\n \u2502 \u2502 \u2502 \u2502\n \u2502 \u2502 [content] \u2502 [content] \u2502\n \u2502 \u2502 \u2502 \u2502\n \u2502 \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524 \u2502\n \u2502 \u2502 companion \u2502 \u2502\n \u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n```\n\n```mdl\ncomponent DeckSettings\n desc: Settings panel rendered in the Composer settings surface.\n props:\n settings: DeckSettings\n actions:\n updateSetting(key: string, value: any)\n```\n\n## Operations\n\n```mdl\nop Open\n desc: |\n Opens one or more subjects (item IDs or navigation paths) in the deck.\n In multi mode uses stack semantics: truncates after the pivot then appends.\n In solo or uninitialised mode, replaces the current subject entirely.\n Validates each target against the app graph and redirects to a 404 node if not found.\n Fires ScrollIntoView, Expose, and an observability event for each newly opened item.\n input:\n subject: string[] # item IDs or navigation paths to open\n pivotId?: string # truncate deck after this item before appending\n key?: string # replace an existing plank whose ID shares this key prefix\n workspace?: string # switch to this workspace before opening\n navigation?: immediate # skip validation; expand path only\n scrollIntoView?: boolean # default true; set false to suppress auto-scroll\n state?: any\n variant?: string\n output: string[]\n effects: [echo:read, layout:state]\n requires: [AppGraph, AttentionCapabilities, ClientCapabilities, DeckCapabilities]\n```\n\n```mdl\nop Close\n desc: |\n Removes one or more subjects from the active deck.\n Computes which plank (if any) should receive attention after removal and\n schedules ScrollIntoView for it.\n input:\n subject: string[]\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities, AttentionCapabilities]\n```\n\n```mdl\nop SetLayoutMode\n desc: |\n Transitions to a new layout mode (multi, solo, solo--fullscreen) or reverts\n to the previously recorded mode. Persists the previous mode so it can be\n restored later. Toggling solo--fullscreen flips the fullscreen flag rather\n than unconditionally setting it.\n input:\n mode?: LayoutMode # target mode; omit if reverting\n subject?: string # item to solo; used only for solo / solo--fullscreen\n revert?: boolean # revert to the previously persisted mode\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateSidebar\n desc: Update the sidebar visibility state (closed / collapsed / expanded).\n input:\n state?: closed | collapsed | expanded\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateComplementary\n desc: Update the complementary-sidebar visibility state and optional panel selection.\n input:\n state?: closed | collapsed | expanded\n panel?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateDialog\n desc: |\n Show or hide the global dialog overlay. Accepts a surface component identifier\n or a boolean state flag. Propagates to EphemeralDeckState.\n input:\n subject?: string # surface component identifier to display\n props?: any # props forwarded to the dialog surface\n state?: boolean # false to close the dialog\n type?: default | alert\n blockAlign?: start | center | end\n overlayClasses?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdatePopover\n desc: |\n Show or hide the global popover. Accepts anchor, content reference, or\n an explicit subject reference.\n input:\n subject?: any # subject reference to pass to the popover surface\n contentRef?: string # surface component identifier for popover content\n anchorId?: string # DOM element ID for positioning the popover anchor\n side?: top | right | bottom | left\n kind?: base | card\n title?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdateCompanion\n desc: Update the companion-pane open state and active variant.\n input:\n open?: boolean\n variant?: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop AdjustPlank\n desc: Apply a PartAdjustment to a specific plank (close, solo, move, companion).\n input:\n id: string\n type: PartAdjustment\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop UpdatePlankSize\n desc: Persist the width of a plank in rem.\n input:\n id: string\n size: number\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop AddToast\n desc: Push a toast notification into the ephemeral queue.\n input:\n title?: string\n body?: string\n duration?: number\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop ScrollIntoView\n desc: Schedule a subject to be scrolled into view when its plank mounts.\n input:\n subject: string\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop SwitchWorkspace\n desc: Switch the active workspace deck, persisting the previous workspace ID.\n input:\n subject: string # workspace ID to switch to\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n```mdl\nop RevertWorkspace\n desc: Revert to the previously active workspace.\n input: void\n output: void\n effects: [layout:state]\n requires: [DeckCapabilities]\n```\n\n## Features\n\n```mdl\nfeat F-1: Multi-Plank Layout (Deck)\n\n req F-1.1:\n when: enableDeck setting is true\n then: deck renders active planks side by side in a scrollable row\n\n req F-1.2:\n when: user opens a new subject with a pivot\n then: deck is truncated after the pivot and the new subject is appended\n\n req F-1.3:\n when: subject is already present in the active deck\n then: deck is unchanged and the existing plank scrolls into view\n\n req F-1.4:\n when: user resizes a plank\n then: new width is persisted in plankSizing and restored on next load\n```\n\n```mdl\nfeat F-2: Solo / Fullscreen Mode\n\n req F-2.1:\n when: user triggers SetLayoutMode with mode=solo\n then: only the targeted plank is visible; other planks move to inactive\n\n req F-2.2:\n when: user triggers SetLayoutMode with mode=solo--fullscreen\n then: plank fills the viewport; sidebar and heading chrome are hidden\n\n req F-2.3:\n when: user triggers SetLayoutMode with revert=true\n then: layout returns to the previously persisted mode\n```\n\n```mdl\nfeat F-3: Sidebar & Complementary Sidebar\n\n req F-3.1:\n when: UpdateSidebar is dispatched with a new state\n then: sidebar transitions between closed / collapsed / expanded and state is persisted\n\n req F-3.2:\n when: UpdateComplementary is dispatched with a panel\n then: complementary sidebar opens to the specified panel\n```\n\n```mdl\nfeat F-4: Dialog & Popover Overlay\n\n req F-4.1:\n when: UpdateDialog is dispatched with a subject component\n then: the matching surface is rendered inside the global dialog overlay\n\n req F-4.2:\n when: UpdatePopover is dispatched with an anchorId\n then: popover is positioned relative to that DOM element\n\n req F-4.3:\n when: dialog or popover state is set to false\n then: overlay is closed and content is unmounted\n```\n\n```mdl\nfeat F-5: URL Navigation & Deep Linking\n\n req F-5.1:\n when: app loads with a URL referencing an item\n then: DeckPlugin resolves the path and opens the item in the deck\n\n req F-5.2:\n when: active deck changes\n then: browser URL is updated to reflect the current plank state\n\n req F-5.3:\n when: a navigation target is not found in the app graph\n then: Open operation redirects to a 404 node\n```\n\n```mdl\nfeat F-6: Toast Notifications\n\n req F-6.1:\n when: AddToast is dispatched\n then: a toast is appended to the queue and displayed for its configured duration\n\n req F-6.2:\n when: ShowUndo is dispatched\n then: a toast with an undo action is shown; invoking it executes the linked undo operation\n```\n\n## Acceptance\n\n```mdl\ntest T-1: Open subject in multi mode\n given: deck is in multi mode with planks A and B active\n when: Open({ subject: ['C'] }) is dispatched\n then:\n - plank C is appended to the right of plank B\n - browser URL reflects the new deck state\n - plank C is scrolled into view\n```\n\n```mdl\ntest T-2: Open with pivot truncation\n given: deck has planks [A, B, C] active\n when: Open({ subject: ['D'], pivotId: 'A' }) is dispatched\n then:\n - active deck becomes [A, D]\n - planks B and C are moved to inactive\n```\n\n```mdl\ntest T-3: Open already-visible subject\n given: deck has plank A active\n when: Open({ subject: ['A'] }) is dispatched\n then:\n - active deck is unchanged\n - plank A scrolls into view\n```\n\n```mdl\ntest T-4: Solo mode\n given: deck is in multi mode\n when: SetLayoutMode({ mode: 'solo', subject: 'A' }) is dispatched\n then:\n - only plank A is visible\n - mode is stored as solo in StoredDeckState\n - previousMode records 'multi' for the active deck\n```\n\n```mdl\ntest T-5: Revert from solo to multi\n given: layout mode is solo, previousMode is multi\n when: SetLayoutMode({ revert: true }) is dispatched\n then:\n - deck returns to multi mode\n - previously inactive planks are restored\n```\n\n```mdl\ntest T-6: Close plank\n given: deck has planks [A, B, C] active; B has attention\n when: Close({ subject: ['B'] }) is dispatched\n then:\n - active deck becomes [A, C]\n - attention moves to an adjacent plank\n - ScrollIntoView is scheduled for the newly attended plank\n```\n\n```mdl\ntest T-7: Toast is shown and dismissed\n given: no toasts are in the queue\n when: AddToast({ title: 'Saved', duration: 3000 }) is dispatched\n then:\n - toast appears in the UI\n - toast is removed after 3 seconds\n```\n\n---\n\n## Appendix: Extension Definitions\n\nExtension block types used in this document are defined below using\nthe core `ext` primitive \u2014 the only construct the base language provides.\n\n```mdl\next type\n uri: org.dxos.mdl.type@1.0\n desc: A named data structure with typed fields and optional literals.\n fields:\n desc?: Prose\n fields?: FieldMap # name[?]: TypeExpr (# inline comment)\n literals?: UnionList # a | b | c\n extends?: TypeRef[]\n```\n\n```mdl\next feat\n uri: org.dxos.mdl.feat@1.0\n desc: A named feature grouping one or more requirements.\n fields:\n desc?: Prose\n req: RequirementList\n nesting: self # feat blocks may contain feat blocks\n```\n\n```mdl\next test\n uri: org.dxos.mdl.test@1.0\n desc: An acceptance scenario expressed as given / when / then steps.\n fields:\n given?: Step | Step[]\n when?: Step | Step[]\n then: Step | Step[]\n tags?: TagList\n```\n\n```mdl\next component\n uri: org.dxos.mdl.component@1.0\n desc: A UI component with props, internal state, slots, actions, and events.\n fields:\n desc?: Prose\n props?: FieldMap # external inputs (immutable inside component)\n state?: FieldMap # internal reactive state\n slots?: FieldMap # named ReactNode injection points\n actions?: ActionMap # methods the component exposes or handles\n emits?: EventMap # events the component raises to its parent\n layout?: CodeBlock # ASCII sketch of visual structure (non-normative)\n```\n\n```mdl\next op\n uri: org.dxos.mdl.op@1.0\n desc: |\n A named operation with typed inputs, outputs, and declared errors.\n Pure ops have no effects or requires. Effectful ops declare both.\n fields:\n desc?: Prose\n input?: FieldMap # named input parameters\n output?: TypeExpr # return type\n errors?: ErrorMap # name: Prose (when this error occurs)\n effects?: EffectList # echo:read | echo:write | http | fs | ...\n requires?: ServiceList # injected service dependencies\n note?: Prose # implementation guidance (non-normative)\n```\n"],
|
|
5
|
+
"mappings": ";;;AAIA,SAASA,qBAAqB;AAE9B,SAASC,iBAAiBC,kBAAkBC,cAAc;AAC1D,SAASC,qBAAqBC,iBAAiB;AAC/C,SAASC,gBAAgBC,yBAAyB;AAElD,SACEC,iBACAC,gBACAC,cACAC,WACAC,qBACAC,kBACAC,WACAC,cACAC,kBACK;AACP,SAASC,YAAY;AACrB,SAASX,oBAAoB;AAC7B,SAASY,kBAAkB;;;ACvB3B;;;AD+BAC,cAAc,KAAA;AAEP,IAAMC,aAAaC,OAAOC,OAAOC,IAAAA,EAAMC;EAC5CC,UAAUC,kBAAkB;IAAEC,UAAUC;EAAgB,CAAA;EACxDH,UAAUI,0BAA0B;IAAEF,UAAUG;EAAiB,CAAA;EACjEL,UAAUM,iBAAiB;IAAEJ,UAAUK;EAAa,CAAA;EACpDP,UAAUQ,sBAAsB;IAAEC,cAAc;SAAIA;SAAiBC;;EAAmB,CAAA;EACxFd,OAAOe,UAAU;IACfC,aAAaC,oBAAoBC;IACjCC,sBAAsB;MAACC,WAAWC;;IAClCf,UAAUgB;EACZ,CAAA;EACAtB,OAAOe,UAAU;IACfC,aAAaO,gBAAgBC,MAAMJ,WAAWC,eAAeI,iBAAiBC,mBAAmB;IACjGpB,UAAUqB;EACZ,CAAA;EACA3B,OAAOe,UAAU;;;;IAIfC,aAAaO,gBAAgBK,MAAMX,oBAAoBC,eAAeD,oBAAoBY,aAAa;IACvGV,sBAAsB;MAACF,oBAAoBa;MAAaV,WAAWW;;IACnEzB,UAAU0B;EACZ,CAAA;EACAhC,OAAOe,UAAU;IACfC,aAAaS,iBAAiBQ;IAC9B3B,UAAU4B;EACZ,CAAA;;;;;EAKAlC,OAAOe,UAAU;IACfC,aAAaO,gBAAgBC,MAAMC,iBAAiBC,qBAAqBN,WAAWW,UAAU;IAC9FzB,UAAU6B;EACZ,CAAA;EACAnC,OAAOe,UAAU;IACfC,aAAaO,gBAAgBC,MAAMC,iBAAiBC,qBAAqBN,WAAWW,UAAU;IAC9FzB,UAAU8B;EACZ,CAAA;EACAhC,UAAUiC,qBAAqB;IAC7BC,OAAO;MAAEC,UAAUrC,KAAKsC;MAAIC,MAAM;MAAcC,SAASC;MAAYC,UAAU;IAAoB;EACrG,CAAA;EACA5C,OAAO6C;AAAI;AAGb,IAAA,qBAAe9C;",
|
|
6
|
+
"names": ["setAutoFreeze", "ActivationEvent", "ActivationEvents", "Plugin", "AppActivationEvents", "AppPlugin", "translations", "stackTranslations", "AppGraphBuilder", "CheckAppScheme", "DeckSettings", "DeckState", "NotificationTracker", "OperationHandler", "ReactRoot", "ReactSurface", "UrlHandler", "meta", "DeckEvents", "setAutoFreeze", "DeckPlugin", "Plugin", "define", "meta", "pipe", "AppPlugin", "addAppGraphModule", "activate", "AppGraphBuilder", "addOperationHandlerModule", "OperationHandler", "addSurfaceModule", "ReactSurface", "addTranslationsModule", "translations", "stackTranslations", "addModule", "activatesOn", "AppActivationEvents", "SetupSettings", "firesAfterActivation", "DeckEvents", "SettingsReady", "DeckSettings", "ActivationEvent", "allOf", "ActivationEvents", "ProcessManagerReady", "CheckAppScheme", "oneOf", "SetupAppGraph", "LayoutReady", "StateReady", "DeckState", "Startup", "ReactRoot", "UrlHandler", "NotificationTracker", "addPluginAssetModule", "asset", "pluginId", "id", "path", "content", "pluginSpec", "mimeType", "make"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import "./chunk-J5LGTIGS.mjs";
|
|
2
|
+
|
|
3
|
+
// src/DeckPlugin.node.ts
|
|
4
|
+
import { Plugin } from "@dxos/app-framework";
|
|
5
|
+
import { AppPlugin } from "@dxos/app-toolkit";
|
|
6
|
+
import { AppGraphBuilder, OperationHandler } from "#capabilities";
|
|
7
|
+
import { meta } from "#meta";
|
|
8
|
+
var DeckPlugin = Plugin.define(meta).pipe(AppPlugin.addAppGraphModule({
|
|
9
|
+
activate: AppGraphBuilder
|
|
10
|
+
}), AppPlugin.addOperationHandlerModule({
|
|
11
|
+
activate: OperationHandler
|
|
12
|
+
}), Plugin.make);
|
|
13
|
+
var DeckPlugin_node_default = DeckPlugin;
|
|
14
|
+
export {
|
|
15
|
+
DeckPlugin,
|
|
16
|
+
DeckPlugin_node_default as default
|
|
17
|
+
};
|
|
18
|
+
//# sourceMappingURL=DeckPlugin.node.mjs.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/DeckPlugin.node.ts"],
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2025 DXOS.org\n//\n\nimport { Plugin } from '@dxos/app-framework';\nimport { AppPlugin } from '@dxos/app-toolkit';\n\nimport { AppGraphBuilder, OperationHandler } from '#capabilities';\nimport { meta } from '#meta';\n\nexport const DeckPlugin = Plugin.define(meta).pipe(\n AppPlugin.addAppGraphModule({ activate: AppGraphBuilder }),\n AppPlugin.addOperationHandlerModule({ activate: OperationHandler }),\n Plugin.make,\n);\n\nexport default DeckPlugin;\n"],
|
|
5
|
+
"mappings": ";;;AAIA,SAASA,cAAc;AACvB,SAASC,iBAAiB;AAE1B,SAASC,iBAAiBC,wBAAwB;AAClD,SAASC,YAAY;AAEd,IAAMC,aAAaL,OAAOM,OAAOF,IAAAA,EAAMG,KAC5CN,UAAUO,kBAAkB;EAAEC,UAAUP;AAAgB,CAAA,GACxDD,UAAUS,0BAA0B;EAAED,UAAUN;AAAiB,CAAA,GACjEH,OAAOW,IAAI;AAGb,IAAA,0BAAeN;",
|
|
6
|
+
"names": ["Plugin", "AppPlugin", "AppGraphBuilder", "OperationHandler", "meta", "DeckPlugin", "define", "pipe", "addAppGraphModule", "activate", "addOperationHandlerModule", "make"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import "./chunk-J5LGTIGS.mjs";
|
|
2
|
+
|
|
3
|
+
// src/DeckPlugin.workerd.ts
|
|
4
|
+
import { Plugin } from "@dxos/app-framework";
|
|
5
|
+
import { AppPlugin } from "@dxos/app-toolkit";
|
|
6
|
+
import { OperationHandler } from "#capabilities";
|
|
7
|
+
import { meta } from "#meta";
|
|
8
|
+
var DeckPlugin = Plugin.define(meta).pipe(AppPlugin.addOperationHandlerModule({
|
|
9
|
+
activate: OperationHandler
|
|
10
|
+
}), Plugin.make);
|
|
11
|
+
var DeckPlugin_workerd_default = DeckPlugin;
|
|
12
|
+
export {
|
|
13
|
+
DeckPlugin,
|
|
14
|
+
DeckPlugin_workerd_default as default
|
|
15
|
+
};
|
|
16
|
+
//# sourceMappingURL=DeckPlugin.workerd.mjs.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/DeckPlugin.workerd.ts"],
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2025 DXOS.org\n//\n\nimport { Plugin } from '@dxos/app-framework';\nimport { AppPlugin } from '@dxos/app-toolkit';\n\nimport { OperationHandler } from '#capabilities';\nimport { meta } from '#meta';\n\nexport const DeckPlugin = Plugin.define(meta).pipe(\n AppPlugin.addOperationHandlerModule({ activate: OperationHandler }),\n Plugin.make,\n);\n\nexport default DeckPlugin;\n"],
|
|
5
|
+
"mappings": ";;;AAIA,SAASA,cAAc;AACvB,SAASC,iBAAiB;AAE1B,SAASC,wBAAwB;AACjC,SAASC,YAAY;AAEd,IAAMC,aAAaJ,OAAOK,OAAOF,IAAAA,EAAMG,KAC5CL,UAAUM,0BAA0B;EAAEC,UAAUN;AAAiB,CAAA,GACjEF,OAAOS,IAAI;AAGb,IAAA,6BAAeL;",
|
|
6
|
+
"names": ["Plugin", "AppPlugin", "OperationHandler", "meta", "DeckPlugin", "define", "pipe", "addOperationHandlerModule", "activate", "make"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import "./chunk-J5LGTIGS.mjs";
|
|
2
|
+
|
|
3
|
+
// src/components/DeckSettings/DeckSettings.tsx
|
|
4
|
+
import React from "react";
|
|
5
|
+
import { useTranslation } from "@dxos/react-ui";
|
|
6
|
+
import { Settings as SettingsForm } from "@dxos/react-ui-form";
|
|
7
|
+
import { meta } from "#meta";
|
|
8
|
+
import { Settings } from "#types";
|
|
9
|
+
var isSocket = !!globalThis.__args;
|
|
10
|
+
var DeckSettings = ({ settings, onSettingsChange }) => {
|
|
11
|
+
const { t } = useTranslation(meta.id);
|
|
12
|
+
return /* @__PURE__ */ React.createElement(SettingsForm.Viewport, null, /* @__PURE__ */ React.createElement(SettingsForm.Section, {
|
|
13
|
+
title: t("settings.title", {
|
|
14
|
+
ns: meta.id
|
|
15
|
+
})
|
|
16
|
+
}, /* @__PURE__ */ React.createElement(SettingsForm.FieldSet, {
|
|
17
|
+
readonly: !onSettingsChange,
|
|
18
|
+
schema: Settings.Settings,
|
|
19
|
+
visible: (path) => path !== "enableNativeRedirect" || !isSocket,
|
|
20
|
+
values: settings,
|
|
21
|
+
onValuesChanged: (values) => onSettingsChange?.(() => values)
|
|
22
|
+
})));
|
|
23
|
+
};
|
|
24
|
+
export {
|
|
25
|
+
DeckSettings as default
|
|
26
|
+
};
|
|
27
|
+
//# sourceMappingURL=DeckSettings-W5I57OXM.mjs.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/components/DeckSettings/DeckSettings.tsx"],
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nimport React from 'react';\n\nimport { type AppSurface } from '@dxos/app-toolkit/ui';\nimport { useTranslation } from '@dxos/react-ui';\nimport { Settings as SettingsForm } from '@dxos/react-ui-form';\n\nimport { meta } from '#meta';\nimport { Settings } from '#types';\n\nconst isSocket = !!(globalThis as any).__args;\n\nexport type DeckSettingsProps = AppSurface.SettingsArticleProps<Settings.Settings>;\n\nexport const DeckSettings = ({ settings, onSettingsChange }: DeckSettingsProps) => {\n const { t } = useTranslation(meta.id);\n\n return (\n <SettingsForm.Viewport>\n <SettingsForm.Section title={t('settings.title', { ns: meta.id })}>\n <SettingsForm.FieldSet\n readonly={!onSettingsChange}\n schema={Settings.Settings}\n visible={(path) => path !== 'enableNativeRedirect' || !isSocket}\n values={settings}\n onValuesChanged={(values) => onSettingsChange?.(() => values)}\n />\n </SettingsForm.Section>\n </SettingsForm.Viewport>\n );\n};\n"],
|
|
5
|
+
"mappings": ";;;AAIA,OAAOA,WAAW;AAGlB,SAASC,sBAAsB;AAC/B,SAASC,YAAYC,oBAAoB;AAEzC,SAASC,YAAY;AACrB,SAASF,gBAAgB;AAEzB,IAAMG,WAAW,CAAC,CAAEC,WAAmBC;AAIhC,IAAMC,eAAe,CAAC,EAAEC,UAAUC,iBAAgB,MAAqB;AAC5E,QAAM,EAAEC,EAAC,IAAKV,eAAeG,KAAKQ,EAAE;AAEpC,SACE,sBAAA,cAACT,aAAaU,UAAQ,MACpB,sBAAA,cAACV,aAAaW,SAAO;IAACC,OAAOJ,EAAE,kBAAkB;MAAEK,IAAIZ,KAAKQ;IAAG,CAAA;KAC7D,sBAAA,cAACT,aAAac,UAAQ;IACpBC,UAAU,CAACR;IACXS,QAAQjB,SAASA;IACjBkB,SAAS,CAACC,SAASA,SAAS,0BAA0B,CAAChB;IACvDiB,QAAQb;IACRc,iBAAiB,CAACD,WAAWZ,mBAAmB,MAAMY,MAAAA;;AAKhE;",
|
|
6
|
+
"names": ["React", "useTranslation", "Settings", "SettingsForm", "meta", "isSocket", "globalThis", "__args", "DeckSettings", "settings", "onSettingsChange", "t", "id", "Viewport", "Section", "title", "ns", "FieldSet", "readonly", "schema", "visible", "path", "values", "onValuesChanged"]
|
|
7
|
+
}
|