結合 UI 框架實現可配置 Vue 表單組件淺析

—— 封面攝於濟州島民宿vue

常規的表單

若是咱們用 UI 框架作管理系統時候,關於表單的代碼咱們不會陌生,大體是這樣的,好比這是一個 iView 框架下的綜合性表單:vuex

<template>
    <Form :model="formItem" :label-width="80">
        <FormItem label="Input">
            <Input v-model="formItem.input" placeholder="Enter something..."></Input>
        </FormItem>
        <FormItem label="Select">
            <Select v-model="formItem.select">
                <Option value="beijing">New York</Option>
                <Option value="shanghai">London</Option>
                <Option value="shenzhen">Sydney</Option>
            </Select>
        </FormItem>
        <FormItem label="Radio">
            <RadioGroup v-model="formItem.radio">
                <Radio label="male">Male</Radio>
                <Radio label="female">Female</Radio>
            </RadioGroup>
        </FormItem>
    </Form>
</template>
<script>
    export default {
        data () {
            return {
                formItem: {
                    input: '',
                    select: '',
                    radio: 'male'
                }
            }
        }
    }
</script>
複製代碼

配置化表單

而我想要的方式是這樣的:api

模板

<template slot="modalContent">
    <AutoForm
        :fileds="projectFields"
        :model="projectFormData"
        :formName="projectFormData"
        class="my-form"
    />
</template>
<script>
// @ is an alias to /src
import { mapState } from 'vuex'
import { projectFields } from '@/utils/fieldsMap'
export default {
    data ()  {
        return {
            // 表單配置列表
            projectFields: projectFields
        }
    },

    computed: {
        ...mapState({
          // 項目列表頁編輯表單
          projectFormData: state => state.project.projectFormData
        })
    },
}
</script>
複製代碼

表單項的數據來源我會利用 Vuex 的 state 裏管理:bash

數據

import { projectFormData } from '@/api/project'

const state = {
      // 項目列表頁編輯
      projectFormData: {
        projectInput: '',
        projectSelect: '',
        projectRadio: ''
      }
}

// getters
const getters = { }

// action
const actions = {
    // 表單項數據獲取
    // 表單項數據提交
}

// mutations
const mutations = { }

export default {
  namespaced: true,
  state,
  getters,
  actions,
  mutations
}
複製代碼

表單項的配置也是經過單文件(fieldsMap.js)管理,方便維護:框架

表單項配置

// 表單配置項
// 注意:tag 和 type 須要根據使用的 UI 框架來匹配。
const projectFields = {
    projectInput: {
        label: '項目Input',
        tag: 'Input',
        type: 'text',
        placeholder: '請輸入項目Input'
    },
    projectSelect: {
        label: '項目下拉Select',
        tag: 'Select',
        options: [
          {
            key: 'beijing',
            value: 'beijing'
          },
          {
            key: 'hangzhou',
            value: 'hangzhou'
          }
        ]
    },
    projectRadio: {
        label: '項目Radio',
        tag: 'RadioGroup',
        options: [
          {
            label: '是'
          },
          {
            label: '否'
          }
        ]
    },
}
複製代碼

OK,整個一個配置表單的文件結構,使用方式就是這樣子,總結一下大體是三部曲:函數

  • 引入 <AutoForm /> 組件。
  • fieldsMap.js 中配置表單項,包括 label、type、tag、options等。
  • Vuex state 中添加數據來源。

剩下的關鍵是 <AutoForm /> 組件是如何實現配置化,其實本質是動態生成表單項(根據配置文件)的過程,對於 iView 來講,就是動態的生成 FormItem,來拼成一個完整的表單。這時咱們就須要用到 vue 提供的 render Api了。ui

首先查看一下官方文檔 render 截圖:this

render

三個參數的簡單用法:spa

<script>
    Vue.component('Line', {
        render: function(h) {
            h('div', {
                props: {} // 傳遞數據
            },'文本 or 子節點')
        }
    })
</script>
複製代碼

瞭解基礎用法後,咱們來看下 <AutoForm /> 組件的實現:雙向綁定

在上代碼以前,咱們先看一下 iView 表單的結構,從外層到內層,Form 容器固定——FormItem 數量動態——Input 類型動態,組件最終是返回一個 Form;根據配置項的數量來決定 FormItem 的數量,動態建立;根據配置項的 tagtype 來決定表單的類型;固然有些例如 Select 的表單項會有 options 下拉選項,也須要單獨生成。

根據上面的分析,那總結關於這個 <AutoForm /> 組件,大體有 FormRenderitemsRendercomponentUse、類型(InputRenderRadioRenderSelectRender)、options (optionsRender) 五個點。

AutoForm.vue

