@dxos/plugin-deck 0.8.3-staging.0fa589b → 0.8.4-main.03d5cd7b56

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 (489) hide show
  1. package/dist/lib/neutral/DeckLayout-6SICSSUF.mjs +278 -0
  2. package/dist/lib/neutral/DeckLayout-6SICSSUF.mjs.map +7 -0
  3. package/dist/lib/neutral/DeckPlugin.mjs +70 -0
  4. package/dist/lib/neutral/DeckPlugin.mjs.map +7 -0
  5. package/dist/lib/neutral/DeckPlugin.node.mjs +18 -0
  6. package/dist/lib/neutral/DeckPlugin.node.mjs.map +7 -0
  7. package/dist/lib/neutral/DeckSettings-W5I57OXM.mjs +27 -0
  8. package/dist/lib/neutral/DeckSettings-W5I57OXM.mjs.map +7 -0
  9. package/dist/lib/neutral/add-toast-2OMTFZ2L.mjs +24 -0
  10. package/dist/lib/neutral/add-toast-2OMTFZ2L.mjs.map +7 -0
  11. package/dist/lib/neutral/adjust-RXTF3IT3.mjs +93 -0
  12. package/dist/lib/neutral/adjust-RXTF3IT3.mjs.map +7 -0
  13. package/dist/lib/neutral/app-graph-builder-GZYYAN5B.mjs +129 -0
  14. package/dist/lib/neutral/app-graph-builder-GZYYAN5B.mjs.map +7 -0
  15. package/dist/lib/neutral/capabilities/index.mjs +23 -0
  16. package/dist/lib/neutral/capabilities/index.mjs.map +7 -0
  17. package/dist/lib/neutral/check-app-scheme-A7FZVNZO.mjs +10 -0
  18. package/dist/lib/neutral/chunk-4MYX2NT3.mjs +1335 -0
  19. package/dist/lib/neutral/chunk-4MYX2NT3.mjs.map +7 -0
  20. package/dist/lib/neutral/chunk-64ODQD6S.mjs +282 -0
  21. package/dist/lib/neutral/chunk-64ODQD6S.mjs.map +7 -0
  22. package/dist/lib/neutral/chunk-GBIGQKYW.mjs +112 -0
  23. package/dist/lib/neutral/chunk-GBIGQKYW.mjs.map +7 -0
  24. package/dist/lib/neutral/chunk-GLB73Q6U.mjs +22 -0
  25. package/dist/lib/neutral/chunk-GLB73Q6U.mjs.map +7 -0
  26. package/dist/lib/neutral/chunk-J5LGTIGS.mjs +10 -0
  27. package/dist/lib/neutral/chunk-JMYWDX4I.mjs +19 -0
  28. package/dist/lib/neutral/chunk-JMYWDX4I.mjs.map +7 -0
  29. package/dist/lib/neutral/chunk-KX4QRGG5.mjs +8 -0
  30. package/dist/lib/neutral/chunk-KX4QRGG5.mjs.map +7 -0
  31. package/dist/lib/neutral/chunk-VHETZ22W.mjs +101 -0
  32. package/dist/lib/neutral/chunk-VHETZ22W.mjs.map +7 -0
  33. package/dist/lib/neutral/chunk-ZYYOSX5I.mjs +69 -0
  34. package/dist/lib/neutral/chunk-ZYYOSX5I.mjs.map +7 -0
  35. package/dist/lib/neutral/close-DG5IVQCE.mjs +44 -0
  36. package/dist/lib/neutral/close-DG5IVQCE.mjs.map +7 -0
  37. package/dist/lib/neutral/components/index.mjs +126 -0
  38. package/dist/lib/neutral/components/index.mjs.map +7 -0
  39. package/dist/lib/neutral/containers/index.mjs +32 -0
  40. package/dist/lib/neutral/containers/index.mjs.map +7 -0
  41. package/dist/lib/neutral/hooks/index.mjs +178 -0
  42. package/dist/lib/neutral/hooks/index.mjs.map +7 -0
  43. package/dist/lib/neutral/index.mjs +38 -0
  44. package/dist/lib/neutral/index.mjs.map +7 -0
  45. package/dist/lib/neutral/meta.json +1 -0
  46. package/dist/lib/neutral/meta.mjs +8 -0
  47. package/dist/lib/neutral/meta.mjs.map +7 -0
  48. package/dist/lib/neutral/open-BGH4LFTC.mjs +147 -0
  49. package/dist/lib/neutral/open-BGH4LFTC.mjs.map +7 -0
  50. package/dist/lib/neutral/operation-handler-266CVMTW.mjs +13 -0
  51. package/dist/lib/neutral/operation-handler-266CVMTW.mjs.map +7 -0
  52. package/dist/lib/neutral/operations/index.mjs +8 -0
  53. package/dist/lib/neutral/operations/index.mjs.map +7 -0
  54. package/dist/lib/neutral/plugin.mjs +16 -0
  55. package/dist/lib/neutral/plugin.mjs.map +7 -0
  56. package/dist/lib/neutral/react-root-WZ6KH27J.mjs +41 -0
  57. package/dist/lib/neutral/react-root-WZ6KH27J.mjs.map +7 -0
  58. package/dist/lib/neutral/react-surface-KAAZDHV3.mjs +34 -0
  59. package/dist/lib/neutral/react-surface-KAAZDHV3.mjs.map +7 -0
  60. package/dist/lib/neutral/revert-workspace-V6D7W6CL.mjs +21 -0
  61. package/dist/lib/neutral/revert-workspace-V6D7W6CL.mjs.map +7 -0
  62. package/dist/lib/neutral/scroll-into-view-UM446Q2T.mjs +21 -0
  63. package/dist/lib/neutral/scroll-into-view-UM446Q2T.mjs.map +7 -0
  64. package/dist/lib/neutral/set-JFFKT32S.mjs +37 -0
  65. package/dist/lib/neutral/set-JFFKT32S.mjs.map +7 -0
  66. package/dist/lib/neutral/set-layout-mode-I4YV5V2Y.mjs +85 -0
  67. package/dist/lib/neutral/set-layout-mode-I4YV5V2Y.mjs.map +7 -0
  68. package/dist/lib/neutral/settings-FE7AC5VS.mjs +34 -0
  69. package/dist/lib/neutral/settings-FE7AC5VS.mjs.map +7 -0
  70. package/dist/lib/neutral/show-undo-ZCMF4J2Q.mjs +59 -0
  71. package/dist/lib/neutral/show-undo-ZCMF4J2Q.mjs.map +7 -0
  72. package/dist/lib/neutral/state-IIDXMQUO.mjs +86 -0
  73. package/dist/lib/neutral/state-IIDXMQUO.mjs.map +7 -0
  74. package/dist/lib/neutral/switch-workspace-CVDYG5YZ.mjs +60 -0
  75. package/dist/lib/neutral/switch-workspace-CVDYG5YZ.mjs.map +7 -0
  76. package/dist/lib/neutral/translations.mjs +57 -0
  77. package/dist/lib/neutral/translations.mjs.map +7 -0
  78. package/dist/lib/neutral/types/index.mjs +34 -0
  79. package/dist/lib/neutral/types/index.mjs.map +7 -0
  80. package/dist/lib/neutral/update-companion-VPCM7FJX.mjs +32 -0
  81. package/dist/lib/neutral/update-companion-VPCM7FJX.mjs.map +7 -0
  82. package/dist/lib/neutral/update-complementary-SEQ6DDWA.mjs +28 -0
  83. package/dist/lib/neutral/update-complementary-SEQ6DDWA.mjs.map +7 -0
  84. package/dist/lib/neutral/update-dialog-VDHYE4QH.mjs +29 -0
  85. package/dist/lib/neutral/update-dialog-VDHYE4QH.mjs.map +7 -0
  86. package/dist/lib/neutral/update-plank-size-ZXOUZD7Y.mjs +26 -0
  87. package/dist/lib/neutral/update-plank-size-ZXOUZD7Y.mjs.map +7 -0
  88. package/dist/lib/neutral/update-popover-VFP32W57.mjs +33 -0
  89. package/dist/lib/neutral/update-popover-VFP32W57.mjs.map +7 -0
  90. package/dist/lib/neutral/update-sidebar-O5HBOXYH.mjs +25 -0
  91. package/dist/lib/neutral/update-sidebar-O5HBOXYH.mjs.map +7 -0
  92. package/dist/lib/neutral/url-handler-3QTKEM2K.mjs +184 -0
  93. package/dist/lib/neutral/url-handler-3QTKEM2K.mjs.map +7 -0
  94. package/dist/types/src/DeckPlugin.d.ts +3 -1
  95. package/dist/types/src/DeckPlugin.d.ts.map +1 -1
  96. package/dist/types/src/DeckPlugin.node.d.ts +4 -0
  97. package/dist/types/src/DeckPlugin.node.d.ts.map +1 -0
  98. package/dist/types/src/DeckPlugin.test.d.ts +2 -0
  99. package/dist/types/src/DeckPlugin.test.d.ts.map +1 -0
  100. package/dist/types/src/capabilities/app-graph-builder.d.ts +4 -2
  101. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -1
  102. package/dist/types/src/capabilities/check-app-scheme.d.ts +17 -2
  103. package/dist/types/src/capabilities/check-app-scheme.d.ts.map +1 -1
  104. package/dist/types/src/capabilities/index.d.ts +196 -12
  105. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  106. package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
  107. package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
  108. package/dist/types/src/capabilities/react-root.d.ts +4 -2
  109. package/dist/types/src/capabilities/react-root.d.ts.map +1 -1
  110. package/dist/types/src/capabilities/react-surface.d.ts +3 -2
  111. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  112. package/dist/types/src/capabilities/settings.d.ts +5 -2
  113. package/dist/types/src/capabilities/settings.d.ts.map +1 -1
  114. package/dist/types/src/capabilities/state.d.ts +136 -46
  115. package/dist/types/src/capabilities/state.d.ts.map +1 -1
  116. package/dist/types/src/capabilities/tools.d.ts +4 -3
  117. package/dist/types/src/capabilities/tools.d.ts.map +1 -1
  118. package/dist/types/src/capabilities/url-handler.d.ts +3 -2
  119. package/dist/types/src/capabilities/url-handler.d.ts.map +1 -1
  120. package/dist/types/src/components/DeckSettings/DeckSettings.d.ts +4 -4
  121. package/dist/types/src/components/DeckSettings/DeckSettings.d.ts.map +1 -1
  122. package/dist/types/src/components/DeckSettings/DeckSettings.stories.d.ts +61 -0
  123. package/dist/types/src/components/DeckSettings/DeckSettings.stories.d.ts.map +1 -0
  124. package/dist/types/src/components/DeckSettings/index.d.ts +1 -1
  125. package/dist/types/src/components/DeckSettings/index.d.ts.map +1 -1
  126. package/dist/types/src/components/Matrix/Matrix.d.ts +41 -0
  127. package/dist/types/src/components/Matrix/Matrix.d.ts.map +1 -0
  128. package/dist/types/src/components/Matrix/Matrix.stories.d.ts +17 -0
  129. package/dist/types/src/components/Matrix/Matrix.stories.d.ts.map +1 -0
  130. package/dist/types/src/components/Matrix/index.d.ts +3 -0
  131. package/dist/types/src/components/Matrix/index.d.ts.map +1 -0
  132. package/dist/types/src/components/index.d.ts +3 -2
  133. package/dist/types/src/components/index.d.ts.map +1 -1
  134. package/dist/types/src/{components/DeckLayout → containers/Deck}/Banner.d.ts +1 -1
  135. package/dist/types/src/containers/Deck/Banner.d.ts.map +1 -0
  136. package/dist/types/src/containers/Deck/Deck.d.ts +25 -0
  137. package/dist/types/src/containers/Deck/Deck.d.ts.map +1 -0
  138. package/dist/types/src/containers/Deck/Deck.stories.d.ts +61 -0
  139. package/dist/types/src/containers/Deck/Deck.stories.d.ts.map +1 -0
  140. package/dist/types/src/containers/Deck/DeckContent.d.ts +7 -0
  141. package/dist/types/src/containers/Deck/DeckContent.d.ts.map +1 -0
  142. package/dist/types/src/containers/Deck/DeckRoot.d.ts +39 -0
  143. package/dist/types/src/containers/Deck/DeckRoot.d.ts.map +1 -0
  144. package/dist/types/src/containers/Deck/DeckViewport.d.ts +16 -0
  145. package/dist/types/src/containers/Deck/DeckViewport.d.ts.map +1 -0
  146. package/dist/types/src/containers/Deck/StatusBar.d.ts.map +1 -0
  147. package/dist/types/src/containers/Deck/index.d.ts +2 -0
  148. package/dist/types/src/containers/Deck/index.d.ts.map +1 -0
  149. package/dist/types/src/containers/DeckLayout/ActiveNode.d.ts.map +1 -0
  150. package/dist/types/src/containers/DeckLayout/DeckLayout.d.ts +5 -0
  151. package/dist/types/src/containers/DeckLayout/DeckLayout.d.ts.map +1 -0
  152. package/dist/types/src/containers/DeckLayout/DeckLayout.stories.d.ts +63 -0
  153. package/dist/types/src/containers/DeckLayout/DeckLayout.stories.d.ts.map +1 -0
  154. package/dist/types/src/containers/DeckLayout/Dialog.d.ts.map +1 -0
  155. package/dist/types/src/containers/DeckLayout/Fallback.d.ts +2 -0
  156. package/dist/types/src/containers/DeckLayout/Fallback.d.ts.map +1 -0
  157. package/dist/types/src/containers/DeckLayout/Popover.d.ts +5 -0
  158. package/dist/types/src/containers/DeckLayout/Popover.d.ts.map +1 -0
  159. package/dist/types/src/containers/DeckLayout/Toast.d.ts +10 -0
  160. package/dist/types/src/containers/DeckLayout/Toast.d.ts.map +1 -0
  161. package/dist/types/src/containers/DeckLayout/constants.d.ts.map +1 -0
  162. package/dist/types/src/containers/DeckLayout/index.d.ts +4 -0
  163. package/dist/types/src/containers/DeckLayout/index.d.ts.map +1 -0
  164. package/dist/types/src/containers/Plank/Plank.d.ts +17 -0
  165. package/dist/types/src/containers/Plank/Plank.d.ts.map +1 -0
  166. package/dist/types/src/containers/Plank/Plank.stories.d.ts +61 -0
  167. package/dist/types/src/containers/Plank/Plank.stories.d.ts.map +1 -0
  168. package/dist/types/src/containers/Plank/PlankComponent.d.ts +17 -0
  169. package/dist/types/src/containers/Plank/PlankComponent.d.ts.map +1 -0
  170. package/dist/types/src/containers/Plank/PlankContent.d.ts +10 -0
  171. package/dist/types/src/containers/Plank/PlankContent.d.ts.map +1 -0
  172. package/dist/types/src/{components → containers}/Plank/PlankControls.d.ts +5 -6
  173. package/dist/types/src/containers/Plank/PlankControls.d.ts.map +1 -0
  174. package/dist/types/src/containers/Plank/PlankError.d.ts +15 -0
  175. package/dist/types/src/containers/Plank/PlankError.d.ts.map +1 -0
  176. package/dist/types/src/{components → containers}/Plank/PlankHeading.d.ts +5 -4
  177. package/dist/types/src/containers/Plank/PlankHeading.d.ts.map +1 -0
  178. package/dist/types/src/{components → containers}/Plank/PlankLoading.d.ts.map +1 -1
  179. package/dist/types/src/containers/Plank/PlankRoot.d.ts +37 -0
  180. package/dist/types/src/containers/Plank/PlankRoot.d.ts.map +1 -0
  181. package/dist/types/src/containers/Plank/index.d.ts +4 -0
  182. package/dist/types/src/containers/Plank/index.d.ts.map +1 -0
  183. package/dist/types/src/containers/Sidebar/ComplementarySidebar.d.ts.map +1 -0
  184. package/dist/types/src/containers/Sidebar/Sidebar.d.ts.map +1 -0
  185. package/dist/types/src/{components → containers}/Sidebar/SidebarButton.d.ts +1 -1
  186. package/dist/types/src/containers/Sidebar/SidebarButton.d.ts.map +1 -0
  187. package/dist/types/src/{components → containers}/Sidebar/index.d.ts.map +1 -1
  188. package/dist/types/src/containers/index.d.ts +6 -0
  189. package/dist/types/src/containers/index.d.ts.map +1 -0
  190. package/dist/types/src/hooks/index.d.ts +2 -0
  191. package/dist/types/src/hooks/index.d.ts.map +1 -1
  192. package/dist/types/src/hooks/useBreakpoints.d.ts +1 -1
  193. package/dist/types/src/hooks/useCompanions.d.ts.map +1 -1
  194. package/dist/types/src/hooks/useDeckCompanions.d.ts +4 -4
  195. package/dist/types/src/hooks/useDeckCompanions.d.ts.map +1 -1
  196. package/dist/types/src/hooks/useDeckState.d.ts +17 -0
  197. package/dist/types/src/hooks/useDeckState.d.ts.map +1 -0
  198. package/dist/types/src/hooks/useHoistStatusbar.d.ts +1 -1
  199. package/dist/types/src/hooks/useHoistStatusbar.d.ts.map +1 -1
  200. package/dist/types/src/hooks/useMainSize.d.ts +2 -2
  201. package/dist/types/src/hooks/useMainSize.d.ts.map +1 -1
  202. package/dist/types/src/hooks/useNodeActionExpander.d.ts +1 -1
  203. package/dist/types/src/hooks/useNodeActionExpander.d.ts.map +1 -1
  204. package/dist/types/src/hooks/useSelectedCompanion.d.ts +13 -0
  205. package/dist/types/src/hooks/useSelectedCompanion.d.ts.map +1 -0
  206. package/dist/types/src/index.d.ts +1 -4
  207. package/dist/types/src/index.d.ts.map +1 -1
  208. package/dist/types/src/layout.d.ts +20 -9
  209. package/dist/types/src/layout.d.ts.map +1 -1
  210. package/dist/types/src/layout.test.d.ts +2 -0
  211. package/dist/types/src/layout.test.d.ts.map +1 -0
  212. package/dist/types/src/meta.d.ts +2 -3
  213. package/dist/types/src/meta.d.ts.map +1 -1
  214. package/dist/types/src/operations/add-toast.d.ts +5 -0
  215. package/dist/types/src/operations/add-toast.d.ts.map +1 -0
  216. package/dist/types/src/operations/adjust.d.ts +5 -0
  217. package/dist/types/src/operations/adjust.d.ts.map +1 -0
  218. package/dist/types/src/operations/close.d.ts +5 -0
  219. package/dist/types/src/operations/close.d.ts.map +1 -0
  220. package/dist/types/src/operations/helpers.d.ts +3 -0
  221. package/dist/types/src/operations/helpers.d.ts.map +1 -0
  222. package/dist/types/src/operations/index.d.ts +3 -0
  223. package/dist/types/src/operations/index.d.ts.map +1 -0
  224. package/dist/types/src/operations/open.d.ts +5 -0
  225. package/dist/types/src/operations/open.d.ts.map +1 -0
  226. package/dist/types/src/operations/revert-workspace.d.ts +5 -0
  227. package/dist/types/src/operations/revert-workspace.d.ts.map +1 -0
  228. package/dist/types/src/operations/scroll-into-view.d.ts +5 -0
  229. package/dist/types/src/operations/scroll-into-view.d.ts.map +1 -0
  230. package/dist/types/src/operations/set-layout-mode.d.ts +9 -0
  231. package/dist/types/src/operations/set-layout-mode.d.ts.map +1 -0
  232. package/dist/types/src/operations/set.d.ts +5 -0
  233. package/dist/types/src/operations/set.d.ts.map +1 -0
  234. package/dist/types/src/operations/show-undo.d.ts +5 -0
  235. package/dist/types/src/operations/show-undo.d.ts.map +1 -0
  236. package/dist/types/src/operations/switch-workspace.d.ts +5 -0
  237. package/dist/types/src/operations/switch-workspace.d.ts.map +1 -0
  238. package/dist/types/src/operations/update-companion.d.ts +5 -0
  239. package/dist/types/src/operations/update-companion.d.ts.map +1 -0
  240. package/dist/types/src/operations/update-complementary.d.ts +5 -0
  241. package/dist/types/src/operations/update-complementary.d.ts.map +1 -0
  242. package/dist/types/src/operations/update-dialog.d.ts +5 -0
  243. package/dist/types/src/operations/update-dialog.d.ts.map +1 -0
  244. package/dist/types/src/operations/update-plank-size.d.ts +5 -0
  245. package/dist/types/src/operations/update-plank-size.d.ts.map +1 -0
  246. package/dist/types/src/operations/update-popover.d.ts +5 -0
  247. package/dist/types/src/operations/update-popover.d.ts.map +1 -0
  248. package/dist/types/src/operations/update-sidebar.d.ts +5 -0
  249. package/dist/types/src/operations/update-sidebar.d.ts.map +1 -0
  250. package/dist/types/src/plugin.d.ts +4 -0
  251. package/dist/types/src/plugin.d.ts.map +1 -0
  252. package/dist/types/src/translations.d.ts +45 -56
  253. package/dist/types/src/translations.d.ts.map +1 -1
  254. package/dist/types/src/types/DeckCapabilities.d.ts +188 -0
  255. package/dist/types/src/types/DeckCapabilities.d.ts.map +1 -0
  256. package/dist/types/src/types/DeckEvents.d.ts +5 -0
  257. package/dist/types/src/types/DeckEvents.d.ts.map +1 -0
  258. package/dist/types/src/types/DeckOperation.d.ts +15 -0
  259. package/dist/types/src/types/DeckOperation.d.ts.map +1 -0
  260. package/dist/types/src/types/Settings.d.ts +11 -0
  261. package/dist/types/src/types/Settings.d.ts.map +1 -0
  262. package/dist/types/src/types/index.d.ts +6 -0
  263. package/dist/types/src/types/index.d.ts.map +1 -0
  264. package/dist/types/src/{types.d.ts → types/schema.d.ts} +66 -66
  265. package/dist/types/src/types/schema.d.ts.map +1 -0
  266. package/dist/types/src/util/index.d.ts +2 -1
  267. package/dist/types/src/util/index.d.ts.map +1 -1
  268. package/dist/types/src/util/layoutAppliesTopbar.d.ts +1 -1
  269. package/dist/types/src/util/layoutAppliesTopbar.d.ts.map +1 -1
  270. package/dist/types/src/util/plank-url-params.d.ts +14 -0
  271. package/dist/types/src/util/plank-url-params.d.ts.map +1 -0
  272. package/dist/types/src/util/plank-url-params.test.d.ts +2 -0
  273. package/dist/types/src/util/plank-url-params.test.d.ts.map +1 -0
  274. package/dist/types/src/util/sanitize-persisted-state.d.ts +19 -0
  275. package/dist/types/src/util/sanitize-persisted-state.d.ts.map +1 -0
  276. package/dist/types/src/util/sanitize-persisted-state.test.d.ts +2 -0
  277. package/dist/types/src/util/sanitize-persisted-state.test.d.ts.map +1 -0
  278. package/dist/types/src/util/set-active.d.ts +19 -4
  279. package/dist/types/src/util/set-active.d.ts.map +1 -1
  280. package/dist/types/src/util/set-active.test.d.ts +2 -0
  281. package/dist/types/src/util/set-active.test.d.ts.map +1 -0
  282. package/dist/types/tsconfig.tsbuildinfo +1 -1
  283. package/package.json +128 -62
  284. package/src/DeckPlugin.node.ts +17 -0
  285. package/src/DeckPlugin.test.ts +27 -0
  286. package/src/DeckPlugin.ts +46 -65
  287. package/src/capabilities/app-graph-builder.ts +112 -126
  288. package/src/capabilities/check-app-scheme.ts +123 -24
  289. package/src/capabilities/index.ts +13 -13
  290. package/src/capabilities/operation-handler.ts +16 -0
  291. package/src/capabilities/react-root.tsx +37 -28
  292. package/src/capabilities/react-surface.tsx +28 -22
  293. package/src/capabilities/settings.ts +30 -19
  294. package/src/capabilities/state.ts +74 -87
  295. package/src/capabilities/tools.ts +65 -55
  296. package/src/capabilities/url-handler.ts +221 -49
  297. package/src/components/DeckSettings/DeckSettings.stories.tsx +38 -0
  298. package/src/components/DeckSettings/DeckSettings.tsx +19 -73
  299. package/src/components/DeckSettings/index.ts +2 -2
  300. package/src/components/Matrix/Matrix.stories.tsx +220 -0
  301. package/src/components/Matrix/Matrix.tsx +205 -0
  302. package/src/components/Matrix/SPEC.md +219 -0
  303. package/src/components/Matrix/index.ts +6 -0
  304. package/src/components/index.ts +6 -3
  305. package/src/containers/Deck/Banner.tsx +41 -0
  306. package/src/containers/Deck/Deck.stories.tsx +81 -0
  307. package/src/containers/Deck/Deck.tsx +21 -0
  308. package/src/containers/Deck/DeckContent.tsx +105 -0
  309. package/src/containers/Deck/DeckRoot.tsx +50 -0
  310. package/src/containers/Deck/DeckViewport.tsx +451 -0
  311. package/src/{components/DeckLayout → containers/Deck}/StatusBar.tsx +4 -4
  312. package/src/containers/Deck/index.ts +5 -0
  313. package/src/{components → containers}/DeckLayout/ActiveNode.tsx +9 -4
  314. package/src/containers/DeckLayout/DeckLayout.stories.tsx +397 -0
  315. package/src/containers/DeckLayout/DeckLayout.tsx +62 -0
  316. package/src/containers/DeckLayout/Dialog.tsx +52 -0
  317. package/src/containers/DeckLayout/Fallback.tsx +24 -0
  318. package/src/containers/DeckLayout/Popover.tsx +161 -0
  319. package/src/{components → containers}/DeckLayout/Toast.tsx +30 -5
  320. package/src/{components → containers}/DeckLayout/constants.ts +1 -0
  321. package/src/{components → containers}/DeckLayout/index.ts +3 -2
  322. package/src/containers/Plank/Plank.stories.tsx +108 -0
  323. package/src/containers/Plank/Plank.tsx +22 -0
  324. package/src/containers/Plank/PlankComponent.tsx +217 -0
  325. package/src/containers/Plank/PlankContent.tsx +45 -0
  326. package/src/{components → containers}/Plank/PlankControls.tsx +45 -38
  327. package/src/containers/Plank/PlankError.tsx +79 -0
  328. package/src/{components → containers}/Plank/PlankHeading.tsx +70 -65
  329. package/src/{components → containers}/Plank/PlankLoading.tsx +1 -1
  330. package/src/containers/Plank/PlankRoot.tsx +49 -0
  331. package/src/{components → containers}/Plank/index.ts +0 -2
  332. package/src/containers/Sidebar/ComplementarySidebar.tsx +208 -0
  333. package/src/containers/Sidebar/Sidebar.tsx +44 -0
  334. package/src/containers/Sidebar/SidebarButton.tsx +100 -0
  335. package/src/containers/index.ts +11 -0
  336. package/src/hooks/index.ts +2 -0
  337. package/src/hooks/useCompanions.ts +3 -3
  338. package/src/hooks/useDeckCompanions.ts +9 -11
  339. package/src/hooks/useDeckState.ts +73 -0
  340. package/src/hooks/useHoistStatusbar.ts +4 -5
  341. package/src/hooks/useMainSize.ts +2 -2
  342. package/src/hooks/useNodeActionExpander.ts +4 -4
  343. package/src/hooks/useSelectedCompanion.ts +32 -0
  344. package/src/index.ts +1 -4
  345. package/src/layout.test.ts +59 -0
  346. package/src/layout.ts +38 -40
  347. package/src/meta.ts +9 -5
  348. package/src/operations/add-toast.ts +24 -0
  349. package/src/operations/adjust.ts +82 -0
  350. package/src/operations/close.ts +35 -0
  351. package/src/operations/helpers.ts +22 -0
  352. package/src/operations/index.ts +24 -0
  353. package/src/operations/open.ts +184 -0
  354. package/src/operations/revert-workspace.ts +22 -0
  355. package/src/operations/scroll-into-view.ts +24 -0
  356. package/src/operations/set-layout-mode.ts +84 -0
  357. package/src/operations/set.ts +36 -0
  358. package/src/operations/show-undo.ts +47 -0
  359. package/src/operations/switch-workspace.ts +60 -0
  360. package/src/operations/update-companion.ts +35 -0
  361. package/src/operations/update-complementary.ts +33 -0
  362. package/src/operations/update-dialog.ts +34 -0
  363. package/src/operations/update-plank-size.ts +28 -0
  364. package/src/operations/update-popover.ts +36 -0
  365. package/src/operations/update-sidebar.ts +28 -0
  366. package/src/plugin.ts +11 -0
  367. package/src/translations.ts +47 -55
  368. package/src/types/DeckCapabilities.ts +34 -0
  369. package/src/types/DeckEvents.ts +21 -0
  370. package/src/types/DeckOperation.ts +51 -0
  371. package/src/types/Settings.ts +42 -0
  372. package/src/types/index.ts +10 -0
  373. package/src/{types.ts → types/schema.ts} +50 -59
  374. package/src/util/index.ts +2 -1
  375. package/src/util/layoutAppliesTopbar.ts +2 -2
  376. package/src/util/plank-url-params.test.ts +85 -0
  377. package/src/util/plank-url-params.ts +36 -0
  378. package/src/util/sanitize-persisted-state.test.ts +79 -0
  379. package/src/util/sanitize-persisted-state.ts +52 -0
  380. package/src/util/set-active.test.ts +106 -0
  381. package/src/util/set-active.ts +50 -30
  382. package/src/vite-env.d.ts +5 -0
  383. package/dist/lib/browser/app-graph-builder-YCO7Y54J.mjs +0 -152
  384. package/dist/lib/browser/app-graph-builder-YCO7Y54J.mjs.map +0 -7
  385. package/dist/lib/browser/check-app-scheme-7AXGR6UT.mjs +0 -32
  386. package/dist/lib/browser/check-app-scheme-7AXGR6UT.mjs.map +0 -7
  387. package/dist/lib/browser/chunk-FX44YX3G.mjs +0 -128
  388. package/dist/lib/browser/chunk-FX44YX3G.mjs.map +0 -7
  389. package/dist/lib/browser/chunk-KLN73CM3.mjs +0 -16
  390. package/dist/lib/browser/chunk-KLN73CM3.mjs.map +0 -7
  391. package/dist/lib/browser/chunk-NSATFAEE.mjs +0 -13
  392. package/dist/lib/browser/chunk-NSATFAEE.mjs.map +0 -7
  393. package/dist/lib/browser/chunk-RAZK4XT5.mjs +0 -127
  394. package/dist/lib/browser/chunk-RAZK4XT5.mjs.map +0 -7
  395. package/dist/lib/browser/chunk-RMCRP7HV.mjs +0 -1494
  396. package/dist/lib/browser/chunk-RMCRP7HV.mjs.map +0 -7
  397. package/dist/lib/browser/chunk-TRFYUEBA.mjs +0 -145
  398. package/dist/lib/browser/chunk-TRFYUEBA.mjs.map +0 -7
  399. package/dist/lib/browser/chunk-YN5OZEGS.mjs +0 -162
  400. package/dist/lib/browser/chunk-YN5OZEGS.mjs.map +0 -7
  401. package/dist/lib/browser/index.mjs +0 -168
  402. package/dist/lib/browser/index.mjs.map +0 -7
  403. package/dist/lib/browser/intent-resolver-EGLWTKJG.mjs +0 -521
  404. package/dist/lib/browser/intent-resolver-EGLWTKJG.mjs.map +0 -7
  405. package/dist/lib/browser/meta.json +0 -1
  406. package/dist/lib/browser/react-root-C4HIM5CI.mjs +0 -43
  407. package/dist/lib/browser/react-root-C4HIM5CI.mjs.map +0 -7
  408. package/dist/lib/browser/react-surface-RV2AZ6WZ.mjs +0 -40
  409. package/dist/lib/browser/react-surface-RV2AZ6WZ.mjs.map +0 -7
  410. package/dist/lib/browser/settings-7VUU3ZJ4.mjs +0 -29
  411. package/dist/lib/browser/settings-7VUU3ZJ4.mjs.map +0 -7
  412. package/dist/lib/browser/state-VJ6E3ADY.mjs +0 -10
  413. package/dist/lib/browser/tools-N57NQ2LH.mjs +0 -88
  414. package/dist/lib/browser/tools-N57NQ2LH.mjs.map +0 -7
  415. package/dist/lib/browser/types.mjs +0 -34
  416. package/dist/lib/browser/url-handler-BUGI6XRE.mjs +0 -70
  417. package/dist/lib/browser/url-handler-BUGI6XRE.mjs.map +0 -7
  418. package/dist/types/src/capabilities/capabilities.d.ts +0 -184
  419. package/dist/types/src/capabilities/capabilities.d.ts.map +0 -1
  420. package/dist/types/src/capabilities/intent-resolver.d.ts +0 -4
  421. package/dist/types/src/capabilities/intent-resolver.d.ts.map +0 -1
  422. package/dist/types/src/components/DeckLayout/ActiveNode.d.ts.map +0 -1
  423. package/dist/types/src/components/DeckLayout/Banner.d.ts.map +0 -1
  424. package/dist/types/src/components/DeckLayout/ContentEmpty.d.ts +0 -3
  425. package/dist/types/src/components/DeckLayout/ContentEmpty.d.ts.map +0 -1
  426. package/dist/types/src/components/DeckLayout/DeckLayout.d.ts +0 -6
  427. package/dist/types/src/components/DeckLayout/DeckLayout.d.ts.map +0 -1
  428. package/dist/types/src/components/DeckLayout/Dialog.d.ts.map +0 -1
  429. package/dist/types/src/components/DeckLayout/Fallback.d.ts +0 -3
  430. package/dist/types/src/components/DeckLayout/Fallback.d.ts.map +0 -1
  431. package/dist/types/src/components/DeckLayout/Popover.d.ts +0 -5
  432. package/dist/types/src/components/DeckLayout/Popover.d.ts.map +0 -1
  433. package/dist/types/src/components/DeckLayout/StatusBar.d.ts.map +0 -1
  434. package/dist/types/src/components/DeckLayout/Toast.d.ts +0 -5
  435. package/dist/types/src/components/DeckLayout/Toast.d.ts.map +0 -1
  436. package/dist/types/src/components/DeckLayout/Topbar.d.ts +0 -3
  437. package/dist/types/src/components/DeckLayout/Topbar.d.ts.map +0 -1
  438. package/dist/types/src/components/DeckLayout/constants.d.ts.map +0 -1
  439. package/dist/types/src/components/DeckLayout/index.d.ts +0 -4
  440. package/dist/types/src/components/DeckLayout/index.d.ts.map +0 -1
  441. package/dist/types/src/components/Plank/Plank.d.ts +0 -27
  442. package/dist/types/src/components/Plank/Plank.d.ts.map +0 -1
  443. package/dist/types/src/components/Plank/Plank.stories.d.ts +0 -8
  444. package/dist/types/src/components/Plank/Plank.stories.d.ts.map +0 -1
  445. package/dist/types/src/components/Plank/PlankControls.d.ts.map +0 -1
  446. package/dist/types/src/components/Plank/PlankError.d.ts +0 -13
  447. package/dist/types/src/components/Plank/PlankError.d.ts.map +0 -1
  448. package/dist/types/src/components/Plank/PlankHeading.d.ts.map +0 -1
  449. package/dist/types/src/components/Plank/index.d.ts +0 -6
  450. package/dist/types/src/components/Plank/index.d.ts.map +0 -1
  451. package/dist/types/src/components/Sidebar/ComplementarySidebar.d.ts.map +0 -1
  452. package/dist/types/src/components/Sidebar/Sidebar.d.ts.map +0 -1
  453. package/dist/types/src/components/Sidebar/SidebarButton.d.ts.map +0 -1
  454. package/dist/types/src/components/fragments.d.ts +0 -4
  455. package/dist/types/src/components/fragments.d.ts.map +0 -1
  456. package/dist/types/src/events.d.ts +0 -4
  457. package/dist/types/src/events.d.ts.map +0 -1
  458. package/dist/types/src/types.d.ts.map +0 -1
  459. package/dist/types/src/util/overscroll.d.ts +0 -47
  460. package/dist/types/src/util/overscroll.d.ts.map +0 -1
  461. package/src/capabilities/capabilities.ts +0 -14
  462. package/src/capabilities/intent-resolver.ts +0 -468
  463. package/src/components/DeckLayout/Banner.tsx +0 -39
  464. package/src/components/DeckLayout/ContentEmpty.tsx +0 -31
  465. package/src/components/DeckLayout/DeckLayout.tsx +0 -301
  466. package/src/components/DeckLayout/Dialog.tsx +0 -36
  467. package/src/components/DeckLayout/Fallback.tsx +0 -28
  468. package/src/components/DeckLayout/Popover.tsx +0 -104
  469. package/src/components/DeckLayout/Topbar.tsx +0 -11
  470. package/src/components/Plank/Plank.stories.tsx +0 -55
  471. package/src/components/Plank/Plank.tsx +0 -266
  472. package/src/components/Plank/PlankError.tsx +0 -51
  473. package/src/components/Sidebar/ComplementarySidebar.tsx +0 -193
  474. package/src/components/Sidebar/Sidebar.tsx +0 -42
  475. package/src/components/Sidebar/SidebarButton.tsx +0 -87
  476. package/src/components/fragments.ts +0 -14
  477. package/src/events.ts +0 -11
  478. package/src/util/overscroll.ts +0 -69
  479. /package/dist/lib/{browser/state-VJ6E3ADY.mjs.map → neutral/check-app-scheme-A7FZVNZO.mjs.map} +0 -0
  480. /package/dist/lib/{browser/types.mjs.map → neutral/chunk-J5LGTIGS.mjs.map} +0 -0
  481. /package/dist/types/src/{components/DeckLayout → containers/Deck}/StatusBar.d.ts +0 -0
  482. /package/dist/types/src/{components → containers}/DeckLayout/ActiveNode.d.ts +0 -0
  483. /package/dist/types/src/{components → containers}/DeckLayout/Dialog.d.ts +0 -0
  484. /package/dist/types/src/{components → containers}/DeckLayout/constants.d.ts +0 -0
  485. /package/dist/types/src/{components → containers}/Plank/PlankLoading.d.ts +0 -0
  486. /package/dist/types/src/{components → containers}/Sidebar/ComplementarySidebar.d.ts +0 -0
  487. /package/dist/types/src/{components → containers}/Sidebar/Sidebar.d.ts +0 -0
  488. /package/dist/types/src/{components → containers}/Sidebar/index.d.ts +0 -0
  489. /package/src/{components → containers}/Sidebar/index.ts +0 -0
