Chris@1296
|
1 module ActiveRecord
|
Chris@1296
|
2 module Acts #:nodoc:
|
Chris@1296
|
3 module List #:nodoc:
|
Chris@1296
|
4 def self.included(base)
|
Chris@1296
|
5 base.extend(ClassMethods)
|
Chris@1296
|
6 end
|
Chris@1296
|
7
|
Chris@1296
|
8 # This +acts_as+ extension provides the capabilities for sorting and reordering a number of objects in a list.
|
Chris@1296
|
9 # The class that has this specified needs to have a +position+ column defined as an integer on
|
Chris@1296
|
10 # the mapped database table.
|
Chris@1296
|
11 #
|
Chris@1296
|
12 # Todo list example:
|
Chris@1296
|
13 #
|
Chris@1296
|
14 # class TodoList < ActiveRecord::Base
|
Chris@1296
|
15 # has_many :todo_items, :order => "position"
|
Chris@1296
|
16 # end
|
Chris@1296
|
17 #
|
Chris@1296
|
18 # class TodoItem < ActiveRecord::Base
|
Chris@1296
|
19 # belongs_to :todo_list
|
Chris@1296
|
20 # acts_as_list :scope => :todo_list
|
Chris@1296
|
21 # end
|
Chris@1296
|
22 #
|
Chris@1296
|
23 # todo_list.first.move_to_bottom
|
Chris@1296
|
24 # todo_list.last.move_higher
|
Chris@1296
|
25 module ClassMethods
|
Chris@1296
|
26 # Configuration options are:
|
Chris@1296
|
27 #
|
Chris@1296
|
28 # * +column+ - specifies the column name to use for keeping the position integer (default: +position+)
|
Chris@1296
|
29 # * +scope+ - restricts what is to be considered a list. Given a symbol, it'll attach <tt>_id</tt>
|
Chris@1296
|
30 # (if it hasn't already been added) and use that as the foreign key restriction. It's also possible
|
Chris@1296
|
31 # to give it an entire string that is interpolated if you need a tighter scope than just a foreign key.
|
Chris@1296
|
32 # Example: <tt>acts_as_list :scope => 'todo_list_id = #{todo_list_id} AND completed = 0'</tt>
|
Chris@1296
|
33 def acts_as_list(options = {})
|
Chris@1296
|
34 configuration = { :column => "position", :scope => "1 = 1" }
|
Chris@1296
|
35 configuration.update(options) if options.is_a?(Hash)
|
Chris@1296
|
36
|
Chris@1296
|
37 configuration[:scope] = "#{configuration[:scope]}_id".intern if configuration[:scope].is_a?(Symbol) && configuration[:scope].to_s !~ /_id$/
|
Chris@1296
|
38
|
Chris@1296
|
39 if configuration[:scope].is_a?(Symbol)
|
Chris@1296
|
40 scope_condition_method = %(
|
Chris@1296
|
41 def scope_condition
|
Chris@1296
|
42 if #{configuration[:scope].to_s}.nil?
|
Chris@1296
|
43 "#{configuration[:scope].to_s} IS NULL"
|
Chris@1296
|
44 else
|
Chris@1296
|
45 "#{configuration[:scope].to_s} = \#{#{configuration[:scope].to_s}}"
|
Chris@1296
|
46 end
|
Chris@1296
|
47 end
|
Chris@1296
|
48 )
|
Chris@1296
|
49 else
|
Chris@1296
|
50 scope_condition_method = "def scope_condition() \"#{configuration[:scope]}\" end"
|
Chris@1296
|
51 end
|
Chris@1296
|
52
|
Chris@1296
|
53 class_eval <<-EOV
|
Chris@1296
|
54 include ActiveRecord::Acts::List::InstanceMethods
|
Chris@1296
|
55
|
Chris@1296
|
56 def acts_as_list_class
|
Chris@1296
|
57 ::#{self.name}
|
Chris@1296
|
58 end
|
Chris@1296
|
59
|
Chris@1296
|
60 def position_column
|
Chris@1296
|
61 '#{configuration[:column]}'
|
Chris@1296
|
62 end
|
Chris@1296
|
63
|
Chris@1296
|
64 #{scope_condition_method}
|
Chris@1296
|
65
|
Chris@1296
|
66 before_destroy :remove_from_list
|
Chris@1296
|
67 before_create :add_to_list_bottom
|
Chris@1296
|
68 EOV
|
Chris@1296
|
69 end
|
Chris@1296
|
70 end
|
Chris@1296
|
71
|
Chris@1296
|
72 # All the methods available to a record that has had <tt>acts_as_list</tt> specified. Each method works
|
Chris@1296
|
73 # by assuming the object to be the item in the list, so <tt>chapter.move_lower</tt> would move that chapter
|
Chris@1296
|
74 # lower in the list of all chapters. Likewise, <tt>chapter.first?</tt> would return +true+ if that chapter is
|
Chris@1296
|
75 # the first in the list of all chapters.
|
Chris@1296
|
76 module InstanceMethods
|
Chris@1296
|
77 # Insert the item at the given position (defaults to the top position of 1).
|
Chris@1296
|
78 def insert_at(position = 1)
|
Chris@1296
|
79 insert_at_position(position)
|
Chris@1296
|
80 end
|
Chris@1296
|
81
|
Chris@1296
|
82 # Swap positions with the next lower item, if one exists.
|
Chris@1296
|
83 def move_lower
|
Chris@1296
|
84 return unless lower_item
|
Chris@1296
|
85
|
Chris@1296
|
86 acts_as_list_class.transaction do
|
Chris@1296
|
87 lower_item.decrement_position
|
Chris@1296
|
88 increment_position
|
Chris@1296
|
89 end
|
Chris@1296
|
90 end
|
Chris@1296
|
91
|
Chris@1296
|
92 # Swap positions with the next higher item, if one exists.
|
Chris@1296
|
93 def move_higher
|
Chris@1296
|
94 return unless higher_item
|
Chris@1296
|
95
|
Chris@1296
|
96 acts_as_list_class.transaction do
|
Chris@1296
|
97 higher_item.increment_position
|
Chris@1296
|
98 decrement_position
|
Chris@1296
|
99 end
|
Chris@1296
|
100 end
|
Chris@1296
|
101
|
Chris@1296
|
102 # Move to the bottom of the list. If the item is already in the list, the items below it have their
|
Chris@1296
|
103 # position adjusted accordingly.
|
Chris@1296
|
104 def move_to_bottom
|
Chris@1296
|
105 return unless in_list?
|
Chris@1296
|
106 acts_as_list_class.transaction do
|
Chris@1296
|
107 decrement_positions_on_lower_items
|
Chris@1296
|
108 assume_bottom_position
|
Chris@1296
|
109 end
|
Chris@1296
|
110 end
|
Chris@1296
|
111
|
Chris@1296
|
112 # Move to the top of the list. If the item is already in the list, the items above it have their
|
Chris@1296
|
113 # position adjusted accordingly.
|
Chris@1296
|
114 def move_to_top
|
Chris@1296
|
115 return unless in_list?
|
Chris@1296
|
116 acts_as_list_class.transaction do
|
Chris@1296
|
117 increment_positions_on_higher_items
|
Chris@1296
|
118 assume_top_position
|
Chris@1296
|
119 end
|
Chris@1296
|
120 end
|
Chris@1296
|
121
|
Chris@1296
|
122 # Move to the given position
|
Chris@1296
|
123 def move_to=(pos)
|
Chris@1296
|
124 case pos.to_s
|
Chris@1296
|
125 when 'highest'
|
Chris@1296
|
126 move_to_top
|
Chris@1296
|
127 when 'higher'
|
Chris@1296
|
128 move_higher
|
Chris@1296
|
129 when 'lower'
|
Chris@1296
|
130 move_lower
|
Chris@1296
|
131 when 'lowest'
|
Chris@1296
|
132 move_to_bottom
|
Chris@1296
|
133 end
|
Chris@1296
|
134 reset_positions_in_list
|
Chris@1296
|
135 end
|
Chris@1296
|
136
|
Chris@1296
|
137 def reset_positions_in_list
|
Chris@1296
|
138 acts_as_list_class.where(scope_condition).reorder("#{position_column} ASC, id ASC").each_with_index do |item, i|
|
Chris@1296
|
139 unless item.send(position_column) == (i + 1)
|
Chris@1296
|
140 acts_as_list_class.update_all({position_column => (i + 1)}, {:id => item.id})
|
Chris@1296
|
141 end
|
Chris@1296
|
142 end
|
Chris@1296
|
143 end
|
Chris@1296
|
144
|
Chris@1296
|
145 # Removes the item from the list.
|
Chris@1296
|
146 def remove_from_list
|
Chris@1296
|
147 if in_list?
|
Chris@1296
|
148 decrement_positions_on_lower_items
|
Chris@1296
|
149 update_attribute position_column, nil
|
Chris@1296
|
150 end
|
Chris@1296
|
151 end
|
Chris@1296
|
152
|
Chris@1296
|
153 # Increase the position of this item without adjusting the rest of the list.
|
Chris@1296
|
154 def increment_position
|
Chris@1296
|
155 return unless in_list?
|
Chris@1296
|
156 update_attribute position_column, self.send(position_column).to_i + 1
|
Chris@1296
|
157 end
|
Chris@1296
|
158
|
Chris@1296
|
159 # Decrease the position of this item without adjusting the rest of the list.
|
Chris@1296
|
160 def decrement_position
|
Chris@1296
|
161 return unless in_list?
|
Chris@1296
|
162 update_attribute position_column, self.send(position_column).to_i - 1
|
Chris@1296
|
163 end
|
Chris@1296
|
164
|
Chris@1296
|
165 # Return +true+ if this object is the first in the list.
|
Chris@1296
|
166 def first?
|
Chris@1296
|
167 return false unless in_list?
|
Chris@1296
|
168 self.send(position_column) == 1
|
Chris@1296
|
169 end
|
Chris@1296
|
170
|
Chris@1296
|
171 # Return +true+ if this object is the last in the list.
|
Chris@1296
|
172 def last?
|
Chris@1296
|
173 return false unless in_list?
|
Chris@1296
|
174 self.send(position_column) == bottom_position_in_list
|
Chris@1296
|
175 end
|
Chris@1296
|
176
|
Chris@1296
|
177 # Return the next higher item in the list.
|
Chris@1296
|
178 def higher_item
|
Chris@1296
|
179 return nil unless in_list?
|
Chris@1296
|
180 acts_as_list_class.find(:first, :conditions =>
|
Chris@1296
|
181 "#{scope_condition} AND #{position_column} = #{(send(position_column).to_i - 1).to_s}"
|
Chris@1296
|
182 )
|
Chris@1296
|
183 end
|
Chris@1296
|
184
|
Chris@1296
|
185 # Return the next lower item in the list.
|
Chris@1296
|
186 def lower_item
|
Chris@1296
|
187 return nil unless in_list?
|
Chris@1296
|
188 acts_as_list_class.find(:first, :conditions =>
|
Chris@1296
|
189 "#{scope_condition} AND #{position_column} = #{(send(position_column).to_i + 1).to_s}"
|
Chris@1296
|
190 )
|
Chris@1296
|
191 end
|
Chris@1296
|
192
|
Chris@1296
|
193 # Test if this record is in a list
|
Chris@1296
|
194 def in_list?
|
Chris@1296
|
195 !send(position_column).nil?
|
Chris@1296
|
196 end
|
Chris@1296
|
197
|
Chris@1296
|
198 private
|
Chris@1296
|
199 def add_to_list_top
|
Chris@1296
|
200 increment_positions_on_all_items
|
Chris@1296
|
201 end
|
Chris@1296
|
202
|
Chris@1296
|
203 def add_to_list_bottom
|
Chris@1296
|
204 self[position_column] = bottom_position_in_list.to_i + 1
|
Chris@1296
|
205 end
|
Chris@1296
|
206
|
Chris@1296
|
207 # Overwrite this method to define the scope of the list changes
|
Chris@1296
|
208 def scope_condition() "1" end
|
Chris@1296
|
209
|
Chris@1296
|
210 # Returns the bottom position number in the list.
|
Chris@1296
|
211 # bottom_position_in_list # => 2
|
Chris@1296
|
212 def bottom_position_in_list(except = nil)
|
Chris@1296
|
213 item = bottom_item(except)
|
Chris@1296
|
214 item ? item.send(position_column) : 0
|
Chris@1296
|
215 end
|
Chris@1296
|
216
|
Chris@1296
|
217 # Returns the bottom item
|
Chris@1296
|
218 def bottom_item(except = nil)
|
Chris@1296
|
219 conditions = scope_condition
|
Chris@1296
|
220 conditions = "#{conditions} AND #{self.class.primary_key} != #{except.id}" if except
|
Chris@1296
|
221 acts_as_list_class.where(conditions).reorder("#{position_column} DESC").first
|
Chris@1296
|
222 end
|
Chris@1296
|
223
|
Chris@1296
|
224 # Forces item to assume the bottom position in the list.
|
Chris@1296
|
225 def assume_bottom_position
|
Chris@1296
|
226 update_attribute(position_column, bottom_position_in_list(self).to_i + 1)
|
Chris@1296
|
227 end
|
Chris@1296
|
228
|
Chris@1296
|
229 # Forces item to assume the top position in the list.
|
Chris@1296
|
230 def assume_top_position
|
Chris@1296
|
231 update_attribute(position_column, 1)
|
Chris@1296
|
232 end
|
Chris@1296
|
233
|
Chris@1296
|
234 # This has the effect of moving all the higher items up one.
|
Chris@1296
|
235 def decrement_positions_on_higher_items(position)
|
Chris@1296
|
236 acts_as_list_class.update_all(
|
Chris@1296
|
237 "#{position_column} = (#{position_column} - 1)", "#{scope_condition} AND #{position_column} <= #{position}"
|
Chris@1296
|
238 )
|
Chris@1296
|
239 end
|
Chris@1296
|
240
|
Chris@1296
|
241 # This has the effect of moving all the lower items up one.
|
Chris@1296
|
242 def decrement_positions_on_lower_items
|
Chris@1296
|
243 return unless in_list?
|
Chris@1296
|
244 acts_as_list_class.update_all(
|
Chris@1296
|
245 "#{position_column} = (#{position_column} - 1)", "#{scope_condition} AND #{position_column} > #{send(position_column).to_i}"
|
Chris@1296
|
246 )
|
Chris@1296
|
247 end
|
Chris@1296
|
248
|
Chris@1296
|
249 # This has the effect of moving all the higher items down one.
|
Chris@1296
|
250 def increment_positions_on_higher_items
|
Chris@1296
|
251 return unless in_list?
|
Chris@1296
|
252 acts_as_list_class.update_all(
|
Chris@1296
|
253 "#{position_column} = (#{position_column} + 1)", "#{scope_condition} AND #{position_column} < #{send(position_column).to_i}"
|
Chris@1296
|
254 )
|
Chris@1296
|
255 end
|
Chris@1296
|
256
|
Chris@1296
|
257 # This has the effect of moving all the lower items down one.
|
Chris@1296
|
258 def increment_positions_on_lower_items(position)
|
Chris@1296
|
259 acts_as_list_class.update_all(
|
Chris@1296
|
260 "#{position_column} = (#{position_column} + 1)", "#{scope_condition} AND #{position_column} >= #{position}"
|
Chris@1296
|
261 )
|
Chris@1296
|
262 end
|
Chris@1296
|
263
|
Chris@1296
|
264 # Increments position (<tt>position_column</tt>) of all items in the list.
|
Chris@1296
|
265 def increment_positions_on_all_items
|
Chris@1296
|
266 acts_as_list_class.update_all(
|
Chris@1296
|
267 "#{position_column} = (#{position_column} + 1)", "#{scope_condition}"
|
Chris@1296
|
268 )
|
Chris@1296
|
269 end
|
Chris@1296
|
270
|
Chris@1296
|
271 def insert_at_position(position)
|
Chris@1296
|
272 remove_from_list
|
Chris@1296
|
273 increment_positions_on_lower_items(position)
|
Chris@1296
|
274 self.update_attribute(position_column, position)
|
Chris@1296
|
275 end
|
Chris@1296
|
276 end
|
Chris@1296
|
277 end
|
Chris@1296
|
278 end
|
Chris@1296
|
279 end
|