把握行业变革关键节点,12 月 19 日 - 20 日,AICon北京站即将重磅启幕! 了解详情
写点什么

ZooTeam 拍了拍你,来看看如何设计动态化表单

  • 2021-04-06
  • 本文字数:3821 字

    阅读完需:约 13 分钟

ZooTeam 拍了拍你,来看看如何设计动态化表单

前言


对于 ToB 业务而言,随着业务的不断壮大,接入的客户逐渐增加,相同页面的差异化的需求越来越多,尤其是在表单层面,小到多一个字段少一个字段这种简单的需求,大到整个页面不变的只剩下一些基础字段。


一旦这种差异化需求随着业务量的增长而膨胀起来。代码中的 IF ELSE 越来越多,项目就越来越难以维护。基于这个问题,比较普遍的解决方案要么是项目拆分,要么相同项目的代码分割。


这两种方案都有维护成本比较大的弊端,那么有没有更好点的解决方案呢。本文就带你了解一下动态化表单搭建。


什么是动态表单


先下个定义,动态表单是页面根据管理端配置的不同的 Schema 结构,动态渲染出不同的表单项的表单。


动态表单一般分两个部分,管理端和渲染端。


管理端配置表单项及相应的简单交互产出 Schema 数据。


渲染端根据 Schema 数据相应的渲染出不同的表单项并实现简单的交互。大致流程如下。



动态表单的实现


表单配置


对于 Schema 数据的配置,考虑到接入业务方的接入成本及维护成本。


管理端采用了可拖拽式的所见即所得配置面板。这里共分为四个部分,备选组件面板,拖拽面板,组件属性面板和表单属性配置(视图属性)。


具体实现如下图:



备选组件面板


左侧备选组件栏里的备选组件共分三种,容器组件,基础组件,自定义组件。


  • 容器组件——是用来存放基础组件的组件。例如表单组件,子表单组件,表格表单组件。这些组件都是内部可以存放其他子组件的组件。属于容器组件。系统内置了 3 个容器组件,对于中台系统而言,容器组件不会太多样化。所以容器组件没有做自定义组件的功能。当然后期如果需要的话也可以扩展出容器组件的自定义组件功能

  • 基础组件——就是普通的表单项。系统内置了 13 个基础组件。这些组件都是基于 Antd 的 React 组件做的二次包装

  • 自定义组件——业务方经常出现一些个性化的表单项,例如某个输入框后面加个链接跳转之类的需求,对于这种组件,系统提供了自定义组件的注册及配置功能。这样业务方就可以自由的维护专属于自己的业务表单组件了。同时也解放出表单系统维护者的时间。组件注册同时也是一个更加轻量级的自定义组件模块。因为不同的组件需要设置不同的参数,所以该组件对应的右侧属性表单也应该是不同。这部分下文组件属性部分会详细说明


拖拽面板


拖拽面板就是维护组件展示关系的面板,同时提供拖拽排序、删除、复制、预览等功能。

具体实现方案采用的是 React-DnD。


组件配置


属性配置面板本身就是个更加轻量级的动态表单实现。


只是 Schema 由开发者直接写死而没有一个可配置的页面而已(自定义组件注册部分例外)。


当在拖拽面板选中一个组件时,组件属性配置面板会渲染出相应组件的可配置项表单, 这里提供一下简单的组件属性配置面板的 Schema 供大家参考。


