@dxos/plugin-kanban 0.8.4-main.f9ba587 → 0.8.4-main.fcc0d83b33

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 (238) hide show
  1. package/dist/lib/browser/blueprints/index.mjs +27 -0
  2. package/dist/lib/browser/blueprints/index.mjs.map +7 -0
  3. package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
  4. package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +7 -0
  5. package/dist/lib/browser/chunk-T32TEM55.mjs +105 -0
  6. package/dist/lib/browser/chunk-T32TEM55.mjs.map +7 -0
  7. package/dist/lib/browser/delete-card-7OSCNCW2.mjs +24 -0
  8. package/dist/lib/browser/delete-card-7OSCNCW2.mjs.map +7 -0
  9. package/dist/lib/browser/delete-card-field-NSB2RE3Z.mjs +42 -0
  10. package/dist/lib/browser/delete-card-field-NSB2RE3Z.mjs.map +7 -0
  11. package/dist/lib/browser/index.mjs +76 -95
  12. package/dist/lib/browser/index.mjs.map +4 -4
  13. package/dist/lib/browser/meta.json +1 -1
  14. package/dist/lib/browser/operations/index.mjs +13 -0
  15. package/dist/lib/browser/operations/index.mjs.map +7 -0
  16. package/dist/lib/browser/restore-card-FO3WERIE.mjs +21 -0
  17. package/dist/lib/browser/restore-card-FO3WERIE.mjs.map +7 -0
  18. package/dist/lib/browser/restore-card-field-U5XYEEOW.mjs +40 -0
  19. package/dist/lib/browser/restore-card-field-U5XYEEOW.mjs.map +7 -0
  20. package/dist/lib/browser/translations.mjs +44 -0
  21. package/dist/lib/browser/translations.mjs.map +7 -0
  22. package/dist/lib/browser/types/index.mjs +154 -7
  23. package/dist/lib/browser/types/index.mjs.map +4 -4
  24. package/dist/lib/node-esm/blueprints/index.mjs +28 -0
  25. package/dist/lib/node-esm/blueprints/index.mjs.map +7 -0
  26. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
  27. package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +7 -0
  28. package/dist/lib/node-esm/chunk-W2RNFBMZ.mjs +106 -0
  29. package/dist/lib/node-esm/chunk-W2RNFBMZ.mjs.map +7 -0
  30. package/dist/lib/node-esm/delete-card-ZIREL6HN.mjs +25 -0
  31. package/dist/lib/node-esm/delete-card-ZIREL6HN.mjs.map +7 -0
  32. package/dist/lib/node-esm/delete-card-field-IPTEGVPP.mjs +43 -0
  33. package/dist/lib/node-esm/delete-card-field-IPTEGVPP.mjs.map +7 -0
  34. package/dist/lib/node-esm/index.mjs +76 -95
  35. package/dist/lib/node-esm/index.mjs.map +4 -4
  36. package/dist/lib/node-esm/meta.json +1 -1
  37. package/dist/lib/node-esm/operations/index.mjs +14 -0
  38. package/dist/lib/node-esm/operations/index.mjs.map +7 -0
  39. package/dist/lib/node-esm/restore-card-WJJ4YB7K.mjs +22 -0
  40. package/dist/lib/node-esm/restore-card-WJJ4YB7K.mjs.map +7 -0
  41. package/dist/lib/node-esm/restore-card-field-L24WJXAW.mjs +41 -0
  42. package/dist/lib/node-esm/restore-card-field-L24WJXAW.mjs.map +7 -0
  43. package/dist/lib/node-esm/translations.mjs +45 -0
  44. package/dist/lib/node-esm/translations.mjs.map +7 -0
  45. package/dist/lib/node-esm/types/index.mjs +154 -7
  46. package/dist/lib/node-esm/types/index.mjs.map +4 -4
  47. package/dist/types/src/KanbanPlugin.d.ts +2 -1
  48. package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
  49. package/dist/types/src/blueprints/index.d.ts +2 -0
  50. package/dist/types/src/blueprints/index.d.ts.map +1 -0
  51. package/dist/types/src/blueprints/kanban-blueprint.d.ts +4 -0
  52. package/dist/types/src/blueprints/kanban-blueprint.d.ts.map +1 -0
  53. package/dist/types/src/capabilities/artifact-definition.d.ts +3 -2
  54. package/dist/types/src/capabilities/artifact-definition.d.ts.map +1 -1
  55. package/dist/types/src/capabilities/blueprint-definition.d.ts +5 -0
  56. package/dist/types/src/capabilities/blueprint-definition.d.ts.map +1 -0
  57. package/dist/types/src/capabilities/index.d.ts +7 -3
  58. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  59. package/dist/types/src/capabilities/migrations.d.ts +5 -0
  60. package/dist/types/src/capabilities/migrations.d.ts.map +1 -0
  61. package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
  62. package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
  63. package/dist/types/src/capabilities/react-surface.d.ts +3 -2
  64. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  65. package/dist/types/src/capabilities/undo-mappings.d.ts +5 -0
  66. package/dist/types/src/capabilities/undo-mappings.d.ts.map +1 -0
  67. package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts +57 -0
  68. package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts.map +1 -0
  69. package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts +72 -0
  70. package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts.map +1 -0
  71. package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts +10 -0
  72. package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts.map +1 -0
  73. package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts +9 -0
  74. package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts.map +1 -0
  75. package/dist/types/src/components/KanbanBoard/index.d.ts +2 -0
  76. package/dist/types/src/components/KanbanBoard/index.d.ts.map +1 -0
  77. package/dist/types/src/components/index.d.ts +1 -2
  78. package/dist/types/src/components/index.d.ts.map +1 -1
  79. package/dist/types/src/containers/KanbanContainer/KanbanContainer.d.ts +6 -0
  80. package/dist/types/src/containers/KanbanContainer/KanbanContainer.d.ts.map +1 -0
  81. package/dist/types/src/containers/KanbanContainer/KanbanContainer.stories.d.ts +79 -0
  82. package/dist/types/src/containers/KanbanContainer/KanbanContainer.stories.d.ts.map +1 -0
  83. package/dist/types/src/containers/KanbanContainer/index.d.ts +2 -0
  84. package/dist/types/src/containers/KanbanContainer/index.d.ts.map +1 -0
  85. package/dist/types/src/containers/KanbanSettings/KanbanSettings.d.ts +13 -0
  86. package/dist/types/src/containers/KanbanSettings/KanbanSettings.d.ts.map +1 -0
  87. package/dist/types/src/containers/KanbanSettings/index.d.ts +2 -0
  88. package/dist/types/src/containers/KanbanSettings/index.d.ts.map +1 -0
  89. package/dist/types/src/containers/index.d.ts +4 -0
  90. package/dist/types/src/containers/index.d.ts.map +1 -0
  91. package/dist/types/src/hooks/index.d.ts +7 -0
  92. package/dist/types/src/hooks/index.d.ts.map +1 -0
  93. package/dist/types/src/hooks/useEchoChangeCallback.d.ts +13 -0
  94. package/dist/types/src/hooks/useEchoChangeCallback.d.ts.map +1 -0
  95. package/dist/types/src/hooks/useItemsProjection.d.ts +10 -0
  96. package/dist/types/src/hooks/useItemsProjection.d.ts.map +1 -0
  97. package/dist/types/src/hooks/useKanbanBoardModel.browser.test.d.ts +2 -0
  98. package/dist/types/src/hooks/useKanbanBoardModel.browser.test.d.ts.map +1 -0
  99. package/dist/types/src/hooks/useKanbanBoardModel.d.ts +16 -0
  100. package/dist/types/src/hooks/useKanbanBoardModel.d.ts.map +1 -0
  101. package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts +22 -0
  102. package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts.map +1 -0
  103. package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts +19 -0
  104. package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts.map +1 -0
  105. package/dist/types/src/hooks/useProjectionModel.d.ts +15 -0
  106. package/dist/types/src/hooks/useProjectionModel.d.ts.map +1 -0
  107. package/dist/types/src/meta.d.ts +2 -3
  108. package/dist/types/src/meta.d.ts.map +1 -1
  109. package/dist/types/src/operations/definitions.d.ts +52 -0
  110. package/dist/types/src/operations/definitions.d.ts.map +1 -0
  111. package/dist/types/src/operations/delete-card-field.d.ts +5 -0
  112. package/dist/types/src/operations/delete-card-field.d.ts.map +1 -0
  113. package/dist/types/src/operations/delete-card.d.ts +5 -0
  114. package/dist/types/src/operations/delete-card.d.ts.map +1 -0
  115. package/dist/types/src/operations/index.d.ts +4 -0
  116. package/dist/types/src/operations/index.d.ts.map +1 -0
  117. package/dist/types/src/operations/restore-card-field.d.ts +5 -0
  118. package/dist/types/src/operations/restore-card-field.d.ts.map +1 -0
  119. package/dist/types/src/operations/restore-card.d.ts +5 -0
  120. package/dist/types/src/operations/restore-card.d.ts.map +1 -0
  121. package/dist/types/src/playwright/board-manager.d.ts +5 -0
  122. package/dist/types/src/playwright/board-manager.d.ts.map +1 -0
  123. package/dist/types/src/playwright/playwright.config.d.ts +3 -0
  124. package/dist/types/src/playwright/playwright.config.d.ts.map +1 -0
  125. package/dist/types/src/playwright/smoke.spec.d.ts +2 -0
  126. package/dist/types/src/playwright/smoke.spec.d.ts.map +1 -0
  127. package/dist/types/src/testing/KanbanCardTileSimple.d.ts +7 -0
  128. package/dist/types/src/testing/KanbanCardTileSimple.d.ts.map +1 -0
  129. package/dist/types/src/testing/index.d.ts +1 -2
  130. package/dist/types/src/testing/index.d.ts.map +1 -1
  131. package/dist/types/src/translations.d.ts +50 -22
  132. package/dist/types/src/translations.d.ts.map +1 -1
  133. package/dist/types/src/types/Kanban.d.ts +109 -0
  134. package/dist/types/src/types/Kanban.d.ts.map +1 -0
  135. package/dist/types/src/types/constants.d.ts +6 -0
  136. package/dist/types/src/types/constants.d.ts.map +1 -0
  137. package/dist/types/src/types/index.d.ts +3 -1
  138. package/dist/types/src/types/index.d.ts.map +1 -1
  139. package/dist/types/src/types/migrations.test.d.ts +2 -0
  140. package/dist/types/src/types/migrations.test.d.ts.map +1 -0
  141. package/dist/types/src/types/schema.d.ts +18 -53
  142. package/dist/types/src/types/schema.d.ts.map +1 -1
  143. package/dist/types/src/types/types.d.ts +26 -9
  144. package/dist/types/src/types/types.d.ts.map +1 -1
  145. package/dist/types/src/util/arrangement.d.ts +72 -0
  146. package/dist/types/src/util/arrangement.d.ts.map +1 -0
  147. package/dist/types/src/util/arrangement.test.d.ts +2 -0
  148. package/dist/types/src/util/arrangement.test.d.ts.map +1 -0
  149. package/dist/types/src/util/index.d.ts +2 -0
  150. package/dist/types/src/util/index.d.ts.map +1 -0
  151. package/dist/types/tsconfig.tsbuildinfo +1 -1
  152. package/package.json +89 -54
  153. package/src/KanbanPlugin.tsx +55 -56
  154. package/src/blueprints/index.ts +5 -0
  155. package/src/blueprints/kanban-blueprint.ts +27 -0
  156. package/src/capabilities/artifact-definition.ts +123 -108
  157. package/src/capabilities/blueprint-definition.ts +17 -0
  158. package/src/capabilities/index.ts +10 -4
  159. package/src/capabilities/migrations.ts +35 -0
  160. package/src/capabilities/operation-handler.ts +14 -0
  161. package/src/capabilities/react-surface.tsx +79 -65
  162. package/src/capabilities/undo-mappings.ts +34 -0
  163. package/src/components/KanbanBoard/KanbanBoard.stories.tsx +146 -0
  164. package/src/components/KanbanBoard/KanbanBoard.tsx +198 -0
  165. package/src/components/KanbanBoard/KanbanCard.tsx +98 -0
  166. package/src/components/KanbanBoard/KanbanColumn.tsx +71 -0
  167. package/src/components/KanbanBoard/index.ts +5 -0
  168. package/src/components/index.ts +1 -2
  169. package/src/containers/KanbanContainer/KanbanContainer.stories.tsx +275 -0
  170. package/src/containers/KanbanContainer/KanbanContainer.tsx +178 -0
  171. package/src/containers/KanbanContainer/index.ts +5 -0
  172. package/src/containers/KanbanSettings/KanbanSettings.tsx +94 -0
  173. package/src/containers/KanbanSettings/index.ts +5 -0
  174. package/src/containers/index.ts +8 -0
  175. package/src/hooks/index.ts +10 -0
  176. package/src/hooks/useEchoChangeCallback.ts +30 -0
  177. package/src/hooks/useItemsProjection.ts +44 -0
  178. package/src/hooks/useKanbanBoardModel.browser.test.ts +235 -0
  179. package/src/hooks/useKanbanBoardModel.ts +157 -0
  180. package/src/hooks/useKanbanColumnEventHandler.ts +106 -0
  181. package/src/hooks/useKanbanItemEventHandler.ts +133 -0
  182. package/src/hooks/useProjectionModel.ts +58 -0
  183. package/src/meta.ts +9 -7
  184. package/src/operations/definitions.ts +63 -0
  185. package/src/operations/delete-card-field.ts +47 -0
  186. package/src/operations/delete-card.ts +23 -0
  187. package/src/operations/index.ts +12 -0
  188. package/src/operations/restore-card-field.ts +41 -0
  189. package/src/operations/restore-card.ts +21 -0
  190. package/src/playwright/board-manager.ts +13 -0
  191. package/src/playwright/playwright.config.ts +19 -0
  192. package/src/playwright/smoke.spec.ts +107 -0
  193. package/src/testing/KanbanCardTileSimple.tsx +82 -0
  194. package/src/testing/index.ts +2 -3
  195. package/src/translations.ts +28 -20
  196. package/src/types/Kanban.ts +151 -0
  197. package/src/types/constants.ts +9 -0
  198. package/src/types/index.ts +3 -1
  199. package/src/types/migrations.test.ts +83 -0
  200. package/src/types/schema.ts +33 -46
  201. package/src/types/types.ts +30 -10
  202. package/src/util/arrangement.test.ts +218 -0
  203. package/src/util/arrangement.ts +177 -0
  204. package/src/util/index.ts +5 -0
  205. package/dist/lib/browser/artifact-definition-H5MOHOV5.mjs +0 -178
  206. package/dist/lib/browser/artifact-definition-H5MOHOV5.mjs.map +0 -7
  207. package/dist/lib/browser/chunk-76UNDUHS.mjs +0 -88
  208. package/dist/lib/browser/chunk-76UNDUHS.mjs.map +0 -7
  209. package/dist/lib/browser/intent-resolver-VJSWAUZD.mjs +0 -297
  210. package/dist/lib/browser/intent-resolver-VJSWAUZD.mjs.map +0 -7
  211. package/dist/lib/browser/react-surface-WMVUSDVR.mjs +0 -305
  212. package/dist/lib/browser/react-surface-WMVUSDVR.mjs.map +0 -7
  213. package/dist/lib/node-esm/artifact-definition-FDRLWGY5.mjs +0 -179
  214. package/dist/lib/node-esm/artifact-definition-FDRLWGY5.mjs.map +0 -7
  215. package/dist/lib/node-esm/chunk-B3AVV7SF.mjs +0 -90
  216. package/dist/lib/node-esm/chunk-B3AVV7SF.mjs.map +0 -7
  217. package/dist/lib/node-esm/intent-resolver-3ZSJN2NS.mjs +0 -298
  218. package/dist/lib/node-esm/intent-resolver-3ZSJN2NS.mjs.map +0 -7
  219. package/dist/lib/node-esm/react-surface-JKYB6SAC.mjs +0 -306
  220. package/dist/lib/node-esm/react-surface-JKYB6SAC.mjs.map +0 -7
  221. package/dist/types/src/capabilities/intent-resolver.d.ts +0 -4
  222. package/dist/types/src/capabilities/intent-resolver.d.ts.map +0 -1
  223. package/dist/types/src/components/KanbanContainer.d.ts +0 -7
  224. package/dist/types/src/components/KanbanContainer.d.ts.map +0 -1
  225. package/dist/types/src/components/KanbanContainer.stories.d.ts +0 -10
  226. package/dist/types/src/components/KanbanContainer.stories.d.ts.map +0 -1
  227. package/dist/types/src/components/KanbanViewEditor.d.ts +0 -8
  228. package/dist/types/src/components/KanbanViewEditor.d.ts.map +0 -1
  229. package/dist/types/src/testing/initialize-kanban.d.ts +0 -17
  230. package/dist/types/src/testing/initialize-kanban.d.ts.map +0 -1
  231. package/dist/types/src/testing/kanban-manager.d.ts +0 -7
  232. package/dist/types/src/testing/kanban-manager.d.ts.map +0 -1
  233. package/src/capabilities/intent-resolver.ts +0 -71
  234. package/src/components/KanbanContainer.stories.tsx +0 -191
  235. package/src/components/KanbanContainer.tsx +0 -95
  236. package/src/components/KanbanViewEditor.tsx +0 -104
  237. package/src/testing/initialize-kanban.ts +0 -128
  238. package/src/testing/kanban-manager.ts +0 -13
