Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit 47b7a72

Browse files
shahatapetebacondarwin
authored andcommitted
feat($rootScope): allow suspending and resuming watchers on scope
This can be very helpful for external modules that help making the digest loop faster by ignoring some of the watchers under some circumstance. example: https://github.com/shahata/angular-viewport-watch
1 parent 0864f73 commit 47b7a72

File tree

2 files changed

+63
-2
lines changed

2 files changed

+63
-2
lines changed

src/ng/rootScope.js

+32-2
Original file line numberDiff line numberDiff line change
@@ -178,6 +178,7 @@ function $RootScopeProvider() {
178178
this.$$childHead = this.$$childTail = null;
179179
this.$root = this;
180180
this.$$destroyed = false;
181+
this.$$suspended = false;
181182
this.$$listeners = {};
182183
this.$$listenerCount = {};
183184
this.$$watchersCount = 0;
@@ -811,7 +812,7 @@ function $RootScopeProvider() {
811812

812813
traverseScopesLoop:
813814
do { // "traverse the scopes" loop
814-
if ((watchers = current.$$watchers)) {
815+
if ((watchers = !current.$$suspended && current.$$watchers)) {
815816
// process our watches
816817
watchers.$$digestWatchIndex = watchers.length;
817818
while (watchers.$$digestWatchIndex--) {
@@ -855,7 +856,7 @@ function $RootScopeProvider() {
855856
// Insanity Warning: scope depth-first traversal
856857
// yes, this code is a bit crazy, but it works and we have tests to prove it!
857858
// this piece should be kept in sync with the traversal in $broadcast
858-
if (!(next = ((current.$$watchersCount && current.$$childHead) ||
859+
if (!(next = ((!current.$$suspended && current.$$watchersCount && current.$$childHead) ||
859860
(current !== target && current.$$nextSibling)))) {
860861
while (current !== target && !(next = current.$$nextSibling)) {
861862
current = current.$parent;
@@ -892,6 +893,35 @@ function $RootScopeProvider() {
892893
$browser.$$checkUrlChange();
893894
},
894895

896+
/**
897+
* @ngdoc method
898+
* @name $rootScope.Scope#$suspend
899+
* @kind function
900+
*
901+
* @description
902+
* Suspend watchers of this scope subtree so that they will not be invoked during digest.
903+
*
904+
* This can be used to optimize your application when you know that running those watchers
905+
* is redundant.
906+
*/
907+
$suspend: function() {
908+
this.$$suspended = true;
909+
},
910+
911+
/**
912+
* @ngdoc method
913+
* @name $rootScope.Scope#$resume
914+
* @kind function
915+
*
916+
* @description
917+
* Resume watchers of this scope subtree in case it was suspended.
918+
*
919+
* It is recommended to digest on this scope after it is resumed to catch any modifications
920+
* that might have happened while it was suspended.
921+
*/
922+
$resume: function() {
923+
this.$$suspended = false;
924+
},
895925

896926
/**
897927
* @ngdoc event

test/ng/rootScopeSpec.js

+31
Original file line numberDiff line numberDiff line change
@@ -1255,6 +1255,37 @@ describe('Scope', function() {
12551255
});
12561256
});
12571257

1258+
1259+
describe('$suspend/$resume', function() {
1260+
it('should suspend watchers on scope', inject(function($rootScope) {
1261+
var watchSpy = jasmine.createSpy('watchSpy');
1262+
$rootScope.$watch(watchSpy);
1263+
$rootScope.$suspend();
1264+
$rootScope.$digest();
1265+
expect(watchSpy).not.toHaveBeenCalled();
1266+
}));
1267+
1268+
it('should resume watchers on scope', inject(function($rootScope) {
1269+
var watchSpy = jasmine.createSpy('watchSpy');
1270+
$rootScope.$watch(watchSpy);
1271+
$rootScope.$suspend();
1272+
$rootScope.$resume();
1273+
$rootScope.$digest();
1274+
expect(watchSpy).toHaveBeenCalled();
1275+
}));
1276+
1277+
it('should suspend watchers on child scope', inject(function($rootScope) {
1278+
var watchSpy = jasmine.createSpy('watchSpy');
1279+
var scope = $rootScope.$new(true);
1280+
scope.$watch(watchSpy);
1281+
$rootScope.$suspend();
1282+
$rootScope.$digest();
1283+
expect(watchSpy).not.toHaveBeenCalled();
1284+
}));
1285+
1286+
});
1287+
1288+
12581289
describe('optimizations', function() {
12591290

12601291
function setupWatches(scope, log) {

0 commit comments

Comments
 (0)