it-logica-application-backbone 1.4.12 → 1.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
data/Gemfile CHANGED
@@ -1,4 +1,4 @@
1
- source "http://rubygems.org"
1
+ source "https://rubygems.org"
2
2
  # Add dependencies required to use your gem here.
3
3
  # Example:
4
4
  # gem "activesupport", ">= 2.3.5"
@@ -8,11 +8,11 @@ source "http://rubygems.org"
8
8
  group :development do
9
9
  gem "shoulda", ">= 0"
10
10
  gem "rdoc", "~> 3.12"
11
- gem "bundler", "~> 1.1.0"
11
+ # gem "bundler", "~> 1.1.0"
12
12
  gem "jeweler", "~> 1.8.4"
13
13
  gem "simplecov", ">= 0"
14
14
  end
15
15
 
16
16
  gem 'htmlentities'
17
17
 
18
- gem 'jquery-rails', '2.1.3'
18
+ gem 'jquery-rails', '2.1.4'
@@ -1,5 +1,5 @@
1
1
  GEM
2
- remote: http://rubygems.org/
2
+ remote: https://rubygems.org/
3
3
  specs:
4
4
  actionpack (3.2.8)
5
5
  activemodel (= 3.2.8)
@@ -20,7 +20,7 @@ GEM
20
20
  builder (3.0.4)
21
21
  erubis (2.7.0)
22
22
  git (1.2.5)
23
- hike (1.2.1)
23
+ hike (1.2.3)
24
24
  htmlentities (4.3.1)
25
25
  i18n (0.6.0)
26
26
  jeweler (1.8.4)
@@ -29,17 +29,17 @@ GEM
29
29
  rake
30
30
  rdoc
31
31
  journey (1.0.4)
32
- jquery-rails (2.1.3)
33
- railties (>= 3.1.0, < 5.0)
34
- thor (~> 0.14)
32
+ jquery-rails (2.1.4)
33
+ railties (>= 3.0, < 5.0)
34
+ thor (>= 0.14, < 2.0)
35
35
  json (1.7.5)
36
36
  multi_json (1.3.6)
37
- rack (1.4.1)
38
- rack-cache (1.2)
37
+ rack (1.4.7)
38
+ rack-cache (1.6.1)
39
39
  rack (>= 0.4)
40
- rack-ssl (1.3.2)
40
+ rack-ssl (1.3.4)
41
41
  rack
42
- rack-test (0.6.2)
42
+ rack-test (0.6.3)
43
43
  rack (>= 1.0)
44
44
  railties (3.2.8)
45
45
  actionpack (= 3.2.8)
@@ -61,21 +61,20 @@ GEM
61
61
  multi_json (~> 1.0)
62
62
  simplecov-html (~> 0.5.3)
63
63
  simplecov-html (0.5.3)
64
- sprockets (2.1.3)
64
+ sprockets (2.1.4)
65
65
  hike (~> 1.2)
66
66
  rack (~> 1.0)
67
67
  tilt (~> 1.1, != 1.3.0)
68
- thor (0.16.0)
69
- tilt (1.3.3)
68
+ thor (0.19.1)
69
+ tilt (1.4.1)
70
70
 
71
71
  PLATFORMS
72
72
  ruby
73
73
 
74
74
  DEPENDENCIES
75
- bundler (~> 1.1.0)
76
75
  htmlentities
77
76
  jeweler (~> 1.8.4)
78
- jquery-rails (= 2.1.3)
77
+ jquery-rails (= 2.1.4)
79
78
  rdoc (~> 3.12)
80
79
  shoulda
81
80
  simplecov
data/VERSION CHANGED
@@ -1 +1 @@
1
- 1.4.12
1
+ 1.5.0
@@ -7,10 +7,6 @@
7
7
  //= require ./backbone_js/jquery-ui-timepicker-addon.js
8
8
 
9
9
 
10
- //= require ./backbone_js/jquery.iframe-transport.js
11
- //= require ./backbone_js/jquery.fileupload.js
12
- //= require ./backbone_js/jquery.fileupload-ui.js
13
-
14
10
  //= require ./backbone_js/tinymce/jscripts/tiny_mce/jquery.tinymce.min.js
15
11
 
16
12
  //= require ./backbone_js/ladas_table_checkbox_pool.js.coffee
@@ -1,1326 +1,2291 @@
1
- /*
2
- * jQuery timepicker addon
3
- * By: Trent Richardson [http://trentrichardson.com]
4
- * Version 1.0.0
5
- * Last Modified: 02/05/2012
6
- *
7
- * Copyright 2012 Trent Richardson
8
- * Dual licensed under the MIT and GPL licenses.
9
- * http://trentrichardson.com/Impromptu/GPL-LICENSE.txt
10
- * http://trentrichardson.com/Impromptu/MIT-LICENSE.txt
11
- *
12
- * HERES THE CSS:
13
- * .ui-timepicker-div .ui-widget-header { margin-bottom: 8px; }
14
- * .ui-timepicker-div dl { text-align: left; }
15
- * .ui-timepicker-div dl dt { height: 25px; margin-bottom: -25px; }
16
- * .ui-timepicker-div dl dd { margin: 0 10px 10px 65px; }
17
- * .ui-timepicker-div td { font-size: 90%; }
18
- * .ui-tpicker-grid-label { background: none; border: none; margin: 0; padding: 0; }
19
- */
20
-
21
- (function($) {
22
-
23
- // Prevent "Uncaught RangeError: Maximum call stack size exceeded"
24
- $.ui.timepicker = $.ui.timepicker || {};
25
- if ($.ui.timepicker.version) {
26
- return;
27
- }
28
-
29
- $.extend($.ui, { timepicker: { version: "1.0.0" } });
30
-
31
- /* Time picker manager.
32
- Use the singleton instance of this class, $.timepicker, to interact with the time picker.
33
- Settings for (groups of) time pickers are maintained in an instance object,
34
- allowing multiple different settings on the same page. */
35
-
36
- function Timepicker() {
37
- this.regional = []; // Available regional settings, indexed by language code
38
- this.regional[''] = { // Default regional settings
39
- currentText: 'Now',
40
- closeText: 'Done',
41
- ampm: false,
42
- amNames: ['AM', 'A'],
43
- pmNames: ['PM', 'P'],
44
- timeFormat: 'hh:mm tt',
45
- timeSuffix: '',
46
- timeOnlyTitle: 'Choose Time',
47
- timeText: 'Time',
48
- hourText: 'Hour',
49
- minuteText: 'Minute',
50
- secondText: 'Second',
51
- millisecText: 'Millisecond',
52
- timezoneText: 'Time Zone'
1
+ /*! jQuery Timepicker Addon - v1.6.3 - 2016-04-20
2
+ * http://trentrichardson.com/examples/timepicker
3
+ * Copyright (c) 2016 Trent Richardson; Licensed MIT */
4
+ (function (factory) {
5
+ if (typeof define === 'function' && define.amd) {
6
+ define(['jquery', 'jquery-ui'], factory);
7
+ } else {
8
+ factory(jQuery);
9
+ }
10
+ }(function ($) {
11
+
12
+ /*
13
+ * Lets not redefine timepicker, Prevent "Uncaught RangeError: Maximum call stack size exceeded"
14
+ */
15
+ $.ui.timepicker = $.ui.timepicker || {};
16
+ if ($.ui.timepicker.version) {
17
+ return;
18
+ }
19
+
20
+ /*
21
+ * Extend jQueryUI, get it started with our version number
22
+ */
23
+ $.extend($.ui, {
24
+ timepicker: {
25
+ version: "1.6.3"
26
+ }
27
+ });
28
+
29
+ /*
30
+ * Timepicker manager.
31
+ * Use the singleton instance of this class, $.timepicker, to interact with the time picker.
32
+ * Settings for (groups of) time pickers are maintained in an instance object,
33
+ * allowing multiple different settings on the same page.
34
+ */
35
+ var Timepicker = function () {
36
+ this.regional = []; // Available regional settings, indexed by language code
37
+ this.regional[''] = { // Default regional settings
38
+ currentText: 'Now',
39
+ closeText: 'Done',
40
+ amNames: ['AM', 'A'],
41
+ pmNames: ['PM', 'P'],
42
+ timeFormat: 'HH:mm',
43
+ timeSuffix: '',
44
+ timeOnlyTitle: 'Choose Time',
45
+ timeText: 'Time',
46
+ hourText: 'Hour',
47
+ minuteText: 'Minute',
48
+ secondText: 'Second',
49
+ millisecText: 'Millisecond',
50
+ microsecText: 'Microsecond',
51
+ timezoneText: 'Time Zone',
52
+ isRTL: false
53
+ };
54
+ this._defaults = { // Global defaults for all the datetime picker instances
55
+ showButtonPanel: true,
56
+ timeOnly: false,
57
+ timeOnlyShowDate: false,
58
+ showHour: null,
59
+ showMinute: null,
60
+ showSecond: null,
61
+ showMillisec: null,
62
+ showMicrosec: null,
63
+ showTimezone: null,
64
+ showTime: true,
65
+ stepHour: 1,
66
+ stepMinute: 1,
67
+ stepSecond: 1,
68
+ stepMillisec: 1,
69
+ stepMicrosec: 1,
70
+ hour: 0,
71
+ minute: 0,
72
+ second: 0,
73
+ millisec: 0,
74
+ microsec: 0,
75
+ timezone: null,
76
+ hourMin: 0,
77
+ minuteMin: 0,
78
+ secondMin: 0,
79
+ millisecMin: 0,
80
+ microsecMin: 0,
81
+ hourMax: 23,
82
+ minuteMax: 59,
83
+ secondMax: 59,
84
+ millisecMax: 999,
85
+ microsecMax: 999,
86
+ minDateTime: null,
87
+ maxDateTime: null,
88
+ maxTime: null,
89
+ minTime: null,
90
+ onSelect: null,
91
+ hourGrid: 0,
92
+ minuteGrid: 0,
93
+ secondGrid: 0,
94
+ millisecGrid: 0,
95
+ microsecGrid: 0,
96
+ alwaysSetTime: true,
97
+ separator: ' ',
98
+ altFieldTimeOnly: true,
99
+ altTimeFormat: null,
100
+ altSeparator: null,
101
+ altTimeSuffix: null,
102
+ altRedirectFocus: true,
103
+ pickerTimeFormat: null,
104
+ pickerTimeSuffix: null,
105
+ showTimepicker: true,
106
+ timezoneList: null,
107
+ addSliderAccess: false,
108
+ sliderAccessArgs: null,
109
+ controlType: 'slider',
110
+ oneLine: false,
111
+ defaultValue: null,
112
+ parse: 'strict',
113
+ afterInject: null
114
+ };
115
+ $.extend(this._defaults, this.regional['']);
53
116
  };
54
- this._defaults = { // Global defaults for all the datetime picker instances
55
- showButtonPanel: true,
56
- timeOnly: false,
57
- showHour: true,
58
- showMinute: true,
59
- showSecond: false,
60
- showMillisec: false,
61
- showTimezone: false,
62
- showTime: true,
63
- stepHour: 1,
64
- stepMinute: 1,
65
- stepSecond: 1,
66
- stepMillisec: 1,
117
+
118
+ $.extend(Timepicker.prototype, {
119
+ $input: null,
120
+ $altInput: null,
121
+ $timeObj: null,
122
+ inst: null,
123
+ hour_slider: null,
124
+ minute_slider: null,
125
+ second_slider: null,
126
+ millisec_slider: null,
127
+ microsec_slider: null,
128
+ timezone_select: null,
129
+ maxTime: null,
130
+ minTime: null,
67
131
  hour: 0,
68
132
  minute: 0,
69
133
  second: 0,
70
134
  millisec: 0,
71
- timezone: '+0000',
72
- hourMin: 0,
73
- minuteMin: 0,
74
- secondMin: 0,
75
- millisecMin: 0,
76
- hourMax: 23,
77
- minuteMax: 59,
78
- secondMax: 59,
79
- millisecMax: 999,
80
- minDateTime: null,
81
- maxDateTime: null,
82
- onSelect: null,
83
- hourGrid: 0,
84
- minuteGrid: 0,
85
- secondGrid: 0,
86
- millisecGrid: 0,
87
- alwaysSetTime: true,
88
- separator: ' ',
89
- altFieldTimeOnly: true,
90
- showTimepicker: true,
91
- timezoneIso8609: false,
135
+ microsec: 0,
136
+ timezone: null,
137
+ hourMinOriginal: null,
138
+ minuteMinOriginal: null,
139
+ secondMinOriginal: null,
140
+ millisecMinOriginal: null,
141
+ microsecMinOriginal: null,
142
+ hourMaxOriginal: null,
143
+ minuteMaxOriginal: null,
144
+ secondMaxOriginal: null,
145
+ millisecMaxOriginal: null,
146
+ microsecMaxOriginal: null,
147
+ ampm: '',
148
+ formattedDate: '',
149
+ formattedTime: '',
150
+ formattedDateTime: '',
92
151
  timezoneList: null,
93
- addSliderAccess: false,
94
- sliderAccessArgs: null
95
- };
96
- $.extend(this._defaults, this.regional['']);
97
- };
98
-
99
- $.extend(Timepicker.prototype, {
100
- $input: null,
101
- $altInput: null,
102
- $timeObj: null,
103
- inst: null,
104
- hour_slider: null,
105
- minute_slider: null,
106
- second_slider: null,
107
- millisec_slider: null,
108
- timezone_select: null,
109
- hour: 0,
110
- minute: 0,
111
- second: 0,
112
- millisec: 0,
113
- timezone: '+0000',
114
- hourMinOriginal: null,
115
- minuteMinOriginal: null,
116
- secondMinOriginal: null,
117
- millisecMinOriginal: null,
118
- hourMaxOriginal: null,
119
- minuteMaxOriginal: null,
120
- secondMaxOriginal: null,
121
- millisecMaxOriginal: null,
122
- ampm: '',
123
- formattedDate: '',
124
- formattedTime: '',
125
- formattedDateTime: '',
126
- timezoneList: null,
127
-
128
- /* Override the default settings for all instances of the time picker.
129
- @param settings object - the new settings to use as defaults (anonymous object)
130
- @return the manager object */
131
- setDefaults: function(settings) {
132
- extendRemove(this._defaults, settings || {});
133
- return this;
134
- },
135
-
136
- //########################################################################
137
- // Create a new Timepicker instance
138
- //########################################################################
139
- _newInst: function($input, o) {
140
- var tp_inst = new Timepicker(),
141
- inlineSettings = {};
142
-
143
- for (var attrName in this._defaults) {
144
- var attrValue = $input.attr('time:' + attrName);
145
- if (attrValue) {
152
+ units: ['hour', 'minute', 'second', 'millisec', 'microsec'],
153
+ support: {},
154
+ control: null,
155
+
156
+ /*
157
+ * Override the default settings for all instances of the time picker.
158
+ * @param {Object} settings object - the new settings to use as defaults (anonymous object)
159
+ * @return {Object} the manager object
160
+ */
161
+ setDefaults: function (settings) {
162
+ extendRemove(this._defaults, settings || {});
163
+ return this;
164
+ },
165
+
166
+ /*
167
+ * Create a new Timepicker instance
168
+ */
169
+ _newInst: function ($input, opts) {
170
+ var tp_inst = new Timepicker(),
171
+ inlineSettings = {},
172
+ fns = {},
173
+ overrides, i;
174
+
175
+ for (var attrName in this._defaults) {
176
+ if (this._defaults.hasOwnProperty(attrName)) {
177
+ var attrValue = $input.attr('time:' + attrName);
178
+ if (attrValue) {
179
+ try {
180
+ inlineSettings[attrName] = eval(attrValue);
181
+ } catch (err) {
182
+ inlineSettings[attrName] = attrValue;
183
+ }
184
+ }
185
+ }
186
+ }
187
+
188
+ overrides = {
189
+ beforeShow: function (input, dp_inst) {
190
+ if ($.isFunction(tp_inst._defaults.evnts.beforeShow)) {
191
+ return tp_inst._defaults.evnts.beforeShow.call($input[0], input, dp_inst, tp_inst);
192
+ }
193
+ },
194
+ onChangeMonthYear: function (year, month, dp_inst) {
195
+ // Update the time as well : this prevents the time from disappearing from the $input field.
196
+ // tp_inst._updateDateTime(dp_inst);
197
+ if ($.isFunction(tp_inst._defaults.evnts.onChangeMonthYear)) {
198
+ tp_inst._defaults.evnts.onChangeMonthYear.call($input[0], year, month, dp_inst, tp_inst);
199
+ }
200
+ },
201
+ onClose: function (dateText, dp_inst) {
202
+ if (tp_inst.timeDefined === true && $input.val() !== '') {
203
+ tp_inst._updateDateTime(dp_inst);
204
+ }
205
+ if ($.isFunction(tp_inst._defaults.evnts.onClose)) {
206
+ tp_inst._defaults.evnts.onClose.call($input[0], dateText, dp_inst, tp_inst);
207
+ }
208
+ }
209
+ };
210
+ for (i in overrides) {
211
+ if (overrides.hasOwnProperty(i)) {
212
+ fns[i] = opts[i] || this._defaults[i] || null;
213
+ }
214
+ }
215
+
216
+ tp_inst._defaults = $.extend({}, this._defaults, inlineSettings, opts, overrides, {
217
+ evnts: fns,
218
+ timepicker: tp_inst // add timepicker as a property of datepicker: $.datepicker._get(dp_inst, 'timepicker');
219
+ });
220
+ tp_inst.amNames = $.map(tp_inst._defaults.amNames, function (val) {
221
+ return val.toUpperCase();
222
+ });
223
+ tp_inst.pmNames = $.map(tp_inst._defaults.pmNames, function (val) {
224
+ return val.toUpperCase();
225
+ });
226
+
227
+ // detect which units are supported
228
+ tp_inst.support = detectSupport(
229
+ tp_inst._defaults.timeFormat +
230
+ (tp_inst._defaults.pickerTimeFormat ? tp_inst._defaults.pickerTimeFormat : '') +
231
+ (tp_inst._defaults.altTimeFormat ? tp_inst._defaults.altTimeFormat : ''));
232
+
233
+ // controlType is string - key to our this._controls
234
+ if (typeof(tp_inst._defaults.controlType) === 'string') {
235
+ if (tp_inst._defaults.controlType === 'slider' && typeof($.ui.slider) === 'undefined') {
236
+ tp_inst._defaults.controlType = 'select';
237
+ }
238
+ tp_inst.control = tp_inst._controls[tp_inst._defaults.controlType];
239
+ }
240
+ // controlType is an object and must implement create, options, value methods
241
+ else {
242
+ tp_inst.control = tp_inst._defaults.controlType;
243
+ }
244
+
245
+ // prep the timezone options
246
+ var timezoneList = [-720, -660, -600, -570, -540, -480, -420, -360, -300, -270, -240, -210, -180, -120, -60,
247
+ 0, 60, 120, 180, 210, 240, 270, 300, 330, 345, 360, 390, 420, 480, 525, 540, 570, 600, 630, 660, 690, 720, 765, 780, 840];
248
+ if (tp_inst._defaults.timezoneList !== null) {
249
+ timezoneList = tp_inst._defaults.timezoneList;
250
+ }
251
+ var tzl = timezoneList.length, tzi = 0, tzv = null;
252
+ if (tzl > 0 && typeof timezoneList[0] !== 'object') {
253
+ for (; tzi < tzl; tzi++) {
254
+ tzv = timezoneList[tzi];
255
+ timezoneList[tzi] = { value: tzv, label: $.timepicker.timezoneOffsetString(tzv, tp_inst.support.iso8601) };
256
+ }
257
+ }
258
+ tp_inst._defaults.timezoneList = timezoneList;
259
+
260
+ // set the default units
261
+ tp_inst.timezone = tp_inst._defaults.timezone !== null ? $.timepicker.timezoneOffsetNumber(tp_inst._defaults.timezone) :
262
+ ((new Date()).getTimezoneOffset() * -1);
263
+ tp_inst.hour = tp_inst._defaults.hour < tp_inst._defaults.hourMin ? tp_inst._defaults.hourMin :
264
+ tp_inst._defaults.hour > tp_inst._defaults.hourMax ? tp_inst._defaults.hourMax : tp_inst._defaults.hour;
265
+ tp_inst.minute = tp_inst._defaults.minute < tp_inst._defaults.minuteMin ? tp_inst._defaults.minuteMin :
266
+ tp_inst._defaults.minute > tp_inst._defaults.minuteMax ? tp_inst._defaults.minuteMax : tp_inst._defaults.minute;
267
+ tp_inst.second = tp_inst._defaults.second < tp_inst._defaults.secondMin ? tp_inst._defaults.secondMin :
268
+ tp_inst._defaults.second > tp_inst._defaults.secondMax ? tp_inst._defaults.secondMax : tp_inst._defaults.second;
269
+ tp_inst.millisec = tp_inst._defaults.millisec < tp_inst._defaults.millisecMin ? tp_inst._defaults.millisecMin :
270
+ tp_inst._defaults.millisec > tp_inst._defaults.millisecMax ? tp_inst._defaults.millisecMax : tp_inst._defaults.millisec;
271
+ tp_inst.microsec = tp_inst._defaults.microsec < tp_inst._defaults.microsecMin ? tp_inst._defaults.microsecMin :
272
+ tp_inst._defaults.microsec > tp_inst._defaults.microsecMax ? tp_inst._defaults.microsecMax : tp_inst._defaults.microsec;
273
+ tp_inst.ampm = '';
274
+ tp_inst.$input = $input;
275
+
276
+ if (tp_inst._defaults.altField) {
277
+ tp_inst.$altInput = $(tp_inst._defaults.altField);
278
+ if (tp_inst._defaults.altRedirectFocus === true) {
279
+ tp_inst.$altInput.css({
280
+ cursor: 'pointer'
281
+ }).focus(function () {
282
+ $input.trigger("focus");
283
+ });
284
+ }
285
+ }
286
+
287
+ if (tp_inst._defaults.minDate === 0 || tp_inst._defaults.minDateTime === 0) {
288
+ tp_inst._defaults.minDate = new Date();
289
+ }
290
+ if (tp_inst._defaults.maxDate === 0 || tp_inst._defaults.maxDateTime === 0) {
291
+ tp_inst._defaults.maxDate = new Date();
292
+ }
293
+
294
+ // datepicker needs minDate/maxDate, timepicker needs minDateTime/maxDateTime..
295
+ if (tp_inst._defaults.minDate !== undefined && tp_inst._defaults.minDate instanceof Date) {
296
+ tp_inst._defaults.minDateTime = new Date(tp_inst._defaults.minDate.getTime());
297
+ }
298
+ if (tp_inst._defaults.minDateTime !== undefined && tp_inst._defaults.minDateTime instanceof Date) {
299
+ tp_inst._defaults.minDate = new Date(tp_inst._defaults.minDateTime.getTime());
300
+ }
301
+ if (tp_inst._defaults.maxDate !== undefined && tp_inst._defaults.maxDate instanceof Date) {
302
+ tp_inst._defaults.maxDateTime = new Date(tp_inst._defaults.maxDate.getTime());
303
+ }
304
+ if (tp_inst._defaults.maxDateTime !== undefined && tp_inst._defaults.maxDateTime instanceof Date) {
305
+ tp_inst._defaults.maxDate = new Date(tp_inst._defaults.maxDateTime.getTime());
306
+ }
307
+ tp_inst.$input.bind('focus', function () {
308
+ tp_inst._onFocus();
309
+ });
310
+
311
+ return tp_inst;
312
+ },
313
+
314
+ /*
315
+ * add our sliders to the calendar
316
+ */
317
+ _addTimePicker: function (dp_inst) {
318
+ var currDT = $.trim((this.$altInput && this._defaults.altFieldTimeOnly) ? this.$input.val() + ' ' + this.$altInput.val() : this.$input.val());
319
+
320
+ this.timeDefined = this._parseTime(currDT);
321
+ this._limitMinMaxDateTime(dp_inst, false);
322
+ this._injectTimePicker();
323
+ this._afterInject();
324
+ },
325
+
326
+ /*
327
+ * parse the time string from input value or _setTime
328
+ */
329
+ _parseTime: function (timeString, withDate) {
330
+ if (!this.inst) {
331
+ this.inst = $.datepicker._getInst(this.$input[0]);
332
+ }
333
+
334
+ if (withDate || !this._defaults.timeOnly) {
335
+ var dp_dateFormat = $.datepicker._get(this.inst, 'dateFormat');
146
336
  try {
147
- inlineSettings[attrName] = eval(attrValue);
337
+ var parseRes = parseDateTimeInternal(dp_dateFormat, this._defaults.timeFormat, timeString, $.datepicker._getFormatConfig(this.inst), this._defaults);
338
+ if (!parseRes.timeObj) {
339
+ return false;
340
+ }
341
+ $.extend(this, parseRes.timeObj);
148
342
  } catch (err) {
149
- inlineSettings[attrName] = attrValue;
343
+ $.timepicker.log("Error parsing the date/time string: " + err +
344
+ "\ndate/time string = " + timeString +
345
+ "\ntimeFormat = " + this._defaults.timeFormat +
346
+ "\ndateFormat = " + dp_dateFormat);
347
+ return false;
150
348
  }
349
+ return true;
350
+ } else {
351
+ var timeObj = $.datepicker.parseTime(this._defaults.timeFormat, timeString, this._defaults);
352
+ if (!timeObj) {
353
+ return false;
354
+ }
355
+ $.extend(this, timeObj);
356
+ return true;
151
357
  }
152
- }
153
- tp_inst._defaults = $.extend({}, this._defaults, inlineSettings, o, {
154
- beforeShow: function(input, dp_inst) {
155
- if ($.isFunction(o.beforeShow))
156
- return o.beforeShow(input, dp_inst, tp_inst);
157
- },
158
- onChangeMonthYear: function(year, month, dp_inst) {
159
- // Update the time as well : this prevents the time from disappearing from the $input field.
160
- tp_inst._updateDateTime(dp_inst);
161
- if ($.isFunction(o.onChangeMonthYear))
162
- o.onChangeMonthYear.call($input[0], year, month, dp_inst, tp_inst);
163
- },
164
- onClose: function(dateText, dp_inst) {
165
- if (tp_inst.timeDefined === true && $input.val() != '')
166
- tp_inst._updateDateTime(dp_inst);
167
- if ($.isFunction(o.onClose))
168
- o.onClose.call($input[0], dateText, dp_inst, tp_inst);
169
- },
170
- timepicker: tp_inst // add timepicker as a property of datepicker: $.datepicker._get(dp_inst, 'timepicker');
171
- });
172
- tp_inst.amNames = $.map(tp_inst._defaults.amNames, function(val) { return val.toUpperCase(); });
173
- tp_inst.pmNames = $.map(tp_inst._defaults.pmNames, function(val) { return val.toUpperCase(); });
174
-
175
- if (tp_inst._defaults.timezoneList === null) {
176
- var timezoneList = [];
177
- for (var i = -11; i <= 12; i++)
178
- timezoneList.push((i >= 0 ? '+' : '-') + ('0' + Math.abs(i).toString()).slice(-2) + '00');
179
- if (tp_inst._defaults.timezoneIso8609)
180
- timezoneList = $.map(timezoneList, function(val) {
181
- return val == '+0000' ? 'Z' : (val.substring(0, 3) + ':' + val.substring(3));
182
- });
183
- tp_inst._defaults.timezoneList = timezoneList;
184
- }
358
+ },
359
+
360
+ /*
361
+ * Handle callback option after injecting timepicker
362
+ */
363
+ _afterInject: function() {
364
+ var o = this.inst.settings;
365
+ if ($.isFunction(o.afterInject)) {
366
+ o.afterInject.call(this);
367
+ }
368
+ },
369
+
370
+ /*
371
+ * generate and inject html for timepicker into ui datepicker
372
+ */
373
+ _injectTimePicker: function () {
374
+ var $dp = this.inst.dpDiv,
375
+ o = this.inst.settings,
376
+ tp_inst = this,
377
+ litem = '',
378
+ uitem = '',
379
+ show = null,
380
+ max = {},
381
+ gridSize = {},
382
+ size = null,
383
+ i = 0,
384
+ l = 0;
385
+
386
+ // Prevent displaying twice
387
+ if ($dp.find("div.ui-timepicker-div").length === 0 && o.showTimepicker) {
388
+ var noDisplay = ' ui_tpicker_unit_hide',
389
+ html = '<div class="ui-timepicker-div' + (o.isRTL ? ' ui-timepicker-rtl' : '') + (o.oneLine && o.controlType === 'select' ? ' ui-timepicker-oneLine' : '') + '"><dl>' + '<dt class="ui_tpicker_time_label' + ((o.showTime) ? '' : noDisplay) + '">' + o.timeText + '</dt>' +
390
+ '<dd class="ui_tpicker_time '+ ((o.showTime) ? '' : noDisplay) + '"><input class="ui_tpicker_time_input" ' + (o.timeInput ? '' : 'disabled') + '/></dd>';
391
+
392
+ // Create the markup
393
+ for (i = 0, l = this.units.length; i < l; i++) {
394
+ litem = this.units[i];
395
+ uitem = litem.substr(0, 1).toUpperCase() + litem.substr(1);
396
+ show = o['show' + uitem] !== null ? o['show' + uitem] : this.support[litem];
397
+
398
+ // Added by Peter Medeiros:
399
+ // - Figure out what the hour/minute/second max should be based on the step values.
400
+ // - Example: if stepMinute is 15, then minMax is 45.
401
+ max[litem] = parseInt((o[litem + 'Max'] - ((o[litem + 'Max'] - o[litem + 'Min']) % o['step' + uitem])), 10);
402
+ gridSize[litem] = 0;
403
+
404
+ html += '<dt class="ui_tpicker_' + litem + '_label' + (show ? '' : noDisplay) + '">' + o[litem + 'Text'] + '</dt>' +
405
+ '<dd class="ui_tpicker_' + litem + (show ? '' : noDisplay) + '"><div class="ui_tpicker_' + litem + '_slider' + (show ? '' : noDisplay) + '"></div>';
406
+
407
+ if (show && o[litem + 'Grid'] > 0) {
408
+ html += '<div style="padding-left: 1px"><table class="ui-tpicker-grid-label"><tr>';
409
+
410
+ if (litem === 'hour') {
411
+ for (var h = o[litem + 'Min']; h <= max[litem]; h += parseInt(o[litem + 'Grid'], 10)) {
412
+ gridSize[litem]++;
413
+ var tmph = $.datepicker.formatTime(this.support.ampm ? 'hht' : 'HH', {hour: h}, o);
414
+ html += '<td data-for="' + litem + '">' + tmph + '</td>';
415
+ }
416
+ }
417
+ else {
418
+ for (var m = o[litem + 'Min']; m <= max[litem]; m += parseInt(o[litem + 'Grid'], 10)) {
419
+ gridSize[litem]++;
420
+ html += '<td data-for="' + litem + '">' + ((m < 10) ? '0' : '') + m + '</td>';
421
+ }
422
+ }
185
423
 
186
- tp_inst.hour = tp_inst._defaults.hour;
187
- tp_inst.minute = tp_inst._defaults.minute;
188
- tp_inst.second = tp_inst._defaults.second;
189
- tp_inst.millisec = tp_inst._defaults.millisec;
190
- tp_inst.ampm = '';
191
- tp_inst.$input = $input;
192
-
193
- if (o.altField)
194
- tp_inst.$altInput = $(o.altField)
195
- .css({ cursor: 'pointer' })
196
- .focus(function(){ $input.trigger("focus"); });
197
-
198
- if(tp_inst._defaults.minDate==0 || tp_inst._defaults.minDateTime==0)
199
- {
200
- tp_inst._defaults.minDate=new Date();
201
- }
202
- if(tp_inst._defaults.maxDate==0 || tp_inst._defaults.maxDateTime==0)
203
- {
204
- tp_inst._defaults.maxDate=new Date();
205
- }
424
+ html += '</tr></table></div>';
425
+ }
426
+ html += '</dd>';
427
+ }
206
428
 
207
- // datepicker needs minDate/maxDate, timepicker needs minDateTime/maxDateTime..
208
- if(tp_inst._defaults.minDate !== undefined && tp_inst._defaults.minDate instanceof Date)
209
- tp_inst._defaults.minDateTime = new Date(tp_inst._defaults.minDate.getTime());
210
- if(tp_inst._defaults.minDateTime !== undefined && tp_inst._defaults.minDateTime instanceof Date)
211
- tp_inst._defaults.minDate = new Date(tp_inst._defaults.minDateTime.getTime());
212
- if(tp_inst._defaults.maxDate !== undefined && tp_inst._defaults.maxDate instanceof Date)
213
- tp_inst._defaults.maxDateTime = new Date(tp_inst._defaults.maxDate.getTime());
214
- if(tp_inst._defaults.maxDateTime !== undefined && tp_inst._defaults.maxDateTime instanceof Date)
215
- tp_inst._defaults.maxDate = new Date(tp_inst._defaults.maxDateTime.getTime());
216
- return tp_inst;
217
- },
218
-
219
- //########################################################################
220
- // add our sliders to the calendar
221
- //########################################################################
222
- _addTimePicker: function(dp_inst) {
223
- var currDT = (this.$altInput && this._defaults.altFieldTimeOnly) ?
224
- this.$input.val() + ' ' + this.$altInput.val() :
225
- this.$input.val();
226
-
227
- this.timeDefined = this._parseTime(currDT);
228
- this._limitMinMaxDateTime(dp_inst, false);
229
- this._injectTimePicker();
230
- },
231
-
232
- //########################################################################
233
- // parse the time string from input value or _setTime
234
- //########################################################################
235
- _parseTime: function(timeString, withDate) {
236
- var regstr = this._defaults.timeFormat.toString()
237
- .replace(/h{1,2}/ig, '(\\d?\\d)')
238
- .replace(/m{1,2}/ig, '(\\d?\\d)')
239
- .replace(/s{1,2}/ig, '(\\d?\\d)')
240
- .replace(/l{1}/ig, '(\\d?\\d?\\d)')
241
- .replace(/t{1,2}/ig, this._getPatternAmpm())
242
- .replace(/z{1}/ig, '(z|[-+]\\d\\d:?\\d\\d)?')
243
- .replace(/\s/g, '\\s?') + this._defaults.timeSuffix + '$',
244
- order = this._getFormatPositions(),
245
- ampm = '',
246
- treg;
247
-
248
- if (!this.inst) this.inst = $.datepicker._getInst(this.$input[0]);
249
-
250
- if (withDate || !this._defaults.timeOnly) {
251
- // the time should come after x number of characters and a space.
252
- // x = at least the length of text specified by the date format
253
- var dp_dateFormat = $.datepicker._get(this.inst, 'dateFormat');
254
- // escape special regex characters in the seperator
255
- var specials = new RegExp("[.*+?|()\\[\\]{}\\\\]", "g");
256
- regstr = '^.{' + dp_dateFormat.length + ',}?' + this._defaults.separator.replace(specials, "\\$&") + regstr;
257
- }
429
+ // Timezone
430
+ var showTz = o.showTimezone !== null ? o.showTimezone : this.support.timezone;
431
+ html += '<dt class="ui_tpicker_timezone_label' + (showTz ? '' : noDisplay) + '">' + o.timezoneText + '</dt>';
432
+ html += '<dd class="ui_tpicker_timezone' + (showTz ? '' : noDisplay) + '"></dd>';
258
433
 
259
- treg = timeString.match(new RegExp(regstr, 'i'));
434
+ // Create the elements from string
435
+ html += '</dl></div>';
436
+ var $tp = $(html);
260
437
 
261
- if (treg) {
262
- if (order.t !== -1) {
263
- if (treg[order.t] === undefined || treg[order.t].length === 0) {
264
- ampm = '';
265
- this.ampm = '';
266
- } else {
267
- ampm = $.inArray(treg[order.t].toUpperCase(), this.amNames) !== -1 ? 'AM' : 'PM';
268
- this.ampm = this._defaults[ampm == 'AM' ? 'amNames' : 'pmNames'][0];
269
- }
270
- }
271
-
272
- if (order.h !== -1) {
273
- if (ampm == 'AM' && treg[order.h] == '12')
274
- this.hour = 0; // 12am = 0 hour
275
- else if (ampm == 'PM' && treg[order.h] != '12')
276
- this.hour = (parseFloat(treg[order.h]) + 12).toFixed(0); // 12pm = 12 hour, any other pm = hour + 12
277
- else this.hour = Number(treg[order.h]);
278
- }
279
-
280
- if (order.m !== -1) this.minute = Number(treg[order.m]);
281
- if (order.s !== -1) this.second = Number(treg[order.s]);
282
- if (order.l !== -1) this.millisec = Number(treg[order.l]);
283
- if (order.z !== -1 && treg[order.z] !== undefined) {
284
- var tz = treg[order.z].toUpperCase();
285
- switch (tz.length) {
286
- case 1: // Z
287
- tz = this._defaults.timezoneIso8609 ? 'Z' : '+0000';
288
- break;
289
- case 5: // +hhmm
290
- if (this._defaults.timezoneIso8609)
291
- tz = tz.substring(1) == '0000'
292
- ? 'Z'
293
- : tz.substring(0, 3) + ':' + tz.substring(3);
294
- break;
295
- case 6: // +hh:mm
296
- if (!this._defaults.timezoneIso8609)
297
- tz = tz == 'Z' || tz.substring(1) == '00:00'
298
- ? '+0000'
299
- : tz.replace(/:/, '');
300
- else if (tz.substring(1) == '00:00')
301
- tz = 'Z';
302
- break;
303
- }
304
- this.timezone = tz;
305
- }
306
-
307
- return true;
438
+ // if we only want time picker...
439
+ if (o.timeOnly === true) {
440
+ $tp.prepend('<div class="ui-widget-header ui-helper-clearfix ui-corner-all">' + '<div class="ui-datepicker-title">' + o.timeOnlyTitle + '</div>' + '</div>');
441
+ $dp.find('.ui-datepicker-header, .ui-datepicker-calendar').hide();
442
+ }
308
443
 
309
- }
310
- return false;
311
- },
312
-
313
- //########################################################################
314
- // pattern for standard and localized AM/PM markers
315
- //########################################################################
316
- _getPatternAmpm: function() {
317
- var markers = [],
318
- o = this._defaults;
319
- if (o.amNames)
320
- $.merge(markers, o.amNames);
321
- if (o.pmNames)
322
- $.merge(markers, o.pmNames);
323
- markers = $.map(markers, function(val) { return val.replace(/[.*+?|()\[\]{}\\]/g, '\\$&'); });
324
- return '(' + markers.join('|') + ')?';
325
- },
326
-
327
- //########################################################################
328
- // figure out position of time elements.. cause js cant do named captures
329
- //########################################################################
330
- _getFormatPositions: function() {
331
- var finds = this._defaults.timeFormat.toLowerCase().match(/(h{1,2}|m{1,2}|s{1,2}|l{1}|t{1,2}|z)/g),
332
- orders = { h: -1, m: -1, s: -1, l: -1, t: -1, z: -1 };
333
-
334
- if (finds)
335
- for (var i = 0; i < finds.length; i++)
336
- if (orders[finds[i].toString().charAt(0)] == -1)
337
- orders[finds[i].toString().charAt(0)] = i + 1;
338
-
339
- return orders;
340
- },
341
-
342
- //########################################################################
343
- // generate and inject html for timepicker into ui datepicker
344
- //########################################################################
345
- _injectTimePicker: function() {
346
- var $dp = this.inst.dpDiv,
347
- o = this._defaults,
348
- tp_inst = this,
349
- // Added by Peter Medeiros:
350
- // - Figure out what the hour/minute/second max should be based on the step values.
351
- // - Example: if stepMinute is 15, then minMax is 45.
352
- hourMax = parseInt((o.hourMax - ((o.hourMax - o.hourMin) % o.stepHour)) ,10),
353
- minMax = parseInt((o.minuteMax - ((o.minuteMax - o.minuteMin) % o.stepMinute)) ,10),
354
- secMax = parseInt((o.secondMax - ((o.secondMax - o.secondMin) % o.stepSecond)) ,10),
355
- millisecMax = parseInt((o.millisecMax - ((o.millisecMax - o.millisecMin) % o.stepMillisec)) ,10),
356
- dp_id = this.inst.id.toString().replace(/([^A-Za-z0-9_])/g, '');
357
-
358
- // Prevent displaying twice
359
- //if ($dp.find("div#ui-timepicker-div-"+ dp_id).length === 0) {
360
- if ($dp.find("div#ui-timepicker-div-"+ dp_id).length === 0 && o.showTimepicker) {
361
- var noDisplay = ' style="display:none;"',
362
- html = '<div class="ui-timepicker-div" id="ui-timepicker-div-' + dp_id + '"><dl>' +
363
- '<dt class="ui_tpicker_time_label" id="ui_tpicker_time_label_' + dp_id + '"' +
364
- ((o.showTime) ? '' : noDisplay) + '>' + o.timeText + '</dt>' +
365
- '<dd class="ui_tpicker_time" id="ui_tpicker_time_' + dp_id + '"' +
366
- ((o.showTime) ? '' : noDisplay) + '></dd>' +
367
- '<dt class="ui_tpicker_hour_label" id="ui_tpicker_hour_label_' + dp_id + '"' +
368
- ((o.showHour) ? '' : noDisplay) + '>' + o.hourText + '</dt>',
369
- hourGridSize = 0,
370
- minuteGridSize = 0,
371
- secondGridSize = 0,
372
- millisecGridSize = 0,
373
- size = null;
374
-
375
- // Hours
376
- html += '<dd class="ui_tpicker_hour"><div id="ui_tpicker_hour_' + dp_id + '"' +
377
- ((o.showHour) ? '' : noDisplay) + '></div>';
378
- if (o.showHour && o.hourGrid > 0) {
379
- html += '<div style="padding-left: 1px"><table class="ui-tpicker-grid-label"><tr>';
380
-
381
- for (var h = o.hourMin; h <= hourMax; h += parseInt(o.hourGrid,10)) {
382
- hourGridSize++;
383
- var tmph = (o.ampm && h > 12) ? h-12 : h;
384
- if (tmph < 10) tmph = '0' + tmph;
385
- if (o.ampm) {
386
- if (h == 0) tmph = 12 +'a';
387
- else if (h < 12) tmph += 'a';
388
- else tmph += 'p';
444
+ // add sliders, adjust grids, add events
445
+ for (i = 0, l = tp_inst.units.length; i < l; i++) {
446
+ litem = tp_inst.units[i];
447
+ uitem = litem.substr(0, 1).toUpperCase() + litem.substr(1);
448
+ show = o['show' + uitem] !== null ? o['show' + uitem] : this.support[litem];
449
+
450
+ // add the slider
451
+ tp_inst[litem + '_slider'] = tp_inst.control.create(tp_inst, $tp.find('.ui_tpicker_' + litem + '_slider'), litem, tp_inst[litem], o[litem + 'Min'], max[litem], o['step' + uitem]);
452
+
453
+ // adjust the grid and add click event
454
+ if (show && o[litem + 'Grid'] > 0) {
455
+ size = 100 * gridSize[litem] * o[litem + 'Grid'] / (max[litem] - o[litem + 'Min']);
456
+ $tp.find('.ui_tpicker_' + litem + ' table').css({
457
+ width: size + "%",
458
+ marginLeft: o.isRTL ? '0' : ((size / (-2 * gridSize[litem])) + "%"),
459
+ marginRight: o.isRTL ? ((size / (-2 * gridSize[litem])) + "%") : '0',
460
+ borderCollapse: 'collapse'
461
+ }).find("td").click(function (e) {
462
+ var $t = $(this),
463
+ h = $t.html(),
464
+ n = parseInt(h.replace(/[^0-9]/g), 10),
465
+ ap = h.replace(/[^apm]/ig),
466
+ f = $t.data('for'); // loses scope, so we use data-for
467
+
468
+ if (f === 'hour') {
469
+ if (ap.indexOf('p') !== -1 && n < 12) {
470
+ n += 12;
471
+ }
472
+ else {
473
+ if (ap.indexOf('a') !== -1 && n === 12) {
474
+ n = 0;
475
+ }
476
+ }
477
+ }
478
+
479
+ tp_inst.control.value(tp_inst, tp_inst[f + '_slider'], litem, n);
480
+
481
+ tp_inst._onTimeChange();
482
+ tp_inst._onSelectHandler();
483
+ }).css({
484
+ cursor: 'pointer',
485
+ width: (100 / gridSize[litem]) + '%',
486
+ textAlign: 'center',
487
+ overflow: 'hidden'
488
+ });
489
+ } // end if grid > 0
490
+ } // end for loop
491
+
492
+ // Add timezone options
493
+ this.timezone_select = $tp.find('.ui_tpicker_timezone').append('<select></select>').find("select");
494
+ $.fn.append.apply(this.timezone_select,
495
+ $.map(o.timezoneList, function (val, idx) {
496
+ return $("<option />").val(typeof val === "object" ? val.value : val).text(typeof val === "object" ? val.label : val);
497
+ }));
498
+ if (typeof(this.timezone) !== "undefined" && this.timezone !== null && this.timezone !== "") {
499
+ var local_timezone = (new Date(this.inst.selectedYear, this.inst.selectedMonth, this.inst.selectedDay, 12)).getTimezoneOffset() * -1;
500
+ if (local_timezone === this.timezone) {
501
+ selectLocalTimezone(tp_inst);
502
+ } else {
503
+ this.timezone_select.val(this.timezone);
504
+ }
505
+ } else {
506
+ if (typeof(this.hour) !== "undefined" && this.hour !== null && this.hour !== "") {
507
+ this.timezone_select.val(o.timezone);
508
+ } else {
509
+ selectLocalTimezone(tp_inst);
389
510
  }
390
- html += '<td>' + tmph + '</td>';
391
511
  }
512
+ this.timezone_select.change(function () {
513
+ tp_inst._onTimeChange();
514
+ tp_inst._onSelectHandler();
515
+ tp_inst._afterInject();
516
+ });
517
+ // End timezone options
392
518
 
393
- html += '</tr></table></div>';
394
- }
395
- html += '</dd>';
519
+ // inject timepicker into datepicker
520
+ var $buttonPanel = $dp.find('.ui-datepicker-buttonpane');
521
+ if ($buttonPanel.length) {
522
+ $buttonPanel.before($tp);
523
+ } else {
524
+ $dp.append($tp);
525
+ }
396
526
 
397
- // Minutes
398
- html += '<dt class="ui_tpicker_minute_label" id="ui_tpicker_minute_label_' + dp_id + '"' +
399
- ((o.showMinute) ? '' : noDisplay) + '>' + o.minuteText + '</dt>'+
400
- '<dd class="ui_tpicker_minute"><div id="ui_tpicker_minute_' + dp_id + '"' +
401
- ((o.showMinute) ? '' : noDisplay) + '></div>';
527
+ this.$timeObj = $tp.find('.ui_tpicker_time_input');
528
+ this.$timeObj.change(function () {
529
+ var timeFormat = tp_inst.inst.settings.timeFormat;
530
+ var parsedTime = $.datepicker.parseTime(timeFormat, this.value);
531
+ var update = new Date();
532
+ if (parsedTime) {
533
+ update.setHours(parsedTime.hour);
534
+ update.setMinutes(parsedTime.minute);
535
+ update.setSeconds(parsedTime.second);
536
+ $.datepicker._setTime(tp_inst.inst, update);
537
+ } else {
538
+ this.value = tp_inst.formattedTime;
539
+ this.blur();
540
+ }
541
+ });
402
542
 
403
- if (o.showMinute && o.minuteGrid > 0) {
404
- html += '<div style="padding-left: 1px"><table class="ui-tpicker-grid-label"><tr>';
543
+ if (this.inst !== null) {
544
+ var timeDefined = this.timeDefined;
545
+ this._onTimeChange();
546
+ this.timeDefined = timeDefined;
547
+ }
405
548
 
406
- for (var m = o.minuteMin; m <= minMax; m += parseInt(o.minuteGrid,10)) {
407
- minuteGridSize++;
408
- html += '<td>' + ((m < 10) ? '0' : '') + m + '</td>';
549
+ // slideAccess integration: http://trentrichardson.com/2011/11/11/jquery-ui-sliders-and-touch-accessibility/
550
+ if (this._defaults.addSliderAccess) {
551
+ var sliderAccessArgs = this._defaults.sliderAccessArgs,
552
+ rtl = this._defaults.isRTL;
553
+ sliderAccessArgs.isRTL = rtl;
554
+
555
+ setTimeout(function () { // fix for inline mode
556
+ if ($tp.find('.ui-slider-access').length === 0) {
557
+ $tp.find('.ui-slider:visible').sliderAccess(sliderAccessArgs);
558
+
559
+ // fix any grids since sliders are shorter
560
+ var sliderAccessWidth = $tp.find('.ui-slider-access:eq(0)').outerWidth(true);
561
+ if (sliderAccessWidth) {
562
+ $tp.find('table:visible').each(function () {
563
+ var $g = $(this),
564
+ oldWidth = $g.outerWidth(),
565
+ oldMarginLeft = $g.css(rtl ? 'marginRight' : 'marginLeft').toString().replace('%', ''),
566
+ newWidth = oldWidth - sliderAccessWidth,
567
+ newMarginLeft = ((oldMarginLeft * newWidth) / oldWidth) + '%',
568
+ css = { width: newWidth, marginRight: 0, marginLeft: 0 };
569
+ css[rtl ? 'marginRight' : 'marginLeft'] = newMarginLeft;
570
+ $g.css(css);
571
+ });
572
+ }
573
+ }
574
+ }, 10);
409
575
  }
576
+ // end slideAccess integration
410
577
 
411
- html += '</tr></table></div>';
578
+ tp_inst._limitMinMaxDateTime(this.inst, true);
579
+ }
580
+ },
581
+
582
+ /*
583
+ * This function tries to limit the ability to go outside the
584
+ * min/max date range
585
+ */
586
+ _limitMinMaxDateTime: function (dp_inst, adjustSliders) {
587
+ var o = this._defaults,
588
+ dp_date = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay);
589
+
590
+ if (!this._defaults.showTimepicker) {
591
+ return;
592
+ } // No time so nothing to check here
593
+
594
+ if ($.datepicker._get(dp_inst, 'minDateTime') !== null && $.datepicker._get(dp_inst, 'minDateTime') !== undefined && dp_date) {
595
+ var minDateTime = $.datepicker._get(dp_inst, 'minDateTime'),
596
+ minDateTimeDate = new Date(minDateTime.getFullYear(), minDateTime.getMonth(), minDateTime.getDate(), 0, 0, 0, 0);
597
+
598
+ if (this.hourMinOriginal === null || this.minuteMinOriginal === null || this.secondMinOriginal === null || this.millisecMinOriginal === null || this.microsecMinOriginal === null) {
599
+ this.hourMinOriginal = o.hourMin;
600
+ this.minuteMinOriginal = o.minuteMin;
601
+ this.secondMinOriginal = o.secondMin;
602
+ this.millisecMinOriginal = o.millisecMin;
603
+ this.microsecMinOriginal = o.microsecMin;
604
+ }
605
+
606
+ if (dp_inst.settings.timeOnly || minDateTimeDate.getTime() === dp_date.getTime()) {
607
+ this._defaults.hourMin = minDateTime.getHours();
608
+ if (this.hour <= this._defaults.hourMin) {
609
+ this.hour = this._defaults.hourMin;
610
+ this._defaults.minuteMin = minDateTime.getMinutes();
611
+ if (this.minute <= this._defaults.minuteMin) {
612
+ this.minute = this._defaults.minuteMin;
613
+ this._defaults.secondMin = minDateTime.getSeconds();
614
+ if (this.second <= this._defaults.secondMin) {
615
+ this.second = this._defaults.secondMin;
616
+ this._defaults.millisecMin = minDateTime.getMilliseconds();
617
+ if (this.millisec <= this._defaults.millisecMin) {
618
+ this.millisec = this._defaults.millisecMin;
619
+ this._defaults.microsecMin = minDateTime.getMicroseconds();
620
+ } else {
621
+ if (this.microsec < this._defaults.microsecMin) {
622
+ this.microsec = this._defaults.microsecMin;
623
+ }
624
+ this._defaults.microsecMin = this.microsecMinOriginal;
625
+ }
626
+ } else {
627
+ this._defaults.millisecMin = this.millisecMinOriginal;
628
+ this._defaults.microsecMin = this.microsecMinOriginal;
629
+ }
630
+ } else {
631
+ this._defaults.secondMin = this.secondMinOriginal;
632
+ this._defaults.millisecMin = this.millisecMinOriginal;
633
+ this._defaults.microsecMin = this.microsecMinOriginal;
634
+ }
635
+ } else {
636
+ this._defaults.minuteMin = this.minuteMinOriginal;
637
+ this._defaults.secondMin = this.secondMinOriginal;
638
+ this._defaults.millisecMin = this.millisecMinOriginal;
639
+ this._defaults.microsecMin = this.microsecMinOriginal;
640
+ }
641
+ } else {
642
+ this._defaults.hourMin = this.hourMinOriginal;
643
+ this._defaults.minuteMin = this.minuteMinOriginal;
644
+ this._defaults.secondMin = this.secondMinOriginal;
645
+ this._defaults.millisecMin = this.millisecMinOriginal;
646
+ this._defaults.microsecMin = this.microsecMinOriginal;
647
+ }
412
648
  }
413
- html += '</dd>';
414
649
 
415
- // Seconds
416
- html += '<dt class="ui_tpicker_second_label" id="ui_tpicker_second_label_' + dp_id + '"' +
417
- ((o.showSecond) ? '' : noDisplay) + '>' + o.secondText + '</dt>'+
418
- '<dd class="ui_tpicker_second"><div id="ui_tpicker_second_' + dp_id + '"'+
419
- ((o.showSecond) ? '' : noDisplay) + '></div>';
650
+ if ($.datepicker._get(dp_inst, 'maxDateTime') !== null && $.datepicker._get(dp_inst, 'maxDateTime') !== undefined && dp_date) {
651
+ var maxDateTime = $.datepicker._get(dp_inst, 'maxDateTime'),
652
+ maxDateTimeDate = new Date(maxDateTime.getFullYear(), maxDateTime.getMonth(), maxDateTime.getDate(), 0, 0, 0, 0);
420
653
 
421
- if (o.showSecond && o.secondGrid > 0) {
422
- html += '<div style="padding-left: 1px"><table><tr>';
654
+ if (this.hourMaxOriginal === null || this.minuteMaxOriginal === null || this.secondMaxOriginal === null || this.millisecMaxOriginal === null) {
655
+ this.hourMaxOriginal = o.hourMax;
656
+ this.minuteMaxOriginal = o.minuteMax;
657
+ this.secondMaxOriginal = o.secondMax;
658
+ this.millisecMaxOriginal = o.millisecMax;
659
+ this.microsecMaxOriginal = o.microsecMax;
660
+ }
423
661
 
424
- for (var s = o.secondMin; s <= secMax; s += parseInt(o.secondGrid,10)) {
425
- secondGridSize++;
426
- html += '<td>' + ((s < 10) ? '0' : '') + s + '</td>';
662
+ if (dp_inst.settings.timeOnly || maxDateTimeDate.getTime() === dp_date.getTime()) {
663
+ this._defaults.hourMax = maxDateTime.getHours();
664
+ if (this.hour >= this._defaults.hourMax) {
665
+ this.hour = this._defaults.hourMax;
666
+ this._defaults.minuteMax = maxDateTime.getMinutes();
667
+ if (this.minute >= this._defaults.minuteMax) {
668
+ this.minute = this._defaults.minuteMax;
669
+ this._defaults.secondMax = maxDateTime.getSeconds();
670
+ if (this.second >= this._defaults.secondMax) {
671
+ this.second = this._defaults.secondMax;
672
+ this._defaults.millisecMax = maxDateTime.getMilliseconds();
673
+ if (this.millisec >= this._defaults.millisecMax) {
674
+ this.millisec = this._defaults.millisecMax;
675
+ this._defaults.microsecMax = maxDateTime.getMicroseconds();
676
+ } else {
677
+ if (this.microsec > this._defaults.microsecMax) {
678
+ this.microsec = this._defaults.microsecMax;
679
+ }
680
+ this._defaults.microsecMax = this.microsecMaxOriginal;
681
+ }
682
+ } else {
683
+ this._defaults.millisecMax = this.millisecMaxOriginal;
684
+ this._defaults.microsecMax = this.microsecMaxOriginal;
685
+ }
686
+ } else {
687
+ this._defaults.secondMax = this.secondMaxOriginal;
688
+ this._defaults.millisecMax = this.millisecMaxOriginal;
689
+ this._defaults.microsecMax = this.microsecMaxOriginal;
690
+ }
691
+ } else {
692
+ this._defaults.minuteMax = this.minuteMaxOriginal;
693
+ this._defaults.secondMax = this.secondMaxOriginal;
694
+ this._defaults.millisecMax = this.millisecMaxOriginal;
695
+ this._defaults.microsecMax = this.microsecMaxOriginal;
696
+ }
697
+ } else {
698
+ this._defaults.hourMax = this.hourMaxOriginal;
699
+ this._defaults.minuteMax = this.minuteMaxOriginal;
700
+ this._defaults.secondMax = this.secondMaxOriginal;
701
+ this._defaults.millisecMax = this.millisecMaxOriginal;
702
+ this._defaults.microsecMax = this.microsecMaxOriginal;
427
703
  }
704
+ }
428
705
 
429
- html += '</tr></table></div>';
706
+ if (dp_inst.settings.minTime!==null) {
707
+ var tempMinTime=new Date("01/01/1970 " + dp_inst.settings.minTime);
708
+ if (this.hour<tempMinTime.getHours()) {
709
+ this.hour=this._defaults.hourMin=tempMinTime.getHours();
710
+ this.minute=this._defaults.minuteMin=tempMinTime.getMinutes();
711
+ } else if (this.hour===tempMinTime.getHours() && this.minute<tempMinTime.getMinutes()) {
712
+ this.minute=this._defaults.minuteMin=tempMinTime.getMinutes();
713
+ } else {
714
+ if (this._defaults.hourMin<tempMinTime.getHours()) {
715
+ this._defaults.hourMin=tempMinTime.getHours();
716
+ this._defaults.minuteMin=tempMinTime.getMinutes();
717
+ } else if (this._defaults.hourMin===tempMinTime.getHours()===this.hour && this._defaults.minuteMin<tempMinTime.getMinutes()) {
718
+ this._defaults.minuteMin=tempMinTime.getMinutes();
719
+ } else {
720
+ this._defaults.minuteMin=0;
721
+ }
722
+ }
430
723
  }
431
- html += '</dd>';
432
724
 
433
- // Milliseconds
434
- html += '<dt class="ui_tpicker_millisec_label" id="ui_tpicker_millisec_label_' + dp_id + '"' +
435
- ((o.showMillisec) ? '' : noDisplay) + '>' + o.millisecText + '</dt>'+
436
- '<dd class="ui_tpicker_millisec"><div id="ui_tpicker_millisec_' + dp_id + '"'+
437
- ((o.showMillisec) ? '' : noDisplay) + '></div>';
725
+ if (dp_inst.settings.maxTime!==null) {
726
+ var tempMaxTime=new Date("01/01/1970 " + dp_inst.settings.maxTime);
727
+ if (this.hour>tempMaxTime.getHours()) {
728
+ this.hour=this._defaults.hourMax=tempMaxTime.getHours();
729
+ this.minute=this._defaults.minuteMax=tempMaxTime.getMinutes();
730
+ } else if (this.hour===tempMaxTime.getHours() && this.minute>tempMaxTime.getMinutes()) {
731
+ this.minute=this._defaults.minuteMax=tempMaxTime.getMinutes();
732
+ } else {
733
+ if (this._defaults.hourMax>tempMaxTime.getHours()) {
734
+ this._defaults.hourMax=tempMaxTime.getHours();
735
+ this._defaults.minuteMax=tempMaxTime.getMinutes();
736
+ } else if (this._defaults.hourMax===tempMaxTime.getHours()===this.hour && this._defaults.minuteMax>tempMaxTime.getMinutes()) {
737
+ this._defaults.minuteMax=tempMaxTime.getMinutes();
738
+ } else {
739
+ this._defaults.minuteMax=59;
740
+ }
741
+ }
742
+ }
438
743
 
439
- if (o.showMillisec && o.millisecGrid > 0) {
440
- html += '<div style="padding-left: 1px"><table><tr>';
744
+ if (adjustSliders !== undefined && adjustSliders === true) {
745
+ var hourMax = parseInt((this._defaults.hourMax - ((this._defaults.hourMax - this._defaults.hourMin) % this._defaults.stepHour)), 10),
746
+ minMax = parseInt((this._defaults.minuteMax - ((this._defaults.minuteMax - this._defaults.minuteMin) % this._defaults.stepMinute)), 10),
747
+ secMax = parseInt((this._defaults.secondMax - ((this._defaults.secondMax - this._defaults.secondMin) % this._defaults.stepSecond)), 10),
748
+ millisecMax = parseInt((this._defaults.millisecMax - ((this._defaults.millisecMax - this._defaults.millisecMin) % this._defaults.stepMillisec)), 10),
749
+ microsecMax = parseInt((this._defaults.microsecMax - ((this._defaults.microsecMax - this._defaults.microsecMin) % this._defaults.stepMicrosec)), 10);
441
750
 
442
- for (var l = o.millisecMin; l <= millisecMax; l += parseInt(o.millisecGrid,10)) {
443
- millisecGridSize++;
444
- html += '<td>' + ((l < 10) ? '0' : '') + l + '</td>';
751
+ if (this.hour_slider) {
752
+ this.control.options(this, this.hour_slider, 'hour', { min: this._defaults.hourMin, max: hourMax, step: this._defaults.stepHour });
753
+ this.control.value(this, this.hour_slider, 'hour', this.hour - (this.hour % this._defaults.stepHour));
754
+ }
755
+ if (this.minute_slider) {
756
+ this.control.options(this, this.minute_slider, 'minute', { min: this._defaults.minuteMin, max: minMax, step: this._defaults.stepMinute });
757
+ this.control.value(this, this.minute_slider, 'minute', this.minute - (this.minute % this._defaults.stepMinute));
758
+ }
759
+ if (this.second_slider) {
760
+ this.control.options(this, this.second_slider, 'second', { min: this._defaults.secondMin, max: secMax, step: this._defaults.stepSecond });
761
+ this.control.value(this, this.second_slider, 'second', this.second - (this.second % this._defaults.stepSecond));
762
+ }
763
+ if (this.millisec_slider) {
764
+ this.control.options(this, this.millisec_slider, 'millisec', { min: this._defaults.millisecMin, max: millisecMax, step: this._defaults.stepMillisec });
765
+ this.control.value(this, this.millisec_slider, 'millisec', this.millisec - (this.millisec % this._defaults.stepMillisec));
766
+ }
767
+ if (this.microsec_slider) {
768
+ this.control.options(this, this.microsec_slider, 'microsec', { min: this._defaults.microsecMin, max: microsecMax, step: this._defaults.stepMicrosec });
769
+ this.control.value(this, this.microsec_slider, 'microsec', this.microsec - (this.microsec % this._defaults.stepMicrosec));
445
770
  }
771
+ }
772
+
773
+ },
446
774
 
447
- html += '</tr></table></div>';
775
+ /*
776
+ * when a slider moves, set the internal time...
777
+ * on time change is also called when the time is updated in the text field
778
+ */
779
+ _onTimeChange: function () {
780
+ if (!this._defaults.showTimepicker) {
781
+ return;
782
+ }
783
+ var hour = (this.hour_slider) ? this.control.value(this, this.hour_slider, 'hour') : false,
784
+ minute = (this.minute_slider) ? this.control.value(this, this.minute_slider, 'minute') : false,
785
+ second = (this.second_slider) ? this.control.value(this, this.second_slider, 'second') : false,
786
+ millisec = (this.millisec_slider) ? this.control.value(this, this.millisec_slider, 'millisec') : false,
787
+ microsec = (this.microsec_slider) ? this.control.value(this, this.microsec_slider, 'microsec') : false,
788
+ timezone = (this.timezone_select) ? this.timezone_select.val() : false,
789
+ o = this._defaults,
790
+ pickerTimeFormat = o.pickerTimeFormat || o.timeFormat,
791
+ pickerTimeSuffix = o.pickerTimeSuffix || o.timeSuffix;
792
+
793
+ if (typeof(hour) === 'object') {
794
+ hour = false;
795
+ }
796
+ if (typeof(minute) === 'object') {
797
+ minute = false;
798
+ }
799
+ if (typeof(second) === 'object') {
800
+ second = false;
801
+ }
802
+ if (typeof(millisec) === 'object') {
803
+ millisec = false;
804
+ }
805
+ if (typeof(microsec) === 'object') {
806
+ microsec = false;
807
+ }
808
+ if (typeof(timezone) === 'object') {
809
+ timezone = false;
448
810
  }
449
- html += '</dd>';
450
811
 
451
- // Timezone
452
- html += '<dt class="ui_tpicker_timezone_label" id="ui_tpicker_timezone_label_' + dp_id + '"' +
453
- ((o.showTimezone) ? '' : noDisplay) + '>' + o.timezoneText + '</dt>';
454
- html += '<dd class="ui_tpicker_timezone" id="ui_tpicker_timezone_' + dp_id + '"' +
455
- ((o.showTimezone) ? '' : noDisplay) + '></dd>';
812
+ if (hour !== false) {
813
+ hour = parseInt(hour, 10);
814
+ }
815
+ if (minute !== false) {
816
+ minute = parseInt(minute, 10);
817
+ }
818
+ if (second !== false) {
819
+ second = parseInt(second, 10);
820
+ }
821
+ if (millisec !== false) {
822
+ millisec = parseInt(millisec, 10);
823
+ }
824
+ if (microsec !== false) {
825
+ microsec = parseInt(microsec, 10);
826
+ }
827
+ if (timezone !== false) {
828
+ timezone = timezone.toString();
829
+ }
456
830
 
457
- html += '</dl></div>';
458
- $tp = $(html);
831
+ var ampm = o[hour < 12 ? 'amNames' : 'pmNames'][0];
459
832
 
460
- // if we only want time picker...
461
- if (o.timeOnly === true) {
462
- $tp.prepend(
463
- '<div class="ui-widget-header ui-helper-clearfix ui-corner-all">' +
464
- '<div class="ui-datepicker-title">' + o.timeOnlyTitle + '</div>' +
465
- '</div>');
466
- $dp.find('.ui-datepicker-header, .ui-datepicker-calendar').hide();
467
- }
468
-
469
- this.hour_slider = $tp.find('#ui_tpicker_hour_'+ dp_id).slider({
470
- orientation: "horizontal",
471
- value: this.hour,
472
- min: o.hourMin,
473
- max: hourMax,
474
- step: o.stepHour,
475
- slide: function(event, ui) {
476
- tp_inst.hour_slider.slider( "option", "value", ui.value);
477
- tp_inst._onTimeChange();
478
- }
479
- });
833
+ // If the update was done in the input field, the input field should not be updated.
834
+ // If the update was done using the sliders, update the input field.
835
+ var hasChanged = (
836
+ hour !== parseInt(this.hour,10) || // sliders should all be numeric
837
+ minute !== parseInt(this.minute,10) ||
838
+ second !== parseInt(this.second,10) ||
839
+ millisec !== parseInt(this.millisec,10) ||
840
+ microsec !== parseInt(this.microsec,10) ||
841
+ (this.ampm.length > 0 && (hour < 12) !== ($.inArray(this.ampm.toUpperCase(), this.amNames) !== -1)) ||
842
+ (this.timezone !== null && timezone !== this.timezone.toString()) // could be numeric or "EST" format, so use toString()
843
+ );
480
844
 
845
+ if (hasChanged) {
481
846
 
482
- // Updated by Peter Medeiros:
483
- // - Pass in Event and UI instance into slide function
484
- this.minute_slider = $tp.find('#ui_tpicker_minute_'+ dp_id).slider({
485
- orientation: "horizontal",
486
- value: this.minute,
487
- min: o.minuteMin,
488
- max: minMax,
489
- step: o.stepMinute,
490
- slide: function(event, ui) {
491
- tp_inst.minute_slider.slider( "option", "value", ui.value);
492
- tp_inst._onTimeChange();
847
+ if (hour !== false) {
848
+ this.hour = hour;
849
+ }
850
+ if (minute !== false) {
851
+ this.minute = minute;
852
+ }
853
+ if (second !== false) {
854
+ this.second = second;
855
+ }
856
+ if (millisec !== false) {
857
+ this.millisec = millisec;
858
+ }
859
+ if (microsec !== false) {
860
+ this.microsec = microsec;
861
+ }
862
+ if (timezone !== false) {
863
+ this.timezone = timezone;
493
864
  }
494
- });
495
865
 
496
- this.second_slider = $tp.find('#ui_tpicker_second_'+ dp_id).slider({
497
- orientation: "horizontal",
498
- value: this.second,
499
- min: o.secondMin,
500
- max: secMax,
501
- step: o.stepSecond,
502
- slide: function(event, ui) {
503
- tp_inst.second_slider.slider( "option", "value", ui.value);
504
- tp_inst._onTimeChange();
866
+ if (!this.inst) {
867
+ this.inst = $.datepicker._getInst(this.$input[0]);
505
868
  }
506
- });
507
869
 
508
- this.millisec_slider = $tp.find('#ui_tpicker_millisec_'+ dp_id).slider({
509
- orientation: "horizontal",
510
- value: this.millisec,
511
- min: o.millisecMin,
512
- max: millisecMax,
513
- step: o.stepMillisec,
514
- slide: function(event, ui) {
515
- tp_inst.millisec_slider.slider( "option", "value", ui.value);
516
- tp_inst._onTimeChange();
870
+ this._limitMinMaxDateTime(this.inst, true);
871
+ }
872
+ if (this.support.ampm) {
873
+ this.ampm = ampm;
874
+ }
875
+
876
+ // Updates the time within the timepicker
877
+ this.formattedTime = $.datepicker.formatTime(o.timeFormat, this, o);
878
+ if (this.$timeObj) {
879
+ if (pickerTimeFormat === o.timeFormat) {
880
+ this.$timeObj.val(this.formattedTime + pickerTimeSuffix);
517
881
  }
518
- });
882
+ else {
883
+ this.$timeObj.val($.datepicker.formatTime(pickerTimeFormat, this, o) + pickerTimeSuffix);
884
+ }
885
+ if (this.$timeObj[0].setSelectionRange) {
886
+ var sPos = this.$timeObj[0].selectionStart;
887
+ var ePos = this.$timeObj[0].selectionEnd;
888
+ this.$timeObj[0].setSelectionRange(sPos, ePos);
889
+ }
890
+ }
519
891
 
520
- this.timezone_select = $tp.find('#ui_tpicker_timezone_'+ dp_id).append('<select></select>').find("select");
521
- $.fn.append.apply(this.timezone_select,
522
- $.map(o.timezoneList, function(val, idx) {
523
- return $("<option />")
524
- .val(typeof val == "object" ? val.value : val)
525
- .text(typeof val == "object" ? val.label : val);
526
- })
527
- );
528
- this.timezone_select.val((typeof this.timezone != "undefined" && this.timezone != null && this.timezone != "") ? this.timezone : o.timezone);
529
- this.timezone_select.change(function() {
530
- tp_inst._onTimeChange();
531
- });
892
+ this.timeDefined = true;
893
+ if (hasChanged) {
894
+ this._updateDateTime();
895
+ //this.$input.focus(); // may automatically open the picker on setDate
896
+ }
897
+ },
898
+
899
+ /*
900
+ * call custom onSelect.
901
+ * bind to sliders slidestop, and grid click.
902
+ */
903
+ _onSelectHandler: function () {
904
+ var onSelect = this._defaults.onSelect || this.inst.settings.onSelect;
905
+ var inputEl = this.$input ? this.$input[0] : null;
906
+ if (onSelect && inputEl) {
907
+ onSelect.apply(inputEl, [this.formattedDateTime, this]);
908
+ }
909
+ },
910
+
911
+ /*
912
+ * update our input with the new date time..
913
+ */
914
+ _updateDateTime: function (dp_inst) {
915
+ dp_inst = this.inst || dp_inst;
916
+ var dtTmp = (dp_inst.currentYear > 0?
917
+ new Date(dp_inst.currentYear, dp_inst.currentMonth, dp_inst.currentDay) :
918
+ new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay)),
919
+ dt = $.datepicker._daylightSavingAdjust(dtTmp),
920
+ //dt = $.datepicker._daylightSavingAdjust(new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay)),
921
+ //dt = $.datepicker._daylightSavingAdjust(new Date(dp_inst.currentYear, dp_inst.currentMonth, dp_inst.currentDay)),
922
+ dateFmt = $.datepicker._get(dp_inst, 'dateFormat'),
923
+ formatCfg = $.datepicker._getFormatConfig(dp_inst),
924
+ timeAvailable = dt !== null && this.timeDefined;
925
+ this.formattedDate = $.datepicker.formatDate(dateFmt, (dt === null ? new Date() : dt), formatCfg);
926
+ var formattedDateTime = this.formattedDate;
927
+
928
+ // if a slider was changed but datepicker doesn't have a value yet, set it
929
+ if (dp_inst.lastVal === "") {
930
+ dp_inst.currentYear = dp_inst.selectedYear;
931
+ dp_inst.currentMonth = dp_inst.selectedMonth;
932
+ dp_inst.currentDay = dp_inst.selectedDay;
933
+ }
934
+
935
+ /*
936
+ * remove following lines to force every changes in date picker to change the input value
937
+ * Bug descriptions: when an input field has a default value, and click on the field to pop up the date picker.
938
+ * If the user manually empty the value in the input field, the date picker will never change selected value.
939
+ */
940
+ //if (dp_inst.lastVal !== undefined && (dp_inst.lastVal.length > 0 && this.$input.val().length === 0)) {
941
+ // return;
942
+ //}
943
+
944
+ if (this._defaults.timeOnly === true && this._defaults.timeOnlyShowDate === false) {
945
+ formattedDateTime = this.formattedTime;
946
+ } else if ((this._defaults.timeOnly !== true && (this._defaults.alwaysSetTime || timeAvailable)) || (this._defaults.timeOnly === true && this._defaults.timeOnlyShowDate === true)) {
947
+ formattedDateTime += this._defaults.separator + this.formattedTime + this._defaults.timeSuffix;
948
+ }
532
949
 
533
- // Add grid functionality
534
- if (o.showHour && o.hourGrid > 0) {
535
- size = 100 * hourGridSize * o.hourGrid / (hourMax - o.hourMin);
536
-
537
- $tp.find(".ui_tpicker_hour table").css({
538
- width: size + "%",
539
- marginLeft: (size / (-2 * hourGridSize)) + "%",
540
- borderCollapse: 'collapse'
541
- }).find("td").each( function(index) {
542
- $(this).click(function() {
543
- var h = $(this).html();
544
- if(o.ampm) {
545
- var ap = h.substring(2).toLowerCase(),
546
- aph = parseInt(h.substring(0,2), 10);
547
- if (ap == 'a') {
548
- if (aph == 12) h = 0;
549
- else h = aph;
550
- } else if (aph == 12) h = 12;
551
- else h = aph + 12;
950
+ this.formattedDateTime = formattedDateTime;
951
+
952
+ if (!this._defaults.showTimepicker) {
953
+ this.$input.val(this.formattedDate);
954
+ } else if (this.$altInput && this._defaults.timeOnly === false && this._defaults.altFieldTimeOnly === true) {
955
+ this.$altInput.val(this.formattedTime);
956
+ this.$input.val(this.formattedDate);
957
+ } else if (this.$altInput) {
958
+ this.$input.val(formattedDateTime);
959
+ var altFormattedDateTime = '',
960
+ altSeparator = this._defaults.altSeparator !== null ? this._defaults.altSeparator : this._defaults.separator,
961
+ altTimeSuffix = this._defaults.altTimeSuffix !== null ? this._defaults.altTimeSuffix : this._defaults.timeSuffix;
962
+
963
+ if (!this._defaults.timeOnly) {
964
+ if (this._defaults.altFormat) {
965
+ altFormattedDateTime = $.datepicker.formatDate(this._defaults.altFormat, (dt === null ? new Date() : dt), formatCfg);
966
+ }
967
+ else {
968
+ altFormattedDateTime = this.formattedDate;
969
+ }
970
+
971
+ if (altFormattedDateTime) {
972
+ altFormattedDateTime += altSeparator;
973
+ }
974
+ }
975
+
976
+ if (this._defaults.altTimeFormat !== null) {
977
+ altFormattedDateTime += $.datepicker.formatTime(this._defaults.altTimeFormat, this, this._defaults) + altTimeSuffix;
978
+ }
979
+ else {
980
+ altFormattedDateTime += this.formattedTime + altTimeSuffix;
981
+ }
982
+ this.$altInput.val(altFormattedDateTime);
983
+ } else {
984
+ this.$input.val(formattedDateTime);
985
+ }
986
+
987
+ this.$input.trigger("change");
988
+ },
989
+
990
+ _onFocus: function () {
991
+ if (!this.$input.val() && this._defaults.defaultValue) {
992
+ this.$input.val(this._defaults.defaultValue);
993
+ var inst = $.datepicker._getInst(this.$input.get(0)),
994
+ tp_inst = $.datepicker._get(inst, 'timepicker');
995
+ if (tp_inst) {
996
+ if (tp_inst._defaults.timeOnly && (inst.input.val() !== inst.lastVal)) {
997
+ try {
998
+ $.datepicker._updateDatepicker(inst);
999
+ } catch (err) {
1000
+ $.timepicker.log(err);
552
1001
  }
553
- tp_inst.hour_slider.slider("option", "value", h);
554
- tp_inst._onTimeChange();
555
- tp_inst._onSelectHandler();
556
- }).css({
557
- cursor: 'pointer',
558
- width: (100 / hourGridSize) + '%',
559
- textAlign: 'center',
560
- overflow: 'hidden'
561
- });
562
- });
1002
+ }
1003
+ }
563
1004
  }
1005
+ },
1006
+
1007
+ /*
1008
+ * Small abstraction to control types
1009
+ * We can add more, just be sure to follow the pattern: create, options, value
1010
+ */
1011
+ _controls: {
1012
+ // slider methods
1013
+ slider: {
1014
+ create: function (tp_inst, obj, unit, val, min, max, step) {
1015
+ var rtl = tp_inst._defaults.isRTL; // if rtl go -60->0 instead of 0->60
1016
+ return obj.prop('slide', null).slider({
1017
+ orientation: "horizontal",
1018
+ value: rtl ? val * -1 : val,
1019
+ min: rtl ? max * -1 : min,
1020
+ max: rtl ? min * -1 : max,
1021
+ step: step,
1022
+ slide: function (event, ui) {
1023
+ tp_inst.control.value(tp_inst, $(this), unit, rtl ? ui.value * -1 : ui.value);
1024
+ tp_inst._onTimeChange();
1025
+ },
1026
+ stop: function (event, ui) {
1027
+ tp_inst._onSelectHandler();
1028
+ }
1029
+ });
1030
+ },
1031
+ options: function (tp_inst, obj, unit, opts, val) {
1032
+ if (tp_inst._defaults.isRTL) {
1033
+ if (typeof(opts) === 'string') {
1034
+ if (opts === 'min' || opts === 'max') {
1035
+ if (val !== undefined) {
1036
+ return obj.slider(opts, val * -1);
1037
+ }
1038
+ return Math.abs(obj.slider(opts));
1039
+ }
1040
+ return obj.slider(opts);
1041
+ }
1042
+ var min = opts.min,
1043
+ max = opts.max;
1044
+ opts.min = opts.max = null;
1045
+ if (min !== undefined) {
1046
+ opts.max = min * -1;
1047
+ }
1048
+ if (max !== undefined) {
1049
+ opts.min = max * -1;
1050
+ }
1051
+ return obj.slider(opts);
1052
+ }
1053
+ if (typeof(opts) === 'string' && val !== undefined) {
1054
+ return obj.slider(opts, val);
1055
+ }
1056
+ return obj.slider(opts);
1057
+ },
1058
+ value: function (tp_inst, obj, unit, val) {
1059
+ if (tp_inst._defaults.isRTL) {
1060
+ if (val !== undefined) {
1061
+ return obj.slider('value', val * -1);
1062
+ }
1063
+ return Math.abs(obj.slider('value'));
1064
+ }
1065
+ if (val !== undefined) {
1066
+ return obj.slider('value', val);
1067
+ }
1068
+ return obj.slider('value');
1069
+ }
1070
+ },
1071
+ // select methods
1072
+ select: {
1073
+ create: function (tp_inst, obj, unit, val, min, max, step) {
1074
+ var sel = '<select class="ui-timepicker-select ui-state-default ui-corner-all" data-unit="' + unit + '" data-min="' + min + '" data-max="' + max + '" data-step="' + step + '">',
1075
+ format = tp_inst._defaults.pickerTimeFormat || tp_inst._defaults.timeFormat;
1076
+
1077
+ for (var i = min; i <= max; i += step) {
1078
+ sel += '<option value="' + i + '"' + (i === val ? ' selected' : '') + '>';
1079
+ if (unit === 'hour') {
1080
+ sel += $.datepicker.formatTime($.trim(format.replace(/[^ht ]/ig, '')), {hour: i}, tp_inst._defaults);
1081
+ }
1082
+ else if (unit === 'millisec' || unit === 'microsec' || i >= 10) { sel += i; }
1083
+ else {sel += '0' + i.toString(); }
1084
+ sel += '</option>';
1085
+ }
1086
+ sel += '</select>';
1087
+
1088
+ obj.children('select').remove();
564
1089
 
565
- if (o.showMinute && o.minuteGrid > 0) {
566
- size = 100 * minuteGridSize * o.minuteGrid / (minMax - o.minuteMin);
567
- $tp.find(".ui_tpicker_minute table").css({
568
- width: size + "%",
569
- marginLeft: (size / (-2 * minuteGridSize)) + "%",
570
- borderCollapse: 'collapse'
571
- }).find("td").each(function(index) {
572
- $(this).click(function() {
573
- tp_inst.minute_slider.slider("option", "value", $(this).html());
1090
+ $(sel).appendTo(obj).change(function (e) {
574
1091
  tp_inst._onTimeChange();
575
1092
  tp_inst._onSelectHandler();
576
- }).css({
577
- cursor: 'pointer',
578
- width: (100 / minuteGridSize) + '%',
579
- textAlign: 'center',
580
- overflow: 'hidden'
1093
+ tp_inst._afterInject();
581
1094
  });
582
- });
1095
+
1096
+ return obj;
1097
+ },
1098
+ options: function (tp_inst, obj, unit, opts, val) {
1099
+ var o = {},
1100
+ $t = obj.children('select');
1101
+ if (typeof(opts) === 'string') {
1102
+ if (val === undefined) {
1103
+ return $t.data(opts);
1104
+ }
1105
+ o[opts] = val;
1106
+ }
1107
+ else { o = opts; }
1108
+ return tp_inst.control.create(tp_inst, obj, $t.data('unit'), $t.val(), o.min>=0 ? o.min : $t.data('min'), o.max || $t.data('max'), o.step || $t.data('step'));
1109
+ },
1110
+ value: function (tp_inst, obj, unit, val) {
1111
+ var $t = obj.children('select');
1112
+ if (val !== undefined) {
1113
+ return $t.val(val);
1114
+ }
1115
+ return $t.val();
1116
+ }
583
1117
  }
1118
+ } // end _controls
584
1119
 
585
- if (o.showSecond && o.secondGrid > 0) {
586
- $tp.find(".ui_tpicker_second table").css({
587
- width: size + "%",
588
- marginLeft: (size / (-2 * secondGridSize)) + "%",
589
- borderCollapse: 'collapse'
590
- }).find("td").each(function(index) {
591
- $(this).click(function() {
592
- tp_inst.second_slider.slider("option", "value", $(this).html());
593
- tp_inst._onTimeChange();
594
- tp_inst._onSelectHandler();
595
- }).css({
596
- cursor: 'pointer',
597
- width: (100 / secondGridSize) + '%',
598
- textAlign: 'center',
599
- overflow: 'hidden'
600
- });
1120
+ });
1121
+
1122
+ $.fn.extend({
1123
+ /*
1124
+ * shorthand just to use timepicker.
1125
+ */
1126
+ timepicker: function (o) {
1127
+ o = o || {};
1128
+ var tmp_args = Array.prototype.slice.call(arguments);
1129
+
1130
+ if (typeof o === 'object') {
1131
+ tmp_args[0] = $.extend(o, {
1132
+ timeOnly: true
601
1133
  });
602
1134
  }
603
1135
 
604
- if (o.showMillisec && o.millisecGrid > 0) {
605
- $tp.find(".ui_tpicker_millisec table").css({
606
- width: size + "%",
607
- marginLeft: (size / (-2 * millisecGridSize)) + "%",
608
- borderCollapse: 'collapse'
609
- }).find("td").each(function(index) {
610
- $(this).click(function() {
611
- tp_inst.millisec_slider.slider("option", "value", $(this).html());
612
- tp_inst._onTimeChange();
613
- tp_inst._onSelectHandler();
614
- }).css({
615
- cursor: 'pointer',
616
- width: (100 / millisecGridSize) + '%',
617
- textAlign: 'center',
618
- overflow: 'hidden'
1136
+ return $(this).each(function () {
1137
+ $.fn.datetimepicker.apply($(this), tmp_args);
1138
+ });
1139
+ },
1140
+
1141
+ /*
1142
+ * extend timepicker to datepicker
1143
+ */
1144
+ datetimepicker: function (o) {
1145
+ o = o || {};
1146
+ var tmp_args = arguments;
1147
+
1148
+ if (typeof(o) === 'string') {
1149
+ if (o === 'getDate' || (o === 'option' && tmp_args.length === 2 && typeof (tmp_args[1]) === 'string')) {
1150
+ return $.fn.datepicker.apply($(this[0]), tmp_args);
1151
+ } else {
1152
+ return this.each(function () {
1153
+ var $t = $(this);
1154
+ $t.datepicker.apply($t, tmp_args);
619
1155
  });
1156
+ }
1157
+ } else {
1158
+ return this.each(function () {
1159
+ var $t = $(this);
1160
+ $t.datepicker($.timepicker._newInst($t, o)._defaults);
620
1161
  });
621
1162
  }
1163
+ }
1164
+ });
622
1165
 
623
- var $buttonPanel = $dp.find('.ui-datepicker-buttonpane');
624
- if ($buttonPanel.length) $buttonPanel.before($tp);
625
- else $dp.append($tp);
1166
+ /*
1167
+ * Public Utility to parse date and time
1168
+ */
1169
+ $.datepicker.parseDateTime = function (dateFormat, timeFormat, dateTimeString, dateSettings, timeSettings) {
1170
+ var parseRes = parseDateTimeInternal(dateFormat, timeFormat, dateTimeString, dateSettings, timeSettings);
1171
+ if (parseRes.timeObj) {
1172
+ var t = parseRes.timeObj;
1173
+ parseRes.date.setHours(t.hour, t.minute, t.second, t.millisec);
1174
+ parseRes.date.setMicroseconds(t.microsec);
1175
+ }
626
1176
 
627
- this.$timeObj = $tp.find('#ui_tpicker_time_'+ dp_id);
1177
+ return parseRes.date;
1178
+ };
628
1179
 
629
- if (this.inst !== null) {
630
- var timeDefined = this.timeDefined;
631
- this._onTimeChange();
632
- this.timeDefined = timeDefined;
633
- }
1180
+ /*
1181
+ * Public utility to parse time
1182
+ */
1183
+ $.datepicker.parseTime = function (timeFormat, timeString, options) {
1184
+ var o = extendRemove(extendRemove({}, $.timepicker._defaults), options || {}),
1185
+ iso8601 = (timeFormat.replace(/\'.*?\'/g, '').indexOf('Z') !== -1);
1186
+
1187
+ // Strict parse requires the timeString to match the timeFormat exactly
1188
+ var strictParse = function (f, s, o) {
1189
+
1190
+ // pattern for standard and localized AM/PM markers
1191
+ var getPatternAmpm = function (amNames, pmNames) {
1192
+ var markers = [];
1193
+ if (amNames) {
1194
+ $.merge(markers, amNames);
1195
+ }
1196
+ if (pmNames) {
1197
+ $.merge(markers, pmNames);
1198
+ }
1199
+ markers = $.map(markers, function (val) {
1200
+ return val.replace(/[.*+?|()\[\]{}\\]/g, '\\$&');
1201
+ });
1202
+ return '(' + markers.join('|') + ')?';
1203
+ };
634
1204
 
635
- //Emulate datepicker onSelect behavior. Call on slidestop.
636
- var onSelectDelegate = function() {
637
- tp_inst._onSelectHandler();
1205
+ // figure out position of time elements.. cause js cant do named captures
1206
+ var getFormatPositions = function (timeFormat) {
1207
+ var finds = timeFormat.toLowerCase().match(/(h{1,2}|m{1,2}|s{1,2}|l{1}|c{1}|t{1,2}|z|'.*?')/g),
1208
+ orders = {
1209
+ h: -1,
1210
+ m: -1,
1211
+ s: -1,
1212
+ l: -1,
1213
+ c: -1,
1214
+ t: -1,
1215
+ z: -1
1216
+ };
1217
+
1218
+ if (finds) {
1219
+ for (var i = 0; i < finds.length; i++) {
1220
+ if (orders[finds[i].toString().charAt(0)] === -1) {
1221
+ orders[finds[i].toString().charAt(0)] = i + 1;
1222
+ }
1223
+ }
1224
+ }
1225
+ return orders;
638
1226
  };
639
- this.hour_slider.bind('slidestop',onSelectDelegate);
640
- this.minute_slider.bind('slidestop',onSelectDelegate);
641
- this.second_slider.bind('slidestop',onSelectDelegate);
642
- this.millisec_slider.bind('slidestop',onSelectDelegate);
643
-
644
- // slideAccess integration: http://trentrichardson.com/2011/11/11/jquery-ui-sliders-and-touch-accessibility/
645
- if (this._defaults.addSliderAccess){
646
- var sliderAccessArgs = this._defaults.sliderAccessArgs;
647
- setTimeout(function(){ // fix for inline mode
648
- if($tp.find('.ui-slider-access').length == 0){
649
- $tp.find('.ui-slider:visible').sliderAccess(sliderAccessArgs);
650
-
651
- // fix any grids since sliders are shorter
652
- var sliderAccessWidth = $tp.find('.ui-slider-access:eq(0)').outerWidth(true);
653
- if(sliderAccessWidth){
654
- $tp.find('table:visible').each(function(){
655
- var $g = $(this),
656
- oldWidth = $g.outerWidth(),
657
- oldMarginLeft = $g.css('marginLeft').toString().replace('%',''),
658
- newWidth = oldWidth - sliderAccessWidth,
659
- newMarginLeft = ((oldMarginLeft * newWidth)/oldWidth) + '%';
660
-
661
- $g.css({ width: newWidth, marginLeft: newMarginLeft });
662
- });
1227
+
1228
+ var regstr = '^' + f.toString()
1229
+ .replace(/([hH]{1,2}|mm?|ss?|[tT]{1,2}|[zZ]|[lc]|'.*?')/g, function (match) {
1230
+ var ml = match.length;
1231
+ switch (match.charAt(0).toLowerCase()) {
1232
+ case 'h':
1233
+ return ml === 1 ? '(\\d?\\d)' : '(\\d{' + ml + '})';
1234
+ case 'm':
1235
+ return ml === 1 ? '(\\d?\\d)' : '(\\d{' + ml + '})';
1236
+ case 's':
1237
+ return ml === 1 ? '(\\d?\\d)' : '(\\d{' + ml + '})';
1238
+ case 'l':
1239
+ return '(\\d?\\d?\\d)';
1240
+ case 'c':
1241
+ return '(\\d?\\d?\\d)';
1242
+ case 'z':
1243
+ return '(z|[-+]\\d\\d:?\\d\\d|\\S+)?';
1244
+ case 't':
1245
+ return getPatternAmpm(o.amNames, o.pmNames);
1246
+ default: // literal escaped in quotes
1247
+ return '(' + match.replace(/\'/g, "").replace(/(\.|\$|\^|\\|\/|\(|\)|\[|\]|\?|\+|\*)/g, function (m) { return "\\" + m; }) + ')?';
1248
+ }
1249
+ })
1250
+ .replace(/\s/g, '\\s?') +
1251
+ o.timeSuffix + '$',
1252
+ order = getFormatPositions(f),
1253
+ ampm = '',
1254
+ treg;
1255
+
1256
+ treg = s.match(new RegExp(regstr, 'i'));
1257
+
1258
+ var resTime = {
1259
+ hour: 0,
1260
+ minute: 0,
1261
+ second: 0,
1262
+ millisec: 0,
1263
+ microsec: 0
1264
+ };
1265
+
1266
+ if (treg) {
1267
+ if (order.t !== -1) {
1268
+ if (treg[order.t] === undefined || treg[order.t].length === 0) {
1269
+ ampm = '';
1270
+ resTime.ampm = '';
1271
+ } else {
1272
+ ampm = $.inArray(treg[order.t].toUpperCase(), $.map(o.amNames, function (x,i) { return x.toUpperCase(); })) !== -1 ? 'AM' : 'PM';
1273
+ resTime.ampm = o[ampm === 'AM' ? 'amNames' : 'pmNames'][0];
1274
+ }
1275
+ }
1276
+
1277
+ if (order.h !== -1) {
1278
+ if (ampm === 'AM' && treg[order.h] === '12') {
1279
+ resTime.hour = 0; // 12am = 0 hour
1280
+ } else {
1281
+ if (ampm === 'PM' && treg[order.h] !== '12') {
1282
+ resTime.hour = parseInt(treg[order.h], 10) + 12; // 12pm = 12 hour, any other pm = hour + 12
1283
+ } else {
1284
+ resTime.hour = Number(treg[order.h]);
663
1285
  }
664
1286
  }
665
- },0);
1287
+ }
1288
+
1289
+ if (order.m !== -1) {
1290
+ resTime.minute = Number(treg[order.m]);
1291
+ }
1292
+ if (order.s !== -1) {
1293
+ resTime.second = Number(treg[order.s]);
1294
+ }
1295
+ if (order.l !== -1) {
1296
+ resTime.millisec = Number(treg[order.l]);
1297
+ }
1298
+ if (order.c !== -1) {
1299
+ resTime.microsec = Number(treg[order.c]);
1300
+ }
1301
+ if (order.z !== -1 && treg[order.z] !== undefined) {
1302
+ resTime.timezone = $.timepicker.timezoneOffsetNumber(treg[order.z]);
1303
+ }
1304
+
1305
+
1306
+ return resTime;
666
1307
  }
667
- // end slideAccess integration
1308
+ return false;
1309
+ };// end strictParse
668
1310
 
669
- }
670
- },
671
-
672
- //########################################################################
673
- // This function tries to limit the ability to go outside the
674
- // min/max date range
675
- //########################################################################
676
- _limitMinMaxDateTime: function(dp_inst, adjustSliders){
677
- var o = this._defaults,
678
- dp_date = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay);
679
-
680
- if(!this._defaults.showTimepicker) return; // No time so nothing to check here
681
-
682
- if($.datepicker._get(dp_inst, 'minDateTime') !== null && $.datepicker._get(dp_inst, 'minDateTime') !== undefined && dp_date){
683
- var minDateTime = $.datepicker._get(dp_inst, 'minDateTime'),
684
- minDateTimeDate = new Date(minDateTime.getFullYear(), minDateTime.getMonth(), minDateTime.getDate(), 0, 0, 0, 0);
685
-
686
- if(this.hourMinOriginal === null || this.minuteMinOriginal === null || this.secondMinOriginal === null || this.millisecMinOriginal === null){
687
- this.hourMinOriginal = o.hourMin;
688
- this.minuteMinOriginal = o.minuteMin;
689
- this.secondMinOriginal = o.secondMin;
690
- this.millisecMinOriginal = o.millisecMin;
691
- }
692
-
693
- if(dp_inst.settings.timeOnly || minDateTimeDate.getTime() == dp_date.getTime()) {
694
- this._defaults.hourMin = minDateTime.getHours();
695
- if (this.hour <= this._defaults.hourMin) {
696
- this.hour = this._defaults.hourMin;
697
- this._defaults.minuteMin = minDateTime.getMinutes();
698
- if (this.minute <= this._defaults.minuteMin) {
699
- this.minute = this._defaults.minuteMin;
700
- this._defaults.secondMin = minDateTime.getSeconds();
701
- } else if (this.second <= this._defaults.secondMin){
702
- this.second = this._defaults.secondMin;
703
- this._defaults.millisecMin = minDateTime.getMilliseconds();
704
- } else {
705
- if(this.millisec < this._defaults.millisecMin)
706
- this.millisec = this._defaults.millisecMin;
707
- this._defaults.millisecMin = this.millisecMinOriginal;
1311
+ // First try JS Date, if that fails, use strictParse
1312
+ var looseParse = function (f, s, o) {
1313
+ try {
1314
+ var d = new Date('2012-01-01 ' + s);
1315
+ if (isNaN(d.getTime())) {
1316
+ d = new Date('2012-01-01T' + s);
1317
+ if (isNaN(d.getTime())) {
1318
+ d = new Date('01/01/2012 ' + s);
1319
+ if (isNaN(d.getTime())) {
1320
+ throw "Unable to parse time with native Date: " + s;
1321
+ }
708
1322
  }
709
- } else {
710
- this._defaults.minuteMin = this.minuteMinOriginal;
711
- this._defaults.secondMin = this.secondMinOriginal;
712
- this._defaults.millisecMin = this.millisecMinOriginal;
713
1323
  }
714
- }else{
715
- this._defaults.hourMin = this.hourMinOriginal;
716
- this._defaults.minuteMin = this.minuteMinOriginal;
717
- this._defaults.secondMin = this.secondMinOriginal;
718
- this._defaults.millisecMin = this.millisecMinOriginal;
1324
+
1325
+ return {
1326
+ hour: d.getHours(),
1327
+ minute: d.getMinutes(),
1328
+ second: d.getSeconds(),
1329
+ millisec: d.getMilliseconds(),
1330
+ microsec: d.getMicroseconds(),
1331
+ timezone: d.getTimezoneOffset() * -1
1332
+ };
1333
+ }
1334
+ catch (err) {
1335
+ try {
1336
+ return strictParse(f, s, o);
1337
+ }
1338
+ catch (err2) {
1339
+ $.timepicker.log("Unable to parse \ntimeString: " + s + "\ntimeFormat: " + f);
1340
+ }
719
1341
  }
1342
+ return false;
1343
+ }; // end looseParse
1344
+
1345
+ if (typeof o.parse === "function") {
1346
+ return o.parse(timeFormat, timeString, o);
1347
+ }
1348
+ if (o.parse === 'loose') {
1349
+ return looseParse(timeFormat, timeString, o);
720
1350
  }
1351
+ return strictParse(timeFormat, timeString, o);
1352
+ };
721
1353
 
722
- if($.datepicker._get(dp_inst, 'maxDateTime') !== null && $.datepicker._get(dp_inst, 'maxDateTime') !== undefined && dp_date){
723
- var maxDateTime = $.datepicker._get(dp_inst, 'maxDateTime'),
724
- maxDateTimeDate = new Date(maxDateTime.getFullYear(), maxDateTime.getMonth(), maxDateTime.getDate(), 0, 0, 0, 0);
725
-
726
- if(this.hourMaxOriginal === null || this.minuteMaxOriginal === null || this.secondMaxOriginal === null){
727
- this.hourMaxOriginal = o.hourMax;
728
- this.minuteMaxOriginal = o.minuteMax;
729
- this.secondMaxOriginal = o.secondMax;
730
- this.millisecMaxOriginal = o.millisecMax;
731
- }
732
-
733
- if(dp_inst.settings.timeOnly || maxDateTimeDate.getTime() == dp_date.getTime()){
734
- this._defaults.hourMax = maxDateTime.getHours();
735
- if (this.hour >= this._defaults.hourMax) {
736
- this.hour = this._defaults.hourMax;
737
- this._defaults.minuteMax = maxDateTime.getMinutes();
738
- if (this.minute >= this._defaults.minuteMax) {
739
- this.minute = this._defaults.minuteMax;
740
- this._defaults.secondMax = maxDateTime.getSeconds();
741
- } else if (this.second >= this._defaults.secondMax) {
742
- this.second = this._defaults.secondMax;
743
- this._defaults.millisecMax = maxDateTime.getMilliseconds();
744
- } else {
745
- if(this.millisec > this._defaults.millisecMax) this.millisec = this._defaults.millisecMax;
746
- this._defaults.millisecMax = this.millisecMaxOriginal;
1354
+ /**
1355
+ * Public utility to format the time
1356
+ * @param {string} format format of the time
1357
+ * @param {Object} time Object not a Date for timezones
1358
+ * @param {Object} [options] essentially the regional[].. amNames, pmNames, ampm
1359
+ * @returns {string} the formatted time
1360
+ */
1361
+ $.datepicker.formatTime = function (format, time, options) {
1362
+ options = options || {};
1363
+ options = $.extend({}, $.timepicker._defaults, options);
1364
+ time = $.extend({
1365
+ hour: 0,
1366
+ minute: 0,
1367
+ second: 0,
1368
+ millisec: 0,
1369
+ microsec: 0,
1370
+ timezone: null
1371
+ }, time);
1372
+
1373
+ var tmptime = format,
1374
+ ampmName = options.amNames[0],
1375
+ hour = parseInt(time.hour, 10);
1376
+
1377
+ if (hour > 11) {
1378
+ ampmName = options.pmNames[0];
1379
+ }
1380
+
1381
+ tmptime = tmptime.replace(/(?:HH?|hh?|mm?|ss?|[tT]{1,2}|[zZ]|[lc]|'.*?')/g, function (match) {
1382
+ switch (match) {
1383
+ case 'HH':
1384
+ return ('0' + hour).slice(-2);
1385
+ case 'H':
1386
+ return hour;
1387
+ case 'hh':
1388
+ return ('0' + convert24to12(hour)).slice(-2);
1389
+ case 'h':
1390
+ return convert24to12(hour);
1391
+ case 'mm':
1392
+ return ('0' + time.minute).slice(-2);
1393
+ case 'm':
1394
+ return time.minute;
1395
+ case 'ss':
1396
+ return ('0' + time.second).slice(-2);
1397
+ case 's':
1398
+ return time.second;
1399
+ case 'l':
1400
+ return ('00' + time.millisec).slice(-3);
1401
+ case 'c':
1402
+ return ('00' + time.microsec).slice(-3);
1403
+ case 'z':
1404
+ return $.timepicker.timezoneOffsetString(time.timezone === null ? options.timezone : time.timezone, false);
1405
+ case 'Z':
1406
+ return $.timepicker.timezoneOffsetString(time.timezone === null ? options.timezone : time.timezone, true);
1407
+ case 'T':
1408
+ return ampmName.charAt(0).toUpperCase();
1409
+ case 'TT':
1410
+ return ampmName.toUpperCase();
1411
+ case 't':
1412
+ return ampmName.charAt(0).toLowerCase();
1413
+ case 'tt':
1414
+ return ampmName.toLowerCase();
1415
+ default:
1416
+ return match.replace(/'/g, "");
1417
+ }
1418
+ });
1419
+
1420
+ return tmptime;
1421
+ };
1422
+
1423
+ /*
1424
+ * the bad hack :/ override datepicker so it doesn't close on select
1425
+ // inspired: http://stackoverflow.com/questions/1252512/jquery-datepicker-prevent-closing-picker-when-clicking-a-date/1762378#1762378
1426
+ */
1427
+ $.datepicker._base_selectDate = $.datepicker._selectDate;
1428
+ $.datepicker._selectDate = function (id, dateStr) {
1429
+ var inst = this._getInst($(id)[0]),
1430
+ tp_inst = this._get(inst, 'timepicker'),
1431
+ was_inline;
1432
+
1433
+ if (tp_inst && inst.settings.showTimepicker) {
1434
+ tp_inst._limitMinMaxDateTime(inst, true);
1435
+ was_inline = inst.inline;
1436
+ inst.inline = inst.stay_open = true;
1437
+ //This way the onSelect handler called from calendarpicker get the full dateTime
1438
+ this._base_selectDate(id, dateStr);
1439
+ inst.inline = was_inline;
1440
+ inst.stay_open = false;
1441
+ this._notifyChange(inst);
1442
+ this._updateDatepicker(inst);
1443
+ } else {
1444
+ this._base_selectDate(id, dateStr);
1445
+ }
1446
+ };
1447
+
1448
+ /*
1449
+ * second bad hack :/ override datepicker so it triggers an event when changing the input field
1450
+ * and does not redraw the datepicker on every selectDate event
1451
+ */
1452
+ $.datepicker._base_updateDatepicker = $.datepicker._updateDatepicker;
1453
+ $.datepicker._updateDatepicker = function (inst) {
1454
+
1455
+ // don't popup the datepicker if there is another instance already opened
1456
+ var input = inst.input[0];
1457
+ if ($.datepicker._curInst && $.datepicker._curInst !== inst && $.datepicker._datepickerShowing && $.datepicker._lastInput !== input) {
1458
+ return;
1459
+ }
1460
+
1461
+ if (typeof(inst.stay_open) !== 'boolean' || inst.stay_open === false) {
1462
+
1463
+ this._base_updateDatepicker(inst);
1464
+
1465
+ // Reload the time control when changing something in the input text field.
1466
+ var tp_inst = this._get(inst, 'timepicker');
1467
+ if (tp_inst) {
1468
+ tp_inst._addTimePicker(inst);
1469
+ }
1470
+ }
1471
+ };
1472
+
1473
+ /*
1474
+ * third bad hack :/ override datepicker so it allows spaces and colon in the input field
1475
+ */
1476
+ $.datepicker._base_doKeyPress = $.datepicker._doKeyPress;
1477
+ $.datepicker._doKeyPress = function (event) {
1478
+ var inst = $.datepicker._getInst(event.target),
1479
+ tp_inst = $.datepicker._get(inst, 'timepicker');
1480
+
1481
+ if (tp_inst) {
1482
+ if ($.datepicker._get(inst, 'constrainInput')) {
1483
+ var ampm = tp_inst.support.ampm,
1484
+ tz = tp_inst._defaults.showTimezone !== null ? tp_inst._defaults.showTimezone : tp_inst.support.timezone,
1485
+ dateChars = $.datepicker._possibleChars($.datepicker._get(inst, 'dateFormat')),
1486
+ datetimeChars = tp_inst._defaults.timeFormat.toString()
1487
+ .replace(/[hms]/g, '')
1488
+ .replace(/TT/g, ampm ? 'APM' : '')
1489
+ .replace(/Tt/g, ampm ? 'AaPpMm' : '')
1490
+ .replace(/tT/g, ampm ? 'AaPpMm' : '')
1491
+ .replace(/T/g, ampm ? 'AP' : '')
1492
+ .replace(/tt/g, ampm ? 'apm' : '')
1493
+ .replace(/t/g, ampm ? 'ap' : '') +
1494
+ " " + tp_inst._defaults.separator +
1495
+ tp_inst._defaults.timeSuffix +
1496
+ (tz ? tp_inst._defaults.timezoneList.join('') : '') +
1497
+ (tp_inst._defaults.amNames.join('')) + (tp_inst._defaults.pmNames.join('')) +
1498
+ dateChars,
1499
+ chr = String.fromCharCode(event.charCode === undefined ? event.keyCode : event.charCode);
1500
+ return event.ctrlKey || (chr < ' ' || !dateChars || datetimeChars.indexOf(chr) > -1);
1501
+ }
1502
+ }
1503
+
1504
+ return $.datepicker._base_doKeyPress(event);
1505
+ };
1506
+
1507
+ /*
1508
+ * Fourth bad hack :/ override _updateAlternate function used in inline mode to init altField
1509
+ * Update any alternate field to synchronise with the main field.
1510
+ */
1511
+ $.datepicker._base_updateAlternate = $.datepicker._updateAlternate;
1512
+ $.datepicker._updateAlternate = function (inst) {
1513
+ var tp_inst = this._get(inst, 'timepicker');
1514
+ if (tp_inst) {
1515
+ var altField = tp_inst._defaults.altField;
1516
+ if (altField) { // update alternate field too
1517
+ var altFormat = tp_inst._defaults.altFormat || tp_inst._defaults.dateFormat,
1518
+ date = this._getDate(inst),
1519
+ formatCfg = $.datepicker._getFormatConfig(inst),
1520
+ altFormattedDateTime = '',
1521
+ altSeparator = tp_inst._defaults.altSeparator ? tp_inst._defaults.altSeparator : tp_inst._defaults.separator,
1522
+ altTimeSuffix = tp_inst._defaults.altTimeSuffix ? tp_inst._defaults.altTimeSuffix : tp_inst._defaults.timeSuffix,
1523
+ altTimeFormat = tp_inst._defaults.altTimeFormat !== null ? tp_inst._defaults.altTimeFormat : tp_inst._defaults.timeFormat;
1524
+
1525
+ altFormattedDateTime += $.datepicker.formatTime(altTimeFormat, tp_inst, tp_inst._defaults) + altTimeSuffix;
1526
+ if (!tp_inst._defaults.timeOnly && !tp_inst._defaults.altFieldTimeOnly && date !== null) {
1527
+ if (tp_inst._defaults.altFormat) {
1528
+ altFormattedDateTime = $.datepicker.formatDate(tp_inst._defaults.altFormat, date, formatCfg) + altSeparator + altFormattedDateTime;
1529
+ }
1530
+ else {
1531
+ altFormattedDateTime = tp_inst.formattedDate + altSeparator + altFormattedDateTime;
747
1532
  }
748
- } else {
749
- this._defaults.minuteMax = this.minuteMaxOriginal;
750
- this._defaults.secondMax = this.secondMaxOriginal;
751
- this._defaults.millisecMax = this.millisecMaxOriginal;
752
1533
  }
753
- }else{
754
- this._defaults.hourMax = this.hourMaxOriginal;
755
- this._defaults.minuteMax = this.minuteMaxOriginal;
756
- this._defaults.secondMax = this.secondMaxOriginal;
757
- this._defaults.millisecMax = this.millisecMaxOriginal;
1534
+ $(altField).val( inst.input.val() ? altFormattedDateTime : "");
758
1535
  }
759
1536
  }
1537
+ else {
1538
+ $.datepicker._base_updateAlternate(inst);
1539
+ }
1540
+ };
760
1541
 
761
- if(adjustSliders !== undefined && adjustSliders === true){
762
- var hourMax = parseInt((this._defaults.hourMax - ((this._defaults.hourMax - this._defaults.hourMin) % this._defaults.stepHour)) ,10),
763
- minMax = parseInt((this._defaults.minuteMax - ((this._defaults.minuteMax - this._defaults.minuteMin) % this._defaults.stepMinute)) ,10),
764
- secMax = parseInt((this._defaults.secondMax - ((this._defaults.secondMax - this._defaults.secondMin) % this._defaults.stepSecond)) ,10),
765
- millisecMax = parseInt((this._defaults.millisecMax - ((this._defaults.millisecMax - this._defaults.millisecMin) % this._defaults.stepMillisec)) ,10);
766
-
767
- if(this.hour_slider)
768
- this.hour_slider.slider("option", { min: this._defaults.hourMin, max: hourMax }).slider('value', this.hour);
769
- if(this.minute_slider)
770
- this.minute_slider.slider("option", { min: this._defaults.minuteMin, max: minMax }).slider('value', this.minute);
771
- if(this.second_slider)
772
- this.second_slider.slider("option", { min: this._defaults.secondMin, max: secMax }).slider('value', this.second);
773
- if(this.millisec_slider)
774
- this.millisec_slider.slider("option", { min: this._defaults.millisecMin, max: millisecMax }).slider('value', this.millisec);
1542
+ /*
1543
+ * Override key up event to sync manual input changes.
1544
+ */
1545
+ $.datepicker._base_doKeyUp = $.datepicker._doKeyUp;
1546
+ $.datepicker._doKeyUp = function (event) {
1547
+ var inst = $.datepicker._getInst(event.target),
1548
+ tp_inst = $.datepicker._get(inst, 'timepicker');
1549
+
1550
+ if (tp_inst) {
1551
+ if (tp_inst._defaults.timeOnly && (inst.input.val() !== inst.lastVal)) {
1552
+ try {
1553
+ $.datepicker._updateDatepicker(inst);
1554
+ } catch (err) {
1555
+ $.timepicker.log(err);
1556
+ }
1557
+ }
775
1558
  }
776
1559
 
777
- },
778
-
779
-
780
- //########################################################################
781
- // when a slider moves, set the internal time...
782
- // on time change is also called when the time is updated in the text field
783
- //########################################################################
784
- _onTimeChange: function() {
785
- var hour = (this.hour_slider) ? this.hour_slider.slider('value') : false,
786
- minute = (this.minute_slider) ? this.minute_slider.slider('value') : false,
787
- second = (this.second_slider) ? this.second_slider.slider('value') : false,
788
- millisec = (this.millisec_slider) ? this.millisec_slider.slider('value') : false,
789
- timezone = (this.timezone_select) ? this.timezone_select.val() : false,
790
- o = this._defaults;
791
-
792
- if (typeof(hour) == 'object') hour = false;
793
- if (typeof(minute) == 'object') minute = false;
794
- if (typeof(second) == 'object') second = false;
795
- if (typeof(millisec) == 'object') millisec = false;
796
- if (typeof(timezone) == 'object') timezone = false;
797
-
798
- if (hour !== false) hour = parseInt(hour,10);
799
- if (minute !== false) minute = parseInt(minute,10);
800
- if (second !== false) second = parseInt(second,10);
801
- if (millisec !== false) millisec = parseInt(millisec,10);
802
-
803
- var ampm = o[hour < 12 ? 'amNames' : 'pmNames'][0];
804
-
805
- // If the update was done in the input field, the input field should not be updated.
806
- // If the update was done using the sliders, update the input field.
807
- var hasChanged = (hour != this.hour || minute != this.minute
808
- || second != this.second || millisec != this.millisec
809
- || (this.ampm.length > 0
810
- && (hour < 12) != ($.inArray(this.ampm.toUpperCase(), this.amNames) !== -1))
811
- || timezone != this.timezone);
812
-
813
- if (hasChanged) {
814
-
815
- if (hour !== false)this.hour = hour;
816
- if (minute !== false) this.minute = minute;
817
- if (second !== false) this.second = second;
818
- if (millisec !== false) this.millisec = millisec;
819
- if (timezone !== false) this.timezone = timezone;
820
-
821
- if (!this.inst) this.inst = $.datepicker._getInst(this.$input[0]);
822
-
823
- this._limitMinMaxDateTime(this.inst, true);
1560
+ return $.datepicker._base_doKeyUp(event);
1561
+ };
1562
+
1563
+ /*
1564
+ * override "Today" button to also grab the time and set it to input field.
1565
+ */
1566
+ $.datepicker._base_gotoToday = $.datepicker._gotoToday;
1567
+ $.datepicker._gotoToday = function (id) {
1568
+ var inst = this._getInst($(id)[0]);
1569
+ this._base_gotoToday(id);
1570
+ var tp_inst = this._get(inst, 'timepicker');
1571
+ if (!tp_inst) {
1572
+ return;
824
1573
  }
825
- if (o.ampm) this.ampm = ampm;
826
-
827
- //this._formatTime();
828
- this.formattedTime = $.datepicker.formatTime(this._defaults.timeFormat, this, this._defaults);
829
- if (this.$timeObj) this.$timeObj.text(this.formattedTime + o.timeSuffix);
830
- this.timeDefined = true;
831
- if (hasChanged) this._updateDateTime();
832
- },
833
-
834
- //########################################################################
835
- // call custom onSelect.
836
- // bind to sliders slidestop, and grid click.
837
- //########################################################################
838
- _onSelectHandler: function() {
839
- var onSelect = this._defaults.onSelect;
840
- var inputEl = this.$input ? this.$input[0] : null;
841
- if (onSelect && inputEl) {
842
- onSelect.apply(inputEl, [this.formattedDateTime, this]);
1574
+
1575
+ var tzoffset = $.timepicker.timezoneOffsetNumber(tp_inst.timezone);
1576
+ var now = new Date();
1577
+ now.setMinutes(now.getMinutes() + now.getTimezoneOffset() + parseInt(tzoffset, 10));
1578
+ this._setTime(inst, now);
1579
+ this._setDate(inst, now);
1580
+ tp_inst._onSelectHandler();
1581
+ };
1582
+
1583
+ /*
1584
+ * Disable & enable the Time in the datetimepicker
1585
+ */
1586
+ $.datepicker._disableTimepickerDatepicker = function (target) {
1587
+ var inst = this._getInst(target);
1588
+ if (!inst) {
1589
+ return;
1590
+ }
1591
+
1592
+ var tp_inst = this._get(inst, 'timepicker');
1593
+ $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1594
+ if (tp_inst) {
1595
+ inst.settings.showTimepicker = false;
1596
+ tp_inst._defaults.showTimepicker = false;
1597
+ tp_inst._updateDateTime(inst);
843
1598
  }
844
- },
845
-
846
- //########################################################################
847
- // left for any backwards compatibility
848
- //########################################################################
849
- _formatTime: function(time, format) {
850
- time = time || { hour: this.hour, minute: this.minute, second: this.second, millisec: this.millisec, ampm: this.ampm, timezone: this.timezone };
851
- var tmptime = (format || this._defaults.timeFormat).toString();
852
-
853
- tmptime = $.datepicker.formatTime(tmptime, time, this._defaults);
854
-
855
- if (arguments.length) return tmptime;
856
- else this.formattedTime = tmptime;
857
- },
858
-
859
- //########################################################################
860
- // update our input with the new date time..
861
- //########################################################################
862
- _updateDateTime: function(dp_inst) {
863
- dp_inst = this.inst || dp_inst;
864
- var dt = $.datepicker._daylightSavingAdjust(new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay)),
865
- dateFmt = $.datepicker._get(dp_inst, 'dateFormat'),
866
- formatCfg = $.datepicker._getFormatConfig(dp_inst),
867
- timeAvailable = dt !== null && this.timeDefined;
868
- this.formattedDate = $.datepicker.formatDate(dateFmt, (dt === null ? new Date() : dt), formatCfg);
869
- var formattedDateTime = this.formattedDate;
870
- if (dp_inst.lastVal !== undefined && (dp_inst.lastVal.length > 0 && this.$input.val().length === 0))
1599
+ };
1600
+
1601
+ $.datepicker._enableTimepickerDatepicker = function (target) {
1602
+ var inst = this._getInst(target);
1603
+ if (!inst) {
871
1604
  return;
1605
+ }
872
1606
 
873
- if (this._defaults.timeOnly === true) {
874
- formattedDateTime = this.formattedTime;
875
- } else if (this._defaults.timeOnly !== true && (this._defaults.alwaysSetTime || timeAvailable)) {
876
- formattedDateTime += this._defaults.separator + this.formattedTime + this._defaults.timeSuffix;
1607
+ var tp_inst = this._get(inst, 'timepicker');
1608
+ $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1609
+ if (tp_inst) {
1610
+ inst.settings.showTimepicker = true;
1611
+ tp_inst._defaults.showTimepicker = true;
1612
+ tp_inst._addTimePicker(inst); // Could be disabled on page load
1613
+ tp_inst._updateDateTime(inst);
877
1614
  }
1615
+ };
878
1616
 
879
- this.formattedDateTime = formattedDateTime;
1617
+ /*
1618
+ * Create our own set time function
1619
+ */
1620
+ $.datepicker._setTime = function (inst, date) {
1621
+ var tp_inst = this._get(inst, 'timepicker');
1622
+ if (tp_inst) {
1623
+ var defaults = tp_inst._defaults;
880
1624
 
881
- if(!this._defaults.showTimepicker) {
882
- this.$input.val(this.formattedDate);
883
- } else if (this.$altInput && this._defaults.altFieldTimeOnly === true) {
884
- this.$altInput.val(this.formattedTime);
885
- this.$input.val(this.formattedDate);
886
- } else if(this.$altInput) {
887
- this.$altInput.val(formattedDateTime);
888
- this.$input.val(formattedDateTime);
889
- } else {
890
- this.$input.val(formattedDateTime);
1625
+ // calling _setTime with no date sets time to defaults
1626
+ tp_inst.hour = date ? date.getHours() : defaults.hour;
1627
+ tp_inst.minute = date ? date.getMinutes() : defaults.minute;
1628
+ tp_inst.second = date ? date.getSeconds() : defaults.second;
1629
+ tp_inst.millisec = date ? date.getMilliseconds() : defaults.millisec;
1630
+ tp_inst.microsec = date ? date.getMicroseconds() : defaults.microsec;
1631
+
1632
+ //check if within min/max times..
1633
+ tp_inst._limitMinMaxDateTime(inst, true);
1634
+
1635
+ tp_inst._onTimeChange();
1636
+ tp_inst._updateDateTime(inst);
891
1637
  }
1638
+ };
892
1639
 
893
- this.$input.trigger("change");
894
- }
1640
+ /*
1641
+ * Create new public method to set only time, callable as $().datepicker('setTime', date)
1642
+ */
1643
+ $.datepicker._setTimeDatepicker = function (target, date, withDate) {
1644
+ var inst = this._getInst(target);
1645
+ if (!inst) {
1646
+ return;
1647
+ }
895
1648
 
896
- });
1649
+ var tp_inst = this._get(inst, 'timepicker');
897
1650
 
898
- $.fn.extend({
899
- //########################################################################
900
- // shorthand just to use timepicker..
901
- //########################################################################
902
- timepicker: function(o) {
903
- o = o || {};
904
- var tmp_args = arguments;
1651
+ if (tp_inst) {
1652
+ this._setDateFromField(inst);
1653
+ var tp_date;
1654
+ if (date) {
1655
+ if (typeof date === "string") {
1656
+ tp_inst._parseTime(date, withDate);
1657
+ tp_date = new Date();
1658
+ tp_date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1659
+ tp_date.setMicroseconds(tp_inst.microsec);
1660
+ } else {
1661
+ tp_date = new Date(date.getTime());
1662
+ tp_date.setMicroseconds(date.getMicroseconds());
1663
+ }
1664
+ if (tp_date.toString() === 'Invalid Date') {
1665
+ tp_date = undefined;
1666
+ }
1667
+ this._setTime(inst, tp_date);
1668
+ }
1669
+ }
905
1670
 
906
- if (typeof o == 'object') tmp_args[0] = $.extend(o, { timeOnly: true });
1671
+ };
907
1672
 
908
- return $(this).each(function() {
909
- $.fn.datetimepicker.apply($(this), tmp_args);
910
- });
911
- },
912
-
913
- //########################################################################
914
- // extend timepicker to datepicker
915
- //########################################################################
916
- datetimepicker: function(o) {
917
- o = o || {};
918
- tmp_args = arguments;
919
-
920
- if (typeof(o) == 'string'){
921
- if(o == 'getDate')
922
- return $.fn.datepicker.apply($(this[0]), tmp_args);
923
- else
924
- return this.each(function() {
925
- var $t = $(this);
926
- $t.datepicker.apply($t, tmp_args);
927
- });
1673
+ /*
1674
+ * override setDate() to allow setting time too within Date object
1675
+ */
1676
+ $.datepicker._base_setDateDatepicker = $.datepicker._setDateDatepicker;
1677
+ $.datepicker._setDateDatepicker = function (target, _date) {
1678
+ var inst = this._getInst(target);
1679
+ var date = _date;
1680
+ if (!inst) {
1681
+ return;
928
1682
  }
929
- else
930
- return this.each(function() {
931
- var $t = $(this);
932
- $t.datepicker($.timepicker._newInst($t, o)._defaults);
933
- });
934
- }
935
- });
936
-
937
- //########################################################################
938
- // format the time all pretty...
939
- // format = string format of the time
940
- // time = a {}, not a Date() for timezones
941
- // options = essentially the regional[].. amNames, pmNames, ampm
942
- //########################################################################
943
- $.datepicker.formatTime = function(format, time, options) {
944
- options = options || {};
945
- options = $.extend($.timepicker._defaults, options);
946
- time = $.extend({hour:0, minute:0, second:0, millisec:0, timezone:'+0000'}, time);
947
-
948
- var tmptime = format;
949
- var ampmName = options['amNames'][0];
950
-
951
- var hour = parseInt(time.hour, 10);
952
- if (options.ampm) {
953
- if (hour > 11){
954
- ampmName = options['pmNames'][0];
955
- if(hour > 12)
956
- hour = hour % 12;
1683
+
1684
+ if (typeof(_date) === 'string') {
1685
+ date = new Date(_date);
1686
+ if (!date.getTime()) {
1687
+ this._base_setDateDatepicker.apply(this, arguments);
1688
+ date = $(target).datepicker('getDate');
1689
+ }
957
1690
  }
958
- if (hour === 0)
959
- hour = 12;
960
- }
961
- tmptime = tmptime.replace(/(?:hh?|mm?|ss?|[tT]{1,2}|[lz])/g, function(match) {
962
- switch (match.toLowerCase()) {
963
- case 'hh': return ('0' + hour).slice(-2);
964
- case 'h': return hour;
965
- case 'mm': return ('0' + time.minute).slice(-2);
966
- case 'm': return time.minute;
967
- case 'ss': return ('0' + time.second).slice(-2);
968
- case 's': return time.second;
969
- case 'l': return ('00' + time.millisec).slice(-3);
970
- case 'z': return time.timezone;
971
- case 't': case 'tt':
972
- if (options.ampm) {
973
- if (match.length == 1)
974
- ampmName = ampmName.charAt(0);
975
- return match.charAt(0) == 'T' ? ampmName.toUpperCase() : ampmName.toLowerCase();
976
- }
977
- return '';
1691
+
1692
+ var tp_inst = this._get(inst, 'timepicker');
1693
+ var tp_date;
1694
+ if (date instanceof Date) {
1695
+ tp_date = new Date(date.getTime());
1696
+ tp_date.setMicroseconds(date.getMicroseconds());
1697
+ } else {
1698
+ tp_date = date;
1699
+ }
1700
+
1701
+ // This is important if you are using the timezone option, javascript's Date
1702
+ // object will only return the timezone offset for the current locale, so we
1703
+ // adjust it accordingly. If not using timezone option this won't matter..
1704
+ // If a timezone is different in tp, keep the timezone as is
1705
+ if (tp_inst && tp_date) {
1706
+ // look out for DST if tz wasn't specified
1707
+ if (!tp_inst.support.timezone && tp_inst._defaults.timezone === null) {
1708
+ tp_inst.timezone = tp_date.getTimezoneOffset() * -1;
1709
+ }
1710
+ date = $.timepicker.timezoneAdjust(date, $.timepicker.timezoneOffsetString(-date.getTimezoneOffset()), tp_inst.timezone);
1711
+ tp_date = $.timepicker.timezoneAdjust(tp_date, $.timepicker.timezoneOffsetString(-tp_date.getTimezoneOffset()), tp_inst.timezone);
978
1712
  }
979
- });
980
1713
 
981
- tmptime = $.trim(tmptime);
982
- return tmptime;
983
- };
984
-
985
- //########################################################################
986
- // the bad hack :/ override datepicker so it doesnt close on select
987
- // inspired: http://stackoverflow.com/questions/1252512/jquery-datepicker-prevent-closing-picker-when-clicking-a-date/1762378#1762378
988
- //########################################################################
989
- $.datepicker._base_selectDate = $.datepicker._selectDate;
990
- $.datepicker._selectDate = function (id, dateStr) {
991
- var inst = this._getInst($(id)[0]),
992
- tp_inst = this._get(inst, 'timepicker');
993
-
994
- if (tp_inst) {
995
- tp_inst._limitMinMaxDateTime(inst, true);
996
- inst.inline = inst.stay_open = true;
997
- //This way the onSelect handler called from calendarpicker get the full dateTime
998
- this._base_selectDate(id, dateStr);
999
- inst.inline = inst.stay_open = false;
1000
- this._notifyChange(inst);
1001
1714
  this._updateDatepicker(inst);
1002
- }
1003
- else this._base_selectDate(id, dateStr);
1004
- };
1005
-
1006
- //#############################################################################################
1007
- // second bad hack :/ override datepicker so it triggers an event when changing the input field
1008
- // and does not redraw the datepicker on every selectDate event
1009
- //#############################################################################################
1010
- $.datepicker._base_updateDatepicker = $.datepicker._updateDatepicker;
1011
- $.datepicker._updateDatepicker = function(inst) {
1012
-
1013
- // don't popup the datepicker if there is another instance already opened
1014
- var input = inst.input[0];
1015
- if($.datepicker._curInst &&
1016
- $.datepicker._curInst != inst &&
1017
- $.datepicker._datepickerShowing &&
1018
- $.datepicker._lastInput != input) {
1019
- return;
1020
- }
1715
+ this._base_setDateDatepicker.apply(this, arguments);
1716
+ this._setTimeDatepicker(target, tp_date, true);
1717
+ };
1021
1718
 
1022
- if (typeof(inst.stay_open) !== 'boolean' || inst.stay_open === false) {
1719
+ /*
1720
+ * override getDate() to allow getting time too within Date object
1721
+ */
1722
+ $.datepicker._base_getDateDatepicker = $.datepicker._getDateDatepicker;
1723
+ $.datepicker._getDateDatepicker = function (target, noDefault) {
1724
+ var inst = this._getInst(target);
1725
+ if (!inst) {
1726
+ return;
1727
+ }
1728
+
1729
+ var tp_inst = this._get(inst, 'timepicker');
1023
1730
 
1024
- this._base_updateDatepicker(inst);
1731
+ if (tp_inst) {
1732
+ // if it hasn't yet been defined, grab from field
1733
+ if (inst.lastVal === undefined) {
1734
+ this._setDateFromField(inst, noDefault);
1735
+ }
1736
+
1737
+ var date = this._getDate(inst);
1738
+
1739
+ var currDT = null;
1740
+
1741
+ if (tp_inst.$altInput && tp_inst._defaults.altFieldTimeOnly) {
1742
+ currDT = tp_inst.$input.val() + ' ' + tp_inst.$altInput.val();
1743
+ }
1744
+ else if (tp_inst.$input.get(0).tagName !== 'INPUT' && tp_inst.$altInput) {
1745
+ /**
1746
+ * in case the datetimepicker has been applied to a non-input tag for inline UI,
1747
+ * and the user has not configured the plugin to display only time in altInput,
1748
+ * pick current date time from the altInput (and hope for the best, for now, until "ER1" is applied)
1749
+ *
1750
+ * @todo ER1. Since altInput can have a totally difference format, convert it to standard format by reading input format from "altFormat" and "altTimeFormat" option values
1751
+ */
1752
+ currDT = tp_inst.$altInput.val();
1753
+ }
1754
+ else {
1755
+ currDT = tp_inst.$input.val();
1756
+ }
1757
+
1758
+ if (date && tp_inst._parseTime(currDT, !inst.settings.timeOnly)) {
1759
+ date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1760
+ date.setMicroseconds(tp_inst.microsec);
1761
+
1762
+ // This is important if you are using the timezone option, javascript's Date
1763
+ // object will only return the timezone offset for the current locale, so we
1764
+ // adjust it accordingly. If not using timezone option this won't matter..
1765
+ if (tp_inst.timezone != null) {
1766
+ // look out for DST if tz wasn't specified
1767
+ if (!tp_inst.support.timezone && tp_inst._defaults.timezone === null) {
1768
+ tp_inst.timezone = date.getTimezoneOffset() * -1;
1769
+ }
1770
+ date = $.timepicker.timezoneAdjust(date, tp_inst.timezone, $.timepicker.timezoneOffsetString(-date.getTimezoneOffset()));
1771
+ }
1772
+ }
1773
+ return date;
1774
+ }
1775
+ return this._base_getDateDatepicker(target, noDefault);
1776
+ };
1777
+
1778
+ /*
1779
+ * override parseDate() because UI 1.8.14 throws an error about "Extra characters"
1780
+ * An option in datapicker to ignore extra format characters would be nicer.
1781
+ */
1782
+ $.datepicker._base_parseDate = $.datepicker.parseDate;
1783
+ $.datepicker.parseDate = function (format, value, settings) {
1784
+ var date;
1785
+ try {
1786
+ date = this._base_parseDate(format, value, settings);
1787
+ } catch (err) {
1788
+ // Hack! The error message ends with a colon, a space, and
1789
+ // the "extra" characters. We rely on that instead of
1790
+ // attempting to perfectly reproduce the parsing algorithm.
1791
+ if (err.indexOf(":") >= 0) {
1792
+ date = this._base_parseDate(format, value.substring(0, value.length - (err.length - err.indexOf(':') - 2)), settings);
1793
+ $.timepicker.log("Error parsing the date string: " + err + "\ndate string = " + value + "\ndate format = " + format);
1794
+ } else {
1795
+ throw err;
1796
+ }
1797
+ }
1798
+ return date;
1799
+ };
1025
1800
 
1026
- // Reload the time control when changing something in the input text field.
1801
+ /*
1802
+ * override formatDate to set date with time to the input
1803
+ */
1804
+ $.datepicker._base_formatDate = $.datepicker._formatDate;
1805
+ $.datepicker._formatDate = function (inst, day, month, year) {
1027
1806
  var tp_inst = this._get(inst, 'timepicker');
1028
- if(tp_inst) tp_inst._addTimePicker(inst);
1029
- }
1030
- };
1031
-
1032
- //#######################################################################################
1033
- // third bad hack :/ override datepicker so it allows spaces and colon in the input field
1034
- //#######################################################################################
1035
- $.datepicker._base_doKeyPress = $.datepicker._doKeyPress;
1036
- $.datepicker._doKeyPress = function(event) {
1037
- var inst = $.datepicker._getInst(event.target),
1038
- tp_inst = $.datepicker._get(inst, 'timepicker');
1039
-
1040
- if (tp_inst) {
1041
- if ($.datepicker._get(inst, 'constrainInput')) {
1042
- var ampm = tp_inst._defaults.ampm,
1043
- dateChars = $.datepicker._possibleChars($.datepicker._get(inst, 'dateFormat')),
1044
- datetimeChars = tp_inst._defaults.timeFormat.toString()
1045
- .replace(/[hms]/g, '')
1046
- .replace(/TT/g, ampm ? 'APM' : '')
1047
- .replace(/Tt/g, ampm ? 'AaPpMm' : '')
1048
- .replace(/tT/g, ampm ? 'AaPpMm' : '')
1049
- .replace(/T/g, ampm ? 'AP' : '')
1050
- .replace(/tt/g, ampm ? 'apm' : '')
1051
- .replace(/t/g, ampm ? 'ap' : '') +
1052
- " " +
1053
- tp_inst._defaults.separator +
1054
- tp_inst._defaults.timeSuffix +
1055
- (tp_inst._defaults.showTimezone ? tp_inst._defaults.timezoneList.join('') : '') +
1056
- (tp_inst._defaults.amNames.join('')) +
1057
- (tp_inst._defaults.pmNames.join('')) +
1058
- dateChars,
1059
- chr = String.fromCharCode(event.charCode === undefined ? event.keyCode : event.charCode);
1060
- return event.ctrlKey || (chr < ' ' || !dateChars || datetimeChars.indexOf(chr) > -1);
1807
+ if (tp_inst) {
1808
+ tp_inst._updateDateTime(inst);
1809
+ return tp_inst.$input.val();
1061
1810
  }
1062
- }
1811
+ return this._base_formatDate(inst);
1812
+ };
1063
1813
 
1064
- return $.datepicker._base_doKeyPress(event);
1065
- };
1814
+ /*
1815
+ * override options setter to add time to maxDate(Time) and minDate(Time). MaxDate
1816
+ */
1817
+ $.datepicker._base_optionDatepicker = $.datepicker._optionDatepicker;
1818
+ $.datepicker._optionDatepicker = function (target, name, value) {
1819
+ var inst = this._getInst(target),
1820
+ name_clone;
1821
+ if (!inst) {
1822
+ return null;
1823
+ }
1066
1824
 
1067
- //#######################################################################################
1068
- // Override key up event to sync manual input changes.
1069
- //#######################################################################################
1070
- $.datepicker._base_doKeyUp = $.datepicker._doKeyUp;
1071
- $.datepicker._doKeyUp = function (event) {
1072
- var inst = $.datepicker._getInst(event.target),
1073
- tp_inst = $.datepicker._get(inst, 'timepicker');
1825
+ var tp_inst = this._get(inst, 'timepicker');
1826
+ if (tp_inst) {
1827
+ var min = null,
1828
+ max = null,
1829
+ onselect = null,
1830
+ overrides = tp_inst._defaults.evnts,
1831
+ fns = {},
1832
+ prop,
1833
+ ret,
1834
+ oldVal,
1835
+ $target;
1836
+ if (typeof name === 'string') { // if min/max was set with the string
1837
+ if (name === 'minDate' || name === 'minDateTime') {
1838
+ min = value;
1839
+ } else if (name === 'maxDate' || name === 'maxDateTime') {
1840
+ max = value;
1841
+ } else if (name === 'onSelect') {
1842
+ onselect = value;
1843
+ } else if (overrides.hasOwnProperty(name)) {
1844
+ if (typeof (value) === 'undefined') {
1845
+ return overrides[name];
1846
+ }
1847
+ fns[name] = value;
1848
+ name_clone = {}; //empty results in exiting function after overrides updated
1849
+ }
1850
+ } else if (typeof name === 'object') { //if min/max was set with the JSON
1851
+ if (name.minDate) {
1852
+ min = name.minDate;
1853
+ } else if (name.minDateTime) {
1854
+ min = name.minDateTime;
1855
+ } else if (name.maxDate) {
1856
+ max = name.maxDate;
1857
+ } else if (name.maxDateTime) {
1858
+ max = name.maxDateTime;
1859
+ }
1860
+ for (prop in overrides) {
1861
+ if (overrides.hasOwnProperty(prop) && name[prop]) {
1862
+ fns[prop] = name[prop];
1863
+ }
1864
+ }
1865
+ }
1866
+ for (prop in fns) {
1867
+ if (fns.hasOwnProperty(prop)) {
1868
+ overrides[prop] = fns[prop];
1869
+ if (!name_clone) { name_clone = $.extend({}, name); }
1870
+ delete name_clone[prop];
1871
+ }
1872
+ }
1873
+ if (name_clone && isEmptyObject(name_clone)) { return; }
1874
+ if (min) { //if min was set
1875
+ if (min === 0) {
1876
+ min = new Date();
1877
+ } else {
1878
+ min = new Date(min);
1879
+ }
1880
+ tp_inst._defaults.minDate = min;
1881
+ tp_inst._defaults.minDateTime = min;
1882
+ } else if (max) { //if max was set
1883
+ if (max === 0) {
1884
+ max = new Date();
1885
+ } else {
1886
+ max = new Date(max);
1887
+ }
1888
+ tp_inst._defaults.maxDate = max;
1889
+ tp_inst._defaults.maxDateTime = max;
1890
+ } else if (onselect) {
1891
+ tp_inst._defaults.onSelect = onselect;
1892
+ }
1074
1893
 
1075
- if (tp_inst) {
1076
- if (tp_inst._defaults.timeOnly && (inst.input.val() != inst.lastVal)) {
1077
- try {
1078
- $.datepicker._updateDatepicker(inst);
1894
+ // Datepicker will override our date when we call _base_optionDatepicker when
1895
+ // calling minDate/maxDate, so we will first grab the value, call
1896
+ // _base_optionDatepicker, then set our value back.
1897
+ if(min || max){
1898
+ $target = $(target);
1899
+ oldVal = $target.datetimepicker('getDate');
1900
+ ret = this._base_optionDatepicker.call($.datepicker, target, name_clone || name, value);
1901
+ $target.datetimepicker('setDate', oldVal);
1902
+ return ret;
1079
1903
  }
1080
- catch (err) {
1081
- $.datepicker.log(err);
1904
+ }
1905
+ if (value === undefined) {
1906
+ return this._base_optionDatepicker.call($.datepicker, target, name);
1907
+ }
1908
+ return this._base_optionDatepicker.call($.datepicker, target, name_clone || name, value);
1909
+ };
1910
+
1911
+ /*
1912
+ * jQuery isEmptyObject does not check hasOwnProperty - if someone has added to the object prototype,
1913
+ * it will return false for all objects
1914
+ */
1915
+ var isEmptyObject = function (obj) {
1916
+ var prop;
1917
+ for (prop in obj) {
1918
+ if (obj.hasOwnProperty(prop)) {
1919
+ return false;
1082
1920
  }
1083
1921
  }
1084
- }
1922
+ return true;
1923
+ };
1085
1924
 
1086
- return $.datepicker._base_doKeyUp(event);
1087
- };
1088
-
1089
- //#######################################################################################
1090
- // override "Today" button to also grab the time.
1091
- //#######################################################################################
1092
- $.datepicker._base_gotoToday = $.datepicker._gotoToday;
1093
- $.datepicker._gotoToday = function(id) {
1094
- var inst = this._getInst($(id)[0]),
1095
- $dp = inst.dpDiv;
1096
- this._base_gotoToday(id);
1097
- var now = new Date();
1098
- var tp_inst = this._get(inst, 'timepicker');
1099
- if (tp_inst && tp_inst._defaults.showTimezone && tp_inst.timezone_select) {
1100
- var tzoffset = now.getTimezoneOffset(); // If +0100, returns -60
1101
- var tzsign = tzoffset > 0 ? '-' : '+';
1102
- tzoffset = Math.abs(tzoffset);
1103
- var tzmin = tzoffset % 60;
1104
- tzoffset = tzsign + ('0' + (tzoffset - tzmin) / 60).slice(-2) + ('0' + tzmin).slice(-2);
1105
- if (tp_inst._defaults.timezoneIso8609)
1106
- tzoffset = tzoffset.substring(0, 3) + ':' + tzoffset.substring(3);
1107
- tp_inst.timezone_select.val(tzoffset);
1108
- }
1109
- this._setTime(inst, now);
1110
- $( '.ui-datepicker-today', $dp).click();
1111
- };
1112
-
1113
- //#######################################################################################
1114
- // Disable & enable the Time in the datetimepicker
1115
- //#######################################################################################
1116
- $.datepicker._disableTimepickerDatepicker = function(target, date, withDate) {
1117
- var inst = this._getInst(target),
1118
- tp_inst = this._get(inst, 'timepicker');
1119
- $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1120
- if (tp_inst) {
1121
- tp_inst._defaults.showTimepicker = false;
1122
- tp_inst._updateDateTime(inst);
1123
- }
1124
- };
1125
-
1126
- $.datepicker._enableTimepickerDatepicker = function(target, date, withDate) {
1127
- var inst = this._getInst(target),
1128
- tp_inst = this._get(inst, 'timepicker');
1129
- $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1130
- if (tp_inst) {
1131
- tp_inst._defaults.showTimepicker = true;
1132
- tp_inst._addTimePicker(inst); // Could be disabled on page load
1133
- tp_inst._updateDateTime(inst);
1134
- }
1135
- };
1136
-
1137
- //#######################################################################################
1138
- // Create our own set time function
1139
- //#######################################################################################
1140
- $.datepicker._setTime = function(inst, date) {
1141
- var tp_inst = this._get(inst, 'timepicker');
1142
- if (tp_inst) {
1143
- var defaults = tp_inst._defaults,
1144
- // calling _setTime with no date sets time to defaults
1145
- hour = date ? date.getHours() : defaults.hour,
1146
- minute = date ? date.getMinutes() : defaults.minute,
1147
- second = date ? date.getSeconds() : defaults.second,
1148
- millisec = date ? date.getMilliseconds() : defaults.millisec;
1149
-
1150
- //check if within min/max times..
1151
- if ((hour < defaults.hourMin || hour > defaults.hourMax) || (minute < defaults.minuteMin || minute > defaults.minuteMax) || (second < defaults.secondMin || second > defaults.secondMax) || (millisec < defaults.millisecMin || millisec > defaults.millisecMax)) {
1152
- hour = defaults.hourMin;
1153
- minute = defaults.minuteMin;
1154
- second = defaults.secondMin;
1155
- millisec = defaults.millisecMin;
1925
+ /*
1926
+ * jQuery extend now ignores nulls!
1927
+ */
1928
+ var extendRemove = function (target, props) {
1929
+ $.extend(target, props);
1930
+ for (var name in props) {
1931
+ if (props[name] === null || props[name] === undefined) {
1932
+ target[name] = props[name];
1933
+ }
1156
1934
  }
1935
+ return target;
1936
+ };
1937
+
1938
+ /*
1939
+ * Determine by the time format which units are supported
1940
+ * Returns an object of booleans for each unit
1941
+ */
1942
+ var detectSupport = function (timeFormat) {
1943
+ var tf = timeFormat.replace(/'.*?'/g, '').toLowerCase(), // removes literals
1944
+ isIn = function (f, t) { // does the format contain the token?
1945
+ return f.indexOf(t) !== -1 ? true : false;
1946
+ };
1947
+ return {
1948
+ hour: isIn(tf, 'h'),
1949
+ minute: isIn(tf, 'm'),
1950
+ second: isIn(tf, 's'),
1951
+ millisec: isIn(tf, 'l'),
1952
+ microsec: isIn(tf, 'c'),
1953
+ timezone: isIn(tf, 'z'),
1954
+ ampm: isIn(tf, 't') && isIn(timeFormat, 'h'),
1955
+ iso8601: isIn(timeFormat, 'Z')
1956
+ };
1957
+ };
1157
1958
 
1158
- tp_inst.hour = hour;
1159
- tp_inst.minute = minute;
1160
- tp_inst.second = second;
1161
- tp_inst.millisec = millisec;
1959
+ /*
1960
+ * Converts 24 hour format into 12 hour
1961
+ * Returns 12 hour without leading 0
1962
+ */
1963
+ var convert24to12 = function (hour) {
1964
+ hour %= 12;
1162
1965
 
1163
- if (tp_inst.hour_slider) tp_inst.hour_slider.slider('value', hour);
1164
- if (tp_inst.minute_slider) tp_inst.minute_slider.slider('value', minute);
1165
- if (tp_inst.second_slider) tp_inst.second_slider.slider('value', second);
1166
- if (tp_inst.millisec_slider) tp_inst.millisec_slider.slider('value', millisec);
1966
+ if (hour === 0) {
1967
+ hour = 12;
1968
+ }
1167
1969
 
1168
- tp_inst._onTimeChange();
1169
- tp_inst._updateDateTime(inst);
1170
- }
1171
- };
1970
+ return String(hour);
1971
+ };
1172
1972
 
1173
- //#######################################################################################
1174
- // Create new public method to set only time, callable as $().datepicker('setTime', date)
1175
- //#######################################################################################
1176
- $.datepicker._setTimeDatepicker = function(target, date, withDate) {
1177
- var inst = this._getInst(target),
1178
- tp_inst = this._get(inst, 'timepicker');
1973
+ var computeEffectiveSetting = function (settings, property) {
1974
+ return settings && settings[property] ? settings[property] : $.timepicker._defaults[property];
1975
+ };
1179
1976
 
1180
- if (tp_inst) {
1181
- this._setDateFromField(inst);
1182
- var tp_date;
1183
- if (date) {
1184
- if (typeof date == "string") {
1185
- tp_inst._parseTime(date, withDate);
1186
- tp_date = new Date();
1187
- tp_date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1188
- }
1189
- else tp_date = new Date(date.getTime());
1190
- if (tp_date.toString() == 'Invalid Date') tp_date = undefined;
1191
- this._setTime(inst, tp_date);
1977
+ /*
1978
+ * Splits datetime string into date and time substrings.
1979
+ * Throws exception when date can't be parsed
1980
+ * Returns {dateString: dateString, timeString: timeString}
1981
+ */
1982
+ var splitDateTime = function (dateTimeString, timeSettings) {
1983
+ // The idea is to get the number separator occurrences in datetime and the time format requested (since time has
1984
+ // fewer unknowns, mostly numbers and am/pm). We will use the time pattern to split.
1985
+ var separator = computeEffectiveSetting(timeSettings, 'separator'),
1986
+ format = computeEffectiveSetting(timeSettings, 'timeFormat'),
1987
+ timeParts = format.split(separator), // how many occurrences of separator may be in our format?
1988
+ timePartsLen = timeParts.length,
1989
+ allParts = dateTimeString.split(separator),
1990
+ allPartsLen = allParts.length;
1991
+
1992
+ if (allPartsLen > 1) {
1993
+ return {
1994
+ dateString: allParts.splice(0, allPartsLen - timePartsLen).join(separator),
1995
+ timeString: allParts.splice(0, timePartsLen).join(separator)
1996
+ };
1192
1997
  }
1193
- }
1194
1998
 
1195
- };
1196
-
1197
- //#######################################################################################
1198
- // override setDate() to allow setting time too within Date object
1199
- //#######################################################################################
1200
- $.datepicker._base_setDateDatepicker = $.datepicker._setDateDatepicker;
1201
- $.datepicker._setDateDatepicker = function(target, date) {
1202
- var inst = this._getInst(target),
1203
- tp_date = (date instanceof Date) ? new Date(date.getTime()) : date;
1204
-
1205
- this._updateDatepicker(inst);
1206
- this._base_setDateDatepicker.apply(this, arguments);
1207
- this._setTimeDatepicker(target, tp_date, true);
1208
- };
1209
-
1210
- //#######################################################################################
1211
- // override getDate() to allow getting time too within Date object
1212
- //#######################################################################################
1213
- $.datepicker._base_getDateDatepicker = $.datepicker._getDateDatepicker;
1214
- $.datepicker._getDateDatepicker = function(target, noDefault) {
1215
- var inst = this._getInst(target),
1216
- tp_inst = this._get(inst, 'timepicker');
1217
-
1218
- if (tp_inst) {
1219
- this._setDateFromField(inst, noDefault);
1220
- var date = this._getDate(inst);
1221
- if (date && tp_inst._parseTime($(target).val(), tp_inst.timeOnly)) date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1999
+ return {
2000
+ dateString: dateTimeString,
2001
+ timeString: ''
2002
+ };
2003
+ };
2004
+
2005
+ /*
2006
+ * Internal function to parse datetime interval
2007
+ * Returns: {date: Date, timeObj: Object}, where
2008
+ * date - parsed date without time (type Date)
2009
+ * timeObj = {hour: , minute: , second: , millisec: , microsec: } - parsed time. Optional
2010
+ */
2011
+ var parseDateTimeInternal = function (dateFormat, timeFormat, dateTimeString, dateSettings, timeSettings) {
2012
+ var date,
2013
+ parts,
2014
+ parsedTime;
2015
+
2016
+ parts = splitDateTime(dateTimeString, timeSettings);
2017
+ date = $.datepicker._base_parseDate(dateFormat, parts.dateString, dateSettings);
2018
+
2019
+ if (parts.timeString === '') {
2020
+ return {
2021
+ date: date
2022
+ };
2023
+ }
2024
+
2025
+ parsedTime = $.datepicker.parseTime(timeFormat, parts.timeString, timeSettings);
2026
+
2027
+ if (!parsedTime) {
2028
+ throw 'Wrong time format';
2029
+ }
2030
+
2031
+ return {
2032
+ date: date,
2033
+ timeObj: parsedTime
2034
+ };
2035
+ };
2036
+
2037
+ /*
2038
+ * Internal function to set timezone_select to the local timezone
2039
+ */
2040
+ var selectLocalTimezone = function (tp_inst, date) {
2041
+ if (tp_inst && tp_inst.timezone_select) {
2042
+ var now = date || new Date();
2043
+ tp_inst.timezone_select.val(-now.getTimezoneOffset());
2044
+ }
2045
+ };
2046
+
2047
+ /*
2048
+ * Create a Singleton Instance
2049
+ */
2050
+ $.timepicker = new Timepicker();
2051
+
2052
+ /**
2053
+ * Get the timezone offset as string from a date object (eg '+0530' for UTC+5.5)
2054
+ * @param {number} tzMinutes if not a number, less than -720 (-1200), or greater than 840 (+1400) this value is returned
2055
+ * @param {boolean} iso8601 if true formats in accordance to iso8601 "+12:45"
2056
+ * @return {string}
2057
+ */
2058
+ $.timepicker.timezoneOffsetString = function (tzMinutes, iso8601) {
2059
+ if (isNaN(tzMinutes) || tzMinutes > 840 || tzMinutes < -720) {
2060
+ return tzMinutes;
2061
+ }
2062
+
2063
+ var off = tzMinutes,
2064
+ minutes = off % 60,
2065
+ hours = (off - minutes) / 60,
2066
+ iso = iso8601 ? ':' : '',
2067
+ tz = (off >= 0 ? '+' : '-') + ('0' + Math.abs(hours)).slice(-2) + iso + ('0' + Math.abs(minutes)).slice(-2);
2068
+
2069
+ if (tz === '+00:00') {
2070
+ return 'Z';
2071
+ }
2072
+ return tz;
2073
+ };
2074
+
2075
+ /**
2076
+ * Get the number in minutes that represents a timezone string
2077
+ * @param {string} tzString formatted like "+0500", "-1245", "Z"
2078
+ * @return {number} the offset minutes or the original string if it doesn't match expectations
2079
+ */
2080
+ $.timepicker.timezoneOffsetNumber = function (tzString) {
2081
+ var normalized = tzString.toString().replace(':', ''); // excuse any iso8601, end up with "+1245"
2082
+
2083
+ if (normalized.toUpperCase() === 'Z') { // if iso8601 with Z, its 0 minute offset
2084
+ return 0;
2085
+ }
2086
+
2087
+ if (!/^(\-|\+)\d{4}$/.test(normalized)) { // possibly a user defined tz, so just give it back
2088
+ return parseInt(tzString, 10);
2089
+ }
2090
+
2091
+ return ((normalized.substr(0, 1) === '-' ? -1 : 1) * // plus or minus
2092
+ ((parseInt(normalized.substr(1, 2), 10) * 60) + // hours (converted to minutes)
2093
+ parseInt(normalized.substr(3, 2), 10))); // minutes
2094
+ };
2095
+
2096
+ /**
2097
+ * No way to set timezone in js Date, so we must adjust the minutes to compensate. (think setDate, getDate)
2098
+ * @param {Date} date
2099
+ * @param {string} fromTimezone formatted like "+0500", "-1245"
2100
+ * @param {string} toTimezone formatted like "+0500", "-1245"
2101
+ * @return {Date}
2102
+ */
2103
+ $.timepicker.timezoneAdjust = function (date, fromTimezone, toTimezone) {
2104
+ var fromTz = $.timepicker.timezoneOffsetNumber(fromTimezone);
2105
+ var toTz = $.timepicker.timezoneOffsetNumber(toTimezone);
2106
+ if (!isNaN(toTz)) {
2107
+ date.setMinutes(date.getMinutes() + (-fromTz) - (-toTz));
2108
+ }
1222
2109
  return date;
1223
- }
1224
- return this._base_getDateDatepicker(target, noDefault);
1225
- };
1226
-
1227
- //#######################################################################################
1228
- // override parseDate() because UI 1.8.14 throws an error about "Extra characters"
1229
- // An option in datapicker to ignore extra format characters would be nicer.
1230
- //#######################################################################################
1231
- $.datepicker._base_parseDate = $.datepicker.parseDate;
1232
- $.datepicker.parseDate = function(format, value, settings) {
1233
- var date;
1234
- try {
1235
- date = this._base_parseDate(format, value, settings);
1236
- } catch (err) {
1237
- if (err.indexOf(":") >= 0) {
1238
- // Hack! The error message ends with a colon, a space, and
1239
- // the "extra" characters. We rely on that instead of
1240
- // attempting to perfectly reproduce the parsing algorithm.
1241
- date = this._base_parseDate(format, value.substring(0,value.length-(err.length-err.indexOf(':')-2)), settings);
1242
- } else {
1243
- // The underlying error was not related to the time
1244
- throw err;
2110
+ };
2111
+
2112
+ /**
2113
+ * Calls `timepicker()` on the `startTime` and `endTime` elements, and configures them to
2114
+ * enforce date range limits.
2115
+ * n.b. The input value must be correctly formatted (reformatting is not supported)
2116
+ * @param {Element} startTime
2117
+ * @param {Element} endTime
2118
+ * @param {Object} options Options for the timepicker() call
2119
+ * @return {jQuery}
2120
+ */
2121
+ $.timepicker.timeRange = function (startTime, endTime, options) {
2122
+ return $.timepicker.handleRange('timepicker', startTime, endTime, options);
2123
+ };
2124
+
2125
+ /**
2126
+ * Calls `datetimepicker` on the `startTime` and `endTime` elements, and configures them to
2127
+ * enforce date range limits.
2128
+ * @param {Element} startTime
2129
+ * @param {Element} endTime
2130
+ * @param {Object} options Options for the `timepicker()` call. Also supports `reformat`,
2131
+ * a boolean value that can be used to reformat the input values to the `dateFormat`.
2132
+ * @param {string} method Can be used to specify the type of picker to be added
2133
+ * @return {jQuery}
2134
+ */
2135
+ $.timepicker.datetimeRange = function (startTime, endTime, options) {
2136
+ $.timepicker.handleRange('datetimepicker', startTime, endTime, options);
2137
+ };
2138
+
2139
+ /**
2140
+ * Calls `datepicker` on the `startTime` and `endTime` elements, and configures them to
2141
+ * enforce date range limits.
2142
+ * @param {Element} startTime
2143
+ * @param {Element} endTime
2144
+ * @param {Object} options Options for the `timepicker()` call. Also supports `reformat`,
2145
+ * a boolean value that can be used to reformat the input values to the `dateFormat`.
2146
+ * @return {jQuery}
2147
+ */
2148
+ $.timepicker.dateRange = function (startTime, endTime, options) {
2149
+ $.timepicker.handleRange('datepicker', startTime, endTime, options);
2150
+ };
2151
+
2152
+ /**
2153
+ * Calls `method` on the `startTime` and `endTime` elements, and configures them to
2154
+ * enforce date range limits.
2155
+ * @param {string} method Can be used to specify the type of picker to be added
2156
+ * @param {Element} startTime
2157
+ * @param {Element} endTime
2158
+ * @param {Object} options Options for the `timepicker()` call. Also supports `reformat`,
2159
+ * a boolean value that can be used to reformat the input values to the `dateFormat`.
2160
+ * @return {jQuery}
2161
+ */
2162
+ $.timepicker.handleRange = function (method, startTime, endTime, options) {
2163
+ options = $.extend({}, {
2164
+ minInterval: 0, // min allowed interval in milliseconds
2165
+ maxInterval: 0, // max allowed interval in milliseconds
2166
+ start: {}, // options for start picker
2167
+ end: {} // options for end picker
2168
+ }, options);
2169
+
2170
+ // for the mean time this fixes an issue with calling getDate with timepicker()
2171
+ var timeOnly = false;
2172
+ if(method === 'timepicker'){
2173
+ timeOnly = true;
2174
+ method = 'datetimepicker';
1245
2175
  }
1246
- }
1247
- return date;
1248
- };
1249
-
1250
- //#######################################################################################
1251
- // override formatDate to set date with time to the input
1252
- //#######################################################################################
1253
- $.datepicker._base_formatDate = $.datepicker._formatDate;
1254
- $.datepicker._formatDate = function(inst, day, month, year){
1255
- var tp_inst = this._get(inst, 'timepicker');
1256
- if(tp_inst) {
1257
- tp_inst._updateDateTime(inst);
1258
- return tp_inst.$input.val();
1259
- }
1260
- return this._base_formatDate(inst);
1261
- };
1262
-
1263
- //#######################################################################################
1264
- // override options setter to add time to maxDate(Time) and minDate(Time). MaxDate
1265
- //#######################################################################################
1266
- $.datepicker._base_optionDatepicker = $.datepicker._optionDatepicker;
1267
- $.datepicker._optionDatepicker = function(target, name, value) {
1268
- var inst = this._getInst(target),
1269
- tp_inst = this._get(inst, 'timepicker');
1270
- if (tp_inst) {
1271
- var min = null, max = null, onselect = null;
1272
- if (typeof name == 'string') { // if min/max was set with the string
1273
- if (name === 'minDate' || name === 'minDateTime' )
1274
- min = value;
1275
- else if (name === 'maxDate' || name === 'maxDateTime')
1276
- max = value;
1277
- else if (name === 'onSelect')
1278
- onselect = value;
1279
- } else if (typeof name == 'object') { //if min/max was set with the JSON
1280
- if (name.minDate)
1281
- min = name.minDate;
1282
- else if (name.minDateTime)
1283
- min = name.minDateTime;
1284
- else if (name.maxDate)
1285
- max = name.maxDate;
1286
- else if (name.maxDateTime)
1287
- max = name.maxDateTime;
2176
+
2177
+ function checkDates(changed, other) {
2178
+ var startdt = startTime[method]('getDate'),
2179
+ enddt = endTime[method]('getDate'),
2180
+ changeddt = changed[method]('getDate');
2181
+
2182
+ if (startdt !== null) {
2183
+ var minDate = new Date(startdt.getTime()),
2184
+ maxDate = new Date(startdt.getTime());
2185
+
2186
+ minDate.setMilliseconds(minDate.getMilliseconds() + options.minInterval);
2187
+ maxDate.setMilliseconds(maxDate.getMilliseconds() + options.maxInterval);
2188
+
2189
+ if (options.minInterval > 0 && minDate > enddt) { // minInterval check
2190
+ endTime[method]('setDate', minDate);
2191
+ }
2192
+ else if (options.maxInterval > 0 && maxDate < enddt) { // max interval check
2193
+ endTime[method]('setDate', maxDate);
2194
+ }
2195
+ else if (startdt > enddt) {
2196
+ other[method]('setDate', changeddt);
2197
+ }
2198
+ }
2199
+ }
2200
+
2201
+ function selected(changed, other, option) {
2202
+ if (!changed.val()) {
2203
+ return;
2204
+ }
2205
+ var date = changed[method].call(changed, 'getDate');
2206
+ if (date !== null && options.minInterval > 0) {
2207
+ if (option === 'minDate') {
2208
+ date.setMilliseconds(date.getMilliseconds() + options.minInterval);
2209
+ }
2210
+ if (option === 'maxDate') {
2211
+ date.setMilliseconds(date.getMilliseconds() - options.minInterval);
2212
+ }
2213
+ }
2214
+
2215
+ if (date.getTime) {
2216
+ other[method].call(other, 'option', option, date);
2217
+ }
1288
2218
  }
1289
- if(min) { //if min was set
1290
- if (min == 0)
1291
- min = new Date();
1292
- else
1293
- min = new Date(min);
1294
-
1295
- tp_inst._defaults.minDate = min;
1296
- tp_inst._defaults.minDateTime = min;
1297
- } else if (max) { //if max was set
1298
- if(max==0)
1299
- max=new Date();
1300
- else
1301
- max= new Date(max);
1302
- tp_inst._defaults.maxDate = max;
1303
- tp_inst._defaults.maxDateTime = max;
1304
- } else if (onselect)
1305
- tp_inst._defaults.onSelect = onselect;
2219
+
2220
+ $.fn[method].call(startTime, $.extend({
2221
+ timeOnly: timeOnly,
2222
+ onClose: function (dateText, inst) {
2223
+ checkDates($(this), endTime);
2224
+ },
2225
+ onSelect: function (selectedDateTime) {
2226
+ selected($(this), endTime, 'minDate');
2227
+ }
2228
+ }, options, options.start));
2229
+ $.fn[method].call(endTime, $.extend({
2230
+ timeOnly: timeOnly,
2231
+ onClose: function (dateText, inst) {
2232
+ checkDates($(this), startTime);
2233
+ },
2234
+ onSelect: function (selectedDateTime) {
2235
+ selected($(this), startTime, 'maxDate');
2236
+ }
2237
+ }, options, options.end));
2238
+
2239
+ checkDates(startTime, endTime);
2240
+
2241
+ selected(startTime, endTime, 'minDate');
2242
+ selected(endTime, startTime, 'maxDate');
2243
+
2244
+ return $([startTime.get(0), endTime.get(0)]);
2245
+ };
2246
+
2247
+ /**
2248
+ * Log error or data to the console during error or debugging
2249
+ * @param {Object} err pass any type object to log to the console during error or debugging
2250
+ * @return {void}
2251
+ */
2252
+ $.timepicker.log = function () {
2253
+ // Older IE (9, maybe 10) throw error on accessing `window.console.log.apply`, so check first.
2254
+ if (window.console && window.console.log && window.console.log.apply) {
2255
+ window.console.log.apply(window.console, Array.prototype.slice.call(arguments));
2256
+ }
2257
+ };
2258
+
2259
+ /*
2260
+ * Add util object to allow access to private methods for testability.
2261
+ */
2262
+ $.timepicker._util = {
2263
+ _extendRemove: extendRemove,
2264
+ _isEmptyObject: isEmptyObject,
2265
+ _convert24to12: convert24to12,
2266
+ _detectSupport: detectSupport,
2267
+ _selectLocalTimezone: selectLocalTimezone,
2268
+ _computeEffectiveSetting: computeEffectiveSetting,
2269
+ _splitDateTime: splitDateTime,
2270
+ _parseDateTimeInternal: parseDateTimeInternal
2271
+ };
2272
+
2273
+ /*
2274
+ * Microsecond support
2275
+ */
2276
+ if (!Date.prototype.getMicroseconds) {
2277
+ Date.prototype.microseconds = 0;
2278
+ Date.prototype.getMicroseconds = function () { return this.microseconds; };
2279
+ Date.prototype.setMicroseconds = function (m) {
2280
+ this.setMilliseconds(this.getMilliseconds() + Math.floor(m / 1000));
2281
+ this.microseconds = m % 1000;
2282
+ return this;
2283
+ };
1306
2284
  }
1307
- if (value === undefined)
1308
- return this._base_optionDatepicker(target, name);
1309
- return this._base_optionDatepicker(target, name, value);
1310
- };
1311
-
1312
- //#######################################################################################
1313
- // jQuery extend now ignores nulls!
1314
- //#######################################################################################
1315
- function extendRemove(target, props) {
1316
- $.extend(target, props);
1317
- for (var name in props)
1318
- if (props[name] === null || props[name] === undefined)
1319
- target[name] = props[name];
1320
- return target;
1321
- };
1322
-
1323
- $.timepicker = new Timepicker(); // singleton instance
1324
- $.timepicker.version = "1.0.0";
1325
-
1326
- })(jQuery);
2285
+
2286
+ /*
2287
+ * Keep up with the version
2288
+ */
2289
+ $.timepicker.version = "1.6.3";
2290
+
2291
+ }));