@dxos/plugin-kanban 0.7.4 → 0.7.5-labs.5f04cf6
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/dist/lib/browser/chunk-FAA5WRZY.mjs +69 -0
- package/dist/lib/browser/chunk-FAA5WRZY.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +66 -145
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/intent-resolver-EWY2KVLT.mjs +70 -0
- package/dist/lib/browser/intent-resolver-EWY2KVLT.mjs.map +7 -0
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/react-surface-47DT36V2.mjs +185 -0
- package/dist/lib/browser/react-surface-47DT36V2.mjs.map +7 -0
- package/dist/lib/browser/types.mjs +11 -0
- package/dist/lib/node/chunk-PLKK6ZJE.cjs +91 -0
- package/dist/lib/node/chunk-PLKK6ZJE.cjs.map +7 -0
- package/dist/lib/node/index.cjs +62 -133
- package/dist/lib/node/index.cjs.map +4 -4
- package/dist/lib/node/intent-resolver-74D63LYS.cjs +84 -0
- package/dist/lib/node/intent-resolver-74D63LYS.cjs.map +7 -0
- package/dist/lib/node/meta.json +1 -1
- package/dist/lib/node/react-surface-V4HKW7LY.cjs +203 -0
- package/dist/lib/node/react-surface-V4HKW7LY.cjs.map +7 -0
- package/dist/lib/node/{types/index.cjs → types.cjs} +7 -10
- package/dist/lib/node/types.cjs.map +7 -0
- package/dist/lib/node-esm/chunk-3753HLJC.mjs +71 -0
- package/dist/lib/node-esm/chunk-3753HLJC.mjs.map +7 -0
- package/dist/lib/node-esm/index.mjs +66 -145
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/intent-resolver-UAORS6YK.mjs +71 -0
- package/dist/lib/node-esm/intent-resolver-UAORS6YK.mjs.map +7 -0
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/react-surface-O5OUSQDL.mjs +186 -0
- package/dist/lib/node-esm/react-surface-O5OUSQDL.mjs.map +7 -0
- package/dist/lib/node-esm/types.mjs +12 -0
- package/dist/types/src/KanbanPlugin.d.ts +1 -3
- package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
- package/dist/types/src/capabilities/index.d.ts +3 -0
- package/dist/types/src/capabilities/index.d.ts.map +1 -0
- package/dist/types/src/capabilities/intent-resolver.d.ts +4 -0
- package/dist/types/src/capabilities/intent-resolver.d.ts.map +1 -0
- package/dist/types/src/capabilities/react-surface.d.ts +4 -0
- package/dist/types/src/capabilities/react-surface.d.ts.map +1 -0
- package/dist/types/src/components/KanbanContainer.d.ts +7 -0
- package/dist/types/src/components/KanbanContainer.d.ts.map +1 -0
- package/dist/types/src/components/KanbanViewEditor.d.ts +8 -0
- package/dist/types/src/components/KanbanViewEditor.d.ts.map +1 -0
- package/dist/types/src/components/index.d.ts +2 -3
- package/dist/types/src/components/index.d.ts.map +1 -1
- package/dist/types/src/index.d.ts +1 -2
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/meta.d.ts +1 -2
- package/dist/types/src/meta.d.ts.map +1 -1
- package/dist/types/src/translations.d.ts +32 -0
- package/dist/types/src/translations.d.ts.map +1 -1
- package/dist/types/src/types.d.ts +67 -0
- package/dist/types/src/types.d.ts.map +1 -0
- package/dist/types/tsconfig.tsbuildinfo +1 -0
- package/package.json +21 -31
- package/src/KanbanPlugin.tsx +49 -101
- package/src/capabilities/index.ts +8 -0
- package/src/capabilities/intent-resolver.ts +39 -0
- package/src/capabilities/react-surface.tsx +28 -0
- package/src/components/KanbanContainer.tsx +82 -0
- package/src/components/KanbanViewEditor.tsx +88 -0
- package/src/components/index.ts +3 -4
- package/src/index.ts +1 -4
- package/src/meta.ts +1 -1
- package/src/translations.ts +8 -2
- package/src/types.ts +75 -0
- package/dist/lib/browser/KanbanMain-I5TMXNIY.mjs +0 -444
- package/dist/lib/browser/KanbanMain-I5TMXNIY.mjs.map +0 -7
- package/dist/lib/browser/chunk-4Y4TZ47E.mjs +0 -47
- package/dist/lib/browser/chunk-4Y4TZ47E.mjs.map +0 -7
- package/dist/lib/browser/chunk-LG4OMN5S.mjs +0 -18
- package/dist/lib/browser/chunk-LG4OMN5S.mjs.map +0 -7
- package/dist/lib/browser/meta.mjs +0 -9
- package/dist/lib/browser/types/index.mjs +0 -14
- package/dist/lib/node/KanbanMain-4OWAWTS4.cjs +0 -453
- package/dist/lib/node/KanbanMain-4OWAWTS4.cjs.map +0 -7
- package/dist/lib/node/chunk-LTR4WYI2.cjs +0 -67
- package/dist/lib/node/chunk-LTR4WYI2.cjs.map +0 -7
- package/dist/lib/node/chunk-MBAGHRFM.cjs +0 -41
- package/dist/lib/node/chunk-MBAGHRFM.cjs.map +0 -7
- package/dist/lib/node/meta.cjs +0 -30
- package/dist/lib/node/meta.cjs.map +0 -7
- package/dist/lib/node/types/index.cjs.map +0 -7
- package/dist/lib/node-esm/KanbanMain-PJC2JMJH.mjs +0 -445
- package/dist/lib/node-esm/KanbanMain-PJC2JMJH.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-2ZBX5F7L.mjs +0 -48
- package/dist/lib/node-esm/chunk-2ZBX5F7L.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-OTZHYV3S.mjs +0 -20
- package/dist/lib/node-esm/chunk-OTZHYV3S.mjs.map +0 -7
- package/dist/lib/node-esm/meta.mjs +0 -10
- package/dist/lib/node-esm/meta.mjs.map +0 -7
- package/dist/lib/node-esm/types/index.mjs +0 -15
- package/dist/lib/node-esm/types/index.mjs.map +0 -7
- package/dist/types/src/components/KanbanBoard.d.ts +0 -6
- package/dist/types/src/components/KanbanBoard.d.ts.map +0 -1
- package/dist/types/src/components/KanbanCard.d.ts +0 -9
- package/dist/types/src/components/KanbanCard.d.ts.map +0 -1
- package/dist/types/src/components/KanbanColumn.d.ts +0 -14
- package/dist/types/src/components/KanbanColumn.d.ts.map +0 -1
- package/dist/types/src/components/KanbanMain.d.ts +0 -7
- package/dist/types/src/components/KanbanMain.d.ts.map +0 -1
- package/dist/types/src/components/util.d.ts +0 -7
- package/dist/types/src/components/util.d.ts.map +0 -1
- package/dist/types/src/stories/testing.d.ts +0 -19
- package/dist/types/src/stories/testing.d.ts.map +0 -1
- package/dist/types/src/types/index.d.ts +0 -3
- package/dist/types/src/types/index.d.ts.map +0 -1
- package/dist/types/src/types/kanban.d.ts +0 -76
- package/dist/types/src/types/kanban.d.ts.map +0 -1
- package/dist/types/src/types/types.d.ts +0 -18
- package/dist/types/src/types/types.d.ts.map +0 -1
- package/src/components/KanbanBoard.tsx +0 -195
- package/src/components/KanbanCard.tsx +0 -82
- package/src/components/KanbanColumn.tsx +0 -143
- package/src/components/KanbanMain.tsx +0 -37
- package/src/components/util.ts +0 -38
- package/src/stories/testing.ts +0 -29
- package/src/types/index.ts +0 -6
- package/src/types/kanban.ts +0 -22
- package/src/types/types.ts +0 -57
- /package/dist/lib/browser/{meta.mjs.map → types.mjs.map} +0 -0
- /package/dist/lib/{browser/types/index.mjs.map → node-esm/types.mjs.map} +0 -0
package/src/KanbanPlugin.tsx
CHANGED
|
@@ -2,110 +2,58 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import
|
|
5
|
+
import {
|
|
6
|
+
createIntent,
|
|
7
|
+
defineModule,
|
|
8
|
+
contributes,
|
|
9
|
+
Capabilities,
|
|
10
|
+
Events,
|
|
11
|
+
definePlugin,
|
|
12
|
+
oneOf,
|
|
13
|
+
} from '@dxos/app-framework';
|
|
14
|
+
import { ClientCapabilities, ClientEvents } from '@dxos/plugin-client';
|
|
15
|
+
import { type Space } from '@dxos/react-client/echo';
|
|
16
|
+
import { KanbanType, translations as kanbanTranslations } from '@dxos/react-ui-kanban';
|
|
6
17
|
|
|
7
|
-
import {
|
|
8
|
-
import {
|
|
9
|
-
import { parseClientPlugin } from '@dxos/plugin-client';
|
|
10
|
-
import { type ActionGroup, createExtension, isActionGroup } from '@dxos/plugin-graph';
|
|
11
|
-
import { SpaceAction } from '@dxos/plugin-space';
|
|
12
|
-
import { loadObjectReferences } from '@dxos/react-client/echo';
|
|
13
|
-
|
|
14
|
-
import { KanbanMain } from './components';
|
|
15
|
-
import meta, { KANBAN_PLUGIN } from './meta';
|
|
18
|
+
import { IntentResolver, ReactSurface } from './capabilities';
|
|
19
|
+
import { KANBAN_PLUGIN, meta } from './meta';
|
|
16
20
|
import translations from './translations';
|
|
17
|
-
import {
|
|
18
|
-
import { KanbanAction, type KanbanPluginProvides } from './types';
|
|
21
|
+
import { KanbanAction } from './types';
|
|
19
22
|
|
|
20
|
-
export const KanbanPlugin = ()
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
23
|
+
export const KanbanPlugin = () =>
|
|
24
|
+
definePlugin(meta, [
|
|
25
|
+
defineModule({
|
|
26
|
+
id: `${meta.id}/module/translations`,
|
|
27
|
+
activatesOn: Events.SetupTranslations,
|
|
28
|
+
activate: () => contributes(Capabilities.Translations, [...translations, ...kanbanTranslations]),
|
|
29
|
+
}),
|
|
30
|
+
defineModule({
|
|
31
|
+
id: `${meta.id}/module/metadata`,
|
|
32
|
+
activatesOn: oneOf(Events.Startup, Events.SetupAppGraph),
|
|
33
|
+
activate: () =>
|
|
34
|
+
contributes(Capabilities.Metadata, {
|
|
35
|
+
id: KanbanType.typename,
|
|
36
|
+
metadata: {
|
|
37
|
+
createObject: (props: { name?: string }, options: { space: Space }) =>
|
|
38
|
+
createIntent(KanbanAction.Create, { ...props, space: options.space }),
|
|
28
39
|
placeholder: ['kanban title placeholder', { ns: KANBAN_PLUGIN }],
|
|
29
40
|
icon: 'ph--kanban--regular',
|
|
30
|
-
// TODO(wittjosiah): Move out of metadata.
|
|
31
|
-
loadReferences: (kanban: KanbanType) => loadObjectReferences(kanban, (kanban) => kanban.columns),
|
|
32
|
-
},
|
|
33
|
-
[KanbanColumnType.typename]: {
|
|
34
|
-
// TODO(wittjosiah): Move out of metadata.
|
|
35
|
-
loadReferences: (column: KanbanColumnType) => loadObjectReferences(column, (column) => column.items),
|
|
36
|
-
},
|
|
37
|
-
[KanbanItemType.typename]: {
|
|
38
|
-
// TODO(wittjosiah): Move out of metadata.
|
|
39
|
-
loadReferences: (item: KanbanItemType) => [], // loadObjectReferences(item, (item) => item.object),
|
|
40
41
|
},
|
|
41
|
-
},
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
actions: ({ node }) => {
|
|
60
|
-
const id = node.id.split('/').at(-1);
|
|
61
|
-
const [spaceId, objectId] = id?.split(':') ?? [];
|
|
62
|
-
const space = client.spaces.get().find((space) => space.id === spaceId);
|
|
63
|
-
const object = objectId && space?.db.getObjectById(objectId);
|
|
64
|
-
const target = objectId ? object : space;
|
|
65
|
-
if (!target) {
|
|
66
|
-
return;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
return [
|
|
70
|
-
{
|
|
71
|
-
id: `${KANBAN_PLUGIN}/create/${node.id}`,
|
|
72
|
-
data: async () => {
|
|
73
|
-
await dispatch([
|
|
74
|
-
{ plugin: KANBAN_PLUGIN, action: KanbanAction.CREATE },
|
|
75
|
-
{ action: SpaceAction.ADD_OBJECT, data: { target } },
|
|
76
|
-
{ action: NavigationAction.OPEN },
|
|
77
|
-
]);
|
|
78
|
-
},
|
|
79
|
-
properties: {
|
|
80
|
-
label: ['create kanban label', { ns: KANBAN_PLUGIN }],
|
|
81
|
-
icon: 'ph--kanban--regular',
|
|
82
|
-
testId: 'kanbanPlugin.createObject',
|
|
83
|
-
},
|
|
84
|
-
},
|
|
85
|
-
];
|
|
86
|
-
},
|
|
87
|
-
});
|
|
88
|
-
},
|
|
89
|
-
},
|
|
90
|
-
surface: {
|
|
91
|
-
component: ({ data, role }) => {
|
|
92
|
-
switch (role) {
|
|
93
|
-
case 'main':
|
|
94
|
-
return data.active instanceof KanbanType ? <KanbanMain kanban={data.active} /> : null;
|
|
95
|
-
default:
|
|
96
|
-
return null;
|
|
97
|
-
}
|
|
98
|
-
},
|
|
99
|
-
},
|
|
100
|
-
intent: {
|
|
101
|
-
resolver: (intent) => {
|
|
102
|
-
switch (intent.action) {
|
|
103
|
-
case KanbanAction.CREATE: {
|
|
104
|
-
return { data: create(KanbanType, { columns: [] }) };
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
},
|
|
108
|
-
},
|
|
109
|
-
},
|
|
110
|
-
};
|
|
111
|
-
};
|
|
42
|
+
}),
|
|
43
|
+
}),
|
|
44
|
+
defineModule({
|
|
45
|
+
id: `${meta.id}/module/schema`,
|
|
46
|
+
activatesOn: ClientEvents.SetupClient,
|
|
47
|
+
activate: () => contributes(ClientCapabilities.Schema, [KanbanType]),
|
|
48
|
+
}),
|
|
49
|
+
defineModule({
|
|
50
|
+
id: `${meta.id}/module/react-surface`,
|
|
51
|
+
activatesOn: Events.Startup,
|
|
52
|
+
activate: ReactSurface,
|
|
53
|
+
}),
|
|
54
|
+
defineModule({
|
|
55
|
+
id: `${meta.id}/module/intent-resolver`,
|
|
56
|
+
activatesOn: Events.SetupIntents,
|
|
57
|
+
activate: IntentResolver,
|
|
58
|
+
}),
|
|
59
|
+
]);
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { contributes, Capabilities, createResolver } from '@dxos/app-framework';
|
|
6
|
+
import { invariant } from '@dxos/invariant';
|
|
7
|
+
import { getSpace } from '@dxos/react-client/echo';
|
|
8
|
+
import { ViewProjection } from '@dxos/schema';
|
|
9
|
+
|
|
10
|
+
import { KANBAN_PLUGIN } from '../meta';
|
|
11
|
+
import { createKanban, KanbanAction } from '../types';
|
|
12
|
+
|
|
13
|
+
export default () =>
|
|
14
|
+
contributes(Capabilities.IntentResolver, [
|
|
15
|
+
createResolver(KanbanAction.Create, async ({ space }) => ({
|
|
16
|
+
data: { object: await createKanban(space) },
|
|
17
|
+
})),
|
|
18
|
+
createResolver(KanbanAction.DeleteCardField, ({ kanban, fieldId, deletionData }, undo) => {
|
|
19
|
+
invariant(kanban.cardView);
|
|
20
|
+
|
|
21
|
+
const schema =
|
|
22
|
+
kanban.cardView.target && getSpace(kanban)?.db.schemaRegistry.getSchema(kanban.cardView.target.query.type);
|
|
23
|
+
invariant(schema);
|
|
24
|
+
const projection = new ViewProjection(schema, kanban.cardView.target!);
|
|
25
|
+
|
|
26
|
+
if (!undo) {
|
|
27
|
+
const { deleted, index } = projection.deleteFieldProjection(fieldId);
|
|
28
|
+
return {
|
|
29
|
+
undoable: {
|
|
30
|
+
message: ['card field deleted label', { ns: KANBAN_PLUGIN }],
|
|
31
|
+
data: { deletionData: { ...deleted, index } },
|
|
32
|
+
},
|
|
33
|
+
};
|
|
34
|
+
} else if (undo && deletionData) {
|
|
35
|
+
const { field, props, index } = deletionData;
|
|
36
|
+
projection.setFieldProjection({ field, props }, index);
|
|
37
|
+
}
|
|
38
|
+
}),
|
|
39
|
+
]);
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import React from 'react';
|
|
6
|
+
|
|
7
|
+
import { Capabilities, contributes, createSurface } from '@dxos/app-framework';
|
|
8
|
+
import { type KanbanType } from '@dxos/react-ui-kanban';
|
|
9
|
+
|
|
10
|
+
import { KanbanContainer, KanbanViewEditor } from '../components';
|
|
11
|
+
import { KANBAN_PLUGIN } from '../meta';
|
|
12
|
+
import { isKanban } from '../types';
|
|
13
|
+
|
|
14
|
+
export default () =>
|
|
15
|
+
contributes(Capabilities.ReactSurface, [
|
|
16
|
+
createSurface({
|
|
17
|
+
id: `${KANBAN_PLUGIN}/kanban`,
|
|
18
|
+
role: ['article', 'section'],
|
|
19
|
+
filter: (data): data is { subject: KanbanType } => isKanban(data.subject),
|
|
20
|
+
component: ({ data, role }) => <KanbanContainer kanban={data.subject} role={role} />,
|
|
21
|
+
}),
|
|
22
|
+
createSurface({
|
|
23
|
+
id: `${KANBAN_PLUGIN}/settings`,
|
|
24
|
+
role: 'complementary--settings',
|
|
25
|
+
filter: (data): data is { subject: KanbanType } => isKanban(data.subject),
|
|
26
|
+
component: ({ data }) => <KanbanViewEditor kanban={data.subject} />,
|
|
27
|
+
}),
|
|
28
|
+
]);
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import React, { useCallback, useEffect, useState } from 'react';
|
|
6
|
+
|
|
7
|
+
import { type EchoSchema } from '@dxos/echo-schema';
|
|
8
|
+
import { invariant } from '@dxos/invariant';
|
|
9
|
+
import { useGlobalFilteredObjects } from '@dxos/plugin-search';
|
|
10
|
+
import { Filter, useQuery, getSpace, create } from '@dxos/react-client/echo';
|
|
11
|
+
import { type KanbanType, useKanbanModel, Kanban } from '@dxos/react-ui-kanban';
|
|
12
|
+
import { StackItem } from '@dxos/react-ui-stack';
|
|
13
|
+
|
|
14
|
+
export const KanbanContainer = ({ kanban }: { kanban: KanbanType; role: string }) => {
|
|
15
|
+
const [cardSchema, setCardSchema] = useState<EchoSchema>();
|
|
16
|
+
const space = getSpace(kanban);
|
|
17
|
+
useEffect(() => {
|
|
18
|
+
if (kanban.cardView?.target?.query?.type && space) {
|
|
19
|
+
const [schema] = space.db.schemaRegistry.query({ typename: kanban.cardView.target.query.type }).runSync();
|
|
20
|
+
if (schema) {
|
|
21
|
+
setCardSchema(schema);
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
}, [kanban.cardView?.target?.query, space]);
|
|
25
|
+
|
|
26
|
+
const objects = useQuery(space, cardSchema ? Filter.schema(cardSchema) : Filter.nothing());
|
|
27
|
+
const filteredObjects = useGlobalFilteredObjects(objects);
|
|
28
|
+
|
|
29
|
+
const model = useKanbanModel({
|
|
30
|
+
kanban,
|
|
31
|
+
cardSchema,
|
|
32
|
+
items: filteredObjects,
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
const handleAddColumn = useCallback((columnValue: string) => model?.addEmptyColumn(columnValue), [model]);
|
|
36
|
+
|
|
37
|
+
const handleAddCard = useCallback(
|
|
38
|
+
(columnValue: string) => {
|
|
39
|
+
if (space && cardSchema) {
|
|
40
|
+
space.db.add(
|
|
41
|
+
create(cardSchema, {
|
|
42
|
+
title: '',
|
|
43
|
+
description: '',
|
|
44
|
+
state: columnValue,
|
|
45
|
+
}),
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
},
|
|
49
|
+
[space, cardSchema],
|
|
50
|
+
);
|
|
51
|
+
|
|
52
|
+
const handleRemoveCard = useCallback(
|
|
53
|
+
(card: { id: string }) => {
|
|
54
|
+
invariant(space);
|
|
55
|
+
space.db.remove(card);
|
|
56
|
+
},
|
|
57
|
+
[space],
|
|
58
|
+
);
|
|
59
|
+
|
|
60
|
+
const handleRemoveEmptyColumn = useCallback(
|
|
61
|
+
(columnValue: string) => {
|
|
62
|
+
model?.removeColumnFromArrangement(columnValue);
|
|
63
|
+
},
|
|
64
|
+
[model],
|
|
65
|
+
);
|
|
66
|
+
|
|
67
|
+
return (
|
|
68
|
+
<StackItem.Content toolbar={false}>
|
|
69
|
+
{model ? (
|
|
70
|
+
<Kanban
|
|
71
|
+
model={model}
|
|
72
|
+
onAddCard={handleAddCard}
|
|
73
|
+
onAddColumn={handleAddColumn}
|
|
74
|
+
onRemoveCard={handleRemoveCard}
|
|
75
|
+
onRemoveEmptyColumn={handleRemoveEmptyColumn}
|
|
76
|
+
/>
|
|
77
|
+
) : (
|
|
78
|
+
<span>Loading</span>
|
|
79
|
+
)}
|
|
80
|
+
</StackItem.Content>
|
|
81
|
+
);
|
|
82
|
+
};
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2023 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import React, { useCallback, useEffect, useMemo, useState } from 'react';
|
|
6
|
+
|
|
7
|
+
import { createIntent, useIntentDispatcher } from '@dxos/app-framework';
|
|
8
|
+
import { Filter, getSpace, useQuery } from '@dxos/react-client/echo';
|
|
9
|
+
import { ViewEditor, Form } from '@dxos/react-ui-form';
|
|
10
|
+
import { type KanbanType, KanbanPropsSchema } from '@dxos/react-ui-kanban';
|
|
11
|
+
import { ViewType } from '@dxos/schema';
|
|
12
|
+
|
|
13
|
+
import { KanbanAction } from '../types';
|
|
14
|
+
|
|
15
|
+
type KanbanViewEditorProps = { kanban: KanbanType };
|
|
16
|
+
|
|
17
|
+
export const KanbanViewEditor = ({ kanban }: KanbanViewEditorProps) => {
|
|
18
|
+
const { dispatchPromise: dispatch } = useIntentDispatcher();
|
|
19
|
+
const space = getSpace(kanban);
|
|
20
|
+
|
|
21
|
+
// TODO(ZaymonFC): The schema registry needs an API where we can query with initial value and
|
|
22
|
+
// endure typename changes. We shouldn't need to manage a subscription at this layer.
|
|
23
|
+
const [schema, setSchema] = useState(
|
|
24
|
+
space && kanban?.cardView?.target?.query?.type
|
|
25
|
+
? space.db.schemaRegistry.getSchema(kanban.cardView.target.query.type)
|
|
26
|
+
: undefined,
|
|
27
|
+
);
|
|
28
|
+
|
|
29
|
+
const views = useQuery(space, Filter.schema(ViewType));
|
|
30
|
+
const currentTypename = useMemo(() => kanban?.cardView?.target?.query?.type, [kanban?.cardView?.target?.query?.type]);
|
|
31
|
+
const updateViewTypename = useCallback(
|
|
32
|
+
(newTypename: string) => {
|
|
33
|
+
if (!schema) {
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
const matchingViews = views.filter((view) => view.query.type === currentTypename);
|
|
37
|
+
for (const view of matchingViews) {
|
|
38
|
+
view.query.type = newTypename;
|
|
39
|
+
}
|
|
40
|
+
schema.updateTypename(newTypename);
|
|
41
|
+
},
|
|
42
|
+
[views, schema],
|
|
43
|
+
);
|
|
44
|
+
|
|
45
|
+
useEffect(() => {
|
|
46
|
+
if (space && kanban?.cardView?.target?.query?.type) {
|
|
47
|
+
const unsubscribe = space.db.schemaRegistry
|
|
48
|
+
.query({ typename: kanban?.cardView?.target?.query?.type })
|
|
49
|
+
.subscribe((query) => {
|
|
50
|
+
const [schema] = query.results;
|
|
51
|
+
if (schema) {
|
|
52
|
+
setSchema(schema);
|
|
53
|
+
}
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
return unsubscribe;
|
|
57
|
+
}
|
|
58
|
+
}, [space, kanban?.cardView?.target?.query?.type]);
|
|
59
|
+
|
|
60
|
+
const handleDelete = useCallback(
|
|
61
|
+
(fieldId: string) => dispatch?.(createIntent(KanbanAction.DeleteCardField, { kanban, fieldId })),
|
|
62
|
+
[dispatch, kanban],
|
|
63
|
+
);
|
|
64
|
+
|
|
65
|
+
if (!space || !schema || !kanban.cardView?.target) {
|
|
66
|
+
return null;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
return (
|
|
70
|
+
<>
|
|
71
|
+
<Form
|
|
72
|
+
schema={KanbanPropsSchema}
|
|
73
|
+
values={{ columnField: kanban.columnField }}
|
|
74
|
+
onSave={({ columnField }) => {
|
|
75
|
+
kanban.columnField = columnField;
|
|
76
|
+
kanban.arrangement = undefined;
|
|
77
|
+
}}
|
|
78
|
+
/>
|
|
79
|
+
<ViewEditor
|
|
80
|
+
registry={space.db.schemaRegistry}
|
|
81
|
+
schema={schema}
|
|
82
|
+
view={kanban.cardView.target}
|
|
83
|
+
onTypenameChanged={updateViewTypename}
|
|
84
|
+
onDelete={handleDelete}
|
|
85
|
+
/>
|
|
86
|
+
</>
|
|
87
|
+
);
|
|
88
|
+
};
|
package/src/components/index.ts
CHANGED
package/src/index.ts
CHANGED
package/src/meta.ts
CHANGED
package/src/translations.ts
CHANGED
|
@@ -2,11 +2,16 @@
|
|
|
2
2
|
// Copyright 2023 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
+
import { KanbanType } from '@dxos/react-ui-kanban';
|
|
6
|
+
|
|
5
7
|
import { KANBAN_PLUGIN } from './meta';
|
|
6
8
|
|
|
7
9
|
export default [
|
|
8
10
|
{
|
|
9
11
|
'en-US': {
|
|
12
|
+
[KanbanType.typename]: {
|
|
13
|
+
'typename label': 'Kanban',
|
|
14
|
+
},
|
|
10
15
|
[KANBAN_PLUGIN]: {
|
|
11
16
|
'plugin name': 'Kanban',
|
|
12
17
|
'kanban title label': 'Title',
|
|
@@ -16,10 +21,11 @@ export default [
|
|
|
16
21
|
'item title label': 'Item title',
|
|
17
22
|
'item title placeholder': 'New item',
|
|
18
23
|
'add column label': 'Add column',
|
|
19
|
-
'add item label': 'Add
|
|
24
|
+
'add item label': 'Add card',
|
|
20
25
|
'delete column label': 'Delete column',
|
|
21
|
-
'delete item label': 'Delete
|
|
26
|
+
'delete item label': 'Delete card',
|
|
22
27
|
'create kanban label': 'Create kanban',
|
|
28
|
+
'card field deleted label': 'Card field deleted',
|
|
23
29
|
},
|
|
24
30
|
},
|
|
25
31
|
},
|
package/src/types.ts
ADDED
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2023 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { S } from '@dxos/echo-schema';
|
|
6
|
+
import { type Space, SpaceSchema } from '@dxos/react-client/echo';
|
|
7
|
+
import { KanbanType } from '@dxos/react-ui-kanban';
|
|
8
|
+
import { initializeKanban } from '@dxos/react-ui-kanban/testing';
|
|
9
|
+
import { FieldSchema } from '@dxos/schema';
|
|
10
|
+
|
|
11
|
+
import { KANBAN_PLUGIN } from './meta';
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Kanban data model.
|
|
15
|
+
* A Kanban board is a collection of columns, each of which contains a collection of items.
|
|
16
|
+
* The layout of columns and items is controlled by models.
|
|
17
|
+
* The underlying data model may be represented by direct object relationships
|
|
18
|
+
* (e.g., a column object containing an array of ordered items) or projections constructed
|
|
19
|
+
* by the model (e.g., a query of items based on metadata within a column object).
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
export namespace KanbanAction {
|
|
23
|
+
const KANBAN_ACTION = `${KANBAN_PLUGIN}/action`;
|
|
24
|
+
|
|
25
|
+
export class Create extends S.TaggedClass<Create>()(`${KANBAN_ACTION}/create`, {
|
|
26
|
+
input: S.Struct({
|
|
27
|
+
name: S.optional(S.String),
|
|
28
|
+
space: SpaceSchema,
|
|
29
|
+
}),
|
|
30
|
+
output: S.Struct({
|
|
31
|
+
object: KanbanType,
|
|
32
|
+
}),
|
|
33
|
+
}) {}
|
|
34
|
+
|
|
35
|
+
export class DeleteCardField extends S.TaggedClass<DeleteCardField>()(`${KANBAN_ACTION}/delete-card-field`, {
|
|
36
|
+
input: S.Struct({
|
|
37
|
+
kanban: KanbanType,
|
|
38
|
+
fieldId: S.String,
|
|
39
|
+
// TODO(wittjosiah): Separate fields for undo data?
|
|
40
|
+
deletionData: S.optional(
|
|
41
|
+
S.Struct({
|
|
42
|
+
field: FieldSchema,
|
|
43
|
+
// TODO(wittjosiah): This creates a type error.
|
|
44
|
+
// props: PropertySchema,
|
|
45
|
+
props: S.Any,
|
|
46
|
+
index: S.Number,
|
|
47
|
+
}),
|
|
48
|
+
),
|
|
49
|
+
}),
|
|
50
|
+
output: S.Void,
|
|
51
|
+
}) {}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// TODO(burdon): Undo?
|
|
55
|
+
// TODO(burdon): Typescript types (replace proto with annotations?)
|
|
56
|
+
// TODO(burdon): Should pure components depend on ECHO? Relationship between ECHO object/array and Observable.
|
|
57
|
+
// TODO(burdon): Can the plugin configure the object based on the data? E.g., how are the models constructed?
|
|
58
|
+
// TODO(burdon): Create models. Simple first based on actual data.
|
|
59
|
+
// Model is always a projection since the dragging state is tentative.
|
|
60
|
+
|
|
61
|
+
// TODO(burdon): Extend model for moving items (in and across columns).
|
|
62
|
+
export interface KanbanModel {
|
|
63
|
+
root: KanbanType;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
export type Location = {
|
|
67
|
+
idx?: number;
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
export const isKanban = (object: unknown): object is KanbanType => object != null && object instanceof KanbanType;
|
|
71
|
+
|
|
72
|
+
export const createKanban = async (space: Space) => {
|
|
73
|
+
const { kanban } = await initializeKanban({ space });
|
|
74
|
+
return kanban;
|
|
75
|
+
};
|