@dxos/plugin-space 0.8.4-main.dedc0f3 → 0.8.4-main.e8ec1fe

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 (337) hide show
  1. package/dist/lib/browser/{CollectionMain-D2B75XBS.mjs → CollectionArticle-WTHWY4YS.mjs} +10 -10
  2. package/dist/lib/browser/CollectionArticle-WTHWY4YS.mjs.map +7 -0
  3. package/dist/lib/browser/ObjectDetailsPanel-2BRUBHP6.mjs +141 -0
  4. package/dist/lib/browser/ObjectDetailsPanel-2BRUBHP6.mjs.map +7 -0
  5. package/dist/lib/browser/ObjectSettings-5LLWCVEK.mjs +144 -0
  6. package/dist/lib/browser/ObjectSettings-5LLWCVEK.mjs.map +7 -0
  7. package/dist/lib/browser/RecordArticle-SXDRWTTU.mjs +116 -0
  8. package/dist/lib/browser/RecordArticle-SXDRWTTU.mjs.map +7 -0
  9. package/dist/lib/browser/{app-graph-builder-AFC6PNAB.mjs → app-graph-builder-DTM7BJ6D.mjs} +107 -96
  10. package/dist/lib/browser/app-graph-builder-DTM7BJ6D.mjs.map +7 -0
  11. package/dist/lib/browser/{app-graph-serializer-Q7QG4EKW.mjs → app-graph-serializer-TIKXF43P.mjs} +19 -19
  12. package/dist/lib/browser/app-graph-serializer-TIKXF43P.mjs.map +7 -0
  13. package/dist/lib/browser/chunk-AFUOMLX6.mjs +167 -0
  14. package/dist/lib/browser/chunk-AFUOMLX6.mjs.map +7 -0
  15. package/dist/lib/browser/{chunk-6GRF7NEF.mjs → chunk-CKACGS7T.mjs} +553 -475
  16. package/dist/lib/browser/chunk-CKACGS7T.mjs.map +7 -0
  17. package/dist/lib/browser/chunk-K5J7ZB5P.mjs +20 -0
  18. package/dist/lib/browser/chunk-K5J7ZB5P.mjs.map +7 -0
  19. package/dist/lib/browser/{chunk-S33AYKSS.mjs → chunk-KFUMADZF.mjs} +135 -62
  20. package/dist/lib/browser/chunk-KFUMADZF.mjs.map +7 -0
  21. package/dist/lib/browser/{chunk-WZYRDFW7.mjs → chunk-VGKOXAPE.mjs} +26 -13
  22. package/dist/lib/browser/chunk-VGKOXAPE.mjs.map +7 -0
  23. package/dist/lib/browser/chunk-VZBIIYFM.mjs +16 -0
  24. package/dist/lib/browser/chunk-VZBIIYFM.mjs.map +7 -0
  25. package/dist/lib/browser/chunk-WJXU4GKV.mjs +19 -0
  26. package/dist/lib/browser/chunk-WJXU4GKV.mjs.map +7 -0
  27. package/dist/lib/browser/{chunk-URST7EEN.mjs → chunk-ZQMSGD5J.mjs} +35 -19
  28. package/dist/lib/browser/chunk-ZQMSGD5J.mjs.map +7 -0
  29. package/dist/lib/browser/{identity-created-T6ZNVE7S.mjs → identity-created-NAXTPQXE.mjs} +5 -5
  30. package/dist/lib/browser/identity-created-NAXTPQXE.mjs.map +7 -0
  31. package/dist/lib/browser/index.mjs +88 -70
  32. package/dist/lib/browser/index.mjs.map +3 -3
  33. package/dist/lib/browser/{intent-resolver-CO262Q44.mjs → intent-resolver-3FNTO3VW.mjs} +89 -76
  34. package/dist/lib/browser/intent-resolver-3FNTO3VW.mjs.map +7 -0
  35. package/dist/lib/browser/meta.json +1 -1
  36. package/dist/lib/browser/react-root-Q7VBWBES.mjs +30 -0
  37. package/dist/lib/browser/react-root-Q7VBWBES.mjs.map +7 -0
  38. package/dist/lib/browser/{react-surface-TPUOK2LC.mjs → react-surface-TTHS332A.mjs} +71 -105
  39. package/dist/lib/browser/react-surface-TTHS332A.mjs.map +7 -0
  40. package/dist/lib/browser/{schema-defs-YDPFZELA.mjs → schema-defs-QPI2JU3X.mjs} +6 -6
  41. package/dist/lib/browser/schema-defs-QPI2JU3X.mjs.map +7 -0
  42. package/dist/lib/browser/{settings-ZYH2JWNI.mjs → settings-45PGPO2V.mjs} +5 -5
  43. package/dist/lib/browser/{settings-ZYH2JWNI.mjs.map → settings-45PGPO2V.mjs.map} +1 -1
  44. package/dist/lib/browser/{spaces-ready-3I5LISH4.mjs → spaces-ready-QCND4DVY.mjs} +15 -14
  45. package/dist/lib/browser/spaces-ready-QCND4DVY.mjs.map +7 -0
  46. package/dist/lib/browser/{state-2RGW7FQG.mjs → state-Q7YRE5KG.mjs} +7 -7
  47. package/dist/lib/browser/state-Q7YRE5KG.mjs.map +7 -0
  48. package/dist/lib/browser/types/index.mjs +2 -2
  49. package/dist/lib/node-esm/{CollectionMain-ZJIFCWKZ.mjs → CollectionArticle-KHXYT3SH.mjs} +10 -10
  50. package/dist/lib/node-esm/CollectionArticle-KHXYT3SH.mjs.map +7 -0
  51. package/dist/lib/node-esm/ObjectDetailsPanel-6PZQIQG3.mjs +142 -0
  52. package/dist/lib/node-esm/ObjectDetailsPanel-6PZQIQG3.mjs.map +7 -0
  53. package/dist/lib/node-esm/ObjectSettings-OQSBOH7K.mjs +145 -0
  54. package/dist/lib/node-esm/ObjectSettings-OQSBOH7K.mjs.map +7 -0
  55. package/dist/lib/node-esm/RecordArticle-NACBH42P.mjs +117 -0
  56. package/dist/lib/node-esm/RecordArticle-NACBH42P.mjs.map +7 -0
  57. package/dist/lib/node-esm/{app-graph-builder-OW2EBYRI.mjs → app-graph-builder-FBJFWI4H.mjs} +107 -96
  58. package/dist/lib/node-esm/app-graph-builder-FBJFWI4H.mjs.map +7 -0
  59. package/dist/lib/node-esm/{app-graph-serializer-6PGC5WR5.mjs → app-graph-serializer-BESQZAYU.mjs} +19 -19
  60. package/dist/lib/node-esm/app-graph-serializer-BESQZAYU.mjs.map +7 -0
  61. package/dist/lib/node-esm/{chunk-RXDT5LA5.mjs → chunk-6VEONPNZ.mjs} +35 -19
  62. package/dist/lib/node-esm/chunk-6VEONPNZ.mjs.map +7 -0
  63. package/dist/lib/node-esm/chunk-6ZQGZBEP.mjs +168 -0
  64. package/dist/lib/node-esm/chunk-6ZQGZBEP.mjs.map +7 -0
  65. package/dist/lib/node-esm/chunk-BDEFTL6K.mjs +18 -0
  66. package/dist/lib/node-esm/chunk-BDEFTL6K.mjs.map +7 -0
  67. package/dist/lib/node-esm/chunk-H4JILUJK.mjs +20 -0
  68. package/dist/lib/node-esm/chunk-H4JILUJK.mjs.map +7 -0
  69. package/dist/lib/node-esm/{chunk-TEUN2E4F.mjs → chunk-OK2L7N2F.mjs} +135 -62
  70. package/dist/lib/node-esm/chunk-OK2L7N2F.mjs.map +7 -0
  71. package/dist/lib/node-esm/{chunk-LGNPMOXU.mjs → chunk-QBRPYAEL.mjs} +26 -13
  72. package/dist/lib/node-esm/chunk-QBRPYAEL.mjs.map +7 -0
  73. package/dist/lib/node-esm/{chunk-NMCD2PSG.mjs → chunk-XGAMJC5C.mjs} +553 -475
  74. package/dist/lib/node-esm/chunk-XGAMJC5C.mjs.map +7 -0
  75. package/dist/lib/node-esm/chunk-Z7BB6HC2.mjs +21 -0
  76. package/dist/lib/node-esm/chunk-Z7BB6HC2.mjs.map +7 -0
  77. package/dist/lib/node-esm/{identity-created-AL7NNCKH.mjs → identity-created-OXLKCJE3.mjs} +5 -5
  78. package/dist/lib/node-esm/identity-created-OXLKCJE3.mjs.map +7 -0
  79. package/dist/lib/node-esm/index.mjs +88 -70
  80. package/dist/lib/node-esm/index.mjs.map +3 -3
  81. package/dist/lib/node-esm/{intent-resolver-EXPK4B76.mjs → intent-resolver-6O5FSB7Z.mjs} +89 -76
  82. package/dist/lib/node-esm/intent-resolver-6O5FSB7Z.mjs.map +7 -0
  83. package/dist/lib/node-esm/meta.json +1 -1
  84. package/dist/lib/node-esm/{react-root-4OQ7MPGW.mjs → react-root-K66W3FMA.mjs} +13 -12
  85. package/dist/lib/node-esm/react-root-K66W3FMA.mjs.map +7 -0
  86. package/dist/lib/node-esm/{react-surface-WXEXIDB7.mjs → react-surface-Y7FTEIDF.mjs} +71 -105
  87. package/dist/lib/node-esm/react-surface-Y7FTEIDF.mjs.map +7 -0
  88. package/dist/lib/node-esm/{schema-defs-OYBCWKAS.mjs → schema-defs-ZS2D47XW.mjs} +6 -6
  89. package/dist/lib/node-esm/schema-defs-ZS2D47XW.mjs.map +7 -0
  90. package/dist/lib/node-esm/{settings-2RYFMMTP.mjs → settings-6FO65BA6.mjs} +5 -5
  91. package/dist/lib/node-esm/{settings-2RYFMMTP.mjs.map → settings-6FO65BA6.mjs.map} +1 -1
  92. package/dist/lib/node-esm/{spaces-ready-JAP22P57.mjs → spaces-ready-P7CKVXBE.mjs} +15 -14
  93. package/dist/lib/node-esm/spaces-ready-P7CKVXBE.mjs.map +7 -0
  94. package/dist/lib/node-esm/{state-7VBVBGNS.mjs → state-362I5BMK.mjs} +7 -7
  95. package/dist/lib/node-esm/state-362I5BMK.mjs.map +7 -0
  96. package/dist/lib/node-esm/types/index.mjs +2 -2
  97. package/dist/types/src/SpacePlugin.d.ts +1 -1
  98. package/dist/types/src/SpacePlugin.d.ts.map +1 -1
  99. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -1
  100. package/dist/types/src/capabilities/capabilities.d.ts +6 -5
  101. package/dist/types/src/capabilities/capabilities.d.ts.map +1 -1
  102. package/dist/types/src/capabilities/intent-resolver.d.ts.map +1 -1
  103. package/dist/types/src/capabilities/react-root.d.ts.map +1 -1
  104. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  105. package/dist/types/src/capabilities/spaces-ready.d.ts.map +1 -1
  106. package/dist/types/src/components/AwaitingObject.d.ts.map +1 -1
  107. package/dist/types/src/components/CollectionArticle.d.ts +6 -0
  108. package/dist/types/src/components/CollectionArticle.d.ts.map +1 -0
  109. package/dist/types/src/components/CollectionSection.d.ts +3 -4
  110. package/dist/types/src/components/CollectionSection.d.ts.map +1 -1
  111. package/dist/types/src/components/CreateDialog/CreateObjectDialog.d.ts +1 -1
  112. package/dist/types/src/components/CreateDialog/CreateObjectDialog.d.ts.map +1 -1
  113. package/dist/types/src/components/CreateDialog/CreateObjectDialog.stories.d.ts +120 -1
  114. package/dist/types/src/components/CreateDialog/CreateObjectDialog.stories.d.ts.map +1 -1
  115. package/dist/types/src/components/CreateDialog/CreateObjectPanel.d.ts +3 -3
  116. package/dist/types/src/components/CreateDialog/CreateObjectPanel.d.ts.map +1 -1
  117. package/dist/types/src/components/CreateDialog/CreateSpaceDialog.d.ts +1 -1
  118. package/dist/types/src/components/CreateDialog/CreateSpaceDialog.d.ts.map +1 -1
  119. package/dist/types/src/components/{JoinDialog.d.ts → JoinDialog/JoinDialog.d.ts} +1 -1
  120. package/dist/types/src/components/JoinDialog/JoinDialog.d.ts.map +1 -0
  121. package/dist/types/src/components/JoinDialog/index.d.ts +2 -0
  122. package/dist/types/src/components/JoinDialog/index.d.ts.map +1 -0
  123. package/dist/types/src/components/MembersContainer/MembersContainer.d.ts.map +1 -0
  124. package/dist/types/src/components/MembersContainer/MembersContainer.stories.d.ts +1707 -0
  125. package/dist/types/src/components/{MembersContainer.stories.d.ts.map → MembersContainer/MembersContainer.stories.d.ts.map} +1 -1
  126. package/dist/types/src/components/MembersContainer/index.d.ts +2 -0
  127. package/dist/types/src/components/MembersContainer/index.d.ts.map +1 -0
  128. package/dist/types/src/components/ObjectDetailsPanel/ObjectDetailsPanel.d.ts +9 -0
  129. package/dist/types/src/components/ObjectDetailsPanel/ObjectDetailsPanel.d.ts.map +1 -0
  130. package/dist/types/src/components/ObjectDetailsPanel/ObjectForm.d.ts +10 -0
  131. package/dist/types/src/components/ObjectDetailsPanel/ObjectForm.d.ts.map +1 -0
  132. package/dist/types/src/components/ObjectDetailsPanel/index.d.ts +3 -0
  133. package/dist/types/src/components/ObjectDetailsPanel/index.d.ts.map +1 -0
  134. package/dist/types/src/components/ObjectRenamePopover/ObjectRenamePopover.d.ts +7 -0
  135. package/dist/types/src/components/ObjectRenamePopover/ObjectRenamePopover.d.ts.map +1 -0
  136. package/dist/types/src/components/ObjectRenamePopover/index.d.ts +2 -0
  137. package/dist/types/src/components/ObjectRenamePopover/index.d.ts.map +1 -0
  138. package/dist/types/src/components/ObjectSettings/BaseObjectSettings.d.ts +2 -2
  139. package/dist/types/src/components/ObjectSettings/BaseObjectSettings.d.ts.map +1 -1
  140. package/dist/types/src/components/ObjectSettings/BaseObjectSettings.stories.d.ts +1540 -0
  141. package/dist/types/src/components/ObjectSettings/BaseObjectSettings.stories.d.ts.map +1 -0
  142. package/dist/types/src/components/ObjectSettings/ForeignKeys.d.ts +1 -1
  143. package/dist/types/src/components/ObjectSettings/ForeignKeys.d.ts.map +1 -1
  144. package/dist/types/src/components/ObjectSettings/index.d.ts +2 -1
  145. package/dist/types/src/components/ObjectSettings/index.d.ts.map +1 -1
  146. package/dist/types/src/components/RecordArticle.d.ts +5 -0
  147. package/dist/types/src/components/RecordArticle.d.ts.map +1 -0
  148. package/dist/types/src/components/{MembersContainer.stories.d.ts → RecordArticle.stories.d.ts} +121 -2
  149. package/dist/types/src/components/RecordArticle.stories.d.ts.map +1 -0
  150. package/dist/types/src/components/SchemaContainer.d.ts +1 -1
  151. package/dist/types/src/components/SchemaContainer.d.ts.map +1 -1
  152. package/dist/types/src/components/SpacePluginSettings.d.ts.map +1 -1
  153. package/dist/types/src/components/{SpacePresence.d.ts → SpacePresence/SpacePresence.d.ts} +4 -3
  154. package/dist/types/src/components/SpacePresence/SpacePresence.d.ts.map +1 -0
  155. package/dist/types/src/components/{SpacePresence.stories.d.ts → SpacePresence/SpacePresence.stories.d.ts} +122 -3
  156. package/dist/types/src/components/{SpacePresence.stories.d.ts.map → SpacePresence/SpacePresence.stories.d.ts.map} +1 -1
  157. package/dist/types/src/components/SpacePresence/index.d.ts +2 -0
  158. package/dist/types/src/components/SpacePresence/index.d.ts.map +1 -0
  159. package/dist/types/src/components/SpaceRenamePopover/SpaceRenamePopover.d.ts +7 -0
  160. package/dist/types/src/components/SpaceRenamePopover/SpaceRenamePopover.d.ts.map +1 -0
  161. package/dist/types/src/components/SpaceRenamePopover/index.d.ts +2 -0
  162. package/dist/types/src/components/SpaceRenamePopover/index.d.ts.map +1 -0
  163. package/dist/types/src/components/SpaceSettings/SpaceSettingsContainer.d.ts.map +1 -1
  164. package/dist/types/src/components/SpaceSettings/SpaceSettingsContainer.stories.d.ts +120 -1
  165. package/dist/types/src/components/SpaceSettings/SpaceSettingsContainer.stories.d.ts.map +1 -1
  166. package/dist/types/src/components/SyncStatus/SyncStatus.d.ts.map +1 -1
  167. package/dist/types/src/components/SyncStatus/SyncStatus.stories.d.ts +121 -2
  168. package/dist/types/src/components/SyncStatus/SyncStatus.stories.d.ts.map +1 -1
  169. package/dist/types/src/components/ViewEditor.d.ts +3 -4
  170. package/dist/types/src/components/ViewEditor.d.ts.map +1 -1
  171. package/dist/types/src/components/index.d.ts +10 -17
  172. package/dist/types/src/components/index.d.ts.map +1 -1
  173. package/dist/types/src/events.d.ts.map +1 -1
  174. package/dist/types/src/helpers/index.d.ts +2 -0
  175. package/dist/types/src/helpers/index.d.ts.map +1 -0
  176. package/dist/types/src/helpers/query.d.ts +8 -0
  177. package/dist/types/src/helpers/query.d.ts.map +1 -0
  178. package/dist/types/src/helpers/query.test.d.ts +2 -0
  179. package/dist/types/src/helpers/query.test.d.ts.map +1 -0
  180. package/dist/types/src/hooks/index.d.ts +1 -0
  181. package/dist/types/src/hooks/index.d.ts.map +1 -1
  182. package/dist/types/src/hooks/useActiveSpace.d.ts.map +1 -1
  183. package/dist/types/src/hooks/useInputSurfaceLookup.d.ts +1 -1
  184. package/dist/types/src/hooks/useInputSurfaceLookup.d.ts.map +1 -1
  185. package/dist/types/src/hooks/usePath.d.ts +1 -1
  186. package/dist/types/src/hooks/usePath.d.ts.map +1 -1
  187. package/dist/types/src/hooks/useTypeOptions.d.ts +10 -0
  188. package/dist/types/src/hooks/useTypeOptions.d.ts.map +1 -0
  189. package/dist/types/src/index.d.ts +1 -0
  190. package/dist/types/src/index.d.ts.map +1 -1
  191. package/dist/types/src/meta.d.ts +0 -1
  192. package/dist/types/src/meta.d.ts.map +1 -1
  193. package/dist/types/src/translations.d.ts +120 -0
  194. package/dist/types/src/translations.d.ts.map +1 -1
  195. package/dist/types/src/types/types.d.ts +70 -55
  196. package/dist/types/src/types/types.d.ts.map +1 -1
  197. package/dist/types/src/util.d.ts +10 -7
  198. package/dist/types/src/util.d.ts.map +1 -1
  199. package/dist/types/tsconfig.tsbuildinfo +1 -1
  200. package/package.json +65 -60
  201. package/src/SpacePlugin.ts +228 -200
  202. package/src/capabilities/app-graph-builder.ts +157 -121
  203. package/src/capabilities/app-graph-serializer.ts +12 -12
  204. package/src/capabilities/capabilities.ts +16 -11
  205. package/src/capabilities/identity-created.ts +2 -2
  206. package/src/capabilities/intent-resolver.ts +75 -59
  207. package/src/capabilities/react-root.tsx +4 -3
  208. package/src/capabilities/react-surface.tsx +65 -141
  209. package/src/capabilities/schema-defs.ts +1 -1
  210. package/src/capabilities/spaces-ready.ts +8 -5
  211. package/src/capabilities/state.ts +2 -2
  212. package/src/components/AwaitingObject.tsx +12 -14
  213. package/src/components/{CollectionMain.tsx → CollectionArticle.tsx} +7 -6
  214. package/src/components/CollectionSection.tsx +8 -6
  215. package/src/components/CreateDialog/CreateObjectDialog.stories.tsx +11 -11
  216. package/src/components/CreateDialog/CreateObjectDialog.tsx +36 -28
  217. package/src/components/CreateDialog/CreateObjectPanel.tsx +8 -8
  218. package/src/components/CreateDialog/CreateSpaceDialog.tsx +30 -11
  219. package/src/components/{JoinDialog.tsx → JoinDialog/JoinDialog.tsx} +26 -10
  220. package/src/components/JoinDialog/index.ts +5 -0
  221. package/src/components/{MembersContainer.stories.tsx → MembersContainer/MembersContainer.stories.tsx} +7 -6
  222. package/src/components/{MembersContainer.tsx → MembersContainer/MembersContainer.tsx} +16 -14
  223. package/src/components/MembersContainer/index.ts +5 -0
  224. package/src/components/MenuFooter.tsx +2 -2
  225. package/src/components/ObjectDetailsPanel/ObjectDetailsPanel.tsx +51 -0
  226. package/src/components/ObjectDetailsPanel/ObjectForm.tsx +72 -0
  227. package/src/components/ObjectDetailsPanel/index.ts +7 -0
  228. package/src/components/{PopoverRenameObject.tsx → ObjectRenamePopover/ObjectRenamePopover.tsx} +6 -6
  229. package/src/components/ObjectRenamePopover/index.ts +5 -0
  230. package/src/components/ObjectSettings/AdvancedObjectSettings.tsx +3 -3
  231. package/src/components/ObjectSettings/BaseObjectSettings.stories.tsx +63 -0
  232. package/src/components/ObjectSettings/BaseObjectSettings.tsx +90 -26
  233. package/src/components/ObjectSettings/ForeignKeys.tsx +4 -4
  234. package/src/components/ObjectSettings/ObjectSettingsContainer.tsx +1 -1
  235. package/src/components/ObjectSettings/index.ts +3 -1
  236. package/src/components/RecordArticle.stories.tsx +115 -0
  237. package/src/components/RecordArticle.tsx +114 -0
  238. package/src/components/SchemaContainer.tsx +23 -26
  239. package/src/components/SpacePluginSettings.tsx +10 -4
  240. package/src/components/{SpacePresence.stories.tsx → SpacePresence/SpacePresence.stories.tsx} +23 -25
  241. package/src/components/{SpacePresence.tsx → SpacePresence/SpacePresence.tsx} +16 -11
  242. package/src/components/SpacePresence/index.ts +5 -0
  243. package/src/components/{PopoverRenameSpace.tsx → SpaceRenamePopover/SpaceRenamePopover.tsx} +7 -6
  244. package/src/components/SpaceRenamePopover/index.ts +5 -0
  245. package/src/components/SpaceSettings/SpaceSettingsContainer.stories.tsx +9 -4
  246. package/src/components/SpaceSettings/SpaceSettingsContainer.tsx +44 -23
  247. package/src/components/SyncStatus/InlineSyncStatus.tsx +8 -8
  248. package/src/components/SyncStatus/SyncStatus.stories.tsx +4 -5
  249. package/src/components/SyncStatus/SyncStatus.tsx +107 -11
  250. package/src/components/ViewEditor.tsx +51 -18
  251. package/src/components/index.ts +7 -8
  252. package/src/events.ts +6 -6
  253. package/src/helpers/index.ts +5 -0
  254. package/src/helpers/query.test.ts +24 -0
  255. package/src/helpers/query.ts +158 -0
  256. package/src/hooks/index.ts +1 -0
  257. package/src/hooks/useActiveSpace.ts +2 -1
  258. package/src/hooks/useInputSurfaceLookup.tsx +8 -3
  259. package/src/hooks/usePath.ts +1 -1
  260. package/src/hooks/useTypeOptions.ts +59 -0
  261. package/src/index.ts +1 -0
  262. package/src/meta.ts +6 -3
  263. package/src/translations.ts +25 -10
  264. package/src/types/types.ts +33 -18
  265. package/src/util.tsx +141 -63
  266. package/dist/lib/browser/CollectionMain-D2B75XBS.mjs.map +0 -7
  267. package/dist/lib/browser/ObjectDetailsPanel-J7KGPJJK.mjs +0 -90
  268. package/dist/lib/browser/ObjectDetailsPanel-J7KGPJJK.mjs.map +0 -7
  269. package/dist/lib/browser/RecordMain-TEBGAVSL.mjs +0 -68
  270. package/dist/lib/browser/RecordMain-TEBGAVSL.mjs.map +0 -7
  271. package/dist/lib/browser/app-graph-builder-AFC6PNAB.mjs.map +0 -7
  272. package/dist/lib/browser/app-graph-serializer-Q7QG4EKW.mjs.map +0 -7
  273. package/dist/lib/browser/chunk-6GRF7NEF.mjs.map +0 -7
  274. package/dist/lib/browser/chunk-CBYL62HG.mjs +0 -19
  275. package/dist/lib/browser/chunk-CBYL62HG.mjs.map +0 -7
  276. package/dist/lib/browser/chunk-ELJDGQTO.mjs +0 -94
  277. package/dist/lib/browser/chunk-ELJDGQTO.mjs.map +0 -7
  278. package/dist/lib/browser/chunk-FBCGT5YY.mjs +0 -13
  279. package/dist/lib/browser/chunk-FBCGT5YY.mjs.map +0 -7
  280. package/dist/lib/browser/chunk-S33AYKSS.mjs.map +0 -7
  281. package/dist/lib/browser/chunk-TUQZO5P4.mjs +0 -20
  282. package/dist/lib/browser/chunk-TUQZO5P4.mjs.map +0 -7
  283. package/dist/lib/browser/chunk-URST7EEN.mjs.map +0 -7
  284. package/dist/lib/browser/chunk-WZYRDFW7.mjs.map +0 -7
  285. package/dist/lib/browser/identity-created-T6ZNVE7S.mjs.map +0 -7
  286. package/dist/lib/browser/intent-resolver-CO262Q44.mjs.map +0 -7
  287. package/dist/lib/browser/react-root-K4IVLFYZ.mjs +0 -29
  288. package/dist/lib/browser/react-root-K4IVLFYZ.mjs.map +0 -7
  289. package/dist/lib/browser/react-surface-TPUOK2LC.mjs.map +0 -7
  290. package/dist/lib/browser/schema-defs-YDPFZELA.mjs.map +0 -7
  291. package/dist/lib/browser/spaces-ready-3I5LISH4.mjs.map +0 -7
  292. package/dist/lib/browser/state-2RGW7FQG.mjs.map +0 -7
  293. package/dist/lib/node-esm/CollectionMain-ZJIFCWKZ.mjs.map +0 -7
  294. package/dist/lib/node-esm/ObjectDetailsPanel-7ADOXZ5W.mjs +0 -91
  295. package/dist/lib/node-esm/ObjectDetailsPanel-7ADOXZ5W.mjs.map +0 -7
  296. package/dist/lib/node-esm/RecordMain-WLYJMYER.mjs +0 -70
  297. package/dist/lib/node-esm/RecordMain-WLYJMYER.mjs.map +0 -7
  298. package/dist/lib/node-esm/app-graph-builder-OW2EBYRI.mjs.map +0 -7
  299. package/dist/lib/node-esm/app-graph-serializer-6PGC5WR5.mjs.map +0 -7
  300. package/dist/lib/node-esm/chunk-FJZNSJC4.mjs +0 -20
  301. package/dist/lib/node-esm/chunk-FJZNSJC4.mjs.map +0 -7
  302. package/dist/lib/node-esm/chunk-HC677WUJ.mjs +0 -21
  303. package/dist/lib/node-esm/chunk-HC677WUJ.mjs.map +0 -7
  304. package/dist/lib/node-esm/chunk-HWNG4MEU.mjs +0 -15
  305. package/dist/lib/node-esm/chunk-HWNG4MEU.mjs.map +0 -7
  306. package/dist/lib/node-esm/chunk-LGNPMOXU.mjs.map +0 -7
  307. package/dist/lib/node-esm/chunk-NMCD2PSG.mjs.map +0 -7
  308. package/dist/lib/node-esm/chunk-Q6AAQLQG.mjs +0 -96
  309. package/dist/lib/node-esm/chunk-Q6AAQLQG.mjs.map +0 -7
  310. package/dist/lib/node-esm/chunk-RXDT5LA5.mjs.map +0 -7
  311. package/dist/lib/node-esm/chunk-TEUN2E4F.mjs.map +0 -7
  312. package/dist/lib/node-esm/identity-created-AL7NNCKH.mjs.map +0 -7
  313. package/dist/lib/node-esm/intent-resolver-EXPK4B76.mjs.map +0 -7
  314. package/dist/lib/node-esm/react-root-4OQ7MPGW.mjs.map +0 -7
  315. package/dist/lib/node-esm/react-surface-WXEXIDB7.mjs.map +0 -7
  316. package/dist/lib/node-esm/schema-defs-OYBCWKAS.mjs.map +0 -7
  317. package/dist/lib/node-esm/spaces-ready-JAP22P57.mjs.map +0 -7
  318. package/dist/lib/node-esm/state-7VBVBGNS.mjs.map +0 -7
  319. package/dist/types/src/components/CollectionMain.d.ts +0 -7
  320. package/dist/types/src/components/CollectionMain.d.ts.map +0 -1
  321. package/dist/types/src/components/JoinDialog.d.ts.map +0 -1
  322. package/dist/types/src/components/MembersContainer.d.ts.map +0 -1
  323. package/dist/types/src/components/ObjectDetailsPanel.d.ts +0 -9
  324. package/dist/types/src/components/ObjectDetailsPanel.d.ts.map +0 -1
  325. package/dist/types/src/components/PersistenceStatus.d.ts +0 -6
  326. package/dist/types/src/components/PersistenceStatus.d.ts.map +0 -1
  327. package/dist/types/src/components/PopoverRenameObject.d.ts +0 -7
  328. package/dist/types/src/components/PopoverRenameObject.d.ts.map +0 -1
  329. package/dist/types/src/components/PopoverRenameSpace.d.ts +0 -7
  330. package/dist/types/src/components/PopoverRenameSpace.d.ts.map +0 -1
  331. package/dist/types/src/components/RecordMain.d.ts +0 -7
  332. package/dist/types/src/components/RecordMain.d.ts.map +0 -1
  333. package/dist/types/src/components/SpacePresence.d.ts.map +0 -1
  334. package/src/components/ObjectDetailsPanel.tsx +0 -79
  335. package/src/components/PersistenceStatus.tsx +0 -83
  336. package/src/components/RecordMain.tsx +0 -43
  337. /package/dist/types/src/components/{MembersContainer.d.ts → MembersContainer/MembersContainer.d.ts} +0 -0