@@ -2,140 +2,126 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { Rx } from '@effect-rx/rx-react';
6
- import { Option, pipe } from 'effect';
5
+ import * as Effect from 'effect/Effect';
7
6
 
8
- import { Capabilities, contributes, createIntent, LayoutAction, type PluginContext } from '@dxos/app-framework';
7
+ import { Capabilities, Capability } from '@dxos/app-framework';
8
+ import { AppCapabilities, AppNode, LayoutOperation } from '@dxos/app-toolkit';
9
+ import { Operation } from '@dxos/compute';
9
10
  import { AttentionCapabilities } from '@dxos/plugin-attention';
10
- import { createExtension, ROOT_ID, rxFromSignal } from '@dxos/plugin-graph';
11
+ import { GraphBuilder, NodeMatcher } from '@dxos/plugin-graph';
11
12
 
12
- import { DeckCapabilities } from './capabilities';
13
- import { DECK_PLUGIN } from '../meta';
13
+ import { meta } from '#meta';
14
+ import { DeckCapabilities } from '#types';
14
15
 
15
- export default (context: PluginContext) =>
16
- contributes(
17
- Capabilities.AppGraphBuilder,
18
- createExtension({
19
- id: DECK_PLUGIN,
20
- actions: (node) =>
21
- Rx.make((get) =>
22
- pipe(
23
- get(node),
24
- Option.flatMap((node) => (node.id === ROOT_ID ? Option.some(node) : Option.none())),
25
- Option.map(() => {
26
- const state = context.getCapability(DeckCapabilities.MutableDeckState);
16
+ export default Capability.makeModule(
17
+ Effect.fnUntraced(function* () {
18
+ const extensions = yield* Effect.all([
19
+ GraphBuilder.createExtension({
20
+ id: 'not-found',
21
+ match: NodeMatcher.whenRoot,
22
+ connector: () => Effect.succeed([AppNode.makeNotFound()]),
23
+ }),
27
24
 
28
- // NOTE(Zan): This is currently disabled.
29
- // TODO(Zan): Fullscreen needs to know the active node and provide that to the layout part.
30
- const _fullscreen = {
31
- id: `${LayoutAction.UpdateLayout._tag}/fullscreen`,
32
- data: async () => {
33
- const { dispatchPromise: dispatch } = context.getCapability(Capabilities.IntentDispatcher);
34
- await dispatch(
35
- createIntent(LayoutAction.SetLayoutMode, { part: 'mode', options: { mode: 'fullscreen' } }),
36
- );
37
- },
38
- properties: {
39
- label: ['toggle fullscreen label', { ns: DECK_PLUGIN }],
40
- icon: 'ph--arrows-out--regular',
41
- keyBinding: {
42
- macos: 'ctrl+meta+f',
43
- windows: 'shift+ctrl+f',
44
- },
45
- },
46
- };
25
+ GraphBuilder.createExtension({
26
+ id: 'root',
27
+ match: NodeMatcher.whenRoot,
28
+ actions: (_node, get) =>
29
+ Effect.gen(function* () {
30
+ // NOTE(Zan): This is currently disabled.
31
+ // TODO(Zan): Fullscreen needs to know the active node and provide that to the layout part.
32
+ // const _fullscreen = {
33
+ // id: `${LayoutAction.UpdateLayout._tag}/fullscreen`,
34
+ // data: async () => {
35
+ // const { dispatchPromise: dispatch } = context.get(Capabilities.IntentDispatcher);
36
+ // await dispatch(
37
+ // createIntent(LayoutAction.SetLayoutMode, { part: 'mode', options: { mode: 'fullscreen' } }),
38
+ // );
39
+ // },
40
+ // properties: {
41
+ // label: ['toggle-fullscreen.label', { ns: meta.id }],
42
+ // icon: 'ph--arrows-out--regular',
43
+ // keyBinding: {
44
+ // macos: 'ctrl+meta+f',
45
+ // windows: 'shift+ctrl+f',
46
+ // },
47
+ // },
48
+ // };
47
49
 
48
- const closeCurrent = {
49
- id: `${LayoutAction.Close._tag}/current`,
50
- data: async () => {
51
- const attention = context.getCapability(AttentionCapabilities.Attention);
52
- const attended = attention.current.at(-1);
53
- if (attended) {
54
- const { dispatchPromise: dispatch } = context.getCapability(Capabilities.IntentDispatcher);
55
- await dispatch(
56
- createIntent(LayoutAction.Close, {
57
- part: 'main',
58
- subject: [attended],
59
- options: { state: false },
60
- }),
61
- );
62
- }
63
- },
64
- properties: {
65
- label: ['close current label', { ns: DECK_PLUGIN }],
66
- icon: 'ph--x--regular',
67
- },
68
- };
50
+ const closeCurrent = {
51
+ id: `${LayoutOperation.Close.meta.key}.current`,
52
+ data: Effect.fnUntraced(function* () {
53
+ const attention = yield* Capability.get(AttentionCapabilities.Attention);
54
+ const attended = attention.getCurrent().at(-1);
55
+ if (attended) {
56
+ yield* Operation.invoke(LayoutOperation.Close, { subject: [attended] });
57
+ }
58
+ }),
59
+ properties: {
60
+ label: ['close-current.label', { ns: meta.id }],
61
+ icon: 'ph--x--regular',
62
+ },
63
+ };
69
64
 
70
- const closeOthers = {
71
- id: `${LayoutAction.Close._tag}/others`,
72
- data: async () => {
73
- const { dispatchPromise: dispatch } = context.getCapability(Capabilities.IntentDispatcher);
74
- const attention = context.getCapability(AttentionCapabilities.Attention);
75
- const attended = attention.current.at(-1);
76
- const ids = state.deck.active.filter((id) => id !== attended) ?? [];
77
- await dispatch(
78
- createIntent(LayoutAction.Close, { part: 'main', subject: ids, options: { state: false } }),
79
- );
80
- },
81
- properties: {
82
- label: ['close others label', { ns: DECK_PLUGIN }],
83
- icon: 'ph--x-square--regular',
84
- },
85
- };
65
+ const closeOthers = {
66
+ id: `${LayoutOperation.Close.meta.key}.others`,
67
+ data: Effect.fnUntraced(function* () {
68
+ const attention = yield* Capability.get(AttentionCapabilities.Attention);
69
+ const deck = yield* DeckCapabilities.getDeck();
70
+ const attended = attention.getCurrent().at(-1);
71
+ const ids = deck.active.filter((id: string) => id !== attended) ?? [];
72
+ yield* Operation.invoke(LayoutOperation.Close, { subject: ids });
73
+ }),
74
+ properties: {
75
+ label: ['close-others.label', { ns: meta.id }],
76
+ icon: 'ph--x-square--regular',
77
+ },
78
+ };
86
79
 
87
- const closeAll = {
88
- id: `${LayoutAction.Close._tag}/all`,
89
- data: async () => {
90
- const { dispatchPromise: dispatch } = context.getCapability(Capabilities.IntentDispatcher);
91
- await dispatch(
92
- createIntent(LayoutAction.Close, {
93
- part: 'main',
94
- subject: state.deck.active,
95
- options: { state: false },
96
- }),
97
- );
98
- },
99
- properties: {
100
- label: ['close all label', { ns: DECK_PLUGIN }],
101
- icon: 'ph--x-circle--regular',
102
- },
103
- };
80
+ const closeAll = {
81
+ id: `${LayoutOperation.Close.meta.key}.all`,
82
+ data: Effect.fnUntraced(function* () {
83
+ const deck = yield* DeckCapabilities.getDeck();
84
+ yield* Operation.invoke(LayoutOperation.Close, { subject: deck.active });
85
+ }),
86
+ properties: {
87
+ label: ['close-all.label', { ns: meta.id }],
88
+ icon: 'ph--x-circle--regular',
89
+ },
90
+ };
104
91
 
105
- const toggleSidebar = {
106
- id: `${LayoutAction.UpdateSidebar._tag}/nav`,
107
- data: async () => {
108
- state.sidebarState = state.sidebarState === 'expanded' ? 'collapsed' : 'expanded';
109
- },
110
- properties: {
111
- label: [
112
- get(
113
- rxFromSignal(() =>
114
- state.sidebarState === 'expanded'
115
- ? 'collapse navigation sidebar label'
116
- : 'open navigation sidebar label',
117
- ),
118
- ),
119
- { ns: DECK_PLUGIN },
120
- ],
121
- icon: 'ph--sidebar--regular',
122
- keyBinding: {
123
- macos: "meta+'",
124
- },
125
- disposition: 'pin-end',
126
- position: 'hoist',
127
- l0Breakpoint: 'lg',
92
+ const state = get(yield* Capability.get(DeckCapabilities.State));
93
+ const deck = state.decks[state.activeDeck];
94
+
95
+ const toggleSidebar = {
96
+ id: `${LayoutOperation.UpdateSidebar.meta.key}.nav`,
97
+ data: Effect.fnUntraced(function* () {
98
+ yield* Capabilities.updateAtomValue(DeckCapabilities.State, (s) => ({
99
+ ...s,
100
+ sidebarState: s.sidebarState === 'expanded' ? ('collapsed' as const) : ('expanded' as const),
101
+ }));
102
+ }),
103
+ properties: {
104
+ label: [
105
+ state.sidebarState === 'expanded'
106
+ ? 'collapse-navigation-sidebar.label'
107
+ : 'open-navigation-sidebar.label',
108
+ { ns: meta.id },
109
+ ],
110
+ icon: 'ph--sidebar--regular',
111
+ keyBinding: {
112
+ macos: "meta+'",
128
113
  },
129
- };
114
+ disposition: 'pin-end',
115
+ position: 'fallback',
116
+ l0Breakpoint: 'lg',
117
+ },
118
+ };
119
+
120
+ return !deck?.solo ? [closeCurrent, closeOthers, closeAll, toggleSidebar] : [toggleSidebar];
121
+ }),
122
+ }),
123
+ ]);
130
124
 
131
- return get(
132
- rxFromSignal(() =>
133
- !state.deck.solo ? [closeCurrent, closeOthers, closeAll, toggleSidebar] : [toggleSidebar],
134
- ),
135
- );
136
- }),
137
- Option.getOrElse(() => []),
138
- ),
139
- ),
140
- }),
141
- );
125
+ return Capability.contributes(AppCapabilities.AppGraphBuilder, extensions.flat());
126
+ }),
127
+ );
@@ -2,39 +2,138 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { Capabilities, contributes, type PluginContext } from '@dxos/app-framework';
5
+ import * as Effect from 'effect/Effect';
6
6
 
7
- import { DECK_PLUGIN } from '../meta';
8
- import { type DeckSettingsProps } from '../types';
7
+ import { Capabilities, Capability } from '@dxos/app-framework';
8
+ import { APP_SCHEME, AppCapabilities, LayoutOperation } from '@dxos/app-toolkit';
9
+ import { runAndForwardErrors } from '@dxos/effect';
10
+ import { isTauri } from '@dxos/util';
9
11
 
10
- const isSocket = !!(globalThis as any).__args;
12
+ import { DeckCapabilities } from '#types';
11
13
 
12
- // TODO(mjamesderocher): Can we get this directly from Socket?
13
- const appScheme = 'composer://';
14
+ /** Identifier for the native redirect dialog surface (defined in welcome plugin). */
15
+ const NATIVE_REDIRECT_DIALOG = 'org.dxos.plugin.welcome.component.native-redirect-dialog';
14
16
 
15
- // TODO(mjamesderocher): Factor out as part of NavigationPlugin.
16
- const checkAppScheme = (url: string) => {
17
- const iframe = document.createElement('iframe');
18
- iframe.style.display = 'none';
19
- document.body.appendChild(iframe);
17
+ const SCHEME_TIMEOUT_MS = 500;
20
18
 
21
- iframe.src = url + window.location.pathname.replace(/^\/+/, '') + window.location.search;
19
+ const isSafari = (): boolean => {
20
+ const ua = navigator.userAgent;
21
+ return ua.includes('Safari') && !ua.includes('Chrome') && !ua.includes('Chromium');
22
+ };
23
+
24
+ /** Check if the current page URL can be converted to a valid custom scheme URL. */
25
+ const canRedirectToScheme = (): boolean => {
26
+ try {
27
+ const schemeUrl = APP_SCHEME + window.location.pathname.replace(/^\/+/, '') + window.location.search;
28
+ new URL(schemeUrl);
29
+ return true;
30
+ } catch {
31
+ return false;
32
+ }
33
+ };
34
+
35
+ /**
36
+ * Try to open the native app via custom scheme.
37
+ * Resolves `true` if the app opened (page lost focus), `false` if it didn't within the timeout.
38
+ */
39
+ const tryOpenNativeApp = (): Promise<boolean> => {
40
+ return new Promise((resolve) => {
41
+ let resolved = false;
42
+
43
+ const onBlur = () => {
44
+ if (!resolved) {
45
+ resolved = true;
46
+ cleanup();
47
+ resolve(true);
48
+ }
49
+ };
50
+
51
+ const cleanup = () => {
52
+ window.removeEventListener('blur', onBlur);
53
+ document.removeEventListener('visibilitychange', onVisibilityChange);
54
+ };
22
55
 
23
- const timer = setTimeout(() => {
24
- document.body.removeChild(iframe);
25
- }, 3000);
56
+ const onVisibilityChange = () => {
57
+ if (document.hidden) {
58
+ onBlur();
59
+ }
60
+ };
26
61
 
27
- window.addEventListener('pagehide', (event) => {
28
- clearTimeout(timer);
29
- document.body.removeChild(iframe);
62
+ window.addEventListener('blur', onBlur);
63
+ document.addEventListener('visibilitychange', onVisibilityChange);
64
+
65
+ // Try the scheme via hidden iframe.
66
+ const schemeUrl = APP_SCHEME + window.location.pathname.replace(/^\/+/, '') + window.location.search;
67
+ const iframe = document.createElement('iframe');
68
+ iframe.style.display = 'none';
69
+ iframe.src = schemeUrl;
70
+ document.body.appendChild(iframe);
71
+ setTimeout(() => iframe.remove(), 3000);
72
+
73
+ // If no blur within timeout, the app isn't running.
74
+ setTimeout(() => {
75
+ if (!resolved) {
76
+ resolved = true;
77
+ cleanup();
78
+ resolve(false);
79
+ }
80
+ }, SCHEME_TIMEOUT_MS);
30
81
  });
31
82
  };
32
83
 
33
- export default async (context: PluginContext) => {
34
- const settings = context.getCapability(Capabilities.SettingsStore).getStore<DeckSettingsProps>(DECK_PLUGIN)?.value;
35
- if (!isSocket && settings?.enableNativeRedirect) {
36
- checkAppScheme(appScheme);
37
- }
84
+ /** Dispatch all NavigationHandler contributions with the current page URL. */
85
+ const dispatchNavigationHandlers = Effect.fn(function* () {
86
+ const url = new URL(window.location.href);
87
+ const handlers = yield* Capability.getAll(AppCapabilities.NavigationHandler);
88
+ yield* Effect.all(
89
+ handlers.map((handler) => handler(url)),
90
+ { concurrency: 'unbounded' },
91
+ );
92
+ });
38
93
 
39
- return contributes(Capabilities.Null, null);
94
+ /**
95
+ * Checks whether the native redirect setting is enabled and the redirect should be attempted.
96
+ * Exported for the URL handler to decide whether to defer NavigationHandler dispatch.
97
+ */
98
+ export const shouldDeferNavigationHandlers = (): boolean => {
99
+ return !isTauri() && !isSafari() && !import.meta.env.DEV && canRedirectToScheme();
40
100
  };
101
+
102
+ /**
103
+ * When running in a web browser (not Tauri) with native redirect enabled,
104
+ * tries to open the native app via custom scheme. Defers NavigationHandlers
105
+ * to prevent the web app from consuming one-time tokens before the native app.
106
+ *
107
+ * If the app opens: shows dialog with "Open here" callback that dispatches handlers.
108
+ * If the app doesn't open: dispatches handlers immediately.
109
+ * In Safari: universal links handle this natively, so the check is skipped.
110
+ */
111
+ export default Capability.makeModule(
112
+ Effect.fnUntraced(function* () {
113
+ const capabilities = yield* Capability.Service;
114
+ const { invoke } = yield* Capability.get(Capabilities.OperationInvoker);
115
+ const settings = yield* Capabilities.getAtomValue(DeckCapabilities.Settings);
116
+
117
+ if (!settings?.enableNativeRedirect || !shouldDeferNavigationHandlers()) {
118
+ return;
119
+ }
120
+
121
+ const appOpened = yield* Effect.promise(() => tryOpenNativeApp());
122
+ if (appOpened) {
123
+ const onOpenHere = () =>
124
+ Effect.gen(function* () {
125
+ yield* dispatchNavigationHandlers();
126
+ yield* invoke(LayoutOperation.UpdateDialog, { state: false });
127
+ }).pipe(Effect.provideService(Capability.Service, capabilities), runAndForwardErrors);
128
+
129
+ yield* invoke(LayoutOperation.UpdateDialog, {
130
+ subject: NATIVE_REDIRECT_DIALOG,
131
+ type: 'alert',
132
+ overlayClasses: 'dark bg-neutral-950!',
133
+ props: { onOpenHere },
134
+ });
135
+ } else {
136
+ yield* dispatchNavigationHandlers();
137
+ }
138
+ }),
139
+ );
@@ -2,17 +2,17 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { lazy } from '@dxos/app-framework';
5
+ import { Capability } from '@dxos/app-framework';
6
+ import { OperationHandlerSet } from '@dxos/compute';
6
7
 
7
- export const AppGraphBuilder = lazy(() => import('./app-graph-builder'));
8
- export const CheckAppScheme = lazy(() => import('./check-app-scheme'));
9
- export const DeckSettings = lazy(() => import('./settings'));
10
- export const DeckState = lazy(() => import('./state'));
11
- export const LayoutIntentResolver = lazy(() => import('./intent-resolver'));
12
- export const ReactRoot = lazy(() => import('./react-root'));
13
- export const ReactSurface = lazy(() => import('./react-surface'));
14
- export const Tools = lazy(() => import('./tools'));
15
- export const UrlHandler = lazy(() => import('./url-handler'));
16
-
17
- export * from './capabilities';
18
- export * from './state';
8
+ export const AppGraphBuilder = Capability.lazy('AppGraphBuilder', () => import('./app-graph-builder'));
9
+ export const CheckAppScheme = Capability.lazy('CheckAppScheme', () => import('./check-app-scheme'));
10
+ export const OperationHandler = Capability.lazy<OperationHandlerSet.OperationHandlerSet>(
11
+ 'OperationHandler',
12
+ () => import('./operation-handler'),
13
+ );
14
+ export const ReactRoot = Capability.lazy('ReactRoot', () => import('./react-root'));
15
+ export const ReactSurface = Capability.lazy('ReactSurface', () => import('./react-surface'));
16
+ export const DeckSettings = Capability.lazy('DeckSettings', () => import('./settings'));
17
+ export const DeckState = Capability.lazy('DeckState', () => import('./state'));
18
+ export const UrlHandler = Capability.lazy('UrlHandler', () => import('./url-handler'));
@@ -0,0 +1,16 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import * as Effect from 'effect/Effect';
6
+
7
+ import { Capabilities, Capability } from '@dxos/app-framework';
8
+ import type { OperationHandlerSet } from '@dxos/compute';
9
+
10
+ import { DeckOperationHandlerSet } from '#operations';
11
+
12
+ export default Capability.makeModule<OperationHandlerSet.OperationHandlerSet>(
13
+ Effect.fnUntraced(function* () {
14
+ return Capability.contributes(Capabilities.OperationHandler, DeckOperationHandlerSet);
15
+ }),
16
+ );
@@ -2,37 +2,46 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import * as Effect from 'effect/Effect';
5
6
  import React, { useCallback } from 'react';
6
7
 
7
- import { Capabilities, contributes, useCapability } from '@dxos/app-framework';
8
+ import { Capabilities, Capability } from '@dxos/app-framework';
8
9
 
9
- import { DeckCapabilities } from './capabilities';
10
- import { DeckLayout } from '../components';
11
- import { DECK_PLUGIN } from '../meta';
10
+ import { DeckLayout } from '#containers';
11
+ import { useDeckState } from '#hooks';
12
+ import { meta } from '#meta';
12
13
 
13
- export default () =>
14
- contributes(Capabilities.ReactRoot, {
15
- id: DECK_PLUGIN,
16
- root: () => {
17
- const layout = useCapability(DeckCapabilities.MutableDeckState);
14
+ export default Capability.makeModule(() =>
15
+ Effect.succeed(
16
+ Capability.contributes(Capabilities.ReactRoot, {
17
+ id: meta.id,
18
+ root: () => {
19
+ const { state, updateEphemeral } = useDeckState();
18
20
 
19
- const handleDismissToast = useCallback(
20
- (id: string) => {
21
- const index = layout.toasts.findIndex((toast) => toast.id === id);
22
- if (index !== -1) {
23
- // Allow time for the toast to animate out.
24
- // TODO(burdon): Factor out and unregister timeout.
25
- setTimeout(() => {
26
- if (layout.toasts[index].id === layout.currentUndoId) {
27
- layout.currentUndoId = undefined;
28
- }
29
- layout.toasts.splice(index, 1);
30
- }, 1_000);
31
- }
32
- },
33
- [layout.toasts],
34
- );
21
+ const handleDismissToast = useCallback(
22
+ (id: string) => {
23
+ const index = state.toasts.findIndex((toast) => toast.id === id);
24
+ if (index !== -1) {
25
+ // Allow time for the toast to animate out.
26
+ // TODO(burdon): Factor out and unregister timeout.
27
+ setTimeout(() => {
28
+ updateEphemeral((s) => {
29
+ const toastToRemove = s.toasts[index];
30
+ const newCurrentUndoId = toastToRemove?.id === s.currentUndoId ? undefined : s.currentUndoId;
31
+ return {
32
+ ...s,
33
+ currentUndoId: newCurrentUndoId,
34
+ toasts: s.toasts.filter((_, i) => i !== index),
35
+ };
36
+ });
37
+ }, 1_000);
38
+ }
39
+ },
40
+ [state.toasts, updateEphemeral],
41
+ );
35
42
 
36
- return <DeckLayout onDismissToast={handleDismissToast} />;
37
- },
38
- });
43
+ return <DeckLayout onDismissToast={handleDismissToast} />;
44
+ },
45
+ }),
46
+ ),
47
+ );
@@ -2,29 +2,35 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import * as Effect from 'effect/Effect';
5
6
  import React from 'react';
6
7
 
7
- import { Capabilities, contributes, createSurface } from '@dxos/app-framework';
8
- import { SettingsStore } from '@dxos/local-storage';
8
+ import { Capabilities, Capability } from '@dxos/app-framework';
9
+ import { Surface, useSettingsState } from '@dxos/app-framework/ui';
10
+ import { NOT_FOUND_PATH } from '@dxos/app-toolkit';
11
+ import { AppSurface, NotFoundArticle } from '@dxos/app-toolkit/ui';
9
12
 
10
- import { DeckSettings, Banner } from '../components';
11
- import { DECK_PLUGIN } from '../meta';
12
- import { type DeckSettingsProps } from '../types';
13
+ import { DeckSettings } from '#components';
14
+ import { meta } from '#meta';
15
+ import { type Settings } from '#types';
13
16
 
14
- export default () =>
15
- contributes(Capabilities.ReactSurface, [
16
- createSurface({
17
- id: `${DECK_PLUGIN}/plugin-settings`,
18
- role: 'article',
19
- filter: (data): data is { subject: SettingsStore<DeckSettingsProps> } =>
20
- data.subject instanceof SettingsStore && data.subject.prefix === DECK_PLUGIN,
21
- component: ({ data: { subject } }) => <DeckSettings settings={subject.value} />,
22
- }),
23
- createSurface({
24
- id: `${DECK_PLUGIN}/banner`,
25
- role: 'banner',
26
- component: ({ data }) => {
27
- return <Banner variant={data.variant} />;
28
- },
29
- }),
30
- ]);
17
+ export default Capability.makeModule(() =>
18
+ Effect.succeed(
19
+ Capability.contributes(Capabilities.ReactSurface, [
20
+ Surface.create({
21
+ id: 'plugin-settings',
22
+ filter: AppSurface.settings(AppSurface.Article, meta.id),
23
+ component: ({ data: { subject } }) => {
24
+ const { settings, updateSettings } = useSettingsState<Settings.Settings>(subject.atom);
25
+ return <DeckSettings settings={settings} onSettingsChange={updateSettings} />;
26
+ },
27
+ }),
28
+ Surface.create({
29
+ id: 'not-found',
30
+ role: 'article',
31
+ filter: (data): data is { attendableId: string } => data.attendableId === NOT_FOUND_PATH,
32
+ component: () => <NotFoundArticle />,
33
+ }),
34
+ ]),
35
+ ),
36
+ );