<script>
export default {
  name: 'Form',
  functional: true,
  render (h, context) {
    let fileds = context.props.fileds // 表單配置 from fieldsMap.js
    let model = context.props.model // 表單數據 from state
    let formName = context.props.formName // 表單名稱惟一
    /**
     * 渲染 FormItem
     */
    function itemsRender () {
      let res = []
      // 遍歷配置項動態生成 FormItem
      Object.keys(fileds).forEach((ele, i) => {
        res.push(
          h('FormItem',
            {
              props: {
                label: fileds[ele].label // FormItem label 屬性
              }
            },
            componentUse(fileds[ele], ele) // 子節點表單類型,利用 componentUse 函數控制
          )
        )
      })

      return res
    }

    /**
     * 表單分發選擇
     * @param { Object } _item - 當前 fields 配置項
     * @param { String } _model - 當前配置項名
     */
    function componentUse (_item, _model) {
      let typeMap = {
        'Input': InputRender,
        'RadioGroup': RadioRender,
        'Select': SelectRender
      }
      let component = typeMap[_item.tag](_item, _model)

      return [component]
    }

    // Input
    function InputRender (_item, _model) {
      return h('Input',
        {
          props: {
            'v-model': `${formName}.${_model}`,
            'placeholder': _item.placeholder,
            'type': _item.type
          },
          on: {
            // iView 組件提供的方法,實現數據雙向綁定
            'on-blur': (e) => {
              model[_model] = e.target.value
            }
          }
        }
      )
    }

    // Radio
    function RadioRender (_item, _model) {
      return h('RadioGroup',
        {
          props: {
            'v-model': `${formName}.${_model}`
          },
          on: {
            'on-change': (e) => {
              model[_model] = e === '是' ? 1 : 0
            }
          }
        },
        _item.options ? optionsRender(_item.options, 'Radio') : []
      )
    }

    // Select
    function SelectRender (_item, _model) {
      return h('Select',
        {
          props: {
            'v-model': `${formName}.${_model}`
          },
          on: {
            'on-change': (e) => {
              model[_model] = e
            }
          }
        },
        _item.options ? optionsRender(_item.options, 'Option') : []
      )
    }

    // 有多選 options 配置 optionsRender
    // Radio
    // Select
    function optionsRender (_options, _tag) {
      let itemRes = []
      _options.forEach((_option, i) => {
        if (_tag === 'Radio') {
          itemRes.push(
            h(_tag,
              {
                props: {
                  'label': _option.label
                }
              }
            )
          )
        } else if (_tag === 'Option') {
          itemRes.push(
            h(_tag,
              {
                props: {
                  'key': _option.key,
                  'value': _option.value
                }
              }
            )
          )
        }
      })

      return itemRes
    }

    let items = itemsRender(h)
    return h(
      'Form',
      {
        class: context.data.staticStyle,
        style: context.data.staticStyle,
        props: context.props
      },
      items
    )
  }
}
</script>

複製代碼

好了,有了上面的鋪路,你就能夠在項目的任何頁面使用配置表單了,這樣你就不用重複去 copy 結構代碼了,使得頁面中的代碼看着清爽;更重要的是分文件管理的方式,有利於維護。其實分頁列表也能夠參考這樣的方式。

一個含分頁列表和基礎表單的文件能夠是這樣的:

<template>
  <div class="hc-project-management">
    <CommonList
      :addSearch="addSearch"
      :columns="columns"
      :data="projectList"
      :pageBean="pageBean"
      :statePath="statePath"
    />
    <MyModal
      :isShow="modal.isShow"
      :title="modal.title"
    >
      <template slot="modalContent">
        <AutoForm
          :fileds="projectFields"
          :model="projectFormData"
          :formName="projectFormData"
          class="my-form"/>
      </template>
    </MyModal>
  </div>
</template>
複製代碼

如何根據 Select 框的選項動態新增表單項

有時候咱們會有像標題描述的需求,當一個下拉菜單選中後,自動的添加或者改變表單項。

實現: 我這邊會在 watch 中監聽 state 中數據變化來添加配置項

watch: {
    // 經過這種語法來watch
    'projectFormData': {
      handler: function (val, oldVal) { // 不能使用箭頭函數 this 指向會出問題
        if (val.projectSelect) {
          this.projectFields = Object.assign({}, this.projectFields, { projectTextarea: {
            label: '項目textarea',
            tag: 'Input',
            type: 'textarea',
            placeholder: '請輸入textarea'
          }})
        }
        console.log(val)
      },
      // 深度觀察
      deep: true
    }
  },
複製代碼

說兩句

  其實配置化仍是常規寫法,都是須要根據自身業務和開發成員等綜合考慮的,好比在配置化時,那麼就須要和組員約定好一個添加表單的流程和寫法,這個是相對固定的,不像常規的那麼自由;又好比,自己咱們這個項目表單的數量只有二、3個,那是否有配置化的必要;再好比,成員間是否定可這樣的寫法,也是須要商量的。可是一旦造成文檔規範,那麼回頭來看,配置化帶來的可維護性、易錯誤定位等好處,就顯得不用付出那麼多成本。

相關文章
相關標籤/搜索