@dxos/plugin-kanban 0.8.4-main.ae835ea → 0.8.4-main.bc674ce
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/blueprint-definition-T2544VMJ.mjs +17 -0
- package/dist/lib/browser/blueprint-definition-T2544VMJ.mjs.map +7 -0
- package/dist/lib/browser/blueprints/index.mjs +8 -0
- package/dist/lib/browser/blueprints/index.mjs.map +7 -0
- package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
- package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +7 -0
- package/dist/lib/browser/chunk-L6N4ZDZ7.mjs +35 -0
- package/dist/lib/browser/chunk-L6N4ZDZ7.mjs.map +7 -0
- package/dist/lib/browser/chunk-XYQO4VL7.mjs +150 -0
- package/dist/lib/browser/chunk-XYQO4VL7.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +58 -65
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/operation-resolver-UEJHX42A.mjs +162 -0
- package/dist/lib/browser/operation-resolver-UEJHX42A.mjs.map +7 -0
- package/dist/lib/browser/react-surface-LFUJAPRL.mjs +236 -0
- package/dist/lib/browser/react-surface-LFUJAPRL.mjs.map +7 -0
- package/dist/lib/browser/types/index.mjs +8 -5
- package/dist/lib/node-esm/blueprint-definition-APJQFSHJ.mjs +18 -0
- package/dist/lib/node-esm/blueprint-definition-APJQFSHJ.mjs.map +7 -0
- package/dist/lib/node-esm/blueprints/index.mjs +9 -0
- package/dist/lib/node-esm/blueprints/index.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs +11 -0
- package/dist/lib/node-esm/chunk-HSLMI22Q.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-NN6JMKIT.mjs +152 -0
- package/dist/lib/node-esm/chunk-NN6JMKIT.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-ZHRMUKTF.mjs +36 -0
- package/dist/lib/node-esm/chunk-ZHRMUKTF.mjs.map +7 -0
- package/dist/lib/node-esm/index.mjs +58 -65
- package/dist/lib/node-esm/index.mjs.map +4 -4
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/operation-resolver-5RPWHZCF.mjs +163 -0
- package/dist/lib/node-esm/operation-resolver-5RPWHZCF.mjs.map +7 -0
- package/dist/lib/node-esm/react-surface-7TSGBRJL.mjs +237 -0
- package/dist/lib/node-esm/react-surface-7TSGBRJL.mjs.map +7 -0
- package/dist/lib/node-esm/types/index.mjs +8 -5
- package/dist/types/src/KanbanPlugin.d.ts +2 -1
- package/dist/types/src/KanbanPlugin.d.ts.map +1 -1
- 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 +22 -0
- package/dist/types/src/blueprints/kanban-blueprint.d.ts.map +1 -0
- package/dist/types/src/capabilities/artifact-definition/artifact-definition.d.ts +12 -0
- package/dist/types/src/capabilities/artifact-definition/artifact-definition.d.ts.map +1 -0
- package/dist/types/src/capabilities/artifact-definition/index.d.ts +3 -0
- package/dist/types/src/capabilities/artifact-definition/index.d.ts.map +1 -0
- package/dist/types/src/capabilities/blueprint-definition/blueprint-definition.d.ts +9 -0
- package/dist/types/src/capabilities/blueprint-definition/blueprint-definition.d.ts.map +1 -0
- package/dist/types/src/capabilities/blueprint-definition/index.d.ts +3 -0
- package/dist/types/src/capabilities/blueprint-definition/index.d.ts.map +1 -0
- package/dist/types/src/capabilities/index.d.ts +3 -3
- package/dist/types/src/capabilities/index.d.ts.map +1 -1
- package/dist/types/src/capabilities/operation-resolver/index.d.ts +3 -0
- package/dist/types/src/capabilities/operation-resolver/index.d.ts.map +1 -0
- package/dist/types/src/capabilities/operation-resolver/operation-resolver.d.ts +5 -0
- package/dist/types/src/capabilities/operation-resolver/operation-resolver.d.ts.map +1 -0
- package/dist/types/src/capabilities/react-surface/index.d.ts +3 -0
- package/dist/types/src/capabilities/react-surface/index.d.ts.map +1 -0
- package/dist/types/src/capabilities/react-surface/react-surface.d.ts +5 -0
- package/dist/types/src/capabilities/react-surface/react-surface.d.ts.map +1 -0
- package/dist/types/src/components/KanbanContainer.d.ts +4 -5
- package/dist/types/src/components/KanbanContainer.d.ts.map +1 -1
- package/dist/types/src/components/KanbanContainer.stories.d.ts +46 -12
- package/dist/types/src/components/KanbanContainer.stories.d.ts.map +1 -1
- package/dist/types/src/components/KanbanViewEditor.d.ts +3 -3
- package/dist/types/src/components/KanbanViewEditor.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/translations.d.ts +21 -9
- package/dist/types/src/translations.d.ts.map +1 -1
- package/dist/types/src/types/schema.d.ts +98 -43
- package/dist/types/src/types/schema.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +63 -47
- package/src/KanbanPlugin.tsx +34 -53
- package/src/blueprints/index.ts +5 -0
- package/src/blueprints/kanban-blueprint.ts +24 -0
- package/src/capabilities/artifact-definition/artifact-definition.ts +150 -0
- package/src/capabilities/artifact-definition/index.ts +7 -0
- package/src/capabilities/blueprint-definition/blueprint-definition.ts +23 -0
- package/src/capabilities/blueprint-definition/index.ts +7 -0
- package/src/capabilities/index.ts +3 -5
- package/src/capabilities/operation-resolver/index.ts +7 -0
- package/src/capabilities/operation-resolver/operation-resolver.ts +133 -0
- package/src/capabilities/react-surface/index.ts +7 -0
- package/src/capabilities/react-surface/react-surface.tsx +86 -0
- package/src/components/KanbanContainer.stories.tsx +197 -104
- package/src/components/KanbanContainer.tsx +37 -46
- package/src/components/KanbanViewEditor.tsx +36 -41
- package/src/meta.ts +2 -2
- package/src/translations.ts +5 -5
- package/src/types/schema.ts +84 -38
- 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/intent-resolver-VVBNS2TO.mjs +0 -111
- package/dist/lib/browser/intent-resolver-VVBNS2TO.mjs.map +0 -7
- package/dist/lib/browser/react-surface-FNXJ6VJX.mjs +0 -255
- package/dist/lib/browser/react-surface-FNXJ6VJX.mjs.map +0 -7
- 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/intent-resolver-ACN7UALP.mjs +0 -112
- package/dist/lib/node-esm/intent-resolver-ACN7UALP.mjs.map +0 -7
- 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/types/src/capabilities/artifact-definition.d.ts +0 -11
- package/dist/types/src/capabilities/artifact-definition.d.ts.map +0 -1
- package/dist/types/src/capabilities/blueprint-definition.d.ts +0 -5
- package/dist/types/src/capabilities/blueprint-definition.d.ts.map +0 -1
- 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/capabilities/react-surface.d.ts +0 -4
- package/dist/types/src/capabilities/react-surface.d.ts.map +0 -1
- package/src/capabilities/artifact-definition.ts +0 -148
- package/src/capabilities/blueprint-definition.ts +0 -30
- package/src/capabilities/intent-resolver.ts +0 -71
- package/src/capabilities/react-surface.tsx +0 -84
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Effect from 'effect/Effect';
|
|
6
|
+
import type * as Schema from 'effect/Schema';
|
|
7
|
+
import React, { useMemo } from 'react';
|
|
8
|
+
|
|
9
|
+
import { Capability, Common } from '@dxos/app-framework';
|
|
10
|
+
import { Database, Obj, Type } from '@dxos/echo';
|
|
11
|
+
import { findAnnotation } from '@dxos/effect';
|
|
12
|
+
import { type FormFieldComponentProps, SelectField, useFormValues } from '@dxos/react-ui-form';
|
|
13
|
+
import { Kanban } from '@dxos/react-ui-kanban/types';
|
|
14
|
+
import { type Collection } from '@dxos/schema';
|
|
15
|
+
|
|
16
|
+
import { KanbanContainer, KanbanViewEditor } from '../../components';
|
|
17
|
+
import { meta } from '../../meta';
|
|
18
|
+
import { PivotColumnAnnotationId } from '../../types';
|
|
19
|
+
|
|
20
|
+
export default Capability.makeModule(() =>
|
|
21
|
+
Effect.succeed(
|
|
22
|
+
Capability.contributes(Common.Capability.ReactSurface, [
|
|
23
|
+
Common.createSurface({
|
|
24
|
+
id: meta.id,
|
|
25
|
+
role: ['article', 'section'],
|
|
26
|
+
filter: (data): data is { subject: Kanban.Kanban } => Obj.instanceOf(Kanban.Kanban, data.subject),
|
|
27
|
+
component: ({ data, role }) => <KanbanContainer role={role} subject={data.subject} />,
|
|
28
|
+
}),
|
|
29
|
+
Common.createSurface({
|
|
30
|
+
id: `${meta.id}/object-settings`,
|
|
31
|
+
role: 'object-settings',
|
|
32
|
+
position: 'hoist',
|
|
33
|
+
filter: (data): data is { subject: Kanban.Kanban } => Obj.instanceOf(Kanban.Kanban, data.subject),
|
|
34
|
+
component: ({ data }) => <KanbanViewEditor object={data.subject} />,
|
|
35
|
+
}),
|
|
36
|
+
Common.createSurface({
|
|
37
|
+
id: `${meta.id}/create-initial-schema-form-[pivot-column]`,
|
|
38
|
+
role: 'form-input',
|
|
39
|
+
filter: (
|
|
40
|
+
data,
|
|
41
|
+
): data is {
|
|
42
|
+
prop: string;
|
|
43
|
+
schema: Schema.Schema<any>;
|
|
44
|
+
target: Database.Database | Collection.Collection | undefined;
|
|
45
|
+
} => {
|
|
46
|
+
const annotation = findAnnotation<boolean>((data.schema as Schema.Schema.All).ast, PivotColumnAnnotationId);
|
|
47
|
+
return !!annotation;
|
|
48
|
+
},
|
|
49
|
+
component: ({ data: { target }, ...inputProps }) => {
|
|
50
|
+
const props = inputProps as any as FormFieldComponentProps;
|
|
51
|
+
const db = Database.isDatabase(target) ? target : target && Obj.getDatabase(target);
|
|
52
|
+
if (!db) {
|
|
53
|
+
return null;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
const { typename } = useFormValues('KanbanForm');
|
|
57
|
+
const [selectedSchema] = useMemo(
|
|
58
|
+
() => db.schemaRegistry.query({ location: ['database', 'runtime'], typename }).runSync(),
|
|
59
|
+
[db, typename],
|
|
60
|
+
);
|
|
61
|
+
const singleSelectColumns = useMemo(() => {
|
|
62
|
+
const properties = Type.toJsonSchema(selectedSchema).properties;
|
|
63
|
+
if (!properties) {
|
|
64
|
+
return [];
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
const columns = Object.entries(properties).reduce<string[]>((acc, [key, value]) => {
|
|
68
|
+
if (typeof value === 'object' && value?.format === 'single-select') {
|
|
69
|
+
acc.push(key);
|
|
70
|
+
}
|
|
71
|
+
return acc;
|
|
72
|
+
}, []);
|
|
73
|
+
|
|
74
|
+
return columns;
|
|
75
|
+
}, [selectedSchema]);
|
|
76
|
+
|
|
77
|
+
if (!typename) {
|
|
78
|
+
return null;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
return <SelectField {...props} options={singleSelectColumns.map((column) => ({ value: column }))} />;
|
|
82
|
+
},
|
|
83
|
+
}),
|
|
84
|
+
]),
|
|
85
|
+
),
|
|
86
|
+
);
|
|
@@ -2,78 +2,98 @@
|
|
|
2
2
|
// Copyright 2024 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import {
|
|
6
|
-
import
|
|
5
|
+
import { RegistryContext } from '@effect-atom/atom-react';
|
|
6
|
+
import { type Decorator, type Meta, type StoryObj } from '@storybook/react-vite';
|
|
7
|
+
import * as Effect from 'effect/Effect';
|
|
8
|
+
import React, { useCallback, useContext } from 'react';
|
|
9
|
+
import { expect, waitFor, within } from 'storybook/test';
|
|
7
10
|
|
|
8
|
-
import { IntentPlugin, SettingsPlugin } from '@dxos/app-framework';
|
|
9
11
|
import { withPluginManager } from '@dxos/app-framework/testing';
|
|
10
12
|
import { Obj, type QueryAST, Type } from '@dxos/echo';
|
|
13
|
+
import { type Mutable } from '@dxos/echo/internal';
|
|
11
14
|
import { invariant } from '@dxos/invariant';
|
|
12
15
|
import { ClientPlugin } from '@dxos/plugin-client';
|
|
13
16
|
import { PreviewPlugin } from '@dxos/plugin-preview';
|
|
14
17
|
import { useGlobalFilteredObjects } from '@dxos/plugin-search';
|
|
15
18
|
import { SpacePlugin } from '@dxos/plugin-space';
|
|
16
|
-
import {
|
|
17
|
-
import { ThemePlugin } from '@dxos/plugin-theme';
|
|
19
|
+
import { StorybookPlugin, corePlugins } from '@dxos/plugin-testing';
|
|
18
20
|
import { faker } from '@dxos/random';
|
|
19
|
-
import {
|
|
20
|
-
import { Filter, useQuery, useSchema, useSpaces } from '@dxos/react-client/echo';
|
|
21
|
+
import { Filter, type Space, useQuery, useSchema, useSpaces } from '@dxos/react-client/echo';
|
|
21
22
|
import { withTheme } from '@dxos/react-ui/testing';
|
|
22
23
|
import { ViewEditor } from '@dxos/react-ui-form';
|
|
23
|
-
import {
|
|
24
|
+
import {
|
|
25
|
+
Kanban as KanbanComponent,
|
|
26
|
+
translations as kanbanTranslations,
|
|
27
|
+
useKanbanModel,
|
|
28
|
+
useProjectionModel,
|
|
29
|
+
} from '@dxos/react-ui-kanban';
|
|
24
30
|
import { Kanban } from '@dxos/react-ui-kanban/types';
|
|
25
|
-
import {
|
|
26
|
-
import {
|
|
27
|
-
import {
|
|
31
|
+
import { JsonFilter } from '@dxos/react-ui-syntax-highlighter';
|
|
32
|
+
import { View, getTypenameFromQuery } from '@dxos/schema';
|
|
33
|
+
import { Organization, Person } from '@dxos/types';
|
|
28
34
|
|
|
29
35
|
import { translations } from '../translations';
|
|
30
36
|
|
|
31
37
|
faker.seed(0);
|
|
32
38
|
|
|
33
|
-
|
|
34
|
-
// Story components.
|
|
35
|
-
//
|
|
36
|
-
|
|
37
|
-
const rollOrg = () => ({
|
|
39
|
+
const createOrg = () => ({
|
|
38
40
|
name: faker.commerce.productName(),
|
|
39
41
|
description: faker.lorem.paragraph(),
|
|
40
42
|
image: faker.image.url(),
|
|
41
43
|
website: faker.internet.url(),
|
|
42
|
-
status: faker.helpers.arrayElement(
|
|
44
|
+
status: faker.helpers.arrayElement(Organization.StatusOptions).id as Organization.Organization['status'],
|
|
43
45
|
});
|
|
44
46
|
|
|
45
|
-
|
|
46
|
-
|
|
47
|
+
//
|
|
48
|
+
// Story setup helpers.
|
|
49
|
+
//
|
|
50
|
+
|
|
51
|
+
type ClientSetupOptions = {
|
|
52
|
+
types?: Type.Entity.Any[];
|
|
53
|
+
onSpaceCreated?: (space: Space) => Promise<void>;
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Creates the standard plugin manager decorator with client configuration.
|
|
58
|
+
*/
|
|
59
|
+
const withKanbanPlugins = ({ types = [], onSpaceCreated }: ClientSetupOptions): Decorator =>
|
|
60
|
+
withPluginManager({
|
|
61
|
+
plugins: [
|
|
62
|
+
...corePlugins(),
|
|
63
|
+
ClientPlugin({
|
|
64
|
+
types: [...types, View.View, Kanban.Kanban],
|
|
65
|
+
onClientInitialized: ({ client }) =>
|
|
66
|
+
Effect.gen(function* () {
|
|
67
|
+
yield* Effect.promise(() => client.halo.createIdentity());
|
|
68
|
+
const space = yield* Effect.promise(() => client.spaces.create());
|
|
69
|
+
yield* Effect.promise(() => space.waitUntilReady());
|
|
70
|
+
yield* Effect.promise(() => onSpaceCreated?.(space) ?? Promise.resolve());
|
|
71
|
+
}),
|
|
72
|
+
}),
|
|
73
|
+
PreviewPlugin(),
|
|
74
|
+
SpacePlugin({}),
|
|
75
|
+
StorybookPlugin({}),
|
|
76
|
+
],
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
//
|
|
80
|
+
// Story components.
|
|
81
|
+
//
|
|
82
|
+
|
|
83
|
+
const DefaultComponent = () => {
|
|
84
|
+
const registry = useContext(RegistryContext);
|
|
47
85
|
const spaces = useSpaces();
|
|
48
86
|
const space = spaces[spaces.length - 1];
|
|
49
|
-
const
|
|
50
|
-
const
|
|
51
|
-
const
|
|
52
|
-
|
|
53
|
-
const
|
|
54
|
-
|
|
55
|
-
useEffect(() => {
|
|
56
|
-
if (views.length && !view) {
|
|
57
|
-
const view = views[0];
|
|
58
|
-
setView(view);
|
|
59
|
-
}
|
|
60
|
-
}, [views]);
|
|
61
|
-
|
|
62
|
-
useEffect(() => {
|
|
63
|
-
if (view?.projection && schema) {
|
|
64
|
-
const jsonSchema = Type.toJsonSchema(schema);
|
|
65
|
-
setProjection(new ProjectionModel(jsonSchema, view.projection));
|
|
66
|
-
}
|
|
67
|
-
// TODO(ZaymonFC): Is there a better way to get notified about deep changes in the json schema?
|
|
68
|
-
// @dmaretskyi? Once resolved, update in multiple places (e.g., storybooks).
|
|
69
|
-
}, [view?.projection, schema, JSON.stringify(schema ? Type.toJsonSchema(schema) : {})]);
|
|
70
|
-
|
|
71
|
-
const objects = useQuery(space, schema ? Filter.type(schema) : Filter.nothing());
|
|
87
|
+
const [object] = useQuery(space?.db, Filter.type(Kanban.Kanban));
|
|
88
|
+
const typename = object?.view.target?.query ? getTypenameFromQuery(object.view.target.query.ast) : undefined;
|
|
89
|
+
const schema = useSchema(space?.db, typename);
|
|
90
|
+
|
|
91
|
+
const objects = useQuery(space?.db, schema ? Filter.type(schema) : Filter.nothing());
|
|
72
92
|
const filteredObjects = useGlobalFilteredObjects(objects);
|
|
73
93
|
|
|
94
|
+
const projection = useProjectionModel(schema, object, registry);
|
|
74
95
|
const model = useKanbanModel({
|
|
75
|
-
|
|
76
|
-
schema,
|
|
96
|
+
object,
|
|
77
97
|
projection,
|
|
78
98
|
items: filteredObjects,
|
|
79
99
|
});
|
|
@@ -81,9 +101,9 @@ const StorybookKanban = () => {
|
|
|
81
101
|
const handleAddCard = useCallback(
|
|
82
102
|
(columnValue: string | undefined) => {
|
|
83
103
|
const path = model?.columnFieldPath;
|
|
84
|
-
if (space && schema && path) {
|
|
104
|
+
if (space && schema && Type.isObjectSchema(schema) && path) {
|
|
85
105
|
const card = Obj.make(schema, {
|
|
86
|
-
...
|
|
106
|
+
...createOrg(),
|
|
87
107
|
[path]: columnValue,
|
|
88
108
|
});
|
|
89
109
|
|
|
@@ -94,100 +114,173 @@ const StorybookKanban = () => {
|
|
|
94
114
|
[space, schema, model],
|
|
95
115
|
);
|
|
96
116
|
|
|
97
|
-
const handleRemoveCard = useCallback((card: { id: string }) => space
|
|
117
|
+
const handleRemoveCard = useCallback((card: { id: string }) => Obj.isObject(card) && space?.db.remove(card), [space]);
|
|
98
118
|
|
|
99
119
|
const handleUpdateQuery = useCallback(
|
|
100
120
|
(newQuery: QueryAST.Query) => {
|
|
101
121
|
invariant(schema);
|
|
102
122
|
invariant(Type.isMutable(schema));
|
|
103
|
-
invariant(view);
|
|
123
|
+
invariant(object.view.target);
|
|
104
124
|
|
|
105
125
|
schema.updateTypename(getTypenameFromQuery(newQuery));
|
|
106
|
-
view.
|
|
126
|
+
Obj.change(object.view.target, (v) => {
|
|
127
|
+
v.query.ast = newQuery as Mutable<typeof newQuery>;
|
|
128
|
+
});
|
|
129
|
+
},
|
|
130
|
+
[object, schema],
|
|
131
|
+
);
|
|
132
|
+
|
|
133
|
+
const handleDeleteField = useCallback(
|
|
134
|
+
(fieldId: string) => {
|
|
135
|
+
if (schema && Type.isMutable(schema) && projection) {
|
|
136
|
+
projection.deleteFieldProjection(fieldId);
|
|
137
|
+
}
|
|
107
138
|
},
|
|
108
|
-
[
|
|
139
|
+
[schema, projection],
|
|
109
140
|
);
|
|
110
141
|
|
|
111
|
-
if (!schema || !view) {
|
|
142
|
+
if (!schema || !object.view.target) {
|
|
112
143
|
return null;
|
|
113
144
|
}
|
|
114
145
|
|
|
115
146
|
return (
|
|
116
|
-
<div className='grow grid grid-cols-[1fr_350px]'>
|
|
147
|
+
<div className='grow grid grid-cols-[1fr_350px] overflow-hidden'>
|
|
117
148
|
{model ? <KanbanComponent model={model} onAddCard={handleAddCard} onRemoveCard={handleRemoveCard} /> : <div />}
|
|
118
|
-
<div className='flex flex-col bs-full border-
|
|
149
|
+
<div className='flex flex-col bs-full overflow-hidden border-l border-separator'>
|
|
119
150
|
<ViewEditor
|
|
151
|
+
classNames='p-2'
|
|
120
152
|
registry={space?.db.schemaRegistry}
|
|
121
153
|
schema={schema}
|
|
122
|
-
view={view}
|
|
154
|
+
view={object.view.target}
|
|
123
155
|
onQueryChanged={handleUpdateQuery}
|
|
124
|
-
onDelete={(
|
|
125
|
-
console.log('[ViewEditor]', 'onDelete', fieldId);
|
|
126
|
-
}}
|
|
156
|
+
onDelete={Type.isMutable(schema) ? handleDeleteField : undefined}
|
|
127
157
|
/>
|
|
128
|
-
<
|
|
129
|
-
{JSON.stringify({ view, schema }, null, 2)}
|
|
130
|
-
</SyntaxHighlighter>
|
|
158
|
+
<JsonFilter data={{ view: object.view.target, schema }} classNames='text-xs' />
|
|
131
159
|
</div>
|
|
132
160
|
</div>
|
|
133
161
|
);
|
|
134
162
|
};
|
|
135
163
|
|
|
136
|
-
type StoryProps = {
|
|
137
|
-
rows?: number;
|
|
138
|
-
};
|
|
139
|
-
|
|
140
164
|
//
|
|
141
165
|
// Story definitions.
|
|
142
166
|
//
|
|
143
167
|
|
|
144
168
|
const meta = {
|
|
145
169
|
title: 'plugins/plugin-kanban/Kanban',
|
|
146
|
-
component:
|
|
147
|
-
render: () => <
|
|
148
|
-
decorators: [
|
|
149
|
-
withTheme,
|
|
150
|
-
withPluginManager({
|
|
151
|
-
plugins: [
|
|
152
|
-
ClientPlugin({
|
|
153
|
-
types: [DataType.Organization, DataType.Person, DataType.View, Kanban.Kanban],
|
|
154
|
-
onClientInitialized: async ({ client }) => {
|
|
155
|
-
await client.halo.createIdentity();
|
|
156
|
-
const space = await client.spaces.create();
|
|
157
|
-
await space.waitUntilReady();
|
|
158
|
-
const { view } = await Kanban.makeView({
|
|
159
|
-
client,
|
|
160
|
-
space,
|
|
161
|
-
typename: DataType.Organization.typename,
|
|
162
|
-
pivotFieldName: 'status',
|
|
163
|
-
});
|
|
164
|
-
space.db.add(view);
|
|
165
|
-
|
|
166
|
-
// TODO(burdon): Replace with sdk/schema/testing.
|
|
167
|
-
Array.from({ length: 80 }).map(() => {
|
|
168
|
-
return space.db.add(Obj.make(DataType.Organization, rollOrg()));
|
|
169
|
-
});
|
|
170
|
-
},
|
|
171
|
-
}),
|
|
172
|
-
SpacePlugin({}),
|
|
173
|
-
IntentPlugin(),
|
|
174
|
-
SettingsPlugin(),
|
|
175
|
-
|
|
176
|
-
// UI
|
|
177
|
-
ThemePlugin({ tx: defaultTx }),
|
|
178
|
-
PreviewPlugin(),
|
|
179
|
-
StorybookLayoutPlugin({}),
|
|
180
|
-
],
|
|
181
|
-
}),
|
|
182
|
-
],
|
|
170
|
+
component: DefaultComponent,
|
|
171
|
+
render: () => <DefaultComponent />,
|
|
172
|
+
decorators: [withTheme],
|
|
183
173
|
parameters: {
|
|
184
174
|
layout: 'fullscreen',
|
|
185
|
-
translations,
|
|
175
|
+
translations: [...translations, ...kanbanTranslations],
|
|
186
176
|
},
|
|
187
|
-
} satisfies Meta<typeof
|
|
177
|
+
} satisfies Meta<typeof DefaultComponent>;
|
|
188
178
|
|
|
189
179
|
export default meta;
|
|
190
180
|
|
|
191
181
|
type Story = StoryObj<typeof meta>;
|
|
192
182
|
|
|
193
|
-
|
|
183
|
+
/**
|
|
184
|
+
* Default story using static runtime schema (immutable).
|
|
185
|
+
* Schema mutations are not allowed.
|
|
186
|
+
*/
|
|
187
|
+
export const Default: Story = {
|
|
188
|
+
decorators: [
|
|
189
|
+
withKanbanPlugins({
|
|
190
|
+
types: [Organization.Organization, Person.Person],
|
|
191
|
+
onSpaceCreated: async (space) => {
|
|
192
|
+
const { view } = await View.makeFromDatabase({
|
|
193
|
+
db: space.db,
|
|
194
|
+
typename: Organization.Organization.typename,
|
|
195
|
+
pivotFieldName: 'status',
|
|
196
|
+
});
|
|
197
|
+
const kanban = Kanban.make({ view });
|
|
198
|
+
space.db.add(kanban);
|
|
199
|
+
|
|
200
|
+
// TODO(burdon): Replace with sdk/schema/testing.
|
|
201
|
+
Array.from({ length: 80 }).map(() => {
|
|
202
|
+
return space.db.add(Obj.make(Organization.Organization, createOrg()));
|
|
203
|
+
});
|
|
204
|
+
},
|
|
205
|
+
}),
|
|
206
|
+
],
|
|
207
|
+
play: async ({ canvasElement }) => {
|
|
208
|
+
const canvas = within(canvasElement);
|
|
209
|
+
|
|
210
|
+
// Wait for the kanban columns to render by finding the status tags.
|
|
211
|
+
// Organization.StatusOptions: prospect, qualified, active, commit, reject.
|
|
212
|
+
const activeTag = await canvas.findByText('Active', undefined, { timeout: 30_000 });
|
|
213
|
+
const prospectTag = await canvas.findByText('Prospect', undefined, { timeout: 10_000 });
|
|
214
|
+
const commitTag = await canvas.findByText('Commit', undefined, { timeout: 10_000 });
|
|
215
|
+
|
|
216
|
+
// Verify all expected columns are rendered.
|
|
217
|
+
await expect(activeTag).toBeTruthy();
|
|
218
|
+
await expect(prospectTag).toBeTruthy();
|
|
219
|
+
await expect(commitTag).toBeTruthy();
|
|
220
|
+
|
|
221
|
+
// Find the column containers.
|
|
222
|
+
const activeColumn = activeTag.closest('[data-dx-stack-item]') as HTMLElement;
|
|
223
|
+
const prospectColumn = prospectTag.closest('[data-dx-stack-item]') as HTMLElement;
|
|
224
|
+
await expect(activeColumn).toBeTruthy();
|
|
225
|
+
await expect(prospectColumn).toBeTruthy();
|
|
226
|
+
|
|
227
|
+
// Wait for cards to render in the columns.
|
|
228
|
+
// Cards have data-dx-item-id attribute from StackItem.Root.
|
|
229
|
+
const getColumnCards = (column: HTMLElement) =>
|
|
230
|
+
Array.from(column.querySelectorAll('[data-dx-item-id]')) as HTMLElement[];
|
|
231
|
+
|
|
232
|
+
await waitFor(() => expect(getColumnCards(activeColumn).length).toBeGreaterThan(0));
|
|
233
|
+
|
|
234
|
+
// Verify cards are distributed across columns.
|
|
235
|
+
const activeCards = getColumnCards(activeColumn);
|
|
236
|
+
const prospectCards = getColumnCards(prospectColumn);
|
|
237
|
+
await expect(activeCards.length).toBeGreaterThan(0);
|
|
238
|
+
await expect(prospectCards.length).toBeGreaterThan(0);
|
|
239
|
+
|
|
240
|
+
// Verify cards have drag handles (first button in toolbar).
|
|
241
|
+
const firstActiveCard = activeCards[0];
|
|
242
|
+
const buttons = firstActiveCard.querySelectorAll('button');
|
|
243
|
+
await expect(buttons.length).toBeGreaterThan(0);
|
|
244
|
+
|
|
245
|
+
// Verify the drop zone exists in each column.
|
|
246
|
+
const activeDropZone = activeColumn.querySelector('.kanban-drop');
|
|
247
|
+
const prospectDropZone = prospectColumn.querySelector('.kanban-drop');
|
|
248
|
+
await expect(activeDropZone).toBeTruthy();
|
|
249
|
+
await expect(prospectDropZone).toBeTruthy();
|
|
250
|
+
|
|
251
|
+
// TODO(wittjosiah): Get drag & drop tests working.
|
|
252
|
+
// See packages/apps/composer-app/src/playwright/stack.spec.ts for reference.
|
|
253
|
+
},
|
|
254
|
+
};
|
|
255
|
+
|
|
256
|
+
/**
|
|
257
|
+
* Story variant that uses a mutable database schema (EchoSchema).
|
|
258
|
+
* This allows testing schema mutations like adding/removing fields.
|
|
259
|
+
*/
|
|
260
|
+
// TODO(wittjosiah): Card previews (e.g., OrganizationCard) are type-specific and hard-coded.
|
|
261
|
+
// They don't use the projection to determine which fields to display, so deleting a field
|
|
262
|
+
// from the schema won't remove it from the card preview. To fix this, the type-specific
|
|
263
|
+
// cards in PreviewPlugin would need to accept and respect the projection prop.
|
|
264
|
+
export const MutableSchema: Story = {
|
|
265
|
+
decorators: [
|
|
266
|
+
withKanbanPlugins({
|
|
267
|
+
onSpaceCreated: async (space) => {
|
|
268
|
+
// Register schema in the database to make it mutable (EchoSchema).
|
|
269
|
+
const [schema] = await space.db.schemaRegistry.register([Organization.Organization]);
|
|
270
|
+
|
|
271
|
+
const { view } = await View.makeFromDatabase({
|
|
272
|
+
db: space.db,
|
|
273
|
+
typename: schema.typename,
|
|
274
|
+
pivotFieldName: 'status',
|
|
275
|
+
});
|
|
276
|
+
const kanban = Kanban.make({ view });
|
|
277
|
+
space.db.add(kanban);
|
|
278
|
+
|
|
279
|
+
// Create test data using the registered schema.
|
|
280
|
+
Array.from({ length: 80 }).map(() => {
|
|
281
|
+
return space.db.add(Obj.make(schema, createOrg()));
|
|
282
|
+
});
|
|
283
|
+
},
|
|
284
|
+
}),
|
|
285
|
+
],
|
|
286
|
+
};
|
|
@@ -2,68 +2,59 @@
|
|
|
2
2
|
// Copyright 2024 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import
|
|
5
|
+
import { RegistryContext } from '@effect-atom/atom-react';
|
|
6
|
+
import React, { useCallback, useContext, useEffect, useState } from 'react';
|
|
6
7
|
|
|
7
|
-
import {
|
|
8
|
+
import { Common } from '@dxos/app-framework';
|
|
9
|
+
import { type SurfaceComponentProps, useCapabilities, useOperationInvoker } from '@dxos/app-framework/react';
|
|
8
10
|
import { Filter, Obj, Type } from '@dxos/echo';
|
|
9
|
-
import { EchoSchema, type TypedObject } from '@dxos/echo/internal';
|
|
10
11
|
import { useGlobalFilteredObjects } from '@dxos/plugin-search';
|
|
11
|
-
import {
|
|
12
|
-
import {
|
|
13
|
-
import { Kanban
|
|
14
|
-
import {
|
|
15
|
-
import {
|
|
12
|
+
import { useQuery } from '@dxos/react-client/echo';
|
|
13
|
+
import { Kanban as KanbanComponent, useKanbanModel, useProjectionModel } from '@dxos/react-ui-kanban';
|
|
14
|
+
import { type Kanban } from '@dxos/react-ui-kanban/types';
|
|
15
|
+
import { Layout } from '@dxos/react-ui-mosaic';
|
|
16
|
+
import { getTypenameFromQuery } from '@dxos/schema';
|
|
16
17
|
|
|
17
|
-
import {
|
|
18
|
+
import { KanbanOperation } from '../types';
|
|
18
19
|
|
|
19
|
-
export
|
|
20
|
-
const client = useClient();
|
|
21
|
-
const [cardSchema, setCardSchema] = useState<TypedObject<any, any>>();
|
|
22
|
-
const [projection, setProjection] = useState<ProjectionModel>();
|
|
23
|
-
const space = getSpace(view);
|
|
24
|
-
const { dispatchPromise: dispatch } = useIntentDispatcher();
|
|
25
|
-
const typename = view.query ? getTypenameFromQuery(view.query.ast) : undefined;
|
|
20
|
+
export type KanbanContainerProps = SurfaceComponentProps<Kanban.Kanban>;
|
|
26
21
|
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
}
|
|
22
|
+
export const KanbanContainer = ({ role, subject: object }: KanbanContainerProps) => {
|
|
23
|
+
const registry = useContext(RegistryContext);
|
|
24
|
+
const schemas = useCapabilities(Common.Capability.Schema);
|
|
25
|
+
const [cardSchema, setCardSchema] = useState<Type.Obj.Any>();
|
|
26
|
+
const db = Obj.getDatabase(object);
|
|
27
|
+
const { invokePromise } = useOperationInvoker();
|
|
28
|
+
const typename = object.view.target?.query ? getTypenameFromQuery(object.view.target.query.ast) : undefined;
|
|
33
29
|
|
|
34
30
|
useEffect(() => {
|
|
35
|
-
const staticSchema =
|
|
31
|
+
const staticSchema = schemas.flat().find((schema) => Type.getTypename(schema) === typename);
|
|
36
32
|
if (staticSchema) {
|
|
37
|
-
|
|
33
|
+
// NOTE: Use functional update to prevent React from calling the schema as a function.
|
|
34
|
+
setCardSchema(() => staticSchema);
|
|
38
35
|
}
|
|
39
|
-
if (!staticSchema && typename &&
|
|
40
|
-
const query =
|
|
36
|
+
if (!staticSchema && typename && db) {
|
|
37
|
+
const query = db.schemaRegistry.query({ typename });
|
|
41
38
|
const unsubscribe = query.subscribe(
|
|
42
39
|
() => {
|
|
43
40
|
const [schema] = query.results;
|
|
44
41
|
if (schema) {
|
|
45
|
-
|
|
42
|
+
// NOTE: Use functional update to prevent React from calling the schema as a function.
|
|
43
|
+
setCardSchema(() => schema);
|
|
46
44
|
}
|
|
47
45
|
},
|
|
48
46
|
{ fire: true },
|
|
49
47
|
);
|
|
50
48
|
return unsubscribe;
|
|
51
49
|
}
|
|
52
|
-
}, [
|
|
53
|
-
|
|
54
|
-
useEffect(() => {
|
|
55
|
-
if (jsonSchema) {
|
|
56
|
-
setProjection(new ProjectionModel(jsonSchema, view.projection));
|
|
57
|
-
}
|
|
58
|
-
// TODO(ZaymonFC): Is there a better way to get notified about deep changes in the json schema?
|
|
59
|
-
}, [view.projection, JSON.stringify(jsonSchema)]);
|
|
50
|
+
}, [schemas, db, typename]);
|
|
60
51
|
|
|
61
|
-
const objects = useQuery(
|
|
52
|
+
const objects = useQuery(db, cardSchema ? Filter.type(cardSchema) : Filter.nothing());
|
|
62
53
|
const filteredObjects = useGlobalFilteredObjects(objects);
|
|
63
54
|
|
|
55
|
+
const projection = useProjectionModel(cardSchema, object, registry);
|
|
64
56
|
const model = useKanbanModel({
|
|
65
|
-
|
|
66
|
-
schema: cardSchema,
|
|
57
|
+
object,
|
|
67
58
|
projection,
|
|
68
59
|
items: filteredObjects,
|
|
69
60
|
});
|
|
@@ -71,25 +62,25 @@ export const KanbanContainer = ({ view }: { view: DataType.View; role: string })
|
|
|
71
62
|
const handleAddCard = useCallback(
|
|
72
63
|
(columnValue: string | undefined) => {
|
|
73
64
|
const path = model?.columnFieldPath;
|
|
74
|
-
if (
|
|
65
|
+
if (db && cardSchema && path) {
|
|
75
66
|
const card = Obj.make(cardSchema, { [path]: columnValue });
|
|
76
|
-
|
|
67
|
+
db.add(card);
|
|
77
68
|
return card.id;
|
|
78
69
|
}
|
|
79
70
|
},
|
|
80
|
-
[
|
|
71
|
+
[db, cardSchema, model],
|
|
81
72
|
);
|
|
82
73
|
|
|
83
74
|
const handleRemoveCard = useCallback(
|
|
84
75
|
(card: { id: string }) => {
|
|
85
|
-
void
|
|
76
|
+
void invokePromise(KanbanOperation.DeleteCard, { card });
|
|
86
77
|
},
|
|
87
|
-
[
|
|
78
|
+
[invokePromise],
|
|
88
79
|
);
|
|
89
80
|
|
|
90
81
|
return (
|
|
91
|
-
<
|
|
92
|
-
{model && <
|
|
93
|
-
</
|
|
82
|
+
<Layout.Main role={role}>
|
|
83
|
+
{model && <KanbanComponent model={model} onAddCard={handleAddCard} onRemoveCard={handleRemoveCard} />}
|
|
84
|
+
</Layout.Main>
|
|
94
85
|
);
|
|
95
86
|
};
|