2022-03-09 17:20:39 +08:00

307 lines
9.1 KiB
TypeScript

import { TreeSelect } from '@formily/antd';
import { Field, onFieldChange } from '@formily/core';
import { ISchema, Schema, useField, useFieldSchema } from '@formily/react';
import React from 'react';
import { useTranslation } from 'react-i18next';
import { findByUid } from '.';
import { createDesignable } from '../..';
import { GeneralSchemaDesigner, SchemaSettings, useAPIClient, useDesignable } from '../../../';
const toItems = (properties = {}) => {
const items = [];
for (const key in properties) {
if (Object.prototype.hasOwnProperty.call(properties, key)) {
const element = properties[key];
const item = {
label: element.title,
value: `${element['x-uid']}||${element['x-component']}`,
};
if (element.properties) {
const children = toItems(element.properties);
if (children?.length) {
item['children'] = children;
}
}
items.push(item);
}
}
return items;
};
const findMenuSchema = (fieldSchema: Schema) => {
let parent = fieldSchema.parent;
while (parent) {
if (parent['x-component'] === 'Menu') {
return parent;
}
parent = parent.parent;
}
};
const InsertMenuItems = (props) => {
const { eventKey, title, insertPosition } = props;
const { t } = useTranslation();
const { dn } = useDesignable();
const fieldSchema = useFieldSchema();
const isSubMenu = fieldSchema['x-component'] === 'Menu.SubMenu';
if (!isSubMenu && insertPosition === 'beforeEnd') {
return null;
}
return (
<SchemaSettings.SubMenu eventKey={eventKey} title={title}>
<SchemaSettings.ModalItem
eventKey={`${insertPosition}group`}
title={t('Group')}
schema={
{
type: 'object',
title: t('Add group'),
properties: {
title: {
'x-decorator': 'FormItem',
'x-component': 'Input',
title: t('Menu item title'),
'x-component-props': {},
// description: `原字段标题:${collectionField?.uiSchema?.title}`,
},
icon: {
title: t('Icon'),
'x-component': 'IconPicker',
'x-decorator': 'FormItem',
},
},
} as ISchema
}
onSubmit={({ title, icon }) => {
dn.insertAdjacent(insertPosition, {
type: 'void',
title,
'x-component': 'Menu.SubMenu',
'x-component-props': {
icon,
},
});
}}
/>
<SchemaSettings.ModalItem
eventKey={`${insertPosition}page`}
title={t('Page')}
schema={
{
type: 'object',
title: t('Add page'),
properties: {
title: {
'x-decorator': 'FormItem',
'x-component': 'Input',
title: t('Menu item title'),
'x-component-props': {},
},
icon: {
title: t('Icon'),
'x-component': 'IconPicker',
'x-decorator': 'FormItem',
},
},
} as ISchema
}
onSubmit={({ title, icon }) => {
dn.insertAdjacent(insertPosition, {
type: 'void',
title,
'x-component': 'Menu.Item',
'x-component-props': {
icon,
},
properties: {
page: {
type: 'void',
'x-component': 'Page',
'x-async': true,
properties: {
grid: {
type: 'void',
'x-component': 'Grid',
'x-initializer': 'BlockInitializers',
properties: {},
},
},
},
},
});
}}
/>
<SchemaSettings.ModalItem
eventKey={`${insertPosition}link`}
title={t('Link')}
schema={
{
type: 'object',
title: t('Add link'),
properties: {
title: {
title: t('Menu item title'),
'x-component': 'Input',
'x-decorator': 'FormItem',
},
icon: {
title: t('Icon'),
'x-component': 'IconPicker',
'x-decorator': 'FormItem',
},
href: {
title: t('Link'),
'x-component': 'Input',
'x-decorator': 'FormItem',
},
},
} as ISchema
}
onSubmit={({ title, icon, href }) => {
dn.insertAdjacent(insertPosition, {
type: 'void',
title,
'x-component': 'Menu.URL',
'x-component-props': {
icon,
href,
},
});
}}
/>
</SchemaSettings.SubMenu>
);
};
export const MenuDesigner = () => {
const field = useField();
const fieldSchema = useFieldSchema();
const api = useAPIClient();
const { dn, refresh } = useDesignable();
const { t } = useTranslation();
const menuSchema = findMenuSchema(fieldSchema);
const items = toItems(menuSchema?.properties);
const effects = (form) => {
onFieldChange('target', (field: Field) => {
const [, component] = field?.value?.split?.('||') || [];
field.query('position').take((f: Field) => {
f.dataSource =
component === 'Menu.SubMenu'
? [
{ label: t('Before'), value: 'beforeBegin' },
{ label: t('After'), value: 'afterEnd' },
{ label: t('Inner'), value: 'beforeEnd' },
]
: [
{ label: t('Before'), value: 'beforeBegin' },
{ label: t('After'), value: 'afterEnd' },
];
});
});
};
return (
<GeneralSchemaDesigner>
<SchemaSettings.ModalItem
title={t('Edit')}
schema={
{
type: 'object',
title: t('Edit menu item'),
properties: {
title: {
title: t('Menu item title'),
'x-decorator': 'FormItem',
'x-component': 'Input',
'x-component-props': {},
},
icon: {
title: t('Menu item icon'),
'x-component': 'IconPicker',
'x-decorator': 'FormItem',
},
},
} as ISchema
}
initialValues={{
title: field.title,
icon: field.componentProps.icon,
}}
onSubmit={({ title, icon }) => {
const schema = {
['x-uid']: fieldSchema['x-uid'],
};
if (title) {
fieldSchema.title = title;
field.title = title;
schema['title'] = title;
refresh();
}
field.componentProps.icon = icon;
schema['x-component-props'] = { icon };
fieldSchema['x-component-props'] = fieldSchema['x-component-props'] || {};
fieldSchema['x-component-props']['icon'] = icon;
dn.emit('patch', {
schema,
});
}}
/>
<SchemaSettings.ModalItem
title={t('Move to')}
components={{ TreeSelect }}
effects={effects}
schema={
{
type: 'object',
title: t('Move to'),
properties: {
target: {
title: t('Target'),
enum: items,
required: true,
'x-decorator': 'FormItem',
'x-component': 'TreeSelect',
'x-component-props': {},
},
position: {
title: t('Position'),
required: true,
enum: [
{ label: t('Before'), value: 'beforeBegin' },
{ label: t('After'), value: 'afterEnd' },
],
default: 'afterEnd',
'x-component': 'Radio.Group',
'x-decorator': 'FormItem',
},
},
} as ISchema
}
onSubmit={({ target, position }) => {
const [uid] = target?.split?.('||') || [];
if (!uid) {
return;
}
const current = findByUid(menuSchema, uid);
const dn = createDesignable({
api,
refresh,
current,
});
dn.loadAPIClientEvents();
dn.insertAdjacent(position, fieldSchema);
}}
/>
<SchemaSettings.Divider />
<InsertMenuItems eventKey={'insertbeforeBegin'} title={t('Insert before')} insertPosition={'beforeBegin'} />
<InsertMenuItems eventKey={'insertafterEnd'} title={t('Insert after')} insertPosition={'afterEnd'} />
<InsertMenuItems eventKey={'insertbeforeEnd'} title={t('Insert inner')} insertPosition={'beforeEnd'} />
<SchemaSettings.Divider />
<SchemaSettings.Remove
confirm={{
title: t('Delete menu item'),
}}
/>
</GeneralSchemaDesigner>
);
};