@dxos/plugin-deck 0.8.4-main.fffef41 → 0.8.4-staging.60fe92afc8

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