@dxos/app-framework 0.8.4-main.8360d9e660 → 0.8.4-main.8baae0fced

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.
Files changed (380) hide show
  1. package/LICENSE +102 -5
  2. package/README.md +1 -1
  3. package/dist/lib/browser/{capability-5OFLR7J4.mjs → capability-K5XIVCQU.mjs} +12 -11
  4. package/dist/lib/browser/capability-K5XIVCQU.mjs.map +7 -0
  5. package/dist/lib/browser/{chunk-WEBSGU5L.mjs → chunk-5AHASNDW.mjs} +20 -5
  6. package/dist/lib/browser/chunk-5AHASNDW.mjs.map +7 -0
  7. package/dist/lib/browser/chunk-5GY3YOEL.mjs +28 -0
  8. package/dist/lib/browser/chunk-5GY3YOEL.mjs.map +7 -0
  9. package/dist/lib/browser/{chunk-272IPLHQ.mjs → chunk-66IXTIVK.mjs} +3 -2
  10. package/dist/lib/browser/chunk-66IXTIVK.mjs.map +7 -0
  11. package/dist/lib/browser/{chunk-JGWCBVKJ.mjs → chunk-BRK6GYNB.mjs} +14 -42
  12. package/dist/lib/browser/chunk-BRK6GYNB.mjs.map +7 -0
  13. package/dist/lib/browser/{chunk-TGX63LTL.mjs → chunk-FJ4765WW.mjs} +1 -1
  14. package/dist/lib/browser/{chunk-TGX63LTL.mjs.map → chunk-FJ4765WW.mjs.map} +2 -2
  15. package/dist/lib/browser/chunk-FO3IYSLV.mjs +68 -0
  16. package/dist/lib/browser/chunk-FO3IYSLV.mjs.map +7 -0
  17. package/dist/lib/browser/{chunk-YNFPIQGB.mjs → chunk-IW44C7UL.mjs} +9 -2
  18. package/dist/lib/browser/chunk-IW44C7UL.mjs.map +7 -0
  19. package/dist/lib/browser/{chunk-TIEBZMTF.mjs → chunk-KFDF7KR3.mjs} +9 -11
  20. package/dist/lib/browser/{chunk-TIEBZMTF.mjs.map → chunk-KFDF7KR3.mjs.map} +3 -3
  21. package/dist/lib/browser/chunk-KLHQNYJ2.mjs +422 -0
  22. package/dist/lib/browser/chunk-KLHQNYJ2.mjs.map +7 -0
  23. package/dist/lib/browser/chunk-QLML5QFJ.mjs +581 -0
  24. package/dist/lib/browser/chunk-QLML5QFJ.mjs.map +7 -0
  25. package/dist/lib/browser/{chunk-GH3M2LIW.mjs → chunk-SLX73WRZ.mjs} +85 -15
  26. package/dist/lib/browser/chunk-SLX73WRZ.mjs.map +7 -0
  27. package/dist/lib/browser/chunk-UVTGHZQF.mjs +513 -0
  28. package/dist/lib/browser/chunk-UVTGHZQF.mjs.map +7 -0
  29. package/dist/lib/browser/chunk-VJ5PFAWC.mjs +1446 -0
  30. package/dist/lib/browser/chunk-VJ5PFAWC.mjs.map +7 -0
  31. package/dist/lib/browser/cli/index.mjs +15 -30
  32. package/dist/lib/browser/cli/index.mjs.map +3 -3
  33. package/dist/lib/browser/common/activation-events.mjs +11 -14
  34. package/dist/lib/browser/common/capabilities.mjs +19 -8
  35. package/dist/lib/browser/core/activation-event.mjs +1 -1
  36. package/dist/lib/browser/core/capability.mjs +5 -1
  37. package/dist/lib/browser/core/plugin-manager.mjs +8 -4
  38. package/dist/lib/browser/core/plugin.mjs +12 -2
  39. package/dist/lib/browser/core/url-loader.mjs +24 -0
  40. package/dist/lib/browser/index.mjs +47 -49
  41. package/dist/lib/browser/index.mjs.map +4 -4
  42. package/dist/lib/browser/meta.json +1 -1
  43. package/dist/lib/browser/process-manager-capability-JIWLN7SU.mjs +89 -0
  44. package/dist/lib/browser/process-manager-capability-JIWLN7SU.mjs.map +7 -0
  45. package/dist/lib/browser/testing/index.mjs +164 -34
  46. package/dist/lib/browser/testing/index.mjs.map +4 -4
  47. package/dist/lib/browser/testing/react.mjs +78 -0
  48. package/dist/lib/browser/testing/react.mjs.map +7 -0
  49. package/dist/lib/browser/ui/index.mjs +24 -15
  50. package/dist/lib/node-esm/{capability-CRHZKL6T.mjs → capability-RLKFFLTB.mjs} +12 -11
  51. package/dist/lib/node-esm/capability-RLKFFLTB.mjs.map +7 -0
  52. package/dist/lib/node-esm/{chunk-FKE4Z3D6.mjs → chunk-37Z53PXZ.mjs} +1 -1
  53. package/dist/lib/node-esm/{chunk-FKE4Z3D6.mjs.map → chunk-37Z53PXZ.mjs.map} +2 -2
  54. package/dist/lib/node-esm/chunk-42UNAKYO.mjs +423 -0
  55. package/dist/lib/node-esm/chunk-42UNAKYO.mjs.map +7 -0
  56. package/dist/lib/node-esm/{chunk-LHCJGNXK.mjs → chunk-6S45OMUP.mjs} +85 -15
  57. package/dist/lib/node-esm/chunk-6S45OMUP.mjs.map +7 -0
  58. package/dist/lib/node-esm/{chunk-3D66SZHP.mjs → chunk-BYHYYJZH.mjs} +14 -42
  59. package/dist/lib/node-esm/chunk-BYHYYJZH.mjs.map +7 -0
  60. package/dist/lib/node-esm/{chunk-SB5ODNPX.mjs → chunk-CTKEZHKF.mjs} +9 -2
  61. package/dist/lib/node-esm/chunk-CTKEZHKF.mjs.map +7 -0
  62. package/dist/lib/node-esm/chunk-JNT72ZCN.mjs +514 -0
  63. package/dist/lib/node-esm/chunk-JNT72ZCN.mjs.map +7 -0
  64. package/dist/lib/node-esm/chunk-KFZEB6BV.mjs +29 -0
  65. package/dist/lib/node-esm/chunk-KFZEB6BV.mjs.map +7 -0
  66. package/dist/lib/node-esm/chunk-LJNUFNDO.mjs +582 -0
  67. package/dist/lib/node-esm/chunk-LJNUFNDO.mjs.map +7 -0
  68. package/dist/lib/node-esm/{chunk-SCDGIGGU.mjs → chunk-OUEMWPIW.mjs} +9 -11
  69. package/dist/lib/node-esm/{chunk-SCDGIGGU.mjs.map → chunk-OUEMWPIW.mjs.map} +3 -3
  70. package/dist/lib/node-esm/{chunk-SQICGJBW.mjs → chunk-PW2VYGOS.mjs} +20 -5
  71. package/dist/lib/node-esm/chunk-PW2VYGOS.mjs.map +7 -0
  72. package/dist/lib/node-esm/chunk-SFYCO3PT.mjs +1447 -0
  73. package/dist/lib/node-esm/chunk-SFYCO3PT.mjs.map +7 -0
  74. package/dist/lib/node-esm/chunk-WK7OIQKI.mjs +70 -0
  75. package/dist/lib/node-esm/chunk-WK7OIQKI.mjs.map +7 -0
  76. package/dist/lib/node-esm/{chunk-6REV5DE7.mjs → chunk-XOCUANHO.mjs} +3 -2
  77. package/dist/lib/node-esm/chunk-XOCUANHO.mjs.map +7 -0
  78. package/dist/lib/node-esm/cli/index.mjs +15 -30
  79. package/dist/lib/node-esm/cli/index.mjs.map +3 -3
  80. package/dist/lib/node-esm/common/activation-events.mjs +11 -14
  81. package/dist/lib/node-esm/common/capabilities.mjs +19 -8
  82. package/dist/lib/node-esm/core/activation-event.mjs +1 -1
  83. package/dist/lib/node-esm/core/capability.mjs +5 -1
  84. package/dist/lib/node-esm/core/plugin-manager.mjs +8 -4
  85. package/dist/lib/node-esm/core/plugin.mjs +12 -2
  86. package/dist/lib/node-esm/core/url-loader.mjs +25 -0
  87. package/dist/lib/node-esm/index.mjs +47 -49
  88. package/dist/lib/node-esm/index.mjs.map +4 -4
  89. package/dist/lib/node-esm/meta.json +1 -1
  90. package/dist/lib/node-esm/process-manager-capability-PHKLO2BL.mjs +90 -0
  91. package/dist/lib/node-esm/process-manager-capability-PHKLO2BL.mjs.map +7 -0
  92. package/dist/lib/node-esm/testing/index.mjs +164 -34
  93. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  94. package/dist/lib/node-esm/testing/react.mjs +79 -0
  95. package/dist/lib/node-esm/testing/react.mjs.map +7 -0
  96. package/dist/lib/node-esm/ui/index.mjs +24 -15
  97. package/dist/plugin/node-esm/index.mjs +893 -0
  98. package/dist/plugin/node-esm/index.mjs.map +7 -0
  99. package/dist/plugin/node-esm/meta.json +1 -0
  100. package/dist/types/src/cli/cli.d.ts +1 -3
  101. package/dist/types/src/cli/cli.d.ts.map +1 -1
  102. package/dist/types/src/common/activation-events.d.ts +10 -13
  103. package/dist/types/src/common/activation-events.d.ts.map +1 -1
  104. package/dist/types/src/common/annotations.d.ts +1 -0
  105. package/dist/types/src/common/annotations.d.ts.map +1 -0
  106. package/dist/types/src/common/capabilities.d.ts +113 -12
  107. package/dist/types/src/common/capabilities.d.ts.map +1 -1
  108. package/dist/types/src/common/operations.d.ts +8 -22
  109. package/dist/types/src/common/operations.d.ts.map +1 -1
  110. package/dist/types/src/core/activation-event.d.ts +4 -4
  111. package/dist/types/src/core/activation-event.d.ts.map +1 -1
  112. package/dist/types/src/core/capability-manager.d.ts +5 -0
  113. package/dist/types/src/core/capability-manager.d.ts.map +1 -1
  114. package/dist/types/src/core/capability.d.ts +13 -2
  115. package/dist/types/src/core/capability.d.ts.map +1 -1
  116. package/dist/types/src/core/edge-registry-plugin-provider.d.ts +30 -0
  117. package/dist/types/src/core/edge-registry-plugin-provider.d.ts.map +1 -0
  118. package/dist/types/src/core/index.d.ts +6 -0
  119. package/dist/types/src/core/index.d.ts.map +1 -1
  120. package/dist/types/src/core/plugin-asset-cache.d.ts +71 -0
  121. package/dist/types/src/core/plugin-asset-cache.d.ts.map +1 -0
  122. package/dist/types/src/core/plugin-manager.d.ts +232 -7
  123. package/dist/types/src/core/plugin-manager.d.ts.map +1 -1
  124. package/dist/types/src/core/plugin-manifest.d.ts +101 -0
  125. package/dist/types/src/core/plugin-manifest.d.ts.map +1 -0
  126. package/dist/types/src/core/plugin-manifest.test.d.ts +2 -0
  127. package/dist/types/src/core/plugin-manifest.test.d.ts.map +1 -0
  128. package/dist/types/src/core/plugin.d.ts +176 -6
  129. package/dist/types/src/core/plugin.d.ts.map +1 -1
  130. package/dist/types/src/core/registry.d.ts +107 -0
  131. package/dist/types/src/core/registry.d.ts.map +1 -0
  132. package/dist/types/src/core/url-loader.d.ts +127 -0
  133. package/dist/types/src/core/url-loader.d.ts.map +1 -0
  134. package/dist/types/src/core/url-loader.test.d.ts +2 -0
  135. package/dist/types/src/core/url-loader.test.d.ts.map +1 -0
  136. package/dist/types/src/helpers.d.ts.map +1 -1
  137. package/dist/types/src/index.d.ts +1 -2
  138. package/dist/types/src/index.d.ts.map +1 -1
  139. package/dist/types/src/plugin-process-manager/ProcessManagerPlugin.d.ts +3 -0
  140. package/dist/types/src/plugin-process-manager/ProcessManagerPlugin.d.ts.map +1 -0
  141. package/dist/types/src/plugin-process-manager/history/capability.d.ts.map +1 -0
  142. package/dist/types/src/plugin-process-manager/history/errors.d.ts +32 -0
  143. package/dist/types/src/plugin-process-manager/history/errors.d.ts.map +1 -0
  144. package/dist/types/src/{plugin-operation → plugin-process-manager}/history/history-tracker.d.ts +1 -1
  145. package/dist/types/src/plugin-process-manager/history/history-tracker.d.ts.map +1 -0
  146. package/dist/types/src/plugin-process-manager/history/history-tracker.test.d.ts.map +1 -0
  147. package/dist/types/src/plugin-process-manager/history/index.d.ts.map +1 -0
  148. package/dist/types/src/{plugin-operation → plugin-process-manager}/history/types.d.ts +1 -1
  149. package/dist/types/src/plugin-process-manager/history/types.d.ts.map +1 -0
  150. package/dist/types/src/{plugin-operation → plugin-process-manager}/history/undo-mapping.d.ts +1 -1
  151. package/dist/types/src/plugin-process-manager/history/undo-mapping.d.ts.map +1 -0
  152. package/dist/types/src/{plugin-operation → plugin-process-manager}/history/undo-registry.d.ts +1 -1
  153. package/dist/types/src/plugin-process-manager/history/undo-registry.d.ts.map +1 -0
  154. package/dist/types/src/plugin-process-manager/history/undo-registry.test.d.ts.map +1 -0
  155. package/dist/types/src/plugin-process-manager/index.d.ts +3 -0
  156. package/dist/types/src/plugin-process-manager/index.d.ts.map +1 -0
  157. package/dist/types/src/plugin-process-manager/meta.d.ts.map +1 -0
  158. package/dist/types/src/plugin-process-manager/process-manager-capability.d.ts +8 -0
  159. package/dist/types/src/plugin-process-manager/process-manager-capability.d.ts.map +1 -0
  160. package/dist/types/src/plugin-process-manager/testing.d.ts +59 -0
  161. package/dist/types/src/plugin-process-manager/testing.d.ts.map +1 -0
  162. package/dist/types/src/testing/harness.d.ts +79 -0
  163. package/dist/types/src/testing/harness.d.ts.map +1 -0
  164. package/dist/types/src/testing/index.d.ts +1 -0
  165. package/dist/types/src/testing/index.d.ts.map +1 -1
  166. package/dist/types/src/testing/react.d.ts +27 -0
  167. package/dist/types/src/testing/react.d.ts.map +1 -0
  168. package/dist/types/src/testing/react.test.d.ts +2 -0
  169. package/dist/types/src/testing/react.test.d.ts.map +1 -0
  170. package/dist/types/src/testing/service.d.ts.map +1 -1
  171. package/dist/types/src/testing/withPluginManager.d.ts.map +1 -1
  172. package/dist/types/src/testing/withPluginManager.stories.d.ts.map +1 -1
  173. package/dist/types/src/ui/components/App/App.d.ts +3 -2
  174. package/dist/types/src/ui/components/App/App.d.ts.map +1 -1
  175. package/dist/types/src/ui/components/App/App.stories.d.ts +2 -2
  176. package/dist/types/src/ui/components/App/App.stories.d.ts.map +1 -1
  177. package/dist/types/src/ui/components/Placeholder/Placeholder.d.ts +64 -0
  178. package/dist/types/src/ui/components/Placeholder/Placeholder.d.ts.map +1 -0
  179. package/dist/types/src/ui/components/Placeholder/Placeholder.stories.d.ts +19 -0
  180. package/dist/types/src/ui/components/Placeholder/Placeholder.stories.d.ts.map +1 -0
  181. package/dist/types/src/ui/components/Placeholder/index.d.ts +2 -0
  182. package/dist/types/src/ui/components/Placeholder/index.d.ts.map +1 -0
  183. package/dist/types/src/ui/components/PluginManager/PluginManagerContext.stories.d.ts.map +1 -1
  184. package/dist/types/src/ui/components/Surface/SurfaceComponent.d.ts +16 -4
  185. package/dist/types/src/ui/components/Surface/SurfaceComponent.d.ts.map +1 -1
  186. package/dist/types/src/ui/components/Surface/SurfaceComponent.stories.d.ts.map +1 -1
  187. package/dist/types/src/ui/components/Surface/SurfaceInfo.d.ts.map +1 -1
  188. package/dist/types/src/ui/components/Surface/SurfaceProfilerContext.d.ts +48 -0
  189. package/dist/types/src/ui/components/Surface/SurfaceProfilerContext.d.ts.map +1 -0
  190. package/dist/types/src/ui/components/Surface/index.d.ts +22 -6
  191. package/dist/types/src/ui/components/Surface/index.d.ts.map +1 -1
  192. package/dist/types/src/ui/components/Surface/types.d.ts +110 -9
  193. package/dist/types/src/ui/components/Surface/types.d.ts.map +1 -1
  194. package/dist/types/src/ui/components/Surface/types.test.d.ts +2 -0
  195. package/dist/types/src/ui/components/Surface/types.test.d.ts.map +1 -0
  196. package/dist/types/src/ui/components/index.d.ts +1 -0
  197. package/dist/types/src/ui/components/index.d.ts.map +1 -1
  198. package/dist/types/src/ui/hooks/index.d.ts +1 -1
  199. package/dist/types/src/ui/hooks/index.d.ts.map +1 -1
  200. package/dist/types/src/ui/hooks/useApp.d.ts +47 -11
  201. package/dist/types/src/ui/hooks/useApp.d.ts.map +1 -1
  202. package/dist/types/src/ui/hooks/useCapabilities.d.ts.map +1 -1
  203. package/dist/types/src/ui/hooks/useLoading.d.ts.map +1 -1
  204. package/dist/types/src/ui/hooks/useProcessManagerRuntime.d.ts +24 -0
  205. package/dist/types/src/ui/hooks/useProcessManagerRuntime.d.ts.map +1 -0
  206. package/dist/types/src/ui/hooks/useSettingsState.d.ts.map +1 -1
  207. package/dist/types/src/vite-plugin/boot-loader/BootLoader.stories.d.ts +34 -0
  208. package/dist/types/src/vite-plugin/boot-loader/BootLoader.stories.d.ts.map +1 -0
  209. package/dist/types/src/vite-plugin/boot-loader/index.d.ts +2 -0
  210. package/dist/types/src/vite-plugin/boot-loader/index.d.ts.map +1 -0
  211. package/dist/types/src/vite-plugin/boot-loader/loader.d.ts +51 -0
  212. package/dist/types/src/vite-plugin/boot-loader/loader.d.ts.map +1 -0
  213. package/dist/types/src/vite-plugin/composer/index.d.ts +34 -0
  214. package/dist/types/src/vite-plugin/composer/index.d.ts.map +1 -0
  215. package/dist/types/src/vite-plugin/import-map/index.d.ts +28 -0
  216. package/dist/types/src/vite-plugin/import-map/index.d.ts.map +1 -0
  217. package/dist/types/src/vite-plugin/index.d.ts +5 -0
  218. package/dist/types/src/vite-plugin/index.d.ts.map +1 -0
  219. package/dist/types/src/vite-plugin/manifest.d.ts +41 -0
  220. package/dist/types/src/vite-plugin/manifest.d.ts.map +1 -0
  221. package/dist/types/src/vite-plugin/manifest.test.d.ts +2 -0
  222. package/dist/types/src/vite-plugin/manifest.test.d.ts.map +1 -0
  223. package/dist/types/src/vite-plugin/packages.d.ts +13 -0
  224. package/dist/types/src/vite-plugin/packages.d.ts.map +1 -0
  225. package/dist/types/tsconfig.tsbuildinfo +1 -1
  226. package/moon.yml +15 -0
  227. package/package.json +53 -54
  228. package/src/cli/cli.ts +2 -7
  229. package/src/common/activation-events.ts +10 -15
  230. package/src/common/annotations.ts +3 -0
  231. package/src/common/capabilities.ts +147 -16
  232. package/src/common/operations.ts +5 -8
  233. package/src/core/activation-event.ts +4 -1
  234. package/src/core/capability-manager.test.ts +1 -1
  235. package/src/core/capability-manager.ts +22 -1
  236. package/src/core/capability.ts +20 -2
  237. package/src/core/edge-registry-plugin-provider.ts +92 -0
  238. package/src/core/index.ts +6 -0
  239. package/src/core/plugin-asset-cache.ts +60 -0
  240. package/src/core/plugin-manager.test.ts +816 -10
  241. package/src/core/plugin-manager.ts +865 -46
  242. package/src/core/plugin-manifest.test.ts +75 -0
  243. package/src/core/plugin-manifest.ts +134 -0
  244. package/src/core/plugin.ts +185 -10
  245. package/src/core/registry.ts +163 -0
  246. package/src/core/url-loader.test.ts +221 -0
  247. package/src/core/url-loader.ts +388 -0
  248. package/src/index.ts +1 -2
  249. package/src/plugin-process-manager/ProcessManagerPlugin.ts +24 -0
  250. package/src/{plugin-operation → plugin-process-manager}/history/capability.ts +1 -2
  251. package/src/plugin-process-manager/history/errors.ts +7 -0
  252. package/src/{plugin-operation → plugin-process-manager}/history/history-tracker.test.ts +37 -43
  253. package/src/{plugin-operation → plugin-process-manager}/history/history-tracker.ts +1 -2
  254. package/src/{plugin-operation → plugin-process-manager}/history/types.ts +1 -1
  255. package/src/{plugin-operation → plugin-process-manager}/history/undo-mapping.ts +1 -1
  256. package/src/{plugin-operation → plugin-process-manager}/history/undo-registry.test.ts +3 -4
  257. package/src/{plugin-operation → plugin-process-manager}/history/undo-registry.ts +1 -1
  258. package/src/{plugin-operation → plugin-process-manager}/index.ts +1 -1
  259. package/src/plugin-process-manager/meta.ts +14 -0
  260. package/src/plugin-process-manager/process-manager-capability.ts +178 -0
  261. package/src/{plugin-operation → plugin-process-manager}/testing.ts +26 -45
  262. package/src/testing/harness.ts +247 -0
  263. package/src/testing/index.ts +1 -0
  264. package/src/testing/react.test.tsx +48 -0
  265. package/src/testing/react.tsx +113 -0
  266. package/src/testing/service.ts +2 -2
  267. package/src/testing/withPluginManager.stories.tsx +1 -2
  268. package/src/testing/withPluginManager.tsx +5 -4
  269. package/src/ui/components/App/App.stories.tsx +5 -11
  270. package/src/ui/components/App/App.tsx +29 -5
  271. package/src/ui/components/Placeholder/Placeholder.stories.tsx +77 -0
  272. package/src/ui/components/Placeholder/Placeholder.tsx +155 -0
  273. package/src/{plugin-runtime → ui/components/Placeholder}/index.ts +1 -1
  274. package/src/ui/components/PluginManager/PluginManagerContext.stories.tsx +5 -4
  275. package/src/ui/components/Surface/SurfaceComponent.stories.tsx +16 -15
  276. package/src/ui/components/Surface/SurfaceComponent.tsx +111 -55
  277. package/src/ui/components/Surface/SurfaceInfo.tsx +0 -1
  278. package/src/ui/components/Surface/SurfaceProfilerContext.tsx +207 -0
  279. package/src/ui/components/Surface/index.ts +35 -1
  280. package/src/ui/components/Surface/types.test.ts +126 -0
  281. package/src/ui/components/Surface/types.ts +164 -12
  282. package/src/ui/components/index.ts +1 -0
  283. package/src/ui/hooks/index.ts +1 -1
  284. package/src/ui/hooks/useApp.test.tsx +2 -2
  285. package/src/ui/hooks/useApp.tsx +216 -17
  286. package/src/ui/hooks/useLoading.tsx +14 -6
  287. package/src/ui/hooks/useProcessManagerRuntime.ts +68 -0
  288. package/src/vite-plugin/boot-loader/BootLoader.stories.tsx +270 -0
  289. package/src/vite-plugin/boot-loader/boot-loader.css +320 -0
  290. package/src/vite-plugin/boot-loader/boot-loader.js +325 -0
  291. package/src/vite-plugin/boot-loader/index.ts +5 -0
  292. package/src/vite-plugin/boot-loader/loader.ts +123 -0
  293. package/src/vite-plugin/composer/index.ts +306 -0
  294. package/src/vite-plugin/import-map/index.ts +527 -0
  295. package/src/vite-plugin/index.ts +10 -0
  296. package/src/vite-plugin/manifest.test.ts +46 -0
  297. package/src/vite-plugin/manifest.ts +57 -0
  298. package/src/vite-plugin/packages.ts +187 -0
  299. package/tsconfig.json +25 -1
  300. package/tsconfig.node.json +1 -1
  301. package/.swc/plugins/linux_x86_64_19.0.0/727453fb3a62f7f1d952a41e051ca8a6f88cadc45cee43c6a4d1aa45f9b75665.wasmer-v7 +0 -0
  302. package/dist/lib/browser/capability-5OFLR7J4.mjs.map +0 -7
  303. package/dist/lib/browser/capability-ZHQDZRP5.mjs +0 -37
  304. package/dist/lib/browser/capability-ZHQDZRP5.mjs.map +0 -7
  305. package/dist/lib/browser/chunk-272IPLHQ.mjs.map +0 -7
  306. package/dist/lib/browser/chunk-3VXJONTI.mjs +0 -933
  307. package/dist/lib/browser/chunk-3VXJONTI.mjs.map +0 -7
  308. package/dist/lib/browser/chunk-7AL6SKIV.mjs +0 -728
  309. package/dist/lib/browser/chunk-7AL6SKIV.mjs.map +0 -7
  310. package/dist/lib/browser/chunk-GH3M2LIW.mjs.map +0 -7
  311. package/dist/lib/browser/chunk-JGWCBVKJ.mjs.map +0 -7
  312. package/dist/lib/browser/chunk-M5IC326L.mjs +0 -34
  313. package/dist/lib/browser/chunk-M5IC326L.mjs.map +0 -7
  314. package/dist/lib/browser/chunk-WEBSGU5L.mjs.map +0 -7
  315. package/dist/lib/browser/chunk-YAFEA4GV.mjs +0 -1
  316. package/dist/lib/browser/chunk-YNFPIQGB.mjs.map +0 -7
  317. package/dist/lib/browser/invoker-capability-YTTQ2OBB.mjs +0 -36
  318. package/dist/lib/browser/invoker-capability-YTTQ2OBB.mjs.map +0 -7
  319. package/dist/lib/node-esm/capability-CRHZKL6T.mjs.map +0 -7
  320. package/dist/lib/node-esm/capability-W5C5464H.mjs +0 -38
  321. package/dist/lib/node-esm/capability-W5C5464H.mjs.map +0 -7
  322. package/dist/lib/node-esm/chunk-3D66SZHP.mjs.map +0 -7
  323. package/dist/lib/node-esm/chunk-6REV5DE7.mjs.map +0 -7
  324. package/dist/lib/node-esm/chunk-CMDON4NG.mjs +0 -934
  325. package/dist/lib/node-esm/chunk-CMDON4NG.mjs.map +0 -7
  326. package/dist/lib/node-esm/chunk-I7FZT4A7.mjs +0 -729
  327. package/dist/lib/node-esm/chunk-I7FZT4A7.mjs.map +0 -7
  328. package/dist/lib/node-esm/chunk-LHCJGNXK.mjs.map +0 -7
  329. package/dist/lib/node-esm/chunk-NHXBSAQR.mjs +0 -35
  330. package/dist/lib/node-esm/chunk-NHXBSAQR.mjs.map +0 -7
  331. package/dist/lib/node-esm/chunk-SB5ODNPX.mjs.map +0 -7
  332. package/dist/lib/node-esm/chunk-SQICGJBW.mjs.map +0 -7
  333. package/dist/lib/node-esm/chunk-Z4TJPSMP.mjs +0 -2
  334. package/dist/lib/node-esm/invoker-capability-BU26474T.mjs +0 -37
  335. package/dist/lib/node-esm/invoker-capability-BU26474T.mjs.map +0 -7
  336. package/dist/types/src/plugin-operation/OperationPlugin.d.ts +0 -3
  337. package/dist/types/src/plugin-operation/OperationPlugin.d.ts.map +0 -1
  338. package/dist/types/src/plugin-operation/history/capability.d.ts.map +0 -1
  339. package/dist/types/src/plugin-operation/history/errors.d.ts +0 -5
  340. package/dist/types/src/plugin-operation/history/errors.d.ts.map +0 -1
  341. package/dist/types/src/plugin-operation/history/history-tracker.d.ts.map +0 -1
  342. package/dist/types/src/plugin-operation/history/history-tracker.test.d.ts.map +0 -1
  343. package/dist/types/src/plugin-operation/history/index.d.ts.map +0 -1
  344. package/dist/types/src/plugin-operation/history/types.d.ts.map +0 -1
  345. package/dist/types/src/plugin-operation/history/undo-mapping.d.ts.map +0 -1
  346. package/dist/types/src/plugin-operation/history/undo-registry.d.ts.map +0 -1
  347. package/dist/types/src/plugin-operation/history/undo-registry.test.d.ts.map +0 -1
  348. package/dist/types/src/plugin-operation/index.d.ts +0 -3
  349. package/dist/types/src/plugin-operation/index.d.ts.map +0 -1
  350. package/dist/types/src/plugin-operation/invoker-capability.d.ts +0 -6
  351. package/dist/types/src/plugin-operation/invoker-capability.d.ts.map +0 -1
  352. package/dist/types/src/plugin-operation/meta.d.ts.map +0 -1
  353. package/dist/types/src/plugin-operation/testing.d.ts +0 -109
  354. package/dist/types/src/plugin-operation/testing.d.ts.map +0 -1
  355. package/dist/types/src/plugin-runtime/RuntimePlugin.d.ts +0 -3
  356. package/dist/types/src/plugin-runtime/RuntimePlugin.d.ts.map +0 -1
  357. package/dist/types/src/plugin-runtime/capability.d.ts +0 -6
  358. package/dist/types/src/plugin-runtime/capability.d.ts.map +0 -1
  359. package/dist/types/src/plugin-runtime/index.d.ts +0 -2
  360. package/dist/types/src/plugin-runtime/index.d.ts.map +0 -1
  361. package/dist/types/src/plugin-runtime/meta.d.ts +0 -3
  362. package/dist/types/src/plugin-runtime/meta.d.ts.map +0 -1
  363. package/dist/types/src/ui/hooks/useOperationResolver.d.ts +0 -19
  364. package/dist/types/src/ui/hooks/useOperationResolver.d.ts.map +0 -1
  365. package/src/plugin-operation/OperationPlugin.ts +0 -25
  366. package/src/plugin-operation/history/errors.ts +0 -11
  367. package/src/plugin-operation/invoker-capability.ts +0 -40
  368. package/src/plugin-operation/meta.ts +0 -11
  369. package/src/plugin-runtime/RuntimePlugin.ts +0 -20
  370. package/src/plugin-runtime/capability.ts +0 -53
  371. package/src/plugin-runtime/meta.ts +0 -11
  372. package/src/ui/hooks/useOperationResolver.ts +0 -40
  373. /package/dist/lib/browser/{chunk-YAFEA4GV.mjs.map → core/url-loader.mjs.map} +0 -0
  374. /package/dist/lib/node-esm/{chunk-Z4TJPSMP.mjs.map → core/url-loader.mjs.map} +0 -0
  375. /package/dist/types/src/{plugin-operation → plugin-process-manager}/history/capability.d.ts +0 -0
  376. /package/dist/types/src/{plugin-operation → plugin-process-manager}/history/history-tracker.test.d.ts +0 -0
  377. /package/dist/types/src/{plugin-operation → plugin-process-manager}/history/index.d.ts +0 -0
  378. /package/dist/types/src/{plugin-operation → plugin-process-manager}/history/undo-registry.test.d.ts +0 -0
  379. /package/dist/types/src/{plugin-operation → plugin-process-manager}/meta.d.ts +0 -0
  380. /package/src/{plugin-operation → plugin-process-manager}/history/index.ts +0 -0
