@dxos/plugin-kanban 0.7.4 → 0.7.5-labs.401163d

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 (131) hide show
  1. package/dist/lib/browser/artifact-FJ7K5S5Y.mjs +173 -0
  2. package/dist/lib/browser/artifact-FJ7K5S5Y.mjs.map +7 -0
  3. package/dist/lib/browser/chunk-ILL6UKZE.mjs +94 -0
  4. package/dist/lib/browser/chunk-ILL6UKZE.mjs.map +7 -0
  5. package/dist/lib/browser/index.mjs +75 -145
  6. package/dist/lib/browser/index.mjs.map +4 -4
  7. package/dist/lib/browser/intent-resolver-EOTUDRIU.mjs +113 -0
  8. package/dist/lib/browser/intent-resolver-EOTUDRIU.mjs.map +7 -0
  9. package/dist/lib/browser/meta.json +1 -1
  10. package/dist/lib/browser/react-surface-SN2UKU4Y.mjs +285 -0
  11. package/dist/lib/browser/react-surface-SN2UKU4Y.mjs.map +7 -0
  12. package/dist/lib/browser/types.mjs +17 -0
  13. package/dist/lib/node/artifact-TT3JYZEY.cjs +189 -0
  14. package/dist/lib/node/artifact-TT3JYZEY.cjs.map +7 -0
  15. package/dist/lib/node/chunk-63KJGDFJ.cjs +119 -0
  16. package/dist/lib/node/chunk-63KJGDFJ.cjs.map +7 -0
  17. package/dist/lib/node/index.cjs +70 -133
  18. package/dist/lib/node/index.cjs.map +4 -4
  19. package/dist/lib/node/intent-resolver-PMBA35UQ.cjs +127 -0
  20. package/dist/lib/node/intent-resolver-PMBA35UQ.cjs.map +7 -0
  21. package/dist/lib/node/meta.json +1 -1
  22. package/dist/lib/node/react-surface-TCUFAH5K.cjs +301 -0
  23. package/dist/lib/node/react-surface-TCUFAH5K.cjs.map +7 -0
  24. package/dist/lib/node/{types/index.cjs → types.cjs} +13 -10
  25. package/dist/lib/node/types.cjs.map +7 -0
  26. package/dist/lib/node-esm/artifact-TKSPF5E2.mjs +174 -0
  27. package/dist/lib/node-esm/artifact-TKSPF5E2.mjs.map +7 -0
  28. package/dist/lib/node-esm/chunk-TZ3QKBLD.mjs +96 -0
  29. package/dist/lib/node-esm/chunk-TZ3QKBLD.mjs.map +7 -0
  30. package/dist/lib/node-esm/index.mjs +75 -145
  31. package/dist/lib/node-esm/index.mjs.map +4 -4
  32. package/dist/lib/node-esm/intent-resolver-VYYMQVBP.mjs +114 -0
  33. package/dist/lib/node-esm/intent-resolver-VYYMQVBP.mjs.map +7 -0
  34. package/dist/lib/node-esm/meta.json +1 -1
  35. package/dist/lib/node-esm/react-surface-HZJXQKYV.mjs +286 -0
  36. package/dist/lib/node-esm/react-surface-HZJXQKYV.mjs.map +7 -0
  37. package/dist/lib/node-esm/types.mjs +18 -0
  38. package/dist/types/src/KanbanPlugin.d.ts +1 -3
  39. package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
  40. package/dist/types/src/capabilities/artifact.d.ts +11 -0
  41. package/dist/types/src/capabilities/artifact.d.ts.map +1 -0
  42. package/dist/types/src/capabilities/index.d.ts +4 -0
  43. package/dist/types/src/capabilities/index.d.ts.map +1 -0
  44. package/dist/types/src/capabilities/intent-resolver.d.ts +4 -0
  45. package/dist/types/src/capabilities/intent-resolver.d.ts.map +1 -0
  46. package/dist/types/src/capabilities/react-surface.d.ts +4 -0
  47. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -0
  48. package/dist/types/src/components/KanbanContainer.d.ts +7 -0
  49. package/dist/types/src/components/KanbanContainer.d.ts.map +1 -0
  50. package/dist/types/src/components/KanbanViewEditor.d.ts +8 -0
  51. package/dist/types/src/components/KanbanViewEditor.d.ts.map +1 -0
  52. package/dist/types/src/components/index.d.ts +2 -3
  53. package/dist/types/src/components/index.d.ts.map +1 -1
  54. package/dist/types/src/index.d.ts +1 -2
  55. package/dist/types/src/index.d.ts.map +1 -1
  56. package/dist/types/src/meta.d.ts +2 -2
  57. package/dist/types/src/meta.d.ts.map +1 -1
  58. package/dist/types/src/translations.d.ts +35 -0
  59. package/dist/types/src/translations.d.ts.map +1 -1
  60. package/dist/types/src/types.d.ts +83 -0
  61. package/dist/types/src/types.d.ts.map +1 -0
  62. package/dist/types/tsconfig.tsbuildinfo +1 -0
  63. package/package.json +25 -31
  64. package/src/KanbanPlugin.tsx +47 -101
  65. package/src/capabilities/artifact.ts +131 -0
  66. package/src/capabilities/index.ts +9 -0
  67. package/src/capabilities/intent-resolver.ts +64 -0
  68. package/src/capabilities/react-surface.tsx +89 -0
  69. package/src/components/KanbanContainer.tsx +84 -0
  70. package/src/components/KanbanViewEditor.tsx +89 -0
  71. package/src/components/index.ts +3 -4
  72. package/src/index.ts +1 -4
  73. package/src/meta.ts +4 -2
  74. package/src/translations.ts +9 -2
  75. package/src/types.ts +102 -0
  76. package/dist/lib/browser/KanbanMain-I5TMXNIY.mjs +0 -444
  77. package/dist/lib/browser/KanbanMain-I5TMXNIY.mjs.map +0 -7
  78. package/dist/lib/browser/chunk-4Y4TZ47E.mjs +0 -47
  79. package/dist/lib/browser/chunk-4Y4TZ47E.mjs.map +0 -7
  80. package/dist/lib/browser/chunk-LG4OMN5S.mjs +0 -18
  81. package/dist/lib/browser/chunk-LG4OMN5S.mjs.map +0 -7
  82. package/dist/lib/browser/meta.mjs +0 -9
  83. package/dist/lib/browser/types/index.mjs +0 -14
  84. package/dist/lib/node/KanbanMain-4OWAWTS4.cjs +0 -453
  85. package/dist/lib/node/KanbanMain-4OWAWTS4.cjs.map +0 -7
  86. package/dist/lib/node/chunk-LTR4WYI2.cjs +0 -67
  87. package/dist/lib/node/chunk-LTR4WYI2.cjs.map +0 -7
  88. package/dist/lib/node/chunk-MBAGHRFM.cjs +0 -41
  89. package/dist/lib/node/chunk-MBAGHRFM.cjs.map +0 -7
  90. package/dist/lib/node/meta.cjs +0 -30
  91. package/dist/lib/node/meta.cjs.map +0 -7
  92. package/dist/lib/node/types/index.cjs.map +0 -7
  93. package/dist/lib/node-esm/KanbanMain-PJC2JMJH.mjs +0 -445
  94. package/dist/lib/node-esm/KanbanMain-PJC2JMJH.mjs.map +0 -7
  95. package/dist/lib/node-esm/chunk-2ZBX5F7L.mjs +0 -48
  96. package/dist/lib/node-esm/chunk-2ZBX5F7L.mjs.map +0 -7
  97. package/dist/lib/node-esm/chunk-OTZHYV3S.mjs +0 -20
  98. package/dist/lib/node-esm/chunk-OTZHYV3S.mjs.map +0 -7
  99. package/dist/lib/node-esm/meta.mjs +0 -10
  100. package/dist/lib/node-esm/meta.mjs.map +0 -7
  101. package/dist/lib/node-esm/types/index.mjs +0 -15
  102. package/dist/lib/node-esm/types/index.mjs.map +0 -7
  103. package/dist/types/src/components/KanbanBoard.d.ts +0 -6
  104. package/dist/types/src/components/KanbanBoard.d.ts.map +0 -1
  105. package/dist/types/src/components/KanbanCard.d.ts +0 -9
  106. package/dist/types/src/components/KanbanCard.d.ts.map +0 -1
  107. package/dist/types/src/components/KanbanColumn.d.ts +0 -14
  108. package/dist/types/src/components/KanbanColumn.d.ts.map +0 -1
  109. package/dist/types/src/components/KanbanMain.d.ts +0 -7
  110. package/dist/types/src/components/KanbanMain.d.ts.map +0 -1
  111. package/dist/types/src/components/util.d.ts +0 -7
  112. package/dist/types/src/components/util.d.ts.map +0 -1
  113. package/dist/types/src/stories/testing.d.ts +0 -19
  114. package/dist/types/src/stories/testing.d.ts.map +0 -1
  115. package/dist/types/src/types/index.d.ts +0 -3
  116. package/dist/types/src/types/index.d.ts.map +0 -1
  117. package/dist/types/src/types/kanban.d.ts +0 -76
  118. package/dist/types/src/types/kanban.d.ts.map +0 -1
  119. package/dist/types/src/types/types.d.ts +0 -18
  120. package/dist/types/src/types/types.d.ts.map +0 -1
  121. package/src/components/KanbanBoard.tsx +0 -195
  122. package/src/components/KanbanCard.tsx +0 -82
  123. package/src/components/KanbanColumn.tsx +0 -143
  124. package/src/components/KanbanMain.tsx +0 -37
  125. package/src/components/util.ts +0 -38
  126. package/src/stories/testing.ts +0 -29
  127. package/src/types/index.ts +0 -6
  128. package/src/types/kanban.ts +0 -22
  129. package/src/types/types.ts +0 -57
  130. /package/dist/lib/browser/{meta.mjs.map → types.mjs.map} +0 -0
  131. /package/dist/lib/{browser/types/index.mjs.map → node-esm/types.mjs.map} +0 -0
