-
Notifications
You must be signed in to change notification settings - Fork 27.4k
Changing only part of date (input) does not change form's pristine state (Chrome) #12207
Comments
I did a little looking into here, it looks like if you type all of the date values (mm/dd/yyyy, say 03/14/2015), validation does occur. This may have to do with how partial date entry is validated - I suspect it has to do with either the fact that it is an input of type date. |
I've noted this issue also. Looking into angular.js I notices that validators are not trigger when a part of the date is changed or cleared. I've als noticed that clearing the complete date (by using backspace in each part) does not trigger any validator. When an input date is cleared I think it should return to its pristine and maybe untouched state. This does not happen. |
As explained in #11622 (comment), I think this is due to the fact that an invalid or partial date sets the input's value to an empty string. It is related to how the browser does/doesn't emit input events upon partial input. |
EDIT: See #12207 (comment). |
@gkalpak There's a condition inside $commitViewValue that forces validation etc. when the input has a native validator, even if the viewValue is empty: https://github.com/angular/angular.js/blob/master/src/ng/directive/ngModel.js#L652 The problem is that there's no change event in the following cases, because the browser does not register a change in the input's value (as gkalpak has pointed out)
I think |
@Narretz, I thought there was another issue, because my original POC (see #12207 (comment)) didn't work. It turned out to be a timing issue. So, triggering @Narretz, wdyt about emulating |
@gkalpak Yeah, let's give it a shot. Do you want to open a PR? And I wonder if we should restrict it to Chrome, because FF and IE don't have datepickers (at least on desktop) |
@Narretz, sure, PR is on the way. |
In date-family input types (`date`, `datetime-local`, `month`, `time`, `week`), the user can interact with parts of the value (e.g. year, month, hours etc) independently. Neverhteless, the actual value of the element is empty (`''`) unless all parts are filled (and valid). Thus, editing a signle part of the value may result in a change in the validity state of the `<input>` (see below), without an accompanying change in the actual value of the element. In such cases, no `input` event is fired by the browser to inform Angular of the change (and the need to re-validate). --- The following scenario describes a series of interactions that would run into the problem (on a browser that supports the `date` input type): 1. Initially empty field. - `input.value`: '' - `input.validity`: {valid: true, badInput: false, ...} 2. The user fills part of the value (e.g. the year) using the keyboard. - `input.value`: '' - `input.validity`: {valid: false, badInput: true, ...} - 'input' event: Not fired (since `input.value` hasn't changed) 3. The user fills the value completely (using either the keyboard or the date-picker). - `input.value`: '<some-valid-value>' - `input.validity`: {valid: true, badInput: false, ...} - 'input' event: Fired 4. The user deletes part of the value (e.g. the year) using the keyboard. - `input.value`: '' (since a partial value is invalid) - `input.validity`: {valid: false, badInput: true, ...} - 'input' event: Fired 5. The user deletes all parts of the value using the keyboard (i.e. clears the field). - `input.value`: '' - `input.validity`: {valid: true, badInput: false, ...} - 'input' event: Not fired (since `input.value` hasn't changed) The problematic cases are (2) and (5), because there is a change in the validity state, but no 'input' event is fired to inform Angular of that change and the need to re-validate. --- This commit fixes the issue by firing an `input` event on `keyup`, correctly triggering re-validation. Fixes angular#12207
In date-family input types (`date`, `datetime-local`, `month`, `time`, `week`), the user can interact with parts of the value (e.g. year, month, hours etc) independently. Neverhteless, the actual value of the element is empty (`''`) unless all parts are filled (and valid). Thus, editing a signle part of the value may result in a change in the validity state of the `<input>` (see below), without an accompanying change in the actual value of the element. In such cases, no `input` event is fired by the browser to inform Angular of the change (and the need to re-validate). --- The following scenario describes a series of interactions that would run into the problem (on a browser that supports the `date` input type): 1. Initially empty field. - `input.value`: '' - `input.validity`: {valid: true, badInput: false, ...} 2. The user fills part of the value (e.g. the year) using the keyboard. - `input.value`: '' - `input.validity`: {valid: false, badInput: true, ...} - 'input' event: Not fired (since `input.value` hasn't changed) 3. The user fills the value completely (using either the keyboard or the date-picker). - `input.value`: '<some-valid-value>' - `input.validity`: {valid: true, badInput: false, ...} - 'input' event: Fired 4. The user deletes part of the value (e.g. the year) using the keyboard. - `input.value`: '' (since a partial value is invalid) - `input.validity`: {valid: false, badInput: true, ...} - 'input' event: Fired 5. The user deletes all parts of the value using the keyboard (i.e. clears the field). - `input.value`: '' - `input.validity`: {valid: true, badInput: false, ...} - 'input' event: Not fired (since `input.value` hasn't changed) The problematic cases are (2) and (5), because there is a change in the validity state, but no 'input' event is fired to inform Angular of that change and the need to re-validate. --- This commit fixes the issue by firing an `input` event on `keyup`, correctly triggering re-validation. Fixes angular#12207
@gkalpak to add to your comment of 11 Sep 2015 you said: when adding data into the input, the change event is only fired after all field (day, month, year) have been filled. This is partially true, it depends which keys you use to edit the date. If I type the following (wrong) date in one flow then no validator is triggered: 31112015. Even if I leave the field it is still not triggered. When I return to the field and change the year it is not triggered. When I change the day or month the it gets striggered. |
@TimoPot, indeed, I was referring to filling the fields with valid data. If not, the input remains empty (aka unchanged). |
Ah missed that. Well actually the input is not empty after all I've just entered a value ;-) am I correct to notice that this issue is not solved yet? |
Also manually invoke $validate() or $setTouched() or $setDirty() on the date field does not help to get the date validated. Is there a known workaround? |
@TimoPot The problem is that for Angular the input is actually empty, because the browser does not set the value property of the input when invalid data is entered (for date and number), and therefore doesn't fire a change / input event. I think we need to listen to key events in addition to the input / change event. I believe @gkalpak worked on a PR for this, we should try to get it in after 1.5. |
The workaround I've used (and is more or less what is included in the PR) is to trigger an I'll resume work on the PR, so we can get this fixed soon. |
Oke. I assume the Demo pen will follow. |
@TimoPot, btw, it's not a proper solution, because one needs to take into account keys that do not change the input (e.g. modifiers etc) and possibly other stuff. But it's good enough for a quick workaround. |
Have a look at this plnkr.
When (in Chrome) I click the year and type something - say 2015 - and blur the field, model value does not change.
However, when I select the date from calendar and then I delete the year value (select it and hit backspace), then field & form error are set as expected:
Not sure if Angular's or Chrome's issue as in Firefox it is working fine (although there is no date picker).
The text was updated successfully, but these errors were encountered: