@dxos/plugin-kanban 0.8.4-main.fd6878d → 0.9.0
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.
- package/LICENSE +102 -5
- package/PLUGIN.mdl +398 -0
- package/README.md +1 -1
- package/dist/lib/neutral/KanbanArticle-T4CPKAZH.mjs +132 -0
- package/dist/lib/neutral/KanbanArticle-T4CPKAZH.mjs.map +7 -0
- package/dist/lib/neutral/KanbanPlugin.mjs +44 -0
- package/dist/lib/neutral/KanbanPlugin.mjs.map +7 -0
- package/dist/lib/neutral/KanbanPlugin.node.mjs +27 -0
- package/dist/lib/neutral/KanbanPlugin.node.mjs.map +7 -0
- package/dist/lib/neutral/KanbanPlugin.workerd.mjs +21 -0
- package/dist/lib/neutral/KanbanPlugin.workerd.mjs.map +7 -0
- package/dist/lib/neutral/KanbanSettings-5WOS4CUE.mjs +83 -0
- package/dist/lib/neutral/KanbanSettings-5WOS4CUE.mjs.map +7 -0
- package/dist/lib/neutral/blueprint-definition-6DV3Q5MC.mjs +15 -0
- package/dist/lib/neutral/blueprint-definition-6DV3Q5MC.mjs.map +7 -0
- package/dist/lib/neutral/blueprints/index.mjs +8 -0
- package/dist/lib/neutral/capabilities/index.mjs +17 -0
- package/dist/lib/neutral/capabilities/index.mjs.map +7 -0
- package/dist/lib/neutral/chunk-6ZHHQWO5.mjs +39 -0
- package/dist/lib/neutral/chunk-6ZHHQWO5.mjs.map +7 -0
- package/dist/lib/neutral/chunk-DAKIZO46.mjs +246 -0
- package/dist/lib/neutral/chunk-DAKIZO46.mjs.map +7 -0
- package/dist/lib/neutral/chunk-J5LGTIGS.mjs +10 -0
- package/dist/lib/neutral/chunk-M5ISZWZU.mjs +8 -0
- package/dist/lib/neutral/chunk-M5ISZWZU.mjs.map +7 -0
- package/dist/lib/neutral/chunk-ZTQW5KQS.mjs +26 -0
- package/dist/lib/neutral/chunk-ZTQW5KQS.mjs.map +7 -0
- package/dist/lib/neutral/components/index.mjs +243 -0
- package/dist/lib/neutral/components/index.mjs.map +7 -0
- package/dist/lib/neutral/containers/index.mjs +11 -0
- package/dist/lib/neutral/containers/index.mjs.map +7 -0
- package/dist/lib/neutral/create-object-DKBSI46K.mjs +40 -0
- package/dist/lib/neutral/create-object-DKBSI46K.mjs.map +7 -0
- package/dist/lib/neutral/delete-card-VNAV3CZV.mjs +24 -0
- package/dist/lib/neutral/delete-card-VNAV3CZV.mjs.map +7 -0
- package/dist/lib/neutral/delete-card-field-XHOLGS6L.mjs +39 -0
- package/dist/lib/neutral/delete-card-field-XHOLGS6L.mjs.map +7 -0
- package/dist/lib/neutral/hooks/index.mjs +431 -0
- package/dist/lib/neutral/hooks/index.mjs.map +7 -0
- package/dist/lib/neutral/index.mjs +34 -0
- package/dist/lib/neutral/index.mjs.map +7 -0
- package/dist/lib/neutral/meta.json +1 -0
- package/dist/lib/neutral/meta.mjs +8 -0
- package/dist/lib/neutral/meta.mjs.map +7 -0
- package/dist/lib/neutral/operation-handler-B7IW6MXU.mjs +13 -0
- package/dist/lib/neutral/operation-handler-B7IW6MXU.mjs.map +7 -0
- package/dist/lib/neutral/operations/index.mjs +8 -0
- package/dist/lib/neutral/operations/index.mjs.map +7 -0
- package/dist/lib/neutral/plugin.mjs +16 -0
- package/dist/lib/neutral/plugin.mjs.map +7 -0
- package/dist/lib/neutral/react-surface-VZEVEJL5.mjs +91 -0
- package/dist/lib/neutral/react-surface-VZEVEJL5.mjs.map +7 -0
- package/dist/lib/neutral/restore-card-EKVEPATL.mjs +21 -0
- package/dist/lib/neutral/restore-card-EKVEPATL.mjs.map +7 -0
- package/dist/lib/neutral/restore-card-field-TQCTGGNO.mjs +37 -0
- package/dist/lib/neutral/restore-card-field-TQCTGGNO.mjs.map +7 -0
- package/dist/lib/neutral/testing/index.mjs +62 -0
- package/dist/lib/neutral/testing/index.mjs.map +7 -0
- package/dist/lib/neutral/translations.mjs +44 -0
- package/dist/lib/neutral/translations.mjs.map +7 -0
- package/dist/lib/neutral/types/index.mjs +22 -0
- package/dist/lib/neutral/types/index.mjs.map +7 -0
- package/dist/lib/neutral/undo-mappings-6CHW6BOF.mjs +42 -0
- package/dist/lib/neutral/undo-mappings-6CHW6BOF.mjs.map +7 -0
- package/dist/types/src/KanbanPlugin.d.ts +3 -1
- package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
- package/dist/types/src/KanbanPlugin.node.d.ts +4 -0
- package/dist/types/src/KanbanPlugin.node.d.ts.map +1 -0
- package/dist/types/src/KanbanPlugin.test.d.ts +2 -0
- package/dist/types/src/KanbanPlugin.test.d.ts.map +1 -0
- package/dist/types/src/KanbanPlugin.workerd.d.ts +4 -0
- package/dist/types/src/KanbanPlugin.workerd.d.ts.map +1 -0
- package/dist/types/src/blueprints/index.d.ts +2 -0
- package/dist/types/src/blueprints/index.d.ts.map +1 -0
- package/dist/types/src/blueprints/kanban-blueprint.d.ts +4 -0
- package/dist/types/src/blueprints/kanban-blueprint.d.ts.map +1 -0
- package/dist/types/src/capabilities/artifact-definition.d.ts +3 -2
- package/dist/types/src/capabilities/artifact-definition.d.ts.map +1 -1
- package/dist/types/src/capabilities/blueprint-definition.d.ts +6 -0
- package/dist/types/src/capabilities/blueprint-definition.d.ts.map +1 -0
- package/dist/types/src/capabilities/create-object.d.ts +11 -0
- package/dist/types/src/capabilities/create-object.d.ts.map +1 -0
- package/dist/types/src/capabilities/index.d.ts +13 -2
- package/dist/types/src/capabilities/index.d.ts.map +1 -1
- package/dist/types/src/capabilities/operation-handler.d.ts +6 -0
- package/dist/types/src/capabilities/operation-handler.d.ts.map +1 -0
- package/dist/types/src/capabilities/react-surface.d.ts +3 -2
- package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
- package/dist/types/src/capabilities/undo-mappings.d.ts +5 -0
- package/dist/types/src/capabilities/undo-mappings.d.ts.map +1 -0
- package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts +37 -0
- package/dist/types/src/components/KanbanBoard/KanbanBoard.d.ts.map +1 -0
- package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts +72 -0
- package/dist/types/src/components/KanbanBoard/KanbanBoard.stories.d.ts.map +1 -0
- package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts +9 -0
- package/dist/types/src/components/KanbanBoard/KanbanCard.d.ts.map +1 -0
- package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts +8 -0
- package/dist/types/src/components/KanbanBoard/KanbanColumn.d.ts.map +1 -0
- package/dist/types/src/components/KanbanBoard/context.d.ts +38 -0
- package/dist/types/src/components/KanbanBoard/context.d.ts.map +1 -0
- package/dist/types/src/components/KanbanBoard/index.d.ts +2 -0
- package/dist/types/src/components/KanbanBoard/index.d.ts.map +1 -0
- package/dist/types/src/components/index.d.ts +1 -2
- package/dist/types/src/components/index.d.ts.map +1 -1
- package/dist/types/src/containers/KanbanArticle/KanbanArticle.d.ts +6 -0
- package/dist/types/src/containers/KanbanArticle/KanbanArticle.d.ts.map +1 -0
- package/dist/types/src/containers/KanbanArticle/KanbanArticle.stories.d.ts +79 -0
- package/dist/types/src/containers/KanbanArticle/KanbanArticle.stories.d.ts.map +1 -0
- package/dist/types/src/containers/KanbanArticle/index.d.ts +2 -0
- package/dist/types/src/containers/KanbanArticle/index.d.ts.map +1 -0
- package/dist/types/src/containers/KanbanSettings/KanbanSettings.d.ts +13 -0
- package/dist/types/src/containers/KanbanSettings/KanbanSettings.d.ts.map +1 -0
- package/dist/types/src/containers/KanbanSettings/index.d.ts +2 -0
- package/dist/types/src/containers/KanbanSettings/index.d.ts.map +1 -0
- package/dist/types/src/containers/index.d.ts +4 -0
- package/dist/types/src/containers/index.d.ts.map +1 -0
- package/dist/types/src/hooks/index.d.ts +7 -0
- package/dist/types/src/hooks/index.d.ts.map +1 -0
- package/dist/types/src/hooks/useEchoChangeCallback.d.ts +13 -0
- package/dist/types/src/hooks/useEchoChangeCallback.d.ts.map +1 -0
- package/dist/types/src/hooks/useItemsProjection.d.ts +10 -0
- package/dist/types/src/hooks/useItemsProjection.d.ts.map +1 -0
- package/dist/types/src/hooks/useKanbanBoardModel.browser.test.d.ts +2 -0
- package/dist/types/src/hooks/useKanbanBoardModel.browser.test.d.ts.map +1 -0
- package/dist/types/src/hooks/useKanbanBoardModel.d.ts +16 -0
- package/dist/types/src/hooks/useKanbanBoardModel.d.ts.map +1 -0
- package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts +22 -0
- package/dist/types/src/hooks/useKanbanColumnEventHandler.d.ts.map +1 -0
- package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts +19 -0
- package/dist/types/src/hooks/useKanbanItemEventHandler.d.ts.map +1 -0
- package/dist/types/src/hooks/useProjectionModel.d.ts +15 -0
- package/dist/types/src/hooks/useProjectionModel.d.ts.map +1 -0
- package/dist/types/src/index.d.ts +3 -1
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/meta.d.ts +2 -3
- package/dist/types/src/meta.d.ts.map +1 -1
- package/dist/types/src/operations/delete-card-field.d.ts +5 -0
- package/dist/types/src/operations/delete-card-field.d.ts.map +1 -0
- package/dist/types/src/operations/delete-card.d.ts +5 -0
- package/dist/types/src/operations/delete-card.d.ts.map +1 -0
- package/dist/types/src/operations/index.d.ts +3 -0
- package/dist/types/src/operations/index.d.ts.map +1 -0
- package/dist/types/src/operations/restore-card-field.d.ts +5 -0
- package/dist/types/src/operations/restore-card-field.d.ts.map +1 -0
- package/dist/types/src/operations/restore-card.d.ts +5 -0
- package/dist/types/src/operations/restore-card.d.ts.map +1 -0
- package/dist/types/src/playwright/board-manager.d.ts +5 -0
- package/dist/types/src/playwright/board-manager.d.ts.map +1 -0
- package/dist/types/src/playwright/playwright.config.d.ts +3 -0
- package/dist/types/src/playwright/playwright.config.d.ts.map +1 -0
- package/dist/types/src/playwright/smoke.spec.d.ts +2 -0
- package/dist/types/src/playwright/smoke.spec.d.ts.map +1 -0
- package/dist/types/src/plugin.d.ts +4 -0
- package/dist/types/src/plugin.d.ts.map +1 -0
- package/dist/types/src/testing/KanbanCardTileSimple.d.ts +7 -0
- package/dist/types/src/testing/KanbanCardTileSimple.d.ts.map +1 -0
- package/dist/types/src/testing/index.d.ts +2 -0
- package/dist/types/src/testing/index.d.ts.map +1 -0
- package/dist/types/src/translations.d.ts +50 -22
- package/dist/types/src/translations.d.ts.map +1 -1
- package/dist/types/src/types/Kanban.d.ts +94 -0
- package/dist/types/src/types/Kanban.d.ts.map +1 -0
- package/dist/types/src/types/KanbanOperation.d.ts +52 -0
- package/dist/types/src/types/KanbanOperation.d.ts.map +1 -0
- package/dist/types/src/types/constants.d.ts +6 -0
- package/dist/types/src/types/constants.d.ts.map +1 -0
- package/dist/types/src/types/index.d.ts +3 -1
- package/dist/types/src/types/index.d.ts.map +1 -1
- package/dist/types/src/types/schema.d.ts +18 -95
- package/dist/types/src/types/schema.d.ts.map +1 -1
- package/dist/types/src/types/types.d.ts +28 -0
- package/dist/types/src/types/types.d.ts.map +1 -1
- package/dist/types/src/util/arrangement.d.ts +72 -0
- package/dist/types/src/util/arrangement.d.ts.map +1 -0
- package/dist/types/src/util/arrangement.test.d.ts +2 -0
- package/dist/types/src/util/arrangement.test.d.ts.map +1 -0
- package/dist/types/src/util/index.d.ts +2 -0
- package/dist/types/src/util/index.d.ts.map +1 -0
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +127 -55
- package/src/KanbanPlugin.node.ts +21 -0
- package/src/KanbanPlugin.test.ts +31 -0
- package/src/KanbanPlugin.tsx +24 -52
- package/src/KanbanPlugin.workerd.ts +18 -0
- package/src/blueprints/index.ts +5 -0
- package/src/blueprints/kanban-blueprint.ts +27 -0
- package/src/capabilities/artifact-definition.ts +119 -114
- package/src/capabilities/blueprint-definition.ts +19 -0
- package/src/capabilities/create-object.ts +40 -0
- package/src/capabilities/index.ts +16 -3
- package/src/capabilities/operation-handler.ts +14 -0
- package/src/capabilities/react-surface.tsx +90 -69
- package/src/capabilities/undo-mappings.ts +34 -0
- package/src/components/KanbanBoard/KanbanBoard.stories.tsx +145 -0
- package/src/components/KanbanBoard/KanbanBoard.tsx +164 -0
- package/src/components/KanbanBoard/KanbanCard.tsx +101 -0
- package/src/components/KanbanBoard/KanbanColumn.tsx +72 -0
- package/src/components/KanbanBoard/context.ts +54 -0
- package/src/components/KanbanBoard/index.ts +5 -0
- package/src/components/index.ts +1 -2
- package/src/containers/KanbanArticle/KanbanArticle.stories.tsx +277 -0
- package/src/containers/KanbanArticle/KanbanArticle.tsx +179 -0
- package/src/containers/KanbanArticle/index.ts +5 -0
- package/src/containers/KanbanSettings/KanbanSettings.tsx +94 -0
- package/src/containers/KanbanSettings/index.ts +5 -0
- package/src/containers/index.ts +8 -0
- package/src/hooks/index.ts +10 -0
- package/src/hooks/useEchoChangeCallback.ts +30 -0
- package/src/hooks/useItemsProjection.ts +44 -0
- package/src/hooks/useKanbanBoardModel.browser.test.ts +230 -0
- package/src/hooks/useKanbanBoardModel.ts +156 -0
- package/src/hooks/useKanbanColumnEventHandler.ts +106 -0
- package/src/hooks/useKanbanItemEventHandler.ts +133 -0
- package/src/hooks/useProjectionModel.ts +58 -0
- package/src/index.ts +3 -2
- package/src/meta.ts +27 -8
- package/src/operations/delete-card-field.ts +42 -0
- package/src/operations/delete-card.ts +23 -0
- package/src/operations/index.ts +10 -0
- package/src/operations/restore-card-field.ts +36 -0
- package/src/operations/restore-card.ts +21 -0
- package/src/playwright/board-manager.ts +13 -0
- package/src/playwright/playwright.config.ts +19 -0
- package/src/playwright/smoke.spec.ts +107 -0
- package/src/plugin.ts +11 -0
- package/src/testing/KanbanCardTileSimple.tsx +82 -0
- package/src/testing/index.ts +5 -0
- package/src/translations.ts +28 -20
- package/src/types/Kanban.ts +134 -0
- package/src/types/KanbanOperation.ts +79 -0
- package/src/types/constants.ts +9 -0
- package/src/types/index.ts +4 -1
- package/src/types/schema.ts +33 -45
- package/src/types/types.ts +35 -0
- package/src/util/arrangement.test.ts +217 -0
- package/src/util/arrangement.ts +177 -0
- package/src/util/index.ts +5 -0
- package/src/vite-env.d.ts +10 -0
- package/dist/lib/browser/chunk-5BR6HNHI.mjs +0 -108
- package/dist/lib/browser/chunk-5BR6HNHI.mjs.map +0 -7
- package/dist/lib/browser/index.mjs +0 -99
- package/dist/lib/browser/index.mjs.map +0 -7
- package/dist/lib/browser/intent-resolver-65UVYRI2.mjs +0 -111
- package/dist/lib/browser/intent-resolver-65UVYRI2.mjs.map +0 -7
- package/dist/lib/browser/meta.json +0 -1
- package/dist/lib/browser/react-surface-KX5U6PJ2.mjs +0 -256
- package/dist/lib/browser/react-surface-KX5U6PJ2.mjs.map +0 -7
- package/dist/lib/browser/types/index.mjs +0 -13
- package/dist/lib/node-esm/chunk-WLFZKRIJ.mjs +0 -110
- package/dist/lib/node-esm/chunk-WLFZKRIJ.mjs.map +0 -7
- package/dist/lib/node-esm/index.mjs +0 -100
- package/dist/lib/node-esm/index.mjs.map +0 -7
- package/dist/lib/node-esm/intent-resolver-R3MYQATZ.mjs +0 -112
- package/dist/lib/node-esm/intent-resolver-R3MYQATZ.mjs.map +0 -7
- package/dist/lib/node-esm/meta.json +0 -1
- package/dist/lib/node-esm/react-surface-BAMERRU5.mjs +0 -257
- package/dist/lib/node-esm/react-surface-BAMERRU5.mjs.map +0 -7
- package/dist/lib/node-esm/types/index.mjs +0 -14
- package/dist/types/src/capabilities/intent-resolver.d.ts +0 -4
- package/dist/types/src/capabilities/intent-resolver.d.ts.map +0 -1
- package/dist/types/src/components/KanbanContainer.d.ts +0 -7
- package/dist/types/src/components/KanbanContainer.d.ts.map +0 -1
- package/dist/types/src/components/KanbanContainer.stories.d.ts +0 -10
- package/dist/types/src/components/KanbanContainer.stories.d.ts.map +0 -1
- package/dist/types/src/components/KanbanViewEditor.d.ts +0 -8
- package/dist/types/src/components/KanbanViewEditor.d.ts.map +0 -1
- package/dist/types/src/types/kanban.d.ts +0 -11
- package/dist/types/src/types/kanban.d.ts.map +0 -1
- package/src/capabilities/intent-resolver.ts +0 -70
- package/src/components/KanbanContainer.stories.tsx +0 -189
- package/src/components/KanbanContainer.tsx +0 -95
- package/src/components/KanbanViewEditor.tsx +0 -60
- package/src/types/kanban.ts +0 -29
- /package/dist/lib/{browser/types → neutral/blueprints}/index.mjs.map +0 -0
- /package/dist/lib/{node-esm/types/index.mjs.map → neutral/chunk-J5LGTIGS.mjs.map} +0 -0
package/src/meta.ts
CHANGED
|
@@ -2,16 +2,35 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import {
|
|
5
|
+
import { Plugin } from '@dxos/app-framework';
|
|
6
|
+
import { DXN } from '@dxos/echo';
|
|
7
|
+
import { trim } from '@dxos/util';
|
|
6
8
|
|
|
7
|
-
export const
|
|
8
|
-
|
|
9
|
-
export const meta: PluginMeta = {
|
|
10
|
-
id: KANBAN_PLUGIN,
|
|
9
|
+
export const meta = Plugin.makeMeta({
|
|
10
|
+
key: DXN.make('org.dxos.plugin.kanban'),
|
|
11
11
|
name: 'Kanban',
|
|
12
|
-
|
|
13
|
-
|
|
12
|
+
author: 'DXOS',
|
|
13
|
+
description: trim`
|
|
14
|
+
Visual project management using customizable kanban boards to track workflow progress.
|
|
15
|
+
Organize any ECHO type into drag-and-drop columns defined by a single-select field, move cards
|
|
16
|
+
between stages, and reorder both columns and cards — all changes are persisted to ECHO and
|
|
17
|
+
replicated to collaborators in real time.
|
|
18
|
+
|
|
19
|
+
Kanban boards come in two variants: view-variant boards source their cards through a linked
|
|
20
|
+
View query (any ECHO type can be pivoted on a single-select field), while items-variant boards
|
|
21
|
+
own their cards explicitly and are used by external sync integrations such as Linear or GitHub.
|
|
22
|
+
Both variants share the same arrangement storage and drag-and-drop mechanics.
|
|
23
|
+
|
|
24
|
+
Board settings let users choose which single-select field drives the column layout and
|
|
25
|
+
optionally hide the uncategorized column for items that have no pivot value.
|
|
26
|
+
Field-level undo support allows deleted card fields to be restored without data loss.
|
|
27
|
+
|
|
28
|
+
The plugin contributes a Kanban blueprint that exposes board creation and card management
|
|
29
|
+
to AI agents, enabling automated project tracking workflows.
|
|
30
|
+
`,
|
|
14
31
|
icon: 'ph--kanban--regular',
|
|
32
|
+
iconHue: 'green',
|
|
15
33
|
source: 'https://github.com/dxos/dxos/tree/main/packages/plugins/plugin-kanban',
|
|
34
|
+
spec: 'PLUGIN.mdl',
|
|
16
35
|
screenshots: ['https://dxos.network/plugin-details-kanban-dark.png'],
|
|
17
|
-
};
|
|
36
|
+
});
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
// Copyright 2025 DXOS.org
|
|
2
|
+
|
|
3
|
+
import * as Effect from 'effect/Effect';
|
|
4
|
+
|
|
5
|
+
import { Capabilities, Capability } from '@dxos/app-framework';
|
|
6
|
+
import { Operation } from '@dxos/compute';
|
|
7
|
+
import { Obj, Type } from '@dxos/echo';
|
|
8
|
+
import { invariant } from '@dxos/invariant';
|
|
9
|
+
import { ProjectionModel, createEchoChangeCallback, getTypeURIFromQuery } from '@dxos/schema';
|
|
10
|
+
|
|
11
|
+
import { KanbanOperation } from '../types';
|
|
12
|
+
|
|
13
|
+
const handler: Operation.WithHandler<typeof KanbanOperation.DeleteCardField> = KanbanOperation.DeleteCardField.pipe(
|
|
14
|
+
Operation.withHandler(
|
|
15
|
+
Effect.fnUntraced(function* ({ view, fieldId }) {
|
|
16
|
+
const registry = yield* Capability.get(Capabilities.AtomRegistry);
|
|
17
|
+
const db = Obj.getDatabase(view);
|
|
18
|
+
invariant(db, 'Database not found');
|
|
19
|
+
const types = yield* Effect.sync(() => db.graph.registry.list().filter(Type.isType));
|
|
20
|
+
const schema = types.find((t) => Type.getURI(t) === getTypeURIFromQuery(view.query.ast));
|
|
21
|
+
invariant(schema, 'Schema not found');
|
|
22
|
+
|
|
23
|
+
invariant(Type.isType(schema), 'expected stored Type.Type for card schema');
|
|
24
|
+
const projection = new ProjectionModel({
|
|
25
|
+
registry,
|
|
26
|
+
view,
|
|
27
|
+
baseSchema: schema.jsonSchema,
|
|
28
|
+
change: createEchoChangeCallback(view, schema),
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
const result = projection.deleteFieldProjection(fieldId);
|
|
32
|
+
|
|
33
|
+
return {
|
|
34
|
+
field: result.deleted.field,
|
|
35
|
+
props: result.deleted.props,
|
|
36
|
+
index: result.index,
|
|
37
|
+
};
|
|
38
|
+
}),
|
|
39
|
+
),
|
|
40
|
+
);
|
|
41
|
+
|
|
42
|
+
export default handler;
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
// Copyright 2025 DXOS.org
|
|
2
|
+
|
|
3
|
+
import * as Effect from 'effect/Effect';
|
|
4
|
+
|
|
5
|
+
import { Operation } from '@dxos/compute';
|
|
6
|
+
import { Obj } from '@dxos/echo';
|
|
7
|
+
import { invariant } from '@dxos/invariant';
|
|
8
|
+
|
|
9
|
+
import { KanbanOperation } from '../types';
|
|
10
|
+
|
|
11
|
+
const handler: Operation.WithHandler<typeof KanbanOperation.DeleteCard> = KanbanOperation.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,10 @@
|
|
|
1
|
+
// Copyright 2025 DXOS.org
|
|
2
|
+
|
|
3
|
+
import { OperationHandlerSet } from '@dxos/compute';
|
|
4
|
+
|
|
5
|
+
export const KanbanOperationHandlerSet = OperationHandlerSet.lazy(
|
|
6
|
+
() => import('./delete-card'),
|
|
7
|
+
() => import('./delete-card-field'),
|
|
8
|
+
() => import('./restore-card'),
|
|
9
|
+
() => import('./restore-card-field'),
|
|
10
|
+
);
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
// Copyright 2025 DXOS.org
|
|
2
|
+
|
|
3
|
+
import * as Effect from 'effect/Effect';
|
|
4
|
+
|
|
5
|
+
import { Capabilities, Capability } from '@dxos/app-framework';
|
|
6
|
+
import { Operation } from '@dxos/compute';
|
|
7
|
+
import { Obj, Type } from '@dxos/echo';
|
|
8
|
+
import { invariant } from '@dxos/invariant';
|
|
9
|
+
import { ProjectionModel, createEchoChangeCallback, getTypeURIFromQuery } from '@dxos/schema';
|
|
10
|
+
|
|
11
|
+
import { KanbanOperation } from '../types';
|
|
12
|
+
|
|
13
|
+
const handler: Operation.WithHandler<typeof KanbanOperation.RestoreCardField> = KanbanOperation.RestoreCardField.pipe(
|
|
14
|
+
Operation.withHandler(
|
|
15
|
+
Effect.fnUntraced(function* ({ view, field, props, index }) {
|
|
16
|
+
const registry = yield* Capability.get(Capabilities.AtomRegistry);
|
|
17
|
+
const db = Obj.getDatabase(view);
|
|
18
|
+
invariant(db, 'Database not found');
|
|
19
|
+
const types = db.graph.registry.list().filter(Type.isType);
|
|
20
|
+
const type = types.find((t) => Type.getURI(t) === getTypeURIFromQuery(view.query.ast));
|
|
21
|
+
invariant(type, 'Schema not found');
|
|
22
|
+
|
|
23
|
+
invariant(Type.isType(type), 'expected stored Type.Type for card schema');
|
|
24
|
+
const projection = new ProjectionModel({
|
|
25
|
+
registry,
|
|
26
|
+
view,
|
|
27
|
+
baseSchema: type.jsonSchema,
|
|
28
|
+
change: createEchoChangeCallback(view, type),
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
projection.setFieldProjection({ field, props }, index);
|
|
32
|
+
}),
|
|
33
|
+
),
|
|
34
|
+
);
|
|
35
|
+
|
|
36
|
+
export default handler;
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
// Copyright 2025 DXOS.org
|
|
2
|
+
|
|
3
|
+
import * as Effect from 'effect/Effect';
|
|
4
|
+
|
|
5
|
+
import { Operation } from '@dxos/compute';
|
|
6
|
+
import { Obj } from '@dxos/echo';
|
|
7
|
+
import { invariant } from '@dxos/invariant';
|
|
8
|
+
|
|
9
|
+
import { KanbanOperation } from '../types';
|
|
10
|
+
|
|
11
|
+
const handler: Operation.WithHandler<typeof KanbanOperation.RestoreCard> = KanbanOperation.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 }, 'bottom');
|
|
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.
|
|
75
|
+
await col1.item(0).dragTo(col2.item(0).locator, { x: 0, y: -30 }, 'top');
|
|
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
|
+
});
|
package/src/plugin.ts
ADDED
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2023 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { Plugin } from '@dxos/app-framework';
|
|
6
|
+
|
|
7
|
+
import { meta } from './meta';
|
|
8
|
+
|
|
9
|
+
export const KanbanPlugin = Plugin.lazy(meta, () => import('#plugin'));
|
|
10
|
+
|
|
11
|
+
export { KanbanOperationHandlerSet } from './operations';
|
|
@@ -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.Header>
|
|
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.Header>
|
|
67
|
+
<Card.Body>
|
|
68
|
+
<Card.Row fullWidth>
|
|
69
|
+
<pre className='p-2 text-xs text-description whitespace-pre-wrap'>
|
|
70
|
+
{JSON.stringify(data, null, 2)}
|
|
71
|
+
</pre>
|
|
72
|
+
</Card.Row>
|
|
73
|
+
</Card.Body>
|
|
74
|
+
</Card.Root>
|
|
75
|
+
</Focus.Item>
|
|
76
|
+
</Mosaic.Tile>
|
|
77
|
+
</Menu.Root>
|
|
78
|
+
);
|
|
79
|
+
},
|
|
80
|
+
);
|
|
81
|
+
|
|
82
|
+
KanbanCardTileSimple.displayName = KANBAN_CARD_TILE_SIMPLE_NAME;
|
package/src/translations.ts
CHANGED
|
@@ -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 { KanbanView } from '@dxos/react-ui-kanban';
|
|
7
7
|
|
|
8
|
-
import { meta } from '
|
|
8
|
+
import { meta } from '#meta';
|
|
9
|
+
import { Kanban } from '#types';
|
|
9
10
|
|
|
10
11
|
export const translations = [
|
|
11
12
|
{
|
|
12
13
|
'en-US': {
|
|
13
|
-
[
|
|
14
|
-
'typename
|
|
15
|
-
'typename
|
|
16
|
-
'typename
|
|
17
|
-
'typename
|
|
18
|
-
'object
|
|
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
|
-
'
|
|
22
|
-
'
|
|
23
|
-
'
|
|
24
|
-
'column
|
|
25
|
-
'
|
|
26
|
-
'
|
|
27
|
-
'add
|
|
28
|
-
'
|
|
29
|
-
'
|
|
30
|
-
'
|
|
31
|
-
'
|
|
32
|
-
'
|
|
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,134 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
6
|
+
|
|
7
|
+
// QueryAST is referenced indirectly through `Type.InstanceType<typeof ...Schema>`
|
|
8
|
+
// (Ref.Ref(View.View) → View.View → QueryAST.Query) in the emitted .d.ts; the
|
|
9
|
+
// namespace import keeps the inferred types portable.
|
|
10
|
+
// eslint-disable-next-line unused-imports/no-unused-imports
|
|
11
|
+
import { DXN, Annotation, Obj, QueryAST, Ref, Type, View } from '@dxos/echo';
|
|
12
|
+
import { FormInputAnnotation, LabelAnnotation } from '@dxos/echo/Annotation';
|
|
13
|
+
import { ViewAnnotation } from '@dxos/schema';
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Per-column entry: ordered card ids plus an optional `hidden` flag that
|
|
17
|
+
* removes the column from the rendered board (and from the model's column
|
|
18
|
+
* list). Today only the uncategorized column is exposed in settings, but
|
|
19
|
+
* the data structure supports per-column hiding generally.
|
|
20
|
+
*/
|
|
21
|
+
const ArrangementColumnEntry = Schema.Struct({
|
|
22
|
+
ids: Schema.Array(Obj.ID),
|
|
23
|
+
hidden: Schema.Boolean.pipe(Schema.optional),
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
/** Keyed by columnValue. */
|
|
27
|
+
const ArrangementColumns = Schema.Record({
|
|
28
|
+
key: Schema.String,
|
|
29
|
+
value: ArrangementColumnEntry,
|
|
30
|
+
}).pipe(FormInputAnnotation.set(false));
|
|
31
|
+
|
|
32
|
+
/** Column order and per-column card ids. */
|
|
33
|
+
export const Arrangement = Schema.Struct({
|
|
34
|
+
order: Schema.Array(Schema.String).pipe(FormInputAnnotation.set(false)),
|
|
35
|
+
columns: ArrangementColumns,
|
|
36
|
+
}).pipe(FormInputAnnotation.set(false));
|
|
37
|
+
|
|
38
|
+
export type Arrangement = Schema.Schema.Type<typeof Arrangement>;
|
|
39
|
+
|
|
40
|
+
//
|
|
41
|
+
// Mirrors the canonical DXOS pattern (see `Trigger.Spec` in
|
|
42
|
+
// `@dxos/functions/src/types/Trigger.ts` and `Sequence.Source` in
|
|
43
|
+
// `@dxos/plugin-zen`): the `Type.makeObject` schema is a flat `Schema.Struct`,
|
|
44
|
+
// and the discriminated union lives one level down as a single field whose
|
|
45
|
+
// variants are tagged with a `kind` literal.
|
|
46
|
+
//
|
|
47
|
+
|
|
48
|
+
/** View-variant: items come from running the View's query (the original behaviour). */
|
|
49
|
+
export const KanbanViewSpec = Schema.Struct({
|
|
50
|
+
kind: Schema.Literal('view').pipe(FormInputAnnotation.set(false)),
|
|
51
|
+
view: Ref.Ref(View.View).pipe(FormInputAnnotation.set(false)),
|
|
52
|
+
});
|
|
53
|
+
export type KanbanViewSpec = Schema.Schema.Type<typeof KanbanViewSpec>;
|
|
54
|
+
|
|
55
|
+
/** Items-variant: kanban owns its items as an explicit ref array (used by externally-synced kanbans). */
|
|
56
|
+
export const KanbanItemsSpec = Schema.Struct({
|
|
57
|
+
kind: Schema.Literal('items').pipe(FormInputAnnotation.set(false)),
|
|
58
|
+
/** Property path on each item that drives column membership (e.g. `'listName'`). */
|
|
59
|
+
pivotField: Schema.String,
|
|
60
|
+
/** Items owned directly by the kanban. */
|
|
61
|
+
items: Schema.Array(Ref.Ref(Obj.Unknown)).pipe(FormInputAnnotation.set(false)),
|
|
62
|
+
});
|
|
63
|
+
export type KanbanItemsSpec = Schema.Schema.Type<typeof KanbanItemsSpec>;
|
|
64
|
+
|
|
65
|
+
/** Discriminated union of source specs. Distinguished by `kind`. */
|
|
66
|
+
export const KanbanSpec = Schema.Union(KanbanViewSpec, KanbanItemsSpec);
|
|
67
|
+
export type KanbanSpec = Schema.Schema.Type<typeof KanbanSpec>;
|
|
68
|
+
|
|
69
|
+
export const Kanban = Schema.Struct({
|
|
70
|
+
name: Schema.String.pipe(Schema.optional),
|
|
71
|
+
arrangement: Arrangement,
|
|
72
|
+
/** How this kanban sources its items. Discriminated by `spec.kind`. */
|
|
73
|
+
spec: KanbanSpec,
|
|
74
|
+
}).pipe(
|
|
75
|
+
LabelAnnotation.set(['name']),
|
|
76
|
+
ViewAnnotation.set(['spec', 'view']),
|
|
77
|
+
Annotation.IconAnnotation.set({ icon: 'ph--kanban--regular', hue: 'green' }),
|
|
78
|
+
Type.makeObject(DXN.make('org.dxos.type.kanban', '0.2.0')),
|
|
79
|
+
);
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Instance type; narrow on `kanban.spec.kind` (or use the guards below).
|
|
83
|
+
*/
|
|
84
|
+
export interface Kanban extends Type.InstanceType<typeof Kanban> {}
|
|
85
|
+
|
|
86
|
+
/** Narrowed view-variant kanban. */
|
|
87
|
+
export type KanbanView = Kanban & { spec: KanbanViewSpec };
|
|
88
|
+
|
|
89
|
+
/** Narrowed items-variant kanban. */
|
|
90
|
+
export type KanbanItems = Kanban & { spec: KanbanItemsSpec };
|
|
91
|
+
|
|
92
|
+
export const isKanbanView = (kanban: Kanban): kanban is KanbanView => kanban.spec.kind === 'view';
|
|
93
|
+
export const isKanbanItems = (kanban: Kanban): kanban is KanbanItems => kanban.spec.kind === 'items';
|
|
94
|
+
|
|
95
|
+
type MakeViewProps = {
|
|
96
|
+
name?: string;
|
|
97
|
+
view: View.View;
|
|
98
|
+
arrangement?: Arrangement;
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
/**
|
|
102
|
+
* Make a view-variant kanban (items sourced via the View's query).
|
|
103
|
+
*/
|
|
104
|
+
export const make = (props: MakeViewProps): Kanban => {
|
|
105
|
+
const { name, view, arrangement } = props;
|
|
106
|
+
const order = arrangement?.order ?? [];
|
|
107
|
+
const columns = arrangement?.columns ?? {};
|
|
108
|
+
return Obj.make(Kanban, {
|
|
109
|
+
name,
|
|
110
|
+
arrangement: { order, columns },
|
|
111
|
+
spec: { kind: 'view' as const, view: Ref.make(view) },
|
|
112
|
+
});
|
|
113
|
+
};
|
|
114
|
+
|
|
115
|
+
type MakeItemsProps = {
|
|
116
|
+
name?: string;
|
|
117
|
+
arrangement?: Arrangement;
|
|
118
|
+
pivotField: string;
|
|
119
|
+
items?: ReadonlyArray<Ref.Ref<Obj.Unknown>>;
|
|
120
|
+
};
|
|
121
|
+
|
|
122
|
+
/**
|
|
123
|
+
* Make an items-variant kanban (items list owned by the kanban itself, e.g. populated by a sync integration).
|
|
124
|
+
*/
|
|
125
|
+
export const makeItems = (props: MakeItemsProps): Kanban => {
|
|
126
|
+
const { name, arrangement, pivotField, items = [] } = props;
|
|
127
|
+
const order = arrangement?.order ?? [];
|
|
128
|
+
const columns = arrangement?.columns ?? {};
|
|
129
|
+
return Obj.make(Kanban, {
|
|
130
|
+
name,
|
|
131
|
+
arrangement: { order, columns },
|
|
132
|
+
spec: { kind: 'items' as const, pivotField, items },
|
|
133
|
+
});
|
|
134
|
+
};
|