vue elementUI组件表单动态验证失效的问题与解决办法

本文介绍了在Vue项目中使用Element UI实现动态表单验证的方法。针对特定场景需求,通过不同尝试,最终找到有效解决方案,包括利用v-if和key属性强制重新渲染表单,以及动态修改rules对象或form-item的rules属性。

一、缘由
在项目中,有一个需求是需要根据条件给表单项添加验证属性prop确定是否验证表项。

二、第一次实现与遇到的问题
比如银行卡号根据输入年龄进行判断,如果大于等于18岁才需要填入银行卡号。最先的想法先设置好el-form的rules,然后通过三元表达式赋值prop属性,实现动态验证,示例如下:

<template>
  <el-form ref="ruleForm" :model="formData" :rules="rules">
    <el-form-item label="名字" prop="name">
      <el-input v-model="formData.name"></el-input>
    </el-form-item>
    <el-form-item label="年龄" prop="age">
      <el-input v-model="formData.age"></el-input>
    </el-form-item>
    <el-form-item label="银行卡号" :prop="formData.age >= 18 ? 'bankCardNo' : ''">
      <el-input v-model="formData.bankCardNo"></el-input>
    </el-form-item>
  </el-form>
</template>
<script>
export default {
  name: 'HelloWorld',
 
  data () {
    return {
      rules: {
        name: [{ required: true, message: '请输入名字', trigger: 'blur' }],
        age: [{ required: true, message: '请输入年龄', trigger: 'blur' }],
        bankCardNo: [{ required: true, message: '请输入银行卡号', trigger: 'blur' }]
      },
      formData: {
        name: '',
        age: '',
        bankCardNo: ''
      }
    }
  }
}
</script>
但是发现虽然页面有了红色星号的验证提示,但实际上当失去焦点的时候并未触发校验,如图所示:

三、遇到问题的原因
所以去查阅了elementUI的官方文档,文档给出了一种动态添加表单项动态的验证的示例,示例代码如下:

<el-form :model="dynamicValidateForm" ref="dynamicValidateForm" label-width="100px" class="demo-dynamic">
  <el-form-item
    prop="email"
    label="邮箱"
    :rules="[
      { required: true, message: '请输入邮箱地址', trigger: 'blur' },
      { type: 'email', message: '请输入正确的邮箱地址', trigger: ['blur', 'change'] }
    ]"
  >
    <el-input v-model="dynamicValidateForm.email"></el-input>
  </el-form-item>
  <el-form-item
    v-for="(domain, index) in dynamicValidateForm.domains"
    :label="'域名' + index"
    :key="domain.key"
    :prop="'domains.' + index + '.value'"
    :rules="{
      required: true, message: '域名不能为空', trigger: 'blur'
    }"
  >
    <el-input v-model="domain.value"></el-input><el-button @click.prevent="removeDomain(domain)">删除</el-button>
  </el-form-item>
  <el-form-item>
    <el-button type="primary" @click="submitForm('dynamicValidateForm')">提交</el-button>
    <el-button @click="addDomain">新增域名</el-button>
    <el-button @click="resetForm('dynamicValidateForm')">重置</el-button>
  </el-form-item>
</el-form>
<script>
  export default {
    data() {
      return {
        dynamicValidateForm: {
          domains: [{
            value: ''
          }],
          email: ''
        }
      };
    },
    methods: {
      submitForm(formName) {
        this.$refs[formName].validate((valid) => {
          if (valid) {
            alert('submit!');
          } else {
            console.log('error submit!!');
            return false;
          }
        });
      },
      resetForm(formName) {
        this.$refs[formName].resetFields();
      },
      removeDomain(item) {
        var index = this.dynamicValidateForm.domains.indexOf(item)
        if (index !== -1) {
          this.dynamicValidateForm.domains.splice(index, 1)
        }
      },
      addDomain() {
        this.dynamicValidateForm.domains.push({
          value: '',
          key: Date.now()
        });
      }
    }
  }
</script>
其运行结果如下:

但官方文档提供的示例与我想要的并不一样,于是上网查找了其他资料。

发现我的方法与官方文档的区别在于,官方文档的示例是通过v-for循环的方式渲染出来的表单,当数组domains变化时会触发表单的重新渲染(起初并未意识到这一点),所以能够有效的动态验证表单,而我的方法只是动态修改了props属性,而并未出发表单的重新渲染,验证器中的prop属性并未更新,还是初次渲染时的空值,知道原因就想到了解决办法。

四、第二次实现与遇到的问题
我的思路是通过v-if和v-else写两个表单项,一个带prop属性,一个不带prop属性,通过强制销毁和创建表单实现表单的重新渲染。

