aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--activerecord/lib/active_record/mixins/list_mixin.rb154
1 files changed, 154 insertions, 0 deletions
diff --git a/activerecord/lib/active_record/mixins/list_mixin.rb b/activerecord/lib/active_record/mixins/list_mixin.rb
new file mode 100644
index 0000000000..9f964f11af
--- /dev/null
+++ b/activerecord/lib/active_record/mixins/list_mixin.rb
@@ -0,0 +1,154 @@
+module ActiveRecord
+ # This mixin provides the capabilities for sorting and reordering a number of objects in list.
+ # The class that has this mixin included needs to have a "position" column defined as an integer on
+ # the mapped database table. Further more, you need to implement the <tt>scope_condition</tt> that
+ # defines how to separate one list from another.
+ #
+ # Todo list example:
+ #
+ # class TodoList < ActiveRecord::Base
+ # has_many :todo_items, :order => "position"
+ # end
+ #
+ # class TodoItem < ActiveRecord::Base
+ # include ListMixin
+ # belongs_to :todo_list
+ #
+ # private
+ # def scope_condition
+ # "todo_list_id = #{todo_list_id}"
+ # end
+ # end
+ #
+ # todo_list.first.move_to_bottom
+ # todo_list.last.move_higher
+ #
+ module ListMixin
+ def self.append_features(base)
+ super
+ base.class_eval do
+ before_destroy :remove_from_list
+ after_create :add_to_list_bottom
+ end
+ end
+
+ # Moving around on the list
+
+ def move_lower
+ return unless lower_item
+ lower_item.decrement_position
+ increment_position
+ end
+
+ def move_higher
+ return unless higher_item
+ higher_item.increment_position
+ decrement_position
+ end
+
+ def move_to_bottom
+ decrement_positions_on_lower_items
+ assume_bottom_position
+ end
+
+ def move_to_top
+ increment_positions_on_higher_items
+ assume_top_position
+ end
+
+
+ # Entering or existing the list
+
+ def add_to_list_top
+ increment_positions_on_all_items
+ end
+
+ def add_to_list_bottom
+ assume_bottom_position
+ end
+
+ def remove_from_list
+ decrement_positions_on_lower_items
+ end
+
+
+ # Changing the position
+
+ def increment_position
+ self.position = position.to_i + 1
+ save
+ end
+
+ def decrement_position
+ self.position = position.to_i - 1
+ save
+ end
+
+
+ # Querying the position
+
+ def first?
+ self.position == 1
+ end
+
+ def last?
+ self.position == bottom_position_in_list
+ end
+
+ private
+ # Overwrite this method to define the scope of the list changes
+ def scope_condition; end
+
+ def higher_item
+ self.class.find_first(
+ "#{scope_condition} AND position = #{(position.to_i - 1).to_s}"
+ )
+ end
+
+ def lower_item
+ self.class.find_first(
+ "#{scope_condition} AND position = #{(position.to_i + 1).to_s}"
+ )
+ end
+
+ def bottom_position_in_list
+ item = bottom_item
+ item ? item.position : 0
+ end
+
+ def bottom_item
+ self.class.find_first(
+ "#{scope_condition} ",
+ "position DESC"
+ )
+ end
+
+ def assume_bottom_position
+ self.position = bottom_position_in_list.to_i + 1
+ save
+ end
+
+ def assume_top_position
+ self.position = 1
+ save
+ end
+
+ def decrement_positions_on_lower_items
+ self.class.update_all(
+ "position = (position - 1)", "#{scope_condition} AND position > #{position.to_i}"
+ )
+ end
+
+ def increment_positions_on_higher_items
+ self.class.update_all(
+ "position = (position + 1)", "#{scope_condition} AND position < #{position.to_i}"
+ )
+ end
+
+ def increment_positions_on_all_items
+ self.class.update_all(
+ "position = (position + 1)", "#{scope_condition}"
+ )
+ end
+ end
+end \ No newline at end of file