Skip to content
This repository has been 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

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

wants to merge 1 commit into from

Conversation

shahata
Copy link
Contributor

@shahata shahata commented Jan 7, 2015

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

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
@shahata
Copy link
Contributor Author

shahata commented Jan 7, 2015

Any thoughts about this? I had to do really hacky tricks in order to do this without changing angular...

@shahata
Copy link
Contributor Author

shahata commented Jan 7, 2015

/cc @petebacondarwin

@lgalfaso
Copy link
Contributor

lgalfaso commented Jan 7, 2015

This issue showed up several times in different forms and shape. Many of us
sees this as an issue that needs a solution, but we need to be sure that we
do not shoot ourself in the foot

@petebacondarwin
Copy link
Contributor

SImilar to @caitp's RFC email for handling sparse arrays in ngRepeat - I think we need to have one for this too. @shahata or @lgalfaso could you follow her template and put one together? Then we can email it out to angular-dev and internally in Google to ensure that we are doing the right thing.

@caitp
Copy link
Contributor

caitp commented Jan 7, 2015

I think we should actually use the angular forum rather than angular-dev, as angular-dev is full of jenkins spam and I don't think people are really using it.

I'm going to put up more/better template docs this week, and present the idea in the team meeting on monday, so I think we're going to hold off on sending out process emails until there's a bit more feedback from the team. But you can share your doc internally before that.

@PatrickJS
Copy link
Member

+1

@petebacondarwin
Copy link
Contributor

I wonder if this falls foul, in a similar way to #10221?
In other words that it would give people ways to break stuff subtly without really knowing why.

@vahan-hartooni
Copy link

+1 on this pull request

I too ended up with the same strategy to pause/suspend watchers (skipping a flagged scope during the digest loop) when we talked about this in #5301

@park9140
Copy link

+1 on this.

@petebacondarwin, @shahata, while there is a significant amount of opportunity here to shoot yourself in the foot, I believe this provides what is an essential feature as applications get more mature in angular.

Here is a sample of the hundreds of articles about performance and tools I keep finding referenced to support making angular a viable system to use in larger applications.

  1. https://github.com/scalyr/angular
  2. http://www.williambrownstreet.net/blog/2013/07/angularjs-my-solution-to-the-ng-repeat-performance-problem/
  3. https://www.exratione.com/2013/12/considering-speed-and-slowness-in-angularjs/

