active_scaffold-sequel 0.7.1 → 0.8.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.
Files changed (36) hide show
  1. data/app/assets/javascripts/active_scaffold.js.erb +12 -14
  2. data/app/assets/javascripts/jquery/active_scaffold.js +610 -348
  3. data/app/assets/javascripts/jquery/date_picker_bridge.js.erb +37 -19
  4. data/app/assets/javascripts/jquery/draggable_lists.js +28 -27
  5. data/app/assets/javascripts/jquery/jquery.editinplace.js +145 -137
  6. data/app/assets/javascripts/jquery/tiny_mce_bridge.js +1 -1
  7. data/app/assets/stylesheets/{active_scaffold.css.scss → active_scaffold.scss} +0 -0
  8. data/app/assets/stylesheets/{active_scaffold_colors.css.scss → active_scaffold_colors.scss} +0 -0
  9. data/app/assets/stylesheets/{active_scaffold_images.css.scss → active_scaffold_images.scss} +0 -0
  10. data/frontends/default/views/_list.html.erb +1 -1
  11. data/frontends/default/views/_list_record.html.erb +2 -1
  12. data/frontends/default/views/_search.html.erb +1 -10
  13. data/frontends/default/views/add_existing.js.erb +0 -3
  14. data/lib/active_scaffold.rb +0 -12
  15. data/lib/active_scaffold/actions/list.rb +8 -3
  16. data/lib/active_scaffold/bridges/carrierwave/form_ui.rb +3 -8
  17. data/lib/active_scaffold/bridges/date_picker.rb +2 -3
  18. data/lib/active_scaffold/bridges/dragonfly/form_ui.rb +3 -7
  19. data/lib/active_scaffold/bridges/tiny_mce.rb +1 -5
  20. data/lib/active_scaffold/bridges/tiny_mce/helpers.rb +1 -5
  21. data/lib/active_scaffold/extensions/action_view_rendering.rb +1 -6
  22. data/lib/active_scaffold/extensions/routing_mapper.rb +1 -1
  23. data/lib/active_scaffold/helpers/controller_helpers.rb +2 -2
  24. data/lib/active_scaffold/helpers/form_column_helpers.rb +3 -10
  25. data/lib/active_scaffold/version.rb +2 -2
  26. data/vendor/assets/javascripts/getprototypeof.js +12 -0
  27. data/vendor/assets/javascripts/jquery-ui-timepicker-addon.js +1882 -1276
  28. metadata +71 -65
  29. checksums.yaml +0 -7
  30. data/app/assets/javascripts/prototype/active_scaffold.js +0 -1103
  31. data/app/assets/javascripts/prototype/dhtml_history.js +0 -870
  32. data/app/assets/javascripts/prototype/form_enhancements.js +0 -117
  33. data/app/assets/javascripts/prototype/rico_corner.js +0 -370
  34. data/app/assets/javascripts/prototype/tiny_mce_bridge.js +0 -7
  35. data/lib/active_scaffold/bridges/calendar_date_select.rb +0 -24
  36. data/lib/active_scaffold/bridges/calendar_date_select/as_cds_bridge.rb +0 -67
@@ -1,7 +1,7 @@
1
1
  var action_link_close = ActiveScaffold.ActionLink.Abstract.prototype.close;
2
2
  ActiveScaffold.ActionLink.Abstract.prototype.close = function() {
3
3
  jQuery(this.adapter).find('textarea.mceEditor').each(function(index, elem) {
4
- tinyMCE.execCommand('mceRemoveControl', false, jQuery(elem).attr('id'));
4
+ tinyMCE.execCommand('mceRemoveEditor', false, jQuery(elem).attr('id'));
5
5
  });
6
6
  action_link_close.apply(this);
7
7
  };
@@ -6,7 +6,7 @@
6
6
  </tr>
7
7
  </thead>
8
8
  <%= render :partial => 'list_messages', :locals => {:columns => columns} %>
9
- <tbody class="records" id="<%= active_scaffold_tbody_id %>">
9
+ <tbody class="records" id="<%= active_scaffold_tbody_id %>" data-refresh-record="<%= url_for(params_for(:action => :index, :id => '--ID--')) %>">
10
10
  <% if !@records.empty? -%>
11
11
  <%= render :partial => 'list_record', :collection => @page.items, :locals => { :hidden => false, :columns => columns, :action_links => active_scaffold_config.action_links.member} %>
12
12
  <% end -%>
@@ -5,9 +5,10 @@ tr_class = cycle("", "even-record")
5
5
  tr_class += " #{list_row_class(record)}" if respond_to? :list_row_class
6
6
  url_options = params_for(:action => :list, :id => record.id)
7
7
  action_links ||= active_scaffold_config.action_links.member
8
+ data_refresh ||= record.id
8
9
  -%>
9
10
 
10
- <tr class="record <%= tr_class %>" id="<%= element_row_id(:action => :list, :id => record.id) %>" data-refresh="<%= url_for(params_for(:action => :row, :id => record.id, :_method => :get)).html_safe %>">
11
+ <tr class="record <%= tr_class %>" id="<%= element_row_id(:action => :list, :id => record.id) %>" data-refresh="<%= data_refresh %>">
11
12
  <%= render :partial => 'list_record_columns', :locals => {:record => record, :columns => columns} %>
12
13
  <%= render :partial => 'list_actions', :locals => {:record => record, :url_options => url_options, :action_links => action_links} unless action_links.empty? %>
13
14
  </tr>
@@ -15,16 +15,7 @@ options['data-loading'] = true unless live_search
15
15
 
16
16
  <script type="text/javascript">
17
17
  //<![CDATA[
18
- <% if ActiveScaffold.js_framework == :prototype %>
19
- new TextFieldWithExample('<%= search_input_id %>', '<%= as_(live_search ? :live_search : :search_terms) %>', {focus: true});
20
- <% end -%>
21
- <% if live_search && ActiveScaffold.js_framework == :prototype -%>
22
- $(<%= search_input_id.to_json.html_safe %>).next().hide();
23
- new Form.Element.DelayedObserver('<%= search_input_id %>', 0.5, function(element, value) {
24
- if (!$(element.id)) return false; // because the element may have been destroyed
25
- $(element).next().click();
26
- });
27
- <% elsif live_search && ActiveScaffold.js_framework == :jquery %>
18
+ <% if live_search %>
28
19
  jQuery(<%= "##{search_input_id}".to_json.html_safe %>).next().hide();
29
20
  jQuery(<%= "##{search_input_id}".to_json.html_safe %>).delayedObserver(0.5, function() {
30
21
  jQuery(<%= "##{search_input_id}".to_json.html_safe %>).parent().trigger("submit");});
@@ -10,9 +10,6 @@ ActiveScaffold.create_record_row('<%= active_scaffold_id %>', '<%= escape_javasc
10
10
  ActiveScaffold.reset_form('<%= element_form_id %>');
11
11
  ActiveScaffold.replace_html('<%= element_messages_id(:action => :add_existing) %>', '<%= escape_javascript(render(:partial => 'form_messages')) %>');
12
12
  <%# have to delay the focus, because there's no "firstElement" in prototype until at least one element is not disabled%>
13
- <% if ActiveScaffold.js_framework == :prototype %>
14
- ActiveScaffold.focus_first_element_of_form.defer('<%= element_form_id %>');
15
- <% end %>
16
13
  <% else %>
17
14
  ActiveScaffold.find_action_link('<%= element_form_id(:action => :new_existing) %>').close();
18
15
  <% end %>
@@ -114,18 +114,6 @@ module ActiveScaffold
114
114
  session[:hover_supported] == false
115
115
  end
116
116
 
117
- def self.js_framework=(framework)
118
- @@js_framework = framework
119
- end
120
-
121
- def self.js_framework
122
- @@js_framework ||= if defined? Jquery
123
- :jquery
124
- elsif defined? PrototypeRails
125
- :prototype
126
- end
127
- end
128
-
129
117
  def self.js_config=(config)
130
118
  @@js_config = config
131
119
  end
@@ -1,14 +1,20 @@
1
1
  module ActiveScaffold::Actions
2
2
  module List
3
3
  def self.included(base)
4
- base.before_filter :list_authorized_filter, :only => [:index, :row]
4
+ base.before_filter :list_authorized_filter, :only => [:index]
5
5
  base.helper_method :list_columns
6
6
  end
7
7
 
8
8
  def index
9
- list
9
+ if params[:id] && !params[:id].is_a?(Array) && request.xhr?
10
+ row
11
+ else
12
+ list
13
+ end
10
14
  end
11
15
 
16
+ protected
17
+
12
18
  # get just a single row
13
19
  def row
14
20
  @record = find_if_allowed(params[:id], :read)
@@ -20,7 +26,6 @@ module ActiveScaffold::Actions
20
26
  respond_to_action(:list)
21
27
  end
22
28
 
23
- protected
24
29
  def list_respond_to_html
25
30
  if params.delete(:embedded)
26
31
  render :action => 'list', :layout => false
@@ -17,13 +17,8 @@ module ActiveScaffold
17
17
  :id => options[:id] + '_cache'
18
18
  }
19
19
 
20
- if ActiveScaffold.js_framework == :jquery
21
- js_remove_file_code = "$(this).prev('input#remove_#{options[:id]}').val('true'); $(this).parent().hide().next().show(); return false;";
22
- js_dont_remove_file_code = "$(this).parents('div.carrierwave_controls').find('input#remove_#{options[:id]}').val('false'); return false;";
23
- else
24
- js_remove_file_code = "$(this).previous('input#remove_#{options[:id]}').value='true'; $(this).up().hide().next().show(); return false;";
25
- js_dont_remove_file_code = "$(this).up('div.carrierwave_controls').down('input#remove_#{options[:id]}').value='false'; return false;";
26
- end
20
+ js_remove_file_code = "jQuery(this).prev('input#remove_#{options[:id]}').val('true'); jQuery(this).parent().hide().next().show(); return false;";
21
+ js_dont_remove_file_code = "jQuery(this).parents('div.carrierwave_controls').find('input#remove_#{options[:id]}').val('false'); return false;";
27
22
 
28
23
  input = file_field(:record, column.name, options.merge(:onchange => js_dont_remove_file_code))
