@dxos/plugin-kanban 0.8.4-main.ae835ea → 0.8.4-main.bc2380dfbc

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 (282) hide show
  1. package/LICENSE +102 -5
  2. package/README.md +1 -1
  3. package/dist/lib/neutral/KanbanArticle-NEJ3LNBO.mjs +132 -0
  4. package/dist/lib/neutral/KanbanArticle-NEJ3LNBO.mjs.map +7 -0
  5. package/dist/lib/neutral/KanbanPlugin.mjs +36 -0
  6. package/dist/lib/neutral/KanbanPlugin.mjs.map +7 -0
  7. package/dist/lib/neutral/KanbanPlugin.node.mjs +27 -0
  8. package/dist/lib/neutral/KanbanPlugin.node.mjs.map +7 -0
  9. package/dist/lib/neutral/KanbanPlugin.workerd.mjs +21 -0
  10. package/dist/lib/neutral/KanbanPlugin.workerd.mjs.map +7 -0
  11. package/dist/lib/neutral/KanbanSettings-G6M47NSK.mjs +83 -0
  12. package/dist/lib/neutral/KanbanSettings-G6M47NSK.mjs.map +7 -0
  13. package/dist/lib/neutral/blueprint-definition-FHVIEKTQ.mjs +15 -0
  14. package/dist/lib/neutral/blueprint-definition-FHVIEKTQ.mjs.map +7 -0
  15. package/dist/lib/neutral/blueprints/index.mjs +8 -0
  16. package/dist/lib/neutral/capabilities/index.mjs +19 -0
  17. package/dist/lib/neutral/capabilities/index.mjs.map +7 -0
  18. package/dist/lib/neutral/chunk-6FPBLOS3.mjs +8 -0
  19. package/dist/lib/neutral/chunk-6FPBLOS3.mjs.map +7 -0
  20. package/dist/lib/neutral/chunk-J5LGTIGS.mjs +10 -0
  21. package/dist/lib/neutral/chunk-KDPM77BD.mjs +21 -0
  22. package/dist/lib/neutral/chunk-KDPM77BD.mjs.map +7 -0
  23. package/dist/lib/neutral/chunk-OQ72EEGU.mjs +254 -0
  24. package/dist/lib/neutral/chunk-OQ72EEGU.mjs.map +7 -0
  25. package/dist/lib/neutral/chunk-ZTQW5KQS.mjs +26 -0
  26. package/dist/lib/neutral/chunk-ZTQW5KQS.mjs.map +7 -0
  27. package/dist/lib/neutral/components/index.mjs +243 -0
  28. package/dist/lib/neutral/components/index.mjs.map +7 -0
  29. package/dist/lib/neutral/containers/index.mjs +11 -0
  30. package/dist/lib/neutral/containers/index.mjs.map +7 -0
  31. package/dist/lib/neutral/create-object-DKBSI46K.mjs +40 -0
  32. package/dist/lib/neutral/create-object-DKBSI46K.mjs.map +7 -0
  33. package/dist/lib/neutral/delete-card-356CBACE.mjs +24 -0
  34. package/dist/lib/neutral/delete-card-356CBACE.mjs.map +7 -0
  35. package/dist/lib/neutral/delete-card-field-IRCZL2BR.mjs +42 -0
  36. package/dist/lib/neutral/delete-card-field-IRCZL2BR.mjs.map +7 -0
  37. package/dist/lib/neutral/hooks/index.mjs +432 -0
  38. package/dist/lib/neutral/hooks/index.mjs.map +7 -0
  39. package/dist/lib/neutral/index.mjs +34 -0
  40. package/dist/lib/neutral/index.mjs.map +7 -0
  41. package/dist/lib/neutral/meta.json +1 -0
  42. package/dist/lib/neutral/meta.mjs +8 -0
  43. package/dist/lib/neutral/meta.mjs.map +7 -0
  44. package/dist/lib/neutral/migrations-IWBT35UT.mjs +31 -0
  45. package/dist/lib/neutral/migrations-IWBT35UT.mjs.map +7 -0
  46. package/dist/lib/neutral/operation-handler-B7IW6MXU.mjs +13 -0
  47. package/dist/lib/neutral/operation-handler-B7IW6MXU.mjs.map +7 -0
  48. package/dist/lib/neutral/operations/index.mjs +8 -0
  49. package/dist/lib/neutral/operations/index.mjs.map +7 -0
  50. package/dist/lib/neutral/plugin.mjs +16 -0
  51. package/dist/lib/neutral/plugin.mjs.map +7 -0
  52. package/dist/lib/neutral/react-surface-QROEHBRW.mjs +93 -0
  53. package/dist/lib/neutral/react-surface-QROEHBRW.mjs.map +7 -0
  54. package/dist/lib/neutral/restore-card-P25Y4YSE.mjs +21 -0
  55. package/dist/lib/neutral/restore-card-P25Y4YSE.mjs.map +7 -0
  56. package/dist/lib/neutral/restore-card-field-NHR3R4XI.mjs +40 -0
  57. package/dist/lib/neutral/restore-card-field-NHR3R4XI.mjs.map +7 -0
  58. package/dist/lib/neutral/testing/index.mjs +60 -0
  59. package/dist/lib/neutral/testing/index.mjs.map +7 -0
  60. package/dist/lib/neutral/translations.mjs +44 -0
  61. package/dist/lib/neutral/translations.mjs.map +7 -0
  62. package/dist/lib/neutral/types/index.mjs +22 -0
  63. package/dist/lib/neutral/types/index.mjs.map +7 -0
  64. package/dist/lib/neutral/undo-mappings-6CHW6BOF.mjs +42 -0
  65. package/dist/lib/neutral/undo-mappings-6CHW6BOF.mjs.map +7 -0
  66. package/dist/types/src/KanbanPlugin.d.ts +3 -1
  67. package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
  68. package/dist/types/src/KanbanPlugin.node.d.ts +4 -0
  69. package/dist/types/src/KanbanPlugin.node.d.ts.map +1 -0
  70. package/dist/types/src/KanbanPlugin.test.d.ts +2 -0
  71. package/dist/types/src/KanbanPlugin.test.d.ts.map +1 -0
  72. package/dist/types/src/KanbanPlugin.workerd.d.ts +4 -0
  73. package/dist/types/src/KanbanPlugin.workerd.d.ts.map +1 -0
  74. package/dist/types/src/blueprints/index.d.ts +2 -0
  75. package/dist/types/src/blueprints/index.d.ts.map +1 -0
  76. package/dist/types/src/blueprints/kanban-blueprint.d.ts +4 -0
  77. package/dist/types/src/blueprints/kanban-blueprint.d.ts.map +1 -0
  78. package/dist/types/src/capabilities/artifact-definition.d.ts +3 -2
  79. package/dist/types/src/capabilities/artifact-definition.d.ts.map +1 -1
  80. package/dist/types/src/capabilities/blueprint-definition.d.ts +5 -4
  81. package/dist/types/src/capabilities/blueprint-definition.d.ts.map +1 -1
  82. package/dist/types/src/capabilities/create-object.d.ts +11 -0
  83. package/dist/types/src/capabilities/create-object.d.ts.map +1 -0
  84. package/dist/types/src/capabilities/index.d.ts +13 -3
  85. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  86. package/dist/types/src/capabilities/migrations.d.ts +6 -0
  87. package/dist/types/src/capabilities/migrations.d.ts.map +1 -0
  88. package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
  89. package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
  90. package/dist/types/src/capabilities/react-surface.d.ts +3 -2
  91. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  92. package/dist/types/src/capabilities/undo-mappings.d.ts +5 -0
  93. package/dist/types/src/capabilities/undo-mappings.d.ts.map +1 -0
  94. package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts +37 -0
  95. package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts.map +1 -0
  96. package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts +72 -0
  97. package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts.map +1 -0
  98. package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts +9 -0
  99. package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts.map +1 -0
  100. package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts +8 -0
  101. package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts.map +1 -0
  102. package/dist/types/src/components/KanbanBoard/context.d.ts +38 -0
  103. package/dist/types/src/components/KanbanBoard/context.d.ts.map +1 -0
  104. package/dist/types/src/components/KanbanBoard/index.d.ts +2 -0
  105. package/dist/types/src/components/KanbanBoard/index.d.ts.map +1 -0
  106. package/dist/types/src/components/index.d.ts +1 -2
  107. package/dist/types/src/components/index.d.ts.map +1 -1
  108. package/dist/types/src/containers/KanbanArticle/KanbanArticle.d.ts +6 -0
  109. package/dist/types/src/containers/KanbanArticle/KanbanArticle.d.ts.map +1 -0
  110. package/dist/types/src/containers/KanbanArticle/KanbanArticle.stories.d.ts +79 -0
  111. package/dist/types/src/containers/KanbanArticle/KanbanArticle.stories.d.ts.map +1 -0
  112. package/dist/types/src/containers/KanbanArticle/index.d.ts +2 -0
  113. package/dist/types/src/containers/KanbanArticle/index.d.ts.map +1 -0
  114. package/dist/types/src/containers/KanbanSettings/KanbanSettings.d.ts +13 -0
  115. package/dist/types/src/containers/KanbanSettings/KanbanSettings.d.ts.map +1 -0
  116. package/dist/types/src/containers/KanbanSettings/index.d.ts +2 -0
  117. package/dist/types/src/containers/KanbanSettings/index.d.ts.map +1 -0
  118. package/dist/types/src/containers/index.d.ts +4 -0
  119. package/dist/types/src/containers/index.d.ts.map +1 -0
  120. package/dist/types/src/hooks/index.d.ts +7 -0
  121. package/dist/types/src/hooks/index.d.ts.map +1 -0
  122. package/dist/types/src/hooks/useEchoChangeCallback.d.ts +13 -0
  123. package/dist/types/src/hooks/useEchoChangeCallback.d.ts.map +1 -0
  124. package/dist/types/src/hooks/useItemsProjection.d.ts +10 -0
  125. package/dist/types/src/hooks/useItemsProjection.d.ts.map +1 -0
  126. package/dist/types/src/hooks/useKanbanBoardModel.browser.test.d.ts +2 -0
  127. package/dist/types/src/hooks/useKanbanBoardModel.browser.test.d.ts.map +1 -0
  128. package/dist/types/src/hooks/useKanbanBoardModel.d.ts +16 -0
  129. package/dist/types/src/hooks/useKanbanBoardModel.d.ts.map +1 -0
  130. package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts +22 -0
  131. package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts.map +1 -0
  132. package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts +19 -0
  133. package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts.map +1 -0
  134. package/dist/types/src/hooks/useProjectionModel.d.ts +15 -0
  135. package/dist/types/src/hooks/useProjectionModel.d.ts.map +1 -0
  136. package/dist/types/src/index.d.ts +3 -1
  137. package/dist/types/src/index.d.ts.map +1 -1
  138. package/dist/types/src/meta.d.ts +2 -2
  139. package/dist/types/src/meta.d.ts.map +1 -1
  140. package/dist/types/src/operations/delete-card-field.d.ts +5 -0
  141. package/dist/types/src/operations/delete-card-field.d.ts.map +1 -0
  142. package/dist/types/src/operations/delete-card.d.ts +5 -0
  143. package/dist/types/src/operations/delete-card.d.ts.map +1 -0
  144. package/dist/types/src/operations/index.d.ts +3 -0
  145. package/dist/types/src/operations/index.d.ts.map +1 -0
  146. package/dist/types/src/operations/restore-card-field.d.ts +5 -0
  147. package/dist/types/src/operations/restore-card-field.d.ts.map +1 -0
  148. package/dist/types/src/operations/restore-card.d.ts +5 -0
  149. package/dist/types/src/operations/restore-card.d.ts.map +1 -0
  150. package/dist/types/src/playwright/board-manager.d.ts +5 -0
  151. package/dist/types/src/playwright/board-manager.d.ts.map +1 -0
  152. package/dist/types/src/playwright/playwright.config.d.ts +3 -0
  153. package/dist/types/src/playwright/playwright.config.d.ts.map +1 -0
  154. package/dist/types/src/playwright/smoke.spec.d.ts +2 -0
  155. package/dist/types/src/playwright/smoke.spec.d.ts.map +1 -0
  156. package/dist/types/src/plugin.d.ts +4 -0
  157. package/dist/types/src/plugin.d.ts.map +1 -0
  158. package/dist/types/src/testing/KanbanCardTileSimple.d.ts +7 -0
  159. package/dist/types/src/testing/KanbanCardTileSimple.d.ts.map +1 -0
  160. package/dist/types/src/testing/index.d.ts +2 -0
  161. package/dist/types/src/testing/index.d.ts.map +1 -0
  162. package/dist/types/src/translations.d.ts +50 -22
  163. package/dist/types/src/translations.d.ts.map +1 -1
  164. package/dist/types/src/types/Kanban.d.ts +108 -0
  165. package/dist/types/src/types/Kanban.d.ts.map +1 -0
  166. package/dist/types/src/types/KanbanOperation.d.ts +52 -0
  167. package/dist/types/src/types/KanbanOperation.d.ts.map +1 -0
  168. package/dist/types/src/types/constants.d.ts +6 -0
  169. package/dist/types/src/types/constants.d.ts.map +1 -0
  170. package/dist/types/src/types/index.d.ts +3 -0
  171. package/dist/types/src/types/index.d.ts.map +1 -1
  172. package/dist/types/src/types/migrations.test.d.ts +2 -0
  173. package/dist/types/src/types/migrations.test.d.ts.map +1 -0
  174. package/dist/types/src/types/schema.d.ts +17 -51
  175. package/dist/types/src/types/schema.d.ts.map +1 -1
  176. package/dist/types/src/types/types.d.ts +28 -0
  177. package/dist/types/src/types/types.d.ts.map +1 -1
  178. package/dist/types/src/util/arrangement.d.ts +72 -0
  179. package/dist/types/src/util/arrangement.d.ts.map +1 -0
  180. package/dist/types/src/util/arrangement.test.d.ts +2 -0
  181. package/dist/types/src/util/arrangement.test.d.ts.map +1 -0
  182. package/dist/types/src/util/index.d.ts +2 -0
  183. package/dist/types/src/util/index.d.ts.map +1 -0
  184. package/dist/types/tsconfig.tsbuildinfo +1 -1
  185. package/package.json +123 -57
  186. package/src/KanbanPlugin.node.ts +21 -0
  187. package/src/KanbanPlugin.test.ts +31 -0
  188. package/src/KanbanPlugin.tsx +28 -56
  189. package/src/KanbanPlugin.workerd.ts +18 -0
  190. package/src/blueprints/index.ts +5 -0
  191. package/src/blueprints/kanban-blueprint.ts +27 -0
  192. package/src/capabilities/artifact-definition.ts +116 -114
  193. package/src/capabilities/blueprint-definition.ts +13 -24
  194. package/src/capabilities/create-object.ts +40 -0
  195. package/src/capabilities/index.ts +12 -4
  196. package/src/capabilities/migrations.ts +35 -0
  197. package/src/capabilities/operation-handler.ts +14 -0
  198. package/src/capabilities/react-surface.tsx +77 -67
  199. package/src/capabilities/undo-mappings.ts +34 -0
  200. package/src/components/KanbanBoard/KanbanBoard.stories.tsx +145 -0
  201. package/src/components/KanbanBoard/KanbanBoard.tsx +164 -0
  202. package/src/components/KanbanBoard/KanbanCard.tsx +101 -0
  203. package/src/components/KanbanBoard/KanbanColumn.tsx +72 -0
  204. package/src/components/KanbanBoard/context.ts +54 -0
  205. package/src/components/KanbanBoard/index.ts +5 -0
  206. package/src/components/index.ts +1 -2
  207. package/src/containers/KanbanArticle/KanbanArticle.stories.tsx +276 -0
  208. package/src/containers/KanbanArticle/KanbanArticle.tsx +178 -0
  209. package/src/containers/KanbanArticle/index.ts +5 -0
  210. package/src/containers/KanbanSettings/KanbanSettings.tsx +94 -0
  211. package/src/containers/KanbanSettings/index.ts +5 -0
  212. package/src/containers/index.ts +8 -0
  213. package/src/hooks/index.ts +10 -0
  214. package/src/hooks/useEchoChangeCallback.ts +30 -0
  215. package/src/hooks/useItemsProjection.ts +44 -0
  216. package/src/hooks/useKanbanBoardModel.browser.test.ts +234 -0
  217. package/src/hooks/useKanbanBoardModel.ts +157 -0
  218. package/src/hooks/useKanbanColumnEventHandler.ts +106 -0
  219. package/src/hooks/useKanbanItemEventHandler.ts +133 -0
  220. package/src/hooks/useProjectionModel.ts +58 -0
  221. package/src/index.ts +3 -2
  222. package/src/meta.ts +3 -3
  223. package/src/operations/delete-card-field.ts +47 -0
  224. package/src/operations/delete-card.ts +23 -0
  225. package/src/operations/index.ts +10 -0
  226. package/src/operations/restore-card-field.ts +41 -0
  227. package/src/operations/restore-card.ts +21 -0
  228. package/src/playwright/board-manager.ts +13 -0
  229. package/src/playwright/playwright.config.ts +19 -0
  230. package/src/playwright/smoke.spec.ts +107 -0
  231. package/src/plugin.ts +11 -0
  232. package/src/testing/KanbanCardTileSimple.tsx +82 -0
  233. package/src/testing/index.ts +5 -0
  234. package/src/translations.ts +28 -20
  235. package/src/types/Kanban.ts +150 -0
  236. package/src/types/KanbanOperation.ts +67 -0
  237. package/src/types/constants.ts +9 -0
  238. package/src/types/index.ts +4 -0
  239. package/src/types/migrations.test.ts +82 -0
  240. package/src/types/schema.ts +32 -44
  241. package/src/types/types.ts +35 -0
  242. package/src/util/arrangement.test.ts +217 -0
  243. package/src/util/arrangement.ts +177 -0
  244. package/src/util/index.ts +5 -0
  245. package/dist/lib/browser/blueprint-definition-UYVX622Q.mjs +0 -28
  246. package/dist/lib/browser/blueprint-definition-UYVX622Q.mjs.map +0 -7
  247. package/dist/lib/browser/chunk-3UDST345.mjs +0 -85
  248. package/dist/lib/browser/chunk-3UDST345.mjs.map +0 -7
  249. package/dist/lib/browser/index.mjs +0 -105
  250. package/dist/lib/browser/index.mjs.map +0 -7
  251. package/dist/lib/browser/intent-resolver-VVBNS2TO.mjs +0 -111
  252. package/dist/lib/browser/intent-resolver-VVBNS2TO.mjs.map +0 -7
  253. package/dist/lib/browser/meta.json +0 -1
  254. package/dist/lib/browser/react-surface-FNXJ6VJX.mjs +0 -255
  255. package/dist/lib/browser/react-surface-FNXJ6VJX.mjs.map +0 -7
  256. package/dist/lib/browser/types/index.mjs +0 -11
  257. package/dist/lib/node-esm/blueprint-definition-42P47FUY.mjs +0 -30
  258. package/dist/lib/node-esm/blueprint-definition-42P47FUY.mjs.map +0 -7
  259. package/dist/lib/node-esm/chunk-JBOARUAT.mjs +0 -87
  260. package/dist/lib/node-esm/chunk-JBOARUAT.mjs.map +0 -7
  261. package/dist/lib/node-esm/index.mjs +0 -106
  262. package/dist/lib/node-esm/index.mjs.map +0 -7
  263. package/dist/lib/node-esm/intent-resolver-ACN7UALP.mjs +0 -112
  264. package/dist/lib/node-esm/intent-resolver-ACN7UALP.mjs.map +0 -7
  265. package/dist/lib/node-esm/meta.json +0 -1
  266. package/dist/lib/node-esm/react-surface-ZHYHCV5N.mjs +0 -256
  267. package/dist/lib/node-esm/react-surface-ZHYHCV5N.mjs.map +0 -7
  268. package/dist/lib/node-esm/types/index.mjs +0 -12
  269. package/dist/types/src/capabilities/intent-resolver.d.ts +0 -4
  270. package/dist/types/src/capabilities/intent-resolver.d.ts.map +0 -1
  271. package/dist/types/src/components/KanbanContainer.d.ts +0 -7
  272. package/dist/types/src/components/KanbanContainer.d.ts.map +0 -1
  273. package/dist/types/src/components/KanbanContainer.stories.d.ts +0 -41
  274. package/dist/types/src/components/KanbanContainer.stories.d.ts.map +0 -1
  275. package/dist/types/src/components/KanbanViewEditor.d.ts +0 -8
  276. package/dist/types/src/components/KanbanViewEditor.d.ts.map +0 -1
  277. package/src/capabilities/intent-resolver.ts +0 -71
  278. package/src/components/KanbanContainer.stories.tsx +0 -193
  279. package/src/components/KanbanContainer.tsx +0 -95
  280. package/src/components/KanbanViewEditor.tsx +0 -64
  281. /package/dist/lib/{browser/types → neutral/blueprints}/index.mjs.map +0 -0
  282. /package/dist/lib/{node-esm/types/index.mjs.map → neutral/chunk-J5LGTIGS.mjs.map} +0 -0
