form-render.vue 4.6 KB
<template>
  <!-- 在row上使用flex,防止表单组件大小不一导致错位 -->
  <component :is="rowComponent" class="eagle-form__flex-wrap" :class="contentClass || 'eagle-form__group-content'">
    <template v-for="(item, index) in list">
      <!-- 表单项有设置分组时 -->
      <component :is="colComponent" v-if="item.group && item.list" :key="index"
        :span="type === 'div' ? undefined : item.group.span || 24" :style="{ width: type === 'div' ? '100%' : undefined }"
      >
        <component :is="rowComponent" class="eagle-form__flex-wrap">
          <!-- 表单分组标题 -->
          <component :is="rowComponent" :class="titleClass || 'eagle-form__group-title'" v-if="item.group.title" style="width: 100%;">
            {{ item.group.title || item.group }}
          </component>
          <!-- 递归本组件 -->
          <form-render :title-class="titleClass" :item-class="itemClass" :list="item.list" :value="value"
            :model="itemKey ? model[itemKey] || {} : model" :itemKey="item.group.key" :type="type"
            @item-change="onItemChange" @form-item-change="onFormItemChange" @item-update="onItemUpdate"
            :span="type === 'div' ? undefined : span * (24 / (item.group.span || 24))"
          ></form-render>
        </component>
      </component>
      <!-- 正常无分组表单项 -->
      <component :is="colComponent" v-else :span="type === 'div' ? undefined : item.span || span" :key="index"
        :style="{ width: type === 'div' && item.style && item.style.width.includes('%') ? item.style.width : undefined, paddingRight: '10px' }"
      >
        <el-form-item :label="item.label" :label-width="item.labelWidth" :prop="item.fullKey" :rules="item.rules" :class="itemClass || 'eagle-form__item'">
          <!-- 自定义组件 -->
          <component :is="item.type" :value="itemValue(item)" @input="v => onInput({ value: v, item })"
            v-on="bindItemEvent(item)" v-bind="item.props" :style="item.style || { maxWidth: '100%' }"
          ></component>
        </el-form-item>
      </component>
    </template>
  </component>
</template>

<script>
export default {
  name: 'form-render',
  props: {
    list: Array,
    model: Object,
    value: Object,
    itemKey: String,
    titleClass: String,
    contentClass: String,
    itemClass: String,
    type: String,
    span: Number,
  },
  computed: {
    rowComponent() {
      return this.type === 'div' ? 'div' : 'el-row';
    },
    colComponent() {
      return this.type === 'div' ? 'div' : 'el-col';
    },
  },
  methods: {
    /**
     * @description 根据表单项的key查询该值
     * @param {Object} item 表单项配置
     * @returns {Any} 返回值
     */
    itemValue(item) {
      if (this.itemKey) { // 如果存在itemKey,即当前项位于嵌套分组内,查询分组名下对应key的值
        const groupItem = this.model[this.itemKey] || {};
        return groupItem[item.key];
      } else { // 否则即意味着不在分组内,直接查询model下对应key的值
        return this.model[item.key];
      }
    },
    /**
     * @description 组件有值输入时的事件
     * @param {Object} data { value => 组件值; item => 表单项配置 }
     */
    onInput({ value, item }) {
      if (this.itemKey) {
        this.$emit('item-change', { [this.itemKey]: { ...this.model[this.itemKey], [item.key]: value } });
      } else {
        this.$emit('item-change', { [item.key]: value });
      }
      this.$emit('form-item-change', { [item.fullKey]: value });
    },
    /**
     * @description 当表单项有改动时的事件
     * @param {Any} 表单项值
     */
    onItemChange(value) {
      if (this.itemKey) {
        this.$emit('item-change', { [this.itemKey]: { ...this.model[this.itemKey], ...value } });
      } else {
        this.$emit('item-change', value);
      }
    },
    /**
     * @description 当表单项校验值有改动时的事件
     * @param {Any} 表单项校验值
     */
    onFormItemChange(value) {
      this.$emit('form-item-change', value);
    },
    /**
     * @description 当表单项有手动更新时的事件
     * @param {Any} 表单项值
     */
    onItemUpdate(value) {
      this.$emit('item-update', value);
    },
    /**
     * @description 绑定表单项事件
     * @param {Object} item 表单项配置
     * @returns {Function} 事件函数
     */
    bindItemEvent(item) {
      if (item.on) {
        if (typeof item.on === 'function') {
          return item.on({ model: this.value, update: e => this.$emit('item-update', e) });
        } else {
          return item.on
        }
      } else {
        return undefined
      }
    },
  }
}
</script>