halorgium-actionpack 3.0.pre

Sign up to get free protection for your applications and to get access to all the features.
Files changed (154) hide show
  1. data/CHANGELOG +5179 -0
  2. data/MIT-LICENSE +21 -0
  3. data/README +409 -0
  4. data/lib/abstract_controller.rb +16 -0
  5. data/lib/abstract_controller/base.rb +158 -0
  6. data/lib/abstract_controller/callbacks.rb +113 -0
  7. data/lib/abstract_controller/exceptions.rb +12 -0
  8. data/lib/abstract_controller/helpers.rb +151 -0
  9. data/lib/abstract_controller/layouts.rb +250 -0
  10. data/lib/abstract_controller/localized_cache.rb +49 -0
  11. data/lib/abstract_controller/logger.rb +61 -0
  12. data/lib/abstract_controller/rendering_controller.rb +188 -0
  13. data/lib/action_controller.rb +72 -0
  14. data/lib/action_controller/base.rb +168 -0
  15. data/lib/action_controller/caching.rb +80 -0
  16. data/lib/action_controller/caching/actions.rb +163 -0
  17. data/lib/action_controller/caching/fragments.rb +116 -0
  18. data/lib/action_controller/caching/pages.rb +154 -0
  19. data/lib/action_controller/caching/sweeping.rb +97 -0
  20. data/lib/action_controller/deprecated.rb +4 -0
  21. data/lib/action_controller/deprecated/integration_test.rb +2 -0
  22. data/lib/action_controller/deprecated/performance_test.rb +1 -0
  23. data/lib/action_controller/dispatch/dispatcher.rb +57 -0
  24. data/lib/action_controller/metal.rb +129 -0
  25. data/lib/action_controller/metal/benchmarking.rb +73 -0
  26. data/lib/action_controller/metal/compatibility.rb +145 -0
  27. data/lib/action_controller/metal/conditional_get.rb +86 -0
  28. data/lib/action_controller/metal/configuration.rb +28 -0
  29. data/lib/action_controller/metal/cookies.rb +105 -0
  30. data/lib/action_controller/metal/exceptions.rb +55 -0
  31. data/lib/action_controller/metal/filter_parameter_logging.rb +77 -0
  32. data/lib/action_controller/metal/flash.rb +162 -0
  33. data/lib/action_controller/metal/head.rb +27 -0
  34. data/lib/action_controller/metal/helpers.rb +115 -0
  35. data/lib/action_controller/metal/hide_actions.rb +47 -0
  36. data/lib/action_controller/metal/http_authentication.rb +312 -0
  37. data/lib/action_controller/metal/layouts.rb +171 -0
  38. data/lib/action_controller/metal/mime_responds.rb +317 -0
  39. data/lib/action_controller/metal/rack_convenience.rb +27 -0
  40. data/lib/action_controller/metal/redirector.rb +22 -0
  41. data/lib/action_controller/metal/render_options.rb +103 -0
  42. data/lib/action_controller/metal/rendering_controller.rb +57 -0
  43. data/lib/action_controller/metal/request_forgery_protection.rb +108 -0
  44. data/lib/action_controller/metal/rescuable.rb +13 -0
  45. data/lib/action_controller/metal/responder.rb +200 -0
  46. data/lib/action_controller/metal/session.rb +15 -0
  47. data/lib/action_controller/metal/session_management.rb +45 -0
  48. data/lib/action_controller/metal/streaming.rb +188 -0
  49. data/lib/action_controller/metal/testing.rb +39 -0
  50. data/lib/action_controller/metal/url_for.rb +41 -0
  51. data/lib/action_controller/metal/verification.rb +130 -0
  52. data/lib/action_controller/middleware.rb +38 -0
  53. data/lib/action_controller/notifications.rb +10 -0
  54. data/lib/action_controller/polymorphic_routes.rb +183 -0
  55. data/lib/action_controller/record_identifier.rb +91 -0
  56. data/lib/action_controller/testing/process.rb +111 -0
  57. data/lib/action_controller/testing/test_case.rb +345 -0
  58. data/lib/action_controller/translation.rb +13 -0
  59. data/lib/action_controller/url_rewriter.rb +204 -0
  60. data/lib/action_controller/vendor/html-scanner.rb +16 -0
  61. data/lib/action_controller/vendor/html-scanner/html/document.rb +68 -0
  62. data/lib/action_controller/vendor/html-scanner/html/node.rb +537 -0
  63. data/lib/action_controller/vendor/html-scanner/html/sanitizer.rb +176 -0
  64. data/lib/action_controller/vendor/html-scanner/html/selector.rb +828 -0
  65. data/lib/action_controller/vendor/html-scanner/html/tokenizer.rb +105 -0
  66. data/lib/action_controller/vendor/html-scanner/html/version.rb +11 -0
  67. data/lib/action_dispatch.rb +70 -0
  68. data/lib/action_dispatch/http/headers.rb +33 -0
  69. data/lib/action_dispatch/http/mime_type.rb +231 -0
  70. data/lib/action_dispatch/http/mime_types.rb +23 -0
  71. data/lib/action_dispatch/http/request.rb +539 -0
  72. data/lib/action_dispatch/http/response.rb +290 -0
  73. data/lib/action_dispatch/http/status_codes.rb +42 -0
  74. data/lib/action_dispatch/http/utils.rb +20 -0
  75. data/lib/action_dispatch/middleware/callbacks.rb +50 -0
  76. data/lib/action_dispatch/middleware/params_parser.rb +79 -0
  77. data/lib/action_dispatch/middleware/rescue.rb +26 -0
  78. data/lib/action_dispatch/middleware/session/abstract_store.rb +208 -0
  79. data/lib/action_dispatch/middleware/session/cookie_store.rb +235 -0
  80. data/lib/action_dispatch/middleware/session/mem_cache_store.rb +47 -0
  81. data/lib/action_dispatch/middleware/show_exceptions.rb +143 -0
  82. data/lib/action_dispatch/middleware/stack.rb +116 -0
  83. data/lib/action_dispatch/middleware/static.rb +44 -0
  84. data/lib/action_dispatch/middleware/string_coercion.rb +29 -0
  85. data/lib/action_dispatch/middleware/templates/rescues/_request_and_response.erb +24 -0
  86. data/lib/action_dispatch/middleware/templates/rescues/_trace.erb +26 -0
  87. data/lib/action_dispatch/middleware/templates/rescues/diagnostics.erb +10 -0
  88. data/lib/action_dispatch/middleware/templates/rescues/layout.erb +29 -0
  89. data/lib/action_dispatch/middleware/templates/rescues/missing_template.erb +2 -0
  90. data/lib/action_dispatch/middleware/templates/rescues/routing_error.erb +10 -0
  91. data/lib/action_dispatch/middleware/templates/rescues/template_error.erb +21 -0
  92. data/lib/action_dispatch/middleware/templates/rescues/unknown_action.erb +2 -0
  93. data/lib/action_dispatch/routing.rb +381 -0
  94. data/lib/action_dispatch/routing/deprecated_mapper.rb +878 -0
  95. data/lib/action_dispatch/routing/mapper.rb +327 -0
  96. data/lib/action_dispatch/routing/route.rb +49 -0
  97. data/lib/action_dispatch/routing/route_set.rb +497 -0
  98. data/lib/action_dispatch/testing/assertions.rb +8 -0
  99. data/lib/action_dispatch/testing/assertions/dom.rb +35 -0
  100. data/lib/action_dispatch/testing/assertions/model.rb +19 -0
  101. data/lib/action_dispatch/testing/assertions/response.rb +145 -0
  102. data/lib/action_dispatch/testing/assertions/routing.rb +144 -0
  103. data/lib/action_dispatch/testing/assertions/selector.rb +639 -0
  104. data/lib/action_dispatch/testing/assertions/tag.rb +123 -0
  105. data/lib/action_dispatch/testing/integration.rb +504 -0
  106. data/lib/action_dispatch/testing/performance_test.rb +15 -0
  107. data/lib/action_dispatch/testing/test_request.rb +83 -0
  108. data/lib/action_dispatch/testing/test_response.rb +131 -0
  109. data/lib/action_pack.rb +24 -0
  110. data/lib/action_pack/version.rb +9 -0
  111. data/lib/action_view.rb +58 -0
  112. data/lib/action_view/base.rb +308 -0
  113. data/lib/action_view/context.rb +44 -0
  114. data/lib/action_view/erb/util.rb +48 -0
  115. data/lib/action_view/helpers.rb +62 -0
  116. data/lib/action_view/helpers/active_model_helper.rb +306 -0
  117. data/lib/action_view/helpers/ajax_helper.rb +68 -0
  118. data/lib/action_view/helpers/asset_tag_helper.rb +830 -0
  119. data/lib/action_view/helpers/atom_feed_helper.rb +198 -0
  120. data/lib/action_view/helpers/cache_helper.rb +39 -0
  121. data/lib/action_view/helpers/capture_helper.rb +168 -0
  122. data/lib/action_view/helpers/date_helper.rb +988 -0
  123. data/lib/action_view/helpers/debug_helper.rb +38 -0
  124. data/lib/action_view/helpers/form_helper.rb +1102 -0
  125. data/lib/action_view/helpers/form_options_helper.rb +600 -0
  126. data/lib/action_view/helpers/form_tag_helper.rb +495 -0
  127. data/lib/action_view/helpers/javascript_helper.rb +208 -0
  128. data/lib/action_view/helpers/number_helper.rb +311 -0
  129. data/lib/action_view/helpers/prototype_helper.rb +1309 -0
  130. data/lib/action_view/helpers/raw_output_helper.rb +9 -0
  131. data/lib/action_view/helpers/record_identification_helper.rb +20 -0
  132. data/lib/action_view/helpers/record_tag_helper.rb +58 -0
  133. data/lib/action_view/helpers/sanitize_helper.rb +259 -0
  134. data/lib/action_view/helpers/scriptaculous_helper.rb +226 -0
  135. data/lib/action_view/helpers/tag_helper.rb +151 -0
  136. data/lib/action_view/helpers/text_helper.rb +594 -0
  137. data/lib/action_view/helpers/translation_helper.rb +39 -0
  138. data/lib/action_view/helpers/url_helper.rb +639 -0
  139. data/lib/action_view/locale/en.yml +117 -0
  140. data/lib/action_view/paths.rb +80 -0
  141. data/lib/action_view/render/partials.rb +342 -0
  142. data/lib/action_view/render/rendering.rb +134 -0
  143. data/lib/action_view/safe_buffer.rb +28 -0
  144. data/lib/action_view/template/error.rb +101 -0
  145. data/lib/action_view/template/handler.rb +36 -0
  146. data/lib/action_view/template/handlers.rb +52 -0
  147. data/lib/action_view/template/handlers/builder.rb +17 -0
  148. data/lib/action_view/template/handlers/erb.rb +53 -0
  149. data/lib/action_view/template/handlers/rjs.rb +18 -0
  150. data/lib/action_view/template/resolver.rb +165 -0
  151. data/lib/action_view/template/template.rb +131 -0
  152. data/lib/action_view/template/text.rb +38 -0
  153. data/lib/action_view/test_case.rb +163 -0
  154. metadata +236 -0
