记录一下开发中对form表单的二次封装及使用

前言

        在我们的日常工作中经常会使用到form表来满足提交需求,那么将form表单封装成组件将会大大提高我们的工作效率,避免更多重复代码的书写,只需一些简单的配置就轻松实现。

一、form表单的封装

<template>
  <div class='form'>
    <el-form
      :label-position="labelPosition"
      :inline="inline"
      :rules="rules"
      ref="formRef"
      :label-width="labelWidth"
      :model="fieldForm"
      :class="customClass"
    >
      <el-form-item
        v-for="(item, index) in fieldList"
        :key="index"
        :prop="item.field"
        :validate-on-rule-change="false"
        :style="{
          flexBasis: item.percent ? item.percent : item.percent ?     
         `${item.percent}%!important` : '',
          width: item.width && isNaN(Number(item.width)) ? item.width +'!important'|| '' 
          : item.width + 'px!important' || '',
        }"
        :class="[item.className || '', `is-${item.formType}`]"
        :rules="getRules(item)"
      >
        <template
          #label
          v-if="item.name"
        >
          <div style="display: inline">
            <a :style="{ color: item.labelColor || '' }">{{ item.name || '' }}</a>
          </div>
        </template>
        <template v-if="item.formType && item.mold && !item.hidden">
          <component
            :is="com['M' + item.mold]"
            v-model="fieldForm[item.field]"
            :item="item"
            :index="index"
            :disabled="disabled"
            :size="size"
            :field-form="fieldForm"
            @change="commonChange"
          />
        </template>
      </el-form-item>
    </el-form>
  </div>
</template>
<script lang="ts" setup>
import { toRefs } from 'vue';
//引入组件
import MInput from './components/MInput.vue';
import MDate from './components/MDate.vue';
import MSelect from './components/MSelect.vue';

//这是自定义表单验证规则
import GenerateRules from './GenerateRules.ts';

//接收父组件传递的参数
const props = defineProps({
  customClass: {
    type: String,
    default: 'is-three-columns',
  },
  rules: {
    type: Object,
    default: () => {
      return {};
    },
  },
  fieldForm: {
    type: Object,
    default: () => {
      return {};
    },
  },
  labelWidth: {
    type: Number,
    default: 90
  },
  fieldList: {
    type: Array,
    default: () => [],
  },
  // 行内表单
  inline: {
    type: Boolean,
    default: false,
  },
  // 对齐方式 'right' | 'left' | 'top'
  labelPosition: {
    type: String,
    default: 'right',
  },
  size: {
    type: String,
    default: 'default',
  },
  disabled: {
    type: Boolean,
    default: false,
  },
});
const { customClass, rules, fieldForm, labelWidth, fieldList } = toRefs(props);

// 定义组件
const com: Record<string, any> = {
  MInput: MInput,
  MDate: MDate,
  MSelect: MSelect,
};

//事件传递
const $emit = defineEmits(['change']);

/**
 * 常规组件change事件
 * @param item 配置数据
 * @param index 下标
 * @param value 选中/输入的值
 * @param data 选中的整条数据
 */
const commonChange = (item, index, value, data) => {
  $emit('change', item, index, value, data);
};

//自定义验证规则
const { getRules } = GenerateRules();

</script>

二、components 

1.MDate.vue

代码如下(示例):

<template>
  <el-date-picker
    v-if="item.formType == 'date'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    clearable
    style="width: 100%"
    type="date"
    :size="size"
    :editable="true"
    value-format="YYYY-MM-DD"
    :placeholder="item.placeholder || '选择日期'"
    @click.stop
    @change="commonChange(item, index, $event)"
  />
  <el-date-picker
    v-else-if="item.formType == 'month'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    clearable
    style="width: 100%"
    type="month"
    :size="size"
    value-format="YYYY-MM"
    :placeholder="item.placeholder || '选择月份'"
    @change="commonChange(item, index, $event)"
  />
  <el-date-picker
    v-else-if="item.formType == 'year'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    clearable
    style="width: 100%"
    type="year"
    :size="size"
    value-format="YYYY"
    :placeholder="item.placeholder || '选择年份'"
    @change="commonChange(item, index, $event)"
  />
  <el-date-picker
    v-else-if="item.formType == 'dateRange'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    :type="item.dateType || 'daterange'"
    :value-format="item.dateValueFormat || 'YYYY-MM-DD'"
    :picker-options="item.pickerOptions || ''"
    clearable
    :size="size"
    style="width: 100%"
    :start-placeholder="item.startPlaceholder || '开始日期'"
    :end-placeholder="item.endPlaceholder || '结束日期'"
    @change="commonChange(item, index, $event)"
  />
  <el-date-picker
    v-else-if="item.formType == 'datetime'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    clearable
    style="width: 100%"
    :size="size"
    type="datetime"
    value-format="YYYY-MM-DD HH:mm:ss"
    :placeholder="item.placeholder || '选择日期'"
    @change="commonChange(item, index, $event)"
  />
