写点什么

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

2021 年 4 月 06 日

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 年 4 月 06 日 09:451467

评论

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

Python OpenCV 图片模糊操作 blur 与 medianBlur

梦想橡皮擦

Python OpenCV 4月日更

构建基于Spring Cloud向Service Mesh框架迁移的解决方案及思路

xcbeyond

架构 云原生 Spring Cloud Service Mesh 解决方案

如何只用一个小时定制一个行业AI 模型?

华为云开发者社区

自然语言处理 华为云 hdc ModelArts Pro AI 模型

Vue3教程:Vue 3.x 快在哪里?

十三

Vue Vue3 Vue 3

EGG NETWORK公链阿凡提以“完全开放式自治”的唯一标准搭建New-DeFi链上生态EFTalk

币圈那点事

「开源免费」基于Vue和Quasar的前端SPA项目crudapi后台管理系统实战之序列号自定义组件(四)

crudapi

Vue crud crudapi 序列号 quasar

想快速重构智慧园区5A系统,这份方案推荐给你

华为云开发者社区

系统 智慧园区 HDC.Cloud 2021 端边云协同架构 南向系统

关于Go语言,你不得不知的并发模式!

博文视点Broadview

如何实现屏幕共享时的多人实时标注?

拍乐云Pano

windows Electron RTC

构建从目标到研发过程的全生命周期管理体验

PingCode

敏捷开发 研发管理 Jira Atlassian 敏捷开发管理 研发工具

Linux scp 命令

一个大红包

4月日更

事务隔离级别实战学习

U+2647

事务隔离级别 事务 四月日更

云智一体 AI开发模式打造电力行业的智慧化范本

百度大脑

云计算 智能云

Rust从0到1-结构体-方法

rust 方法 struct 结构体 method

传统金融体系vs新金融体系,区块链改变了什么?

CECBC区块链专委会

金融

“区块链新基建”可否发展可信平台?

电微13828808271

区块链+ 区块链新基建

并发的HashMap为什么会引起死循环?

Java小咖秀

hashmap 并发 并发容器

文字识别:关键信息提取的3种探索方法

华为云开发者社区

深度学习 文字识别 图结构 关键信息提取 栅格

释放千行百业数据价值,华为云DAYU有一套

华为云开发者社区

大数据 数据湖 华为云 数据价值 dayu

maven中心仓库OSSRH使用简介

程序那些事

Java maven 程序那些事

世界五百强第一的沃尔玛在用区块链做什么

CECBC区块链专委会

区块链

区块链BaaS服务平台开发,助推中小企业快速落地

13828808769

区块链+ #区块链#

“区块链+”司法合约,电子认证不造假

电微13828808271

区块链+

科技“智”造:智慧工厂这样规划,既高效又节能

一只数据鲸鱼

数据可视化 工业物联网 智慧园区 智慧工厂

公安合成作战平台建设解决方案,智慧警务情报研判系统开发

WX13823153201

国人开源的运维监控系统——WGCLOUD

王逅逅

运维 Grafana Prometheus zabbix 开源项目

在多人音视频聊天中插入现场直播

anyRTC开发者

android 前端 音视频 WebRTC RTC

强化区块链应用 破解知识产权运营难题

CECBC区块链专委会

区块链

从金融到物联网 区块链的落地应用将如何改变世界?

CECBC区块链专委会

区块链

区块链给普通人带来的机会!

CECBC区块链专委会

区块链

SumSwap与金色财经共为 首席创新合作大会在上海拉开帷幕

币圈资讯

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