Mercurial > hg > soundsoftware-site
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 |