29
24
  content_tag( :div,
@@ -42,4 +37,4 @@ module ActiveScaffold
42
37
  end
43
38
  end
44
39
  end
45
- end
40
+ end
@@ -1,12 +1,11 @@
1
1
  module ActiveScaffold::Bridges
2
2
  class DatePicker < ActiveScaffold::DataStructures::Bridge
3
3
  autoload :Helper, 'active_scaffold/bridges/date_picker/helper'
4
+
4
5
  def self.install
5
6
  require File.join(File.dirname(__FILE__), "date_picker/ext.rb")
6
7
  end
7
- def self.install?
8
- ActiveScaffold.js_framework == :jquery
9
- end
8
+
10
9
  def self.localization
11
10
  "jQuery(function($){
12
11
  if (typeof($.datepicker) === 'object') {
@@ -6,12 +6,8 @@ module ActiveScaffold
6
6
  input = file_field(:record, column.name, options)
7
7
  dragonfly = @record.send("#{column.name}")
8
8
  if dragonfly.present?
9
- if ActiveScaffold.js_framework == :jquery
10
- js_remove_file_code = "$(this).prev().val('true'); $(this).parent().hide().next().show(); return false;";
11
- else
12
- js_remove_file_code = "$(this).previous().value='true'; $(this).up().hide().next().show(); return false;";
13
- end
14
-
9
+ js_remove_file_code = "jQuery(this).prev().val('true'); jQuery(this).parent().hide().next().show(); return false;";
10
+
15
11
  content = active_scaffold_column_dragonfly(column, @record)
16
12
  content_tag(:div,
17
13
  content + " | " +
@@ -24,4 +20,4 @@ module ActiveScaffold
24
20
  end
25
21
  end
26
22
  end
27
- end
23
+ end
@@ -8,10 +8,6 @@ class ActiveScaffold::Bridges::TinyMce < ActiveScaffold::DataStructures::Bridge
8
8
  end
9
9
 
10
10
  def self.javascripts
11
- if ActiveScaffold.js_framework == :jquery
12
- ['tinymce-jquery', 'jquery/tiny_mce_bridge']
13
- else
14
- ['tinymce', 'prototype/tiny_mce_bridge']
15
- end
11
+ ['tinymce-jquery', 'jquery/tiny_mce_bridge']
16
12
  end
17
13
  end
@@ -26,11 +26,7 @@ class ActiveScaffold::Bridges::TinyMce
26
26
  end
27
27
 
28
28
  def onsubmit_with_tiny_mce
29
- if ActiveScaffold.js_framework == :jquery
30
- submit_js = 'tinyMCE.triggerSave();$(\'textarea.mceEditor\').each(function(index, elem) { tinyMCE.execCommand(\'mceRemoveControl\', false, $(elem).attr(\'id\')); });'
31
- else
32
- submit_js = 'tinyMCE.triggerSave();this.select(\'textarea.mceEditor\').each(function(elem) { tinyMCE.execCommand(\'mceRemoveControl\', false, elem.id); });'
33
- end
29
+ submit_js = 'tinyMCE.triggerSave();jQuery(\'textarea.mceEditor\').each(function(index, elem) { tinyMCE.execCommand(\'mceRemoveControl\', false, jQuery(elem).attr(\'id\')); });'
34
30
  [onsubmit_without_tiny_mce, submit_js].compact.join ';'
35
31
  end
36
32
  end
@@ -80,12 +80,7 @@ module ActionView::Helpers #:nodoc:
80
80
  model = remote_controller.to_s.sub(/.*\//, '').singularize
81
81
  content_tag(:div, :class => 'active-scaffold-header') do
82
82
  content_tag :h2, link_to(args.first[:label] || active_scaffold_config_for(model).list.label, url, :remote => true)
83
- end <<
84
- if ActiveScaffold.js_framework == :prototype
85
- javascript_tag("new Ajax.Updater('#{id}', '#{url}', {method: 'get', evalScripts: true});")
86
- elsif ActiveScaffold.js_framework == :jquery
87
- javascript_tag("jQuery('##{id}').load('#{url}');")
88
- end
83
+ end << javascript_tag("jQuery('##{id}').load('#{url}');")
89
84
  end
90
85
  else
91
86
  options = args.first
@@ -2,7 +2,7 @@ module ActionDispatch
2
2
  module Routing
3
3
  ACTIVE_SCAFFOLD_CORE_ROUTING = {
4
4
  :collection => {:show_search => :get, :render_field => :get},
5
- :member => {:row => :get, :update_column => :post, :render_field => :get}
5
+ :member => {:update_column => :post, :render_field => :get}
6
6
  }
7
7
  ACTIVE_SCAFFOLD_ASSOCIATION_ROUTING = {
8
8
  :collection => {:edit_associated => :get, :new_existing => :get, :add_existing => :post},
@@ -57,13 +57,13 @@ module ActiveScaffold
57
57
 
58
58
  def render_parent_options
59
59
  if nested_singular_association?
60
- {:controller => nested.parent_scaffold.controller_path, :action => :row, :id => nested.parent_id}
60
+ {:controller => nested.parent_scaffold.controller_path, :action => :index, :id => nested.parent_id}
61
61
  elsif params[:parent_sti]
62
62
  options = {:controller => params[:parent_sti], :action => render_parent_action(params[:parent_sti])}
63
63
  if render_parent_action(params[:parent_sti]) == :index
64
64
  options.merge(params.slice(:eid))
65
65
  else
66
- options.merge({:id => @record.id})
66
+ options.merge({:action => :index, :id => @record.id})
67
67
  end
68
68
  end
69
69
  end
@@ -298,16 +298,9 @@ module ActiveScaffold
298
298
  end
299
299
 
300
300
  def active_scaffold_add_existing_input(options)
301
- if ActiveScaffold.js_framework == :prototype && controller.respond_to?(:record_select_config)
302
- remote_controller = active_scaffold_controller_for(record_select_config.model).controller_path
303
- options.merge!(:controller => remote_controller)
304
- options.merge!(active_scaffold_input_text_options)
305
- record_select_field(options[:name], @record, options)
306
- else
307
- select_options = options_for_select(options_for_association(nested.association)) #unless column.through_association?
308
- select_options ||= options_for_select(active_scaffold_config.model.all.collect {|c| [h(c.to_label), c.id]})
309
- select_tag 'associated_id', ('<option value="">' + as_(:_select_) + '</option>' + select_options).html_safe unless select_options.empty?
310
- end
301
+ select_options = options_for_select(options_for_association(nested.association)) #unless column.through_association?
302
+ select_options ||= options_for_select(active_scaffold_config.model.all.collect {|c| [h(c.to_label), c.id]})
303
+ select_tag 'associated_id', ('<option value="">' + as_(:_select_) + '</option>' + select_options).html_safe unless select_options.empty?
311
304
  end
312
305
 
313
306
  def active_scaffold_add_existing_label
@@ -1,8 +1,8 @@
1
1
  module ActiveScaffold
2
2
  module Version
3
3
  MAJOR = 0
4
- MINOR = 7
5
- PATCH = 1
4
+ MINOR = 8
5
+ PATCH = 0
6
6
 
7
7
  STRING = [MAJOR, MINOR, PATCH].compact.join('.')
8
8
  end
@@ -0,0 +1,12 @@
1
+ if ( typeof Object.getPrototypeOf !== "function" ) {
2
+ if ( typeof "test".__proto__ === "object" ) {
3
+ Object.getPrototypeOf = function(object){
4
+ return object.__proto__;
5
+ };
6
+ } else {
7
+ Object.getPrototypeOf = function(object){
8
+ // May break if the constructor has been tampered with
9
+ return object.constructor.prototype;
10
+ };
11
+ }
12
+ }
@@ -1,1276 +1,1882 @@
1
- /*
2
- * jQuery timepicker addon
3
- * By: Trent Richardson [http://trentrichardson.com]
4
- * Version 0.9.7
5
- * Last Modified: 10/02/2011
6
- *
7
- * Copyright 2011 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; }
16
- * .ui-timepicker-div dl dd { margin: -25px 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
- $.extend($.ui, { timepicker: { version: "0.9.7" } });
24
-
25
- /* Time picker manager.
26
- Use the singleton instance of this class, $.timepicker, to interact with the time picker.
27
- Settings for (groups of) time pickers are maintained in an instance object,
28
- allowing multiple different settings on the same page. */
29
-
30
- function Timepicker() {
31
- this.regional = []; // Available regional settings, indexed by language code
32
- this.regional[''] = { // Default regional settings
33
- currentText: 'Now',
34
- closeText: 'Done',
35
- ampm: false,
36
- amNames: ['AM', 'A'],
37
- pmNames: ['PM', 'P'],
38
- timeFormat: 'hh:mm tt',
39
- timeSuffix: '',
40
- timeOnlyTitle: 'Choose Time',
41
- timeText: 'Time',
42
- hourText: 'Hour',
43
- minuteText: 'Minute',
44
- secondText: 'Second',
45
- millisecText: 'Millisecond',
46
- timezoneText: 'Time Zone'
47
- };
48
- this._defaults = { // Global defaults for all the datetime picker instances
49
- showButtonPanel: true,
50
- timeOnly: false,
51
- showHour: true,
52
- showMinute: true,
53
- showSecond: false,
54
- showMillisec: false,
55
- showTimezone: false,
56
- showTime: true,
57
- stepHour: 0.05,
58
- stepMinute: 0.05,
59
- stepSecond: 0.05,
60
- stepMillisec: 0.5,
61
- hour: 0,
62
- minute: 0,
63
- second: 0,
64
- millisec: 0,
65
- timezone: '+0000',
66
- hourMin: 0,
67
- minuteMin: 0,
68
- secondMin: 0,
69
- millisecMin: 0,
70
- hourMax: 23,
71
- minuteMax: 59,
72
- secondMax: 59,
73
- millisecMax: 999,
74
- minDateTime: null,
75
- maxDateTime: null,
76
- onSelect: null,
77
- hourGrid: 0,
78
- minuteGrid: 0,
79
- secondGrid: 0,
80
- millisecGrid: 0,
81
- alwaysSetTime: true,
82
- separator: ' ',
83
- altFieldTimeOnly: true,
84
- showTimepicker: true,
85
- timezoneIso8609: false,
86
- timezoneList: null
87
- };
88
- $.extend(this._defaults, this.regional['']);
89
- }
90
-
91
- $.extend(Timepicker.prototype, {
92
- $input: null,
93
- $altInput: null,
94
- $timeObj: null,
95
- inst: null,
96
- hour_slider: null,
97
- minute_slider: null,
98
- second_slider: null,
99
- millisec_slider: null,
100
- timezone_select: null,
101
- hour: 0,
102
- minute: 0,
103
- second: 0,
104
- millisec: 0,
105
- timezone: '+0000',
106
- hourMinOriginal: null,
107
- minuteMinOriginal: null,
108
- secondMinOriginal: null,
109
- millisecMinOriginal: null,
110
- hourMaxOriginal: null,
111
- minuteMaxOriginal: null,
112
- secondMaxOriginal: null,
113
- millisecMaxOriginal: null,
114
- ampm: '',
115
- formattedDate: '',
116
- formattedTime: '',
117
- formattedDateTime: '',
118
- timezoneList: null,
119
-
120
- /* Override the default settings for all instances of the time picker.
121
- @param settings object - the new settings to use as defaults (anonymous object)
122
- @return the manager object */
123
- setDefaults: function(settings) {
124
- extendRemove(this._defaults, settings || {});
125
- return this;
126
- },
127
-
128
- //########################################################################
129
- // Create a new Timepicker instance
130
- //########################################################################
131
- _newInst: function($input, o) {
132
- var tp_inst = new Timepicker(),
133
- inlineSettings = {};
134
-
135
- for (var attrName in this._defaults) {
136
- var attrValue = $input.attr('time:' + attrName);
137
- if (attrValue) {
138
- try {
139
- inlineSettings[attrName] = eval(attrValue);
140
- } catch (err) {
141
- inlineSettings[attrName] = attrValue;
142
- }
143
- }
144
- }
145
- tp_inst._defaults = $.extend({}, this._defaults, inlineSettings, o, {
146
- beforeShow: function(input, dp_inst) {
147
- if ($.isFunction(o.beforeShow))
148
- o.beforeShow(input, dp_inst, tp_inst);
149
- },
150
- onChangeMonthYear: function(year, month, dp_inst) {
151
- // Update the time as well : this prevents the time from disappearing from the $input field.
152
- tp_inst._updateDateTime(dp_inst);
153
- if ($.isFunction(o.onChangeMonthYear))
154
- o.onChangeMonthYear.call($input[0], year, month, dp_inst, tp_inst);
155
- },
156
- onClose: function(dateText, dp_inst) {
157
- if (tp_inst.timeDefined === true && $input.val() != '')
158
- tp_inst._updateDateTime(dp_inst);
159
- if ($.isFunction(o.onClose))
160
- o.onClose.call($input[0], dateText, dp_inst, tp_inst);
161
- },
162
- timepicker: tp_inst // add timepicker as a property of datepicker: $.datepicker._get(dp_inst, 'timepicker');
163
- });
164
- tp_inst.amNames = $.map(tp_inst._defaults.amNames, function(val) { return val.toUpperCase() });
165
- tp_inst.pmNames = $.map(tp_inst._defaults.pmNames, function(val) { return val.toUpperCase() });
166
-
167
- if (tp_inst._defaults.timezoneList === null) {
168
- var timezoneList = [];
169
- for (var i = -11; i <= 12; i++)
170
- timezoneList.push((i >= 0 ? '+' : '-') + ('0' + Math.abs(i).toString()).slice(-2) + '00');
171
- if (tp_inst._defaults.timezoneIso8609)
172
- timezoneList = $.map(timezoneList, function(val) {
173
- return val == '+0000' ? 'Z' : (val.substring(0, 3) + ':' + val.substring(3));
174
- });
175
- tp_inst._defaults.timezoneList = timezoneList;
176
- }
177
-
178
- tp_inst.hour = tp_inst._defaults.hour;
179
- tp_inst.minute = tp_inst._defaults.minute;
180
- tp_inst.second = tp_inst._defaults.second;
181
- tp_inst.millisec = tp_inst._defaults.millisec;
182
- tp_inst.ampm = '';
183
- tp_inst.$input = $input;
184
-
185
- if (o.altField)
186
- tp_inst.$altInput = $(o.altField)
187
- .css({ cursor: 'pointer' })
188
- .focus(function(){ $input.trigger("focus"); });
189
-
190
- if(tp_inst._defaults.minDate==0 || tp_inst._defaults.minDateTime==0)
191
- {
192
- tp_inst._defaults.minDate=new Date();
193
- }
194
- if(tp_inst._defaults.maxDate==0 || tp_inst._defaults.maxDateTime==0)
195
- {
196
- tp_inst._defaults.maxDate=new Date();
197
- }
198
-
199
- // datepicker needs minDate/maxDate, timepicker needs minDateTime/maxDateTime..
200
- if(tp_inst._defaults.minDate !== undefined && tp_inst._defaults.minDate instanceof Date)
201
- tp_inst._defaults.minDateTime = new Date(tp_inst._defaults.minDate.getTime());
202
- if(tp_inst._defaults.minDateTime !== undefined && tp_inst._defaults.minDateTime instanceof Date)
203
- tp_inst._defaults.minDate = new Date(tp_inst._defaults.minDateTime.getTime());
204
- if(tp_inst._defaults.maxDate !== undefined && tp_inst._defaults.maxDate instanceof Date)
205
- tp_inst._defaults.maxDateTime = new Date(tp_inst._defaults.maxDate.getTime());
206
- if(tp_inst._defaults.maxDateTime !== undefined && tp_inst._defaults.maxDateTime instanceof Date)
207
- tp_inst._defaults.maxDate = new Date(tp_inst._defaults.maxDateTime.getTime());
208
- return tp_inst;
209
- },
210
-
211
- //########################################################################
212
- // add our sliders to the calendar
213
- //########################################################################
214
- _addTimePicker: function(dp_inst) {
215
- var currDT = (this.$altInput && this._defaults.altFieldTimeOnly) ?
216
- this.$input.val() + ' ' + this.$altInput.val() :
217
- this.$input.val();
218
-
219
- this.timeDefined = this._parseTime(currDT);
220
- this._limitMinMaxDateTime(dp_inst, false);
221
- this._injectTimePicker();
222
- },
223
-
224
- //########################################################################
225
- // parse the time string from input value or _setTime
226
- //########################################################################
227
- _parseTime: function(timeString, withDate) {
228
- var regstr = this._defaults.timeFormat.toString()
229
- .replace(/h{1,2}/ig, '(\\d?\\d)')
230
- .replace(/m{1,2}/ig, '(\\d?\\d)')
231
- .replace(/s{1,2}/ig, '(\\d?\\d)')
232
- .replace(/l{1}/ig, '(\\d?\\d?\\d)')
233
- .replace(/t{1,2}/ig, this._getPatternAmpm())
234
- .replace(/z{1}/ig, '(z|[-+]\\d\\d:?\\d\\d)?')
235
- .replace(/\s/g, '\\s?') + this._defaults.timeSuffix + '$',
236
- order = this._getFormatPositions(),
237
- ampm = '',
238
- treg;
239
-
240
- if (!this.inst) this.inst = $.datepicker._getInst(this.$input[0]);
241
-
242
- if (withDate || !this._defaults.timeOnly) {
243
- // the time should come after x number of characters and a space.
244
- // x = at least the length of text specified by the date format
245
- var dp_dateFormat = $.datepicker._get(this.inst, 'dateFormat');
246
- // escape special regex characters in the seperator
247
- var specials = new RegExp("[.*+?|()\\[\\]{}\\\\]", "g");
248
- regstr = '.{' + dp_dateFormat.length + ',}' + this._defaults.separator.replace(specials, "\\$&") + regstr;
249
- }
250
-
251
- treg = timeString.match(new RegExp(regstr, 'i'));
252
-
253
- if (treg) {
254
- if (order.t !== -1) {
255
- if (treg[order.t] === undefined || treg[order.t].length === 0) {
256
- ampm = '';
257
- this.ampm = '';
258
- } else {
259
- ampm = $.inArray(treg[order.t].toUpperCase(), this.amNames) !== -1 ? 'AM' : 'PM';
260
- this.ampm = this._defaults[ampm == 'AM' ? 'amNames' : 'pmNames'][0];
261
- }
262
- }
263
-
264
- if (order.h !== -1) {
265
- if (ampm == 'AM' && treg[order.h] == '12')
266
- this.hour = 0; // 12am = 0 hour
267
- else if (ampm == 'PM' && treg[order.h] != '12')
268
- this.hour = (parseFloat(treg[order.h]) + 12).toFixed(0); // 12pm = 12 hour, any other pm = hour + 12
269
- else this.hour = Number(treg[order.h]);
270
- }
271
-
272
- if (order.m !== -1) this.minute = Number(treg[order.m]);
273
- if (order.s !== -1) this.second = Number(treg[order.s]);
274
- if (order.l !== -1) this.millisec = Number(treg[order.l]);
275
- if (order.z !== -1 && treg[order.z] !== undefined) {
276
- var tz = treg[order.z].toUpperCase();
277
- switch (tz.length) {
278
- case 1: // Z
279
- tz = this._defaults.timezoneIso8609 ? 'Z' : '+0000';
280
- break;
281
- case 5: // +hhmm
282
- if (this._defaults.timezoneIso8609)
283
- tz = tz.substring(1) == '0000'
284
- ? 'Z'
285
- : tz.substring(0, 3) + ':' + tz.substring(3);
286
- break;
287
- case 6: // +hh:mm
288
- if (!this._defaults.timezoneIso8609)
289
- tz = tz == 'Z' || tz.substring(1) == '00:00'
290
- ? '+0000'
291
- : tz.replace(/:/, '');
292
- else if (tz.substring(1) == '00:00')
293
- tz = 'Z';
294
- break;
295
- }
296
- this.timezone = tz;
297
- }
298
-
299
- return true;
300
-
301
- }
302
- return false;
303
- },
304
-
305
- //########################################################################
306
- // pattern for standard and localized AM/PM markers
307
- //########################################################################
308
- _getPatternAmpm: function() {
309
- var markers = [];
310
- o = this._defaults;
311
- if (o.amNames)
312
- $.merge(markers, o.amNames);
313
- if (o.pmNames)
314
- $.merge(markers, o.pmNames);
315
- markers = $.map(markers, function(val) { return val.replace(/[.*+?|()\[\]{}\\]/g, '\\$&') });
316
- return '(' + markers.join('|') + ')?';
317
- },
318
-
319
- //########################################################################
320
- // figure out position of time elements.. cause js cant do named captures
321
- //########################################################################
322
- _getFormatPositions: function() {
323
- var finds = this._defaults.timeFormat.toLowerCase().match(/(h{1,2}|m{1,2}|s{1,2}|l{1}|t{1,2}|z)/g),
324
- orders = { h: -1, m: -1, s: -1, l: -1, t: -1, z: -1 };
325
-
326
- if (finds)
327
- for (var i = 0; i < finds.length; i++)
328
- if (orders[finds[i].toString().charAt(0)] == -1)
329
- orders[finds[i].toString().charAt(0)] = i + 1;
330
-
331
- return orders;
332
- },
333
-
334
- //########################################################################
335
- // generate and inject html for timepicker into ui datepicker
336
- //########################################################################
337
- _injectTimePicker: function() {
338
- var $dp = this.inst.dpDiv,
339
- o = this._defaults,
340
- tp_inst = this,
341
- // Added by Peter Medeiros:
342
- // - Figure out what the hour/minute/second max should be based on the step values.
343
- // - Example: if stepMinute is 15, then minMax is 45.
344
- hourMax = (o.hourMax - ((o.hourMax - o.hourMin) % o.stepHour)).toFixed(0),
345
- minMax = (o.minuteMax - ((o.minuteMax - o.minuteMin) % o.stepMinute)).toFixed(0),
346
- secMax = (o.secondMax - ((o.secondMax - o.secondMin) % o.stepSecond)).toFixed(0),
347
- millisecMax = (o.millisecMax - ((o.millisecMax - o.millisecMin) % o.stepMillisec)).toFixed(0),
348
- dp_id = this.inst.id.toString().replace(/([^A-Za-z0-9_])/g, '');
349
-
350
- // Prevent displaying twice
351
- //if ($dp.find("div#ui-timepicker-div-"+ dp_id).length === 0) {
352
- if ($dp.find("div#ui-timepicker-div-"+ dp_id).length === 0 && o.showTimepicker) {
353
- var noDisplay = ' style="display:none;"',
354
- html = '<div class="ui-timepicker-div" id="ui-timepicker-div-' + dp_id + '"><dl>' +
355
- '<dt class="ui_tpicker_time_label" id="ui_tpicker_time_label_' + dp_id + '"' +
356
- ((o.showTime) ? '' : noDisplay) + '>' + o.timeText + '</dt>' +
357
- '<dd class="ui_tpicker_time" id="ui_tpicker_time_' + dp_id + '"' +
358
- ((o.showTime) ? '' : noDisplay) + '></dd>' +
359
- '<dt class="ui_tpicker_hour_label" id="ui_tpicker_hour_label_' + dp_id + '"' +
360
- ((o.showHour) ? '' : noDisplay) + '>' + o.hourText + '</dt>',
361
- hourGridSize = 0,
362
- minuteGridSize = 0,
363
- secondGridSize = 0,
364
- millisecGridSize = 0,
365
- size;
366
-
367
- // Hours
368
- if (o.showHour && o.hourGrid > 0) {
369
- html += '<dd class="ui_tpicker_hour">' +
370
- '<div id="ui_tpicker_hour_' + dp_id + '"' + ((o.showHour) ? '' : noDisplay) + '></div>' +
371
- '<div style="padding-left: 1px"><table class="ui-tpicker-grid-label"><tr>';
372
-
373
- for (var h = o.hourMin; h <= hourMax; h += parseInt(o.hourGrid,10)) {
374
- hourGridSize++;
375
- var tmph = (o.ampm && h > 12) ? h-12 : h;
376
- if (tmph < 10) tmph = '0' + tmph;
377
- if (o.ampm) {
378
- if (h == 0) tmph = 12 +'a';
379
- else if (h < 12) tmph += 'a';
380
- else tmph += 'p';
381
- }
382
- html += '<td>' + tmph + '</td>';
383
- }
384
-
385
- html += '</tr></table></div>' +
386
- '</dd>';
387
- } else html += '<dd class="ui_tpicker_hour" id="ui_tpicker_hour_' + dp_id + '"' +
388
- ((o.showHour) ? '' : noDisplay) + '></dd>';
389
-
390
- html += '<dt class="ui_tpicker_minute_label" id="ui_tpicker_minute_label_' + dp_id + '"' +
391
- ((o.showMinute) ? '' : noDisplay) + '>' + o.minuteText + '</dt>';
392
-
393
- // Minutes
394
- if (o.showMinute && o.minuteGrid > 0) {
395
- html += '<dd class="ui_tpicker_minute ui_tpicker_minute_' + o.minuteGrid + '">' +
396
- '<div id="ui_tpicker_minute_' + dp_id + '"' +
397
- ((o.showMinute) ? '' : noDisplay) + '></div>' +
398
- '<div style="padding-left: 1px"><table class="ui-tpicker-grid-label"><tr>';
399
-
400
- for (var m = o.minuteMin; m <= minMax; m += parseInt(o.minuteGrid,10)) {
401
- minuteGridSize++;
402
- html += '<td>' + ((m < 10) ? '0' : '') + m + '</td>';
403
- }
404
-
405
- html += '</tr></table></div>' +
406
- '</dd>';
407
- } else html += '<dd class="ui_tpicker_minute" id="ui_tpicker_minute_' + dp_id + '"' +
408
- ((o.showMinute) ? '' : noDisplay) + '></dd>';
409
-
410
- // Seconds
411
- html += '<dt class="ui_tpicker_second_label" id="ui_tpicker_second_label_' + dp_id + '"' +
412
- ((o.showSecond) ? '' : noDisplay) + '>' + o.secondText + '</dt>';
413
-
414
- if (o.showSecond && o.secondGrid > 0) {
415
- html += '<dd class="ui_tpicker_second ui_tpicker_second_' + o.secondGrid + '">' +
416
- '<div id="ui_tpicker_second_' + dp_id + '"' +
417
- ((o.showSecond) ? '' : noDisplay) + '></div>' +
418
- '<div style="padding-left: 1px"><table><tr>';
419
-
420
- for (var s = o.secondMin; s <= secMax; s += parseInt(o.secondGrid,10)) {
421
- secondGridSize++;
422
- html += '<td>' + ((s < 10) ? '0' : '') + s + '</td>';
423
- }
424
-
425
- html += '</tr></table></div>' +
426
- '</dd>';
427
- } else html += '<dd class="ui_tpicker_second" id="ui_tpicker_second_' + dp_id + '"' +
428
- ((o.showSecond) ? '' : noDisplay) + '></dd>';
429
-
430
- // Milliseconds
431
- html += '<dt class="ui_tpicker_millisec_label" id="ui_tpicker_millisec_label_' + dp_id + '"' +
432
- ((o.showMillisec) ? '' : noDisplay) + '>' + o.millisecText + '</dt>';
433
-
434
- if (o.showMillisec && o.millisecGrid > 0) {
435
- html += '<dd class="ui_tpicker_millisec ui_tpicker_millisec_' + o.millisecGrid + '">' +
436
- '<div id="ui_tpicker_millisec_' + dp_id + '"' +
437
- ((o.showMillisec) ? '' : noDisplay) + '></div>' +
438
- '<div style="padding-left: 1px"><table><tr>';
439
-
440
- for (var l = o.millisecMin; l <= millisecMax; l += parseInt(o.millisecGrid,10)) {
441
- millisecGridSize++;
442
- html += '<td>' + ((l < 10) ? '0' : '') + s + '</td>';
443
- }
444
-
445
- html += '</tr></table></div>' +
446
- '</dd>';
447
- } else html += '<dd class="ui_tpicker_millisec" id="ui_tpicker_millisec_' + dp_id + '"' +
448
- ((o.showMillisec) ? '' : noDisplay) + '></dd>';
449
-
450
- // Timezone
451
- html += '<dt class="ui_tpicker_timezone_label" id="ui_tpicker_timezone_label_' + dp_id + '"' +
452
- ((o.showTimezone) ? '' : noDisplay) + '>' + o.timezoneText + '</dt>';
453
- html += '<dd class="ui_tpicker_timezone" id="ui_tpicker_timezone_' + dp_id + '"' +
454
- ((o.showTimezone) ? '' : noDisplay) + '></dd>';
455
-
456
- html += '</dl></div>';
457
- $tp = $(html);
458
-
459
- // if we only want time picker...
460
- if (o.timeOnly === true) {
461
- $tp.prepend(
462
- '<div class="ui-widget-header ui-helper-clearfix ui-corner-all">' +
463
- '<div class="ui-datepicker-title">' + o.timeOnlyTitle + '</div>' +
464
- '</div>');
465
- $dp.find('.ui-datepicker-header, .ui-datepicker-calendar').hide();
466
- }
467
-
468
- this.hour_slider = $tp.find('#ui_tpicker_hour_'+ dp_id).slider({
469
- orientation: "horizontal",
470
- value: this.hour,
471
- min: o.hourMin,
472
- max: hourMax,
473
- step: o.stepHour,
474
- slide: function(event, ui) {
475
- tp_inst.hour_slider.slider( "option", "value", ui.value);
476
- tp_inst._onTimeChange();
477
- }
478
- });
479
-
480
- // Updated by Peter Medeiros:
481
- // - Pass in Event and UI instance into slide function
482
- this.minute_slider = $tp.find('#ui_tpicker_minute_'+ dp_id).slider({
483
- orientation: "horizontal",
484
- value: this.minute,
485
- min: o.minuteMin,
486
- max: minMax,
487
- step: o.stepMinute,
488
- slide: function(event, ui) {
489
- // update the global minute slider instance value with the current slider value
490
- tp_inst.minute_slider.slider( "option", "value", ui.value);
491
- tp_inst._onTimeChange();
492
- }
493
- });
494
-
495
- this.second_slider = $tp.find('#ui_tpicker_second_'+ dp_id).slider({
496
- orientation: "horizontal",
497
- value: this.second,
498
- min: o.secondMin,
499
- max: secMax,
500
- step: o.stepSecond,
501
- slide: function(event, ui) {
502
- tp_inst.second_slider.slider( "option", "value", ui.value);
503
- tp_inst._onTimeChange();
504
- }
505
- });
506
-
507
- this.millisec_slider = $tp.find('#ui_tpicker_millisec_'+ dp_id).slider({
508
- orientation: "horizontal",
509
- value: this.millisec,
510
- min: o.millisecMin,
511
- max: millisecMax,
512
- step: o.stepMillisec,
513
- slide: function(event, ui) {
514
- tp_inst.millisec_slider.slider( "option", "value", ui.value);
515
- tp_inst._onTimeChange();
516
- }
517
- });
518
-
519
- this.timezone_select = $tp.find('#ui_tpicker_timezone_'+ dp_id).append('<select></select>').find("select");
520
- $.fn.append.apply(this.timezone_select,
521
- $.map(o.timezoneList, function(val, idx) {
522
- return $("<option />")
523
- .val(typeof val == "object" ? val.value : val)
524
- .text(typeof val == "object" ? val.label : val);
525
- })
526
- );
527
- this.timezone_select.val((typeof this.timezone != "undefined" && this.timezone != null && this.timezone != "") ? this.timezone : o.timezone);
528
- this.timezone_select.change(function() {
529
- tp_inst._onTimeChange();
530
- });
531
-
532
- // Add grid functionality
533
- if (o.showHour && o.hourGrid > 0) {
534
- size = 100 * hourGridSize * o.hourGrid / (hourMax - o.hourMin);
535
-
536
- $tp.find(".ui_tpicker_hour table").css({
537
- width: size + "%",
538
- marginLeft: (size / (-2 * hourGridSize)) + "%",
539
- borderCollapse: 'collapse'
540
- }).find("td").each( function(index) {
541
- $(this).click(function() {
542
- var h = $(this).html();
543
- if(o.ampm) {
544
- var ap = h.substring(2).toLowerCase(),
545
- aph = parseInt(h.substring(0,2), 10);
546
- if (ap == 'a') {
547
- if (aph == 12) h = 0;
548
- else h = aph;
549
- } else if (aph == 12) h = 12;
550
- else h = aph + 12;
551
- }
552
- tp_inst.hour_slider.slider("option", "value", h);
553
- tp_inst._onTimeChange();
554
- tp_inst._onSelectHandler();
555
- }).css({
556
- cursor: 'pointer',
557
- width: (100 / hourGridSize) + '%',
558
- textAlign: 'center',
559
- overflow: 'hidden'
560
- });
561
- });
562
- }
563
-
564
- if (o.showMinute && o.minuteGrid > 0) {
565
- size = 100 * minuteGridSize * o.minuteGrid / (minMax - o.minuteMin);
566
- $tp.find(".ui_tpicker_minute table").css({
567
- width: size + "%",
568
- marginLeft: (size / (-2 * minuteGridSize)) + "%",
569
- borderCollapse: 'collapse'
570
- }).find("td").each(function(index) {
571
- $(this).click(function() {
572
- tp_inst.minute_slider.slider("option", "value", $(this).html());
573
- tp_inst._onTimeChange();
574
- tp_inst._onSelectHandler();
575
- }).css({
576
- cursor: 'pointer',
577
- width: (100 / minuteGridSize) + '%',
578
- textAlign: 'center',
579
- overflow: 'hidden'
580
- });
581
- });
582
- }
583
-
584
- if (o.showSecond && o.secondGrid > 0) {
585
- $tp.find(".ui_tpicker_second table").css({
586
- width: size + "%",
587
- marginLeft: (size / (-2 * secondGridSize)) + "%",
588
- borderCollapse: 'collapse'
589
- }).find("td").each(function(index) {
590
- $(this).click(function() {
591
- tp_inst.second_slider.slider("option", "value", $(this).html());
592
- tp_inst._onTimeChange();
593
- tp_inst._onSelectHandler();
594
- }).css({
595
- cursor: 'pointer',
596
- width: (100 / secondGridSize) + '%',
597
- textAlign: 'center',
598
- overflow: 'hidden'
599
- });
600
- });
601
- }
602
-
603
- if (o.showMillisec && o.millisecGrid > 0) {
604
- $tp.find(".ui_tpicker_millisec table").css({
605
- width: size + "%",
606
- marginLeft: (size / (-2 * millisecGridSize)) + "%",
607
- borderCollapse: 'collapse'
608
- }).find("td").each(function(index) {
609
- $(this).click(function() {
610
- tp_inst.millisec_slider.slider("option", "value", $(this).html());
611
- tp_inst._onTimeChange();
612
- tp_inst._onSelectHandler();
613
- }).css({
614
- cursor: 'pointer',
615
- width: (100 / millisecGridSize) + '%',
616
- textAlign: 'center',
617
- overflow: 'hidden'
618
- });
619
- });
620
- }
621
-
622
- var $buttonPanel = $dp.find('.ui-datepicker-buttonpane');
623
- if ($buttonPanel.length) $buttonPanel.before($tp);
624
- else $dp.append($tp);
625
-
626
- this.$timeObj = $tp.find('#ui_tpicker_time_'+ dp_id);
627
-
628
- if (this.inst !== null) {
629
- var timeDefined = this.timeDefined;
630
- this._onTimeChange();
631
- this.timeDefined = timeDefined;
632
- }
633
-
634
- //Emulate datepicker onSelect behavior. Call on slidestop.
635
- var onSelectDelegate = function() {
636
- tp_inst._onSelectHandler();
637
- };
638
- this.hour_slider.bind('slidestop',onSelectDelegate);
639
- this.minute_slider.bind('slidestop',onSelectDelegate);
640
- this.second_slider.bind('slidestop',onSelectDelegate);
641
- this.millisec_slider.bind('slidestop',onSelectDelegate);
642
- }
643
- },
644
-
645
- //########################################################################
646
- // This function tries to limit the ability to go outside the
647
- // min/max date range
648
- //########################################################################
649
- _limitMinMaxDateTime: function(dp_inst, adjustSliders){
650
- var o = this._defaults,
651
- dp_date = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay);
652
-
653
- if(!this._defaults.showTimepicker) return; // No time so nothing to check here
654
-
655
- if($.datepicker._get(dp_inst, 'minDateTime') !== null && $.datepicker._get(dp_inst, 'minDateTime') !== undefined && dp_date){
656
- var minDateTime = $.datepicker._get(dp_inst, 'minDateTime'),
657
- minDateTimeDate = new Date(minDateTime.getFullYear(), minDateTime.getMonth(), minDateTime.getDate(), 0, 0, 0, 0);
658
-
659
- if(this.hourMinOriginal === null || this.minuteMinOriginal === null || this.secondMinOriginal === null || this.millisecMinOriginal === null){
660
- this.hourMinOriginal = o.hourMin;
661
- this.minuteMinOriginal = o.minuteMin;
662
- this.secondMinOriginal = o.secondMin;
663
- this.millisecMinOriginal = o.millisecMin;
664
- }
665
-
666
- if(dp_inst.settings.timeOnly || minDateTimeDate.getTime() == dp_date.getTime()) {
667
- this._defaults.hourMin = minDateTime.getHours();
668
- if (this.hour <= this._defaults.hourMin) {
669
- this.hour = this._defaults.hourMin;
670
- this._defaults.minuteMin = minDateTime.getMinutes();
671
- if (this.minute <= this._defaults.minuteMin) {
672
- this.minute = this._defaults.minuteMin;
673
- this._defaults.secondMin = minDateTime.getSeconds();
674
- } else if (this.second <= this._defaults.secondMin){
675
- this.second = this._defaults.secondMin;
676
- this._defaults.millisecMin = minDateTime.getMilliseconds();
677
- } else {
678
- if(this.millisec < this._defaults.millisecMin)
679
- this.millisec = this._defaults.millisecMin;
680
- this._defaults.millisecMin = this.millisecMinOriginal;
681
- }
682
- } else {
683
- this._defaults.minuteMin = this.minuteMinOriginal;
684
- this._defaults.secondMin = this.secondMinOriginal;
685
- this._defaults.millisecMin = this.millisecMinOriginal;
686
- }
687
- }else{
688
- this._defaults.hourMin = this.hourMinOriginal;
689
- this._defaults.minuteMin = this.minuteMinOriginal;
690
- this._defaults.secondMin = this.secondMinOriginal;
691
- this._defaults.millisecMin = this.millisecMinOriginal;
692
- }
693
- }
694
-
695
- if($.datepicker._get(dp_inst, 'maxDateTime') !== null && $.datepicker._get(dp_inst, 'maxDateTime') !== undefined && dp_date){
696
- var maxDateTime = $.datepicker._get(dp_inst, 'maxDateTime'),
697
- maxDateTimeDate = new Date(maxDateTime.getFullYear(), maxDateTime.getMonth(), maxDateTime.getDate(), 0, 0, 0, 0);
698
-
699
- if(this.hourMaxOriginal === null || this.minuteMaxOriginal === null || this.secondMaxOriginal === null){
700
- this.hourMaxOriginal = o.hourMax;
701
- this.minuteMaxOriginal = o.minuteMax;
702
- this.secondMaxOriginal = o.secondMax;
703
- this.millisecMaxOriginal = o.millisecMax;
704
- }
705
-
706
- if(dp_inst.settings.timeOnly || maxDateTimeDate.getTime() == dp_date.getTime()){
707
- this._defaults.hourMax = maxDateTime.getHours();
708
- if (this.hour >= this._defaults.hourMax) {
709
- this.hour = this._defaults.hourMax;
710
- this._defaults.minuteMax = maxDateTime.getMinutes();
711
- if (this.minute >= this._defaults.minuteMax) {
712
- this.minute = this._defaults.minuteMax;
713
- this._defaults.secondMax = maxDateTime.getSeconds();
714
- } else if (this.second >= this._defaults.secondMax) {
715
- this.second = this._defaults.secondMax;
716
- this._defaults.millisecMax = maxDateTime.getMilliseconds();
717
- } else {
718
- if(this.millisec > this._defaults.millisecMax) this.millisec = this._defaults.millisecMax;
719
- this._defaults.millisecMax = this.millisecMaxOriginal;
720
- }
721
- } else {
722
- this._defaults.minuteMax = this.minuteMaxOriginal;
723
- this._defaults.secondMax = this.secondMaxOriginal;
724
- this._defaults.millisecMax = this.millisecMaxOriginal;
725
- }
726
- }else{
727
- this._defaults.hourMax = this.hourMaxOriginal;
728
- this._defaults.minuteMax = this.minuteMaxOriginal;
729
- this._defaults.secondMax = this.secondMaxOriginal;
730
- this._defaults.millisecMax = this.millisecMaxOriginal;
731
- }
732
- }
733
-
734
- if(adjustSliders !== undefined && adjustSliders === true){
735
- var hourMax = (this._defaults.hourMax - ((this._defaults.hourMax - this._defaults.hourMin) % this._defaults.stepHour)).toFixed(0),
736
- minMax = (this._defaults.minuteMax - ((this._defaults.minuteMax - this._defaults.minuteMin) % this._defaults.stepMinute)).toFixed(0),
737
- secMax = (this._defaults.secondMax - ((this._defaults.secondMax - this._defaults.secondMin) % this._defaults.stepSecond)).toFixed(0),
738
- millisecMax = (this._defaults.millisecMax - ((this._defaults.millisecMax - this._defaults.millisecMin) % this._defaults.stepMillisec)).toFixed(0);
739
-
740
- if(this.hour_slider)
741
- this.hour_slider.slider("option", { min: this._defaults.hourMin, max: hourMax }).slider('value', this.hour);
742
- if(this.minute_slider)
743
- this.minute_slider.slider("option", { min: this._defaults.minuteMin, max: minMax }).slider('value', this.minute);
744
- if(this.second_slider)
745
- this.second_slider.slider("option", { min: this._defaults.secondMin, max: secMax }).slider('value', this.second);
746
- if(this.millisec_slider)
747
- this.millisec_slider.slider("option", { min: this._defaults.millisecMin, max: millisecMax }).slider('value', this.millisec);
748
- }
749
-
750
- },
751
-
752
-
753
- //########################################################################
754
- // when a slider moves, set the internal time...
755
- // on time change is also called when the time is updated in the text field
756
- //########################################################################
757
- _onTimeChange: function() {
758
- var hour = (this.hour_slider) ? this.hour_slider.slider('value') : false,
759
- minute = (this.minute_slider) ? this.minute_slider.slider('value') : false,
760
- second = (this.second_slider) ? this.second_slider.slider('value') : false,
761
- millisec = (this.millisec_slider) ? this.millisec_slider.slider('value') : false,
762
- timezone = (this.timezone_select) ? this.timezone_select.val() : false,
763
- o = this._defaults;
764
-
765
- if (typeof(hour) == 'object') hour = false;
766
- if (typeof(minute) == 'object') minute = false;
767
- if (typeof(second) == 'object') second = false;
768
- if (typeof(millisec) == 'object') millisec = false;
769
- if (typeof(timezone) == 'object') timezone = false;
770
-
771
- if (hour !== false) hour = parseInt(hour,10);
772
- if (minute !== false) minute = parseInt(minute,10);
773
- if (second !== false) second = parseInt(second,10);
774
- if (millisec !== false) millisec = parseInt(millisec,10);
775
-
776
- var ampm = o[hour < 12 ? 'amNames' : 'pmNames'][0];
777
-
778
- // If the update was done in the input field, the input field should not be updated.
779
- // If the update was done using the sliders, update the input field.
780
- var hasChanged = (hour != this.hour || minute != this.minute
781
- || second != this.second || millisec != this.millisec
782
- || (this.ampm.length > 0
783
- && (hour < 12) != ($.inArray(this.ampm.toUpperCase(), this.amNames) !== -1))
784
- || timezone != this.timezone);
785
-
786
- if (hasChanged) {
787
-
788
- if (hour !== false)this.hour = hour;
789
- if (minute !== false) this.minute = minute;
790
- if (second !== false) this.second = second;
791
- if (millisec !== false) this.millisec = millisec;
792
- if (timezone !== false) this.timezone = timezone;
793
-
794
- if (!this.inst) this.inst = $.datepicker._getInst(this.$input[0]);
795
-
796
- this._limitMinMaxDateTime(this.inst, true);
797
- }
798
- if (o.ampm) this.ampm = ampm;
799
-
800
- this._formatTime();
801
- if (this.$timeObj) this.$timeObj.text(this.formattedTime + o.timeSuffix);
802
- this.timeDefined = true;
803
- if (hasChanged) this._updateDateTime();
804
- },
805
-
806
- //########################################################################
807
- // call custom onSelect.
808
- // bind to sliders slidestop, and grid click.
809
- //########################################################################
810
- _onSelectHandler: function() {
811
- var onSelect = this._defaults.onSelect;
812
- var inputEl = this.$input ? this.$input[0] : null;
813
- if (onSelect && inputEl) {
814
- onSelect.apply(inputEl, [this.formattedDateTime, this]);
815
- }
816
- },
817
-
818
- //########################################################################
819
- // format the time all pretty...
820
- //########################################################################
821
- _formatTime: function(time, format, ampm) {
822
- if (ampm == undefined) ampm = this._defaults.ampm;
823
- time = time || { hour: this.hour, minute: this.minute, second: this.second, millisec: this.millisec, ampm: this.ampm, timezone: this.timezone };
824
- var tmptime = (format || this._defaults.timeFormat).toString();
825
-
826
- var hour = parseInt(time.hour, 10);
827
- if (ampm) {
828
- if (!$.inArray(time.ampm.toUpperCase(), this.amNames) !== -1)
829
- hour = hour % 12;
830
- if (hour === 0)
831
- hour = 12;
832
- }
833
- tmptime = tmptime.replace(/(?:hh?|mm?|ss?|[tT]{1,2}|[lz])/g, function(match) {
834
- switch (match.toLowerCase()) {
835
- case 'hh': return ('0' + hour).slice(-2);
836
- case 'h': return hour;
837
- case 'mm': return ('0' + time.minute).slice(-2);
838
- case 'm': return time.minute;
839
- case 'ss': return ('0' + time.second).slice(-2);
840
- case 's': return time.second;
841
- case 'l': return ('00' + time.millisec).slice(-3);
842
- case 'z': return time.timezone;
843
- case 't': case 'tt':
844
- if (ampm) {
845
- var _ampm = time.ampm;
846
- if (match.length == 1)
847
- _ampm = _ampm.charAt(0);
848
- return match.charAt(0) == 'T' ? _ampm.toUpperCase() : _ampm.toLowerCase();
849
- }
850
- return '';
851
- }
852
- });
853
-
854
- if (arguments.length) return tmptime;
855
- else this.formattedTime = tmptime;
856
- },
857
-
858
- //########################################################################
859
- // update our input with the new date time..
860
- //########################################################################
861
- _updateDateTime: function(dp_inst) {
862
- dp_inst = this.inst || dp_inst,
863
- dt = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay),
864
- dateFmt = $.datepicker._get(dp_inst, 'dateFormat'),
865
- formatCfg = $.datepicker._getFormatConfig(dp_inst),
866
- timeAvailable = dt !== null && this.timeDefined;
867
- this.formattedDate = $.datepicker.formatDate(dateFmt, (dt === null ? new Date() : dt), formatCfg);
868
- var formattedDateTime = this.formattedDate;
869
- if (dp_inst.lastVal !== undefined && (dp_inst.lastVal.length > 0 && this.$input.val().length === 0))
870
- return;
871
-
872
- if (this._defaults.timeOnly === true) {
873
- formattedDateTime = this.formattedTime;
874
- } else if (this._defaults.timeOnly !== true && (this._defaults.alwaysSetTime || timeAvailable)) {
875
- formattedDateTime += this._defaults.separator + this.formattedTime + this._defaults.timeSuffix;
876
- }
877
-
878
- this.formattedDateTime = formattedDateTime;
879
-
880
- if(!this._defaults.showTimepicker) {
881
- this.$input.val(this.formattedDate);
882
- } else if (this.$altInput && this._defaults.altFieldTimeOnly === true) {
883
- this.$altInput.val(this.formattedTime);
884
- this.$input.val(this.formattedDate);
885
- } else if(this.$altInput) {
886
- this.$altInput.val(formattedDateTime);
887
- this.$input.val(formattedDateTime);
888
- } else {
889
- this.$input.val(formattedDateTime);
890
- }
891
-
892
- this.$input.trigger("change");
893
- }
894
-
895
- });
896
-
897
- $.fn.extend({
898
- //########################################################################
899
- // shorthand just to use timepicker..
900
- //########################################################################
901
- timepicker: function(o) {
902
- o = o || {};
903
- var tmp_args = arguments;
904
-
905
- if (typeof o == 'object') tmp_args[0] = $.extend(o, { timeOnly: true });
906
-
907
- return $(this).each(function() {
908
- $.fn.datetimepicker.apply($(this), tmp_args);
909
- });
910
- },
911
-
912
- //########################################################################
913
- // extend timepicker to datepicker
914
- //########################################################################
915
- datetimepicker: function(o) {
916
- o = o || {};
917
- var $input = this,
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
- });
928
- }
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
- // the bad hack :/ override datepicker so it doesnt close on select
939
- // inspired: http://stackoverflow.com/questions/1252512/jquery-datepicker-prevent-closing-picker-when-clicking-a-date/1762378#1762378
940
- //########################################################################
941
- $.datepicker._base_selectDate = $.datepicker._selectDate;
942
- $.datepicker._selectDate = function (id, dateStr) {
943
- var inst = this._getInst($(id)[0]),
944
- tp_inst = this._get(inst, 'timepicker');
945
-
946
- if (tp_inst) {
947
- tp_inst._limitMinMaxDateTime(inst, true);
948
- inst.inline = inst.stay_open = true;
949
- //This way the onSelect handler called from calendarpicker get the full dateTime
950
- this._base_selectDate(id, dateStr);
951
- inst.inline = inst.stay_open = false;
952
- this._notifyChange(inst);
953
- this._updateDatepicker(inst);
954
- }
955
- else this._base_selectDate(id, dateStr);
956
- };
957
-
958
- //#############################################################################################
959
- // second bad hack :/ override datepicker so it triggers an event when changing the input field
960
- // and does not redraw the datepicker on every selectDate event
961
- //#############################################################################################
962
- $.datepicker._base_updateDatepicker = $.datepicker._updateDatepicker;
963
- $.datepicker._updateDatepicker = function(inst) {
964
-
965
- // don't popup the datepicker if there is another instance already opened
966
- var input = inst.input[0];
967
- if($.datepicker._curInst &&
968
- $.datepicker._curInst != inst &&
969
- $.datepicker._datepickerShowing &&
970
- $.datepicker._lastInput != input) {
971
- return;
972
- }
973
-
974
- if (typeof(inst.stay_open) !== 'boolean' || inst.stay_open === false) {
975
-
976
- this._base_updateDatepicker(inst);
977
-
978
- // Reload the time control when changing something in the input text field.
979
- var tp_inst = this._get(inst, 'timepicker');
980
- if(tp_inst) tp_inst._addTimePicker(inst);
981
- }
982
- };
983
-
984
- //#######################################################################################
985
- // third bad hack :/ override datepicker so it allows spaces and colon in the input field
986
- //#######################################################################################
987
- $.datepicker._base_doKeyPress = $.datepicker._doKeyPress;
988
- $.datepicker._doKeyPress = function(event) {
989
- var inst = $.datepicker._getInst(event.target),
990
- tp_inst = $.datepicker._get(inst, 'timepicker');
991
-
992
- if (tp_inst) {
993
- if ($.datepicker._get(inst, 'constrainInput')) {
994
- var ampm = tp_inst._defaults.ampm,
995
- dateChars = $.datepicker._possibleChars($.datepicker._get(inst, 'dateFormat')),
996
- datetimeChars = tp_inst._defaults.timeFormat.toString()
997
- .replace(/[hms]/g, '')
998
- .replace(/TT/g, ampm ? 'APM' : '')
999
- .replace(/Tt/g, ampm ? 'AaPpMm' : '')
1000
- .replace(/tT/g, ampm ? 'AaPpMm' : '')
1001
- .replace(/T/g, ampm ? 'AP' : '')
1002
- .replace(/tt/g, ampm ? 'apm' : '')
1003
- .replace(/t/g, ampm ? 'ap' : '') +
1004
- " " +
1005
- tp_inst._defaults.separator +
1006
- tp_inst._defaults.timeSuffix +
1007
- (tp_inst._defaults.showTimezone ? tp_inst._defaults.timezoneList.join('') : '') +
1008
- (tp_inst._defaults.amNames.join('')) +
1009
- (tp_inst._defaults.pmNames.join('')) +
1010
- dateChars,
1011
- chr = String.fromCharCode(event.charCode === undefined ? event.keyCode : event.charCode);
1012
- return event.ctrlKey || (chr < ' ' || !dateChars || datetimeChars.indexOf(chr) > -1);
1013
- }
1014
- }
1015
-
1016
- return $.datepicker._base_doKeyPress(event);
1017
- };
1018
-
1019
- //#######################################################################################
1020
- // Override key up event to sync manual input changes.
1021
- //#######################################################################################
1022
- $.datepicker._base_doKeyUp = $.datepicker._doKeyUp;
1023
- $.datepicker._doKeyUp = function (event) {
1024
- var inst = $.datepicker._getInst(event.target),
1025
- tp_inst = $.datepicker._get(inst, 'timepicker');
1026
-
1027
- if (tp_inst) {
1028
- if (tp_inst._defaults.timeOnly && (inst.input.val() != inst.lastVal)) {
1029
- try {
1030
- $.datepicker._updateDatepicker(inst);
1031
- }
1032
- catch (err) {
1033
- $.datepicker.log(err);
1034
- }
1035
- }
1036
- }
1037
-
1038
- return $.datepicker._base_doKeyUp(event);
1039
- };
1040
-
1041
- //#######################################################################################
1042
- // override "Today" button to also grab the time.
1043
- //#######################################################################################
1044
- $.datepicker._base_gotoToday = $.datepicker._gotoToday;
1045
- $.datepicker._gotoToday = function(id) {
1046
- var inst = this._getInst($(id)[0]),
1047
- $dp = inst.dpDiv;
1048
- this._base_gotoToday(id);
1049
- var now = new Date();
1050
- var tp_inst = this._get(inst, 'timepicker');
1051
- if (tp_inst._defaults.showTimezone && tp_inst.timezone_select) {
1052
- var tzoffset = now.getTimezoneOffset(); // If +0100, returns -60
1053
- var tzsign = tzoffset > 0 ? '-' : '+';
1054
- tzoffset = Math.abs(tzoffset);
1055
- var tzmin = tzoffset % 60
1056
- tzoffset = tzsign + ('0' + (tzoffset - tzmin) / 60).slice(-2) + ('0' + tzmin).slice(-2);
1057
- if (tp_inst._defaults.timezoneIso8609)
1058
- tzoffset = tzoffset.substring(0, 3) + ':' + tzoffset.substring(3);
1059
- tp_inst.timezone_select.val(tzoffset);
1060
- }
1061
- this._setTime(inst, now);
1062
- $( '.ui-datepicker-today', $dp).click();
1063
- };
1064
-
1065
- //#######################################################################################
1066
- // Disable & enable the Time in the datetimepicker
1067
- //#######################################################################################
1068
- $.datepicker._disableTimepickerDatepicker = function(target, date, withDate) {
1069
- var inst = this._getInst(target),
1070
- tp_inst = this._get(inst, 'timepicker');
1071
- $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1072
- if (tp_inst) {
1073
- tp_inst._defaults.showTimepicker = false;
1074
- tp_inst._updateDateTime(inst);
1075
- }
1076
- };
1077
-
1078
- $.datepicker._enableTimepickerDatepicker = function(target, date, withDate) {
1079
- var inst = this._getInst(target),
1080
- tp_inst = this._get(inst, 'timepicker');
1081
- $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1082
- if (tp_inst) {
1083
- tp_inst._defaults.showTimepicker = true;
1084
- tp_inst._addTimePicker(inst); // Could be disabled on page load
1085
- tp_inst._updateDateTime(inst);
1086
- }
1087
- };
1088
-
1089
- //#######################################################################################
1090
- // Create our own set time function
1091
- //#######################################################################################
1092
- $.datepicker._setTime = function(inst, date) {
1093
- var tp_inst = this._get(inst, 'timepicker');
1094
- if (tp_inst) {
1095
- var defaults = tp_inst._defaults,
1096
- // calling _setTime with no date sets time to defaults
1097
- hour = date ? date.getHours() : defaults.hour,
1098
- minute = date ? date.getMinutes() : defaults.minute,
1099
- second = date ? date.getSeconds() : defaults.second,
1100
- millisec = date ? date.getMilliseconds() : defaults.millisec;
1101
-
1102
- //check if within min/max times..
1103
- 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)) {
1104
- hour = defaults.hourMin;
1105
- minute = defaults.minuteMin;
1106
- second = defaults.secondMin;
1107
- millisec = defaults.millisecMin;
1108
- }
1109
-
1110
- tp_inst.hour = hour;
1111
- tp_inst.minute = minute;
1112
- tp_inst.second = second;
1113
- tp_inst.millisec = millisec;
1114
-
1115
- if (tp_inst.hour_slider) tp_inst.hour_slider.slider('value', hour);
1116
- if (tp_inst.minute_slider) tp_inst.minute_slider.slider('value', minute);
1117
- if (tp_inst.second_slider) tp_inst.second_slider.slider('value', second);
1118
- if (tp_inst.millisec_slider) tp_inst.millisec_slider.slider('value', millisec);
1119
-
1120
- tp_inst._onTimeChange();
1121
- tp_inst._updateDateTime(inst);
1122
- }
1123
- };
1124
-
1125
- //#######################################################################################
1126
- // Create new public method to set only time, callable as $().datepicker('setTime', date)
1127
- //#######################################################################################
1128
- $.datepicker._setTimeDatepicker = function(target, date, withDate) {
1129
- var inst = this._getInst(target),
1130
- tp_inst = this._get(inst, 'timepicker');
1131
-
1132
- if (tp_inst) {
1133
- this._setDateFromField(inst);
1134
- var tp_date;
1135
- if (date) {
1136
- if (typeof date == "string") {
1137
- tp_inst._parseTime(date, withDate);
1138
- tp_date = new Date();
1139
- tp_date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1140
- }
1141
- else tp_date = new Date(date.getTime());
1142
- if (tp_date.toString() == 'Invalid Date') tp_date = undefined;
1143
- this._setTime(inst, tp_date);
1144
- }
1145
- }
1146
-
1147
- };
1148
-
1149
- //#######################################################################################
1150
- // override setDate() to allow setting time too within Date object
1151
- //#######################################################################################
1152
- $.datepicker._base_setDateDatepicker = $.datepicker._setDateDatepicker;
1153
- $.datepicker._setDateDatepicker = function(target, date) {
1154
- var inst = this._getInst(target),
1155
- tp_date = (date instanceof Date) ? new Date(date.getTime()) : date;
1156
-
1157
- this._updateDatepicker(inst);
1158
- this._base_setDateDatepicker.apply(this, arguments);
1159
- this._setTimeDatepicker(target, tp_date, true);
1160
- };
1161
-
1162
- //#######################################################################################
1163
- // override getDate() to allow getting time too within Date object
1164
- //#######################################################################################
1165
- $.datepicker._base_getDateDatepicker = $.datepicker._getDateDatepicker;
1166
- $.datepicker._getDateDatepicker = function(target, noDefault) {
1167
- var inst = this._getInst(target),
1168
- tp_inst = this._get(inst, 'timepicker');
1169
-
1170
- if (tp_inst) {
1171
- this._setDateFromField(inst, noDefault);
1172
- var date = this._getDate(inst);
1173
- if (date && tp_inst._parseTime($(target).val(), tp_inst.timeOnly)) date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1174
- return date;
1175
- }
1176
- return this._base_getDateDatepicker(target, noDefault);
1177
- };
1178
-
1179
- //#######################################################################################
1180
- // override parseDate() because UI 1.8.14 throws an error about "Extra characters"
1181
- // An option in datapicker to ignore extra format characters would be nicer.
1182
- //#######################################################################################
1183
- $.datepicker._base_parseDate = $.datepicker.parseDate;
1184
- $.datepicker.parseDate = function(format, value, settings) {
1185
- var date;
1186
- try {
1187
- date = this._base_parseDate(format, value, settings);
1188
- } catch (err) {
1189
- // Hack! The error message ends with a colon, a space, and
1190
- // the "extra" characters. We rely on that instead of
1191
- // attempting to perfectly reproduce the parsing algorithm.
1192
- date = this._base_parseDate(format, value.substring(0,value.length-(err.length-err.indexOf(':')-2)), settings);
1193
- }
1194
- return date;
1195
- };
1196
-
1197
- //#######################################################################################
1198
- // override formatDate to set date with time to the input
1199
- //#######################################################################################
1200
- $.datepicker._base_formatDate=$.datepicker._formatDate;
1201
- $.datepicker._formatDate = function(inst, day, month, year){
1202
- var tp_inst = this._get(inst, 'timepicker');
1203
- if(tp_inst)
1204
- {
1205
- if(day)
1206
- var b = this._base_formatDate(inst, day, month, year);
1207
- tp_inst._updateDateTime();
1208
- return tp_inst.$input.val();
1209
- }
1210
- return this._base_formatDate(inst);
1211
- }
1212
-
1213
- //#######################################################################################
1214
- // override options setter to add time to maxDate(Time) and minDate(Time). MaxDate
1215
- //#######################################################################################
1216
- $.datepicker._base_optionDatepicker = $.datepicker._optionDatepicker;
1217
- $.datepicker._optionDatepicker = function(target, name, value) {
1218
- var inst = this._getInst(target),
1219
- tp_inst = this._get(inst, 'timepicker');
1220
- if (tp_inst) {
1221
- var min,max,onselect;
1222
- if (typeof name == 'string') { // if min/max was set with the string
1223
- if (name==='minDate' || name==='minDateTime' )
1224
- min = value;
1225
- else if (name==='maxDate' || name==='maxDateTime')
1226
- max = value;
1227
- else if (name==='onSelect')
1228
- onselect=value;
1229
- } else if (typeof name == 'object') { //if min/max was set with the JSON
1230
- if(name.minDate)
1231
- min = name.minDate;
1232
- else if (name.minDateTime)
1233
- min = name.minDateTime;
1234
- else if (name.maxDate)
1235
- max = name.maxDate;
1236
- else if (name.maxDateTime)
1237
- max = name.maxDateTime;
1238
- }
1239
- if(min){ //if min was set
1240
- if(min==0)
1241
- min=new Date();
1242
- else
1243
- min= new Date(min);
1244
-
1245
- tp_inst._defaults.minDate = min;
1246
- tp_inst._defaults.minDateTime = min;
1247
- } else if (max){ //if max was set
1248
- if(max==0)
1249
- max=new Date();
1250
- else
1251
- max= new Date(max);
1252
- tp_inst._defaults.maxDate = max;
1253
- tp_inst._defaults.maxDateTime = max;
1254
- }
1255
- else if (onselect)
1256
- tp_inst._defaults.onSelect=onselect;
1257
- }
1258
- this._base_optionDatepicker(target, name, value);
1259
- };
1260
-
1261
- //#######################################################################################
1262
- // jQuery extend now ignores nulls!
1263
- //#######################################################################################
1264
- function extendRemove(target, props) {
1265
- $.extend(target, props);
1266
- for (var name in props)
1267
- if (props[name] === null || props[name] === undefined)
1268
- target[name] = props[name];
1269
- return target;
1270
- }
1271
-
1272
- $.timepicker = new Timepicker(); // singleton instance
1273
- $.timepicker.version = "0.9.7";
1274
-
1275
- })(jQuery);
1276
-
1
+ /*
2
+ * jQuery timepicker addon
3
+ * By: Trent Richardson [http://trentrichardson.com]
4
+ * Version 1.1.1
5
+ * Last Modified: 11/07/2012
6
+ *
7
+ * Copyright 2012 Trent Richardson
8
+ * You may use this project under MIT or GPL licenses.
9
+ * http://trentrichardson.com/Impromptu/GPL-LICENSE.txt
10
+ * http://trentrichardson.com/Impromptu/MIT-LICENSE.txt
11
+ */
12
+
13
+ /*jslint evil: true, white: false, undef: false, nomen: false */
14
+
15
+ (function($) {
16
+
17
+ /*
18
+ * Lets not redefine timepicker, Prevent "Uncaught RangeError: Maximum call stack size exceeded"
19
+ */
20
+ $.ui.timepicker = $.ui.timepicker || {};
21
+ if ($.ui.timepicker.version) {
22
+ return;
23
+ }
24
+
25
+ /*
26
+ * Extend jQueryUI, get it started with our version number
27
+ */
28
+ $.extend($.ui, {
29
+ timepicker: {
30
+ version: "1.1.1"
31
+ }
32
+ });
33
+
34
+ /*
35
+ * Timepicker manager.
36
+ * Use the singleton instance of this class, $.timepicker, to interact with the time picker.
37
+ * Settings for (groups of) time pickers are maintained in an instance object,
38
+ * allowing multiple different settings on the same page.
39
+ */
40
+ function Timepicker() {
41
+ this.regional = []; // Available regional settings, indexed by language code
42
+ this.regional[''] = { // Default regional settings
43
+ currentText: 'Now',
44
+ closeText: 'Done',
45
+ amNames: ['AM', 'A'],
46
+ pmNames: ['PM', 'P'],
47
+ timeFormat: 'HH:mm',
48
+ timeSuffix: '',
49
+ timeOnlyTitle: 'Choose Time',
50
+ timeText: 'Time',
51
+ hourText: 'Hour',
52
+ minuteText: 'Minute',
53
+ secondText: 'Second',
54
+ millisecText: 'Millisecond',
55
+ timezoneText: 'Time Zone',
56
+ isRTL: false
57
+ };
58
+ this._defaults = { // Global defaults for all the datetime picker instances
59
+ showButtonPanel: true,
60
+ timeOnly: false,
61
+ showHour: true,
62
+ showMinute: true,
63
+ showSecond: false,
64
+ showMillisec: false,
65
+ showTimezone: false,
66
+ showTime: true,
67
+ stepHour: 1,
68
+ stepMinute: 1,
69
+ stepSecond: 1,
70
+ stepMillisec: 1,
71
+ hour: 0,
72
+ minute: 0,
73
+ second: 0,
74
+ millisec: 0,
75
+ timezone: null,
76
+ useLocalTimezone: false,
77
+ defaultTimezone: "+0000",
78
+ hourMin: 0,
79
+ minuteMin: 0,
80
+ secondMin: 0,
81
+ millisecMin: 0,
82
+ hourMax: 23,
83
+ minuteMax: 59,
84
+ secondMax: 59,
85
+ millisecMax: 999,
86
+ minDateTime: null,
87
+ maxDateTime: null,
88
+ onSelect: null,
89
+ hourGrid: 0,
90
+ minuteGrid: 0,
91
+ secondGrid: 0,
92
+ millisecGrid: 0,
93
+ alwaysSetTime: true,
94
+ separator: ' ',
95
+ altFieldTimeOnly: true,
96
+ altTimeFormat: null,
97
+ altSeparator: null,
98
+ altTimeSuffix: null,
99
+ pickerTimeFormat: null,
100
+ pickerTimeSuffix: null,
101
+ showTimepicker: true,
102
+ timezoneIso8601: false,
103
+ timezoneList: null,
104
+ addSliderAccess: false,
105
+ sliderAccessArgs: null,
106
+ controlType: 'slider',
107
+ defaultValue: null,
108
+ parse: 'strict'
109
+ };
110
+ $.extend(this._defaults, this.regional['']);
111
+ }
112
+
113
+ $.extend(Timepicker.prototype, {
114
+ $input: null,
115
+ $altInput: null,
116
+ $timeObj: null,
117
+ inst: null,
118
+ hour_slider: null,
119
+ minute_slider: null,
120
+ second_slider: null,
121
+ millisec_slider: null,
122
+ timezone_select: null,
123
+ hour: 0,
124
+ minute: 0,
125
+ second: 0,
126
+ millisec: 0,
127
+ timezone: null,
128
+ defaultTimezone: "+0000",
129
+ hourMinOriginal: null,
130
+ minuteMinOriginal: null,
131
+ secondMinOriginal: null,
132
+ millisecMinOriginal: null,
133
+ hourMaxOriginal: null,
134
+ minuteMaxOriginal: null,
135
+ secondMaxOriginal: null,
136
+ millisecMaxOriginal: null,
137
+ ampm: '',
138
+ formattedDate: '',
139
+ formattedTime: '',
140
+ formattedDateTime: '',
141
+ timezoneList: null,
142
+ units: ['hour','minute','second','millisec'],
143
+ control: null,
144
+
145
+ /*
146
+ * Override the default settings for all instances of the time picker.
147
+ * @param settings object - the new settings to use as defaults (anonymous object)
148
+ * @return the manager object
149
+ */
150
+ setDefaults: function(settings) {
151
+ extendRemove(this._defaults, settings || {});
152
+ return this;
153
+ },
154
+
155
+ /*
156
+ * Create a new Timepicker instance
157
+ */
158
+ _newInst: function($input, o) {
159
+ var tp_inst = new Timepicker(),
160
+ inlineSettings = {},
161
+ fns = {},
162
+ overrides, i;
163
+
164
+ for (var attrName in this._defaults) {
165
+ if(this._defaults.hasOwnProperty(attrName)){
166
+ var attrValue = $input.data(attrName.toLowerCase());
167
+ if (attrValue) {
168
+ try {
169
+ inlineSettings[attrName] = eval(attrValue);
170
+ } catch (err) {
171
+ inlineSettings[attrName] = attrValue;
172
+ }
173
+ }
174
+ }
175
+ }
176
+ overrides = {
177
+ beforeShow: function (input, dp_inst) {
178
+ if ($.isFunction(tp_inst._defaults.evnts.beforeShow)) {
179
+ return tp_inst._defaults.evnts.beforeShow.call($input[0], input, dp_inst, tp_inst);
180
+ }
181
+ },
182
+ onChangeMonthYear: function (year, month, dp_inst) {
183
+ // Update the time as well : this prevents the time from disappearing from the $input field.
184
+ tp_inst._updateDateTime(dp_inst);
185
+ if ($.isFunction(tp_inst._defaults.evnts.onChangeMonthYear)) {
186
+ tp_inst._defaults.evnts.onChangeMonthYear.call($input[0], year, month, dp_inst, tp_inst);
187
+ }
188
+ },
189
+ onClose: function (dateText, dp_inst) {
190
+ if (tp_inst.timeDefined === true && $input.val() !== '') {
191
+ tp_inst._updateDateTime(dp_inst);
192
+ }
193
+ if ($.isFunction(tp_inst._defaults.evnts.onClose)) {
194
+ tp_inst._defaults.evnts.onClose.call($input[0], dateText, dp_inst, tp_inst);
195
+ }
196
+ }
197
+ };
198
+ for (i in overrides) {
199
+ if (overrides.hasOwnProperty(i)) {
200
+ fns[i] = o[i] || null;
201
+ }
202
+ }
203
+ tp_inst._defaults = $.extend({}, this._defaults, inlineSettings, o, overrides, {
204
+ evnts:fns,
205
+ timepicker: tp_inst // add timepicker as a property of datepicker: $.datepicker._get(dp_inst, 'timepicker');
206
+ });
207
+ tp_inst.amNames = $.map(tp_inst._defaults.amNames, function(val) {
208
+ return val.toUpperCase();
209
+ });
210
+ tp_inst.pmNames = $.map(tp_inst._defaults.pmNames, function(val) {
211
+ return val.toUpperCase();
212
+ });
213
+
214
+ // controlType is string - key to our this._controls
215
+ if(typeof(tp_inst._defaults.controlType) === 'string'){
216
+ if($.fn[tp_inst._defaults.controlType] === undefined){
217
+ tp_inst._defaults.controlType = 'select';
218
+ }
219
+ tp_inst.control = tp_inst._controls[tp_inst._defaults.controlType];
220
+ }
221
+ // controlType is an object and must implement create, options, value methods
222
+ else{
223
+ tp_inst.control = tp_inst._defaults.controlType;
224
+ }
225
+
226
+ if (tp_inst._defaults.timezoneList === null) {
227
+ var timezoneList = ['-1200', '-1100', '-1000', '-0930', '-0900', '-0800', '-0700', '-0600', '-0500', '-0430', '-0400', '-0330', '-0300', '-0200', '-0100', '+0000',
228
+ '+0100', '+0200', '+0300', '+0330', '+0400', '+0430', '+0500', '+0530', '+0545', '+0600', '+0630', '+0700', '+0800', '+0845', '+0900', '+0930',
229
+ '+1000', '+1030', '+1100', '+1130', '+1200', '+1245', '+1300', '+1400'];
230
+
231
+ if (tp_inst._defaults.timezoneIso8601) {
232
+ timezoneList = $.map(timezoneList, function(val) {
233
+ return val == '+0000' ? 'Z' : (val.substring(0, 3) + ':' + val.substring(3));
234
+ });
235
+ }
236
+ tp_inst._defaults.timezoneList = timezoneList;
237
+ }
238
+
239
+ tp_inst.timezone = tp_inst._defaults.timezone;
240
+ tp_inst.hour = tp_inst._defaults.hour;
241
+ tp_inst.minute = tp_inst._defaults.minute;
242
+ tp_inst.second = tp_inst._defaults.second;
243
+ tp_inst.millisec = tp_inst._defaults.millisec;
244
+ tp_inst.ampm = '';
245
+ tp_inst.$input = $input;
246
+
247
+ if (o.altField) {
248
+ tp_inst.$altInput = $(o.altField).css({
249
+ cursor: 'pointer'
250
+ }).focus(function() {
251
+ $input.trigger("focus");
252
+ });
253
+ }
254
+
255
+ if (tp_inst._defaults.minDate === 0 || tp_inst._defaults.minDateTime === 0) {
256
+ tp_inst._defaults.minDate = new Date();
257
+ }
258
+ if (tp_inst._defaults.maxDate === 0 || tp_inst._defaults.maxDateTime === 0) {
259
+ tp_inst._defaults.maxDate = new Date();
260
+ }
261
+
262
+ // datepicker needs minDate/maxDate, timepicker needs minDateTime/maxDateTime..
263
+ if (tp_inst._defaults.minDate !== undefined && tp_inst._defaults.minDate instanceof Date) {
264
+ tp_inst._defaults.minDateTime = new Date(tp_inst._defaults.minDate.getTime());
265
+ }
266
+ if (tp_inst._defaults.minDateTime !== undefined && tp_inst._defaults.minDateTime instanceof Date) {
267
+ tp_inst._defaults.minDate = new Date(tp_inst._defaults.minDateTime.getTime());
268
+ }
269
+ if (tp_inst._defaults.maxDate !== undefined && tp_inst._defaults.maxDate instanceof Date) {
270
+ tp_inst._defaults.maxDateTime = new Date(tp_inst._defaults.maxDate.getTime());
271
+ }
272
+ if (tp_inst._defaults.maxDateTime !== undefined && tp_inst._defaults.maxDateTime instanceof Date) {
273
+ tp_inst._defaults.maxDate = new Date(tp_inst._defaults.maxDateTime.getTime());
274
+ }
275
+ tp_inst.$input.bind('focus', function() {
276
+ tp_inst._onFocus();
277
+ });
278
+
279
+ return tp_inst;
280
+ },
281
+
282
+ /*
283
+ * add our sliders to the calendar
284
+ */
285
+ _addTimePicker: function(dp_inst) {
286
+ var currDT = (this.$altInput && this._defaults.altFieldTimeOnly) ? this.$input.val() + ' ' + this.$altInput.val() : this.$input.val();
287
+
288
+ this.timeDefined = this._parseTime(currDT);
289
+ this._limitMinMaxDateTime(dp_inst, false);
290
+ this._injectTimePicker();
291
+ },
292
+
293
+ /*
294
+ * parse the time string from input value or _setTime
295
+ */
296
+ _parseTime: function(timeString, withDate) {
297
+ if (!this.inst) {
298
+ this.inst = $.datepicker._getInst(this.$input[0]);
299
+ }
300
+
301
+ if (withDate || !this._defaults.timeOnly) {
302
+ var dp_dateFormat = $.datepicker._get(this.inst, 'dateFormat');
303
+ try {
304
+ var parseRes = parseDateTimeInternal(dp_dateFormat, this._defaults.timeFormat, timeString, $.datepicker._getFormatConfig(this.inst), this._defaults);
305
+ if (!parseRes.timeObj) {
306
+ return false;
307
+ }
308
+ $.extend(this, parseRes.timeObj);
309
+ } catch (err) {
310
+ $.datepicker.log("Error parsing the date/time string: " + err +
311
+ "\ndate/time string = " + timeString +
312
+ "\ntimeFormat = " + this._defaults.timeFormat +
313
+ "\ndateFormat = " + dp_dateFormat);
314
+ return false;
315
+ }
316
+ return true;
317
+ } else {
318
+ var timeObj = $.datepicker.parseTime(this._defaults.timeFormat, timeString, this._defaults);
319
+ if (!timeObj) {
320
+ return false;
321
+ }
322
+ $.extend(this, timeObj);
323
+ return true;
324
+ }
325
+ },
326
+
327
+ /*
328
+ * generate and inject html for timepicker into ui datepicker
329
+ */
330
+ _injectTimePicker: function() {
331
+ var $dp = this.inst.dpDiv,
332
+ o = this.inst.settings,
333
+ tp_inst = this,
334
+ litem = '',
335
+ uitem = '',
336
+ max = {},
337
+ gridSize = {},
338
+ size = null;
339
+
340
+ // Prevent displaying twice
341
+ if ($dp.find("div.ui-timepicker-div").length === 0 && o.showTimepicker) {
342
+ var noDisplay = ' style="display:none;"',
343
+ html = '<div class="ui-timepicker-div'+ (o.isRTL? ' ui-timepicker-rtl' : '') +'"><dl>' + '<dt class="ui_tpicker_time_label"' + ((o.showTime) ? '' : noDisplay) + '>' + o.timeText + '</dt>' +
344
+ '<dd class="ui_tpicker_time"' + ((o.showTime) ? '' : noDisplay) + '></dd>';
345
+
346
+ // Create the markup
347
+ for(var i=0,l=this.units.length; i<l; i++){
348
+ litem = this.units[i];
349
+ uitem = litem.substr(0,1).toUpperCase() + litem.substr(1);
350
+ // Added by Peter Medeiros:
351
+ // - Figure out what the hour/minute/second max should be based on the step values.
352
+ // - Example: if stepMinute is 15, then minMax is 45.
353
+ max[litem] = parseInt((o[litem+'Max'] - ((o[litem+'Max'] - o[litem+'Min']) % o['step'+uitem])), 10);
354
+ gridSize[litem] = 0;
355
+
356
+ html += '<dt class="ui_tpicker_'+ litem +'_label"' + ((o['show'+uitem]) ? '' : noDisplay) + '>' + o[litem +'Text'] + '</dt>' +
357
+ '<dd class="ui_tpicker_'+ litem +'"><div class="ui_tpicker_'+ litem +'_slider"' + ((o['show'+uitem]) ? '' : noDisplay) + '></div>';
358
+
359
+ if (o['show'+uitem] && o[litem+'Grid'] > 0) {
360
+ html += '<div style="padding-left: 1px"><table class="ui-tpicker-grid-label"><tr>';
361
+
362
+ if(litem == 'hour'){
363
+ for (var h = o[litem+'Min']; h <= max[litem]; h += parseInt(o[litem+'Grid'], 10)) {
364
+ gridSize[litem]++;
365
+ var tmph = $.datepicker.formatTime(useAmpm(o.pickerTimeFormat || o.timeFormat)? 'hht':'HH', {hour:h}, o);
366
+ html += '<td data-for="'+litem+'">' + tmph + '</td>';
367
+ }
368
+ }
369
+ else{
370
+ for (var m = o[litem+'Min']; m <= max[litem]; m += parseInt(o[litem+'Grid'], 10)) {
371
+ gridSize[litem]++;
372
+ html += '<td data-for="'+litem+'">' + ((m < 10) ? '0' : '') + m + '</td>';
373
+ }
374
+ }
375
+
376
+ html += '</tr></table></div>';
377
+ }
378
+ html += '</dd>';
379
+ }
380
+
381
+ // Timezone
382
+ html += '<dt class="ui_tpicker_timezone_label"' + ((o.showTimezone) ? '' : noDisplay) + '>' + o.timezoneText + '</dt>';
383
+ html += '<dd class="ui_tpicker_timezone" ' + ((o.showTimezone) ? '' : noDisplay) + '></dd>';
384
+
385
+ // Create the elements from string
386
+ html += '</dl></div>';
387
+ var $tp = $(html);
388
+
389
+ // if we only want time picker...
390
+ if (o.timeOnly === true) {
391
+ $tp.prepend('<div class="ui-widget-header ui-helper-clearfix ui-corner-all">' + '<div class="ui-datepicker-title">' + o.timeOnlyTitle + '</div>' + '</div>');
392
+ $dp.find('.ui-datepicker-header, .ui-datepicker-calendar').hide();
393
+ }
394
+
395
+ // add sliders, adjust grids, add events
396
+ for(var i=0,l=tp_inst.units.length; i<l; i++){
397
+ litem = tp_inst.units[i];
398
+ uitem = litem.substr(0,1).toUpperCase() + litem.substr(1);
399
+
400
+ // add the slider
401
+ 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]);
402
+
403
+ // adjust the grid and add click event
404
+ if (o['show'+uitem] && o[litem+'Grid'] > 0) {
405
+ size = 100 * gridSize[litem] * o[litem+'Grid'] / (max[litem] - o[litem+'Min']);
406
+ $tp.find('.ui_tpicker_'+litem+' table').css({
407
+ width: size + "%",
408
+ marginLeft: o.isRTL? '0' : ((size / (-2 * gridSize[litem])) + "%"),
409
+ marginRight: o.isRTL? ((size / (-2 * gridSize[litem])) + "%") : '0',
410
+ borderCollapse: 'collapse'
411
+ }).find("td").click(function(e){
412
+ var $t = $(this),
413
+ h = $t.html(),
414
+ n = parseInt(h.replace(/[^0-9]/g),10),
415
+ ap = h.replace(/[^apm]/ig),
416
+ f = $t.data('for'); // loses scope, so we use data-for
417
+
418
+ if(f == 'hour'){
419
+ if(ap.indexOf('p') !== -1 && n < 12){
420
+ n += 12;
421
+ }
422
+ else{
423
+ if(ap.indexOf('a') !== -1 && n === 12){
424
+ n = 0;
425
+ }
426
+ }
427
+ }
428
+
429
+ tp_inst.control.value(tp_inst, tp_inst[f+'_slider'], litem, n);
430
+
431
+ tp_inst._onTimeChange();
432
+ tp_inst._onSelectHandler();
433
+ })
434
+ .css({
435
+ cursor: 'pointer',
436
+ width: (100 / gridSize[litem]) + '%',
437
+ textAlign: 'center',
438
+ overflow: 'hidden'
439
+ });
440
+ } // end if grid > 0
441
+ } // end for loop
442
+
443
+ // Add timezone options
444
+ this.timezone_select = $tp.find('.ui_tpicker_timezone').append('<select></select>').find("select");
445
+ $.fn.append.apply(this.timezone_select,
446
+ $.map(o.timezoneList, function(val, idx) {
447
+ return $("<option />").val(typeof val == "object" ? val.value : val).text(typeof val == "object" ? val.label : val);
448
+ }));
449
+ if (typeof(this.timezone) != "undefined" && this.timezone !== null && this.timezone !== "") {
450
+ var local_date = new Date(this.inst.selectedYear, this.inst.selectedMonth, this.inst.selectedDay, 12);
451
+ var local_timezone = $.timepicker.timeZoneOffsetString(local_date);
452
+ if (local_timezone == this.timezone) {
453
+ selectLocalTimeZone(tp_inst);
454
+ } else {
455
+ this.timezone_select.val(this.timezone);
456
+ }
457
+ } else {
458
+ if (typeof(this.hour) != "undefined" && this.hour !== null && this.hour !== "") {
459
+ this.timezone_select.val(o.defaultTimezone);
460
+ } else {
461
+ selectLocalTimeZone(tp_inst);
462
+ }
463
+ }
464
+ this.timezone_select.change(function() {
465
+ tp_inst._defaults.useLocalTimezone = false;
466
+ tp_inst._onTimeChange();
467
+ });
468
+ // End timezone options
469
+
470
+ // inject timepicker into datepicker
471
+ var $buttonPanel = $dp.find('.ui-datepicker-buttonpane');
472
+ if ($buttonPanel.length) {
473
+ $buttonPanel.before($tp);
474
+ } else {
475
+ $dp.append($tp);
476
+ }
477
+
478
+ this.$timeObj = $tp.find('.ui_tpicker_time');
479
+
480
+ if (this.inst !== null) {
481
+ var timeDefined = this.timeDefined;
482
+ this._onTimeChange();
483
+ this.timeDefined = timeDefined;
484
+ }
485
+
486
+ // slideAccess integration: http://trentrichardson.com/2011/11/11/jquery-ui-sliders-and-touch-accessibility/
487
+ if (this._defaults.addSliderAccess) {
488
+ var sliderAccessArgs = this._defaults.sliderAccessArgs,
489
+ rtl = this._defaults.isRTL;
490
+ sliderAccessArgs.isRTL = rtl;
491
+
492
+ setTimeout(function() { // fix for inline mode
493
+ if ($tp.find('.ui-slider-access').length === 0) {
494
+ $tp.find('.ui-slider:visible').sliderAccess(sliderAccessArgs);
495
+
496
+ // fix any grids since sliders are shorter
497
+ var sliderAccessWidth = $tp.find('.ui-slider-access:eq(0)').outerWidth(true);
498
+ if (sliderAccessWidth) {
499
+ $tp.find('table:visible').each(function() {
500
+ var $g = $(this),
501
+ oldWidth = $g.outerWidth(),
502
+ oldMarginLeft = $g.css(rtl? 'marginRight':'marginLeft').toString().replace('%', ''),
503
+ newWidth = oldWidth - sliderAccessWidth,
504
+ newMarginLeft = ((oldMarginLeft * newWidth) / oldWidth) + '%',
505
+ css = { width: newWidth, marginRight: 0, marginLeft: 0 };
506
+ css[rtl? 'marginRight':'marginLeft'] = newMarginLeft;
507
+ $g.css(css);
508
+ });
509
+ }
510
+ }
511
+ }, 10);
512
+ }
513
+ // end slideAccess integration
514
+
515
+ }
516
+ },
517
+
518
+ /*
519
+ * This function tries to limit the ability to go outside the
520
+ * min/max date range
521
+ */
522
+ _limitMinMaxDateTime: function(dp_inst, adjustSliders) {
523
+ var o = this._defaults,
524
+ dp_date = new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay);
525
+
526
+ if (!this._defaults.showTimepicker) {
527
+ return;
528
+ } // No time so nothing to check here
529
+
530
+ if ($.datepicker._get(dp_inst, 'minDateTime') !== null && $.datepicker._get(dp_inst, 'minDateTime') !== undefined && dp_date) {
531
+ var minDateTime = $.datepicker._get(dp_inst, 'minDateTime'),
532
+ minDateTimeDate = new Date(minDateTime.getFullYear(), minDateTime.getMonth(), minDateTime.getDate(), 0, 0, 0, 0);
533
+
534
+ if (this.hourMinOriginal === null || this.minuteMinOriginal === null || this.secondMinOriginal === null || this.millisecMinOriginal === null) {
535
+ this.hourMinOriginal = o.hourMin;
536
+ this.minuteMinOriginal = o.minuteMin;
537
+ this.secondMinOriginal = o.secondMin;
538
+ this.millisecMinOriginal = o.millisecMin;
539
+ }
540
+
541
+ if (dp_inst.settings.timeOnly || minDateTimeDate.getTime() == dp_date.getTime()) {
542
+ this._defaults.hourMin = minDateTime.getHours();
543
+ if (this.hour <= this._defaults.hourMin) {
544
+ this.hour = this._defaults.hourMin;
545
+ this._defaults.minuteMin = minDateTime.getMinutes();
546
+ if (this.minute <= this._defaults.minuteMin) {
547
+ this.minute = this._defaults.minuteMin;
548
+ this._defaults.secondMin = minDateTime.getSeconds();
549
+ if (this.second <= this._defaults.secondMin) {
550
+ this.second = this._defaults.secondMin;
551
+ this._defaults.millisecMin = minDateTime.getMilliseconds();
552
+ } else {
553
+ if (this.millisec < this._defaults.millisecMin) {
554
+ this.millisec = this._defaults.millisecMin;
555
+ }
556
+ this._defaults.millisecMin = this.millisecMinOriginal;
557
+ }
558
+ } else {
559
+ this._defaults.secondMin = this.secondMinOriginal;
560
+ this._defaults.millisecMin = this.millisecMinOriginal;
561
+ }
562
+ } else {
563
+ this._defaults.minuteMin = this.minuteMinOriginal;
564
+ this._defaults.secondMin = this.secondMinOriginal;
565
+ this._defaults.millisecMin = this.millisecMinOriginal;
566
+ }
567
+ } else {
568
+ this._defaults.hourMin = this.hourMinOriginal;
569
+ this._defaults.minuteMin = this.minuteMinOriginal;
570
+ this._defaults.secondMin = this.secondMinOriginal;
571
+ this._defaults.millisecMin = this.millisecMinOriginal;
572
+ }
573
+ }
574
+
575
+ if ($.datepicker._get(dp_inst, 'maxDateTime') !== null && $.datepicker._get(dp_inst, 'maxDateTime') !== undefined && dp_date) {
576
+ var maxDateTime = $.datepicker._get(dp_inst, 'maxDateTime'),
577
+ maxDateTimeDate = new Date(maxDateTime.getFullYear(), maxDateTime.getMonth(), maxDateTime.getDate(), 0, 0, 0, 0);
578
+
579
+ if (this.hourMaxOriginal === null || this.minuteMaxOriginal === null || this.secondMaxOriginal === null) {
580
+ this.hourMaxOriginal = o.hourMax;
581
+ this.minuteMaxOriginal = o.minuteMax;
582
+ this.secondMaxOriginal = o.secondMax;
583
+ this.millisecMaxOriginal = o.millisecMax;
584
+ }
585
+
586
+ if (dp_inst.settings.timeOnly || maxDateTimeDate.getTime() == dp_date.getTime()) {
587
+ this._defaults.hourMax = maxDateTime.getHours();
588
+ if (this.hour >= this._defaults.hourMax) {
589
+ this.hour = this._defaults.hourMax;
590
+ this._defaults.minuteMax = maxDateTime.getMinutes();
591
+ if (this.minute >= this._defaults.minuteMax) {
592
+ this.minute = this._defaults.minuteMax;
593
+ this._defaults.secondMax = maxDateTime.getSeconds();
594
+ } else if (this.second >= this._defaults.secondMax) {
595
+ this.second = this._defaults.secondMax;
596
+ this._defaults.millisecMax = maxDateTime.getMilliseconds();
597
+ } else {
598
+ if (this.millisec > this._defaults.millisecMax) {
599
+ this.millisec = this._defaults.millisecMax;
600
+ }
601
+ this._defaults.millisecMax = this.millisecMaxOriginal;
602
+ }
603
+ } else {
604
+ this._defaults.minuteMax = this.minuteMaxOriginal;
605
+ this._defaults.secondMax = this.secondMaxOriginal;
606
+ this._defaults.millisecMax = this.millisecMaxOriginal;
607
+ }
608
+ } else {
609
+ this._defaults.hourMax = this.hourMaxOriginal;
610
+ this._defaults.minuteMax = this.minuteMaxOriginal;
611
+ this._defaults.secondMax = this.secondMaxOriginal;
612
+ this._defaults.millisecMax = this.millisecMaxOriginal;
613
+ }
614
+ }
615
+
616
+ if (adjustSliders !== undefined && adjustSliders === true) {
617
+ var hourMax = parseInt((this._defaults.hourMax - ((this._defaults.hourMax - this._defaults.hourMin) % this._defaults.stepHour)), 10),
618
+ minMax = parseInt((this._defaults.minuteMax - ((this._defaults.minuteMax - this._defaults.minuteMin) % this._defaults.stepMinute)), 10),
619
+ secMax = parseInt((this._defaults.secondMax - ((this._defaults.secondMax - this._defaults.secondMin) % this._defaults.stepSecond)), 10),
620
+ millisecMax = parseInt((this._defaults.millisecMax - ((this._defaults.millisecMax - this._defaults.millisecMin) % this._defaults.stepMillisec)), 10);
621
+
622
+ if (this.hour_slider) {
623
+ this.control.options(this, this.hour_slider, 'hour', { min: this._defaults.hourMin, max: hourMax });
624
+ this.control.value(this, this.hour_slider, 'hour', this.hour - this.hour % this._defaults.stepHour);
625
+ }
626
+ if (this.minute_slider) {
627
+ this.control.options(this, this.minute_slider, 'minute', { min: this._defaults.minuteMin, max: minMax });
628
+ this.control.value(this, this.minute_slider, 'minute', this.minute - this.minute % this._defaults.stepMinute);
629
+ }
630
+ if (this.second_slider) {
631
+ this.control.options(this, this.second_slider, 'second', { min: this._defaults.secondMin, max: secMax });
632
+ this.control.value(this, this.second_slider, 'second', this.second - this.second % this._defaults.stepSecond);
633
+ }
634
+ if (this.millisec_slider) {
635
+ this.control.options(this, this.millisec_slider, 'millisec', { min: this._defaults.millisecMin, max: millisecMax });
636
+ this.control.value(this, this.millisec_slider, 'millisec', this.millisec - this.millisec % this._defaults.stepMillisec);
637
+ }
638
+ }
639
+
640
+ },
641
+
642
+ /*
643
+ * when a slider moves, set the internal time...
644
+ * on time change is also called when the time is updated in the text field
645
+ */
646
+ _onTimeChange: function() {
647
+ var hour = (this.hour_slider) ? this.control.value(this, this.hour_slider, 'hour') : false,
648
+ minute = (this.minute_slider) ? this.control.value(this, this.minute_slider, 'minute') : false,
649
+ second = (this.second_slider) ? this.control.value(this, this.second_slider, 'second') : false,
650
+ millisec = (this.millisec_slider) ? this.control.value(this, this.millisec_slider, 'millisec') : false,
651
+ timezone = (this.timezone_select) ? this.timezone_select.val() : false,
652
+ o = this._defaults,
653
+ pickerTimeFormat = o.pickerTimeFormat || o.timeFormat,
654
+ pickerTimeSuffix = o.pickerTimeSuffix || o.timeSuffix;
655
+
656
+ if (typeof(hour) == 'object') {
657
+ hour = false;
658
+ }
659
+ if (typeof(minute) == 'object') {
660
+ minute = false;
661
+ }
662
+ if (typeof(second) == 'object') {
663
+ second = false;
664
+ }
665
+ if (typeof(millisec) == 'object') {
666
+ millisec = false;
667
+ }
668
+ if (typeof(timezone) == 'object') {
669
+ timezone = false;
670
+ }
671
+
672
+ if (hour !== false) {
673
+ hour = parseInt(hour, 10);
674
+ }
675
+ if (minute !== false) {
676
+ minute = parseInt(minute, 10);
677
+ }
678
+ if (second !== false) {
679
+ second = parseInt(second, 10);
680
+ }
681
+ if (millisec !== false) {
682
+ millisec = parseInt(millisec, 10);
683
+ }
684
+
685
+ var ampm = o[hour < 12 ? 'amNames' : 'pmNames'][0];
686
+
687
+ // If the update was done in the input field, the input field should not be updated.
688
+ // If the update was done using the sliders, update the input field.
689
+ var hasChanged = (hour != this.hour || minute != this.minute || second != this.second || millisec != this.millisec
690
+ || (this.ampm.length > 0 && (hour < 12) != ($.inArray(this.ampm.toUpperCase(), this.amNames) !== -1))
691
+ || ((this.timezone === null && timezone != this.defaultTimezone) || (this.timezone !== null && timezone != this.timezone)));
692
+
693
+ if (hasChanged) {
694
+
695
+ if (hour !== false) {
696
+ this.hour = hour;
697
+ }
698
+ if (minute !== false) {
699
+ this.minute = minute;
700
+ }
701
+ if (second !== false) {
702
+ this.second = second;
703
+ }
704
+ if (millisec !== false) {
705
+ this.millisec = millisec;
706
+ }
707
+ if (timezone !== false) {
708
+ this.timezone = timezone;
709
+ }
710
+
711
+ if (!this.inst) {
712
+ this.inst = $.datepicker._getInst(this.$input[0]);
713
+ }
714
+
715
+ this._limitMinMaxDateTime(this.inst, true);
716
+ }
717
+ if (useAmpm(o.timeFormat)) {
718
+ this.ampm = ampm;
719
+ }
720
+
721
+ // Updates the time within the timepicker
722
+ this.formattedTime = $.datepicker.formatTime(o.timeFormat, this, o);
723
+ if (this.$timeObj) {
724
+ if(pickerTimeFormat === o.timeFormat){
725
+ this.$timeObj.text(this.formattedTime + pickerTimeSuffix);
726
+ }
727
+ else{
728
+ this.$timeObj.text($.datepicker.formatTime(pickerTimeFormat, this, o) + pickerTimeSuffix);
729
+ }
730
+ }
731
+
732
+ this.timeDefined = true;
733
+ if (hasChanged) {
734
+ this._updateDateTime();
735
+ }
736
+ },
737
+
738
+ /*
739
+ * call custom onSelect.
740
+ * bind to sliders slidestop, and grid click.
741
+ */
742
+ _onSelectHandler: function() {
743
+ var onSelect = this._defaults.onSelect || this.inst.settings.onSelect;
744
+ var inputEl = this.$input ? this.$input[0] : null;
745
+ if (onSelect && inputEl) {
746
+ onSelect.apply(inputEl, [this.formattedDateTime, this]);
747
+ }
748
+ },
749
+
750
+ /*
751
+ * update our input with the new date time..
752
+ */
753
+ _updateDateTime: function(dp_inst) {
754
+ dp_inst = this.inst || dp_inst;
755
+ var dt = $.datepicker._daylightSavingAdjust(new Date(dp_inst.selectedYear, dp_inst.selectedMonth, dp_inst.selectedDay)),
756
+ dateFmt = $.datepicker._get(dp_inst, 'dateFormat'),
757
+ formatCfg = $.datepicker._getFormatConfig(dp_inst),
758
+ timeAvailable = dt !== null && this.timeDefined;
759
+ this.formattedDate = $.datepicker.formatDate(dateFmt, (dt === null ? new Date() : dt), formatCfg);
760
+ var formattedDateTime = this.formattedDate;
761
+
762
+ /*
763
+ * remove following lines to force every changes in date picker to change the input value
764
+ * Bug descriptions: when an input field has a default value, and click on the field to pop up the date picker.
765
+ * If the user manually empty the value in the input field, the date picker will never change selected value.
766
+ */
767
+ //if (dp_inst.lastVal !== undefined && (dp_inst.lastVal.length > 0 && this.$input.val().length === 0)) {
768
+ // return;
769
+ //}
770
+
771
+ if (this._defaults.timeOnly === true) {
772
+ formattedDateTime = this.formattedTime;
773
+ } else if (this._defaults.timeOnly !== true && (this._defaults.alwaysSetTime || timeAvailable)) {
774
+ formattedDateTime += this._defaults.separator + this.formattedTime + this._defaults.timeSuffix;
775
+ }
776
+
777
+ this.formattedDateTime = formattedDateTime;
778
+
779
+ if (!this._defaults.showTimepicker) {
780
+ this.$input.val(this.formattedDate);
781
+ } else if (this.$altInput && this._defaults.altFieldTimeOnly === true) {
782
+ this.$altInput.val(this.formattedTime);
783
+ this.$input.val(this.formattedDate);
784
+ } else if (this.$altInput) {
785
+ this.$input.val(formattedDateTime);
786
+ var altFormattedDateTime = '',
787
+ altSeparator = this._defaults.altSeparator ? this._defaults.altSeparator : this._defaults.separator,
788
+ altTimeSuffix = this._defaults.altTimeSuffix ? this._defaults.altTimeSuffix : this._defaults.timeSuffix;
789
+
790
+ if (this._defaults.altFormat) altFormattedDateTime = $.datepicker.formatDate(this._defaults.altFormat, (dt === null ? new Date() : dt), formatCfg);
791
+ else altFormattedDateTime = this.formattedDate;
792
+ if (altFormattedDateTime) altFormattedDateTime += altSeparator;
793
+ if (this._defaults.altTimeFormat) altFormattedDateTime += $.datepicker.formatTime(this._defaults.altTimeFormat, this, this._defaults) + altTimeSuffix;
794
+ else altFormattedDateTime += this.formattedTime + altTimeSuffix;
795
+ this.$altInput.val(altFormattedDateTime);
796
+ } else {
797
+ this.$input.val(formattedDateTime);
798
+ }
799
+
800
+ this.$input.trigger("change");
801
+ },
802
+
803
+ _onFocus: function() {
804
+ if (!this.$input.val() && this._defaults.defaultValue) {
805
+ this.$input.val(this._defaults.defaultValue);
806
+ var inst = $.datepicker._getInst(this.$input.get(0)),
807
+ tp_inst = $.datepicker._get(inst, 'timepicker');
808
+ if (tp_inst) {
809
+ if (tp_inst._defaults.timeOnly && (inst.input.val() != inst.lastVal)) {
810
+ try {
811
+ $.datepicker._updateDatepicker(inst);
812
+ } catch (err) {
813
+ $.datepicker.log(err);
814
+ }
815
+ }
816
+ }
817
+ }
818
+ },
819
+
820
+ /*
821
+ * Small abstraction to control types
822
+ * We can add more, just be sure to follow the pattern: create, options, value
823
+ */
824
+ _controls: {
825
+ // slider methods
826
+ slider: {
827
+ create: function(tp_inst, obj, unit, val, min, max, step){
828
+ var rtl = tp_inst._defaults.isRTL; // if rtl go -60->0 instead of 0->60
829
+ return obj.prop('slide', null).slider({
830
+ orientation: "horizontal",
831
+ value: rtl? val*-1 : val,
832
+ min: rtl? max*-1 : min,
833
+ max: rtl? min*-1 : max,
834
+ step: step,
835
+ slide: function(event, ui) {
836
+ tp_inst.control.value(tp_inst, $(this), unit, rtl? ui.value*-1:ui.value);
837
+ tp_inst._onTimeChange();
838
+ },
839
+ stop: function(event, ui) {
840
+ tp_inst._onSelectHandler();
841
+ }
842
+ });
843
+ },
844
+ options: function(tp_inst, obj, unit, opts, val){
845
+ if(tp_inst._defaults.isRTL){
846
+ if(typeof(opts) == 'string'){
847
+ if(opts == 'min' || opts == 'max'){
848
+ if(val !== undefined)
849
+ return obj.slider(opts, val*-1);
850
+ return Math.abs(obj.slider(opts));
851
+ }
852
+ return obj.slider(opts);
853
+ }
854
+ var min = opts.min,
855
+ max = opts.max;
856
+ opts.min = opts.max = null;
857
+ if(min !== undefined)
858
+ opts.max = min * -1;
859
+ if(max !== undefined)
860
+ opts.min = max * -1;
861
+ return obj.slider(opts);
862
+ }
863
+ if(typeof(opts) == 'string' && val !== undefined)
864
+ return obj.slider(opts, val);
865
+ return obj.slider(opts);
866
+ },
867
+ value: function(tp_inst, obj, unit, val){
868
+ if(tp_inst._defaults.isRTL){
869
+ if(val !== undefined)
870
+ return obj.slider('value', val*-1);
871
+ return Math.abs(obj.slider('value'));
872
+ }
873
+ if(val !== undefined)
874
+ return obj.slider('value', val);
875
+ return obj.slider('value');
876
+ }
877
+ },
878
+ // select methods
879
+ select: {
880
+ create: function(tp_inst, obj, unit, val, min, max, step){
881
+ var sel = '<select class="ui-timepicker-select" data-unit="'+ unit +'" data-min="'+ min +'" data-max="'+ max +'" data-step="'+ step +'">',
882
+ ul = tp_inst._defaults.timeFormat.indexOf('t') !== -1? 'toLowerCase':'toUpperCase',
883
+ m = 0;
884
+
885
+ for(var i=min; i<=max; i+=step){
886
+ sel += '<option value="'+ i +'"'+ (i==val? ' selected':'') +'>';
887
+ if(unit == 'hour' && useAmpm(tp_inst._defaults.pickerTimeFormat || tp_inst._defaults.timeFormat))
888
+ sel += $.datepicker.formatTime("hh TT", {hour:i}, tp_inst._defaults);
889
+ else if(unit == 'millisec' || i >= 10) sel += i;
890
+ else sel += '0'+ i.toString();
891
+ sel += '</option>';
892
+ }
893
+ sel += '</select>';
894
+
895
+ obj.children('select').remove();
896
+
897
+ $(sel).appendTo(obj).change(function(e){
898
+ tp_inst._onTimeChange();
899
+ tp_inst._onSelectHandler();
900
+ });
901
+
902
+ return obj;
903
+ },
904
+ options: function(tp_inst, obj, unit, opts, val){
905
+ var o = {},
906
+ $t = obj.children('select');
907
+ if(typeof(opts) == 'string'){
908
+ if(val === undefined)
909
+ return $t.data(opts);
910
+ o[opts] = val;
911
+ }
912
+ else o = opts;
913
+ return tp_inst.control.create(tp_inst, obj, $t.data('unit'), $t.val(), o.min || $t.data('min'), o.max || $t.data('max'), o.step || $t.data('step'));
914
+ },
915
+ value: function(tp_inst, obj, unit, val){
916
+ var $t = obj.children('select');
917
+ if(val !== undefined)
918
+ return $t.val(val);
919
+ return $t.val();
920
+ }
921
+ }
922
+ } // end _controls
923
+
924
+ });
925
+
926
+ $.fn.extend({
927
+ /*
928
+ * shorthand just to use timepicker..
929
+ */
930
+ timepicker: function(o) {
931
+ o = o || {};
932
+ var tmp_args = Array.prototype.slice.call(arguments);
933
+
934
+ if (typeof o == 'object') {
935
+ tmp_args[0] = $.extend(o, {
936
+ timeOnly: true
937
+ });
938
+ }
939
+
940
+ return $(this).each(function() {
941
+ $.fn.datetimepicker.apply($(this), tmp_args);
942
+ });
943
+ },
944
+
945
+ /*
946
+ * extend timepicker to datepicker
947
+ */
948
+ datetimepicker: function(o) {
949
+ o = o || {};
950
+ var tmp_args = arguments;
951
+
952
+ if (typeof(o) == 'string') {
953
+ if (o == 'getDate') {
954
+ return $.fn.datepicker.apply($(this[0]), tmp_args);
955
+ } else {
956
+ return this.each(function() {
957
+ var $t = $(this);
958
+ $t.datepicker.apply($t, tmp_args);
959
+ });
960
+ }
961
+ } else {
962
+ return this.each(function() {
963
+ var $t = $(this);
964
+ $t.datepicker($.timepicker._newInst($t, o)._defaults);
965
+ });
966
+ }
967
+ }
968
+ });
969
+
970
+ /*
971
+ * Public Utility to parse date and time
972
+ */
973
+ $.datepicker.parseDateTime = function(dateFormat, timeFormat, dateTimeString, dateSettings, timeSettings) {
974
+ var parseRes = parseDateTimeInternal(dateFormat, timeFormat, dateTimeString, dateSettings, timeSettings);
975
+ if (parseRes.timeObj) {
976
+ var t = parseRes.timeObj;
977
+ parseRes.date.setHours(t.hour, t.minute, t.second, t.millisec);
978
+ }
979
+
980
+ return parseRes.date;
981
+ };
982
+
983
+ /*
984
+ * Public utility to parse time
985
+ */
986
+ $.datepicker.parseTime = function(timeFormat, timeString, options) {
987
+ var o = extendRemove(extendRemove({}, $.timepicker._defaults), options || {});
988
+
989
+ // Strict parse requires the timeString to match the timeFormat exactly
990
+ var strictParse = function(f, s, o){
991
+
992
+ // pattern for standard and localized AM/PM markers
993
+ var getPatternAmpm = function(amNames, pmNames) {
994
+ var markers = [];
995
+ if (amNames) {
996
+ $.merge(markers, amNames);
997
+ }
998
+ if (pmNames) {
999
+ $.merge(markers, pmNames);
1000
+ }
1001
+ markers = $.map(markers, function(val) {
1002
+ return val.replace(/[.*+?|()\[\]{}\\]/g, '\\$&');
1003
+ });
1004
+ return '(' + markers.join('|') + ')?';
1005
+ };
1006
+
1007
+ // figure out position of time elements.. cause js cant do named captures
1008
+ var getFormatPositions = function(timeFormat) {
1009
+ var finds = timeFormat.toLowerCase().match(/(h{1,2}|m{1,2}|s{1,2}|l{1}|t{1,2}|z|'.*?')/g),
1010
+ orders = {
1011
+ h: -1,
1012
+ m: -1,
1013
+ s: -1,
1014
+ l: -1,
1015
+ t: -1,
1016
+ z: -1
1017
+ };
1018
+
1019
+ if (finds) {
1020
+ for (var i = 0; i < finds.length; i++) {
1021
+ if (orders[finds[i].toString().charAt(0)] == -1) {
1022
+ orders[finds[i].toString().charAt(0)] = i + 1;
1023
+ }
1024
+ }
1025
+ }
1026
+ return orders;
1027
+ };
1028
+
1029
+ var regstr = '^' + f.toString()
1030
+ .replace(/([hH]{1,2}|mm?|ss?|[tT]{1,2}|[lz]|'.*?')/g, function (match) {
1031
+ switch (match.charAt(0).toLowerCase()) {
1032
+ case 'h': return '(\\d?\\d)';
1033
+ case 'm': return '(\\d?\\d)';
1034
+ case 's': return '(\\d?\\d)';
1035
+ case 'l': return '(\\d?\\d?\\d)';
1036
+ case 'z': return '(z|[-+]\\d\\d:?\\d\\d|\\S+)?';
1037
+ case 't': return getPatternAmpm(o.amNames, o.pmNames);
1038
+ default: // literal escaped in quotes
1039
+ return '(' + match.replace(/\'/g, "").replace(/(\.|\$|\^|\\|\/|\(|\)|\[|\]|\?|\+|\*)/g, function (m) { return "\\" + m; }) + ')?';
1040
+ }
1041
+ })
1042
+ .replace(/\s/g, '\\s?') +
1043
+ o.timeSuffix + '$',
1044
+ order = getFormatPositions(f),
1045
+ ampm = '',
1046
+ treg;
1047
+
1048
+ treg = s.match(new RegExp(regstr, 'i'));
1049
+
1050
+ var resTime = {
1051
+ hour: 0,
1052
+ minute: 0,
1053
+ second: 0,
1054
+ millisec: 0
1055
+ };
1056
+
1057
+ if (treg) {
1058
+ if (order.t !== -1) {
1059
+ if (treg[order.t] === undefined || treg[order.t].length === 0) {
1060
+ ampm = '';
1061
+ resTime.ampm = '';
1062
+ } else {
1063
+ ampm = $.inArray(treg[order.t].toUpperCase(), o.amNames) !== -1 ? 'AM' : 'PM';
1064
+ resTime.ampm = o[ampm == 'AM' ? 'amNames' : 'pmNames'][0];
1065
+ }
1066
+ }
1067
+
1068
+ if (order.h !== -1) {
1069
+ if (ampm == 'AM' && treg[order.h] == '12') {
1070
+ resTime.hour = 0; // 12am = 0 hour
1071
+ } else {
1072
+ if (ampm == 'PM' && treg[order.h] != '12') {
1073
+ resTime.hour = parseInt(treg[order.h], 10) + 12; // 12pm = 12 hour, any other pm = hour + 12
1074
+ } else {
1075
+ resTime.hour = Number(treg[order.h]);
1076
+ }
1077
+ }
1078
+ }
1079
+
1080
+ if (order.m !== -1) {
1081
+ resTime.minute = Number(treg[order.m]);
1082
+ }
1083
+ if (order.s !== -1) {
1084
+ resTime.second = Number(treg[order.s]);
1085
+ }
1086
+ if (order.l !== -1) {
1087
+ resTime.millisec = Number(treg[order.l]);
1088
+ }
1089
+ if (order.z !== -1 && treg[order.z] !== undefined) {
1090
+ var tz = treg[order.z].toUpperCase();
1091
+ switch (tz.length) {
1092
+ case 1:
1093
+ // Z
1094
+ tz = o.timezoneIso8601 ? 'Z' : '+0000';
1095
+ break;
1096
+ case 5:
1097
+ // +hhmm
1098
+ if (o.timezoneIso8601) {
1099
+ tz = tz.substring(1) == '0000' ? 'Z' : tz.substring(0, 3) + ':' + tz.substring(3);
1100
+ }
1101
+ break;
1102
+ case 6:
1103
+ // +hh:mm
1104
+ if (!o.timezoneIso8601) {
1105
+ tz = tz == 'Z' || tz.substring(1) == '00:00' ? '+0000' : tz.replace(/:/, '');
1106
+ } else {
1107
+ if (tz.substring(1) == '00:00') {
1108
+ tz = 'Z';
1109
+ }
1110
+ }
1111
+ break;
1112
+ }
1113
+ resTime.timezone = tz;
1114
+ }
1115
+
1116
+
1117
+ return resTime;
1118
+ }
1119
+ return false;
1120
+ };// end strictParse
1121
+
1122
+ // First try JS Date, if that fails, use strictParse
1123
+ var looseParse = function(f,s,o){
1124
+ try{
1125
+ var d = new Date('2012-01-01 '+ s);
1126
+ return {
1127
+ hour: d.getHours(),
1128
+ minutes: d.getMinutes(),
1129
+ seconds: d.getSeconds(),
1130
+ millisec: d.getMilliseconds(),
1131
+ timezone: $.timepicker.timeZoneOffsetString(d)
1132
+ };
1133
+ }
1134
+ catch(err){
1135
+ try{
1136
+ return strictParse(f,s,o);
1137
+ }
1138
+ catch(err2){
1139
+ $.datepicker.log("Unable to parse \ntimeString: "+ s +"\ntimeFormat: "+ f);
1140
+ }
1141
+ }
1142
+ return false;
1143
+ }; // end looseParse
1144
+
1145
+ if(typeof o.parse === "function"){
1146
+ return o.parse(timeFormat, timeString, o)
1147
+ }
1148
+ if(o.parse === 'loose'){
1149
+ return looseParse(timeFormat, timeString, o);
1150
+ }
1151
+ return strictParse(timeFormat, timeString, o);
1152
+ };
1153
+
1154
+ /*
1155
+ * Public utility to format the time
1156
+ * format = string format of the time
1157
+ * time = a {}, not a Date() for timezones
1158
+ * options = essentially the regional[].. amNames, pmNames, ampm
1159
+ */
1160
+ $.datepicker.formatTime = function(format, time, options) {
1161
+ options = options || {};
1162
+ options = $.extend({}, $.timepicker._defaults, options);
1163
+ time = $.extend({
1164
+ hour: 0,
1165
+ minute: 0,
1166
+ second: 0,
1167
+ millisec: 0,
1168
+ timezone: '+0000'
1169
+ }, time);
1170
+
1171
+ var tmptime = format,
1172
+ ampmName = options.amNames[0],
1173
+ hour = parseInt(time.hour, 10);
1174
+
1175
+ if (hour > 11) {
1176
+ ampmName = options.pmNames[0];
1177
+ }
1178
+
1179
+ tmptime = tmptime.replace(/(?:HH?|hh?|mm?|ss?|[tT]{1,2}|[lz]|('.*?'|".*?"))/g, function(match) {
1180
+ switch (match) {
1181
+ case 'HH':
1182
+ return ('0' + hour).slice(-2);
1183
+ case 'H':
1184
+ return hour;
1185
+ case 'hh':
1186
+ return ('0' + convert24to12(hour)).slice(-2);
1187
+ case 'h':
1188
+ return convert24to12(hour);
1189
+ case 'mm':
1190
+ return ('0' + time.minute).slice(-2);
1191
+ case 'm':
1192
+ return time.minute;
1193
+ case 'ss':
1194
+ return ('0' + time.second).slice(-2);
1195
+ case 's':
1196
+ return time.second;
1197
+ case 'l':
1198
+ return ('00' + time.millisec).slice(-3);
1199
+ case 'z':
1200
+ return time.timezone === null? options.defaultTimezone : time.timezone;
1201
+ case 'T':
1202
+ return ampmName.charAt(0).toUpperCase();
1203
+ case 'TT':
1204
+ return ampmName.toUpperCase();
1205
+ case 't':
1206
+ return ampmName.charAt(0).toLowerCase();
1207
+ case 'tt':
1208
+ return ampmName.toLowerCase();
1209
+ default:
1210
+ return match.replace(/\'/g, "") || "'";
1211
+ }
1212
+ });
1213
+
1214
+ tmptime = $.trim(tmptime);
1215
+ return tmptime;
1216
+ };
1217
+
1218
+ /*
1219
+ * the bad hack :/ override datepicker so it doesnt close on select
1220
+ // inspired: http://stackoverflow.com/questions/1252512/jquery-datepicker-prevent-closing-picker-when-clicking-a-date/1762378#1762378
1221
+ */
1222
+ $.datepicker._base_selectDate = $.datepicker._selectDate;
1223
+ $.datepicker._selectDate = function(id, dateStr) {
1224
+ var inst = this._getInst($(id)[0]),
1225
+ tp_inst = this._get(inst, 'timepicker');
1226
+
1227
+ if (tp_inst) {
1228
+ tp_inst._limitMinMaxDateTime(inst, true);
1229
+ inst.inline = inst.stay_open = true;
1230
+ //This way the onSelect handler called from calendarpicker get the full dateTime
1231
+ this._base_selectDate(id, dateStr);
1232
+ inst.inline = inst.stay_open = false;
1233
+ this._notifyChange(inst);
1234
+ this._updateDatepicker(inst);
1235
+ } else {
1236
+ this._base_selectDate(id, dateStr);
1237
+ }
1238
+ };
1239
+
1240
+ /*
1241
+ * second bad hack :/ override datepicker so it triggers an event when changing the input field
1242
+ * and does not redraw the datepicker on every selectDate event
1243
+ */
1244
+ $.datepicker._base_updateDatepicker = $.datepicker._updateDatepicker;
1245
+ $.datepicker._updateDatepicker = function(inst) {
1246
+
1247
+ // don't popup the datepicker if there is another instance already opened
1248
+ var input = inst.input[0];
1249
+ if ($.datepicker._curInst && $.datepicker._curInst != inst && $.datepicker._datepickerShowing && $.datepicker._lastInput != input) {
1250
+ return;
1251
+ }
1252
+
1253
+ if (typeof(inst.stay_open) !== 'boolean' || inst.stay_open === false) {
1254
+
1255
+ this._base_updateDatepicker(inst);
1256
+
1257
+ // Reload the time control when changing something in the input text field.
1258
+ var tp_inst = this._get(inst, 'timepicker');
1259
+ if (tp_inst) {
1260
+ tp_inst._addTimePicker(inst);
1261
+
1262
+ if (tp_inst._defaults.useLocalTimezone) { //checks daylight saving with the new date.
1263
+ var date = new Date(inst.selectedYear, inst.selectedMonth, inst.selectedDay, 12);
1264
+ selectLocalTimeZone(tp_inst, date);
1265
+ tp_inst._onTimeChange();
1266
+ }
1267
+ }
1268
+ }
1269
+ };
1270
+
1271
+ /*
1272
+ * third bad hack :/ override datepicker so it allows spaces and colon in the input field
1273
+ */
1274
+ $.datepicker._base_doKeyPress = $.datepicker._doKeyPress;
1275
+ $.datepicker._doKeyPress = function(event) {
1276
+ var inst = $.datepicker._getInst(event.target),
1277
+ tp_inst = $.datepicker._get(inst, 'timepicker');
1278
+
1279
+ if (tp_inst) {
1280
+ if ($.datepicker._get(inst, 'constrainInput')) {
1281
+ var ampm = useAmpm(tp_inst._defaults.timeFormat),
1282
+ dateChars = $.datepicker._possibleChars($.datepicker._get(inst, 'dateFormat')),
1283
+ datetimeChars = tp_inst._defaults.timeFormat.toString()
1284
+ .replace(/[hms]/g, '')
1285
+ .replace(/TT/g, ampm ? 'APM' : '')
1286
+ .replace(/Tt/g, ampm ? 'AaPpMm' : '')
1287
+ .replace(/tT/g, ampm ? 'AaPpMm' : '')
1288
+ .replace(/T/g, ampm ? 'AP' : '')
1289
+ .replace(/tt/g, ampm ? 'apm' : '')
1290
+ .replace(/t/g, ampm ? 'ap' : '') +
1291
+ " " + tp_inst._defaults.separator +
1292
+ tp_inst._defaults.timeSuffix +
1293
+ (tp_inst._defaults.showTimezone ? tp_inst._defaults.timezoneList.join('') : '') +
1294
+ (tp_inst._defaults.amNames.join('')) + (tp_inst._defaults.pmNames.join('')) +
1295
+ dateChars,
1296
+ chr = String.fromCharCode(event.charCode === undefined ? event.keyCode : event.charCode);
1297
+ return event.ctrlKey || (chr < ' ' || !dateChars || datetimeChars.indexOf(chr) > -1);
1298
+ }
1299
+ }
1300
+
1301
+ return $.datepicker._base_doKeyPress(event);
1302
+ };
1303
+
1304
+ /*
1305
+ * Fourth bad hack :/ override _updateAlternate function used in inline mode to init altField
1306
+ */
1307
+ $.datepicker._base_updateAlternate = $.datepicker._updateAlternate;
1308
+ /* Update any alternate field to synchronise with the main field. */
1309
+ $.datepicker._updateAlternate = function(inst) {
1310
+ var tp_inst = this._get(inst, 'timepicker');
1311
+ if(tp_inst){
1312
+ var altField = tp_inst._defaults.altField;
1313
+ if (altField) { // update alternate field too
1314
+ var altFormat = tp_inst._defaults.altFormat || tp_inst._defaults.dateFormat,
1315
+ date = this._getDate(inst),
1316
+ formatCfg = $.datepicker._getFormatConfig(inst),
1317
+ altFormattedDateTime = '',
1318
+ altSeparator = tp_inst._defaults.altSeparator ? tp_inst._defaults.altSeparator : tp_inst._defaults.separator,
1319
+ altTimeSuffix = tp_inst._defaults.altTimeSuffix ? tp_inst._defaults.altTimeSuffix : tp_inst._defaults.timeSuffix,
1320
+ altTimeFormat = tp_inst._defaults.altTimeFormat !== null ? tp_inst._defaults.altTimeFormat : tp_inst._defaults.timeFormat;
1321
+
1322
+ altFormattedDateTime += $.datepicker.formatTime(altTimeFormat, tp_inst, tp_inst._defaults) + altTimeSuffix;
1323
+ if(!tp_inst._defaults.timeOnly && !tp_inst._defaults.altFieldTimeOnly){
1324
+ if(tp_inst._defaults.altFormat)
1325
+ altFormattedDateTime = $.datepicker.formatDate(tp_inst._defaults.altFormat, (date === null ? new Date() : date), formatCfg) + altSeparator + altFormattedDateTime;
1326
+ else altFormattedDateTime = tp_inst.formattedDate + altSeparator + altFormattedDateTime;
1327
+ }
1328
+ $(altField).val(altFormattedDateTime);
1329
+ }
1330
+ }
1331
+ else{
1332
+ $.datepicker._base_updateAlternate(inst);
1333
+ }
1334
+ };
1335
+
1336
+ /*
1337
+ * Override key up event to sync manual input changes.
1338
+ */
1339
+ $.datepicker._base_doKeyUp = $.datepicker._doKeyUp;
1340
+ $.datepicker._doKeyUp = function(event) {
1341
+ var inst = $.datepicker._getInst(event.target),
1342
+ tp_inst = $.datepicker._get(inst, 'timepicker');
1343
+
1344
+ if (tp_inst) {
1345
+ if (tp_inst._defaults.timeOnly && (inst.input.val() != inst.lastVal)) {
1346
+ try {
1347
+ $.datepicker._updateDatepicker(inst);
1348
+ } catch (err) {
1349
+ $.datepicker.log(err);
1350
+ }
1351
+ }
1352
+ }
1353
+
1354
+ return $.datepicker._base_doKeyUp(event);
1355
+ };
1356
+
1357
+ /*
1358
+ * override "Today" button to also grab the time.
1359
+ */
1360
+ $.datepicker._base_gotoToday = $.datepicker._gotoToday;
1361
+ $.datepicker._gotoToday = function(id) {
1362
+ var inst = this._getInst($(id)[0]),
1363
+ $dp = inst.dpDiv;
1364
+ this._base_gotoToday(id);
1365
+ var tp_inst = this._get(inst, 'timepicker');
1366
+ selectLocalTimeZone(tp_inst);
1367
+ var now = new Date();
1368
+ this._setTime(inst, now);
1369
+ $('.ui-datepicker-today', $dp).click();
1370
+ };
1371
+
1372
+ /*
1373
+ * Disable & enable the Time in the datetimepicker
1374
+ */
1375
+ $.datepicker._disableTimepickerDatepicker = function(target) {
1376
+ var inst = this._getInst(target);
1377
+ if (!inst) {
1378
+ return;
1379
+ }
1380
+
1381
+ var tp_inst = this._get(inst, 'timepicker');
1382
+ $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1383
+ if (tp_inst) {
1384
+ tp_inst._defaults.showTimepicker = false;
1385
+ tp_inst._updateDateTime(inst);
1386
+ }
1387
+ };
1388
+
1389
+ $.datepicker._enableTimepickerDatepicker = function(target) {
1390
+ var inst = this._getInst(target);
1391
+ if (!inst) {
1392
+ return;
1393
+ }
1394
+
1395
+ var tp_inst = this._get(inst, 'timepicker');
1396
+ $(target).datepicker('getDate'); // Init selected[Year|Month|Day]
1397
+ if (tp_inst) {
1398
+ tp_inst._defaults.showTimepicker = true;
1399
+ tp_inst._addTimePicker(inst); // Could be disabled on page load
1400
+ tp_inst._updateDateTime(inst);
1401
+ }
1402
+ };
1403
+
1404
+ /*
1405
+ * Create our own set time function
1406
+ */
1407
+ $.datepicker._setTime = function(inst, date) {
1408
+ var tp_inst = this._get(inst, 'timepicker');
1409
+ if (tp_inst) {
1410
+ var defaults = tp_inst._defaults;
1411
+
1412
+ // calling _setTime with no date sets time to defaults
1413
+ tp_inst.hour = date ? date.getHours() : defaults.hour;
1414
+ tp_inst.minute = date ? date.getMinutes() : defaults.minute;
1415
+ tp_inst.second = date ? date.getSeconds() : defaults.second;
1416
+ tp_inst.millisec = date ? date.getMilliseconds() : defaults.millisec;
1417
+
1418
+ //check if within min/max times..
1419
+ tp_inst._limitMinMaxDateTime(inst, true);
1420
+
1421
+ tp_inst._onTimeChange();
1422
+ tp_inst._updateDateTime(inst);
1423
+ }
1424
+ };
1425
+
1426
+ /*
1427
+ * Create new public method to set only time, callable as $().datepicker('setTime', date)
1428
+ */
1429
+ $.datepicker._setTimeDatepicker = function(target, date, withDate) {
1430
+ var inst = this._getInst(target);
1431
+ if (!inst) {
1432
+ return;
1433
+ }
1434
+
1435
+ var tp_inst = this._get(inst, 'timepicker');
1436
+
1437
+ if (tp_inst) {
1438
+ this._setDateFromField(inst);
1439
+ var tp_date;
1440
+ if (date) {
1441
+ if (typeof date == "string") {
1442
+ tp_inst._parseTime(date, withDate);
1443
+ tp_date = new Date();
1444
+ tp_date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1445
+ } else {
1446
+ tp_date = new Date(date.getTime());
1447
+ }
1448
+ if (tp_date.toString() == 'Invalid Date') {
1449
+ tp_date = undefined;
1450
+ }
1451
+ this._setTime(inst, tp_date);
1452
+ }
1453
+ }
1454
+
1455
+ };
1456
+
1457
+ /*
1458
+ * override setDate() to allow setting time too within Date object
1459
+ */
1460
+ $.datepicker._base_setDateDatepicker = $.datepicker._setDateDatepicker;
1461
+ $.datepicker._setDateDatepicker = function(target, date) {
1462
+ var inst = this._getInst(target);
1463
+ if (!inst) {
1464
+ return;
1465
+ }
1466
+
1467
+ var tp_date = (date instanceof Date) ? new Date(date.getTime()) : date;
1468
+
1469
+ this._updateDatepicker(inst);
1470
+ this._base_setDateDatepicker.apply(this, arguments);
1471
+ this._setTimeDatepicker(target, tp_date, true);
1472
+ };
1473
+
1474
+ /*
1475
+ * override getDate() to allow getting time too within Date object
1476
+ */
1477
+ $.datepicker._base_getDateDatepicker = $.datepicker._getDateDatepicker;
1478
+ $.datepicker._getDateDatepicker = function(target, noDefault) {
1479
+ var inst = this._getInst(target);
1480
+ if (!inst) {
1481
+ return;
1482
+ }
1483
+
1484
+ var tp_inst = this._get(inst, 'timepicker');
1485
+
1486
+ if (tp_inst) {
1487
+ // if it hasn't yet been defined, grab from field
1488
+ if(inst.lastVal === undefined){
1489
+ this._setDateFromField(inst, noDefault);
1490
+ }
1491
+
1492
+ var date = this._getDate(inst);
1493
+ if (date && tp_inst._parseTime($(target).val(), tp_inst.timeOnly)) {
1494
+ date.setHours(tp_inst.hour, tp_inst.minute, tp_inst.second, tp_inst.millisec);
1495
+ }
1496
+ return date;
1497
+ }
1498
+ return this._base_getDateDatepicker(target, noDefault);
1499
+ };
1500
+
1501
+ /*
1502
+ * override parseDate() because UI 1.8.14 throws an error about "Extra characters"
1503
+ * An option in datapicker to ignore extra format characters would be nicer.
1504
+ */
1505
+ $.datepicker._base_parseDate = $.datepicker.parseDate;
1506
+ $.datepicker.parseDate = function(format, value, settings) {
1507
+ var date;
1508
+ try {
1509
+ date = this._base_parseDate(format, value, settings);
1510
+ } catch (err) {
1511
+ // Hack! The error message ends with a colon, a space, and
1512
+ // the "extra" characters. We rely on that instead of
1513
+ // attempting to perfectly reproduce the parsing algorithm.
1514
+ date = this._base_parseDate(format, value.substring(0,value.length-(err.length-err.indexOf(':')-2)), settings);
1515
+ $.datepicker.log("Error parsing the date string: " + err + "\ndate string = " + value + "\ndate format = " + format);
1516
+ }
1517
+ return date;
1518
+ };
1519
+
1520
+ /*
1521
+ * override formatDate to set date with time to the input
1522
+ */
1523
+ $.datepicker._base_formatDate = $.datepicker._formatDate;
1524
+ $.datepicker._formatDate = function(inst, day, month, year) {
1525
+ var tp_inst = this._get(inst, 'timepicker');
1526
+ if (tp_inst) {
1527
+ tp_inst._updateDateTime(inst);
1528
+ return tp_inst.$input.val();
1529
+ }
1530
+ return this._base_formatDate(inst);
1531
+ };
1532
+
1533
+ /*
1534
+ * override options setter to add time to maxDate(Time) and minDate(Time). MaxDate
1535
+ */
1536
+ $.datepicker._base_optionDatepicker = $.datepicker._optionDatepicker;
1537
+ $.datepicker._optionDatepicker = function(target, name, value) {
1538
+ var inst = this._getInst(target),
1539
+ name_clone;
1540
+ if (!inst) {
1541
+ return null;
1542
+ }
1543
+
1544
+ var tp_inst = this._get(inst, 'timepicker');
1545
+ if (tp_inst) {
1546
+ var min = null,
1547
+ max = null,
1548
+ onselect = null,
1549
+ overrides = tp_inst._defaults.evnts,
1550
+ fns = {},
1551
+ prop;
1552
+ if (typeof name == 'string') { // if min/max was set with the string
1553
+ if (name === 'minDate' || name === 'minDateTime') {
1554
+ min = value;
1555
+ } else if (name === 'maxDate' || name === 'maxDateTime') {
1556
+ max = value;
1557
+ } else if (name === 'onSelect') {
1558
+ onselect = value;
1559
+ } else if (overrides.hasOwnProperty(name)) {
1560
+ if (typeof (value) === 'undefined') {
1561
+ return overrides[name];
1562
+ }
1563
+ fns[name] = value;
1564
+ name_clone = {}; //empty results in exiting function after overrides updated
1565
+ }
1566
+ } else if (typeof name == 'object') { //if min/max was set with the JSON
1567
+ if (name.minDate) {
1568
+ min = name.minDate;
1569
+ } else if (name.minDateTime) {
1570
+ min = name.minDateTime;
1571
+ } else if (name.maxDate) {
1572
+ max = name.maxDate;
1573
+ } else if (name.maxDateTime) {
1574
+ max = name.maxDateTime;
1575
+ }
1576
+ for (prop in overrides) {
1577
+ if (overrides.hasOwnProperty(prop) && name[prop]) {
1578
+ fns[prop] = name[prop];
1579
+ }
1580
+ }
1581
+ }
1582
+ for (prop in fns) {
1583
+ if (fns.hasOwnProperty(prop)) {
1584
+ overrides[prop] = fns[prop];
1585
+ if (!name_clone) { name_clone = $.extend({}, name);}
1586
+ delete name_clone[prop];
1587
+ }
1588
+ }
1589
+ if (name_clone && isEmptyObject(name_clone)) { return; }
1590
+ if (min) { //if min was set
1591
+ if (min === 0) {
1592
+ min = new Date();
1593
+ } else {
1594
+ min = new Date(min);
1595
+ }
1596
+ tp_inst._defaults.minDate = min;
1597
+ tp_inst._defaults.minDateTime = min;
1598
+ } else if (max) { //if max was set
1599
+ if (max === 0) {
1600
+ max = new Date();
1601
+ } else {
1602
+ max = new Date(max);
1603
+ }
1604
+ tp_inst._defaults.maxDate = max;
1605
+ tp_inst._defaults.maxDateTime = max;
1606
+ } else if (onselect) {
1607
+ tp_inst._defaults.onSelect = onselect;
1608
+ }
1609
+ }
1610
+ if (value === undefined) {
1611
+ return this._base_optionDatepicker.call($.datepicker, target, name);
1612
+ }
1613
+ return this._base_optionDatepicker.call($.datepicker, target, name_clone || name, value);
1614
+ };
1615
+ /*
1616
+ * jQuery isEmptyObject does not check hasOwnProperty - if someone has added to the object prototype,
1617
+ * it will return false for all objects
1618
+ */
1619
+ var isEmptyObject = function(obj) {
1620
+ var prop;
1621
+ for (prop in obj) {
1622
+ if (obj.hasOwnProperty(obj)) {
1623
+ return false;
1624
+ }
1625
+ }
1626
+ return true;
1627
+ };
1628
+
1629
+ /*
1630
+ * jQuery extend now ignores nulls!
1631
+ */
1632
+ var extendRemove = function(target, props) {
1633
+ $.extend(target, props);
1634
+ for (var name in props) {
1635
+ if (props[name] === null || props[name] === undefined) {
1636
+ target[name] = props[name];
1637
+ }
1638
+ }
1639
+ return target;
1640
+ };
1641
+
1642
+ /*
1643
+ * Determine by the time format if should use ampm
1644
+ * Returns true if should use ampm, false if not
1645
+ */
1646
+ var useAmpm = function(timeFormat){
1647
+ return (timeFormat.indexOf('t') !== -1 && timeFormat.indexOf('h') !== -1);
1648
+ };
1649
+
1650
+ /*
1651
+ * Converts 24 hour format into 12 hour
1652
+ * Returns 12 hour without leading 0
1653
+ */
1654
+ var convert24to12 = function(hour) {
1655
+ if (hour > 12) {
1656
+ hour = hour - 12;
1657
+ }
1658
+
1659
+ if (hour == 0) {
1660
+ hour = 12;
1661
+ }
1662
+
1663
+ return String(hour);
1664
+ };
1665
+
1666
+ /*
1667
+ * Splits datetime string into date ans time substrings.
1668
+ * Throws exception when date can't be parsed
1669
+ * Returns [dateString, timeString]
1670
+ */
1671
+ var splitDateTime = function(dateFormat, dateTimeString, dateSettings, timeSettings) {
1672
+ try {
1673
+ // The idea is to get the number separator occurances in datetime and the time format requested (since time has
1674
+ // fewer unknowns, mostly numbers and am/pm). We will use the time pattern to split.
1675
+ var separator = timeSettings && timeSettings.separator ? timeSettings.separator : $.timepicker._defaults.separator,
1676
+ format = timeSettings && timeSettings.timeFormat ? timeSettings.timeFormat : $.timepicker._defaults.timeFormat,
1677
+ timeParts = format.split(separator), // how many occurances of separator may be in our format?
1678
+ timePartsLen = timeParts.length,
1679
+ allParts = dateTimeString.split(separator),
1680
+ allPartsLen = allParts.length;
1681
+
1682
+ if (allPartsLen > 1) {
1683
+ return [
1684
+ allParts.splice(0,allPartsLen-timePartsLen).join(separator),
1685
+ allParts.splice(0,timePartsLen).join(separator)
1686
+ ];
1687
+ }
1688
+
1689
+ } catch (err) {
1690
+ $.datepicker.log('Could not split the date from the time. Please check the following datetimepicker options' +
1691
+ "\nthrown error: " + err +
1692
+ "\ndateTimeString" + dateTimeString +
1693
+ "\ndateFormat = " + dateFormat +
1694
+ "\nseparator = " + timeSettings.separator +
1695
+ "\ntimeFormat = " + timeSettings.timeFormat);
1696
+
1697
+ if (err.indexOf(":") >= 0) {
1698
+ // Hack! The error message ends with a colon, a space, and
1699
+ // the "extra" characters. We rely on that instead of
1700
+ // attempting to perfectly reproduce the parsing algorithm.
1701
+ var dateStringLength = dateTimeString.length - (err.length - err.indexOf(':') - 2),
1702
+ timeString = dateTimeString.substring(dateStringLength);
1703
+
1704
+ return [$.trim(dateTimeString.substring(0, dateStringLength)), $.trim(dateTimeString.substring(dateStringLength))];
1705
+
1706
+ } else {
1707
+ throw err;
1708
+ }
1709
+ }
1710
+ return [dateTimeString, ''];
1711
+ };
1712
+
1713
+ /*
1714
+ * Internal function to parse datetime interval
1715
+ * Returns: {date: Date, timeObj: Object}, where
1716
+ * date - parsed date without time (type Date)
1717
+ * timeObj = {hour: , minute: , second: , millisec: } - parsed time. Optional
1718
+ */
1719
+ var parseDateTimeInternal = function(dateFormat, timeFormat, dateTimeString, dateSettings, timeSettings) {
1720
+ var date;
1721
+ var splitRes = splitDateTime(dateFormat, dateTimeString, dateSettings, timeSettings);
1722
+ date = $.datepicker._base_parseDate(dateFormat, splitRes[0], dateSettings);
1723
+ if (splitRes[1] !== '') {
1724
+ var timeString = splitRes[1],
1725
+ parsedTime = $.datepicker.parseTime(timeFormat, timeString, timeSettings);
1726
+
1727
+ if (parsedTime === null) {
1728
+ throw 'Wrong time format';
1729
+ }
1730
+ return {
1731
+ date: date,
1732
+ timeObj: parsedTime
1733
+ };
1734
+ } else {
1735
+ return {
1736
+ date: date
1737
+ };
1738
+ }
1739
+ };
1740
+
1741
+ /*
1742
+ * Internal function to set timezone_select to the local timezone
1743
+ */
1744
+ var selectLocalTimeZone = function(tp_inst, date) {
1745
+ if (tp_inst && tp_inst.timezone_select) {
1746
+ tp_inst._defaults.useLocalTimezone = true;
1747
+ var now = typeof date !== 'undefined' ? date : new Date();
1748
+ var tzoffset = $.timepicker.timeZoneOffsetString(now);
1749
+ if (tp_inst._defaults.timezoneIso8601) {
1750
+ tzoffset = tzoffset.substring(0, 3) + ':' + tzoffset.substring(3);
1751
+ }
1752
+ tp_inst.timezone_select.val(tzoffset);
1753
+ }
1754
+ };
1755
+
1756
+ /*
1757
+ * Create a Singleton Insance
1758
+ */
1759
+ $.timepicker = new Timepicker();
1760
+
1761
+ /**
1762
+ * Get the timezone offset as string from a date object (eg '+0530' for UTC+5.5)
1763
+ * @param date
1764
+ * @return string
1765
+ */
1766
+ $.timepicker.timeZoneOffsetString = function(date) {
1767
+ var off = date.getTimezoneOffset() * -1,
1768
+ minutes = off % 60,
1769
+ hours = (off - minutes) / 60;
1770
+ return (off >= 0 ? '+' : '-') + ('0' + (hours * 101).toString()).substr(-2) + ('0' + (minutes * 101).toString()).substr(-2);
1771
+ };
1772
+
1773
+ /**
1774
+ * Calls `timepicker()` on the `startTime` and `endTime` elements, and configures them to
1775
+ * enforce date range limits.
1776
+ * n.b. The input value must be correctly formatted (reformatting is not supported)
1777
+ * @param Element startTime
1778
+ * @param Element endTime
1779
+ * @param obj options Options for the timepicker() call
1780
+ * @return jQuery
1781
+ */
1782
+ $.timepicker.timeRange = function(startTime, endTime, options) {
1783
+ return $.timepicker.handleRange('timepicker', startTime, endTime, options);
1784
+ };
1785
+
1786
+ /**
1787
+ * Calls `datetimepicker` on the `startTime` and `endTime` elements, and configures them to
1788
+ * enforce date range limits.
1789
+ * @param Element startTime
1790
+ * @param Element endTime
1791
+ * @param obj options Options for the `timepicker()` call. Also supports `reformat`,
1792
+ * a boolean value that can be used to reformat the input values to the `dateFormat`.
1793
+ * @param string method Can be used to specify the type of picker to be added
1794
+ * @return jQuery
1795
+ */
1796
+ $.timepicker.dateTimeRange = function(startTime, endTime, options) {
1797
+ $.timepicker.dateRange(startTime, endTime, options, 'datetimepicker');
1798
+ };
1799
+
1800
+ /**
1801
+ * Calls `method` on the `startTime` and `endTime` elements, and configures them to
1802
+ * enforce date range limits.
1803
+ * @param Element startTime
1804
+ * @param Element endTime
1805
+ * @param obj options Options for the `timepicker()` call. Also supports `reformat`,
1806
+ * a boolean value that can be used to reformat the input values to the `dateFormat`.
1807
+ * @param string method Can be used to specify the type of picker to be added
1808
+ * @return jQuery
1809
+ */
1810
+ $.timepicker.dateRange = function(startTime, endTime, options, method) {
1811
+ method = method || 'datepicker';
1812
+ $.timepicker.handleRange(method, startTime, endTime, options);
1813
+ };
1814
+
1815
+ /**
1816
+ * Calls `method` on the `startTime` and `endTime` elements, and configures them to
1817
+ * enforce date range limits.
1818
+ * @param string method Can be used to specify the type of picker to be added
1819
+ * @param Element startTime
1820
+ * @param Element endTime
1821
+ * @param obj options Options for the `timepicker()` call. Also supports `reformat`,
1822
+ * a boolean value that can be used to reformat the input values to the `dateFormat`.
1823
+ * @return jQuery
1824
+ */
1825
+ $.timepicker.handleRange = function(method, startTime, endTime, options) {
1826
+ $.fn[method].call(startTime, $.extend({
1827
+ onClose: function(dateText, inst) {
1828
+ checkDates(this, endTime, dateText);
1829
+ },
1830
+ onSelect: function(selectedDateTime) {
1831
+ selected(this, endTime, 'minDate');
1832
+ }
1833
+ }, options, options.start));
1834
+ $.fn[method].call(endTime, $.extend({
1835
+ onClose: function(dateText, inst) {
1836
+ checkDates(this, startTime, dateText);
1837
+ },
1838
+ onSelect: function(selectedDateTime) {
1839
+ selected(this, startTime, 'maxDate');
1840
+ }
1841
+ }, options, options.end));
1842
+ // timepicker doesn't provide access to its 'timeFormat' option,
1843
+ // nor could I get datepicker.formatTime() to behave with times, so I
1844
+ // have disabled reformatting for timepicker
1845
+ if (method != 'timepicker' && options.reformat) {
1846
+ $([startTime, endTime]).each(function() {
1847
+ var format = $(this)[method].call($(this), 'option', 'dateFormat'),
1848
+ date = new Date($(this).val());
1849
+ if ($(this).val() && date) {
1850
+ $(this).val($.datepicker.formatDate(format, date));
1851
+ }
1852
+ });
1853
+ }
1854
+ checkDates(startTime, endTime, startTime.val());
1855
+
1856
+ function checkDates(changed, other, dateText) {
1857
+ if (other.val() && (new Date(startTime.val()) > new Date(endTime.val()))) {
1858
+ other.val(dateText);
1859
+ }
1860
+ }
1861
+ selected(startTime, endTime, 'minDate');
1862
+ selected(endTime, startTime, 'maxDate');
1863
+
1864
+ function selected(changed, other, option) {
1865
+ if (!$(changed).val()) {
1866
+ return;
1867
+ }
1868
+ var date = $(changed)[method].call($(changed), 'getDate');
1869
+ // timepicker doesn't implement 'getDate' and returns a jQuery
1870
+ if (date.getTime) {
1871
+ $(other)[method].call($(other), 'option', option, date);
1872
+ }
1873
+ }
1874
+ return $([startTime.get(0), endTime.get(0)]);
1875
+ };
1876
+
1877
+ /*
1878
+ * Keep up with the version
1879
+ */
1880
+ $.timepicker.version = "1.1.1";
1881
+
1882
+ })(jQuery);