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 #16308

Merged
merged 1 commit into from
Dec 11, 2017
Merged
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
100 changes: 97 additions & 3 deletions src/ng/rootScope.js
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,7 @@ function $RootScopeProvider() {
this.$$watchersCount = 0;
this.$id = nextUid();
this.$$ChildScope = null;
this.$$suspended = false;
}
ChildScope.prototype = parent;
return ChildScope;
Expand Down Expand Up @@ -178,6 +179,7 @@ function $RootScopeProvider() {
this.$$childHead = this.$$childTail = null;
this.$root = this;
this.$$destroyed = false;
this.$$suspended = false;
this.$$listeners = {};
this.$$listenerCount = {};
this.$$watchersCount = 0;
Expand Down Expand Up @@ -808,7 +810,7 @@ function $RootScopeProvider() {

traverseScopesLoop:
do { // "traverse the scopes" loop
if ((watchers = current.$$watchers)) {
if ((watchers = !current.$$suspended && current.$$watchers)) {
// process our watches
watchers.$$digestWatchIndex = watchers.length;
while (watchers.$$digestWatchIndex--) {
Expand Down Expand Up @@ -852,7 +854,9 @@ 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.$$watchersCount && current.$$childHead) ||
// (though it differs due to having the extra check for $$suspended and does not
// check $$listenerCount)
if (!(next = ((!current.$$suspended && current.$$watchersCount && current.$$childHead) ||
(current !== target && current.$$nextSibling)))) {
while (current !== target && !(next = current.$$nextSibling)) {
current = current.$parent;
Expand Down Expand Up @@ -889,6 +893,95 @@ function $RootScopeProvider() {
$browser.$$checkUrlChange();
},

/**
* @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.
*
* **Warning**
*
* Suspending scopes from the digest cycle can have unwanted and difficult to debug results.
* Only use this approach if you are confident that you know what you are doing and have
* ample tests to ensure that bindings get updated as you expect.
*
* Some of the things to consider are:
Copy link
Contributor

@Narretz Narretz Nov 1, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A few other things:

  • isolate scopes are also suspended (as they are still children of the parent and not the rootScope)

  • resolved promises e.g. from explicit $q deferreds, timeouts, intervals, and http calls are not bound to scopes and will trigger a global digest if you e.g. execute http calls from a component which has its scope suspended.

*
* * Any external event on a directive/component will not trigger a digest while the hosting
* scope is suspended - even if the event handler calls `$apply()` or `$rootScope.$digest()`.
* * Transcluded content exists on a scope that inherits from outside a directive but exists
* as a child of the directive's containing scope. If the containing scope is suspended the
* transcluded scope will also be suspended, even if the scope from which the transcluded
* scope inherits is not suspended.
* * Multiple directives trying to manage the suspended status of a scope can confuse each other:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes good point. Perhaps a .$isSuspended() getter or similar so a directive can find out whether its scope is already suspended?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm this brings up an additional more complex point.

If a parent scope is suspended, but the current scope is not, this scope will not take part in any digests even though it is not officially suspended.

I think that we don't care, since directives that want to suspend a scope don't really care whether an ancestor scope has been suspended. They are only interested in managing their own scope.

The only time this is a problem is if there are multiple directives trying to manage the suspended status of a single scope.

Copy link
Contributor

@poshest poshest Nov 2, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I see the problem. It's a fair assumption that if you asked for $isSuspended() you'd want true returned if it was disabled because of a parent, but I think the stuff you put in the docs explaining this issue solves it.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Out of interest, how hard would it be to implement a .$ancestorIsSuspended()?

Unfortunately my only case for it is if people use $scope.$digest at some point below the $suspend'ed element and want to avoid that $digest in those cases. Since you discourage $scope.$digest, I am girding myself for rejection here, but it would be useful to me, and .$ancestorIsSuspended() would save me writing a custom "recurse up the $parents" function.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To find out if an ancestor is suspended, you only need to read the $parent scope until you find one that is suspended.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

An example is shown here.
@poshest, if you want it to be available on every scope, you can put it on $rootScope's prototype.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Awesome! Thanks guys, and for the code example @gkalpak :)

* * A call to `$suspend()` on an already suspended scope is a no-op.
* * A call to `$resume()` on a non-suspended scope is a no-op.
* * If two directives suspend a scope, then one of them resumes the scope, the scope will no
* longer be suspended. This could result in the other directive believing a scope to be
* suspended when it is not.
* * If a parent scope is suspended then all its descendants will be also excluded from future
* digests whether or not they have been suspended themselves. Note that this also applies to
* isolate child scopes.
* * Calling `$digest()` directly on a descendant of a suspended scope will still run the watchers
* for that scope and its descendants. When digesting we only check whether the current scope is
* locally suspended, rather than checking whether it has a suspended ancestor.
* * Calling `$resume()` on a scope that has a suspended ancestor will not cause the scope to be
* included in future digests until all its ancestors have been resumed.
* * Resolved promises, e.g. from explicit `$q` deferreds and `$http` calls, trigger `$apply()`
* against the `$rootScope` and so will still trigger a global digest even if the promise was
* initiated by a component that lives on a suspended scope.
*/
$suspend: function() {
this.$$suspended = true;
},

/**
* @ngdoc method
* @name $rootScope.Scope#$isSuspended
* @kind function
*
* @description
* Call this method to determine if this scope has been explicitly suspended. It will not
* tell you whether an ancestor has been suspended.
* To determine if this scope will be excluded from a digest triggered at the $rootScope,
* for example, you must check all its ancestors:
*
* ```
* function isExcludedFromDigest(scope) {
* while(scope) {
* if (scope.$isSuspended()) return true;
* scope = scope.$parent;
* }
* return false;
* ```
*
* Be aware that a scope may not be included in digests if it has a suspended ancestor,
* even if `$isSuspended()` returns false.
*
* @returns true if the current scope has been suspended.
*/
$isSuspended: function() {
return this.$$suspended;
},

/**
* @ngdoc method
* @name $rootScope.Scope#$resume
* @kind function
*
* @description
* Resume watchers of this scope subtree in case it was suspended.
*
* See {@link $rootScope.Scope#$suspend} for information about the dangers of using this approach.
*/
$resume: function() {
this.$$suspended = false;
},
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

IMO, $resume should also set dirty = true and lastDirtyWatch = null (in case already during a $digest).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If $resume triggered an async apply then we wouldn't need to worry about this, no?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct (I think 😁).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It appears that calling resume from a watch handler is fine, because the fact that the watch handler was called means that the digest is already marked as dirty. So a new digest has already been triggered.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

And we don't seem to need to worry about the short circuiting of lastDirtyWatch:

  • if the resumed scope is "after" the last dirty watch, then it will have been digested on that turn of the digest cycle.
  • if the resumed scope is "before" the last dirty watch, then it will get digested before we hit the short circuit.

Copy link
Contributor Author

@petebacondarwin petebacondarwin Nov 21, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So we don't need to run any applyAsync etc.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Interesting! 👍


/**
* @ngdoc event
Expand Down Expand Up @@ -1289,7 +1382,8 @@ 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 $digest
// (though it differs due to having the extra check for $$listenerCount)
// (though it differs due to having the extra check for $$listenerCount and
// does not check $$suspended)
if (!(next = ((current.$$listenerCount[name] && current.$$childHead) ||
(current !== target && current.$$nextSibling)))) {
while (current !== target && !(next = current.$$nextSibling)) {
Expand Down
122 changes: 122 additions & 0 deletions test/ng/rootScopeSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -1255,6 +1255,128 @@ describe('Scope', function() {
});
});


describe('$suspend/$resume/$isSuspended', 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) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe test that $resume works in this case as well?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

var watchSpy = jasmine.createSpy('watchSpy');
var scope = $rootScope.$new(true);
scope.$watch(watchSpy);
$rootScope.$suspend();
$rootScope.$digest();
expect(watchSpy).not.toHaveBeenCalled();
}));

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

it('should resume digesting immediately if `$resume` is called from an ancestor scope watch handler', inject(function($rootScope) {
var watchSpy = jasmine.createSpy('watchSpy');
var scope = $rootScope.$new();

// Setup a handler that will toggle the scope suspension
$rootScope.$watch('a', function(a) { if (a) scope.$resume(); else scope.$suspend(); });

// Spy on the scope watches being called
scope.$watch(watchSpy);

// Trigger a digest that should suspend the scope from within the watch handler
$rootScope.$apply('a = false');
// The scope is suspended before it gets to do a digest
expect(watchSpy).not.toHaveBeenCalled();

// Trigger a digest that should resume the scope from within the watch handler
$rootScope.$apply('a = true');
// The watch handler that resumes the scope is in the parent, so the resumed scope will digest immediately
expect(watchSpy).toHaveBeenCalled();
}));

it('should resume digesting immediately if `$resume` is called from a non-ancestor scope watch handler', inject(function($rootScope) {
var watchSpy = jasmine.createSpy('watchSpy');
var scope = $rootScope.$new();
var sibling = $rootScope.$new();

// Setup a handler that will toggle the scope suspension
sibling.$watch('a', function(a) { if (a) scope.$resume(); else scope.$suspend(); });

// Spy on the scope watches being called
scope.$watch(watchSpy);

// Trigger a digest that should suspend the scope from within the watch handler
$rootScope.$apply('a = false');
// The scope is suspended by the sibling handler after the scope has already digested
expect(watchSpy).toHaveBeenCalled();
watchSpy.calls.reset();

// Trigger a digest that should resume the scope from within the watch handler
$rootScope.$apply('a = true');
// The watch handler that resumes the scope marks the digest as dirty, so it will run an extra digest
expect(watchSpy).toHaveBeenCalled();
}));

it('should not suspend watchers on parent or sibling scopes', inject(function($rootScope) {
var watchSpyParent = jasmine.createSpy('watchSpyParent');
var watchSpyChild = jasmine.createSpy('watchSpyChild');
var watchSpySibling = jasmine.createSpy('watchSpySibling');

var parent = $rootScope.$new();
parent.$watch(watchSpyParent);
var child = parent.$new();
child.$watch(watchSpyChild);
var sibling = parent.$new();
sibling.$watch(watchSpySibling);

child.$suspend();
$rootScope.$digest();
expect(watchSpyParent).toHaveBeenCalled();
expect(watchSpyChild).not.toHaveBeenCalled();
expect(watchSpySibling).toHaveBeenCalled();
}));

it('should return true from `$isSuspended()` when a scope is suspended', inject(function($rootScope) {
$rootScope.$suspend();
expect($rootScope.$isSuspended()).toBe(true);
$rootScope.$resume();
expect($rootScope.$isSuspended()).toBe(false);
}));

it('should return false from `$isSuspended()` for a non-suspended scope that has a suspended ancestor', inject(function($rootScope) {
var childScope = $rootScope.$new();
$rootScope.$suspend();
expect(childScope.$isSuspended()).toBe(false);
childScope.$suspend();
expect(childScope.$isSuspended()).toBe(true);
childScope.$resume();
expect(childScope.$isSuspended()).toBe(false);
$rootScope.$resume();
expect(childScope.$isSuspended()).toBe(false);
}));
});
Copy link
Member

@gkalpak gkalpak Nov 3, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ideally, there should be some tests about suspending/resuming during a $digest.
(E.g. to cover basic cases, but also things like #16308 (comment).)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll see what I can add.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.



describe('optimizations', function() {

function setupWatches(scope, log) {
Expand Down