• 低代码平台前端的设计与实现(一)渲染模块的基本实现


    这两年低代码平台的话题愈来愈火,一眼望去全是关于低代码开发的概念,鲜有关于低代码平台的设计实现。本文将以实际的代码入手,逐步介绍如何打造一款低开的平台。

    低开概念我们不再赘述,但对于低开的前端来说,至少要有以下3个要素:

    1. 使用能被更多用户(甚至不是开发人员)容易接受的DSL(领域特定语言),用以描述页面结构以及相关UI上下文。
    2. 内部具有构建引擎,能够将DSL JSON构建为React组件树,交给React进行渲染。
    3. 提供设计器(Designer)支持以拖拉拽方式来快速处理DSL,方便用户快速完成页面设计。

    本文我们首先着眼于如何进行构建,后面的文章我们再详细介绍设计器的实现思路。

    DSL

    对于页面UI来说,我们总是可以将界面通过树状结构进行描述:

    1. 页面
        1-1. 标题
           1-1-1. 文字
        1-2. 内容面板
           1-2-1. 一个输入框
    

    如果采用xml来描述,可以是如下的形式:

    <page>
        <title>标题文字title>
        <content>
            <input>input>
        content>
    page>
    

    当然,xml作为DSL有以下的两个问题:

    1. 内容存在较大的信息冗余(page标签、title标签,都有重复的字符)。
    2. 前端需要引入单独处理xml的库

    自然,我们很容易想到另一个数据描述方案:JSON。使用JSON来描述上述的页面,我们可以如下设计:

    {
        "type": "page",
        "children": [
            {
                "type": "title",
                "props": {
                    "value": "标题文字"
                }
            },
            {
                "type": "content",
                "children": [
                    {
                        "type": "input"
                    }
                ]
            }
        ]
    }
    

    初看JSON可能觉得内容比起xml更多,但是在前端我们拥有原生处理JSON的能力,这一点就很体现优势。

    回顾一下JSON的方案,我们首先定义一个基本的数据结构:组件节点(ComponentNode),它至少有如下的内容:

    1. componentName属性:表明当前组件节点的名称。
    2. children属性:一个ComponentNode数组,存放所有的子节点。
    3. props:该元素的属性列表,可以应用到当前的组件节点,产生作用。

    例如,对于一个页面(page),该页面有一个属性配置背景色(backgroundColor),该页面中有一个按钮(button),并且该按钮有一个属性配置按钮的尺寸(size),此外还有一个输入框(input)。

    {
        "componentName": "page",
        "props": {
            "backgroundColor": "pink", // page的 backgroundColor 配置
        },
        "children": [
            {
                "componentName": "button",
                "props": {
                    "size": "default" // button的size配置
                }
            },
            {
                "componentName": "input"
            }
        ]
    }
    

    同时,我们需要设计一下组件节点属性props这个字段。考虑到DSL中的props最终将会送入到对应React组件的props,我们有必要进行一定的设计与处理来保证React接收到的正确性。首先,我们先假设,props里面的每一个prop属性对应的值目前只支持string、number字面量(后续我们会设计表达式或者事件等,这里先简单设计)。也就是说,props的类型定义为:

    /**
     * 组件节点每一个属性的类型
     */
    export type ComponentNodePropType = string | number;
    
    export interface ComponentNode {
      // ... ...
      props: {
        [propName: string]: ComponentNodePropType;
      }
      // ... ...
    }
    

    在我们的平台中,我们定义如下的结构:

    /**
     * 组件节点每一个属性的类型
     */
    export type ComponentNodePropType = string | number;
    
    /**
     * 组件节点
     */
    export type ComponentNode = {
        /**
         * 组件节点唯一名称
         */
        componentName: string;
        /**
         * 组件各种属性集合
         */
        props: {
            [propName: string]: ComponentNodePropType;
        };
        /**
         * 组件节点子节点
         */
        children?: Array<ComponentNode>;
    }
    

    构建

    上文讨论了我们低开平台的DSL中关于组件节点的定义,但是DSL组件节点数据如果没有转换构建为UI组件并渲染在界面上,是没有任何意义的。我们必须要有构建引擎支持将JSON转换为web页面的内容。接下来我们将继续分析讨论如何完成ComponentNode到UI的转换处理。

    组件构造映射表

    首先,我们会有一个容器,来专门存放componentName与对应组件的构造方法(类组件、函数组件,甚至是一般的html组件字符串),就像如下的一个表:

    import {Button, Input} from "antd";
    import React from "react";
    
    
    /**
     * lite-lc内置的文本字面量节点,支持string、number
     */
    const Text = ({value}: { value: string | number }) => {
        return <>{value};
    }
    
    export const COMPONENT_MAP = {
        'page': 'div', // page直接使用div
        'button': Button,
        'input': Input,
        'text': Text
    }
    
    

    当然,平台还设计了一个内置默认的组件名为"text"的文本节点。主要用于某些组件的子节点直接是一个文本内容的场景来进行映射:

    {
      "componentName": "button",
      "children": [{
        "componentName": "text",
        "props": {
          "value": "hello, button"
        }
      }]
    }
    

    构建引擎(BuildEngine)

    接下来是实现我们的构建引擎(BuildEngine,叫引擎高大上)。构建引擎的核心功能是读取由DSL转为的ComponentNode,然后以递归深度遍历的方式不断读取ComponentNode及其子节点,根据ComponentNode对应的数据(譬如)componentName,从前面我们编写的COMPONENT_MAP中获取对应组件构造方法来将ComponentNode构建为一个又一个ReactNode。

    010-ComponentNode-build-flow

    代码如下:

    import {ComponentNode} from "../meta/ComponentNode";
    import {COMPONENT_MAP} from "../component-map/ComponentMap";
    import React from "react";
    
    export class BuildEngine {
    
        /**
         * 构建:通过传入 ComponentNode 信息,得到该节点对应供React渲染的ReactNode
         * @param componentNode
         */
        build(componentNode: ComponentNode) {
            return this.innerBuild(componentNode);
        }
    
        /**
         * 构建:通过传入 ComponentNode 信息,得到该节点对应供React渲染的ReactNode
         * @param componentNode
         */
        private innerBuild(componentNode: ComponentNode) {
    
            if (!componentNode) {
                return undefined;
            }
    
            const {componentName, children, props} = componentNode;
    
            // 如果有子元素,则递归调用自身,获取子元素处理后的ReactNode
            const childrenReactNode =
                (children || []).map((childNode) => {
                    return this.innerBuild(childNode);
                });
    
            // 通过 COMPONENT_MAP 来查找对应组件的构造器
            const componentConstructor = COMPONENT_MAP[componentName];
    
            return React.createElement(
                componentConstructor,
                {...props},
                childrenReactNode.length > 0 ? childrenReactNode : undefined
            )
        }
    }
    

    需要注意,这个Engine的公共API是build,由外部调用,仅需要传入根节点ComponentNode即可得到整个节点数的UI组件树(ReactNode)。为了后续我们优化内部的API结构,我们内部使用innerBuild作为内部处理的实际方法。

    效果展示

    建立一个样例项目,编写一个简单的样例:

    import {BuildEngine} from "@lite-lc/core";
    import {ChangeEvent, useState} from "react";
    import {Input} from 'antd';
    
    export function SimpleExample() {
    
        // 使用构建引擎
        const [buildEngine] = useState(new BuildEngine());
    
        // 使用state存储一个schema的字符串
        const [componentNodeJson, setComponentNodeJson] = useState(JSON.stringify({
            "componentName": "page",
            "children": [
                {
                    "componentName": "button",
                    "props": {
                        "size": "small",
                        "type": "primary"
                    },
                    "children": [
                        {
                            "componentName": "text",
                            "props": {
                                "value": "hello, my button."
                            }
                        }
                    ]
                },
                {
                    "componentName": "input"
                }
            ]
        }, null, 2))
    
        let reactNode;
        try {
            const eleNode = JSON.parse(componentNodeJson);
            reactNode = buildEngine.build(eleNode);
        } catch (e) {
            // 序列化出异常,返回JSON格式出错
            reactNode = <div>JSON格式出错div>
        }
    
        return (
            <div style={{width: '100%', height: '100%', padding: '10px'}}>
                <div style={{width: '100%', height: 'calc(50%)'}}>
                    <Input.TextArea
                        rows={4}
                        value={componentNodeJson}
                        onChange={(e: ChangeEvent<HTMLTextAreaElement>) => {
                            const value = e.target.value;
                            // 编辑框发生修改,重新设置JSON
                            setComponentNodeJson(value);
                        }}/>
                div>
                <div style={{width: '100%', height: 'calc(50%)', border: '1px solid gray'}}>
                    {reactNode}
                div>
            div>
        );
    }
    

    020-base-effect

    设计优化

    路径设计

    目前为止,我们已经设计了一个简单的构建引擎。但是还有两个需要解决的问题:

    1. 循环创建的ReactNode数组没有添加key,会导致React渲染性能问题。
    2. 构建的过程中,无法定位当前ComponentNode的所在位置。

    我们先讨论问题2。对于该问题具体是指:我们希望能够记录每一个节点在整个树状的定位。

    {
        "componentName": "page",
        "children": [
            {
                "componentName": "panel",
                "children": [
                    {    
                        "componentName": "input"
                    },
                    {
                        "componentName": "button",
                    }
                ]
            },
            {    
                "componentName": "input"
            }
        ]
    }
    

    对于上述的每一个type,都应当有其标志其唯一的一个key。可以知道,每一个元素的路径是唯一的:

    • page:/page
    • panel:/page/panel@0
    • 第一个input:/page/panel@0/input@0。page下面有个panel(面板)元素,位于page的子节点第0号位置(基于0作为起始)。panel下面有个input元素,位于panel的子节点第0号位置。
    • button:/page/panel@0/button@1
    • 第二个input:/page/input@1

    也就是说,路径由'/'拼接,每一级路径由'@'分割组件名称componentName和index,index表明该节点处于上一级节点(也就是父级节点)的children数组的位置索引(基于0起始)。

    那么,如何生成这样一个路径信息呢?只需要在build的遍历ComponentNode过程中记录即可,基于之前构建引擎的innerBuild的递归调用,现在只需要进行简单的修改方法:

    // BuildEngine.ts代码
    -    private innerBuild(componentNode: ComponentNode): ReactNode | undefined {
    +    private innerBuild(componentNode: ComponentNode, path: string): ReactNode | undefined {
             if (!componentNode) {
                 return undefined;
             }
    				 // ... ...
             // 递归调用自身,获取子元素处理后的ReactNode
             const childrenReactNode =
    -            (children || []).map((childNode) => {
    -               return this.innerBuild(childNode);
    -            });
    +            (children || []).map((childNode, index) => {
    +                // 子元素路径:
    +                // 父级路径(也就是当前path)+ '/' + 子元素名称 + '@' + 子元素所在索引
    +                const childPath = `${path}/${childNode.componentName}@${index}`;
    +                return this.innerBuild(childNode, childPath);
    +            });
    

    首先,我们修改了innerBuild方法入参,增加了参数path,用以表示当前节点所在的路径;其次,在生成子元素调用innertBuild的地方,将path作为基准,根据上述规则"${componentName}@${index}",来生成子元素节点的路径,并传入到的递归调用的innerBuild中。

    当然,build内部调用innerBuild的时候,需要构造一个起始节点的path,传入innerBuild。

    // BuildEngine.ts代码
        build(componentNode: ComponentNode) {
    -       return this.innerBuild(componentNode);
    +       // 起始节点,需要构造一个起始path传入innerBuild
    +       // 根节点由于不属于某一个父级的子元素,所以不存在'@${index}'
    +       return this.innerBuild(componentNode, '/' + componentNode.componentName);
        }
    

    再回到innerBuild关于使用React.createElement的部分,考虑到现在已经有了path作为每一个组件唯一的路径标识。我们可以将该path作为每一个组件的key,让React创建元素的时候,将这个path作为key添加到组件实例上,进而解决Warning: Each child in a list should have a unique "key" prop.组件为一个key属性问题。相关改动代码如下:

    // innerBuild中最后的返回ReactNode部分
            return React.createElement(
                componentConstructor,
    -           {...props},
    +           {...props, key: path}, // 将path作为key
                childrenReactNode.length > 0 ? childrenReactNode : undefined
            )
    

    关于构建的总结

    目前为止,我们设计了一套十分精简的根据DSL组件节点树转换为ReactNode的构建引擎,内部基于antd5组件的组件构建ReactNode,通过接收JSON遍历节点构建出ReactNode,再交给React渲染出对应结构的页面。该构建引擎需要考虑,React渲染时候元素的时候,需要一个唯一key来表示对应组件。本系列,我们由浅入深逐步建立整个低代码平台。下篇文章,笔者将开始介绍设计器Designer的实现。

    附录

    本章内容对应代码已经推送到github上

    w4ngzhen/lite-lc (github.com)

    main分支与最新文章同步,对应章节将会有对应的tag来标识。

    且按照文章里各段介绍顺序完成了提交:

    modify: BuildEngine递归增加path标识组件唯一性,并作为key交给react创建ReactNode。
    add: 新增BuildEngine并导出相关类型;修改样例代码,验证BuildEngine流程。
    add: 新增组件名称与组件构造器映射的数据容器,用于构建过程中根据对应组件名称构造对应的组件实例。
    add: ComponentNode 映射 JSON DSL
    init: 项目初始化,添加core and example 基础文件(使用antd5)。
    
  • 相关阅读:
    log4j2---基于vulhub的log4j2漏洞复现---反弹shell
    QT基础 柱状图
    element表单非必填数据的验证写法(自定义验证规则)
    【AICFD案例教程】汽车外气动-AI加速
    Java.lang.Class类 getTypeParameters()方法有什么功能呢?
    docker打包x86镜像包的详细步骤
    【Python】常用距离计算方法
    【RT-Thread】nxp rt10xx 设备驱动框架之--Pin搭建和使用
    浏览器——Microsoft Edge
    Hive窗口函数回顾
  • 原文地址:https://www.cnblogs.com/w4ngzhen/p/16705454.html