diff --git a/src/ng/directive/input.js b/src/ng/directive/input.js
index 6b93d55e67cb..81f98dcc8515 100644
--- a/src/ng/directive/input.js
+++ b/src/ng/directive/input.js
@@ -1721,6 +1721,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$
function($scope, $exceptionHandler, $attr, $element, $parse, $animate, $timeout, $rootScope, $q, $interpolate) {
this.$viewValue = Number.NaN;
this.$modelValue = Number.NaN;
+ this.$rawModelValue = undefined; // stores the parsed modelValue / model set from scope regardless of validity.
this.$validators = {};
this.$asyncValidators = {};
this.$parsers = [];
@@ -1975,14 +1976,51 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$
* @name ngModel.NgModelController#$validate
*
* @description
- * Runs each of the registered validators (first synchronous validators and then asynchronous validators).
+ * Runs each of the registered validators (first synchronous validators and then
+ * asynchronous validators).
+ * If the validity changes to invalid, the model will be set to `undefined`,
+ * unless {@link ngModelOptions `ngModelOptions.allowInvalid`} is `true`.
+ * If the validity changes to valid, it will set the model to the last available valid
+ * modelValue, i.e. either the last parsed value or the last value set from the scope.
*/
this.$validate = function() {
// ignore $validate before model is initialized
if (isNumber(ctrl.$modelValue) && isNaN(ctrl.$modelValue)) {
return;
}
- this.$$parseAndValidate();
+
+ var viewValue = ctrl.$$lastCommittedViewValue;
+ // Note: we use the $$rawModelValue as $modelValue might have been
+ // set to undefined during a view -> model update that found validation
+ // errors. We can't parse the view here, since that could change
+ // the model although neither viewValue nor the model on the scope changed
+ var modelValue = ctrl.$$rawModelValue;
+
+ // Check if the there's a parse error, so we don't unset it accidentially
+ var parserName = ctrl.$$parserName || 'parse';
+ var parserValid = ctrl.$error[parserName] ? false : undefined;
+
+ var prevValid = ctrl.$valid;
+ var prevModelValue = ctrl.$modelValue;
+
+ var allowInvalid = ctrl.$options && ctrl.$options.allowInvalid;
+
+ ctrl.$$runValidators(parserValid, modelValue, viewValue, function(allValid) {
+ // If there was no change in validity, don't update the model
+ // This prevents changing an invalid modelValue to undefined
+ if (!allowInvalid && prevValid !== allValid) {
+ // Note: Don't check ctrl.$valid here, as we could have
+ // external validators (e.g. calculated on the server),
+ // that just call $setValidity and need the model value
+ // to calculate their validity.
+ ctrl.$modelValue = allValid ? modelValue : undefined;
+
+ if (ctrl.$modelValue !== prevModelValue) {
+ ctrl.$$writeModelToScope();
+ }
+ }
+ });
+
};
this.$$runValidators = function(parseValid, modelValue, viewValue, doneCallback) {
@@ -2130,6 +2168,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$
}
var prevModelValue = ctrl.$modelValue;
var allowInvalid = ctrl.$options && ctrl.$options.allowInvalid;
+ ctrl.$$rawModelValue = modelValue;
if (allowInvalid) {
ctrl.$modelValue = modelValue;
writeToModelIfNeeded();
@@ -2254,7 +2293,7 @@ var NgModelController = ['$scope', '$exceptionHandler', '$attrs', '$element', '$
// if scope model value and ngModel value are out of sync
// TODO(perf): why not move this to the action fn?
if (modelValue !== ctrl.$modelValue) {
- ctrl.$modelValue = modelValue;
+ ctrl.$modelValue = ctrl.$$rawModelValue = modelValue;
var formatters = ctrl.$formatters,
idx = formatters.length;
diff --git a/test/ng/directive/inputSpec.js b/test/ng/directive/inputSpec.js
index e96022facf6e..b157ab9bbf58 100644
--- a/test/ng/directive/inputSpec.js
+++ b/test/ng/directive/inputSpec.js
@@ -484,72 +484,190 @@ describe('NgModelController', function() {
});
- describe('validations pipeline', function() {
+ describe('validation', function() {
- it('should perform validations when $validate() is called', function() {
- scope.$apply('value = ""');
+ describe('$validate', function() {
+ it('should perform validations when $validate() is called', function() {
+ scope.$apply('value = ""');
- var validatorResult = false;
- ctrl.$validators.someValidator = function(value) {
- return validatorResult;
- };
+ var validatorResult = false;
+ ctrl.$validators.someValidator = function(value) {
+ return validatorResult;
+ };
- ctrl.$validate();
+ ctrl.$validate();
- expect(ctrl.$valid).toBe(false);
+ expect(ctrl.$valid).toBe(false);
- validatorResult = true;
- ctrl.$validate();
+ validatorResult = true;
+ ctrl.$validate();
- expect(ctrl.$valid).toBe(true);
- });
+ expect(ctrl.$valid).toBe(true);
+ });
- it('should always perform validations using the parsed model value', function() {
- var captures;
- ctrl.$validators.raw = function() {
- captures = arguments;
- return captures[0];
- };
+ it('should pass the last parsed modelValue to the validators', function() {
+ ctrl.$parsers.push(function(modelValue) {
+ return modelValue + 'def';
+ });
- ctrl.$parsers.push(function(value) {
- return value.toUpperCase();
+ ctrl.$setViewValue('abc');
+
+ ctrl.$validators.test = function(modelValue, viewValue) {
+ return true;
+ };
+
+ spyOn(ctrl.$validators, 'test');
+
+ ctrl.$validate();
+
+ expect(ctrl.$validators.test).toHaveBeenCalledWith('abcdef', 'abc');
});
- ctrl.$setViewValue('my-value');
+ it('should set the model to undefined when it becomes invalid', function() {
+ var valid = true;
+ ctrl.$validators.test = function(modelValue, viewValue) {
+ return valid;
+ };
- expect(captures).toEqual(['MY-VALUE', 'my-value']);
- });
+ scope.$apply('value = "abc"');
+ expect(scope.value).toBe('abc');
- it('should always perform validations using the formatted view value', function() {
- var captures;
- ctrl.$validators.raw = function() {
- captures = arguments;
- return captures[0];
- };
+ valid = false;
+ ctrl.$validate();
- ctrl.$formatters.push(function(value) {
- return value + '...';
+ expect(scope.value).toBeUndefined();
+ });
+
+ it('should update the model when it becomes valid', function() {
+ var valid = true;
+ ctrl.$validators.test = function(modelValue, viewValue) {
+ return valid;
+ };
+
+ scope.$apply('value = "abc"');
+ expect(scope.value).toBe('abc');
+
+ valid = false;
+ ctrl.$validate();
+ expect(scope.value).toBeUndefined();
+
+ valid = true;
+ ctrl.$validate();
+ expect(scope.value).toBe('abc');
+ });
+
+ it('should not update the model when it is valid, but there is a parse error', function() {
+ ctrl.$parsers.push(function(modelValue) {
+ return undefined;
+ });
+
+ ctrl.$setViewValue('abc');
+ expect(ctrl.$error.parse).toBe(true);
+ expect(scope.value).toBeUndefined();
+
+ ctrl.$validators.test = function(modelValue, viewValue) {
+ return true;
+ };
+
+ ctrl.$validate();
+ expect(ctrl.$error).toEqual({parse: true});
+ expect(scope.value).toBeUndefined();
+ });
+
+ it('should not set an invalid model to undefined when validity is the same', function() {
+ ctrl.$validators.test = function() {
+ return false;
+ };
+
+ scope.$apply('value = "invalid"');
+ expect(ctrl.$valid).toBe(false);
+ expect(scope.value).toBe('invalid');
+
+ ctrl.$validate();
+ expect(ctrl.$valid).toBe(false);
+ expect(scope.value).toBe('invalid');
});
- scope.$apply('value = "matias"');
+ it('should not change a model that has a formatter', function() {
+ ctrl.$validators.test = function() {
+ return true;
+ };
+
+ ctrl.$formatters.push(function(modelValue) {
+ return 'xyz';
+ });
+
+ scope.$apply('value = "abc"');
+ expect(ctrl.$viewValue).toBe('xyz');
+
+ ctrl.$validate();
+ expect(scope.value).toBe('abc');
+ });
+
+ it('should not change a model that has a parser', function() {
+ ctrl.$validators.test = function() {
+ return true;
+ };
+
+ ctrl.$parsers.push(function(modelValue) {
+ return 'xyz';
+ });
+
+ scope.$apply('value = "abc"');
- expect(captures).toEqual(['matias', 'matias...']);
+ ctrl.$validate();
+ expect(scope.value).toBe('abc');
+ });
});
- it('should only perform validations if the view value is different', function() {
- var count = 0;
- ctrl.$validators.countMe = function() {
- count++;
- };
+ describe('view -> model update', function() {
+ it('should always perform validations using the parsed model value', function() {
+ var captures;
+ ctrl.$validators.raw = function() {
+ captures = arguments;
+ return captures[0];
+ };
- ctrl.$setViewValue('my-value');
- expect(count).toBe(1);
+ ctrl.$parsers.push(function(value) {
+ return value.toUpperCase();
+ });
- ctrl.$setViewValue('my-value');
- expect(count).toBe(1);
+ ctrl.$setViewValue('my-value');
- ctrl.$setViewValue('your-value');
- expect(count).toBe(2);
+ expect(captures).toEqual(['MY-VALUE', 'my-value']);
+ });
+
+ it('should always perform validations using the formatted view value', function() {
+ var captures;
+ ctrl.$validators.raw = function() {
+ captures = arguments;
+ return captures[0];
+ };
+
+ ctrl.$formatters.push(function(value) {
+ return value + '...';
+ });
+
+ scope.$apply('value = "matias"');
+
+ expect(captures).toEqual(['matias', 'matias...']);
+ });
+
+ it('should only perform validations if the view value is different', function() {
+ var count = 0;
+ ctrl.$validators.countMe = function() {
+ count++;
+ };
+
+ ctrl.$setViewValue('my-value');
+ expect(count).toBe(1);
+
+ ctrl.$setViewValue('my-value');
+ expect(count).toBe(1);
+
+ ctrl.$setViewValue('your-value');
+ expect(count).toBe(2);
+ });
});
it('should perform validations twice each time the model value changes within a digest', function() {
@@ -934,6 +1052,7 @@ describe('NgModelController', function() {
});
});
+
describe('ngModel', function() {
var EMAIL_REGEXP = /^[a-z0-9!#$%&'*+\/=?^_`{|}~.-]+@[a-z0-9]([a-z0-9-]*[a-z0-9])?(\.[a-z0-9]([a-z0-9-]*[a-z0-9])?)*$/i;
@@ -1348,7 +1467,6 @@ describe('input', function() {
expect(scope.form.$$renameControl).not.toHaveBeenCalled();
});
-
describe('compositionevents', function() {
it('should not update the model between "compositionstart" and "compositionend" on non android', inject(function($sniffer) {
$sniffer.android = false;
@@ -2311,6 +2429,14 @@ describe('input', function() {
expect(inputElm).toBeValid();
expect(scope.form.input.$error.minlength).not.toBe(true);
});
+
+ it('should validate when the model is initalized as a number', function() {
+ scope.value = 12345;
+ compileInput('');
+ expect(scope.value).toBe(12345);
+ expect(scope.form.input.$error.minlength).toBeUndefined();
+ });
+
});
@@ -2409,6 +2535,13 @@ describe('input', function() {
expect(scope.value).toBe('12345');
});
+ it('should validate when the model is initalized as a number', function() {
+ scope.value = 12345;
+ compileInput('');
+ expect(scope.value).toBe(12345);
+ expect(scope.form.input.$error.maxlength).toBeUndefined();
+ });
+
});