@@ -2,20 +2,18 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
- import { type Meta, type StoryObj } from '@storybook/react-vite';
5
+ import { type Meta } from '@storybook/react-vite';
8
6
  import React from 'react';
9
7
 
10
8
  import { IdentityDid, PublicKey } from '@dxos/keys';
11
9
  import { HaloSpaceMember, SpaceMember } from '@dxos/react-client/echo';
12
- import { withLayout, withTheme } from '@dxos/storybook-utils';
10
+ import { withTheme } from '@dxos/react-ui/testing';
13
11
 
14
- import { translations } from '../translations';
12
+ import { translations } from '../../translations';
15
13
 
16
14
  import { FullPresence, type Member, type MemberPresenceProps, SmallPresence } from './SpacePresence';
17
15
 
18
- const nViewers = (n: number, currentlyAttended = true): Member[] =>
16
+ const viewers = (n: number, currentlyAttended = true): Member[] =>
19
17
  Array.from({ length: n }, () => ({
20
18
  role: HaloSpaceMember.Role.ADMIN,
21
19
  identity: { did: IdentityDid.random(), identityKey: PublicKey.random() },
@@ -24,6 +22,16 @@ const nViewers = (n: number, currentlyAttended = true): Member[] =>
24
22
  currentlyAttended,
25
23
  }));
