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

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 (225) 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-A3PBV3S5.mjs +105 -0
  4. package/dist/lib/browser/chunk-A3PBV3S5.mjs.map +7 -0
  5. package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
  6. package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +7 -0
  7. package/dist/lib/browser/delete-card-VPNVIWOA.mjs +32 -0
  8. package/dist/lib/browser/delete-card-VPNVIWOA.mjs.map +7 -0
  9. package/dist/lib/browser/delete-card-field-4HHF2GYX.mjs +50 -0
  10. package/dist/lib/browser/delete-card-field-4HHF2GYX.mjs.map +7 -0
  11. package/dist/lib/browser/index.mjs +101 -84
  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-4GG2RYKR.mjs +29 -0
  17. package/dist/lib/browser/restore-card-4GG2RYKR.mjs.map +7 -0
  18. package/dist/lib/browser/restore-card-field-3T26ACYX.mjs +48 -0
  19. package/dist/lib/browser/restore-card-field-3T26ACYX.mjs.map +7 -0
  20. package/dist/lib/browser/types/index.mjs +94 -7
  21. package/dist/lib/browser/types/index.mjs.map +4 -4
  22. package/dist/lib/node-esm/blueprints/index.mjs +28 -0
  23. package/dist/lib/node-esm/blueprints/index.mjs.map +7 -0
  24. package/dist/lib/node-esm/chunk-6LELYA2G.mjs +106 -0
  25. package/dist/lib/node-esm/chunk-6LELYA2G.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/delete-card-5PW5OMFN.mjs +33 -0
  29. package/dist/lib/node-esm/delete-card-5PW5OMFN.mjs.map +7 -0
  30. package/dist/lib/node-esm/delete-card-field-KPJU2AQ3.mjs +51 -0
  31. package/dist/lib/node-esm/delete-card-field-KPJU2AQ3.mjs.map +7 -0
  32. package/dist/lib/node-esm/index.mjs +101 -84
  33. package/dist/lib/node-esm/index.mjs.map +4 -4
  34. package/dist/lib/node-esm/meta.json +1 -1
  35. package/dist/lib/node-esm/operations/index.mjs +14 -0
  36. package/dist/lib/node-esm/operations/index.mjs.map +7 -0
  37. package/dist/lib/node-esm/restore-card-X2TKMU5A.mjs +30 -0
  38. package/dist/lib/node-esm/restore-card-X2TKMU5A.mjs.map +7 -0
  39. package/dist/lib/node-esm/restore-card-field-IUTL4RTR.mjs +49 -0
  40. package/dist/lib/node-esm/restore-card-field-IUTL4RTR.mjs.map +7 -0
  41. package/dist/lib/node-esm/types/index.mjs +94 -7
  42. package/dist/lib/node-esm/types/index.mjs.map +4 -4
  43. package/dist/types/src/KanbanPlugin.d.ts +2 -1
  44. package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
  45. package/dist/types/src/blueprints/index.d.ts +2 -0
  46. package/dist/types/src/blueprints/index.d.ts.map +1 -0
  47. package/dist/types/src/blueprints/kanban-blueprint.d.ts +4 -0
  48. package/dist/types/src/blueprints/kanban-blueprint.d.ts.map +1 -0
  49. package/dist/types/src/capabilities/artifact-definition.d.ts +3 -2
  50. package/dist/types/src/capabilities/artifact-definition.d.ts.map +1 -1
  51. package/dist/types/src/capabilities/blueprint-definition.d.ts +6 -0
  52. package/dist/types/src/capabilities/blueprint-definition.d.ts.map +1 -0
  53. package/dist/types/src/capabilities/index.d.ts +6 -3
  54. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  55. package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
  56. package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
  57. package/dist/types/src/capabilities/react-surface.d.ts +3 -2
  58. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  59. package/dist/types/src/capabilities/undo-mappings.d.ts +5 -0
  60. package/dist/types/src/capabilities/undo-mappings.d.ts.map +1 -0
  61. package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts +57 -0
  62. package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts.map +1 -0
  63. package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts +72 -0
  64. package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts.map +1 -0
  65. package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts +10 -0
  66. package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts.map +1 -0
  67. package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts +9 -0
  68. package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts.map +1 -0
  69. package/dist/types/src/components/KanbanBoard/index.d.ts +2 -0
  70. package/dist/types/src/components/KanbanBoard/index.d.ts.map +1 -0
  71. package/dist/types/src/components/index.d.ts +1 -2
  72. package/dist/types/src/components/index.d.ts.map +1 -1
  73. package/dist/types/src/containers/KanbanContainer/KanbanContainer.d.ts +6 -0
  74. package/dist/types/src/containers/KanbanContainer/KanbanContainer.d.ts.map +1 -0
  75. package/dist/types/src/containers/KanbanContainer/KanbanContainer.stories.d.ts +79 -0
  76. package/dist/types/src/containers/KanbanContainer/KanbanContainer.stories.d.ts.map +1 -0
  77. package/dist/types/src/containers/KanbanContainer/index.d.ts +2 -0
  78. package/dist/types/src/containers/KanbanContainer/index.d.ts.map +1 -0
  79. package/dist/types/src/containers/KanbanViewEditor/KanbanViewEditor.d.ts +6 -0
  80. package/dist/types/src/containers/KanbanViewEditor/KanbanViewEditor.d.ts.map +1 -0
  81. package/dist/types/src/containers/KanbanViewEditor/index.d.ts +2 -0
  82. package/dist/types/src/containers/KanbanViewEditor/index.d.ts.map +1 -0
  83. package/dist/types/src/containers/index.d.ts +4 -0
  84. package/dist/types/src/containers/index.d.ts.map +1 -0
  85. package/dist/types/src/hooks/index.d.ts +6 -0
  86. package/dist/types/src/hooks/index.d.ts.map +1 -0
  87. package/dist/types/src/hooks/useEchoChangeCallback.d.ts +13 -0
  88. package/dist/types/src/hooks/useEchoChangeCallback.d.ts.map +1 -0
  89. package/dist/types/src/hooks/useKanbanBoardModel.d.ts +16 -0
  90. package/dist/types/src/hooks/useKanbanBoardModel.d.ts.map +1 -0
  91. package/dist/types/src/hooks/useKanbanBoardModel.test.d.ts +2 -0
  92. package/dist/types/src/hooks/useKanbanBoardModel.test.d.ts.map +1 -0
  93. package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts +22 -0
  94. package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts.map +1 -0
  95. package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts +19 -0
  96. package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts.map +1 -0
  97. package/dist/types/src/hooks/useProjectionModel.d.ts +15 -0
  98. package/dist/types/src/hooks/useProjectionModel.d.ts.map +1 -0
  99. package/dist/types/src/meta.d.ts +2 -3
  100. package/dist/types/src/meta.d.ts.map +1 -1
  101. package/dist/types/src/operations/definitions.d.ts +52 -0
  102. package/dist/types/src/operations/definitions.d.ts.map +1 -0
  103. package/dist/types/src/operations/delete-card-field.d.ts +5 -0
  104. package/dist/types/src/operations/delete-card-field.d.ts.map +1 -0
  105. package/dist/types/src/operations/delete-card.d.ts +5 -0
  106. package/dist/types/src/operations/delete-card.d.ts.map +1 -0
  107. package/dist/types/src/operations/index.d.ts +4 -0
  108. package/dist/types/src/operations/index.d.ts.map +1 -0
  109. package/dist/types/src/operations/restore-card-field.d.ts +5 -0
  110. package/dist/types/src/operations/restore-card-field.d.ts.map +1 -0
  111. package/dist/types/src/operations/restore-card.d.ts +5 -0
  112. package/dist/types/src/operations/restore-card.d.ts.map +1 -0
  113. package/dist/types/src/playwright/board-manager.d.ts +5 -0
  114. package/dist/types/src/playwright/board-manager.d.ts.map +1 -0
  115. package/dist/types/src/playwright/playwright.config.d.ts +3 -0
  116. package/dist/types/src/playwright/playwright.config.d.ts.map +1 -0
  117. package/dist/types/src/playwright/smoke.spec.d.ts +2 -0
  118. package/dist/types/src/playwright/smoke.spec.d.ts.map +1 -0
  119. package/dist/types/src/testing/KanbanCardTileSimple.d.ts +7 -0
  120. package/dist/types/src/testing/KanbanCardTileSimple.d.ts.map +1 -0
  121. package/dist/types/src/testing/index.d.ts +1 -2
  122. package/dist/types/src/testing/index.d.ts.map +1 -1
  123. package/dist/types/src/translations.d.ts +50 -22
  124. package/dist/types/src/translations.d.ts.map +1 -1
  125. package/dist/types/src/types/Kanban.d.ts +37 -0
  126. package/dist/types/src/types/Kanban.d.ts.map +1 -0
  127. package/dist/types/src/types/constants.d.ts +6 -0
  128. package/dist/types/src/types/constants.d.ts.map +1 -0
  129. package/dist/types/src/types/index.d.ts +3 -1
  130. package/dist/types/src/types/index.d.ts.map +1 -1
  131. package/dist/types/src/types/schema.d.ts +4 -53
  132. package/dist/types/src/types/schema.d.ts.map +1 -1
  133. package/dist/types/src/types/types.d.ts +26 -9
  134. package/dist/types/src/types/types.d.ts.map +1 -1
  135. package/dist/types/src/util/arrangement.d.ts +68 -0
  136. package/dist/types/src/util/arrangement.d.ts.map +1 -0
  137. package/dist/types/src/util/arrangement.test.d.ts +2 -0
  138. package/dist/types/src/util/arrangement.test.d.ts.map +1 -0
  139. package/dist/types/src/util/index.d.ts +2 -0
  140. package/dist/types/src/util/index.d.ts.map +1 -0
  141. package/dist/types/tsconfig.tsbuildinfo +1 -1
  142. package/package.json +88 -48
  143. package/src/KanbanPlugin.tsx +50 -55
  144. package/src/blueprints/index.ts +5 -0
  145. package/src/blueprints/kanban-blueprint.ts +28 -0
  146. package/src/capabilities/artifact-definition.ts +123 -108
  147. package/src/capabilities/blueprint-definition.ts +17 -0
  148. package/src/capabilities/index.ts +9 -4
  149. package/src/capabilities/operation-handler.ts +14 -0
  150. package/src/capabilities/react-surface.tsx +73 -66
  151. package/src/capabilities/undo-mappings.ts +34 -0
  152. package/src/components/KanbanBoard/KanbanBoard.stories.tsx +142 -0
  153. package/src/components/KanbanBoard/KanbanBoard.tsx +193 -0
  154. package/src/components/KanbanBoard/KanbanCard.tsx +86 -0
  155. package/src/components/KanbanBoard/KanbanColumn.tsx +69 -0
  156. package/src/components/KanbanBoard/index.ts +5 -0
  157. package/src/components/index.ts +1 -2
  158. package/src/containers/KanbanContainer/KanbanContainer.stories.tsx +271 -0
  159. package/src/containers/KanbanContainer/KanbanContainer.tsx +96 -0
  160. package/src/containers/KanbanContainer/index.ts +5 -0
  161. package/src/containers/KanbanViewEditor/KanbanViewEditor.tsx +63 -0
  162. package/src/containers/KanbanViewEditor/index.ts +5 -0
  163. package/src/containers/index.ts +8 -0
  164. package/src/hooks/index.ts +9 -0
  165. package/src/hooks/useEchoChangeCallback.ts +30 -0
  166. package/src/hooks/useKanbanBoardModel.test.ts +235 -0
  167. package/src/hooks/useKanbanBoardModel.ts +144 -0
  168. package/src/hooks/useKanbanColumnEventHandler.ts +106 -0
  169. package/src/hooks/useKanbanItemEventHandler.ts +133 -0
  170. package/src/hooks/useProjectionModel.ts +58 -0
  171. package/src/meta.ts +9 -7
  172. package/src/operations/definitions.ts +63 -0
  173. package/src/operations/delete-card-field.ts +47 -0
  174. package/src/operations/delete-card.ts +23 -0
  175. package/src/operations/index.ts +12 -0
  176. package/src/operations/restore-card-field.ts +41 -0
  177. package/src/operations/restore-card.ts +21 -0
  178. package/src/playwright/board-manager.ts +13 -0
  179. package/src/playwright/playwright.config.ts +19 -0
  180. package/src/playwright/smoke.spec.ts +107 -0
  181. package/src/testing/KanbanCardTileSimple.tsx +82 -0
  182. package/src/testing/index.ts +2 -3
  183. package/src/translations.ts +28 -20
  184. package/src/types/Kanban.ts +71 -0
  185. package/src/types/constants.ts +9 -0
  186. package/src/types/index.ts +3 -1
  187. package/src/types/schema.ts +15 -46
  188. package/src/types/types.ts +30 -10
  189. package/src/util/arrangement.test.ts +208 -0
  190. package/src/util/arrangement.ts +167 -0
  191. package/src/util/index.ts +5 -0
  192. package/dist/lib/browser/artifact-definition-H5MOHOV5.mjs +0 -178
  193. package/dist/lib/browser/artifact-definition-H5MOHOV5.mjs.map +0 -7
  194. package/dist/lib/browser/chunk-76UNDUHS.mjs +0 -88
  195. package/dist/lib/browser/chunk-76UNDUHS.mjs.map +0 -7
  196. package/dist/lib/browser/intent-resolver-VJSWAUZD.mjs +0 -297
  197. package/dist/lib/browser/intent-resolver-VJSWAUZD.mjs.map +0 -7
  198. package/dist/lib/browser/react-surface-WMVUSDVR.mjs +0 -305
  199. package/dist/lib/browser/react-surface-WMVUSDVR.mjs.map +0 -7
  200. package/dist/lib/node-esm/artifact-definition-FDRLWGY5.mjs +0 -179
  201. package/dist/lib/node-esm/artifact-definition-FDRLWGY5.mjs.map +0 -7
  202. package/dist/lib/node-esm/chunk-B3AVV7SF.mjs +0 -90
  203. package/dist/lib/node-esm/chunk-B3AVV7SF.mjs.map +0 -7
  204. package/dist/lib/node-esm/intent-resolver-3ZSJN2NS.mjs +0 -298
  205. package/dist/lib/node-esm/intent-resolver-3ZSJN2NS.mjs.map +0 -7
  206. package/dist/lib/node-esm/react-surface-JKYB6SAC.mjs +0 -306
  207. package/dist/lib/node-esm/react-surface-JKYB6SAC.mjs.map +0 -7
  208. package/dist/types/src/capabilities/intent-resolver.d.ts +0 -4
  209. package/dist/types/src/capabilities/intent-resolver.d.ts.map +0 -1
  210. package/dist/types/src/components/KanbanContainer.d.ts +0 -7
  211. package/dist/types/src/components/KanbanContainer.d.ts.map +0 -1
  212. package/dist/types/src/components/KanbanContainer.stories.d.ts +0 -10
  213. package/dist/types/src/components/KanbanContainer.stories.d.ts.map +0 -1
  214. package/dist/types/src/components/KanbanViewEditor.d.ts +0 -8
  215. package/dist/types/src/components/KanbanViewEditor.d.ts.map +0 -1
  216. package/dist/types/src/testing/initialize-kanban.d.ts +0 -17
  217. package/dist/types/src/testing/initialize-kanban.d.ts.map +0 -1
  218. package/dist/types/src/testing/kanban-manager.d.ts +0 -7
  219. package/dist/types/src/testing/kanban-manager.d.ts.map +0 -1
  220. package/src/capabilities/intent-resolver.ts +0 -71
  221. package/src/components/KanbanContainer.stories.tsx +0 -191
  222. package/src/components/KanbanContainer.tsx +0 -95
  223. package/src/components/KanbanViewEditor.tsx +0 -104
  224. package/src/testing/initialize-kanban.ts +0 -128
  225. package/src/testing/kanban-manager.ts +0 -13
