nocobase/docs/en-US/development/client/ui-schema-designer/designable.md

350 lines
8.7 KiB
Markdown
Raw Normal View History

2022-12-21 01:55:59 +00:00
# Schema design capabilities
2022-12-21 01:55:59 +00:00
The design capabilities of Schema are mainly
2022-12-21 01:55:59 +00:00
- Neighborhood insertion, which can be used to
- Insertion of new schema nodes
- Drag-and-drop movement of existing schema nodes
- schema parameter modification
2022-12-21 01:55:59 +00:00
The core designer APIs and parameters are
2022-12-21 01:55:59 +00:00
- Designer API: `createDesignable()` & `useDesignable()`
- Schema parameters: `x-designer`, used to adapt the designer component
2022-12-21 01:55:59 +00:00
## Designer API
### createDesignable()
```ts
import { Schema } from '@nocobase/client';
const current = new Schema({
type: 'void',
'x-component': 'div',
});
const {
2022-12-21 01:55:59 +00:00
designable, // whether it is configurable
remove,
2022-12-21 01:55:59 +00:00
insertAdjacent, // insert at a position, four positions: beforeBegin, afterBegin, beforeEnd, afterEnd
insertBeforeBegin, // insert in front of the current node
insertAfterBegin, // insert in front of the first child node of the current node
insertBeforeEnd, // after the last child of the current node
insertAfterEnd, // after the current node
} = createDesignable({
current,
});
const newSchema = {
type: 'void',
name: 'hello',
'x-component': 'Hello',
};
insertAfterBegin(newSchema);
console.log(current.toJSON());
{
type: 'void',
'x-component': 'div',
properties: {
hello: {
type: 'void',
'x-component': 'Hello',
},
},
}
```
### useDesignable()
2022-12-21 01:55:59 +00:00
React Hook scenarios can also use `useDesignable()` to get the API of the current schema component designer
```ts
const {
2022-12-21 01:55:59 +00:00
designable, // whether it is configurable
remove,
2022-12-21 01:55:59 +00:00
insertAdjacent, // insert at a position, four positions: beforeBegin, afterBegin, beforeEnd, afterEnd
insertBeforeBegin, // insert in front of the current node
insertAfterBegin, // insert in front of the first child node of the current node
insertBeforeEnd, // after the last child of the current node
insertAfterEnd, // after the current node
} = useDesignable();
const schema = {
name: uid(),
'x-component': 'Hello',
};
2022-12-21 01:55:59 +00:00
// Insert in front of the current node
insertBeforeBegin(schema);
2022-12-21 01:55:59 +00:00
// Equivalent to
insertAdjacent('beforeBegin', schema);
2022-12-21 01:55:59 +00:00
// insert in front of the first child of the current node
insertAfterBegin(schema);
2022-12-21 01:55:59 +00:00
// Equivalent to
insertAdjacent('afterBegin', schema);
2022-12-21 01:55:59 +00:00
// after the last child of the current node
insertBeforeEnd(schema);
2022-12-21 01:55:59 +00:00
// Equivalent to
insertAdjacent('beforeEnd', schema);
2022-12-21 01:55:59 +00:00
// After the current node
insertAfterEnd(schema);
2022-12-21 01:55:59 +00:00
// Equivalent to
insertAdjacent('afterEnd', schema);
```
2022-12-21 01:55:59 +00:00
## Neighborhood insertion
2022-12-21 01:55:59 +00:00
Similar to the DOM's [insert adjacent](https://dom.spec.whatwg.org/#insert-adjacent) concept, Schema also provides the `insertAdjacent()` method for solving the insertion of adjacent positions.
2022-12-21 01:55:59 +00:00
The four adjacent positions
```ts
{
properties: {
2022-12-21 01:55:59 +00:00
// beforeBegin insert before the current node
node1: {
properties: {
2022-12-21 01:55:59 +00:00
// afterBegin inserted before the first child of the current node
// ...
2022-12-21 01:55:59 +00:00
// beforeEnd after the last child of the current node
},
},
2022-12-21 01:55:59 +00:00
// afterEnd after the current node
},
}
```
2022-12-21 01:55:59 +00:00
Like HTML tags, the components of the Schema component library can be combined with each other and inserted in reasonable proximity as needed via the insertAdjacent API.
2022-12-21 01:55:59 +00:00
### Inserting a new schema node
2022-12-21 01:55:59 +00:00
Within a Schema component, a new node can be inserted directly into the adjacent position of the current Schema with `useDesignable()`.
2022-12-21 01:55:59 +00:00
Example
```tsx
import React from 'react';
import { SchemaComponentProvider, SchemaComponent, useDesignable } from '@nocobase/client';
import { observer, Schema, useFieldSchema } from '@formily/react';
import { Button, Space } from 'antd';
import { uid } from '@formily/shared';
const Hello = observer((props) => {
const { insertAdjacent } = useDesignable();
const fieldSchema = useFieldSchema();
return (
<div>
<h1>{fieldSchema.name}</h1>
<Space>
<Button
onClick={() => {
insertAdjacent('beforeBegin', {
'x-component': 'Hello',
});
}}
>
before begin
</Button>
<Button
onClick={() => {
insertAdjacent('afterBegin', {
'x-component': 'Hello',
});
}}
>
after begin
</Button>
<Button
onClick={() => {
insertAdjacent('beforeEnd', {
'x-component': 'Hello',
});
}}
>
before end
</Button>
<Button
onClick={() => {
insertAdjacent('afterEnd', {
'x-component': 'Hello',
});
}}
>
after end
</Button>
</Space>
<div style={{ margin: 50 }}>{props.children}</div>
</div>
);
});
const Page = observer((props) => {
return <div>{props.children}</div>;
});
export default () => {
return (
<SchemaComponentProvider components={{ Page, Hello }}>
<SchemaComponent
schema={{
type: 'void',
name: 'page',
'x-component': 'Page',
properties: {
hello1: {
type: 'void',
'x-component': 'Hello',
},
},
}}
/>
</SchemaComponentProvider>
);
}
```
2022-12-21 01:55:59 +00:00
### Drag-and-drop movement of existing schema nodes
2022-12-21 01:55:59 +00:00
Methods such as insertAdjacent can also be used to drag and drop nodes
```tsx
import React from 'react';
import { uid } from '@formily/shared';
import { observer, useField, useFieldSchema } from '@formily/react';
import { DndContext, DragEndEvent, useDraggable, useDroppable } from '@dnd-kit/core';
import { SchemaComponent, SchemaComponentProvider, createDesignable, useDesignable } from '@nocobase/client';
const useDragEnd = () => {
const { refresh } = useDesignable();
return ({ active, over }: DragEndEvent) => {
const activeSchema = active?.data?.current?.schema;
const overSchema = over?.data?.current?.schema;
if (!activeSchema || !overSchema) {
return;
}
const dn = createDesignable({
current: overSchema,
});
dn.on('insertAdjacent', refresh);
dn.insertBeforeBeginOrAfterEnd(activeSchema);
};
};
const Page = observer((props) => {
return <DndContext onDragEnd={useDragEnd()}>{props.children}</DndContext>;
});
function Draggable(props) {
const { attributes, listeners, setNodeRef, transform } = useDraggable({
id: props.id,
data: props.data,
});
const style = transform
? {
transform: `translate3d(${transform.x}px, ${transform.y}px, 0)`,
}
: undefined;
return (
<button ref={setNodeRef} style={style} {...listeners} {...attributes}>
{props.children}
</button>
);
}
function Droppable(props) {
const { isOver, setNodeRef } = useDroppable({
id: props.id,
data: props.data,
});
const style = {
color: isOver ? 'green' : undefined,
};
return (
<div ref={setNodeRef} style={style}>
{props.children}
</div>
);
}
const Block = observer((props) => {
const field = useField();
const fieldSchema = useFieldSchema();
return (
<Droppable id={field.address.toString()} data={{ schema: fieldSchema }}>
<div style={{ marginBottom: 20, padding: '20px', background: '#f1f1f1' }}>
Block {fieldSchema.name}{' '}
<Draggable id={field.address.toString()} data={{ schema: fieldSchema }}>
Drag
</Draggable>
</div>
</Droppable>
);
});
export default function App() {
return (
<SchemaComponentProvider components={{ Page, Block }}>
<SchemaComponent
schema={{
type: 'void',
name: 'page',
'x-component': 'Page',
properties: {
block1: {
'x-component': 'Block',
},
block2: {
'x-component': 'Block',
},
block3: {
'x-component': 'Block',
},
},
}}
/>
</SchemaComponentProvider>
);
}
```
2022-12-21 01:55:59 +00:00
## Applications of `x-designer`
2022-12-21 01:55:59 +00:00
`x-designer` is usually used only in wrapper components such as BlockItem, CardItem, FormItem, etc.
```ts
{
type: 'object',
properties: {
title: {
type: 'string',
title: '标题',
'x-decorator': 'FormItem',
'x-component': 'Input',
'x-designer': 'FormItem.Designer',
},
status: {
type: 'string',
title: '状态',
'x-decorator': 'FormItem',
'x-component': 'Select',
'x-designer': 'FormItem.Designer',
},
},
}
```
2022-12-21 01:55:59 +00:00
Note: The Schema designer provided by NocoBase is directly embedded in the interface in the form of a toolbar. When the UI configuration is activated (`designable = true`), the `x-designer` component (designer toolbar) will be displayed and the current schema component can be updated through the toolbar.
2022-12-21 01:55:59 +00:00
- Drag and Drop: DndContext + DragHandler
- Inserting new nodes: SchemaInitializer
- Parameter configuration: SchemaSettings