Merge pull request #241 from glyptodon/date-time-compat

GUAC-1213: Provide lenient date/time parsing
This commit is contained in:
James Muehlner
2015-08-11 13:47:59 -07:00
6 changed files with 218 additions and 2 deletions

View File

@@ -38,6 +38,13 @@ angular.module('form').controller('dateFieldController', ['$scope', '$injector',
*/
$scope.modelOptions = {
/**
* Space-delimited list of events on which the model will be updated.
*
* @type String
*/
updateOn : 'blur',
/**
* The time zone to use when reading/writing the Date object of the
* model.
@@ -62,7 +69,14 @@ angular.module('form').controller('dateFieldController', ['$scope', '$injector',
* set.
*/
var parseDate = function parseDate(str) {
return new Date(str + 'T00:00Z');
// Parse date, return blank if invalid
var parsedDate = new Date(str + 'T00:00Z');
if (isNaN(parsedDate.getTime()))
return null;
return parsedDate;
};
// Update typed value when model is changed

View File

@@ -38,6 +38,13 @@ angular.module('form').controller('timeFieldController', ['$scope', '$injector',
*/
$scope.modelOptions = {
/**
* Space-delimited list of events on which the model will be updated.
*
* @type String
*/
updateOn : 'blur',
/**
* The time zone to use when reading/writing the Date object of the
* model.
@@ -62,7 +69,14 @@ angular.module('form').controller('timeFieldController', ['$scope', '$injector',
* set.
*/
var parseTime = function parseTime(str) {
return new Date('1970-01-01T' + str + 'Z');
// Parse time, return blank if invalid
var parsedDate = new Date('1970-01-01T' + str + 'Z');
if (isNaN(parsedDate.getTime()))
return null;
return parsedDate;
};
// Update typed value when model is changed

View File

@@ -0,0 +1,83 @@
/*
* Copyright (C) 2015 Glyptodon LLC
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
/**
* A directive which modifies the parsing and formatting of ngModel when used
* on an HTML5 date input field, relaxing the otherwise strict parsing and
* validation behavior. The behavior of this directive for other input elements
* is undefined.
*/
angular.module('form').directive('guacLenientDate', ['$injector',
function guacLenientDate($injector) {
// Required services
var $filter = $injector.get('$filter');
/**
* Directive configuration object.
*
* @type Object.<String, Object>
*/
var config = {
restrict : 'A',
require : 'ngModel'
};
// Linking function
config.link = function linkGuacLenientDate($scope, $element, $attrs, ngModel) {
// Parse date strings leniently
ngModel.$parsers = [function parse(viewValue) {
// If blank, return null
if (!viewValue)
return null;
// Match basic date pattern
var match = /([0-9]*)(?:-([0-9]*)(?:-([0-9]*))?)?/.exec(viewValue);
if (!match)
return null;
// Determine year, month, and day based on pattern
var year = parseInt(match[1] || '0') || new Date().getFullYear();
var month = parseInt(match[2] || '0') || 1;
var day = parseInt(match[3] || '0') || 1;
// Convert to Date object
var parsedDate = new Date(Date.UTC(year, month - 1, day));
if (isNaN(parsedDate.getTime()))
return null;
return parsedDate;
}];
// Format date strings as "yyyy-MM-dd"
ngModel.$formatters = [function format(modelValue) {
return modelValue ? $filter('date')(modelValue, 'yyyy-MM-dd', 'UTC') : '';
}];
};
return config;
}]);

View File

@@ -0,0 +1,103 @@
/*
* Copyright (C) 2015 Glyptodon LLC
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
/**
* A directive which modifies the parsing and formatting of ngModel when used
* on an HTML5 time input field, relaxing the otherwise strict parsing and
* validation behavior. The behavior of this directive for other input elements
* is undefined.
*/
angular.module('form').directive('guacLenientTime', ['$injector',
function guacLenientTime($injector) {
// Required services
var $filter = $injector.get('$filter');
/**
* Directive configuration object.
*
* @type Object.<String, Object>
*/
var config = {
restrict : 'A',
require : 'ngModel'
};
// Linking function
config.link = function linkGuacLenientTIme($scope, $element, $attrs, ngModel) {
// Parse time strings leniently
ngModel.$parsers = [function parse(viewValue) {
// If blank, return null
if (!viewValue)
return null;
// Match basic time pattern
var match = /([0-9]*)(?::([0-9]*)(?::([0-9]*))?)?(?:\s*(a|p))?/.exec(viewValue.toLowerCase());
if (!match)
return null;
// Determine hour, minute, and second based on pattern
var hour = parseInt(match[1] || '0');
var minute = parseInt(match[2] || '0');
var second = parseInt(match[3] || '0');
// Handle AM/PM
if (match[4]) {
// Interpret 12 AM as 00:00 and 12 PM as 12:00
if (hour === 12)
hour = 0;
// Increment hour to evening if PM
if (match[4] === 'p')
hour += 12;
}
// Wrap seconds and minutes into minutes and hours
minute += second / 60; second %= 60;
hour += minute / 60; minute %= 60;
// Constrain hours to 0 - 23
hour %= 24;
// Convert to Date object
var parsedDate = new Date(Date.UTC(1970, 0, 1, hour, minute, second));
if (isNaN(parsedDate.getTime()))
return null;
return parsedDate;
}];
// Format time strings as "HH:mm:ss"
ngModel.$formatters = [function format(modelValue) {
return modelValue ? $filter('date')(modelValue, 'HH:mm:ss', 'UTC') : '';
}];
};
return config;
}]);

View File

@@ -2,6 +2,7 @@
<input type="date"
ng-model="typedValue"
ng-model-options="modelOptions"
guac-lenient-date
placeholder="{{'FORM.FIELD_PLACEHOLDER_DATE' | translate}}"
autocorrect="off"
autocapitalize="off"/>

View File

@@ -2,6 +2,7 @@
<input type="time"
ng-model="typedValue"
ng-model-options="modelOptions"
guac-lenient-time
placeholder="{{'FORM.FIELD_PLACEHOLDER_TIME' | translate}}"
autocorrect="off"
autocapitalize="off"/>