package/src/meta.ts CHANGED
@@ -2,16 +2,18 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { type PluginMeta } from '@dxos/app-framework';
5
+ import { type Plugin } from '@dxos/app-framework';
6
+ import { trim } from '@dxos/util';
6
7
 
7
- export const KANBAN_PLUGIN = 'dxos.org/plugin/kanban';
8
-
9
- export const meta: PluginMeta = {
10
- id: KANBAN_PLUGIN,
8
+ export const meta: Plugin.Meta = {
9
+ id: 'org.dxos.plugin.kanban',
11
10
  name: 'Kanban',
12
- description:
13
- 'Kanban allows you to explore Table data in sorted columns defined by your custom schema. You can use Kanbans to track progress or trigger custom automations when cards are moved from one state to another.',
11
+ description: trim`
12
+ Visual project management using customizable kanban boards to track workflow progress.
13
+ Organize table data into columns, drag and drop items between stages, and trigger automations based on status changes.
14
+ `,
14
15
  icon: 'ph--kanban--regular',
16
+ iconHue: 'green',
15
17
  source: 'https://github.com/dxos/dxos/tree/main/packages/plugins/plugin-kanban',
16
18
  screenshots: ['https://dxos.network/plugin-details-kanban-dark.png'],
17
19
  };
@@ -0,0 +1,63 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Schema from 'effect/Schema';
4
+
5
+ import { Capability } from '@dxos/app-framework';
6
+ import { View } from '@dxos/echo';
7
+ import { Operation } from '@dxos/operation';
8
+
9
+ import { meta } from '#meta';
10
+
11
+ const KANBAN_OPERATION = `${meta.id}.operation`;
12
+
13
+ export const DeleteCardFieldOutput = Schema.Struct({
14
+ field: View.FieldSchema.annotations({ description: 'The deleted field schema.' }),
15
+ props: Schema.Any.annotations({ description: 'The deleted field properties.' }),
16
+ index: Schema.Number.annotations({ description: 'The index the field was at.' }),
17
+ });
18
+
19
+ export type DeleteCardFieldOutput = Schema.Schema.Type<typeof DeleteCardFieldOutput>;
20
+
21
+ export const DeleteCardField = Operation.make({
22
+ meta: { key: `${KANBAN_OPERATION}.delete-card-field`, name: 'Delete Card Field' },
23
+ services: [Capability.Service],
24
+ input: Schema.Struct({
25
+ view: View.View,
26
+ fieldId: Schema.String,
27
+ }),
28
+ output: DeleteCardFieldOutput,
29
+ });
30
+
31
+ export const DeleteCardOutput = Schema.Struct({
32
+ card: Schema.Any.annotations({ description: 'The deleted card.' }),
33
+ });
34
+
35
+ export type DeleteCardOutput = Schema.Schema.Type<typeof DeleteCardOutput>;
36
+
37
+ export const DeleteCard = Operation.make({
38
+ meta: { key: `${KANBAN_OPERATION}.delete-card`, name: 'Delete Card' },
39
+ input: Schema.Struct({
40
+ card: Schema.Any,
41
+ }),
42
+ output: DeleteCardOutput,
43
+ });
44
+
45
+ export const RestoreCardField = Operation.make({
46
+ meta: { key: `${KANBAN_OPERATION}.restore-card-field`, name: 'Restore Card Field' },
47
+ services: [Capability.Service],
48
+ input: Schema.Struct({
49
+ view: View.View.annotations({ description: 'The view to restore the field to.' }),
50
+ field: View.FieldSchema.annotations({ description: 'The field schema to restore.' }),
51
+ props: Schema.Any.annotations({ description: 'The field properties to restore.' }),
52
+ index: Schema.Number.annotations({ description: 'The index to restore the field at.' }),
53
+ }),
54
+ output: Schema.Void,
55
+ });
56
+
57
+ export const RestoreCard = Operation.make({
58
+ meta: { key: `${KANBAN_OPERATION}.restore-card`, name: 'Restore Card' },
59
+ input: Schema.Struct({
60
+ card: Schema.Any.annotations({ description: 'The card to restore.' }),
61
+ }),
62
+ output: Schema.Void,
63
+ });
@@ -0,0 +1,47 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Effect from 'effect/Effect';
4
+
5
+ import { Capabilities, Capability } from '@dxos/app-framework';
6
+ import { JsonSchema, Obj } from '@dxos/echo';
7
+ import { type EchoSchema } from '@dxos/echo/internal';
8
+ import { invariant } from '@dxos/invariant';
9
+ import { Operation } from '@dxos/operation';
10
+ import { ProjectionModel, createEchoChangeCallback, getTypenameFromQuery } from '@dxos/schema';
11
+
12
+ import { DeleteCardField } from './definitions';
13
+
14
+ const handler: Operation.WithHandler<typeof DeleteCardField> = DeleteCardField.pipe(
15
+ Operation.withHandler(
16
+ Effect.fnUntraced(function* ({ view, fieldId }) {
17
+ const registry = yield* Capability.get(Capabilities.AtomRegistry);
18
+ const db = Obj.getDatabase(view);
19
+ invariant(db, 'Database not found');
20
+ const schema = yield* Effect.promise(() =>
21
+ db.schemaRegistry
22
+ .query({
23
+ typename: getTypenameFromQuery(view.query.ast)!,
24
+ location: ['database', 'runtime'],
25
+ })
26
+ .first(),
27
+ );
28
+
29
+ const projection = new ProjectionModel({
30
+ registry,
31
+ view,
32
+ baseSchema: JsonSchema.toJsonSchema(schema),
33
+ change: createEchoChangeCallback(view, schema as EchoSchema),
34
+ });
35
+
36
+ const result = projection.deleteFieldProjection(fieldId);
37
+
38
+ return {
39
+ field: result.deleted.field,
40
+ props: result.deleted.props,
41
+ index: result.index,
42
+ };
43
+ }),
44
+ ),
45
+ );
46
+
47
+ export default handler;
@@ -0,0 +1,23 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Effect from 'effect/Effect';
4
+
5
+ import { Obj } from '@dxos/echo';
6
+ import { invariant } from '@dxos/invariant';
7
+ import { Operation } from '@dxos/operation';
8
+
9
+ import { DeleteCard } from './definitions';
10
+
11
+ const handler: Operation.WithHandler<typeof DeleteCard> = DeleteCard.pipe(
12
+ Operation.withHandler(({ card }) =>
13
+ Effect.sync(() => {
14
+ const db = Obj.getDatabase(card);
15
+ invariant(db);
16
+ db.remove(card);
17
+
18
+ return { card };
19
+ }),
20
+ ),
21
+ );
22
+
23
+ export default handler;
@@ -0,0 +1,12 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import { OperationHandlerSet } from '@dxos/operation';
4
+
5
+ export * as KanbanOperation from './definitions';
6
+
7
+ export const KanbanOperationHandlerSet = OperationHandlerSet.lazy(
8
+ () => import('./delete-card'),
9
+ () => import('./delete-card-field'),
10
+ () => import('./restore-card'),
11
+ () => import('./restore-card-field'),
12
+ );
@@ -0,0 +1,41 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Effect from 'effect/Effect';
4
+
5
+ import { Capabilities, Capability } from '@dxos/app-framework';
6
+ import { JsonSchema, Obj } from '@dxos/echo';
7
+ import { type EchoSchema } from '@dxos/echo/internal';
8
+ import { invariant } from '@dxos/invariant';
9
+ import { Operation } from '@dxos/operation';
10
+ import { ProjectionModel, createEchoChangeCallback, getTypenameFromQuery } from '@dxos/schema';
11
+
12
+ import { RestoreCardField } from './definitions';
13
+
14
+ const handler: Operation.WithHandler<typeof RestoreCardField> = RestoreCardField.pipe(
15
+ Operation.withHandler(
16
+ Effect.fnUntraced(function* ({ view, field, props, index }) {
17
+ const registry = yield* Capability.get(Capabilities.AtomRegistry);
18
+ const db = Obj.getDatabase(view);
19
+ invariant(db, 'Database not found');
20
+ const schema = yield* Effect.promise(() =>
21
+ db.schemaRegistry
22
+ .query({
23
+ typename: getTypenameFromQuery(view.query.ast)!,
24
+ location: ['database', 'runtime'],
25
+ })
26
+ .first(),
27
+ );
28
+
29
+ const projection = new ProjectionModel({
30
+ registry,
31
+ view,
32
+ baseSchema: JsonSchema.toJsonSchema(schema),
33
+ change: createEchoChangeCallback(view, schema as EchoSchema),
34
+ });
35
+
36
+ projection.setFieldProjection({ field, props }, index);
37
+ }),
38
+ ),
39
+ );
40
+
41
+ export default handler;
@@ -0,0 +1,21 @@
1
+ // Copyright 2025 DXOS.org
2
+
3
+ import * as Effect from 'effect/Effect';
4
+
5
+ import { Obj } from '@dxos/echo';
6
+ import { invariant } from '@dxos/invariant';
7
+ import { Operation } from '@dxos/operation';
8
+
9
+ import { RestoreCard } from './definitions';
10
+
11
+ const handler: Operation.WithHandler<typeof RestoreCard> = RestoreCard.pipe(
12
+ Operation.withHandler(({ card }) =>
13
+ Effect.sync(() => {
14
+ const db = Obj.getDatabase(card);
15
+ invariant(db);
16
+ db.add(card);
17
+ }),
18
+ ),
19
+ );
20
+
21
+ export default handler;
@@ -0,0 +1,13 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { BoardManager as MosaicBoardManager } from '@dxos/react-ui-mosaic/playwright';
6
+
7
+ export class BoardManager extends MosaicBoardManager {
8
+ async waitUntilReady(): Promise<void> {
9
+ await this.columns().first().waitFor({ state: 'visible' });
10
+ await this.columns().nth(2).waitFor({ state: 'visible' });
11
+ await this.column(1).items().first().waitFor({ state: 'visible' });
12
+ }
13
+ }
@@ -0,0 +1,19 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { defineConfig } from '@playwright/test';
6
+
7
+ import { e2ePreset } from '@dxos/test-utils/playwright';
8
+
9
+ export default defineConfig({
10
+ ...e2ePreset(import.meta.dirname),
11
+ // TODO(wittjosiah): Stories are slow to start up.
12
+ timeout: 60_000,
13
+ // TODO(wittjosiah): Avoid hard-coding ports.
14
+ webServer: {
15
+ command: 'pnpm storybook dev --ci --quiet --port=9011 --config-dir=.storybook',
16
+ port: 9011,
17
+ reuseExistingServer: false,
18
+ },
19
+ });
@@ -0,0 +1,107 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { type Page, expect, test } from '@playwright/test';
6
+
7
+ import { setupPage, storybookUrl } from '@dxos/test-utils/playwright';
8
+
9
+ import { BoardManager } from './board-manager';
10
+
11
+ const PORT = 9011;
12
+ const STORY_URL = storybookUrl('plugins-plugin-kanban-containers-kanban--mutable-schema', PORT);
13
+
14
+ test.describe('Kanban MutableSchema', () => {
15
+ let page: Page;
16
+ let board: BoardManager;
17
+
18
+ test.beforeEach(async ({ browser }) => {
19
+ // Larger viewport to avoid triggering scroll-assist behaviour on simple drag operations.
20
+ ({ page } = await setupPage(browser, { url: STORY_URL, viewportSize: { width: 1920, height: 1080 } }));
21
+ board = new BoardManager(page.locator('body'));
22
+ await board.waitUntilReady();
23
+ });
24
+
25
+ test.afterEach(async () => {
26
+ await page.close();
27
+ });
28
+
29
+ test('rearrange columns', async () => {
30
+ const col1Label = await board.column(1).title().textContent();
31
+ const col2Label = await board.column(2).title().textContent();
32
+ expect(col1Label).not.toBeNull();
33
+ expect(col2Label).not.toBeNull();
34
+
35
+ await board.column(1).dragTo(board.column(2).header());
36
+
37
+ await expect(board.column(1).title()).toHaveText(col2Label!);
38
+ await expect(board.column(2).title()).toHaveText(col1Label!);
39
+ });
40
+
41
+ test('rearrange within column', async () => {
42
+ // Column 0 is uncategorized (empty). Use column 1 (first status column).
43
+ const column = board.column(1);
44
+ const countBefore = await column.items().count();
45
+
46
+ const firstLabel = await column.item(0).title().textContent();
47
+ const secondLabel = await column.item(1).title().textContent();
48
+ expect(firstLabel).not.toBeNull();
49
+ expect(secondLabel).not.toBeNull();
50
+
51
+ // Drag first item below the second item.
52
+ await column.item(0).dragTo(column.item(1).locator, { x: 0, y: 200 });
53
+
54
+ // Item count should stay the same.
55
+ await expect(column.items()).toHaveCount(countBefore);
56
+
57
+ // The first item should now be what was previously the second item.
58
+ await expect(column.item(0).title()).toHaveText(secondLabel!);
59
+
60
+ // The original first item should now be at index 1.
61
+ await expect(column.item(1).title()).toHaveText(firstLabel!);
62
+ });
63
+
64
+ test('drag to beginning of another column', async () => {
65
+ // Column 0 is uncategorized (empty). Use columns 1 and 2 (both have items).
66
+ const col1 = board.column(1);
67
+ const col2 = board.column(2);
68
+
69
+ const col1CountBefore = await col1.items().count();
70
+ const col2CountBefore = await col2.items().count();
71
+ const draggedLabel = await col1.item(0).title().textContent();
72
+ expect(draggedLabel).not.toBeNull();
73
+
74
+ // Drop above first item. Kanban cards are taller; use larger negative y so we land in top half.
75
+ await col1.item(0).dragTo(col2.item(0).locator, { x: 0, y: -30 });
76
+
77
+ await expect(col1.items()).toHaveCount(col1CountBefore - 1);
78
+ await expect(col2.items()).toHaveCount(col2CountBefore + 1);
79
+ await expect(col2.item(0).title()).toHaveText(draggedLabel!);
80
+ });
81
+
82
+ test('drag into empty column', async () => {
83
+ // Uncategorized is column 0 (empty); first populated column is at index 1.
84
+ const emptyColumn = board.column(0);
85
+ const sourceColumn = board.column(1);
86
+
87
+ const sourceCountBefore = await sourceColumn.items().count();
88
+ const draggedLabel = await sourceColumn.item(0).title().textContent();
89
+ expect(draggedLabel).not.toBeNull();
90
+
91
+ await sourceColumn.item(0).dragTo(emptyColumn.header(), { x: 0, y: 40 });
92
+
93
+ await expect(sourceColumn.items()).toHaveCount(sourceCountBefore - 1);
94
+ await expect(emptyColumn.items()).toHaveCount(1);
95
+ await expect(emptyColumn.item(0).title()).toHaveText(draggedLabel!);
96
+ });
97
+
98
+ test('create new item', async () => {
99
+ // Use first populated column.
100
+ const column = board.column(1);
101
+ const countBefore = await column.items().count();
102
+
103
+ await column.addItem();
104
+
105
+ await expect(column.items()).toHaveCount(countBefore + 1);
106
+ });
107
+ });
@@ -0,0 +1,82 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { forwardRef, useCallback, useMemo, useState } from 'react';
6
+
7
+ import { Obj } from '@dxos/echo';
8
+ import { Card, Toolbar, useTranslation } from '@dxos/react-ui';
9
+ import { Menu, createMenuAction } from '@dxos/react-ui-menu';
10
+ import { Focus, Mosaic, useBoard } from '@dxos/react-ui-mosaic';
11
+
12
+ import { type KanbanCardProps, useKanbanBoard } from '#components';
13
+ import { meta } from '#meta';
14
+
15
+ const KANBAN_CARD_TILE_SIMPLE_NAME = 'KanbanCardTileSimple';
16
+
17
+ /**
18
+ * Card tile without Surface; for stories and tests when plugin manager is not available.
19
+ */
20
+ export const KanbanCardTileSimple = forwardRef<HTMLDivElement, KanbanCardProps>(
21
+ ({ data, location, debug }, forwardedRef) => {
22
+ const { t } = useTranslation(meta.id);
23
+ const { model } = useBoard(KANBAN_CARD_TILE_SIMPLE_NAME);
24
+ const { onCardRemove } = useKanbanBoard(KANBAN_CARD_TILE_SIMPLE_NAME);
25
+ const [dragHandle, setDragHandle] = useState<HTMLButtonElement | null>(null);
26
+ const dragHandleRef = useCallback((el: HTMLButtonElement | null) => setDragHandle(el), []);
27
+
28
+ const menuItems = useMemo(
29
+ () =>
30
+ onCardRemove
31
+ ? [
32
+ createMenuAction('remove', () => onCardRemove(data), {
33
+ label: t('remove-card.label'),
34
+ icon: 'ph--trash--regular',
35
+ }),
36
+ ]
37
+ : [],
38
+ [onCardRemove, data, t],
39
+ );
40
+
41
+ return (
42
+ <Menu.Root>
43
+ <Mosaic.Tile
44
+ asChild
45
+ id={model.getItemId(data)}
46
+ data={data}
47
+ location={location}
48
+ debug={debug}
49
+ dragHandle={dragHandle}
50
+ >
51
+ <Focus.Item asChild>
52
+ <Card.Root ref={forwardedRef} data-testid='board-item'>
53
+ <Card.Toolbar>
54
+ <Card.DragHandle ref={dragHandleRef} />
55
+ <Card.Title>{Obj.getLabel(data)}</Card.Title>
56
+ {/* TODO(wittjosiah): Reconcile with Card.Menu. */}
57
+ <Menu.Trigger asChild disabled={!menuItems?.length}>
58
+ <Toolbar.IconButton
59
+ iconOnly
60
+ variant='ghost'
61
+ icon='ph--dots-three-vertical--regular'
62
+ label={t('action-menu.label')}
63
+ />
64
+ </Menu.Trigger>
65
+ <Menu.Content items={menuItems} />
66
+ </Card.Toolbar>
67
+ <Card.Content>
68
+ <Card.Section>
69
+ <pre className='p-2 text-xs text-description whitespace-pre-wrap'>
70
+ {JSON.stringify(data, null, 2)}
71
+ </pre>
72
+ </Card.Section>
73
+ </Card.Content>
74
+ </Card.Root>
75
+ </Focus.Item>
76
+ </Mosaic.Tile>
77
+ </Menu.Root>
78
+ );
79
+ },
80
+ );
81
+
82
+ KanbanCardTileSimple.displayName = KANBAN_CARD_TILE_SIMPLE_NAME;
@@ -1,6 +1,5 @@
1
1
  //
2
- // Copyright 2023 DXOS.org
2
+ // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- export * from './kanban-manager';
6
- export * from './initialize-kanban';
5
+ export { KanbanCardTileSimple } from './KanbanCardTileSimple';
@@ -2,34 +2,42 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
+ import { Type } from '@dxos/echo';
5
6
  import { type Resource } from '@dxos/react-ui';
6
- import { KanbanType } from '@dxos/react-ui-kanban';
7
7
 
8
- import { meta } from './meta';
8
+ import { meta } from '#meta';
9
+ import { Kanban } from '#types';
9
10
 
10
11
  export const translations = [
11
12
  {
12
13
  'en-US': {
13
- [KanbanType.typename]: {
14
- 'typename label': 'Kanban',
15
- 'typename label_zero': 'Kanbans',
16
- 'typename label_one': 'Kanban',
17
- 'typename label_other': 'Kanbans',
18
- 'object name placeholder': 'New kanban',
14
+ [Type.getTypename(Kanban.Kanban)]: {
15
+ 'typename.label': 'Kanban',
16
+ 'typename.label_zero': 'Kanbans',
17
+ 'typename.label_one': 'Kanban',
18
+ 'typename.label_other': 'Kanbans',
19
+ 'object-name.placeholder': 'New kanban',
20
+ 'add-object.label': 'Add kanban',
21
+ 'rename-object.label': 'Rename kanban',
22
+ 'delete-object.label': 'Delete kanban',
23
+ 'object-deleted.label': 'Kanban deleted',
19
24
  },
20
25
  [meta.id]: {
21
- 'plugin name': 'Kanban',
22
- 'kanban title label': 'Title',
23
- 'column title label': 'Column title',
24
- 'column title placeholder': 'New column',
25
- 'item title label': 'Item title',
26
- 'item title placeholder': 'New item',
27
- 'add column label': 'Add column',
28
- 'add item label': 'Add card',
29
- 'delete column label': 'Delete column',
30
- 'delete item label': 'Delete card',
31
- 'card field deleted label': 'Card field deleted',
32
- 'card deleted label': 'Card deleted',
26
+ 'action-menu.label': 'Actions',
27
+ 'plugin.name': 'Kanban',
28
+ 'kanban-title.label': 'Title',
29
+ 'column-title.label': 'Column title',
30
+ 'column-title.placeholder': 'New column',
31
+ 'add-column.label': 'Add column',
32
+ 'add-card.label': 'Add card',
33
+ 'new-column-name.label': 'New column name',
34
+ 'remove-card.label': 'Remove card',
35
+ 'remove-empty-column.label': 'Remove empty column',
36
+ 'column-drag-handle.label': 'Drag to rearrange',
37
+ 'delete-column.label': 'Delete column',
38
+ 'card-field-deleted.label': 'Card field deleted',
39
+ 'card-deleted.label': 'Card deleted',
40
+ 'select-pivot.placeholder': 'Select a pivot column in board settings to display columns.',
33
41
  },
34
42
  },
35
43
  },
@@ -0,0 +1,71 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import * as Schema from 'effect/Schema';
6
+
7
+ import { Annotation, Obj, Ref, Type } from '@dxos/echo';
8
+ import { View } from '@dxos/echo';
9
+ import { FormInputAnnotation, LabelAnnotation } from '@dxos/echo/internal';
10
+ import { ViewAnnotation } from '@dxos/schema';
11
+
12
+ /** Per-column entry (ids order, optional hidden). */
13
+ const ArrangementColumnEntry = Schema.Struct({
14
+ ids: Schema.Array(Obj.ID),
15
+ hidden: Schema.optional(Schema.Boolean),
16
+ });
17
+
18
+ /** Keyed by columnValue. */
19
+ const ArrangementColumns = Schema.Record({
20
+ key: Schema.String,
21
+ value: ArrangementColumnEntry,
22
+ }).pipe(FormInputAnnotation.set(false));
23
+
24
+ /** Column order and per-column card ids. */
25
+ export const Arrangement = Schema.Struct({
26
+ order: Schema.Array(Schema.String).pipe(FormInputAnnotation.set(false)),
27
+ columns: ArrangementColumns,
28
+ }).pipe(FormInputAnnotation.set(false));
29
+
30
+ export type Arrangement = Schema.Schema.Type<typeof Arrangement>;
31
+
32
+ export const Kanban = Schema.Struct({
33
+ name: Schema.optional(Schema.String),
34
+
35
+ view: Ref.Ref(View.View).pipe(FormInputAnnotation.set(false)),
36
+
37
+ /** Column display order and per-column card ids. */
38
+ arrangement: Arrangement,
39
+ }).pipe(
40
+ Type.object({
41
+ typename: 'org.dxos.type.kanban',
42
+ version: '0.1.0',
43
+ }),
44
+ LabelAnnotation.set(['name']),
45
+ ViewAnnotation.set(true),
46
+ Annotation.IconAnnotation.set({
47
+ icon: 'ph--kanban--regular',
48
+ hue: 'green',
49
+ }),
50
+ );
51
+
52
+ /** Instance type; use Kanban.Kanban in type position so namespace has .Kanban as type and .KanbanSchema as schema. */
53
+ export interface Kanban extends Schema.Schema.Type<typeof Kanban> {}
54
+
55
+ type MakeProps = Omit<Partial<Obj.MakeProps<typeof Kanban>>, 'view'> & {
56
+ view: View.View;
57
+ };
58
+
59
+ /**
60
+ * Make a kanban as a view of a data set.
61
+ */
62
+ export const make = (props: MakeProps): Kanban => {
63
+ const { name, view, arrangement } = props;
64
+ const order = arrangement?.order ?? [];
65
+ const columns = arrangement?.columns ?? {};
66
+ return Obj.make(Kanban, {
67
+ name,
68
+ view: Ref.make(view),
69
+ arrangement: { order, columns },
70
+ });
71
+ };
@@ -0,0 +1,9 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export const UNCATEGORIZED_VALUE = '__uncategorized__' as const;
6
+ export const UNCATEGORIZED_ATTRIBUTES = {
7
+ title: 'Uncategorized',
8
+ color: 'neutral',
9
+ } as const;
@@ -2,5 +2,7 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ export * from './constants';
6
+ export * as Kanban from './Kanban';
5
7
  export * from './schema';
6
- export * from './types';
8
+ export type * from './types';