@dxos/plugin-kanban 0.7.4 → 0.7.5-labs.071a3e2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (122) hide show
  1. package/dist/lib/browser/chunk-7PFZLUDZ.mjs +72 -0
  2. package/dist/lib/browser/chunk-7PFZLUDZ.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +66 -145
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/intent-resolver-4SJJSDOD.mjs +76 -0
  6. package/dist/lib/browser/intent-resolver-4SJJSDOD.mjs.map +7 -0
  7. package/dist/lib/browser/meta.json +1 -1
  8. package/dist/lib/browser/react-surface-5WOH5E2K.mjs +240 -0
  9. package/dist/lib/browser/react-surface-5WOH5E2K.mjs.map +7 -0
  10. package/dist/lib/browser/types.mjs +11 -0
  11. package/dist/lib/node/chunk-IR3NCB4Y.cjs +94 -0
  12. package/dist/lib/node/chunk-IR3NCB4Y.cjs.map +7 -0
  13. package/dist/lib/node/index.cjs +62 -133
  14. package/dist/lib/node/index.cjs.map +4 -4
  15. package/dist/lib/node/intent-resolver-X3NB4IA6.cjs +90 -0
  16. package/dist/lib/node/intent-resolver-X3NB4IA6.cjs.map +7 -0
  17. package/dist/lib/node/meta.json +1 -1
  18. package/dist/lib/node/react-surface-A5TP26JG.cjs +258 -0
  19. package/dist/lib/node/react-surface-A5TP26JG.cjs.map +7 -0
  20. package/dist/lib/node/{types/index.cjs → types.cjs} +7 -10
  21. package/dist/lib/node/types.cjs.map +7 -0
  22. package/dist/lib/node-esm/chunk-LUAZTJHF.mjs +74 -0
  23. package/dist/lib/node-esm/chunk-LUAZTJHF.mjs.map +7 -0
  24. package/dist/lib/node-esm/index.mjs +66 -145
  25. package/dist/lib/node-esm/index.mjs.map +4 -4
  26. package/dist/lib/node-esm/intent-resolver-KFSXWDSV.mjs +77 -0
  27. package/dist/lib/node-esm/intent-resolver-KFSXWDSV.mjs.map +7 -0
  28. package/dist/lib/node-esm/meta.json +1 -1
  29. package/dist/lib/node-esm/react-surface-W4HLVD2O.mjs +241 -0
  30. package/dist/lib/node-esm/react-surface-W4HLVD2O.mjs.map +7 -0
  31. package/dist/lib/node-esm/types.mjs +12 -0
  32. package/dist/types/src/KanbanPlugin.d.ts +1 -3
  33. package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
  34. package/dist/types/src/capabilities/index.d.ts +3 -0
  35. package/dist/types/src/capabilities/index.d.ts.map +1 -0
  36. package/dist/types/src/capabilities/intent-resolver.d.ts +4 -0
  37. package/dist/types/src/capabilities/intent-resolver.d.ts.map +1 -0
  38. package/dist/types/src/capabilities/react-surface.d.ts +4 -0
  39. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -0
  40. package/dist/types/src/components/KanbanContainer.d.ts +7 -0
  41. package/dist/types/src/components/KanbanContainer.d.ts.map +1 -0
  42. package/dist/types/src/components/KanbanViewEditor.d.ts +8 -0
  43. package/dist/types/src/components/KanbanViewEditor.d.ts.map +1 -0
  44. package/dist/types/src/components/index.d.ts +2 -3
  45. package/dist/types/src/components/index.d.ts.map +1 -1
  46. package/dist/types/src/index.d.ts +1 -2
  47. package/dist/types/src/index.d.ts.map +1 -1
  48. package/dist/types/src/meta.d.ts +2 -2
  49. package/dist/types/src/meta.d.ts.map +1 -1
  50. package/dist/types/src/translations.d.ts +32 -0
  51. package/dist/types/src/translations.d.ts.map +1 -1
  52. package/dist/types/src/types.d.ts +67 -0
  53. package/dist/types/src/types.d.ts.map +1 -0
  54. package/dist/types/tsconfig.tsbuildinfo +1 -0
  55. package/package.json +21 -31
  56. package/src/KanbanPlugin.tsx +41 -101
  57. package/src/capabilities/index.ts +8 -0
  58. package/src/capabilities/intent-resolver.ts +45 -0
  59. package/src/capabilities/react-surface.tsx +28 -0
  60. package/src/components/KanbanContainer.tsx +80 -0
  61. package/src/components/KanbanViewEditor.tsx +106 -0
  62. package/src/components/index.ts +3 -4
  63. package/src/index.ts +1 -4
  64. package/src/meta.ts +4 -2
  65. package/src/translations.ts +8 -2
  66. package/src/types.ts +75 -0
  67. package/dist/lib/browser/KanbanMain-I5TMXNIY.mjs +0 -444
  68. package/dist/lib/browser/KanbanMain-I5TMXNIY.mjs.map +0 -7
  69. package/dist/lib/browser/chunk-4Y4TZ47E.mjs +0 -47
  70. package/dist/lib/browser/chunk-4Y4TZ47E.mjs.map +0 -7
  71. package/dist/lib/browser/chunk-LG4OMN5S.mjs +0 -18
  72. package/dist/lib/browser/chunk-LG4OMN5S.mjs.map +0 -7
  73. package/dist/lib/browser/meta.mjs +0 -9
  74. package/dist/lib/browser/types/index.mjs +0 -14
  75. package/dist/lib/node/KanbanMain-4OWAWTS4.cjs +0 -453
  76. package/dist/lib/node/KanbanMain-4OWAWTS4.cjs.map +0 -7
  77. package/dist/lib/node/chunk-LTR4WYI2.cjs +0 -67
  78. package/dist/lib/node/chunk-LTR4WYI2.cjs.map +0 -7
  79. package/dist/lib/node/chunk-MBAGHRFM.cjs +0 -41
  80. package/dist/lib/node/chunk-MBAGHRFM.cjs.map +0 -7
  81. package/dist/lib/node/meta.cjs +0 -30
  82. package/dist/lib/node/meta.cjs.map +0 -7
  83. package/dist/lib/node/types/index.cjs.map +0 -7
  84. package/dist/lib/node-esm/KanbanMain-PJC2JMJH.mjs +0 -445
  85. package/dist/lib/node-esm/KanbanMain-PJC2JMJH.mjs.map +0 -7
  86. package/dist/lib/node-esm/chunk-2ZBX5F7L.mjs +0 -48
  87. package/dist/lib/node-esm/chunk-2ZBX5F7L.mjs.map +0 -7
  88. package/dist/lib/node-esm/chunk-OTZHYV3S.mjs +0 -20
  89. package/dist/lib/node-esm/chunk-OTZHYV3S.mjs.map +0 -7
  90. package/dist/lib/node-esm/meta.mjs +0 -10
  91. package/dist/lib/node-esm/meta.mjs.map +0 -7
  92. package/dist/lib/node-esm/types/index.mjs +0 -15
  93. package/dist/lib/node-esm/types/index.mjs.map +0 -7
  94. package/dist/types/src/components/KanbanBoard.d.ts +0 -6
  95. package/dist/types/src/components/KanbanBoard.d.ts.map +0 -1
  96. package/dist/types/src/components/KanbanCard.d.ts +0 -9
  97. package/dist/types/src/components/KanbanCard.d.ts.map +0 -1
  98. package/dist/types/src/components/KanbanColumn.d.ts +0 -14
  99. package/dist/types/src/components/KanbanColumn.d.ts.map +0 -1
  100. package/dist/types/src/components/KanbanMain.d.ts +0 -7
  101. package/dist/types/src/components/KanbanMain.d.ts.map +0 -1
  102. package/dist/types/src/components/util.d.ts +0 -7
  103. package/dist/types/src/components/util.d.ts.map +0 -1
  104. package/dist/types/src/stories/testing.d.ts +0 -19
  105. package/dist/types/src/stories/testing.d.ts.map +0 -1
  106. package/dist/types/src/types/index.d.ts +0 -3
  107. package/dist/types/src/types/index.d.ts.map +0 -1
  108. package/dist/types/src/types/kanban.d.ts +0 -76
  109. package/dist/types/src/types/kanban.d.ts.map +0 -1
  110. package/dist/types/src/types/types.d.ts +0 -18
  111. package/dist/types/src/types/types.d.ts.map +0 -1
  112. package/src/components/KanbanBoard.tsx +0 -195
  113. package/src/components/KanbanCard.tsx +0 -82
  114. package/src/components/KanbanColumn.tsx +0 -143
  115. package/src/components/KanbanMain.tsx +0 -37
  116. package/src/components/util.ts +0 -38
  117. package/src/stories/testing.ts +0 -29
  118. package/src/types/index.ts +0 -6
  119. package/src/types/kanban.ts +0 -22
  120. package/src/types/types.ts +0 -57
  121. /package/dist/lib/browser/{meta.mjs.map → types.mjs.map} +0 -0
  122. /package/dist/lib/{browser/types/index.mjs.map → node-esm/types.mjs.map} +0 -0