@@ -0,0 +1,35 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import * as Effect from 'effect/Effect';
6
+
7
+ import { Capability } from '@dxos/app-framework';
8
+ import { defineObjectMigration } from '@dxos/client/echo';
9
+ import { ClientCapabilities } from '@dxos/plugin-client/types';
10
+
11
+ import { Kanban } from '#types';
12
+
13
+ /**
14
+ * v0.1.0 → v0.2.0: nests the existing `view` ref under a `spec: { kind: 'view', view }`
15
+ * discriminated union. v0.2.0 introduced an items-variant for externally-synced kanbans;
16
+ * pre-migration objects were all view-based by construction.
17
+ */
18
+ const migrations = [
19
+ defineObjectMigration({
20
+ from: Kanban.KanbanV1,
21
+ to: Kanban.Kanban,
22
+ transform: async (from) => ({
23
+ name: from.name,
24
+ arrangement: from.arrangement,
25
+ spec: { kind: 'view' as const, view: from.view },
26
+ }),
27
+ onMigration: async () => {},
28
+ }),
29
+ ];
30
+
31
+ export default Capability.makeModule(
32
+ Effect.fnUntraced(function* () {
33
+ return Capability.contributes(ClientCapabilities.Migration, migrations);
34
+ }),
35
+ );
@@ -0,0 +1,14 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Effect from 'effect/Effect';
4
+
5
+ import { Capabilities, Capability } from '@dxos/app-framework';
6
+ import type { OperationHandlerSet } from '@dxos/compute';
7
+
8
+ import { KanbanOperationHandlerSet } from '#operations';
9
+
10
+ export default Capability.makeModule<OperationHandlerSet.OperationHandlerSet>(
11
+ Effect.fnUntraced(function* () {
12
+ return Capability.contributes(Capabilities.OperationHandler, KanbanOperationHandlerSet);
13
+ }),
14
+ );
@@ -2,80 +2,94 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { type Schema } from 'effect';
5
+ import * as Effect from 'effect/Effect';
6
+ import type * as Schema from 'effect/Schema';
7
+ import * as SchemaAST from 'effect/SchemaAST';
6
8
  import React, { useMemo } from 'react';
