Files
angular.js/test/ng/directive/inputSpec.js
Peter Bacon Darwin 3e42b22b0e refact(input): split input.js into smaller files
The input.js file is unnecessarily large, containing many directives including the
vast `ngModel`. This change moves ngModel and a few other directives into their
own files, which will make maintenance easier.
2014-12-24 13:07:07 +00:00

3765 lines
119 KiB
JavaScript

'use strict';
describe('input', function() {
var formElm, inputElm, scope, $compile, $sniffer, $browser, changeInputValueTo, currentSpec;
function compileInput(inputHtml, mockValidity) {
inputElm = jqLite(inputHtml);
if (isObject(mockValidity)) {
VALIDITY_STATE_PROPERTY = 'ngMockValidity';
inputElm.prop(VALIDITY_STATE_PROPERTY, mockValidity);
currentSpec.after(function() {
VALIDITY_STATE_PROPERTY = 'validity';
});
}
formElm = jqLite('<form name="form"></form>');
formElm.append(inputElm);
$compile(formElm)(scope);
scope.$digest();
}
var attrs;
beforeEach(function() { currentSpec = this; });
afterEach(function() { currentSpec = null; });
beforeEach(module(function($compileProvider) {
$compileProvider.directive('attrCapture', function() {
return function(scope, element, $attrs) {
attrs = $attrs;
};
});
}));
beforeEach(inject(function($injector, _$sniffer_, _$browser_) {
$sniffer = _$sniffer_;
$browser = _$browser_;
$compile = $injector.get('$compile');
scope = $injector.get('$rootScope');
changeInputValueTo = function(value) {
inputElm.val(value);
browserTrigger(inputElm, $sniffer.hasEvent('input') ? 'input' : 'change');
};
}));
afterEach(function() {
dealoc(formElm);
});
it('should bind to a model', function() {
compileInput('<input type="text" ng-model="name" name="alias" ng-change="change()" />');
scope.$apply("name = 'misko'");
expect(inputElm.val()).toBe('misko');
});
it('should not set readonly or disabled property on ie7', function() {
this.addMatchers({
toBeOff: function(attributeName) {
var actualValue = this.actual.attr(attributeName);
this.message = function() {
return "Attribute '" + attributeName + "' expected to be off but was '" + actualValue +
"' in: " + angular.mock.dump(this.actual);
};
return !actualValue || actualValue == 'false';
}
});
compileInput('<input type="text" ng-model="name" name="alias"/>');
expect(inputElm.prop('readOnly')).toBe(false);
expect(inputElm.prop('disabled')).toBe(false);
expect(inputElm).toBeOff('readOnly');
expect(inputElm).toBeOff('readonly');
expect(inputElm).toBeOff('disabled');
});
it('should update the model on "blur" event', function() {
compileInput('<input type="text" ng-model="name" name="alias" ng-change="change()" />');
changeInputValueTo('adam');
expect(scope.name).toEqual('adam');
});
it('should not add the property to the scope if name is unspecified', function() {
inputElm = jqLite('<input type="text" ng-model="name">');
formElm = jqLite('<form name="form"></form>');
formElm.append(inputElm);
$compile(formElm)(scope);
spyOn(scope.form, '$addControl').andCallThrough();
spyOn(scope.form, '$$renameControl').andCallThrough();
scope.$digest();
expect(scope.form['undefined']).toBeUndefined();
expect(scope.form.$addControl).not.toHaveBeenCalled();
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;
compileInput('<input type="text" ng-model="name" name="alias"" />');
changeInputValueTo('a');
expect(scope.name).toEqual('a');
browserTrigger(inputElm, 'compositionstart');
changeInputValueTo('adam');
expect(scope.name).toEqual('a');
browserTrigger(inputElm, 'compositionend');
changeInputValueTo('adam');
expect(scope.name).toEqual('adam');
}));
it('should update the model between "compositionstart" and "compositionend" on android', inject(function($sniffer) {
$sniffer.android = true;
compileInput('<input type="text" ng-model="name" name="alias"" />');
changeInputValueTo('a');
expect(scope.name).toEqual('a');
browserTrigger(inputElm, 'compositionstart');
changeInputValueTo('adam');
expect(scope.name).toEqual('adam');
browserTrigger(inputElm, 'compositionend');
changeInputValueTo('adam2');
expect(scope.name).toEqual('adam2');
}));
});
it('should update the model on "compositionend"', function() {
compileInput('<input type="text" ng-model="name" name="alias" />');
browserTrigger(inputElm, 'compositionstart');
changeInputValueTo('caitp');
expect(scope.name).toBeUndefined();
browserTrigger(inputElm, 'compositionend');
expect(scope.name).toEqual('caitp');
});
describe("IE placeholder input events", function() {
//IE fires an input event whenever a placeholder visually changes, essentially treating it as a value
//Events:
// placeholder attribute change: *input*
// focus (which visually removes the placeholder value): focusin focus *input*
// blur (which visually creates the placeholder value): focusout *input* blur
//However none of these occur if the placeholder is not visible at the time of the event.
//These tests try simulate various scenerios which do/do-not fire the extra input event
it('should not dirty the model on an input event in response to a placeholder change', function() {
compileInput('<input type="text" placeholder="Test" attr-capture ng-model="unsetValue" name="name" />');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('Test');
expect(inputElm).toBePristine();
attrs.$set('placeholder', '');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('');
expect(inputElm).toBePristine();
attrs.$set('placeholder', 'Test Again');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('Test Again');
expect(inputElm).toBePristine();
attrs.$set('placeholder', undefined);
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe(undefined);
expect(inputElm).toBePristine();
changeInputValueTo('foo');
expect(inputElm).toBeDirty();
});
it('should not dirty the model on an input event in response to a interpolated placeholder change', inject(function($rootScope) {
compileInput('<input type="text" placeholder="{{ph}}" ng-model="unsetValue" name="name" />');
msie && browserTrigger(inputElm, 'input');
expect(inputElm).toBePristine();
$rootScope.ph = 1;
$rootScope.$digest();
msie && browserTrigger(inputElm, 'input');
expect(inputElm).toBePristine();
$rootScope.ph = "";
$rootScope.$digest();
msie && browserTrigger(inputElm, 'input');
expect(inputElm).toBePristine();
changeInputValueTo('foo');
expect(inputElm).toBeDirty();
}));
it('should dirty the model on an input event while in focus even if the placeholder changes', inject(function($rootScope) {
$rootScope.ph = 'Test';
compileInput('<input type="text" ng-attr-placeholder="{{ph}}" ng-model="unsetValue" name="name" />');
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusin');
browserTrigger(inputElm, 'focus');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('Test');
expect(inputElm).toBePristine();
$rootScope.ph = 'Test Again';
$rootScope.$digest();
expect(inputElm).toBePristine();
changeInputValueTo('foo');
expect(inputElm).toBeDirty();
}));
it('should not dirty the model on an input event in response to a ng-attr-placeholder change', inject(function($rootScope) {
compileInput('<input type="text" ng-attr-placeholder="{{ph}}" ng-model="unsetValue" name="name" />');
expect(inputElm).toBePristine();
$rootScope.ph = 1;
$rootScope.$digest();
msie && browserTrigger(inputElm, 'input');
expect(inputElm).toBePristine();
$rootScope.ph = "";
$rootScope.$digest();
msie && browserTrigger(inputElm, 'input');
expect(inputElm).toBePristine();
changeInputValueTo('foo');
expect(inputElm).toBeDirty();
}));
it('should not dirty the model on an input event in response to a focus', inject(function($sniffer) {
compileInput('<input type="text" placeholder="Test" ng-model="unsetValue" name="name" />');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('Test');
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusin');
browserTrigger(inputElm, 'focus');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('Test');
expect(inputElm).toBePristine();
changeInputValueTo('foo');
expect(inputElm).toBeDirty();
}));
it('should not dirty the model on an input event in response to a blur', inject(function($sniffer) {
compileInput('<input type="text" placeholder="Test" ng-model="unsetValue" name="name" />');
msie && browserTrigger(inputElm, 'input');
expect(inputElm.attr('placeholder')).toBe('Test');
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusin');
browserTrigger(inputElm, 'focus');
msie && browserTrigger(inputElm, 'input');
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusout');
msie && browserTrigger(inputElm, 'input');
browserTrigger(inputElm, 'blur');
expect(inputElm).toBePristine();
changeInputValueTo('foo');
expect(inputElm).toBeDirty();
}));
it('should dirty the model on an input event if there is a placeholder and value', inject(function($rootScope) {
$rootScope.name = 'foo';
compileInput('<input type="text" placeholder="Test" ng-model="name" value="init" name="name" />');
expect(inputElm.val()).toBe($rootScope.name);
expect(inputElm).toBePristine();
changeInputValueTo('bar');
expect(inputElm).toBeDirty();
}));
it('should dirty the model on an input event if there is a placeholder and value after focusing', inject(function($rootScope) {
$rootScope.name = 'foo';
compileInput('<input type="text" placeholder="Test" ng-model="name" value="init" name="name" />');
expect(inputElm.val()).toBe($rootScope.name);
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusin');
browserTrigger(inputElm, 'focus');
changeInputValueTo('bar');
expect(inputElm).toBeDirty();
}));
it('should dirty the model on an input event if there is a placeholder and value after bluring', inject(function($rootScope) {
$rootScope.name = 'foo';
compileInput('<input type="text" placeholder="Test" ng-model="name" value="init" name="name" />');
expect(inputElm.val()).toBe($rootScope.name);
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusin');
browserTrigger(inputElm, 'focus');
expect(inputElm).toBePristine();
browserTrigger(inputElm, 'focusout');
browserTrigger(inputElm, 'blur');
changeInputValueTo('bar');
expect(inputElm).toBeDirty();
}));
});
it('should interpolate input names', function() {
scope.nameID = '47';
compileInput('<input type="text" ng-model="name" name="name{{nameID}}" />');
expect(scope.form.name47.$pristine).toBeTruthy();
changeInputValueTo('caitp');
expect(scope.form.name47.$dirty).toBeTruthy();
});
it('should rename form controls in form when interpolated name changes', function() {
scope.nameID = "A";
compileInput('<input type="text" ng-model="name" name="name{{nameID}}" />');
expect(scope.form.nameA.$name).toBe('nameA');
var oldModel = scope.form.nameA;
scope.nameID = "B";
scope.$digest();
expect(scope.form.nameA).toBeUndefined();
expect(scope.form.nameB).toBe(oldModel);
expect(scope.form.nameB.$name).toBe('nameB');
});
it('should rename form controls in null form when interpolated name changes', function() {
var element = $compile('<input type="text" ng-model="name" name="name{{nameID}}" />')(scope);
scope.nameID = "A";
scope.$digest();
var model = element.controller('ngModel');
expect(model.$name).toBe('nameA');
scope.nameID = "B";
scope.$digest();
expect(model.$name).toBe('nameB');
});
describe('"change" event', function() {
function assertBrowserSupportsChangeEvent(inputEventSupported) {
// Force browser to report a lack of an 'input' event
$sniffer.hasEvent = function(eventName) {
if (eventName === 'input' && !inputEventSupported) {
return false;
}
return true;
};
compileInput('<input type="text" ng-model="name" name="alias" />');
inputElm.val('mark');
browserTrigger(inputElm, 'change');
expect(scope.name).toEqual('mark');
}
it('should update the model event if the browser does not support the "input" event',function() {
assertBrowserSupportsChangeEvent(false);
});
it('should update the model event if the browser supports the "input" ' +
'event so that form auto complete works',function() {
assertBrowserSupportsChangeEvent(true);
});
if (!_jqLiteMode) {
describe('double $digest when triggering an event using jQuery', function() {
function run() {
$sniffer.hasEvent = function(eventName) {
return eventName !== 'input';
};
compileInput('<input type="text" ng-model="name" name="alias" ng-change="change()" />');
scope.field = 'fake field';
scope.$watch('field', function() {
// We need to use _originalTrigger since trigger is modified by Angular Scenario.
inputElm._originalTrigger('change');
});
scope.$apply();
}
it('should not cause the double $digest with non isolate scopes', function() {
run();
});
it('should not cause the double $digest with isolate scopes', function() {
scope = scope.$new(true);
run();
});
});
}
});
describe('"keydown", "paste" and "cut" events', function() {
beforeEach(function() {
// Force browser to report a lack of an 'input' event
$sniffer.hasEvent = function(eventName) {
return eventName !== 'input';
};
});
it('should update the model on "paste" event if the input value changes', function() {
compileInput('<input type="text" ng-model="name" name="alias" ng-change="change()" />');
browserTrigger(inputElm, 'keydown');
$browser.defer.flush();
expect(inputElm).toBePristine();
inputElm.val('mark');
browserTrigger(inputElm, 'paste');
$browser.defer.flush();
expect(scope.name).toEqual('mark');
});
it('should update the model on "cut" event', function() {
compileInput('<input type="text" ng-model="name" name="alias" ng-change="change()" />');
inputElm.val('john');
browserTrigger(inputElm, 'cut');
$browser.defer.flush();
expect(scope.name).toEqual('john');
});
it('should cancel the delayed dirty if a change occurs', function() {
compileInput('<input type="text" ng-model="name" />');
var ctrl = inputElm.controller('ngModel');
browserTrigger(inputElm, 'keydown', {target: inputElm[0]});
inputElm.val('f');
browserTrigger(inputElm, 'change');
expect(inputElm).toBeDirty();
ctrl.$setPristine();
scope.$apply();
$browser.defer.flush();
expect(inputElm).toBePristine();
});
});
it('should update the model and trim the value', function() {
compileInput('<input type="text" ng-model="name" name="alias" ng-change="change()" />');
changeInputValueTo(' a ');
expect(scope.name).toEqual('a');
});
it('should update the model and not trim the value', function() {
compileInput('<input type="text" ng-model="name" name="alias" ng-trim="false" />');
changeInputValueTo(' a ');
expect(scope.name).toEqual(' a ');
});
describe('ngModelOptions attributes', function() {
it('should allow overriding the model update trigger event on text inputs', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur\' }"' +
'/>');
changeInputValueTo('a');
expect(scope.name).toBeUndefined();
browserTrigger(inputElm, 'blur');
expect(scope.name).toEqual('a');
});
it('should not dirty the input if nothing was changed before updateOn trigger', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur\' }"' +
'/>');
browserTrigger(inputElm, 'blur');
expect(scope.form.alias.$pristine).toBeTruthy();
});
it('should allow overriding the model update trigger event on text areas', function() {
compileInput(
'<textarea ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur\' }"' +
'/>');
changeInputValueTo('a');
expect(scope.name).toBeUndefined();
browserTrigger(inputElm, 'blur');
expect(scope.name).toEqual('a');
});
it('should bind the element to a list of events', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur mousemove\' }"' +
'/>');
changeInputValueTo('a');
expect(scope.name).toBeUndefined();
browserTrigger(inputElm, 'blur');
expect(scope.name).toEqual('a');
changeInputValueTo('b');
expect(scope.name).toEqual('a');
browserTrigger(inputElm, 'mousemove');
expect(scope.name).toEqual('b');
});
it('should allow keeping the default update behavior on text inputs', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'default\' }"' +
'/>');
changeInputValueTo('a');
expect(scope.name).toEqual('a');
});
it('should allow overriding the model update trigger event on checkboxes', function() {
compileInput(
'<input type="checkbox" ng-model="checkbox" ' +
'ng-model-options="{ updateOn: \'blur\' }"' +
'/>');
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(undefined);
browserTrigger(inputElm, 'blur');
expect(scope.checkbox).toBe(true);
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(true);
});
it('should allow keeping the default update behavior on checkboxes', function() {
compileInput(
'<input type="checkbox" ng-model="checkbox" ' +
'ng-model-options="{ updateOn: \'blur default\' }"' +
'/>');
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(true);
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(false);
});
it('should allow overriding the model update trigger event on radio buttons', function() {
compileInput(
'<input type="radio" ng-model="color" value="white" ' +
'ng-model-options="{ updateOn: \'blur\'}"' +
'/>' +
'<input type="radio" ng-model="color" value="red" ' +
'ng-model-options="{ updateOn: \'blur\'}"' +
'/>' +
'<input type="radio" ng-model="color" value="blue" ' +
'ng-model-options="{ updateOn: \'blur\'}"' +
'/>');
scope.$apply("color = 'white'");
browserTrigger(inputElm[2], 'click');
expect(scope.color).toBe('white');
browserTrigger(inputElm[2], 'blur');
expect(scope.color).toBe('blue');
});
it('should allow keeping the default update behavior on radio buttons', function() {
compileInput(
'<input type="radio" ng-model="color" value="white" ' +
'ng-model-options="{ updateOn: \'blur default\' }"' +
'/>' +
'<input type="radio" ng-model="color" value="red" ' +
'ng-model-options="{ updateOn: \'blur default\' }"' +
'/>' +
'<input type="radio" ng-model="color" value="blue" ' +
'ng-model-options="{ updateOn: \'blur default\' }"' +
'/>');
scope.$apply("color = 'white'");
browserTrigger(inputElm[2], 'click');
expect(scope.color).toBe('blue');
});
it('should trigger only after timeout in text inputs', inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ debounce: 10000 }"' +
'/>');
changeInputValueTo('a');
changeInputValueTo('b');
changeInputValueTo('c');
expect(scope.name).toEqual(undefined);
$timeout.flush(2000);
expect(scope.name).toEqual(undefined);
$timeout.flush(9000);
expect(scope.name).toEqual('c');
}));
it('should trigger only after timeout in checkboxes', inject(function($timeout) {
compileInput(
'<input type="checkbox" ng-model="checkbox" ' +
'ng-model-options="{ debounce: 10000 }"' +
'/>');
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(undefined);
$timeout.flush(2000);
expect(scope.checkbox).toBe(undefined);
$timeout.flush(9000);
expect(scope.checkbox).toBe(true);
}));
it('should trigger only after timeout in radio buttons', inject(function($timeout) {
compileInput(
'<input type="radio" ng-model="color" value="white" />' +
'<input type="radio" ng-model="color" value="red" ' +
'ng-model-options="{ debounce: 20000 }"' +
'/>' +
'<input type="radio" ng-model="color" value="blue" ' +
'ng-model-options="{ debounce: 30000 }"' +
'/>');
browserTrigger(inputElm[0], 'click');
expect(scope.color).toBe('white');
browserTrigger(inputElm[1], 'click');
expect(scope.color).toBe('white');
$timeout.flush(12000);
expect(scope.color).toBe('white');
$timeout.flush(10000);
expect(scope.color).toBe('red');
}));
it('should not trigger digest while debouncing', inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ debounce: 10000 }"' +
'/>');
var watchSpy = jasmine.createSpy('watchSpy');
scope.$watch(watchSpy);
changeInputValueTo('a');
expect(watchSpy).not.toHaveBeenCalled();
$timeout.flush(10000);
expect(watchSpy).toHaveBeenCalled();
}));
it('should allow selecting different debounce timeouts for each event',
inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{' +
'updateOn: \'default blur\', ' +
'debounce: { default: 10000, blur: 5000 }' +
'}"' +
'/>');
changeInputValueTo('a');
expect(scope.checkbox).toBe(undefined);
$timeout.flush(6000);
expect(scope.checkbox).toBe(undefined);
$timeout.flush(4000);
expect(scope.name).toEqual('a');
changeInputValueTo('b');
browserTrigger(inputElm, 'blur');
$timeout.flush(4000);
expect(scope.name).toEqual('a');
$timeout.flush(2000);
expect(scope.name).toEqual('b');
}));
it('should allow selecting different debounce timeouts for each event on checkboxes', inject(function($timeout) {
compileInput('<input type="checkbox" ng-model="checkbox" ' +
'ng-model-options="{ ' +
'updateOn: \'default blur\', debounce: { default: 10000, blur: 5000 } }"' +
'/>');
inputElm[0].checked = false;
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(undefined);
$timeout.flush(8000);
expect(scope.checkbox).toBe(undefined);
$timeout.flush(3000);
expect(scope.checkbox).toBe(true);
inputElm[0].checked = true;
browserTrigger(inputElm, 'click');
browserTrigger(inputElm, 'blur');
$timeout.flush(3000);
expect(scope.checkbox).toBe(true);
$timeout.flush(3000);
expect(scope.checkbox).toBe(false);
}));
it('should allow selecting 0 for non-default debounce timeouts for each event on checkboxes', inject(function($timeout) {
compileInput('<input type="checkbox" ng-model="checkbox" ' +
'ng-model-options="{ ' +
'updateOn: \'default blur\', debounce: { default: 10000, blur: 0 } }"' +
'/>');
inputElm[0].checked = false;
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(undefined);
$timeout.flush(8000);
expect(scope.checkbox).toBe(undefined);
$timeout.flush(3000);
expect(scope.checkbox).toBe(true);
inputElm[0].checked = true;
browserTrigger(inputElm, 'click');
browserTrigger(inputElm, 'blur');
$timeout.flush(0);
expect(scope.checkbox).toBe(false);
}));
it('should inherit model update settings from ancestor elements', inject(function($timeout) {
var doc = $compile(
'<form name="test" ' +
'ng-model-options="{ debounce: 10000, updateOn: \'blur\' }" >' +
'<input type="text" ng-model="name" name="alias" />' +
'</form>')(scope);
scope.$digest();
inputElm = doc.find('input').eq(0);
changeInputValueTo('a');
expect(scope.name).toEqual(undefined);
browserTrigger(inputElm, 'blur');
expect(scope.name).toBe(undefined);
$timeout.flush(2000);
expect(scope.name).toBe(undefined);
$timeout.flush(9000);
expect(scope.name).toEqual('a');
dealoc(doc);
}));
it('should flush debounced events when calling $commitViewValue directly', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ debounce: 1000 }" />');
changeInputValueTo('a');
expect(scope.name).toEqual(undefined);
scope.form.alias.$commitViewValue();
expect(scope.name).toEqual('a');
});
it('should cancel debounced events when calling $commitViewValue', inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ debounce: 1000 }"/>');
changeInputValueTo('a');
scope.form.alias.$commitViewValue();
expect(scope.name).toEqual('a');
scope.form.alias.$setPristine();
$timeout.flush(1000);
expect(scope.form.alias.$pristine).toBeTruthy();
}));
it('should reset input val if rollbackViewValue called during pending update', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur\' }" />');
changeInputValueTo('a');
expect(inputElm.val()).toBe('a');
scope.form.alias.$rollbackViewValue();
expect(inputElm.val()).toBe('');
browserTrigger(inputElm, 'blur');
expect(inputElm.val()).toBe('');
});
it('should allow canceling pending updates', inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur\' }" />');
changeInputValueTo('a');
expect(scope.name).toEqual(undefined);
scope.form.alias.$rollbackViewValue();
expect(scope.name).toEqual(undefined);
browserTrigger(inputElm, 'blur');
expect(scope.name).toEqual(undefined);
}));
it('should allow canceling debounced updates', inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ debounce: 10000 }" />');
changeInputValueTo('a');
expect(scope.name).toEqual(undefined);
$timeout.flush(2000);
scope.form.alias.$rollbackViewValue();
expect(scope.name).toEqual(undefined);
$timeout.flush(10000);
expect(scope.name).toEqual(undefined);
}));
it('should handle model updates correctly even if rollbackViewValue is not invoked', function() {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ updateOn: \'blur\' }" />');
changeInputValueTo('a');
scope.$apply("name = 'b'");
browserTrigger(inputElm, 'blur');
expect(scope.name).toBe('b');
});
it('should reset input val if rollbackViewValue called during debounce', inject(function($timeout) {
compileInput(
'<input type="text" ng-model="name" name="alias" ' +
'ng-model-options="{ debounce: 2000 }" />');
changeInputValueTo('a');
expect(inputElm.val()).toBe('a');
scope.form.alias.$rollbackViewValue();
expect(inputElm.val()).toBe('');
$timeout.flush(3000);
expect(inputElm.val()).toBe('');
}));
it('should not try to invoke a model if getterSetter is false', function() {
compileInput(
'<input type="text" ng-model="name" ' +
'ng-model-options="{ getterSetter: false }" />');
var spy = scope.name = jasmine.createSpy('setterSpy');
changeInputValueTo('a');
expect(spy).not.toHaveBeenCalled();
expect(inputElm.val()).toBe('a');
});
it('should not try to invoke a model if getterSetter is not set', function() {
compileInput('<input type="text" ng-model="name" />');
var spy = scope.name = jasmine.createSpy('setterSpy');
changeInputValueTo('a');
expect(spy).not.toHaveBeenCalled();
expect(inputElm.val()).toBe('a');
});
it('should try to invoke a function model if getterSetter is true', function() {
compileInput(
'<input type="text" ng-model="name" ' +
'ng-model-options="{ getterSetter: true }" />');
var spy = scope.name = jasmine.createSpy('setterSpy').andCallFake(function() {
return 'b';
});
scope.$apply();
expect(inputElm.val()).toBe('b');
changeInputValueTo('a');
expect(inputElm.val()).toBe('b');
expect(spy).toHaveBeenCalledWith('a');
expect(scope.name).toBe(spy);
});
it('should assign to non-function models if getterSetter is true', function() {
compileInput(
'<input type="text" ng-model="name" ' +
'ng-model-options="{ getterSetter: true }" />');
scope.name = 'c';
changeInputValueTo('d');
expect(inputElm.val()).toBe('d');
expect(scope.name).toBe('d');
});
it('should fail on non-assignable model binding if getterSetter is false', function() {
expect(function() {
compileInput('<input type="text" ng-model="accessor(user, \'name\')" />');
}).toThrowMinErr('ngModel', 'nonassign', 'Expression \'accessor(user, \'name\')\' is non-assignable.');
});
it('should not fail on non-assignable model binding if getterSetter is true', function() {
compileInput(
'<input type="text" ng-model="accessor(user, \'name\')" ' +
'ng-model-options="{ getterSetter: true }" />');
});
it('should invoke a model in the correct context if getterSetter is true', function() {
compileInput(
'<input type="text" ng-model="someService.getterSetter" ' +
'ng-model-options="{ getterSetter: true }" />');
scope.someService = {
value: 'a',
getterSetter: function(newValue) {
this.value = newValue || this.value;
return this.value;
}
};
spyOn(scope.someService, 'getterSetter').andCallThrough();
scope.$apply();
expect(inputElm.val()).toBe('a');
expect(scope.someService.getterSetter).toHaveBeenCalledWith();
expect(scope.someService.value).toBe('a');
changeInputValueTo('b');
expect(scope.someService.getterSetter).toHaveBeenCalledWith('b');
expect(scope.someService.value).toBe('b');
scope.someService.value = 'c';
scope.$apply();
expect(inputElm.val()).toBe('c');
expect(scope.someService.getterSetter).toHaveBeenCalledWith();
});
it('should assign invalid values to the scope if allowInvalid is true', function() {
compileInput('<input type="text" name="input" ng-model="value" maxlength="1" ' +
'ng-model-options="{allowInvalid: true}" />');
changeInputValueTo('12345');
expect(scope.value).toBe('12345');
expect(inputElm).toBeInvalid();
});
it('should not assign not parsable values to the scope if allowInvalid is true', function() {
compileInput('<input type="number" name="input" ng-model="value" ' +
'ng-model-options="{allowInvalid: true}" />', {
valid: false,
badInput: true
});
changeInputValueTo('abcd');
expect(scope.value).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should update the scope before async validators execute if allowInvalid is true', inject(function($q) {
compileInput('<input type="text" name="input" ng-model="value" ' +
'ng-model-options="{allowInvalid: true}" />');
var defer;
scope.form.input.$asyncValidators.promiseValidator = function(value) {
defer = $q.defer();
return defer.promise;
};
changeInputValueTo('12345');
expect(scope.value).toBe('12345');
expect(scope.form.input.$pending.promiseValidator).toBe(true);
defer.reject();
scope.$digest();
expect(scope.value).toBe('12345');
expect(inputElm).toBeInvalid();
}));
it('should update the view before async validators execute if allowInvalid is true', inject(function($q) {
compileInput('<input type="text" name="input" ng-model="value" ' +
'ng-model-options="{allowInvalid: true}" />');
var defer;
scope.form.input.$asyncValidators.promiseValidator = function(value) {
defer = $q.defer();
return defer.promise;
};
scope.$apply('value = \'12345\'');
expect(inputElm.val()).toBe('12345');
expect(scope.form.input.$pending.promiseValidator).toBe(true);
defer.reject();
scope.$digest();
expect(inputElm.val()).toBe('12345');
expect(inputElm).toBeInvalid();
}));
it('should not call ng-change listeners twice if the model did not change with allowInvalid', function() {
compileInput('<input type="text" name="input" ng-model="value" ' +
'ng-model-options="{allowInvalid: true}" ng-change="changed()" />');
scope.changed = jasmine.createSpy('changed');
scope.form.input.$parsers.push(function(value) {
return 'modelValue';
});
changeInputValueTo('input1');
expect(scope.value).toBe('modelValue');
expect(scope.changed).toHaveBeenCalledOnce();
changeInputValueTo('input2');
expect(scope.value).toBe('modelValue');
expect(scope.changed).toHaveBeenCalledOnce();
});
});
it('should allow complex reference binding', function() {
compileInput('<input type="text" ng-model="obj[\'abc\'].name"/>');
scope.$apply("obj = { abc: { name: 'Misko'} }");
expect(inputElm.val()).toEqual('Misko');
});
it('should ignore input without ngModel directive', function() {
compileInput('<input type="text" name="whatever" required />');
changeInputValueTo('');
expect(inputElm.hasClass('ng-valid')).toBe(false);
expect(inputElm.hasClass('ng-invalid')).toBe(false);
expect(inputElm.hasClass('ng-pristine')).toBe(false);
expect(inputElm.hasClass('ng-dirty')).toBe(false);
});
it('should report error on assignment error', function() {
expect(function() {
compileInput('<input type="text" ng-model="throw \'\'">');
}).toThrowMinErr("$parse", "syntax", "Syntax Error: Token '''' is an unexpected token at column 7 of the expression [throw ''] starting at [''].");
});
it("should render as blank if null", function() {
compileInput('<input type="text" ng-model="age" />');
scope.$apply('age = null');
expect(scope.age).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should render 0 even if it is a number', function() {
compileInput('<input type="text" ng-model="value" />');
scope.$apply('value = 0');
expect(inputElm.val()).toBe('0');
});
it('should render the $viewValue when $modelValue is empty', function() {
compileInput('<input type="text" ng-model="value" />');
var ctrl = inputElm.controller('ngModel');
ctrl.$modelValue = null;
expect(ctrl.$isEmpty(ctrl.$modelValue)).toBe(true);
ctrl.$viewValue = 'abc';
ctrl.$render();
expect(inputElm.val()).toBe('abc');
});
describe('pattern', function() {
it('should validate in-lined pattern', function() {
compileInput('<input type="text" ng-model="value" ng-pattern="/^\\d\\d\\d-\\d\\d-\\d\\d\\d\\d$/" />');
changeInputValueTo('x000-00-0000x');
expect(inputElm).toBeInvalid();
changeInputValueTo('000-00-0000');
expect(inputElm).toBeValid();
changeInputValueTo('000-00-0000x');
expect(inputElm).toBeInvalid();
changeInputValueTo('123-45-6789');
expect(inputElm).toBeValid();
changeInputValueTo('x');
expect(inputElm).toBeInvalid();
});
it('should listen on ng-pattern when pattern is observed', function() {
var value, patternVal = /^\w+$/;
compileInput('<input type="text" ng-model="value" ng-pattern="pat" attr-capture />');
attrs.$observe('pattern', function(v) {
value = attrs.pattern;
});
scope.$apply(function() {
scope.pat = patternVal;
});
expect(value).toBe(patternVal);
});
it('should validate in-lined pattern with modifiers', function() {
compileInput('<input type="text" ng-model="value" ng-pattern="/^abc?$/i" />');
changeInputValueTo('aB');
expect(inputElm).toBeValid();
changeInputValueTo('xx');
expect(inputElm).toBeInvalid();
});
it('should validate pattern from scope', function() {
scope.regexp = /^\d\d\d-\d\d-\d\d\d\d$/;
compileInput('<input type="text" ng-model="value" ng-pattern="regexp" />');
changeInputValueTo('x000-00-0000x');
expect(inputElm).toBeInvalid();
changeInputValueTo('000-00-0000');
expect(inputElm).toBeValid();
changeInputValueTo('000-00-0000x');
expect(inputElm).toBeInvalid();
changeInputValueTo('123-45-6789');
expect(inputElm).toBeValid();
changeInputValueTo('x');
expect(inputElm).toBeInvalid();
scope.$apply(function() {
scope.regexp = /abc?/;
});
changeInputValueTo('ab');
expect(inputElm).toBeValid();
changeInputValueTo('xx');
expect(inputElm).toBeInvalid();
});
it('should perform validations when the ngPattern scope value changes', function() {
scope.regexp = /^[a-z]+$/;
compileInput('<input type="text" ng-model="value" ng-pattern="regexp" />');
changeInputValueTo('abcdef');
expect(inputElm).toBeValid();
changeInputValueTo('123');
expect(inputElm).toBeInvalid();
scope.$apply(function() {
scope.regexp = /^\d+$/;
});
expect(inputElm).toBeValid();
changeInputValueTo('abcdef');
expect(inputElm).toBeInvalid();
scope.$apply(function() {
scope.regexp = '';
});
expect(inputElm).toBeValid();
});
it('should register "pattern" with the model validations when the pattern attribute is used', function() {
compileInput('<input type="text" name="input" ng-model="value" pattern="^\\d+$" />');
changeInputValueTo('abcd');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.pattern).toBe(true);
changeInputValueTo('12345');
expect(inputElm).toBeValid();
expect(scope.form.input.$error.pattern).not.toBe(true);
});
it('should not throw an error when scope pattern can\'t be found', function() {
expect(function() {
compileInput('<input type="text" ng-model="foo" ng-pattern="fooRegexp" />');
scope.$apply("foo = 'bar'");
}).not.toThrowMatching(/^\[ngPattern:noregexp\] Expected fooRegexp to be a RegExp but was/);
});
it('should throw an error when the scope pattern is not a regular expression', function() {
expect(function() {
compileInput('<input type="text" ng-model="foo" ng-pattern="fooRegexp" />');
scope.$apply(function() {
scope.fooRegexp = {};
scope.foo = 'bar';
});
}).toThrowMatching(/^\[ngPattern:noregexp\] Expected fooRegexp to be a RegExp but was/);
});
it('should be invalid if entire string does not match pattern', function() {
compileInput('<input type="text" name="test" ng-model="value" pattern="\\d{4}">');
changeInputValueTo('1234');
expect(scope.form.test.$error.pattern).not.toBe(true);
expect(inputElm).toBeValid();
changeInputValueTo('123');
expect(scope.form.test.$error.pattern).toBe(true);
expect(inputElm).not.toBeValid();
changeInputValueTo('12345');
expect(scope.form.test.$error.pattern).toBe(true);
expect(inputElm).not.toBeValid();
});
it('should be cope with patterns that start with ^', function() {
compileInput('<input type="text" name="test" ng-model="value" pattern="^\\d{4}">');
changeInputValueTo('1234');
expect(scope.form.test.$error.pattern).not.toBe(true);
expect(inputElm).toBeValid();
changeInputValueTo('123');
expect(scope.form.test.$error.pattern).toBe(true);
expect(inputElm).not.toBeValid();
changeInputValueTo('12345');
expect(scope.form.test.$error.pattern).toBe(true);
expect(inputElm).not.toBeValid();
});
it('should be cope with patterns that end with $', function() {
compileInput('<input type="text" name="test" ng-model="value" pattern="\\d{4}$">');
changeInputValueTo('1234');
expect(scope.form.test.$error.pattern).not.toBe(true);
expect(inputElm).toBeValid();
changeInputValueTo('123');
expect(scope.form.test.$error.pattern).toBe(true);
expect(inputElm).not.toBeValid();
changeInputValueTo('12345');
expect(scope.form.test.$error.pattern).toBe(true);
expect(inputElm).not.toBeValid();
});
});
describe('minlength', function() {
it('should invalidate values that are shorter than the given minlength', function() {
compileInput('<input type="text" ng-model="value" ng-minlength="3" />');
changeInputValueTo('aa');
expect(inputElm).toBeInvalid();
changeInputValueTo('aaa');
expect(inputElm).toBeValid();
});
it('should listen on ng-minlength when minlength is observed', function() {
var value = 0;
compileInput('<input type="text" ng-model="value" ng-minlength="min" attr-capture />');
attrs.$observe('minlength', function(v) {
value = int(attrs.minlength);
});
scope.$apply('min = 5');
expect(value).toBe(5);
});
it('should observe the standard minlength attribute and register it as a validator on the model', function() {
compileInput('<input type="text" name="input" ng-model="value" minlength="{{ min }}" />');
scope.$apply('min = 10');
changeInputValueTo('12345');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.minlength).toBe(true);
scope.$apply('min = 5');
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('<input type="text" name="input" ng-model="value" minlength="3" />');
expect(scope.value).toBe(12345);
expect(scope.form.input.$error.minlength).toBeUndefined();
});
});
describe('maxlength', function() {
it('should invalidate values that are longer than the given maxlength', function() {
compileInput('<input type="text" ng-model="value" ng-maxlength="5" />');
changeInputValueTo('aaaaaaaa');
expect(inputElm).toBeInvalid();
changeInputValueTo('aaa');
expect(inputElm).toBeValid();
});
it('should only accept empty values when maxlength is 0', function() {
compileInput('<input type="text" ng-model="value" ng-maxlength="0" />');
changeInputValueTo('');
expect(inputElm).toBeValid();
changeInputValueTo('a');
expect(inputElm).toBeInvalid();
});
it('should accept values of any length when maxlength is negative', function() {
compileInput('<input type="text" ng-model="value" ng-maxlength="-1" />');
changeInputValueTo('');
expect(inputElm).toBeValid();
changeInputValueTo('aaaaaaaaaa');
expect(inputElm).toBeValid();
});
it('should accept values of any length when maxlength is non-numeric', function() {
compileInput('<input type="text" ng-model="value" ng-maxlength="{{maxlength}}" />');
changeInputValueTo('aaaaaaaaaa');
scope.$apply('maxlength = "5"');
expect(inputElm).toBeInvalid();
scope.$apply('maxlength = "abc"');
expect(inputElm).toBeValid();
scope.$apply('maxlength = ""');
expect(inputElm).toBeValid();
scope.$apply('maxlength = null');
expect(inputElm).toBeValid();
scope.someObj = {};
scope.$apply('maxlength = someObj');
expect(inputElm).toBeValid();
});
it('should listen on ng-maxlength when maxlength is observed', function() {
var value = 0;
compileInput('<input type="text" ng-model="value" ng-maxlength="max" attr-capture />');
attrs.$observe('maxlength', function(v) {
value = int(attrs.maxlength);
});
scope.$apply('max = 10');
expect(value).toBe(10);
});
it('should observe the standard maxlength attribute and register it as a validator on the model', function() {
compileInput('<input type="text" name="input" ng-model="value" maxlength="{{ max }}" />');
scope.$apply('max = 1');
changeInputValueTo('12345');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.maxlength).toBe(true);
scope.$apply('max = 6');
expect(inputElm).toBeValid();
expect(scope.form.input.$error.maxlength).not.toBe(true);
});
it('should assign the correct model after an observed validator became valid', function() {
compileInput('<input type="text" name="input" ng-model="value" maxlength="{{ max }}" />');
scope.$apply('max = 1');
changeInputValueTo('12345');
expect(scope.value).toBeUndefined();
scope.$apply('max = 6');
expect(scope.value).toBe('12345');
});
it('should assign the correct model after an observed validator became invalid', function() {
compileInput('<input type="text" name="input" ng-model="value" maxlength="{{ max }}" />');
scope.$apply('max = 6');
changeInputValueTo('12345');
expect(scope.value).toBe('12345');
scope.$apply('max = 1');
expect(scope.value).toBeUndefined();
});
it('should leave the value as invalid if observed maxlength changed, but is still invalid', function() {
compileInput('<input type="text" name="input" ng-model="value" maxlength="{{ max }}" />');
scope.$apply('max = 1');
changeInputValueTo('12345');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.maxlength).toBe(true);
expect(scope.value).toBeUndefined();
scope.$apply('max = 3');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.maxlength).toBe(true);
expect(scope.value).toBeUndefined();
});
it('should not notify if observed maxlength changed, but is still invalid', function() {
compileInput('<input type="text" name="input" ng-model="value" ng-change="ngChangeSpy()" ' +
'maxlength="{{ max }}" />');
scope.$apply('max = 1');
changeInputValueTo('12345');
scope.ngChangeSpy = jasmine.createSpy();
scope.$apply('max = 3');
expect(scope.ngChangeSpy).not.toHaveBeenCalled();
});
it('should leave the model untouched when validating before model initialization', function() {
scope.value = '12345';
compileInput('<input type="text" name="input" ng-model="value" minlength="3" />');
expect(scope.value).toBe('12345');
});
it('should validate when the model is initalized as a number', function() {
scope.value = 12345;
compileInput('<input type="text" name="input" ng-model="value" maxlength="10" />');
expect(scope.value).toBe(12345);
expect(scope.form.input.$error.maxlength).toBeUndefined();
});
});
// INPUT TYPES
describe('month', function() {
it('should throw if model is not a Date object', function() {
compileInput('<input type="month" ng-model="january"/>');
expect(function() {
scope.$apply(function() {
scope.january = '2013-01';
});
}).toThrowMinErr('ngModel', 'datefmt', 'Expected `2013-01` to be a date');
});
it('should set the view if the model is a valid Date object', function() {
compileInput('<input type="month" ng-model="march"/>');
scope.$apply(function() {
scope.march = new Date(2013, 2, 1);
});
expect(inputElm.val()).toBe('2013-03');
});
it('should set the model undefined if the input is an invalid month string', function() {
compileInput('<input type="month" ng-model="value"/>');
scope.$apply(function() {
scope.value = new Date(2013, 0, 1);
});
expect(inputElm.val()).toBe('2013-01');
try {
//set to text for browsers with datetime-local validation.
inputElm[0].setAttribute('type', 'text');
} catch (e) {
//for IE8
}
changeInputValueTo('stuff');
expect(inputElm.val()).toBe('stuff');
expect(scope.value).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should render as blank if null', function() {
compileInput('<input type="month" ng-model="test" />');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should come up blank when no value specified', function() {
compileInput('<input type="month" ng-model="test" />');
expect(inputElm.val()).toBe('');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toBe('');
});
it('should parse empty string to null', function() {
compileInput('<input type="month" ng-model="test" />');
scope.$apply(function() {
scope.test = new Date(2011, 0, 1);
});
changeInputValueTo('');
expect(scope.test).toBeNull();
expect(inputElm).toBeValid();
});
it('should use UTC if specified in the options', function() {
compileInput('<input type="month" ng-model="value" ng-model-options="{timezone: \'UTC\'}" />');
changeInputValueTo('2013-07');
expect(+scope.value).toBe(Date.UTC(2013, 6, 1));
scope.$apply(function() {
scope.value = new Date(Date.UTC(2014, 6, 1));
});
expect(inputElm.val()).toBe('2014-07');
});
it('should label parse errors as `month`', function() {
compileInput('<input type="month" ng-model="val" name="alias" />', {
valid: false,
badInput: true
});
changeInputValueTo('xxx');
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.month).toBeTruthy();
});
it('should only change the month of a bound date', function() {
compileInput('<input type="month" ng-model="value" ng-model-options="{timezone: \'UTC\'}" />');
scope.$apply(function() {
scope.value = new Date(Date.UTC(2013, 7, 1, 1, 0, 0, 0));
});
changeInputValueTo('2013-12');
expect(+scope.value).toBe(Date.UTC(2013, 11, 1, 1, 0, 0, 0));
expect(inputElm.val()).toBe('2013-12');
});
describe('min', function() {
var scope;
beforeEach(inject(function($rootScope) {
scope = $rootScope;
$rootScope.minVal = '2013-01';
compileInput('<input type="month" ng-model="value" name="alias" min="{{ minVal }}" />');
}));
it('should invalidate', function() {
changeInputValueTo('2012-12');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('2013-07');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2013, 6, 1));
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should revalidate when the min value changes', function() {
changeInputValueTo('2013-07');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.min).toBeFalsy();
scope.minVal = '2014-01';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.min).toBeTruthy();
});
});
describe('max', function() {
var scope;
beforeEach(inject(function($rootScope) {
scope = $rootScope;
$rootScope.maxVal = '2013-01';
compileInput('<input type="month" ng-model="value" name="alias" max="{{ maxVal }}" />');
}));
it('should validate', function() {
changeInputValueTo('2012-03');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2012, 2, 1));
expect(scope.form.alias.$error.max).toBeFalsy();
});
it('should invalidate', function() {
changeInputValueTo('2013-05');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeUndefined();
expect(scope.form.alias.$error.max).toBeTruthy();
});
it('should revalidate when the max value changes', function() {
changeInputValueTo('2012-07');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.max).toBeFalsy();
scope.maxVal = '2012-01';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.max).toBeTruthy();
});
});
});
describe('week', function() {
it('should throw if model is not a Date object', function() {
compileInput('<input type="week" ng-model="secondWeek"/>');
expect(function() {
scope.$apply(function() {
scope.secondWeek = '2013-W02';
});
}).toThrowMinErr('ngModel', 'datefmt', 'Expected `2013-W02` to be a date');
});
it('should set the view if the model is a valid Date object', function() {
compileInput('<input type="week" ng-model="secondWeek"/>');
scope.$apply(function() {
scope.secondWeek = new Date(2013, 0, 11);
});
expect(inputElm.val()).toBe('2013-W02');
});
it('should not affect the hours or minutes of a bound date', function() {
compileInput('<input type="week" ng-model="secondWeek"/>');
scope.$apply(function() {
scope.secondWeek = new Date(2013, 0, 11, 1, 0, 0, 0);
});
changeInputValueTo('2013-W03');
expect(+scope.secondWeek).toBe(+new Date(2013, 0, 17, 1, 0, 0, 0));
});
it('should set the model undefined if the input is an invalid week string', function() {
compileInput('<input type="week" ng-model="value"/>');
scope.$apply(function() {
scope.value = new Date(2013, 0, 11);
});
expect(inputElm.val()).toBe('2013-W02');
try {
//set to text for browsers with datetime-local validation.
inputElm[0].setAttribute('type', 'text');
} catch (e) {
//for IE8
}
changeInputValueTo('stuff');
expect(inputElm.val()).toBe('stuff');
expect(scope.value).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should render as blank if null', function() {
compileInput('<input type="week" ng-model="test" />');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should come up blank when no value specified', function() {
compileInput('<input type="week" ng-model="test" />');
expect(inputElm.val()).toBe('');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toBe('');
});
it('should parse empty string to null', function() {
compileInput('<input type="week" ng-model="test" />');
scope.$apply(function() {
scope.test = new Date(2011, 0, 1);
});
changeInputValueTo('');
expect(scope.test).toBeNull();
expect(inputElm).toBeValid();
});
it('should use UTC if specified in the options', function() {
compileInput('<input type="week" ng-model="value" ng-model-options="{timezone: \'UTC\'}" />');
changeInputValueTo('2013-W03');
expect(+scope.value).toBe(Date.UTC(2013, 0, 17));
scope.$apply(function() {
scope.value = new Date(Date.UTC(2014, 0, 17));
});
expect(inputElm.val()).toBe('2014-W03');
});
it('should label parse errors as `week`', function() {
compileInput('<input type="week" ng-model="val" name="alias" />', {
valid: false,
badInput: true
});
changeInputValueTo('yyy');
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.week).toBeTruthy();
});
describe('min', function() {
var scope;
beforeEach(inject(function($rootScope) {
scope = $rootScope;
$rootScope.minVal = '2013-W01';
compileInput('<input type="week" ng-model="value" name="alias" min="{{ minVal }}" />');
}));
it('should invalidate', function() {
changeInputValueTo('2012-W12');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('2013-W03');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2013, 0, 17));
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should revalidate when the min value changes', function() {
changeInputValueTo('2013-W03');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.min).toBeFalsy();
scope.minVal = '2014-W01';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.min).toBeTruthy();
});
});
describe('max', function() {
beforeEach(inject(function($rootScope) {
$rootScope.maxVal = '2013-W01';
scope = $rootScope;
compileInput('<input type="week" ng-model="value" name="alias" max="{{ maxVal }}" />');
}));
it('should validate', function() {
changeInputValueTo('2012-W01');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2012, 0, 5));
expect(scope.form.alias.$error.max).toBeFalsy();
});
it('should invalidate', function() {
changeInputValueTo('2013-W03');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeUndefined();
expect(scope.form.alias.$error.max).toBeTruthy();
});
it('should revalidate when the max value changes', function() {
changeInputValueTo('2012-W03');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.max).toBeFalsy();
scope.maxVal = '2012-W01';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.max).toBeTruthy();
});
});
});
describe('datetime-local', function() {
it('should throw if model is not a Date object', function() {
compileInput('<input type="datetime-local" ng-model="lunchtime"/>');
expect(function() {
scope.$apply(function() {
scope.lunchtime = '2013-12-16T11:30:00';
});
}).toThrowMinErr('ngModel', 'datefmt', 'Expected `2013-12-16T11:30:00` to be a date');
});
it('should set the view if the model if a valid Date object.', function() {
compileInput('<input type="datetime-local" ng-model="halfSecondToNextYear"/>');
scope.$apply(function() {
scope.halfSecondToNextYear = new Date(2013, 11, 31, 23, 59, 59, 500);
});
expect(inputElm.val()).toBe('2013-12-31T23:59:59.500');
});
it('should set the model undefined if the view is invalid', function() {
compileInput('<input type="datetime-local" ng-model="breakMe"/>');
scope.$apply(function() {
scope.breakMe = new Date(2009, 0, 6, 16, 25, 0);
});
expect(inputElm.val()).toBe('2009-01-06T16:25:00.000');
try {
//set to text for browsers with datetime-local validation.
inputElm[0].setAttribute('type', 'text');
} catch (e) {
//for IE8
}
changeInputValueTo('stuff');
expect(inputElm.val()).toBe('stuff');
expect(scope.breakMe).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should render as blank if null', function() {
compileInput('<input type="datetime-local" ng-model="test" />');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should come up blank when no value specified', function() {
compileInput('<input type="datetime-local" ng-model="test" />');
expect(inputElm.val()).toBe('');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toBe('');
});
it('should parse empty string to null', function() {
compileInput('<input type="datetime-local" ng-model="test" />');
scope.$apply(function() {
scope.test = new Date(2011, 0, 1);
});
changeInputValueTo('');
expect(scope.test).toBeNull();
expect(inputElm).toBeValid();
});
it('should use UTC if specified in the options', function() {
compileInput('<input type="datetime-local" ng-model="value" ng-model-options="{timezone: \'UTC\'}" />');
changeInputValueTo('2000-01-01T01:02');
expect(+scope.value).toBe(Date.UTC(2000, 0, 1, 1, 2, 0));
scope.$apply(function() {
scope.value = new Date(Date.UTC(2001, 0, 1, 1, 2, 0));
});
expect(inputElm.val()).toBe('2001-01-01T01:02:00.000');
});
it('should allow to specify the milliseconds', function() {
compileInput('<input type="datetime-local" ng-model="value"" />');
changeInputValueTo('2000-01-01T01:02:03.500');
expect(+scope.value).toBe(+new Date(2000, 0, 1, 1, 2, 3, 500));
});
it('should allow to specify single digit milliseconds', function() {
compileInput('<input type="datetime-local" ng-model="value"" />');
changeInputValueTo('2000-01-01T01:02:03.4');
expect(+scope.value).toBe(+new Date(2000, 0, 1, 1, 2, 3, 400));
});
it('should allow to specify the seconds', function() {
compileInput('<input type="datetime-local" ng-model="value"" />');
changeInputValueTo('2000-01-01T01:02:03');
expect(+scope.value).toBe(+new Date(2000, 0, 1, 1, 2, 3));
scope.$apply(function() {
scope.value = new Date(2001, 0, 1, 1, 2, 3);
});
expect(inputElm.val()).toBe('2001-01-01T01:02:03.000');
});
it('should allow to skip the seconds', function() {
compileInput('<input type="datetime-local" ng-model="value"" />');
changeInputValueTo('2000-01-01T01:02');
expect(+scope.value).toBe(+new Date(2000, 0, 1, 1, 2, 0));
});
it('should label parse errors as `datetimelocal`', function() {
compileInput('<input type="datetime-local" ng-model="val" name="alias" />', {
valid: false,
badInput: true
});
changeInputValueTo('zzz');
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.datetimelocal).toBeTruthy();
});
describe('min', function() {
var scope;
beforeEach(inject(function($rootScope) {
$rootScope.minVal = '2000-01-01T12:30:00';
scope = $rootScope;
compileInput('<input type="datetime-local" ng-model="value" name="alias" min="{{ minVal }}" />');
}));
it('should invalidate', function() {
changeInputValueTo('1999-12-31T01:02:00');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('2000-01-01T23:02:00');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2000, 0, 1, 23, 2, 0));
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should revalidate when the min value changes', function() {
changeInputValueTo('2000-02-01T01:02:00');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.min).toBeFalsy();
scope.minVal = '2010-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.min).toBeTruthy();
});
});
describe('max', function() {
var scope;
beforeEach(inject(function($rootScope) {
$rootScope.maxVal = '2019-01-01T01:02:00';
scope = $rootScope;
compileInput('<input type="datetime-local" ng-model="value" name="alias" max="{{ maxVal }}" />');
}));
it('should invalidate', function() {
changeInputValueTo('2019-12-31T01:02:00');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.max).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('2000-01-01T01:02:00');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2000, 0, 1, 1, 2, 0));
expect(scope.form.alias.$error.max).toBeFalsy();
});
it('should revalidate when the max value changes', function() {
changeInputValueTo('2000-02-01T01:02:00');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.max).toBeFalsy();
scope.maxVal = '2000-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.max).toBeTruthy();
});
});
it('should validate even if max value changes on-the-fly', function() {
scope.max = '2013-01-01T01:02:00';
compileInput('<input type="datetime-local" ng-model="value" name="alias" max="{{max}}" />');
changeInputValueTo('2014-01-01T12:34:00');
expect(inputElm).toBeInvalid();
scope.max = '2001-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = '2024-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if min value changes on-the-fly', function() {
scope.min = '2013-01-01T01:02:00';
compileInput('<input type="datetime-local" ng-model="value" name="alias" min="{{min}}" />');
changeInputValueTo('2010-01-01T12:34:00');
expect(inputElm).toBeInvalid();
scope.min = '2014-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = '2009-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if ng-max value changes on-the-fly', function() {
scope.max = '2013-01-01T01:02:00';
compileInput('<input type="datetime-local" ng-model="value" name="alias" ng-max="max" />');
changeInputValueTo('2014-01-01T12:34:00');
expect(inputElm).toBeInvalid();
scope.max = '2001-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = '2024-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if ng-min value changes on-the-fly', function() {
scope.min = '2013-01-01T01:02:00';
compileInput('<input type="datetime-local" ng-model="value" name="alias" ng-min="min" />');
changeInputValueTo('2010-01-01T12:34:00');
expect(inputElm).toBeInvalid();
scope.min = '2014-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = '2009-01-01T01:02:00';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('time', function() {
it('should throw if model is not a Date object', function() {
compileInput('<input type="time" ng-model="lunchtime"/>');
expect(function() {
scope.$apply(function() {
scope.lunchtime = '11:30:00';
});
}).toThrowMinErr('ngModel', 'datefmt', 'Expected `11:30:00` to be a date');
});
it('should set the view if the model if a valid Date object.', function() {
compileInput('<input type="time" ng-model="threeFortyOnePm"/>');
scope.$apply(function() {
scope.threeFortyOnePm = new Date(1970, 0, 1, 15, 41, 0, 500);
});
expect(inputElm.val()).toBe('15:41:00.500');
});
it('should set the model undefined if the view is invalid', function() {
compileInput('<input type="time" ng-model="breakMe"/>');
scope.$apply(function() {
scope.breakMe = new Date(1970, 0, 1, 16, 25, 0);
});
expect(inputElm.val()).toBe('16:25:00.000');
try {
//set to text for browsers with time validation.
inputElm[0].setAttribute('type', 'text');
} catch (e) {
//for IE8
}
changeInputValueTo('stuff');
expect(inputElm.val()).toBe('stuff');
expect(scope.breakMe).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should render as blank if null', function() {
compileInput('<input type="time" ng-model="test" />');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should come up blank when no value specified', function() {
compileInput('<input type="time" ng-model="test" />');
expect(inputElm.val()).toBe('');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toBe('');
});
it('should parse empty string to null', function() {
compileInput('<input type="time" ng-model="test" />');
scope.$apply(function() {
scope.test = new Date(2011, 0, 1);
});
changeInputValueTo('');
expect(scope.test).toBeNull();
expect(inputElm).toBeValid();
});
it('should use UTC if specified in the options', function() {
compileInput('<input type="time" ng-model="value" ng-model-options="{timezone: \'UTC\'}" />');
changeInputValueTo('23:02:00');
expect(+scope.value).toBe(Date.UTC(1970, 0, 1, 23, 2, 0));
scope.$apply(function() {
scope.value = new Date(Date.UTC(1971, 0, 1, 23, 2, 0));
});
expect(inputElm.val()).toBe('23:02:00.000');
});
it('should allow to specify the milliseconds', function() {
compileInput('<input type="time" ng-model="value"" />');
changeInputValueTo('01:02:03.500');
expect(+scope.value).toBe(+new Date(1970, 0, 1, 1, 2, 3, 500));
});
it('should allow to specify single digit milliseconds', function() {
compileInput('<input type="time" ng-model="value"" />');
changeInputValueTo('01:02:03.4');
expect(+scope.value).toBe(+new Date(1970, 0, 1, 1, 2, 3, 400));
});
it('should allow to specify the seconds', function() {
compileInput('<input type="time" ng-model="value"" />');
changeInputValueTo('01:02:03');
expect(+scope.value).toBe(+new Date(1970, 0, 1, 1, 2, 3));
scope.$apply(function() {
scope.value = new Date(1970, 0, 1, 1, 2, 3);
});
expect(inputElm.val()).toBe('01:02:03.000');
});
it('should allow to skip the seconds', function() {
compileInput('<input type="time" ng-model="value"" />');
changeInputValueTo('01:02');
expect(+scope.value).toBe(+new Date(1970, 0, 1, 1, 2, 0));
});
it('should label parse errors as `time`', function() {
compileInput('<input type="time" ng-model="val" name="alias" />', {
valid: false,
badInput: true
});
changeInputValueTo('mmm');
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.time).toBeTruthy();
});
it('should only change hours and minute of a bound date', function() {
compileInput('<input type="time" ng-model="value"" />');
scope.$apply(function() {
scope.value = new Date(2013, 2, 3, 1, 0, 0);
});
changeInputValueTo('01:02');
expect(+scope.value).toBe(+new Date(2013, 2, 3, 1, 2, 0));
});
describe('min', function() {
var scope;
beforeEach(inject(function($rootScope) {
$rootScope.minVal = '09:30:00';
scope = $rootScope;
compileInput('<input type="time" ng-model="value" name="alias" min="{{ minVal }}" />');
}));
it('should invalidate', function() {
changeInputValueTo('01:02:00');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('23:02:00');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(1970, 0, 1, 23, 2, 0));
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should revalidate when the min value changes', function() {
changeInputValueTo('23:02:00');
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.min).toBeFalsy();
scope.minVal = '23:55:00';
scope.$digest();
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.min).toBeTruthy();
});
});
describe('max', function() {
beforeEach(function() {
compileInput('<input type="time" ng-model="value" name="alias" max="22:30:00" />');
});
it('should invalidate', function() {
changeInputValueTo('23:00:00');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.max).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('05:30:00');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(1970, 0, 1, 5, 30, 0));
expect(scope.form.alias.$error.max).toBeFalsy();
});
});
it('should validate even if max value changes on-the-fly', function() {
scope.max = '4:02:00';
compileInput('<input type="time" ng-model="value" name="alias" max="{{max}}" />');
changeInputValueTo('05:34:00');
expect(inputElm).toBeInvalid();
scope.max = '06:34:00';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if min value changes on-the-fly', function() {
scope.min = '08:45:00';
compileInput('<input type="time" ng-model="value" name="alias" min="{{min}}" />');
changeInputValueTo('06:15:00');
expect(inputElm).toBeInvalid();
scope.min = '05:50:00';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if ng-max value changes on-the-fly', function() {
scope.max = '4:02:00';
compileInput('<input type="time" ng-model="value" name="alias" ng-max="max" />');
changeInputValueTo('05:34:00');
expect(inputElm).toBeInvalid();
scope.max = '06:34:00';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if ng-min value changes on-the-fly', function() {
scope.min = '08:45:00';
compileInput('<input type="time" ng-model="value" name="alias" ng-min="min" />');
changeInputValueTo('06:15:00');
expect(inputElm).toBeInvalid();
scope.min = '05:50:00';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('date', function() {
it('should throw if model is not a Date object.', function() {
compileInput('<input type="date" ng-model="birthday"/>');
expect(function() {
scope.$apply(function() {
scope.birthday = '1977-10-22';
});
}).toThrowMinErr('ngModel', 'datefmt', 'Expected `1977-10-22` to be a date');
});
it('should set the view to empty when the model is an InvalidDate', function() {
compileInput('<input type="date" ng-model="val"/>');
// reset the element type to text otherwise newer browsers
// would always set the input.value to empty for invalid dates...
inputElm.attr('type', 'text');
scope.$apply(function() {
scope.val = new Date('a');
});
expect(inputElm.val()).toBe('');
});
it('should set the view if the model if a valid Date object.', function() {
compileInput('<input type="date" ng-model="christmas"/>');
scope.$apply(function() {
scope.christmas = new Date(2013, 11, 25);
});
expect(inputElm.val()).toBe('2013-12-25');
});
it('should set the model undefined if the view is invalid', function() {
compileInput('<input type="date" ng-model="arrMatey"/>');
scope.$apply(function() {
scope.arrMatey = new Date(2014, 8, 14);
});
expect(inputElm.val()).toBe('2014-09-14');
try {
//set to text for browsers with date validation.
inputElm[0].setAttribute('type', 'text');
} catch (e) {
//for IE8
}
changeInputValueTo('1-2-3');
expect(inputElm.val()).toBe('1-2-3');
expect(scope.arrMatey).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should render as blank if null', function() {
compileInput('<input type="date" ng-model="test" />');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should come up blank when no value specified', function() {
compileInput('<input type="date" ng-model="test" />');
expect(inputElm.val()).toBe('');
scope.$apply('test = null');
expect(scope.test).toBeNull();
expect(inputElm.val()).toBe('');
});
it('should parse empty string to null', function() {
compileInput('<input type="date" ng-model="test" />');
scope.$apply(function() {
scope.test = new Date(2011, 0, 1);
});
changeInputValueTo('');
expect(scope.test).toBeNull();
expect(inputElm).toBeValid();
});
it('should use UTC if specified in the options', function() {
compileInput('<input type="date" ng-model="value" ng-model-options="{timezone: \'UTC\'}" />');
changeInputValueTo('2000-01-01');
expect(+scope.value).toBe(Date.UTC(2000, 0, 1));
scope.$apply(function() {
scope.value = new Date(Date.UTC(2001, 0, 1));
});
expect(inputElm.val()).toBe('2001-01-01');
});
it('should label parse errors as `date`', function() {
compileInput('<input type="date" ng-model="val" name="alias" />', {
valid: false,
badInput: true
});
changeInputValueTo('nnn');
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.date).toBeTruthy();
});
it('should work with multiple date types bound to the same model', function() {
formElm = jqLite('<form name="form"></form>');
var timeElm = jqLite('<input type="time" ng-model="val" />'),
monthElm = jqLite('<input type="month" ng-model="val" />'),
weekElm = jqLite('<input type="week" ng-model="val" />');
formElm.append(timeElm);
formElm.append(monthElm);
formElm.append(weekElm);
$compile(formElm)(scope);
scope.$apply(function() {
scope.val = new Date(2013, 1, 2, 3, 4, 5, 6);
});
expect(timeElm.val()).toBe('03:04:05.006');
expect(monthElm.val()).toBe('2013-02');
expect(weekElm.val()).toBe('2013-W05');
changeGivenInputTo(monthElm, '2012-02');
expect(monthElm.val()).toBe('2012-02');
expect(timeElm.val()).toBe('03:04:05.006');
expect(weekElm.val()).toBe('2012-W05');
changeGivenInputTo(timeElm, '04:05:06');
expect(monthElm.val()).toBe('2012-02');
expect(timeElm.val()).toBe('04:05:06');
expect(weekElm.val()).toBe('2012-W05');
changeGivenInputTo(weekElm, '2014-W01');
expect(monthElm.val()).toBe('2014-01');
expect(timeElm.val()).toBe('04:05:06.000');
expect(weekElm.val()).toBe('2014-W01');
expect(+scope.val).toBe(+new Date(2014, 0, 2, 4, 5, 6, 0));
function changeGivenInputTo(inputElm, value) {
inputElm.val(value);
browserTrigger(inputElm, $sniffer.hasEvent('input') ? 'input' : 'change');
}
});
describe('min', function() {
beforeEach(function() {
compileInput('<input type="date" ng-model="value" name="alias" min="2000-01-01" />');
});
it('should invalidate', function() {
changeInputValueTo('1999-12-31');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('2000-01-01');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2000, 0, 1));
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should parse ISO-based date strings as a valid min date value', inject(function($rootScope) {
var scope = $rootScope.$new();
var element = $compile('<form name="myForm">' +
'<input name="myControl" type="date" min="{{ min }}" ng-model="value">' +
'</form>')(scope);
var inputElm = element.find('input');
scope.value = new Date(2010, 1, 1, 0, 0, 0);
scope.min = new Date(2014, 10, 10, 0, 0, 0);
scope.$digest();
expect(scope.myForm.myControl.$error.min).toBeTruthy();
dealoc(element);
}));
});
describe('max', function() {
beforeEach(function() {
compileInput('<input type="date" ng-model="value" name="alias" max="2019-01-01" />');
});
it('should invalidate', function() {
changeInputValueTo('2019-12-31');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.max).toBeTruthy();
});
it('should validate', function() {
changeInputValueTo('2000-01-01');
expect(inputElm).toBeValid();
expect(+scope.value).toBe(+new Date(2000, 0, 1));
expect(scope.form.alias.$error.max).toBeFalsy();
});
it('should parse ISO-based date strings as a valid max date value', inject(function($rootScope) {
var scope = $rootScope.$new();
var element = $compile('<form name="myForm">' +
'<input name="myControl" type="date" max="{{ max }}" ng-model="value">' +
'</form>')(scope);
var inputElm = element.find('input');
scope.value = new Date(2020, 1, 1, 0, 0, 0);
scope.max = new Date(2014, 10, 10, 0, 0, 0);
scope.$digest();
expect(scope.myForm.myControl.$error.max).toBeTruthy();
dealoc(element);
}));
});
it('should validate even if max value changes on-the-fly', function() {
scope.max = '2013-01-01';
compileInput('<input type="date" ng-model="value" name="alias" max="{{max}}" />');
changeInputValueTo('2014-01-01');
expect(inputElm).toBeInvalid();
scope.max = '2001-01-01';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = '2021-01-01';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if min value changes on-the-fly', function() {
scope.min = '2013-01-01';
compileInput('<input type="date" ng-model="value" name="alias" min="{{min}}" />');
changeInputValueTo('2010-01-01');
expect(inputElm).toBeInvalid();
scope.min = '2014-01-01';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = '2009-01-01';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if ng-max value changes on-the-fly', function() {
scope.max = '2013-01-01';
compileInput('<input type="date" ng-model="value" name="alias" ng-max="max" />');
changeInputValueTo('2014-01-01');
expect(inputElm).toBeInvalid();
scope.max = '2001-01-01';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = '2021-01-01';
scope.$digest();
expect(inputElm).toBeValid();
});
it('should validate even if ng-min value changes on-the-fly', function() {
scope.min = '2013-01-01';
compileInput('<input type="date" ng-model="value" name="alias" ng-min="min" />');
changeInputValueTo('2010-01-01');
expect(inputElm).toBeInvalid();
scope.min = '2014-01-01';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = '2009-01-01';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('number', function() {
it('should reset the model if view is invalid', function() {
compileInput('<input type="number" ng-model="age"/>');
scope.$apply('age = 123');
expect(inputElm.val()).toBe('123');
try {
// to allow non-number values, we have to change type so that
// the browser which have number validation will not interfere with
// this test. IE8 won't allow it hence the catch.
inputElm[0].setAttribute('type', 'text');
} catch (e) {}
changeInputValueTo('123X');
expect(inputElm.val()).toBe('123X');
expect(scope.age).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should render as blank if null', function() {
compileInput('<input type="number" ng-model="age" />');
scope.$apply('age = null');
expect(scope.age).toBeNull();
expect(inputElm.val()).toEqual('');
});
it('should come up blank when no value specified', function() {
compileInput('<input type="number" ng-model="age" />');
expect(inputElm.val()).toBe('');
scope.$apply('age = null');
expect(scope.age).toBeNull();
expect(inputElm.val()).toBe('');
});
it('should parse empty string to null', function() {
compileInput('<input type="number" ng-model="age" />');
scope.$apply('age = 10');
changeInputValueTo('');
expect(scope.age).toBeNull();
expect(inputElm).toBeValid();
});
it('should only invalidate the model if suffering from bad input when the data is parsed', function() {
compileInput('<input type="number" ng-model="age" />', {
valid: false,
badInput: true
});
expect(scope.age).toBeUndefined();
expect(inputElm).toBeValid();
changeInputValueTo('this-will-fail-because-of-the-badInput-flag');
expect(scope.age).toBeUndefined();
expect(inputElm).toBeInvalid();
});
it('should validate number if transition from bad input to empty string', function() {
var validity = {
valid: false,
badInput: true
};
compileInput('<input type="number" ng-model="age" />', validity);
changeInputValueTo('10a');
validity.badInput = false;
validity.valid = true;
changeInputValueTo('');
expect(scope.age).toBeNull();
expect(inputElm).toBeValid();
});
it('should validate with undefined viewValue when $validate() called', function() {
compileInput('<input type="number" name="alias" ng-model="value" />');
scope.form.alias.$validate();
expect(inputElm).toBeValid();
expect(scope.form.alias.$error.number).toBeUndefined();
});
it('should throw if the model value is not a number', function() {
expect(function() {
scope.value = 'one';
compileInput('<input type="number" ng-model="value" />');
}).toThrowMinErr('ngModel', 'numfmt', "Expected `one` to be a number");
});
describe('min', function() {
it('should validate', function() {
compileInput('<input type="number" ng-model="value" name="alias" min="10" />');
changeInputValueTo('1');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
changeInputValueTo('100');
expect(inputElm).toBeValid();
expect(scope.value).toBe(100);
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should validate even if min value changes on-the-fly', function() {
scope.min = 10;
compileInput('<input type="number" ng-model="value" name="alias" min="{{min}}" />');
changeInputValueTo('15');
expect(inputElm).toBeValid();
scope.min = 20;
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = null;
scope.$digest();
expect(inputElm).toBeValid();
scope.min = '20';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = 'abc';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('ngMin', function() {
it('should validate', function() {
compileInput('<input type="number" ng-model="value" name="alias" ng-min="50" />');
changeInputValueTo('1');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeFalsy();
expect(scope.form.alias.$error.min).toBeTruthy();
changeInputValueTo('100');
expect(inputElm).toBeValid();
expect(scope.value).toBe(100);
expect(scope.form.alias.$error.min).toBeFalsy();
});
it('should validate even if the ngMin value changes on-the-fly', function() {
scope.min = 10;
compileInput('<input type="number" ng-model="value" name="alias" ng-min="min" />');
changeInputValueTo('15');
expect(inputElm).toBeValid();
scope.min = 20;
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = null;
scope.$digest();
expect(inputElm).toBeValid();
scope.min = '20';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.min = 'abc';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('max', function() {
it('should validate', function() {
compileInput('<input type="number" ng-model="value" name="alias" max="10" />');
changeInputValueTo('20');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeUndefined();
expect(scope.form.alias.$error.max).toBeTruthy();
changeInputValueTo('0');
expect(inputElm).toBeValid();
expect(scope.value).toBe(0);
expect(scope.form.alias.$error.max).toBeFalsy();
});
it('should validate even if max value changes on-the-fly', function() {
scope.max = 10;
compileInput('<input type="number" ng-model="value" name="alias" max="{{max}}" />');
changeInputValueTo('5');
expect(inputElm).toBeValid();
scope.max = 0;
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = null;
scope.$digest();
expect(inputElm).toBeValid();
scope.max = '4';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = 'abc';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('ngMax', function() {
it('should validate', function() {
compileInput('<input type="number" ng-model="value" name="alias" ng-max="5" />');
changeInputValueTo('20');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeUndefined();
expect(scope.form.alias.$error.max).toBeTruthy();
changeInputValueTo('0');
expect(inputElm).toBeValid();
expect(scope.value).toBe(0);
expect(scope.form.alias.$error.max).toBeFalsy();
});
it('should validate even if the ngMax value changes on-the-fly', function() {
scope.max = 10;
compileInput('<input type="number" ng-model="value" name="alias" ng-max="max" />');
changeInputValueTo('5');
expect(inputElm).toBeValid();
scope.max = 0;
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = null;
scope.$digest();
expect(inputElm).toBeValid();
scope.max = '4';
scope.$digest();
expect(inputElm).toBeInvalid();
scope.max = 'abc';
scope.$digest();
expect(inputElm).toBeValid();
});
});
describe('required', function() {
it('should be valid even if value is 0', function() {
compileInput('<input type="number" ng-model="value" name="alias" required />');
changeInputValueTo('0');
expect(inputElm).toBeValid();
expect(scope.value).toBe(0);
expect(scope.form.alias.$error.required).toBeFalsy();
});
it('should be valid even if value 0 is set from model', function() {
compileInput('<input type="number" ng-model="value" name="alias" required />');
scope.$apply('value = 0');
expect(inputElm).toBeValid();
expect(inputElm.val()).toBe('0');
expect(scope.form.alias.$error.required).toBeFalsy();
});
it('should register required on non boolean elements', function() {
compileInput('<div ng-model="value" name="alias" required>');
scope.$apply("value = ''");
expect(inputElm).toBeInvalid();
expect(scope.form.alias.$error.required).toBeTruthy();
});
it('should not invalidate number if ng-required=false and viewValue has not been committed', function() {
compileInput('<input type="number" ng-model="value" name="alias" ng-required="required">');
scope.$apply("required = false");
expect(inputElm).toBeValid();
});
});
describe('ngRequired', function() {
describe('when the ngRequired expression initially evaluates to true', function() {
it('should be valid even if value is 0', function() {
compileInput('<input type="number" ng-model="value" name="numberInput" ng-required="true" />');
changeInputValueTo('0');
expect(inputElm).toBeValid();
expect(scope.value).toBe(0);
expect(scope.form.numberInput.$error.required).toBeFalsy();
});
it('should be valid even if value 0 is set from model', function() {
compileInput('<input type="number" ng-model="value" name="numberInput" ng-required="true" />');
scope.$apply('value = 0');
expect(inputElm).toBeValid();
expect(inputElm.val()).toBe('0');
expect(scope.form.numberInput.$error.required).toBeFalsy();
});
it('should register required on non boolean elements', function() {
compileInput('<div ng-model="value" name="numberInput" ng-required="true">');
scope.$apply("value = ''");
expect(inputElm).toBeInvalid();
expect(scope.form.numberInput.$error.required).toBeTruthy();
});
it('should change from invalid to valid when the value is empty and the ngRequired expression changes to false', function() {
compileInput('<input type="number" ng-model="value" name="numberInput" ng-required="ngRequiredExpr" />');
scope.$apply('ngRequiredExpr = true');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeUndefined();
expect(scope.form.numberInput.$error.required).toBeTruthy();
scope.$apply('ngRequiredExpr = false');
expect(inputElm).toBeValid();
expect(scope.value).toBeUndefined();
expect(scope.form.numberInput.$error.required).toBeFalsy();
});
});
describe('when the ngRequired expression initially evaluates to false', function() {
it('should be valid even if value is empty', function() {
compileInput('<input type="number" ng-model="value" name="numberInput" ng-required="false" />');
expect(inputElm).toBeValid();
expect(scope.value).toBeUndefined();
expect(scope.form.numberInput.$error.required).toBeFalsy();
expect(scope.form.numberInput.$error.number).toBeFalsy();
});
it('should be valid if value is non-empty', function() {
compileInput('<input type="number" ng-model="value" name="numberInput" ng-required="false" />');
changeInputValueTo('42');
expect(inputElm).toBeValid();
expect(scope.value).toBe(42);
expect(scope.form.numberInput.$error.required).toBeFalsy();
});
it('should not register required on non boolean elements', function() {
compileInput('<div ng-model="value" name="numberInput" ng-required="false">');
scope.$apply("value = ''");
expect(inputElm).toBeValid();
expect(scope.form.numberInput.$error.required).toBeFalsy();
});
it('should change from valid to invalid when the value is empty and the ngRequired expression changes to true', function() {
compileInput('<input type="number" ng-model="value" name="numberInput" ng-required="ngRequiredExpr" />');
scope.$apply('ngRequiredExpr = false');
expect(inputElm).toBeValid();
expect(scope.value).toBeUndefined();
expect(scope.form.numberInput.$error.required).toBeFalsy();
scope.$apply('ngRequiredExpr = true');
expect(inputElm).toBeInvalid();
expect(scope.value).toBeUndefined();
expect(scope.form.numberInput.$error.required).toBeTruthy();
});
});
});
describe('minlength', function() {
it('should invalidate values that are shorter than the given minlength', function() {
compileInput('<input type="number" ng-model="value" ng-minlength="3" />');
changeInputValueTo('12');
expect(inputElm).toBeInvalid();
changeInputValueTo('123');
expect(inputElm).toBeValid();
});
it('should listen on ng-minlength when minlength is observed', function() {
var value = 0;
compileInput('<input type="number" ng-model="value" ng-minlength="min" attr-capture />');
attrs.$observe('minlength', function(v) {
value = int(attrs.minlength);
});
scope.$apply(function() {
scope.min = 5;
});
expect(value).toBe(5);
});
it('should observe the standard minlength attribute and register it as a validator on the model', function() {
compileInput('<input type="number" name="input" ng-model="value" minlength="{{ min }}" />');
scope.$apply(function() {
scope.min = 10;
});
changeInputValueTo('12345');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.minlength).toBe(true);
scope.$apply(function() {
scope.min = 5;
});
expect(inputElm).toBeValid();
expect(scope.form.input.$error.minlength).not.toBe(true);
});
});
describe('maxlength', function() {
it('should invalidate values that are longer than the given maxlength', function() {
compileInput('<input type="number" ng-model="value" ng-maxlength="5" />');
changeInputValueTo('12345678');
expect(inputElm).toBeInvalid();
changeInputValueTo('123');
expect(inputElm).toBeValid();
});
it('should listen on ng-maxlength when maxlength is observed', function() {
var value = 0;
compileInput('<input type="number" ng-model="value" ng-maxlength="max" attr-capture />');
attrs.$observe('maxlength', function(v) {
value = int(attrs.maxlength);
});
scope.$apply(function() {
scope.max = 10;
});
expect(value).toBe(10);
});
it('should observe the standard maxlength attribute and register it as a validator on the model', function() {
compileInput('<input type="number" name="input" ng-model="value" maxlength="{{ max }}" />');
scope.$apply(function() {
scope.max = 1;
});
changeInputValueTo('12345');
expect(inputElm).toBeInvalid();
expect(scope.form.input.$error.maxlength).toBe(true);
scope.$apply(function() {
scope.max = 6;
});
expect(inputElm).toBeValid();
expect(scope.form.input.$error.maxlength).not.toBe(true);
});
});
});
describe('email', function() {
it('should validate e-mail', function() {
compileInput('<input type="email" ng-model="email" name="alias" />');
var widget = scope.form.alias;
changeInputValueTo('vojta@google.com');
expect(scope.email).toBe('vojta@google.com');
expect(inputElm).toBeValid();
expect(widget.$error.email).toBeFalsy();
changeInputValueTo('invalid@');
expect(scope.email).toBeUndefined();
expect(inputElm).toBeInvalid();
expect(widget.$error.email).toBeTruthy();
});
describe('EMAIL_REGEXP', function() {
/* global EMAIL_REGEXP: false */
it('should validate email', function() {
expect(EMAIL_REGEXP.test('a@b.com')).toBe(true);
expect(EMAIL_REGEXP.test('a@b.museum')).toBe(true);
expect(EMAIL_REGEXP.test('a@B.c')).toBe(true);
expect(EMAIL_REGEXP.test('a@.b.c')).toBe(false);
expect(EMAIL_REGEXP.test('a@-b.c')).toBe(false);
expect(EMAIL_REGEXP.test('a@b-.c')).toBe(false);
expect(EMAIL_REGEXP.test('a@3b.c')).toBe(true);
expect(EMAIL_REGEXP.test('a@b')).toBe(true);
});
});
});
describe('url', function() {
it('should validate url', function() {
compileInput('<input type="url" ng-model="url" name="alias" />');
var widget = scope.form.alias;
changeInputValueTo('http://www.something.com');
expect(scope.url).toBe('http://www.something.com');
expect(inputElm).toBeValid();
expect(widget.$error.url).toBeFalsy();
changeInputValueTo('invalid.com');
expect(scope.url).toBeUndefined();
expect(inputElm).toBeInvalid();
expect(widget.$error.url).toBeTruthy();
});
describe('URL_REGEXP', function() {
/* global URL_REGEXP: false */
it('should validate url', function() {
expect(URL_REGEXP.test('http://server:123/path')).toBe(true);
expect(URL_REGEXP.test('a@B.c')).toBe(false);
});
});
});
describe('radio', function() {
it('should update the model', function() {
compileInput(
'<input type="radio" ng-model="color" value="white" />' +
'<input type="radio" ng-model="color" value="red" />' +
'<input type="radio" ng-model="color" value="blue" />');
scope.$apply("color = 'white'");
expect(inputElm[0].checked).toBe(true);
expect(inputElm[1].checked).toBe(false);
expect(inputElm[2].checked).toBe(false);
scope.$apply("color = 'red'");
expect(inputElm[0].checked).toBe(false);
expect(inputElm[1].checked).toBe(true);
expect(inputElm[2].checked).toBe(false);
browserTrigger(inputElm[2], 'click');
expect(scope.color).toBe('blue');
});
it('should allow {{expr}} as value', function() {
scope.some = 11;
compileInput(
'<input type="radio" ng-model="value" value="{{some}}" />' +
'<input type="radio" ng-model="value" value="{{other}}" />');
scope.$apply(function() {
scope.value = 'blue';
scope.some = 'blue';
scope.other = 'red';
});
expect(inputElm[0].checked).toBe(true);
expect(inputElm[1].checked).toBe(false);
browserTrigger(inputElm[1], 'click');
expect(scope.value).toBe('red');
scope.$apply("other = 'non-red'");
expect(inputElm[0].checked).toBe(false);
expect(inputElm[1].checked).toBe(false);
});
});
describe('checkbox', function() {
it('should ignore checkbox without ngModel directive', function() {
compileInput('<input type="checkbox" name="whatever" required />');
changeInputValueTo('');
expect(inputElm.hasClass('ng-valid')).toBe(false);
expect(inputElm.hasClass('ng-invalid')).toBe(false);
expect(inputElm.hasClass('ng-pristine')).toBe(false);
expect(inputElm.hasClass('ng-dirty')).toBe(false);
});
it('should format booleans', function() {
compileInput('<input type="checkbox" ng-model="name" />');
scope.$apply("name = false");
expect(inputElm[0].checked).toBe(false);
scope.$apply("name = true");
expect(inputElm[0].checked).toBe(true);
});
it('should support type="checkbox" with non-standard capitalization', function() {
compileInput('<input type="checkBox" ng-model="checkbox" />');
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(true);
browserTrigger(inputElm, 'click');
expect(scope.checkbox).toBe(false);
});
it('should allow custom enumeration', function() {
compileInput('<input type="checkbox" ng-model="name" ng-true-value="\'y\'" ' +
'ng-false-value="\'n\'">');
scope.$apply("name = 'y'");
expect(inputElm[0].checked).toBe(true);
scope.$apply("name = 'n'");
expect(inputElm[0].checked).toBe(false);
scope.$apply("name = 'something else'");
expect(inputElm[0].checked).toBe(false);
browserTrigger(inputElm, 'click');
expect(scope.name).toEqual('y');
browserTrigger(inputElm, 'click');
expect(scope.name).toEqual('n');
});
it('should throw if ngTrueValue is present and not a constant expression', function() {
expect(function() {
compileInput('<input type="checkbox" ng-model="value" ng-true-value="yes" />');
}).toThrowMinErr('ngModel', 'constexpr', "Expected constant expression for `ngTrueValue`, but saw `yes`.");
});
it('should throw if ngFalseValue is present and not a constant expression', function() {
expect(function() {
compileInput('<input type="checkbox" ng-model="value" ng-false-value="no" />');
}).toThrowMinErr('ngModel', 'constexpr', "Expected constant expression for `ngFalseValue`, but saw `no`.");
});
it('should not throw if ngTrueValue or ngFalseValue are not present', function() {
expect(function() {
compileInput('<input type="checkbox" ng-model="value" />');
}).not.toThrow();
});
it('should be required if false', function() {
compileInput('<input type="checkbox" ng-model="value" required />');
browserTrigger(inputElm, 'click');
expect(inputElm[0].checked).toBe(true);
expect(inputElm).toBeValid();
browserTrigger(inputElm, 'click');
expect(inputElm[0].checked).toBe(false);
expect(inputElm).toBeInvalid();
});
it('should set the ngTrueValue when required directive is present', function() {
compileInput('<input type="checkbox" ng-model="value" required ng-true-value="\'yes\'" />');
expect(inputElm).toBeInvalid();
browserTrigger(inputElm, 'click');
expect(inputElm[0].checked).toBe(true);
expect(inputElm).toBeValid();
});
});
describe('textarea', function() {
it("should process textarea", function() {
compileInput('<textarea ng-model="name"></textarea>');
inputElm = formElm.find('textarea');
scope.$apply("name = 'Adam'");
expect(inputElm.val()).toEqual('Adam');
changeInputValueTo('Shyam');
expect(scope.name).toEqual('Shyam');
changeInputValueTo('Kai');
expect(scope.name).toEqual('Kai');
});
it('should ignore textarea without ngModel directive', function() {
compileInput('<textarea name="whatever" required></textarea>');
inputElm = formElm.find('textarea');
changeInputValueTo('');
expect(inputElm.hasClass('ng-valid')).toBe(false);
expect(inputElm.hasClass('ng-invalid')).toBe(false);
expect(inputElm.hasClass('ng-pristine')).toBe(false);
expect(inputElm.hasClass('ng-dirty')).toBe(false);
});
});
describe('ngList', function() {
it('should parse text into an array', function() {
compileInput('<input type="text" ng-model="list" ng-list />');
// model -> view
scope.$apply("list = ['x', 'y', 'z']");
expect(inputElm.val()).toBe('x, y, z');
// view -> model
changeInputValueTo('1, 2, 3');
expect(scope.list).toEqual(['1', '2', '3']);
});
it("should not clobber text if model changes due to itself", function() {
// When the user types 'a,b' the 'a,' stage parses to ['a'] but if the
// $parseModel function runs it will change to 'a', in essence preventing
// the user from ever typing ','.
compileInput('<input type="text" ng-model="list" ng-list />');
changeInputValueTo('a ');
expect(inputElm.val()).toEqual('a ');
expect(scope.list).toEqual(['a']);
changeInputValueTo('a ,');
expect(inputElm.val()).toEqual('a ,');
expect(scope.list).toEqual(['a']);
changeInputValueTo('a , ');
expect(inputElm.val()).toEqual('a , ');
expect(scope.list).toEqual(['a']);
changeInputValueTo('a , b');
expect(inputElm.val()).toEqual('a , b');
expect(scope.list).toEqual(['a', 'b']);
});
it('should convert empty string to an empty array', function() {
compileInput('<input type="text" ng-model="list" ng-list />');
changeInputValueTo('');
expect(scope.list).toEqual([]);
});
it('should be invalid if required and empty', function() {
compileInput('<input type="text" ng-list ng-model="list" required>');
changeInputValueTo('');
expect(scope.list).toBeUndefined();
expect(inputElm).toBeInvalid();
changeInputValueTo('a,b');
expect(scope.list).toEqual(['a','b']);
expect(inputElm).toBeValid();
});
describe('with a custom separator', function() {
it('should split on the custom separator', function() {
compileInput('<input type="text" ng-model="list" ng-list=":" />');
changeInputValueTo('a,a');
expect(scope.list).toEqual(['a,a']);
changeInputValueTo('a:b');
expect(scope.list).toEqual(['a', 'b']);
});
it("should join the list back together with the custom separator", function() {
compileInput('<input type="text" ng-model="list" ng-list=" : " />');
scope.$apply(function() {
scope.list = ['x', 'y', 'z'];
});
expect(inputElm.val()).toBe('x : y : z');
});
});
describe('(with ngTrim undefined or true)', function() {
it('should ignore separator whitespace when splitting', function() {
compileInput('<input type="text" ng-model="list" ng-list=" | " />');
changeInputValueTo('a|b');
expect(scope.list).toEqual(['a', 'b']);
});
it('should trim whitespace from each list item', function() {
compileInput('<input type="text" ng-model="list" ng-list="|" />');
changeInputValueTo('a | b');
expect(scope.list).toEqual(['a', 'b']);
});
});
describe('(with ngTrim set to false)', function() {
it('should use separator whitespace when splitting', function() {
compileInput('<input type="text" ng-model="list" ng-trim="false" ng-list=" | " />');
changeInputValueTo('a|b');
expect(scope.list).toEqual(['a|b']);
changeInputValueTo('a | b');
expect(scope.list).toEqual(['a','b']);
});
it("should not trim whitespace from each list item", function() {
compileInput('<input type="text" ng-model="list" ng-trim="false" ng-list="|" />');
changeInputValueTo('a | b');
expect(scope.list).toEqual(['a ',' b']);
});
it("should support splitting on newlines", function() {
compileInput('<textarea type="text" ng-model="list" ng-trim="false" ng-list="&#10;"></textarea');
changeInputValueTo('a\nb');
expect(scope.list).toEqual(['a','b']);
});
});
});
describe('required', function() {
it('should allow bindings via ngRequired', function() {
compileInput('<input type="text" ng-model="value" ng-required="required" />');
scope.$apply("required = false");
changeInputValueTo('');
expect(inputElm).toBeValid();
scope.$apply("required = true");
expect(inputElm).toBeInvalid();
scope.$apply("value = 'some'");
expect(inputElm).toBeValid();
changeInputValueTo('');
expect(inputElm).toBeInvalid();
scope.$apply("required = false");
expect(inputElm).toBeValid();
});
it('should invalid initial value with bound required', function() {
compileInput('<input type="text" ng-model="value" required="{{required}}" />');
scope.$apply('required = true');
expect(inputElm).toBeInvalid();
});
it('should be $invalid but $pristine if not touched', function() {
compileInput('<input type="text" ng-model="name" name="alias" required />');
scope.$apply("name = null");
expect(inputElm).toBeInvalid();
expect(inputElm).toBePristine();
changeInputValueTo('');
expect(inputElm).toBeInvalid();
expect(inputElm).toBeDirty();
});
it('should allow empty string if not required', function() {
compileInput('<input type="text" ng-model="foo" />');
changeInputValueTo('a');
changeInputValueTo('');
expect(scope.foo).toBe('');
});
it('should set $invalid when model undefined', function() {
compileInput('<input type="text" ng-model="notDefined" required />');
expect(inputElm).toBeInvalid();
});
it('should consider bad input as an error before any other errors are considered', function() {
compileInput('<input type="text" ng-model="value" required />', { badInput: true });
var ctrl = inputElm.controller('ngModel');
ctrl.$parsers.push(function() {
return undefined;
});
changeInputValueTo('abc123');
expect(ctrl.$error.parse).toBe(true);
expect(inputElm).toHaveClass('ng-invalid-parse');
expect(inputElm).toBeInvalid(); // invalid because of the number validator
});
it('should allow `false` as a valid value when the input type is not "checkbox"', function() {
compileInput('<input type="radio" ng-value="true" ng-model="answer" required />' +
'<input type="radio" ng-value="false" ng-model="answer" required />');
scope.$apply();
expect(inputElm).toBeInvalid();
scope.$apply("answer = true");
expect(inputElm).toBeValid();
scope.$apply("answer = false");
expect(inputElm).toBeValid();
});
});
describe('ngChange', function() {
it('should $eval expression after new value is set in the model', function() {
compileInput('<input type="text" ng-model="value" ng-change="change()" />');
scope.change = jasmine.createSpy('change').andCallFake(function() {
expect(scope.value).toBe('new value');
});
changeInputValueTo('new value');
expect(scope.change).toHaveBeenCalledOnce();
});
it('should not $eval the expression if changed from model', function() {
compileInput('<input type="text" ng-model="value" ng-change="change()" />');
scope.change = jasmine.createSpy('change');
scope.$apply('value = true');
expect(scope.change).not.toHaveBeenCalled();
});
it('should $eval ngChange expression on checkbox', function() {
compileInput('<input type="checkbox" ng-model="foo" ng-change="changeFn()">');
scope.changeFn = jasmine.createSpy('changeFn');
expect(scope.changeFn).not.toHaveBeenCalled();
browserTrigger(inputElm, 'click');
expect(scope.changeFn).toHaveBeenCalledOnce();
});
it('should be able to change the model and via that also update the view', function() {
compileInput('<input type="text" ng-model="value" ng-change="value=\'b\'" />');
changeInputValueTo('a');
expect(inputElm.val()).toBe('b');
});
});
describe('ngValue', function() {
it('should update the dom "value" property and attribute', function() {
compileInput('<input type="submit" ng-value="value">');
scope.$apply("value = 'something'");
expect(inputElm[0].value).toBe('something');
expect(inputElm[0].getAttribute('value')).toBe('something');
});
it('should evaluate and set constant expressions', function() {
compileInput('<input type="radio" ng-model="selected" ng-value="true">' +
'<input type="radio" ng-model="selected" ng-value="false">' +
'<input type="radio" ng-model="selected" ng-value="1">');
browserTrigger(inputElm[0], 'click');
expect(scope.selected).toBe(true);
browserTrigger(inputElm[1], 'click');
expect(scope.selected).toBe(false);
browserTrigger(inputElm[2], 'click');
expect(scope.selected).toBe(1);
});
it('should watch the expression', function() {
compileInput('<input type="radio" ng-model="selected" ng-value="value">');
scope.$apply(function() {
scope.selected = scope.value = {some: 'object'};
});
expect(inputElm[0].checked).toBe(true);
scope.$apply(function() {
scope.value = {some: 'other'};
});
expect(inputElm[0].checked).toBe(false);
browserTrigger(inputElm, 'click');
expect(scope.selected).toBe(scope.value);
});
it('should work inside ngRepeat', function() {
compileInput(
'<input type="radio" ng-repeat="i in items" ng-model="$parent.selected" ng-value="i.id">');
scope.$apply(function() {
scope.items = [{id: 1}, {id: 2}];
scope.selected = 1;
});
inputElm = formElm.find('input');
expect(inputElm[0].checked).toBe(true);
expect(inputElm[1].checked).toBe(false);
browserTrigger(inputElm.eq(1), 'click');
expect(scope.selected).toBe(2);
});
it('should work inside ngRepeat with primitive values', function() {
compileInput(
'<div ng-repeat="i in items">' +
'<input type="radio" name="sel_{{i.id}}" ng-model="i.selected" ng-value="true">' +
'<input type="radio" name="sel_{{i.id}}" ng-model="i.selected" ng-value="false">' +
'</div>');
scope.$apply(function() {
scope.items = [{id: 1, selected: true}, {id: 2, selected: false}];
});
inputElm = formElm.find('input');
expect(inputElm[0].checked).toBe(true);
expect(inputElm[1].checked).toBe(false);
expect(inputElm[2].checked).toBe(false);
expect(inputElm[3].checked).toBe(true);
browserTrigger(inputElm.eq(1), 'click');
expect(scope.items[0].selected).toBe(false);
});
it('should work inside ngRepeat without name attribute', function() {
compileInput(
'<div ng-repeat="i in items">' +
'<input type="radio" ng-model="i.selected" ng-value="true">' +
'<input type="radio" ng-model="i.selected" ng-value="false">' +
'</div>');
scope.$apply(function() {
scope.items = [{id: 1, selected: true}, {id: 2, selected: false}];
});
inputElm = formElm.find('input');
expect(inputElm[0].checked).toBe(true);
expect(inputElm[1].checked).toBe(false);
expect(inputElm[2].checked).toBe(false);
expect(inputElm[3].checked).toBe(true);
browserTrigger(inputElm.eq(1), 'click');
expect(scope.items[0].selected).toBe(false);
});
});
describe('password', function() {
// Under no circumstances should input[type=password] trim inputs
it('should not trim if ngTrim is unspecified', function() {
compileInput('<input type="password" ng-model="password">');
changeInputValueTo(' - - untrimmed - - ');
expect(scope.password.length).toBe(' - - untrimmed - - '.length);
});
it('should not trim if ngTrim !== false', function() {
compileInput('<input type="password" ng-model="password" ng-trim="true">');
changeInputValueTo(' - - untrimmed - - ');
expect(scope.password.length).toBe(' - - untrimmed - - '.length);
});
it('should not trim if ngTrim === false', function() {
compileInput('<input type="password" ng-model="password" ng-trim="false">');
changeInputValueTo(' - - untrimmed - - ');
expect(scope.password.length).toBe(' - - untrimmed - - '.length);
});
});
});