@@ -0,0 +1,198 @@
1
+ require 'set'
2
+
3
+ # Adds easy defaults to writing Atom feeds with the Builder template engine (this does not work on ERb or any other
4
+ # template languages).
5
+ module ActionView
6
+ module Helpers #:nodoc:
7
+ module AtomFeedHelper
8
+ # Full usage example:
9
+ #
10
+ # config/routes.rb:
11
+ # ActionController::Routing::Routes.draw do |map|
12
+ # map.resources :posts
13
+ # map.root :controller => "posts"
14
+ # end
15
+ #
16
+ # app/controllers/posts_controller.rb:
17
+ # class PostsController < ApplicationController::Base
18
+ # # GET /posts.html
19
+ # # GET /posts.atom
20
+ # def index
21
+ # @posts = Post.find(:all)
22
+ #
23
+ # respond_to do |format|
24
+ # format.html
25
+ # format.atom
26
+ # end
27
+ # end
28
+ # end
29
+ #
30
+ # app/views/posts/index.atom.builder:
31
+ # atom_feed do |feed|
32
+ # feed.title("My great blog!")
33
+ # feed.updated(@posts.first.created_at)
34
+ #
35
+ # for post in @posts
36
+ # feed.entry(post) do |entry|
37
+ # entry.title(post.title)
38
+ # entry.content(post.body, :type => 'html')
39
+ #
40
+ # entry.author do |author|
41
+ # author.name("DHH")
42
+ # end
43
+ # end
44
+ # end
45
+ # end
46
+ #
47
+ # The options for atom_feed are:
48
+ #
49
+ # * <tt>:language</tt>: Defaults to "en-US".
50
+ # * <tt>:root_url</tt>: The HTML alternative that this feed is doubling for. Defaults to / on the current host.
51
+ # * <tt>:url</tt>: The URL for this feed. Defaults to the current URL.
52
+ # * <tt>:id</tt>: The id for this feed. Defaults to "tag:#{request.host},#{options[:schema_date]}:#{request.request_uri.split(".")[0]}"
53
+ # * <tt>:schema_date</tt>: The date at which the tag scheme for the feed was first used. A good default is the year you
54
+ # created the feed. See http://feedvalidator.org/docs/error/InvalidTAG.html for more information. If not specified,
55
+ # 2005 is used (as an "I don't care" value).
56
+ # * <tt>:instruct</tt>: Hash of XML processing instructions in the form {target => {attribute => value, }} or {target => [{attribute => value, }, ]}
57
+ #
58
+ # Other namespaces can be added to the root element:
59
+ #
60
+ # app/views/posts/index.atom.builder:
61
+ # atom_feed({'xmlns:app' => 'http://www.w3.org/2007/app',
62
+ # 'xmlns:openSearch' => 'http://a9.com/-/spec/opensearch/1.1/'}) do |feed|
63
+ # feed.title("My great blog!")
64
+ # feed.updated((@posts.first.created_at))
65
+ # feed.tag!(openSearch:totalResults, 10)
66
+ #
67
+ # for post in @posts
68
+ # feed.entry(post) do |entry|
69
+ # entry.title(post.title)
70
+ # entry.content(post.body, :type => 'html')
71
+ # entry.tag!('app:edited', Time.now)
72
+ #
73
+ # entry.author do |author|
74
+ # author.name("DHH")
75
+ # end
76
+ # end
77
+ # end
78
+ # end
79
+ #
80
+ # The Atom spec defines five elements (content rights title subtitle
81
+ # summary) which may directly contain xhtml content if :type => 'xhtml'
82
+ # is specified as an attribute. If so, this helper will take care of
83
+ # the enclosing div and xhtml namespace declaration. Example usage:
84
+ #
85
+ # entry.summary :type => 'xhtml' do |xhtml|
86
+ # xhtml.p pluralize(order.line_items.count, "line item")
87
+ # xhtml.p "Shipped to #{order.address}"
88
+ # xhtml.p "Paid by #{order.pay_type}"
89
+ # end
90
+ #
91
+ #
92
+ # atom_feed yields an AtomFeedBuilder instance. Nested elements yield
93
+ # an AtomBuilder instance.
94
+ def atom_feed(options = {}, &block)
95
+ if options[:schema_date]
96
+ options[:schema_date] = options[:schema_date].strftime("%Y-%m-%d") if options[:schema_date].respond_to?(:strftime)
97
+ else
98
+ options[:schema_date] = "2005" # The Atom spec copyright date
99
+ end
100
+
101
+ xml = options.delete(:xml) || eval("xml", block.binding)
102
+ xml.instruct!
103
+ if options[:instruct]
104
+ options[:instruct].each do |target,attrs|
105
+ if attrs.respond_to?(:keys)
106
+ xml.instruct!(target, attrs)
107
+ elsif attrs.respond_to?(:each)
108
+ attrs.each { |attr_group| xml.instruct!(target, attr_group) }
109
+ end
110
+ end
111
+ end
112
+
113
+ feed_opts = {"xml:lang" => options[:language] || "en-US", "xmlns" => 'http://www.w3.org/2005/Atom'}
114
+ feed_opts.merge!(options).reject!{|k,v| !k.to_s.match(/^xml/)}
115
+
116
+ xml.feed(feed_opts) do
117
+ xml.id(options[:id] || "tag:#{request.host},#{options[:schema_date]}:#{request.request_uri.split(".")[0]}")
118
+ xml.link(:rel => 'alternate', :type => 'text/html', :href => options[:root_url] || (request.protocol + request.host_with_port))
119
+ xml.link(:rel => 'self', :type => 'application/atom+xml', :href => options[:url] || request.url)
120
+
121
+ yield AtomFeedBuilder.new(xml, self, options)
122
+ end
123
+ end
124
+
125
+ class AtomBuilder
126
+ XHTML_TAG_NAMES = %w(content rights title subtitle summary).to_set
127
+
128
+ def initialize(xml)
129
+ @xml = xml
130
+ end
131
+
132
+ private
133
+ # Delegate to xml builder, first wrapping the element in a xhtml
134
+ # namespaced div element if the method and arguments indicate
135
+ # that an xhtml_block? is desired.
136
+ def method_missing(method, *arguments, &block)
137
+ if xhtml_block?(method, arguments)
138
+ @xml.__send__(method, *arguments) do
139
+ @xml.div(:xmlns => 'http://www.w3.org/1999/xhtml') do |xhtml|
140
+ block.call(xhtml)
141
+ end
142
+ end
143
+ else
144
+ @xml.__send__(method, *arguments, &block)
145
+ end
146
+ end
147
+
148
+ # True if the method name matches one of the five elements defined
149
+ # in the Atom spec as potentially containing XHTML content and
150
+ # if :type => 'xhtml' is, in fact, specified.
151
+ def xhtml_block?(method, arguments)
152
+ if XHTML_TAG_NAMES.include?(method.to_s)
153
+ last = arguments.last
154
+ last.is_a?(Hash) && last[:type].to_s == 'xhtml'
155
+ end
156
+ end
157
+ end
158
+
159
+ class AtomFeedBuilder < AtomBuilder
160
+ def initialize(xml, view, feed_options = {})
161
+ @xml, @view, @feed_options = xml, view, feed_options
162
+ end
163
+
164
+ # Accepts a Date or Time object and inserts it in the proper format. If nil is passed, current time in UTC is used.
165
+ def updated(date_or_time = nil)
166
+ @xml.updated((date_or_time || Time.now.utc).xmlschema)
167
+ end
168
+
169
+ # Creates an entry tag for a specific record and prefills the id using class and id.
170
+ #
171
+ # Options:
172
+ #
173
+ # * <tt>:published</tt>: Time first published. Defaults to the created_at attribute on the record if one such exists.
174
+ # * <tt>:updated</tt>: Time of update. Defaults to the updated_at attribute on the record if one such exists.
175
+ # * <tt>:url</tt>: The URL for this entry. Defaults to the polymorphic_url for the record.
176
+ # * <tt>:id</tt>: The ID for this entry. Defaults to "tag:#{@view.request.host},#{@feed_options[:schema_date]}:#{record.class}/#{record.id}"
177
+ def entry(record, options = {})
178
+ @xml.entry do
179
+ @xml.id(options[:id] || "tag:#{@view.request.host},#{@feed_options[:schema_date]}:#{record.class}/#{record.id}")
180
+
181
+ if options[:published] || (record.respond_to?(:created_at) && record.created_at)
182
+ @xml.published((options[:published] || record.created_at).xmlschema)
183
+ end
184
+
185
+ if options[:updated] || (record.respond_to?(:updated_at) && record.updated_at)
186
+ @xml.updated((options[:updated] || record.updated_at).xmlschema)
187
+ end
188
+
189
+ @xml.link(:rel => 'alternate', :type => 'text/html', :href => options[:url] || @view.polymorphic_url(record))
190
+
191
+ yield AtomBuilder.new(@xml)
192
+ end
193
+ end
194
+ end
195
+
196
+ end
197
+ end
198
+ end
@@ -0,0 +1,39 @@
1
+ module ActionView
2
+ module Helpers
3
+ # This helper to exposes a method for caching of view fragments.
4
+ # See ActionController::Caching::Fragments for usage instructions.
5
+ module CacheHelper
6
+ # A method for caching fragments of a view rather than an entire
7
+ # action or page. This technique is useful caching pieces like
8
+ # menus, lists of news topics, static HTML fragments, and so on.
9
+ # This method takes a block that contains the content you wish
10
+ # to cache. See ActionController::Caching::Fragments for more
11
+ # information.
12
+ #
13
+ # ==== Examples
14
+ # If you wanted to cache a navigation menu, you could do the
15
+ # following.
16
+ #
17
+ # <% cache do %>
18
+ # <%= render :partial => "menu" %>
19
+ # <% end %>
20
+ #
21
+ # You can also cache static content...
22
+ #
23
+ # <% cache do %>
24
+ # <p>Hello users! Welcome to our website!</p>
25
+ # <% end %>
26
+ #
27
+ # ...and static content mixed with RHTML content.
28
+ #
29
+ # <% cache do %>
30
+ # Topics:
31
+ # <%= render :partial => "topics", :collection => @topic_list %>
32
+ # <i>Topics listed alphabetically</i>
33
+ # <% end %>
34
+ def cache(name = {}, options = nil, &block)
35
+ @controller.fragment_for(output_buffer, name, options, &block)
36
+ end
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,168 @@
1
+ module ActionView
2
+ module Helpers
3
+ # CaptureHelper exposes methods to let you extract generated markup which
4
+ # can be used in other parts of a template or layout file.
5
+ # It provides a method to capture blocks into variables through capture and
6
+ # a way to capture a block of markup for use in a layout through content_for.
7
+ module CaptureHelper
8
+ # The capture method allows you to extract part of a template into a
9
+ # variable. You can then use this variable anywhere in your templates or layout.
10
+ #
11
+ # ==== Examples
12
+ # The capture method can be used in ERb templates...
13
+ #
14
+ # <% @greeting = capture do %>
15
+ # Welcome to my shiny new web page! The date and time is
16
+ # <%= Time.now %>
17
+ # <% end %>
18
+ #
19
+ # ...and Builder (RXML) templates.
20
+ #
21
+ # @timestamp = capture do
22
+ # "The current timestamp is #{Time.now}."
23
+ # end
24
+ #
25
+ # You can then use that variable anywhere else. For example:
26
+ #
27
+ # <html>
28
+ # <head><title><%= @greeting %></title></head>
29
+ # <body>
30
+ # <b><%= @greeting %></b>
31
+ # </body></html>
32
+ #
33
+ def capture(*args, &block)
34
+ # Return captured buffer in erb.
35
+ if block_called_from_erb?(block)
36
+ with_output_buffer { block.call(*args) }
37
+ else
38
+ # Return block result otherwise, but protect buffer also.
39
+ with_output_buffer { return block.call(*args) }
40
+ end
41
+ end
42
+
43
+ # Calling content_for stores a block of markup in an identifier for later use.
44
+ # You can make subsequent calls to the stored content in other templates or the layout
45
+ # by passing the identifier as an argument to <tt>yield</tt>.
46
+ #
47
+ # ==== Examples
48
+ #
49
+ # <% content_for :not_authorized do %>
50
+ # alert('You are not authorized to do that!')
51
+ # <% end %>
52
+ #
53
+ # You can then use <tt>yield :not_authorized</tt> anywhere in your templates.
54
+ #
55
+ # <%= yield :not_authorized if current_user.nil? %>
56
+ #
57
+ # You can also use this syntax alongside an existing call to <tt>yield</tt> in a layout. For example:
58
+ #
59
+ # <%# This is the layout %>
60
+ # <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
61
+ # <head>
62
+ # <title>My Website</title>
63
+ # <%= yield :script %>
64
+ # </head>
65
+ # <body>
66
+ # <%= yield %>
67
+ # </body>
68
+ # </html>
69
+ #
70
+ # And now, we'll create a view that has a content_for call that
71
+ # creates the <tt>script</tt> identifier.
72
+ #
73
+ # <%# This is our view %>
74
+ # Please login!
75
+ #
76
+ # <% content_for :script do %>
77
+ # <script type="text/javascript">alert('You are not authorized to view this page!')</script>
78
+ # <% end %>
79
+ #
80
+ # Then, in another view, you could to do something like this:
81
+ #
82
+ # <%= link_to_remote 'Logout', :action => 'logout' %>
83
+ #
84
+ # <% content_for :script do %>
85
+ # <%= javascript_include_tag :defaults %>
86
+ # <% end %>
87
+ #
88
+ # That will place <script> tags for Prototype, Scriptaculous, and application.js (if it exists)
89
+ # on the page; this technique is useful if you'll only be using these scripts in a few views.
90
+ #
91
+ # Note that content_for concatenates the blocks it is given for a particular
92
+ # identifier in order. For example:
93
+ #
94
+ # <% content_for :navigation do %>
95
+ # <li><%= link_to 'Home', :action => 'index' %></li>
96
+ # <% end %>
97
+ #
98
+ # <%# Add some other content, or use a different template: %>
99
+ #
100
+ # <% content_for :navigation do %>
101
+ # <li><%= link_to 'Login', :action => 'login' %></li>
102
+ # <% end %>
103
+ #
104
+ # Then, in another template or layout, this code would render both links in order:
105
+ #
106
+ # <ul><%= yield :navigation %></ul>
107
+ #
108
+ # Lastly, simple content can be passed as a parameter:
109
+ #
110
+ # <% content_for :script, javascript_include_tag(:defaults) %>
111
+ #
112
+ # WARNING: content_for is ignored in caches. So you shouldn't use it
113
+ # for elements that will be fragment cached.
114
+ def content_for(name, content = nil, &block)
115
+ content = capture(&block) if block_given?
116
+ return @_content_for[name] << content if content
117
+ @_content_for[name]
118
+ end
119
+
120
+ # content_for? simply checks whether any content has been captured yet using content_for
121
+ # Useful to render parts of your layout differently based on what is in your views.
122
+ #
123
+ # ==== Examples
124
+ #
125
+ # Perhaps you will use different css in you layout if no content_for :right_column
126
+ #
127
+ # <%# This is the layout %>
128
+ # <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
129
+ # <head>
130
+ # <title>My Website</title>
131
+ # <%= yield :script %>
132
+ # </head>
133
+ # <body class="<%= content_for?(:right_col) ? 'one-column' : 'two-column' %>">
134
+ # <%= yield %>
135
+ # <%= yield :right_col %>
136
+ # </body>
137
+ # </html>
138
+ def content_for?(name)
139
+ @_content_for[name].present?
140
+ end
141
+
142
+ # Use an alternate output buffer for the duration of the block.
143
+ # Defaults to a new empty string.
144
+ def with_output_buffer(buf = nil) #:nodoc:
145
+ unless buf
146
+ buf = ActionView::SafeBuffer.new
147
+ buf.force_encoding(output_buffer.encoding) if buf.respond_to?(:force_encoding)
148
+ end
149
+ self.output_buffer, old_buffer = buf, output_buffer
150
+ yield
151
+ output_buffer
152
+ ensure
153
+ self.output_buffer = old_buffer
154
+ end
155
+
156
+ # Add the output buffer to the response body and start a new one.
157
+ def flush_output_buffer #:nodoc:
158
+ if output_buffer && !output_buffer.empty?
159
+ response.body_parts << output_buffer
160
+ new = ''
161
+ new.force_encoding(output_buffer.encoding) if new.respond_to?(:force_encoding)
162
+ self.output_buffer = new
163
+ nil
164
+ end
165
+ end
166
+ end
167
+ end
168
+ end
@@ -0,0 +1,988 @@
1
+ require "date"
2
+ require 'action_view/helpers/tag_helper'
3
+
4
+ module ActionView
5
+ module Helpers
6
+ # The Date Helper primarily creates select/option tags for different kinds of dates and date elements. All of the
7
+ # select-type methods share a number of common options that are as follows:
8
+ #
9
+ # * <tt>:prefix</tt> - overwrites the default prefix of "date" used for the select names. So specifying "birthday"
10
+ # would give birthday[month] instead of date[month] if passed to the select_month method.
11
+ # * <tt>:include_blank</tt> - set to true if it should be possible to set an empty date.
12
+ # * <tt>:discard_type</tt> - set to true if you want to discard the type part of the select name. If set to true,
13
+ # the select_month method would use simply "date" (which can be overwritten using <tt>:prefix</tt>) instead of
14
+ # "date[month]".
15
+ module DateHelper
16
+ # Reports the approximate distance in time between two Time or Date objects or integers as seconds.
17
+ # Set <tt>include_seconds</tt> to true if you want more detailed approximations when distance < 1 min, 29 secs
18
+ # Distances are reported based on the following table:
19
+ #
20
+ # 0 <-> 29 secs # => less than a minute
21
+ # 30 secs <-> 1 min, 29 secs # => 1 minute
22
+ # 1 min, 30 secs <-> 44 mins, 29 secs # => [2..44] minutes
23
+ # 44 mins, 30 secs <-> 89 mins, 29 secs # => about 1 hour
24
+ # 89 mins, 29 secs <-> 23 hrs, 59 mins, 29 secs # => about [2..24] hours
25
+ # 23 hrs, 59 mins, 29 secs <-> 47 hrs, 59 mins, 29 secs # => 1 day
26
+ # 47 hrs, 59 mins, 29 secs <-> 29 days, 23 hrs, 59 mins, 29 secs # => [2..29] days
27
+ # 29 days, 23 hrs, 59 mins, 30 secs <-> 59 days, 23 hrs, 59 mins, 29 secs # => about 1 month
28
+ # 59 days, 23 hrs, 59 mins, 30 secs <-> 1 yr minus 1 sec # => [2..12] months
29
+ # 1 yr <-> 1 yr, 3 months # => about 1 year
30
+ # 1 yr, 3 months <-> 1 yr, 9 months # => over 1 year
31
+ # 1 yr, 9 months <-> 2 yr minus 1 sec # => almost 2 years
32
+ # 2 yrs <-> max time or date # => (same rules as 1 yr)
33
+ #
34
+ # With <tt>include_seconds</tt> = true and the difference < 1 minute 29 seconds:
35
+ # 0-4 secs # => less than 5 seconds
36
+ # 5-9 secs # => less than 10 seconds
37
+ # 10-19 secs # => less than 20 seconds
38
+ # 20-39 secs # => half a minute
39
+ # 40-59 secs # => less than a minute
40
+ # 60-89 secs # => 1 minute
41
+ #
42
+ # ==== Examples
43
+ # from_time = Time.now
44
+ # distance_of_time_in_words(from_time, from_time + 50.minutes) # => about 1 hour
45
+ # distance_of_time_in_words(from_time, 50.minutes.from_now) # => about 1 hour
46
+ # distance_of_time_in_words(from_time, from_time + 15.seconds) # => less than a minute
47
+ # distance_of_time_in_words(from_time, from_time + 15.seconds, true) # => less than 20 seconds
48
+ # distance_of_time_in_words(from_time, 3.years.from_now) # => about 3 years
49
+ # distance_of_time_in_words(from_time, from_time + 60.hours) # => about 3 days
50
+ # distance_of_time_in_words(from_time, from_time + 45.seconds, true) # => less than a minute
51
+ # distance_of_time_in_words(from_time, from_time - 45.seconds, true) # => less than a minute
52
+ # distance_of_time_in_words(from_time, 76.seconds.from_now) # => 1 minute
53
+ # distance_of_time_in_words(from_time, from_time + 1.year + 3.days) # => about 1 year
54
+ # distance_of_time_in_words(from_time, from_time + 3.years + 6.months) # => over 3 years
55
+ # distance_of_time_in_words(from_time, from_time + 4.years + 9.days + 30.minutes + 5.seconds) # => about 4 years
56
+ #
57
+ # to_time = Time.now + 6.years + 19.days
58
+ # distance_of_time_in_words(from_time, to_time, true) # => about 6 years
59
+ # distance_of_time_in_words(to_time, from_time, true) # => about 6 years
60
+ # distance_of_time_in_words(Time.now, Time.now) # => less than a minute
61
+ #
62
+ def distance_of_time_in_words(from_time, to_time = 0, include_seconds = false, options = {})
63
+ from_time = from_time.to_time if from_time.respond_to?(:to_time)
64
+ to_time = to_time.to_time if to_time.respond_to?(:to_time)
65
+ distance_in_minutes = (((to_time - from_time).abs)/60).round
66
+ distance_in_seconds = ((to_time - from_time).abs).round
67
+
68
+ I18n.with_options :locale => options[:locale], :scope => :'datetime.distance_in_words' do |locale|
69
+ case distance_in_minutes
70
+ when 0..1
71
+ return distance_in_minutes == 0 ?
72
+ locale.t(:less_than_x_minutes, :count => 1) :
73
+ locale.t(:x_minutes, :count => distance_in_minutes) unless include_seconds
74
+
75
+ case distance_in_seconds
76
+ when 0..4 then locale.t :less_than_x_seconds, :count => 5
77
+ when 5..9 then locale.t :less_than_x_seconds, :count => 10
78
+ when 10..19 then locale.t :less_than_x_seconds, :count => 20
79
+ when 20..39 then locale.t :half_a_minute
80
+ when 40..59 then locale.t :less_than_x_minutes, :count => 1
81
+ else locale.t :x_minutes, :count => 1
82
+ end
83
+
84
+ when 2..44 then locale.t :x_minutes, :count => distance_in_minutes
85
+ when 45..89 then locale.t :about_x_hours, :count => 1
86
+ when 90..1439 then locale.t :about_x_hours, :count => (distance_in_minutes.to_f / 60.0).round
87
+ when 1440..2529 then locale.t :x_days, :count => 1
88
+ when 2530..43199 then locale.t :x_days, :count => (distance_in_minutes.to_f / 1440.0).round
89
+ when 43200..86399 then locale.t :about_x_months, :count => 1
90
+ when 86400..525599 then locale.t :x_months, :count => (distance_in_minutes.to_f / 43200.0).round
91
+ else
92
+ distance_in_years = distance_in_minutes / 525600
93
+ minute_offset_for_leap_year = (distance_in_years / 4) * 1440
94
+ remainder = ((distance_in_minutes - minute_offset_for_leap_year) % 525600)
95
+ if remainder < 131400
96
+ locale.t(:about_x_years, :count => distance_in_years)
97
+ elsif remainder < 394200
98
+ locale.t(:over_x_years, :count => distance_in_years)
99
+ else
100
+ locale.t(:almost_x_years, :count => distance_in_years + 1)
101
+ end
102
+ end
103
+ end
104
+ end
105
+
106
+ # Like distance_of_time_in_words, but where <tt>to_time</tt> is fixed to <tt>Time.now</tt>.
107
+ #
108
+ # ==== Examples
109
+ # time_ago_in_words(3.minutes.from_now) # => 3 minutes
110
+ # time_ago_in_words(Time.now - 15.hours) # => 15 hours
111
+ # time_ago_in_words(Time.now) # => less than a minute
112
+ #
113
+ # from_time = Time.now - 3.days - 14.minutes - 25.seconds # => 3 days
114
+ def time_ago_in_words(from_time, include_seconds = false)
115
+ distance_of_time_in_words(from_time, Time.now, include_seconds)
116
+ end
117
+
118
+ alias_method :distance_of_time_in_words_to_now, :time_ago_in_words
119
+
120
+ # Returns a set of select tags (one for year, month, and day) pre-selected for accessing a specified date-based
121
+ # attribute (identified by +method+) on an object assigned to the template (identified by +object+).
122
+ #
123
+ #
124
+ # ==== Options
125
+ # * <tt>:use_month_numbers</tt> - Set to true if you want to use month numbers rather than month names (e.g.
126
+ # "2" instead of "February").
127
+ # * <tt>:use_short_month</tt> - Set to true if you want to use abbreviated month names instead of full
128
+ # month names (e.g. "Feb" instead of "February").
129
+ # * <tt>:add_month_numbers</tt> - Set to true if you want to use both month numbers and month names (e.g.
130
+ # "2 - February" instead of "February").
131
+ # * <tt>:use_month_names</tt> - Set to an array with 12 month names if you want to customize month names.
132
+ # Note: You can also use Rails' i18n functionality for this.
133
+ # * <tt>:date_separator</tt> - Specifies a string to separate the date fields. Default is "" (i.e. nothing).
134
+ # * <tt>:start_year</tt> - Set the start year for the year select. Default is <tt>Time.now.year - 5</tt>.
135
+ # * <tt>:end_year</tt> - Set the end year for the year select. Default is <tt>Time.now.year + 5</tt>.
136
+ # * <tt>:discard_day</tt> - Set to true if you don't want to show a day select. This includes the day
137
+ # as a hidden field instead of showing a select field. Also note that this implicitly sets the day to be the
138
+ # first of the given month in order to not create invalid dates like 31 February.
139
+ # * <tt>:discard_month</tt> - Set to true if you don't want to show a month select. This includes the month
140
+ # as a hidden field instead of showing a select field. Also note that this implicitly sets :discard_day to true.
141
+ # * <tt>:discard_year</tt> - Set to true if you don't want to show a year select. This includes the year
142
+ # as a hidden field instead of showing a select field.
143
+ # * <tt>:order</tt> - Set to an array containing <tt>:day</tt>, <tt>:month</tt> and <tt>:year</tt> to
144
+ # customize the order in which the select fields are shown. If you leave out any of the symbols, the respective
145
+ # select will not be shown (like when you set <tt>:discard_xxx => true</tt>. Defaults to the order defined in
146
+ # the respective locale (e.g. [:year, :month, :day] in the en locale that ships with Rails).
147
+ # * <tt>:include_blank</tt> - Include a blank option in every select field so it's possible to set empty
148
+ # dates.
149
+ # * <tt>:default</tt> - Set a default date if the affected date isn't set or is nil.
150
+ # * <tt>:disabled</tt> - Set to true if you want show the select fields as disabled.
151
+ # * <tt>:prompt</tt> - Set to true (for a generic prompt), a prompt string or a hash of prompt strings
152
+ # for <tt>:year</tt>, <tt>:month</tt>, <tt>:day</tt>, <tt>:hour</tt>, <tt>:minute</tt> and <tt>:second</tt>.
153
+ # Setting this option prepends a select option with a generic prompt (Day, Month, Year, Hour, Minute, Seconds)
154
+ # or the given prompt string.
155
+ #
156
+ # If anything is passed in the +html_options+ hash it will be applied to every select tag in the set.
157
+ #
158
+ # NOTE: Discarded selects will default to 1. So if no month select is available, January will be assumed.
159
+ #
160
+ # ==== Examples
161
+ # # Generates a date select that when POSTed is stored in the post variable, in the written_on attribute
162
+ # date_select("post", "written_on")
163
+ #
164
+ # # Generates a date select that when POSTed is stored in the post variable, in the written_on attribute,
165
+ # # with the year in the year drop down box starting at 1995.
166
+ # date_select("post", "written_on", :start_year => 1995)
167
+ #
168
+ # # Generates a date select that when POSTed is stored in the post variable, in the written_on attribute,
169
+ # # with the year in the year drop down box starting at 1995, numbers used for months instead of words,
170
+ # # and without a day select box.
171
+ # date_select("post", "written_on", :start_year => 1995, :use_month_numbers => true,
172
+ # :discard_day => true, :include_blank => true)
173
+ #
174
+ # # Generates a date select that when POSTed is stored in the post variable, in the written_on attribute
175
+ # # with the fields ordered as day, month, year rather than month, day, year.
176
+ # date_select("post", "written_on", :order => [:day, :month, :year])
177
+ #
178
+ # # Generates a date select that when POSTed is stored in the user variable, in the birthday attribute
179
+ # # lacking a year field.
180
+ # date_select("user", "birthday", :order => [:month, :day])
181
+ #
182
+ # # Generates a date select that when POSTed is stored in the user variable, in the birthday attribute
183
+ # # which is initially set to the date 3 days from the current date
184
+ # date_select("post", "written_on", :default => 3.days.from_now)
185
+ #
186
+ # # Generates a date select that when POSTed is stored in the credit_card variable, in the bill_due attribute
187
+ # # that will have a default day of 20.
188
+ # date_select("credit_card", "bill_due", :default => { :day => 20 })
189
+ #
190
+ # # Generates a date select with custom prompts
191
+ # date_select("post", "written_on", :prompt => { :day => 'Select day', :month => 'Select month', :year => 'Select year' })
192
+ #
193
+ # The selects are prepared for multi-parameter assignment to an Active Record object.
194
+ #
195
+ # Note: If the day is not included as an option but the month is, the day will be set to the 1st to ensure that
196
+ # all month choices are valid.
197
+ def date_select(object_name, method, options = {}, html_options = {})
198
+ InstanceTag.new(object_name, method, self, options.delete(:object)).to_date_select_tag(options, html_options)
199
+ end
200
+
201
+ # Returns a set of select tags (one for hour, minute and optionally second) pre-selected for accessing a
202
+ # specified time-based attribute (identified by +method+) on an object assigned to the template (identified by
203
+ # +object+). You can include the seconds with <tt>:include_seconds</tt>.
204
+ #
205
+ # This method will also generate 3 input hidden tags, for the actual year, month and day unless the option
206
+ # <tt>:ignore_date</tt> is set to +true+.
207
+ #
208
+ # If anything is passed in the html_options hash it will be applied to every select tag in the set.
209
+ #
210
+ # ==== Examples
211
+ # # Creates a time select tag that, when POSTed, will be stored in the post variable in the sunrise attribute
212
+ # time_select("post", "sunrise")
213
+ #
214
+ # # Creates a time select tag that, when POSTed, will be stored in the order variable in the submitted
215
+ # # attribute
216
+ # time_select("order", "submitted")
217
+ #
218
+ # # Creates a time select tag that, when POSTed, will be stored in the mail variable in the sent_at attribute
219
+ # time_select("mail", "sent_at")
220
+ #
221
+ # # Creates a time select tag with a seconds field that, when POSTed, will be stored in the post variables in
222
+ # # the sunrise attribute.
223
+ # time_select("post", "start_time", :include_seconds => true)
224
+ #
225
+ # # Creates a time select tag with a seconds field that, when POSTed, will be stored in the entry variables in
226
+ # # the submission_time attribute.
227
+ # time_select("entry", "submission_time", :include_seconds => true)
228
+ #
229
+ # # You can set the :minute_step to 15 which will give you: 00, 15, 30 and 45.
230
+ # time_select 'game', 'game_time', {:minute_step => 15}
231
+ #
232
+ # # Creates a time select tag with a custom prompt. Use :prompt => true for generic prompts.
233
+ # time_select("post", "written_on", :prompt => {:hour => 'Choose hour', :minute => 'Choose minute', :second => 'Choose seconds'})
234
+ # time_select("post", "written_on", :prompt => {:hour => true}) # generic prompt for hours
235
+ # time_select("post", "written_on", :prompt => true) # generic prompts for all
236
+ #
237
+ # The selects are prepared for multi-parameter assignment to an Active Record object.
238
+ #
239
+ # Note: If the day is not included as an option but the month is, the day will be set to the 1st to ensure that
240
+ # all month choices are valid.
241
+ def time_select(object_name, method, options = {}, html_options = {})
242
+ InstanceTag.new(object_name, method, self, options.delete(:object)).to_time_select_tag(options, html_options)
243
+ end
244
+
245
+ # Returns a set of select tags (one for year, month, day, hour, and minute) pre-selected for accessing a
246
+ # specified datetime-based attribute (identified by +method+) on an object assigned to the template (identified
247
+ # by +object+).
248
+ #
249
+ # If anything is passed in the html_options hash it will be applied to every select tag in the set.
250
+ #
251
+ # ==== Examples
252
+ # # Generates a datetime select that, when POSTed, will be stored in the post variable in the written_on
253
+ # # attribute
254
+ # datetime_select("post", "written_on")
255
+ #
256
+ # # Generates a datetime select with a year select that starts at 1995 that, when POSTed, will be stored in the
257
+ # # post variable in the written_on attribute.
258
+ # datetime_select("post", "written_on", :start_year => 1995)
259
+ #
260
+ # # Generates a datetime select with a default value of 3 days from the current time that, when POSTed, will
261
+ # # be stored in the trip variable in the departing attribute.
262
+ # datetime_select("trip", "departing", :default => 3.days.from_now)
263
+ #
264
+ # # Generates a datetime select that discards the type that, when POSTed, will be stored in the post variable
265
+ # # as the written_on attribute.
266
+ # datetime_select("post", "written_on", :discard_type => true)
267
+ #
268
+ # # Generates a datetime select with a custom prompt. Use :prompt=>true for generic prompts.
269
+ # datetime_select("post", "written_on", :prompt => {:day => 'Choose day', :month => 'Choose month', :year => 'Choose year'})
270
+ # datetime_select("post", "written_on", :prompt => {:hour => true}) # generic prompt for hours
271
+ # datetime_select("post", "written_on", :prompt => true) # generic prompts for all
272
+ #
273
+ # The selects are prepared for multi-parameter assignment to an Active Record object.
274
+ def datetime_select(object_name, method, options = {}, html_options = {})
275
+ InstanceTag.new(object_name, method, self, options.delete(:object)).to_datetime_select_tag(options, html_options)
276
+ end
277
+
278
+ # Returns a set of html select-tags (one for year, month, day, hour, and minute) pre-selected with the
279
+ # +datetime+. It's also possible to explicitly set the order of the tags using the <tt>:order</tt> option with
280
+ # an array of symbols <tt>:year</tt>, <tt>:month</tt> and <tt>:day</tt> in the desired order. If you do not
281
+ # supply a Symbol, it will be appended onto the <tt>:order</tt> passed in. You can also add
282
+ # <tt>:date_separator</tt>, <tt>:datetime_separator</tt> and <tt>:time_separator</tt> keys to the +options+ to
283
+ # control visual display of the elements.
284
+ #
285
+ # If anything is passed in the html_options hash it will be applied to every select tag in the set.
286
+ #
287
+ # ==== Examples
288
+ # my_date_time = Time.now + 4.days
289
+ #
290
+ # # Generates a datetime select that defaults to the datetime in my_date_time (four days after today)
291
+ # select_datetime(my_date_time)
292
+ #
293
+ # # Generates a datetime select that defaults to today (no specified datetime)
294
+ # select_datetime()
295
+ #
296
+ # # Generates a datetime select that defaults to the datetime in my_date_time (four days after today)
297
+ # # with the fields ordered year, month, day rather than month, day, year.
298
+ # select_datetime(my_date_time, :order => [:year, :month, :day])
299
+ #
300
+ # # Generates a datetime select that defaults to the datetime in my_date_time (four days after today)
301
+ # # with a '/' between each date field.
302
+ # select_datetime(my_date_time, :date_separator => '/')
303
+ #
304
+ # # Generates a datetime select that defaults to the datetime in my_date_time (four days after today)
305
+ # # with a date fields separated by '/', time fields separated by '' and the date and time fields
306
+ # # separated by a comma (',').
307
+ # select_datetime(my_date_time, :date_separator => '/', :time_separator => '', :datetime_separator => ',')
308
+ #
309
+ # # Generates a datetime select that discards the type of the field and defaults to the datetime in
310
+ # # my_date_time (four days after today)
311
+ # select_datetime(my_date_time, :discard_type => true)
312
+ #
313
+ # # Generates a datetime select that defaults to the datetime in my_date_time (four days after today)
314
+ # # prefixed with 'payday' rather than 'date'
315
+ # select_datetime(my_date_time, :prefix => 'payday')
316
+ #
317
+ # # Generates a datetime select with a custom prompt. Use :prompt=>true for generic prompts.
318
+ # select_datetime(my_date_time, :prompt => {:day => 'Choose day', :month => 'Choose month', :year => 'Choose year'})
319
+ # select_datetime(my_date_time, :prompt => {:hour => true}) # generic prompt for hours
320
+ # select_datetime(my_date_time, :prompt => true) # generic prompts for all
321
+ #
322
+ def select_datetime(datetime = Time.current, options = {}, html_options = {})
323
+ DateTimeSelector.new(datetime, options, html_options).select_datetime
324
+ end
325
+
326
+ # Returns a set of html select-tags (one for year, month, and day) pre-selected with the +date+.
327
+ # It's possible to explicitly set the order of the tags using the <tt>:order</tt> option with an array of
328
+ # symbols <tt>:year</tt>, <tt>:month</tt> and <tt>:day</tt> in the desired order. If you do not supply a Symbol,
329
+ # it will be appended onto the <tt>:order</tt> passed in.
330
+ #
331
+ # If anything is passed in the html_options hash it will be applied to every select tag in the set.
332
+ #
333
+ # ==== Examples
334
+ # my_date = Time.today + 6.days
335
+ #
336
+ # # Generates a date select that defaults to the date in my_date (six days after today)
337
+ # select_date(my_date)
338
+ #
339
+ # # Generates a date select that defaults to today (no specified date)
340
+ # select_date()
341
+ #
342
+ # # Generates a date select that defaults to the date in my_date (six days after today)
343
+ # # with the fields ordered year, month, day rather than month, day, year.
344
+ # select_date(my_date, :order => [:year, :month, :day])
345
+ #
346
+ # # Generates a date select that discards the type of the field and defaults to the date in
347
+ # # my_date (six days after today)
348
+ # select_date(my_date, :discard_type => true)
349
+ #
350
+ # # Generates a date select that defaults to the date in my_date,
351
+ # # which has fields separated by '/'
352
+ # select_date(my_date, :date_separator => '/')
353
+ #
354
+ # # Generates a date select that defaults to the datetime in my_date (six days after today)
355
+ # # prefixed with 'payday' rather than 'date'
356
+ # select_date(my_date, :prefix => 'payday')
357
+ #
358
+ # # Generates a date select with a custom prompt. Use :prompt=>true for generic prompts.
359
+ # select_date(my_date, :prompt => {:day => 'Choose day', :month => 'Choose month', :year => 'Choose year'})
360
+ # select_date(my_date, :prompt => {:hour => true}) # generic prompt for hours
361
+ # select_date(my_date, :prompt => true) # generic prompts for all
362
+ #
363
+ def select_date(date = Date.current, options = {}, html_options = {})
364
+ DateTimeSelector.new(date, options, html_options).select_date
365
+ end
366
+
367
+ # Returns a set of html select-tags (one for hour and minute)
368
+ # You can set <tt>:time_separator</tt> key to format the output, and
369
+ # the <tt>:include_seconds</tt> option to include an input for seconds.
370
+ #
371
+ # If anything is passed in the html_options hash it will be applied to every select tag in the set.
372
+ #
373
+ # ==== Examples
374
+ # my_time = Time.now + 5.days + 7.hours + 3.minutes + 14.seconds
375
+ #
376
+ # # Generates a time select that defaults to the time in my_time
377
+ # select_time(my_time)
378
+ #
379
+ # # Generates a time select that defaults to the current time (no specified time)
380
+ # select_time()
381
+ #
382
+ # # Generates a time select that defaults to the time in my_time,
383
+ # # which has fields separated by ':'
384
+ # select_time(my_time, :time_separator => ':')
385
+ #
386
+ # # Generates a time select that defaults to the time in my_time,
387
+ # # that also includes an input for seconds
388
+ # select_time(my_time, :include_seconds => true)
389
+ #
390
+ # # Generates a time select that defaults to the time in my_time, that has fields
391
+ # # separated by ':' and includes an input for seconds
392
+ # select_time(my_time, :time_separator => ':', :include_seconds => true)
393
+ #
394
+ # # Generates a time select with a custom prompt. Use :prompt=>true for generic prompts.
395
+ # select_time(my_time, :prompt => {:day => 'Choose day', :month => 'Choose month', :year => 'Choose year'})
396
+ # select_time(my_time, :prompt => {:hour => true}) # generic prompt for hours
397
+ # select_time(my_time, :prompt => true) # generic prompts for all
398
+ #
399
+ def select_time(datetime = Time.current, options = {}, html_options = {})
400
+ DateTimeSelector.new(datetime, options, html_options).select_time
401
+ end
402
+
403
+ # Returns a select tag with options for each of the seconds 0 through 59 with the current second selected.
404
+ # The <tt>second</tt> can also be substituted for a second number.
405
+ # Override the field name using the <tt>:field_name</tt> option, 'second' by default.
406
+ #
407
+ # ==== Examples
408
+ # my_time = Time.now + 16.minutes
409
+ #
410
+ # # Generates a select field for seconds that defaults to the seconds for the time in my_time
411
+ # select_second(my_time)
412
+ #
413
+ # # Generates a select field for seconds that defaults to the number given
414
+ # select_second(33)
415
+ #
416
+ # # Generates a select field for seconds that defaults to the seconds for the time in my_time
417
+ # # that is named 'interval' rather than 'second'
418
+ # select_second(my_time, :field_name => 'interval')
419
+ #
420
+ # # Generates a select field for seconds with a custom prompt. Use :prompt=>true for a
421
+ # # generic prompt.
422
+ # select_minute(14, :prompt => 'Choose seconds')
423
+ #
424
+ def select_second(datetime, options = {}, html_options = {})
425
+ DateTimeSelector.new(datetime, options, html_options).select_second
426
+ end
427
+
428
+ # Returns a select tag with options for each of the minutes 0 through 59 with the current minute selected.
429
+ # Also can return a select tag with options by <tt>minute_step</tt> from 0 through 59 with the 00 minute
430
+ # selected. The <tt>minute</tt> can also be substituted for a minute number.
431
+ # Override the field name using the <tt>:field_name</tt> option, 'minute' by default.
432
+ #
433
+ # ==== Examples
434
+ # my_time = Time.now + 6.hours
435
+ #
436
+ # # Generates a select field for minutes that defaults to the minutes for the time in my_time
437
+ # select_minute(my_time)
438
+ #
439
+ # # Generates a select field for minutes that defaults to the number given
440
+ # select_minute(14)
441
+ #
442
+ # # Generates a select field for minutes that defaults to the minutes for the time in my_time
443
+ # # that is named 'stride' rather than 'second'
444
+ # select_minute(my_time, :field_name => 'stride')
445
+ #
446
+ # # Generates a select field for minutes with a custom prompt. Use :prompt=>true for a
447
+ # # generic prompt.
448
+ # select_minute(14, :prompt => 'Choose minutes')
449
+ #
450
+ def select_minute(datetime, options = {}, html_options = {})
451
+ DateTimeSelector.new(datetime, options, html_options).select_minute
452
+ end
453
+
454
+ # Returns a select tag with options for each of the hours 0 through 23 with the current hour selected.
455
+ # The <tt>hour</tt> can also be substituted for a hour number.
456
+ # Override the field name using the <tt>:field_name</tt> option, 'hour' by default.
457
+ #
458
+ # ==== Examples
459
+ # my_time = Time.now + 6.hours
460
+ #
461
+ # # Generates a select field for hours that defaults to the hour for the time in my_time
462
+ # select_hour(my_time)
463
+ #
464
+ # # Generates a select field for hours that defaults to the number given
465
+ # select_hour(13)
466
+ #
467
+ # # Generates a select field for hours that defaults to the minutes for the time in my_time
468
+ # # that is named 'stride' rather than 'second'
469
+ # select_hour(my_time, :field_name => 'stride')
470
+ #
471
+ # # Generates a select field for hours with a custom prompt. Use :prompt => true for a
472
+ # # generic prompt.
473
+ # select_hour(13, :prompt =>'Choose hour')
474
+ #
475
+ def select_hour(datetime, options = {}, html_options = {})
476
+ DateTimeSelector.new(datetime, options, html_options).select_hour
477
+ end
478
+
479
+ # Returns a select tag with options for each of the days 1 through 31 with the current day selected.
480
+ # The <tt>date</tt> can also be substituted for a hour number.
481
+ # Override the field name using the <tt>:field_name</tt> option, 'day' by default.
482
+ #
483
+ # ==== Examples
484
+ # my_date = Time.today + 2.days
485
+ #
486
+ # # Generates a select field for days that defaults to the day for the date in my_date
487
+ # select_day(my_time)
488
+ #
489
+ # # Generates a select field for days that defaults to the number given
490
+ # select_day(5)
491
+ #
492
+ # # Generates a select field for days that defaults to the day for the date in my_date
493
+ # # that is named 'due' rather than 'day'
494
+ # select_day(my_time, :field_name => 'due')
495
+ #
496
+ # # Generates a select field for days with a custom prompt. Use :prompt => true for a
497
+ # # generic prompt.
498
+ # select_day(5, :prompt => 'Choose day')
499
+ #
500
+ def select_day(date, options = {}, html_options = {})
501
+ DateTimeSelector.new(date, options, html_options).select_day
502
+ end
503
+
504
+ # Returns a select tag with options for each of the months January through December with the current month
505
+ # selected. The month names are presented as keys (what's shown to the user) and the month numbers (1-12) are
506
+ # used as values (what's submitted to the server). It's also possible to use month numbers for the presentation
507
+ # instead of names -- set the <tt>:use_month_numbers</tt> key in +options+ to true for this to happen. If you
508
+ # want both numbers and names, set the <tt>:add_month_numbers</tt> key in +options+ to true. If you would prefer
509
+ # to show month names as abbreviations, set the <tt>:use_short_month</tt> key in +options+ to true. If you want
510
+ # to use your own month names, set the <tt>:use_month_names</tt> key in +options+ to an array of 12 month names.
511
+ # Override the field name using the <tt>:field_name</tt> option, 'month' by default.
512
+ #
513
+ # ==== Examples
514
+ # # Generates a select field for months that defaults to the current month that
515
+ # # will use keys like "January", "March".
516
+ # select_month(Date.today)
517
+ #
518
+ # # Generates a select field for months that defaults to the current month that
519
+ # # is named "start" rather than "month"
520
+ # select_month(Date.today, :field_name => 'start')
521
+ #
522
+ # # Generates a select field for months that defaults to the current month that
523
+ # # will use keys like "1", "3".
524
+ # select_month(Date.today, :use_month_numbers => true)
525
+ #
526
+ # # Generates a select field for months that defaults to the current month that
527
+ # # will use keys like "1 - January", "3 - March".
528
+ # select_month(Date.today, :add_month_numbers => true)
529
+ #
530
+ # # Generates a select field for months that defaults to the current month that
531
+ # # will use keys like "Jan", "Mar".
532
+ # select_month(Date.today, :use_short_month => true)
533
+ #
534
+ # # Generates a select field for months that defaults to the current month that
535
+ # # will use keys like "Januar", "Marts."
536
+ # select_month(Date.today, :use_month_names => %w(Januar Februar Marts ...))
537
+ #
538
+ # # Generates a select field for months with a custom prompt. Use :prompt => true for a
539
+ # # generic prompt.
540
+ # select_month(14, :prompt => 'Choose month')
541
+ #
542
+ def select_month(date, options = {}, html_options = {})
543
+ DateTimeSelector.new(date, options, html_options).select_month
544
+ end
545
+
546
+ # Returns a select tag with options for each of the five years on each side of the current, which is selected.
547
+ # The five year radius can be changed using the <tt>:start_year</tt> and <tt>:end_year</tt> keys in the
548
+ # +options+. Both ascending and descending year lists are supported by making <tt>:start_year</tt> less than or
549
+ # greater than <tt>:end_year</tt>. The <tt>date</tt> can also be substituted for a year given as a number.
550
+ # Override the field name using the <tt>:field_name</tt> option, 'year' by default.
551
+ #
552
+ # ==== Examples
553
+ # # Generates a select field for years that defaults to the current year that
554
+ # # has ascending year values
555
+ # select_year(Date.today, :start_year => 1992, :end_year => 2007)
556
+ #
557
+ # # Generates a select field for years that defaults to the current year that
558
+ # # is named 'birth' rather than 'year'
559
+ # select_year(Date.today, :field_name => 'birth')
560
+ #
561
+ # # Generates a select field for years that defaults to the current year that
562
+ # # has descending year values
563
+ # select_year(Date.today, :start_year => 2005, :end_year => 1900)
564
+ #
565
+ # # Generates a select field for years that defaults to the year 2006 that
566
+ # # has ascending year values
567
+ # select_year(2006, :start_year => 2000, :end_year => 2010)
568
+ #
569
+ # # Generates a select field for years with a custom prompt. Use :prompt => true for a
570
+ # # generic prompt.
571
+ # select_year(14, :prompt => 'Choose year')
572
+ #
573
+ def select_year(date, options = {}, html_options = {})
574
+ DateTimeSelector.new(date, options, html_options).select_year
575
+ end
576
+ end
577
+
578
+ class DateTimeSelector #:nodoc:
579
+ extend ActiveSupport::Memoizable
580
+ include ActionView::Helpers::TagHelper
581
+
582
+ DEFAULT_PREFIX = 'date'.freeze unless const_defined?('DEFAULT_PREFIX')
583
+ POSITION = {
584
+ :year => 1, :month => 2, :day => 3, :hour => 4, :minute => 5, :second => 6
585
+ }.freeze unless const_defined?('POSITION')
586
+
587
+ def initialize(datetime, options = {}, html_options = {})
588
+ @options = options.dup
589
+ @html_options = html_options.dup
590
+ @datetime = datetime
591
+ end
592
+
593
+ def select_datetime
594
+ # TODO: Remove tag conditional
595
+ # Ideally we could just join select_date and select_date for the tag case
596
+ if @options[:tag] && @options[:ignore_date]
597
+ select_time
598
+ elsif @options[:tag]
599
+ order = date_order.dup
600
+ order -= [:hour, :minute, :second]
601
+
602
+ @options[:discard_year] ||= true unless order.include?(:year)
603
+ @options[:discard_month] ||= true unless order.include?(:month)
604
+ @options[:discard_day] ||= true if @options[:discard_month] || !order.include?(:day)
605
+ @options[:discard_minute] ||= true if @options[:discard_hour]
606
+ @options[:discard_second] ||= true unless @options[:include_seconds] && !@options[:discard_minute]
607
+
608
+ # If the day is hidden and the month is visible, the day should be set to the 1st so all month choices are
609
+ # valid (otherwise it could be 31 and february wouldn't be a valid date)
610
+ if @datetime && @options[:discard_day] && !@options[:discard_month]
611
+ @datetime = @datetime.change(:day => 1)
612
+ end
613
+
614
+ [:day, :month, :year].each { |o| order.unshift(o) unless order.include?(o) }
615
+ order += [:hour, :minute, :second] unless @options[:discard_hour]
616
+
617
+ build_selects_from_types(order)
618
+ else
619
+ "#{select_date}#{@options[:datetime_separator]}#{select_time}"
620
+ end
621
+ end
622
+
623
+ def select_date
624
+ order = date_order.dup
625
+
626
+ # TODO: Remove tag conditional
627
+ if @options[:tag]
628
+ @options[:discard_hour] = true
629
+ @options[:discard_minute] = true
630
+ @options[:discard_second] = true
631
+
632
+ @options[:discard_year] ||= true unless order.include?(:year)
633
+ @options[:discard_month] ||= true unless order.include?(:month)
634
+ @options[:discard_day] ||= true if @options[:discard_month] || !order.include?(:day)
635
+
636
+ # If the day is hidden and the month is visible, the day should be set to the 1st so all month choices are
637
+ # valid (otherwise it could be 31 and february wouldn't be a valid date)
638
+ if @datetime && @options[:discard_day] && !@options[:discard_month]
639
+ @datetime = @datetime.change(:day => 1)
640
+ end
641
+ end
642
+
643
+ [:day, :month, :year].each { |o| order.unshift(o) unless order.include?(o) }
644
+
645
+ build_selects_from_types(order)
646
+ end
647
+
648
+ def select_time
649
+ order = []
650
+
651
+ # TODO: Remove tag conditional
652
+ if @options[:tag]
653
+ @options[:discard_month] = true
654
+ @options[:discard_year] = true
655
+ @options[:discard_day] = true
656
+ @options[:discard_second] ||= true unless @options[:include_seconds]
657
+
658
+ order += [:year, :month, :day] unless @options[:ignore_date]
659
+ end
660
+
661
+ order += [:hour, :minute]
662
+ order << :second if @options[:include_seconds]
663
+
664
+ build_selects_from_types(order)
665
+ end
666
+
667
+ def select_second
668
+ if @options[:use_hidden] || @options[:discard_second]
669
+ build_hidden(:second, sec) if @options[:include_seconds]
670
+ else
671
+ build_options_and_select(:second, sec)
672
+ end
673
+ end
674
+
675
+ def select_minute
676
+ if @options[:use_hidden] || @options[:discard_minute]
677
+ build_hidden(:minute, min)
678
+ else
679
+ build_options_and_select(:minute, min, :step => @options[:minute_step])
680
+ end
681
+ end
682
+
683
+ def select_hour
684
+ if @options[:use_hidden] || @options[:discard_hour]
685
+ build_hidden(:hour, hour)
686
+ else
687
+ build_options_and_select(:hour, hour, :end => 23)
688
+ end
689
+ end
690
+
691
+ def select_day
692
+ if @options[:use_hidden] || @options[:discard_day]
693
+ build_hidden(:day, day)
694
+ else
695
+ build_options_and_select(:day, day, :start => 1, :end => 31, :leading_zeros => false)
696
+ end
697
+ end
698
+
699
+ def select_month
700
+ if @options[:use_hidden] || @options[:discard_month]
701
+ build_hidden(:month, month)
702
+ else
703
+ month_options = []
704
+ 1.upto(12) do |month_number|
705
+ options = { :value => month_number }
706
+ options[:selected] = "selected" if month == month_number
707
+ month_options << content_tag(:option, month_name(month_number), options) + "\n"
708
+ end
709
+ build_select(:month, month_options.join)
710
+ end
711
+ end
712
+
713
+ def select_year
714
+ if !@datetime || @datetime == 0
715
+ val = ''
716
+ middle_year = Date.today.year
717
+ else
718
+ val = middle_year = year
719
+ end
720
+
721
+ if @options[:use_hidden] || @options[:discard_year]
722
+ build_hidden(:year, val)
723
+ else
724
+ options = {}
725
+ options[:start] = @options[:start_year] || middle_year - 5
726
+ options[:end] = @options[:end_year] || middle_year + 5
727
+ options[:step] = options[:start] < options[:end] ? 1 : -1
728
+ options[:leading_zeros] = false
729
+
730
+ build_options_and_select(:year, val, options)
731
+ end
732
+ end
733
+
734
+ private
735
+ %w( sec min hour day month year ).each do |method|
736
+ define_method(method) do
737
+ @datetime.kind_of?(Fixnum) ? @datetime : @datetime.send(method) if @datetime
738
+ end
739
+ end
740
+
741
+ # Returns translated month names, but also ensures that a custom month
742
+ # name array has a leading nil element
743
+ def month_names
744
+ month_names = @options[:use_month_names] || translated_month_names
745
+ month_names.unshift(nil) if month_names.size < 13
746
+ month_names
747
+ end
748
+ memoize :month_names
749
+
750
+ # Returns translated month names
751
+ # => [nil, "January", "February", "March",
752
+ # "April", "May", "June", "July",
753
+ # "August", "September", "October",
754
+ # "November", "December"]
755
+ #
756
+ # If :use_short_month option is set
757
+ # => [nil, "Jan", "Feb", "Mar", "Apr", "May", "Jun",
758
+ # "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
759
+ def translated_month_names
760
+ begin
761
+ key = @options[:use_short_month] ? :'date.abbr_month_names' : :'date.month_names'
762
+ I18n.translate(key, :locale => @options[:locale])
763
+ end
764
+ end
765
+
766
+ # Lookup month name for number
767
+ # month_name(1) => "January"
768
+ #
769
+ # If :use_month_numbers option is passed
770
+ # month_name(1) => 1
771
+ #
772
+ # If :add_month_numbers option is passed
773
+ # month_name(1) => "1 - January"
774
+ def month_name(number)
775
+ if @options[:use_month_numbers]
776
+ number
777
+ elsif @options[:add_month_numbers]
778
+ "#{number} - #{month_names[number]}"
779
+ else
780
+ month_names[number]
781
+ end
782
+ end
783
+
784
+ def date_order
785
+ @options[:order] || translated_date_order
786
+ end
787
+ memoize :date_order
788
+
789
+ def translated_date_order
790
+ begin
791
+ I18n.translate(:'date.order', :locale => @options[:locale]) || []
792
+ end
793
+ end
794
+
795
+ # Build full select tag from date type and options
796
+ def build_options_and_select(type, selected, options = {})
797
+ build_select(type, build_options(selected, options))
798
+ end
799
+
800
+ # Build select option html from date value and options
801
+ # build_options(15, :start => 1, :end => 31)
802
+ # => "<option value="1">1</option>
803
+ # <option value=\"2\">2</option>
804
+ # <option value=\"3\">3</option>..."
805
+ def build_options(selected, options = {})
806
+ start = options.delete(:start) || 0
807
+ stop = options.delete(:end) || 59
808
+ step = options.delete(:step) || 1
809
+ leading_zeros = options.delete(:leading_zeros).nil? ? true : false
810
+
811
+ select_options = []
812
+ start.step(stop, step) do |i|
813
+ value = leading_zeros ? sprintf("%02d", i) : i
814
+ tag_options = { :value => value }
815
+ tag_options[:selected] = "selected" if selected == i
816
+ select_options << content_tag(:option, value, tag_options)
817
+ end
818
+ select_options.join("\n") + "\n"
819
+ end
820
+
821
+ # Builds select tag from date type and html select options
822
+ # build_select(:month, "<option value="1">January</option>...")
823
+ # => "<select id="post_written_on_2i" name="post[written_on(2i)]">
824
+ # <option value="1">January</option>...
825
+ # </select>"
826
+ def build_select(type, select_options_as_html)
827
+ select_options = {
828
+ :id => input_id_from_type(type),
829
+ :name => input_name_from_type(type)
830
+ }.merge(@html_options)
831
+ select_options.merge!(:disabled => 'disabled') if @options[:disabled]
832
+
833
+ select_html = "\n"
834
+ select_html << content_tag(:option, '', :value => '') + "\n" if @options[:include_blank]
835
+ select_html << prompt_option_tag(type, @options[:prompt]) + "\n" if @options[:prompt]
836
+ select_html << select_options_as_html.to_s
837
+
838
+ content_tag(:select, select_html, select_options) + "\n"
839
+ end
840
+
841
+ # Builds a prompt option tag with supplied options or from default options
842
+ # prompt_option_tag(:month, :prompt => 'Select month')
843
+ # => "<option value="">Select month</option>"
844
+ def prompt_option_tag(type, options)
845
+ default_options = {:year => false, :month => false, :day => false, :hour => false, :minute => false, :second => false}
846
+
847
+ case options
848
+ when Hash
849
+ prompt = default_options.merge(options)[type.to_sym]
850
+ when String
851
+ prompt = options
852
+ else
853
+ prompt = I18n.translate(('datetime.prompts.' + type.to_s).to_sym, :locale => @options[:locale])
854
+ end
855
+
856
+ prompt ? content_tag(:option, prompt, :value => '') : ''
857
+ end
858
+
859
+ # Builds hidden input tag for date part and value
860
+ # build_hidden(:year, 2008)
861
+ # => "<input id="post_written_on_1i" name="post[written_on(1i)]" type="hidden" value="2008" />"
862
+ def build_hidden(type, value)
863
+ tag(:input, {
864
+ :type => "hidden",
865
+ :id => input_id_from_type(type),
866
+ :name => input_name_from_type(type),
867
+ :value => value
868
+ }) + "\n"
869
+ end
870
+
871
+ # Returns the name attribute for the input tag
872
+ # => post[written_on(1i)]
873
+ def input_name_from_type(type)
874
+ prefix = @options[:prefix] || ActionView::Helpers::DateTimeSelector::DEFAULT_PREFIX
875
+ prefix += "[#{@options[:index]}]" if @options.has_key?(:index)
876
+
877
+ field_name = @options[:field_name] || type
878
+ if @options[:include_position]
879
+ field_name += "(#{ActionView::Helpers::DateTimeSelector::POSITION[type]}i)"
880
+ end
881
+
882
+ @options[:discard_type] ? prefix : "#{prefix}[#{field_name}]"
883
+ end
884
+
885
+ # Returns the id attribute for the input tag
886
+ # => "post_written_on_1i"
887
+ def input_id_from_type(type)
888
+ input_name_from_type(type).gsub(/([\[\(])|(\]\[)/, '_').gsub(/[\]\)]/, '')
889
+ end
890
+
891
+ # Given an ordering of datetime components, create the selection HTML
892
+ # and join them with their appropriate separators.
893
+ def build_selects_from_types(order)
894
+ select = ''
895
+ order.reverse.each do |type|
896
+ separator = separator(type) unless type == order.first # don't add on last field
897
+ select.insert(0, separator.to_s + send("select_#{type}").to_s)
898
+ end
899
+ select
900
+ end
901
+
902
+ # Returns the separator for a given datetime component
903
+ def separator(type)
904
+ case type
905
+ when :month, :day
906
+ @options[:date_separator]
907
+ when :hour
908
+ (@options[:discard_year] && @options[:discard_day]) ? "" : @options[:datetime_separator]
909
+ when :minute
910
+ @options[:time_separator]
911
+ when :second
912
+ @options[:include_seconds] ? @options[:time_separator] : ""
913
+ end
914
+ end
915
+ end
916
+
917
+ class InstanceTag #:nodoc:
918
+ def to_date_select_tag(options = {}, html_options = {})
919
+ datetime_selector(options, html_options).select_date.html_safe!
920
+ end
921
+
922
+ def to_time_select_tag(options = {}, html_options = {})
923
+ datetime_selector(options, html_options).select_time.html_safe!
924
+ end
925
+
926
+ def to_datetime_select_tag(options = {}, html_options = {})
927
+ datetime_selector(options, html_options).select_datetime.html_safe!
928
+ end
929
+
930
+ private
931
+ def datetime_selector(options, html_options)
932
+ datetime = value(object) || default_datetime(options)
933
+
934
+ options = options.dup
935
+ options[:field_name] = @method_name
936
+ options[:include_position] = true
937
+ options[:prefix] ||= @object_name
938
+ options[:index] = @auto_index if @auto_index && !options.has_key?(:index)
939
+ options[:datetime_separator] ||= ' &mdash; '
940
+ options[:time_separator] ||= ' : '
941
+
942
+ DateTimeSelector.new(datetime, options.merge(:tag => true), html_options)
943
+ end
944
+
945
+ def default_datetime(options)
946
+ return if options[:include_blank] || options[:prompt]
947
+
948
+ case options[:default]
949
+ when nil
950
+ Time.current
951
+ when Date, Time
952
+ options[:default]
953
+ else
954
+ default = options[:default].dup
955
+
956
+ # Rename :minute and :second to :min and :sec
957
+ default[:min] ||= default[:minute]
958
+ default[:sec] ||= default[:second]
959
+
960
+ time = Time.current
961
+
962
+ [:year, :month, :day, :hour, :min, :sec].each do |key|
963
+ default[key] ||= time.send(key)
964
+ end
965
+
966
+ Time.utc_time(
967
+ default[:year], default[:month], default[:day],
968
+ default[:hour], default[:min], default[:sec]
969
+ )
970
+ end
971
+ end
972
+ end
973
+
974
+ class FormBuilder
975
+ def date_select(method, options = {}, html_options = {})
976
+ @template.date_select(@object_name, method, objectify_options(options), html_options)
977
+ end
978
+
979
+ def time_select(method, options = {}, html_options = {})
980
+ @template.time_select(@object_name, method, objectify_options(options), html_options)
981
+ end
982
+
983
+ def datetime_select(method, options = {}, html_options = {})
984
+ @template.datetime_select(@object_name, method, objectify_options(options), html_options)
985
+ end
986
+ end
987
+ end
988
+ end