summaryrefslogtreecommitdiffstats
path: root/lib/redmine
diff options
context:
space:
mode:
Diffstat (limited to 'lib/redmine')
-rw-r--r--lib/redmine/acts/positioned.rb134
1 files changed, 134 insertions, 0 deletions
diff --git a/lib/redmine/acts/positioned.rb b/lib/redmine/acts/positioned.rb
new file mode 100644
index 000000000..a322f1346
--- /dev/null
+++ b/lib/redmine/acts/positioned.rb
@@ -0,0 +1,134 @@
+# Redmine - project management software
+# Copyright (C) 2006-2016 Jean-Philippe Lang
+#
+# This program is free software; you can redistribute it and/or
+# modify it under the terms of the GNU General Public License
+# as published by the Free Software Foundation; either version 2
+# of the License, or (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+
+module Redmine
+ module Acts
+ module Positioned
+ def self.included(base)
+ base.extend ClassMethods
+ end
+
+ # This extension provides the capabilities for reordering objects in a list.
+ # The class needs to have a +position+ column defined as an integer on the
+ # mapped database table.
+ module ClassMethods
+ # Configuration options are:
+ #
+ # * +scope+ - restricts what is to be considered a list. Must be a symbol
+ # or an array of symbols
+ def acts_as_positioned(options = {})
+ class_attribute :positioned_options
+ self.positioned_options = {:scope => Array(options[:scope])}
+
+ send :include, Redmine::Acts::Positioned::InstanceMethods
+
+ before_save :set_default_position
+ after_save :update_position
+ after_destroy :remove_position
+ end
+ end
+
+ module InstanceMethods
+ def self.included(base)
+ base.extend ClassMethods
+ end
+
+ # Move to the given position
+ # For compatibility with the previous way of sorting items
+ def move_to=(pos)
+ case pos.to_s
+ when 'highest'
+ self.position = 1
+ when 'higher'
+ self.position -= 1 if position > 1
+ when 'lower'
+ self.position += 1
+ when 'lowest'
+ self.position = nil
+ set_default_position
+ end
+ end
+
+ private
+
+ def position_scope
+ build_position_scope {|c| send(c)}
+ end
+
+ def position_scope_was
+ build_position_scope {|c| send("#{c}_was")}
+ end
+
+ def build_position_scope
+ condition_hash = self.class.positioned_options[:scope].inject({}) do |h, column|
+ h[column] = yield(column)
+ h
+ end
+ self.class.where(condition_hash)
+ end
+
+ def set_default_position
+ if position.nil?
+ self.position = position_scope.maximum(:position).to_i + (new_record? ? 1 : 0)
+ end
+ end
+
+ def update_position
+ if !new_record? && position_scope_changed?
+ remove_position
+ insert_position
+ elsif position_changed?
+ if position_was.nil?
+ insert_position
+ else
+ shift_positions
+ end
+ end
+ end
+
+ def insert_position
+ position_scope.where("position >= ? AND id <> ?", position, id).update_all("position = position + 1")
+ end
+
+ def remove_position
+ position_scope_was.where("position >= ? AND id <> ?", position_was, id).update_all("position = position - 1")
+ end
+
+ def position_scope_changed?
+ (changed & self.class.positioned_options[:scope].map(&:to_s)).any?
+ end
+
+ def shift_positions
+ offset = position_was <=> position
+ min, max = [position, position_was].sort
+ r = position_scope.where("id <> ? AND position BETWEEN ? AND ?", id, min, max).update_all("position = position + #{offset}")
+ if r != max - min
+ reset_positions_in_list
+ end
+ end
+
+ def reset_positions_in_list
+ position_scope.reorder(:position, :id).pluck(:id).each_with_index do |record_id, p|
+ self.class.where(:id => record_id).update_all(:position => p+1)
+ end
+ end
+ end
+ end
+ end
+end
+
+ActiveRecord::Base.send :include, Redmine::Acts::Positioned