—— 封面攝於濟州島民宿
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
三個參數的簡單用法:spa
<script>
Vue.component('Line', {
render: function(h) {
h('div', {
props: {} // 傳遞數據
},'文本 or 子節點')
}
})
</script>
複製代碼
瞭解基礎用法後,咱們來看下 <AutoForm />
組件的實現:雙向綁定
在上代碼以前,咱們先看一下 iView
表單的結構,從外層到內層,Form 容器固定
——FormItem 數量動態
——Input 類型動態
,組件最終是返回一個 Form
;根據配置項的數量來決定 FormItem
的數量,動態建立;根據配置項的 tag
和 type
來決定表單的類型;固然有些例如 Select
的表單項會有 options
下拉選項,也須要單獨生成。
根據上面的分析,那總結關於這個 <AutoForm />
組件,大體有 FormRender
、itemsRender
、componentUse
、類型(InputRender
、RadioRender
、SelectRender
)、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>
複製代碼
有時候咱們會有像標題描述的需求,當一個下拉菜單選中後,自動的添加或者改變表單項。
實現: 我這邊會在 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個,那是否有配置化的必要;再好比,成員間是否定可這樣的寫法,也是須要商量的。可是一旦造成文檔規範,那麼回頭來看,配置化帶來的可維護性、易錯誤定位等好處,就顯得不用付出那麼多成本。