26
24
 
25
+ const meta = {
26
+ title: 'plugins/plugin-space/SpacePresence',
27
+ decorators: [withTheme],
28
+ parameters: {
29
+ translations,
30
+ },
31
+ } satisfies Meta<typeof IdentityDid>;
32
+
33
+ export default meta;
34
+
27
35
  export const Full = (props: MemberPresenceProps) => {
28
36
  const p: MemberPresenceProps = {
29
37
  ...props,
@@ -32,31 +40,31 @@ export const Full = (props: MemberPresenceProps) => {
32
40
  return (
33
41
  <div className='p-4'>
34
42
  <div className='p-3'>
35
- <FullPresence members={nViewers(1)} {...p} />
43
+ <FullPresence members={viewers(1)} {...p} />
36
44
  </div>
37
45
  <div className='p-3'>
38
- <FullPresence members={nViewers(2)} {...p} />
46
+ <FullPresence members={viewers(2)} {...p} />
39
47
  </div>
40
48
  <div className='p-3'>
41
- <FullPresence members={nViewers(3)} {...p} />
49
+ <FullPresence members={viewers(3)} {...p} />
42
50
  </div>
43
51
  <div className='p-3'>
44
- <FullPresence members={nViewers(3, false)} {...p} />
52
+ <FullPresence members={viewers(3, false)} {...p} />
45
53
  </div>
46
54
  <div className='p-3'>
47
- <FullPresence members={nViewers(4)} {...p} />
55
+ <FullPresence members={viewers(4)} {...p} />
48
56
  </div>
49
57
  <div className='p-3'>
50
- <FullPresence members={nViewers(5)} {...p} />
58
+ <FullPresence members={viewers(5)} {...p} />
51
59
  </div>
52
60
  <div className='p-3'>
53
- <FullPresence members={nViewers(5, false)} {...p} />
61
+ <FullPresence members={viewers(5, false)} {...p} />
54
62
  </div>
55
63
  <div className='p-3'>
56
- <FullPresence members={nViewers(10)} {...p} />
64
+ <FullPresence members={viewers(10)} {...p} />
57
65
  </div>
58
66
  <div className='p-3'>
59
- <FullPresence members={nViewers(100)} {...p} />
67
+ <FullPresence members={viewers(100)} {...p} />
60
68
  </div>
61
69
  </div>
62
70
  );
@@ -98,13 +106,3 @@ export const Small = () => {
98
106
  </div>
99
107
  );
100
108
  };
101
-
102
- const meta = {
103
- title: 'plugins/plugin-space/SpacePresence',
104
- decorators: [withTheme, withLayout()],
105
- parameters: { translations },
106
- } satisfies Meta<typeof IdentityDid>;
107
-
108
- export default meta;
109
-
110
- type Story = StoryObj<typeof meta>;
@@ -2,14 +2,14 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { Option } from 'effect';
5
+ import * as Option from 'effect/Option';
6
6
  import React, { forwardRef, useCallback, useEffect, useState } from 'react';
7
7
 
8
- import { useAppGraph, useCapability } from '@dxos/app-framework';
8
+ import { useAppGraph, useCapability } from '@dxos/app-framework/react';
9
9
  import { generateName } from '@dxos/display-name';
10
- import { type Type } from '@dxos/echo';
10
+ import { Obj, type Type } from '@dxos/echo';
11
11
  import { PublicKey, useClient } from '@dxos/react-client';
12
- import { type SpaceMember, fullyQualifiedId, getSpace, useMembers } from '@dxos/react-client/echo';
12
+ import { type SpaceMember, getSpace, useMembers } from '@dxos/react-client/echo';
13
13
  import { type Identity, useIdentity } from '@dxos/react-client/halo';
14
14
  import {
15
15
  Avatar,
@@ -27,10 +27,10 @@ import {
27
27
  import { AttentionGlyph, type AttentionGlyphProps, useAttended, useAttention } from '@dxos/react-ui-attention';
28
28
  import { ComplexMap, keyToFallback } from '@dxos/util';
29
29
 
30
- import { SpaceCapabilities } from '../capabilities';
31
- import { usePath } from '../hooks';
32
- import { SPACE_PLUGIN } from '../meta';
33
- import type { ObjectViewerProps } from '../types';
30
+ import { SpaceCapabilities } from '../../capabilities';
31
+ import { usePath } from '../../hooks';
32
+ import { meta } from '../../meta';
33
+ import { type ObjectViewerProps } from '../../types';
34
34
 
35
35
  // TODO(thure): Get/derive these values from protocol
36
36
  const REFRESH_INTERVAL = 5000;
@@ -42,7 +42,12 @@ const noViewers = new ComplexMap<PublicKey, ObjectViewerProps>(PublicKey.hash);
42
42
  // TODO(wittjosiah): Factor out?
43
43
  const getName = (identity: Identity) => identity.profile?.displayName ?? generateName(identity.identityKey.toHex());
44
44
 
45
- export const SpacePresence = ({ object, spaceKey }: { object: Type.Expando; spaceKey?: PublicKey }) => {
45
+ export type SpacePresenceProps = {
46
+ object: Type.Expando;
47
+ spaceKey?: PublicKey;
48
+ };
49
+
50
+ export const SpacePresence = ({ object, spaceKey }: SpacePresenceProps) => {
46
51
  // TODO(wittjosiah): Doesn't need to be mutable but readonly type messes with ComplexMap.
47
52
  const spaceState = useCapability(SpaceCapabilities.MutableState);
48
53
  const client = useClient();
@@ -70,7 +75,7 @@ export const SpacePresence = ({ object, spaceKey }: { object: Type.Expando; spac
70
75
  return null;
71
76
  }
72
77
 
73
- const currentObjectViewers = spaceState.viewersByObject[fullyQualifiedId(object)] ?? noViewers;
78
+ const currentObjectViewers = spaceState.viewersByObject[Obj.getDXN(object).toString()] ?? noViewers;
74
79
 
75
80
  const membersForObject = spaceMembers
76
81
  .filter((member) => memberOnline(member) && memberIsNotSelf(member))
@@ -254,7 +259,7 @@ export type SmallPresenceProps = {
254
259
  } & Pick<AttentionGlyphProps, 'attended' | 'containsAttended'>;
255
260
 
256
261
  export const SmallPresence = ({ count = 0, attended, containsAttended }: SmallPresenceProps) => {
257
- const { t } = useTranslation(SPACE_PLUGIN);
262
+ const { t } = useTranslation(meta.id);
258
263
 
259
264
  return (
260
265
  <Tooltip.Trigger asChild content={t('presence label', { count })} side='bottom'>
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './SpacePresence';
@@ -4,16 +4,17 @@
4
4
 
5
5
  import React, { useCallback, useRef, useState } from 'react';
6
6
 
7
- import { LayoutAction, createIntent, useIntentDispatcher } from '@dxos/app-framework';
8
- import { type Space } from '@dxos/react-client/echo';
7
+ import { LayoutAction, createIntent } from '@dxos/app-framework';
8
+ import { useIntentDispatcher } from '@dxos/app-framework/react';
9
+ import { type Space } from '@dxos/client/echo';
9
10
  import { Button, Input, Popover, useTranslation } from '@dxos/react-ui';
10
11
 
11
- import { SPACE_PLUGIN } from '../meta';
12
+ import { meta } from '../../meta';
12
13
 
13
- export const POPOVER_RENAME_SPACE = `${SPACE_PLUGIN}/PopoverRenameSpace`;
14
+ export const SPACE_RENAME_POPOVER = `${meta.id}/SpaceRenamePopover`;
14
15
 
15
- export const PopoverRenameSpace = ({ space }: { space: Space }) => {
16
- const { t } = useTranslation(SPACE_PLUGIN);
16
+ export const SpaceRenamePopover = ({ space }: { space: Space }) => {
17
+ const { t } = useTranslation(meta.id);
17
18
  const doneButton = useRef<HTMLButtonElement>(null);
18
19
  const [name, setName] = useState(space.properties.name ?? '');
19
20
  const { dispatchPromise: dispatch } = useIntentDispatcher();
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './SpaceRenamePopover';
@@ -2,13 +2,13 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
5
  import { type Meta, type StoryObj } from '@storybook/react-vite';
8
6
  import React from 'react';
9
7
 
8
+ import { IntentPlugin } from '@dxos/app-framework';
9
+ import { withPluginManager } from '@dxos/app-framework/testing';
10
10
  import { useClientProvider, withClientProvider } from '@dxos/react-client/testing';
11
- import { withTheme } from '@dxos/storybook-utils';
11
+ import { withTheme } from '@dxos/react-ui/testing';
12
12
 
13
13
  import { translations } from '../../translations';
14
14
 
@@ -27,7 +27,12 @@ const meta = {
27
27
  title: 'plugins/plugin-space/SpaceSettingsContainer',
28
28
  component: SpaceSettingsContainer,
29
29
  render: Story,
30
- decorators: [withClientProvider({ createIdentity: true, createSpace: true }), withTheme],
30
+ decorators: [
31
+ withTheme,
32
+ withClientProvider({ createIdentity: true, createSpace: true }),
33
+ // TODO(wittjosiah): Try to write story which does not depend on plugin manager.
34
+ withPluginManager({ plugins: [IntentPlugin()] }),
35
+ ],
31
36
  parameters: {
32
37
  translations,
33
38
  layout: 'centered',
@@ -2,15 +2,17 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { Schema, pipe } from 'effect';
5
+ import * as Function from 'effect/Function';
6
+ import * as Schema from 'effect/Schema';
6
7
  import React, { type ChangeEvent, useCallback, useMemo, useState } from 'react';
7
8
 
8
- import { LayoutAction, chain, createIntent, useIntentDispatcher } from '@dxos/app-framework';
9
+ import { LayoutAction, chain, createIntent } from '@dxos/app-framework';
10
+ import { useIntentDispatcher } from '@dxos/app-framework/react';
9
11
  import { log } from '@dxos/log';
10
12
  import { EdgeReplicationSetting } from '@dxos/protocols/proto/dxos/echo/metadata';
11
13
  import { useClient } from '@dxos/react-client';
12
14
  import { type Space, SpaceState } from '@dxos/react-client/echo';
13
- import { Button, Input, useMulticastObservable, useTranslation } from '@dxos/react-ui';
15
+ import { Button, Input, useFileDownload, useMulticastObservable, useTranslation } from '@dxos/react-ui';
14
16
  import {
15
17
  ControlItem,
16
18
  ControlItemInput,
@@ -22,11 +24,15 @@ import {
22
24
  import { HuePicker, IconPicker } from '@dxos/react-ui-pickers';
23
25
  import { StackItem } from '@dxos/react-ui-stack';
24
26
 
25
- import { SPACE_PLUGIN } from '../../meta';
27
+ import { meta } from '../../meta';
26
28
  import { SpaceAction, SpaceForm } from '../../types';
27
29
 
28
30
  const FormSchema = SpaceForm.pipe(
29
- Schema.extend(Schema.Struct({ archived: Schema.Boolean.annotations({ title: 'Archive space' }) })),
31
+ Schema.extend(
32
+ Schema.Struct({
33
+ archived: Schema.Boolean.annotations({ title: 'Archive Space' }),
34
+ }),
35
+ ),
30
36
  );
31
37
 
32
38
  export type SpaceSettingsContainerProps = {
@@ -35,7 +41,7 @@ export type SpaceSettingsContainerProps = {
35
41
 
36
42
  // TODO(wittjosiah): Handle space migrations here?
37
43
  export const SpaceSettingsContainer = ({ space }: SpaceSettingsContainerProps) => {
38
- const { t } = useTranslation(SPACE_PLUGIN);
44
+ const { t } = useTranslation(meta.id);
39
45
  const { dispatchPromise: dispatch } = useIntentDispatcher();
40
46
  const client = useClient();
41
47
  const archived = useMulticastObservable(space.state) === SpaceState.SPACE_INACTIVE;
@@ -69,9 +75,12 @@ export const SpaceSettingsContainer = ({ space }: SpaceSettingsContainerProps) =
69
75
  }
70
76
  if (properties.archived && !archived) {
71
77
  void dispatch(
72
- pipe(
78
+ Function.pipe(
73
79
  createIntent(SpaceAction.Close, { space }),
74
- chain(LayoutAction.SwitchWorkspace, { part: 'workspace', subject: client.spaces.default.id }),
80
+ chain(LayoutAction.SwitchWorkspace, {
81
+ part: 'workspace',
82
+ subject: client.spaces.default.id,
83
+ }),
75
84
  ),
76
85
  );
77
86
  } else if (!properties.archived && archived) {
@@ -111,31 +120,25 @@ export const SpaceSettingsContainer = ({ space }: SpaceSettingsContainerProps) =
111
120
  );
112
121
  },
113
122
  icon: ({ type, label, getValue, onValueChange }) => {
114
- const handleChange = useCallback((nextEmoji: string) => onValueChange(type, nextEmoji), [onValueChange, type]);
115
- const handleEmojiReset = useCallback(() => onValueChange(type, undefined), [onValueChange, type]);
123
+ const handleChange = useCallback((icon: string) => onValueChange(type, icon), [onValueChange, type]);
124
+ const handleReset = useCallback(() => onValueChange(type, undefined), [onValueChange, type]);
116
125
  return (
117
126
  <ControlItem title={label} description={t('icon description')}>
118
127
  <IconPicker
119
128
  value={getValue()}
120
129
  onChange={handleChange}
121
- onReset={handleEmojiReset}
130
+ onReset={handleReset}
122
131
  classNames='justify-self-end'
123
- iconSize={5}
124
132
  />
125
133
  </ControlItem>
126
134
  );
127
135
  },
128
136
  hue: ({ type, label, getValue, onValueChange }) => {
129
137
  const handleChange = useCallback((nextHue: string) => onValueChange(type, nextHue), [onValueChange, type]);
130
- const handleHueReset = useCallback(() => onValueChange(type, undefined), [onValueChange, type]);
138
+ const handleReset = useCallback(() => onValueChange(type, undefined), [onValueChange, type]);
131
139
  return (
132
140
  <ControlItem title={label} description={t('hue description')}>
133
- <HuePicker
134
- value={getValue()}
135
- onChange={handleChange}
136
- onReset={handleHueReset}
137
- classNames='[--hue-preview-size:1.25rem] justify-self-end'
138
- />
141
+ <HuePicker value={getValue()} onChange={handleChange} onReset={handleReset} classNames='justify-self-end' />
139
142
  </ControlItem>
140
143
  );
141
144
  },
@@ -151,7 +154,11 @@ export const SpaceSettingsContainer = ({ space }: SpaceSettingsContainerProps) =
151
154
  const handleChange = useCallback(() => onValueChange(type, !getValue()), [onValueChange, type, getValue]);
152
155
  return (
153
156
  <ControlItemInput title={label} description={t('archive space description')}>
154
- <Button disabled={space === client.spaces.default} onClick={handleChange}>
157
+ <Button
158
+ disabled={space === client.spaces.default}
159
+ variant={getValue() ? 'default' : 'destructive'}
160
+ onClick={handleChange}
161
+ >
155
162
  {getValue() ? t('unarchive space label') : t('archive space label')}
156
163
  </Button>
157
164
  </ControlItemInput>
@@ -161,12 +168,20 @@ export const SpaceSettingsContainer = ({ space }: SpaceSettingsContainerProps) =
161
168
  [t, space],
162
169
  );
163
170
 
171
+ const download = useFileDownload();
172
+ const handleBackup = useCallback(async () => {
173
+ const archive = await space.internal.export();
174
+ download(new Blob([archive.contents as Uint8Array<ArrayBuffer>]), archive.filename);
175
+ }, [space, download]);
176
+
164
177
  return (
165
- <StackItem.Content classNames='block overflow-y-auto'>
178
+ <StackItem.Content scrollable>
166
179
  <ControlPage>
167
180
  <ControlSection
168
- title={t('space properties settings verbose label', { ns: SPACE_PLUGIN })}
169
- description={t('space properties settings description', { ns: SPACE_PLUGIN })}
181
+ title={t('space properties settings verbose label', { ns: meta.id })}
182
+ description={t('space properties settings description', {
183
+ ns: meta.id,
184
+ })}
170
185
  >
171
186
  <Form
172
187
  schema={FormSchema}
@@ -178,6 +193,12 @@ export const SpaceSettingsContainer = ({ space }: SpaceSettingsContainerProps) =
178
193
  classNames='container-max-width grid grid-cols-1 md:grid-cols-[1fr_min-content]'
179
194
  />
180
195
  </ControlSection>
196
+ <ControlItemInput
197
+ title={t('backup space label', { ns: meta.id })}
198
+ description={t('backup space description', { ns: meta.id })}
199
+ >
200
+ <Button onClick={handleBackup}>{t('download backup')}</Button>
201
+ </ControlItemInput>
181
202
  </ControlPage>
182
203
  </StackItem.Content>
183
204
  );
@@ -2,10 +2,10 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { Option } from 'effect';
5
+ import * as Option from 'effect/Option';
6
6
  import React, { useEffect, useState } from 'react';
7
7
 
8
- import { useAppGraph } from '@dxos/app-framework';
8
+ import { useAppGraph } from '@dxos/app-framework/react';
9
9
  import { EdgeStatus } from '@dxos/protocols/proto/dxos/client/services';
10
10
  import { EdgeReplicationSetting } from '@dxos/protocols/proto/dxos/echo/metadata';
11
11
  import { useClient } from '@dxos/react-client';
@@ -14,14 +14,14 @@ import { Tooltip, useTranslation } from '@dxos/react-ui';
14
14
  import { AttentionGlyph, useAttended, useAttention } from '@dxos/react-ui-attention';
15
15
 
16
16
  import { usePath } from '../../hooks';
17
- import { SPACE_PLUGIN } from '../../meta';
17
+ import { meta } from '../../meta';
18
18
 
19
- const useEdgeStatus = (): EdgeStatus => {
20
- const [status, setStatus] = useState(EdgeStatus.NOT_CONNECTED);
19
+ const useEdgeStatus = (): EdgeStatus.ConnectionState => {
20
+ const [status, setStatus] = useState(EdgeStatus.ConnectionState.NOT_CONNECTED);
21
21
  const client = useClient();
22
22
  useEffect(() => {
23
23
  client.services.services.EdgeAgentService?.queryEdgeStatus().subscribe(({ status }) => {
24
- setStatus(status);
24
+ setStatus(status.state);
25
25
  });
26
26
  }, [client]);
27
27
 
@@ -29,7 +29,7 @@ const useEdgeStatus = (): EdgeStatus => {
29
29
  };
30
30
 
31
31
  export const InlineSyncStatus = ({ space, open }: { space: Space; open?: boolean }) => {
32
- const { t } = useTranslation(SPACE_PLUGIN);
32
+ const { t } = useTranslation(meta.id);
33
33
  const id = space.id;
34
34
  const { hasAttention, isAncestor, isRelated } = useAttention(id);
35
35
  const isAttended = hasAttention || isAncestor || isRelated;
@@ -43,7 +43,7 @@ export const InlineSyncStatus = ({ space, open }: { space: Space; open?: boolean
43
43
  const path = usePath(graph, startOfAttention);
44
44
  const containsAttended = !open && !isAttended && id && Option.isSome(path) ? path.value.includes(id) : false;
45
45
 
46
- const connectedToEdge = useEdgeStatus() === EdgeStatus.CONNECTED;
46
+ const connectedToEdge = useEdgeStatus() === EdgeStatus.ConnectionState.CONNECTED;
47
47
  // TODO(wittjosiah): This is not reactive.
48
48
  const edgeSyncEnabled = space.internal.data.edgeReplication === EdgeReplicationSetting.ENABLED;
49
49
  const syncState = useSpaceSyncState(space);
@@ -2,11 +2,10 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
5
  import { type Meta, type StoryObj } from '@storybook/react-vite';
8
6
 
9
- import { withTheme } from '@dxos/storybook-utils';
7
+ import { withClientProvider } from '@dxos/react-client/testing';
8
+ import { withTheme } from '@dxos/react-ui/testing';
10
9
 
11
10
  import { translations } from '../../translations';
12
11
 
@@ -15,10 +14,10 @@ import { SyncStatusIndicator } from './SyncStatus';
15
14
  const meta = {
16
15
  title: 'plugins/plugin-space/SyncStatusIndicator',
17
16
  component: SyncStatusIndicator,
18
- decorators: [withTheme],
17
+ decorators: [withTheme, withClientProvider({ createIdentity: true })],
19
18
  parameters: {
20
- translations,
21
19
  layout: 'centered',
20
+ translations,
22
21
  },
23
22
  } satisfies Meta<typeof SyncStatusIndicator>;
24
23
 
@@ -5,11 +5,16 @@
5
5
  import React, { useEffect, useState } from 'react';
6
6
 
7
7
  import { StatusBar } from '@dxos/plugin-status-bar';
8
+ import { EdgeStatus } from '@dxos/protocols/proto/dxos/client/services';
9
+ import { type QueryEdgeStatusResponse } from '@dxos/protocols/proto/dxos/client/services';
8
10
  import { useClient } from '@dxos/react-client';
11
+ import { useStream } from '@dxos/react-client/devtools';
9
12
  import { type SpaceSyncStateMap, getSyncSummary, useSyncState } from '@dxos/react-client/echo';
10
- import { Icon, useTranslation } from '@dxos/react-ui';
13
+ import { Icon, Popover, useTranslation } from '@dxos/react-ui';
14
+ import { mx } from '@dxos/react-ui-theme';
15
+ import { Unit, type UnitFormat } from '@dxos/util';
11
16
 
12
- import { SPACE_PLUGIN } from '../../meta';
17
+ import { meta } from '../../meta';
13
18
 
14
19
  import { createClientSaveTracker } from './save-tracker';
15
20
  import { getIcon, getStatus } from './status';
@@ -20,18 +25,13 @@ export const SyncStatus = () => {
20
25
  const client = useClient();
21
26
  const state = useSyncState();
22
27
  const [saved, setSaved] = useState(true);
23
-
24
- useEffect(() => {
25
- return createClientSaveTracker(client, (state) => {
26
- setSaved(state === 'saved');
27
- });
28
- }, []);
28
+ useEffect(() => createClientSaveTracker(client, (state) => setSaved(state === 'saved')), []);
29
29
 
30
30
  return <SyncStatusIndicator state={state} saved={saved} />;
31
31
  };
32
32
 
33
33
  export const SyncStatusIndicator = ({ state, saved }: { state: SpaceSyncStateMap; saved: boolean }) => {
34
- const { t } = useTranslation(SPACE_PLUGIN);
34
+ const { t } = useTranslation(meta.id);
35
35
  const summary = getSyncSummary(state);
36
36
  const offline = Object.values(state).length === 0;
37
37
  const needsToUpload = summary.differentDocuments > 0 || summary.missingOnRemote > 0;
@@ -53,7 +53,103 @@ export const SyncStatusIndicator = ({ state, saved }: { state: SpaceSyncStateMap
53
53
  }, [offline, needsToUpload, needsToDownload]);
54
54
 
55
55
  const title = t(`${status} label`);
56
- const icon = <Icon icon={getIcon(status)} size={4} classNames={classNames} />;
56
+ const icon = <Icon icon={getIcon(status)} classNames={classNames} />;
57
+
58
+ return (
59
+ <Popover.Root>
60
+ <Popover.Trigger asChild>
61
+ <StatusBar.Item title={title}>{icon}</StatusBar.Item>
62
+ </Popover.Trigger>
63
+ <Popover.Portal>
64
+ <Popover.Content>
65
+ <EdgeConnectionPopover />
66
+ <Popover.Arrow />
67
+ </Popover.Content>
68
+ </Popover.Portal>
69
+ </Popover.Root>
70
+ );
71
+ };
72
+
73
+ const useEdgeStatus = (): EdgeStatus | undefined => {
74
+ const client = useClient();
75
+ const { status } = useStream(
76
+ () => client.services.services.EdgeAgentService!.queryEdgeStatus(),
77
+ {} as QueryEdgeStatusResponse,
78
+ );
79
+ return status;
80
+ };
81
+
82
+ const EdgeConnectionPopover = () => {
83
+ const status = useEdgeStatus();
84
+ const { t } = useTranslation(meta.id);
85
+
86
+ const isConnected = status?.state === EdgeStatus.ConnectionState.CONNECTED;
87
+
88
+ return (
89
+ <div className='min-is-[240px] p-2'>
90
+ {/* Connection Status Header */}
91
+ <div className='flex items-center gap-2 mbe-2'>
92
+ <Icon
93
+ icon={isConnected ? 'ph--check-circle--regular' : 'ph--warning-circle--regular'}
94
+ classNames={mx(isConnected ? 'text-successText' : 'text-errorText animate-pulse')}
95
+ />
96
+ <span className='font-medium text-sm'>
97
+ {isConnected ? t('sync edge connected label') : t('sync edge disconnected label')}
98
+ </span>
99
+ </div>
100
+
101
+ {/* Connection Details */}
102
+ {status?.state === EdgeStatus.ConnectionState.NOT_CONNECTED && (
103
+ <div className='flex items-center gap-2 text-sm text-description'>
104
+ <Icon icon='ph--cloud-x--regular' />
105
+ <span>{t('sync no connection label')}</span>
106
+ </div>
107
+ )}
108
+
109
+ {status?.state === EdgeStatus.ConnectionState.CONNECTED && (
110
+ <div className='space-y-2'>
111
+ {/* Latency */}
112
+ <div className='flex items-center justify-between text-sm'>
113
+ <div className='flex items-center gap-2 text-description'>
114
+ <Icon icon='ph--timer--regular' />
115
+ <span>{t('sync latency label')}</span>
116
+ </div>
117
+ <UnitValue value={status.rtt} format={Unit.Millisecond} />
118
+ </div>
119
+
120
+ {/* Upload Speed */}
121
+ <div className='flex items-center justify-between text-sm'>
122
+ <div className='flex items-center gap-2 text-description'>
123
+ <Icon icon='ph--arrow-up--regular' />
124
+ <span>{t('sync upload label')}</span>
125
+ </div>
126
+ <UnitValue value={status.rateBytesUp} format={Unit.Kilobyte} suffix='/s' />
127
+ </div>
128
+
129
+ {/* Download Speed */}
130
+ <div className='flex items-center justify-between text-sm'>
131
+ <div className='flex items-center gap-2 text-sm text-description'>
132
+ <Icon icon='ph--arrow-down--regular' />
133
+ <span>{t('sync download label')}</span>
134
+ </div>
135
+ <UnitValue value={status.rateBytesDown} format={Unit.Kilobyte} suffix='/s' />
136
+ </div>
137
+ </div>
138
+ )}
139
+ </div>
140
+ );
141
+ };
57
142
 
58
- return <StatusBar.Item title={title}>{icon}</StatusBar.Item>;
143
+ // TODO(burdon): Factor out.
144
+ const UnitValue = ({ value: input, format, suffix }: { value: number; format: UnitFormat; suffix?: string }) => {
145
+ const { formattedValue, unit } = format(input);
146
+ return (
147
+ <span className='font-mono'>
148
+ {formattedValue}
149
+ <span className='mis-1 text-subdued'>
150
+ {unit.symbol}
151
+ {suffix}
152
+ </span>
153
+ </span>
154
+ );
59
155
  };