el-form resetFields無效和validate無效的可能原因及解決方法
問題導入
在使用 el-form
過程中,尤其是表單驗證這一塊,官方提供的 reserFields
方法以及驗證方面存在一些坑,在此記錄一下,給大傢提供可能的解決辦法。
簡單實例
1、官方案例
先來看看官方提供的案例,對應的官網 表單驗證
對應的代碼
<el-form :model="ruleForm" :rules="rules" ref="ruleForm" label-width="100px" class="demo-ruleForm"> <el-form-item label="活動名稱" prop="name"> <el-input v-model="ruleForm.name"></el-input> </el-form-item> <el-form-item label="活動區域" prop="region"> <el-select v-model="ruleForm.region" placeholder="請選擇活動區域"> <el-option label="區域一" value="shanghai"></el-option> <el-option label="區域二" value="beijing"></el-option> </el-select> </el-form-item> <el-form-item label="活動時間" required> <el-col :span="11"> <el-form-item prop="date1"> <el-date-picker type="date" placeholder="選擇日期" v-model="ruleForm.date1" style="width: 100%;"></el-date-picker> </el-form-item> </el-col> <el-col class="line" :span="2">-</el-col> <el-col :span="11"> <el-form-item prop="date2"> <el-time-picker placeholder="選擇時間" v-model="ruleForm.date2" style="width: 100%;"></el-time-picker> </el-form-item> </el-col> </el-form-item> <el-form-item label="即時配送" prop="delivery"> <el-switch v-model="ruleForm.delivery"></el-switch> </el-form-item> <el-form-item label="活動性質" prop="type"> <el-checkbox-group v-model="ruleForm.type"> <el-checkbox label="美食/餐廳線上活動" name="type"></el-checkbox> <el-checkbox label="地推活動" name="type"></el-checkbox> <el-checkbox label="線下主題活動" name="type"></el-checkbox> <el-checkbox label="單純品牌曝光" name="type"></el-checkbox> </el-checkbox-group> </el-form-item> <el-form-item label="特殊資源" prop="resource"> <el-radio-group v-model="ruleForm.resource"> <el-radio label="線上品牌商贊助"></el-radio> <el-radio label="線下場地免費"></el-radio> </el-radio-group> </el-form-item> <el-form-item label="活動形式" prop="desc"> <el-input type="textarea" v-model="ruleForm.desc"></el-input> </el-form-item> <el-form-item> <el-button type="primary" @click="submitForm('ruleForm')">立即創建</el-button> <el-button @click="resetForm('ruleForm')">重置</el-button> </el-form-item> </el-form> <script> export default { data() { return { ruleForm: { name: '', region: '', date1: '', date2: '', delivery: false, type: [], resource: '', desc: '' }, rules: { name: [ { required: true, message: '請輸入活動名稱', trigger: 'blur' }, { min: 3, max: 5, message: '長度在 3 到 5 個字符', trigger: 'blur' } ], region: [ { required: true, message: '請選擇活動區域', trigger: 'change' } ], date1: [ { type: 'date', required: true, message: '請選擇日期', trigger: 'change' } ], date2: [ { type: 'date', required: true, message: '請選擇時間', trigger: 'change' } ], type: [ { type: 'array', required: true, message: '請至少選擇一個活動性質', trigger: 'change' } ], resource: [ { required: true, message: '請選擇活動資源', trigger: 'change' } ], desc: [ { required: true, message: '請填寫活動形式', trigger: 'blur' } ] } }; }, 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(); } } } </script>
對應的字段含義,在官網上都有詳細的解釋,那麼如果類似的表單重置 resetFields
時候無法成功,那麼可能的原因有哪些呢?(以下皆由個人實踐得出,具體原因有些不太懂,歡迎解釋補充)
resetFields 失敗可能原因
1:
el-form
中使用 v-model="ruleForm"
代替瞭:model="ruleForm"
,正確的應為後者。
2:
el-form-item
中的 prop
屬性設置錯誤,官網給出瞭解釋。即 prop=a
, v-model=Form.a
,Js中表單數據字段Form:{ a:'', b: [] }
,需要一一對應,不能出現差錯。
3:
<el-button @click="resetForm('ruleForm')">重置</el-button>
resetForm(formName)的參數一定要和 el-form
中 ref ="formName"
一致。
2、個人案例
項目開發過程中,通常 el-form
都是嵌套在el-dialog
中。在表格頁面中,點擊添加或者修改按鈕,彈出同一個表單,因為兩個操作都是類似的組件,不同的組件可以用 v-if
屏蔽掉,這也是大多數開發中公用組件的一個體現。
舉個個例:
通過添加按鈕和編輯按鈕都能夠打開這個dialog。
EditRole.vue
<template> <div style="text-align: left;"> <el-button class="add-button" type="success" @click="dialogFormVisible = true">添加角色</el-button> <el-dialog :title="title" :visible.sync="dialogFormVisible" @close="clear" > <el-form :model="form" ref="roleForm" :rules="rules"> <el-form-item label="角色名稱" :label-width="formLabelWidth" prop="name"> <el-input v-model="form.name" autocomplete="off" placeholder="請輸入新角色英文名稱"></el-input> </el-form-item> <el-row> <el-col :span="16"> <el-form-item label="中文名稱" :label-width="formLabelWidth" prop="nameZh"> <el-input v-model="form.nameZh" autocomplete="off" placeholder="請輸入中文名稱"></el-input> </el-form-item> </el-col> <el-col :span="8"> <el-form-item label="是否可用" :label-width="formLabelWidth" prop="enabled"> <el-switch v-model="form.enabled" :active-value="1" :inactive-value="0" active-color="#13ce66" inactive-color="#ff4949"> </el-switch> </el-form-item> </el-col> </el-row> <el-form-item label="功能簡述" :label-width="formLabelWidth" prop="description"> <el-input v-model="form.description" autocomplete="off" type="textarea" :autosize="{ minRows: 2, maxRows: 4}" placeholder="請輸入角色功能簡述"></el-input> </el-form-item> <el-form-item prop="id" style="height: 0"> <el-input type="hidden" v-model="form.roleId" autocomplete="off"></el-input> </el-form-item> </el-form> <div slot="footer" class="dialog-footer"> <el-button @click="dialogFormVisible = false">取 消</el-button> <el-button type="info" @click="resetForm('form')">重置</el-button> <el-button type="primary" @click="submitForm('roleForm')">確 定</el-button> </div> </el-dialog> </div> </template> <script> export default { name: "EditRole", data () { return { form: { roleId: '', name: '', nameZh: '', enabled: 1, description: '', }, // 表單驗證規則 rules: { name: [ { required: true, message: '請輸入新角色英文名稱', trigger: 'blur' }, ], nameZh: [ { required: true, message: '請輸入角色中文名稱', trigger: 'blur' }, ], description: [ { required: true, message: '請賦予角色簡要的功能描述', trigger: 'blur' } ], }, dialogFormVisible: false, formLabelWidth: '120px', isShow: true, //是否顯示父級菜單,添加的時候顯示,修改的時候不讓修改父級菜單。 title: '添加角色', } }, methods: { clear () { if (this.form.id !== '') { // 不為空,那麼就是修改操作 this.isShow = false; this.title = '修改角色' } // 關閉彈框後清空數據, this.resetForm('roleForm') this.isShow = true; this.title = '添加角色' }, resetForm(formName) { // 重置表格內容 this.$refs[formName].resetFields(); }, submitForm (formName) { this.$refs[formName].validate((valid) => { if (valid) { ... }else { this.$notify({ type: 'error', message: '數據未填寫完整,請仔細核對!' }) return false; } }) }, } } </script> <style scoped> .add-button { margin: 18px 0 0 10px; } </style>
resetFields失效及解決辦法
先添加後編輯
一般打開這個頁面後,我們測試數據的時候,都是先點擊添加按鈕,關閉dialog後,再點擊編輯按鈕。
此時你使用this.$refs[formName].resetFields();
是能夠將表單重置為空的,因為初始化的時候,本來form數據就是空的。
先編輯後添加
但是若你先點擊編輯按鈕,通過父組件傳遞瞭 本行數據
到 el-dialog 的 el-form
時,退出後再點擊添加按鈕,此時會發現數據被污染,表單竟然不是空的,而是出現瞭剛才編輯那行的數據。圖示:
關閉彈窗時,我們設置瞭clear函數,clear函數為關閉dialog執行的函數。其中調用瞭resetForm
函數,即利用this.$refs[formName].resetFields()
來希望重置數據。但是再點擊添加按鈕時,數據並沒有清除。
這個現象的原因是因為dialog是懶加載的,通過表格中的編輯按鈕打開dialog時,傳過來瞭父組件行數據
。即初始化時數據並不是空的,而resetFields()
函數重置時,初始化的值是啥就是啥
。
瞭解本質後,那麼解決這個問題就簡單瞭,我們關閉彈出時,不使用resetFields()
函數不就行瞭。而是這樣:
resetForm(formName) { console.log("誰要重置數據啊?" + formName) // 重置表格內容 this.form= { roleId: '', name: '', nameZh: '', enabled: 1, description: '', } },
改動後結果:
相當於編輯退出後,將form表單初始化為空,這樣就無法出現污染瞭。但是這樣也有個問題,重置數據後,無法移除校驗效果。好在官方給瞭我們方法,隻需要添加一行。
resetForm(formName) { console.log("誰要重置數據啊?" + formName) // 重置表格內容 this.form= { roleId: '', name: '', nameZh: '', enabled: 1, description: '', }, // 不使用resetFields後,重置時無法移除校驗效果,使用以下函數移除 this.$refs[formName].clearValidate(); },
validate失效及解決辦法
當使用表單驗證的時候,如果正常的驗證都無誤,提交時,我們通過以下代碼來判斷是否驗證成功。
submitForm (formName) { this.$refs[formName].validate((valid) => { if (valid) { ... }else { this.$notify({ type: 'error', message: '數據未填寫完整,請仔細核對!' }) return false; } }) },
但是使用自定義驗證規則是,註意規則中回調函數callback()
一定要寫,官方文檔也寫的十分清楚。否則 if(valid)
條件無法判斷,而且它是沒有報錯的。
如自定義郵箱校驗的時候:
data() { const validateEmail = (rule, value, callback) => { let regEmail = /^([a-zA-Z0-9_-])+@([a-zA-Z0-9_-])+(\.[a-zA-Z0-9_-])+/; if (!regEmail.test(this.form.email)) { return callback(new Error("郵箱格式不正確")); // 有一個if 就要 一個 callback } return callback(); // 返回一個回調函數給驗證,是表單validate生效 }; return { // 表單數據 form: { user_id: '', ... email: '', }, // 表單驗證規則 rules: { ... email: [ { required: true, message: '請輸入真實郵箱地址,方便找回密碼', validator: validateEmail, trigger: 'blur' }, ], }, } },
callback()
回調函數必須要被調用。
這是目前碰到的表單問題,大傢有什麼好的解決方案或者新問題,歡迎在評論區留言,幫助大傢解決問題!
3、model is required for validate to work!
2021/4/21
今天給表的表單添加驗證規則的時候,el-form-item
都配置好瞭prop
屬性,但是無效,F12後給出一個警告
然後回去看el-form
時候,才發現 v-model =
沒有改成 :model
,造成錯誤,在此更新一下。
結尾
到此這篇關於el-form resetFields無效和validate無效的可能原因及解決方法的文章就介紹到這瞭,更多相關el-form resetFields無效和validate無效內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- vue表單驗證rules及validator驗證器的使用方法實例
- Vue之ElementUI Form表單校驗
- element ui提交表單返回成功後自動清空表單的值的實現代碼
- 使用ElementUI循環生成的Form表單添加校驗
- v-for中動態校驗el-form表單項的實踐