@dxos/plugin-kanban 0.8.4-main.ae835ea → 0.8.4-main.bc2380dfbc
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +102 -5
- package/README.md +1 -1
- package/dist/lib/neutral/KanbanArticle-NEJ3LNBO.mjs +132 -0
- package/dist/lib/neutral/KanbanArticle-NEJ3LNBO.mjs.map +7 -0
- package/dist/lib/neutral/KanbanPlugin.mjs +36 -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-G6M47NSK.mjs +83 -0
- package/dist/lib/neutral/KanbanSettings-G6M47NSK.mjs.map +7 -0
- package/dist/lib/neutral/blueprint-definition-FHVIEKTQ.mjs +15 -0
- package/dist/lib/neutral/blueprint-definition-FHVIEKTQ.mjs.map +7 -0
- package/dist/lib/neutral/blueprints/index.mjs +8 -0
- package/dist/lib/neutral/capabilities/index.mjs +19 -0
- package/dist/lib/neutral/capabilities/index.mjs.map +7 -0
- package/dist/lib/neutral/chunk-6FPBLOS3.mjs +8 -0
- package/dist/lib/neutral/chunk-6FPBLOS3.mjs.map +7 -0
- package/dist/lib/neutral/chunk-J5LGTIGS.mjs +10 -0
- package/dist/lib/neutral/chunk-KDPM77BD.mjs +21 -0
- package/dist/lib/neutral/chunk-KDPM77BD.mjs.map +7 -0
- package/dist/lib/neutral/chunk-OQ72EEGU.mjs +254 -0
- package/dist/lib/neutral/chunk-OQ72EEGU.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-356CBACE.mjs +24 -0
- package/dist/lib/neutral/delete-card-356CBACE.mjs.map +7 -0
- package/dist/lib/neutral/delete-card-field-IRCZL2BR.mjs +42 -0
- package/dist/lib/neutral/delete-card-field-IRCZL2BR.mjs.map +7 -0
- package/dist/lib/neutral/hooks/index.mjs +432 -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/migrations-IWBT35UT.mjs +31 -0
- package/dist/lib/neutral/migrations-IWBT35UT.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-QROEHBRW.mjs +93 -0
- package/dist/lib/neutral/react-surface-QROEHBRW.mjs.map +7 -0
- package/dist/lib/neutral/restore-card-P25Y4YSE.mjs +21 -0
- package/dist/lib/neutral/restore-card-P25Y4YSE.mjs.map +7 -0
- package/dist/lib/neutral/restore-card-field-NHR3R4XI.mjs +40 -0
- package/dist/lib/neutral/restore-card-field-NHR3R4XI.mjs.map +7 -0
- package/dist/lib/neutral/testing/index.mjs +60 -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 +5 -4
- package/dist/types/src/capabilities/blueprint-definition.d.ts.map +1 -1
- 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 -3
- package/dist/types/src/capabilities/index.d.ts.map +1 -1
- package/dist/types/src/capabilities/migrations.d.ts +6 -0
- package/dist/types/src/capabilities/migrations.d.ts.map +1 -0
- 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 -2
- 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 +108 -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 -0
- package/dist/types/src/types/index.d.ts.map +1 -1
- package/dist/types/src/types/migrations.test.d.ts +2 -0
- package/dist/types/src/types/migrations.test.d.ts.map +1 -0
- package/dist/types/src/types/schema.d.ts +17 -51
- 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 +123 -57
- package/src/KanbanPlugin.node.ts +21 -0
- package/src/KanbanPlugin.test.ts +31 -0
- package/src/KanbanPlugin.tsx +28 -56
- 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 +116 -114
- package/src/capabilities/blueprint-definition.ts +13 -24
- package/src/capabilities/create-object.ts +40 -0
- package/src/capabilities/index.ts +12 -4
- package/src/capabilities/migrations.ts +35 -0
- package/src/capabilities/operation-handler.ts +14 -0
- package/src/capabilities/react-surface.tsx +77 -67
- 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 +276 -0
- package/src/containers/KanbanArticle/KanbanArticle.tsx +178 -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 +234 -0
- package/src/hooks/useKanbanBoardModel.ts +157 -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 +3 -3
- package/src/operations/delete-card-field.ts +47 -0
- package/src/operations/delete-card.ts +23 -0
- package/src/operations/index.ts +10 -0
- package/src/operations/restore-card-field.ts +41 -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 +150 -0
- package/src/types/KanbanOperation.ts +67 -0
- package/src/types/constants.ts +9 -0
- package/src/types/index.ts +4 -0
- package/src/types/migrations.test.ts +82 -0
- package/src/types/schema.ts +32 -44
- 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/dist/lib/browser/blueprint-definition-UYVX622Q.mjs +0 -28
- package/dist/lib/browser/blueprint-definition-UYVX622Q.mjs.map +0 -7
- package/dist/lib/browser/chunk-3UDST345.mjs +0 -85
- package/dist/lib/browser/chunk-3UDST345.mjs.map +0 -7
- package/dist/lib/browser/index.mjs +0 -105
- package/dist/lib/browser/index.mjs.map +0 -7
- package/dist/lib/browser/intent-resolver-VVBNS2TO.mjs +0 -111
- package/dist/lib/browser/intent-resolver-VVBNS2TO.mjs.map +0 -7
- package/dist/lib/browser/meta.json +0 -1
- package/dist/lib/browser/react-surface-FNXJ6VJX.mjs +0 -255
- package/dist/lib/browser/react-surface-FNXJ6VJX.mjs.map +0 -7
- package/dist/lib/browser/types/index.mjs +0 -11
- package/dist/lib/node-esm/blueprint-definition-42P47FUY.mjs +0 -30
- package/dist/lib/node-esm/blueprint-definition-42P47FUY.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-JBOARUAT.mjs +0 -87
- package/dist/lib/node-esm/chunk-JBOARUAT.mjs.map +0 -7
- package/dist/lib/node-esm/index.mjs +0 -106
- package/dist/lib/node-esm/index.mjs.map +0 -7
- package/dist/lib/node-esm/intent-resolver-ACN7UALP.mjs +0 -112
- package/dist/lib/node-esm/intent-resolver-ACN7UALP.mjs.map +0 -7
- package/dist/lib/node-esm/meta.json +0 -1
- package/dist/lib/node-esm/react-surface-ZHYHCV5N.mjs +0 -256
- package/dist/lib/node-esm/react-surface-ZHYHCV5N.mjs.map +0 -7
- package/dist/lib/node-esm/types/index.mjs +0 -12
- 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 -41
- 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/src/capabilities/intent-resolver.ts +0 -71
- package/src/components/KanbanContainer.stories.tsx +0 -193
- package/src/components/KanbanContainer.tsx +0 -95
- package/src/components/KanbanViewEditor.tsx +0 -64
- /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/index.ts
CHANGED
package/src/meta.ts
CHANGED
|
@@ -2,11 +2,11 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import { type
|
|
5
|
+
import { type Plugin } from '@dxos/app-framework';
|
|
6
6
|
import { trim } from '@dxos/util';
|
|
7
7
|
|
|
8
|
-
export const meta:
|
|
9
|
-
id: 'dxos.
|
|
8
|
+
export const meta: Plugin.Meta = {
|
|
9
|
+
id: 'org.dxos.plugin.kanban',
|
|
10
10
|
name: 'Kanban',
|
|
11
11
|
description: trim`
|
|
12
12
|
Visual project management using customizable kanban boards to track workflow progress.
|
|
@@ -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 { Operation } from '@dxos/compute';
|
|
7
|
+
import { JsonSchema, Obj } from '@dxos/echo';
|
|
8
|
+
import { type EchoSchema } from '@dxos/echo/internal';
|
|
9
|
+
import { invariant } from '@dxos/invariant';
|
|
10
|
+
import { ProjectionModel, createEchoChangeCallback, getTypenameFromQuery } from '@dxos/schema';
|
|
11
|
+
|
|
12
|
+
import { KanbanOperation } from '../types';
|
|
13
|
+
|
|
14
|
+
const handler: Operation.WithHandler<typeof KanbanOperation.DeleteCardField> = KanbanOperation.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 { 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,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 { Operation } from '@dxos/compute';
|
|
7
|
+
import { JsonSchema, Obj } from '@dxos/echo';
|
|
8
|
+
import { type EchoSchema } from '@dxos/echo/internal';
|
|
9
|
+
import { invariant } from '@dxos/invariant';
|
|
10
|
+
import { ProjectionModel, createEchoChangeCallback, getTypenameFromQuery } from '@dxos/schema';
|
|
11
|
+
|
|
12
|
+
import { KanbanOperation } from '../types';
|
|
13
|
+
|
|
14
|
+
const handler: Operation.WithHandler<typeof KanbanOperation.RestoreCardField> = KanbanOperation.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 { 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.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;
|
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 { Kanban } from '@dxos/react-ui-kanban/types';
|
|
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
|
-
[Kanban.Kanban
|
|
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,150 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
6
|
+
|
|
7
|
+
import { Annotation, Obj, Ref, Type, View } from '@dxos/echo';
|
|
8
|
+
import { FormInputAnnotation, LabelAnnotation } from '@dxos/echo/internal';
|
|
9
|
+
import { ViewAnnotation } from '@dxos/schema';
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Per-column entry: ordered card ids plus an optional `hidden` flag that
|
|
13
|
+
* removes the column from the rendered board (and from the model's column
|
|
14
|
+
* list). Today only the uncategorized column is exposed in settings, but
|
|
15
|
+
* the data structure supports per-column hiding generally.
|
|
16
|
+
*/
|
|
17
|
+
const ArrangementColumnEntry = Schema.Struct({
|
|
18
|
+
ids: Schema.Array(Obj.ID),
|
|
19
|
+
hidden: Schema.Boolean.pipe(Schema.optional),
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
/** Keyed by columnValue. */
|
|
23
|
+
const ArrangementColumns = Schema.Record({
|
|
24
|
+
key: Schema.String,
|
|
25
|
+
value: ArrangementColumnEntry,
|
|
26
|
+
}).pipe(FormInputAnnotation.set(false));
|
|
27
|
+
|
|
28
|
+
/** Column order and per-column card ids. */
|
|
29
|
+
export const Arrangement = Schema.Struct({
|
|
30
|
+
order: Schema.Array(Schema.String).pipe(FormInputAnnotation.set(false)),
|
|
31
|
+
columns: ArrangementColumns,
|
|
32
|
+
}).pipe(FormInputAnnotation.set(false));
|
|
33
|
+
|
|
34
|
+
export type Arrangement = Schema.Schema.Type<typeof Arrangement>;
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* v1: pre-existing Kanban shape. Retained as the source for the v1→v2 migration.
|
|
38
|
+
*/
|
|
39
|
+
export const KanbanV1 = Schema.Struct({
|
|
40
|
+
name: Schema.String.pipe(Schema.optional),
|
|
41
|
+
view: Ref.Ref(View.View).pipe(FormInputAnnotation.set(false)),
|
|
42
|
+
arrangement: Arrangement,
|
|
43
|
+
}).pipe(
|
|
44
|
+
Type.object({
|
|
45
|
+
typename: 'org.dxos.type.kanban',
|
|
46
|
+
version: '0.1.0',
|
|
47
|
+
}),
|
|
48
|
+
);
|
|
49
|
+
|
|
50
|
+
//
|
|
51
|
+
// v2 — `spec` is a discriminated union of how items are sourced.
|
|
52
|
+
//
|
|
53
|
+
// Mirrors the canonical DXOS pattern (see `Trigger.Spec` in
|
|
54
|
+
// `@dxos/functions/src/types/Trigger.ts` and `Sequence.Source` in
|
|
55
|
+
// `@dxos/plugin-zen`): the `Type.object` schema is a flat `Schema.Struct`,
|
|
56
|
+
// and the discriminated union lives one level down as a single field whose
|
|
57
|
+
// variants are tagged with a `kind` literal.
|
|
58
|
+
//
|
|
59
|
+
|
|
60
|
+
/** View-variant: items come from running the View's query (the original behaviour). */
|
|
61
|
+
export const KanbanViewSpec = Schema.Struct({
|
|
62
|
+
kind: Schema.Literal('view').pipe(FormInputAnnotation.set(false)),
|
|
63
|
+
view: Ref.Ref(View.View).pipe(FormInputAnnotation.set(false)),
|
|
64
|
+
});
|
|
65
|
+
export type KanbanViewSpec = Schema.Schema.Type<typeof KanbanViewSpec>;
|
|
66
|
+
|
|
67
|
+
/** Items-variant: kanban owns its items as an explicit ref array (used by externally-synced kanbans). */
|
|
68
|
+
export const KanbanItemsSpec = Schema.Struct({
|
|
69
|
+
kind: Schema.Literal('items').pipe(FormInputAnnotation.set(false)),
|
|
70
|
+
/** Property path on each item that drives column membership (e.g. `'listName'`). */
|
|
71
|
+
pivotField: Schema.String,
|
|
72
|
+
/** Items owned directly by the kanban. */
|
|
73
|
+
items: Schema.Array(Ref.Ref(Obj.Unknown)).pipe(FormInputAnnotation.set(false)),
|
|
74
|
+
});
|
|
75
|
+
export type KanbanItemsSpec = Schema.Schema.Type<typeof KanbanItemsSpec>;
|
|
76
|
+
|
|
77
|
+
/** Discriminated union of source specs. Distinguished by `kind`. */
|
|
78
|
+
export const KanbanSpec = Schema.Union(KanbanViewSpec, KanbanItemsSpec);
|
|
79
|
+
export type KanbanSpec = Schema.Schema.Type<typeof KanbanSpec>;
|
|
80
|
+
|
|
81
|
+
export const Kanban = Schema.Struct({
|
|
82
|
+
name: Schema.String.pipe(Schema.optional),
|
|
83
|
+
arrangement: Arrangement,
|
|
84
|
+
/** How this kanban sources its items. Discriminated by `spec.kind`. */
|
|
85
|
+
spec: KanbanSpec,
|
|
86
|
+
}).pipe(
|
|
87
|
+
Type.object({
|
|
88
|
+
typename: 'org.dxos.type.kanban',
|
|
89
|
+
version: '0.2.0',
|
|
90
|
+
}),
|
|
91
|
+
LabelAnnotation.set(['name']),
|
|
92
|
+
ViewAnnotation.set(['spec', 'view']),
|
|
93
|
+
Annotation.IconAnnotation.set({
|
|
94
|
+
icon: 'ph--kanban--regular',
|
|
95
|
+
hue: 'green',
|
|
96
|
+
}),
|
|
97
|
+
);
|
|
98
|
+
|
|
99
|
+
/** Instance type; narrow on `kanban.spec.kind` (or use the guards below). */
|
|
100
|
+
export interface Kanban extends Schema.Schema.Type<typeof Kanban> {}
|
|
101
|
+
|
|
102
|
+
/** Narrowed view-variant kanban. */
|
|
103
|
+
export type KanbanView = Kanban & { spec: KanbanViewSpec };
|
|
104
|
+
|
|
105
|
+
/** Narrowed items-variant kanban. */
|
|
106
|
+
export type KanbanItems = Kanban & { spec: KanbanItemsSpec };
|
|
107
|
+
|
|
108
|
+
export const isKanbanView = (kanban: Kanban): kanban is KanbanView => kanban.spec.kind === 'view';
|
|
109
|
+
export const isKanbanItems = (kanban: Kanban): kanban is KanbanItems => kanban.spec.kind === 'items';
|
|
110
|
+
|
|
111
|
+
type MakeViewProps = {
|
|
112
|
+
name?: string;
|
|
113
|
+
view: View.View;
|
|
114
|
+
arrangement?: Arrangement;
|
|
115
|
+
};
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Make a view-variant kanban (items sourced via the View's query).
|
|
119
|
+
*/
|
|
120
|
+
export const make = (props: MakeViewProps): Kanban => {
|
|
121
|
+
const { name, view, arrangement } = props;
|
|
122
|
+
const order = arrangement?.order ?? [];
|
|
123
|
+
const columns = arrangement?.columns ?? {};
|
|
124
|
+
return Obj.make(Kanban, {
|
|
125
|
+
name,
|
|
126
|
+
arrangement: { order, columns },
|
|
127
|
+
spec: { kind: 'view' as const, view: Ref.make(view) },
|
|
128
|
+
});
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
type MakeItemsProps = {
|
|
132
|
+
name?: string;
|
|
133
|
+
arrangement?: Arrangement;
|
|
134
|
+
pivotField: string;
|
|
135
|
+
items?: ReadonlyArray<Ref.Ref<Obj.Unknown>>;
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Make an items-variant kanban (items list owned by the kanban itself, e.g. populated by a sync integration).
|
|
140
|
+
*/
|
|
141
|
+
export const makeItems = (props: MakeItemsProps): Kanban => {
|
|
142
|
+
const { name, arrangement, pivotField, items = [] } = props;
|
|
143
|
+
const order = arrangement?.order ?? [];
|
|
144
|
+
const columns = arrangement?.columns ?? {};
|
|
145
|
+
return Obj.make(Kanban, {
|
|
146
|
+
name,
|
|
147
|
+
arrangement: { order, columns },
|
|
148
|
+
spec: { kind: 'items' as const, pivotField, items },
|
|
149
|
+
});
|
|
150
|
+
};
|