diff --git a/jquery-ui-timepicker-addon.js b/jquery-ui-timepicker-addon.js old mode 100755 new mode 100644 index 0a10de7..ffd0549 --- a/jquery-ui-timepicker-addon.js +++ b/jquery-ui-timepicker-addon.js @@ -1,8 +1,8 @@ /* * jQuery timepicker addon * By: Trent Richardson [http://trentrichardson.com] -* Version 0.8 -* Last Modified: 11/14/2010 +* Version 0.8-alt +* Last Modified: 11/15/2010 by Charles Phillips * * Copyright 2010 Trent Richardson * Dual licensed under the MIT and GPL licenses. @@ -18,864 +18,697 @@ */ (function($) { - function Timepicker(singleton) { - if(typeof(singleton) === 'boolean' && singleton === true) { - this.regional = []; // Available regional settings, indexed by language code - this.regional[''] = { // Default regional settings - currentText: 'Now', - ampm: false, - timeFormat: 'hh:mm tt', - timeOnlyTitle: 'Choose Time', - timeText: 'Time', - hourText: 'Hour', - minuteText: 'Minute', - secondText: 'Second' - }; - this.defaults = { // Global defaults for all the datetime picker instances - showButtonPanel: true, - timeOnly: false, - showHour: true, - showMinute: true, - showSecond: false, - showTime: true, - stepHour: 0.05, - stepMinute: 0.05, - stepSecond: 0.05, - hour: 0, - minute: 0, - second: 0, - hourMin: 0, - minuteMin: 0, - secondMin: 0, - hourMax: 23, - minuteMax: 59, - secondMax: 59, - hourGrid: 0, - minuteGrid: 0, - secondGrid: 0, - alwaysSetTime: true - }; - $.extend(this.defaults, this.regional['']); - } else { - this.defaults = $.extend({}, $.timepicker.defaults); - } + +$.extend($.ui, { timepicker: { version: "0.8-alt" } }); - } +/* Time picker manager. + Use the singleton instance of this class, $.timepicker, to interact with the time picker. + Settings for (groups of) time pickers are maintained in an instance object, + allowing multiple different settings on the same page. */ - Timepicker.prototype = { - $input: null, - $altInput: null, - $timeObj: null, - inst: null, - hour_slider: null, - minute_slider: null, - second_slider: null, +function Timepicker() { + this.regional = []; // Available regional settings, indexed by language code + this.regional[''] = { // Default regional settings + currentText: 'Now', + ampm: false, + timeFormat: 'hh:mm tt', + timeOnlyTitle: 'Choose Time', + timeText: 'Time', + hourText: 'Hour', + minuteText: 'Minute', + secondText: 'Second' + }; + this._defaults = { // Global defaults for all the datetime picker instances + showButtonPanel: true, + timeOnly: false, + showHour: true, + showMinute: true, + showSecond: false, + showTime: true, + stepHour: 0.05, + stepMinute: 0.05, + stepSecond: 0.05, hour: 0, minute: 0, second: 0, - ampm: '', - formattedDate: '', - formattedTime: '', - formattedDateTime: '', + hourMin: 0, + minuteMin: 0, + secondMin: 0, + hourMax: 23, + minuteMax: 59, + secondMax: 59, + hourGrid: 0, + minuteGrid: 0, + secondGrid: 0, + alwaysSetTime: true + }; + $.extend(this._defaults, this.regional['']); +} - //######################################################################## - // add our sliders to the calendar - //######################################################################## - addTimePicker: function(dp_inst) { - var tp_inst = this; - var currDT; - if ((this.$altInput) && this.$altInput !== null) - { - currDT = this.$input.val() + ' ' + this.$altInput.val(); - } - else - { - currDT = this.$input.val(); - } - var regstr = this.defaults.timeFormat.toString() +$.extend(Timepicker.prototype, { + $input: null, + $altInput: null, + $timeObj: null, + inst: null, + hour_slider: null, + minute_slider: null, + second_slider: null, + hour: 0, + minute: 0, + second: 0, + ampm: '', + formattedDate: '', + formattedTime: '', + formattedDateTime: '', + + /* Override the default settings for all instances of the time picker. + @param settings object - the new settings to use as defaults (anonymous object) + @return the manager object */ + setDefaults: function(settings) { + extendRemove(this._defaults, settings || {}); + return this; + }, + + //######################################################################## + // add our sliders to the calendar + //######################################################################## + _addTimePicker: function() { + var currDT = (this.$altInput) ? + this.$input.val() + ' ' + this.$altInput.val() : + this.$input.val(), + + regstr = this._defaults.timeFormat.toString() .replace(/h{1,2}/ig, '(\\d?\\d)') .replace(/m{1,2}/ig, '(\\d?\\d)') .replace(/s{1,2}/ig, '(\\d?\\d)') .replace(/t{1,2}/ig, '(am|pm|a|p)?') - .replace(/\s/g, '\\s?') + '$'; + .replace(/\s/g, '\\s?') + '$', - if (!this.defaults.timeOnly) { - //the time should come after x number of characters and a space. x = at least the length of text specified by the date format - var dp_dateFormat = $.datepicker._get(dp_inst, 'dateFormat'); - regstr = '.{' + dp_dateFormat.length + ',}\\s+' + regstr; + treg = currDT.match(new RegExp(regstr, 'i')), + order = this._getFormatPositions(); + + if (!this._defaults.timeOnly) { + //the time should come after x number of characters and a space. x = at least the length of text specified by the date format + var dp_dateFormat = $.datepicker._get(this.inst, 'dateFormat'); + regstr = '.{' + dp_dateFormat.length + ',}\\s+' + regstr; + } + + if (treg) { + if (order.t !== -1) + this.ampm = ((treg[order.t] === undefined || treg[order.t].length === 0) ? '' : (treg[order.t].charAt(0).toUpperCase() == 'A') ? 'AM' : 'PM').toUpperCase(); + + if (order.h !== -1) { + if (this.ampm == 'AM' && treg[order.h] == '12') + this.hour = 0; // 12am = 0 hour + else if (this.ampm == 'PM' && treg[order.h] != '12') + this.hour = (parseFloat(treg[order.h]) + 12).toFixed(0); // 12pm = 12 hour, any other pm = hour + 12 + else this.hour = treg[order.h]; } - var order = this.getFormatPositions(); - var treg = currDT.match(new RegExp(regstr, 'i')); + if (order.m !== -1) this.minute = treg[order.m]; + if (order.s !== -1) this.second = treg[order.s]; + } - if (treg) { - if (order.t !== -1) { - this.ampm = ((treg[order.t] === undefined || treg[order.t].length === 0) ? '' : (treg[order.t].charAt(0).toUpperCase() == 'A') ? 'AM' : 'PM').toUpperCase(); - } + this.timeDefined = (treg) ? true : false; + this._injectTimePicker(); + }, - if (order.h !== -1) { - if (this.ampm == 'AM' && treg[order.h] == '12') { - // 12am = 0 hour - this.hour = 0; - } else if (this.ampm == 'PM' && treg[order.h] != '12') { - // 12pm = 12 hour, any other pm = hour + 12 - this.hour = (parseFloat(treg[order.h]) + 12).toFixed(0); - } else { - this.hour = treg[order.h]; - } - } + //######################################################################## + // figure out position of time elements.. cause js cant do named captures + //######################################################################## + _getFormatPositions: function() { + var finds = this._defaults.timeFormat.toLowerCase().match(/(h{1,2}|m{1,2}|s{1,2}|t{1,2})/g), + orders = { h: -1, m: -1, s: -1, t: -1 }; - if (order.m !== -1) { - this.minute = treg[order.m]; - } + if (finds) + for (var i = 0; i < finds.length; i++) + if (orders[finds[i].toString().charAt(0)] == -1) + orders[finds[i].toString().charAt(0)] = i + 1; - if (order.s !== -1) { - this.second = treg[order.s]; - } - } + return orders; + }, - tp_inst.timeDefined = (treg) ? true : false; - - if (typeof(dp_inst.stay_open) !== 'boolean' || dp_inst.stay_open === false) { - // wait for datepicker to create itself.. 60% of the time it works every time.. - setTimeout(function() { - tp_inst.injectTimePicker(dp_inst, tp_inst); - }, 10); - } else { - tp_inst.injectTimePicker(dp_inst, tp_inst); - } - - }, - - //######################################################################## - // figure out position of time elements.. cause js cant do named captures - //######################################################################## - getFormatPositions: function() { - var finds = this.defaults.timeFormat.toLowerCase().match(/(h{1,2}|m{1,2}|s{1,2}|t{1,2})/g); - var orders = { h: -1, m: -1, s: -1, t: -1 }; - - if (finds) { - for (var i = 0; i < finds.length; i++) { - if (orders[finds[i].toString().charAt(0)] == -1) { - orders[finds[i].toString().charAt(0)] = i + 1; - } - } - } - - return orders; - }, - - //######################################################################## - // generate and inject html for timepicker into ui datepicker - //######################################################################## - injectTimePicker: function(dp_inst, tp_inst) - { - var $dp = dp_inst.dpDiv; - var opts = tp_inst.defaults; - var dp_id = dp_inst.id.toString().replace(/([^A-Za-z0-9_])/g, ''); - + //######################################################################## + // generate and inject html for timepicker into ui datepicker + //######################################################################## + _injectTimePicker: function() { + var $dp = this.inst.dpDiv, + o = this._defaults, + tp_inst = this, // Added by Peter Medeiros: // - Figure out what the hour/minute/second max should be based on the step values. // - Example: if stepMinute is 15, then minMax is 45. - var hourMax = opts.hourMax - (opts.hourMax % opts.stepHour); - var minMax = opts.minuteMax - (opts.minuteMax % opts.stepMinute); - var secMax = opts.secondMax - (opts.secondMax % opts.stepSecond); + hourMax = o.hourMax - (o.hourMax % o.stepHour), + minMax = o.minuteMax - (o.minuteMax % o.stepMinute), + secMax = o.secondMax - (o.secondMax % o.stepSecond), + dp_id = this.inst.id.toString().replace(/([^A-Za-z0-9_])/g, ''); - // Prevent displaying twice - if ($dp.find("div#ui-timepicker-div-" + dp_id).length === 0){ - var noDisplay = ' style="display:none;"'; - var html = - '
' + - '
' + opts.timeText + '
' + - '
' + - '
' + opts.hourText + '
'; - var hourGridSize = 0; - var minuteGridSize = 0; - var secondGridSize = 0; - var size = 0; - - if (opts.showHour && opts.hourGrid > 0) - { - html += '
' + - '
' + - '
'; - - for (var h = 0; h <= hourMax; h += opts.hourGrid) - { - hourGridSize++; - - var tmph = h; - if (opts.ampm && h > 12){ - tmph = h - 12; - } - else{ - tmph = h; - } - - if (tmph < 10){ - tmph = '0' + tmph; - } + // Prevent displaying twice + if ($dp.find("div#ui-timepicker-div-"+ dp_id).length === 0) { + var noDisplay = ' style="display:none;"', + html = + '
' + + '
' + o.timeText + '
' + + '
' + + '
' + o.hourText + '
', + hourGridSize = 0, + minuteGridSize = 0, + secondGridSize = 0, + size; + + if (o.showHour && o.hourGrid > 0) { + html += '
' + + '
' + + '
'; + + for (var h = 0; h < hourMax; h += o.hourGrid) { + hourGridSize++; + var tmph = (o.ampm && h > 12) ? h-12 : h; + if (tmph < 10) tmph = '0' + tmph; + if (o.ampm) { + if (h == 0) tmph = 12 +'a'; + else if (h < 12) tmph += 'a'; + else tmph += 'p'; + } + html += ''; + } + + html += '
' + tmph + '
' + + '
'; + } else html += '
'; - if (opts.ampm) - { - if (h === 0){ - tmph = 12 + 'a'; - } - else if (h < 12){ - tmph += 'a'; - } - else{ - tmph += 'p'; - } - } - html += '' + tmph + ''; - } - - html += '
' + - ''; + html += '
' + o.minuteText + '
'; + + if (o.showMinute && o.minuteGrid > 0) { + html += '
' + + '
' + + '
'; + + for (var m = 0; m < minMax; m += o.minuteGrid) { + minuteGridSize++; + html += ''; } - else - { - html += '
'; + + html += '
' + ((m < 10) ? '0' : '') + m + '
' + + '
'; + } else html += '
'; + + html += '
' + o.secondText + '
'; + + if (o.showSecond && o.secondGrid > 0) { + html += '
' + + '
' + + '
'; + + for (var s = 0; s < secMax; s += o.secondGrid) { + secondGridSize++; + html += ''; } + + html += '
' + ((s < 10) ? '0' : '') + s + '
' + + '
'; + } else html += '
'; - html += '
' + opts.minuteText + '
'; - - if (opts.showMinute && opts.minuteGrid > 0) - { - html += '
' + - '
' + - '
'; - - for (var m = 0; m <= minMax; m += opts.minuteGrid) - { - minuteGridSize++; - html += ''; - } - - html += '
' + ((m < 10) ? '0' : '') + m + '
' + - '
'; - } - else - { - html += '
'; - } - - html += '
' + opts.secondText + '
'; - - if (opts.showSecond && opts.secondGrid > 0) - { - html += '
' + - '
' + - '
'; - - for (var s = 0; s <= secMax; s += opts.secondGrid) - { - secondGridSize++; - html += ''; - } - - html += '
' + ((s < 10) ? '0' : '') + s + '
' + - '
'; - } - else - { - html += '
'; - } - - html += ''; - $tp = $(html); - + html += ''; + $tp = $(html); + // if we only want time picker... - if (opts.timeOnly === true) - { - $tp.prepend( - '
' + - '
' + opts.timeOnlyTitle + '
' + - '
'); - $dp.find('.ui-datepicker-header, .ui-datepicker-calendar').hide(); - } - - tp_inst.hour_slider = $tp.find('#ui_tpicker_hour_' + dp_id).slider({ - orientation: "horizontal", - value: tp_inst.hour, - min: opts.hourMin, - max: hourMax, - step: opts.stepHour, - slide: function(event, ui) - { - tp_inst.hour_slider.slider("option", "value", ui.value); - tp_inst.onTimeChange(dp_inst, tp_inst); - } - }); - - // Updated by Peter Medeiros: - // - Pass in Event and UI instance into slide function - tp_inst.minute_slider = $tp.find('#ui_tpicker_minute_' + dp_id).slider({ - orientation: "horizontal", - value: tp_inst.minute, - min: opts.minuteMin, - max: minMax, - step: opts.stepMinute, - slide: function(event, ui) - { - // update the global minute slider instance value with the current slider value - tp_inst.minute_slider.slider("option", "value", ui.value); - tp_inst.onTimeChange(dp_inst, tp_inst); - } - }); - - tp_inst.second_slider = $tp.find('#ui_tpicker_second_' + dp_id).slider({ - orientation: "horizontal", - value: tp_inst.second, - min: opts.secondMin, - max: secMax, - step: opts.stepSecond, - slide: function(event, ui) - { - tp_inst.second_slider.slider("option", "value", ui.value); - tp_inst.onTimeChange(dp_inst, tp_inst); - } - }); - - // Add grid functionality - if (opts.showHour && opts.hourGrid > 0) - { - size = 100 * hourGridSize * opts.hourGrid / (hourMax - opts.hourMin); - - $tp.find(".ui_tpicker_hour table").css({ - 'width': size + "%", - 'margin-left': (size / (-2 * hourGridSize)) + "%", - 'border-collapse': 'collapse' - }); - $tp.find(".ui_tpicker_hour td").each( - function(index) - { - $(this).click( - function() - { - var h = $(this).html(); - if (opts.ampm) - { - var ap = h.substring(2).toLowerCase(); - var aph = parseInt(h.substring(0, 2), 10); - - if (ap == 'a') - { - if (aph == 12){ - h = 0; - } - else{ - h = aph; - } - } else - { - if (aph == 12){ - h = 12; - } - else{ - h = aph + 12; - } - } - } - tp_inst.hour_slider.slider("option", "value", h); - tp_inst.onTimeChange(dp_inst, tp_inst); - } - ); - $(this).css({ - 'cursor': "pointer", - 'width': (100 / hourGridSize) + '%', - 'text-align': 'center', - 'overflow': 'hidden' - }); - } - ); - } - - if (opts.showMinute && opts.minuteGrid > 0) - { - size = 100 * minuteGridSize * opts.minuteGrid / (minMax - opts.minuteMin); - - $tp.find(".ui_tpicker_minute table").css({ - 'width': size + "%", - 'margin-left': (size / (-2 * minuteGridSize)) + "%", - 'border-collapse': 'collapse' - }); - $tp.find(".ui_tpicker_minute td").each( - function(index) - { - $(this).click( - function() - { - tp_inst.minute_slider.slider("option", "value", $(this).html()); - tp_inst.onTimeChange(dp_inst, tp_inst); - } - ); - $(this).css({ - 'cursor': "pointer", - 'width': (100 / minuteGridSize) + '%', - 'text-align': 'center', - 'overflow': 'hidden' - }); - } - ); - } - - if (opts.showSecond && opts.secondGrid > 0) - { - size = 100 * secondGridSize * opts.secondGrid / (secMax - opts.secondMin); - - $tp.find(".ui_tpicker_second table").css({ - 'width': size + "%", - 'margin-left': (size / (-2 * secondGridSize)) + "%", - 'border-collapse': 'collapse' - }); - $tp.find(".ui_tpicker_second td").each( - function(index) - { - $(this).click( - function() - { - tp_inst.second_slider.slider("option", "value", $(this).html()); - tp_inst.onTimeChange(dp_inst, tp_inst); - } - ); - $(this).css({ - 'cursor': "pointer", - 'width': (100 / secondGridSize) + '%', - 'text-align': 'center', - 'overflow': 'hidden' - }); - } - ); - } - - if (this.defaults.showButtonPanel) { - $dp.find('.ui-datepicker-buttonpane').before($tp); - } - else { - $dp.append($tp); - } - - tp_inst.$timeObj = $('#ui_tpicker_time_' + dp_id); - - if (dp_inst !== null) - { - var timeDefined = tp_inst.timeDefined; - tp_inst.onTimeChange(dp_inst, tp_inst, true); - tp_inst.timeDefined = timeDefined; - } + if (o.timeOnly === true) { + $tp.prepend( + '
' + + '
' + o.timeOnlyTitle + '
' + + '
'); + $dp.find('.ui-datepicker-header, .ui-datepicker-calendar').hide(); } - }, - - //######################################################################## - // when a slider moves.. - // on time change is also called when the time is updated in the text field - //######################################################################## - onTimeChange: function(dp_inst, tp_inst, force) { - var hour = (tp_inst.hour_slider)? tp_inst.hour_slider.slider('value') : tp_inst.hour; - var minute = (tp_inst.minute_slider)? tp_inst.minute_slider.slider('value') : tp_inst.minute; - var second = (tp_inst.second_slider)? tp_inst.second_slider.slider('value') : tp_inst.second; - var ampm = (hour < 11.5) ? 'AM' : 'PM'; - hour = (hour >= 11.5 && hour < 12) ? 12 : hour; - var hasChanged = false; - - // If the update was done in the input field, this field should not be updated. - // If the update was done using the sliders, update the input field. - if (tp_inst.hour != hour || tp_inst.minute != minute || tp_inst.second != second || (tp_inst.ampm.length > 0 && tp_inst.ampm != ampm) || (force !== undefined && force === true)) { - hasChanged = true; - } - - tp_inst.hour = parseFloat(hour).toFixed(0); - tp_inst.minute = parseFloat(minute).toFixed(0); - tp_inst.second = parseFloat(second).toFixed(0); - tp_inst.ampm = ampm; - - tp_inst.formatTime(tp_inst); - if(tp_inst.$timeObj){ - tp_inst.$timeObj.text(tp_inst.formattedTime); + + this.hour_slider = $tp.find('#ui_tpicker_hour_'+ dp_id).slider({ + orientation: "horizontal", + value: this.hour, + min: o.hourMin, + max: hourMax, + step: o.stepHour, + slide: function(event, ui) { + tp_inst.hour_slider.slider( "option", "value", ui.value); + tp_inst._onTimeChange(); + } + }); + + // Updated by Peter Medeiros: + // - Pass in Event and UI instance into slide function + this.minute_slider = $tp.find('#ui_tpicker_minute_'+ dp_id).slider({ + orientation: "horizontal", + value: this.minute, + min: o.minuteMin, + max: minMax, + step: o.stepMinute, + slide: function(event, ui) { + // update the global minute slider instance value with the current slider value + tp_inst.minute_slider.slider( "option", "value", ui.value); + tp_inst._onTimeChange(); + } + }); + + this.second_slider = $tp.find('#ui_tpicker_second_'+ dp_id).slider({ + orientation: "horizontal", + value: this.second, + min: o.secondMin, + max: secMax, + step: o.stepSecond, + slide: function(event, ui) { + tp_inst.second_slider.slider( "option", "value", ui.value); + tp_inst._onTimeChange(); + } + }); + + // Add grid functionality + if (o.showHour && o.hourGrid > 0) { + size = 100 * hourGridSize * o.hourGrid / (hourMax - o.hourMin); + + $tp.find(".ui_tpicker_hour table").css({ + width: size + "%", + marginLeft: (size / (-2 * hourGridSize)) + "%", + borderCollapse: 'collapse' + }).find("td").each( function(index) { + $(this).click(function() { + var h = $(this).html(); + if(o.ampm) { + var ap = h.substring(2).toLowerCase(), + aph = parseInt(h.substring(0,2)); + if (ap == 'a') { + if (aph == 12) h = 0; + else h = aph; + } else if (aph == 12) h = 12; + else h = aph + 12; + } + tp_inst.hour_slider.slider("option", "value", h); + tp_inst._onTimeChange(); + }).css({ + cursor: 'pointer', + width: (100 / hourGridSize) + '%', + textAlign: 'center', + overflow: 'hidden' + }); + }); } - if (hasChanged) { - tp_inst.updateDateTime(dp_inst, tp_inst); - tp_inst.timeDefined = true; - } - }, - - //######################################################################## - // format the time all pretty... - //######################################################################## - formatTime: function(tp_inst) { - var tmptime = tp_inst.defaults.timeFormat.toString(); - var hour12 = ((tp_inst.ampm == 'AM') ? (tp_inst.hour) : (tp_inst.hour % 12)); - hour12 = (Number(hour12) === 0) ? 12 : hour12; - - if (tp_inst.defaults.ampm === true) { - tmptime = tmptime.toString() - .replace(/hh/g, ((hour12 < 10) ? '0' : '') + hour12) - .replace(/h/g, hour12) - .replace(/mm/g, ((tp_inst.minute < 10) ? '0' : '') + tp_inst.minute) - .replace(/m/g, tp_inst.minute) - .replace(/ss/g, ((tp_inst.second < 10) ? '0' : '') + tp_inst.second) - .replace(/s/g, tp_inst.second) - .replace(/TT/g, tp_inst.ampm.toUpperCase()) - .replace(/tt/g, tp_inst.ampm.toLowerCase()) - .replace(/T/g, tp_inst.ampm.charAt(0).toUpperCase()) - .replace(/t/g, tp_inst.ampm.charAt(0).toLowerCase()); - - } else { - tmptime = tmptime.toString() - .replace(/hh/g, ((tp_inst.hour < 10) ? '0' : '') + tp_inst.hour) - .replace(/h/g, tp_inst.hour) - .replace(/mm/g, ((tp_inst.minute < 10) ? '0' : '') + tp_inst.minute) - .replace(/m/g, tp_inst.minute) - .replace(/ss/g, ((tp_inst.second < 10) ? '0' : '') + tp_inst.second) - .replace(/s/g, tp_inst.second); - tmptime = $.trim(tmptime.replace(/t/gi, '')); - } - - tp_inst.formattedTime = tmptime; - return tp_inst.formattedTime; - }, - - //######################################################################## - // update our input with the new date time.. - //######################################################################## - updateDateTime: function(dp_inst, tp_inst) { - var dt = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay); - var dateFmt = $.datepicker._get(dp_inst, 'dateFormat'); - var formatCfg = $.datepicker._getFormatConfig(dp_inst); - this.formattedDate = $.datepicker.formatDate(dateFmt, (dt === null ? new Date() : dt), formatCfg); - var formattedDateTime = this.formattedDate; - var timeAvailable = ((dt !== null && tp_inst.timeDefined) !== true)? false : true; - - if (dp_inst.lastVal !== undefined && (dp_inst.lastVal.length > 0 && this.$input.val().length === 0)) { - return; - } - - if(this.defaults.timeOnly === true){ - formattedDateTime = this.formattedTime; - } - else if (this.defaults.timeOnly !== true && (this.defaults.alwaysSetTime || timeAvailable)) { - if ((this.$altInput) && this.$altInput !== null) - { - this.$altInput.val(this.formattedTime); - } - else{ - formattedDateTime += ' ' + this.formattedTime; - } + if (o.showMinute && o.minuteGrid > 0) { + size = 100 * minuteGridSize * o.minuteGrid / (minMax - o.minuteMin); + $tp.find(".ui_tpicker_minute table").css({ + width: size + "%", + marginLeft: (size / (-2 * minuteGridSize)) + "%", + borderCollapse: 'collapse' + }).find("td").each(function(index) { + $(this).click(function() { + tp_inst.minute_slider.slider("option", "value", $(this).html()); + tp_inst._onTimeChange(); + }).css({ + cursor: 'pointer', + width: (100 / minuteGridSize) + '%', + textAlign: 'center', + overflow: 'hidden' + }); + }); } - this.formattedDateTime = formattedDateTime; - if(!dp_inst.inline && this.$input){ - this.$input.val(formattedDateTime); - this.$input.trigger("change"); + if (o.showSecond && o.secondGrid > 0) { + $tp.find(".ui_tpicker_second table").css({ + width: size + "%", + marginLeft: (size / (-2 * secondGridSize)) + "%", + borderCollapse: 'collapse' + }).find("td").each(function(index) { + $(this).click(function() { + tp_inst.second_slider.slider("option", "value", $(this).html()); + tp_inst._onTimeChange(); + }).css({ + cursor: 'pointer', + width: (100 / secondGridSize) + '%', + textAlign: 'center', + overflow: 'hidden' + }); + }); + } + + if (this._defaults.showButtonPanel) $dp.find('.ui-datepicker-buttonpane').before($tp); + else $dp.append($tp); + + this.$timeObj = $('#ui_tpicker_time_'+ dp_id); + + if (this.inst !== null) { + var timeDefined = this.timeDefined; + this._onTimeChange(); + this.timeDefined = timeDefined; } - }, - - setDefaults: function(settings) { - extendRemove(this.defaults, settings || {}); - return this; } - }; + }, + + //######################################################################## + // when a slider moves.. + // on time change is also called when the time is updated in the text field + //######################################################################## + _onTimeChange: function(force) { + var hour = (this.hour_slider) ? this.hour_slider.slider('value') : this.hour, + minute = (this.minute_slider) ? this.minute_slider.slider('value') : this.minute, + second = (this.second_slider) ? this.second_slider.slider('value') : this.second, + ampm = (hour < 11.5) ? 'AM' : 'PM', + hasChanged = false; + hour = (hour >= 11.5 && hour < 12) ? 12 : hour; + + // If the update was done in the input field, this field should not be updated. + // If the update was done using the sliders, update the input field. + if (force || this.hour != hour || this.minute != minute || this.second != second || (this.ampm.length > 0 && this.ampm != ampm)) + hasChanged = true; + + this.hour = parseFloat(hour).toFixed(0); + this.minute = parseFloat(minute).toFixed(0); + this.second = parseFloat(second).toFixed(0); + this.ampm = ampm; + + this._formatTime(); + if (this.$timeObj) this.$timeObj.text(this.formattedTime); + + if (hasChanged) { + this._updateDateTime(); + this.timeDefined = true; + } + }, + + //######################################################################## + // format the time all pretty... + //######################################################################## + _formatTime: function() { + var tmptime = this._defaults.timeFormat.toString(); + var hour12 = ((this.ampm == 'AM') ? (this.hour) : (this.hour % 12)); + hour12 = (Number(hour12) === 0) ? 12 : hour12; + + if (this._defaults.ampm === true) { + tmptime = tmptime.toString() + .replace(/hh/g, ((hour12 < 10) ? '0' : '') + hour12) + .replace(/h/g, hour12) + .replace(/mm/g, ((this.minute < 10) ? '0' : '') + this.minute) + .replace(/m/g, this.minute) + .replace(/ss/g, ((this.second < 10) ? '0' : '') + this.second) + .replace(/s/g, this.second) + .replace(/TT/g, this.ampm.toUpperCase()) + .replace(/tt/g, this.ampm.toLowerCase()) + .replace(/T/g, this.ampm.charAt(0).toUpperCase()) + .replace(/t/g, this.ampm.charAt(0).toLowerCase()); + } else { + tmptime = tmptime.toString() + .replace(/hh/g, ((this.hour < 10) ? '0' : '') + this.hour) + .replace(/h/g, this.hour) + .replace(/mm/g, ((this.minute < 10) ? '0' : '') + this.minute) + .replace(/m/g, this.minute) + .replace(/ss/g, ((this.second < 10) ? '0' : '') + this.second) + .replace(/s/g, this.second); + tmptime = $.trim(tmptime.replace(/t/gi, '')); + } + + this.formattedTime = tmptime; + return this.formattedTime; + }, + + //######################################################################## + // update our input with the new date time.. + //######################################################################## + _updateDateTime: function() { + var dp_inst = this.inst, + dt = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay), + dateFmt = $.datepicker._get(dp_inst, 'dateFormat'), + formatCfg = $.datepicker._getFormatConfig(dp_inst), + timeAvailable = dt !== null && this.timeDefined; + this.formattedDate = $.datepicker.formatDate(dateFmt, (dt === null ? new Date() : dt), formatCfg); + var formattedDateTime = this.formattedDate; + + if (dp_inst.lastVal !== undefined && (dp_inst.lastVal.length > 0 && this.$input.val().length === 0)) + return; + + if (this._defaults.timeOnly === true) formattedDateTime = this.formattedTime; + else if (this._defaults.timeOnly !== true && (this._defaults.alwaysSetTime || timeAvailable)) { + if (this.$altInput) this.$altInput.val(this.formattedTime); + else formattedDateTime += ' ' + this.formattedTime; + } + + this.formattedDateTime = formattedDateTime; + this.$input.val(formattedDateTime).trigger("change"); + } + +}); + +$.fn.extend({ + //######################################################################## + // shorthand just to use timepicker.. + //######################################################################## + timepicker: function(o) { + if (typeof o == 'object') o = $.extend(o, { timeOnly: true }); + return this.each(function() { + $(this).datetimepicker(o, arguments[1], arguments[2], arguments[3], arguments[4]); + }); + }, //######################################################################## // extend timepicker to datepicker //######################################################################## - jQuery.fn.datetimepicker = function(o) { - var opts = (o === undefined ? {} : o); - var input = $(this); - - if(typeof(o) == 'string') - { - if(o == 'setDate'){ - return input.datepicker(o, arguments[1]); - } - if(o == 'option' && typeof(arguments[1]) == 'string'){ - return input.datepicker(o, arguments[1], arguments[2]); - } - if(o == 'dialog'){ - return input.datepicker(o, arguments[1], arguments[2], arguments[3], arguments[4]); - } - return input.datepicker(o); - } + datetimepicker: function(o) { + o = o || {}; + var $input = this; - var tp = new Timepicker(); - var inlineSettings = {}; - - for (var attrName in tp.defaults) { - var attrValue = input.attr('time:' + attrName); - if (attrValue) { - try { - inlineSettings[attrName] = eval(attrValue); - } catch (err) { - inlineSettings[attrName] = attrValue; + if (typeof(o) == 'string') { + if (o == 'setDate') return this.each(function() { + $(this).datepicker(o, arguments[1]); + }); + else if(o == 'option' && typeof(arguments[1]) == 'string') return this.each(function() { + $(this).datepicker(o, arguments[1], arguments[2]); + }); + else if(o == 'dialog') return this.each(function() { + $(this).datepicker(o, arguments[1], arguments[2], arguments[3], arguments[4]); + }); + else return this.each(function() { + $(this).datepicker(o); + }); + } else { + var tp_inst = new Timepicker(), + inlineSettings = {}; + + for (var attrName in tp_inst._defaults) { + var attrValue = $input.attr('time:' + attrName); + if (attrValue) { + try { + inlineSettings[attrName] = eval(attrValue); + } catch (err) { + inlineSettings[attrName] = attrValue; + } } } - } - tp.defaults = $.extend(tp.defaults, inlineSettings); - - var beforeShowFunc = function(input, inst) { - tp.hour = tp.defaults.hour; - tp.minute = tp.defaults.minute; - tp.second = tp.defaults.second; - tp.ampm = ''; - tp.$input = $(input); - if(opts.altField !== undefined && opts.altField != ''){ - tp.$altInput = $($.datepicker._get(inst, 'altField')); - } - tp.inst = inst; - tp.addTimePicker(inst); - if ($.isFunction(opts.beforeShow)) { - opts.beforeShow(input, inst); - } - }; - - var onChangeMonthYearFunc = function(year, month, inst) { - // Update the time as well : this prevents the time from disappearing from the input field. - tp.updateDateTime(inst, tp); - if ($.isFunction(opts.onChangeMonthYear)) { - opts.onChangeMonthYear(year, month, inst); - } - }; - - var onCloseFunc = function(dateText, inst) { - if(tp.timeDefined === true && input.val() != '') { - tp.updateDateTime(inst, tp); - } - if ($.isFunction(opts.onClose)) { - opts.onClose(dateText, inst); - } - }; - - // make the alt field trigger the picker if its set - if ((opts.altField) && opts.altField !== null){ - var me = $(opts.altField); - - me.css({ 'cursor': 'pointer' }); - me.focus(function(){ - input.trigger("focus"); + tp_inst._defaults = $.extend({}, tp_inst._defaults, inlineSettings, o, { + beforeShow: function(input, dp_inst) { + tp_inst.hour = tp_inst._defaults.hour; + tp_inst.minute = tp_inst._defaults.minute; + tp_inst.second = tp_inst._defaults.second; + tp_inst.ampm = ''; + tp_inst.$input = $(input); + if (o.altField) { + tp_inst.$altinput = $($.datepicker._get(dp_inst, 'altField')) + .css({ cursor: 'pointer' }) + .focus(function(){ + $input.trigger("focus"); + }); + } + tp_inst.inst = dp_inst; + tp_inst._addTimePicker(); + if ($.isFunction(o.beforeShow)) { + o.beforeShow(input, dp_inst); + } + }, + onChangeMonthYear: function(year, month, dp_inst) { + // Update the time as well : this prevents the time from disappearing from the $input field. + tp_inst._updateDateTime(dp_inst); + if ($.isFunction(o.onChangeMonthYear)) { + o.onChangeMonthYear(year, month, dp_inst); + } + }, + onClose: function(dateText, dp_inst) { + if(tp_inst.timeDefined === true && $input.val() != '') { + tp_inst._updateDateTime(dp_inst); + } + if ($.isFunction(o.onClose)) { + o.onClose(dateText, dp_inst); + } + }, + timepicker: tp_inst // add timepicker as a property of datepicker: $.datepicker._get(dp_inst, 'timepicker'); + }); + + return this.each(function() { + $(this).datepicker(tp_inst._defaults); }); } - - tp.defaults = $.extend({}, tp.defaults, opts, { - beforeShow: beforeShowFunc, - onChangeMonthYear: onChangeMonthYearFunc, - onClose: onCloseFunc, - timepicker: tp // add timepicker as a property of datepicker: $.datepicker._get(dp_inst, 'timepicker'); - }); - - return input.datepicker(tp.defaults); - - }; - - //######################################################################## - // shorthand just to use timepicker.. - //######################################################################## - jQuery.fn.timepicker = function(opts) { - if(typeof opts == 'object'){ - opts = $.extend(opts, { timeOnly: true }); - } - - return $(this).datetimepicker(opts, arguments[1], arguments[2], arguments[3], arguments[4]); - }; - - //######################################################################## - // the bad hack :/ override datepicker so it doesnt close on select - // inspired: http://stackoverflow.com/questions/1252512/jquery-datepicker-prevent-closing-picker-when-clicking-a-date/1762378#1762378 - //######################################################################## - $.datepicker._base_selectDate = $.datepicker._selectDate; - $.datepicker._selectDate = function (id, dateStr) { - var target = $(id); - var inst = this._getInst(target[0]); - var tp_inst = $.datepicker._get(inst, 'timepicker'); - - if(tp_inst){ - inst.inline = true; - inst.stay_open = true; - $.datepicker._base_selectDate(id, dateStr); - inst.stay_open = false; - inst.inline = false; - this._notifyChange(inst); - this._updateDatepicker(inst); - } - else{ - $.datepicker._base_selectDate(id, dateStr); - } - }; - - //############################################################################################# - // second bad hack :/ override datepicker so it triggers an event when changing the input field - // and does not redraw the datepicker on every selectDate event - //############################################################################################# - $.datepicker._base_updateDatepicker = $.datepicker._updateDatepicker; - $.datepicker._updateDatepicker = function(inst) { - if (typeof(inst.stay_open) !== 'boolean' || inst.stay_open === false) { - this._base_updateDatepicker(inst); - // Reload the time control when changing something in the input text field. - this._beforeShow(inst.input, inst); - } - }; - - $.datepicker._beforeShow = function(input, inst) { - var beforeShow = this._get(inst, 'beforeShow'); - if (beforeShow) { - inst.stay_open = true; - beforeShow.apply((inst.input ? inst.input[0] : null), [inst.input, inst]); - inst.stay_open = false; - } - }; - - //####################################################################################### - // third bad hack :/ override datepicker so it allows spaces and colan in the input field - //####################################################################################### - $.datepicker._base_doKeyPress = $.datepicker._doKeyPress; - $.datepicker._doKeyPress = function(event) { - var inst = $.datepicker._getInst(event.target); - var tp_inst = $.datepicker._get(inst, 'timepicker'); - - if(tp_inst){ - if ($.datepicker._get(inst, 'constrainInput')) { - var dateChars = $.datepicker._possibleChars($.datepicker._get(inst, 'dateFormat')); - var chr = String.fromCharCode(event.charCode === undefined ? event.keyCode : event.charCode); - var chrl = chr.toLowerCase(); - // keyCode == 58 => ":" - // keyCode == 32 => " " - return event.ctrlKey || (chr < ' ' || !dateChars || dateChars.indexOf(chr) > -1 || event.keyCode == 58 || event.keyCode == 32 || chr == ':' || chr == ' ' || chrl == 'a' || chrl == 'p' || chrl == 'm'); - } - } - else{ - return $.datepicker._base_doKeyPress(event); - } - - }; - - //####################################################################################### - // Override key up event to sync manual input changes. - //####################################################################################### - $.datepicker._base_doKeyUp = $.datepicker._doKeyUp; - $.datepicker._doKeyUp = function (event) { - - var inst = $.datepicker._getInst(event.target); - var tp_inst = $.datepicker._get(inst, 'timepicker'); - - if (tp_inst !== null) { - if (tp_inst.defaults.timeOnly && (inst.input.val() != inst.lastVal)) { - try { - $.datepicker._updateDatepicker(inst); - } - catch (err) { - $.datepicker.log(err); - } - } - } - - return $.datepicker._base_doKeyUp(event); - }; - - //####################################################################################### - // override "Today" button to also grab the time. - //####################################################################################### - $.datepicker._base_gotoToday = $.datepicker._gotoToday; - $.datepicker._gotoToday = function(id) { - $.datepicker._base_gotoToday(id); - - var target = $(id); - var dp_inst = this._getInst(target[0]); - - this._setTime(dp_inst, new Date()); - - }; - - //####################################################################################### - // Create our on set time function - //####################################################################################### - $.datepicker._setTime = function(inst, date) { - - var tp_inst = $.datepicker._get(inst, 'timepicker'); - - if(tp_inst){ - var hour = date.getHours(); - var minute = date.getMinutes(); - var second = date.getSeconds(); - - //check if within min/max times.. - if( (hour < tp_inst.defaults.hourMin || hour > tp_inst.defaults.hourMax) || (minute < tp_inst.defaults.minuteMin || minute > tp_inst.defaults.minuteMax) || (second < tp_inst.defaults.secondMin || second > tp_inst.defaults.secondMax) ){ - hour = tp_inst.defaults.hourMin; - minute = tp_inst.defaults.minuteMin; - second = tp_inst.defaults.secondMin; - } - - if(tp_inst.hour_slider && tp_inst.minute_slider && tp_inst.second_slider){ - tp_inst.hour_slider.slider('value', hour ); - tp_inst.minute_slider.slider('value', minute ); - tp_inst.second_slider.slider('value', second ); - } - else{ - tp_inst.hour = hour; - tp_inst.minute = minute; - tp_inst.second = second; - } - - tp_inst.onTimeChange(inst, tp_inst, true); - } - }; - - //####################################################################################### - // override getDate() to allow getting time too within date object - //####################################################################################### - $.datepicker._base_setDate = $.datepicker._setDate; - $.datepicker._setDate = function(inst, date, noChange) { - var tp_inst = $.datepicker._get(inst, 'timepicker'); - - if(!date){ - date = new Date(); - } - - var tp_date = new Date(date.getYear(), date.getMonth(), date.getDay(), date.getHours(), date.getMinutes(), date.getSeconds()); - - $.datepicker._updateDatepicker(inst); - - $.datepicker._base_setDate(inst, date, noChange); - - if(tp_inst){ - this._setTime(inst, tp_date); - } - - }; - - //####################################################################################### - // override getDate() to allow getting time too within date object - //####################################################################################### - $.datepicker._base_getDate = $.datepicker._getDate; - $.datepicker._getDate = function(inst) { - - var tp_inst = $.datepicker._get(inst, 'timepicker'); - - if(tp_inst){ - var startDate = (!inst.currentYear || (inst.input && inst.input.val() == '') ? null : - this._daylightSavingAdjust(new Date( - inst.currentYear, inst.currentMonth, inst.currentDay, tp_inst.hour, tp_inst.minute, tp_inst.second))); - return startDate; - } - - return $.datepicker._base_getDate(inst); - }; - - //####################################################################################### - // jQuery extend now ignores nulls! - //####################################################################################### - function extendRemove(target, props) { - $.extend(target, props); - for (var name in props){ - if (props[name] === null || props[name] === undefined){ - target[name] = props[name]; - } - } - return target; } +}); + +//######################################################################## +// the bad hack :/ override datepicker so it doesnt close on select +// inspired: http://stackoverflow.com/questions/1252512/jquery-datepicker-prevent-closing-picker-when-clicking-a-date/1762378#1762378 +//######################################################################## +$.datepicker._base_selectDate = $.datepicker._selectDate; +$.datepicker._selectDate = function (id, dateStr) { + var inst = this._getInst($(id)[0]), + tp_inst = $.datepicker._get(inst, 'timepicker'); + + if (tp_inst) { + inst.inline = inst.stay_open = true; + inst.stay_open = inst.inline = false; + $.datepicker._base_selectDate(id, dateStr); + this._notifyChange(inst); + this._updateDatepicker(inst); + } + else $.datepicker._base_selectDate(id, dateStr); +}; + +//############################################################################################# +// second bad hack :/ override datepicker so it triggers an event when changing the input field +// and does not redraw the datepicker on every selectDate event +//############################################################################################# +$.datepicker._base_updateDatepicker = $.datepicker._updateDatepicker; +$.datepicker._updateDatepicker = function(inst) { + if (typeof(inst.stay_open) !== 'boolean' || inst.stay_open === false) { + this._base_updateDatepicker(inst); + // Reload the time control when changing something in the input text field. + this._beforeShow(inst.input, inst); + } +}; + +$.datepicker._beforeShow = function(input, inst) { + var beforeShow = this._get(inst, 'beforeShow'); + if (beforeShow) { + inst.stay_open = true; + beforeShow.apply((inst.input ? inst.input[0] : null), [inst.input, inst]); + inst.stay_open = false; + } +}; + +//####################################################################################### +// third bad hack :/ override datepicker so it allows spaces and colan in the input field +//####################################################################################### +$.datepicker._base_doKeyPress = $.datepicker._doKeyPress; +$.datepicker._doKeyPress = function(event) { + var inst = $.datepicker._getInst(event.target), + tp_inst = $.datepicker._get(inst, 'timepicker'); + + if (tp_inst) { + if ($.datepicker._get(inst, 'constrainInput')) { + var dateChars = $.datepicker._possibleChars($.datepicker._get(inst, 'dateFormat')); + var chr = String.fromCharCode(event.charCode === undefined ? event.keyCode : event.charCode); + var chrl = chr.toLowerCase(); + // keyCode == 58 => ":" + // keyCode == 32 => " " + return event.ctrlKey || (chr < ' ' || !dateChars || dateChars.indexOf(chr) > -1 || event.keyCode == 58 || event.keyCode == 32 || chr == ':' || chr == ' ' || chrl == 'a' || chrl == 'p' || chrl == 'm'); + } + } else return $.datepicker._base_doKeyPress(event); + +}; + +//####################################################################################### +// Override key up event to sync manual input changes. +//####################################################################################### +$.datepicker._base_doKeyUp = $.datepicker._doKeyUp; +$.datepicker._doKeyUp = function (event) { + var inst = $.datepicker._getInst(event.target), + tp_inst = $.datepicker._get(inst, 'timepicker'); + + if (tp_inst) { + if (tp_inst._defaults.timeOnly && (inst.input.val() != inst.lastVal)) { + try { + $.datepicker._updateDatepicker(inst); + } + catch (err) { + $.datepicker.log(err); + } + } + } + + return $.datepicker._base_doKeyUp(event); +}; + +//####################################################################################### +// override "Today" button to also grab the time. +//####################################################################################### +$.datepicker._base_gotoToday = $.datepicker._gotoToday; +$.datepicker._gotoToday = function(id) { + $.datepicker._base_gotoToday(id); + this._setTime(this._getInst($(id)[0]), new Date()); +}; + +//####################################################################################### +// Create our own set time function +//####################################################################################### +$.datepicker._setTime = function(inst, date) { + var tp_inst = $.datepicker._get(inst, 'timepicker'); + + if(tp_inst){ + var hour = date.getHours(), + minute = date.getMinutes(), + second = date.getSeconds(); + + //check if within min/max times.. + if ((hour < tp_inst._defaults.hourMin || hour > tp_inst._defaults.hourMax) || (minute < tp_inst._defaults.minuteMin || minute > tp_inst._defaults.minuteMax) || (second < tp_inst._defaults.secondMin || second > tp_inst._defaults.secondMax)) { + hour = tp_inst._defaults.hourMin; + minute = tp_inst._defaults.minuteMin; + second = tp_inst._defaults.secondMin; + } + + if (tp_inst.hour_slider && tp_inst.minute_slider && tp_inst.second_slider) { + tp_inst.hour_slider.slider('value', hour); + tp_inst.minute_slider.slider('value', minute); + tp_inst.second_slider.slider('value', second); + } else { + tp_inst.hour = hour; + tp_inst.minute = minute; + tp_inst.second = second; + } + + tp_inst._onTimeChange(true); + } +}; + +//####################################################################################### +// override setDate() to allow getting time too within Date object +//####################################################################################### +$.datepicker._base_setDate = $.datepicker._setDate; +$.datepicker._setDate = function(inst, date, noChange) { + date = date || new Date(); + var tp_inst = $.datepicker._get(inst, 'timepicker'), + tp_date = new Date(date.getTime()); + + $.datepicker._updateDatepicker(inst); + $.datepicker._base_setDate(inst, date, noChange); + if (tp_inst) this._setTime(inst, tp_date); +}; + +//####################################################################################### +// override getDate() to allow getting time too within Date object +//####################################################################################### +$.datepicker._base_getDate = $.datepicker._getDate; +$.datepicker._getDate = function(inst) { + var tp_inst = $.datepicker._get(inst, 'timepicker'); + if (tp_inst) + return startDate = (!inst.currentYear || (inst.input && inst.input.val() == '')) ? + null : + this._daylightSavingAdjust(new Date(inst.currentYear, inst.currentMonth, inst.currentDay, tp_inst.hour, tp_inst.minute, tp_inst.second)); + else return $.datepicker._base_getDate(inst); +}; + + +//####################################################################################### +// jQuery extend now ignores nulls! +//####################################################################################### +function extendRemove(target, props) { + $.extend(target, props); + for (var name in props) + if (props[name] == null || props[name] == undefined) + target[name] = props[name]; + return target; +} + +$.timepicker = new Timepicker(); // singleton instance +$.timepicker.version = "0.8-alt"; - $.timepicker = new Timepicker(true); // singleton instance })(jQuery); - -