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,186 @@
|
|
1
|
+
module ActiveScaffold
|
2
|
+
module Constraints
|
3
|
+
|
4
|
+
protected
|
5
|
+
|
6
|
+
# Returns the current constraints
|
7
|
+
def active_scaffold_constraints
|
8
|
+
@active_scaffold_constraints ||= active_scaffold_session_storage[:constraints] || {}
|
9
|
+
end
|
10
|
+
|
11
|
+
def set_active_scaffold_constraints
|
12
|
+
associations_by_params = {}
|
13
|
+
active_scaffold_config.model.reflect_on_all_associations.each do |association|
|
14
|
+
associations_by_params[association.klass.name.foreign_key] = association.name unless association.options[:polymorphic]
|
15
|
+
end
|
16
|
+
params.each do |key, value|
|
17
|
+
active_scaffold_constraints[associations_by_params[key]] = value if associations_by_params.include? key
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
# For each enabled action, adds the constrained columns to the ActionColumns object (if it exists).
|
22
|
+
# This lets the ActionColumns object skip constrained columns.
|
23
|
+
#
|
24
|
+
# If the constraint value is a Hash, then we assume the constraint is a multi-level association constraint (the reverse of a has_many :through) and we do NOT register the constraint column.
|
25
|
+
def register_constraints_with_action_columns(association_constrained_fields = [], exclude_actions = [])
|
26
|
+
debugger
|
27
|
+
constrained_fields = active_scaffold_constraints.reject{|k, v| v.is_a? Hash}.keys.collect{|k| k.to_sym}
|
28
|
+
constrained_fields = constrained_fields | association_constrained_fields
|
29
|
+
if self.class.uses_active_scaffold?
|
30
|
+
# we actually want to do this whether constrained_fields exist or not, so that we can reset the array when they don't
|
31
|
+
active_scaffold_config.actions.each do |action_name|
|
32
|
+
next if exclude_actions.include?(action_name)
|
33
|
+
action = active_scaffold_config.send(action_name)
|
34
|
+
next unless action.respond_to? :columns
|
35
|
+
action.columns.constraint_columns = constrained_fields
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
# Returns search conditions based on the current scaffold constraints.
|
41
|
+
#
|
42
|
+
# Supports constraints based on either a column name (in which case it checks for an association
|
43
|
+
# or just uses the search_sql) or a database field name.
|
44
|
+
#
|
45
|
+
# All of this work is primarily to support nested scaffolds in a manner generally useful for other
|
46
|
+
# embedded scaffolds.
|
47
|
+
def conditions_from_constraints
|
48
|
+
conditions = nil
|
49
|
+
debugger
|
50
|
+
active_scaffold_constraints.each do |k, v|
|
51
|
+
column = active_scaffold_config.columns[k]
|
52
|
+
constraint_condition = if column
|
53
|
+
# Assume this is a multi-level association constraint.
|
54
|
+
# example:
|
55
|
+
# data model: Park -> Den -> Bear
|
56
|
+
# constraint: :den => {:park => 5}
|
57
|
+
if v.is_a? Hash
|
58
|
+
far_association = column.association.klass.reflect_on_association(v.keys.first)
|
59
|
+
field = far_association.klass.primary_key
|
60
|
+
table = far_association.table_name
|
61
|
+
|
62
|
+
active_scaffold_includes.concat([{k => v.keys.first}]) # e.g. {:den => :park}
|
63
|
+
constraint_condition_for("#{table}.#{field}", v.values.first)
|
64
|
+
|
65
|
+
# association column constraint
|
66
|
+
elsif column.association
|
67
|
+
if column.association.macro == :has_and_belongs_to_many
|
68
|
+
active_scaffold_habtm_joins.concat column.includes
|
69
|
+
else
|
70
|
+
active_scaffold_includes.concat column.includes
|
71
|
+
end
|
72
|
+
condition_from_association_constraint(column.association, v)
|
73
|
+
|
74
|
+
# regular column constraints
|
75
|
+
elsif column.searchable?
|
76
|
+
active_scaffold_includes.concat column.includes
|
77
|
+
constraint_condition_for(column.search_sql, v)
|
78
|
+
end
|
79
|
+
# unknown-to-activescaffold-but-real-database-column constraint
|
80
|
+
elsif active_scaffold_config.model.column_names.include? k.to_s
|
81
|
+
constraint_condition_for(k.to_s, v)
|
82
|
+
else
|
83
|
+
raise ActiveScaffold::MalformedConstraint, constraint_error(active_scaffold_config.model, k), caller
|
84
|
+
end
|
85
|
+
|
86
|
+
conditions = merge_conditions(conditions, constraint_condition)
|
87
|
+
end
|
88
|
+
|
89
|
+
conditions
|
90
|
+
end
|
91
|
+
|
92
|
+
# We do NOT want to use .search_sql. If anything, search_sql will refer
|
93
|
+
# to a human-searchable value on the associated record.
|
94
|
+
def condition_from_association_constraint(association, value)
|
95
|
+
# when the reverse association is a :belongs_to, the id for the associated object only exists as
|
96
|
+
# the primary_key on the other table. so for :has_one and :has_many (when the reverse is :belongs_to),
|
97
|
+
# we have to use the other model's primary_key.
|
98
|
+
#
|
99
|
+
# please see the relevant tests for concrete examples.
|
100
|
+
field = if [:has_one, :has_many].include?(association.macro)
|
101
|
+
association.klass.primary_key
|
102
|
+
elsif [:has_and_belongs_to_many].include?(association.macro)
|
103
|
+
association.association_foreign_key
|
104
|
+
else
|
105
|
+
association.options[:foreign_key] || association.name.to_s.foreign_key
|
106
|
+
end
|
107
|
+
|
108
|
+
table = case association.macro
|
109
|
+
when :has_and_belongs_to_many
|
110
|
+
association.options[:join_table]
|
111
|
+
|
112
|
+
when :belongs_to
|
113
|
+
active_scaffold_config.model.table_name
|
114
|
+
|
115
|
+
else
|
116
|
+
association.table_name
|
117
|
+
end
|
118
|
+
|
119
|
+
if association.options[:primary_key]
|
120
|
+
value = association.klass.find(value).send(association.options[:primary_key])
|
121
|
+
end
|
122
|
+
|
123
|
+
condition = constraint_condition_for("#{table}.#{field}", value)
|
124
|
+
if association.options[:polymorphic]
|
125
|
+
begin
|
126
|
+
parent_scaffold = "#{session_info[:parent_scaffold].to_s.camelize}Controller".constantize
|
127
|
+
condition = merge_conditions(
|
128
|
+
condition,
|
129
|
+
constraint_condition_for("#{table}.#{association.name}_type", parent_scaffold.active_scaffold_config.model_id.to_s)
|
130
|
+
)
|
131
|
+
rescue ActiveScaffold::ControllerNotFound
|
132
|
+
nil
|
133
|
+
end
|
134
|
+
end
|
135
|
+
|
136
|
+
condition
|
137
|
+
end
|
138
|
+
|
139
|
+
def constraint_error(klass, column_name)
|
140
|
+
"Malformed constraint `#{klass}##{column_name}'. If it's a legitimate column, and you are using a nested scaffold, please specify or double-check the reverse association name."
|
141
|
+
end
|
142
|
+
|
143
|
+
# Applies constraints to the given record.
|
144
|
+
#
|
145
|
+
# Searches through the known columns for association columns. If the given constraint is an association,
|
146
|
+
# it assumes that the constraint value is an id. It then does a association.klass.find with the value
|
147
|
+
# and adds the associated object to the record.
|
148
|
+
#
|
149
|
+
# For some operations ActiveRecord will automatically update the database. That's not always ok.
|
150
|
+
# If it *is* ok (e.g. you're in a transaction), then set :allow_autosave to true.
|
151
|
+
def apply_constraints_to_record(record, options = {})
|
152
|
+
options[:allow_autosave] = false if options[:allow_autosave].nil?
|
153
|
+
|
154
|
+
active_scaffold_constraints.each do |k, v|
|
155
|
+
column = active_scaffold_config.columns[k]
|
156
|
+
if column and column.association
|
157
|
+
if column.plural_association?
|
158
|
+
record.send("#{k}").send(:<<, column.association.klass.find(v))
|
159
|
+
elsif column.association.options[:polymorphic]
|
160
|
+
record.send("#{k}=", params[:parent_model].constantize.find(v))
|
161
|
+
else # regular singular association
|
162
|
+
record.send("#{k}=", column.association.klass.find(v))
|
163
|
+
|
164
|
+
# setting the belongs_to side of a has_one isn't safe. if the has_one was already
|
165
|
+
# specified, rails won't automatically clear out the previous associated record.
|
166
|
+
#
|
167
|
+
# note that we can't take the extra step to correct this unless we're permitted to
|
168
|
+
# run operations where activerecord auto-saves the object.
|
169
|
+
reverse = column.association.klass.reflect_on_association(column.association.reverse)
|
170
|
+
if reverse.macro == :has_one and options[:allow_autosave]
|
171
|
+
record.send(k).send("#{column.association.reverse}=", record)
|
172
|
+
end
|
173
|
+
end
|
174
|
+
else
|
175
|
+
record.send("#{k}=", v)
|
176
|
+
end
|
177
|
+
end
|
178
|
+
end
|
179
|
+
|
180
|
+
private
|
181
|
+
|
182
|
+
def constraint_condition_for(sql, value)
|
183
|
+
value.nil? ? "#{sql} IS NULL" : ["#{sql} = ?", value]
|
184
|
+
end
|
185
|
+
end
|
186
|
+
end
|
@@ -0,0 +1,140 @@
|
|
1
|
+
module ActiveScaffold::DataStructures
|
2
|
+
# A set of columns. These structures can be nested for organization.
|
3
|
+
class ActionColumns < ActiveScaffold::DataStructures::Set
|
4
|
+
include ActiveScaffold::Configurable
|
5
|
+
|
6
|
+
# this lets us refer back to the action responsible for this link, if it exists.
|
7
|
+
# the immediate need here is to get the crud_type so we can dynamically filter columns from the set.
|
8
|
+
attr_accessor :action
|
9
|
+
|
10
|
+
# labels are useful for the Create/Update forms, when we display columns in a grouped fashion and want to name them separately
|
11
|
+
attr_writer :label
|
12
|
+
def label
|
13
|
+
as_(@label) if @label
|
14
|
+
end
|
15
|
+
def name
|
16
|
+
@label
|
17
|
+
end
|
18
|
+
|
19
|
+
# Whether this column set is collapsed by default in contexts where collapsing is supported
|
20
|
+
attr_accessor :collapsed
|
21
|
+
|
22
|
+
# nests a subgroup in the column set
|
23
|
+
def add_subgroup(label, &proc)
|
24
|
+
columns = ActiveScaffold::DataStructures::ActionColumns.new
|
25
|
+
columns.label = label
|
26
|
+
columns.action = self.action
|
27
|
+
columns.configure &proc
|
28
|
+
self.exclude columns.collect_columns
|
29
|
+
self.add columns
|
30
|
+
end
|
31
|
+
|
32
|
+
def include?(item)
|
33
|
+
@set.each do |c|
|
34
|
+
return true if !c.is_a? Symbol and c.include? item
|
35
|
+
return true if c == item.to_sym
|
36
|
+
end
|
37
|
+
return false
|
38
|
+
end
|
39
|
+
|
40
|
+
def names
|
41
|
+
self.collect(&:name)
|
42
|
+
end
|
43
|
+
|
44
|
+
def names_without_auth_check
|
45
|
+
Array(@set)
|
46
|
+
end
|
47
|
+
|
48
|
+
protected
|
49
|
+
|
50
|
+
def collect_columns
|
51
|
+
@set.collect {|col| col.is_a?(ActiveScaffold::DataStructures::ActionColumns) ? col.collect_columns : col}
|
52
|
+
end
|
53
|
+
|
54
|
+
# called during clone or dup. makes the clone/dup deeper.
|
55
|
+
def initialize_copy(from)
|
56
|
+
@set = from.instance_variable_get('@set').clone
|
57
|
+
end
|
58
|
+
|
59
|
+
# A package of stuff to add after the configuration block. This is an attempt at making a certain level of functionality inaccessible during configuration, to reduce possible breakage from misuse.
|
60
|
+
# The bulk of the package is a means of connecting the referential column set (ActionColumns) with the actual column objects (Columns). This lets us iterate over the set and yield real column objects.
|
61
|
+
module AfterConfiguration
|
62
|
+
# Redefine the each method to yield actual Column objects.
|
63
|
+
# It will skip constrained and unauthorized columns.
|
64
|
+
#
|
65
|
+
# Options:
|
66
|
+
# * :flatten - whether to recursively iterate on nested sets. default is false.
|
67
|
+
# * :for - the record (or class) being iterated over. used for column-level security. default is the class.
|
68
|
+
def each(options = {}, &proc)
|
69
|
+
options[:for] ||= @columns.active_record_class
|
70
|
+
self.unauthorized_columns = []
|
71
|
+
@set.each do |item|
|
72
|
+
unless item.is_a? ActiveScaffold::DataStructures::ActionColumns
|
73
|
+
item = (@columns[item] || ActiveScaffold::DataStructures::Column.new(item.to_sym, @columns.active_record_class))
|
74
|
+
next if self.skip_column?(item, options)
|
75
|
+
end
|
76
|
+
if item.is_a? ActiveScaffold::DataStructures::ActionColumns and options.has_key?(:flatten) and options[:flatten]
|
77
|
+
item.each(options, &proc)
|
78
|
+
else
|
79
|
+
yield item
|
80
|
+
end
|
81
|
+
end
|
82
|
+
end
|
83
|
+
|
84
|
+
def collect_visible(options = {}, &proc)
|
85
|
+
columns = []
|
86
|
+
options[:for] ||= @columns.active_record_class
|
87
|
+
self.unauthorized_columns = []
|
88
|
+
@set.each do |item|
|
89
|
+
unless item.is_a? ActiveScaffold::DataStructures::ActionColumns
|
90
|
+
item = (@columns[item] || ActiveScaffold::DataStructures::Column.new(item.to_sym, @columns.active_record_class))
|
91
|
+
next if self.skip_column?(item, options)
|
92
|
+
end
|
93
|
+
if item.is_a? ActiveScaffold::DataStructures::ActionColumns and options.has_key?(:flatten) and options[:flatten]
|
94
|
+
columns = columns + item.collect(options, &proc)
|
95
|
+
else
|
96
|
+
columns << item
|
97
|
+
end
|
98
|
+
end
|
99
|
+
columns
|
100
|
+
end
|
101
|
+
|
102
|
+
def skip_column?(column, options)
|
103
|
+
result = false
|
104
|
+
# skip if this matches a constrained column
|
105
|
+
result = true if constraint_columns.include?(column.name.to_sym)
|
106
|
+
# skip if this matches the field_name of a constrained column
|
107
|
+
result = true if column.field_name and constraint_columns.include?(column.field_name.to_sym)
|
108
|
+
# skip this field if it's not authorized
|
109
|
+
unless options[:for].authorized_for?(:action => options[:action], :crud_type => options[:crud_type] || self.action.crud_type, :column => column.name)
|
110
|
+
self.unauthorized_columns << column.name.to_sym
|
111
|
+
result = true
|
112
|
+
end
|
113
|
+
return result
|
114
|
+
end
|
115
|
+
|
116
|
+
# registers a set of column objects (recursively, for all nested ActionColumns)
|
117
|
+
def set_columns(columns)
|
118
|
+
@columns = columns
|
119
|
+
# iterate over @set instead of self to avoid dealing with security queries
|
120
|
+
@set.each do |item|
|
121
|
+
item.set_columns(columns) if item.respond_to? :set_columns
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
125
|
+
attr_writer :constraint_columns
|
126
|
+
def constraint_columns
|
127
|
+
@constraint_columns ||= []
|
128
|
+
end
|
129
|
+
|
130
|
+
attr_writer :unauthorized_columns
|
131
|
+
def unauthorized_columns
|
132
|
+
@unauthorized_columns ||= []
|
133
|
+
end
|
134
|
+
|
135
|
+
def length
|
136
|
+
((@set - self.constraint_columns) - self.unauthorized_columns).length
|
137
|
+
end
|
138
|
+
end
|
139
|
+
end
|
140
|
+
end
|
@@ -3,15 +3,15 @@ module ActiveScaffold::DataStructures
|
|
3
3
|
# provides a quick way to set any property of the object from a hash
|
4
4
|
def initialize(action, options = {})
|
5
5
|
# set defaults
|
6
|
-
self.action = action
|
6
|
+
self.action = action
|
7
7
|
self.label = action
|
8
8
|
self.confirm = false
|
9
9
|
self.type = :collection
|
10
10
|
self.inline = true
|
11
11
|
self.method = :get
|
12
|
-
self.crud_type = :delete if [:destroy].include?(action.to_sym)
|
13
|
-
self.crud_type = :create if [:create, :new].include?(action.to_sym)
|
14
|
-
self.crud_type = :update if [:edit, :update].include?(action.to_sym)
|
12
|
+
self.crud_type = :delete if [:destroy].include?(action.try(:to_sym))
|
13
|
+
self.crud_type = :create if [:create, :new].include?(action.try(:to_sym))
|
14
|
+
self.crud_type = :update if [:edit, :update].include?(action.try(:to_sym))
|
15
15
|
self.crud_type ||= :read
|
16
16
|
self.parameters = {}
|
17
17
|
self.html_options = {}
|
@@ -0,0 +1,179 @@
|
|
1
|
+
module ActiveScaffold::DataStructures
|
2
|
+
class ActionLink
|
3
|
+
# provides a quick way to set any property of the object from a hash
|
4
|
+
def initialize(action, options = {})
|
5
|
+
# set defaults
|
6
|
+
self.action = action.to_s
|
7
|
+
self.label = action
|
8
|
+
self.confirm = false
|
9
|
+
self.type = :collection
|
10
|
+
self.inline = true
|
11
|
+
self.method = :get
|
12
|
+
self.crud_type = :delete if [:destroy].include?(action.try(:to_sym))
|
13
|
+
self.crud_type = :create if [:create, :new].include?(action.try(:to_sym))
|
14
|
+
self.crud_type = :update if [:edit, :update].include?(action.try(:to_sym))
|
15
|
+
self.crud_type ||= :read
|
16
|
+
self.parameters = {}
|
17
|
+
self.html_options = {}
|
18
|
+
self.column = nil
|
19
|
+
self.image = nil
|
20
|
+
self.dynamic_parameters = nil
|
21
|
+
|
22
|
+
# apply quick properties
|
23
|
+
options.each_pair do |k, v|
|
24
|
+
setter = "#{k}="
|
25
|
+
self.send(setter, v) if self.respond_to? setter
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
# the action-path for this link. what page to request? this is required!
|
30
|
+
attr_accessor :action
|
31
|
+
|
32
|
+
# the controller for this action link. if nil, the current controller should be assumed.
|
33
|
+
attr_writer :controller
|
34
|
+
|
35
|
+
def controller
|
36
|
+
@controller = @controller.call if @controller.is_a?(Proc)
|
37
|
+
@controller
|
38
|
+
end
|
39
|
+
|
40
|
+
def static_controller?
|
41
|
+
!(@controller.is_a?(Proc) || (@controller == :polymorph))
|
42
|
+
end
|
43
|
+
|
44
|
+
# a hash of request parameters
|
45
|
+
attr_accessor :parameters
|
46
|
+
|
47
|
+
# a block for dynamic_parameters
|
48
|
+
attr_accessor :dynamic_parameters
|
49
|
+
|
50
|
+
# the RESTful method
|
51
|
+
attr_accessor :method
|
52
|
+
|
53
|
+
# what string to use to represent this action
|
54
|
+
attr_writer :label
|
55
|
+
def label
|
56
|
+
@label.is_a?(Symbol) ? as_(@label) : @label
|
57
|
+
end
|
58
|
+
|
59
|
+
# image to use {:name => 'arrow.png', :size => '16x16'}
|
60
|
+
attr_accessor :image
|
61
|
+
|
62
|
+
# if the action requires confirmation
|
63
|
+
def confirm=(value)
|
64
|
+
@dhtml_confirm = nil if value
|
65
|
+
@confirm = value
|
66
|
+
end
|
67
|
+
def confirm(label = '')
|
68
|
+
@confirm.is_a?(String) ? @confirm : as_(@confirm, :label => label)
|
69
|
+
end
|
70
|
+
def confirm?
|
71
|
+
!!@confirm
|
72
|
+
end
|
73
|
+
|
74
|
+
# if the action uses a DHTML based (i.e. 2-phase) confirmation
|
75
|
+
attr_accessor :dhtml_confirm
|
76
|
+
def dhtml_confirm=(value)
|
77
|
+
@confirm = nil if value
|
78
|
+
@dhtml_confirm = value
|
79
|
+
end
|
80
|
+
def dhtml_confirm?
|
81
|
+
!!@dhtml_confirm
|
82
|
+
end
|
83
|
+
|
84
|
+
# what method to call on the controller to see if this action_link should be visible
|
85
|
+
# note that this is only the UI part of the security. to prevent URL hax0rz, you also need security on requests (e.g. don't execute update method unless authorized).
|
86
|
+
attr_writer :security_method
|
87
|
+
def security_method
|
88
|
+
@security_method || "#{self.action}_authorized?"
|
89
|
+
end
|
90
|
+
|
91
|
+
def security_method_set?
|
92
|
+
!!@security_method
|
93
|
+
end
|
94
|
+
|
95
|
+
attr_accessor :ignore_method
|
96
|
+
|
97
|
+
# the crud type of the (eventual?) action. different than :method, because this crud action may not be imminent.
|
98
|
+
# this is used to determine record-level authorization (e.g. record.authorized_for?(:crud_type => link.crud_type).
|
99
|
+
# options are :create, :read, :update, and :delete
|
100
|
+
attr_accessor :crud_type
|
101
|
+
|
102
|
+
# an "inline" link is inserted into the existing page
|
103
|
+
# exclusive with popup? and page?
|
104
|
+
def inline=(val)
|
105
|
+
@inline = (val == true)
|
106
|
+
self.popup = self.page = false if @inline
|
107
|
+
end
|
108
|
+
def inline?; @inline end
|
109
|
+
|
110
|
+
# a "popup" link displays in a separate (browser?) window. this will eventually take arguments.
|
111
|
+
# exclusive with inline? and page?
|
112
|
+
def popup=(val)
|
113
|
+
@popup = (val == true)
|
114
|
+
if @popup
|
115
|
+
self.inline = self.page = false
|
116
|
+
|
117
|
+
# the :method parameter doesn't mix with the :popup parameter
|
118
|
+
# when/if we start using DHTML popups, we can bring :method back
|
119
|
+
self.method = nil
|
120
|
+
end
|
121
|
+
end
|
122
|
+
def popup?; @popup end
|
123
|
+
|
124
|
+
# a "page" link displays by reloading the current page
|
125
|
+
# exclusive with inline? and popup?
|
126
|
+
def page=(val)
|
127
|
+
@page = (val == true)
|
128
|
+
if @page
|
129
|
+
self.inline = self.popup = false
|
130
|
+
|
131
|
+
# when :method is defined, ActionView adds an onclick to use a form ...
|
132
|
+
# so it's best to just empty out :method whenever possible.
|
133
|
+
# we only ever need to know @method = :get for things that default to POST.
|
134
|
+
# the only things that default to POST are forms and ajax calls.
|
135
|
+
# when @page = true, we don't use ajax.
|
136
|
+
self.method = nil if method == :get
|
137
|
+
end
|
138
|
+
end
|
139
|
+
def page?; @page end
|
140
|
+
|
141
|
+
# where the result of this action should insert in the display.
|
142
|
+
# for :type => :collection, supported values are:
|
143
|
+
# :top
|
144
|
+
# :bottom
|
145
|
+
# :replace (for updating the entire table)
|
146
|
+
# false (no attempt at positioning)
|
147
|
+
# for :type => :member, supported values are:
|
148
|
+
# :before
|
149
|
+
# :replace
|
150
|
+
# :after
|
151
|
+
# false (no attempt at positioning)
|
152
|
+
attr_writer :position
|
153
|
+
def position
|
154
|
+
return @position unless @position.nil? or @position == true
|
155
|
+
return :replace if self.type == :member
|
156
|
+
return :top if self.type == :collection
|
157
|
+
raise "what should the default position be for #{self.type}?"
|
158
|
+
end
|
159
|
+
|
160
|
+
# what type of link this is. currently supported values are :collection and :member.
|
161
|
+
attr_accessor :type
|
162
|
+
|
163
|
+
# html options for the link
|
164
|
+
attr_accessor :html_options
|
165
|
+
|
166
|
+
# nested action_links are referencing a column
|
167
|
+
attr_accessor :column
|
168
|
+
|
169
|
+
# indicates that this a nested_link
|
170
|
+
def nested_link?
|
171
|
+
@column || (parameters && parameters[:named_scope])
|
172
|
+
end
|
173
|
+
|
174
|
+
# Internal use: generated eid for this action_link
|
175
|
+
attr_accessor :eid
|
176
|
+
|
177
|
+
|
178
|
+
end
|
179
|
+
end
|
@@ -0,0 +1,124 @@
|
|
1
|
+
module ActiveScaffold::DataStructures
|
2
|
+
class NestedInfo
|
3
|
+
def self.get(model, session_storage)
|
4
|
+
if session_storage[:nested].nil?
|
5
|
+
nil
|
6
|
+
else
|
7
|
+
session_info = session_storage[:nested].clone
|
8
|
+
begin
|
9
|
+
debugger
|
10
|
+
session_info[:parent_scaffold] = "#{session_info[:parent_scaffold].to_s.camelize}Controller".constantize
|
11
|
+
session_info[:parent_model] = session_info[:parent_scaffold].active_scaffold_config.model
|
12
|
+
session_info[:association] = session_info[:parent_model].reflect_on_association(session_info[:name])
|
13
|
+
unless session_info[:association].nil?
|
14
|
+
ActiveScaffold::DataStructures::NestedInfoAssociation.new(model, session_info)
|
15
|
+
else
|
16
|
+
ActiveScaffold::DataStructures::NestedInfoScope.new(model, session_info)
|
17
|
+
end
|
18
|
+
rescue ActiveScaffold::ControllerNotFound
|
19
|
+
nil
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
attr_accessor :association, :child_association, :parent_model, :parent_scaffold, :parent_id, :constrained_fields, :scope
|
25
|
+
|
26
|
+
def initialize(model, session_info)
|
27
|
+
@parent_model = session_info[:parent_model]
|
28
|
+
@parent_id = session_info[:parent_id]
|
29
|
+
@parent_scaffold = session_info[:parent_scaffold]
|
30
|
+
end
|
31
|
+
|
32
|
+
def new_instance?
|
33
|
+
result = @new_instance.nil?
|
34
|
+
@new_instance = false
|
35
|
+
result
|
36
|
+
end
|
37
|
+
|
38
|
+
def parent_scope
|
39
|
+
parent_model.find(parent_id)
|
40
|
+
end
|
41
|
+
|
42
|
+
def habtm?
|
43
|
+
false
|
44
|
+
end
|
45
|
+
|
46
|
+
def belongs_to?
|
47
|
+
false
|
48
|
+
end
|
49
|
+
|
50
|
+
def has_one?
|
51
|
+
false
|
52
|
+
end
|
53
|
+
|
54
|
+
def readonly?
|
55
|
+
false
|
56
|
+
end
|
57
|
+
|
58
|
+
def sorted?
|
59
|
+
false
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
class NestedInfoAssociation < NestedInfo
|
64
|
+
def initialize(model, session_info)
|
65
|
+
super(model, session_info)
|
66
|
+
@association = session_info[:association]
|
67
|
+
iterate_model_associations(model)
|
68
|
+
end
|
69
|
+
|
70
|
+
def habtm?
|
71
|
+
association.macro == :has_and_belongs_to_many
|
72
|
+
end
|
73
|
+
|
74
|
+
def belongs_to?
|
75
|
+
association.belongs_to?
|
76
|
+
end
|
77
|
+
|
78
|
+
def has_one?
|
79
|
+
association.macro == :has_one
|
80
|
+
end
|
81
|
+
|
82
|
+
def readonly?
|
83
|
+
if association.options.has_key? :readonly
|
84
|
+
association.options[:readonly]
|
85
|
+
else
|
86
|
+
association.options.has_key? :through
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
def sorted?
|
91
|
+
association.options.has_key? :order
|
92
|
+
end
|
93
|
+
|
94
|
+
def default_sorting
|
95
|
+
association.options[:order]
|
96
|
+
end
|
97
|
+
|
98
|
+
protected
|
99
|
+
|
100
|
+
def iterate_model_associations(model)
|
101
|
+
@constrained_fields = []
|
102
|
+
@constrained_fields << association.foreign_key.to_sym unless association.belongs_to?
|
103
|
+
model.reflect_on_all_associations.each do |current|
|
104
|
+
if !current.belongs_to? && association.foreign_key == current.association_foreign_key
|
105
|
+
constrained_fields << current.name.to_sym
|
106
|
+
@child_association = current if current.klass == @parent_model
|
107
|
+
end
|
108
|
+
if association.foreign_key == current.foreign_key
|
109
|
+
# show columns for has_many and has_one child associationes
|
110
|
+
constrained_fields << current.name.to_sym if current.belongs_to?
|
111
|
+
@child_association = current
|
112
|
+
end
|
113
|
+
end
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
class NestedInfoScope < NestedInfo
|
118
|
+
def initialize(model, session_info)
|
119
|
+
super(model, session_info)
|
120
|
+
@scope = session_info[:name]
|
121
|
+
@constrained_fields = []
|
122
|
+
end
|
123
|
+
end
|
124
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
# wrap the action rendering for ActiveScaffold controllers
|
2
|
+
module ActionController #:nodoc:
|
3
|
+
class Base
|
4
|
+
def render_with_active_scaffold(*args, &block)
|
5
|
+
if self.class.uses_active_scaffold? and params[:adapter] and @rendering_adapter.nil?
|
6
|
+
@rendering_adapter = true # recursion control
|
7
|
+
# if we need an adapter, then we render the actual stuff to a string and insert it into the adapter template
|
8
|
+
opts = args.blank? ? Hash.new : args.first
|
9
|
+
render :partial => params[:adapter][1..-1],
|
10
|
+
:locals => {:payload => render_to_string(opts.merge(:layout => false), &block).html_safe},
|
11
|
+
:use_full_path => true, :layout => false
|
12
|
+
@rendering_adapter = nil # recursion control
|
13
|
+
else
|
14
|
+
render_without_active_scaffold(*args, &block)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
alias_method_chain :render, :active_scaffold
|
18
|
+
|
19
|
+
# Rails 2.x implementation is post-initialization on :active_scaffold method
|
20
|
+
end
|
21
|
+
end
|
22
|
+
|