jcnetdev-will_paginate 2.3.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. data/CHANGELOG +92 -0
  2. data/LICENSE +18 -0
  3. data/README.rdoc +131 -0
  4. data/Rakefile +62 -0
  5. data/examples/apple-circle.gif +0 -0
  6. data/examples/index.haml +69 -0
  7. data/examples/index.html +92 -0
  8. data/examples/pagination.css +90 -0
  9. data/examples/pagination.sass +91 -0
  10. data/init.rb +1 -0
  11. data/lib/will_paginate.rb +86 -0
  12. data/lib/will_paginate/array.rb +16 -0
  13. data/lib/will_paginate/collection.rb +145 -0
  14. data/lib/will_paginate/core_ext.rb +32 -0
  15. data/lib/will_paginate/finder.rb +247 -0
  16. data/lib/will_paginate/named_scope.rb +132 -0
  17. data/lib/will_paginate/named_scope_patch.rb +39 -0
  18. data/lib/will_paginate/version.rb +9 -0
  19. data/lib/will_paginate/view_helpers.rb +373 -0
  20. data/test/boot.rb +21 -0
  21. data/test/collection_test.rb +140 -0
  22. data/test/console +8 -0
  23. data/test/database.yml +22 -0
  24. data/test/finder_test.rb +434 -0
  25. data/test/fixtures/admin.rb +3 -0
  26. data/test/fixtures/developer.rb +13 -0
  27. data/test/fixtures/developers_projects.yml +13 -0
  28. data/test/fixtures/project.rb +15 -0
  29. data/test/fixtures/projects.yml +6 -0
  30. data/test/fixtures/replies.yml +29 -0
  31. data/test/fixtures/reply.rb +7 -0
  32. data/test/fixtures/schema.rb +38 -0
  33. data/test/fixtures/topic.rb +6 -0
  34. data/test/fixtures/topics.yml +30 -0
  35. data/test/fixtures/user.rb +2 -0
  36. data/test/fixtures/users.yml +35 -0
  37. data/test/helper.rb +37 -0
  38. data/test/lib/activerecord_test_case.rb +36 -0
  39. data/test/lib/activerecord_test_connector.rb +69 -0
  40. data/test/lib/load_fixtures.rb +11 -0
  41. data/test/lib/view_test_process.rb +165 -0
  42. data/test/tasks.rake +56 -0
  43. data/test/view_test.rb +355 -0
  44. metadata +140 -0