</template>
<script lang="ts" setup>
import { toRefs, PropType } from 'vue';
import Hooks from '../Hooks.ts';
// 定义defineProps
const props = defineProps({
  modelValue: {
    type: [Number, String],
    default: '',
  },
  item: {
    type: Object,
    default: () => {},
  },
  disabled: {
    type: Boolean,
    default: false,
  },
  size: {
    type: String as PropType <''| 'large' | 'small'>,
    default: 'small',
  },
  index: {
    type: Number,
    default: null,
  },
  fieldForm: {
    type: Object,
    default: () => {},
  },
});
const { fieldForm } = toRefs(props);

// 定义$emit
const $emit = defineEmits(['change']);

const { commonChange } = Hooks($emit);
</script>

2.MInput.vue

代码如下(示例):

<template>
  <el-input
    v-if="item.formType == 'text' && !item.hidden"
    v-model="fieldForm[item.field]"
    :size="size || ''"
    :disabled="item.disabled || disabled"
    :maxlength="item.maxlength"
    :placeholder="item.placeholder"
    :type="item.formType"
    :show-word-limit="item.showLimit"
    :title="fieldForm[item.field]"
    clearable
    :readonly="item.readonly"
    :name="item.field"
    @input="commonChange(item, index, $event)"
  />
  <el-input-number
    v-else-if="['floatnumber', 'number'].includes(item.formType)"
    v-model="fieldForm[item.field]"
    :placeholder="item.placeholder"
    :disabled="item.disabled || disabled"
    :controls="item.controls || false"
    :controls-position="item.position || 'right'"
    clearable
    :size="size"
    @change="commonChange(item, index, $event)"
  />
  <el-input
    v-else-if="item.formType === 'textarea'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    :rows="item.rows || 2"
    :autosize="!item.autosize && { minRows: item.rows || 2 }"
    :maxlength="item.maxlength || 800"
    :placeholder="item.placeholder"
    :type="item.formType"
    clearable
    :size="size"
    :resize="item.resize || 'none'"
    @input="commonChange(item, index, $event)"
  />
</template>
<script lang="ts" setup>
import { toRefs} from 'vue';
import Hooks from '../Hooks.ts';

// 定义props 
const props = defineProps({
  modelValue: {
    default: '',
  },
  item: {
    type: Object,
    default: () => {},
  },
  disabled: {
    type: Boolean,
    default: false,
  },
  size: {
    type: String,
    default: 'small',
  },
  index: {
    type: Number,
    default: null,
  },
  fieldForm: {
    type: Object,
    default: () => {},
  },
});
const { fieldForm } = toRefs(props);

// 定义$emit
const $emit = defineEmits(['change']);

const { commonChange } = Hooks($emit);
</script>

3.MInput.vue

代码如下(示例):