@@ -2,110 +2,50 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import React from 'react';
5
+ import { createIntent, defineModule, contributes, Capabilities, Events, definePlugin } from '@dxos/app-framework';
6
+ import { ClientCapabilities, ClientEvents } from '@dxos/plugin-client';
7
+ import { type Space } from '@dxos/react-client/echo';
8
+ import { KanbanType, translations as kanbanTranslations } from '@dxos/react-ui-kanban';
6
9
 
7
- import { resolvePlugin, type PluginDefinition, parseIntentPlugin, NavigationAction } from '@dxos/app-framework';
8
- import { create } from '@dxos/live-object';
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';
10
+ import { IntentResolver, ReactSurface } from './capabilities';
11
+ import { KANBAN_PLUGIN, meta } from './meta';
16
12
  import translations from './translations';
17
- import { KanbanColumnType, KanbanItemType, KanbanType } from './types';
18
- import { KanbanAction, type KanbanPluginProvides } from './types';
13
+ import { KanbanAction } from './types';
19
14
 
20
- export const KanbanPlugin = (): PluginDefinition<KanbanPluginProvides> => {
21
- return {
22
- meta,
23
- provides: {
24
- metadata: {
25
- records: {
26
- [KanbanType.typename]: {
27
- createObject: KanbanAction.CREATE,
15
+ export const KanbanPlugin = () =>
16
+ definePlugin(meta, [
17
+ defineModule({
18
+ id: `${meta.id}/module/translations`,
19
+ activatesOn: Events.SetupTranslations,
20
+ activate: () => contributes(Capabilities.Translations, [...translations, ...kanbanTranslations]),
21
+ }),
22
+ defineModule({
23
+ id: `${meta.id}/module/metadata`,
24
+ activatesOn: Events.SetupMetadata,
25
+ activate: () =>
26
+ contributes(Capabilities.Metadata, {
27
+ id: KanbanType.typename,
28
+ metadata: {
29
+ createObject: (props: { name?: string }, options: { space: Space }) =>
30
+ createIntent(KanbanAction.Create, { ...props, space: options.space }),
28
31
  placeholder: ['kanban title placeholder', { ns: KANBAN_PLUGIN }],
29
32
  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
33
  },
41
- },
42
- },
43
- echo: {
44
- schema: [KanbanType],
45
- system: [KanbanColumnType, KanbanItemType],
46
- },
47
- translations,
48
- graph: {
49
- builder: (plugins) => {
50
- const client = resolvePlugin(plugins, parseClientPlugin)?.provides.client;
51
- const dispatch = resolvePlugin(plugins, parseIntentPlugin)?.provides.intent.dispatch;
52
- if (!client || !dispatch) {
53
- return [];
54
- }
55
-
56
- return createExtension({
57
- id: KanbanAction.CREATE,
58
- filter: (node): node is ActionGroup => isActionGroup(node) && node.id.startsWith(SpaceAction.ADD_OBJECT),
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
- };
34
+ }),
35
+ }),
36
+ defineModule({
37
+ id: `${meta.id}/module/schema`,
38
+ activatesOn: ClientEvents.SetupSchema,
39
+ activate: () => contributes(ClientCapabilities.Schema, [KanbanType]),
40
+ }),
41
+ defineModule({
42
+ id: `${meta.id}/module/react-surface`,
43
+ activatesOn: Events.SetupSurfaces,
44
+ activate: ReactSurface,
45
+ }),
46
+ defineModule({
47
+ id: `${meta.id}/module/intent-resolver`,
48
+ activatesOn: Events.SetupIntents,
49
+ activate: IntentResolver,
50
+ }),
51
+ ]);
@@ -0,0 +1,8 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { lazy } from '@dxos/app-framework';
6
+
7
+ export const IntentResolver = lazy(() => import('./intent-resolver'));
8
+ export const ReactSurface = lazy(() => import('./react-surface'));
@@ -0,0 +1,45 @@
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({
16
+ intent: KanbanAction.Create,
17
+ resolve: async ({ space }) => ({
18
+ data: { object: await createKanban(space) },
19
+ }),
20
+ }),
21
+ createResolver({
22
+ intent: KanbanAction.DeleteCardField,
23
+ resolve: ({ kanban, fieldId, deletionData }, undo) => {
24
+ invariant(kanban.cardView);
25
+
26
+ const schema =
27
+ kanban.cardView.target && getSpace(kanban)?.db.schemaRegistry.getSchema(kanban.cardView.target.query.type);
28
+ invariant(schema);
29
+ const projection = new ViewProjection(schema, kanban.cardView.target!);
30
+
31
+ if (!undo) {
32
+ const { deleted, index } = projection.deleteFieldProjection(fieldId);
33
+ return {
34
+ undoable: {
35
+ message: ['card field deleted label', { ns: KANBAN_PLUGIN }],
36
+ data: { deletionData: { ...deleted, index } },
37
+ },
38
+ };
39
+ } else if (undo && deletionData) {
40
+ const { field, props, index } = deletionData;
41
+ projection.setFieldProjection({ field, props }, index);
42
+ }
43
+ },
44
+ }),
45
+ ]);
@@ -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,80 @@
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
+ import { ViewProjection } from '@dxos/schema';
14
+
15
+ export const KanbanContainer = ({ kanban }: { kanban: KanbanType; role: string }) => {
16
+ const [cardSchema, setCardSchema] = useState<EchoSchema>();
17
+ const [projection, setProjection] = useState<ViewProjection>();
18
+ const space = getSpace(kanban);
19
+
20
+ useEffect(() => {
21
+ if (kanban.cardView?.target?.query?.type && space) {
22
+ const query = space.db.schemaRegistry.query({ typename: kanban.cardView.target.query.type });
23
+ const unsubscribe = query.subscribe(
24
+ () => {
25
+ const [schema] = query.results;
26
+ if (schema) {
27
+ setCardSchema(schema);
28
+ }
29
+ },
30
+ { fire: true },
31
+ );
32
+ return unsubscribe;
33
+ }
34
+ }, [kanban.cardView?.target?.query, space]);
35
+
36
+ useEffect(() => {
37
+ if (kanban.cardView?.target && cardSchema) {
38
+ setProjection(new ViewProjection(cardSchema, kanban.cardView.target));
39
+ }
40
+ // TODO(ZaymonFC): Is there a better way to get notified about deep changes in the json schema?
41
+ }, [kanban.cardView?.target, cardSchema, JSON.stringify(cardSchema?.jsonSchema)]);
42
+
43
+ const objects = useQuery(space, cardSchema ? Filter.schema(cardSchema) : Filter.nothing());
44
+ const filteredObjects = useGlobalFilteredObjects(objects);
45
+
46
+ const model = useKanbanModel({
47
+ kanban,
48
+ cardSchema,
49
+ projection,
50
+ items: filteredObjects,
51
+ });
52
+
53
+ const handleAddCard = useCallback(
54
+ (columnValue: string | undefined) => {
55
+ const path = model?.columnFieldPath;
56
+ if (space && cardSchema && path) {
57
+ space.db.add(create(cardSchema, { [path]: columnValue }));
58
+ }
59
+ },
60
+ [space, cardSchema, model],
61
+ );
62
+
63
+ const handleRemoveCard = useCallback(
64
+ (card: { id: string }) => {
65
+ invariant(space);
66
+ space.db.remove(card);
67
+ },
68
+ [space],
69
+ );
70
+
71
+ return (
72
+ <StackItem.Content toolbar={false}>
73
+ {model ? (
74
+ <Kanban model={model} onAddCard={handleAddCard} onRemoveCard={handleRemoveCard} />
75
+ ) : (
76
+ <span>Loading</span>
77
+ )}
78
+ </StackItem.Content>
79
+ );
80
+ };
@@ -0,0 +1,106 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { useCallback, useEffect, useMemo, useState } from 'react';
6
+
7
+ import { createIntent, useIntentDispatcher } from '@dxos/app-framework';
8
+ import { type EchoSchema, FormatEnum } from '@dxos/echo-schema';
9
+ import { invariant } from '@dxos/invariant';
10
+ import { Filter, getSpace, useQuery } from '@dxos/react-client/echo';
11
+ import { ViewEditor, Form, SelectInput, type CustomInputMap } from '@dxos/react-ui-form';
12
+ import { type KanbanType, KanbanSettingsSchema } from '@dxos/react-ui-kanban';
13
+ import { ViewType, ViewProjection } from '@dxos/schema';
14
+
15
+ import { KanbanAction } from '../types';
16
+
17
+ type KanbanViewEditorProps = { kanban: KanbanType };
18
+
19
+ export const KanbanViewEditor = ({ kanban }: KanbanViewEditorProps) => {
20
+ const { dispatchPromise: dispatch } = useIntentDispatcher();
21
+ const space = getSpace(kanban);
22
+
23
+ const [schema, setSchema] = useState<EchoSchema | undefined>();
24
+
25
+ useEffect(() => {
26
+ if (space && kanban?.cardView?.target?.query?.type) {
27
+ const query = space.db.schemaRegistry.query({ typename: kanban.cardView.target.query.type });
28
+ const unsubscribe = query.subscribe(
29
+ () => {
30
+ const [schema] = query.results;
31
+ if (schema) {
32
+ setSchema(schema);
33
+ }
34
+ },
35
+ { fire: true },
36
+ );
37
+ return unsubscribe;
38
+ }
39
+ }, [space, kanban?.cardView?.target?.query?.type]);
40
+
41
+ const views = useQuery(space, Filter.schema(ViewType));
42
+ const currentTypename = useMemo(() => kanban?.cardView?.target?.query?.type, [kanban?.cardView?.target?.query?.type]);
43
+ const updateViewTypename = useCallback(
44
+ (newTypename: string) => {
45
+ invariant(schema);
46
+ const matchingViews = views.filter((view) => view.query.type === currentTypename);
47
+ for (const view of matchingViews) {
48
+ view.query.type = newTypename;
49
+ }
50
+ schema.updateTypename(newTypename);
51
+ },
52
+ [views, schema],
53
+ );
54
+
55
+ const handleDelete = useCallback(
56
+ (fieldId: string) => dispatch?.(createIntent(KanbanAction.DeleteCardField, { kanban, fieldId })),
57
+ [dispatch, kanban],
58
+ );
59
+
60
+ const projection = useMemo(() => {
61
+ if (kanban?.cardView?.target && schema) {
62
+ return new ViewProjection(schema, kanban.cardView.target);
63
+ }
64
+ }, [kanban?.cardView?.target, schema, JSON.stringify(schema)]);
65
+
66
+ const selectFields = useMemo(() => {
67
+ if (!projection) {
68
+ return [];
69
+ }
70
+
71
+ return projection
72
+ .getFieldProjections()
73
+ .filter((field) => field.props.format === FormatEnum.SingleSelect)
74
+ .map(({ field }) => ({ value: field.id, label: field.path }));
75
+ }, [projection]);
76
+
77
+ const onSave = useCallback(
78
+ (values: Partial<{ columnFieldId: string }>) => {
79
+ kanban.columnFieldId = values.columnFieldId;
80
+ },
81
+ [kanban],
82
+ );
83
+
84
+ const initialValues = useMemo(() => ({ columnFieldId: kanban.columnFieldId }), [kanban]);
85
+ const custom: CustomInputMap = useMemo(
86
+ () => ({ columnFieldId: (props) => <SelectInput {...props} options={selectFields} /> }),
87
+ [selectFields],
88
+ );
89
+
90
+ if (!space || !schema || !kanban.cardView?.target) {
91
+ return null;
92
+ }
93
+
94
+ return (
95
+ <>
96
+ <Form schema={KanbanSettingsSchema} values={initialValues} onSave={onSave} autoSave Custom={custom} />
97
+ <ViewEditor
98
+ registry={space.db.schemaRegistry}
99
+ schema={schema}
100
+ view={kanban.cardView.target}
101
+ onTypenameChanged={updateViewTypename}
102
+ onDelete={handleDelete}
103
+ />
104
+ </>
105
+ );
106
+ };
@@ -1,7 +1,6 @@
1
1
  //
2
- // Copyright 2023 DXOS.org
2
+ // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { lazy } from 'react';
6
-
7
- export const KanbanMain = lazy(() => import('./KanbanMain'));
5
+ export * from './KanbanContainer';
6
+ export * from './KanbanViewEditor';
package/src/index.ts CHANGED
@@ -2,8 +2,5 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { KanbanPlugin } from './KanbanPlugin';
6
-
7
- export default KanbanPlugin;
8
-
9
5
  export * from './KanbanPlugin';
6
+ export * from './meta';
package/src/meta.ts CHANGED
@@ -6,11 +6,13 @@ import { type PluginMeta } from '@dxos/app-framework';
6
6
 
7
7
  export const KANBAN_PLUGIN = 'dxos.org/plugin/kanban';
8
8
 
9
- export default {
9
+ export const meta = {
10
10
  id: KANBAN_PLUGIN,
11
11
  name: 'Kanban',
12
- description: 'Kanban board for managing tasks.',
12
+ description:
13
+ 'Kanban allows you to explore Table data in sorted columns defined by your custom schema. You can use Kanbans to track progress or trigger custom automations when cards are moved from one state to another.',
13
14
  icon: 'ph--kanban--regular',
14
15
  source: 'https://github.com/dxos/dxos/tree/main/packages/plugins/experimental/plugin-kanban',
15
16
  tags: ['experimental'],
17
+ screenshots: ['https://dxos.network/plugin-details-kanban-dark.png'],
16
18
  } satisfies PluginMeta;
@@ -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 item',
24
+ 'add item label': 'Add card',
20
25
  'delete column label': 'Delete column',
21
- 'delete item label': 'Delete item',
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
+ };