@@ -0,0 +1,276 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { RegistryContext } from '@effect-atom/atom-react';
6
+ import { type Decorator, type Meta, type StoryObj } from '@storybook/react-vite';
7
+ import * as Effect from 'effect/Effect';
8
+ import React, { useCallback, useContext, useMemo } from 'react';
9
+ import { expect, waitFor, within } from 'storybook/test';
10
+
11
+ import { withPluginManager } from '@dxos/app-framework/testing';
12
+ import { Surface } from '@dxos/app-framework/ui';
13
+ import { AppSurface } from '@dxos/app-toolkit/ui';
14
+ import { Filter, Obj, type QueryAST, Type, View } from '@dxos/echo';
15
+ import { type Mutable } from '@dxos/echo/internal';
16
+ import { invariant } from '@dxos/invariant';
17
+ // `/plugin` entrypoints used here for the same reason as `corePlugins()` —
18
+ // see `@dxos/plugin-testing/src/core.ts` for the rationale.
19
+ import { ClientPlugin } from '@dxos/plugin-client/testing';
20
+ import { initializeIdentity } from '@dxos/plugin-client/testing';
21
+ import { PreviewPlugin } from '@dxos/plugin-preview/testing';
22
+ import { SpacePlugin } from '@dxos/plugin-space/testing';
23
+ import { StorybookPlugin, corePlugins } from '@dxos/plugin-testing';
24
+ import { random } from '@dxos/random';
25
+ import { type Space, useQuery, useSchema, useSpaces } from '@dxos/react-client/echo';
26
+ import { ViewEditor } from '@dxos/react-ui-form';
27
+ import { Syntax } from '@dxos/react-ui-syntax-highlighter';
28
+ import { withLayout } from '@dxos/react-ui/testing';
29
+ import { ViewModel, getTypenameFromQuery } from '@dxos/schema';
30
+ // TODO(wittjosiah): Replace with echo/testing.
31
+ import { Organization, Person } from '@dxos/types';
32
+
33
+ import { useProjectionModel } from '#hooks';
34
+ import { translations } from '#translations';
35
+ import { Kanban } from '#types';
36
+
37
+ import { KanbanPlugin } from '../../KanbanPlugin';
38
+
39
+ random.seed(0);
40
+
41
+ const createOrg = (status?: Organization.Organization['status']) => ({
42
+ name: random.commerce.productName(),
43
+ description: random.lorem.paragraph(),
44
+ image: random.image.url(),
45
+ website: random.internet.url(),
46
+ status: (status ?? random.helpers.arrayElement(Organization.StatusOptions).id) as Organization.Organization['status'],
47
+ });
48
+
49
+ //
50
+ // Story setup helpers.
51
+ //
52
+
53
+ type ClientSetupOptions = {
54
+ types?: Type.AnyEntity[];
55
+ onSpaceCreated?: (space: Space) => Promise<void>;
56
+ };
57
+
58
+ /**
59
+ * Creates the standard plugin manager decorator with client configuration.
60
+ * Includes KanbanPlugin so the Surface resolves to KanbanArticle.
61
+ */
62
+ const withKanbanPlugins = ({ types = [], onSpaceCreated }: ClientSetupOptions): Decorator =>
63
+ withPluginManager({
64
+ plugins: [
65
+ ...corePlugins(),
66
+ ClientPlugin({
67
+ types: [...types, View.View, Kanban.Kanban],
68
+ onClientInitialized: ({ client }) =>
69
+ Effect.gen(function* () {
70
+ yield* initializeIdentity(client);
71
+ const space = yield* Effect.promise(() => client.spaces.create());
72
+ yield* Effect.promise(() => space.waitUntilReady());
73
+ yield* Effect.promise(() => onSpaceCreated?.(space) ?? Promise.resolve());
74
+ }),
75
+ }),
76
+ PreviewPlugin(),
77
+ SpacePlugin({}),
78
+ StorybookPlugin({}),
79
+ KanbanPlugin(),
80
+ ],
81
+ });
82
+
83
+ /**
84
+ * Renders the first Kanban in the space via Surface (resolves to KanbanArticle),
85
+ * with a sidebar containing ViewEditor and Json filter.
86
+ */
87
+ const DefaultComponent = () => {
88
+ const registry = useContext(RegistryContext);
89
+ const spaces = useSpaces();
90
+ const space = spaces[spaces.length - 1];
91
+ const [kanban] = useQuery(space?.db, Filter.type(Kanban.Kanban));
92
+ const viewRef = kanban && kanban.spec.kind === 'view' ? kanban.spec.view : undefined;
93
+ const view = viewRef?.target;
94
+ const typename = view?.query ? getTypenameFromQuery(view.query.ast) : undefined;
95
+ const schema = useSchema(space?.db, typename);
96
+ const projection = useProjectionModel(schema, kanban, registry);
97
+
98
+ const data = useMemo(() => (kanban ? { subject: kanban, attendableId: 'story' } : undefined), [kanban]);
99
+
100
+ const handleUpdateQuery = useCallback(
101
+ (newQuery: QueryAST.Query) => {
102
+ invariant(schema);
103
+ invariant(view);
104
+ if (Type.isMutable(schema)) {
105
+ schema.updateTypename(getTypenameFromQuery(newQuery));
106
+ }
107
+ Obj.update(view, (view) => {
108
+ view.query.ast = newQuery as Mutable<QueryAST.Query>;
109
+ });
110
+ },
111
+ [view, schema],
112
+ );
113
+
114
+ const handleDeleteField = useCallback(
115
+ (fieldId: string) => {
116
+ if (schema && Type.isMutable(schema) && projection) {
117
+ projection.deleteFieldProjection(fieldId);
118
+ }
119
+ },
120
+ [schema, projection],
121
+ );
122
+
123
+ if (!schema || !view) {
124
+ return null;
125
+ }
126
+
127
+ return (
128
+ <div className='grow grid grid-cols-[1fr_350px] overflow-hidden h-full w-full'>
129
+ <Surface.Surface type={AppSurface.Article} data={data} limit={1} />
130
+ <div className='flex flex-col h-full overflow-hidden border-l border-separator'>
131
+ <ViewEditor
132
+ registry={space?.db.schemaRegistry}
133
+ schema={schema}
134
+ view={view}
135
+ onQueryChanged={handleUpdateQuery}
136
+ onDelete={schema && Type.isMutable(schema) ? handleDeleteField : undefined}
137
+ />
138
+ <Syntax.Root data={{ view, schema }}>
139
+ <Syntax.Content>
140
+ <Syntax.Filter />
141
+ <Syntax.Viewport>
142
+ <Syntax.Code classNames='text-xs' />
143
+ </Syntax.Viewport>
144
+ </Syntax.Content>
145
+ </Syntax.Root>
146
+ </div>
147
+ </div>
148
+ );
149
+ };
150
+
151
+ //
152
+ // Story definitions.
153
+ //
154
+
155
+ const meta = {
156
+ title: 'plugins/plugin-kanban/containers/Kanban',
157
+ component: DefaultComponent,
158
+ render: () => <DefaultComponent />,
159
+ decorators: [withLayout({ layout: 'fullscreen' })],
160
+ parameters: {
161
+ layout: 'fullscreen',
162
+ translations,
163
+ },
164
+ } satisfies Meta<typeof DefaultComponent>;
165
+
166
+ export default meta;
167
+
168
+ type Story = StoryObj<typeof meta>;
169
+
170
+ /**
171
+ * Default story using static runtime schema (immutable).
172
+ * Schema mutations are not allowed.
173
+ */
174
+ export const Default: Story = {
175
+ decorators: [
176
+ withKanbanPlugins({
177
+ types: [Organization.Organization, Person.Person],
178
+ onSpaceCreated: async (space) => {
179
+ const { view } = await ViewModel.makeFromDatabase({
180
+ db: space.db,
181
+ typename: Organization.Organization.typename,
182
+ pivotFieldName: 'status',
183
+ });
184
+ const kanban = Kanban.make({ view });
185
+ space.db.add(kanban);
186
+
187
+ Array.from({ length: 10 }).map(() => {
188
+ return space.db.add(Obj.make(Organization.Organization, createOrg()));
189
+ });
190
+ },
191
+ }),
192
+ ],
193
+ play: async ({ canvasElement }) => {
194
+ const canvas = within(canvasElement);
195
+
196
+ // Wait for the kanban columns to render by finding the status tags.
197
+ // Organization.StatusOptions: prospect, qualified, active, commit, reject.
198
+ const activeTag = await canvas.findByText('Active', undefined, { timeout: 12_000 });
199
+ const prospectTag = await canvas.findByText('Prospect', undefined, { timeout: 12_000 });
200
+ const commitTag = await canvas.findByText('Commit', undefined, { timeout: 12_000 });
201
+
202
+ // Verify all expected columns are rendered.
203
+ await expect(activeTag).toBeTruthy();
204
+ await expect(prospectTag).toBeTruthy();
205
+ await expect(commitTag).toBeTruthy();
206
+
207
+ // Find the column containers (Board uses data-testid="board-column").
208
+ const activeColumn = activeTag.closest('[data-testid="board-column"]') as HTMLElement;
209
+ const prospectColumn = prospectTag.closest('[data-testid="board-column"]') as HTMLElement;
210
+ await expect(activeColumn).toBeTruthy();
211
+ await expect(prospectColumn).toBeTruthy();
212
+
213
+ // Wait for cards to render in the columns (Board items use data-testid="board-item").
214
+ const getColumnCards = (column: HTMLElement) =>
215
+ Array.from(column.querySelectorAll('[data-testid="board-item"]')) as HTMLElement[];
216
+
217
+ await waitFor(() => expect(getColumnCards(activeColumn).length).toBeGreaterThan(0));
218
+
219
+ // Verify cards are distributed across columns.
220
+ const activeCards = getColumnCards(activeColumn);
221
+ const prospectCards = getColumnCards(prospectColumn);
222
+ await expect(activeCards.length).toBeGreaterThan(0);
223
+ await expect(prospectCards.length).toBeGreaterThan(0);
224
+
225
+ // Verify cards have drag handles (Card.Toolbar includes drag handle).
226
+ const firstActiveCard = activeCards[0];
227
+ const buttons = firstActiveCard.querySelectorAll('button');
228
+ await expect(buttons.length).toBeGreaterThan(0);
229
+
230
+ // Verify add-card action exists in columns (optional footer).
231
+ const activeAddItem = activeColumn.querySelector('[data-testid="board-column-add-item"]');
232
+ const prospectAddItem = prospectColumn.querySelector('[data-testid="board-column-add-item"]');
233
+ await expect(activeAddItem).toBeTruthy();
234
+ await expect(prospectAddItem).toBeTruthy();
235
+
236
+ // TODO(wittjosiah): Get drag & drop tests working.
237
+ // See packages/apps/composer-app/src/playwright/stack.spec.ts for reference.
238
+ },
239
+ };
240
+
241
+ /**
242
+ * Story variant that uses a mutable database schema (EchoSchema).
243
+ * This allows testing schema mutations like adding/removing fields.
244
+ */
245
+ // TODO(wittjosiah): Card previews (e.g., OrganizationCard) are type-specific and hard-coded.
246
+ // They don't use the projection to determine which fields to display, so deleting a field
247
+ // from the schema won't remove it from the card preview. To fix this, the type-specific
248
+ // cards in PreviewPlugin would need to accept and respect the projection prop.
249
+ export const MutableSchema: Story = {
250
+ decorators: [
251
+ withKanbanPlugins({
252
+ onSpaceCreated: async (space) => {
253
+ // Register schema in the database to make it mutable (EchoSchema).
254
+ const [schema] = await space.db.schemaRegistry.register([Organization.Organization]);
255
+
256
+ const { view } = await ViewModel.makeFromDatabase({
257
+ db: space.db,
258
+ typename: schema.typename,
259
+ pivotFieldName: 'status',
260
+ });
261
+ const kanban = Kanban.make({ view });
262
+ space.db.add(kanban);
263
+
264
+ // Create test data using the registered schema.
265
+ const requiredOrgs = [
266
+ ...Array.from({ length: 2 }, () => createOrg('prospect')),
267
+ ...Array.from({ length: 5 }, () => createOrg('qualified')),
268
+ ...Array.from({ length: 1 }, () => createOrg('active')),
269
+ ...Array.from({ length: 1 }, () => createOrg('commit')),
270
+ ...Array.from({ length: 1 }, () => createOrg('reject')),
271
+ ];
272
+ requiredOrgs.forEach((org) => space.db.add(Obj.make(schema, org)));
273
+ },
274
+ }),
275
+ ],
276
+ };
@@ -0,0 +1,178 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { Atom, RegistryContext } from '@effect-atom/atom-react';
6
+ import React, { useCallback, useContext, useMemo } from 'react';
7
+
8
+ import { useCapabilities, useOperationInvoker } from '@dxos/app-framework/ui';
9
+ import { AppCapabilities } from '@dxos/app-toolkit';
10
+ import { type AppSurface } from '@dxos/app-toolkit/ui';
11
+ import { Filter, Obj, Query, type Ref, Type } from '@dxos/echo';
12
+ import { AtomObj, AtomQuery } from '@dxos/echo-atom';
13
+ import { useObject, useSchema } from '@dxos/react-client/echo';
14
+ import { Panel, Toolbar } from '@dxos/react-ui';
15
+ import { getTagFromQuery, getTypenameFromQuery } from '@dxos/schema';
16
+
17
+ import { KanbanBoard } from '#components';
18
+ import { useEchoChangeCallback, useItemsProjection, useProjectionModel } from '#hooks';
19
+ import { KanbanOperation } from '#types';
20
+ import { Kanban } from '#types';
21
+
22
+ export type KanbanArticleProps = AppSurface.ObjectArticleProps<Kanban.Kanban>;
23
+
24
+ export const KanbanArticle = (props: KanbanArticleProps) => {
25
+ // Branch on `kanban.spec.kind`: view-variant runs a typename query through
26
+ // `useProjectionModel`; items-variant dereferences `kanban.spec.items` and
27
+ // uses a stub projection from `useItemsProjection`.
28
+ return Kanban.isKanbanItems(props.subject) ? (
29
+ <ItemsKanbanArticle {...props} subject={props.subject} />
30
+ ) : (
31
+ <ViewKanbanArticle {...props} />
32
+ );
33
+ };
34
+
35
+ const ViewKanbanArticle = ({ role, subject: object }: KanbanArticleProps) => {
36
+ const registry = useContext(RegistryContext);
37
+ const schemas = useCapabilities(AppCapabilities.Schema);
38
+ const db = Obj.getDatabase(object);
39
+ const { invokePromise } = useOperationInvoker();
40
+ const [view] = useObject(object.spec.kind === 'view' ? object.spec.view : undefined);
41
+ const typename = view?.query ? getTypenameFromQuery(view.query.ast) : undefined;
42
+ const tag = view?.query ? getTagFromQuery(view.query.ast) : undefined;
43
+
44
+ const schemaFromDb = useSchema(db, typename);
45
+ const cardSchema = useMemo(
46
+ () => schemaFromDb ?? schemas.flat().find((schema) => Type.getTypename(schema) === typename),
47
+ [schemaFromDb, schemas, typename],
48
+ );
49
+
50
+ const items = useMemo(() => {
51
+ if (!db) {
52
+ return null;
53
+ }
54
+ const baseFilter = cardSchema ? Filter.type(cardSchema) : Filter.nothing();
55
+ const query = tag ? Query.select(baseFilter).select(Filter.tag(tag)) : Query.select(baseFilter);
56
+ return AtomQuery.make(db, query);
57
+ }, [db, cardSchema, tag]);
58
+
59
+ const projection = useProjectionModel(cardSchema, object, registry);
60
+ const change = useEchoChangeCallback(object);
61
+
62
+ const pivotFieldId = view?.projection?.pivotFieldId;
63
+ const columnFieldPath =
64
+ projection && pivotFieldId ? projection.tryGetFieldProjection(pivotFieldId)?.props.property : undefined;
65
+
66
+ const handleCardAdd = useCallback(
67
+ (columnValue: string | undefined) => {
68
+ if (db && cardSchema && columnFieldPath) {
69
+ const card = Obj.make(cardSchema, { [columnFieldPath]: columnValue });
70
+ db.add(card);
71
+ return card.id;
72
+ }
73
+ },
74
+ [db, cardSchema, columnFieldPath],
75
+ );
76
+
77
+ const handleCardRemove = useCallback(
78
+ (card: { id: string }) => {
79
+ void invokePromise(KanbanOperation.DeleteCard, { card });
80
+ },
81
+ [invokePromise],
82
+ );
83
+
84
+ if (!object || !db || !items || !projection || !change) {
85
+ return null;
86
+ }
87
+
88
+ return (
89
+ <Panel.Root role={role}>
90
+ <Panel.Toolbar asChild>
91
+ <Toolbar.Root />
92
+ </Panel.Toolbar>
93
+ <KanbanBoard.Root
94
+ kanban={object}
95
+ projection={projection}
96
+ items={items}
97
+ change={change}
98
+ onCardAdd={handleCardAdd}
99
+ onCardRemove={handleCardRemove}
100
+ >
101
+ <Panel.Content asChild>
102
+ <KanbanBoard.Content />
103
+ </Panel.Content>
104
+ </KanbanBoard.Root>
105
+ </Panel.Root>
106
+ );
107
+ };
108
+
109
+ type ItemsKanbanArticleProps = Omit<KanbanArticleProps, 'subject'> & { subject: Kanban.KanbanItems };
110
+
111
+ const ItemsKanbanArticle = ({ role, subject: object }: ItemsKanbanArticleProps) => {
112
+ const db = Obj.getDatabase(object);
113
+ const projection = useItemsProjection(object);
114
+ const change = useEchoChangeCallback(object);
115
+
116
+ // TODO(wittjosiah): pass refs (not loaded objects) through to the kanban
117
+ // board and let `KanbanCard` subscribe to its own ref via `useObject`.
118
+ // Today this atom subscribes to *every* item — any one changing causes the
119
+ // container (and the model's per-column atoms) to recompute. With cards
120
+ // subscribing themselves, the container only needs the refs and the
121
+ // per-card render is independent. Requires:
122
+ // - `KanbanCard` to accept `Ref<Obj.Unknown>` as `data` and call
123
+ // `useObject(ref)` internally.
124
+ // - The model to handle a ref-bearing item shape (id from
125
+ // `ref.dxn.asEchoDXN()?.echoId`) and use arrangement-only ordering
126
+ // for items-variant (no pivot-value fallback, since refs don't expose
127
+ // the pivot field without loading).
128
+ // - `Mosaic.isItem` to accept the ref wrapper alongside `Obj.isObject`.
129
+ const itemsAtom = useMemo(
130
+ () =>
131
+ Atom.make((get) => {
132
+ const out: Obj.Unknown[] = [];
133
+ for (const ref of object.spec.items as ReadonlyArray<Ref.Ref<Obj.Unknown>>) {
134
+ const target = get(AtomObj.make(ref));
135
+ if (target == null) {
136
+ continue;
137
+ }
138
+ // Drop soft-deleted cards (e.g. Trello-closed cards). The ref
139
+ // stays in `spec.items` so arrangement is preserved, but the card
140
+ // shouldn't render.
141
+ if (Obj.isDeleted(target)) {
142
+ continue;
143
+ }
144
+ out.push(target as unknown as Obj.Unknown);
145
+ }
146
+ return out;
147
+ }),
148
+ [object.spec.items],
149
+ );
150
+
151
+ const handleCardRemove = useCallback(() => undefined, []);
152
+
153
+ if (!object || !db || !change) {
154
+ return null;
155
+ }
156
+
157
+ // TODO(wittjosiah): wire `onCardAdd` to the create-object flow so
158
+ // users can add items directly from the kanban (currently the column's
159
+ // "+" button is hidden because `onCardAdd` is undefined).
160
+ return (
161
+ <Panel.Root role={role}>
162
+ <Panel.Toolbar asChild>
163
+ <Toolbar.Root />
164
+ </Panel.Toolbar>
165
+ <KanbanBoard.Root
166
+ kanban={object}
167
+ projection={projection}
168
+ items={itemsAtom}
169
+ change={change}
170
+ onCardRemove={handleCardRemove}
171
+ >
172
+ <Panel.Content asChild>
173
+ <KanbanBoard.Content />
174
+ </Panel.Content>
175
+ </KanbanBoard.Root>
176
+ </Panel.Root>
177
+ );
178
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ export { KanbanArticle as default } from './KanbanArticle';
@@ -0,0 +1,94 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { RegistryContext } from '@effect-atom/atom-react';
6
+ import React, { useCallback, useContext, useMemo } from 'react';
7
+
8
+ import { type AppSurface } from '@dxos/app-toolkit/ui';
9
+ import { Obj } from '@dxos/echo';
10
+ import { Format } from '@dxos/echo/internal';
11
+ import { useObject, useSchema } from '@dxos/react-client/echo';
12
+ import { Form, type FormFieldMap, SelectField } from '@dxos/react-ui-form';
13
+ import { getTypenameFromQuery } from '@dxos/schema';
14
+
15
+ import { useProjectionModel } from '#hooks';
16
+ import { type Kanban, KanbanSettingsSchema, KanbanViewSettingsSchema, UNCATEGORIZED_VALUE } from '#types';
17
+
18
+ export type KanbanSettingsProps = AppSurface.ObjectPropertiesProps<Kanban.Kanban>;
19
+
20
+ /**
21
+ * Settings panel for a Kanban. Renders fields common to every kanban
22
+ * (currently the "Hide uncategorized column" toggle); for view-variant
23
+ * kanbans an additional "Column field" picker drives the View's pivot
24
+ * field. Items-variant kanbans use a hardcoded `spec.pivotField`, so that
25
+ * field is omitted there.
26
+ */
27
+ export const KanbanSettings = ({ subject: object }: KanbanSettingsProps) => {
28
+ const registry = useContext(RegistryContext);
29
+ const db = Obj.getDatabase(object);
30
+ const isView = object.spec.kind === 'view';
31
+ const [view, updateView] = useObject(object.spec.kind === 'view' ? object.spec.view : undefined);
32
+ const [, updateKanban] = useObject(object);
33
+ const currentTypename = view?.query ? getTypenameFromQuery(view.query.ast) : undefined;
34
+ const schema = useSchema(db, currentTypename);
35
+ const projection = useProjectionModel(schema, object, registry);
36
+
37
+ const fieldProjections = projection?.getFieldProjections() ?? [];
38
+ const selectFields = useMemo(
39
+ () =>
40
+ fieldProjections
41
+ .filter((field) => field.props.format === Format.TypeFormat.SingleSelect)
42
+ .map(({ field }) => ({ value: field.id, label: field.path })),
43
+ [fieldProjections],
44
+ );
45
+
46
+ const hideUncategorized = object.arrangement.columns[UNCATEGORIZED_VALUE]?.hidden ?? false;
47
+
48
+ const handleValuesChanged = useCallback(
49
+ (values: Partial<{ columnFieldId: string; hideUncategorized: boolean }>) => {
50
+ if (isView && values.columnFieldId != null) {
51
+ updateView((view) => {
52
+ view.projection.pivotFieldId = values.columnFieldId!;
53
+ });
54
+ }
55
+ if (values.hideUncategorized !== undefined) {
56
+ updateKanban((kanban) => {
57
+ const existing = kanban.arrangement.columns[UNCATEGORIZED_VALUE];
58
+ if (existing) {
59
+ existing.hidden = values.hideUncategorized;
60
+ } else {
61
+ kanban.arrangement.columns[UNCATEGORIZED_VALUE] = {
62
+ ids: [],
63
+ hidden: values.hideUncategorized,
64
+ };
65
+ }
66
+ });
67
+ }
68
+ },
69
+ [isView, updateView, updateKanban],
70
+ );
71
+
72
+ const initialValues = useMemo(
73
+ () => ({
74
+ ...(isView ? { columnFieldId: view?.projection.pivotFieldId } : {}),
75
+ hideUncategorized,
76
+ }),
77
+ [isView, view?.projection.pivotFieldId, hideUncategorized],
78
+ );
79
+
80
+ const fieldMap: FormFieldMap = useMemo(
81
+ () => ({ columnFieldId: (props) => <SelectField {...props} options={selectFields} /> }),
82
+ [selectFields],
83
+ );
84
+
85
+ // Schema is picked by `kanban.spec.kind` — they have different shapes,
86
+ // so cast for `Form.Root`'s single-schema prop.
87
+ const settingsSchema = (isView ? KanbanViewSettingsSchema : KanbanSettingsSchema) as any;
88
+
89
+ return (
90
+ <Form.Root schema={settingsSchema} values={initialValues} fieldMap={fieldMap} onValuesChanged={handleValuesChanged}>
91
+ <Form.FieldSet />
92
+ </Form.Root>
93
+ );
94
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ export { KanbanSettings as default } from './KanbanSettings';
@@ -0,0 +1,8 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { type ComponentType, lazy } from 'react';
6
+
7
+ export const KanbanArticle: ComponentType<any> = lazy(() => import('./KanbanArticle'));
8
+ export const KanbanSettings: ComponentType<any> = lazy(() => import('./KanbanSettings'));
@@ -0,0 +1,10 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './useEchoChangeCallback';
6
+ export * from './useItemsProjection';
7
+ export * from './useKanbanBoardModel';
8
+ export * from './useKanbanColumnEventHandler';
9
+ export * from './useKanbanItemEventHandler';
10
+ export * from './useProjectionModel';
@@ -0,0 +1,30 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { useMemo } from 'react';
6
+
7
+ import { Obj } from '@dxos/echo';
8
+
9
+ import { type Kanban, type KanbanChangeCallback } from '#types';
10
+
11
+ /**
12
+ * Creates a change callback for ECHO-backed kanban and items (plain function, no hooks).
13
+ * Use this when the kanban and items are stored in the ECHO database.
14
+ */
15
+ export const createEchoChangeCallback = <T extends Obj.Unknown>(kanban: Kanban.Kanban): KanbanChangeCallback<T> => ({
16
+ kanban: (mutate) => Obj.update(kanban, (kanban) => mutate(kanban)),
17
+ setItemField: (item, field, value) => {
18
+ Obj.update(item, (item: any) => {
19
+ item[field] = value;
20
+ });
21
+ },
22
+ });
23
+
24
+ /**
25
+ * Returns a memoized ECHO change callback for the given kanban.
26
+ * Returns null when kanban is undefined.
27
+ */
28
+ export const useEchoChangeCallback = <T extends Obj.Unknown = Obj.Unknown>(
29
+ kanban: Kanban.Kanban | undefined,
30
+ ): KanbanChangeCallback<T> | null => useMemo(() => (kanban ? createEchoChangeCallback<T>(kanban) : null), [kanban]);
@@ -0,0 +1,44 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { Atom } from '@effect-atom/atom-react';
6
+ import { useMemo } from 'react';
7
+
8
+ import type { ProjectionModel } from '@dxos/schema';
9
+
10
+ import { type Kanban } from '#types';
11
+
12
+ /**
13
+ * Minimal `ProjectionModel` for `spec.kind === 'items'` (no View). Supplies `pivotField`
14
+ * and column options from `arrangement.columns` keys—written by sync so columns exist
15
+ * before refs hydrate. Stubs `getFieldProjections` / `getHiddenProperties` for shared
16
+ * board/card UI; hides the pivot on the card body (column shows it); Expando cards render title only.
17
+ */
18
+ export const useItemsProjection = (kanban: Kanban.KanbanItems): ProjectionModel => {
19
+ return useMemo(() => {
20
+ const pivotField = kanban.spec.pivotField;
21
+
22
+ const optionIds = Object.keys(kanban.arrangement?.columns ?? {});
23
+ const options = optionIds.map((id) => ({ id, title: id, color: 'neutral' as const }));
24
+
25
+ const fieldProjection: any = {
26
+ field: { id: pivotField, path: pivotField },
27
+ props: { property: pivotField, options },
28
+ };
29
+
30
+ const fields = Atom.make(() => [fieldProjection.field]);
31
+
32
+ const stub: Pick<ProjectionModel, 'tryGetFieldProjection' | 'getFieldProjections' | 'getHiddenProperties'> & {
33
+ fields: typeof fields;
34
+ } = {
35
+ fields,
36
+ tryGetFieldProjection: (id: string) => (id === pivotField ? fieldProjection : undefined),
37
+ getFieldProjections: () => [],
38
+ getHiddenProperties: () => [pivotField],
39
+ };
40
+
41
+ // TODO(wittjosiah): Refactor ProjectionModel to be an interface that we can fulfill.
42
+ return stub as unknown as ProjectionModel;
43
+ }, [kanban.arrangement?.columns, kanban.spec.pivotField]);
44
+ };