<template>
  <el-select
    v-if="item.formType == 'select'"
    v-model="fieldForm[item.field]"
    :disabled="item.disabled || disabled"
    :clearable="item.clearable || true"
    :placeholder="item.placeholder"
    :multiple="item.multiple"
    :collapse-tags="item.collapse"
    :collapse-tags-tooltip="item.tooltip"
    :size="size"
    :value-key="item.props?.valueKey || 'id'"
    filterable
    style="width: 100%"
    @change="commonChange(item, index, $event)"
  >
    <el-option
      v-for="(items, i) in item.basicGroup
        ? getGroupValue(item)
        : item.setting"
      :key="i"
      :label="!isEmptyValue(items[item.props?.value || 'value']) ?                 
      items[item.props?.label || 'label'] || items.name : items"
      :value="!isEmptyValue(items[item.props?.value || 'value']) ? 
      items[item.props?.value || 'value'] : items"
    />
  </el-select>
  <el-radio-group
    v-else-if="item.formType == 'radio-group'"
    :size="size"
    :disabled="item.disabled || disabled"
    v-model="fieldForm[item.field]"
    @change="commonChange(item, index, $event)"
  >
    <el-radio
      :label="items.value"
      v-for="(items, i) in item.setting"
      :key="i"
    >{{ items.label }}</el-radio>
  </el-radio-group>
  <el-switch
    v-else-if="item.formType == 'switch'"
    v-model="fieldForm[item.field]"
    inline-prompt
    :size="size"
    :active-color="item.active&&item.active.color || '#67c23a'"
    :inactive-color="item.inactive&&item.inactive.color || '#f56c6c'"
    :active-text="item.active&&item.active.text || '是'"
    :inactive-text="item.inactive&&item.inactive.text || '否'"
  />
  <el-checkbox-group
    v-else-if="item.formType == 'checkbox-group'"
    :size="size"
    :disabled="item.disabled || disabled"
    v-model="fieldForm[item.field]"
    @change="commonChange(item, index, $event)">
    <el-checkbox
      :label="items.label"
      v-for="(items, i) in item.setting"
      :key="i"
    />
  </el-checkbox-group>
</template>
<script lang="ts" setup>
import Hooks from '../Hooks.ts';
import { toRefs} from 'vue';

// 定义props 
const props = defineProps({
  modelValue: {
    default: '',
  },
  item: {
    type: Object,
    default: () => { },
  },
  disabled: {
    type: Boolean,
    default: false,
  },
  size: {
    type: String,
    default: 'default',
  },
  index: {
    type: Number,
    default: null,
  },
  fieldForm: {
    type: Object,
    default: () => { },
  },
});
const { fieldForm, size } = toRefs(props);

// 定义$emit
const $emit = defineEmits(['change']);

const { commonChange, isEmptyValue } = Hooks($emit);
</script>

4.Hooks.ts

代码如下(示例):

export default function ($emit: any) {
    
 /**
 * 常规组件change事件
 */
  const commonChange = (item: any, index: any, value: any, data: any) => {
    $emit('change', item, index, value, data);
  };


  /**
  * 判断是空值
  */
  const isEmptyValue = (value: any) => {
    return value === null || value == undefined;
  };

  return {
    commonChange,
    isEmptyValue,
  };
}

5.Hooks.ts

代码如下(示例):

/**
 * Created by yxk at 2020/6/1
 * 单个自定义字段生成校验规则
 */