7
9
 
8
- import { Capabilities, contributes, createSurface, useCapabilities } from '@dxos/app-framework';
9
- import { Type } from '@dxos/echo';
10
+ import { Capabilities, Capability } from '@dxos/app-framework';
11
+ import { Surface } from '@dxos/app-framework/ui';
12
+ import { AppSurface } from '@dxos/app-toolkit/ui';
13
+ import { Database, JsonSchema, Obj } from '@dxos/echo';
14
+ import { type Collection } from '@dxos/echo';
10
15
  import { findAnnotation } from '@dxos/effect';
11
- import { ClientCapabilities } from '@dxos/plugin-client';
12
- import { getSpace, isSpace, type Space } from '@dxos/react-client/echo';
13
- import { type InputProps, SelectInput, useFormValues } from '@dxos/react-ui-form';
14
- import { type KanbanType } from '@dxos/react-ui-kanban';
15
- import { type DataType } from '@dxos/schema';
16
+ import { type FormFieldComponentProps, SelectField, useFormValues } from '@dxos/react-ui-form';
16
17
 
17
- import { KanbanContainer, KanbanViewEditor } from '../components';
18
- import { meta } from '../meta';
19
- import { isKanban, PivotColumnAnnotationId } from '../types';
18
+ import { KanbanContainer, KanbanSettings } from '#containers';
19
+ import { Kanban, PivotColumnAnnotationId } from '#types';
20
20
 