<template>
  <el-form ref="ruleForm" :model="formData" :rules="rules">
    <el-form-item label="名字" prop="name">
      <el-input v-model="formData.name"></el-input>
    </el-form-item>
    <el-form-item label="年龄" prop="age">
      <el-input v-model="formData.age"></el-input>
    </el-form-item>
    <el-form-item label="银行卡号" v-if="formData.age >= 18" prop="bankCardNo">
      <el-input v-model="formData.bankCardNo"></el-input>
    </el-form-item>
    <el-form-item label="银行卡号" v-else>
      <el-input v-model="formData.bankCardNo"></el-input>
    </el-form-item>
  </el-form>
</template>
然而并没啥用,还是没有生效!!!

实际上出现这个问题是因为vue的diff算法,在diff算法中会尽可能的复用组件,所以此处看上根据formData.age是否大于等于18进行切换,但实际上diff算法会复用组件,并没有重新渲染,所以需要配合key属性。

官方文档是这样介绍的:

Vue代码风格指南

五、最终解决办法
完整代码如下:

<template>
  <el-form ref="ruleForm" :model="formData" :rules="rules">
    <el-form-item label="名字" prop="name">
      <el-input v-model="formData.name"></el-input>
    </el-form-item>
    <el-form-item label="年龄" prop="age">
      <el-input v-model="formData.age"></el-input>
    </el-form-item>
    <el-form-item label="银行卡号" key="1" v-if="formData.age >= 18" prop="bankCardNo">
      <el-input v-model="formData.bankCardNo"></el-input>
    </el-form-item>
    <el-form-item label="银行卡号" key="2" v-else>
      <el-input v-model="formData.bankCardNo"></el-input>
    </el-form-item>
  </el-form>
</template>
<script>
export default {
  name: 'HelloWorld',
 
  data () {
    return {
      rules: {
        name: [{ required: true, message: '请输入名字', trigger: 'blur' }],
        age: [{ required: true, message: '请输入年龄', trigger: 'blur' }],
        bankCardNo: [{ required: true, message: '请输入银行卡号', trigger: 'blur' }]
      },
      formData: {
        name: '',
        age: '',
        bankCardNo: ''
      }
    }
  }
}
</script>
最终效果如图,当年龄大于等于18时就会触发验证器。

2020年4月3日更新
上述方法没有真正理解原因,可以查看本人新文章《深入了解Element Form表单动态验证问题》,提及两种新解决办法:

1.  form-item保留prop属性,根据条件动态修改form组件的rules对象

watch: {
  'formData.age': {
    immediate: true,
    handler (newVal) {
      if (newVal >= 18) {
        this.addRule('bankCardNo', [{ required: true, message: '请输入银行卡号', trigger: 'blur' }])
      } else {
        this.addRule('bankCardNo', [])
      }
    }
  }
}
 
data () {
  return {
    rules: {}
  }
},
 
methods: {
  addRule (prop, rule) {
    this.$set(this.rules, prop, rule)
  }
}
2. form-item保留prop属性,根据条件动态修改form-item组件的rules属性。比如下面两种方式,因为rules属性接受数据和对象

<el-form-item label="银行卡号" prop="bankCardNo" :rules="formData.age >= 18 ? [{ required: true, message: '请输入银行卡号', trigger: 'blur' }] : []">
  <el-input v-model="formData.bankCardNo"></el-input>
</el-form-item>
 
<!-- 或者 -->
<el-form-item label="银行卡号" prop="bankCardNo" :rules="formData.age >= 18 ? { required: true, message: '请输入银行卡号', trigger: 'blur' } : []">
  <el-input v-model="formData.bankCardNo"></el-input>
</el-form-item>
 

