240 lines
7.1 KiB
TypeScript
240 lines
7.1 KiB
TypeScript
![]() |
/*
|
|||
|
* @Author: yehuozhili
|
|||
|
* @Date: 2021-03-14 05:42:13
|
|||
|
* @LastEditors: yehuozhili
|
|||
|
* @LastEditTime: 2021-07-05 23:35:05
|
|||
|
* @FilePath: \DooringV2\packages\dooring-v2-lib\src\components\rightConfig.tsx
|
|||
|
*/
|
|||
|
import { CreateOptionsRes } from '../core/components/formTypes';
|
|||
|
import { IBlockType, IStoreData } from '../core/store/storetype';
|
|||
|
import { store } from '../runtime/store';
|
|||
|
import { PropsWithChildren, useEffect, useMemo, useState } from 'react';
|
|||
|
import React from 'react';
|
|||
|
import { Tabs, Input, Row, Col } from 'antd';
|
|||
|
import UserConfig from '../config';
|
|||
|
import { RGBColor, SketchPicker } from 'react-color';
|
|||
|
import { rgba2Obj } from '../core/utils';
|
|||
|
import deepcopy from 'deepcopy';
|
|||
|
|
|||
|
interface RightConfigProps {
|
|||
|
state: IStoreData;
|
|||
|
config: UserConfig;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
*
|
|||
|
* 这里一个需要异步拿取当前注册组件的配置项,另外需要异步加载所需的配置项。
|
|||
|
* @param {*} props
|
|||
|
* @returns
|
|||
|
*/
|
|||
|
function RightConfig(props: PropsWithChildren<RightConfigProps>) {
|
|||
|
const [menuSelect, setMenuSelect] = useState('0');
|
|||
|
const [current, setCurrent] = useState<IBlockType | null>(null);
|
|||
|
const rightMapRenderListCategory = useMemo(() => {
|
|||
|
return props.config.getConfig().rightRenderListCategory;
|
|||
|
}, [props.config]);
|
|||
|
useEffect(() => {
|
|||
|
const fn = () => {
|
|||
|
let item: IBlockType | undefined;
|
|||
|
store.getData().block.some((v) => {
|
|||
|
if (v.focus) {
|
|||
|
item = v;
|
|||
|
}
|
|||
|
return v.focus === true;
|
|||
|
});
|
|||
|
if (item) {
|
|||
|
setCurrent({ ...item });
|
|||
|
} else {
|
|||
|
setCurrent(null);
|
|||
|
}
|
|||
|
};
|
|||
|
const unregist = store.subscribe(fn);
|
|||
|
return () => {
|
|||
|
unregist();
|
|||
|
};
|
|||
|
}, []);
|
|||
|
const render = useMemo(() => {
|
|||
|
return (type: string, current: IBlockType) => {
|
|||
|
const fn = () => props.config.getComponentRegister().getComp(current.name);
|
|||
|
const data = fn();
|
|||
|
// 这里不可能拿不到组件,因为点击的那个组件已经渲染出来了
|
|||
|
if (data) {
|
|||
|
const renderList = data.props[type];
|
|||
|
if (renderList) {
|
|||
|
return renderList.map((v, i) => {
|
|||
|
const Component = props.config.getFormRegister().formMap[v.type];
|
|||
|
if (!Component) {
|
|||
|
console.error(`you might forgot to regist form component ${v.type}`);
|
|||
|
return null;
|
|||
|
}
|
|||
|
return (
|
|||
|
<Component
|
|||
|
key={i}
|
|||
|
data={v as CreateOptionsRes<any, any>}
|
|||
|
current={current}
|
|||
|
config={props.config}
|
|||
|
></Component>
|
|||
|
);
|
|||
|
});
|
|||
|
} else {
|
|||
|
return <div>还没有配置属性</div>;
|
|||
|
}
|
|||
|
}
|
|||
|
return null;
|
|||
|
};
|
|||
|
}, [props.config]);
|
|||
|
|
|||
|
const initColor = useMemo(() => {
|
|||
|
return props.config.getStoreChanger().isEdit()
|
|||
|
? rgba2Obj(props.config.getStoreChanger().getOrigin()?.now.globalState.containerColor)
|
|||
|
: rgba2Obj(props.config.getStore().getData().globalState.containerColor);
|
|||
|
}, [props.config]);
|
|||
|
const [color, setColor] = useState<RGBColor>(initColor);
|
|||
|
const [colorPickerVisible, setColorPickerVisible] = useState(false);
|
|||
|
const initTitle = useMemo(() => {
|
|||
|
const title = props.config.getStoreChanger().isEdit()
|
|||
|
? props.config.getStoreChanger().getOrigin()?.now.globalState.title
|
|||
|
: props.config.getStore().getData().globalState.title;
|
|||
|
return title;
|
|||
|
}, [props.config]);
|
|||
|
const [title, setTitle] = useState(initTitle);
|
|||
|
|
|||
|
const customGlobal = props.config.getConfig().rightGlobalCustom;
|
|||
|
return (
|
|||
|
<div
|
|||
|
className="ant-menu"
|
|||
|
style={{
|
|||
|
height: '100%',
|
|||
|
width: '400px',
|
|||
|
overflow: 'auto',
|
|||
|
padding: '0 10px',
|
|||
|
lineHeight: 1.5715,
|
|||
|
}}
|
|||
|
>
|
|||
|
{current && (
|
|||
|
<Tabs
|
|||
|
activeKey={menuSelect}
|
|||
|
style={{ width: '100%' }}
|
|||
|
onChange={(e) => {
|
|||
|
setMenuSelect(e);
|
|||
|
}}
|
|||
|
>
|
|||
|
{rightMapRenderListCategory.map((v, i) => {
|
|||
|
return (
|
|||
|
<Tabs.TabPane tab={v.icon} key={i + ''}>
|
|||
|
<div
|
|||
|
className="scrollbar"
|
|||
|
style={{
|
|||
|
height: 'calc(100vh - 110px)',
|
|||
|
overflow: 'auto',
|
|||
|
}}
|
|||
|
>
|
|||
|
{v.custom && v.customRender && v.customRender(v.type, current)}
|
|||
|
{!v.custom && render(v.type, current)}
|
|||
|
</div>
|
|||
|
</Tabs.TabPane>
|
|||
|
);
|
|||
|
})}
|
|||
|
</Tabs>
|
|||
|
)}
|
|||
|
{!current && !customGlobal && (
|
|||
|
<div style={{ padding: '20px' }}>
|
|||
|
<Row style={{ padding: '10 0 20px 0', fontWeight: 'bold' }}>全局设置</Row>
|
|||
|
<Row style={{ padding: '10px 0' }}>
|
|||
|
<Col span={6}>标题</Col>
|
|||
|
<Col span={18}>
|
|||
|
<Input
|
|||
|
value={title}
|
|||
|
onChange={(e) => {
|
|||
|
const val = e.target.value;
|
|||
|
setTitle(val);
|
|||
|
const isEdit = props.config.getStoreChanger().isEdit();
|
|||
|
if (isEdit) {
|
|||
|
const originData: IStoreData = deepcopy(
|
|||
|
props.config.getStoreChanger().getOrigin()!.now
|
|||
|
);
|
|||
|
originData.globalState.title = val;
|
|||
|
props.config.getStoreChanger().updateOrigin(originData);
|
|||
|
} else {
|
|||
|
const originData = deepcopy(props.config.getStore().getData());
|
|||
|
originData.globalState.title = val;
|
|||
|
props.config.getStore().setData(originData);
|
|||
|
}
|
|||
|
}}
|
|||
|
/>
|
|||
|
</Col>
|
|||
|
</Row>
|
|||
|
<Row style={{ padding: '10px 0' }}>
|
|||
|
<Col span={6}>容器底色</Col>
|
|||
|
<Col span={18}>
|
|||
|
{
|
|||
|
<div style={{ position: 'relative' }}>
|
|||
|
<div
|
|||
|
onClick={() => {
|
|||
|
setColorPickerVisible((pre) => !pre);
|
|||
|
}}
|
|||
|
style={{
|
|||
|
background: '#fff',
|
|||
|
borderRadius: '1px',
|
|||
|
boxShadow: '0 0 0 1px rgba(0,0,0,.1)',
|
|||
|
cursor: 'pointer',
|
|||
|
display: 'inline-block',
|
|||
|
}}
|
|||
|
>
|
|||
|
<div
|
|||
|
style={{
|
|||
|
width: '20px',
|
|||
|
height: '20px',
|
|||
|
borderRadius: '2px',
|
|||
|
background: `rgba(${color.r}, ${color.g}, ${color.b}, ${color.a})`,
|
|||
|
}}
|
|||
|
/>
|
|||
|
</div>
|
|||
|
{colorPickerVisible && (
|
|||
|
<>
|
|||
|
<div style={{ position: 'absolute', zIndex: 2000 }}>
|
|||
|
<SketchPicker
|
|||
|
color={color}
|
|||
|
onChange={(c) => {
|
|||
|
const newcolor = c.rgb;
|
|||
|
setColor(newcolor);
|
|||
|
const isEdit = props.config.getStoreChanger().isEdit();
|
|||
|
if (isEdit) {
|
|||
|
const originData: IStoreData = deepcopy(
|
|||
|
props.config.getStoreChanger().getOrigin()!.now
|
|||
|
);
|
|||
|
originData.globalState.containerColor = `rgba(${newcolor.r}, ${newcolor.g}, ${newcolor.b}, ${newcolor.a})`;
|
|||
|
props.config.getStoreChanger().updateOrigin(originData);
|
|||
|
} else {
|
|||
|
const originData = deepcopy(props.config.getStore().getData());
|
|||
|
originData.globalState.containerColor = `rgba(${newcolor.r}, ${newcolor.g}, ${newcolor.b}, ${newcolor.a})`;
|
|||
|
props.config.getStore().setData(originData);
|
|||
|
}
|
|||
|
}}
|
|||
|
></SketchPicker>
|
|||
|
</div>
|
|||
|
<div
|
|||
|
style={{
|
|||
|
position: 'fixed',
|
|||
|
top: '0px',
|
|||
|
right: '0px',
|
|||
|
bottom: '0px',
|
|||
|
left: '0px',
|
|||
|
zIndex: 1000,
|
|||
|
}}
|
|||
|
onClick={() => setColorPickerVisible(false)}
|
|||
|
/>
|
|||
|
</>
|
|||
|
)}
|
|||
|
</div>
|
|||
|
}
|
|||
|
</Col>
|
|||
|
</Row>
|
|||
|
</div>
|
|||
|
)}
|
|||
|
{!current && customGlobal && customGlobal}
|
|||
|
</div>
|
|||
|
);
|
|||
|
}
|
|||
|
export default RightConfig;
|