@@ -0,0 +1,247 @@
1
+ require 'will_paginate/core_ext'
2
+
3
+ module WillPaginate
4
+ # A mixin for ActiveRecord::Base. Provides +per_page+ class method
5
+ # and hooks things up to provide paginating finders.
6
+ #
7
+ # Find out more in WillPaginate::Finder::ClassMethods
8
+ #
9
+ module Finder
10
+ def self.included(base)
11
+ base.extend ClassMethods
12
+ class << base
13
+ alias_method_chain :method_missing, :paginate
14
+ # alias_method_chain :find_every, :paginate
15
+ define_method(:per_page) { 30 } unless respond_to?(:per_page)
16
+ end
17
+ end
18
+
19
+ # = Paginating finders for ActiveRecord models
20
+ #
21
+ # WillPaginate adds +paginate+, +per_page+ and other methods to
22
+ # ActiveRecord::Base class methods and associations. It also hooks into
23
+ # +method_missing+ to intercept pagination calls to dynamic finders such as
24
+ # +paginate_by_user_id+ and translate them to ordinary finders
25
+ # (+find_all_by_user_id+ in this case).
26
+ #
27
+ # In short, paginating finders are equivalent to ActiveRecord finders; the
28
+ # only difference is that we start with "paginate" instead of "find" and
29
+ # that <tt>:page</tt> is required parameter:
30
+ #
31
+ # @posts = Post.paginate :all, :page => params[:page], :order => 'created_at DESC'
32
+ #
33
+ # In paginating finders, "all" is implicit. There is no sense in paginating
34
+ # a single record, right? So, you can drop the <tt>:all</tt> argument:
35
+ #
36
+ # Post.paginate(...) => Post.find :all
37
+ # Post.paginate_all_by_something => Post.find_all_by_something
38
+ # Post.paginate_by_something => Post.find_all_by_something
39
+ #
40
+ # == The importance of the <tt>:order</tt> parameter
41
+ #
42
+ # In ActiveRecord finders, <tt>:order</tt> parameter specifies columns for
43
+ # the <tt>ORDER BY</tt> clause in SQL. It is important to have it, since
44
+ # pagination only makes sense with ordered sets. Without the <tt>ORDER
45
+ # BY</tt> clause, databases aren't required to do consistent ordering when
46
+ # performing <tt>SELECT</tt> queries; this is especially true for
47
+ # PostgreSQL.
48
+ #
49
+ # Therefore, make sure you are doing ordering on a column that makes the
50
+ # most sense in the current context. Make that obvious to the user, also.
51
+ # For perfomance reasons you will also want to add an index to that column.
52
+ module ClassMethods
53
+ # This is the main paginating finder.
54
+ #
55
+ # == Special parameters for paginating finders
56
+ # * <tt>:page</tt> -- REQUIRED, but defaults to 1 if false or nil
57
+ # * <tt>:per_page</tt> -- defaults to <tt>CurrentModel.per_page</tt> (which is 30 if not overridden)
58
+ # * <tt>:total_entries</tt> -- use only if you manually count total entries
59
+ # * <tt>:count</tt> -- additional options that are passed on to +count+
60
+ # * <tt>:finder</tt> -- name of the ActiveRecord finder used (default: "find")
61
+ #
62
+ # All other options (+conditions+, +order+, ...) are forwarded to +find+
63
+ # and +count+ calls.
64
+ def paginate(*args, &block)
65
+ options = args.pop
66
+ page, per_page, total_entries = wp_parse_options(options)
67
+ finder = (options[:finder] || 'find').to_s
68
+
69
+ if finder == 'find'
70
+ # an array of IDs may have been given:
71
+ total_entries ||= (Array === args.first and args.first.size)
72
+ # :all is implicit
73
+ args.unshift(:all) if args.empty?
74
+ end
75
+
76
+ WillPaginate::Collection.create(page, per_page, total_entries) do |pager|
77
+ count_options = options.except :page, :per_page, :total_entries, :finder
78
+ find_options = count_options.except(:count).update(:offset => pager.offset, :limit => pager.per_page)
79
+
80
+ args << find_options
81
+ # @options_from_last_find = nil
82
+ pager.replace send(finder, *args, &block)
83
+
84
+ # magic counting for user convenience:
85
+ pager.total_entries = wp_count(count_options, args, finder) unless pager.total_entries
86
+ end
87
+ end
88
+
89
+ # Iterates through all records by loading one page at a time. This is useful
90
+ # for migrations or any other use case where you don't want to load all the
91
+ # records in memory at once.
92
+ #
93
+ # It uses +paginate+ internally; therefore it accepts all of its options.
94
+ # You can specify a starting page with <tt>:page</tt> (default is 1). Default
95
+ # <tt>:order</tt> is <tt>"id"</tt>, override if necessary.
96
+ #
97
+ # See http://weblog.jamisbuck.org/2007/4/6/faking-cursors-in-activerecord where
98
+ # Jamis Buck describes this and also uses a more efficient way for MySQL.
99
+ def paginated_each(options = {}, &block)
100
+ options = { :order => 'id', :page => 1 }.merge options
101
+ options[:page] = options[:page].to_i
102
+ options[:total_entries] = 0 # skip the individual count queries
103
+ total = 0
104
+
105
+ begin
106
+ collection = paginate(options)
107
+ total += collection.each(&block).size
108
+ options[:page] += 1
109
+ end until collection.size < collection.per_page
110
+
111
+ total
112
+ end
113
+
114
+ # Wraps +find_by_sql+ by simply adding LIMIT and OFFSET to your SQL string
115
+ # based on the params otherwise used by paginating finds: +page+ and
116
+ # +per_page+.
117
+ #
118
+ # Example:
119
+ #
120
+ # @developers = Developer.paginate_by_sql ['select * from developers where salary > ?', 80000],
121
+ # :page => params[:page], :per_page => 3
122
+ #
123
+ # A query for counting rows will automatically be generated if you don't
124
+ # supply <tt>:total_entries</tt>. If you experience problems with this
125
+ # generated SQL, you might want to perform the count manually in your
126
+ # application.
127
+ #
128
+ def paginate_by_sql(sql, options)
129
+ WillPaginate::Collection.create(*wp_parse_options(options)) do |pager|
130
+ query = sanitize_sql(sql)
131
+ original_query = query.dup
132
+ # add limit, offset
133
+ add_limit! query, :offset => pager.offset, :limit => pager.per_page
134
+ # perfom the find
135
+ pager.replace find_by_sql(query)
136
+
137
+ unless pager.total_entries
138
+ count_query = original_query.sub /\bORDER\s+BY\s+[\w`,\s]+$/mi, ''
139
+ count_query = "SELECT COUNT(*) FROM (#{count_query})"
140
+
141
+ unless ['oracle', 'oci'].include?(self.connection.adapter_name.downcase)
142
+ count_query << ' AS count_table'
143
+ end
144
+ # perform the count query
145
+ pager.total_entries = count_by_sql(count_query)
146
+ end
147
+ end
148
+ end
149
+
150
+ def respond_to?(method, include_priv = false) #:nodoc:
151
+ case method.to_sym
152
+ when :paginate, :paginate_by_sql
153
+ true
154
+ else
155
+ super(method.to_s.sub(/^paginate/, 'find'), include_priv)
156
+ end
157
+ end
158
+
159
+ protected
160
+
161
+ def method_missing_with_paginate(method, *args, &block) #:nodoc:
162
+ # did somebody tried to paginate? if not, let them be
163
+ unless method.to_s.index('paginate') == 0
164
+ return method_missing_without_paginate(method, *args, &block)
165
+ end
166
+
167
+ # paginate finders are really just find_* with limit and offset
168
+ finder = method.to_s.sub('paginate', 'find')
169
+ finder.sub!('find', 'find_all') if finder.index('find_by_') == 0
170
+
171
+ options = args.pop
172
+ raise ArgumentError, 'parameter hash expected' unless options.respond_to? :symbolize_keys
173
+ options = options.dup
174
+ options[:finder] = finder
175
+ args << options
176
+
177
+ paginate(*args, &block)
178
+ end
179
+
180
+ # Does the not-so-trivial job of finding out the total number of entries
181
+ # in the database. It relies on the ActiveRecord +count+ method.
182
+ def wp_count(options, args, finder)
183
+ excludees = [:count, :order, :limit, :offset, :readonly]
184
+ unless options[:select] and options[:select] =~ /^\s*DISTINCT\b/i
185
+ excludees << :select # only exclude the select param if it doesn't begin with DISTINCT
186
+ end
187
+
188
+ # count expects (almost) the same options as find
189
+ count_options = options.except *excludees
190
+
191
+ # merge the hash found in :count
192
+ # this allows you to specify :select, :order, or anything else just for the count query
193
+ count_options.update options[:count] if options[:count]
194
+
195
+ # we may be in a model or an association proxy
196
+ klass = (@owner and @reflection) ? @reflection.klass : self
197
+
198
+ # forget about includes if they are irrelevant (Rails 2.1)
199
+ if count_options[:include] and
200
+ klass.private_methods.include?('references_eager_loaded_tables?') and
201
+ !klass.send(:references_eager_loaded_tables?, count_options)
202
+ count_options.delete :include
203
+ end
204
+
205
+ # we may have to scope ...
206
+ counter = Proc.new { count(count_options) }
207
+
208
+ count = if finder.index('find_') == 0 and klass.respond_to?(scoper = finder.sub('find', 'with'))
209
+ # scope_out adds a 'with_finder' method which acts like with_scope, if it's present
210
+ # then execute the count with the scoping provided by the with_finder
211
+ send(scoper, &counter)
212
+ elsif match = /^find_(all_by|by)_([_a-zA-Z]\w*)$/.match(finder)
213
+ # extract conditions from calls like "paginate_by_foo_and_bar"
214
+ attribute_names = extract_attribute_names_from_match(match)
215
+ conditions = construct_attributes_from_arguments(attribute_names, args)
216
+ with_scope(:find => { :conditions => conditions }, &counter)
217
+ else
218
+ counter.call
219
+ end
220
+
221
+ count.respond_to?(:length) ? count.length : count
222
+ end
223
+
224
+ def wp_parse_options(options) #:nodoc:
225
+ raise ArgumentError, 'parameter hash expected' unless options.respond_to? :symbolize_keys
226
+ options = options.symbolize_keys
227
+ raise ArgumentError, ':page parameter required' unless options.key? :page
228
+
229
+ if options[:count] and options[:total_entries]
230
+ raise ArgumentError, ':count and :total_entries are mutually exclusive'
231
+ end
232
+
233
+ page = options[:page] || 1
234
+ per_page = options[:per_page] || self.per_page
235
+ total = options[:total_entries]
236
+ [page, per_page, total]
237
+ end
238
+
239
+ private
240
+
241
+ # def find_every_with_paginate(options)
242
+ # @options_from_last_find = options
243
+ # find_every_without_paginate(options)
244
+ # end
245
+ end
246
+ end
247
+ end
@@ -0,0 +1,132 @@
1
+ ## stolen from: http://dev.rubyonrails.org/browser/trunk/activerecord/lib/active_record/named_scope.rb?rev=9084
2
+
3
+ module WillPaginate
4
+ # This is a feature backported from Rails 2.1 because of its usefullness not only with will_paginate,
5
+ # but in other aspects when managing complex conditions that you want to be reusable.
6
+ module NamedScope
7
+ # All subclasses of ActiveRecord::Base have two named_scopes:
8
+ # * <tt>all</tt>, which is similar to a <tt>find(:all)</tt> query, and
9
+ # * <tt>scoped</tt>, which allows for the creation of anonymous scopes, on the fly:
10
+ #
11
+ # Shirt.scoped(:conditions => {:color => 'red'}).scoped(:include => :washing_instructions)
12
+ #
13
+ # These anonymous scopes tend to be useful when procedurally generating complex queries, where passing
14
+ # intermediate values (scopes) around as first-class objects is convenient.
15
+ def self.included(base)
16
+ base.class_eval do
17
+ extend ClassMethods
18
+ named_scope :all
19
+ named_scope :scoped, lambda { |scope| scope }
20
+ end
21
+ end
22
+
23
+ module ClassMethods
24
+ def scopes #:nodoc:
25
+ read_inheritable_attribute(:scopes) || write_inheritable_attribute(:scopes, {})
26
+ end
27
+
28
+ # Adds a class method for retrieving and querying objects. A scope represents a narrowing of a database query,
29
+ # such as <tt>:conditions => {:color => :red}, :select => 'shirts.*', :include => :washing_instructions</tt>.
30
+ #
31
+ # class Shirt < ActiveRecord::Base
32
+ # named_scope :red, :conditions => {:color => 'red'}
33
+ # named_scope :dry_clean_only, :joins => :washing_instructions, :conditions => ['washing_instructions.dry_clean_only = ?', true]
34
+ # end
35
+ #
36
+ # The above calls to <tt>named_scope</tt> define class methods <tt>Shirt.red</tt> and <tt>Shirt.dry_clean_only</tt>. <tt>Shirt.red</tt>,
37
+ # in effect, represents the query <tt>Shirt.find(:all, :conditions => {:color => 'red'})</tt>.
38
+ #
39
+ # Unlike Shirt.find(...), however, the object returned by <tt>Shirt.red</tt> is not an Array; it resembles the association object
40
+ # constructed by a <tt>has_many</tt> declaration. For instance, you can invoke <tt>Shirt.red.find(:first)</tt>, <tt>Shirt.red.count</tt>,
41
+ # <tt>Shirt.red.find(:all, :conditions => {:size => 'small'})</tt>. Also, just
42
+ # as with the association objects, name scopes acts like an Array, implementing Enumerable; <tt>Shirt.red.each(&block)</tt>,
43
+ # <tt>Shirt.red.first</tt>, and <tt>Shirt.red.inject(memo, &block)</tt> all behave as if Shirt.red really were an Array.
44
+ #
45
+ # These named scopes are composable. For instance, <tt>Shirt.red.dry_clean_only</tt> will produce all shirts that are both red and dry clean only.
46
+ # Nested finds and calculations also work with these compositions: <tt>Shirt.red.dry_clean_only.count</tt> returns the number of garments
47
+ # for which these criteria obtain. Similarly with <tt>Shirt.red.dry_clean_only.average(:thread_count)</tt>.
48
+ #
49
+ # All scopes are available as class methods on the ActiveRecord descendent upon which the scopes were defined. But they are also available to
50
+ # <tt>has_many</tt> associations. If,
51
+ #
52
+ # class Person < ActiveRecord::Base
53
+ # has_many :shirts
54
+ # end
55
+ #
56
+ # then <tt>elton.shirts.red.dry_clean_only</tt> will return all of Elton's red, dry clean
57
+ # only shirts.
58
+ #
59
+ # Named scopes can also be procedural.
60
+ #
61
+ # class Shirt < ActiveRecord::Base
62
+ # named_scope :colored, lambda { |color|
63
+ # { :conditions => { :color => color } }
64
+ # }
65
+ # end
66
+ #
67
+ # In this example, <tt>Shirt.colored('puce')</tt> finds all puce shirts.
68
+ #
69
+ # Named scopes can also have extensions, just as with <tt>has_many</tt> declarations:
70
+ #
71
+ # class Shirt < ActiveRecord::Base
72
+ # named_scope :red, :conditions => {:color => 'red'} do
73
+ # def dom_id
74
+ # 'red_shirts'
75
+ # end
76
+ # end
77
+ # end
78
+ #
79
+ def named_scope(name, options = {}, &block)
80
+ scopes[name] = lambda do |parent_scope, *args|
81
+ Scope.new(parent_scope, case options
82
+ when Hash
83
+ options
84
+ when Proc
85
+ options.call(*args)
86
+ end, &block)
87
+ end
88
+ (class << self; self end).instance_eval do
89
+ define_method name do |*args|
90
+ scopes[name].call(self, *args)
91
+ end
92
+ end
93
+ end
94
+ end
95
+
96
+ class Scope #:nodoc:
97
+ attr_reader :proxy_scope, :proxy_options
98
+ [].methods.each { |m| delegate m, :to => :proxy_found unless m =~ /(^__|^nil\?|^send|class|extend|find|count|sum|average|maximum|minimum|paginate)/ }
99
+ delegate :scopes, :with_scope, :to => :proxy_scope
100
+
101
+ def initialize(proxy_scope, options, &block)
102
+ [options[:extend]].flatten.each { |extension| extend extension } if options[:extend]
103
+ extend Module.new(&block) if block_given?
104
+ @proxy_scope, @proxy_options = proxy_scope, options.except(:extend)
105
+ end
106
+
107
+ def reload
108
+ load_found; self
109
+ end
110
+
111
+ protected
112
+ def proxy_found
113
+ @found || load_found
114
+ end
115
+
116
+ private
117
+ def method_missing(method, *args, &block)
118
+ if scopes.include?(method)
119
+ scopes[method].call(self, *args)
120
+ else
121
+ with_scope :find => proxy_options do
122
+ proxy_scope.send(method, *args, &block)
123
+ end
124
+ end
125
+ end
126
+
127
+ def load_found
128
+ @found = find(:all)
129
+ end
130
+ end
131
+ end
132
+ end
@@ -0,0 +1,39 @@
1
+ ## based on http://dev.rubyonrails.org/changeset/9084
2
+
3
+ ActiveRecord::Associations::AssociationProxy.class_eval do
4
+ protected
5
+ def with_scope(*args, &block)
6
+ @reflection.klass.send :with_scope, *args, &block
7
+ end
8
+ end
9
+
10
+ [ ActiveRecord::Associations::AssociationCollection,
11
+ ActiveRecord::Associations::HasManyThroughAssociation ].each do |klass|
12
+ klass.class_eval do
13
+ protected
14
+ alias :method_missing_without_scopes :method_missing_without_paginate
15
+ def method_missing_without_paginate(method, *args, &block)
16
+ if @reflection.klass.scopes.include?(method)
17
+ @reflection.klass.scopes[method].call(self, *args, &block)
18
+ else
19
+ method_missing_without_scopes(method, *args, &block)
20
+ end
21
+ end
22
+ end
23
+ end
24
+
25
+ # Rails 1.2.6
26
+ ActiveRecord::Associations::HasAndBelongsToManyAssociation.class_eval do
27
+ protected
28
+ def method_missing(method, *args, &block)
29
+ if @target.respond_to?(method) || (!@reflection.klass.respond_to?(method) && Class.respond_to?(method))
30
+ super
31
+ elsif @reflection.klass.scopes.include?(method)
32
+ @reflection.klass.scopes[method].call(self, *args)
33
+ else
34
+ @reflection.klass.with_scope(:find => { :conditions => @finder_sql, :joins => @join_sql, :readonly => false }) do
35
+ @reflection.klass.send(method, *args, &block)
36
+ end
37
+ end
38
+ end
39
+ end if ActiveRecord::Base.respond_to? :find_first
@@ -0,0 +1,9 @@
1
+ module WillPaginate
2
+ module VERSION
3
+ MAJOR = 2
4
+ MINOR = 3
5
+ TINY = 3
6
+
7
+ STRING = [MAJOR, MINOR, TINY].join('.')
8
+ end
9
+ end
@@ -0,0 +1,373 @@
1
+ require 'will_paginate/core_ext'
2
+
3
+ module WillPaginate
4
+ # = Will Paginate view helpers
5
+ #
6
+ # Currently there is only one view helper: +will_paginate+. It renders the
7
+ # pagination links for the given collection. The helper itself is lightweight
8
+ # and serves only as a wrapper around link renderer instantiation; the
9
+ # renderer then does all the hard work of generating the HTML.
10
+ #
11
+ # == Global options for helpers
12
+ #
13
+ # Options for pagination helpers are optional and get their default values from the
14
+ # WillPaginate::ViewHelpers.pagination_options hash. You can write to this hash to
15
+ # override default options on the global level:
16
+ #
17
+ # WillPaginate::ViewHelpers.pagination_options[:prev_label] = 'Previous page'
18
+ #
19
+ # By putting this into your environment.rb you can easily translate link texts to previous
20
+ # and next pages, as well as override some other defaults to your liking.
21
+ module ViewHelpers
22
+ # default options that can be overridden on the global level
23
+ @@pagination_options = {
24
+ :class => 'pagination',
25
+ :prev_label => '&laquo; Previous',
26
+ :next_label => 'Next &raquo;',
27
+ :inner_window => 4, # links around the current page
28
+ :outer_window => 1, # links around beginning and end
29
+ :separator => ' ', # single space is friendly to spiders and non-graphic browsers
30
+ :param_name => :page,
31
+ :params => nil,
32
+ :renderer => 'WillPaginate::LinkRenderer',
33
+ :page_links => true,
34
+ :container => true
35
+ }
36
+ mattr_reader :pagination_options
37
+
38
+ # Renders Digg/Flickr-style pagination for a WillPaginate::Collection
39
+ # object. Nil is returned if there is only one page in total; no point in
40
+ # rendering the pagination in that case...
41
+ #
42
+ # ==== Options
43
+ # * <tt>:class</tt> -- CSS class name for the generated DIV (default: "pagination")
44
+ # * <tt>:prev_label</tt> -- default: "« Previous"
45
+ # * <tt>:next_label</tt> -- default: "Next »"
46
+ # * <tt>:inner_window</tt> -- how many links are shown around the current page (default: 4)
47
+ # * <tt>:outer_window</tt> -- how many links are around the first and the last page (default: 1)
48
+ # * <tt>:separator</tt> -- string separator for page HTML elements (default: single space)
49
+ # * <tt>:param_name</tt> -- parameter name for page number in URLs (default: <tt>:page</tt>)
50
+ # * <tt>:params</tt> -- additional parameters when generating pagination links
51
+ # (eg. <tt>:controller => "foo", :action => nil</tt>)
52
+ # * <tt>:renderer</tt> -- class name, class or instance of a link renderer (default:
53
+ # <tt>WillPaginate::LinkRenderer</tt>)
54
+ # * <tt>:page_links</tt> -- when false, only previous/next links are rendered (default: true)
55
+ # * <tt>:container</tt> -- toggles rendering of the DIV container for pagination links, set to
56
+ # false only when you are rendering your own pagination markup (default: true)
57
+ # * <tt>:id</tt> -- HTML ID for the container (default: nil). Pass +true+ to have the ID
58
+ # automatically generated from the class name of objects in collection: for example, paginating
59
+ # ArticleComment models would yield an ID of "article_comments_pagination".
60
+ #
61
+ # All options beside listed ones are passed as HTML attributes to the container
62
+ # element for pagination links (the DIV). For example:
63
+ #
64
+ # <%= will_paginate @posts, :id => 'wp_posts' %>
65
+ #
66
+ # ... will result in:
67
+ #
68
+ # <div class="pagination" id="wp_posts"> ... </div>
69
+ #
70
+ # ==== Using the helper without arguments
71
+ # If the helper is called without passing in the collection object, it will
72
+ # try to read from the instance variable inferred by the controller name.
73
+ # For example, calling +will_paginate+ while the current controller is
74
+ # PostsController will result in trying to read from the <tt>@posts</tt>
75
+ # variable. Example:
76
+ #
77
+ # <%= will_paginate :id => true %>
78
+ #
79
+ # ... will result in <tt>@post</tt> collection getting paginated:
80
+ #
81
+ # <div class="pagination" id="posts_pagination"> ... </div>
82
+ #
83
+ def will_paginate(collection = nil, options = {})
84
+ options, collection = collection, nil if collection.is_a? Hash
85
+ unless collection or !controller
86
+ collection_name = "@#{controller.controller_name}"
87
+ collection = instance_variable_get(collection_name)
88
+ raise ArgumentError, "The #{collection_name} variable appears to be empty. Did you " +
89
+ "forget to pass the collection object for will_paginate?" unless collection
90
+ end
91
+ # early exit if there is nothing to render
92
+ return nil unless WillPaginate::ViewHelpers.total_pages_for_collection(collection) > 1
93
+
94
+ options = options.symbolize_keys.reverse_merge WillPaginate::ViewHelpers.pagination_options
95
+
96
+ # get the renderer instance
97
+ renderer = case options[:renderer]
98
+ when String
99
+ options[:renderer].to_s.constantize.new
100
+ when Class
101
+ options[:renderer].new
102
+ else
103
+ options[:renderer]
104
+ end
105
+ # render HTML for pagination
106
+ renderer.prepare collection, options, self
107
+ renderer.to_html
108
+ end
109
+
110
+ # Wrapper for rendering pagination links at both top and bottom of a block
111
+ # of content.
112
+ #
113
+ # <% paginated_section @posts do %>
114
+ # <ol id="posts">
115
+ # <% for post in @posts %>
116
+ # <li> ... </li>
117
+ # <% end %>
118
+ # </ol>
119
+ # <% end %>
120
+ #
121
+ # will result in:
122
+ #
123
+ # <div class="pagination"> ... </div>
124
+ # <ol id="posts">
125
+ # ...
126
+ # </ol>
127
+ # <div class="pagination"> ... </div>
128
+ #
129
+ # Arguments are passed to a <tt>will_paginate</tt> call, so the same options
130
+ # apply. Don't use the <tt>:id</tt> option; otherwise you'll finish with two
131
+ # blocks of pagination links sharing the same ID (which is invalid HTML).
132
+ def paginated_section(*args, &block)
133
+ pagination = will_paginate(*args).to_s
134
+ content = pagination + capture(&block) + pagination
135
+ concat content, block.binding
136
+ end
137
+
138
+ # Renders a helpful message with numbers of displayed vs. total entries.
139
+ # You can use this as a blueprint for your own, similar helpers.
140
+ #
141
+ # <%= page_entries_info @posts %>
142
+ # #-> Displaying posts 6 - 10 of 26 in total
143
+ #
144
+ # By default, the message will use the humanized class name of objects
145
+ # in collection: for instance, "project types" for ProjectType models.
146
+ # Override this to your liking with the <tt>:entry_name</tt> parameter:
147
+ #
148
+ # <%= page_entries_info @posts, :entry_name => 'item' %>
149
+ # #-> Displaying items 6 - 10 of 26 in total
150
+ def page_entries_info(collection, options = {})
151
+ entry_name = options[:entry_name] ||
152
+ (collection.empty?? 'entry' : collection.first.class.name.underscore.sub('_', ' '))
153
+
154
+ if collection.total_pages < 2
155
+ case collection.size
156
+ when 0; "No #{entry_name.pluralize} found"
157
+ when 1; "Displaying <b>1</b> #{entry_name}"
158
+ else; "Displaying <b>all #{collection.size}</b> #{entry_name.pluralize}"
159
+ end
160
+ else
161
+ %{Displaying #{entry_name.pluralize} <b>%d&nbsp;-&nbsp;%d</b> of <b>%d</b> in total} % [
162
+ collection.offset + 1,
163
+ collection.offset + collection.length,
164
+ collection.total_entries
165
+ ]
166
+ end
167
+ end
168
+
169
+ def self.total_pages_for_collection(collection) #:nodoc:
170
+ if collection.respond_to?('page_count') and !collection.respond_to?('total_pages')
171
+ WillPaginate::Deprecation.warn <<-MSG
172
+ You are using a paginated collection of class #{collection.class.name}
173
+ which conforms to the old API of WillPaginate::Collection by using
174
+ `page_count`, while the current method name is `total_pages`. Please
175
+ upgrade yours or 3rd-party code that provides the paginated collection.
176
+ MSG
177
+ class << collection
178
+ def total_pages; page_count; end
179
+ end
180
+ end
181
+ collection.total_pages
182
+ end
183
+ end
184
+
185
+ # This class does the heavy lifting of actually building the pagination
186
+ # links. It is used by +will_paginate+ helper internally.
187
+ class LinkRenderer
188
+
189
+ # The gap in page links is represented by:
190
+ #
191
+ # <span class="gap">&hellip;</span>
192
+ attr_accessor :gap_marker
193
+
194
+ def initialize
195
+ @gap_marker = '<span class="gap">&hellip;</span>'
196
+ end
197
+
198
+ # * +collection+ is a WillPaginate::Collection instance or any other object
199
+ # that conforms to that API
200
+ # * +options+ are forwarded from +will_paginate+ view helper
201
+ # * +template+ is the reference to the template being rendered
202
+ def prepare(collection, options, template)
203
+ @collection = collection
204
+ @options = options
205
+ @template = template
206
+
207
+ # reset values in case we're re-using this instance
208
+ @total_pages = @param_name = @url_string = nil
209
+ end
210
+
211
+ # Process it! This method returns the complete HTML string which contains
212
+ # pagination links. Feel free to subclass LinkRenderer and change this
213
+ # method as you see fit.
214
+ def to_html
215
+ links = @options[:page_links] ? windowed_links : []
216
+ # previous/next buttons
217
+ links.unshift page_link_or_span(@collection.previous_page, 'disabled prev_page', @options[:prev_label])
218
+ links.push page_link_or_span(@collection.next_page, 'disabled next_page', @options[:next_label])
219
+
220
+ html = links.join(@options[:separator])
221
+ @options[:container] ? @template.content_tag(:div, html, html_attributes) : html
222
+ end
223
+
224
+ # Returns the subset of +options+ this instance was initialized with that
225
+ # represent HTML attributes for the container element of pagination links.
226
+ def html_attributes
227
+ return @html_attributes if @html_attributes
228
+ @html_attributes = @options.except *(WillPaginate::ViewHelpers.pagination_options.keys - [:class])
229
+ # pagination of Post models will have the ID of "posts_pagination"
230
+ if @options[:container] and @options[:id] === true
231
+ @html_attributes[:id] = @collection.first.class.name.underscore.pluralize + '_pagination'
232
+ end
233
+ @html_attributes
234
+ end
235
+
236
+ protected
237
+
238
+ # Collects link items for visible page numbers.
239
+ def windowed_links
240
+ prev = nil
241
+
242
+ visible_page_numbers.inject [] do |links, n|
243
+ # detect gaps:
244
+ links << gap_marker if prev and n > prev + 1
245
+ links << page_link_or_span(n, 'current')
246
+ prev = n
247
+ links
248
+ end
249
+ end
250
+
251
+ # Calculates visible page numbers using the <tt>:inner_window</tt> and
252
+ # <tt>:outer_window</tt> options.
253
+ def visible_page_numbers
254
+ inner_window, outer_window = @options[:inner_window].to_i, @options[:outer_window].to_i
255
+ window_from = current_page - inner_window
256
+ window_to = current_page + inner_window
257
+
258
+ # adjust lower or upper limit if other is out of bounds
259
+ if window_to > total_pages
260
+ window_from -= window_to - total_pages
261
+ window_to = total_pages
262
+ end
263
+ if window_from < 1
264
+ window_to += 1 - window_from
265
+ window_from = 1
266
+ window_to = total_pages if window_to > total_pages
267
+ end
268
+
269
+ visible = (1..total_pages).to_a
270
+ left_gap = (2 + outer_window)...window_from
271
+ right_gap = (window_to + 1)...(total_pages - outer_window)
272
+ visible -= left_gap.to_a if left_gap.last - left_gap.first > 1
273
+ visible -= right_gap.to_a if right_gap.last - right_gap.first > 1
274
+
275
+ visible
276
+ end
277
+
278
+ def page_link_or_span(page, span_class, text = nil)
279
+ text ||= page.to_s
280
+
281
+ if page and page != current_page
282
+ classnames = span_class && span_class.index(' ') && span_class.split(' ', 2).last
283
+ page_link page, text, :rel => rel_value(page), :class => classnames
284
+ else
285
+ page_span page, text, :class => span_class
286
+ end
287
+ end
288
+
289
+ def page_link(page, text, attributes = {})
290
+ @template.link_to text, url_for(page), attributes
291
+ end
292
+
293
+ def page_span(page, text, attributes = {})
294
+ @template.content_tag :span, text, attributes
295
+ end
296
+
297
+ # Returns URL params for +page_link_or_span+, taking the current GET params
298
+ # and <tt>:params</tt> option into account.
299
+ def url_for(page)
300
+ page_one = page == 1
301
+ unless @url_string and !page_one
302
+ @url_params = {}
303
+ # page links should preserve GET parameters
304
+ stringified_merge @url_params, @template.params if @template.request.get?
305
+ stringified_merge @url_params, @options[:params] if @options[:params]
306
+
307
+ if complex = param_name.index(/[^\w-]/)
308
+ page_param = (defined?(CGIMethods) ? CGIMethods : ActionController::AbstractRequest).
309
+ parse_query_parameters("#{param_name}=#{page}")
310
+
311
+ stringified_merge @url_params, page_param
312
+ else
313
+ @url_params[param_name] = page_one ? 1 : 2
314
+ end
315
+
316
+ url = @template.url_for(@url_params)
317
+ return url if page_one
318
+
319
+ if complex
320
+ @url_string = url.sub(%r!((?:\?|&amp;)#{CGI.escape param_name}=)#{page}!, '\1@')
321
+ return url
322
+ else
323
+ @url_string = url
324
+ @url_params[param_name] = 3
325
+ @template.url_for(@url_params).split(//).each_with_index do |char, i|
326
+ if char == '3' and url[i, 1] == '2'
327
+ @url_string[i] = '@'
328
+ break
329
+ end
330
+ end
331
+ end
332
+ end
333
+ # finally!
334
+ @url_string.sub '@', page.to_s
335
+ end
336
+
337
+ private
338
+
339
+ def rel_value(page)
340
+ case page
341
+ when @collection.previous_page; 'prev' + (page == 1 ? ' start' : '')
342
+ when @collection.next_page; 'next'
343
+ when 1; 'start'
344
+ end
345
+ end
346
+
347
+ def current_page
348
+ @collection.current_page
349
+ end
350
+
351
+ def total_pages
352
+ @total_pages ||= WillPaginate::ViewHelpers.total_pages_for_collection(@collection)
353
+ end
354
+
355
+ def param_name
356
+ @param_name ||= @options[:param_name].to_s
357
+ end
358
+
359
+ # Recursively merge into target hash by using stringified keys from the other one
360
+ def stringified_merge(target, other)
361
+ other.each do |key, value|
362
+ key = key.to_s # this line is what it's all about!
363
+ existing = target[key]
364
+
365
+ if value.is_a?(Hash) and (existing.is_a?(Hash) or existing.nil?)
366
+ stringified_merge(existing || (target[key] = {}), value)
367
+ else
368
+ target[key] = value
369
+ end
370
+ end
371
+ end
372
+ end
373
+ end