几何图形计算器
40.01M · 2026-03-05
封装一个可以扩展的表单生成器,尤其是对一些中后台系统、低代码平台、企业级项目,会非常有价值。
我们先看一个普通的表单:
<el-form :model="form">
<el-form-item label="用户名">
<el-input v-model="form.username" />
</el-form-item>
<el-form-item label="年龄">
<el-input-number v-model="form.age" />
</el-form-item>
</el-form>
问题在哪?
这时候我们会思考一个问题:
理想状态是这样:
const schema = [
{
type: 'input',
label: '用户名',
field: 'username',
props: {
placeholder: '请输入用户名'
}
},
{
type: 'number',
label: '年龄',
field: 'age',
}
]
然后我们写一个 <SchemaForm />:
<SchemaForm :schema="schema" v-model="formData" />
这就是:
一个成熟的表单生成系统,至少包含 5 个层次:
Schema 配置层
↓
字段解析层
↓
组件映射层
↓
状态管理层
↓
渲染引擎层
我们逐层拆解。
最重要的一步:
const componentMap = {
input: ElInput,
number: ElInputNumber,
select: ElSelect,
}
渲染逻辑:
const Component = componentMap[item.type]
return h(Component, {
...item.props,
modelValue: formData[item.field],
'onUpdate:modelValue': (val) => {
formData[item.field] = val
}
})
这样我们就实现了:
一个“玩具级”表单生成器和一个“工程级”的区别在于:
必须支持:
{
type: 'input',
field: 'company',
visible: (form) => form.role === 'admin'
}
解析时:
if (typeof item.visible === 'function') {
return item.visible(formData)
}
{
type: 'select',
field: 'province',
onChange: (val, form) => {
form.city = ''
}
}
{
type: 'select',
field: 'user',
asyncOptions: () => fetchUserList()
}
<SchemaForm>
<template #customField="{ field }">
<MyCustomComponent />
</template>
</SchemaForm>
很多人会这样写:
const formData = reactive({})
但在复杂场景中:
你需要抽象出一个 FormStore
class FormStore {
values = reactive({})
errors = reactive({})
touched = reactive({})
setFieldValue(field, value) {
this.values[field] = value
}
validate() {}
}
真正成熟的系统会支持:
| 能力 | 说明 |
|---|---|
| 嵌套表单 | object / array 结构 |
| 动态增删字段 | 表单列表 |
| 表单分组 | step 表单 |
| 表单布局系统 | grid / col 配置 |
| 表单 JSON 导出 | 支持保存配置 |
| 拖拽编辑器 | 低代码场景 |
| 远程 schema | 后端下发表单配置 |
原因很简单:
Ant Design Pro、阿里飞冰、字节内部平台,核心都在做这件事。
不是为了“少写代码”。
而是: