diff --git a/src/ng/directive/input.js b/src/ng/directive/input.js
index 2d3210d52649..9726d8c7dc48 100644
--- a/src/ng/directive/input.js
+++ b/src/ng/directive/input.js
@@ -340,8 +340,8 @@ var inputType = {
*
* @param {string} ngModel Assignable angular expression to data-bind to.
* @param {string=} name Property name of the form under which the control is published.
- * @param {string=} ngTrueValue The value to which the expression should be set when selected.
- * @param {string=} ngFalseValue The value to which the expression should be set when not selected.
+ * @param {string=} ngTrueValue The string or number value to which the expression should be set when selected.
+ * @param {string=} ngFalseValue The string or number value to which the expression should be set when not selected.
* @param {string=} ngChange Angular expression to be executed when input changes due to user
* interaction with the input element.
*
@@ -641,11 +641,8 @@ function radioInputType(scope, element, attr, ctrl) {
}
function checkboxInputType(scope, element, attr, ctrl) {
- var trueValue = attr.ngTrueValue,
- falseValue = attr.ngFalseValue;
-
- if (!isString(trueValue)) trueValue = true;
- if (!isString(falseValue)) falseValue = false;
+ var trueValue = typedValue(attr.ngTrueValue, true),
+ falseValue = typedValue(attr.ngFalseValue, false);
element.on('click', function() {
scope.$apply(function() {
@@ -666,6 +663,14 @@ function checkboxInputType(scope, element, attr, ctrl) {
});
}
+function typedValue(value, defaultValue) {
+ if(isString(value)) {
+ var number = parseFloat(value);
+ return !isNaN(number) && isFinite(number) ? number : value;
+ } else {
+ return defaultValue;
+ }
+}
/**
* @ngdoc directive
diff --git a/test/ng/directive/inputSpec.js b/test/ng/directive/inputSpec.js
index 36fb754d580c..209ac454ba42 100644
--- a/test/ng/directive/inputSpec.js
+++ b/test/ng/directive/inputSpec.js
@@ -876,32 +876,165 @@ describe('input', function() {
});
- it('should allow custom enumeration', function() {
- compileInput('');
+ describe('should allow custom string enumeration', function() {
+ it('with non empty strings', function() {
+ compileInput('');
- scope.$apply(function() {
- scope.name = 'y';
+ scope.$apply(function() {
+ scope.name = 'y';
+ });
+ expect(inputElm[0].checked).toBe(true);
+
+ scope.$apply(function() {
+ scope.name = 'n';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ scope.$apply(function() {
+ scope.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');
});
- expect(inputElm[0].checked).toBe(true);
- scope.$apply(function() {
- scope.name = 'n';
+ it('with empty strings', function() {
+ compileInput('');
+
+ scope.$apply(function() {
+ scope.name = 'something';
+ });
+ expect(inputElm[0].checked).toBe(true);
+
+ scope.$apply(function() {
+ scope.name = '';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ scope.$apply(function() {
+ scope.name = 'something else';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual('something');
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual('');
+ });
+ });
+
+ describe('should allow custom numerical enumeration', function() {
+ it('with positive numbers', function() {
+ compileInput('');
+
+ scope.$apply(function() {
+ scope.name = 1;
+ });
+ expect(inputElm[0].checked).toBe(true);
+
+ scope.$apply(function() {
+ scope.name = 0;
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ scope.$apply(function() {
+ scope.name = 'something else';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(1);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(0);
});
- expect(inputElm[0].checked).toBe(false);
- scope.$apply(function() {
- scope.name = 'something else';
+ it('with negative numbers', function() {
+ compileInput('');
+
+ scope.$apply(function() {
+ scope.name = -1;
+ });
+ expect(inputElm[0].checked).toBe(true);
+
+ scope.$apply(function() {
+ scope.name = 0;
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ scope.$apply(function() {
+ scope.name = 'something else';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(-1);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(0);
});
- expect(inputElm[0].checked).toBe(false);
- browserTrigger(inputElm, 'click');
- expect(scope.name).toEqual('y');
+ it('with floating point numbers', function() {
+ compileInput('');
- browserTrigger(inputElm, 'click');
- expect(scope.name).toEqual('n');
- });
+ scope.$apply(function() {
+ scope.name = 1.3;
+ });
+ expect(inputElm[0].checked).toBe(true);
+
+ scope.$apply(function() {
+ scope.name = -1.3;
+ });
+ expect(inputElm[0].checked).toBe(false);
+ scope.$apply(function() {
+ scope.name = 'something else';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(1.3);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(-1.3);
+ });
+
+ it('with exponential numbers', function() {
+ compileInput('');
+
+ scope.$apply(function() {
+ scope.name = 1e3;
+ });
+ expect(inputElm[0].checked).toBe(true);
+
+ scope.$apply(function() {
+ scope.name = 1e-3;
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ scope.$apply(function() {
+ scope.name = 'something else';
+ });
+ expect(inputElm[0].checked).toBe(false);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(1e3);
+
+ browserTrigger(inputElm, 'click');
+ expect(scope.name).toEqual(1e-3);
+ });
+ });
it('should be required if false', function() {
compileInput('');