1. 程式人生 > >Vue實戰指南之依賴注入(provide / inject)

Vue實戰指南之依賴注入(provide / inject)

案例

UI美眉說咱家的選項選單太醜了,小哥哥能不能美化一下呀,灑家自然是說小意思啦~
自定義一個select元件,so easy~

簡單粗暴型:

<el-select v-model="favourite" :option="[]"></el-select>

option作為資料進來就ok啦。

然後發現下列問題:

  • key-value,不是所有的介面都是id-name
  • optiondisabled 怎麼辦?
  • option存在幾種情況怎麼辦?
  • ...

回頭看看原生的寫法是這樣:

<select v-model="favourite">
    <option value="1">Vue</option>
    <option value="2">React</option>
    <option value="3">Angular</option>
</select>

還要加個el-option元件,靈活自由型:

<el-select v-model="favourite">
    <el-option value="1">Vue</el-option>
    <el-option value="2">React</el-option>
    <el-option value="3">Angular</el-option>
</el-select>

好啦,這樣設計就能完美解決之前的幾個問題。
接著要解決選擇了某一個el-option,怎麼告訴el-select

$parent是一種選擇,那麼el-select當前的值又怎麼告訴el-option你被選中了呢~ 筆者沒有繼續去深究,因為看到了APIprovide/inject

官方說明:

允許一個祖先元件向其所有子孫後代注入一個依賴,不論元件層次有多深(這也是使用$parent不好實現的地方),並在起上下游關係成立的時間裡始終生效。

不論元件層次有多深,這個簡直太爽了,不用再關心dom層級,只要在祖先元件內部就可以一直使用祖先元件提供的provide

用法

下面貼出一部分select的實現:

  • provideObject | () => Object
  • injectArray<string> | { [key: string]: string | Symbol | Object }

el-select

export default {
  name: "el-select",
  provide() {
    return {
      select: this
    };
  }
}

el-option

export default {
    name:'el-option',
    inject:['select'],
    created(){
      if(this.select.value===this.value){
        this.select.label=this.label;
      }
    }
}

總結

provide/inject 是解決元件之間的通訊問題的利器,不受層級結構的限制。
但也不是隨便去濫用,通訊代表著耦合:

provideinject 主要為高階外掛/元件庫提供用例。並不推薦直接用於應用程式程式碼中。
官方文件:
https://cn.vuejs.org/v2/api/#provide-inject
https://cn.vuejs.org/v2/guide/components-edge-cases.html#%E4%BE%9D%E8%B5%96%E6%B3%A8%E5%85%A5