radiant-forum-extension 3.0.0.rc3 → 3.0.0.rc4

Sign up to get free protection for your applications and to get access to all the features.
@@ -1,31 +1,11 @@
1
1
  - include_stylesheet 'admin/forum_dashboard'
2
2
 
3
- #recent_comments.dashboard_module
4
- .header
5
- %h2
6
- = t('forum_extension.pages_with_latest_comments')
7
- .blockcontent
8
- - latest = Page.last_commented(6)
9
- - if latest.any?
10
- - latest.each do |page|
11
- .page
12
- %h3
13
- = link_to("#{icon} #{node_title}", edit_admin_page_url(page), :title => page.url, :class => 'page')
14
- = link_to("→", page.path, :class => 'public')
15
- %p.minor
16
- =t('forum_extension.comment_count_from', :count => page.posts.count)
17
- = link_to topic.replied_by.name, reader_url(topic.replied_by)
18
- = friendly_date(topic.replied_at)
19
- - else
20
- %p.minor
21
- =t('dashboard_extension.no_show')
22
-
23
3
  #recent_topics.dashboard_module
24
4
  .header
25
5
  %h2
26
6
  = t('forum_extension.latest_forum_activity')
27
7
  .blockcontent
28
- - latest = Topic.latest(4)
8
+ - latest = Topic.latest(8)
29
9
  - if latest.any?
30
10
  - latest.each do |topic|
31
11
  - post = topic.posts.last
@@ -40,12 +20,34 @@
40
20
  = t('forum_extension.started_by')
41
21
  = link_to topic.reader.name, reader_url(topic.reader)
42
22
  = friendly_date(topic.created_at)
43
- %p.excerpt
44
- = truncate_words(scrub_html(post.body), 16)
45
23
  %p.minor
46
24
  =t('forum_extension.administer_by_browsing')
47
25
  - else
48
26
  %p.minor
49
27
  =t('dashboard_extension.no_show')
50
-
51
-
28
+
29
+
30
+ #recent_comments.dashboard_module
31
+ .header
32
+ %h2
33
+ = t('forum_extension.pages_with_latest_comments')
34
+ .blockcontent
35
+ - if Radiant.config['forum.allow_page_comments?']
36
+ - latest = Page.last_commented(8)
37
+ - if latest.any?
38
+ - latest.each do |page|
39
+ .page
40
+ %h3
41
+ = link_to("#{icon} #{node_title}", edit_admin_page_url(page), :title => page.url, :class => 'page')
42
+ = link_to("→", page.path, :class => 'public')
43
+ %p.minor
44
+ =t('forum_extension.comment_count_from', :count => page.posts.count)
45
+ = link_to topic.replied_by.name, reader_url(topic.replied_by)
46
+ = friendly_date(topic.replied_at)
47
+ - else
48
+ %p.minor
49
+ =t('dashboard_extension.no_show')
50
+ - else
51
+ %p.minor
52
+ =t('forum_extension.comments_disabled')
53
+
@@ -54,6 +54,7 @@ en:
54
54
  other: "%{count} comments, most recently from "
55
55
  comment_on_page: "a comment added to the page"
56
56
  comments_closed: "Comments closed."
57
+ comments_disabled: "Page comments are disabled"
57
58
  comments_on: "Comments on"
58
59
  comments: "comments"
59
60
  confirm_removal_of_comment: "Are you sure you want to delete this comment from %{author}? There's no way to get it back if you change your mind."
@@ -6,6 +6,10 @@ class ForumExtension < Radiant::Extension
6
6
  description RadiantForumExtension::DESCRIPTION
7
7
  url RadiantForumExtension::URL
8
8
 
9
+ extension_config do |config|
10
+ config.gem 'acts_as_list', :version => "~> 0.1.2"
11
+ end
12
+
9
13
  def activate
10
14
  ActiveRecord::Base.send :include, CommentableModel # provides has_comments class method that is used here by topics and pages but can be called from any model
11
15
  Reader.send :include, ForumReader # has topics and posts
@@ -33,7 +37,7 @@ class ForumExtension < Radiant::Extension
33
37
 
34
38
  if admin.respond_to?(:dashboard)
35
39
  Admin::DashboardController.send :helper, ForumHelper
36
- admin.dashboard.index.add :main, 'forum_dashboard', :after => 'draft_pages'
40
+ admin.dashboard.index.add :main, 'forum_dashboard', :before => 'recent_assets'
37
41
  end
38
42
 
39
43
  tab("Forum") do
@@ -1,20 +1,6 @@
1
1
  module CommentableModel # for inclusion into ActiveRecord::Base
2
2
  def self.included(base)
3
3
  base.extend ClassMethods
4
- base.class_eval do
5
- named_scope :last_commented, lambda { |count| {
6
- :conditions => "replied_at IS NOT NULL",
7
- :order => "replied_at DESC",
8
- :limit => count
9
- }}
10
- named_scope :most_commented, lambda { |count| {
11
- :select => "topics.*, count(posts.id) AS post_count",
12
- :joins => "INNER JOIN posts ON posts.topic_id = topics.id",
13
- :group => "topics.id",
14
- :order => "post_count DESC",
15
- :limit => count
16
- }}
17
- end
18
4
  end
19
5
 
20
6
  module ClassMethods
@@ -30,6 +16,18 @@ module CommentableModel # for inclusion into ActiveRecord::Base
30
16
  extend CommentableModel::CommentableClassMethods
31
17
  include CommentableModel::CommentableInstanceMethods
32
18
  }