//类型判断 这些就不贴出来啦 
import { isEmpty, isObject, isArray } from '@/utils/types.ts';
export default function () {
  /**
   * 唯一性校验
   * @param data
   * @returns {Promise<unknown>}
   * @constructor
  */
  const UniquePromise = (data: any) => {
    return new Promise((resolve: any) => {
      resolve(data);
    });
  };
  /**
 * 生成单个字段的验证规则
 * @param item 字段信息
 * @returns {[]}
 */
  const getRules = (item: any) => {
    const tempList = [];
    // 验证必填
    if (item.isNull === 1) {
      if (item.formType === 'detail_table') {
        tempList.push({
          validator: ({ item }: any, value: any, callback: any) => {
            if (getDetailTableIsEmpty(item.fieldExtendList, value)) {
              callback(new Error(item.name + '不能为空'));
            } else {
              callback();
            }
          },
          item: item,
          trigger: ['blur', 'change']
        });
      } else if (item.formType === 'checkbox' || item.formType === 'dep-select') {
        tempList.push({
          validator: ({ item }: any, value: any, callback: any) => {
            if (!isArray(value) || value.length === 0) {
              callback(new Error(item.name + '不能为空'));
            } else {
              const emptyObj = value.find((valueItem: any) => isEmpty(valueItem));
              emptyObj === '' ? callback(new Error(item.name + '不能为空')) : callback();
            }
          },
          item: item,
          trigger: ['blur', 'change']
        });
      } else {
        tempList.push({
          required: true,
          message: item.name + '不能为空',
          trigger: ['blur', 'change']
        });
      }
    }

    // 验证唯一
    if (item.isUnique === 1 && UniquePromise) {
      const validateUnique = (rule: any, value: any, callback: any) => {
        if (isEmpty(value)) {
          callback();
        } else {
          // 验证唯一
          UniquePromise({
            field: item,
            rule: rule,
            value: value
          }).then(() => {
            callback();
          }).catch(msg => {
            callback(new Error(msg || '验证出错'));
          });
        }
      };
      tempList.push({
        validator: validateUnique,
        item: item,
        trigger:
          item.formType === 'checkbox' || item.formType === 'select' ?
            ['change', 'blur'] :
            ['blur']
      });
    }

    return tempList;
  };

  /**
  * 获取数值规则
  */
  const getNumberRule = (rule: any, value: any, callback: any) => {
    const field = rule.item;

    const arr = String(value).split('.');

    const len = String(value)
      .replace('.', '')
      .replace('-', '')
      .length;
    const maxlength = field.formType === 'percent' ? 10 : 15;

    const min = isEmpty(field.minNumRestrict) ? -Infinity : Number(field.minNumRestrict 
               || -Infinity);
    const max = isEmpty(field.maxNumRestrict) ? Infinity : Number(field.maxNumRestrict || 
    Infinity);

    if (len > maxlength) {
      callback(new Error(`最多支持${maxlength}位数字(包含小数位)`));
    } else if (isEmpty(field.precisions) && String(value).includes('.')) {
      // null 不支持小数  0 不限制小数位
      callback(new Error(`不支持小数`));
    } else if (arr.length > 1 && arr[1].length > Number(field.precisions)) {
      callback(new Error(`小数位不能大于${field.precisions}`));
    } else if (value < min) {
      callback(new Error(`不能小于${min}`));
    } else if (value > max) {
      callback(new Error(`不能大于${max}`));
    } else {
      callback();
    }
  };

  /**
  * 判断明细表格是否是空
  * @param {*} fieldList
  * @param {*} valueObj
  */
  const getDetailTableIsEmpty = (fieldList: any, valueObjs: any) => {
    for (let index = 0; index < valueObjs.length; index++) {
      const valueObj = valueObjs[index];
      if (judgeFormValueIsEmpty(fieldList, valueObj)) {
        return true;
      }
    }
    return false;
  };
  /**
  * 判断对象值是否是空
  */
  const judgeFormValueIsEmpty = (fieldList: any, valueObj: any) => {
    for (let index = 0; index < fieldList.length; index++) {
      const field = fieldList[index];
      const value = valueObj[field.fieldName];
      if (field.formType === 'location') {
        if (isObject(value) && (!isEmpty(value.lat) || !isEmpty(value.lng) ||                     
        !isEmpty(value.address))) {
          return false;
        }
      } else if (!isEmpty(value)) {
        return false;
      }
    }
    return true;
  };
  return {
    UniquePromise,
    getRules
  };
}

6.form表单的使用

<div class="form_container">
      <Form
        :field-list="fieldList"
        :field-form="fieldForm" />
   </div>
const formData = reactive({
  fieldForm: {},
  fieldList: [
    {
      field: 'name',
      name: '活动名称',
      mold: 'Input',
      formType: 'text',
      // 检验唯一性
      isNull:1
    },
    {
      field: 'region',
      name: '活动区域',
      mold: 'Select',
      formType: 'select',
      setting: [
        {
          label: '区域一',
          value: 0
        },
        {
          label: '区域二',
          value: 0
        }
      ]
    },
    {
      field: 'date',
      name: '活动区域',
      mold: 'Date',
      formType: 'datetime',
    },
    {
      field: 'deliver',
      name: '即时配送',
      mold: 'Select',
      formType: 'switch',
    },
    {
      field: 'type',
      name: '活动性质',
      mold: 'Select',
      formType: 'checkbox-group',
      setting: [
        {
          label: '美食/餐厅线上活动',
        },
        {
          label: '地推活动',
        },
        {
          label: '线下主题活动',
        },
        {
          label: '单纯品牌曝光',
        },
      ]
    },
    {
      field: 'resource',
      name: '资源',
      mold: 'Select',
      formType: 'radio-group',
      setting: [
        {
          label: '线上品牌商赞助',
          value:0
        },
        {
          label: '线下场地免费',
          value:1
        },
      ]
    },
    {
      field: 'desc',
      name: '活动形式',
      mold: 'Input',
      formType: 'textarea',
      rows:3
    },
  ]
});

 出来的效果就是这样子滴

element 现成的可编辑表格封装控件 element form表单封装_vue.js

 

总结

         以上就是form表单的全部代码