preface
Recently, I have wrapped some components myself and have nothing to do, just like looking at the element-UI source code. By the way, I only focus on the logic level, and I have time to look at the style processing behind
The main content
The elder-UI verification is implemented by executing addField and removeField operations in the elder-Form created declaration cycle. Listen for events emitted by the current instance’s $emit method
// el-form created() { this.$on('el.form.addField', (field) => { if (field) { this.fields.push(field); }}); /* istanbul ignore next */ this.$on('el.form.removeField', (field) => { if (field.prop) { this.fields.splice(this.fields.indexOf(field), 1); }}); },Copy the code
// Emit mounted() {if (this.prop) {this.dispatch('ElForm', 'el.form.addfield ', [this]); // Emit mounted() {if (this.prop) {this.dispatch('ElForm',' el.form.addfield ', [this]); let initialValue = this.fieldValue; if (Array.isArray(initialValue)) { initialValue = [].concat(initialValue); } Object.defineProperty(this, 'initialValue', { value: initialValue }); this.addValidateEvents(); }},Copy the code
The dispatch method in el-form-item mainly uses an encapsulated dispatch method to distribute custom events on specified components. In the relevant source emitter. this.dispatch(‘ElForm’, ‘el.form.addField’, [this]); This means sending an el.form.addField event on the el-From component whose value is the current component instance, so the fields in the El-Form component are all instances of the el-Form-item component
submitForm(formName) { this.$refs[formName].validate((valid) => { if (valid) { alert('submit! '); } else { console.log('error submit!! '); return false; }}); },Copy the code
At this point, look at the implementation of the validate method
// el-form validate(callback) { if (! this.model) { console.warn('[Element Warn][Form]model is required for validate to work! '); return; } let promise; // if no callback, return promise if (typeof callback ! == 'function' && window.Promise) { promise = new window.Promise((resolve, reject) => { callback = function(valid) { valid ? resolve(valid) : reject(valid); }; }); } let valid = true; let count = 0; Callback if (this.fields. Length === 0 && callback) {callback(true); } let invalidFields = {}; this.fields.forEach(field => { field.validate('', (message, field) => { if (message) { valid = false; } invalidFields = objectAssign({}, invalidFields, field); if (typeof callback === 'function' && ++count === this.fields.length) { callback(valid, invalidFields); }}); }); if (promise) { return promise; }},Copy the code
// el-form-item validate(trigger, callback = noop) { this.validateDisabled = false; const rules = this.getFilteredRule(trigger); if ((! rules || rules.length === 0) && this.required === undefined) { callback(); return true; } this.validateState = 'validating'; const descriptor = {}; if (rules && rules.length > 0) { rules.forEach(rule => { delete rule.trigger; }); } descriptor[this.prop] = rules; const validator = new AsyncValidator(descriptor); const model = {}; model[this.prop] = this.fieldValue; validator.validate(model, { firstFields: true }, (errors, invalidFields) => { console.log(model) this.validateState = ! errors ? 'success' : 'error'; this.validateMessage = errors ? errors[0].message : ''; callback(this.validateMessage, invalidFields); this.elForm && this.elForm.$emit('validate', this.prop, ! errors, this.validateMessage || null); }); },Copy the code
Validate can be passed as a callback, or if it is not passed as a promise, which declares a valid value true. If the el-form-item validate method throws a message indicating that the validation fails, the valid value is set to false and the validation fails. If the el-form-item validate method throws a message indicating that the validation fails, the valid value is set to false and the validation fails. Here we focus on the validate method of the el-form-item sub-component. This method mainly verifies all rules, ignoring the value of rules trigger. If there are rules, it will directly pass. And then we put every el-form-item prop in descriptor, and then we do new AsyncValidator(Descriptor); Model [this.prop] = this.fieldValue; This. FieldValue gets the value of each field in the form and validates it with the validate method of the AsyncValidator library
// computed form() { let parent = this.$parent; let parentName = parent.$options.componentName; while (parentName ! == 'ElForm') { if (parentName === 'ElFormItem') { this.isNested = true; } parent = parent.$parent; parentName = parent.$options.componentName; } return parent; }, fieldValue() { const model = this.form.model; if (! model || ! this.prop) { return; } let path = this.prop; if (path.indexOf(':') ! == -1) { path = path.replace(/:/, '.'); } return getPropByPath(model, path, true).v; },Copy the code
// util.js export function getPropByPath(obj, path, strict) { let tempObj = obj; path = path.replace(/[(\w+)]/g, ".$1"); path = path.replace(/^./, ""); let keyArr = path.split("."); let i = 0; for (let len = keyArr.length; i < len - 1; ++i) { if (! tempObj && ! strict) break; let key = keyArr[i]; if (key in tempObj) { tempObj = tempObj[key]; } else { if (strict) { throw new Error("please transfer a valid prop path to form item!" ); } break; } } return { o: tempObj, k: keyArr[i], v: tempObj ? tempObj[keyArr[i]] : null, }; }Copy the code
You can see that you get the field of the model object of the el-Form component, and then if you see prop A:B:C, convert it to A,B,C, and call return getPropByPath(Model, Path, true).v; The getPropByPath method is basically just an operation that fetches the value of each field in the model object, and there are two cases. The first case is, if prop is a normal key, it fetches it, and the final format looks something like this
Return {o: {name: ", age: ", address: 'Hz '}, k: 'name', // Common format does not go for loop v: null} return {o: {name: ", age: ', address: 'Hertz'}, k: 'age, don't go / / common format for loop v: return null} {o: {name: ", the age: "', address: 'Hertz'}, k: 'address', // Common format does not go for loop v: 'hz'}Copy the code
If you are adding or subtracting form items dynamically, such as the example in the document
<el-form-item v-for="(domain, Index) in dynamicValidateForm. Domains: "label =" 'domain' + index "is: the key =" domain. The key ": prop = 'domains.'" + index + 'value'" :rules="{required: true, message: 'Domain name cannot be empty ', trigger: 'blur'}" >< el-input v-model="domain.value"></el-input><el-button @click.prevent="removeDomain(domain)"> delete </el-button> </el-form-item>Copy the code
Prop =”‘ ‘domains.’ + index + ‘. Value ‘”, so getPropByPath returns something like this. 1636695934176, value: “”}, {key: 1636695934176, value: [‘domains’, index, ‘value’], and then the props will split into an array of [‘domains’, index, ‘value’]. So the tempObj will be reassigned to [{key: 1636695934176, value: “”}, {key: 1636695934176, value: “”}], and then the for loop will fetch the corresponding index value {key: 1636695934176, value: “”} let len = keyarr.length; i < len – 1; Plus plus I, so I break out of the loop and I return something like this
return {
o: tempObj, // {key: 1636695934176, value: ""}
k: keyArr[i], // 'value'
v: tempObj ? tempObj[keyArr[i]] : null, // ''
};
Copy the code
If it passes the validateMessage, it will be null; otherwise, it will be the value of Message defined in Rules. If message has a value, it will not pass the validateMessage, and the value of Valid will be false
At the end
Here is mainly about the verification function is how to achieve, not very difficult, learning record, there may be some wrong place, behind may be perfect