国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

仿ElementUI實現一個Form表單的實現代碼

2019-11-19 11:43:41
字體:
來源:轉載
供稿:網友

使用組件就像流水線上的工人;設計組件就像設計流水線的人,設計好了給工人使用。

完整項目地址:仿 ElementtUI 實現一個 Form 表單

一. 目標

仿 ElementUI 實現一個簡單的 Form 表單,主要實現以下四點:

  • Form
  • FormItem
  • Input
  • 表單驗證

我們先看一下 ElementUI 中 Form 表單的基本用法

<el-form :model="ruleForm" :rules="rules" ref="loginForm">     <el-form-item label="用戶名" prop="name">    	<el-input v-model="ruleForm.name"></el-input>   </el-form-item>     <el-form-item label="密碼" prop="pwd">    	<el-input v-model="ruleForm.pwd"></el-input>   </el-form-item>     <el-form-item>    	<el-button type="primary" @click="submitForm('loginForm')">登錄</el-button>   </el-form-item>  </el-form>

在 ElementUI 的表單中,主要進行了 3 層嵌套關系, Form 是最外面一層, FormItem 是中間一層,最內層是 Input 或者 Button 。

二. 創建項目

我們通過 Vue CLI 3.x 創建項目。

使用 vue create e-form 創建一個目錄。

使用 npm run serve 啟動項目。

三. Form 組件設計

ElementUI 中的表單叫做 el-form ,我們設計的表單就叫 e-form 。

為了實現 e-form 表單,我們參考 ElementUI 的表單用法,總結出以下我們需要設計的功能。

  • e-form 負責全局校驗,并提供插槽;
  • e-form-item 負責單一項校驗及顯示錯誤信息,并提供插槽;
  • e-input 負責數據雙向綁定;

1. Input 的設計

我們首先觀察一下 ElementUI 中的 Input 組件:

<el-input v-model="ruleForm.name"></el-input>

在上面的代碼中,我們發現 input 標簽可以實現一個雙向數據綁定,而實現雙向數據綁定需要我們在 input 標簽上做兩件事。

  • 要綁定 value
  • 要響應 input 事件

當我們完成這兩件事以后,我們就可以完成一個 v-model 的語法糖了。

我們創建一個 Input.vue 文件:

