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

feat($rootScope): allow suspending and resuming watchers on scope #10658

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 32 additions & 2 deletions src/ng/rootScope.js
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,7 @@ function $RootScopeProvider() {
this.$$childHead = this.$$childTail = null;
this.$root = this;
this.$$destroyed = false;
this.$$suspended = false;
this.$$listeners = {};
this.$$listenerCount = {};
this.$$isolateBindings = null;
Expand Down Expand Up @@ -751,7 +752,7 @@ function $RootScopeProvider() {

traverseScopesLoop:
do { // "traverse the scopes" loop
if ((watchers = current.$$watchers)) {
if ((watchers = !current.$$suspended && current.$$watchers)) {
// process our watches
length = watchers.length;
while (length--) {
Expand Down Expand Up @@ -794,7 +795,7 @@ function $RootScopeProvider() {
// Insanity Warning: scope depth-first traversal
// yes, this code is a bit crazy, but it works and we have tests to prove it!
// this piece should be kept in sync with the traversal in $broadcast
if (!(next = (current.$$childHead ||
if (!(next = (!current.$$suspended && current.$$childHead ||
(current !== target && current.$$nextSibling)))) {
while (current !== target && !(next = current.$$nextSibling)) {
current = current.$parent;
Expand Down Expand Up @@ -825,6 +826,35 @@ function $RootScopeProvider() {
}
},

/**
* @ngdoc method
* @name $rootScope.Scope#$suspend
* @kind function
*
* @description
* Suspend watchers of this scope subtree so that they will not be invoked during digest.
*
* This can be used to optimize your application when you know that running those watchers
* is redundant.
*/
$suspend: function() {
this.$$suspended = true;
},

/**
* @ngdoc method
* @name $rootScope.Scope#$resume
* @kind function
*
* @description
* Resume watchers of this scope subtree in case it was suspended.
*
* It is recommended to digest on this scope after it is resumed to catch any modifications
* that might have happened while it was suspended.
*/
$resume: function() {
this.$$suspended = false;
},

/**
* @ngdoc event
Expand Down
30 changes: 30 additions & 0 deletions test/ng/rootScopeSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -775,6 +775,36 @@ describe('Scope', function() {
});


describe('$suspend/$resume', function() {
it('should suspend watchers on scope', inject(function($rootScope) {
var watchSpy = jasmine.createSpy('watchSpy');
$rootScope.$watch(watchSpy);
$rootScope.$suspend();
$rootScope.$digest();
expect(watchSpy).not.toHaveBeenCalled();
}));

it('should resume watchers on scope', inject(function($rootScope) {
var watchSpy = jasmine.createSpy('watchSpy');
$rootScope.$watch(watchSpy);
$rootScope.$suspend();
$rootScope.$resume();
$rootScope.$digest();
expect(watchSpy).toHaveBeenCalled();
}));

it('should suspend watchers on child scope', inject(function($rootScope) {
var watchSpy = jasmine.createSpy('watchSpy');
var scope = $rootScope.$new(true);
scope.$watch(watchSpy);
$rootScope.$suspend();
$rootScope.$digest();
expect(watchSpy).not.toHaveBeenCalled();
}));

});


describe('optimizations', function() {

function setupWatches(scope, log) {
Expand Down