annotate lib/redmine/pagination.rb @ 1295:622f24f53b42 redmine-2.3

Update to Redmine SVN revision 11972 on 2.3-stable branch
author Chris Cannam
date Fri, 14 Jun 2013 09:02:21 +0100
parents
children e248c7af89ec
rev   line source
Chris@1295 1 # encoding: utf-8
Chris@1295 2 #
Chris@1295 3 # Redmine - project management software
Chris@1295 4 # Copyright (C) 2006-2013 Jean-Philippe Lang
Chris@1295 5 #
Chris@1295 6 # This program is free software; you can redistribute it and/or
Chris@1295 7 # modify it under the terms of the GNU General Public License
Chris@1295 8 # as published by the Free Software Foundation; either version 2
Chris@1295 9 # of the License, or (at your option) any later version.
Chris@1295 10 #
Chris@1295 11 # This program is distributed in the hope that it will be useful,
Chris@1295 12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
Chris@1295 13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Chris@1295 14 # GNU General Public License for more details.
Chris@1295 15 #
Chris@1295 16 # You should have received a copy of the GNU General Public License
Chris@1295 17 # along with this program; if not, write to the Free Software
Chris@1295 18 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Chris@1295 19
Chris@1295 20 module Redmine
Chris@1295 21 module Pagination
Chris@1295 22 class Paginator
Chris@1295 23 attr_reader :item_count, :per_page, :page, :page_param
Chris@1295 24
Chris@1295 25 def initialize(*args)
Chris@1295 26 if args.first.is_a?(ActionController::Base)
Chris@1295 27 args.shift
Chris@1295 28 ActiveSupport::Deprecation.warn "Paginator no longer takes a controller instance as the first argument. Remove it from #new arguments."
Chris@1295 29 end
Chris@1295 30 item_count, per_page, page, page_param = *args
Chris@1295 31
Chris@1295 32 @item_count = item_count
Chris@1295 33 @per_page = per_page
Chris@1295 34 page = (page || 1).to_i
Chris@1295 35 if page < 1
Chris@1295 36 page = 1
Chris@1295 37 end
Chris@1295 38 @page = page
Chris@1295 39 @page_param = page_param || :page
Chris@1295 40 end
Chris@1295 41
Chris@1295 42 def offset
Chris@1295 43 (page - 1) * per_page
Chris@1295 44 end
Chris@1295 45
Chris@1295 46 def first_page
Chris@1295 47 if item_count > 0
Chris@1295 48 1
Chris@1295 49 end
Chris@1295 50 end
Chris@1295 51
Chris@1295 52 def previous_page
Chris@1295 53 if page > 1
Chris@1295 54 page - 1
Chris@1295 55 end
Chris@1295 56 end
Chris@1295 57
Chris@1295 58 def next_page
Chris@1295 59 if last_item < item_count
Chris@1295 60 page + 1
Chris@1295 61 end
Chris@1295 62 end
Chris@1295 63
Chris@1295 64 def last_page
Chris@1295 65 if item_count > 0
Chris@1295 66 (item_count - 1) / per_page + 1
Chris@1295 67 end
Chris@1295 68 end
Chris@1295 69
Chris@1295 70 def first_item
Chris@1295 71 item_count == 0 ? 0 : (offset + 1)
Chris@1295 72 end
Chris@1295 73
Chris@1295 74 def last_item
Chris@1295 75 l = first_item + per_page - 1
Chris@1295 76 l > item_count ? item_count : l
Chris@1295 77 end
Chris@1295 78
Chris@1295 79 def linked_pages
Chris@1295 80 pages = []
Chris@1295 81 if item_count > 0
Chris@1295 82 pages += [first_page, page, last_page]
Chris@1295 83 pages += ((page-2)..(page+2)).to_a.select {|p| p > first_page && p < last_page}
Chris@1295 84 end
Chris@1295 85 pages = pages.compact.uniq.sort
Chris@1295 86 if pages.size > 1
Chris@1295 87 pages
Chris@1295 88 else
Chris@1295 89 []
Chris@1295 90 end
Chris@1295 91 end
Chris@1295 92
Chris@1295 93 def items_per_page
Chris@1295 94 ActiveSupport::Deprecation.warn "Paginator#items_per_page will be removed. Use #per_page instead."
Chris@1295 95 per_page
Chris@1295 96 end
Chris@1295 97
Chris@1295 98 def current
Chris@1295 99 ActiveSupport::Deprecation.warn "Paginator#current will be removed. Use .offset instead of .current.offset."
Chris@1295 100 self
Chris@1295 101 end
Chris@1295 102 end
Chris@1295 103
Chris@1295 104 # Paginates the given scope or model. Returns a Paginator instance and
Chris@1295 105 # the collection of objects for the current page.
Chris@1295 106 #
Chris@1295 107 # Options:
Chris@1295 108 # :parameter name of the page parameter
Chris@1295 109 #
Chris@1295 110 # Examples:
Chris@1295 111 # @user_pages, @users = paginate User.where(:status => 1)
Chris@1295 112 #
Chris@1295 113 def paginate(scope, options={})
Chris@1295 114 options = options.dup
Chris@1295 115 finder_options = options.extract!(
Chris@1295 116 :conditions,
Chris@1295 117 :order,
Chris@1295 118 :joins,
Chris@1295 119 :include,
Chris@1295 120 :select
Chris@1295 121 )
Chris@1295 122 if scope.is_a?(Symbol) || finder_options.values.compact.any?
Chris@1295 123 return deprecated_paginate(scope, finder_options, options)
Chris@1295 124 end
Chris@1295 125
Chris@1295 126 paginator = paginator(scope.count, options)
Chris@1295 127 collection = scope.limit(paginator.per_page).offset(paginator.offset).to_a
Chris@1295 128
Chris@1295 129 return paginator, collection
Chris@1295 130 end
Chris@1295 131
Chris@1295 132 def deprecated_paginate(arg, finder_options, options={})
Chris@1295 133 ActiveSupport::Deprecation.warn "#paginate with a Symbol and/or find options is depreceted and will be removed. Use a scope instead."
Chris@1295 134 klass = arg.is_a?(Symbol) ? arg.to_s.classify.constantize : arg
Chris@1295 135 scope = klass.scoped(finder_options)
Chris@1295 136 paginate(scope, options)
Chris@1295 137 end
Chris@1295 138
Chris@1295 139 def paginator(item_count, options={})
Chris@1295 140 options.assert_valid_keys :parameter, :per_page
Chris@1295 141
Chris@1295 142 page_param = options[:parameter] || :page
Chris@1295 143 page = (params[page_param] || 1).to_i
Chris@1295 144 per_page = options[:per_page] || per_page_option
Chris@1295 145 Paginator.new(item_count, per_page, page, page_param)
Chris@1295 146 end
Chris@1295 147
Chris@1295 148 module Helper
Chris@1295 149 include Redmine::I18n
Chris@1295 150
Chris@1295 151 # Renders the pagination links for the given paginator.
Chris@1295 152 #
Chris@1295 153 # Options:
Chris@1295 154 # :per_page_links if set to false, the "Per page" links are not rendered
Chris@1295 155 #
Chris@1295 156 def pagination_links_full(*args)
Chris@1295 157 pagination_links_each(*args) do |text, parameters, options|
Chris@1295 158 if block_given?
Chris@1295 159 yield text, parameters, options
Chris@1295 160 else
Chris@1295 161 link_to text, params.merge(parameters), options
Chris@1295 162 end
Chris@1295 163 end
Chris@1295 164 end
Chris@1295 165
Chris@1295 166 # Yields the given block with the text and parameters
Chris@1295 167 # for each pagination link and returns a string that represents the links
Chris@1295 168 def pagination_links_each(paginator, count=nil, options={}, &block)
Chris@1295 169 options.assert_valid_keys :per_page_links
Chris@1295 170
Chris@1295 171 per_page_links = options.delete(:per_page_links)
Chris@1295 172 per_page_links = false if count.nil?
Chris@1295 173 page_param = paginator.page_param
Chris@1295 174
Chris@1295 175 html = ''
Chris@1295 176 if paginator.previous_page
Chris@1295 177 # \xc2\xab(utf-8) = &#171;
Chris@1295 178 text = "\xc2\xab " + l(:label_previous)
Chris@1295 179 html << yield(text, {page_param => paginator.previous_page}, :class => 'previous') + ' '
Chris@1295 180 end
Chris@1295 181
Chris@1295 182 previous = nil
Chris@1295 183 paginator.linked_pages.each do |page|
Chris@1295 184 if previous && previous != page - 1
Chris@1295 185 html << content_tag('span', '...', :class => 'spacer') + ' '
Chris@1295 186 end
Chris@1295 187 if page == paginator.page
Chris@1295 188 html << content_tag('span', page.to_s, :class => 'current page')
Chris@1295 189 else
Chris@1295 190 html << yield(page.to_s, {page_param => page}, :class => 'page')
Chris@1295 191 end
Chris@1295 192 html << ' '
Chris@1295 193 previous = page
Chris@1295 194 end
Chris@1295 195
Chris@1295 196 if paginator.next_page
Chris@1295 197 # \xc2\xbb(utf-8) = &#187;
Chris@1295 198 text = l(:label_next) + " \xc2\xbb"
Chris@1295 199 html << yield(text, {page_param => paginator.next_page}, :class => 'next') + ' '
Chris@1295 200 end
Chris@1295 201
Chris@1295 202 html << content_tag('span', "(#{paginator.first_item}-#{paginator.last_item}/#{paginator.item_count})", :class => 'items') + ' '
Chris@1295 203
Chris@1295 204 if per_page_links != false && links = per_page_links(paginator, &block)
Chris@1295 205 html << content_tag('span', links.to_s, :class => 'per-page')
Chris@1295 206 end
Chris@1295 207
Chris@1295 208 html.html_safe
Chris@1295 209 end
Chris@1295 210
Chris@1295 211 # Renders the "Per page" links.
Chris@1295 212 def per_page_links(paginator, &block)
Chris@1295 213 values = per_page_options(paginator.per_page, paginator.item_count)
Chris@1295 214 if values.any?
Chris@1295 215 links = values.collect do |n|
Chris@1295 216 if n == paginator.per_page
Chris@1295 217 content_tag('span', n.to_s)
Chris@1295 218 else
Chris@1295 219 yield(n, :per_page => n, paginator.page_param => nil)
Chris@1295 220 end
Chris@1295 221 end
Chris@1295 222 l(:label_display_per_page, links.join(', ')).html_safe
Chris@1295 223 end
Chris@1295 224 end
Chris@1295 225
Chris@1295 226 def per_page_options(selected=nil, item_count=nil)
Chris@1295 227 options = Setting.per_page_options_array
Chris@1295 228 if item_count && options.any?
Chris@1295 229 if item_count > options.first
Chris@1295 230 max = options.detect {|value| value >= item_count} || item_count
Chris@1295 231 else
Chris@1295 232 max = item_count
Chris@1295 233 end
Chris@1295 234 options = options.select {|value| value <= max || value == selected}
Chris@1295 235 end
Chris@1295 236 if options.empty? || (options.size == 1 && options.first == selected)
Chris@1295 237 []
Chris@1295 238 else
Chris@1295 239 options
Chris@1295 240 end
Chris@1295 241 end
Chris@1295 242 end
Chris@1295 243 end
Chris@1295 244 end