19
+ named_scope :last_commented, lambda { |count| {
20
+ :conditions => "replied_at IS NOT NULL",
21
+ :order => "replied_at DESC",
22
+ :limit => count
23
+ }}
24
+ named_scope :most_commented, lambda { |count| {
25
+ :select => "#{self.table_name}.*, count(posts.id) AS post_count",
26
+ :joins => "INNER JOIN posts ON posts.#{self.class.to_s.downcase}_id = #{self.table_name}.id",
27
+ :group => "#{self.table_name}.id",
28
+ :order => "post_count DESC",
29
+ :limit => count
30
+ }}
33
31
  end
34
32
  end
35
33
 
@@ -1,5 +1,5 @@
1
1
  module RadiantForumExtension
2
- VERSION = '3.0.0.rc3'
2
+ VERSION = '3.0.0.rc4'
3
3
  SUMMARY = %q{Forum and Comment Extension for Radiant CMS}
4
4
  DESCRIPTION = %q{Nice clean forums and page comments for inclusion in your radiant site.}
5
5
  URL = "http://radiant.spanner.org/forum"
@@ -0,0 +1,27 @@
1
+ @import admin/grid
2
+
3
+ #dashboard
4
+ #recent_topics
5
+ +grid(4)
6
+ height: 500px
7
+ #recent_comments
8
+ +grid(4)
9
+ height: 500px
10
+
11
+ .blockcontent
12
+ h3
13
+ margin-top: 0.25em
14
+ a.topic
15
+ color: black
16
+ text-decoration: none
17
+ &:hover
18
+ color: #06c
19
+ text-decoration: underline
20
+ span.context
21
+ font-size: 10px
22
+ font-weight: normal
23
+ color: #bbb8ac
24
+ a
25
+ color: #939187
26
+ p.excerpt
27
+ margin-top: 0.25em
@@ -237,7 +237,11 @@ p.add_comment, p.add_reply
237
237
 
238
238
  span.formnote
239
239
  +small
240
-
240
+
241
+ // dashboard module
242
+
243
+
244
+
241
245
  // emoticons in message display are interpolated by our redcloth extension as <img class="emoticon something" />
242
246
 
243
247
  img.emoticon
@@ -13,6 +13,7 @@ Gem::Specification.new do |s|
13
13
  s.description = RadiantForumExtension::DESCRIPTION
14
14
 
15
15
  s.add_dependency 'radiant-reader-extension', "~> 2.0.0.rc3"
16
+ s.add_dependency 'acts_as_list', "~> 0.1.2"
16
17
 
17
18
  ignores = if File.exist?('.gitignore')
18
19
  File.read('.gitignore').split("\n").inject([]) {|a,p| a + Dir[p] }
metadata CHANGED
@@ -1,15 +1,15 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: radiant-forum-extension
3
3
  version: !ruby/object:Gem::Version
4
- hash: 15424115
4
+ hash: 15424125
5
5
  prerelease: 6
6
6
  segments:
7
7
  - 3
8
8
  - 0
9
9
  - 0
10
10
  - rc
11
- - 3
12
- version: 3.0.0.rc3
11
+ - 4
12
+ version: 3.0.0.rc4
13
13
  platform: ruby
14
14
  authors:
15
15
  - William Ross
@@ -37,6 +37,22 @@ dependencies:
37
37
  version: 2.0.0.rc3
38
38
  type: :runtime
39
39
  version_requirements: *id001
40
+ - !ruby/object:Gem::Dependency
41
+ name: acts_as_list
42
+ prerelease: false
43
+ requirement: &id002 !ruby/object:Gem::Requirement
44
+ none: false
45
+ requirements:
46
+ - - ~>
47
+ - !ruby/object:Gem::Version
48
+ hash: 31
49
+ segments:
50
+ - 0
51
+ - 1
52
+ - 2
53
+ version: 0.1.2
54
+ type: :runtime
55
+ version_requirements: *id002
40
56
  description: Nice clean forums and page comments for inclusion in your radiant site.
41
57
  email:
42
58
  - radiant@spanner.org
@@ -216,8 +232,10 @@ files:
216
232
  - public/images/furniture/wait_16_grey.gif
217
233
  - public/javascripts/forum.js
218
234
  - public/stylesheets/sass/admin/forum.sass
235
+ - public/stylesheets/sass/admin/forum_dashboard.sass
219
236
  - public/stylesheets/sass/forum.sass
220
237
  - public/stylesheets/sass/gallery.sass
238
+ - radiant-forum-extension-3.0.0.rc4.gem
221
239
  - radiant-forum-extension.gemspec
222
240
  - Rakefile
223
241
  - README.md
@@ -241,14 +259,10 @@ files:
241
259
  - spec/models/topic_spec.rb
242
260
  - spec/spec.opts
243
261
  - spec/spec_helper.rb
244
- - vendor/plugins/acts_as_list/init.rb
245
- - vendor/plugins/acts_as_list/lib/active_record/acts/list.rb
246
- - vendor/plugins/acts_as_list/README
247
- - vendor/plugins/acts_as_list/test/list_test.rb
248
262
  homepage: http://radiant.spanner.org/forum
249
263
  licenses: []
250
264
 
251
- post_install_message: "\n Add this to your radiant project with:\n\n config.gem 'radiant-forum-extension', :version => '~> 3.0.0.rc3'\n\n "
265
+ post_install_message: "\n Add this to your radiant project with:\n\n config.gem 'radiant-forum-extension', :version => '~> 3.0.0.rc4'\n\n "
252
266
  rdoc_options: []
253
267
 
254
268
  require_paths:
@@ -1,23 +0,0 @@
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
@@ -1,3 +0,0 @@
1
- $:.unshift "#{File.dirname(__FILE__)}/lib"
2
- require 'active_record/acts/list'
3
- ActiveRecord::Base.class_eval { include ActiveRecord::Acts::List }
@@ -1,256 +0,0 @@
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
@@ -1,332 +0,0 @@
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