diff options
| author | hukl <hukl@eight.local> | 2009-01-31 13:04:43 +0100 |
|---|---|---|
| committer | hukl <hukl@eight.local> | 2009-01-31 13:04:43 +0100 |
| commit | be83d467b5b6f92b0a6a175ee365d043f250d631 (patch) | |
| tree | 87209c119576d1ac7ff7dcf26d3e65b706c91fef /vendor/plugins | |
| parent | 89d3dc4a676ee82cc6bad4d9d00535897318f1c3 (diff) | |
added acts_as_list plugin
Diffstat (limited to 'vendor/plugins')
| -rw-r--r-- | vendor/plugins/acts_as_list/README | 23 | ||||
| -rw-r--r-- | vendor/plugins/acts_as_list/init.rb | 3 | ||||
| -rw-r--r-- | vendor/plugins/acts_as_list/lib/active_record/acts/list.rb | 256 | ||||
| -rw-r--r-- | vendor/plugins/acts_as_list/test/list_test.rb | 332 |
4 files changed, 614 insertions, 0 deletions
diff --git a/vendor/plugins/acts_as_list/README b/vendor/plugins/acts_as_list/README new file mode 100644 index 0000000..36ae318 --- /dev/null +++ b/vendor/plugins/acts_as_list/README | |||
| @@ -0,0 +1,23 @@ | |||
| 1 | ActsAsList | ||
| 2 | ========== | ||
| 3 | |||
| 4 | This acts_as extension provides the capabilities for sorting and reordering a number of objects in a list. The class that has this specified needs to have a +position+ column defined as an integer on the mapped database table. | ||
| 5 | |||
| 6 | |||
| 7 | Example | ||
| 8 | ======= | ||
| 9 | |||
| 10 | class TodoList < ActiveRecord::Base | ||
| 11 | has_many :todo_items, :order => "position" | ||
| 12 | end | ||
| 13 | |||
| 14 | class TodoItem < ActiveRecord::Base | ||
| 15 | belongs_to :todo_list | ||
| 16 | acts_as_list :scope => :todo_list | ||
| 17 | end | ||
| 18 | |||
| 19 | todo_list.first.move_to_bottom | ||
| 20 | todo_list.last.move_higher | ||
| 21 | |||
| 22 | |||
| 23 | Copyright (c) 2007 David Heinemeier Hansson, released under the MIT license \ No newline at end of file | ||
diff --git a/vendor/plugins/acts_as_list/init.rb b/vendor/plugins/acts_as_list/init.rb new file mode 100644 index 0000000..eb87e87 --- /dev/null +++ b/vendor/plugins/acts_as_list/init.rb | |||
| @@ -0,0 +1,3 @@ | |||
| 1 | $:.unshift "#{File.dirname(__FILE__)}/lib" | ||
| 2 | require 'active_record/acts/list' | ||
| 3 | ActiveRecord::Base.class_eval { include ActiveRecord::Acts::List } | ||
diff --git a/vendor/plugins/acts_as_list/lib/active_record/acts/list.rb b/vendor/plugins/acts_as_list/lib/active_record/acts/list.rb new file mode 100644 index 0000000..00d8692 --- /dev/null +++ b/vendor/plugins/acts_as_list/lib/active_record/acts/list.rb | |||
| @@ -0,0 +1,256 @@ | |||
| 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 | # Removes the item from the list. | ||
| 123 | def remove_from_list | ||
| 124 | if in_list? | ||
| 125 | decrement_positions_on_lower_items | ||
| 126 | update_attribute position_column, nil | ||
| 127 | end | ||
| 128 | end | ||
| 129 | |||
| 130 | # Increase the position of this item without adjusting the rest of the list. | ||
| 131 | def increment_position | ||
| 132 | return unless in_list? | ||
| 133 | update_attribute position_column, self.send(position_column).to_i + 1 | ||
| 134 | end | ||
| 135 | |||
| 136 | # Decrease the position of this item without adjusting the rest of the list. | ||
| 137 | def decrement_position | ||
| 138 | return unless in_list? | ||
| 139 | update_attribute position_column, self.send(position_column).to_i - 1 | ||
| 140 | end | ||
| 141 | |||
| 142 | # Return +true+ if this object is the first in the list. | ||
| 143 | def first? | ||
| 144 | return false unless in_list? | ||
| 145 | self.send(position_column) == 1 | ||
| 146 | end | ||
| 147 | |||
| 148 | # Return +true+ if this object is the last in the list. | ||
| 149 | def last? | ||
| 150 | return false unless in_list? | ||
| 151 | self.send(position_column) == bottom_position_in_list | ||
| 152 | end | ||
| 153 | |||
| 154 | # Return the next higher item in the list. | ||
| 155 | def higher_item | ||
| 156 | return nil unless in_list? | ||
| 157 | acts_as_list_class.find(:first, :conditions => | ||
| 158 | "#{scope_condition} AND #{position_column} = #{(send(position_column).to_i - 1).to_s}" | ||
| 159 | ) | ||
| 160 | end | ||
| 161 | |||
| 162 | # Return the next lower item in the list. | ||
| 163 | def lower_item | ||
| 164 | return nil unless in_list? | ||
| 165 | acts_as_list_class.find(:first, :conditions => | ||
| 166 | "#{scope_condition} AND #{position_column} = #{(send(position_column).to_i + 1).to_s}" | ||
| 167 | ) | ||
| 168 | end | ||
| 169 | |||
| 170 | # Test if this record is in a list | ||
| 171 | def in_list? | ||
| 172 | !send(position_column).nil? | ||
| 173 | end | ||
| 174 | |||
| 175 | private | ||
| 176 | def add_to_list_top | ||
| 177 | increment_positions_on_all_items | ||
| 178 | end | ||
| 179 | |||
| 180 | def add_to_list_bottom | ||
| 181 | self[position_column] = bottom_position_in_list.to_i + 1 | ||
| 182 | end | ||
| 183 | |||
| 184 | # Overwrite this method to define the scope of the list changes | ||
| 185 | def scope_condition() "1" end | ||
| 186 | |||
| 187 | # Returns the bottom position number in the list. | ||
| 188 | # bottom_position_in_list # => 2 | ||
| 189 | def bottom_position_in_list(except = nil) | ||
| 190 | item = bottom_item(except) | ||
| 191 | item ? item.send(position_column) : 0 | ||
| 192 | end | ||
| 193 | |||
| 194 | # Returns the bottom item | ||
| 195 | def bottom_item(except = nil) | ||
| 196 | conditions = scope_condition | ||
| 197 | conditions = "#{conditions} AND #{self.class.primary_key} != #{except.id}" if except | ||
| 198 | acts_as_list_class.find(:first, :conditions => conditions, :order => "#{position_column} DESC") | ||
| 199 | end | ||
| 200 | |||
| 201 | # Forces item to assume the bottom position in the list. | ||
| 202 | def assume_bottom_position | ||
| 203 | update_attribute(position_column, bottom_position_in_list(self).to_i + 1) | ||
| 204 | end | ||
| 205 | |||
| 206 | # Forces item to assume the top position in the list. | ||
| 207 | def assume_top_position | ||
| 208 | update_attribute(position_column, 1) | ||
| 209 | end | ||
| 210 | |||
| 211 | # This has the effect of moving all the higher items up one. | ||
| 212 | def decrement_positions_on_higher_items(position) | ||
| 213 | acts_as_list_class.update_all( | ||
| 214 | "#{position_column} = (#{position_column} - 1)", "#{scope_condition} AND #{position_column} <= #{position}" | ||
| 215 | ) | ||
| 216 | end | ||
| 217 | |||
| 218 | # This has the effect of moving all the lower items up one. | ||
| 219 | def decrement_positions_on_lower_items | ||
| 220 | return unless in_list? | ||
| 221 | acts_as_list_class.update_all( | ||
| 222 | "#{position_column} = (#{position_column} - 1)", "#{scope_condition} AND #{position_column} > #{send(position_column).to_i}" | ||
| 223 | ) | ||
| 224 | end | ||
| 225 | |||
| 226 | # This has the effect of moving all the higher items down one. | ||
| 227 | def increment_positions_on_higher_items | ||
| 228 | return unless in_list? | ||
| 229 | acts_as_list_class.update_all( | ||
| 230 | "#{position_column} = (#{position_column} + 1)", "#{scope_condition} AND #{position_column} < #{send(position_column).to_i}" | ||
| 231 | ) | ||
| 232 | end | ||
| 233 | |||
| 234 | # This has the effect of moving all the lower items down one. | ||
| 235 | def increment_positions_on_lower_items(position) | ||
| 236 | acts_as_list_class.update_all( | ||
| 237 | "#{position_column} = (#{position_column} + 1)", "#{scope_condition} AND #{position_column} >= #{position}" | ||
| 238 | ) | ||
| 239 | end | ||
| 240 | |||
| 241 | # Increments position (<tt>position_column</tt>) of all items in the list. | ||
| 242 | def increment_positions_on_all_items | ||
| 243 | acts_as_list_class.update_all( | ||
| 244 | "#{position_column} = (#{position_column} + 1)", "#{scope_condition}" | ||
| 245 | ) | ||
| 246 | end | ||
| 247 | |||
| 248 | def insert_at_position(position) | ||
| 249 | remove_from_list | ||
| 250 | increment_positions_on_lower_items(position) | ||
| 251 | self.update_attribute(position_column, position) | ||
| 252 | end | ||
| 253 | end | ||
| 254 | end | ||
| 255 | end | ||
| 256 | end | ||
diff --git a/vendor/plugins/acts_as_list/test/list_test.rb b/vendor/plugins/acts_as_list/test/list_test.rb new file mode 100644 index 0000000..e89cb8e --- /dev/null +++ b/vendor/plugins/acts_as_list/test/list_test.rb | |||
| @@ -0,0 +1,332 @@ | |||
| 1 | require 'test/unit' | ||
| 2 | |||
| 3 | require 'rubygems' | ||
| 4 | gem 'activerecord', '>= 1.15.4.7794' | ||
| 5 | require 'active_record' | ||
| 6 | |||
| 7 | require "#{File.dirname(__FILE__)}/../init" | ||
| 8 | |||
| 9 | ActiveRecord::Base.establish_connection(:adapter => "sqlite3", :dbfile => ":memory:") | ||
| 10 | |||
| 11 | def setup_db | ||
| 12 | ActiveRecord::Schema.define(:version => 1) do | ||
| 13 | create_table :mixins do |t| | ||
| 14 | t.column :pos, :integer | ||
| 15 | t.column :parent_id, :integer | ||
| 16 | t.column :created_at, :datetime | ||
| 17 | t.column :updated_at, :datetime | ||
| 18 | end | ||
| 19 | end | ||
| 20 | end | ||
| 21 | |||
| 22 | def teardown_db | ||
| 23 | ActiveRecord::Base.connection.tables.each do |table| | ||
| 24 | ActiveRecord::Base.connection.drop_table(table) | ||
| 25 | end | ||
| 26 | end | ||
| 27 | |||
| 28 | class Mixin < ActiveRecord::Base | ||
| 29 | end | ||
| 30 | |||
| 31 | class ListMixin < Mixin | ||
| 32 | acts_as_list :column => "pos", :scope => :parent | ||
| 33 | |||
| 34 | def self.table_name() "mixins" end | ||
| 35 | end | ||
| 36 | |||
| 37 | class ListMixinSub1 < ListMixin | ||
| 38 | end | ||
| 39 | |||
| 40 | class ListMixinSub2 < ListMixin | ||
| 41 | end | ||
| 42 | |||
| 43 | class ListWithStringScopeMixin < ActiveRecord::Base | ||
| 44 | acts_as_list :column => "pos", :scope => 'parent_id = #{parent_id}' | ||
| 45 | |||
| 46 | def self.table_name() "mixins" end | ||
| 47 | end | ||
| 48 | |||
| 49 | |||
| 50 | class ListTest < Test::Unit::TestCase | ||
| 51 | |||
| 52 | def setup | ||
| 53 | setup_db | ||
| 54 | (1..4).each { |counter| ListMixin.create! :pos => counter, :parent_id => 5 } | ||
| 55 | end | ||
| 56 | |||
| 57 | def teardown | ||
| 58 | teardown_db | ||
| 59 | end | ||
| 60 | |||
| 61 | def test_reordering | ||
| 62 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 63 | |||
| 64 | ListMixin.find(2).move_lower | ||
| 65 | assert_equal [1, 3, 2, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 66 | |||
| 67 | ListMixin.find(2).move_higher | ||
| 68 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 69 | |||
| 70 | ListMixin.find(1).move_to_bottom | ||
| 71 | assert_equal [2, 3, 4, 1], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 72 | |||
| 73 | ListMixin.find(1).move_to_top | ||
| 74 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 75 | |||
| 76 | ListMixin.find(2).move_to_bottom | ||
| 77 | assert_equal [1, 3, 4, 2], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 78 | |||
| 79 | ListMixin.find(4).move_to_top | ||
| 80 | assert_equal [4, 1, 3, 2], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 81 | end | ||
| 82 | |||
| 83 | def test_move_to_bottom_with_next_to_last_item | ||
| 84 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 85 | ListMixin.find(3).move_to_bottom | ||
| 86 | assert_equal [1, 2, 4, 3], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 87 | end | ||
| 88 | |||
| 89 | def test_next_prev | ||
| 90 | assert_equal ListMixin.find(2), ListMixin.find(1).lower_item | ||
| 91 | assert_nil ListMixin.find(1).higher_item | ||
| 92 | assert_equal ListMixin.find(3), ListMixin.find(4).higher_item | ||
| 93 | assert_nil ListMixin.find(4).lower_item | ||
| 94 | end | ||
| 95 | |||
| 96 | def test_injection | ||
| 97 | item = ListMixin.new(:parent_id => 1) | ||
| 98 | assert_equal "parent_id = 1", item.scope_condition | ||
| 99 | assert_equal "pos", item.position_column | ||
| 100 | end | ||
| 101 | |||
| 102 | def test_insert | ||
| 103 | new = ListMixin.create(:parent_id => 20) | ||
| 104 | assert_equal 1, new.pos | ||
| 105 | assert new.first? | ||
| 106 | assert new.last? | ||
| 107 | |||
| 108 | new = ListMixin.create(:parent_id => 20) | ||
| 109 | assert_equal 2, new.pos | ||
| 110 | assert !new.first? | ||
| 111 | assert new.last? | ||
| 112 | |||
| 113 | new = ListMixin.create(:parent_id => 20) | ||
| 114 | assert_equal 3, new.pos | ||
| 115 | assert !new.first? | ||
| 116 | assert new.last? | ||
| 117 | |||
| 118 | new = ListMixin.create(:parent_id => 0) | ||
| 119 | assert_equal 1, new.pos | ||
| 120 | assert new.first? | ||
| 121 | assert new.last? | ||
| 122 | end | ||
| 123 | |||
| 124 | def test_insert_at | ||
| 125 | new = ListMixin.create(:parent_id => 20) | ||
| 126 | assert_equal 1, new.pos | ||
| 127 | |||
| 128 | new = ListMixin.create(:parent_id => 20) | ||
| 129 | assert_equal 2, new.pos | ||
| 130 | |||
| 131 | new = ListMixin.create(:parent_id => 20) | ||
| 132 | assert_equal 3, new.pos | ||
| 133 | |||
| 134 | new4 = ListMixin.create(:parent_id => 20) | ||
| 135 | assert_equal 4, new4.pos | ||
| 136 | |||
| 137 | new4.insert_at(3) | ||
| 138 | assert_equal 3, new4.pos | ||
| 139 | |||
| 140 | new.reload | ||
| 141 | assert_equal 4, new.pos | ||
| 142 | |||
| 143 | new.insert_at(2) | ||
| 144 | assert_equal 2, new.pos | ||
| 145 | |||
| 146 | new4.reload | ||
| 147 | assert_equal 4, new4.pos | ||
| 148 | |||
| 149 | new5 = ListMixin.create(:parent_id => 20) | ||
| 150 | assert_equal 5, new5.pos | ||
| 151 | |||
| 152 | new5.insert_at(1) | ||
| 153 | assert_equal 1, new5.pos | ||
| 154 | |||
| 155 | new4.reload | ||
| 156 | assert_equal 5, new4.pos | ||
| 157 | end | ||
| 158 | |||
| 159 | def test_delete_middle | ||
| 160 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 161 | |||
| 162 | ListMixin.find(2).destroy | ||
| 163 | |||
| 164 | assert_equal [1, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 165 | |||
| 166 | assert_equal 1, ListMixin.find(1).pos | ||
| 167 | assert_equal 2, ListMixin.find(3).pos | ||
| 168 | assert_equal 3, ListMixin.find(4).pos | ||
| 169 | |||
| 170 | ListMixin.find(1).destroy | ||
| 171 | |||
| 172 | assert_equal [3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 173 | |||
| 174 | assert_equal 1, ListMixin.find(3).pos | ||
| 175 | assert_equal 2, ListMixin.find(4).pos | ||
| 176 | end | ||
| 177 | |||
| 178 | def test_with_string_based_scope | ||
| 179 | new = ListWithStringScopeMixin.create(:parent_id => 500) | ||
| 180 | assert_equal 1, new.pos | ||
| 181 | assert new.first? | ||
| 182 | assert new.last? | ||
| 183 | end | ||
| 184 | |||
| 185 | def test_nil_scope | ||
| 186 | new1, new2, new3 = ListMixin.create, ListMixin.create, ListMixin.create | ||
| 187 | new2.move_higher | ||
| 188 | assert_equal [new2, new1, new3], ListMixin.find(:all, :conditions => 'parent_id IS NULL', :order => 'pos') | ||
| 189 | end | ||
| 190 | |||
| 191 | |||
| 192 | def test_remove_from_list_should_then_fail_in_list? | ||
| 193 | assert_equal true, ListMixin.find(1).in_list? | ||
| 194 | ListMixin.find(1).remove_from_list | ||
| 195 | assert_equal false, ListMixin.find(1).in_list? | ||
| 196 | end | ||
| 197 | |||
| 198 | def test_remove_from_list_should_set_position_to_nil | ||
| 199 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 200 | |||
| 201 | ListMixin.find(2).remove_from_list | ||
| 202 | |||
| 203 | assert_equal [2, 1, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 204 | |||
| 205 | assert_equal 1, ListMixin.find(1).pos | ||
| 206 | assert_equal nil, ListMixin.find(2).pos | ||
| 207 | assert_equal 2, ListMixin.find(3).pos | ||
| 208 | assert_equal 3, ListMixin.find(4).pos | ||
| 209 | end | ||
| 210 | |||
| 211 | def test_remove_before_destroy_does_not_shift_lower_items_twice | ||
| 212 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 213 | |||
| 214 | ListMixin.find(2).remove_from_list | ||
| 215 | ListMixin.find(2).destroy | ||
| 216 | |||
| 217 | assert_equal [1, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5', :order => 'pos').map(&:id) | ||
| 218 | |||
| 219 | assert_equal 1, ListMixin.find(1).pos | ||
| 220 | assert_equal 2, ListMixin.find(3).pos | ||
| 221 | assert_equal 3, ListMixin.find(4).pos | ||
| 222 | end | ||
| 223 | |||
| 224 | end | ||
| 225 | |||
| 226 | class ListSubTest < Test::Unit::TestCase | ||
| 227 | |||
| 228 | def setup | ||
| 229 | setup_db | ||
| 230 | (1..4).each { |i| ((i % 2 == 1) ? ListMixinSub1 : ListMixinSub2).create! :pos => i, :parent_id => 5000 } | ||
| 231 | end | ||
| 232 | |||
| 233 | def teardown | ||
| 234 | teardown_db | ||
| 235 | end | ||
| 236 | |||
| 237 | def test_reordering | ||
| 238 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 239 | |||
| 240 | ListMixin.find(2).move_lower | ||
| 241 | assert_equal [1, 3, 2, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 242 | |||
| 243 | ListMixin.find(2).move_higher | ||
| 244 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 245 | |||
| 246 | ListMixin.find(1).move_to_bottom | ||
| 247 | assert_equal [2, 3, 4, 1], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 248 | |||
| 249 | ListMixin.find(1).move_to_top | ||
| 250 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 251 | |||
| 252 | ListMixin.find(2).move_to_bottom | ||
| 253 | assert_equal [1, 3, 4, 2], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 254 | |||
| 255 | ListMixin.find(4).move_to_top | ||
| 256 | assert_equal [4, 1, 3, 2], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 257 | end | ||
| 258 | |||
| 259 | def test_move_to_bottom_with_next_to_last_item | ||
| 260 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 261 | ListMixin.find(3).move_to_bottom | ||
| 262 | assert_equal [1, 2, 4, 3], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 263 | end | ||
| 264 | |||
| 265 | def test_next_prev | ||
| 266 | assert_equal ListMixin.find(2), ListMixin.find(1).lower_item | ||
| 267 | assert_nil ListMixin.find(1).higher_item | ||
| 268 | assert_equal ListMixin.find(3), ListMixin.find(4).higher_item | ||
| 269 | assert_nil ListMixin.find(4).lower_item | ||
| 270 | end | ||
| 271 | |||
| 272 | def test_injection | ||
| 273 | item = ListMixin.new("parent_id"=>1) | ||
| 274 | assert_equal "parent_id = 1", item.scope_condition | ||
| 275 | assert_equal "pos", item.position_column | ||
| 276 | end | ||
| 277 | |||
| 278 | def test_insert_at | ||
| 279 | new = ListMixin.create("parent_id" => 20) | ||
| 280 | assert_equal 1, new.pos | ||
| 281 | |||
| 282 | new = ListMixinSub1.create("parent_id" => 20) | ||
| 283 | assert_equal 2, new.pos | ||
| 284 | |||
| 285 | new = ListMixinSub2.create("parent_id" => 20) | ||
| 286 | assert_equal 3, new.pos | ||
| 287 | |||
| 288 | new4 = ListMixin.create("parent_id" => 20) | ||
| 289 | assert_equal 4, new4.pos | ||
| 290 | |||
| 291 | new4.insert_at(3) | ||
| 292 | assert_equal 3, new4.pos | ||
| 293 | |||
| 294 | new.reload | ||
| 295 | assert_equal 4, new.pos | ||
| 296 | |||
| 297 | new.insert_at(2) | ||
| 298 | assert_equal 2, new.pos | ||
| 299 | |||
| 300 | new4.reload | ||
| 301 | assert_equal 4, new4.pos | ||
| 302 | |||
| 303 | new5 = ListMixinSub1.create("parent_id" => 20) | ||
| 304 | assert_equal 5, new5.pos | ||
| 305 | |||
| 306 | new5.insert_at(1) | ||
| 307 | assert_equal 1, new5.pos | ||
| 308 | |||
| 309 | new4.reload | ||
| 310 | assert_equal 5, new4.pos | ||
| 311 | end | ||
| 312 | |||
| 313 | def test_delete_middle | ||
| 314 | assert_equal [1, 2, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 315 | |||
| 316 | ListMixin.find(2).destroy | ||
| 317 | |||
| 318 | assert_equal [1, 3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 319 | |||
| 320 | assert_equal 1, ListMixin.find(1).pos | ||
| 321 | assert_equal 2, ListMixin.find(3).pos | ||
| 322 | assert_equal 3, ListMixin.find(4).pos | ||
| 323 | |||
| 324 | ListMixin.find(1).destroy | ||
| 325 | |||
| 326 | assert_equal [3, 4], ListMixin.find(:all, :conditions => 'parent_id = 5000', :order => 'pos').map(&:id) | ||
| 327 | |||
| 328 | assert_equal 1, ListMixin.find(3).pos | ||
| 329 | assert_equal 2, ListMixin.find(4).pos | ||
| 330 | end | ||
| 331 | |||
| 332 | end | ||