The common thread I find in every performance article is that we need to reduce watches in order to ensure performance. The new bind once syntax is based entirely on that goal, and the overuse of (https://github.com/Pasvaz/bindonce) by people advocating speed in angular. It gives you many varied and awful patterns for shooting yourself in the foot.

This PR is a much more subtle methodology for achieving the same intent as bind-once, it allows a developer to reduce the quantity of watches in his application based on his application's needs. More so, this particular ability allows angular library developers to write more performant directives.

Lets say a developer wants to build a general purpose tab directive. As is users of that directive can easily shoot themselves in the foot by not properly using bind once and limiting total watches. With this capability a plugin developer can disable the scope of non rendered tabs.

In a slightly more complex example. Development of data grids. Currently, in order to really display a large grid of data you absolutely must do virtual rendering, as is done in UI-Grid. For most data sets in modern browsers we could progressively render and leave the static html on the page with little to no performance drop. However, in angular we can't really achieve static output without making it truly static. Imagine if we made a simple directive that progressively rendered the content down the page, then disabled the scope processing for all offscreen components. Digesting components as they scroll into view becomes trivial if all we need to do is update the on screen components to be enabled, and trigger a digest. As an offshoot we don't have to do hacks like https://github.com/shahata/angular-viewport-watch, or https://github.com/scalyr/angular, in order to remove watchers.

TLDR; Those of us who know how to not shoot ourselves in the foot with this can use it to write libraries that help others not shoot themselves in the foot in other ways.

@lgalfaso
Copy link
Contributor

Hi, @park9140 brings some interesting points, so I would like to write some of the issues that this PR brings.

TL;DR
Angular is full of sharp edges that should be smooth out, having the ability to suspend a scope will add another. The valid use cases for this feature are limited at best.

Consistency model

Right now, angular consistency model is very simple. Eg, the two-way binding in ng-model is "once there is a change on one side, reflect it in the other side", there is no room for inconsistency. With scope: { '=foo' }, the two-way binding is "in case of a conflict, then the parent wins". Both of these will need to be revisited as developers will need a predictable mechanism that stipulates how conflicts are resolved (and we cannot claim that you can only suspend a scope that does not have an ng-model)

Async

If a directive performs anything asynchronous that needs to be reflected, then this directive cannot be inside a scope that gets suspended. The main issue is that in most cases a directive should not care what other directives are inside of it, they should only care of the behavior they bring to the table.

Transclude

If a directive uses a transclusion, then it should never suspend a scope (unless it has complete control of the content being transcluded). The reason being that even when the scope look like a tree, the inherence model does not follow the same tree and a suspended scope will also suspend parts of something that looks like an outer scope, but in reality is a child scope.

Extending another directive

A directive that uses suspend cannot be safely extended, this is, without understanding all the ins and outs of when this happens. In most cases, this implies that whoever does this will need to read the code of the other directive.

Creating a directive on an html standard element

Given that a directive does not have control of what directives will be executed inside it (as a developer may decide to create a directive on a standard element), then a directive is never sure that by suspending a scope it is breaking some other directive. In most cases, this implies that you will need to control 100% of all the directives in an app and reusability can be problematic.

Ownership of $scope

Right now, there is no clear ownership model between directives and scope. Some people may think that a directive sits on top of a scope (this is that the scope owns the directive) or that they are independent. This change makes a bold claim, a directive that suspends a scope, it owns the scope (as it changes it behavior and everything in it).

There are other minor nits in the PR that are not clear how things are affected. Eg, if a scope is suspended twice, should it need to be resumed twice for it to be active? This is important when there are two directives in the same scope use this trick. Each of the options opens another source of conflicts.

If a feature is made part of the core, we are saying that this feature is good for broad use, not made-to-order for a handful of directives. If we claim that the way to make your application more performant is this, then it needs to support the use cases that developers are used to. I am not sure that this feature can make these claims.

@park9140
Copy link

@lgalfaso,

Consistency Model

This is not as simplistic as you have pointed out, enormous numbers of directives bypass this by entirely bypassing scope as soon as they require any level of 'bare metal' performance. Anything that executes outside the digest cycle currently has very limited options for restoring consistency after the completion of any given bare metal action. Many take the easy route out and trigger something that calls $rootScope.$apply() in order to attempt to ensure consistency. Core angular code does this regularly. 90% of the time I see a digest cycle called it is absolutely unnecessary.

Async

I do not understand why async operations done by a directive can't be done when the directive is suspended. I suppose you might have a concurrency issue where an async operation could have the wrong data on the scope at the time of processing on a suspended scope, but you shouldn't be accessing scope from a non digest locked operation anyway, and if you are operating on a scope that is suspended the scope interacting portion of your async operation will not run until the scope is resumed.

Transclude
Extending another directive
Creating a directive on a standard html element

Im starting to think that a directive is necessarily the proper place to be doing this kind of control. It seems like something that should be reserved for use by your application controllers. Pattern wise it really isn't up to a directive to do this kind of application control. In every situation where I have had the need for this it really should be handled by a controller.

Ownership of scope

I'd argue that it is fairly clear that scope is a shared resource, except under certain circumstances. Isolate scopes are definitely owned by the directive that makes them. When there is a controller for a section the scope should be under that controller's scope of control.

I think I can agree that there are issues with this methodology being a public api and open to everyone to just extend. However, the core capability absolutely must exist for angular to really be able to handle larger applications.

Just today I implemented this in my application. The application needs to be able to quickly switch up a multiple annotations on a large grid of data. Using it in my controllers only, I was quickly able to decrease the points of UI thread lockout caused by digesting a much to large active digest tree. I rewrote ng-show and ng-hide to make large numbers of hidden annotations not get digested, but still allowed the core html to be rendered in the initial render state so that displaying annotations could be done very efficiently even with the enormous dataset.

TLDR; In the large majority of cases a modern browser can handle way more rendered dom than can be actively controlled by angular in a single cohesive scope tree. Isolating application layers such that global digests are controlled to only work on active code is essential.

@lgalfaso
Copy link
Contributor

@park9140 I think we mostly in sync, in fact many of the issues that you are pointing out (and the possible solutions) are implemented in Angular 2. Now, for 1.x, many of these changes would imply breaking changes that would be hard for developers to adapt to and need to be documented, and this PR documentation changes are super minimal.

I do not understand why async operations done by a directive can't be done when the directive is suspended.

If there is a sub-directive that makes a change that needs to be reflected on the DOM, then it will not be reflected. This implies that this sub-directive will need to contact all parent controllers that suspended a scope and ask them to resume it once so this change it reflected. This can make the solution worst than what we have now.

Im starting to think that a directive is necessarily the proper place to be doing this kind of control.

A directive cannot extend another directive (and this is something developers do a lot)
I am not sure how you would fix the issues with transclusion
A directive is not aware of all the directives that would run in it's template, so I do not know how it should know that it is safe to suspend the scope it sits on

I think I can agree that there are issues with this methodology being a public api and open to everyone to just extend. However, the core capability absolutely must exist for angular to really be able to handle larger applications.

Agree 100%, something must be made. In fact, this is not the first attempt at solving this issue, but whatever solution ends up in the core needs to be flexible enough to be useful for the majority of users while at the same time do not create any new sharp edges that people have to be aware of.
Angular is getting faster with every new version, anyhow I understand that there is some frustration as the pace that Angular is getting faster is slower than angular applications are getting bigger.
We are all aware of this and are always looking into ways to make Angular faster, but this needs to be done in a way that does not involve a lot of work for developers to upgrade to these new features, nor break common patterns used today.

@piernik
Copy link

piernik commented Mar 7, 2015

If I can - I'm using one time bindings. Now to reset those bindings (to calculate them once again) I'm using <directive ng-if="vm.reset"> and with $timeout I change vm.reset to false and after 10 ms again to true. But then directive (and all DOM element) disapears for a second - blinks.

How about giving new angular build-in directive <directive ng-reset="vm.reset">? If ng-reset would be set to true directive's controller is executed once more and caluculates new one-time bindings. After cycle is done angular sets ng-reset value to false.
You have only one watcher for directive and ability to calculate all directive's one-time bindings once more with new values (set before vm.reset=true)

Good idea?

@lgalfaso
Copy link
Contributor

It ends up that landing this would change some constraints in a way that makes the migration to Angular 2 harder.

It is a -1 from me

@petebacondarwin
Copy link
Contributor

@lgalfaso - can you elaborate?

@lgalfaso
Copy link
Contributor

@petebacondarwin will start with isolated directives as it is simpler. In Angular 1, the two statements below are either both true or both false

  • watches on $scope are executed on all digest calls
  • $element is attached to $document

If both of these statement are true, then the directive is active (invented word just to explain this easier). If both are false, then the directive is inactive (and if things are right, the GC should take care of it including $element). With this change, this is no longer true, as it is possible for watchers in the scope not to get called and $element to be attached to $document (and the directive should remain active).
Now, Angular 2 incorporates the concept of hydrate and dehydrate. In a dehydrated state, elements are kept for later reuse, but all directives should remain inactive (in fact, not even inactive as completely out of the elements and GC is possible).

This is, by landing the PR we are breaking something that devs might rely on (that by itself is not necessarily bad if the end result is something better), but it also does so in a way that is less aligned to Angular 2.

@petebacondarwin petebacondarwin modified the milestones: 1.6.3, Purgatory Feb 6, 2017
@PatrickJS
Copy link
Member

nice, this feature is already in Angular 2/4+ as ChangeDetectorRef.detach() ChangeDetectorRef.reattach() ChangeDetectorRef.detectChanges()

@petebacondarwin
Copy link
Contributor

@gdi2290 - yes, but it has slightly more reliable behaviour in Angular; in AngularJS there are going to be a load of corner cases that could bite us.

@petebacondarwin
Copy link
Contributor

Closing in favour of #16308

@poshest
Copy link
Contributor

poshest commented Nov 4, 2017

Has anyone here used .suspend() as implemented in this PR? I found this issue in this PR's reincarnation.

Eg @park9140, you said you "rewrote ng-show and ng-hide". Was it using a modified AngularJS like in this PR?

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.