Vue - 自定義組件如何雙向綁定

来源:https://www.cnblogs.com/chanwahfung/archive/2020/01/10/12173902.html
-Advertisement-
Play Games

前言 無論在任何的語言或框架中,我們都提倡代碼的復用性。對於Vue來說也是如此,相同的代碼邏輯會被封裝成組件,除了復用之外,更重要的是統一管理提高開發效率。我真就接手過一個項目,多個頁面都會用到的列表,沒有去封裝列表組件,只要有一點改動,每個頁面都得加上。很肯定的說,沒有用組件化開發的Vue項目是沒 ...


前言

無論在任何的語言或框架中,我們都提倡代碼的復用性。對於Vue來說也是如此,相同的代碼邏輯會被封裝成組件,除了復用之外,更重要的是統一管理提高開發效率。我真就接手過一個項目,多個頁面都會用到的列表,沒有去封裝列表組件,只要有一點改動,每個頁面都得加上。很肯定的說,沒有用組件化開發的Vue項目是沒有靈魂的。所以如何封裝一個優雅且復用性高的組件成為我們必需的技能。

Tab自定義組件

首先來看一個Tab組件的實現,看看它存在什麼問題,哪裡可以改進?

效果

組件

<template>
  <div class="tabs">
    <div 
      class="tab-item" 
      :class="{'tab--active':item===activeName}"
      v-for="(item,index) in tabs" 
      :key="index" 
      @click="tabChange(item)">
      {{item}}
    </div>
  </div>
</template>

<script>
export default {
  props:{
    tabs:{
      type: Array,
      default: ()=> []
    },
    activeName:{
      type: String,
      default: ''
    }
  },
  methods:{
    tabChange(item){
      this.$emit('tabChange',item)
    }
  },
}
</script>

使用

<template>
  <div>
    <Tabs :tabs="tabs" :activeName="activeName" @tabChange="tabChange" />
  </div>
</template>

<script>
import Tabs from '../components/Tabs'
export default {
  components:{
    Tabs
  },
  data(){
    return{
      tabs:['黃金體驗','敗者食塵','緋紅之王','白金之星','波紋疾走'],
      activeName: '黃金體驗'
    }
  },
  methods:{
    tabChange(item){
      this.activeName = item
    }
  },
}
</script>

分析

這個組件最大的問題就是,activeName 需要使用者額外通過事件來手動更新,假如有另一個使用者接手,在不知道這種情況下使用,會出現tab沒有切換的情況。然後要去看組件內部實現,再回來修改代碼,很顯然這樣的組件是失敗的。本著所有的臟活累活都由組件實現的原則,理想的狀態應該是使用者不需要管理 activeName,而是由組件內部去更新。

如何改進

修改prop?

可能有人會想到,既然要由內部管理,那在組件內部修改prop的值是不是就可以了?來看下這樣的做法是否可行
修改組件tabChange方法,在點擊時更新prop的值

tabChange(item){
  this.activeName = item
  this.$emit('tabChange',item)
}

使用時,控制台拋出警告

由於prop是單向數據流,父級prop的更新會向下流動到子組件中,相反的在子組件內部直接更新狀態,會導致數據的流向不明確。例如,在父組件中有多個子組件依賴同一個屬性,其中一個子組件更新該屬性,會引發其餘子組件發生改變,發生問題時不容易被找到,因此Vue不推薦我們這樣做。另外,在父組件發生更新時,子組件的prop會被刷新為最新的值。
單向數據流: https://cn.vuejs.org/v2/guide/components-props.html#%E5%8D%95%E5%90%91%E6%95%B0%E6%8D%AE%E6%B5%81

正解:model選項

改進組件

組件model選項

允許一個自定義組件在使用 v-model 時定製 prop 和 event。預設情況下,一個組件上的 v-model 會把 value 用作 prop 且把 input 用作 event,但是一些輸入類型比如單選框和覆選框按鈕可能想使用 value prop 來達到不同的目的。使用 model 選項可以迴避這些情況產生的衝突。

model: https://cn.vuejs.org/v2/api/#model

在model選項里,我們可以綁定一個屬性,併為其添加事件,只需在調用方法時傳入值即可更新屬性。

<script>
export default {
  model:{
    prop: 'activeName',
    event: 'update'
  },
  props:{
    tabs:{
      type: Array,
      default: ()=> []
    },
    activeName:{
      type: String,
      default: ''
    }
  },
  methods:{
    tabChange(item){
      this.$emit('update',item) // 這裡更新activeName
      this.$emit('tabChange',item)
    }
  }
}
</script>

註意你仍然需要在組件的 props 選項里聲明 prop。

使用

使用組件雙向綁定後,屬性在組件內部更新時,父組件的 activeName 也會隨之更新,這樣組件使用起來顯得更優雅

