Skip to content

necfe/vue-schema

Repository files navigation

VueSchema

VueSchema 是一种可以用 JSON 或 JS Object 来表示 Vue 组件的规范。

特色

  • 将一份 Schema 数据渲染成页面中的组件
  • 根据existhidden控制显隐藏。exist等效于v-if,对组件有创建销毁的特性;hidden等效于v-show,只控制display是否为none
  • 动态绑定属性
  • 使用依赖项控制联动
  • Schema 数据无限合并
  • JSON 能够完整表示 Vue 的相关功能
  • JSON 与 Vue template 的互相转换工具

指南

VueSchema

class VueSchema {
    version: string;            // 当前配置所使用的规范版本
    model: Object;              // 表单中的字段,通常在表单页面中使用。内部的属性会被双向绑定
    uiSchema: UINode;           // UISchema,它表示整个组件的结构,可以取代 Vue template。由 UINode 类型递归表示
    dependencies: Array;        // 组件间的依赖关系
    grayDependencies: Array;    // 使用灰测的依赖关系,与 dependencies 无异
    watchers: Array;            // Vue 的 watchers
}

UINode

class UINode {
    tag: string;                // 节点的 tagName
    parent: UINode;             // 父节点
    children: Array<UINode>;    // 所有子节点
    child: UINode;              // 第一个子节点。为了方便
    text: string;               // 文本,如果没有子节点,则按文本渲染
    others: Object;             // 其它字段,用于保留自定义数据
    bindingAttrs: Object;       // 动态绑定属性
    bindingText: Function;      // 动态绑定文本

    // 以下为 Vue 渲染函数中 data 对象的字段
    // https://cn.vuejs.org/v2/guide/render-function.html#%E6%B7%B1%E5%85%A5-data-%E5%AF%B9%E8%B1%A1
    class: Object;
    style: Object;
    attrs: Object;              // 静态绑定属性
    props: Object;
    domProps: Object;
    on: Object;                 // 事件
    nativeOn: Object;
    directives: Array;
    scopedSlots: Object;
    slot: string;
    key: string;
    ref: string;                // 关联名。根据它 Vue 会在 this.$refs 会引用渲染后的实例,VueSchema 会在 uiRefs 引用此节点
}

正规化

在 new UINode 时,可以使用配置的简化形式,UINode 会自动正规化。如:

{
    tag: 'u-button',
    color: 'primary',
    ':disabled': () => this.canSubmit,
    ':text': () => this.create ? '立即创建' : '设置',
}

会转换为

{
    tag: 'u-button',
    attrs: {
        color: 'primary',
    },
    bindingAttrs: {
        disabled: () => this.canSubmit,
    },
    bindingText: () => this.create ? '立即创建' : '设置'
}

所有未知的静态属性都会移入attrs,所有未知的动态属性都会移入bindingAttrs。因为设置组件时,属性不能出现 UINode 中所有的保留字。

自动猜测子组件

如果没有传tag属性,则会根据父组件的名称猜测子组件。

{
    "tag": "u-radios",
    "children": [
        { "text": "直接创建", "value": "normal" },
        { "text": "从快照创建", "value": "snapshot" }
    ]
}
{
    "tag": "u-select",
    "children": [
        { "text": "面向容器服务", "value": "NCS" },
        { "text": "面向云服务器", "value": "NVM" }
    ]
}

会自动猜测为

{
    "tag": "u-radios",
    "children": [
        { "tag": "u-radio", "text": "直接创建", "value": "normal" },
        { "tag": "u-radio", "text": "从快照创建", "value": "snapshot" }
    ]
}
{
    "tag": "u-select",
    "children": [
        { "tag": "u-select-item", "text": "面向容器服务", "value": "NCS" },
        { "tag": "u-select-item", "text": "面向云服务器", "value": "NVM" }
    ]
}

动态绑定属性

这个功能类似于 Vue Template 中的v-bind:attr,只需属性名前加个冒号,值为一个函数。

{
    ':disabled': () => this.canSubmit
}

v0.4.0 之后,支持字符串表达式,这样在 JSON 中也可以方便地使用。

":disabled": "this.canSubmit"

类似 Vue Template,this 也可以省略。

依赖项控制联动

在配置中设置依赖项,很容易就能实现组件间的联动效果。目前依赖项 Dep 有两种。

depend

depend 字段为 Condition 类型,根据它 resolve 的布尔结果,直接赋值给需要控制 UINode 的属性。

这个经常用于一些布尔属性,比如控制组件的显隐、禁用等等。

{
    "$ref": "snapshot",                 // 需要控制 的 UINode 引用名
    "prop": "exist",                    // 需要控制的属性
    "depend": {                         // 依赖于。Condition 类型
        "model.createType": "snapshot"
    }
}

if

if 字段中的 condition 为 Condition 类型,根据它 resolve 的结果,判断给需要控制 UINode 的属性赋值 then 还是 else

{
    "$ref": "InternetMaxBandwidth", "prop": "label", "if": {
        "condition": { "model.NetworkChargeType": "BANDWIDTH" },
        "then": "公网带宽",
        "else": "最大公网带宽"
    }
}

Condition 类型

Condition 类型用于表示复合型的布尔计算。

  • 当它为一个对象时,结果按照对象中的各值进行“与”运算
  • 当它为一个数组时,结果按照数组中的各值进行“或”运算
  • 当它为一个简单类型时,结果直接将它转换为布尔值
  • 对象中的键是一个 JSON 路径,如:gray.nbsvolumeLabels['cn-east-1a']。上下文支持:schema(当前配置)、gray(灰测对象)、this(Vue 实例)以及this中的所有属性。
  • 如果这个键开头加!,则会对结果进行“非”运算

下面为一个复杂的例子:

[{
    "model.createType": "snapshot"
}, {
    "this.model.azName": "cn-east-1a",
    "!gray.nbsvolumeLabels['cn-east-1a']": 1
}]

Watchers

每个 watcher 结构很简单,为一个数组,它直接表示 Vue 的 $watch 待传入的各个参数。如:

['model.createType', (createType) => {
    if (createType === 'normal')
        this.model.volume = 10;
}],
[() => [this.model.createType, this.model.format], () => {
    this.getSnapshotList();
}],

其实上述的动态绑定属性与依赖项控制联动都会转换为 watcher 的形式。

无限合并

使用schema.merge(schema2),可以合并另一份数据。schema2只需为schema的部分数据即可。

uiSchema 的合并有两种方式。

根据 uiSchema

根据 uiRefs

构建

npm run build

开发

cd demo
vusion dev