@@ -0,0 +1,64 @@
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, initialSchema, initialPivotColumn }) => ({
18
+ data: { object: await createKanban({ space, initialSchema, initialPivotColumn }) },
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
+ createResolver({
46
+ intent: KanbanAction.DeleteCard,
47
+ resolve: ({ card }, undo) => {
48
+ const space = getSpace(card);
49
+ invariant(space);
50
+
51
+ if (!undo) {
52
+ space.db.remove(card);
53
+ return {
54
+ undoable: {
55
+ message: ['card deleted label', { ns: KANBAN_PLUGIN }],
56
+ data: { card },
57
+ },
58
+ };
59
+ } else {
60
+ space.db.add(card);
61
+ }
62
+ },
63
+ }),
64
+ ]);
@@ -0,0 +1,89 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { useMemo } from 'react';
6
+
7
+ import { Capabilities, contributes, createSurface } from '@dxos/app-framework';
8
+ import { type S } from '@dxos/echo-schema';
9
+ import { findAnnotation } from '@dxos/effect';
10
+ import { type CollectionType } from '@dxos/plugin-space/types';
11
+ import { getSpace, isSpace, type Space } from '@dxos/react-client/echo';
12
+ import { type InputProps, SelectInput, useFormValues } from '@dxos/react-ui-form';
13
+ import { type KanbanType } from '@dxos/react-ui-kanban';
14
+
15
+ import { KanbanContainer, KanbanViewEditor } from '../components';
16
+ import { KANBAN_PLUGIN } from '../meta';
17
+ import { isKanban, InitialSchemaAnnotationId, InitialPivotColumnAnnotationId } from '../types';
18
+
19
+ export default () =>
20
+ contributes(Capabilities.ReactSurface, [
21
+ createSurface({
22
+ id: `${KANBAN_PLUGIN}/kanban`,
23
+ role: ['article', 'section'],
24
+ filter: (data): data is { subject: KanbanType } => isKanban(data.subject),
25
+ component: ({ data, role }) => <KanbanContainer kanban={data.subject} role={role} />,
26
+ }),
27
+ createSurface({
28
+ id: `${KANBAN_PLUGIN}/settings`,
29
+ role: 'complementary--settings',
30
+ filter: (data): data is { subject: KanbanType } => isKanban(data.subject),
31
+ component: ({ data }) => <KanbanViewEditor kanban={data.subject} />,
32
+ }),
33
+ createSurface({
34
+ id: `${KANBAN_PLUGIN}/create-initial-schema-form-[schema]`,
35
+ role: 'form-input',
36
+ filter: (data): data is { prop: string; schema: S.Schema<any>; target: Space | CollectionType | undefined } => {
37
+ const annotation = findAnnotation<boolean>((data.schema as S.Schema.All).ast, InitialSchemaAnnotationId);
38
+ return !!annotation;
39
+ },
40
+ component: ({ data: { target }, ...inputProps }) => {
41
+ const props = inputProps as any as InputProps;
42
+ const space = isSpace(target) ? target : getSpace(target);
43
+ if (!space) {
44
+ return null;
45
+ }
46
+ const schemata = space?.db.schemaRegistry.query().runSync();
47
+
48
+ return <SelectInput {...props} options={schemata.map((schema) => ({ value: schema.typename }))} />;
49
+ },
50
+ }),
51
+ createSurface({
52
+ id: `${KANBAN_PLUGIN}/create-initial-schema-form-[pivot-column]`,
53
+ role: 'form-input',
54
+ filter: (data): data is { prop: string; schema: S.Schema<any>; target: Space | CollectionType | undefined } => {
55
+ const annotation = findAnnotation<boolean>((data.schema as S.Schema.All).ast, InitialPivotColumnAnnotationId);
56
+ return !!annotation;
57
+ },
58
+ component: ({ data: { target }, ...inputProps }) => {
59
+ const props = inputProps as any as InputProps;
60
+ const space = isSpace(target) ? target : getSpace(target);
61
+ if (!space) {
62
+ return null;
63
+ }
64
+ const { initialSchema } = useFormValues();
65
+ const [selectedSchema] = space?.db.schemaRegistry.query({ typename: initialSchema }).runSync();
66
+
67
+ const singleSelectColumns = useMemo(() => {
68
+ if (!selectedSchema?.jsonSchema?.properties) {
69
+ return [];
70
+ }
71
+
72
+ const columns = Object.entries(selectedSchema.jsonSchema.properties).reduce<string[]>((acc, [key, value]) => {
73
+ if (typeof value === 'object' && value?.format === 'single-select') {
74
+ acc.push(key);
75
+ }
76
+ return acc;
77
+ }, []);
78
+
79
+ return columns;
80
+ }, [selectedSchema?.jsonSchema]);
81
+
82
+ if (!initialSchema) {
83
+ return null;
84
+ }
85
+
86
+ return <SelectInput {...props} options={singleSelectColumns.map((column) => ({ value: column }))} />;
87
+ },
88
+ }),
89
+ ]);
@@ -0,0 +1,84 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import React, { useCallback, useEffect, useState } from 'react';
6
+
7
+ import { createIntent, useIntentDispatcher } from '@dxos/app-framework';
8
+ import { type EchoSchema } from '@dxos/echo-schema';
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
+ import { KanbanAction } from '../types';
16
+
17
+ export const KanbanContainer = ({ kanban }: { kanban: KanbanType; role: string }) => {
18
+ const [cardSchema, setCardSchema] = useState<EchoSchema>();
19
+ const [projection, setProjection] = useState<ViewProjection>();
20
+ const space = getSpace(kanban);
21
+ const { dispatchPromise: dispatch } = useIntentDispatcher();
22
+
23
+ useEffect(() => {
24
+ if (kanban.cardView?.target?.query?.type && space) {
25
+ const query = space.db.schemaRegistry.query({ typename: kanban.cardView.target.query.type });
26
+ const unsubscribe = query.subscribe(
27
+ () => {
28
+ const [schema] = query.results;
29
+ if (schema) {
30
+ setCardSchema(schema);
31
+ }
32
+ },
33
+ { fire: true },
34
+ );
35
+ return unsubscribe;
36
+ }
37
+ }, [kanban.cardView?.target?.query, space]);
38
+
39
+ useEffect(() => {
40
+ if (kanban.cardView?.target && cardSchema) {
41
+ setProjection(new ViewProjection(cardSchema, kanban.cardView.target));
42
+ }
43
+ // TODO(ZaymonFC): Is there a better way to get notified about deep changes in the json schema?
44
+ }, [kanban.cardView?.target, cardSchema, JSON.stringify(cardSchema?.jsonSchema)]);
45
+
46
+ const objects = useQuery(space, cardSchema ? Filter.schema(cardSchema) : Filter.nothing());
47
+ const filteredObjects = useGlobalFilteredObjects(objects);
48
+
49
+ const model = useKanbanModel({
50
+ kanban,
51
+ cardSchema,
52
+ projection,
53
+ items: filteredObjects,
54
+ });
55
+
56
+ const handleAddCard = useCallback(
57
+ (columnValue: string | undefined) => {
58
+ const path = model?.columnFieldPath;
59
+ if (space && cardSchema && path) {
60
+ const card = create(cardSchema, { [path]: columnValue });
61
+ space.db.add(card);
62
+ return card.id;
63
+ }
64
+ },
65
+ [space, cardSchema, model],
66
+ );
67
+
68
+ const handleRemoveCard = useCallback(
69
+ (card: { id: string }) => {
70
+ void dispatch(createIntent(KanbanAction.DeleteCard, { card }));
71
+ },
72
+ [dispatch],
73
+ );
74
+
75
+ return (
76
+ <StackItem.Content toolbar={false}>
77
+ {model ? (
78
+ <Kanban model={model} onAddCard={handleAddCard} onRemoveCard={handleRemoveCard} />
79
+ ) : (
80
+ <span>Loading</span>
81
+ )}
82
+ </StackItem.Content>
83
+ );
84
+ };
@@ -0,0 +1,89 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { useCallback, useMemo } from 'react';
6
+
7
+ import { createIntent, useIntentDispatcher } from '@dxos/app-framework';
8
+ import { FormatEnum } from '@dxos/echo-schema';
9
+ import { invariant } from '@dxos/invariant';
10
+ import { Filter, getSpace, useQuery, useSchema } 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
+ const currentTypename = useMemo(() => kanban?.cardView?.target?.query?.type, [kanban?.cardView?.target?.query?.type]);
23
+ const schema = useSchema(space, currentTypename);
24
+ const views = useQuery(space, Filter.schema(ViewType));
25
+
26
+ const updateViewTypename = useCallback(
27
+ (newTypename: string) => {
28
+ invariant(schema);
29
+ const matchingViews = views.filter((view) => view.query.type === currentTypename);
30
+ for (const view of matchingViews) {
31
+ view.query.type = newTypename;
32
+ }
33
+ schema.updateTypename(newTypename);
34
+ },
35
+ [views, schema],
36
+ );
37
+
38
+ const handleDelete = useCallback(
39
+ (fieldId: string) => dispatch?.(createIntent(KanbanAction.DeleteCardField, { kanban, fieldId })),
40
+ [dispatch, kanban],
41
+ );
42
+
43
+ const projection = useMemo(() => {
44
+ if (kanban?.cardView?.target && schema) {
45
+ return new ViewProjection(schema, kanban.cardView.target);
46
+ }
47
+ }, [kanban?.cardView?.target, schema, JSON.stringify(schema)]);
48
+
49
+ const selectFields = useMemo(() => {
50
+ if (!projection) {
51
+ return [];
52
+ }
53
+
54
+ return projection
55
+ .getFieldProjections()
56
+ .filter((field) => field.props.format === FormatEnum.SingleSelect)
57
+ .map(({ field }) => ({ value: field.id, label: field.path }));
58
+ }, [projection]);
59
+
60
+ const onSave = useCallback(
61
+ (values: Partial<{ columnFieldId: string }>) => {
62
+ kanban.columnFieldId = values.columnFieldId;
63
+ },
64
+ [kanban],
65
+ );
66
+
67
+ const initialValues = useMemo(() => ({ columnFieldId: kanban.columnFieldId }), [kanban]);
68
+ const custom: CustomInputMap = useMemo(
69
+ () => ({ columnFieldId: (props) => <SelectInput {...props} options={selectFields} /> }),
70
+ [selectFields],
71
+ );
72
+
73
+ if (!space || !schema || !kanban.cardView?.target) {
74
+ return null;
75
+ }
76
+
77
+ return (
78
+ <>
79
+ <Form schema={KanbanSettingsSchema} values={initialValues} onSave={onSave} autoSave Custom={custom} />
80
+ <ViewEditor
81
+ registry={space.db.schemaRegistry}
82
+ schema={schema}
83
+ view={kanban.cardView.target}
84
+ onTypenameChanged={updateViewTypename}
85
+ onDelete={handleDelete}
86
+ />
87
+ </>
88
+ );
89
+ };
@@ -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,12 @@ 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',
29
+ 'card deleted label': 'Card deleted',
23
30
  },