[{  label: '是否可见',  code: 'visible',  widget: 'switchBtn',  initialValue: true},{  label: '是否可编辑', // 标签文案  code: 'code', // 字段编码  widget: 'switchBtn', // 组件类型  initialValue: true, // 初始值 默认可编辑  hide: 'exp: visible === false', // 是否隐藏  required: true // 是否必填}]
复制代码


细心的同学会发现 hide 字段写了个表达式。这里通过 exp: 开头作为一个表达式的标识。表达式的可以使用的变量是属性表单内的值。


比如上面这个例子,visible 是上面定义了一个是否可见的字段。如果当前选中的这个组件不可见的话,是否可编辑本身就无从谈起,所以直接隐藏掉。


容器属性 共有的属性有标题、编码、是否可见、以及容器结构是否对数据透明。


前面三个好理解。容器结构是否对数据透明是什么呢?


前面说过,我们的容器组件是可多层嵌套的,那问题来了,数据咋办,表单嵌套会导致数据也跟着嵌套。所以这里参考了阿里的 Formily 开源表单方案。使用一个 skip ,来使其对数据透明。即:


{  "title": "表单",  "type": "form",  "fields": [{    "name": "name",    "label": "姓名"  }, {    "title": "子表单",    "skip": true, // 表单结构对数据透明    "name": "item",    "type": "form",    "fields": [      {        "name": "object",        "label": "物品"      }, {        "name": "brand",        "label": "品牌"      }    ]  }]}
复制代码


skip 为 false 时返回的数据为:

{  "name": "简名",  "item": {    "object": "电脑",    "brand": "Mac"  }}
复制代码


skip 为 true 时返回的数据为:

{  "name": "简名",  "object": "电脑",  "brand": "Mac"}
复制代码


组件属性 分为基本属性和组件属性,基本属性是所有属性共有的。标题,编码,是否可见,是否必填等属性都是基本属性。组件属性则是组件私有的属性。


比如 Select 组件会需要一个数据来源,以及该组件是否多选之类的。基本属性直接写死。组件私有属性则通过远程数据库维护。自定义组件的注册就需要涉及到这部分的数据管理。


自定义组件的注册表单如下:



其中组件可配置属性就是组件私有的属性的定义,注册时定义,配置该组件时赋值,渲染端渲染时应用。可配置属性还需要支持表达式的填写。


比如某个组件需要远程数据,url 提供了,但是参数需要取当前时间,这个时候就需要组件属性支持表达式的解析或者少量代码读写运行了。


这些属性除了组件自定义属性以外,还有组件默认值,组件自定义校验,组件 onChange 事件。

以自定义校验为例:



表单属性配置(视图属性)


这部分在上图中没有显示,是在组件属性右侧。表单属性分两部分,交互规则和接口绑定。



交互规则 表单交互规则在表单级别绑定,而不是在字段级别。进行就近配置的目的,是为了方便管理,进入一个表单配置,该表单的交互在右侧一目了然。


接口绑定 则是表单渲染过程中有可能涉及到一些远程数据的读取,比如默认值等。这部分数据的配置需要用到远程数据。表单上绑定了接口之后,表单初始化之前先发请求获取绑定接口的数据,相应的表单组件里就可以使用到该数据进行初始化。


管理端数据流转


管理端的功能是构建出一个目标 Schema。


每个备选组件都有基本信息和相应的组件可配置属性信息。


组件基本信息主要用于组件面板展现。组件可配置属性需要在右侧属性配置时渲染成一个表单给使用者去配置,故而组件可配置信息又是一个简化版的 Schema,这里称为组件级 Schema。


在拖拽页面中添加一个组件,通过解析组件的组件级 Schema 及组件放置位置给目标 Schema 添加一个组件数据。


然后在拖拽页面中选中该组件,右侧属性配置会相应渲染出组件级 Schema 所描述的表单给用户配置填写。用户配置时直接修改目标 Schema 中相应选中组件的信息。


数据流转图大致如下:


表单动态渲染


因为表单页面还会有各种定制化的需求,表单渲染端这里采用组件的形式,提供了两个组件,一个组件作为表单页面的外层包裹组件主要功能是发请求获取相应的 schema.json 数据。


另一个组件就是通过上层组件的数据渲染相应的表单。示例:


import { FormPageWrap, MainForm } from './index';@FormPageWrap({  prefix: '/api/budget', // 业务方接口前缀  getFormParams: (props) => { // 获取表单结构参数    return {};  },  getDataParams: (props) => { // 获取表单回填数据参数    return {};  }})export default class FormPage extends Component {  render() {    return (      <div>        // 表单各种额外显示内容        <MainForm // 表单渲染组件          customComponent={{            test: TestComp,          }}          extraParams={{}}        />        // 表单各种额外显示内容      </div>    );  }}
复制代码


内部实现则是根据 Schema 渲染相应的组件。


待完善


目前系统部分功能还有待完善。具体有几点:


  • 自定义组件的异步加载。当一个表单需要新增加一个自定义组件时,项目需要重新构建发版。如果自定义组件可以单独发布,就可以做到及时添加一个自定义组件,不需要项目重新构建发布了。当然如果自定义组件太多,异步加载还是会有些性能问题。而这就需要做到同页面下多组件代码合并了

  • 一些配置的沉淀复用。比如某些经常配置的表单块。可以沉淀为常用组件。直接选择使用,可进一步简化配置流程

  • 同页面下的一些相同区块,如果每个页面都单独维护,会极大的增加维护成本、抽取并联动,可以极大的减少维护表单的成本


展望


对于动态化表单的能力远不止目前看到的动态表单搭建:


  • 对于管理端流转出来的 Schema 数据可以进行二次加工,从而实现对于用户的权限,业务配置等能力的扩展

  • 管理端配置出来的 Schema 不止可以用在动态表单渲染中,还可以作为数据模型去描述一个静态数据的结构。从而提供各业务系统配置数据的构建能力

  • 前端渲染组件也不一定要和管理端的 Schema 完全耦合在一起。单独拿来使用也是完全没问题的,这样对于某些简化版动态表单的能力也能做到支持



头图:Unsplash

作者:简名

原文:https://mp.weixin.qq.com/s/zvegeJAOUW6tdlLUM_-Xzw

原文:ZooTeam 拍了拍你,来看看如何设计动态化表单

来源:政采云前端团队 - 微信公众号 [ID:Zoo-Team]

转载:著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

2021-04-06 09:454001

评论

发布
暂无评论
发现更多内容

六一儿童节,看我用ModelArts让8090梦回童年

华为云开发者联盟

AI 美食 童年 modelarts 六一

工业制造业在数字化时代的三大发展方向

CECBC

智慧工厂VR拆解零件——3D虚实现实可视化系统

一只数据鲸鱼

数据可视化 工业互联网 vr 智慧工厂 零件拆解

实现接口幂等性的四种方案!

李阿柯

面试 编程之路 幂等性

Python3 Note 函数注解

awen

Python Function 函数注解

因为一个字符校对问题,我的大厂面试挂了

华为云开发者联盟

MySQL 字符 字符校对 语句 MySQL5.7

从VMWare安装到Nginx配置

wildpig

nginx vmware Centos 7

GitHub上收录400余篇任正非的讲话稿

不脱发的程序猿

GitHub 开源 程序人生 任正非讲话

计算机视觉常用图像数据集标记平台

不脱发的程序猿

人工智能 计算机视觉 图像处理 图像数据集标记平台

六一限定,致每一个追光者

白洞计划

Serverless over Storage

焱融科技

云计算 容器 云原生 高性能 文件存储

BoCloud博云微服务平台3.0正式发布:让微服务转型路径更清晰

BoCloud博云

微服务

Android studio 在外置硬盘运行项目报.lock 的错误

三爻

flutter android Mac Android Studio

Python3 Note 对象初始化

awen

Python 生命周期 对象初始化

Python3 Note __slots__

awen

Python slots

人生算法:涌现,在自己身上发挥群体智慧

石云升

读书笔记 6月日更

苏州源控电子科技怎么样?名副其实的行业新星

Geek_8a195c

数字人民币有望为全球贸易结算开辟新视窗

CECBC

【LeetCode】包含min函数的栈Java题解

Albert

算法 LeetCode 6月日更

☕️【Java技术之旅】深入学习JIT编译器实现机制(原理篇)

码界西柚

Java 编译器 JIT 6月日更

mPaaS 月度小报 | 应用上线前都应该检查哪些指标?CodeHub#5回顾:小程序容器加持下的技术架构“提质增效”

蚂蚁集团移动开发平台 mPaaS

小程序 移动开发 mPaaS

用敏捷扑克做需求评审的3大优势,你get了吗?

LigaAI

高效工作 团队管理 产品思考

安迈云首席战略官于晓晖:去中心化云计算构建Web3.0世界

DT极客

一文带你认识队列数据结构

华为云开发者联盟

Java 数据结构 数组 队列

NUCLEO-L432KC实现GPIO控制(STM32L432KC)

不脱发的程序猿

嵌入式 stm32 单片机 NUCLEO-L432KC STM32L432KC

全球案例 | 一家财富500强公司利用 Jira 和 Jira Align 将万人级团队的生产力提高了 30%

Atlassian

管理 DevOps 敏捷 Jira 协同办公

模块五总结

竹林七贤

个推CTO谈数据中台(上):从要求、方法论到应用实践

个推

大数据 数据中台 数字化转型 数据智能

智慧光伏能源-园区光伏发电能源管控可视化

一只数据鲸鱼

数据可视化 智慧园区 智慧能源 能源管理 光伏发电

“图发展”与“保安全”:大数据今后怎么玩?

CECBC

净筹6亿美元:微盟正在加速拉开差距

ToB行业头条

SaaS 微盟

ZooTeam 拍了拍你,来看看如何设计动态化表单_文化 & 方法_政采云前端团队_InfoQ精选文章