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

perf(ngOptions): use documentFragment to populate select #14400

Merged
merged 1 commit into from
Apr 13, 2016

Conversation

Narretz
Copy link
Contributor

@Narretz Narretz commented Apr 9, 2016

What kind of change does this PR introduce? (Bug fix, feature, docs update, ...)
perf/fix

What is the current behavior? (You can also link to an open issue here)

  • ngoptions is slow in ie with many options
  • strange freezes in ie in some cases

Does this PR introduce a breaking change?
a tiny one possibly - now the select element is completely emptied before the options are appended

Please check if the PR fulfills these requirements

Other information:

This changes the way option elements are generated when the ngOption collection changes.
Previously, we would re-use option elements when possible (updating their text and
label). Now, we first remove all currently displayed options and the create new options for the
collection. The new options are first appended to a documentFragment, which is in the end appended
to the selectElement.

Using documentFragment improves render performance in IE with large option collections
(> 100 elements) considerably.

Always creating new options fixes issues in IE where the select would become unresponsive to user
input.

Fixes #13607
Fixes #12076

@Narretz Narretz force-pushed the fix-select-freeze branch from 1026b4d to 8e2100d Compare April 9, 2016 10:11
@Narretz Narretz added this to the 1.5.4 milestone Apr 9, 2016
@Narretz Narretz force-pushed the fix-select-freeze branch 3 times, most recently from 340dd24 to d73ab07 Compare April 12, 2016 18:32
@petebacondarwin
Copy link
Contributor

I checked this with the ngOptions benchmark the project and there is no discernable difference in speed on Chrome, Firefox or Safarin on OS/X. So I am happy to go with using this much simpler (and less buggy when it comes to IE) algorithm.

@petebacondarwin
Copy link
Contributor

@Narretz please merge into master and 1.5.x


options = ngOptions.getOptions();

var groupMap = {};
var currentElement = selectElement[0].firstChild;
var listFragment = $document[0].createDocumentFragment();
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Just noticed that it should be possible to create the documentFragment only once per ngOptions (instead of once per update), as the fragment contents are removed once it has been appended to a container

@gkalpak
Copy link
Member

gkalpak commented Apr 13, 2016

So, except for a couple of minor (ignorable) comments it LGTM.
I wonder if it might break people's test (the same way it broke ours).

@Narretz
Copy link
Contributor Author

Narretz commented Apr 13, 2016

Wrt to the tests - maybe. But the test was only working before because it relied on the implementation (option elements are reused). And tests shouldn't do that.

This changes the way option elements are generated when the ngOption collection changes.
Previously, we would re-use option elements when possible (updating their text and
label). Now, we first remove all currently displayed options and the create new options for the
collection. The new options are first appended to a documentFragment, which is in the end appended
to the selectElement.

Using documentFragment improves render performance in IE with large option collections
(> 100 elements) considerably.

Creating new options from scratch fixes issues in IE where the select would become unresponsive
to user input.

Fixes angular#13607
Fixes angular#13239
Fixes angular#12076
@Narretz Narretz force-pushed the fix-select-freeze branch from 091bbc8 to 97b3e00 Compare April 13, 2016 16:40
@Narretz Narretz merged commit 97b3e00 into angular:master Apr 13, 2016
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
4 participants