Vue自定義組件實現(xiàn)?v-model?的幾種方式
前言
在 Vue 中,v-model 是一個常用的指令,用于實現(xiàn)表單元素和組件之間的雙向綁定。當我們使用原生的表單元素時,直接使用 v-model 是很方便的,但是對于自定義組件來說,要實現(xiàn)類似的雙向綁定功能就需要一些額外的處理。
本篇文章將介紹幾種在自定義組件中實現(xiàn) v-model 的方式,主要如下:
使用
v-model
屬性:適用于表單元素定義
model
屬性:適用于非表單元素
除了以上的兩種方式外,還有我們通常使用的.sync 修飾符(2.3.0+ 新增),主要區(qū)別在于使用方式的不同,前者直接使用 v-model,后者使用 .sync
修飾符進行綁定
一. 單向數(shù)據(jù)流
所有的 prop 都使得其父子 prop 之間形成了一個單向下行綁定:父級 prop 的更新會向下流動到子組件中,但是反過來則不行。這樣會防止從子組件意外變更父級組件的狀態(tài),從而導致你的應用的數(shù)據(jù)流向難以理解。
注意:每次父級組件發(fā)生變更時,子組件中所有的 prop 都將會刷新為最新的值。這意味著你不應該在一個子組件內(nèi)部改變 prop。雖然可以,但是如果你這樣做了,Vue 會在瀏覽器的控制臺中發(fā)出警告。
二. 基礎原理淺析
v-model 實際上就是 props:value
和 $emit('input')
的組合語法糖,簡單來說,v-model 的使用其實做了兩個比較重要的操作,理解這兩個操作,我們就可以輕松實現(xiàn)組件的自定義 v-model
v-bind 綁定 value 屬性的值 - props:value;
v-on 綁定 input 事件監(jiān)聽到函數(shù)中,函數(shù)會獲取最新的值賦值到綁定的屬性中 - $emit('input');
三. 實現(xiàn) v-model 的兩種方式
1. 直接使用 v-model 屬性
以 input 表單元素為例,在 vue 中,我們可以直接使用 v-model 進行綁定數(shù)據(jù),當我們在實現(xiàn)自定義組件custom-component
進行封裝 input 時,我們的組件對外暴露時也需要使用 v-model,看下面我們應該如何實現(xiàn):
在自定義組件中,我們可以通過在組件內(nèi)部使用 value 屬性和手動觸發(fā) input 事件來實現(xiàn) v-model 的雙向綁定效果。具體實現(xiàn)如下:
<template> <input v-model="newValue" /> </template> <script> export default { props: { value: { type: Boolean, default: false, }, }, computed: { newValue: { get() { return this.value; }, set(val) { this.$emit("input", val); }, }, }, }; </script>
使用custom-component
<custom-component v-model="newValue" />
在上述代碼中,我們通過 value 屬性接收父組件傳入的值,并且使用 computed 的 set 屬性用來監(jiān)聽值的更新,手動觸發(fā) $emit('input', value)
來將新值傳遞給父組件,從而實現(xiàn)雙向綁定的效果。
2. 定義 model 屬性
Vue 允許我們在定義組件中通過定義 model 屬性來簡化 v-model 的使用。通過定義 model 屬性,我們可以指定組件中哪個屬性的值應該作 v-model 的值。
假如我們實現(xiàn)一個計數(shù)器的組件custom-counter
,在頁面中顯示兩個按鈕,點擊按鈕可以進行數(shù)值的加減操作,具體示例如下:
<template> <div> <button @click="increment">+</button> <span>{{ value }}</span> <button @click="decrement">-</button> </div> </template> <script> export default { // 當 model 為默認值時,可以將其省略 model: { prop: "value", // 默認是 value event: "input", // 默認是 input }, props: { value: { type: Number, default: 0, }, }, methods: { increment() { this.$emit("input", this.value + 1); }, decrement() { this.$emit("input", this.value - 1); }, }, }; </script>
提示:當 model 為默認值時,可以將其省略
父組件中使用custom-counter
<template> <div> <custom-counter v-model="count"></custom-counter> <p>計數(shù)器的值為:{{ count }}</p> </div> </template> <script> import CustomCounter from "./CustomCounter.vue"; export default { components: { CustomCounter, }, data() { return { count: 0, }; }, }; </script>
演示效果如下圖所示:
在上面的示例中,CustomCounter
組件接收一個value
屬性來接收父組件傳遞的值,并在點擊按鈕時修改value
屬性的值。通過調(diào)用this.$emit('input', newValue)
觸發(fā)input
事件,將新的value
值傳遞給父組件進行更新。
在父組件中,使用v-model
將父組件中的count
屬性綁定到CustomCounter
組件的value
屬性上,綁定 input 事件監(jiān)聽到函數(shù)中,從而實現(xiàn)了數(shù)據(jù)的雙向綁定。
model 是允許 vue 自定義組件使用 v-model 的關鍵,雖然有時我們不顯性的使用它,也不影響我們在自定義組件中使用 v-model 指令,這只是因為被設置默認值。而有的時候,顯示的使用,并自定義 model 的 prop 和 event 會有益。
注意:允許一個自定義組件在使用 v-model 時定制 prop 和 event。默認情況下,一個組件上的 v-model 會把 value 用作 prop 且把 input 用作 event,但是一些輸入類型比如單選框和復選框按鈕可能想使用 value prop 來達到不同的目的。使用 model 選項可以回避這些情況產(chǎn)生的沖突。
四. .sync 修飾符
在有些情況下,我們可能需要對一個 prop 進行 “雙向綁定”。但是,真正的雙向綁定會帶來維護上的問題,因為子組件可以變更父組件,且在父組件和子組件兩側(cè)都沒有明顯的變更來源。
這也是為什么我們推薦以 update:myPropName
的模式觸發(fā)事件取而代之。
舉個例子,還是上面的計數(shù)器的例子,在一個包含 count
prop 的計數(shù)組件中,我們可以用以下方法表達對其賦新值的意圖:
this.$emit("update:count", newCount);
然后父組件可以監(jiān)聽那個事件并根據(jù)需要更新一個本地的數(shù)據(jù) property。例如:
<custom-counter v-bind:count="count" v-on:update:count="count = $event" />
我們?yōu)檫@種模式提供一個縮寫,即 .sync 修飾符,如下:
<custom-counter :count.sync="count"></custom-counter>
以上的兩種寫法是等價的
注意:帶有 .sync 修飾符的 v-bind 不能和表達式一起使用 (例如 v-bind:count.sync=”count + 1” 是無效的)。取而代之的是,你只能提供你想要綁定的 property 名,類似 v-model。
五. 總結(jié)
本篇文章介紹了在 Vue 自定義組中實現(xiàn) v-model 的幾種方式,包括定義 model 屬性和使用 v-model 屬性。通過以上案例實際分析,相信大家都已經(jīng)了解這幾種方式的創(chuàng)建以及應用場景,以便更加靈活地應用到實際項目中。
以上面兩種方式為例,我們可以靈活地根據(jù)自己的需求選擇合適的方式來實現(xiàn)自定義組件 v-model。如果是表單元素,可以直接使用 v-model 屬性;如果是非表單元素,可以通過定義 model 屬性指定組件中哪個屬性的值應該作 v-model 的值。
除此之外,使用 .sync
修飾符也是進行自定義組件雙向綁定的優(yōu)秀選擇,因此在實際開發(fā)中,選擇適合自己項目需求的方式是最重要的。
以上就是Vue自定義組件實現(xiàn) v-model 的幾種方式的詳細內(nèi)容,更多關于Vue實現(xiàn)v-model的資料請關注腳本之家其它相關文章!
相關文章
在?Vue?項目中如何引用?assets?文件夾中的圖片(方式匯總)
Vue中引用assets文件夾中的圖片有多種方式,在.vue文件的模板部分,使用相對路徑或通過@別名引用圖片,在CSS中,通過相對路徑或@別名引用圖片作為背景,在JavaScript中,通過import語句導入圖片資源,并使用v-bind在模板中綁定顯示,這些方法均可有效管理和引用項目中的圖片資源2024-09-09Vue-cli assets SubDirectory及PublicPath區(qū)別詳解
這篇文章主要介紹了Vue-cli assets SubDirectory及PublicPath區(qū)別詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-08-08vuex狀態(tài)管理數(shù)據(jù)狀態(tài)查詢與更改方式
這篇文章主要介紹了vuex狀態(tài)管理數(shù)據(jù)狀態(tài)查詢與更改方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-04-04vue3+ts+element-plus實際開發(fā)之統(tǒng)一調(diào)用彈窗封裝的詳細過程
這篇文章主要介紹了vue3+ts+element-plus實際開發(fā)之統(tǒng)一調(diào)用彈窗封裝的詳細過程,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-03-03