@dxos/app-framework 0.8.4-main.a4bbb77 → 0.8.4-main.abd8ff62ef
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/.storybook/main.mts +2 -4
- package/.storybook/preview.mts +2 -2
- package/dist/lib/browser/capability-Q5XRXRD2.mjs +38 -0
- package/dist/lib/browser/capability-Q5XRXRD2.mjs.map +7 -0
- package/dist/lib/browser/capability-V7LR4LQN.mjs +35 -0
- package/dist/lib/browser/capability-V7LR4LQN.mjs.map +7 -0
- package/dist/lib/browser/chunk-23D4SJUE.mjs +42 -0
- package/dist/lib/browser/chunk-23D4SJUE.mjs.map +7 -0
- package/dist/lib/browser/chunk-3JWJXGLK.mjs +79 -0
- package/dist/lib/browser/chunk-3JWJXGLK.mjs.map +7 -0
- package/dist/lib/browser/chunk-3ZS2A3DN.mjs +907 -0
- package/dist/lib/browser/chunk-3ZS2A3DN.mjs.map +7 -0
- package/dist/lib/browser/chunk-45CHLTBV.mjs +34 -0
- package/dist/lib/browser/chunk-45CHLTBV.mjs.map +7 -0
- package/dist/lib/browser/chunk-5LAIGWLU.mjs +467 -0
- package/dist/lib/browser/chunk-5LAIGWLU.mjs.map +7 -0
- package/dist/lib/browser/chunk-66IXTIVK.mjs +48 -0
- package/dist/lib/browser/chunk-66IXTIVK.mjs.map +7 -0
- package/dist/lib/browser/chunk-FJ4765WW.mjs +8 -0
- package/dist/lib/browser/chunk-FJ4765WW.mjs.map +7 -0
- package/dist/lib/browser/chunk-G7SDBRKH.mjs +1 -0
- package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
- package/dist/lib/browser/chunk-JXCBZSBJ.mjs +372 -0
- package/dist/lib/browser/chunk-JXCBZSBJ.mjs.map +7 -0
- package/dist/lib/browser/chunk-MX5DKEJH.mjs +584 -0
- package/dist/lib/browser/chunk-MX5DKEJH.mjs.map +7 -0
- package/dist/lib/browser/chunk-WBHCSOBW.mjs +80 -0
- package/dist/lib/browser/chunk-WBHCSOBW.mjs.map +7 -0
- package/dist/lib/browser/chunk-Z55LVAGN.mjs +213 -0
- package/dist/lib/browser/chunk-Z55LVAGN.mjs.map +7 -0
- package/dist/lib/browser/chunk-ZGJAZSNE.mjs +142 -0
- package/dist/lib/browser/chunk-ZGJAZSNE.mjs.map +7 -0
- package/dist/lib/browser/cli/index.mjs +74 -0
- package/dist/lib/browser/cli/index.mjs.map +7 -0
- package/dist/lib/browser/common/activation-events.mjs +24 -0
- package/dist/lib/browser/common/capabilities.mjs +46 -0
- package/dist/lib/browser/core/activation-event.mjs +20 -0
- package/dist/lib/browser/core/activation-event.mjs.map +7 -0
- package/dist/lib/browser/core/capability.mjs +30 -0
- package/dist/lib/browser/core/capability.mjs.map +7 -0
- package/dist/lib/browser/core/plugin-manager.mjs +17 -0
- package/dist/lib/browser/core/plugin-manager.mjs.map +7 -0
- package/dist/lib/browser/core/plugin.mjs +37 -0
- package/dist/lib/browser/core/plugin.mjs.map +7 -0
- package/dist/lib/browser/core/url-loader.mjs +20 -0
- package/dist/lib/browser/core/url-loader.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +95 -148
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/invoker-capability-LNX4CGIV.mjs +44 -0
- package/dist/lib/browser/invoker-capability-LNX4CGIV.mjs.map +7 -0
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/testing/index.mjs +227 -41
- package/dist/lib/browser/testing/index.mjs.map +4 -4
- package/dist/lib/browser/testing/react.mjs +78 -0
- package/dist/lib/browser/testing/react.mjs.map +7 -0
- package/dist/lib/browser/ui/index.mjs +48 -0
- package/dist/lib/browser/ui/index.mjs.map +7 -0
- package/dist/lib/node-esm/capability-EW5GJCI6.mjs +39 -0
- package/dist/lib/node-esm/capability-EW5GJCI6.mjs.map +7 -0
- package/dist/lib/node-esm/capability-YKBMMD53.mjs +36 -0
- package/dist/lib/node-esm/capability-YKBMMD53.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-37Z53PXZ.mjs +10 -0
- package/dist/lib/node-esm/chunk-37Z53PXZ.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-6XW6LET6.mjs +35 -0
- package/dist/lib/node-esm/chunk-6XW6LET6.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-D347W3KO.mjs +143 -0
- package/dist/lib/node-esm/chunk-D347W3KO.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-D5PO2WXX.mjs +373 -0
- package/dist/lib/node-esm/chunk-D5PO2WXX.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-HTBJU5FX.mjs +214 -0
- package/dist/lib/node-esm/chunk-HTBJU5FX.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-KM2F6GH6.mjs +468 -0
- package/dist/lib/node-esm/chunk-KM2F6GH6.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-OZ7DZA5Z.mjs +2 -0
- package/dist/lib/node-esm/chunk-OZ7DZA5Z.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-Q7XBFII4.mjs +908 -0
- package/dist/lib/node-esm/chunk-Q7XBFII4.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-SBS2YMPT.mjs +43 -0
- package/dist/lib/node-esm/chunk-SBS2YMPT.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-SDJ4B2LU.mjs +80 -0
- package/dist/lib/node-esm/chunk-SDJ4B2LU.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-WFSRZKBP.mjs +81 -0
- package/dist/lib/node-esm/chunk-WFSRZKBP.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-WKTLE7MG.mjs +585 -0
- package/dist/lib/node-esm/chunk-WKTLE7MG.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-XOCUANHO.mjs +49 -0
- package/dist/lib/node-esm/chunk-XOCUANHO.mjs.map +7 -0
- package/dist/lib/node-esm/cli/index.mjs +75 -0
- package/dist/lib/node-esm/cli/index.mjs.map +7 -0
- package/dist/lib/node-esm/common/activation-events.mjs +25 -0
- package/dist/lib/node-esm/common/activation-events.mjs.map +7 -0
- package/dist/lib/node-esm/common/capabilities.mjs +47 -0
- package/dist/lib/node-esm/common/capabilities.mjs.map +7 -0
- package/dist/lib/node-esm/core/activation-event.mjs +21 -0
- package/dist/lib/node-esm/core/activation-event.mjs.map +7 -0
- package/dist/lib/node-esm/core/capability.mjs +31 -0
- package/dist/lib/node-esm/core/capability.mjs.map +7 -0
- package/dist/lib/node-esm/core/plugin-manager.mjs +18 -0
- package/dist/lib/node-esm/core/plugin-manager.mjs.map +7 -0
- package/dist/lib/node-esm/core/plugin.mjs +38 -0
- package/dist/lib/node-esm/core/plugin.mjs.map +7 -0
- package/dist/lib/node-esm/core/url-loader.mjs +21 -0
- package/dist/lib/node-esm/core/url-loader.mjs.map +7 -0
- package/dist/lib/node-esm/index.mjs +95 -148
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/invoker-capability-O4T5PHLA.mjs +45 -0
- package/dist/lib/node-esm/invoker-capability-O4T5PHLA.mjs.map +7 -0
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/testing/index.mjs +227 -41
- package/dist/lib/node-esm/testing/index.mjs.map +4 -4
- package/dist/lib/node-esm/testing/react.mjs +79 -0
- package/dist/lib/node-esm/testing/react.mjs.map +7 -0
- package/dist/lib/node-esm/ui/index.mjs +49 -0
- package/dist/lib/node-esm/ui/index.mjs.map +7 -0
- package/dist/plugin/node-esm/index.mjs +832 -0
- package/dist/plugin/node-esm/index.mjs.map +7 -0
- package/dist/plugin/node-esm/meta.json +1 -0
- package/dist/types/src/cli/cli.d.ts +39 -0
- package/dist/types/src/cli/cli.d.ts.map +1 -0
- package/dist/types/src/cli/index.d.ts +2 -0
- package/dist/types/src/cli/index.d.ts.map +1 -0
- package/dist/types/src/common/activation-events.d.ts +27 -0
- package/dist/types/src/common/activation-events.d.ts.map +1 -0
- package/dist/types/src/common/capabilities.d.ts +110 -197
- package/dist/types/src/common/capabilities.d.ts.map +1 -1
- package/dist/types/src/common/index.d.ts +4 -8
- package/dist/types/src/common/index.d.ts.map +1 -1
- package/dist/types/src/common/operations.d.ts +19 -0
- package/dist/types/src/common/operations.d.ts.map +1 -0
- package/dist/types/src/common/translations.d.ts +8 -8
- package/dist/types/src/common/translations.d.ts.map +1 -1
- package/dist/types/src/context.d.ts +5 -0
- package/dist/types/src/context.d.ts.map +1 -0
- package/dist/types/src/core/{events.d.ts → activation-event.d.ts} +11 -11
- package/dist/types/src/core/activation-event.d.ts.map +1 -0
- package/dist/types/src/core/capability-manager.d.ts +48 -0
- package/dist/types/src/core/capability-manager.d.ts.map +1 -0
- package/dist/types/src/core/capability-manager.test.d.ts +2 -0
- package/dist/types/src/core/capability-manager.test.d.ts.map +1 -0
- package/dist/types/src/core/capability.d.ts +156 -0
- package/dist/types/src/core/capability.d.ts.map +1 -0
- package/dist/types/src/core/index.d.ts +8 -4
- package/dist/types/src/core/index.d.ts.map +1 -1
- package/dist/types/src/core/plugin-asset-cache.d.ts +71 -0
- package/dist/types/src/core/plugin-asset-cache.d.ts.map +1 -0
- package/dist/types/src/core/plugin-manager.d.ts +122 -0
- package/dist/types/src/core/plugin-manager.d.ts.map +1 -0
- package/dist/types/src/core/plugin-manager.test.d.ts +2 -0
- package/dist/types/src/core/plugin-manager.test.d.ts.map +1 -0
- package/dist/types/src/core/plugin-manifest.d.ts +76 -0
- package/dist/types/src/core/plugin-manifest.d.ts.map +1 -0
- package/dist/types/src/core/plugin-manifest.test.d.ts +2 -0
- package/dist/types/src/core/plugin-manifest.test.d.ts.map +1 -0
- package/dist/types/src/core/plugin.d.ts +207 -39
- package/dist/types/src/core/plugin.d.ts.map +1 -1
- package/dist/types/src/core/url-loader.d.ts +112 -0
- package/dist/types/src/core/url-loader.d.ts.map +1 -0
- package/dist/types/src/core/url-loader.test.d.ts +2 -0
- package/dist/types/src/core/url-loader.test.d.ts.map +1 -0
- package/dist/types/src/helpers.d.ts.map +1 -1
- package/dist/types/src/index.d.ts +3 -4
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/plugin-operation/OperationPlugin.d.ts +3 -0
- package/dist/types/src/plugin-operation/OperationPlugin.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/capability.d.ts +7 -0
- package/dist/types/src/plugin-operation/history/capability.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/errors.d.ts +32 -0
- package/dist/types/src/plugin-operation/history/errors.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/history-tracker.d.ts +18 -0
- package/dist/types/src/plugin-operation/history/history-tracker.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/history-tracker.test.d.ts +2 -0
- package/dist/types/src/plugin-operation/history/history-tracker.test.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/index.d.ts +6 -0
- package/dist/types/src/plugin-operation/history/index.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/types.d.ts +13 -0
- package/dist/types/src/plugin-operation/history/types.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/undo-mapping.d.ts +101 -0
- package/dist/types/src/plugin-operation/history/undo-mapping.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/undo-registry.d.ts +23 -0
- package/dist/types/src/plugin-operation/history/undo-registry.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/history/undo-registry.test.d.ts +2 -0
- package/dist/types/src/plugin-operation/history/undo-registry.test.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/index.d.ts +3 -0
- package/dist/types/src/plugin-operation/index.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/invoker-capability.d.ts +6 -0
- package/dist/types/src/plugin-operation/invoker-capability.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/meta.d.ts +3 -0
- package/dist/types/src/plugin-operation/meta.d.ts.map +1 -0
- package/dist/types/src/plugin-operation/testing.d.ts +59 -0
- package/dist/types/src/plugin-operation/testing.d.ts.map +1 -0
- package/dist/types/src/plugin-runtime/RuntimePlugin.d.ts +3 -0
- package/dist/types/src/plugin-runtime/RuntimePlugin.d.ts.map +1 -0
- package/dist/types/src/plugin-runtime/capability.d.ts +6 -0
- package/dist/types/src/plugin-runtime/capability.d.ts.map +1 -0
- package/dist/types/src/plugin-runtime/index.d.ts +2 -0
- package/dist/types/src/plugin-runtime/index.d.ts.map +1 -0
- package/dist/types/src/plugin-runtime/meta.d.ts +3 -0
- package/dist/types/src/plugin-runtime/meta.d.ts.map +1 -0
- package/dist/types/src/testing/harness.d.ts +67 -0
- package/dist/types/src/testing/harness.d.ts.map +1 -0
- package/dist/types/src/testing/index.d.ts +2 -0
- package/dist/types/src/testing/index.d.ts.map +1 -1
- package/dist/types/src/testing/react.d.ts +27 -0
- package/dist/types/src/testing/react.d.ts.map +1 -0
- package/dist/types/src/testing/react.test.d.ts +2 -0
- package/dist/types/src/testing/react.test.d.ts.map +1 -0
- package/dist/types/src/testing/service.d.ts +8 -0
- package/dist/types/src/testing/service.d.ts.map +1 -0
- package/dist/types/src/testing/withPluginManager.d.ts +7 -6
- package/dist/types/src/testing/withPluginManager.d.ts.map +1 -1
- package/dist/types/src/testing/withPluginManager.stories.d.ts.map +1 -1
- package/dist/types/src/ui/components/App/App.d.ts +9 -0
- package/dist/types/src/ui/components/App/App.d.ts.map +1 -0
- package/dist/types/src/ui/components/App/App.stories.d.ts +19 -0
- package/dist/types/src/ui/components/App/App.stories.d.ts.map +1 -0
- package/dist/types/src/ui/components/App/index.d.ts +2 -0
- package/dist/types/src/ui/components/App/index.d.ts.map +1 -0
- package/dist/types/src/ui/components/Placeholder/Placeholder.d.ts +64 -0
- package/dist/types/src/ui/components/Placeholder/Placeholder.d.ts.map +1 -0
- package/dist/types/src/ui/components/Placeholder/Placeholder.stories.d.ts +19 -0
- package/dist/types/src/ui/components/Placeholder/Placeholder.stories.d.ts.map +1 -0
- package/dist/types/src/ui/components/Placeholder/index.d.ts +2 -0
- package/dist/types/src/ui/components/Placeholder/index.d.ts.map +1 -0
- package/dist/types/src/{playground/playground.stories.d.ts → ui/components/PluginManager/PluginManagerContext.stories.d.ts} +5 -3
- package/dist/types/src/ui/components/PluginManager/PluginManagerContext.stories.d.ts.map +1 -0
- package/dist/types/src/ui/components/PluginManager/PluginManagerProvider.d.ts +10 -0
- package/dist/types/src/ui/components/PluginManager/PluginManagerProvider.d.ts.map +1 -0
- package/dist/types/src/ui/components/PluginManager/index.d.ts +2 -0
- package/dist/types/src/ui/components/PluginManager/index.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/SurfaceComponent.d.ts +24 -0
- package/dist/types/src/ui/components/Surface/SurfaceComponent.d.ts.map +1 -0
- package/dist/types/src/{components/App.stories.d.ts → ui/components/Surface/SurfaceComponent.stories.d.ts} +1 -1
- package/dist/types/src/ui/components/Surface/SurfaceComponent.stories.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/SurfaceInfo.d.ts +11 -0
- package/dist/types/src/ui/components/Surface/SurfaceInfo.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/SurfaceProfilerContext.d.ts +48 -0
- package/dist/types/src/ui/components/Surface/SurfaceProfilerContext.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/context.d.ts +5 -0
- package/dist/types/src/ui/components/Surface/context.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/index.d.ts +36 -0
- package/dist/types/src/ui/components/Surface/index.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/types.d.ts +197 -0
- package/dist/types/src/ui/components/Surface/types.d.ts.map +1 -0
- package/dist/types/src/ui/components/Surface/types.test.d.ts +2 -0
- package/dist/types/src/ui/components/Surface/types.test.d.ts.map +1 -0
- package/dist/types/src/ui/components/index.d.ts +5 -0
- package/dist/types/src/ui/components/index.d.ts.map +1 -0
- package/dist/types/src/ui/hooks/index.d.ts +6 -0
- package/dist/types/src/ui/hooks/index.d.ts.map +1 -0
- package/dist/types/src/ui/hooks/useApp.d.ts +88 -0
- package/dist/types/src/ui/hooks/useApp.d.ts.map +1 -0
- package/dist/types/src/ui/hooks/useApp.test.d.ts +2 -0
- package/dist/types/src/ui/hooks/useApp.test.d.ts.map +1 -0
- package/dist/types/src/ui/hooks/useCapabilities.d.ts +31 -0
- package/dist/types/src/ui/hooks/useCapabilities.d.ts.map +1 -0
- package/dist/types/src/{components → ui/hooks}/useLoading.d.ts +1 -2
- package/dist/types/src/ui/hooks/useLoading.d.ts.map +1 -0
- package/dist/types/src/ui/hooks/useSettingsState.d.ts +10 -0
- package/dist/types/src/ui/hooks/useSettingsState.d.ts.map +1 -0
- package/dist/types/src/ui/hooks/useSurface.d.ts +3 -0
- package/dist/types/src/ui/hooks/useSurface.d.ts.map +1 -0
- package/dist/types/src/ui/index.d.ts +3 -0
- package/dist/types/src/ui/index.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/boot-loader/BootLoader.stories.d.ts +34 -0
- package/dist/types/src/vite-plugin/boot-loader/BootLoader.stories.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/boot-loader/index.d.ts +52 -0
- package/dist/types/src/vite-plugin/boot-loader/index.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/composer/index.d.ts +34 -0
- package/dist/types/src/vite-plugin/composer/index.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/import-map/index.d.ts +28 -0
- package/dist/types/src/vite-plugin/import-map/index.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/index.d.ts +5 -0
- package/dist/types/src/vite-plugin/index.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/manifest.d.ts +37 -0
- package/dist/types/src/vite-plugin/manifest.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/manifest.test.d.ts +2 -0
- package/dist/types/src/vite-plugin/manifest.test.d.ts.map +1 -0
- package/dist/types/src/vite-plugin/packages.d.ts +13 -0
- package/dist/types/src/vite-plugin/packages.d.ts.map +1 -0
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/moon.yml +25 -6
- package/package.json +112 -55
- package/src/cli/cli.ts +107 -0
- package/src/{components → cli}/index.ts +1 -1
- package/src/common/activation-events.ts +44 -0
- package/src/common/capabilities.ts +169 -210
- package/src/common/index.ts +4 -8
- package/src/common/operations.ts +35 -0
- package/src/common/translations.ts +18 -10
- package/src/context.ts +9 -0
- package/src/core/{events.ts → activation-event.ts} +10 -7
- package/src/core/capability-manager.test.ts +151 -0
- package/src/core/capability-manager.ts +192 -0
- package/src/core/capability.ts +247 -0
- package/src/core/index.ts +8 -4
- package/src/core/plugin-asset-cache.ts +60 -0
- package/src/core/plugin-manager.test.ts +1354 -0
- package/src/core/plugin-manager.ts +1025 -0
- package/src/core/plugin-manifest.test.ts +48 -0
- package/src/core/plugin-manifest.ts +102 -0
- package/src/core/plugin.ts +365 -45
- package/src/core/url-loader.test.ts +178 -0
- package/src/core/url-loader.ts +337 -0
- package/src/index.ts +3 -4
- package/src/plugin-operation/OperationPlugin.ts +24 -0
- package/src/plugin-operation/history/capability.ts +36 -0
- package/src/plugin-operation/history/errors.ts +7 -0
- package/src/plugin-operation/history/history-tracker.test.ts +374 -0
- package/src/plugin-operation/history/history-tracker.ts +128 -0
- package/src/plugin-operation/history/index.ts +9 -0
- package/src/plugin-operation/history/types.ts +17 -0
- package/src/plugin-operation/history/undo-mapping.ts +135 -0
- package/src/plugin-operation/history/undo-registry.test.ts +72 -0
- package/src/plugin-operation/history/undo-registry.ts +54 -0
- package/src/plugin-operation/index.ts +6 -0
- package/src/plugin-operation/invoker-capability.ts +55 -0
- package/src/plugin-operation/meta.ts +11 -0
- package/src/plugin-operation/testing.ts +155 -0
- package/src/plugin-runtime/RuntimePlugin.ts +19 -0
- package/src/plugin-runtime/capability.ts +53 -0
- package/src/{playground/layout → plugin-runtime}/index.ts +1 -1
- package/src/plugin-runtime/meta.ts +11 -0
- package/src/testing/harness.ts +229 -0
- package/src/testing/index.ts +2 -0
- package/src/testing/react.test.tsx +48 -0
- package/src/testing/react.tsx +113 -0
- package/src/testing/service.ts +52 -0
- package/src/testing/withPluginManager.stories.tsx +8 -9
- package/src/testing/withPluginManager.tsx +68 -40
- package/src/ui/components/App/App.stories.tsx +92 -0
- package/src/ui/components/App/App.tsx +81 -0
- package/src/{playground/debug → ui/components/App}/index.ts +1 -1
- package/src/ui/components/Placeholder/Placeholder.stories.tsx +77 -0
- package/src/ui/components/Placeholder/Placeholder.tsx +155 -0
- package/src/{playground/logger → ui/components/Placeholder}/index.ts +1 -1
- package/src/ui/components/PluginManager/PluginManagerContext.stories.tsx +185 -0
- package/src/{react → ui/components/PluginManager}/PluginManagerProvider.ts +3 -3
- package/src/ui/components/PluginManager/index.ts +5 -0
- package/src/ui/components/Surface/SurfaceComponent.stories.tsx +144 -0
- package/src/ui/components/Surface/SurfaceComponent.tsx +303 -0
- package/src/ui/components/Surface/SurfaceInfo.tsx +107 -0
- package/src/ui/components/Surface/SurfaceProfilerContext.tsx +207 -0
- package/src/ui/components/Surface/context.ts +12 -0
- package/src/ui/components/Surface/index.ts +54 -0
- package/src/ui/components/Surface/types.test.ts +126 -0
- package/src/ui/components/Surface/types.ts +269 -0
- package/src/ui/components/index.ts +8 -0
- package/src/ui/hooks/index.ts +9 -0
- package/src/ui/hooks/useApp.test.tsx +159 -0
- package/src/ui/hooks/useApp.tsx +413 -0
- package/src/ui/hooks/useCapabilities.ts +67 -0
- package/src/{components → ui/hooks}/useLoading.tsx +16 -10
- package/src/ui/hooks/useSettingsState.ts +26 -0
- package/src/ui/hooks/useSurface.ts +13 -0
- package/src/ui/index.ts +6 -0
- package/src/vite-plugin/boot-loader/BootLoader.stories.tsx +263 -0
- package/src/vite-plugin/boot-loader/boot-loader.css +294 -0
- package/src/vite-plugin/boot-loader/boot-loader.js +274 -0
- package/src/vite-plugin/boot-loader/index.ts +112 -0
- package/src/vite-plugin/composer/index.ts +277 -0
- package/src/vite-plugin/import-map/index.ts +524 -0
- package/src/vite-plugin/index.ts +10 -0
- package/src/vite-plugin/manifest.test.ts +24 -0
- package/src/vite-plugin/manifest.ts +50 -0
- package/src/vite-plugin/packages.ts +188 -0
- package/tsconfig.json +18 -15
- package/tsconfig.node.json +2 -4
- package/typedoc.json +2 -4
- package/vitest.config.ts +1 -1
- package/.swc/plugins/linux_x86_64_19.0.0/727453fb3a62f7f1d952a41e051ca8a6f88cadc45cee43c6a4d1aa45f9b75665.wasmer-v7 +0 -0
- package/.swc/plugins/linux_x86_64_19.0.0/fce1bdb8e20a094e4af08bad09cc81497ed0e2e7c51223b07d371063cca18429.wasmer-v7 +0 -0
- package/dist/lib/browser/app-graph-builder-XH4OYQLC.mjs +0 -137
- package/dist/lib/browser/app-graph-builder-XH4OYQLC.mjs.map +0 -7
- package/dist/lib/browser/chunk-6V54SRFL.mjs +0 -1638
- package/dist/lib/browser/chunk-6V54SRFL.mjs.map +0 -7
- package/dist/lib/browser/chunk-RGKMLI6U.mjs +0 -35
- package/dist/lib/browser/chunk-RGKMLI6U.mjs.map +0 -7
- package/dist/lib/browser/chunk-ZZVFNUHZ.mjs +0 -467
- package/dist/lib/browser/chunk-ZZVFNUHZ.mjs.map +0 -7
- package/dist/lib/browser/intent-dispatcher-VFMJVO2M.mjs +0 -11
- package/dist/lib/browser/intent-resolver-ICAPD4JL.mjs +0 -39
- package/dist/lib/browser/intent-resolver-ICAPD4JL.mjs.map +0 -7
- package/dist/lib/browser/store-7ZGMHOGB.mjs +0 -30
- package/dist/lib/browser/store-7ZGMHOGB.mjs.map +0 -7
- package/dist/lib/browser/worker.mjs +0 -77
- package/dist/lib/node-esm/app-graph-builder-C7H22SOL.mjs +0 -138
- package/dist/lib/node-esm/app-graph-builder-C7H22SOL.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-AXSZKZFD.mjs +0 -468
- package/dist/lib/node-esm/chunk-AXSZKZFD.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-LKPMRTRR.mjs +0 -37
- package/dist/lib/node-esm/chunk-LKPMRTRR.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-SOVTUUAY.mjs +0 -1640
- package/dist/lib/node-esm/chunk-SOVTUUAY.mjs.map +0 -7
- package/dist/lib/node-esm/intent-dispatcher-SAPOKSLZ.mjs +0 -12
- package/dist/lib/node-esm/intent-resolver-CRNJ6BMD.mjs +0 -40
- package/dist/lib/node-esm/intent-resolver-CRNJ6BMD.mjs.map +0 -7
- package/dist/lib/node-esm/store-H4F4RMYD.mjs +0 -31
- package/dist/lib/node-esm/store-H4F4RMYD.mjs.map +0 -7
- package/dist/lib/node-esm/worker.mjs +0 -78
- package/dist/types/src/common/collaboration.d.ts +0 -20
- package/dist/types/src/common/collaboration.d.ts.map +0 -1
- package/dist/types/src/common/events.d.ts +0 -52
- package/dist/types/src/common/events.d.ts.map +0 -1
- package/dist/types/src/common/file.d.ts +0 -14
- package/dist/types/src/common/file.d.ts.map +0 -1
- package/dist/types/src/common/graph.d.ts +0 -21
- package/dist/types/src/common/graph.d.ts.map +0 -1
- package/dist/types/src/common/layout.d.ts +0 -279
- package/dist/types/src/common/layout.d.ts.map +0 -1
- package/dist/types/src/common/surface.d.ts +0 -59
- package/dist/types/src/common/surface.d.ts.map +0 -1
- package/dist/types/src/components/App.d.ts +0 -10
- package/dist/types/src/components/App.d.ts.map +0 -1
- package/dist/types/src/components/App.stories.d.ts.map +0 -1
- package/dist/types/src/components/DefaultFallback.d.ts +0 -8
- package/dist/types/src/components/DefaultFallback.d.ts.map +0 -1
- package/dist/types/src/components/index.d.ts +0 -2
- package/dist/types/src/components/index.d.ts.map +0 -1
- package/dist/types/src/components/useApp.d.ts +0 -44
- package/dist/types/src/components/useApp.d.ts.map +0 -1
- package/dist/types/src/components/useLoading.d.ts.map +0 -1
- package/dist/types/src/core/capabilities.d.ts +0 -117
- package/dist/types/src/core/capabilities.d.ts.map +0 -1
- package/dist/types/src/core/capabilities.test.d.ts +0 -2
- package/dist/types/src/core/capabilities.test.d.ts.map +0 -1
- package/dist/types/src/core/events.d.ts.map +0 -1
- package/dist/types/src/core/manager.d.ts +0 -126
- package/dist/types/src/core/manager.d.ts.map +0 -1
- package/dist/types/src/core/manager.test.d.ts +0 -2
- package/dist/types/src/core/manager.test.d.ts.map +0 -1
- package/dist/types/src/playground/debug/Debug.d.ts +0 -6
- package/dist/types/src/playground/debug/Debug.d.ts.map +0 -1
- package/dist/types/src/playground/debug/index.d.ts +0 -2
- package/dist/types/src/playground/debug/index.d.ts.map +0 -1
- package/dist/types/src/playground/debug/plugin.d.ts +0 -2
- package/dist/types/src/playground/debug/plugin.d.ts.map +0 -1
- package/dist/types/src/playground/generator/Main.d.ts +0 -6
- package/dist/types/src/playground/generator/Main.d.ts.map +0 -1
- package/dist/types/src/playground/generator/Toolbar.d.ts +0 -6
- package/dist/types/src/playground/generator/Toolbar.d.ts.map +0 -1
- package/dist/types/src/playground/generator/generator.d.ts +0 -7
- package/dist/types/src/playground/generator/generator.d.ts.map +0 -1
- package/dist/types/src/playground/generator/index.d.ts +0 -3
- package/dist/types/src/playground/generator/index.d.ts.map +0 -1
- package/dist/types/src/playground/generator/plugin.d.ts +0 -2
- package/dist/types/src/playground/generator/plugin.d.ts.map +0 -1
- package/dist/types/src/playground/layout/Layout.d.ts +0 -8
- package/dist/types/src/playground/layout/Layout.d.ts.map +0 -1
- package/dist/types/src/playground/layout/index.d.ts +0 -2
- package/dist/types/src/playground/layout/index.d.ts.map +0 -1
- package/dist/types/src/playground/layout/plugin.d.ts +0 -2
- package/dist/types/src/playground/layout/plugin.d.ts.map +0 -1
- package/dist/types/src/playground/logger/Toolbar.d.ts +0 -6
- package/dist/types/src/playground/logger/Toolbar.d.ts.map +0 -1
- package/dist/types/src/playground/logger/index.d.ts +0 -2
- package/dist/types/src/playground/logger/index.d.ts.map +0 -1
- package/dist/types/src/playground/logger/plugin.d.ts +0 -2
- package/dist/types/src/playground/logger/plugin.d.ts.map +0 -1
- package/dist/types/src/playground/logger/schema.d.ts +0 -13
- package/dist/types/src/playground/logger/schema.d.ts.map +0 -1
- package/dist/types/src/playground/playground.stories.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/IntentPlugin.d.ts +0 -2
- package/dist/types/src/plugin-intent/IntentPlugin.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/actions.d.ts +0 -36
- package/dist/types/src/plugin-intent/actions.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/errors.d.ts +0 -16
- package/dist/types/src/plugin-intent/errors.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/index.d.ts +0 -6
- package/dist/types/src/plugin-intent/index.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/intent-dispatcher.d.ts +0 -139
- package/dist/types/src/plugin-intent/intent-dispatcher.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/intent-dispatcher.test.d.ts +0 -2
- package/dist/types/src/plugin-intent/intent-dispatcher.test.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/intent.d.ts +0 -63
- package/dist/types/src/plugin-intent/intent.d.ts.map +0 -1
- package/dist/types/src/plugin-intent/meta.d.ts +0 -3
- package/dist/types/src/plugin-intent/meta.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/SettingsPlugin.d.ts +0 -2
- package/dist/types/src/plugin-settings/SettingsPlugin.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/actions.d.ts +0 -25
- package/dist/types/src/plugin-settings/actions.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/app-graph-builder.d.ts +0 -4
- package/dist/types/src/plugin-settings/app-graph-builder.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/index.d.ts +0 -3
- package/dist/types/src/plugin-settings/index.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/intent-resolver.d.ts +0 -4
- package/dist/types/src/plugin-settings/intent-resolver.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/meta.d.ts +0 -3
- package/dist/types/src/plugin-settings/meta.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/store.d.ts +0 -5
- package/dist/types/src/plugin-settings/store.d.ts.map +0 -1
- package/dist/types/src/plugin-settings/translations.d.ts +0 -11
- package/dist/types/src/plugin-settings/translations.d.ts.map +0 -1
- package/dist/types/src/react/ErrorBoundary.d.ts +0 -30
- package/dist/types/src/react/ErrorBoundary.d.ts.map +0 -1
- package/dist/types/src/react/IntentContext.d.ts +0 -8
- package/dist/types/src/react/IntentContext.d.ts.map +0 -1
- package/dist/types/src/react/PluginManagerProvider.d.ts +0 -10
- package/dist/types/src/react/PluginManagerProvider.d.ts.map +0 -1
- package/dist/types/src/react/Surface.d.ts +0 -12
- package/dist/types/src/react/Surface.d.ts.map +0 -1
- package/dist/types/src/react/Surface.stories.d.ts +0 -17
- package/dist/types/src/react/Surface.stories.d.ts.map +0 -1
- package/dist/types/src/react/common.d.ts +0 -13
- package/dist/types/src/react/common.d.ts.map +0 -1
- package/dist/types/src/react/index.d.ts +0 -7
- package/dist/types/src/react/index.d.ts.map +0 -1
- package/dist/types/src/react/useCapabilities.d.ts +0 -13
- package/dist/types/src/react/useCapabilities.d.ts.map +0 -1
- package/dist/types/src/react/useIntentResolver.d.ts +0 -3
- package/dist/types/src/react/useIntentResolver.d.ts.map +0 -1
- package/dist/types/src/worker.d.ts +0 -4
- package/dist/types/src/worker.d.ts.map +0 -1
- package/src/common/collaboration.ts +0 -18
- package/src/common/events.ts +0 -79
- package/src/common/file.ts +0 -22
- package/src/common/graph.ts +0 -30
- package/src/common/layout.ts +0 -277
- package/src/common/surface.ts +0 -83
- package/src/components/App.stories.tsx +0 -33
- package/src/components/App.tsx +0 -59
- package/src/components/DefaultFallback.tsx +0 -26
- package/src/components/useApp.tsx +0 -164
- package/src/core/capabilities.test.ts +0 -136
- package/src/core/capabilities.ts +0 -259
- package/src/core/manager.test.ts +0 -516
- package/src/core/manager.ts +0 -597
- package/src/playground/debug/Debug.tsx +0 -39
- package/src/playground/debug/plugin.ts +0 -16
- package/src/playground/generator/Main.tsx +0 -71
- package/src/playground/generator/Toolbar.tsx +0 -47
- package/src/playground/generator/generator.ts +0 -48
- package/src/playground/generator/index.ts +0 -6
- package/src/playground/generator/plugin.ts +0 -22
- package/src/playground/layout/Layout.tsx +0 -33
- package/src/playground/layout/plugin.ts +0 -18
- package/src/playground/logger/Toolbar.tsx +0 -30
- package/src/playground/logger/plugin.ts +0 -41
- package/src/playground/logger/schema.ts +0 -12
- package/src/playground/playground.stories.tsx +0 -46
- package/src/plugin-intent/IntentPlugin.ts +0 -20
- package/src/plugin-intent/actions.ts +0 -31
- package/src/plugin-intent/errors.ts +0 -40
- package/src/plugin-intent/index.ts +0 -9
- package/src/plugin-intent/intent-dispatcher.test.ts +0 -279
- package/src/plugin-intent/intent-dispatcher.ts +0 -334
- package/src/plugin-intent/intent.ts +0 -154
- package/src/plugin-intent/meta.ts +0 -10
- package/src/plugin-settings/SettingsPlugin.ts +0 -34
- package/src/plugin-settings/actions.ts +0 -25
- package/src/plugin-settings/app-graph-builder.ts +0 -159
- package/src/plugin-settings/index.ts +0 -6
- package/src/plugin-settings/intent-resolver.ts +0 -35
- package/src/plugin-settings/meta.ts +0 -10
- package/src/plugin-settings/store.ts +0 -33
- package/src/plugin-settings/translations.ts +0 -19
- package/src/react/ErrorBoundary.tsx +0 -54
- package/src/react/IntentContext.tsx +0 -35
- package/src/react/Surface.stories.tsx +0 -101
- package/src/react/Surface.tsx +0 -86
- package/src/react/common.ts +0 -13
- package/src/react/index.ts +0 -10
- package/src/react/useCapabilities.ts +0 -31
- package/src/react/useIntentResolver.ts +0 -22
- package/src/worker.ts +0 -11
- /package/dist/lib/browser/{intent-dispatcher-VFMJVO2M.mjs.map → chunk-G7SDBRKH.mjs.map} +0 -0
- /package/dist/lib/browser/{worker.mjs.map → chunk-J5LGTIGS.mjs.map} +0 -0
- /package/dist/lib/{node-esm/intent-dispatcher-SAPOKSLZ.mjs.map → browser/common/activation-events.mjs.map} +0 -0
- /package/dist/lib/{node-esm/worker.mjs.map → browser/common/capabilities.mjs.map} +0 -0
|
@@ -0,0 +1,413 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { RegistryContext } from '@effect-atom/atom-react';
|
|
6
|
+
import * as Effect from 'effect/Effect';
|
|
7
|
+
import * as Fiber from 'effect/Fiber';
|
|
8
|
+
import * as PubSub from 'effect/PubSub';
|
|
9
|
+
import * as Queue from 'effect/Queue';
|
|
10
|
+
import React, { type FC, useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
|
11
|
+
|
|
12
|
+
import { runAndForwardErrors } from '@dxos/effect';
|
|
13
|
+
import { invariant } from '@dxos/invariant';
|
|
14
|
+
import { log } from '@dxos/log';
|
|
15
|
+
import { ErrorBoundary, ErrorFallback, type FallbackProps } from '@dxos/react-error-boundary';
|
|
16
|
+
import { useAsyncEffect, useDefaultValue } from '@dxos/react-hooks';
|
|
17
|
+
import { ContextProtocolProvider } from '@dxos/web-context-react';
|
|
18
|
+
|
|
19
|
+
import { ActivationEvents, Capabilities } from '../../common';
|
|
20
|
+
import { PluginManagerContext } from '../../context';
|
|
21
|
+
import { type ActivationEvent, type Plugin, PluginManager } from '../../core';
|
|
22
|
+
import { App, PluginManagerProvider } from '../components';
|
|
23
|
+
|
|
24
|
+
const ENABLED_KEY = 'org.dxos.app-framework.enabled';
|
|
25
|
+
|
|
26
|
+
export type StartupProgress = {
|
|
27
|
+
/** Number of modules that have been activated. */
|
|
28
|
+
activated: number;
|
|
29
|
+
/** Total number of modules registered. */
|
|
30
|
+
total: number;
|
|
31
|
+
/** Fractional progress (0-1). */
|
|
32
|
+
progress: number;
|
|
33
|
+
/**
|
|
34
|
+
* Raw activation event key (e.g. `org.dxos.app-framework.event.startup`).
|
|
35
|
+
* Set on event-level transitions, *and* on module-level transitions where
|
|
36
|
+
* it carries the parent activation event that first triggered the
|
|
37
|
+
* module's load (plumbed through `_loadCapabilitiesForModules` →
|
|
38
|
+
* `_loadModule`). Consumers can use this either as the primary id (when
|
|
39
|
+
* {@link module} is absent) or as an extra "context" field alongside
|
|
40
|
+
* {@link module}.
|
|
41
|
+
*/
|
|
42
|
+
event?: string;
|
|
43
|
+
/**
|
|
44
|
+
* Raw module id (e.g. `org.dxos.plugin.observability.module.ReactSurface`)
|
|
45
|
+
* when the in-flight activation is module-level. When present,
|
|
46
|
+
* {@link event} may also be set, identifying the parent activation that
|
|
47
|
+
* triggered this module's load.
|
|
48
|
+
*/
|
|
49
|
+
module?: string;
|
|
50
|
+
/**
|
|
51
|
+
* Pre-humanized label for the currently surfaced transition (module
|
|
52
|
+
* label if {@link module} is set, otherwise the event label), supplied
|
|
53
|
+
* for consumers that want a sensible default. Hosts that prefer to
|
|
54
|
+
* render their own label can read the raw {@link event}/{@link module}
|
|
55
|
+
* fields and ignore this — the framework leaves the policy choice
|
|
56
|
+
* (which transitions to surface, how to format them, whether to drop
|
|
57
|
+
* sub-modules entirely) to the host's `Placeholder`.
|
|
58
|
+
*/
|
|
59
|
+
humanizedName?: string;
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
export type PlaceholderProps = {
|
|
63
|
+
stage?: number;
|
|
64
|
+
progress?: StartupProgress;
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
export type UseAppOptions = {
|
|
68
|
+
pluginManager?: PluginManager.PluginManager;
|
|
69
|
+
pluginLoader?: PluginManager.ManagerOptions['pluginLoader'];
|
|
70
|
+
onPluginRemove?: PluginManager.ManagerOptions['onRemove'];
|
|
71
|
+
plugins?: Plugin.Plugin[];
|
|
72
|
+
core?: string[];
|
|
73
|
+
defaults?: string[];
|
|
74
|
+
/**
|
|
75
|
+
* Additional activation events to fire before startup.
|
|
76
|
+
* These are fired alongside SetupReactSurface before the Startup event.
|
|
77
|
+
*/
|
|
78
|
+
setupEvents?: ActivationEvent.ActivationEvent[];
|
|
79
|
+
cacheEnabled?: boolean;
|
|
80
|
+
safeMode?: boolean;
|
|
81
|
+
debounce?: number;
|
|
82
|
+
timeout?: number;
|
|
83
|
+
fallback?: FC<FallbackProps>;
|
|
84
|
+
placeholder?: FC<PlaceholderProps>;
|
|
85
|
+
};
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* Expected usage is for this to be the entrypoint of the application.
|
|
89
|
+
* Initializes plugins and renders the root components.
|
|
90
|
+
*
|
|
91
|
+
* @example
|
|
92
|
+
* const plugins = [LayoutPlugin(), MyPlugin()];
|
|
93
|
+
* const core = [LayoutPluginId];
|
|
94
|
+
* const default = [MyPluginId];
|
|
95
|
+
* const fallback = <div>Initializing Plugins...</div>;
|
|
96
|
+
* const App = useApp({ plugins, core, default, fallback });
|
|
97
|
+
* createRoot(document.getElementById('root')!).render(
|
|
98
|
+
* <StrictMode>
|
|
99
|
+
* <App />
|
|
100
|
+
* </StrictMode>,
|
|
101
|
+
* );
|
|
102
|
+
*
|
|
103
|
+
* @param params.pluginLoader A function which loads new plugins.
|
|
104
|
+
* @param params.plugins All plugins available to the application.
|
|
105
|
+
* @param params.core Core plugins which will always be enabled.
|
|
106
|
+
* @param params.defaults Default plugins are enabled by default but can be disabled by the user.
|
|
107
|
+
* @param params.cacheEnabled Whether to cache enabled plugins in localStorage.
|
|
108
|
+
* @param params.safeMode Whether to enable safe mode, which disables optional plugins.
|
|
109
|
+
* @param params.fallback Fallback component to render if an error occurs during startup.
|
|
110
|
+
* @param params.placeholder Placeholder component to render during startup.
|
|
111
|
+
*/
|
|
112
|
+
export const useApp = ({
|
|
113
|
+
pluginManager,
|
|
114
|
+
pluginLoader: pluginLoaderProp,
|
|
115
|
+
onPluginRemove,
|
|
116
|
+
plugins: pluginsProp,
|
|
117
|
+
core: coreProp,
|
|
118
|
+
defaults: defaultsProp,
|
|
119
|
+
setupEvents: setupEventsProp,
|
|
120
|
+
placeholder,
|
|
121
|
+
fallback = ErrorFallback,
|
|
122
|
+
cacheEnabled = false,
|
|
123
|
+
safeMode = false,
|
|
124
|
+
debounce = 0,
|
|
125
|
+
timeout = 30_000,
|
|
126
|
+
}: UseAppOptions) => {
|
|
127
|
+
const plugins = useDefaultValue(pluginsProp, () => []);
|
|
128
|
+
const core = useDefaultValue(coreProp, () => plugins.map(({ meta }) => meta.id));
|
|
129
|
+
const defaults = useDefaultValue(defaultsProp, () => []);
|
|
130
|
+
const setupEvents = useDefaultValue(setupEventsProp, () => []);
|
|
131
|
+
|
|
132
|
+
const pluginLoader = useMemo(
|
|
133
|
+
() =>
|
|
134
|
+
pluginLoaderProp ??
|
|
135
|
+
((id: string) =>
|
|
136
|
+
Effect.sync(() => {
|
|
137
|
+
const plugin = plugins.find((plugin) => plugin.meta.id === id);
|
|
138
|
+
invariant(plugin, `Plugin not found: ${id}`);
|
|
139
|
+
return plugin;
|
|
140
|
+
})),
|
|
141
|
+
[pluginLoaderProp, plugins],
|
|
142
|
+
);
|
|
143
|
+
|
|
144
|
+
const readyRef = useRef(false);
|
|
145
|
+
const [ready, setReady] = useState(false);
|
|
146
|
+
const errorRef = useRef<unknown>(null);
|
|
147
|
+
const [error, setError] = useState<unknown>(null);
|
|
148
|
+
const [startupProgress, setStartupProgress] = useState<StartupProgress>({
|
|
149
|
+
activated: 0,
|
|
150
|
+
total: 0,
|
|
151
|
+
progress: 0,
|
|
152
|
+
});
|
|
153
|
+
// TODO(wittjosiah): Migrate to Atom.kvs for isomorphic storage.
|
|
154
|
+
const cached: string[] = useMemo(() => JSON.parse(localStorage.getItem(ENABLED_KEY) ?? '[]'), []);
|
|
155
|
+
const enabled = useMemo(
|
|
156
|
+
() => (safeMode ? [] : cacheEnabled && cached.length > 0 ? cached : defaults),
|
|
157
|
+
[safeMode, cacheEnabled, cached, defaults],
|
|
158
|
+
);
|
|
159
|
+
const isExternalManager = !!pluginManager;
|
|
160
|
+
const manager = useMemo(() => {
|
|
161
|
+
const mgr = pluginManager ?? PluginManager.make({ pluginLoader, plugins, core, enabled, onRemove: onPluginRemove });
|
|
162
|
+
log('useApp: useMemo created/reused manager', { provided: !!pluginManager });
|
|
163
|
+
return mgr;
|
|
164
|
+
}, [pluginManager, pluginLoader, plugins, core, enabled, onPluginRemove]);
|
|
165
|
+
|
|
166
|
+
useEffect(() => {
|
|
167
|
+
if (!cacheEnabled) {
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
return manager.registry.subscribe(manager.enabled, (value) => {
|
|
171
|
+
localStorage.setItem(ENABLED_KEY, JSON.stringify(value));
|
|
172
|
+
});
|
|
173
|
+
}, [cacheEnabled, manager]);
|
|
174
|
+
|
|
175
|
+
useEffect(() => {
|
|
176
|
+
setupDevtools(manager);
|
|
177
|
+
}, [manager]);
|
|
178
|
+
|
|
179
|
+
useAsyncEffect(async () => {
|
|
180
|
+
log('useApp: effect mount');
|
|
181
|
+
|
|
182
|
+
manager.capabilities.contribute({
|
|
183
|
+
interface: Capabilities.PluginManager,
|
|
184
|
+
implementation: manager,
|
|
185
|
+
module: 'org.dxos.app-framework.plugin-manager',
|
|
186
|
+
});
|
|
187
|
+
|
|
188
|
+
manager.capabilities.contribute({
|
|
189
|
+
interface: Capabilities.AtomRegistry,
|
|
190
|
+
implementation: manager.registry,
|
|
191
|
+
module: 'org.dxos.app-framework.atom-registry',
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
const fiber = Effect.gen(function* () {
|
|
195
|
+
const queue = yield* PubSub.subscribe(manager.activation);
|
|
196
|
+
const listener = yield* Effect.forkDaemon(
|
|
197
|
+
Queue.take(queue).pipe(
|
|
198
|
+
Effect.tap(({ event, state, module, error: error$ }) =>
|
|
199
|
+
Effect.sync(() => {
|
|
200
|
+
// Event-level Startup activated (no `module` field) fires once,
|
|
201
|
+
// after every module triggered by Startup has finished. Module
|
|
202
|
+
// activations now also carry their parent event id (so the trace
|
|
203
|
+
// can attribute a module to its triggering event), which means
|
|
204
|
+
// each module activated under Startup publishes
|
|
205
|
+
// `{ event: 'startup', state: 'activated', module: <id> }`. Without
|
|
206
|
+
// the `!module` guard the listener would mark the app ready on
|
|
207
|
+
// the *first* such module rather than waiting for the event-level
|
|
208
|
+
// completion — leaving downstream capabilities (operation-invoker,
|
|
209
|
+
// app-graph, …) un-registered when the placeholder dismisses.
|
|
210
|
+
if (event === ActivationEvents.Startup.id && state === 'activated' && !module) {
|
|
211
|
+
clearTimeout(timeoutId);
|
|
212
|
+
setReady(true);
|
|
213
|
+
readyRef.current = true;
|
|
214
|
+
// Trigger startup profiler dump if available.
|
|
215
|
+
(globalThis as any).composer?.profiler?.dump();
|
|
216
|
+
// Notify any host observability layer that startup completed.
|
|
217
|
+
// A `CustomEvent` keeps this generic — app-framework doesn't
|
|
218
|
+
// import a provider, and consumers can capture the startup
|
|
219
|
+
// summary without us picking one.
|
|
220
|
+
if (typeof window !== 'undefined') {
|
|
221
|
+
window.dispatchEvent(new CustomEvent('app-framework:startup-activated'));
|
|
222
|
+
}
|
|
223
|
+
return;
|
|
224
|
+
}
|
|
225
|
+
// `activating` is the start-of-load signal. Surface the raw
|
|
226
|
+
// `module` (or `event`) plus a pre-humanized label so the
|
|
227
|
+
// host placeholder can decide what to render — show
|
|
228
|
+
// everything, suppress noisy sub-modules, group by plugin,
|
|
229
|
+
// or apply its own formatting. We intentionally do NOT touch
|
|
230
|
+
// these fields on `activated`: pairing the two would cause
|
|
231
|
+
// back-to-back identical updates to the host's effect, which
|
|
232
|
+
// the boot loader treats as a re-trigger because
|
|
233
|
+
// `progress.progress` moved. Leaving the label alone on
|
|
234
|
+
// completion keeps it accurate ("now activating X") until
|
|
235
|
+
// the next module starts.
|
|
236
|
+
if (module && state === 'activating' && !readyRef.current) {
|
|
237
|
+
setStartupProgress((current) => ({
|
|
238
|
+
...current,
|
|
239
|
+
// `event` here is the activation event that first
|
|
240
|
+
// triggered this module load (plumbed through
|
|
241
|
+
// `_loadCapabilitiesForModules` → `_loadModule`).
|
|
242
|
+
// Falsy/empty falls back to undefined so consumers can
|
|
243
|
+
// tell "no parent context" from "parent context: <X>".
|
|
244
|
+
event: event || undefined,
|
|
245
|
+
module,
|
|
246
|
+
humanizedName: humanizeModuleId(module),
|
|
247
|
+
}));
|
|
248
|
+
}
|
|
249
|
+
// Update the activation count when a module commits. The
|
|
250
|
+
// ring's fraction comes from this; `event`/`module`/
|
|
251
|
+
// `humanizedName` were set by the matching `activating`
|
|
252
|
+
// message above and are left alone so the count can advance
|
|
253
|
+
// without re-firing the host's status callback.
|
|
254
|
+
if (module && state === 'activated' && !readyRef.current) {
|
|
255
|
+
const active = manager.getActive();
|
|
256
|
+
const total = manager.getModules().length;
|
|
257
|
+
setStartupProgress((current) => ({
|
|
258
|
+
...current,
|
|
259
|
+
activated: active.length,
|
|
260
|
+
total,
|
|
261
|
+
progress: total > 0 ? active.length / total : 0,
|
|
262
|
+
}));
|
|
263
|
+
}
|
|
264
|
+
// Event-level `activating` (no `module`) — fired at the start
|
|
265
|
+
// of `_activateModulesForEvent` and recursively for each
|
|
266
|
+
// before/after event. Surfaces a label during the gap before
|
|
267
|
+
// the first module-level message lands; subsequent module
|
|
268
|
+
// updates immediately overwrite this with a more specific
|
|
269
|
+
// label.
|
|
270
|
+
if (event && !module && state === 'activating' && !readyRef.current) {
|
|
271
|
+
setStartupProgress((current) => ({
|
|
272
|
+
...current,
|
|
273
|
+
event,
|
|
274
|
+
module: undefined,
|
|
275
|
+
humanizedName: humanizeEventKey(event),
|
|
276
|
+
}));
|
|
277
|
+
}
|
|
278
|
+
if (error$ && !readyRef.current) {
|
|
279
|
+
setError(error$);
|
|
280
|
+
errorRef.current = error$;
|
|
281
|
+
}
|
|
282
|
+
}),
|
|
283
|
+
),
|
|
284
|
+
Effect.forever,
|
|
285
|
+
),
|
|
286
|
+
);
|
|
287
|
+
|
|
288
|
+
yield* Effect.all([
|
|
289
|
+
...setupEvents.map((event) => manager.activate(event)),
|
|
290
|
+
manager.activate(ActivationEvents.SetupReactSurface),
|
|
291
|
+
manager.activate(ActivationEvents.Startup),
|
|
292
|
+
]);
|
|
293
|
+
|
|
294
|
+
return yield* Fiber.join(listener);
|
|
295
|
+
}).pipe(Effect.scoped, Effect.runFork);
|
|
296
|
+
|
|
297
|
+
// Set up a timeout for startup.
|
|
298
|
+
const timeoutId = setTimeout(() => {
|
|
299
|
+
if (!readyRef.current && !errorRef.current) {
|
|
300
|
+
log.warn('startup timeout diagnostic', {
|
|
301
|
+
eventsFired: manager.getEventsFired(),
|
|
302
|
+
activeModules: manager.getActive(),
|
|
303
|
+
pendingReset: manager.getPendingReset(),
|
|
304
|
+
});
|
|
305
|
+
void runAndForwardErrors(Fiber.interrupt(fiber));
|
|
306
|
+
setError(new Error(`Startup timed out after ${timeout}ms`));
|
|
307
|
+
}
|
|
308
|
+
}, timeout);
|
|
309
|
+
|
|
310
|
+
return () => {
|
|
311
|
+
log('useApp: effect cleanup');
|
|
312
|
+
clearTimeout(timeoutId);
|
|
313
|
+
void runAndForwardErrors(Fiber.interrupt(fiber));
|
|
314
|
+
if (!isExternalManager) {
|
|
315
|
+
void runAndForwardErrors(manager.shutdown());
|
|
316
|
+
}
|
|
317
|
+
};
|
|
318
|
+
}, [manager]);
|
|
319
|
+
|
|
320
|
+
const progressRef = useRef(startupProgress);
|
|
321
|
+
progressRef.current = startupProgress;
|
|
322
|
+
|
|
323
|
+
return useCallback(
|
|
324
|
+
() => (
|
|
325
|
+
<ErrorBoundary name='app' FallbackComponent={fallback}>
|
|
326
|
+
<PluginManagerProvider value={manager}>
|
|
327
|
+
<ContextProtocolProvider value={manager} context={PluginManagerContext}>
|
|
328
|
+
<RegistryContext.Provider value={manager.registry}>
|
|
329
|
+
<App
|
|
330
|
+
placeholder={placeholder}
|
|
331
|
+
ready={ready}
|
|
332
|
+
error={error}
|
|
333
|
+
debounce={debounce}
|
|
334
|
+
progress={progressRef.current}
|
|
335
|
+
/>
|
|
336
|
+
</RegistryContext.Provider>
|
|
337
|
+
</ContextProtocolProvider>
|
|
338
|
+
</PluginManagerProvider>
|
|
339
|
+
</ErrorBoundary>
|
|
340
|
+
),
|
|
341
|
+
[fallback, manager, placeholder, ready, error],
|
|
342
|
+
);
|
|
343
|
+
};
|
|
344
|
+
|
|
345
|
+
const setupDevtools = (manager: PluginManager.PluginManager) => {
|
|
346
|
+
(globalThis as any).composer ??= {};
|
|
347
|
+
(globalThis as any).composer.manager = manager;
|
|
348
|
+
};
|
|
349
|
+
|
|
350
|
+
/**
|
|
351
|
+
* Extracts a human-readable label from a module ID.
|
|
352
|
+
*
|
|
353
|
+
* Module IDs follow `org.dxos.plugin.<plugin-slug>.module.<module-name>`,
|
|
354
|
+
* where `<plugin-slug>` is kebab-case and `<module-name>` is either an
|
|
355
|
+
* explicit string (often kebab-case, e.g. `'observability'`, `'namespace'`)
|
|
356
|
+
* or a capability tag in PascalCase from `Capability.getModuleTag(...)`
|
|
357
|
+
* (e.g. `'ReactSurface'`, `'AppGraphBuilder'`). The output is
|
|
358
|
+
* `"Title Case Plugin: kebab-module"` so the visible status names both the
|
|
359
|
+
* plugin and the aspect being activated, helping disambiguate the multiple
|
|
360
|
+
* modules a plugin contributes.
|
|
361
|
+
*
|
|
362
|
+
* Examples:
|
|
363
|
+
* - "org.dxos.plugin.markdown.module.ReactSurface" → "Markdown: react-surface"
|
|
364
|
+
* - "org.dxos.plugin.observability.module.AppGraphBuilder" → "Observability: app-graph-builder"
|
|
365
|
+
* - "org.dxos.plugin.observability.module.observability" → "Observability"
|
|
366
|
+
* (the module name matches the plugin slug — collapsed to avoid
|
|
367
|
+
* "Observability: observability" noise.)
|
|
368
|
+
*/
|
|
369
|
+
const humanizeModuleId = (moduleId: string): string => {
|
|
370
|
+
const match = moduleId.match(/\.plugin\.([^.]+)\.module\.(.+)$/);
|
|
371
|
+
if (!match) {
|
|
372
|
+
// Fallback: use the last segment.
|
|
373
|
+
const parts = moduleId.split('.');
|
|
374
|
+
return parts[parts.length - 1];
|
|
375
|
+
}
|
|
376
|
+
const [, pluginSlug, moduleName] = match;
|
|
377
|
+
const pluginLabel = pluginSlug
|
|
378
|
+
.split('-')
|
|
379
|
+
.map((word) => word.charAt(0).toUpperCase() + word.slice(1))
|
|
380
|
+
.join(' ');
|
|
381
|
+
// Normalise the module name to kebab-case so PascalCase capability tags
|
|
382
|
+
// ("ReactSurface") read consistently with explicit kebab IDs
|
|
383
|
+
// ("operation-handler"). The two-step substitution handles consecutive
|
|
384
|
+
// uppercase runs (`URLLoader` → `url-loader`) without splitting them
|
|
385
|
+
// mid-acronym.
|
|
386
|
+
const moduleLabel = moduleName
|
|
387
|
+
.replace(/([a-z0-9])([A-Z])/g, '$1-$2')
|
|
388
|
+
.replace(/([A-Z]+)([A-Z][a-z])/g, '$1-$2')
|
|
389
|
+
.toLowerCase();
|
|
390
|
+
// Capability modules whose name matches the plugin slug (e.g. the
|
|
391
|
+
// observability plugin's `observability` capability module) would render
|
|
392
|
+
// as "Observability: observability" — drop the redundant suffix.
|
|
393
|
+
if (moduleLabel === pluginSlug) {
|
|
394
|
+
return pluginLabel;
|
|
395
|
+
}
|
|
396
|
+
return `${pluginLabel}: ${moduleLabel}`;
|
|
397
|
+
};
|
|
398
|
+
|
|
399
|
+
/**
|
|
400
|
+
* Extracts a human-readable label from an activation event key.
|
|
401
|
+
* E.g., "org.dxos.app-framework.event.setup-react-surface" → "Setup React Surface".
|
|
402
|
+
*/
|
|
403
|
+
const humanizeEventKey = (eventKey: string): string => {
|
|
404
|
+
// Strip a leading specifier (composite key form: "<id>:<specifier>").
|
|
405
|
+
const id = eventKey.split(':')[0];
|
|
406
|
+
// Match the trailing segment after `.event.`.
|
|
407
|
+
const match = id.match(/\.event\.(.+)$/);
|
|
408
|
+
const slug = match ? match[1] : (id.split('.').pop() ?? id);
|
|
409
|
+
return slug
|
|
410
|
+
.split('-')
|
|
411
|
+
.map((word) => word.charAt(0).toUpperCase() + word.slice(1))
|
|
412
|
+
.join(' ');
|
|
413
|
+
};
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { type Atom, useAtomValue } from '@effect-atom/atom-react';
|
|
6
|
+
import { useCallback } from 'react';
|
|
7
|
+
|
|
8
|
+
import { invariant } from '@dxos/invariant';
|
|
9
|
+
|
|
10
|
+
import { Capabilities } from '../../common';
|
|
11
|
+
import { type Capability } from '../../core';
|
|
12
|
+
import { usePluginManager } from '../components';
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Hook to request capabilities from the plugin context.
|
|
16
|
+
* @returns An array of capabilities.
|
|
17
|
+
*/
|
|
18
|
+
export const useCapabilities = <T>(interfaceDef: Capability.InterfaceDef<T>) => {
|
|
19
|
+
const manager = usePluginManager();
|
|
20
|
+
return useAtomValue(manager.capabilities.atom(interfaceDef));
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Hook to request a capability from the plugin context.
|
|
25
|
+
* @returns The capability.
|
|
26
|
+
* @throws If no capability is found.
|
|
27
|
+
*/
|
|
28
|
+
export const useCapability = <T>(interfaceDef: Capability.InterfaceDef<T>) => {
|
|
29
|
+
const capabilities = useCapabilities(interfaceDef);
|
|
30
|
+
invariant(capabilities.length > 0, `No capability found for ${interfaceDef.identifier}`);
|
|
31
|
+
return capabilities[0];
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Hook to get the current value of an atom capability.
|
|
36
|
+
* Automatically subscribes to changes.
|
|
37
|
+
* @example const settings = useAtomCapability(ThreadCapabilities.Settings);
|
|
38
|
+
*/
|
|
39
|
+
export const useAtomCapability = <T>(atomCapability: Capability.InterfaceDef<Atom.Atom<T>>): T => {
|
|
40
|
+
const atom = useCapability(atomCapability);
|
|
41
|
+
return useAtomValue(atom);
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Hook to get value and updater for an atom capability.
|
|
46
|
+
* Returns [currentValue, updateFn] similar to useState.
|
|
47
|
+
* @example const [settings, updateSettings] = useAtomCapabilityState(ThreadCapabilities.Settings);
|
|
48
|
+
*/
|
|
49
|
+
export const useAtomCapabilityState = <T>(
|
|
50
|
+
atomCapability: Capability.InterfaceDef<Atom.Writable<T>>,
|
|
51
|
+
): [T, (fn: (current: T) => T) => void] => {
|
|
52
|
+
const registry = useCapability(Capabilities.AtomRegistry);
|
|
53
|
+
const atom = useCapability(atomCapability);
|
|
54
|
+
const value = useAtomValue(atom);
|
|
55
|
+
const update = useCallback(
|
|
56
|
+
(fn: (current: T) => T) => {
|
|
57
|
+
registry.set(atom, fn(registry.get(atom)));
|
|
58
|
+
},
|
|
59
|
+
[registry, atom],
|
|
60
|
+
);
|
|
61
|
+
return [value, update];
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Hook to get the operation invoker capability.
|
|
66
|
+
*/
|
|
67
|
+
export const useOperationInvoker = (): Capabilities.OperationInvoker => useCapability(Capabilities.OperationInvoker);
|
|
@@ -2,9 +2,7 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import { useEffect, useState } from 'react';
|
|
6
|
-
|
|
7
|
-
import { type AppProps } from './App';
|
|
5
|
+
import { useEffect, useRef, useState } from 'react';
|
|
8
6
|
|
|
9
7
|
export enum LoadingState {
|
|
10
8
|
Loading = 0,
|
|
@@ -23,8 +21,16 @@ export enum LoadingState {
|
|
|
23
21
|
* 2: Fade-out - Fade out the loading animation.
|
|
24
22
|
* 3: Done - Remove the placeholder.
|
|
25
23
|
*/
|
|
26
|
-
export const useLoading = (
|
|
24
|
+
export const useLoading = (ready: boolean, debounce = 0) => {
|
|
27
25
|
const [stage, setStage] = useState<LoadingState>(LoadingState.Loading);
|
|
26
|
+
// Mirror `ready` into a ref so the interval's `setStage` callback can read
|
|
27
|
+
// the latest value without depending on the effect re-running. The pure
|
|
28
|
+
// closure-capture pattern (with `ready` in deps) sticks the FSM at
|
|
29
|
+
// `FadeIn` whenever HMR doesn't propagate the dep change end-to-end —
|
|
30
|
+
// a ref sidesteps that entirely and fires the read on every tick.
|
|
31
|
+
const readyRef = useRef(ready);
|
|
32
|
+
readyRef.current = ready;
|
|
33
|
+
|
|
28
34
|
useEffect(() => {
|
|
29
35
|
if (!debounce) {
|
|
30
36
|
return;
|
|
@@ -32,18 +38,18 @@ export const useLoading = (state: AppProps['state'], debounce = 0) => {
|
|
|
32
38
|
|
|
33
39
|
const i = setInterval(() => {
|
|
34
40
|
setStage((stage) => {
|
|
41
|
+
const isReady = readyRef.current;
|
|
35
42
|
switch (stage) {
|
|
36
43
|
case LoadingState.Loading: {
|
|
37
|
-
if (!
|
|
44
|
+
if (!isReady) {
|
|
38
45
|
return LoadingState.FadeIn;
|
|
39
|
-
} else {
|
|
40
|
-
clearInterval(i);
|
|
41
|
-
return LoadingState.Done;
|
|
42
46
|
}
|
|
47
|
+
clearInterval(i);
|
|
48
|
+
return LoadingState.Done;
|
|
43
49
|
}
|
|
44
50
|
|
|
45
51
|
case LoadingState.FadeIn: {
|
|
46
|
-
if (
|
|
52
|
+
if (isReady) {
|
|
47
53
|
return LoadingState.FadeOut;
|
|
48
54
|
}
|
|
49
55
|
break;
|
|
@@ -63,7 +69,7 @@ export const useLoading = (state: AppProps['state'], debounce = 0) => {
|
|
|
63
69
|
}, [debounce]);
|
|
64
70
|
|
|
65
71
|
if (!debounce) {
|
|
66
|
-
return
|
|
72
|
+
return ready ? LoadingState.Done : LoadingState.Loading;
|
|
67
73
|
}
|
|
68
74
|
|
|
69
75
|
return stage;
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { type Atom, RegistryContext, useAtomValue } from '@effect-atom/atom-react';
|
|
6
|
+
import { useCallback, useContext } from 'react';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Hook to read and update a settings atom.
|
|
10
|
+
* Returns the current value and an update function.
|
|
11
|
+
*/
|
|
12
|
+
export const useSettingsState = <T>(
|
|
13
|
+
atom: Atom.Writable<T>,
|
|
14
|
+
): { settings: T; updateSettings: (fn: (current: T) => T) => void } => {
|
|
15
|
+
const registry = useContext(RegistryContext);
|
|
16
|
+
const settings = useAtomValue(atom);
|
|
17
|
+
|
|
18
|
+
const updateSettings = useCallback(
|
|
19
|
+
(fn: (current: T) => T) => {
|
|
20
|
+
registry.set(atom, fn(registry.get(atom)));
|
|
21
|
+
},
|
|
22
|
+
[registry, atom],
|
|
23
|
+
);
|
|
24
|
+
|
|
25
|
+
return { settings, updateSettings };
|
|
26
|
+
};
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { useContext } from 'react';
|
|
6
|
+
|
|
7
|
+
import { raise } from '@dxos/debug';
|
|
8
|
+
|
|
9
|
+
import { Surface } from '../components';
|
|
10
|
+
|
|
11
|
+
export const useSurface = (): Surface.Context => {
|
|
12
|
+
return useContext(Surface.Context) ?? raise(new Error('Missing SurfaceContext'));
|
|
13
|
+
};
|