-
Notifications
You must be signed in to change notification settings - Fork 27.4k
ngOptions slow performance in IE due to option rerendering #15801
Comments
Thanks for the detailed report, I will investigate later. Two thoughts:
|
Hi,
I hope I have resolved the doubts. Thank you for you answer! |
Would you like to open a PR that fixes the ngIf bug, removes the double initial updating, and any other possible simplifications? This would also show us which of our tests will fail with these changes |
This bug was reported as part of angular#15801
This bug was reported as part of #15801
This bug was reported as part of #15801
Avoid double execution of `updateOptions()` method, which causes a complete repainting of all `<option>` elements. Fixes angular#15801 Closes angular#15812
Avoid double execution of `updateOptions()` method, which causes a complete repainting of all `<option>` elements. Fixes angular#15801 Closes angular#15812 Close angular#16071
Do you want to request a feature or report a bug?
Bug
What is the current behavior?
Currently, when we create a
<select>
with ngOptions, theupdateOptions()
method is called twice, causing a complete repainting of the options. The first time is called manually and the second time is called from the getWatchables watch. This is the code:This could be a bug, or not. In my opinion it doesn't matter if we reuse the elements, but with the current implementation it's a performance flaw.
The problem appears with the perf(ngOptions): use documentFragment to populate select commit when the behavior of updateOptions() was changed.
The main problem with this commit occurs when
options
are already defined (the options have changed), all current options are removed from the DOM one by one. In slow browsers (like IE), removing a child is an expensive operation. With this commit you have tried to solve a performance problem, creating a worse one.The real performance problem is solved with this commit perf(ngOptions): avoid calls to
element.value
. I tried to revert the commit Revert 'perf(ngOptions): use documentFragment to populate select' commit and the performance in IE is now similar to Chrome.I create a plunker to show the difference: http://plnkr.co/edit/T83r9GXG6NfMZqSR5jXZ (to change between Angular current implementation and the revert of the commit Revert 'perf(ngOptions): use documentFragment to populate select' commit change the commented
<scripts src="">
). To be able to calculate the render time of<select>
I've made a wrap inside a custom directive. The time it takes to render is displayed in the browser console. The<select>
have 10000 items to render (IE will not be able to load the<select>
with the current implementation). Change between the angular.js scripts and see the difference.I'll show you another bug introduced in this commit. The new updateOptions method starts with:
I read the comment, then I tried to create the following
<select>
:The surprise was that no ngIf comment came out. The reason? All content in the
<select>
is deleted before it is created (selectElement.empty()
called before the firstupdateOptions()
is called in postLink). Then it would be the same to change the previous code for this one a little (we don't need to access to theelement.parentNode
) faster:To show you how badly implemented it is, if we see the method
updateOptionElement
, there is an if to check if the label is different ... Why it needs to check? The option element is new, no label, no value. At most it could look if it is undefined, because in some browsers it is expensive to change the textContent property.What is the expected behavior?
I already commented it in the previous section.
What is the motivation / use case for changing the behavior?
Performance flaw.
Which versions of AngularJS, and which browser / OS are affected by this issue? Did this work in previous versions of AngularJS? Please also test with the latest stable and snapshot (https://code.angularjs.org/snapshot/) versions.
Angular 1.6.0 and later (all browsers, but with IE is very noticeable).
Other information (e.g. stacktraces, related issues, suggestions how to fix)
Revert the perf(ngOptions): use documentFragment to populate select commit.
If you want to use a documentFragment, you can use it when options is empty (first rendering).
The text was updated successfully, but these errors were encountered: