vue2.x中$attrs的使用方法教程
最近筆者在做大屏項目的時候,由於組件數據傳遞,一層傳遞一層,使用vuex
或者pinia
又顯得過於笨重。故而想起瞭那個傳說中的v-bind="$attrs"
以及v-on="$listeners"
,下面就來聊下使用:
上圖組件之間的關系如下:
-
ComponentGrandParent
為最外層父級組件(爺) -
ComponentParent
為中間層父級組件(父) -
ComponentChild
為子組件
ComponentGrandParent
組件想把props
傳遞給ComponentChild
就通常需要在ComponentParent
中通過屬性一個個的傳遞
//ComponentParent組件 <template> <ComponentChild :propa="prop1" :propb="prop2" :propc="prop3" ... /> </template>
如果需要傳遞的屬性多,而且ComponentParent
中沒有用到的ComponentGrandParent
傳遞過來的屬性的時候,就很尷尬,很不優雅,有時候還需要在寫watch
監聽傳遞過來的數據,然後再賦值給data
中的prop1
,然後再傳遞給ComponentChild
使用$attrs
能解決上述問題,那麼什麼是$attrs
呢?
透傳 Attributes 是指由父組件傳入,且沒有被子組件聲明為props
或是組件自定義事件的 attributes
和事件處理函數。默認情況下,若是單一根節點組件,$attrs
中的所有屬性都是直接自動繼承自組件的根元素。
大白話講就是沒有父組件傳遞過來的props
,在子組件中沒有對應的props
聲明,那麼在子組件中就可以通過v-on:$attrs
將父組件的props
透傳給孫組件,在二次封裝一些elementui
的組件有奇效
紙上得來終覺淺,下面來看下實際的使用,目錄結構如下:
src
├─ blocks
│ └─ PassVal
│ ├─ components
│ │ └─ PassInput.vue
│ └─ PassVal.vue
├─ view
│ ├─ basic
│ │ └─ BasicView.vue
├─ App.vue
└─ main.js
BasicView.vue(父組件)
中引入PassVal.vue(子組件)
;PassVal.vue(子組件)
中引入PassInput.vue(孫組件)
//BasicView.vue代碼如下: <template> <div class="basic"> <h3><i class="title_icon"></i>基礎知識</h3> <PassVal placeholder="我是placerholder" :clearable="true" :defaultVal="defaultVal" @changGrandChildVal="changGrandChildVal" /> </div> </template> <script> import PassVal from "@/blocks/PassVal/PassVal.vue"; export default { data() { return { defaultVal: "測試透傳", }; }, components: { PassVal, }, methods: { /** * @function input值修改回調函數 */ changGrandChildVal(val) { console.log("PassInput組件的值變瞭", val); }, }, }; </script> <style scoped> .basic { width: 100%; height: 100%; } </style>
- 在
PassVal
想要傳遞三個屬性placeholder="我是placerholder"
、:clearable="true"
以及:defaultVal="defaultVal"
接著來看下在PassVal
中的處理:
//PassVal.vue <template> <div class="passval"> <el-divider content-position="left">1-$attr和 $listeners</el-divider> <div class="container"> <div class="flex-two"> <passInput v-bind="$attrs" v-on="$listeners"></passInput> </div> </div> </div> </template> <script> import PassInput from "./components/PassInput.vue"; export default { components: { PassInput, }, props: { defaultVal: { type: String, default: "輸入框默認值", }, }, }; </script> <style scoped> .passval { width: 100%; height: 100%; } </style>
- 這裡通過
v-bind="$attrs"
和v-on="$listeners"
將屬性和方法透傳下去 - 如果在
PassVal.vue
中有關於來自父組件BasicView.vue
相關的props
聲明,那麼v-bind="$attrs"
透傳的屬性會將聲明的這個屬性剔除,透傳餘下的porps
屬性。- 例如:如果在
PassVal.vue
中的props
中聲明defaultVal
,那麼父組件BasicView.vue
傳遞過來的defaultVal
將無法通過v-bind="$attrs"
透傳給子組件PassInput.vue
- 例如:如果在
而在PassInput.vue
組件中
<template> <div class="pass-input"> <el-input v-bind="$attrs" v-model="value" @input="inputHandler"></el-input> </div> </template> <script> export default { name: "PassInput", created() { console.log("我是$attrs", this.$attrs); console.log("我是$listeners", this.$listeners); }, data() { return { value: "", }; }, methods: { /** * @function el-input的輸入回調函數 */ inputHandler(val) { this.$emit("changGrandChildVal", val); }, }, }; </script> <style scoped> .pass-input { width: 100%; height: 100%; } </style>
從而實現瞭優雅的屬性透傳,在組件封裝中比較有用。
生命周期中的console.log("我是$attrs", this.$attrs)和console.log("我是$listeners", this.$listeners);
以結果如下:
總結
到此這篇關於vue2.x中$attrs使用的文章就介紹到這瞭,更多相關vue2.x $attrs使用內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 簡單聊聊vue2.x的$attrs和$listeners
- vue $attrs和$listeners的使用與區別
- Vue祖孫組件如何實現傳值
- vue3中$attrs的變化與inheritAttrs的使用詳解
- vue 組件通信的多種方式