@dxos/plugin-deck 0.8.4-main.4a85c3132b → 0.8.4-main.51f1e5ca51

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 (548) hide show
  1. package/dist/lib/browser/add-toast-JWJOQNVL.mjs +24 -0
  2. package/dist/lib/browser/add-toast-JWJOQNVL.mjs.map +7 -0
  3. package/dist/lib/browser/adjust-RG6LIQLX.mjs +95 -0
  4. package/dist/lib/browser/adjust-RG6LIQLX.mjs.map +7 -0
  5. package/dist/lib/browser/chunk-GLB73Q6U.mjs +22 -0
  6. package/dist/lib/browser/chunk-GLB73Q6U.mjs.map +7 -0
  7. package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
  8. package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +7 -0
  9. package/dist/lib/browser/{chunk-BJDEG7YZ.mjs → chunk-LSZ47AY3.mjs} +3 -29
  10. package/dist/lib/browser/chunk-LSZ47AY3.mjs.map +7 -0
  11. package/dist/lib/browser/chunk-MARE7ZFK.mjs +215 -0
  12. package/dist/lib/browser/chunk-MARE7ZFK.mjs.map +7 -0
  13. package/dist/lib/browser/chunk-XM263CQU.mjs +72 -0
  14. package/dist/lib/browser/chunk-XM263CQU.mjs.map +7 -0
  15. package/dist/lib/browser/{chunk-SKBPLX5T.mjs → chunk-ZYYOSX5I.mjs} +22 -34
  16. package/dist/lib/browser/chunk-ZYYOSX5I.mjs.map +7 -0
  17. package/dist/lib/browser/close-URA766JO.mjs +44 -0
  18. package/dist/lib/browser/close-URA766JO.mjs.map +7 -0
  19. package/dist/lib/browser/index.mjs +78 -125
  20. package/dist/lib/browser/index.mjs.map +4 -4
  21. package/dist/lib/browser/meta.json +1 -1
  22. package/dist/lib/browser/open-OPFOOHCQ.mjs +147 -0
  23. package/dist/lib/browser/open-OPFOOHCQ.mjs.map +7 -0
  24. package/dist/lib/browser/operations/index.mjs +13 -0
  25. package/dist/lib/browser/operations/index.mjs.map +7 -0
  26. package/dist/lib/browser/revert-workspace-KP2Y4QLG.mjs +21 -0
  27. package/dist/lib/browser/revert-workspace-KP2Y4QLG.mjs.map +7 -0
  28. package/dist/lib/browser/scroll-into-view-S5H2FGSQ.mjs +21 -0
  29. package/dist/lib/browser/scroll-into-view-S5H2FGSQ.mjs.map +7 -0
  30. package/dist/lib/browser/set-AEKFFSIX.mjs +37 -0
  31. package/dist/lib/browser/set-AEKFFSIX.mjs.map +7 -0
  32. package/dist/lib/browser/set-layout-mode-CWIGQDFZ.mjs +85 -0
  33. package/dist/lib/browser/set-layout-mode-CWIGQDFZ.mjs.map +7 -0
  34. package/dist/lib/browser/show-undo-XFVEDKYC.mjs +59 -0
  35. package/dist/lib/browser/show-undo-XFVEDKYC.mjs.map +7 -0
  36. package/dist/lib/browser/switch-workspace-DSAAXZSI.mjs +60 -0
  37. package/dist/lib/browser/switch-workspace-DSAAXZSI.mjs.map +7 -0
  38. package/dist/lib/browser/translations.mjs +57 -0
  39. package/dist/lib/browser/translations.mjs.map +7 -0
  40. package/dist/lib/browser/types/index.mjs +8 -16
  41. package/dist/lib/browser/update-companion-IDBBBHT3.mjs +32 -0
  42. package/dist/lib/browser/update-companion-IDBBBHT3.mjs.map +7 -0
  43. package/dist/lib/browser/update-complementary-55PNO5MD.mjs +28 -0
  44. package/dist/lib/browser/update-complementary-55PNO5MD.mjs.map +7 -0
  45. package/dist/lib/browser/update-dialog-VNARIANC.mjs +29 -0
  46. package/dist/lib/browser/update-dialog-VNARIANC.mjs.map +7 -0
  47. package/dist/lib/browser/update-plank-size-H63PRRXS.mjs +28 -0
  48. package/dist/lib/browser/update-plank-size-H63PRRXS.mjs.map +7 -0
  49. package/dist/lib/browser/update-popover-ODTPZ3JL.mjs +33 -0
  50. package/dist/lib/browser/update-popover-ODTPZ3JL.mjs.map +7 -0
  51. package/dist/lib/browser/update-sidebar-JDVQ36WK.mjs +25 -0
  52. package/dist/lib/browser/update-sidebar-JDVQ36WK.mjs.map +7 -0
  53. package/dist/lib/node-esm/add-toast-JUTN4GTI.mjs +25 -0
  54. package/dist/lib/node-esm/add-toast-JUTN4GTI.mjs.map +7 -0
  55. package/dist/lib/node-esm/adjust-P5QK33P4.mjs +96 -0
  56. package/dist/lib/node-esm/adjust-P5QK33P4.mjs.map +7 -0
  57. package/dist/lib/node-esm/chunk-FO7YDJZ3.mjs +216 -0
  58. package/dist/lib/node-esm/chunk-FO7YDJZ3.mjs.map +7 -0
  59. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
  60. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +7 -0
  61. package/dist/lib/node-esm/{chunk-DGTRKKWZ.mjs → chunk-KY7ZCANM.mjs} +3 -29
  62. package/dist/lib/node-esm/chunk-KY7ZCANM.mjs.map +7 -0
  63. package/dist/lib/node-esm/chunk-L7OVX7CV.mjs +73 -0
  64. package/dist/lib/node-esm/chunk-L7OVX7CV.mjs.map +7 -0
  65. package/dist/lib/node-esm/chunk-SHSR2CYO.mjs +24 -0
  66. package/dist/lib/node-esm/chunk-SHSR2CYO.mjs.map +7 -0
  67. package/dist/lib/node-esm/{chunk-IR6ODCNC.mjs → chunk-WGBUTT5K.mjs} +22 -34
  68. package/dist/lib/node-esm/chunk-WGBUTT5K.mjs.map +7 -0
  69. package/dist/lib/node-esm/close-U5VC7ERP.mjs +45 -0
  70. package/dist/lib/node-esm/close-U5VC7ERP.mjs.map +7 -0
  71. package/dist/lib/node-esm/index.mjs +78 -125
  72. package/dist/lib/node-esm/index.mjs.map +4 -4
  73. package/dist/lib/node-esm/meta.json +1 -1
  74. package/dist/lib/node-esm/open-4B3P5XGS.mjs +148 -0
  75. package/dist/lib/node-esm/open-4B3P5XGS.mjs.map +7 -0
  76. package/dist/lib/node-esm/operations/index.mjs +14 -0
  77. package/dist/lib/node-esm/operations/index.mjs.map +7 -0
  78. package/dist/lib/node-esm/revert-workspace-QD63M2U7.mjs +22 -0
  79. package/dist/lib/node-esm/revert-workspace-QD63M2U7.mjs.map +7 -0
  80. package/dist/lib/node-esm/scroll-into-view-G5M2EWK7.mjs +22 -0
  81. package/dist/lib/node-esm/scroll-into-view-G5M2EWK7.mjs.map +7 -0
  82. package/dist/lib/node-esm/set-ERJCXZBY.mjs +38 -0
  83. package/dist/lib/node-esm/set-ERJCXZBY.mjs.map +7 -0
  84. package/dist/lib/node-esm/set-layout-mode-BOZD7744.mjs +86 -0
  85. package/dist/lib/node-esm/set-layout-mode-BOZD7744.mjs.map +7 -0
  86. package/dist/lib/node-esm/show-undo-H5BRBYDC.mjs +60 -0
  87. package/dist/lib/node-esm/show-undo-H5BRBYDC.mjs.map +7 -0
  88. package/dist/lib/node-esm/switch-workspace-F54PXE5N.mjs +61 -0
  89. package/dist/lib/node-esm/switch-workspace-F54PXE5N.mjs.map +7 -0
  90. package/dist/lib/node-esm/translations.mjs +58 -0
  91. package/dist/lib/node-esm/translations.mjs.map +7 -0
  92. package/dist/lib/node-esm/types/index.mjs +8 -16
  93. package/dist/lib/node-esm/update-companion-47LMKL5R.mjs +33 -0
  94. package/dist/lib/node-esm/update-companion-47LMKL5R.mjs.map +7 -0
  95. package/dist/lib/node-esm/update-complementary-7VYYG7L2.mjs +29 -0
  96. package/dist/lib/node-esm/update-complementary-7VYYG7L2.mjs.map +7 -0
  97. package/dist/lib/node-esm/update-dialog-CABL7OWW.mjs +30 -0
  98. package/dist/lib/node-esm/update-dialog-CABL7OWW.mjs.map +7 -0
  99. package/dist/lib/node-esm/update-plank-size-WYTSMZF2.mjs +29 -0
  100. package/dist/lib/node-esm/update-plank-size-WYTSMZF2.mjs.map +7 -0
  101. package/dist/lib/node-esm/update-popover-YZDYGL7G.mjs +34 -0
  102. package/dist/lib/node-esm/update-popover-YZDYGL7G.mjs.map +7 -0
  103. package/dist/lib/node-esm/update-sidebar-HV736LUN.mjs +26 -0
  104. package/dist/lib/node-esm/update-sidebar-HV736LUN.mjs.map +7 -0
  105. package/dist/types/src/DeckPlugin.d.ts.map +1 -1
  106. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -0
  107. package/dist/types/src/capabilities/check-app-scheme.d.ts +19 -0
  108. package/dist/types/src/capabilities/check-app-scheme.d.ts.map +1 -0
  109. package/dist/types/src/capabilities/index.d.ts +198 -9
  110. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  111. package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
  112. package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
  113. package/dist/types/src/capabilities/react-root.d.ts.map +1 -0
  114. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -0
  115. package/dist/types/src/capabilities/settings.d.ts +7 -0
  116. package/dist/types/src/capabilities/settings.d.ts.map +1 -0
  117. package/dist/types/src/capabilities/{state/state.d.ts → state.d.ts} +38 -22
  118. package/dist/types/src/capabilities/state.d.ts.map +1 -0
  119. package/dist/types/src/capabilities/{tools/tools.d.ts → tools.d.ts} +1 -1
  120. package/dist/types/src/capabilities/tools.d.ts.map +1 -0
  121. package/dist/types/src/capabilities/{url-handler/url-handler.d.ts → url-handler.d.ts} +1 -1
  122. package/dist/types/src/capabilities/url-handler.d.ts.map +1 -0
  123. package/dist/types/src/components/DeckSettings/DeckSettings.d.ts +6 -0
  124. package/dist/types/src/components/DeckSettings/DeckSettings.d.ts.map +1 -0
  125. package/dist/types/src/components/DeckSettings/DeckSettings.stories.d.ts +61 -0
  126. package/dist/types/src/components/DeckSettings/DeckSettings.stories.d.ts.map +1 -0
  127. package/dist/types/src/components/DeckSettings/index.d.ts +2 -0
  128. package/dist/types/src/components/DeckSettings/index.d.ts.map +1 -0
  129. package/dist/types/src/components/Matrix/Matrix.d.ts +41 -0
  130. package/dist/types/src/components/Matrix/Matrix.d.ts.map +1 -0
  131. package/dist/types/src/components/Matrix/Matrix.stories.d.ts +17 -0
  132. package/dist/types/src/components/Matrix/Matrix.stories.d.ts.map +1 -0
  133. package/dist/types/src/components/Matrix/index.d.ts +3 -0
  134. package/dist/types/src/components/Matrix/index.d.ts.map +1 -0
  135. package/dist/types/src/components/index.d.ts +3 -1
  136. package/dist/types/src/components/index.d.ts.map +1 -1
  137. package/dist/types/src/{components/DeckLayout → containers/Deck}/Banner.d.ts +1 -1
  138. package/dist/types/src/containers/Deck/Banner.d.ts.map +1 -0
  139. package/dist/types/src/containers/Deck/Deck.d.ts +25 -0
  140. package/dist/types/src/containers/Deck/Deck.d.ts.map +1 -0
  141. package/dist/types/src/containers/Deck/Deck.stories.d.ts +61 -0
  142. package/dist/types/src/containers/Deck/Deck.stories.d.ts.map +1 -0
  143. package/dist/types/src/containers/Deck/DeckContent.d.ts +7 -0
  144. package/dist/types/src/containers/Deck/DeckContent.d.ts.map +1 -0
  145. package/dist/types/src/containers/Deck/DeckRoot.d.ts +39 -0
  146. package/dist/types/src/containers/Deck/DeckRoot.d.ts.map +1 -0
  147. package/dist/types/src/containers/Deck/DeckViewport.d.ts +16 -0
  148. package/dist/types/src/containers/Deck/DeckViewport.d.ts.map +1 -0
  149. package/dist/types/src/containers/Deck/StatusBar.d.ts.map +1 -0
  150. package/dist/types/src/containers/Deck/index.d.ts +2 -0
  151. package/dist/types/src/containers/Deck/index.d.ts.map +1 -0
  152. package/dist/types/src/{components → containers}/DeckLayout/ActiveNode.d.ts.map +1 -1
  153. package/dist/types/src/containers/DeckLayout/DeckLayout.d.ts.map +1 -0
  154. package/dist/types/src/containers/DeckLayout/DeckLayout.stories.d.ts +63 -0
  155. package/dist/types/src/containers/DeckLayout/DeckLayout.stories.d.ts.map +1 -0
  156. package/dist/types/src/containers/DeckLayout/Dialog.d.ts.map +1 -0
  157. package/dist/types/src/containers/DeckLayout/Fallback.d.ts +2 -0
  158. package/dist/types/src/containers/DeckLayout/Fallback.d.ts.map +1 -0
  159. package/dist/types/src/containers/DeckLayout/Popover.d.ts +5 -0
  160. package/dist/types/src/containers/DeckLayout/Popover.d.ts.map +1 -0
  161. package/dist/types/src/{components → containers}/DeckLayout/Toast.d.ts +1 -1
  162. package/dist/types/src/containers/DeckLayout/Toast.d.ts.map +1 -0
  163. package/dist/types/src/containers/DeckLayout/constants.d.ts.map +1 -0
  164. package/dist/types/src/containers/DeckLayout/index.d.ts +4 -0
  165. package/dist/types/src/containers/DeckLayout/index.d.ts.map +1 -0
  166. package/dist/types/src/containers/Plank/Plank.d.ts +17 -0
  167. package/dist/types/src/containers/Plank/Plank.d.ts.map +1 -0
  168. package/dist/types/src/containers/Plank/Plank.stories.d.ts +61 -0
  169. package/dist/types/src/containers/Plank/Plank.stories.d.ts.map +1 -0
  170. package/dist/types/src/containers/Plank/PlankComponent.d.ts +17 -0
  171. package/dist/types/src/containers/Plank/PlankComponent.d.ts.map +1 -0
  172. package/dist/types/src/containers/Plank/PlankContent.d.ts +10 -0
  173. package/dist/types/src/containers/Plank/PlankContent.d.ts.map +1 -0
  174. package/dist/types/src/{components → containers}/Plank/PlankControls.d.ts +5 -6
  175. package/dist/types/src/containers/Plank/PlankControls.d.ts.map +1 -0
  176. package/dist/types/src/{components → containers}/Plank/PlankError.d.ts +2 -2
  177. package/dist/types/src/containers/Plank/PlankError.d.ts.map +1 -0
  178. package/dist/types/src/{components → containers}/Plank/PlankHeading.d.ts +3 -2
  179. package/dist/types/src/containers/Plank/PlankHeading.d.ts.map +1 -0
  180. package/dist/types/src/{components → containers}/Plank/PlankLoading.d.ts.map +1 -1
  181. package/dist/types/src/containers/Plank/PlankRoot.d.ts +37 -0
  182. package/dist/types/src/containers/Plank/PlankRoot.d.ts.map +1 -0
  183. package/dist/types/src/containers/Plank/index.d.ts +4 -0
  184. package/dist/types/src/containers/Plank/index.d.ts.map +1 -0
  185. package/dist/types/src/containers/Sidebar/ComplementarySidebar.d.ts.map +1 -0
  186. package/dist/types/src/containers/Sidebar/Sidebar.d.ts.map +1 -0
  187. package/dist/types/src/{components → containers}/Sidebar/SidebarButton.d.ts +1 -1
  188. package/dist/types/src/containers/Sidebar/SidebarButton.d.ts.map +1 -0
  189. package/dist/types/src/{components → containers}/Sidebar/index.d.ts.map +1 -1
  190. package/dist/types/src/containers/index.d.ts +4 -2
  191. package/dist/types/src/containers/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 +2 -2
  195. package/dist/types/src/hooks/useDeckCompanions.d.ts.map +1 -1
  196. package/dist/types/src/hooks/useDeckState.d.ts +3 -3
  197. package/dist/types/src/hooks/useDeckState.d.ts.map +1 -1
  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.map +1 -1
  201. package/dist/types/src/hooks/useNodeActionExpander.d.ts.map +1 -1
  202. package/dist/types/src/hooks/useSelectedCompanion.d.ts.map +1 -1
  203. package/dist/types/src/layout.d.ts +20 -9
  204. package/dist/types/src/layout.d.ts.map +1 -1
  205. package/dist/types/src/layout.test.d.ts +2 -0
  206. package/dist/types/src/layout.test.d.ts.map +1 -0
  207. package/dist/types/src/operations/add-toast.d.ts +5 -0
  208. package/dist/types/src/operations/add-toast.d.ts.map +1 -0
  209. package/dist/types/src/operations/adjust.d.ts +5 -0
  210. package/dist/types/src/operations/adjust.d.ts.map +1 -0
  211. package/dist/types/src/operations/close.d.ts +5 -0
  212. package/dist/types/src/operations/close.d.ts.map +1 -0
  213. package/dist/types/src/operations/definitions.d.ts +15 -0
  214. package/dist/types/src/operations/definitions.d.ts.map +1 -0
  215. package/dist/types/src/operations/helpers.d.ts +3 -0
  216. package/dist/types/src/operations/helpers.d.ts.map +1 -0
  217. package/dist/types/src/operations/index.d.ts +4 -0
  218. package/dist/types/src/operations/index.d.ts.map +1 -0
  219. package/dist/types/src/operations/open.d.ts +5 -0
  220. package/dist/types/src/operations/open.d.ts.map +1 -0
  221. package/dist/types/src/operations/revert-workspace.d.ts +5 -0
  222. package/dist/types/src/operations/revert-workspace.d.ts.map +1 -0
  223. package/dist/types/src/operations/scroll-into-view.d.ts +5 -0
  224. package/dist/types/src/operations/scroll-into-view.d.ts.map +1 -0
  225. package/dist/types/src/operations/set-layout-mode.d.ts +9 -0
  226. package/dist/types/src/operations/set-layout-mode.d.ts.map +1 -0
  227. package/dist/types/src/operations/set.d.ts +5 -0
  228. package/dist/types/src/operations/set.d.ts.map +1 -0
  229. package/dist/types/src/operations/show-undo.d.ts +5 -0
  230. package/dist/types/src/operations/show-undo.d.ts.map +1 -0
  231. package/dist/types/src/operations/switch-workspace.d.ts +5 -0
  232. package/dist/types/src/operations/switch-workspace.d.ts.map +1 -0
  233. package/dist/types/src/operations/update-companion.d.ts +5 -0
  234. package/dist/types/src/operations/update-companion.d.ts.map +1 -0
  235. package/dist/types/src/operations/update-complementary.d.ts +5 -0
  236. package/dist/types/src/operations/update-complementary.d.ts.map +1 -0
  237. package/dist/types/src/operations/update-dialog.d.ts +5 -0
  238. package/dist/types/src/operations/update-dialog.d.ts.map +1 -0
  239. package/dist/types/src/operations/update-plank-size.d.ts +5 -0
  240. package/dist/types/src/operations/update-plank-size.d.ts.map +1 -0
  241. package/dist/types/src/operations/update-popover.d.ts +5 -0
  242. package/dist/types/src/operations/update-popover.d.ts.map +1 -0
  243. package/dist/types/src/operations/update-sidebar.d.ts +5 -0
  244. package/dist/types/src/operations/update-sidebar.d.ts.map +1 -0
  245. package/dist/types/src/translations.d.ts +42 -56
  246. package/dist/types/src/translations.d.ts.map +1 -1
  247. package/dist/types/src/types/Settings.d.ts +11 -0
  248. package/dist/types/src/types/Settings.d.ts.map +1 -0
  249. package/dist/types/src/types/capabilities.d.ts +41 -40
  250. package/dist/types/src/types/capabilities.d.ts.map +1 -1
  251. package/dist/types/src/types/events.d.ts.map +1 -1
  252. package/dist/types/src/types/index.d.ts +1 -0
  253. package/dist/types/src/types/index.d.ts.map +1 -1
  254. package/dist/types/src/types/schema.d.ts +44 -104
  255. package/dist/types/src/types/schema.d.ts.map +1 -1
  256. package/dist/types/src/util/index.d.ts +2 -1
  257. package/dist/types/src/util/index.d.ts.map +1 -1
  258. package/dist/types/src/util/layoutAppliesTopbar.d.ts +1 -1
  259. package/dist/types/src/util/layoutAppliesTopbar.d.ts.map +1 -1
  260. package/dist/types/src/util/plank-url-params.d.ts +14 -0
  261. package/dist/types/src/util/plank-url-params.d.ts.map +1 -0
  262. package/dist/types/src/util/plank-url-params.test.d.ts +2 -0
  263. package/dist/types/src/util/plank-url-params.test.d.ts.map +1 -0
  264. package/dist/types/src/util/sanitize-persisted-state.d.ts +19 -0
  265. package/dist/types/src/util/sanitize-persisted-state.d.ts.map +1 -0
  266. package/dist/types/src/util/sanitize-persisted-state.test.d.ts +2 -0
  267. package/dist/types/src/util/sanitize-persisted-state.test.d.ts.map +1 -0
  268. package/dist/types/src/util/set-active.d.ts +2 -2
  269. package/dist/types/src/util/set-active.d.ts.map +1 -1
  270. package/dist/types/src/util/set-active.test.d.ts +2 -0
  271. package/dist/types/src/util/set-active.test.d.ts.map +1 -0
  272. package/dist/types/tsconfig.tsbuildinfo +1 -1
  273. package/package.json +71 -45
  274. package/src/DeckPlugin.ts +10 -16
  275. package/src/capabilities/app-graph-builder.ts +127 -0
  276. package/src/capabilities/check-app-scheme.ts +139 -0
  277. package/src/capabilities/index.ts +14 -10
  278. package/src/capabilities/operation-handler.ts +16 -0
  279. package/src/capabilities/{react-root/react-root.tsx → react-root.tsx} +3 -3
  280. package/src/capabilities/{react-surface/react-surface.tsx → react-surface.tsx} +12 -14
  281. package/src/capabilities/{settings/settings.ts → settings.ts} +4 -6
  282. package/src/capabilities/{state/state.ts → state.ts} +13 -27
  283. package/src/capabilities/{tools/tools.ts → tools.ts} +6 -6
  284. package/src/capabilities/url-handler.ts +231 -0
  285. package/src/components/DeckSettings/DeckSettings.stories.tsx +38 -0
  286. package/src/components/DeckSettings/DeckSettings.tsx +34 -0
  287. package/src/components/DeckSettings/index.ts +5 -0
  288. package/src/components/Matrix/Matrix.stories.tsx +220 -0
  289. package/src/components/Matrix/Matrix.tsx +205 -0
  290. package/src/components/Matrix/SPEC.md +219 -0
  291. package/src/components/Matrix/index.ts +6 -0
  292. package/src/components/index.ts +6 -2
  293. package/src/{components/DeckLayout → containers/Deck}/Banner.tsx +5 -3
  294. package/src/containers/Deck/Deck.stories.tsx +81 -0
  295. package/src/containers/Deck/Deck.tsx +21 -0
  296. package/src/containers/Deck/DeckContent.tsx +105 -0
  297. package/src/containers/Deck/DeckRoot.tsx +50 -0
  298. package/src/containers/Deck/DeckViewport.tsx +456 -0
  299. package/src/containers/Deck/index.ts +5 -0
  300. package/src/{components → containers}/DeckLayout/ActiveNode.tsx +7 -3
  301. package/src/containers/DeckLayout/DeckLayout.stories.tsx +397 -0
  302. package/src/containers/DeckLayout/DeckLayout.tsx +62 -0
  303. package/src/{components → containers}/DeckLayout/Dialog.tsx +11 -4
  304. package/src/{components → containers}/DeckLayout/Fallback.tsx +3 -3
  305. package/src/{components → containers}/DeckLayout/Popover.tsx +62 -43
  306. package/src/{components → containers}/DeckLayout/Toast.tsx +1 -1
  307. package/src/{components → containers}/DeckLayout/constants.ts +1 -0
  308. package/src/{components → containers}/DeckLayout/index.ts +3 -2
  309. package/src/containers/Plank/Plank.stories.tsx +108 -0
  310. package/src/containers/Plank/Plank.tsx +22 -0
  311. package/src/containers/Plank/PlankComponent.tsx +217 -0
  312. package/src/containers/Plank/PlankContent.tsx +45 -0
  313. package/src/{components → containers}/Plank/PlankControls.tsx +37 -28
  314. package/src/{components → containers}/Plank/PlankError.tsx +20 -4
  315. package/src/{components → containers}/Plank/PlankHeading.tsx +53 -42
  316. package/src/containers/Plank/PlankRoot.tsx +49 -0
  317. package/src/{components → containers}/Plank/index.ts +0 -2
  318. package/src/{components → containers}/Sidebar/ComplementarySidebar.tsx +65 -80
  319. package/src/{components → containers}/Sidebar/Sidebar.tsx +8 -6
  320. package/src/{components → containers}/Sidebar/SidebarButton.tsx +16 -10
  321. package/src/containers/index.ts +5 -2
  322. package/src/hooks/useCompanions.ts +1 -1
  323. package/src/hooks/useDeckCompanions.ts +3 -7
  324. package/src/hooks/useDeckState.ts +11 -20
  325. package/src/hooks/useHoistStatusbar.ts +1 -1
  326. package/src/hooks/useSelectedCompanion.ts +4 -10
  327. package/src/layout.test.ts +59 -0
  328. package/src/layout.ts +38 -40
  329. package/src/meta.ts +1 -1
  330. package/src/operations/add-toast.ts +24 -0
  331. package/src/operations/adjust.ts +83 -0
  332. package/src/operations/close.ts +35 -0
  333. package/src/operations/definitions.ts +49 -0
  334. package/src/operations/helpers.ts +22 -0
  335. package/src/operations/index.ts +26 -0
  336. package/src/operations/open.ts +184 -0
  337. package/src/operations/revert-workspace.ts +22 -0
  338. package/src/operations/scroll-into-view.ts +24 -0
  339. package/src/operations/set-layout-mode.ts +84 -0
  340. package/src/operations/set.ts +36 -0
  341. package/src/operations/show-undo.ts +47 -0
  342. package/src/operations/switch-workspace.ts +60 -0
  343. package/src/operations/update-companion.ts +35 -0
  344. package/src/operations/update-complementary.ts +33 -0
  345. package/src/operations/update-dialog.ts +34 -0
  346. package/src/operations/update-plank-size.ts +29 -0
  347. package/src/operations/update-popover.ts +36 -0
  348. package/src/operations/update-sidebar.ts +28 -0
  349. package/src/translations.ts +42 -55
  350. package/src/types/Settings.ts +42 -0
  351. package/src/types/capabilities.ts +8 -6
  352. package/src/types/events.ts +2 -2
  353. package/src/types/index.ts +1 -0
  354. package/src/types/schema.ts +36 -116
  355. package/src/util/index.ts +2 -1
  356. package/src/util/layoutAppliesTopbar.ts +1 -1
  357. package/src/util/plank-url-params.test.ts +85 -0
  358. package/src/util/plank-url-params.ts +36 -0
  359. package/src/util/sanitize-persisted-state.test.ts +79 -0
  360. package/src/util/sanitize-persisted-state.ts +52 -0
  361. package/src/util/set-active.test.ts +106 -0
  362. package/src/util/set-active.ts +2 -2
  363. package/src/vite-env.d.ts +5 -0
  364. package/dist/lib/browser/Banner-DR6B7XTO.mjs +0 -15
  365. package/dist/lib/browser/Banner-DR6B7XTO.mjs.map +0 -7
  366. package/dist/lib/browser/DeckSettings-HB5TKG2F.mjs +0 -96
  367. package/dist/lib/browser/DeckSettings-HB5TKG2F.mjs.map +0 -7
  368. package/dist/lib/browser/app-graph-builder-22XAL7YE.mjs +0 -123
  369. package/dist/lib/browser/app-graph-builder-22XAL7YE.mjs.map +0 -7
  370. package/dist/lib/browser/check-app-scheme-ZV3QWNNZ.mjs +0 -33
  371. package/dist/lib/browser/check-app-scheme-ZV3QWNNZ.mjs.map +0 -7
  372. package/dist/lib/browser/chunk-BJDEG7YZ.mjs.map +0 -7
  373. package/dist/lib/browser/chunk-FBL4WVYZ.mjs +0 -16
  374. package/dist/lib/browser/chunk-FBL4WVYZ.mjs.map +0 -7
  375. package/dist/lib/browser/chunk-OAPMV5ZR.mjs +0 -280
  376. package/dist/lib/browser/chunk-OAPMV5ZR.mjs.map +0 -7
  377. package/dist/lib/browser/chunk-Q4GUCIRE.mjs +0 -200
  378. package/dist/lib/browser/chunk-Q4GUCIRE.mjs.map +0 -7
  379. package/dist/lib/browser/chunk-SKBPLX5T.mjs.map +0 -7
  380. package/dist/lib/browser/chunk-YGOOWRJE.mjs +0 -1390
  381. package/dist/lib/browser/chunk-YGOOWRJE.mjs.map +0 -7
  382. package/dist/lib/browser/operation-resolver-BLZGJWLO.mjs +0 -580
  383. package/dist/lib/browser/operation-resolver-BLZGJWLO.mjs.map +0 -7
  384. package/dist/lib/browser/react-root-4ESZAM6D.mjs +0 -49
  385. package/dist/lib/browser/react-root-4ESZAM6D.mjs.map +0 -7
  386. package/dist/lib/browser/react-surface-KYQGCALN.mjs +0 -44
  387. package/dist/lib/browser/react-surface-KYQGCALN.mjs.map +0 -7
  388. package/dist/lib/browser/settings-RY2TBSGP.mjs +0 -40
  389. package/dist/lib/browser/settings-RY2TBSGP.mjs.map +0 -7
  390. package/dist/lib/browser/state-CSTTMJ43.mjs +0 -106
  391. package/dist/lib/browser/state-CSTTMJ43.mjs.map +0 -7
  392. package/dist/lib/browser/toolkit-EONRPYA5.mjs +0 -55
  393. package/dist/lib/browser/toolkit-EONRPYA5.mjs.map +0 -7
  394. package/dist/lib/browser/url-handler-EM3SYHHX.mjs +0 -95
  395. package/dist/lib/browser/url-handler-EM3SYHHX.mjs.map +0 -7
  396. package/dist/lib/node-esm/Banner-HR7DPUZU.mjs +0 -16
  397. package/dist/lib/node-esm/Banner-HR7DPUZU.mjs.map +0 -7
  398. package/dist/lib/node-esm/DeckSettings-MWTR7HU2.mjs +0 -97
  399. package/dist/lib/node-esm/DeckSettings-MWTR7HU2.mjs.map +0 -7
  400. package/dist/lib/node-esm/app-graph-builder-GNNNCWGN.mjs +0 -124
  401. package/dist/lib/node-esm/app-graph-builder-GNNNCWGN.mjs.map +0 -7
  402. package/dist/lib/node-esm/check-app-scheme-ZNISVRFH.mjs +0 -34
  403. package/dist/lib/node-esm/check-app-scheme-ZNISVRFH.mjs.map +0 -7
  404. package/dist/lib/node-esm/chunk-DGTRKKWZ.mjs.map +0 -7
  405. package/dist/lib/node-esm/chunk-EGFOT3DE.mjs +0 -18
  406. package/dist/lib/node-esm/chunk-EGFOT3DE.mjs.map +0 -7
  407. package/dist/lib/node-esm/chunk-IR6ODCNC.mjs.map +0 -7
  408. package/dist/lib/node-esm/chunk-L2NPJPGL.mjs +0 -281
  409. package/dist/lib/node-esm/chunk-L2NPJPGL.mjs.map +0 -7
  410. package/dist/lib/node-esm/chunk-O4IOJICP.mjs +0 -201
  411. package/dist/lib/node-esm/chunk-O4IOJICP.mjs.map +0 -7
  412. package/dist/lib/node-esm/chunk-Z43MBISY.mjs +0 -1391
  413. package/dist/lib/node-esm/chunk-Z43MBISY.mjs.map +0 -7
  414. package/dist/lib/node-esm/operation-resolver-5BYDNQND.mjs +0 -581
  415. package/dist/lib/node-esm/operation-resolver-5BYDNQND.mjs.map +0 -7
  416. package/dist/lib/node-esm/react-root-YO7IIC75.mjs +0 -50
  417. package/dist/lib/node-esm/react-root-YO7IIC75.mjs.map +0 -7
  418. package/dist/lib/node-esm/react-surface-G7VRKT4U.mjs +0 -45
  419. package/dist/lib/node-esm/react-surface-G7VRKT4U.mjs.map +0 -7
  420. package/dist/lib/node-esm/settings-GYLWWOBK.mjs +0 -41
  421. package/dist/lib/node-esm/settings-GYLWWOBK.mjs.map +0 -7
  422. package/dist/lib/node-esm/state-RQXTBWUX.mjs +0 -107
  423. package/dist/lib/node-esm/state-RQXTBWUX.mjs.map +0 -7
  424. package/dist/lib/node-esm/toolkit-EU3Z2R4H.mjs +0 -56
  425. package/dist/lib/node-esm/toolkit-EU3Z2R4H.mjs.map +0 -7
  426. package/dist/lib/node-esm/url-handler-VUK2LBJV.mjs +0 -96
  427. package/dist/lib/node-esm/url-handler-VUK2LBJV.mjs.map +0 -7
  428. package/dist/types/src/capabilities/app-graph-builder/app-graph-builder.d.ts.map +0 -1
  429. package/dist/types/src/capabilities/app-graph-builder/index.d.ts +0 -3
  430. package/dist/types/src/capabilities/app-graph-builder/index.d.ts.map +0 -1
  431. package/dist/types/src/capabilities/check-app-scheme/check-app-scheme.d.ts +0 -5
  432. package/dist/types/src/capabilities/check-app-scheme/check-app-scheme.d.ts.map +0 -1
  433. package/dist/types/src/capabilities/check-app-scheme/index.d.ts +0 -3
  434. package/dist/types/src/capabilities/check-app-scheme/index.d.ts.map +0 -1
  435. package/dist/types/src/capabilities/operation-resolver/index.d.ts +0 -3
  436. package/dist/types/src/capabilities/operation-resolver/index.d.ts.map +0 -1
  437. package/dist/types/src/capabilities/operation-resolver/operation-resolver.d.ts +0 -5
  438. package/dist/types/src/capabilities/operation-resolver/operation-resolver.d.ts.map +0 -1
  439. package/dist/types/src/capabilities/react-root/index.d.ts +0 -6
  440. package/dist/types/src/capabilities/react-root/index.d.ts.map +0 -1
  441. package/dist/types/src/capabilities/react-root/react-root.d.ts.map +0 -1
  442. package/dist/types/src/capabilities/react-surface/index.d.ts +0 -3
  443. package/dist/types/src/capabilities/react-surface/index.d.ts.map +0 -1
  444. package/dist/types/src/capabilities/react-surface/react-surface.d.ts.map +0 -1
  445. package/dist/types/src/capabilities/settings/index.d.ts +0 -19
  446. package/dist/types/src/capabilities/settings/index.d.ts.map +0 -1
  447. package/dist/types/src/capabilities/settings/settings.d.ts +0 -22
  448. package/dist/types/src/capabilities/settings/settings.d.ts.map +0 -1
  449. package/dist/types/src/capabilities/state/index.d.ts +0 -172
  450. package/dist/types/src/capabilities/state/index.d.ts.map +0 -1
  451. package/dist/types/src/capabilities/state/state.d.ts.map +0 -1
  452. package/dist/types/src/capabilities/toolkit/index.d.ts +0 -3
  453. package/dist/types/src/capabilities/toolkit/index.d.ts.map +0 -1
  454. package/dist/types/src/capabilities/toolkit/toolkit.d.ts +0 -27
  455. package/dist/types/src/capabilities/toolkit/toolkit.d.ts.map +0 -1
  456. package/dist/types/src/capabilities/tools/index.d.ts +0 -3
  457. package/dist/types/src/capabilities/tools/index.d.ts.map +0 -1
  458. package/dist/types/src/capabilities/tools/tools.d.ts.map +0 -1
  459. package/dist/types/src/capabilities/url-handler/index.d.ts +0 -3
  460. package/dist/types/src/capabilities/url-handler/index.d.ts.map +0 -1
  461. package/dist/types/src/capabilities/url-handler/url-handler.d.ts.map +0 -1
  462. package/dist/types/src/components/DeckLayout/Banner.d.ts.map +0 -1
  463. package/dist/types/src/components/DeckLayout/ContentEmpty.d.ts +0 -3
  464. package/dist/types/src/components/DeckLayout/ContentEmpty.d.ts.map +0 -1
  465. package/dist/types/src/components/DeckLayout/DeckLayout.d.ts.map +0 -1
  466. package/dist/types/src/components/DeckLayout/DeckLayout.stories.d.ts +0 -75
  467. package/dist/types/src/components/DeckLayout/DeckLayout.stories.d.ts.map +0 -1
  468. package/dist/types/src/components/DeckLayout/DeckMain.d.ts +0 -3
  469. package/dist/types/src/components/DeckLayout/DeckMain.d.ts.map +0 -1
  470. package/dist/types/src/components/DeckLayout/Dialog.d.ts.map +0 -1
  471. package/dist/types/src/components/DeckLayout/Fallback.d.ts +0 -3
  472. package/dist/types/src/components/DeckLayout/Fallback.d.ts.map +0 -1
  473. package/dist/types/src/components/DeckLayout/Popover.d.ts +0 -5
  474. package/dist/types/src/components/DeckLayout/Popover.d.ts.map +0 -1
  475. package/dist/types/src/components/DeckLayout/StatusBar.d.ts.map +0 -1
  476. package/dist/types/src/components/DeckLayout/Toast.d.ts.map +0 -1
  477. package/dist/types/src/components/DeckLayout/Topbar.d.ts +0 -3
  478. package/dist/types/src/components/DeckLayout/Topbar.d.ts.map +0 -1
  479. package/dist/types/src/components/DeckLayout/constants.d.ts.map +0 -1
  480. package/dist/types/src/components/DeckLayout/index.d.ts +0 -4
  481. package/dist/types/src/components/DeckLayout/index.d.ts.map +0 -1
  482. package/dist/types/src/components/Plank/Plank.d.ts +0 -27
  483. package/dist/types/src/components/Plank/Plank.d.ts.map +0 -1
  484. package/dist/types/src/components/Plank/Plank.stories.d.ts +0 -108
  485. package/dist/types/src/components/Plank/Plank.stories.d.ts.map +0 -1
  486. package/dist/types/src/components/Plank/PlankControls.d.ts.map +0 -1
  487. package/dist/types/src/components/Plank/PlankError.d.ts.map +0 -1
  488. package/dist/types/src/components/Plank/PlankHeading.d.ts.map +0 -1
  489. package/dist/types/src/components/Plank/index.d.ts +0 -6
  490. package/dist/types/src/components/Plank/index.d.ts.map +0 -1
  491. package/dist/types/src/components/Sidebar/ComplementarySidebar.d.ts.map +0 -1
  492. package/dist/types/src/components/Sidebar/Sidebar.d.ts.map +0 -1
  493. package/dist/types/src/components/Sidebar/SidebarButton.d.ts.map +0 -1
  494. package/dist/types/src/components/fragments.d.ts +0 -7
  495. package/dist/types/src/components/fragments.d.ts.map +0 -1
  496. package/dist/types/src/containers/Banner/Banner.d.ts +0 -2
  497. package/dist/types/src/containers/Banner/Banner.d.ts.map +0 -1
  498. package/dist/types/src/containers/Banner/index.d.ts +0 -3
  499. package/dist/types/src/containers/Banner/index.d.ts.map +0 -1
  500. package/dist/types/src/containers/DeckSettings/DeckSettings.d.ts +0 -8
  501. package/dist/types/src/containers/DeckSettings/DeckSettings.d.ts.map +0 -1
  502. package/dist/types/src/containers/DeckSettings/index.d.ts +0 -3
  503. package/dist/types/src/containers/DeckSettings/index.d.ts.map +0 -1
  504. package/dist/types/src/util/overscroll.d.ts +0 -47
  505. package/dist/types/src/util/overscroll.d.ts.map +0 -1
  506. package/src/capabilities/app-graph-builder/app-graph-builder.ts +0 -119
  507. package/src/capabilities/app-graph-builder/index.ts +0 -7
  508. package/src/capabilities/check-app-scheme/check-app-scheme.ts +0 -45
  509. package/src/capabilities/check-app-scheme/index.ts +0 -7
  510. package/src/capabilities/operation-resolver/index.ts +0 -10
  511. package/src/capabilities/operation-resolver/operation-resolver.ts +0 -560
  512. package/src/capabilities/react-root/index.ts +0 -7
  513. package/src/capabilities/react-surface/index.ts +0 -7
  514. package/src/capabilities/settings/index.ts +0 -7
  515. package/src/capabilities/state/index.ts +0 -7
  516. package/src/capabilities/toolkit/index.ts +0 -7
  517. package/src/capabilities/toolkit/toolkit.ts +0 -64
  518. package/src/capabilities/tools/index.ts +0 -7
  519. package/src/capabilities/url-handler/index.ts +0 -7
  520. package/src/capabilities/url-handler/url-handler.ts +0 -96
  521. package/src/components/DeckLayout/ContentEmpty.tsx +0 -30
  522. package/src/components/DeckLayout/DeckLayout.stories.tsx +0 -53
  523. package/src/components/DeckLayout/DeckLayout.tsx +0 -34
  524. package/src/components/DeckLayout/DeckMain.tsx +0 -287
  525. package/src/components/DeckLayout/Topbar.tsx +0 -11
  526. package/src/components/Plank/Plank.stories.tsx +0 -64
  527. package/src/components/Plank/Plank.tsx +0 -302
  528. package/src/components/fragments.ts +0 -19
  529. package/src/containers/Banner/Banner.tsx +0 -5
  530. package/src/containers/Banner/index.ts +0 -7
  531. package/src/containers/DeckSettings/DeckSettings.tsx +0 -113
  532. package/src/containers/DeckSettings/index.ts +0 -7
  533. package/src/util/overscroll.ts +0 -69
  534. /package/dist/types/src/capabilities/{app-graph-builder/app-graph-builder.d.ts → app-graph-builder.d.ts} +0 -0
  535. /package/dist/types/src/capabilities/{react-root/react-root.d.ts → react-root.d.ts} +0 -0
  536. /package/dist/types/src/capabilities/{react-surface/react-surface.d.ts → react-surface.d.ts} +0 -0
  537. /package/dist/types/src/{components/DeckLayout → containers/Deck}/StatusBar.d.ts +0 -0
  538. /package/dist/types/src/{components → containers}/DeckLayout/ActiveNode.d.ts +0 -0
  539. /package/dist/types/src/{components → containers}/DeckLayout/DeckLayout.d.ts +0 -0
  540. /package/dist/types/src/{components → containers}/DeckLayout/Dialog.d.ts +0 -0
  541. /package/dist/types/src/{components → containers}/DeckLayout/constants.d.ts +0 -0
  542. /package/dist/types/src/{components → containers}/Plank/PlankLoading.d.ts +0 -0
  543. /package/dist/types/src/{components → containers}/Sidebar/ComplementarySidebar.d.ts +0 -0
  544. /package/dist/types/src/{components → containers}/Sidebar/Sidebar.d.ts +0 -0
  545. /package/dist/types/src/{components → containers}/Sidebar/index.d.ts +0 -0
  546. /package/src/{components/DeckLayout → containers/Deck}/StatusBar.tsx +0 -0
  547. /package/src/{components → containers}/Plank/PlankLoading.tsx +0 -0
  548. /package/src/{components → containers}/Sidebar/index.ts +0 -0
