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

<select> broken: model/view out of sync #7544

Closed
iffy opened this issue May 22, 2014 · 13 comments
Closed

<select> broken: model/view out of sync #7544

iffy opened this issue May 22, 2014 · 13 comments

Comments

@iffy
Copy link
Contributor

iffy commented May 22, 2014

I'm seeing a <select> display one value but have a model with a different value. I think it has something to do with the initial "empty" <option> when the model is null.

Here's some minimal reproducing code (there's also a fiddle below):

<!doctype html>
<html>
<body ng-app="test" ng-controller="TestCtrl as ctrl">
    {{ ctrl.selected }}
    <select ng-model="ctrl.selected.id" ng-options="thing.id as thing.name for thing in ctrl.things|orderBy:'name'" autofocus></select>

<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.2.16/angular.min.js"></script>
<script>
var app = angular.module('test', []);

app.controller('TestCtrl', function() {
    var ctrl = this;
    ctrl.selected = {};
    ctrl.things = [
        {name: 'Health', id: "health.id"},
        {name: 'Haircuts', id: "haircuts.id"},
    ];
    return ctrl;
});
</script>
</body>
</html>

To duplicate with the fiddle below:

  1. click somewhere in the rendered pane
  2. then press tab to focus the <select>
  3. then quickly type he
  4. Notice that the select is showing "Health" but the model is for "Haircuts"

http://jsfiddle.net/VRE2L/

I can duplicate this bug on 1.2.10 and 1.2.16 using Chrome (haven't tested other browsers)

@iffy
Copy link
Contributor Author

iffy commented May 22, 2014

If I include track by thing.id in ng-options then the model is correct (after pressing tab then typing he) but the view doesn't update at all :(

Here's a fiddle:

http://jsfiddle.net/AgNUM/

@iffy
Copy link
Contributor Author

iffy commented May 22, 2014

The problem (or maybe symptom) is that the change handler is not being called even though the value is changing. Using this patch (based on 1.2.16):

diff --git a/src/ng/directive/select.js b/src/ng/directive/select.js
index 628d217..fe9b61a 100644
--- a/src/ng/directive/select.js
+++ b/src/ng/directive/select.js
@@ -342,7 +342,14 @@ var selectDirective = ['$compile', '$parse', function($compile,   $parse) {
         // clear contents, we'll add what's needed based on the model
         selectElement.empty();

+        selectElement.on('keydown', function(ev) {
+          console.log('keydown', ev.keyCode, selectElement.val());
+        });
+        selectElement.on('keyup', function(ev) {
+          console.log('keyup', ev.keyCode, selectElement.val());
+        });
         selectElement.on('change', function() {
+          console.log('change');
           scope.$apply(function() {
             var optionGroup,
                 collection = valuesFn(scope) || [],

I see the following in the console when I type he:

keydown 72 ?
change
keyup 72 0
keydown 69 0
keyup 69 1

So you can see that the val (as returned by .val()) does indeed change when pressing e but the change handler is not called.

@iffy iffy changed the title select model/view out of sync <select> broken: model/view out of sync May 22, 2014
@btford btford added this to the 1.3.0 milestone May 22, 2014
@btford
Copy link
Contributor

btford commented May 22, 2014

I can confirm that this is an issue in Chrome.

@btford btford self-assigned this May 22, 2014
@pkozlowski-opensource
Copy link
Member

Ha, actually I was also looking at this one :-) From my playing with the issue it looks like the problem goes away if the orderBy filter is not used: http://jsfiddle.net/y9UxU/

Or maybe I'm just not clicking fast enough....

@iffy
Copy link
Contributor Author

iffy commented May 22, 2014

@pkozlowski-opensource The problem is still there even without the orderBy. In your fiddle if you:

  1. Click the rendered pane
  2. Press tab (no mouse)
  3. Type ha
  4. You will see it display "Haircuts" in the dropdown and the model is "health.id"

Afaict, if you use the mouse to select things there is no problem.

Edit: And I should re-point out that using track by changes the behavior but introduces other buggy behavior.

@iffy
Copy link
Contributor Author

iffy commented May 22, 2014

In Firefox (29.0.1) it behaves differently (there's never a discrepancy between model and view) because the change event isn't triggered until blur.

@IgorMinar
Copy link
Contributor

we should take a look at this after #6564 is fixed

@gkalpak
Copy link
Member

gkalpak commented Oct 2, 2014

This might be realated to #9134 (and a bunch similar issues), which is related to a Chrome bug - has been reported here.

@tbosch tbosch self-assigned this Oct 3, 2014
@jeffbcross
Copy link
Contributor

Created a plnkr from the original fiddle (easier for me): http://plnkr.co/edit/vu1buCX8Hb6ePQm2kDLF?p=preview

I was able to see that the change event is firing in Chrome, so this seems like an issue in Angular.

@tbosch
Copy link
Contributor

tbosch commented Oct 4, 2014

@jeffbcross I discussed this with @IgorMinar, have some ideas about the cause and an idea for the solution. Will take this from here...

@gkalpak
Copy link
Member

gkalpak commented Oct 4, 2014

@jeffbcross, I don't think it's an issue with Angular, as it also happens without Angular: plunkr
(originally posted here: #9134 (comment))

And yes, the change event is not fired when you remove a preceeding <option>.

This seems like a browser bug to me, but the question is: Should we account for that and make Angular not step into the bug, or should we wait for the bug to get fixed ?

(/cc @tbosch )

@jeffbcross
Copy link
Contributor

yes sorry I was too hasty :)

@btford btford modified the milestones: 1.3.0-rc.5, 1.3.0-rc.6 Oct 8, 2014
@Narretz
Copy link
Contributor

Narretz commented Aug 13, 2015

Closing as dupe of #9134

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

No branches or pull requests

8 participants