active_scaffold 3.0.23 → 3.0.24
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/frontends/default/views/_action_group.html.erb +1 -1
- data/frontends/default/views/_action_group.html.erb~ +24 -0
- data/frontends/default/views/_form.html.erb~ +26 -0
- data/frontends/default/views/_form_association.html.erb~ +19 -0
- data/frontends/default/views/_form_association_footer.html.erb~ +16 -6
- data/frontends/default/views/_horizontal_subform.html.erb~ +29 -0
- data/frontends/default/views/_horizontal_subform_header.html.erb~ +3 -2
- data/frontends/default/views/_list_actions.html.erb~ +15 -0
- data/frontends/default/views/_list_inline_adapter.html.erb~ +10 -0
- data/frontends/default/views/_list_messages.html.erb~ +30 -0
- data/frontends/default/views/_list_pagination.html.erb~ +11 -0
- data/frontends/default/views/_list_pagination_links.html.erb~ +0 -0
- data/frontends/default/views/_render_field.js.erb~ +23 -0
- data/frontends/default/views/_row.html.erb~ +6 -0
- data/frontends/default/views/_vertical_subform.html.erb~ +12 -0
- data/frontends/default/views/edit_associated.js.erb~ +13 -0
- data/frontends/default/views/on_create.js.rjs +2 -2
- data/frontends/default/views/render_field.js.erb~ +1 -0
- data/lib/active_scaffold/actions/core.rb~ +13 -5
- data/lib/active_scaffold/actions/create.rb~ +149 -0
- data/lib/active_scaffold/actions/list.rb~ +196 -0
- data/lib/active_scaffold/actions/nested.rb +6 -2
- data/lib/active_scaffold/actions/nested.rb~ +252 -0
- data/lib/active_scaffold/actions/search.rb~ +49 -0
- data/lib/active_scaffold/actions/subform.rb~ +27 -0
- data/lib/active_scaffold/actions/update.rb~ +149 -0
- data/lib/active_scaffold/attribute_params.rb~ +202 -0
- data/lib/active_scaffold/bridges/record_select/{lib/record_select_bridge.rb~ → helpers.rb~} +7 -16
- data/lib/active_scaffold/bridges/shared/date_bridge.rb~ +209 -0
- data/lib/active_scaffold/config/create.rb +4 -4
- data/lib/active_scaffold/config/nested.rb +1 -0
- data/lib/active_scaffold/config/nested.rb~ +41 -0
- data/lib/active_scaffold/config/search.rb~ +74 -0
- data/lib/active_scaffold/constraints.rb~ +186 -0
- data/lib/active_scaffold/data_structures/action_columns.rb~ +140 -0
- data/lib/active_scaffold/data_structures/action_link.rb +4 -4
- data/lib/active_scaffold/data_structures/action_link.rb~ +179 -0
- data/lib/active_scaffold/data_structures/nested_info.rb~ +124 -0
- data/lib/active_scaffold/extensions/action_controller_rendering.rb~ +22 -0
- data/lib/active_scaffold/extensions/action_view_rendering.rb~ +108 -0
- data/lib/active_scaffold/extensions/cache_association.rb~ +12 -0
- data/lib/active_scaffold/extensions/reverse_associations.rb~ +64 -0
- data/lib/active_scaffold/extensions/routing_mapper.rb~ +34 -0
- data/lib/active_scaffold/extensions/unsaved_associated.rb~ +62 -0
- data/lib/active_scaffold/finder.rb~ +370 -0
- data/lib/active_scaffold/helpers/controller_helpers.rb~ +101 -0
- data/lib/active_scaffold/helpers/form_column_helpers.rb~ +321 -0
- data/lib/active_scaffold/helpers/id_helpers.rb~ +123 -0
- data/lib/active_scaffold/helpers/list_column_helpers.rb +9 -6
- data/lib/active_scaffold/helpers/list_column_helpers.rb~ +368 -0
- data/lib/active_scaffold/helpers/search_column_helpers.rb~ +94 -46
- data/lib/active_scaffold/helpers/view_helpers.rb +1 -1
- data/lib/active_scaffold/helpers/view_helpers.rb~ +353 -0
- data/lib/active_scaffold/version.rb +1 -1
- data/lib/active_scaffold.rb +1 -1
- data/lib/active_scaffold.rb~ +362 -0
- metadata +110 -76
- data/lib/active_scaffold/bridges/dragonfly/bridge.rb~ +0 -12
- data/lib/active_scaffold/bridges/dragonfly/lib/dragonfly_bridge.rb~ +0 -36
- data/lib/active_scaffold/bridges/dragonfly/lib/dragonfly_bridge_helpers.rb~ +0 -12
- data/lib/active_scaffold/bridges/dragonfly/lib/form_ui.rb~ +0 -27
- data/lib/active_scaffold/bridges/dragonfly/lib/list_ui.rb~ +0 -16
@@ -0,0 +1,368 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
module ActiveScaffold
|
3
|
+
module Helpers
|
4
|
+
# Helpers that assist with the rendering of a List Column
|
5
|
+
module ListColumnHelpers
|
6
|
+
def get_column_value(record, column)
|
7
|
+
begin
|
8
|
+
# check for an override helper
|
9
|
+
value = if column_override? column
|
10
|
+
# we only pass the record as the argument. we previously also passed the formatted_value,
|
11
|
+
# but mike perham pointed out that prohibited the usage of overrides to improve on the
|
12
|
+
# performance of our default formatting. see issue #138.
|
13
|
+
send(column_override(column), record)
|
14
|
+
# second, check if the dev has specified a valid list_ui for this column
|
15
|
+
elsif column.list_ui and override_column_ui?(column.list_ui)
|
16
|
+
send(override_column_ui(column.list_ui), column, record)
|
17
|
+
elsif column.column and override_column_ui?(column.column.type)
|
18
|
+
send(override_column_ui(column.column.type), column, record)
|
19
|
+
else
|
20
|
+
format_column_value(record, column)
|
21
|
+
end
|
22
|
+
|
23
|
+
value = ' '.html_safe if value.nil? or (value.respond_to?(:empty?) and value.empty?) # fix for IE 6
|
24
|
+
return value
|
25
|
+
rescue Exception => e
|
26
|
+
logger.error Time.now.to_s + "#{e.inspect} -- on the ActiveScaffold column = :#{column.name} in #{controller.class}"
|
27
|
+
raise e
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
# TODO: move empty_field_text and logic in here?
|
32
|
+
# TODO: we need to distinguish between the automatic links *we* create and the ones that the dev specified. some logic may not apply if the dev specified the link.
|
33
|
+
def render_list_column(text, column, record)
|
34
|
+
if column.link
|
35
|
+
link = column.link
|
36
|
+
associated = record.send(column.association.name) if column.association
|
37
|
+
url_options = params_for(:action => nil, :id => record.id)
|
38
|
+
|
39
|
+
# setup automatic link
|
40
|
+
if column.autolink? && column.singular_association? # link to inline form
|
41
|
+
link = action_link_to_inline_form(column, record, associated, text)
|
42
|
+
return text if link.nil?
|
43
|
+
else
|
44
|
+
url_options[:link] = text
|
45
|
+
end
|
46
|
+
|
47
|
+
if column_link_authorized?(link, column, record, associated)
|
48
|
+
render_action_link(link, url_options, record)
|
49
|
+
else
|
50
|
+
"<a class='disabled'>#{text}</a>".html_safe
|
51
|
+
end
|
52
|
+
else
|
53
|
+
text = active_scaffold_inplace_edit(record, column, {:formatted_column => text}) if inplace_edit?(record, column)
|
54
|
+
text
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
# setup the action link to inline form
|
59
|
+
def action_link_to_inline_form(column, record, associated, text)
|
60
|
+
link = column.link.clone
|
61
|
+
link.label = text
|
62
|
+
if column.polymorphic_association?
|
63
|
+
polymorphic_controller = controller_path_for_activerecord(record.send(column.association.name).class)
|
64
|
+
return link if polymorphic_controller.nil?
|
65
|
+
link.controller = polymorphic_controller
|
66
|
+
end
|
67
|
+
configure_column_link(link, associated, column.actions_for_association_links)
|
68
|
+
end
|
69
|
+
|
70
|
+
def configure_column_link(link, associated, actions)
|
71
|
+
if column_empty?(associated) # if association is empty, we only can link to create form
|
72
|
+
if actions.include?(:new)
|
73
|
+
link.action = 'new'
|
74
|
+
link.crud_type = :create
|
75
|
+
link.label = as_(:create_new)
|
76
|
+
end
|
77
|
+
elsif actions.include?(:edit)
|
78
|
+
link.action = 'edit'
|
79
|
+
link.crud_type = :update
|
80
|
+
elsif actions.include?(:show)
|
81
|
+
link.action = 'show'
|
82
|
+
link.crud_type = :read
|
83
|
+
elsif actions.include?(:list)
|
84
|
+
link.action = 'index'
|
85
|
+
link.crud_type = :read
|
86
|
+
end
|
87
|
+
link if link.crud_type.present?
|
88
|
+
end
|
89
|
+
|
90
|
+
def column_link_authorized?(link, column, record, associated)
|
91
|
+
if column.association
|
92
|
+
associated_for_authorized = if associated.nil? || (associated.respond_to?(:blank?) && associated.blank?)
|
93
|
+
column.association.klass
|
94
|
+
elsif [:has_many, :has_and_belongs_to_many].include? column.association.macro
|
95
|
+
associated.first
|
96
|
+
else
|
97
|
+
associated
|
98
|
+
end
|
99
|
+
authorized = associated_for_authorized.authorized_for?(:crud_type => link.crud_type)
|
100
|
+
authorized = authorized and record.authorized_for?(:crud_type => :update, :column => column.name) if link.crud_type == :create
|
101
|
+
authorized
|
102
|
+
else
|
103
|
+
record.authorized_for?(:crud_type => link.crud_type)
|
104
|
+
end
|
105
|
+
end
|
106
|
+
|
107
|
+
# There are two basic ways to clean a column's value: h() and sanitize(). The latter is useful
|
108
|
+
# when the column contains *valid* html data, and you want to just disable any scripting. People
|
109
|
+
# can always use field overrides to clean data one way or the other, but having this override
|
110
|
+
# lets people decide which way it should happen by default.
|
111
|
+
#
|
112
|
+
# Why is it not a configuration option? Because it seems like a somewhat rare request. But it
|
113
|
+
# could eventually be an option in config.list (and config.show, I guess).
|
114
|
+
def clean_column_value(v)
|
115
|
+
h(v)
|
116
|
+
end
|
117
|
+
|
118
|
+
##
|
119
|
+
## Overrides
|
120
|
+
##
|
121
|
+
def active_scaffold_column_text(column, record)
|
122
|
+
truncate(clean_column_value(record.send(column.name)), :length => column.options[:truncate] || 50)
|
123
|
+
end
|
124
|
+
|
125
|
+
def active_scaffold_column_checkbox(column, record)
|
126
|
+
options = {:disabled => true, :id => nil, :object => record}
|
127
|
+
options.delete(:disabled) if inplace_edit?(record, column)
|
128
|
+
check_box(:record, column.name, options)
|
129
|
+
end
|
130
|
+
|
131
|
+
def column_override_name(column, class_prefix = false)
|
132
|
+
"#{clean_class_name(column.active_record_class.name) + '_' if class_prefix}#{clean_column_name(column.name)}_column"
|
133
|
+
end
|
134
|
+
|
135
|
+
def column_override(column)
|
136
|
+
method_with_class = column_override_name(column, true)
|
137
|
+
return method_with_class if respond_to?(method_with_class)
|
138
|
+
method = column_override_name(column)
|
139
|
+
method if respond_to?(method)
|
140
|
+
end
|
141
|
+
alias_method :column_override?, :column_override
|
142
|
+
|
143
|
+
def override_column_ui?(list_ui)
|
144
|
+
respond_to?(override_column_ui(list_ui))
|
145
|
+
end
|
146
|
+
|
147
|
+
# the naming convention for overriding column types with helpers
|
148
|
+
def override_column_ui(list_ui)
|
149
|
+
"active_scaffold_column_#{list_ui}"
|
150
|
+
end
|
151
|
+
|
152
|
+
##
|
153
|
+
## Formatting
|
154
|
+
##
|
155
|
+
def format_column_value(record, column, value = nil)
|
156
|
+
value ||= record.send(column.name) unless record.nil?
|
157
|
+
if value && column.association # cache association size before calling column_empty?
|
158
|
+
associated_size = value.size if column.plural_association? and column.associated_number? # get count before cache association
|
159
|
+
cache_association(value, column) if column.plural_association?
|
160
|
+
end
|
161
|
+
if column.association.nil? or column_empty?(value)
|
162
|
+
if column.form_ui == :select && column.options[:options]
|
163
|
+
text, val = column.options[:options].find {|text, val| (val.nil? ? text : val).to_s == value.to_s}
|
164
|
+
value = active_scaffold_translated_option(column, text, val).first if text
|
165
|
+
end
|
166
|
+
if value.is_a? Numeric
|
167
|
+
format_number_value(value, column.options)
|
168
|
+
else
|
169
|
+
format_value(value, column.options)
|
170
|
+
end
|
171
|
+
else
|
172
|
+
format_association_value(value, column, associated_size)
|
173
|
+
end
|
174
|
+
end
|
175
|
+
|
176
|
+
def format_number_value(value, options = {})
|
177
|
+
value = case options[:format]
|
178
|
+
when :size
|
179
|
+
number_to_human_size(value, options[:i18n_options] || {})
|
180
|
+
when :percentage
|
181
|
+
number_to_percentage(value, options[:i18n_options] || {})
|
182
|
+
when :currency
|
183
|
+
number_to_currency(value, options[:i18n_options] || {})
|
184
|
+
when :i18n_number
|
185
|
+
number_with_delimiter(value, options[:i18n_options] || {})
|
186
|
+
else
|
187
|
+
value
|
188
|
+
end
|
189
|
+
clean_column_value(value)
|
190
|
+
end
|
191
|
+
|
192
|
+
def format_association_value(value, column, size)
|
193
|
+
case column.association.macro
|
194
|
+
when :has_one, :belongs_to
|
195
|
+
if column.polymorphic_association?
|
196
|
+
format_value("#{value.class.model_name.human}: #{value.to_label}")
|
197
|
+
else
|
198
|
+
format_value(value.to_label)
|
199
|
+
end
|
200
|
+
when :has_many, :has_and_belongs_to_many
|
201
|
+
if column.associated_limit.nil?
|
202
|
+
firsts = value.collect { |v| clean_column_value(v.to_label) }
|
203
|
+
else
|
204
|
+
firsts = value.first(column.associated_limit)
|
205
|
+
firsts.collect! { |v| clean_column_value(v.to_label) }
|
206
|
+
firsts[column.associated_limit] = '…' if value.size > column.associated_limit
|
207
|
+
end
|
208
|
+
if column.associated_limit == 0
|
209
|
+
size if column.associated_number?
|
210
|
+
else
|
211
|
+
joined_associated = format_value(firsts.join(active_scaffold_config.list.association_join_text))
|
212
|
+
joined_associated << " (#{size})" if column.associated_number? and column.associated_limit and value.size > column.associated_limit
|
213
|
+
joined_associated
|
214
|
+
end
|
215
|
+
end
|
216
|
+
end
|
217
|
+
|
218
|
+
def format_value(column_value, options = {})
|
219
|
+
value = if column_empty?(column_value)
|
220
|
+
active_scaffold_config.list.empty_field_text
|
221
|
+
elsif column_value.is_a?(Time) || column_value.is_a?(Date)
|
222
|
+
l(column_value, :format => options[:format] || :default)
|
223
|
+
elsif [FalseClass, TrueClass].include?(column_value.class)
|
224
|
+
as_(column_value.to_s.to_sym)
|
225
|
+
else
|
226
|
+
column_value.to_s
|
227
|
+
end
|
228
|
+
clean_column_value(value)
|
229
|
+
end
|
230
|
+
|
231
|
+
def cache_association(value, column)
|
232
|
+
# we are not using eager loading, cache firsts records in order not to query the database in a future
|
233
|
+
unless value.loaded?
|
234
|
+
# load at least one record, is needed for column_empty? and checking permissions
|
235
|
+
if column.associated_limit.nil?
|
236
|
+
Rails.logger.warn "ActiveScaffold: Enable eager loading for #{column.name} association to reduce SQL queries"
|
237
|
+
else
|
238
|
+
value.target = value.find(:all, :limit => column.associated_limit + 1, :select => column.select_columns)
|
239
|
+
end
|
240
|
+
end
|
241
|
+
end
|
242
|
+
|
243
|
+
# ==========
|
244
|
+
# = Inline Edit =
|
245
|
+
# ==========
|
246
|
+
|
247
|
+
def inplace_edit?(record, column)
|
248
|
+
if column.inplace_edit
|
249
|
+
editable = controller.send(:update_authorized?, record) if controller.respond_to?(:update_authorized?)
|
250
|
+
editable = record.authorized_for?(:crud_type => :update, :column => column.name) if editable.nil? || editable == true
|
251
|
+
editable
|
252
|
+
end
|
253
|
+
end
|
254
|
+
|
255
|
+
def inplace_edit_cloning?(column)
|
256
|
+
column.inplace_edit != :ajax and (override_form_field?(column) or column.form_ui or (column.column and override_input?(column.column.type)))
|
257
|
+
end
|
258
|
+
|
259
|
+
def format_inplace_edit_column(record,column)
|
260
|
+
if column.list_ui == :checkbox
|
261
|
+
active_scaffold_column_checkbox(column, record)
|
262
|
+
else
|
263
|
+
format_column_value(record, column)
|
264
|
+
end
|
265
|
+
end
|
266
|
+
|
267
|
+
def active_scaffold_inplace_edit(record, column, options = {})
|
268
|
+
formatted_column = options[:formatted_column] || format_column_value(record, column)
|
269
|
+
id_options = {:id => record.id.to_s, :action => 'update_column', :name => column.name.to_s}
|
270
|
+
tag_options = {:id => element_cell_id(id_options), :class => "in_place_editor_field",
|
271
|
+
:title => as_(:click_to_edit), 'data-ie_id' => record.id.to_s}
|
272
|
+
|
273
|
+
content_tag(:span, formatted_column, tag_options)
|
274
|
+
end
|
275
|
+
|
276
|
+
def inplace_edit_control(column)
|
277
|
+
if inplace_edit?(active_scaffold_config.model, column) and inplace_edit_cloning?(column)
|
278
|
+
@record = new_model
|
279
|
+
column = column.clone
|
280
|
+
column.options = column.options.clone
|
281
|
+
column.form_ui = :select if (column.association && column.form_ui.nil?)
|
282
|
+
content_tag(:div, active_scaffold_input_for(column), {:style => "display:none;", :class => inplace_edit_control_css_class})
|
283
|
+
end
|
284
|
+
end
|
285
|
+
|
286
|
+
def inplace_edit_control_css_class
|
287
|
+
"as_inplace_pattern"
|
288
|
+
end
|
289
|
+
|
290
|
+
def inplace_edit_tag_attributes(column)
|
291
|
+
tag_options = {}
|
292
|
+
tag_options['data-ie_url'] = url_for({:controller => params_for[:controller], :action => "update_column", :column => column.name, :id => '__id__'})
|
293
|
+
tag_options['data-ie_cancel_text'] = column.options[:cancel_text] || as_(:cancel)
|
294
|
+
tag_options['data-ie_loading_text'] = column.options[:loading_text] || as_(:loading)
|
295
|
+
tag_options['data-ie_save_text'] = column.options[:save_text] || as_(:update)
|
296
|
+
tag_options['data-ie_saving_text'] = column.options[:saving_text] || as_(:saving)
|
297
|
+
tag_options['data-ie_rows'] = column.options[:rows] || 5 if column.column.try(:type) == :text
|
298
|
+
tag_options['data-ie_cols'] = column.options[:cols] if column.options[:cols]
|
299
|
+
tag_options['data-ie_size'] = column.options[:size] if column.options[:size]
|
300
|
+
|
301
|
+
if column.list_ui == :checkbox
|
302
|
+
tag_options['data-ie_mode'] = :inline_checkbox
|
303
|
+
elsif inplace_edit_cloning?(column)
|
304
|
+
tag_options['data-ie_mode'] = :clone
|
305
|
+
elsif column.inplace_edit == :ajax
|
306
|
+
url = url_for(:controller => params_for[:controller], :action => 'render_field', :id => '__id__', :column => column.name, :update_column => column.name, :in_place_editing => true, :escape => false)
|
307
|
+
plural = column.plural_association? && !override_form_field?(column) && [:select, :record_select].include?(column.form_ui)
|
308
|
+
tag_options['data-ie_render_url'] = url
|
309
|
+
tag_options['data-ie_mode'] = :ajax
|
310
|
+
tag_options['data-ie_plural'] = plural
|
311
|
+
end
|
312
|
+
tag_options
|
313
|
+
end
|
314
|
+
|
315
|
+
def mark_column_heading
|
316
|
+
if active_scaffold_config.mark.mark_all_mode == :page then
|
317
|
+
all_marked = true
|
318
|
+
@page.items.each do |record|
|
319
|
+
all_marked = false if !marked_records.entries.include?(record.id)
|
320
|
+
end
|
321
|
+
else
|
322
|
+
all_marked = (marked_records.length >= @page.pager.count)
|
323
|
+
end
|
324
|
+
tag_options = {:id => "#{controller_id}_mark_heading", :class => "mark_heading in_place_editor_field"}
|
325
|
+
tag_options['data-ie_url'] = url_for({:controller => params_for[:controller], :action => 'mark_all', :eid => params[:eid]})
|
326
|
+
content_tag(:span, check_box_tag("#{controller_id}_mark_heading_span_input", !all_marked, all_marked), tag_options)
|
327
|
+
end
|
328
|
+
|
329
|
+
def render_column_heading(column, sorting, sort_direction)
|
330
|
+
tag_options = {:id => active_scaffold_column_header_id(column), :class => column_heading_class(column, sorting), :title => column.description}
|
331
|
+
tag_options.merge!(inplace_edit_tag_attributes(column)) if column.inplace_edit
|
332
|
+
content_tag(:th, column_heading_value(column, sorting, sort_direction) + inplace_edit_control(column), tag_options)
|
333
|
+
end
|
334
|
+
|
335
|
+
|
336
|
+
def column_heading_value(column, sorting, sort_direction)
|
337
|
+
if column.sortable?
|
338
|
+
options = {:id => nil, :class => "as_sort",
|
339
|
+
'data-page-history' => controller_id,
|
340
|
+
:remote => true, :method => :get}
|
341
|
+
url_options = params_for(:action => :index, :page => 1,
|
342
|
+
:sort => column.name, :sort_direction => sort_direction)
|
343
|
+
link_to column.label, url_options, options
|
344
|
+
else
|
345
|
+
if column.name != :marked
|
346
|
+
content_tag(:p, column.label)
|
347
|
+
else
|
348
|
+
mark_column_heading
|
349
|
+
end
|
350
|
+
end
|
351
|
+
end
|
352
|
+
|
353
|
+
def render_nested_view(action_links, url_options, record)
|
354
|
+
rendered = []
|
355
|
+
action_links.member.each do |link|
|
356
|
+
if link.nested_link? && link.column && @nested_auto_open[link.column.name] && @records.length <= @nested_auto_open[link.column.name] && controller.respond_to?(:render_component_into_view)
|
357
|
+
link_url_options = {:adapter => '_list_inline_adapter', :format => :js}.merge(action_link_url_options(link, url_options, record, options = {:reuse_eid => true}))
|
358
|
+
link_id = get_action_link_id(link_url_options, record, link.column)
|
359
|
+
rendered << (controller.send(:render_component_into_view, link_url_options) + javascript_tag("ActiveScaffold.ActionLink.get('#{link_id}').set_opened();"))
|
360
|
+
end
|
361
|
+
end
|
362
|
+
rendered.join(' ').html_safe
|
363
|
+
end
|
364
|
+
|
365
|
+
end
|
366
|
+
end
|
367
|
+
end
|
368
|
+
|
@@ -40,7 +40,7 @@ module ActiveScaffold
|
|
40
40
|
# for textual fields we pass different options
|
41
41
|
text_types = [:text, :string, :integer, :float, :decimal]
|
42
42
|
options = active_scaffold_input_text_options(options) if text_types.include?(column.column.type)
|
43
|
-
|
43
|
+
text_field(:record, column.name, options.merge(column.options))
|
44
44
|
end
|
45
45
|
end
|
46
46
|
end
|
@@ -59,37 +59,26 @@ module ActiveScaffold
|
|
59
59
|
associated = options.delete :value
|
60
60
|
associated = [associated].compact unless associated.is_a? Array
|
61
61
|
associated.collect!(&:to_i)
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
options[:name] += '[]'
|
68
|
-
select_options.each_with_index do |option, i|
|
69
|
-
label, id = option
|
70
|
-
this_id = "#{options[:id]}_#{i}_id"
|
71
|
-
html << "<li>"
|
72
|
-
html << check_box_tag(options[:name], id, associated.include?(id), :id => this_id)
|
73
|
-
html << "<label for='#{this_id}'>"
|
74
|
-
html << label
|
75
|
-
html << "</label>"
|
76
|
-
html << "</li>"
|
62
|
+
|
63
|
+
if column.association
|
64
|
+
select_options = options_for_association(column.association, false)
|
65
|
+
else
|
66
|
+
select_options = Array(column.options[:options])
|
77
67
|
end
|
68
|
+
return as_(:no_options) if select_options.empty?
|
78
69
|
|
79
|
-
|
80
|
-
html << javascript_tag("new DraggableLists('#{options[:id]}')") if column.options[:draggable_lists]
|
81
|
-
html
|
70
|
+
active_scaffold_checkbox_list(column, select_options, associated, options)
|
82
71
|
end
|
83
72
|
|
84
73
|
def active_scaffold_search_select(column, html_options)
|
85
74
|
associated = html_options.delete :value
|
86
75
|
if column.association
|
87
76
|
associated = associated.is_a?(Array) ? associated.map(&:to_i) : associated.to_i unless associated.nil?
|
88
|
-
method = column.association.macro == :belongs_to ? column.association.
|
89
|
-
|
77
|
+
method = column.association.macro == :belongs_to ? column.association.foreign_key : column.name
|
78
|
+
select_options = options_for_association(column.association, true)
|
90
79
|
else
|
91
80
|
method = column.name
|
92
|
-
|
81
|
+
select_options = Array(column.options[:options])
|
93
82
|
end
|
94
83
|
|
95
84
|
options = { :selected => associated }.merge! column.options
|
@@ -118,11 +107,43 @@ module ActiveScaffold
|
|
118
107
|
end
|
119
108
|
# we can't use checkbox ui because it's not possible to decide whether search for this field or not
|
120
109
|
alias_method :active_scaffold_search_checkbox, :active_scaffold_search_boolean
|
110
|
+
|
111
|
+
def active_scaffold_search_null(column, options)
|
112
|
+
select_options = []
|
113
|
+
select_options << [as_(:_select_), nil]
|
114
|
+
select_options.concat ActiveScaffold::Finder::NullComparators.collect {|comp| [as_(comp), comp]}
|
115
|
+
select_tag(options[:name], options_for_select(select_options, field_search_params[column.name]))
|
116
|
+
end
|
121
117
|
|
122
118
|
def field_search_params_range_values(column)
|
123
119
|
values = field_search_params[column.name]
|
124
120
|
return nil if values.nil?
|
125
|
-
return values[:opt], values[:from], values[:to]
|
121
|
+
return values[:opt], (values[:from].blank? ? nil : values[:from]), (values[:to].blank? ? nil : values[:to])
|
122
|
+
|
123
|
+
end
|
124
|
+
|
125
|
+
def active_scaffold_search_range_string?(column)
|
126
|
+
(column.column && column.column.text?) || column.search_ui == :string
|
127
|
+
end
|
128
|
+
|
129
|
+
def include_null_comparators?(column)
|
130
|
+
return column.options[:null_comparators] if column.options.has_key? :null_comparators
|
131
|
+
if column.association
|
132
|
+
column.association.macro != :belongs_to || active_scaffold_config.columns[column.association.primary_key_name].column.try(:null)
|
133
|
+
else
|
134
|
+
column.column.try(:null)
|
135
|
+
end
|
136
|
+
end
|
137
|
+
|
138
|
+
def active_scaffold_search_range_comparator_options(column)
|
139
|
+
select_options = ActiveScaffold::Finder::NumericComparators.collect {|comp| [as_(comp.downcase.to_sym), comp]}
|
140
|
+
if active_scaffold_search_range_string?(column)
|
141
|
+
select_options.unshift *ActiveScaffold::Finder::StringComparators.collect {|title, comp| [as_(title), comp]}
|
142
|
+
end
|
143
|
+
if include_null_comparators? column
|
144
|
+
select_options += ActiveScaffold::Finder::NullComparators.collect {|comp| [as_(comp), comp]}
|
145
|
+
end
|
146
|
+
select_options
|
126
147
|
end
|
127
148
|
|
128
149
|
def include_null_comparators?(column)
|
@@ -136,20 +157,29 @@ module ActiveScaffold
|
|
136
157
|
|
137
158
|
def active_scaffold_search_range(column, options)
|
138
159
|
opt_value, from_value, to_value = field_search_params_range_values(column)
|
139
|
-
select_options = ActiveScaffold::Finder::NumericComparators.collect {|comp| [as_(comp.downcase.to_sym), comp]}
|
140
|
-
select_options.unshift *ActiveScaffold::Finder::StringComparators.collect {|title, comp| [as_(title), comp]} if column.options[:string_comparators] || column.column.try(:text?)
|
141
|
-
select_options += ActiveScaffold::Finder::NullComparators.collect {|comp| [as_(comp.downcase.to_sym), comp]} if include_null_comparators? column
|
142
160
|
|
143
|
-
|
144
|
-
|
161
|
+
select_options = active_scaffold_search_range_comparator_options(column)
|
162
|
+
if active_scaffold_search_range_string?(column)
|
163
|
+
text_field_size = 15
|
164
|
+
opt_value ||= '%?%'
|
165
|
+
else
|
166
|
+
text_field_size = 10
|
167
|
+
opt_value ||= '='
|
168
|
+
end
|
169
|
+
|
170
|
+
from_value = controller.class.condition_value_for_numeric(column, from_value)
|
171
|
+
to_value = controller.class.condition_value_for_numeric(column, to_value)
|
172
|
+
from_value = format_number_value(from_value, column.options) if from_value.is_a?(Numeric)
|
173
|
+
to_value = format_number_value(to_value, column.options) if to_value.is_a?(Numeric)
|
174
|
+
html = select_tag("#{options[:name]}[opt]",
|
145
175
|
options_for_select(select_options, opt_value),
|
146
176
|
:id => "#{options[:id]}_opt",
|
147
|
-
:
|
148
|
-
html << text_field_tag("#{options[:name]}[from]", from_value, active_scaffold_input_text_options(:id => options[:id], :size =>
|
149
|
-
html << content_tag(:span, ' - ' + text_field_tag("#{options[:name]}[to]", to_value,
|
150
|
-
active_scaffold_input_text_options(:id => "#{options[:id]}_to", :size =>
|
151
|
-
:id => "#{options[:id]}_between", :style => "display:none")
|
152
|
-
html
|
177
|
+
:class => "as_search_range_option")
|
178
|
+
html << ' ' << text_field_tag("#{options[:name]}[from]", from_value, active_scaffold_input_text_options(:id => options[:id], :size => text_field_size))
|
179
|
+
html << ' ' << content_tag(:span, (' - ' + text_field_tag("#{options[:name]}[to]", to_value,
|
180
|
+
active_scaffold_input_text_options(:id => "#{options[:id]}_to", :size => text_field_size))).html_safe,
|
181
|
+
:id => "#{options[:id]}_between", :class => "as_search_range_between", :style => "display:#{(opt_value == 'BETWEEN') ? '' : 'none'}")
|
182
|
+
content_tag :span, html, :class => 'search_range'
|
153
183
|
end
|
154
184
|
alias_method :active_scaffold_search_integer, :active_scaffold_search_range
|
155
185
|
alias_method :active_scaffold_search_decimal, :active_scaffold_search_range
|
@@ -164,10 +194,9 @@ module ActiveScaffold
|
|
164
194
|
opt_value, from_value, to_value = field_search_params_range_values(column)
|
165
195
|
options = column.options.merge(options)
|
166
196
|
helper = "select_#{'date' unless options[:discard_date]}#{'time' unless options[:discard_time]}"
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
html * ' - '
|
197
|
+
|
198
|
+
send(helper, field_search_datetime_value(from_value), {:include_blank => true, :prefix => "#{options[:name]}[from]"}.merge(options)) <<
|
199
|
+
' - '.html_safe << send(helper, field_search_datetime_value(to_value), {:include_blank => true, :prefix => "#{options[:name]}[to]"}.merge(options))
|
171
200
|
end
|
172
201
|
|
173
202
|
def active_scaffold_search_date(column, options)
|
@@ -183,16 +212,9 @@ module ActiveScaffold
|
|
183
212
|
##
|
184
213
|
|
185
214
|
def override_search_field(column)
|
186
|
-
method = override_search_field_name(column)
|
187
|
-
return method if respond_to?(method)
|
188
|
-
old_method = override_search_field_name(column, true)
|
189
|
-
if respond_to?(old_method)
|
190
|
-
ActiveSupport::Deprecation.warn("You are using an old naming schema for overrides, you should name the helper #{method} instead of #{old_method}")
|
191
|
-
old_method
|
192
|
-
end
|
193
215
|
method_with_class = override_search_field_name(column, true)
|
194
216
|
return method_with_class if respond_to?(method_with_class)
|
195
|
-
method =
|
217
|
+
method = override_search_field_name(column)
|
196
218
|
method if respond_to?(method)
|
197
219
|
end
|
198
220
|
alias_method :override_search_field?, :override_search_field
|
@@ -210,6 +232,32 @@ module ActiveScaffold
|
|
210
232
|
def override_search(form_ui)
|
211
233
|
"active_scaffold_search_#{form_ui}"
|
212
234
|
end
|
235
|
+
|
236
|
+
def visibles_and_hiddens(search_config)
|
237
|
+
visibles = []
|
238
|
+
hiddens = []
|
239
|
+
search_config.columns.each do |column|
|
240
|
+
next unless column.search_sql
|
241
|
+
if search_config.optional_columns.include?(column.name) && !searched_by?(column)
|
242
|
+
hiddens << column
|
243
|
+
else
|
244
|
+
visibles << column
|
245
|
+
end
|
246
|
+
end
|
247
|
+
return visibles, hiddens
|
248
|
+
end
|
249
|
+
|
250
|
+
def searched_by?(column)
|
251
|
+
value = field_search_params[column.name]
|
252
|
+
case value
|
253
|
+
when Hash
|
254
|
+
!value['from'].blank?
|
255
|
+
when String
|
256
|
+
!value.blank?
|
257
|
+
else
|
258
|
+
false
|
259
|
+
end
|
260
|
+
end
|
213
261
|
end
|
214
262
|
end
|
215
263
|
end
|
@@ -161,7 +161,7 @@ module ActiveScaffold
|
|
161
161
|
|
162
162
|
def action_link_html_options(link, url_options, record, html_options)
|
163
163
|
link_id = get_action_link_id(url_options, record, link.column)
|
164
|
-
html_options.reverse_merge! link.html_options.merge(:class => link.action)
|
164
|
+
html_options.reverse_merge! link.html_options.merge(:class => link.action.to_s)
|
165
165
|
|
166
166
|
# Needs to be in html_options to as the adding _method to the url is no longer supported by Rails
|
167
167
|
html_options[:method] = link.method if link.method != :get
|