<template> <div>  <!-- 1. 綁定 value   		 2. 響應 input 事件		-->  <input type="text" :value="valueInInput" @input="handleInput"> </div></template><script>export default { name: "EInput", props: {  value: { // 解釋一   type: String,   default: '',  } }, data() {  return {   valueInInput: this.value // 解釋二  }; }, methods: {   handleInput(event) {     this.valueInInput = event.target.value; // 解釋三    	this.$emit('input', this.valueInInput); // 解釋四   } },};</script>

我們對上面的代碼做一點解釋:

**解釋一:**既然我們想做一個 Input 組件,那么接收的值必然是父組件傳進來的,并且當父組件沒有傳進來值的時候,我們可以它一個默認值 "" 。

**解釋二:**我們在設計組件的時候,要遵循單向數據流的原則:父組件傳進來的值,我們只能用,不能改。那么將父組件傳進來的值進行一個賦值操作,賦值給 Input 組件內部的 valueInInput ,如果這個值發生變動,我們就修改內部的值 valueInInput 。這樣我們既可以處理數據的變動,又不會直接修改父組件傳進來的值。

**解釋三:**當 Input 中的值發生變動時,觸發 @input 事件,此時我們通過 event.target.value 獲取到變化后的值,將它重新賦值給內部的 valueInInput 。

**解釋四:**完成了內部賦值之后,我們需要做的就是將變化后的值通知父組件,這里我們用 this.$emit 向上派發事件。其中第一個參數為事件名,第二個參數為變化的值。

完成了以上四步,一個實現了雙向數據綁定的簡單的 Input 組件就設計完成了。此時我們可以在 App.vue 中引入 Input 組件觀察一下結果。

<template> <div id="app">  <e-input v-model="initValue"></e-input>  <div>{{ initValue }}</div> </div></template><script>import EInput from './components/Input.vue';export default { name: "app", components: {  EInput }, data() {  return {   initValue: '223',  }; },};</script>

 

2. FormItem 的設計

<el-form-item label="用戶名" prop="name">		<el-input v-model="ruleForm.name"></el-input></el-form-item>

在 ElementUI 的 formItem 中,我們可以看到:

  1. 需要 label 來顯示名稱;
  2. 需要 prop 來校驗當前項;
  3. 需要給 inputbutton 預留插槽;

根據上面的需求,我們可以創建出自己的 formItem ,新建一個 FormItem.vue 文件 。

<template>  <div>    <!-- 解釋一 -->    <label v-if="label">{{ label }}</label>    <div>      <!-- 解釋二 -->      <slot></slot>      <!-- 解釋三 -->      <p v-if="validateState === 'error'" class="error">{{ validateMessage }}</p>    </div>  </div></template><script>  export default {    name: "EFormItem",    props: {   			label: { type: String, default: '' },   			prop: { type: String, default: '' } 			},    data() {      return {        validateState: '',        validateMessage: ''      }    },  }</script><style scoped>.error {  color: red;}</style>

和上面一樣,我們接著對上面的代碼進行一些解釋:

**解釋一:**根據 ElementUI 中的用法,我們知道 label 是父組件傳來,且當傳入時我們展示,不傳入時不展示。

解釋二: slot 是一個預留的槽位,我們可以在其中放入 input 或其他組件、元素。

解釋三: p 標簽是用來展示錯誤信息的,如果驗證狀態為 error 時,就顯示。

此時,我們的 FormItem 組件也可以使用了。同樣,我們在 App.vue 中引入該組件。

<template> <div id="app">    <e-form-item label="用戶名" prop="name">   	<e-input v-model="ruleForm.name"></e-input>  </e-form-item>    <e-form-item label="密碼" prop="pwd">   	<e-input v-model="ruleForm.pwd"></e-input>  </e-form-item>    <div>   {{ ruleForm }}  </div>   </div></template><script>import EInput from './components/Input.vue';import EFormItem from './components/FormItem.vue';export default { name: "app", components: {  EInput,  EFormItem }, data() {  return {   ruleForm: {    name: '',    pwd: '',   },  }; },};</script>

 

3. Form 的設計

到現在,我們已經完成了最內部的 input 以及中間層的 FormItem 的設計,現在我們開始設計最外層的 Form 組件。

當層級過多并且組件間需要進行數據傳遞時,Vue 為我們提供了 provideinject API,方便我們跨層級傳遞數據。

我們舉個例子來簡單實現一下 provideinject 。在 App.vue 中,我們提供數據(provide)。

export default { name: "app", provide() {  return {   msg: '哥是最外層提供的數據'  } }};</script>

接著,我們在最內層的 Input.vue 中注入數據,觀察結果。

<template> <div>  <!-- 1、綁定 value   2、響應 input 事件-->  <input type="text" :value="valueInInput" @input="handleInput">  <div>{{ msg }}</div> </div></template><script>export default { name: "EInput", inject: [ 'msg' ], props: {  value: {   type: String,   default: '',  } }, data() {  return {   valueInInput: this.value  }; }, methods: {   handleInput(event) {     this.valueInInput = event.target.value;    	this.$emit('input', this.valueInInput);   } },};</script>

根據上圖,我們可以看到無論跨越多少層級, provideinject 可以非常方便的實現數據的傳遞。

理解了上面的知識點后,我們可以開始設計 Form 組件了。

<el-form :model="ruleForm" :rules="rules" ref="loginForm">	</el-form>

根據 ElementUI 中表單的用法,我們知道 Form 組件需要實現以下功能:

  • 提供數據模型 model;
  • 提供校驗規則 rules;
  • 提供槽位,里面放我們的 FormItem 等組件;

根據上面的需求,我們創建一個 Form.vue 組件:

<template>  <form>    <slot></slot>  </form></template><script>  export default {    name: 'EForm',    props: { // 解釋一      model: {        type: Object,        required: true      },      rules: {        type: Object      }    },    provide() { // 解釋二      return {        eForm: this // 解釋三      }    }  }</script>

解釋一:該組件需要用戶傳遞進來一個數據模型 model 進來,類型為 Object 。 rules 為可傳項。

解釋二:為了讓各個層級都能使用 Form 中的數據,需要依靠 provide 函數提供數據。

解釋三:直接將組件的實例傳遞下去。

完成了 Form 組件的設計,我們在 App.vue 中使用一下:

<template> <div id="app">    <e-form :model="ruleForm" :rules="rules">      <e-form-item label="用戶名" prop="name">    <e-input v-model="ruleForm.name"></e-input>   </e-form-item>      <e-form-item label="密碼" prop="pwd">    <e-input v-model="ruleForm.pwd"></e-input>   </e-form-item>      <e-form-item>    <button>提交</button>   </e-form-item>   	</e-form> </div></template><script>import EInput from './components/Input.vue';import EFormItem from './components/FormItem.vue';import EForm from "./components/Form";export default { name: "app", components: {  EInput,  EFormItem,  EForm }, data() {  return {   ruleForm: {    name: '',    pwd: '',   },   rules: {    name: [{ required: true }],    pwd: [{ required: true }]   },  }; },};</script>

到目前為止,我們的基本功能就已經實現了,除了提交與驗證規則外,所有的組件幾乎與 ElementUI 中的表單一模一樣了。下面我們就開始實現校驗功能。

4. 設計校驗規則

在上面設計的組件中,我們知道校驗當前項和展示錯誤信息的工作是在 FormItem 組件中,但是數據的變化是在 Input 組件中,所以 FormItemInput 組件是有數據傳遞的。當 Input 中的數據變化時,要告訴 FormItem ,讓 FormItem 進行校驗,并展示錯誤。

首先,我們修改一下 Input 組件:

methods: {  handlerInput(event) {   this.valueInInput = event.target.value;   this.$emit("input", this.valueInInput);      // 數據變了,定向通知 FormItem 校驗   this.dispatch('EFormItem', 'validate', this.valueInput);  },		// 查找指定 name 的組件,  dispatch(componentName, eventName, params) {   var parent = this.$parent || this.$root;   var name = parent.$options.name;   while (parent && (!name || name !== componentName)) {    parent = parent.$parent;    if (parent) {     name = parent.$options.name;    }   }   if (parent) {    parent.$emit.apply(parent, [eventName].concat(params));   }  } }

這里,我們不能用 this.$emit 直接派發事件,因為在 FormItem 組件中, Input 組件的位置只是一個插槽,無法做事件監聽,所以此時我們讓 FormItem 自己派發事件,并自己監聽。修改 FormItem 組件,在 created 中監聽該事件。

created() {	this.$on('validate', this.validate);}

Input 組件中的數據變化時, FormItem 組件監聽到 validate 事件后,執行 validate 函數。

下面,我們就要處理我們的 validate 函數了。而在 ElementUI 中,驗證用到了一個底層庫async-validator,我們可以通過 npm 安裝這個包。

npm i async-validator

async-validator 是一個可以對數據進行異步校驗的庫,具體的用法可以參考上面的鏈接。我們通過這個庫來完成我們的 validate 函數。繼續看 FormItem.vue 這個文件:

<template> <div>  <label v-if="label">{{ label }}</label>  <div>   <slot></slot>   <p v-if="validateState === 'error' " class="error">{{ validateMessage }}</p>  </div> </div></template><script>import AsyncValidator from "async-validator";export default { name: "EFormItem", props: {			label: { type: String, default: '' },			prop: { type: String, default: '' } }, inject: ["eForm"], // 解釋一 created() {  this.$on("validate", this.validate); }, mounted() { // 解釋二  if (this.prop) { // 解釋三   this.dispatch('EForm', 'addFiled', this);  } }, data() {  return {   validateMessage: "",   validateState: ""  }; }, methods: {  validate() {    // 解釋四   return new Promise(resolve => {    // 解釋五    const descriptor = {     // name: this.form.rules.name =>     // name: [ { require: true }, { ... } ]    };    descriptor[this.prop] = this.eForm.rules[this.prop];    // 校驗器    const validator = new AsyncValidator(descriptor);    const model = {};    model[this.prop] = this.eForm.model[this.prop];    // 異步校驗    validator.validate(model, errors => {     if (errors) {      this.validateState = "error";      this.validateMessage = errors[0].message;      resolve(false);     } else {      this.validateState = "";      this.validateMessage = "";      resolve(true);     }    });   });  },  // 查找上級指定名稱的組件  dispatch(componentName, eventName, params) {   var parent = this.$parent || this.$root;   var name = parent.$options.name;   while (parent && (!name || name !== componentName)) {    parent = parent.$parent;    if (parent) {     name = parent.$options.name;    }   }   if (parent) {    parent.$emit.apply(parent, [eventName].concat(params));   }  } }};</script><style scoped>.error { color: red;}</style>

我們對上面的代碼做一個解釋。

解釋一:注入 Form 組件提供的數據 - Form 組件的實例,下面就可以使用 this.eForm.xxx 來使用 Form 中的數據了。

解釋二:因為我們需要在 Form 組件中校驗所有的 FormItem ,所以當 FormItem 掛載完成后,需要派發一個事件告訴 Form :你可以校驗我了。

解釋三:當 FormItem 中有 prop 屬性的時候才校驗,沒有的時候不校驗。比如提交按鈕就不需要校驗。

<e-form-item>		<input type="submit" @click="submitForm()" value="提交"></e-form-item>

**解釋四:**返回一個 promise 對象,批量處理所有異步校驗的結果。

解釋五: descriptor 對象是 async-validator 的用法,采用鍵值對的形式,用來檢查當前項。比如:

// 檢查當前項// async-validator 給出的例子name: {		type: "string",		required: true,		validator: (rule, value) => value === 'muji',}

FormItem 中檢查當前項完成了,現在我們需要處理一下 Form 組件中的全局校驗。表單提交時,需要對 form 進行一個全局校驗。大致的思路是:循環遍歷表單中的所有派發上來的 FormItem ,讓每一個 FormItem 執行自己的校驗函數,如果有一個為 false ,則校驗不通過;否則,校驗通過。我們通過代碼實現一下:

<template> <form>  <slot></slot> </form></template><script>  export default {    props: {      model: { type: Object, required: true },      rules: { type: Object }    },    provide() {     return {       eForm: this, // provide this component's instance     }     },	   data() {      return {        fileds: [],      }    },    created() {      // 解釋一     	this.fileds = [];      this.$on('addFiled', filed => this.fileds.push(filed));    },    methods: {      async validate(cb) { // 解釋二        // 解釋三        const eachFiledResultArray = this.fileds.map(filed => filed.validate());        // 解釋四        const results = await Promise.all(eachFiledResultArray);        let ret = true;        results.forEach(valid => {          if (!valid) {            ret = false;          }        });        cb(ret);      }    },  }</script><style lang="scss" scoped></style>

解釋一:用 fileds 緩存需要校驗的表單項,因為我們在 FormItem 中派發了事件。只有需要校驗的 FormItem 會被派發到這里,而且都會保存在數組中。

if (this.prop) {   this.dispatch('EForm', 'addFiled', this);}

解釋二:當點擊提交按鈕時,會觸發這個事件。

解釋三:遍歷所有被添加到 fileds 中的 FormItem 項,讓每一項單獨去驗證,會返回 Promise 的 truefalse 。將所有的結果,放在一個數組 eachFiledResultArray 中。

解釋四:獲取所有的結果,統一進行處理,其中有一個結果為 false ,驗證就不能通過。

至此,一個最簡化版本的仿 ElementUI 的表單就實現了。

 

四. 總結

當然上面的代碼還有很多可以優化的地方,比如說 dispatch 函數,我們可以寫一遍,使用的時候用 mixin 導入。由于篇幅關系,這里就不做處理了。

通過這次實現,我們首先總結一下其中所涉及的知識點。

  • 父組件傳遞給子組件用 props
  • 子組件派發事件,用 $emit
  • 跨層級數據交互,用 provide 和 inject
  • 用 slot 可以預留插槽

其次是一些思想:

  • 單項數據流:父組件傳遞給子組件的值,子組件內部只能用,不能修改。
  • 組件內部的 name 屬性,可以通過 this.$parent.$options.name 查找。
  • 想要批量處理很多異步的結果,可以用 promise 對象。

最后,文章會首先發布在我的 Github ,以及公眾號上,歡迎關注,歡迎 star。

 以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 满洲里市| 滁州市| 蓬莱市| 电白县| 花垣县| 德江县| 青冈县| 什邡市| 怀安县| 吉水县| 和政县| 洞头县| 汨罗市| 苍梧县| 浑源县| 精河县| 永平县| 大理市| 通山县| 平南县| 长垣县| 海林市| 区。| 中西区| 望奎县| 潜山县| 靖边县| 宜都市| 定安县| 台东市| 临汾市| 遂昌县| 浑源县| 大厂| 龙州县| 海门市| 江川县| 雷波县| 柏乡县| 正蓝旗| 龙泉市|