@@ -0,0 +1,247 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { type Registry } from '@effect-atom/atom-react';
6
+ import * as Duration from 'effect/Duration';
7
+ import * as Effect from 'effect/Effect';
8
+ import * as PubSub from 'effect/PubSub';
9
+ import * as Queue from 'effect/Queue';
10
+
11
+ import { type Operation } from '@dxos/compute';
12
+ import { runAndForwardErrors } from '@dxos/effect';
13
+ import { invariant } from '@dxos/invariant';
14
+
15
+ import { ActivationEvents, Capabilities } from '../common';
16
+ import { ActivationEvent, type Capability, type CapabilityManager, type Plugin, PluginManager } from '../core';
17
+
18
+ export type TestAppOptions = {
19
+ /**
20
+ * Plugins to register. Plugins whose `meta.tags` includes `'system'` are treated as core
21
+ * (force-enabled). For test convenience, plugins without a `'system'` tag are enabled by
22
+ * default unless `enabled` is provided.
23
+ */
24
+ plugins: Plugin.Plugin[];
25
+ /** Plugin ids that are enabled by default in addition to core. Defaults to all non-system plugin ids. */
26
+ enabled?: string[];
27
+ /** Additional activation events fired alongside SetupReactSurface. */
28
+ setupEvents?: ActivationEvent.ActivationEvent[];
29
+ /**
30
+ * Whether to automatically fire SetupReactSurface + Startup during setup.
31
+ * Defaults to true.
32
+ */
33
+ autoStart?: boolean;
34
+ /**
35
+ * Whether to register the PluginManager + AtomRegistry framework capabilities.
36
+ * Defaults to true.
37
+ */
38
+ registerFrameworkCapabilities?: boolean;
39
+ };
40
+
41
+ /**
42
+ * A running plugin manager plus helpers for driving it in tests.
43
+ */
44
+ export interface TestHarness {
45
+ readonly manager: PluginManager.PluginManager;
46
+ readonly capabilities: CapabilityManager.CapabilityManager;
47
+ readonly registry: Registry.Registry;
48
+
49
+ /** Activate the given event. Equivalent to `manager.activate(event)`. */
50
+ fire(event: ActivationEvent.ActivationEvent | string): Promise<boolean>;
51
+ /** Re-activate all modules that were activated by the given event. */
52
+ reset(event: ActivationEvent.ActivationEvent | string): Promise<boolean>;
53
+
54
+ /** Returns the first contributed capability for the given interface. Throws if none are present. */
55
+ get<T>(iface: Capability.InterfaceDef<T>): T;
56
+ /** Returns all contributed capabilities for the given interface. */
57
+ getAll<T>(iface: Capability.InterfaceDef<T>): T[];
58
+ /** Waits until at least one capability is contributed for the given interface. */
59
+ waitForCapability<T>(iface: Capability.InterfaceDef<T>, opts?: { timeout?: number }): Promise<T>;
60
+ /** Waits until the given activation event has completed. */
61
+ waitForEvent(event: ActivationEvent.ActivationEvent | string, opts?: { timeout?: number }): Promise<void>;
62
+
63
+ /** Invokes an operation through the `Capabilities.OperationInvoker` capability. */
64
+ invoke<I, O>(op: Operation.Definition<I, O>, ...args: void extends I ? [input?: I] : [input: I]): Promise<O>;
65
+
66
+ /**
67
+ * Waits for `Capabilities.ProcessManagerRuntime` and runs the given effect on it.
68
+ * Convenience around `waitForCapability(ProcessManagerRuntime).runPromise(effect)`.
69
+ */
70
+ runPromise<A, E>(
71
+ effect: Effect.Effect<A, E, Capabilities.ProcessManagerRuntimeServices>,
72
+ options?: { readonly timeout?: number; readonly signal?: AbortSignal },
73
+ ): Promise<A>;
74
+
75
+ enable(id: string): Promise<boolean>;
76
+ disable(id: string): Promise<boolean>;
77
+
78
+ /** Shuts down the underlying plugin manager. */
79
+ dispose(): Promise<void>;
80
+
81
+ /** Async-disposable support so tests can use `await using harness = ...`. */
82
+ [Symbol.asyncDispose](): Promise<void>;
83
+ }
84
+
85
+ const DEFAULT_TIMEOUT_MS = 5_000;
86
+
87
+ /**
88
+ * Creates a TestHarness with the same bootstrap sequence that `useApp` performs,
89
+ * minus the React provider tree.
90
+ *
91
+ * For Node-only tests, this is enough to fire activation events, read
92
+ * capabilities, and invoke operations.
93
+ *
94
+ * For React tests, pass the returned harness to `render` or `renderSurface`
95
+ * from `@dxos/app-framework/testing-react`.
96
+ */
97
+ export const createTestApp = async (opts: TestAppOptions): Promise<TestHarness> => {
98
+ const {
99
+ plugins,
100
+ enabled = plugins.filter(({ meta }) => !meta.tags?.includes('system')).map((plugin) => plugin.meta.id),
101
+ setupEvents = [],
102
+ autoStart = true,
103
+ registerFrameworkCapabilities = true,
104
+ } = opts;
105
+
106
+ const pluginLoader = (id: string) =>
107
+ Effect.sync(() => {
108
+ const plugin = plugins.find((plugin) => plugin.meta.id === id);
109
+ invariant(plugin, `Plugin not found: ${id}`);
110
+ return { plugin };
111
+ });
112
+
113
+ const manager = PluginManager.make({ pluginLoader, plugins, enabled });
114
+
115
+ if (registerFrameworkCapabilities) {
116
+ manager.capabilities.contribute({
117
+ interface: Capabilities.PluginManager,
118
+ implementation: manager,
119
+ module: 'org.dxos.app-framework.plugin-manager',
120
+ });
121
+ manager.capabilities.contribute({
122
+ interface: Capabilities.AtomRegistry,
123
+ implementation: manager.registry,
124
+ module: 'org.dxos.app-framework.atom-registry',
125
+ });
126
+ }
127
+
128
+ if (autoStart) {
129
+ try {
130
+ await runAndForwardErrors(
131
+ Effect.all([
132
+ ...setupEvents.map((event) => manager.activate(event)),
133
+ manager.activate(ActivationEvents.SetupReactSurface),
134
+ manager.activate(ActivationEvents.Startup),
135
+ ]),
136
+ );
137
+ } catch (err) {
138
+ await runAndForwardErrors(manager.shutdown()).catch(() => undefined);
139
+ throw err;
140
+ }
141
+ }
142
+
143
+ return new TestHarnessImpl(manager);
144
+ };
145
+
146
+ class TestHarnessImpl implements TestHarness {
147
+ constructor(readonly manager: PluginManager.PluginManager) {}
148
+
149
+ get capabilities(): CapabilityManager.CapabilityManager {
150
+ return this.manager.capabilities;
151
+ }
152
+
153
+ get registry(): Registry.Registry {
154
+ return this.manager.registry;
155
+ }
156
+
157
+ fire(event: ActivationEvent.ActivationEvent | string): Promise<boolean> {
158
+ return runAndForwardErrors(this.manager.activate(event));
159
+ }
160
+
161
+ reset(event: ActivationEvent.ActivationEvent | string): Promise<boolean> {
162
+ return runAndForwardErrors(this.manager.reset(event));
163
+ }
164
+
165
+ get<T>(iface: Capability.InterfaceDef<T>): T {
166
+ return this.manager.capabilities.get(iface);
167
+ }
168
+
169
+ getAll<T>(iface: Capability.InterfaceDef<T>): T[] {
170
+ return this.manager.capabilities.getAll(iface);
171
+ }
172
+
173
+ waitForCapability<T>(iface: Capability.InterfaceDef<T>, opts?: { timeout?: number }): Promise<T> {
174
+ const timeout = opts?.timeout ?? DEFAULT_TIMEOUT_MS;
175
+ return runAndForwardErrors(
176
+ this.manager.capabilities.waitFor(iface).pipe(
177
+ Effect.timeoutFail({
178
+ duration: Duration.millis(timeout),
179
+ onTimeout: () => timeoutError(iface.identifier),
180
+ }),
181
+ ),
182
+ );
183
+ }
184
+
185
+ waitForEvent(event: ActivationEvent.ActivationEvent | string, opts?: { timeout?: number }): Promise<void> {
186
+ const key = typeof event === 'string' ? event : ActivationEvent.eventKey(event);
187
+ const timeout = opts?.timeout ?? DEFAULT_TIMEOUT_MS;
188
+
189
+ const program = Effect.gen(this, function* () {
190
+ const queue = yield* PubSub.subscribe(this.manager.activation);
191
+ // Re-check after subscribing to avoid a race where the event fires
192
+ // between the caller invoking this and the subscription being installed.
193
+ if (this.manager.getEventsFired().includes(key)) {
194
+ return;
195
+ }
196
+ while (true) {
197
+ const message = yield* Queue.take(queue);
198
+ if (message.event === key && message.state === 'activated') {
199
+ return;
200
+ }
201
+ }
202
+ }).pipe(
203
+ Effect.scoped,
204
+ Effect.timeoutFail({
205
+ duration: Duration.millis(timeout),
206
+ onTimeout: () => timeoutError(key),
207
+ }),
208
+ );
209
+
210
+ return runAndForwardErrors(program);
211
+ }
212
+
213
+ async invoke<I, O>(op: Operation.Definition<I, O>, ...args: [input?: I]): Promise<O> {
214
+ const invoker = await this.waitForCapability(Capabilities.OperationInvoker);
215
+ const result = await invoker.invokePromise(op as any, ...(args as [any]));
216
+ if (result.error) {
217
+ throw result.error;
218
+ }
219
+ return result.data as O;
220
+ }
221
+
222
+ async runPromise<A, E>(
223
+ effect: Effect.Effect<A, E, Capabilities.ProcessManagerRuntimeServices>,
224
+ options?: { readonly timeout?: number; readonly signal?: AbortSignal },
225
+ ): Promise<A> {
226
+ const runtime = await this.waitForCapability(Capabilities.ProcessManagerRuntime, { timeout: options?.timeout });
227
+ return runtime.runPromise(effect, options?.signal ? { signal: options.signal } : undefined);
228
+ }
229
+
230
+ enable(id: string): Promise<boolean> {
231
+ return runAndForwardErrors(this.manager.enable(id));
232
+ }
233
+
234
+ disable(id: string): Promise<boolean> {
235
+ return runAndForwardErrors(this.manager.disable(id));
236
+ }
237
+
238
+ async dispose(): Promise<void> {
239
+ await runAndForwardErrors(this.manager.shutdown());
240
+ }
241
+
242
+ [Symbol.asyncDispose](): Promise<void> {
243
+ return this.dispose();
244
+ }
245
+ }
246
+
247
+ const timeoutError = (id: string) => new Error(`Timed out waiting for ${id}`);
@@ -2,5 +2,6 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ export * from './harness';
5
6
  export * from './service';
