基于 huxy-admin 模板创建。
低代码就是尽量少写代码,定义好业务组件,通过可视化操作实现开发工作。它主要受众是开发者。
无代码(no-code)即不需要写代码就能完成开发,更加偏向业务层的定制。
提效降本、质量保障、降低开发门槛。
低代码可以提升开发效率,保障系统稳定性,也降低了开发门槛,可以直接可视化开发。
- 不灵活。适用于通用业务领域,对定制化需求不友好。
- 不可控。业务拓展性、可维护性较低。
- 不好用。开发不想用,业务不会用。
低代码或许可以降低开发门槛,但复杂度并不会降低。可视化开发的自由度越高,组件粒度就越细,配置复杂度就越高。
前端低代码开发不仅是界面开发,应该还包含工程化、项目管理、api 接口、权限控制等一些列的开发提效。
一个页面其实就是一棵树,所以不管是拖拽还是配置,最终完成的就是一棵数据树。所以我们可以通过json schema
来进行页面设计。
工程化、layout 设计、权限和 i18n、API 管理 这些都是一些管理平台的基础设施,前面也讲过,大家可以去看看。
const schema = {
type: 'a',
props: {},
children: [],
};
- type:标签名或组件名。组件可以是 UI 组件或业务组件,先注册再使用。
- props:属性配置。组件的属性可根据组件库或自定义组件使用文档去配置。如果属性里面含有组件,可依照 schema 渲染原则执行。
- children:子节点。可以是文本节点,组件,或子元素列表。
const render = (schema, params) => {
schema = Array.isArray(schema) ? schema : [schema];
const dom = schema.map((item, i) => {
let {type, props, children} = item;
type = (type || 'div').trim();
const first = type.charAt(0);
type = first.toUpperCase() === first ? components[type] || 'div' : type;
props = {
key: i,
...formatProps(props, params),
};
children = Array.isArray(children) ? render(children, params) : [formatChildren(children || props.children, params) ?? null];
return createElement(type, props, ...children);
});
return dom;
};
- components:我们注册的组件。
- formatProps、formatChildren:将 props 或 children 转换为我们需要的运行时的值。主要用于我们自定义的组件。props 或 children 可以是函数,可以传递我们需要的参数 params,最终返回我们需要的数据。
- render:通过 react 的
createElement(type,props,...children)
渲染。
属性解析
const render = (schema, params) => {
schema = Array.isArray(schema) ? schema : [schema];
const dom = schema.map((item, i) => {
let {type, props, children} = item;
type = (type || 'div').trim();
const first = type.charAt(0);
type = first.toUpperCase() === first ? components[type] || 'div' : type;
props = {
key: i,
...formatProps(props, params),
};
children = Array.isArray(children) ? render(children, params) : [formatChildren(children || props.children, params) ?? null];
return createElement(type, props, ...children);
});
return dom;
};
可使用自定义函数,组件内部数据作为参数,来获取属性值。或直接使用全局 configs。
例如:
{
prop:'test',
isPending:`{true}`,
style:`{{width:'800px'}}`,
handle:`{self=>self.rules}`,
onClick:`{()=>e=>alert('hello')}`,
}
通过 {code}
将 code 字符串转换为运行时代码。
判断并提取字符串代码
const matchedStr = (str, c = ['{', '}']) => str?.trim?.().match(new RegExp(`^${c[0]}([\\s\\S]*)${c[1]}$`))?.[1];
执行字符串代码
const str2code = (str, hasReturn = false) => {
str = hasReturn ? str : `return ${str};`;
const exec = Function(str);
return exec();
};
str2code 会直接执行并返回结果,如果返回的是函数会执行函数并返回结果。如果我们需要返回函数,就要包裹一层函数。例如:
onClick
,{()=>e=>alert('hello')}
。
路由设置
路由配置可直接在页面配置,存入后台,使用时获取路由配置即可。
{
path:'/low-code',
name:'低代码',
icon:'CoffeeOutlined',
denied:false,
children:[
{
path:'/dom',
name:'原生dom',
icon:'CodeOutlined',
componentPath:'/lowCode',
loadData:{
pageSchema,
},
},
{
path:'/ui',
name:'UI组件',
icon:'CodeOutlined',
componentPath:'/lowCode',
loadData:{
pageSchema,
},
},
{
path:'/users',
name:'业务组件',
icon:'CodeOutlined',
componentPath:'/lowCode',
loadData:{
pageSchema,
},
},
{
path:'/users/add',
name:'新增用户',
componentPath:'/lowCode',
loadData:{
pageSchema,
},
},
{
path:'/users/edit/:id',
name:'编辑用户',
componentPath:'/lowCode',
loadData:{
pageSchema,
},
},
],
}
如果整个系统都是通过 schema
数据配置生成的,那么我们只需一个渲染器,通过路由 id 获取到 shcema
数据,然后渲染成当前路由页面。所以只需一个渲染文件即可。
根据 projectId
、routerId
获取路由页面数据。
const pageSchema = async ({id}) => {
const {result} = await apiList.listSchemaFn({routerId: id, projectId: defProject._id});
return {result};
};
通过设置路由 loadData
来提前请求数据,页面直接获取即可。详细使用见 useRouter
const pageSchema = async ({id}) => {
const {result} = await apiList.listSchemaFn({routerId: id, projectId: defProject._id});
return {result};
};
页面渲染
const Index = props => {
const {pageSchema} = props;
if (pageSchema == null || pageSchema.pending) {
return <Spinner global />;
}
return customRender(pageSchema.result || [], {}, props);
};
首先我们创建一个项目,如图所示。本示例使用 控制台
项目演示。
根据 dom 元素属性自行配置。
当我们设计好页面时,可以随时回到项目路由查看改页面,也可点击预览查看,符合预期效果后保存即可。
原生标签和基础组件只能设计出一些静态展示效果,我们可以自定义一些业务组件,给页面加入交互性。
以 table
和 form
为例,简单设计一个用户管理页面。
为 table
设置了自定义属性 actions
、columns
、searchSchema
、modalSchema
{
actions,
columns,
searchSchema,
modalSchema,
}
- actions:定义事件
- columns:表头设计
- searchSchema:搜索表单
- modalSchema:弹窗表单
事件定义可自行定义 action name
,共页面使用,apiName
从我们 API 系统里面选。
预览
可实时进行页面预览,也提供了撤销重做操作。
props 编辑
const editProps = values => {
const tree = editNodes(schemaTree, selectedKey, {props: values}, 'key');
setSchema(tree);
record(clone(tree));
setDisableUndo(false);
};
每次编辑都会触发 schema
更新,并会记录每次操作的数据,使用 record
函数记录,便于我们完成撤销重做功能。
cacheData 函数
const {record, undo, redo, clean} = cacheData();
撤销重做
const undoDesign = () => {
const {index, data} = undo();
setSchema(data);
if (index === 0) {
setDisableUndo(true);
}
setDisableRedo(false);
};
const redoDesign = () => {
const {index, length, data} = redo();
setSchema(data);
if (index === length - 1) {
setDisableRedo(true);
}
setDisableUndo(false);
};
组件移动
提供了组件移动功能,可根据需要自行拖动。
const onDrop = info => {
const fromId = info.dragNode.key;
const toId = info.node.key;
const dropPosition = info.dropPosition;
const tree = moveNodes(schemaTree, fromId, toId, dropPosition, 'key');
setSchema(tree);
};
可点击按钮或链接查看效果。
页面 schema
:
用户管理页面
页面 schema
:
编辑页面
页面 schema
:
低代码更多的是用来当作提升开发效率的一个工具,在我们当前业务范围内,写少量代码封装好业务组件,即可进行可视化开发。
平台的通用性和灵活性,需要我们在实际业务中去权衡。
我们需要认清,没有一劳永逸的方法,只有在不断探索中提升。