摘要:分別是表達式,事件傳遞,和的計算屬性。問題我有一個,其中包含一些狀態信息比如連接狀態和電量。現在我需要做一個去展示這些狀態。因為渲染的時候也許已經更新了。
TL;DR
這篇文章講解了三種根據 service 的狀態更新 directive 的做法。分別是 $watch 表達式,事件傳遞,和 controller 的計算屬性。
問題我有一個 readerService ,其中包含一些狀態信息(比如連接狀態和電量)。現在我需要做一個 directive 去展示這些狀態。因為它只需要從 readerService 中獲取數據,不需要任何外部傳值,所以我直接把 service 注入進去。但如何更新就成了一個問題。
service 的代碼如下。
const STATUS = { DETACH: "DETACH", ATTACH: "ATTACH", READY: "READY" } class ReaderService { constructor() { this.STATUS = STATUS // The status will be changed by some callbacks this.status = STATUS.DETACH } } angular.module("app").service("readerService", readerService)
directive 代碼如下:
angular.module("app").directive("readerIndicator", (readerService) => { const STATUS = readerService.STATUS const STATUS_DISPLAY = { [STATUS.DETACH]: "Disconnected", [STATUS.ATTACH]: "Connecting...", [STATUS.READY]: "Connected", } return { restrict: "E", scope: {}, template: `{{statusDisplay}}`, link(scope) { // Set and change scope.statusDisplay here } } })
我嘗試過以下幾種辦法,下面一一介紹。
方法一:$watch第一個想到的方法就是在 directive 中用 $watch 去監視 readerService.status。因為它不是 directive scope 的屬性,所以我們需要用一個函數來包裹它。Angular 會在 dirty-checking 時計算和比較新舊值,只有狀態真的發生了改變才會觸發回調。
// In directive link(scope) { scope.$watch(() => readerService.status, (status) => { scope.statusDisplay = STATUS_DISPLAY[status] }) }
這個做法足夠簡單高效,只要涉及 readerService.status 改變的代碼會觸發 dirty-checking ,directive 就會自動更新。service 不需要修改任何代碼。
但如果有多個 directive 的屬性都受 service status 的影響,那 $watch 代碼就看得比較晦澀了。尤其是 $watch 修改的值會影響其他的值的時候。比如:
// In directive link(scope) { scope.$watch(() => readerService.status, (status) => { scope.statusDisplay = STATUS_DISPLAY[status] scope.showBattery = status !== STATUS.DETACH }) scope.$watch("showBattery", () => { // some other things depend on showBattery }) }
這種時候聲明式的編程風格會更容易看懂,比如 Ember 或 Vue 里面的 computed property 。這個待會討論。
方法二:$broadcast/$emit + $on這種思路是 service 每次狀態改變都發送一個事件,然后 directive 監聽事件來改變狀態。因為 directive 渲染的時候也許 status 已經更新了。所以我們需要在 link 中計算一個初始值。
我最開始是用 $broadcast 去做的。代碼如下:
// In service setStatus(value) { this.status = value // Need to inject $rootScope this.$rootScope.$broadcast("reader.statusChanged", this.status) } // In directive link(scope) { scope.statusDisplay = STATUS_DISPLAY[nfcReaderService.status] scope.$on("reader.statusChanged", (event, status) => { scope.statusDisplay = STATUS_DISPLAY[status] }) }
但馬上發現 $broadcast 之后 UI 更新總要等 1 秒多(不過 $on 回調倒是很快)。Google 一番后知道原因是 $broadcast 是向下層所有 scope 廣播,廣播完成后再 dirty-checking 。一個更好的做法是使用 $emit ,它只會向上傳遞事件,不過不管發送事件還是監聽事件都得用 $rootScope 。
修改后的代碼如下:
// In service setStatus(value) { this.status = value // Use $emit instead of $broadcast this.$rootScope.$emit("reader.statusChanged", this.status) } // In directive link(scope) { scope.statusDisplay = STATUS_DISPLAY[nfcReaderService.status] // Use $rootScope instead of scope $rootScope.$on("reader.statusChanged", (event, status) => { scope.statusDisplay = STATUS_DISPLAY[status] }) }
如果因為某些原因不得不用 $broadcast 的話,你可以在 $on 回調最后用 $digest 或 $apply 強制觸發 dirty-checking ,這也可以達到快速更新 UI 的目的。
方法三:controller + property我個人覺得前兩個方法能解決問題,但代碼維護性都不太好。$watch 在屬性相互關聯的情況下非常難看懂,$emit/$on 需要把一些邏輯寫兩次(初始化 directive 時和回調執行時)。方法一中我提到了有些時候聲明式的屬性比 $watch 更容易看懂。這個方法就是使用 controller 。directive 可以設置自己的 controller 作為數據來源(或者說 view model),我們可以把那些需要計算的屬性作為 controller 的屬性。這樣 dirty-checking 時它們就會自動計算。
// In directive class ReaderController { constructor($scope, readerService) { this.readerService = readerService } get statusDisplay() { return STATUS_DISPLAY[this.readerService.status] } } return { // ... controller: ReaderController, controllerAs: "vm", template: `{{vm.statusDisplay}}` }
這樣一來,大部分邏輯都可以挪到 controller 中。如果沒有 DOM 操作我們甚至可以不寫 link 方法。也沒必要加入額外的 $watch 和 $on 。只是因為 dirty-checking 的特性,綁定到 template 的屬性往往會多計算幾次。所以屬性必須非常簡單。大部分情況下這不會有什么問題。
參考資料$rootScope.Scope
Angular API ,可以看看里面對 $watch ,$broadcast ,$emit , $on 的描述。
$rootScope.$emit() vs $rootScope.$broadcast()
$emit 和 $broadcast 的性能比較。注意后來的 Angular 已經解決了性能差異,兩者相差無幾。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/79084.html
摘要:共享數據的最佳策略是什么呢用一些變態的控制器繼承方案嗎當然不是,最簡單容易的方式就是使用服務。概括創建一個服務去存放你的數據,并給數據創建和的方法。 原文鏈接 : Sharing Data Between Controllers? Best Practice: Use a Service原文作者 : DAVE CEDDIA譯者 : 李林璞(web前端領域)譯者注:翻譯如有疏漏,歡迎指出...
摘要:組件還包含數據事件的輸入與輸出,生命周期鉤子和使用單向數據流以及從父組件上獲取數據的事件對象備份。 說明:參照了Angular1.x+es2015的中文翻譯,并將個人覺得不合適、不正確的地方進行了修改,歡迎批評指正。 架構,文件結構,組件,單向數據流以及最佳實踐 來自@toddmotto團隊的實用編碼指南 Angular 的編碼風格以及架構已經使用ES2015進行重寫,這些在Angul...
摘要:場景和問題這幾天在寫一個。因為狀態的改變由另一個插件控制,不在的中。前者強制觸發一次,后者讓一段代碼執行完成后觸發。所以直接在回調中使用就可以解決問題。參考資料對異常的描述。這種異常附帶在線文檔的方式還是很方便的。 TL;DR 這是一個關于 $rootScope:inprog 錯誤在什么樣的情況下被觸發,和如何解決的故事。 場景和問題 這幾天在寫一個 service 。這個 servi...
摘要:因為我們正好要使用,此時我們就可以用的服務來監聽了當然,我們不能忘了清空訂閱最后一步就是在有返回的時候調用表單的現在我們可以在組件里顯示提醒了 原文:Connect Angular Forms to @ngrx/store showImg(https://segmentfault.com/img/bVRpdX?w=1000&h=674); 這篇文章中,我們將要討論如何用ngrx/eff...
閱讀 2701·2021-10-12 10:12
閱讀 2341·2021-09-02 15:41
閱讀 2573·2019-08-30 15:55
閱讀 1404·2019-08-30 13:05
閱讀 2440·2019-08-29 11:21
閱讀 3539·2019-08-28 17:53
閱讀 3033·2019-08-26 13:39
閱讀 806·2019-08-26 11:50