6
7
  export * from './withPluginManager';
@@ -0,0 +1,48 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import * as Effect from 'effect/Effect';
6
+ import React from 'react';
7
+ import { describe, test } from 'vitest';
8
+
9
+ import { ActivationEvents, Capabilities } from '../common';
10
+ import { Capability, Plugin } from '../core';
11
+ import { Surface } from '../ui';
12
+ import { createTestApp } from './harness';
13
+ import { render, renderSurface } from './react';
14
+
15
+ const testMeta = { id: 'org.dxos.plugin.test.react-harness', name: 'ReactHarnessTest' };
16
+
17
+ const TestPlugin = Plugin.define(testMeta).pipe(
18
+ Plugin.addModule({
19
+ id: 'surfaces',
20
+ activatesOn: ActivationEvents.SetupReactSurface,
21
+ activate: () =>
22
+ Effect.succeed(
23
+ Capability.contributes(Capabilities.ReactSurface, [
24
+ Surface.create<{ message: string }>({
25
+ id: 'greeting',
26
+ role: 'greeting',
27
+ component: ({ data }) => <span data-testid='greeting'>hello {data.message}</span>,
28
+ }),
29
+ ]),
30
+ ),
31
+ }),
32
+ Plugin.make,
33
+ );
34
+
35
+ describe('testing/react', () => {
36
+ test('render wraps ui in the harness provider tree', async ({ expect }) => {
37
+ await using harness = await createTestApp({ plugins: [TestPlugin()] });
38
+ const view = render(harness, <span data-testid='hello'>world</span>);
39
+ expect(view.getByTestId('hello').textContent).toBe('world');
40
+ });
41
+
42
+ test('renderSurface mounts a surface registered by a plugin', async ({ expect }) => {
43
+ await using harness = await createTestApp({ plugins: [TestPlugin()] });
44
+ const view = renderSurface(harness, { role: 'greeting', data: { message: 'plugins' } });
45
+ const node = await view.findByTestId('greeting');
46
+ expect(node.textContent).toBe('hello plugins');
47
+ });
48
+ });
@@ -0,0 +1,113 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { RegistryContext } from '@effect-atom/atom-react';
6
+ import { render as rtlRender, type RenderOptions, type RenderResult } from '@testing-library/react';
7
+ import React, { type FC, Fragment, type PropsWithChildren, type ReactNode } from 'react';
8
+
9
+ import { ContextProtocolProvider } from '@dxos/web-context-react';
10
+
11
+ import { Capabilities } from '../common';
12
+ import { PluginManagerContext } from '../context';
13
+ import { topologicalSort } from '../helpers';
14
+ import { PluginManagerProvider } from '../ui/components/PluginManager/PluginManagerProvider';
15
+ import { SurfaceComponent } from '../ui/components/Surface/SurfaceComponent';
16
+ import { type TestHarness } from './harness';
17
+
18
+ export type HarnessRenderOptions = Omit<RenderOptions, 'wrapper'> & {
19
+ /** Additional providers to wrap around the harness tree (applied innermost first). */
20
+ reactContexts?: FC<PropsWithChildren>[];
21
+ };
22
+
23
+ /**
24
+ * Renders `ui` wrapped in the same provider tree as `useApp` plus every contributed
25
+ * `Capabilities.ReactContext`.
26
+ */
27
+ export const render = (harness: TestHarness, ui: ReactNode, options?: HarnessRenderOptions): RenderResult => {
28
+ const { reactContexts = [], ...rest } = options ?? {};
29
+ const Wrapper = ({ children }: PropsWithChildren) => (
30
+ <HarnessProviders harness={harness} extra={reactContexts}>
31
+ {children}
32
+ </HarnessProviders>
33
+ );
34
+ return rtlRender(<>{ui}</>, { ...rest, wrapper: Wrapper });
35
+ };
36
+
37
+ export type RenderSurfaceProps = {
38
+ role: string;
39
+ data?: unknown;
40
+ limit?: number;
41
+ fallback?: FC<{ error: Error; data?: any }>;
42
+ placeholder?: ReactNode;
43
+ };
44
+
45
+ /**
46
+ * Renders a `Surface` with the given role/data inside the harness provider tree.
47
+ */
48
+ export const renderSurface = (
49
+ harness: TestHarness,
50
+ props: RenderSurfaceProps,
51
+ options?: HarnessRenderOptions,
52
+ ): RenderResult => {
53
+ const { role, data, limit, fallback, placeholder } = props;
54
+ return render(
55
+ harness,
56
+ <SurfaceComponent role={role} data={data as any} limit={limit} fallback={fallback} placeholder={placeholder} />,
57
+ options,
58
+ );
59
+ };
60
+
61
+ type HarnessProvidersProps = PropsWithChildren<{
62
+ harness: TestHarness;
63
+ extra: FC<PropsWithChildren>[];
64
+ }>;
65
+
66
+ const HarnessProviders = ({ harness, extra, children }: HarnessProvidersProps) => {
67
+ const contributed = harness.getAll(Capabilities.ReactContext);
68
+ const ContributedContext = composeContexts(contributed);
69
+ const ExtraContext = composeExtra(extra);
70
+ return (
71
+ <PluginManagerProvider value={harness.manager}>
72
+ <ContextProtocolProvider value={harness.manager} context={PluginManagerContext}>
73
+ <RegistryContext.Provider value={harness.registry}>
74
+ <ContributedContext>
75
+ <ExtraContext>{children}</ExtraContext>
76
+ </ContributedContext>
77
+ </RegistryContext.Provider>
78
+ </ContextProtocolProvider>
79
+ </PluginManagerProvider>
80
+ );
81
+ };
82
+
83
+ const composeContexts = (contexts: Capabilities.ReactContext[]): FC<PropsWithChildren> => {
84
+ if (contexts.length === 0) {
85
+ return Passthrough;
86
+ }
87
+ return topologicalSort(contexts)
88
+ .map(({ context }) => context)
89
+ .reduce((Acc, Next) => ({ children }: PropsWithChildren) => (
90
+ <Acc>
91
+ <Next>{children}</Next>
92
+ </Acc>
93
+ ));
94
+ };
95
+
96
+ // Composes in innermost-first order: the first context in the array wraps
97
+ // `children` directly; subsequent contexts wrap the accumulator.
98
+ const composeExtra = (contexts: FC<PropsWithChildren>[]): FC<PropsWithChildren> => {
99
+ if (contexts.length === 0) {
100
+ return Passthrough;
101
+ }
102
+ return contexts.reduce<FC<PropsWithChildren>>(
103
+ (Acc, Next) =>
104
+ ({ children }: PropsWithChildren) => (
105
+ <Next>
106
+ <Acc>{children}</Acc>
107
+ </Next>
108
+ ),
109
+ Passthrough,
110
+ );
111
+ };
112
+
113
+ const Passthrough: FC<PropsWithChildren> = ({ children }) => <Fragment>{children}</Fragment>;
@@ -24,13 +24,13 @@ export const fromPlugins = (plugins: Plugin.Plugin[]) =>
24
24
  Effect.sync(() => {
25
25
  const plugin = plugins.find((plugin) => plugin.meta.id === id);
26
26
  invariant(plugin, `Plugin not found: ${id}`);
27
- return plugin;
27
+ return { plugin };
28
28
  });