<Tabs :tabs="tabs" v-model="activeName" />

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 判定樹和哈夫曼樹 分類與判定樹 這個小節有個比較重要的概念,就是 記住即可 哈夫曼樹與哈夫曼演算法 首先瞭解一下什麼是哈夫曼樹 給定一組值p~1~,...p~k~,如何構造一棵有k個葉子且分別以這些值為權的判定樹,使得其平均比較次數最小。滿足上述條件的判定樹稱為哈夫曼樹。 哈夫曼率先給出了一個求哈夫曼 ...
  • 場景 狀態欄上顯示通知效果 註: 博客: https://blog.csdn.net/badao_liumang_qizhi關註公眾號 霸道的程式猿 獲取編程相關電子書、教程推送與免費下載。 實現 新建NotificationActivity,通過getSystemService方法獲取通知管理器。 ...
  • 個人博客 "http://www.milovetingting.cn" 四大組件的工作過程 四大組件:Activity、Service、BroadcastReceiver、ContentProvider 四大組件的運行狀態 Android的四大組件中除了BroadcastReceiver,其它三種組 ...
  • 先 "pod setup" 再 "pod install". ...
  • 場景 app中常見的對話框。 簡單的帶確定取消按鈕的對話框 帶列表的對話框 帶單項選擇的對話框 帶多項選擇的對話框 註: 博客: https://blog.csdn.net/badao_liumang_qizhi 關註公眾號 霸道的程式猿 獲取編程相關電子書、教程推送與免費下載。 實現 將佈局改為L ...
  • 以前一直用的安卓原生Toast,個人感覺Toast這東西,沒必要花功夫,知道看到了Toasty這東西,立刻被圈粉了,真的非常好看。 項目地址 我們都知道,安卓原生Toast的用法是 1 Toast.makeText(MainActivity.this,"Toast顯示內容",Toast.LENGTH ...
  • 場景 效果 註: 博客: https://blog.csdn.net/badao_liumang_qizhi 關註公眾號 霸道的程式猿 獲取編程相關電子書、教程推送與免費下載。 實現 將佈局改為LinearLayout,並通過android:orientation="vertical">設置為垂直布 ...
  • 1.SharedPreferences介紹 SharedPreferences,它是一個輕量級的配置文件類,用於保存軟體配置參數. 採用xml文件形式存儲在/data/data/包名/shared_prefs/下 優點在於: 輕量級,以鍵值對的方式進行存儲,使用方便,易於理解 程式卸載後會也會一併被 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:本文代碼示例演示瞭如何在WPF中使用LiveCharts庫創建動態條形圖。通過創建數據模型、ViewModel和在XAML中使用`CartesianChart`控制項,你可以輕鬆實現圖表的數據綁定和動態更新。我將通過清晰的步驟指南包括詳細的中文註釋,幫助你快速理解並應用這一功能。 先上效果: 在 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • 概述:本示例演示了在WPF應用程式中實現多語言支持的詳細步驟。通過資源字典和數據綁定,以及使用語言管理器類,應用程式能夠在運行時動態切換語言。這種方法使得多語言支持更加靈活,便於維護,同時提供清晰的代碼結構。 在WPF中實現多語言的一種常見方法是使用資源字典和數據綁定。以下是一個詳細的步驟和示例源代 ...
  • 描述(做一個簡單的記錄): 事件(event)的本質是一個委托;(聲明一個事件: public event TestDelegate eventTest;) 委托(delegate)可以理解為一個符合某種簽名的方法類型;比如:TestDelegate委托的返回數據類型為string,參數為 int和 ...
  • 1、AOT適合場景 Aot適合工具類型的項目使用,優點禁止反編 ,第一次啟動快,業務型項目或者反射多的項目不適合用AOT AOT更新記錄: 實實在在經過實踐的AOT ORM 5.1.4.117 +支持AOT 5.1.4.123 +支持CodeFirst和非同步方法 5.1.4.129-preview1 ...
  • 總說周知,UWP 是運行在沙盒裡面的,所有許可權都有嚴格限制,和沙盒外交互也需要特殊的通道,所以從根本杜絕了 UWP 毒瘤的存在。但是實際上 UWP 只是一個應用模型,本身是沒有什麼許可權管理的,許可權管理全靠 App Container 沙盒控制,如果我們脫離了這個沙盒,UWP 就會放飛自我了。那麼有沒... ...
  • 目錄條款17:讓介面容易被正確使用,不易被誤用(Make interfaces easy to use correctly and hard to use incorrectly)限制類型和值規定能做和不能做的事提供行為一致的介面條款19:設計class猶如設計type(Treat class de ...
  • title: 從零開始:Django項目的創建與配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 後端開發 tags: Django WebDev Python ORM Security Deployment Op ...
  • 1、BOM對象 BOM:Broswer object model,即瀏覽器提供我們開發者在javascript用於操作瀏覽器的對象。 1.1、window對象 視窗方法 // BOM Browser object model 瀏覽器對象模型 // js中最大的一個對象.整個瀏覽器視窗出現的所有東西都 ...