当前位置:   article > 正文

vue elementUI组件表单动态验证失效的问题与解决办法_elementui 动态表单 自定义效验不生效

elementui 动态表单 自定义效验不生效

一、缘由

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

二、第一次实现与遇到的问题

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

  1. <template>
  2. <el-form ref="ruleForm" :model="formData" :rules="rules">
  3. <el-form-item label="名字" prop="name">
  4. <el-input v-model="formData.name"></el-input>
  5. </el-form-item>
  6. <el-form-item label="年龄" prop="age">
  7. <el-input v-model="formData.age"></el-input>
  8. </el-form-item>
  9. <el-form-item label="银行卡号" :prop="formData.age >= 18 ? 'bankCardNo' : ''">
  10. <el-input v-model="formData.bankCardNo"></el-input>
  11. </el-form-item>
  12. </el-form>
  13. </template>
  14. <script>
  15. export default {
  16. name: 'HelloWorld',
  17. data () {
  18. return {
  19. rules: {
  20. name: [{ required: true, message: '请输入名字', trigger: 'blur' }],
  21. age: [{ required: true, message: '请输入年龄', trigger: 'blur' }],
  22. bankCardNo: [{ required: true, message: '请输入银行卡号', trigger: 'blur' }]
  23. },
  24. formData: {
  25. name: '',
  26. age: '',
  27. bankCardNo: ''
  28. }
  29. }
  30. }
  31. }
  32. </script>

但是发现虽然页面有了红色星号的验证提示,但实际上当失去焦点的时候并未触发校验,如图所示:

三、遇到问题的原因

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

  1. <el-form :model="dynamicValidateForm" ref="dynamicValidateForm" label-width="100px" class="demo-dynamic">
  2. <el-form-item
  3. prop="email"
  4. label="邮箱"
  5. :rules="[
  6. { required: true, message: '请输入邮箱地址', trigger: 'blur' },
  7. { type: 'email', message: '请输入正确的邮箱地址', trigger: ['blur', 'change'] }
  8. ]"
  9. >
  10. <el-input v-model="dynamicValidateForm.email"></el-input>
  11. </el-form-item>
  12. <el-form-item
  13. v-for="(domain, index) in dynamicValidateForm.domains"
  14. :label="'域名' + index"
  15. :key="domain.key"
  16. :prop="'domains.' + index + '.value'"
  17. :rules="{
  18. required: true, message: '域名不能为空', trigger: 'blur'
  19. }"
  20. >
  21. <el-input v-model="domain.value"></el-input><el-button @click.prevent="removeDomain(domain)">删除</el-button>
  22. </el-form-item>
  23. <el-form-item>
  24. <el-button type="primary" @click="submitForm('dynamicValidateForm')">提交</el-button>
  25. <el-button @click="addDomain">新增域名</el-button>
  26. <el-button @click="resetForm('dynamicValidateForm')">重置</el-button>
  27. </el-form-item>
  28. </el-form>
  29. <script>
  30. export default {
  31. data() {
  32. return {
  33. dynamicValidateForm: {
  34. domains: [{
  35. value: ''
  36. }],
  37. email: ''
  38. }
  39. };
  40. },
  41. methods: {
  42. submitForm(formName) {
  43. this.$refs[formName].validate((valid) => {
  44. if (valid) {
  45. alert('submit!');
  46. } else {
  47. console.log('error submit!!');
  48. return false;
  49. }
  50. });
  51. },
  52. resetForm(formName) {
  53. this.$refs[formName].resetFields();
  54. },
  55. removeDomain(item) {
  56. var index = this.dynamicValidateForm.domains.indexOf(item)
  57. if (index !== -1) {
  58. this.dynamicValidateForm.domains.splice(index, 1)
  59. }
  60. },
  61. addDomain() {
  62. this.dynamicValidateForm.domains.push({
  63. value: '',
  64. key: Date.now()
  65. });
  66. }
  67. }
  68. }
  69. </script>

其运行结果如下:

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

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

四、第二次实现与遇到的问题

我的思路是通过v-if和v-else写两个表单项,一个带prop属性,一个不带prop属性,通过强制销毁和创建表单实现表单的重新渲染。

  1. <template>
  2. <el-form ref="ruleForm" :model="formData" :rules="rules">
  3. <el-form-item label="名字" prop="name">
  4. <el-input v-model="formData.name"></el-input>
  5. </el-form-item>
  6. <el-form-item label="年龄" prop="age">
  7. <el-input v-model="formData.age"></el-input>
  8. </el-form-item>
  9. <el-form-item label="银行卡号" v-if="formData.age >= 18" prop="bankCardNo">
  10. <el-input v-model="formData.bankCardNo"></el-input>
  11. </el-form-item>
  12. <el-form-item label="银行卡号" v-else>
  13. <el-input v-model="formData.bankCardNo"></el-input>
  14. </el-form-item>
  15. </el-form>
  16. </template>

然而并没啥用,还是没有生效!!!

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

官方文档是这样介绍的:

Vue代码风格指南

五、最终解决办法

完整代码如下:

  1. <template>
  2. <el-form ref="ruleForm" :model="formData" :rules="rules">
  3. <el-form-item label="名字" prop="name">
  4. <el-input v-model="formData.name"></el-input>
  5. </el-form-item>
  6. <el-form-item label="年龄" prop="age">
  7. <el-input v-model="formData.age"></el-input>
  8. </el-form-item>
  9. <el-form-item label="银行卡号" key="1" v-if="formData.age >= 18" prop="bankCardNo">
  10. <el-input v-model="formData.bankCardNo"></el-input>
  11. </el-form-item>
  12. <el-form-item label="银行卡号" key="2" v-else>
  13. <el-input v-model="formData.bankCardNo"></el-input>
  14. </el-form-item>
  15. </el-form>
  16. </template>
  17. <script>
  18. export default {
  19. name: 'HelloWorld',
  20. data () {
  21. return {
  22. rules: {
  23. name: [{ required: true, message: '请输入名字', trigger: 'blur' }],
  24. age: [{ required: true, message: '请输入年龄', trigger: 'blur' }],
  25. bankCardNo: [{ required: true, message: '请输入银行卡号', trigger: 'blur' }]
  26. },
  27. formData: {
  28. name: '',
  29. age: '',
  30. bankCardNo: ''
  31. }
  32. }
  33. }
  34. }
  35. </script>

最终效果如图,当年龄大于等于18时就会触发验证器。

2020年4月3日更新

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

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

  1. watch: {
  2. 'formData.age': {
  3. immediate: true,
  4. handler (newVal) {
  5. if (newVal >= 18) {
  6. this.addRule('bankCardNo', [{ required: true, message: '请输入银行卡号', trigger: 'blur' }])
  7. } else {
  8. this.addRule('bankCardNo', [])
  9. }
  10. }
  11. }
  12. }
  13. data () {
  14. return {
  15. rules: {}
  16. }
  17. },
  18. methods: {
  19. addRule (prop, rule) {
  20. this.$set(this.rules, prop, rule)
  21. }
  22. }

2. form-item保留prop属性,根据条件动态修改form-item组件的rules属性。比如下面两种方式,因为rules属性接受数据和对象

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

 

六、总结

还需要多多了解对elementUI组件的一些底层原理,虽然看过不少有关vue框架底层原理和diff算法的资料,但在实际开发过程中容易忽略这些知识点,还是需要多踩坑多学习

《深入了解Element Form表单动态验证问题》https://blog.csdn.net/sinat_36521655/article/details/105294419

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/不正经/article/detail/472380
推荐阅读
相关标签
  

闽ICP备14008679号