29
29
 
30
30
  const manager = PluginManager.make({
31
31
  pluginLoader,
32
32
  plugins,
33
- core: plugins.map((plugin) => plugin.meta.id),
33
+ enabled: plugins.map((plugin) => plugin.meta.id),
34
34
  });
35
35
 
36
36
  manager.capabilities.contribute({
@@ -10,7 +10,6 @@ import { withTheme } from '@dxos/react-ui/testing';
10
10
  import { Capabilities } from '../common';
11
11
  import { Capability } from '../core';
12
12
  import { Surface } from '../ui';
13
-
14
13
  import { withPluginManager } from './withPluginManager';
15
14
 
16
15
  const DefaultStory = () => {
@@ -24,7 +23,7 @@ const DefaultStory = () => {
24
23
  };
25
24
 
26
25
  const meta = {
27
- title: 'sdk/app-framework/withPluginManager',
26
+ title: 'sdk/app-framework/testing/withPluginManager',
28
27
  render: DefaultStory,
29
28
  decorators: [
30
29
  withTheme(),
@@ -21,13 +21,15 @@ import { type UseAppOptions, useApp } from '../ui';
21
21
  export const setupPluginManager = ({
22
22
  capabilities,
23
23
  plugins = [],
24
- core = plugins.map(({ meta }) => meta.id),
25
24
  ...options
26
25
  }: UseAppOptions & Pick<WithPluginManagerOptions, 'capabilities'> = {}) => {
26
+ // Auto-enable every non-system plugin so stories don't have to spell out
27
+ // enablement. System-tagged plugins are force-enabled by the manager.
28
+ const enabled = plugins.filter(({ meta }) => !meta.tags?.includes('system')).map(({ meta }) => meta.id);
27
29
  const pluginManager = PluginManager.make({
28
30
  pluginLoader: () => raise(new Error('Not implemented')),
29
31
  plugins: [StoryPlugin, ...plugins],
30
- core: [StoryPlugin.meta.id, ...core],
32
+ enabled,
31
33
  ...options,
32
34
  });
33
35
 
@@ -75,7 +77,6 @@ export const withPluginManager = <Args,>(init: WithPluginManagerInitializer<Args
75
77
  // Storybook replaces the full context object often, so key manager ownership by story id.
76
78
  useEffect(() => {
77
79
  const pluginManager = setupPluginManager(options);
78
-
79
80
  const capability = Capability.contributes(Capabilities.ReactRoot, {
80
81
  id: storyId,
81
82
  root: () => <Story />,
@@ -115,13 +116,13 @@ const WithPluginManagerApp = ({ fireEvents, pluginManager, setupEvents, storyId
115
116
  }, [fireEvents, pluginManager, storyId]);
116
117
 
117
118
  const App = useApp({ pluginManager, setupEvents });
118
-
119
119
  return <App />;
120
120
  };
121
121
 
122
122
  const storyMeta = {
123
123
  id: 'org.dxos.app-framework.story',
124
124
  name: 'Story',
125
+ tags: ['system'],
125
126
  };
126
127
 
127
128
  // No-op plugin to ensure there exists at least one plugin for the startup event.
@@ -16,6 +16,7 @@ import { useApp } from '../../hooks';
16
16
  const TestPlugin = Plugin.define<{ error?: boolean }>({
17
17
  id: 'org.dxos.plugin.test',
18
18
  name: 'Test Plugin',
19
+ tags: ['system'],
19
20
  }).pipe(
20
21
  Plugin.addModule(({ error }) => ({
21
22
  id: 'TestMain',
@@ -46,20 +47,13 @@ const TestPlugin = Plugin.define<{ error?: boolean }>({
46
47
  Plugin.make,
47
48
  );
48
49
 
49
- const core = [TestPlugin.meta.id];
50
+ type DefaultStoryProps = { plugins?: Plugin.Plugin[] };
50
51
 
51
- type StoryProps = { plugins?: Plugin.Plugin[] };
52
-
53
- const DefaultStory = ({ plugins }: StoryProps) => {
52
+ const DefaultStory = ({ plugins }: DefaultStoryProps) => {
54
53
  const App = useApp({
55
54
  plugins,
56
- core,
57
55
  placeholder: () => {
58
- return (
59
- <div role='none' className='text-description'>
60
- Loading...
61
- </div>
62
- );
56
+ return <div className='text-description'>Loading...</div>;
63
57
  },
64
58
  });
65
59
 
@@ -67,7 +61,7 @@ const DefaultStory = ({ plugins }: StoryProps) => {
67
61
  };
68
62
 
69
63
  const meta = {
70
- title: 'sdk/app-framework/App',
64
+ title: 'sdk/app-framework/components/App',
71
65
  render: DefaultStory,
72
66
  decorators: [withTheme()],
73
67
  parameters: {
@@ -2,21 +2,45 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import React, { type PropsWithChildren } from 'react';
5
+ import React, { type PropsWithChildren, useEffect } from 'react';
6
6
 
7
7
  import { Capabilities } from '../../../common';
8
8
  import { topologicalSort } from '../../../helpers';
9
- import { LoadingState, type UseAppOptions, useCapabilities, useLoading } from '../../hooks';
9
+ import { LoadingState, type StartupProgress, type UseAppOptions, useCapabilities, useLoading } from '../../hooks';
10
10
 
11
11
  export type AppProps = Pick<UseAppOptions, 'placeholder' | 'debounce'> & {
12
12
  ready: boolean;
13
13
  error: unknown;
14
+ progress?: StartupProgress;
14
15
  };
15
16
 
16
- export const App = ({ placeholder: Placeholder, ready, error, debounce }: AppProps) => {
17
+ const FIRST_INTERACTIVE_MARK = 'app-framework:first-interactive';
18
+
19
+ export const App = ({ placeholder: Placeholder, ready, error, debounce, progress }: AppProps) => {
17
20
  const reactContexts = useCapabilities(Capabilities.ReactContext);
18
21
  const reactRoots = useCapabilities(Capabilities.ReactRoot);
19
22
  const stage = useLoading(ready, debounce);
23
+ const placeholderDismissed = stage >= LoadingState.Done;
24
+
25
+ // Emit a once-per-app `app-framework:first-interactive` mark the first time
26
+ // the placeholder is dismissed and the real app shell renders. Closes the
27
+ // gap between `Startup` activated and the first interactive paint.
28
+ //
29
+ // Also dismisses the native-DOM boot loader from
30
+ // `@dxos/app-framework/vite-plugin` here — the framework owns the
31
+ // handoff-complete signal, so dismissing in `App` covers both hosts that
32
+ // wire in a `Placeholder` (which may have its own dismiss path) and hosts
33
+ // that don't (the loader would otherwise sit forever as a `z-index: 10`
34
+ // overlay covering the rendered shell).
35
+ useEffect(() => {
36
+ if (!placeholderDismissed) {
37
+ return;
38
+ }
39
+ if (performance.getEntriesByName(FIRST_INTERACTIVE_MARK).length === 0) {
40
+ performance.mark(FIRST_INTERACTIVE_MARK);
41
+ }
42
+ (window as { __bootLoader?: { dismiss?: () => void } }).__bootLoader?.dismiss?.();
43
+ }, [placeholderDismissed]);
20
44
 
21
45
  if (error) {
22
46
  // This triggers the error boundary to provide UI feedback for the startup error.
@@ -24,12 +48,12 @@ export const App = ({ placeholder: Placeholder, ready, error, debounce }: AppPro
24
48
  }
25
49
 
26
50
  // TODO(wittjosiah): Consider using Suspense instead.
27
- if (stage < LoadingState.Done) {
51
+ if (!placeholderDismissed) {
28
52
  if (!Placeholder) {
29
53
  return null;
30
54
  }
31
55
 
32
- return <Placeholder stage={stage} />;
56
+ return <Placeholder stage={stage} progress={progress} />;
33
57
  }
34
58
 
35
59
  const ComposedContext = composeContexts(reactContexts);
@@ -0,0 +1,77 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
6
+ import React, { useEffect, useState } from 'react';
7
+ import { expect, within } from 'storybook/test';
8
+
9
+ import { Composer } from '@dxos/brand';
10
+ import { Input, Panel, Toolbar } from '@dxos/react-ui';
11
+ import { withLayout, withTheme } from '@dxos/react-ui/testing';
12
+
13
+ import { Placeholder } from './Placeholder';
14
+
15
+ /**
16
+ * The story renders the brand-agnostic `Placeholder` from `@dxos/app-framework/ui`
17
+ * with the Composer brand icon as its `logo`, so the storybook mirrors what
18
+ * composer-app actually mounts in production. The render function form lets
19
+ * the SVG accept the placeholder's animation `className` directly.
20
+ */
21
+ const meta = {
22
+ title: 'sdk/app-framework/components/Placeholder',
23
+ component: Placeholder,
24
+ render: () => {
25
+ const [stage, setStage] = useState(0);
26
+ const [auto, setAuto] = useState(false);
27
+ useEffect(() => {
28
+ if (!auto) {
29
+ return;
30
+ }
31
+
32
+ const interval = setInterval(() => {
33
+ setStage((prev) => {
34
+ return prev >= 2 ? 0 : prev + 1;
35
+ });
36
+ }, 3_000);
37
+
38
+ return () => clearInterval(interval);
39
+ }, [auto]);
40
+
41
+ return (
42
+ <Panel.Root>
43
+ <Panel.Toolbar asChild>
44
+ <Toolbar.Root classNames='gap-2'>
45
+ <Input.Root>
46
+ <Input.Checkbox checked={auto} onCheckedChange={(checked) => setAuto(checked === true)} />
47
+ <Input.Label>Auto</Input.Label>
48
+ </Input.Root>
49
+ <Toolbar.Button onClick={() => setStage((prev) => (prev < 2 ? prev + 1 : 0))}>Next</Toolbar.Button>
50
+ <Toolbar.Text>{stage}</Toolbar.Text>
51
+ </Toolbar.Root>
52
+ </Panel.Toolbar>
53
+ <Panel.Content>
54
+ <Placeholder stage={stage} logo={(logoProps) => <Composer {...logoProps} />} />
55
+ </Panel.Content>
56
+ </Panel.Root>
57
+ );
58
+ },
59
+ decorators: [withTheme(), withLayout({ layout: 'fullscreen' })],
60
+ tags: ['test'],
61
+ } satisfies Meta<typeof Placeholder>;
62
+
63
+ export default meta;
64
+
65
+ type Story = StoryObj<typeof meta>;
66
+
67
+ export const Default: Story = {
68
+ args: {
69
+ stage: 0,
70
+ },
71
+ play: async ({ canvasElement }) => {
72
+ const canvas = within(canvasElement);
73
+ // Smoke test: the toolbar's `Next` button is rendered alongside the
74
+ // placeholder, so the story mounted without throwing.
75
+ await expect(canvas.getByRole('button', { name: 'Next' })).toBeInTheDocument();
76
+ },
77
+ };