@@ -10,18 +10,13 @@ import { AppCapabilities } from '@dxos/app-toolkit';
10
10
  import { createKvsStore } from '@dxos/effect';
11
11
  import { invariant } from '@dxos/invariant';
12
12
 
13
- import { meta } from '../../meta';
14
- import {
15
- DeckCapabilities,
16
- type DeckEphemeralStateProps,
17
- type DeckStateProps,
18
- DeckStateSchema,
19
- defaultDeck,
20
- getMode,
21
- } from '../../types';
13
+ import { meta } from '#meta';
14
+ import { DeckCapabilities, type EphemeralDeckState, StoredDeckState, defaultDeck, getMode } from '#types';
15
+
16
+ import { sanitizePersistedState } from '../util';
22
17
 
23
18
  /** Default persisted state. */
24
- const defaultDeckState: DeckStateProps = {
19
+ const defaultDeckState: StoredDeckState = {
25
20
  sidebarState: 'expanded',
26
21
  complementarySidebarState: 'collapsed',
27
22
  complementarySidebarPanel: undefined,
@@ -34,7 +29,7 @@ const defaultDeckState: DeckStateProps = {
34
29
  };
35
30
 
36
31
  /** Default ephemeral state. */
37
- const defaultDeckEphemeralState: DeckEphemeralStateProps = {
32
+ const defaultDeckEphemeralState: EphemeralDeckState = {
38
33
  dialogContent: null,
39
34
  dialogOpen: false,
40
35
  dialogBlockAlign: undefined,
@@ -54,28 +49,19 @@ export default Capability.makeModule(
54
49
 
55
50
  // Persisted state using KVS store.
56
51
  const stateAtom = createKvsStore({
57
- key: `${meta.id}/state`,
58
- schema: DeckStateSchema,
52
+ key: `${meta.id}.state`,
53
+ schema: StoredDeckState,
59
54
  defaultValue: () => ({ ...defaultDeckState }),
60
55
  });
61
56
 
62
57
  // Ephemeral state (not persisted, but kept alive to prevent GC resets).
63
- const ephemeralAtom = Atom.make<DeckEphemeralStateProps>({ ...defaultDeckEphemeralState }).pipe(Atom.keepAlive);
58
+ const ephemeralAtom = Atom.make<EphemeralDeckState>({ ...defaultDeckEphemeralState }).pipe(Atom.keepAlive);
64
59
 
65
- // Don't allow fullscreen mode to be persisted to prevent getting stuck in it.
60
+ // Sanitize persisted state on startup (see sanitizePersistedState for details).
66
61
  const currentState = registry.get(stateAtom);
67
- const currentDeck = currentState.decks[currentState.activeDeck];
68
- if (currentDeck?.fullscreen) {
69
- registry.set(stateAtom, {
70
- ...currentState,
71
- decks: {
72
- ...currentState.decks,
73
- [currentState.activeDeck]: {
74
- ...currentDeck,
75
- fullscreen: false,
76
- },
77
- },
78
- });
62
+ const sanitizedState = sanitizePersistedState(currentState);
63
+ if (sanitizedState !== currentState) {
64
+ registry.set(stateAtom, sanitizedState);
79
65
  }
80
66
 
81
67
  // Create derived layout atom (read-only) from both state atoms.
@@ -16,18 +16,18 @@ import {
16
16
  createIntent,
17
17
  } from '@dxos/app-framework';
18
18
  import { type OperationInvoker } from '@dxos/app-framework/plugin-operation/invoker';
19
+ import { LayoutOperation } from '@dxos/app-toolkit';
19
20
  import { invariant } from '@dxos/invariant';
20
21
  import { trim } from '@dxos/util';
21
22
 
22
- import { meta } from '../../meta';
23
- import { DeckOperation } from '../../types';
23
+ import { meta } from '#meta';
24
24
 
25
25
  // TODO(burdon): Factor out.
26
26
  declare global {
27
27
  interface ToolContextExtensions {
28
28
  dispatch?: PromiseIntentDispatcher;
29
29
  pivotId?: string;
30
- part?: 'deck' | 'dialog';
30
+ part?: 'multi' | 'dialog';
31
31
  }
32
32
  }
33
33
 
@@ -58,9 +58,9 @@ export default Capability.makeModule(() =>
58
58
  invariant(pivotId, 'No pivot ID');
59
59
  invariant(invokePromise, 'No operation invoker');
60
60
 
61
- if (part === 'deck') {
62
- const { error } = await invokePromise(DeckOperation.ChangeCompanion, {
63
- companion: id,
61
+ if (part === 'multi') {
62
+ const { error } = await invokePromise(LayoutOperation.UpdateCompanion, {
63
+ subject: id,
64
64
  });
65
65
  if (error) {
66
66
  return ToolResult.Error(error.message);
@@ -0,0 +1,231 @@
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 {
9
+ AppCapabilities,
10
+ LayoutOperation,
11
+ NOT_FOUND_PATH,
12
+ expandPath,
13
+ fromUrlPath,
14
+ getWorkspaceFromPath,
15
+ toUrlPath,
16
+ } from '@dxos/app-toolkit';
17
+ import { Operation } from '@dxos/compute';
18
+ import { runAndForwardErrors } from '@dxos/effect';
19
+ import { invariant } from '@dxos/invariant';
20
+ import { log } from '@dxos/log';
21
+ import { Node } from '@dxos/plugin-graph';
22
+ import { isTauri } from '@dxos/util';
23
+
24
+ import { DeckCapabilities, type StoredDeckState, defaultDeck } from '#types';
25
+
26
+ import { updateActiveDeck } from '../operations/helpers';
27
+ import { deserializePlanks, serializePlanks, stripPlanks } from '../util';
28
+ import { shouldDeferNavigationHandlers } from './check-app-scheme';
29
+
30
+ /** Dispatch all NavigationHandler contributions with a given URL. */
31
+ const dispatchNavigationHandlers = Effect.fn(function* (url: URL) {
32
+ const handlers = yield* Capability.getAll(AppCapabilities.NavigationHandler);
33
+ yield* Effect.all(
34
+ handlers.map((handler) => handler(url)),
35
+ { concurrency: 'unbounded' },
36
+ );
37
+ });
38
+
39
+ export default Capability.makeModule(
40
+ Effect.fnUntraced(function* () {
41
+ const operationService = yield* Capability.get(Capabilities.OperationInvoker);
42
+ const capabilities = yield* Capability.Service;
43
+ const registry = yield* Capability.get(Capabilities.AtomRegistry);
44
+ const stateAtom = yield* Capability.get(DeckCapabilities.State);
45
+ const settingsAtom = yield* Capability.get(DeckCapabilities.Settings);
46
+
47
+ const provideServices = <A, E>(effect: Effect.Effect<A, E, Capability.Service | Operation.Service>) =>
48
+ effect.pipe(
49
+ Effect.provideService(Capability.Service, capabilities),
50
+ Effect.provideService(Operation.Service, operationService),
51
+ );
52
+
53
+ // Helper to get state.
54
+ const getState = () => registry.get(stateAtom);
55
+
56
+ // Helper to get computed deck from state.
57
+ const getDeck = () => {
58
+ const state = getState();
59
+ const deck = state.decks[state.activeDeck];
60
+ invariant(deck, `Deck not found: ${state.activeDeck}`);
61
+ return deck;
62
+ };
63
+
64
+ // Helper to update state.
65
+ const updateState = (fn: (current: StoredDeckState) => StoredDeckState) => {
66
+ registry.set(stateAtom, fn(getState()));
67
+ };
68
+
69
+ const handleNavigation = Effect.fn(function* (url?: URL) {
70
+ const { graph } = yield* Capability.get(AppCapabilities.AppGraph);
71
+ const resolvedUrl = url ?? new URL(window.location.href);
72
+ // When native redirect is active, check-app-scheme owns the initial dispatch
73
+ // to prevent one-time tokens from being consumed before the native app can use them.
74
+ const settings = registry.get(settingsAtom);
75
+ const deferHandlers = settings?.enableNativeRedirect && shouldDeferNavigationHandlers();
76
+ if (!deferHandlers) {
77
+ yield* dispatchNavigationHandlers(resolvedUrl);
78
+ }
79
+
80
+ const pathname = resolvedUrl.pathname;
81
+ const state = getState();
82
+ if (pathname === '/reset') {
83
+ updateState((s) => ({
84
+ ...s,
85
+ activeDeck: 'default',
86
+ decks: {
87
+ default: { ...defaultDeck },
88
+ },
89
+ }));
90
+ window.location.pathname = '/';
91
+ return;
92
+ }
93
+
94
+ const qualifiedId = fromUrlPath(pathname);
95
+ const workspace = getWorkspaceFromPath(qualifiedId);
96
+ if (workspace !== Node.RootId && workspace !== state.activeDeck) {
97
+ yield* Operation.invoke(LayoutOperation.SwitchWorkspace, { subject: workspace });
98
+ }
99
+
100
+ const deck = getDeck();
101
+ const activeId = qualifiedId !== workspace ? qualifiedId : undefined;
102
+ if (activeId) {
103
+ // Ensure the object referenced by the URL is open in the deck.
104
+ // Open validates the target and may redirect to 404, returning the resolved IDs.
105
+ const resolvedIds = yield* Operation.invoke(LayoutOperation.Open, { subject: [activeId] });
106
+ // If not already in solo mode, switch to solo for the resolved target.
107
+ if (!deck.solo) {
108
+ yield* Operation.invoke(LayoutOperation.SetLayoutMode, {
109
+ subject: resolvedIds?.[0] ?? activeId,
110
+ mode: 'solo',
111
+ });
112
+ }
113
+ } else if (deck.solo && deck.solo !== NOT_FOUND_PATH) {
114
+ // Stay in solo mode; redirect URL to reflect the current solo item.
115
+ // Do not switch to deck mode here — only explicit user action should change layout mode.
116
+ const path = toUrlPath(deck.solo);
117
+ if (window.location.pathname !== path) {
118
+ history.replaceState(null, '', `${path}${stripPlanks(window.location.search)}`);
119
+ }
120
+ } else if (!activeId && !deck.solo) {
121
+ // Multi-mode: restore planks from query params.
122
+ const plankIds = deserializePlanks(resolvedUrl);
123
+ if (plankIds.length > 0) {
124
+ for (const plankId of plankIds) {
125
+ expandPath(graph, plankId);
126
+ }
127
+ updateState((state) => updateActiveDeck(state, { active: plankIds, initialized: true }));
128
+ }
129
+ }
130
+ });
131
+
132
+ const onPopState = () => void runAndForwardErrors(provideServices(handleNavigation()));
133
+
134
+ // Initial navigation.
135
+ yield* provideServices(handleNavigation());
136
+ window.addEventListener('popstate', onPopState);
137
+
138
+ // Tauri deep link support.
139
+ let unlistenDeepLink: (() => void) | undefined;
140
+ if (isTauri()) {
141
+ yield* Effect.gen(function* () {
142
+ const { getCurrent, onOpenUrl } = yield* Effect.promise(() => import('@tauri-apps/plugin-deep-link'));
143
+
144
+ const launchUrls = yield* Effect.promise(() => getCurrent());
145
+ if (launchUrls && launchUrls.length > 0) {
146
+ log('app launched with deep links', { urls: launchUrls });
147
+ for (const urlString of launchUrls) {
148
+ yield* provideServices(handleDeepLink(urlString, handleNavigation));
149
+ }
150
+ }
151
+
152
+ unlistenDeepLink = yield* Effect.promise(() =>
153
+ onOpenUrl((urls) => {
154
+ for (const urlString of urls) {
155
+ void runAndForwardErrors(provideServices(handleDeepLink(urlString, handleNavigation)));
156
+ }
157
+ }),
158
+ );
159
+ }).pipe(
160
+ Effect.catchAll((error) => Effect.sync(() => log.warn('failed to initialize deep link listener', { error }))),
161
+ );
162
+ }
163
+
164
+ // Sync URL with layout state changes.
165
+ let lastSolo: string | undefined;
166
+ let lastActiveDeck: string | undefined;
167
+ let lastActiveKey: string | undefined;
168
+ const unsubscribe = registry.subscribe(stateAtom, () => {
169
+ const state = getState();
170
+ const deck = getDeck();
171
+ const solo = deck.solo;
172
+ const activeDeck = state.activeDeck;
173
+ const activeKey = solo ? undefined : JSON.stringify(deck.active);
174
+
175
+ if (solo !== lastSolo || activeDeck !== lastActiveDeck || activeKey !== lastActiveKey) {
176
+ lastSolo = solo;
177
+ lastActiveDeck = activeDeck;
178
+ lastActiveKey = activeKey;
179
+
180
+ const path = solo && solo !== NOT_FOUND_PATH ? toUrlPath(solo) : toUrlPath(activeDeck);
181
+ const search = !solo
182
+ ? serializePlanks(deck.active, window.location.search)
183
+ : stripPlanks(window.location.search);
184
+ const newUrl = `${path}${search}`;
185
+
186
+ if (`${window.location.pathname}${window.location.search}` !== newUrl) {
187
+ history.pushState(null, '', newUrl);
188
+ }
189
+ }
190
+ });
191
+
192
+ return Capability.contributes(Capabilities.Null, null, () =>
193
+ Effect.sync(() => {
194
+ window.removeEventListener('popstate', onPopState);
195
+ unsubscribe();
196
+ unlistenDeepLink?.();
197
+ }),
198
+ );
199
+ }),
200
+ );
201
+
202
+ /** Check if a path is a redirect path handled elsewhere (e.g., OAuth). */
203
+ const isRedirectPath = (pathname: string): boolean => pathname.startsWith('/redirect/');
204
+
205
+ /** Handle a deep link URL string. Merges query params into window.location and navigates. */
206
+ const handleDeepLink = Effect.fn(function* (urlString: string, navigate: (url?: URL) => Effect.Effect<void, any, any>) {
207
+ log('deep link received', { url: urlString });
208
+
209
+ const deepLinkUrl = new URL(urlString);
210
+
211
+ // For custom schemes (e.g., composer://a/b/c), new URL() treats the first segment as the
212
+ // hostname. Reconstruct the full path from hostname + pathname.
213
+ const fullPath =
214
+ deepLinkUrl.protocol !== 'https:' && deepLinkUrl.protocol !== 'http:' && deepLinkUrl.hostname
215
+ ? '/' + deepLinkUrl.hostname + deepLinkUrl.pathname
216
+ : deepLinkUrl.pathname;
217
+
218
+ if (isRedirectPath(fullPath)) {
219
+ return;
220
+ }
221
+
222
+ // Merge deep link query params into the current window URL so handlers can read them.
223
+ const current = new URL(window.location.href);
224
+ if (deepLinkUrl.search) {
225
+ deepLinkUrl.searchParams.forEach((value, key) => current.searchParams.set(key, value));
226
+ }
227
+ current.pathname = fullPath;
228
+ history.replaceState(null, '', current.pathname + current.search);
229
+
230
+ yield* navigate(current);
231
+ });
@@ -0,0 +1,38 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
6
+
7
+ import { withLayout, withTheme } from '@dxos/react-ui/testing';
8
+
9
+ import { translations } from '#translations';
10
+
11
+ import { DeckSettings } from './DeckSettings';
12
+
13
+ const meta = {
14
+ title: 'plugins/plugin-deck/components/DeckSettings',
15
+ tags: ['settings'],
16
+ component: DeckSettings,
17
+ decorators: [withTheme(), withLayout({ layout: 'fullscreen' })],
18
+ parameters: {
19
+ layout: 'fullscreen',
20
+ translations,
21
+ },
22
+ } satisfies Meta<typeof DeckSettings>;
23
+
24
+ export default meta;
25
+
26
+ type Story = StoryObj<typeof meta>;
27
+
28
+ export const Default: Story = {
29
+ args: {
30
+ settings: {
31
+ enableDeck: true,
32
+ encapsulatedPlanks: false,
33
+ enableStatusbar: true,
34
+ showHints: true,
35
+ enableNativeRedirect: false,
36
+ },
37
+ },
38
+ };
@@ -0,0 +1,34 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import React from 'react';
6
+
7
+ import { type AppSurface } from '@dxos/app-toolkit/ui';
8
+ import { useTranslation } from '@dxos/react-ui';
9
+ import { Settings as SettingsForm } from '@dxos/react-ui-form';
10
+
11
+ import { meta } from '#meta';
12
+ import { Settings } from '#types';
13
+
14
+ const isSocket = !!(globalThis as any).__args;
15
+
16
+ export type DeckSettingsProps = AppSurface.SettingsArticleProps<Settings.Settings>;
17
+
18
+ export const DeckSettings = ({ settings, onSettingsChange }: DeckSettingsProps) => {
19
+ const { t } = useTranslation(meta.id);
20
+
21
+ return (
22
+ <SettingsForm.Viewport>
23
+ <SettingsForm.Section title={t('settings.title', { ns: meta.id })}>
24
+ <SettingsForm.FieldSet
25
+ readonly={!onSettingsChange}
26
+ schema={Settings.Settings}
27
+ visible={(path) => path !== 'enableNativeRedirect' || !isSocket}
28
+ values={settings}
29
+ onValuesChanged={(values) => onSettingsChange?.(() => values)}
30
+ />
31
+ </SettingsForm.Section>
32
+ </SettingsForm.Viewport>
33
+ );
34
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export { DeckSettings as default } from './DeckSettings';
@@ -0,0 +1,220 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
6
+ import React, { useCallback, useMemo, useRef, useState } from 'react';
7
+
8
+ import { Capabilities, Capability, Plugin } from '@dxos/app-framework';
9
+ import { withPluginManager } from '@dxos/app-framework/testing';
10
+ import { Surface } from '@dxos/app-framework/ui';
11
+ import { Graph } from '@dxos/app-graph';
12
+ import { AppActivationEvents, AppPlugin } from '@dxos/app-toolkit';
13
+ import { Obj } from '@dxos/echo';
14
+ import { corePlugins } from '@dxos/plugin-testing';
15
+ import { random } from '@dxos/random';
16
+ import { Focus, Panel, Toolbar } from '@dxos/react-ui';
17
+ import { useAttentionAttributes } from '@dxos/react-ui-attention';
18
+ import { withAttention } from '@dxos/react-ui-attention/testing';
19
+ import { type MosaicTileProps, Mosaic } from '@dxos/react-ui-mosaic';
20
+ import { StackContext } from '@dxos/react-ui-stack';
21
+ import { Syntax } from '@dxos/react-ui-syntax-highlighter';
22
+ import { Loading, withLayout, withTheme } from '@dxos/react-ui/testing';
23
+ import { Text } from '@dxos/schema';
24
+ import { Organization, Person } from '@dxos/types';
25
+
26
+ import { DeckState, OperationHandler } from '#capabilities';
27
+ import { meta as pluginMeta } from '#meta';
28
+ import { translations } from '#translations';
29
+
30
+ import { Plank } from '../../containers/Plank';
31
+ import { Matrix, type MatrixController, type MatrixRootProps } from './Matrix';
32
+
33
+ random.seed(123);
34
+
35
+ const TestPlugin = Plugin.define(pluginMeta).pipe(
36
+ Plugin.addModule({
37
+ id: Capability.getModuleTag(DeckState),
38
+ activatesOn: AppActivationEvents.AppGraphReady,
39
+ activate: () => DeckState(),
40
+ }),
41
+ AppPlugin.addOperationHandlerModule({
42
+ activate: OperationHandler,
43
+ }),
44
+ Plugin.make,
45
+ );
46
+
47
+ /**
48
+ * Simple tile with JSON display and attention tracking.
49
+ */
50
+ const StoryTile = (props: MosaicTileProps<Obj.Any>) => {
51
+ const attentionAttrs = useAttentionAttributes(props.data.id);
52
+ return (
53
+ <Mosaic.Tile {...props} asChild>
54
+ <Focus.Item asChild border current={props.current}>
55
+ <Panel.Root classNames='dx-current dx-hover w-full md:w-[50rem] snap-start shrink-0' {...attentionAttrs}>
56
+ <Panel.Toolbar asChild>
57
+ <Toolbar.Root>
58
+ <p>{Obj.getLabel(props.data)}</p>
59
+ </Toolbar.Root>
60
+ </Panel.Toolbar>
61
+ <Syntax.Root data={props.data}>
62
+ <Panel.Content asChild>
63
+ <Syntax.Content>
64
+ <Syntax.Viewport>
65
+ <Syntax.Code />
66
+ </Syntax.Viewport>
67
+ </Syntax.Content>
68
+ </Panel.Content>
69
+ </Syntax.Root>
70
+ </Panel.Root>
71
+ </Focus.Item>
72
+ </Mosaic.Tile>
73
+ );
74
+ };
75
+
76
+ /**
77
+ * Tile that wraps a Plank for content rendering.
78
+ */
79
+ const PlankTile = (props: MosaicTileProps<Obj.Any>) => {
80
+ const graph = useMemo(() => Graph.make(), []);
81
+ return (
82
+ <StackContext.Provider value={{ orientation: 'horizontal', size: 'contain', rail: true }}>
83
+ <Plank.Root layoutMode='multi' part='multi' graph={graph}>
84
+ <Mosaic.Tile {...props} asChild>
85
+ <Plank.Content solo={false} companion={false} encapsulate={false}>
86
+ <Plank.Component
87
+ id={props.data.id}
88
+ part='multi'
89
+ node={{
90
+ id: props.data.id,
91
+ data: props.data,
92
+ type: 'test',
93
+ properties: {},
94
+ }}
95
+ />
96
+ </Plank.Content>
97
+ </Mosaic.Tile>
98
+ </Plank.Root>
99
+ </StackContext.Provider>
100
+ );
101
+ };
102
+
103
+ const TestExtension = Capability.contributes(
104
+ Capabilities.ReactSurface,
105
+ Surface.create({
106
+ id: 'story-article',
107
+ role: 'article',
108
+ component: ({ data: { subject } }) => {
109
+ if (!subject) {
110
+ return <Loading />;
111
+ }
112
+
113
+ return (
114
+ <Syntax.Root data={subject}>
115
+ <Syntax.Content>
116
+ <Syntax.Viewport>
117
+ <Syntax.Code />
118
+ </Syntax.Viewport>
119
+ </Syntax.Content>
120
+ </Syntax.Root>
121
+ );
122
+ },
123
+ }),
124
+ );
125
+
126
+ type DefaultStoryProps = Pick<MatrixRootProps, 'Tile'>;
127
+
128
+ const DefaultStory = ({ Tile }: DefaultStoryProps) => {
129
+ const items = useMemo(
130
+ () => [
131
+ Organization.make({ name: random.company.name() }),
132
+ Person.make({ fullName: random.person.fullName() }),
133
+ Text.make({ name: 'Bio', content: random.lorem.paragraphs(10) }),
134
+ Text.make({ name: 'Companion', content: 'Companion panel for Bio' }),
135
+ ],
136
+ [],
137
+ );
138
+
139
+ const controller = useRef<MatrixController>(null);
140
+ const [current, setCurrent] = useState<string | undefined>(items[0]?.id);
141
+
142
+ const handleCurrentChange = useCallback((id: string | undefined) => {
143
+ setCurrent(id);
144
+ }, []);
145
+
146
+ const handlePrev = useCallback(() => {
147
+ const index = items.findIndex((item) => item.id === current);
148
+ const prev = items[Math.max(0, index - 1)];
149
+ if (prev) {
150
+ controller.current?.scrollTo(prev.id);
151
+ }
152
+ }, [items, current]);
153
+
154
+ const handleNext = useCallback(() => {
155
+ const index = items.findIndex((item) => item.id === current);
156
+ const next = items[Math.min(items.length - 1, index + 1)];
157
+ if (next) {
158
+ controller.current?.scrollTo(next.id);
159
+ }
160
+ }, [items, current]);
161
+
162
+ const currentIndex = items.findIndex((item) => item.id === current);
163
+
164
+ return (
165
+ <Mosaic.Root>
166
+ <Matrix.Root Tile={Tile} items={items} current={current} onCurrentChange={handleCurrentChange} ref={controller}>
167
+ <Panel.Root>
168
+ <Panel.Toolbar asChild>
169
+ <Toolbar.Root>
170
+ <Toolbar.IconButton icon='ph--caret-left--regular' iconOnly label='Back' onClick={handlePrev} />
171
+ <Toolbar.IconButton icon='ph--caret-right--regular' iconOnly label='Forward' onClick={handleNext} />
172
+ <Toolbar.Text>
173
+ {currentIndex + 1} / {items.length}
174
+ </Toolbar.Text>
175
+ </Toolbar.Root>
176
+ </Panel.Toolbar>
177
+ <Panel.Content asChild>
178
+ <Matrix.Content>
179
+ <Matrix.Viewport />
180
+ </Matrix.Content>
181
+ </Panel.Content>
182
+ </Panel.Root>
183
+ </Matrix.Root>
184
+ </Mosaic.Root>
185
+ );
186
+ };
187
+
188
+ const meta = {
189
+ title: 'plugins/plugin-deck/components/Matrix',
190
+ component: DefaultStory,
191
+ decorators: [withTheme(), withAttention(), withLayout({ layout: 'fullscreen' })],
192
+ parameters: {
193
+ layout: 'fullscreen',
194
+ },
195
+ } satisfies Meta<MatrixRootProps>;
196
+
197
+ export default meta;
198
+
199
+ type Story = StoryObj<typeof meta>;
200
+
201
+ export const Default: Story = {
202
+ args: {
203
+ Tile: StoryTile,
204
+ },
205
+ };
206
+
207
+ export const WithPlank: Story = {
208
+ decorators: [
209
+ withPluginManager({
210
+ plugins: [...corePlugins(), TestPlugin()],
211
+ capabilities: [TestExtension],
212
+ }),
213
+ ],
214
+ parameters: {
215
+ translations,
216
+ },
217
+ args: {
218
+ Tile: PlankTile,
219
+ },
220
+ };