21
- export default () =>
22
- contributes(Capabilities.ReactSurface, [
23
- createSurface({
24
- id: meta.id,
25
- role: ['article', 'section'],
26
- filter: (data): data is { subject: KanbanType } => isKanban(data.subject),
27
- component: ({ data, role }) => <KanbanContainer kanban={data.subject} role={role} />,
28
- }),
29
- createSurface({
30
- id: `${meta.id}/object-settings`,
31
- role: 'object-settings',
32
- filter: (data): data is { subject: KanbanType } => isKanban(data.subject),
33
- component: ({ data }) => <KanbanViewEditor kanban={data.subject} />,
34
- }),
35
- createSurface({
36
- id: `${meta.id}/create-initial-schema-form-[pivot-column]`,
37
- role: 'form-input',
38
- filter: (
39
- data,
40
- ): data is { prop: string; schema: Schema.Schema<any>; target: Space | DataType.Collection | undefined } => {
41
- const annotation = findAnnotation<boolean>((data.schema as Schema.Schema.All).ast, PivotColumnAnnotationId);
42
- return !!annotation;
43
- },
44
- component: ({ data: { target }, ...inputProps }) => {
45
- const props = inputProps as any as InputProps;
46
- const space = isSpace(target) ? target : getSpace(target);
47
- if (!space) {
48
- return null;
49
- }
50
- const { typename } = useFormValues();
51
- // TODO(wittjosiah): Unify this schema lookup.
52
- const schemaWhitelists = useCapabilities(ClientCapabilities.SchemaWhiteList);
53
- const staticSchema = schemaWhitelists.flat().find((schema) => Type.getTypename(schema) === typename);
54
- const [selectedSchema] = space?.db.schemaRegistry.query({ typename }).runSync();
21
+ export default Capability.makeModule(() =>
22
+ Effect.succeed(
23
+ Capability.contributes(Capabilities.ReactSurface, [
24
+ Surface.create({
25
+ id: 'root',
26
+ // TODO(wittjosiah): Split into multiple surfaces if this filter proves too strict for non-article roles.
27
+ filter: AppSurface.oneOf(
28
+ AppSurface.object(AppSurface.Article, Kanban.Kanban),
29
+ AppSurface.object(AppSurface.Section, Kanban.Kanban),
30
+ ),
31
+ component: ({ data, role }) => <KanbanContainer role={role} subject={data.subject} />,
32
+ }),
33
+ Surface.create({
34
+ id: 'object-properties',
35
+ position: 'hoist',
36
+ filter: AppSurface.object(AppSurface.ObjectProperties, Kanban.Kanban),
37
+ component: ({ data }) => <KanbanSettings subject={data.subject} />,
38
+ }),
39
+ Surface.create({
40
+ id: 'create-initial-schema-form-[pivot-column]',
41
+ role: 'form-input',
42
+ filter: (
43
+ data,
44
+ ): data is {
45
+ prop: string;
46
+ schema: Schema.Schema<any>;
47
+ target: Database.Database | Collection.Collection | undefined;
48
+ fieldPropertyAst?: SchemaAST.AST;
49
+ } => {
50
+ const annotation = findAnnotation<boolean>((data.schema as Schema.Schema.All).ast, PivotColumnAnnotationId);
51
+ return !!annotation;
52
+ },
53
+ component: ({ data: { target, fieldPropertyAst }, ...inputProps }) => {
54
+ const ast = fieldPropertyAst;
55
+ if (!ast) {
56
+ return null;
57
+ }
55
58
 
56
- const singleSelectColumns = useMemo(() => {
57
- const properties = staticSchema
58
- ? Type.toJsonSchema(staticSchema).properties
59
- : selectedSchema?.jsonSchema?.properties;
60
- if (!properties) {
61
- return [];
59
+ const props = { ...inputProps, type: ast } as any as FormFieldComponentProps;
60
+ const db = Database.isDatabase(target) ? target : target && Obj.getDatabase(target);
61
+ if (!db) {
62
+ return null;
62
63
  }
63
64
 
64
- const columns = Object.entries(properties).reduce<string[]>((acc, [key, value]) => {
65
- if (typeof value === 'object' && value?.format === 'single-select') {
66
- acc.push(key);
65
+ const { typename } = useFormValues('KanbanForm');
66
+ const [selectedSchema] = useMemo(
67
+ () => db.schemaRegistry.query({ location: ['database', 'runtime'], typename }).runSync(),
68
+ [db, typename],
69
+ );
70
+ const singleSelectColumns = useMemo(() => {
71
+ const properties = JsonSchema.toJsonSchema(selectedSchema).properties;
72
+ if (!properties) {
73
+ return [];
67
74
  }
68
- return acc;
69
- }, []);
70
75
 
71
- return columns;
72
- }, [selectedSchema?.jsonSchema, staticSchema]);
76
+ const columns = Object.entries(properties).reduce<string[]>((acc, [key, value]) => {
77
+ if (typeof value === 'object' && value?.format === 'single-select') {
78
+ acc.push(key);
79
+ }
80
+ return acc;
81
+ }, []);
82
+
83
+ return columns;
84
+ }, [selectedSchema]);
73
85
 
74
- if (!typename) {
75
- return null;
76
- }
86
+ if (!typename) {
87
+ return null;
88
+ }
77
89
 
78
- return <SelectInput {...props} options={singleSelectColumns.map((column) => ({ value: column }))} />;
79
- },
80
- }),
81
- ]);
90
+ return <SelectField {...props} options={singleSelectColumns.map((column) => ({ value: column }))} />;
91
+ },
92
+ }),
93
+ ]),
94
+ ),
95
+ );
@@ -0,0 +1,34 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Effect from 'effect/Effect';
4
+
5
+ import { Capabilities, Capability, UndoMapping } from '@dxos/app-framework';
6
+
7
+ import { meta } from '#meta';
8
+ import { KanbanOperation } from '#operations';
9
+
10
+ export default Capability.makeModule(() =>
11
+ Effect.succeed(
12
+ Capability.contributes(Capabilities.UndoMapping, [
13
+ UndoMapping.make({
14
+ operation: KanbanOperation.DeleteCardField,
15
+ inverse: KanbanOperation.RestoreCardField,
16
+ deriveContext: (input, output) => ({
17
+ view: input.view,
18
+ field: output.field,
19
+ props: output.props,
20
+ index: output.index,
21
+ }),
22
+ message: ['card-field-deleted.label', { ns: meta.id }],
23
+ }),
24
+ UndoMapping.make({
25
+ operation: KanbanOperation.DeleteCard,
26
+ inverse: KanbanOperation.RestoreCard,
27
+ deriveContext: (_input, output) => ({
28
+ card: output.card,
29
+ }),
30
+ message: ['card-deleted.label', { ns: meta.id }],
31
+ }),
32
+ ]),
33
+ ),
34
+ );
@@ -0,0 +1,146 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { Atom, RegistryContext } from '@effect-atom/atom-react';
6
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
7
+ import React, { useCallback, useContext, useEffect, useMemo, useState } from 'react';
8
+
9
+ import { Filter, JsonSchema, Obj, Query } from '@dxos/echo';
10
+ import { type View } from '@dxos/echo';
11
+ import { random } from '@dxos/random';
12
+ import { withMosaic } from '@dxos/react-ui-mosaic/testing';
13
+ import { Loading, withLayout, withTheme } from '@dxos/react-ui/testing';
14
+ import { ProjectionModel, ViewModel, createEchoChangeCallback } from '@dxos/schema';
15
+ import { withRegistry } from '@dxos/storybook-utils';
16
+ import { Organization } from '@dxos/types';
17
+
18
+ import { createEchoChangeCallback as createKanbanChangeCallback } from '#hooks';
19
+ import { KanbanCardTileSimple } from '#testing';
20
+ import { translations } from '#translations';
21
+ import { Kanban } from '#types';
22
+
23
+ import { KanbanBoard } from './KanbanBoard';
24
+
25
+ random.seed(1);
26
+
27
+ const createOrg = () => ({
28
+ name: random.commerce.productName(),
29
+ description: random.lorem.sentence(),
30
+ image: random.image.url(),
31
+ website: random.internet.url(),
32
+ status: random.helpers.arrayElement(Organization.StatusOptions).id as Organization.Organization['status'],
33
+ });
34
+
35
+ /**
36
+ * In-memory Kanban board: View + ProjectionModel + Kanban + items, no plugin manager or Space.
37
+ * Similar to react-ui-kanban Kanban.stories.tsx.
38
+ */
39
+ const DefaultStory = () => {
40
+ const registry = useContext(RegistryContext);
41
+ const items = useMemo(() => Atom.make<Obj.Unknown[]>([]), []);
42
+ const [state, setState] = useState<{
43
+ view: View.View;
44
+ kanban: Kanban.Kanban;
45
+ projection: ProjectionModel;
46
+ change: ReturnType<typeof createKanbanChangeCallback>;
47
+ }>();
48
+
49
+ useEffect(() => {
50
+ const view = ViewModel.make({
51
+ query: Query.select(Filter.typename(Organization.Organization.typename)),
52
+ jsonSchema: JsonSchema.toJsonSchema(Organization.Organization),
53
+ pivotFieldName: 'status',
54
+ });
55
+ const kanban = Kanban.make({ view });
56
+ const change = createKanbanChangeCallback(kanban);
57
+ const projection = new ProjectionModel({
58
+ registry,
59
+ view,
60
+ baseSchema: JsonSchema.toJsonSchema(Organization.Organization),
61
+ change: createEchoChangeCallback(view),
62
+ });
63
+ projection.normalizeView();
64
+
65
+ const statuses = Organization.StatusOptions.map((o) => o.id);
66
+ const initialItems = Array.from({ length: 12 }, () =>
67
+ Obj.make(Organization.Organization, {
68
+ ...createOrg(),
69
+ status: random.helpers.arrayElement(statuses) as Organization.Organization['status'],
70
+ }),
71
+ );
72
+
73
+ setState({ view, kanban, projection, change });
74
+ registry.set(items, initialItems);
75
+ }, [registry, items]);
76
+
77
+ const columnFieldPath =
78
+ state?.projection.tryGetFieldProjection(state.projection.getFieldId('status') ?? '')?.props.property ?? 'status';
79
+
80
+ const handleCardAdd = useCallback(
81
+ (columnValue: string | undefined) => {
82
+ if (!state || !columnFieldPath || !registry) {
83
+ return undefined;
84
+ }
85
+ const card = Obj.make(Organization.Organization, {
86
+ ...createOrg(),
87
+ ...(columnFieldPath ? { [columnFieldPath]: columnValue } : {}),
88
+ });
89
+ const current = registry.get(items) ?? [];
90
+ registry.set(items, [...current, card]);
91
+ return card.id;
92
+ },
93
+ [state, columnFieldPath, registry, items],
94
+ );
95
+
96
+ const handleCardRemove = useCallback(
97
+ (card: Obj.Unknown) => {
98
+ if (!registry) {
99
+ return;
100
+ }
101
+ const current = registry.get(items) ?? [];
102
+ registry.set(
103
+ items,
104
+ current.filter((i) => i.id !== card.id),
105
+ );
106
+ },
107
+ [registry, items],
108
+ );
109
+
110
+ if (!state) {
111
+ return <Loading />;
112
+ }
113
+
114
+ return (
115
+ <KanbanBoard.Root
116
+ kanban={state.kanban}
117
+ projection={state.projection}
118
+ items={items}
119
+ itemTile={KanbanCardTileSimple}
120
+ change={state.change}
121
+ onCardAdd={handleCardAdd}
122
+ onCardRemove={handleCardRemove}
123
+ >
124
+ <KanbanBoard.Content />
125
+ </KanbanBoard.Root>
126
+ );
127
+ };
128
+
129
+ const meta = {
130
+ title: 'plugins/plugin-kanban/components/KanbanBoard',
131
+ component: DefaultStory,
132
+ decorators: [withTheme(), withLayout({ layout: 'fullscreen' }), withMosaic(), withRegistry],
133
+ parameters: {
134
+ layout: 'fullscreen',
135
+ translations,
136
+ },
137
+ } satisfies Meta<typeof DefaultStory>;
138
+
139
+ export default meta;
140
+
141
+ type Story = StoryObj<typeof meta>;
142
+
143
+ /**
144
+ * In-memory board with Echo-shaped objects. No plugin manager, client, or Space.
145
+ */
146
+ export const Default: Story = {};
@@ -0,0 +1,198 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type Atom, RegistryContext } from '@effect-atom/atom-react';
6
+ import { createContext } from '@radix-ui/react-context';
7
+ import React, {
8
+ type ComponentPropsWithoutRef,
9
+ type ComponentType,
10
+ type PropsWithChildren,
11
+ useCallback,
12
+ useContext,
13
+ useMemo,
14
+ } from 'react';
15
+
16
+ import { Obj } from '@dxos/echo';
17
+ import { useTranslation } from '@dxos/react-ui';
18
+ import { Board, useBoard } from '@dxos/react-ui-mosaic';
19
+ import type { ProjectionModel } from '@dxos/schema';
20
+ import { composable, composableProps } from '@dxos/ui-theme';
21
+
22
+ import { useKanbanBoardModel, useKanbanColumnEventHandler } from '#hooks';
23
+ import { meta } from '#meta';
24
+ import { type Kanban, type KanbanChangeCallback, UNCATEGORIZED_ATTRIBUTES, UNCATEGORIZED_VALUE } from '#types';
25
+
26
+ import { KanbanCard, type KanbanCardProps } from './KanbanCard';
27
+ import { KanbanColumn, type KanbanColumnProps } from './KanbanColumn';
28
+
29
+ // TODO(burdon): Rename Kanban.
30
+
31
+ //
32
+ // Context
33
+ //
34
+
35
+ const KANBAN_BOARD_NAME = 'KanbanBoard.Context';
36
+
37
+ /**
38
+ * Context value for the Kanban board.
39
+ * Items are Echo objects (Obj.Unknown).
40
+ */
41
+ type KanbanBoardContextValue = {
42
+ kanbanId: string;
43
+ projection: ProjectionModel | undefined;
44
+ columnFieldPath: string | undefined;
45
+ change: KanbanChangeCallback<Obj.Unknown>;
46
+ pivotFieldId: string | undefined;
47
+ getPivotAttributes: (columnValue: string) => { title: string; color: string };
48
+ itemTile?: ComponentType<KanbanCardProps>; // TODO(burdon): Prop.
49
+ onCardAdd?: (columnValue: string | undefined) => string | undefined;
50
+ onCardRemove?: (card: Obj.Unknown) => void;
51
+ };
52
+
53
+ const [KanbanBoardContext, useKanbanBoard] = createContext<KanbanBoardContextValue>(KANBAN_BOARD_NAME, {
54
+ kanbanId: 'never',
55
+ projection: undefined,
56
+ columnFieldPath: undefined,
57
+ change: { kanban: () => {}, setItemField: () => {} },
58
+ pivotFieldId: undefined,
59
+ getPivotAttributes: (id: string) =>
60
+ id === UNCATEGORIZED_VALUE ? UNCATEGORIZED_ATTRIBUTES : { title: id, color: 'neutral' },
61
+ itemTile: (() => null) as ComponentType<KanbanCardProps>,
62
+ });
63
+
64
+ //
65
+ // Root
66
+ //
67
+
68
+ const KANBAN_BOARD_ROOT = 'KanbanBoard.Root';
69
+
70
+ type KanbanBoardRootProps = PropsWithChildren<
71
+ {
72
+ kanban: Kanban.Kanban;
73
+ /** Required when providing context; Root derives columnFieldPath, pivotFieldId, getPivotAttributes from kanban + projection. */
74
+ projection: ProjectionModel;
75
+ /** Atom of items (e.g. from AtomQuery for DB, or Atom.make([]) for in-memory). */
76
+ items: Atom.Atom<Obj.Unknown[]>;
77
+ onCardAdd?: (columnValue: string | undefined) => string | undefined;
78
+ onCardRemove?: (card: Obj.Unknown) => void;
79
+ } & Pick<KanbanBoardContextValue, 'change' | 'itemTile'> &
80
+ ComponentPropsWithoutRef<'div'>
81
+ >;
82
+
83
+ export const KanbanBoardRoot = ({
84
+ children,
85
+ kanban,
86
+ projection,
87
+ items,
88
+ change,
89
+ itemTile = KanbanCard,
90
+ onCardAdd,
91
+ onCardRemove,
92
+ }: KanbanBoardRootProps) => {
93
+ const registry = useContext(RegistryContext);
94
+ const { t } = useTranslation(meta.id);
95
+ const model = useKanbanBoardModel(kanban, projection, items, registry);
96
+ const columns = model?.getColumns?.() ?? [];
97
+ const view = kanban?.spec.kind === 'view' ? kanban.spec.view.target : undefined;
98
+ const pivotFieldId = view?.projection?.pivotFieldId;
99
+ const columnFieldPath = useMemo(() => {
100
+ // Items-variant kanbans use the property name itself as the pivot field
101
+ // (no view/projection translation layer).
102
+ if (kanban?.spec.kind === 'items') {
103
+ return kanban.spec.pivotField;
104
+ }
105
+ if (pivotFieldId === undefined || !projection) {
106
+ return undefined;
107
+ }
108
+
109
+ return projection.tryGetFieldProjection(pivotFieldId)?.props.property;
110
+ }, [kanban?.spec, projection, pivotFieldId]);
111
+
112
+ const getPivotAttributes = useCallback<KanbanBoardContextValue['getPivotAttributes']>(
113
+ (columnValue) => {
114
+ if (columnValue === UNCATEGORIZED_VALUE) {
115
+ return UNCATEGORIZED_ATTRIBUTES;
116
+ }
117
+
118
+ const options = projection?.tryGetFieldProjection(pivotFieldId ?? '')?.props.options ?? [];
119
+ const option = options.find((option) => option.id === columnValue);
120
+ return option ?? ({ title: columnValue, color: 'neutral' } as const);
121
+ },
122
+ [projection, pivotFieldId],
123
+ );
124
+
125
+ if (columns.length === 0) {
126
+ return (
127
+ <div role='none' className='flex flex-1 items-center justify-center p-8 text-center text-description'>
128
+ {t('select-pivot.placeholder')}
129
+ </div>
130
+ );
131
+ }
132
+
133
+ return (
134
+ <KanbanBoardContext
135
+ kanbanId={Obj.getDXN(kanban).toString()}
136
+ projection={projection}
137
+ columnFieldPath={columnFieldPath}
138
+ pivotFieldId={pivotFieldId}
139
+ getPivotAttributes={getPivotAttributes}
140
+ itemTile={itemTile}
141
+ change={change}
142
+ onCardAdd={onCardAdd}
143
+ onCardRemove={onCardRemove}
144
+ >
145
+ <Board.Root model={model}>{children}</Board.Root>
146
+ </KanbanBoardContext>
147
+ );
148
+ };
149
+
150
+ KanbanBoardRoot.displayName = KANBAN_BOARD_ROOT;
151
+
152
+ //
153
+ // KanbanBoardContent
154
+ //
155
+
156
+ const KANBAN_BOARD_CONTENT = 'KanbanBoard.Content';
157
+
158
+ type KanbanBoardContentProps = {};
159
+
160
+ export const KanbanBoardContent = composable<HTMLDivElement, KanbanBoardContentProps>((props, forwardedRef) => {
161
+ const { model } = useBoard(KANBAN_BOARD_CONTENT);
162
+ const { kanbanId, projection, pivotFieldId, change } = useKanbanBoard(KANBAN_BOARD_CONTENT);
163
+
164
+ const columnEventHandler = useKanbanColumnEventHandler({
165
+ id: `${kanbanId}-columns`,
166
+ model,
167
+ projection: projection ?? undefined,
168
+ pivotFieldId: pivotFieldId ?? undefined,
169
+ change,
170
+ });
171
+
172
+ return (
173
+ <Board.Content
174
+ {...composableProps(props)}
175
+ ref={forwardedRef}
176
+ id={kanbanId}
177
+ eventHandler={columnEventHandler}
178
+ Tile={KanbanColumn}
179
+ />
180
+ );
181
+ });
182
+
183
+ KanbanBoardContent.displayName = KANBAN_BOARD_CONTENT;
184
+
185
+ //
186
+ // KanbanBoard
187
+ //
188
+
189
+ export const KanbanBoard = {
190
+ Root: KanbanBoardRoot,
191
+ Content: KanbanBoardContent,
192
+ Column: KanbanColumn,
193
+ Card: KanbanCard,
194
+ };
195
+
196
+ export { useKanbanBoard };
197
+
198
+ export type { KanbanBoardRootProps as KanbanBoardProps, KanbanCardProps, KanbanColumnProps };