From 81245b01f9837d2d24336be5620b2b50857995e3 Mon Sep 17 00:00:00 2001 From: Philipp Denzler Date: Tue, 2 Dec 2014 13:02:48 +0100 Subject: [PATCH] fix(digitsValidator): Use combination of Number constructor and toString() to split number to integer and fraction parts - fixes a bug when the current locale uses ',' as decimal separator --- src/core/validators/digitsValidator.js | 20 ++++++++++++-------- src/core/validators/digitsValidator.spec.js | 6 ++++++ 2 files changed, 18 insertions(+), 8 deletions(-) diff --git a/src/core/validators/digitsValidator.js b/src/core/validators/digitsValidator.js index f22c58d..c38b03c 100644 --- a/src/core/validators/digitsValidator.js +++ b/src/core/validators/digitsValidator.js @@ -2,12 +2,16 @@ angular.module('valdr') .factory('valdrDigitsValidator', ['valdrUtil', function (valdrUtil) { - var decimalSeparator = 1.1.toLocaleString().substring(1, 2); // jshint ignore:line - - var removeAnythingButDigitsAndDecimalSeparator = function (value) { - var regex = new RegExp('[^' + decimalSeparator + '\\d]', 'g'); - // at this point 'value' can still be a number or a string or... - return value.toLocaleString().replace(regex, ''); + // matches everything except digits and '.' as decimal separator + var regexp = new RegExp('[^.\\d]', 'g'); + + /** + * By converting to number and back to string using toString(), we make sure that '.' is used as decimal separator + * and not the locale specific decimal separator. + * As we already checked for NaN at this point, we can do this safely. + */ + var toStringWithoutThousandSeparators = function (value) { + return Number(value).toString().replace(regexp, ''); }; var isNotLongerThan = function (valueAsString, maxLengthConstraint) { @@ -19,8 +23,8 @@ angular.module('valdr') fractionConstraint = constraint.fraction, cleanValueAsString, integerAndFraction; - cleanValueAsString = removeAnythingButDigitsAndDecimalSeparator(value); - integerAndFraction = cleanValueAsString.split(decimalSeparator); + cleanValueAsString = toStringWithoutThousandSeparators(value); + integerAndFraction = cleanValueAsString.split('.'); return isNotLongerThan(integerAndFraction[0], integerConstraint) && isNotLongerThan(integerAndFraction[1], fractionConstraint); diff --git a/src/core/validators/digitsValidator.spec.js b/src/core/validators/digitsValidator.spec.js index e752188..96a2528 100644 --- a/src/core/validators/digitsValidator.spec.js +++ b/src/core/validators/digitsValidator.spec.js @@ -47,6 +47,12 @@ describe('valdrDigitsValidator', function () { expect(digitsValidator.validate('47\'11', constraint)).toBe(false); }); + it('should validate correctly with fraction > 4', function() { + constraint.fraction = 4; + expect(digitsValidator.validate(1000.00001, constraint)).toBe(false); + expect(digitsValidator.validate(1000.0001, constraint)).toBe(true) + }); + it('should not choke on integer:1 conditions', function () { constraint.integer = 1; expect(digitsValidator.validate('10', constraint)).toBe(false);