<think>我们正在解决VueElementUI表单校验失效问题。根据提供的引用,我们可以总结出以下常见原因及解决方案:原因1:嵌套布局导致引用[1]中提到,在el-form-item内部嵌套了el-row和el-col可能导致校验失效。这是因为ElementUI表单校验需要精确的DOM结构,过多的嵌套可能破坏校验机制。解决方案:尽量减少在el-form-item内部使用布局组件,若必须使用,确保每个el-form-item直接包含表单项(如el-select)。原因2:初始数据未声明引用[2]中提到,未在data中声明初始值可能导致校验失效(虽然该引用中未复现,但这是一个常见问题)。解决方案:确保每个需要校验表单字段都在data中声明,即使为空值。例如:```javascriptdata(){return{addForm:{tableName:'',//必须声明//...其他字段}}}```原因3:多个表单混合校验(v-if导致)引用[3]指出,如果使用v-if切换表单,由于Vue2的虚拟DOM优化,可能造成新旧表单校验规则混合。解决方案:为每个表单外层添加一个唯一key,或者使用div包裹表单并确保结构同(打破虚拟DOM的复用)。原因4:trigger触发方式选择错误引用[4]中提到,同的表单组件需要设置同的trigger:-输入框(el-input)通常使用blur(失去焦点)-选择框(el-select)、日期选择器等使用change(值改变)解决方案:根据组件类型设置正确的trigger:```javascriptrules:{//输入框使用blurname:[{required:true,message:'请输入',trigger:'blur'}],//选择框使用changetableName:[{required:true,message:'请选择',trigger:'change'}]}```原因5:prop属性字段名一致在el-form-item上指定的prop必须表单对象中的字段名一致,且必须是字符串(能是变量)。解决方案:检查每个el-form-item的prop是否v-model绑定的字段名完全一致。原因6:表单引用问题有时需要给el-form设置ref,并在提交时调用validate方法。解决方案:确保正确使用ref进行校验:```html<el-formref="formRef":model="addForm"><!--表单项--></el-form>``````javascriptmethods:{submit(){this.$refs.formRef.validate(valid=>{if(valid){//提交}})}}```原因7:自定义校验规则错误如果使用了自定义校验函数,函数中必须调用callback(),否则校验会卡住。解决方案:检查自定义规则是否正确使用callback:```javascriptrule:(rule,value,callback)=>{if(!value){returncallback(newError('请输入'))}callback()//必须调用}```综合以上,请按以下步骤排查:1.检查DOM结构,避免el-form-item内嵌套复杂布局,若必须嵌套,确保el-form-item直接包含表单项2.检查data中是否声明了所有表单字段的初始值3.检查trigger设置是否合理(根据组件类型)4.检查prop是否表单字段名一致5.检查ref调用validate的代码是否正确6.检查自定义校验规则是否调用了callback如果问题仍未解决,可以尝试:-更新ElementUI到最新版本-提供一个可复现问题的在线示例(如CodeSandbox)根据引用[3]的提示,如果页面中存在多个表单,且通过v-if切换,可以给每个表单添加同的key:```html<el-formv-if="showForm1"key="form1">...</el-form><el-formv-if="showForm2"key="form2">...</el-form>```或者为表单包裹一个div并保持结构同(如添加同的类名)。如果上述方法都能解决,可能是个别场景下的bug,建议在ElementUI的GitHub仓库搜索类似issue。希望这些方法能帮助您解决问题。</think> ## Vue Element UI 表单校验失效 原因及解决方案 在使用 Vue 和 Element UI 的表单校验时,常见的失效原因及解决方法如下: ### 🔍 常见失效原因 1. **组件嵌套问题** `el-form-item` 中嵌套 `el-row/el-col` 可能导致校验失效[^1]。 ```html <!-- 错误示例 --> <el-form-item prop="tableName"> <el-row> <el-col :span="12"> <el-select v-model="addForm.tableName">...</el-select> </el-col> </el-row> </el-form-item> ``` 2. **v-if 导致虚拟 DOM 复用问题** 使用 `v-if` 切换表单时,新旧表单校验可能混合导致失效[^3]。 3. **trigger 触发方式配置错误** 同类型组件同触发方式[^4]: - 输入框:`trigger: 'blur'` - 下拉框/选择器:`trigger: 'change'` 4. **初始数据未声明** 未在 `data` 中声明字段初始值可能导致问题[^2]: ```javascript data() { return { form: { name: '' // 必须声明 } } } ``` ### 🔧 解决方案 1. **简化组件嵌套结构** ```html <!-- 正确方式 --> <el-row> <el-col :span="12"> <el-form-item prop="tableName"> <el-select v-model="addForm.tableName">...</el-select> </el-form-item> </el-col> </el-row> ``` 2. **解决 v-if 导致的问题** ```html <div v-if="showForm" :key="formKey"> <!-- 添加唯一 key --> <el-form>...</el-form> </div> ``` ```javascript methods: { resetForm() { this.formKey = Date.now() // 切换时改变 key } } ``` 3. **正确配置 trigger** ```javascript rules: { // 下拉框使用 change selectField: [ { required: true, message: '请选择', trigger: 'change' } ], // 输入框使用 blur inputField: [ { required: true, message: '请输入', trigger: 'blur' } ] } ``` 4. **验证初始数据声明** ```javascript data() { return { form: { name: '', // 文本字段 selectValue: null, // 选择器字段 dateValue: '' // 日期字段 } } } ``` 5. **检查 prop 命名一致性** ```html <el-form-item prop="userName"> <!-- prop 名 --> <el-input v-model="form.userName"></el-input> <!-- 字段名必须一致 --> </el-form-item> ``` ### ✅ 推荐调试步骤 1. 检查浏览器控制台是否有报错 2. 确保 `el-form` 添加了 `:model` 属性 3. 确认 `ref` 引用名称正确(提交时调用) 4. 简化表单结构,逐步排除嵌套问题 5. 尝试手动触发校验测试: ```javascript this.$refs.formRef.validateField('fieldName') ``` 📌 **关键提示**:Element UI 表单校验对 DOM 结构敏感,保持表单结构简单能减少大多数问题。若使用动态表单,务必确保每个表单项有独立的 `key`[^3]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值