24
31
  },
25
32
  },
package/src/types.ts ADDED
@@ -0,0 +1,102 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { TitleAnnotationId } from '@effect/schema/AST';
6
+
7
+ import { S } from '@dxos/echo-schema';
8
+ import { type Space, SpaceSchema } from '@dxos/react-client/echo';
9
+ import { KanbanType } from '@dxos/react-ui-kanban';
10
+ import { initializeKanban } from '@dxos/react-ui-kanban/testing';
11
+ import { FieldSchema } from '@dxos/schema';
12
+
13
+ import { KANBAN_PLUGIN } from './meta';
14
+
15
+ /**
16
+ * Kanban data model.
17
+ * A Kanban board is a collection of columns, each of which contains a collection of items.
18
+ * The layout of columns and items is controlled by models.
19
+ * The underlying data model may be represented by direct object relationships
20
+ * (e.g., a column object containing an array of ordered items) or projections constructed
21
+ * by the model (e.g., a query of items based on metadata within a column object).
22
+ */
23
+
24
+ export const InitialSchemaAnnotationId = Symbol.for('@dxos/plugin-kanban/annotation/InitialSchema');
25
+ export const InitialPivotColumnAnnotationId = Symbol.for('@dxos/plugin-kanban/annotation/InitialPivotColumn');
26
+
27
+ export const CreateKanbanSchema = S.Struct({
28
+ name: S.optional(S.String),
29
+ initialSchema: S.optional(
30
+ S.String.annotations({
31
+ [InitialSchemaAnnotationId]: true,
32
+ [TitleAnnotationId]: 'Select card schema (leave empty to start fresh)',
33
+ }),
34
+ ),
35
+ initialPivotColumn: S.optional(
36
+ S.String.annotations({
37
+ [InitialPivotColumnAnnotationId]: true,
38
+ [TitleAnnotationId]: 'Pivot column',
39
+ }),
40
+ ),
41
+ });
42
+
43
+ export type CreateKanbanType = S.Schema.Type<typeof CreateKanbanSchema>;
44
+
45
+ export namespace KanbanAction {
46
+ const KANBAN_ACTION = `${KANBAN_PLUGIN}/action`;
47
+
48
+ export class Create extends S.TaggedClass<Create>()(`${KANBAN_ACTION}/create`, {
49
+ input: S.extend(S.Struct({ space: SpaceSchema }), CreateKanbanSchema),
50
+ output: S.Struct({
51
+ object: KanbanType,
52
+ }),
53
+ }) {}
54
+
55
+ export class DeleteCardField extends S.TaggedClass<DeleteCardField>()(`${KANBAN_ACTION}/delete-card-field`, {
56
+ input: S.Struct({
57
+ kanban: KanbanType,
58
+ fieldId: S.String,
59
+ // TODO(wittjosiah): Separate fields for undo data?
60
+ deletionData: S.optional(
61
+ S.Struct({
62
+ field: FieldSchema,
63
+ // TODO(wittjosiah): This creates a type error.
64
+ // props: PropertySchema,
65
+ props: S.Any,
66
+ index: S.Number,
67
+ }),
68
+ ),
69
+ }),
70
+ output: S.Void,
71
+ }) {}
72
+
73
+ export class DeleteCard extends S.TaggedClass<DeleteCard>()(`${KANBAN_ACTION}/delete-card`, {
74
+ input: S.Struct({
75
+ card: S.Any, // The card object to delete
76
+ }),
77
+ output: S.Void,
78
+ }) {}
79
+ }
80
+
81
+ // TODO(burdon): Undo?
82
+ // TODO(burdon): Typescript types (replace proto with annotations?)
83
+ // TODO(burdon): Should pure components depend on ECHO? Relationship between ECHO object/array and Observable.
84
+ // TODO(burdon): Can the plugin configure the object based on the data? E.g., how are the models constructed?
85
+ // TODO(burdon): Create models. Simple first based on actual data.
86
+ // Model is always a projection since the dragging state is tentative.
87
+
88
+ // TODO(burdon): Extend model for moving items (in and across columns).
89
+ export interface KanbanModel {
90
+ root: KanbanType;
91
+ }
92
+
93
+ export type Location = {
94
+ idx?: number;
95
+ };
96
+
97
+ export const isKanban = (object: unknown): object is KanbanType => object != null && object instanceof KanbanType;
98
+
99
+ export const createKanban = async (props: { space: Space; initialSchema?: string; initialPivotColumn?: string }) => {
100
+ const { kanban } = await initializeKanban(props);
101
+ return kanban;
102
+ };