comparison .svn/pristine/a2/a265852a9b9824ef68e490e52d5427ad89e62e3e.svn-base @ 1517:dffacf8a6908 redmine-2.5

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