Module: Merb::PaginationHelper

Defined in:
lib/merb-pagination/pagination_helper.rb

Instance Method Summary collapse

Instance Method Details

#paginate(current_page, page_count, options = {}) ⇒ Object

Given a page count and the current page, we generate a set of pagination links.

  • We use an inner and outer window into a list of links. For a set of

20 pages with the current page being 10: outer_window:

1 2  ..... 19 20

inner_window

5 6 7 8 9 10 11 12 13 14

This is totally adjustable, or can be turned off by giving the :inner_window setting a value of nil.

  • Options

:class => <em>css_class</em

The CSS class to be given to the paginator div. Defaults to ‘paginated’

:prev_label => text_for_previous_link

Defaults to ‘&laquo; Previous ’

:next_labe => text_for_next_link

Defaults to ‘ Next &raquo;’

:left_cut_label => text_for_cut

Used when the page numbers need to be cut off to prevent the set of pagination links from being too long. Defaults to ‘&larr;’

:right_cut_label => text_for_cut

Same as :left_cut_label but for the right side of numbers. Defaults to ‘&rarr;’

:outer_window => number_of_pages

Sets the number of pages to include in the outer ‘window’ Defaults to 2

:inner_window => number_of_pages

Sets the number of pags to include in the inner ‘window’ Defaults to 10

:default_css => use_paginator_provided_css

Use the default CSS provided by the paginator. If you want to do
your own custom styling of the paginator from scratch, set this to 
false.
Defaults to true

:page_param => name_of_page_paramiter

Sets the name of the paramiter the paginator uses to return what
page is being requested.
Defaults to 'page'

:url => url_for_links

Provides the base url to use in the page navigation links.
Defaults to ''


50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
# File 'lib/merb-pagination/pagination_helper.rb', line 50

def paginate(current_page, page_count, options = {})
  options.reverse_merge!({
    :class           => 'paginated',
    :prev_label      => '&laquo; Previous',
    :next_label      => 'Next &raquo;',
    :left_cut_label  => '&larr;',
    :right_cut_label => '&rarr;',
    :outer_window    => 2,
    :inner_window    => 10,
    :default_css     => true,
    :page_param      => 'page',
    :url             => ''
  })
  url = options.delete :url
  url << (url.include?('?') ? '&' : '?') << options[:page_param]
  
  pages = { 
    :all => (1 .. page_count).to_a, 
    :left => [], 
    :center => [], 
    :right => [] 
  }
  
  # Only worry about using our 'windows' if the page count is less then 
  # our windows combined.
  if options[:inner_window].nil? or ((options[:outer_window] *2) + options[:inner_window] + 2) >= page_count
    pages[:center] = pages[:all]
  else
    pages[:left] = pages[:all][0, options[:outer_window]]
    pages[:right] = pages[:all][page_count - options[:outer_window], options[:outer_window]]
    pages[:center] = case current_page
    # allow the inner 'window' to shift to right when close to the left edge
    # Ex: 1 2 [3] 4 5 6 7 8 9 ... 20
    when -infinity .. (options[:inner_window] / 2) +3
      pages[:all][options[:outer_window], options[:inner_window]] + 
        [options[:right_cut_label]]
    # allow the inner 'window' to shift left when close to the right edge
    # Ex: 1 2 ... 12 13 14 15 16 [17] 18 19 20
    when (page_count - (options[:inner_window] / 2.0).ceil) -1 .. infinity
      [options[:left_cut_label]] +
        pages[:all][page_count - options[:inner_window] - options[:outer_window], options[:inner_window]]
    # Display the unshifed window
    # ex: 1 2 ... 5 6 7 [8] 9 10 11 ... 19 20
    else
      [options[:left_cut_label]] + 
        pages[:all][current_page - (options[:inner_window] / 2) -1, options[:inner_window]] +
        [options[:right_cut_label]]
    end
  end
  
  Builder::XmlMarkup.new.div(:class => options[:class]) do |b|
    b.style {|s| s << %Q{
      div.#{options[:class]} ul, div.#{options[:class]} ul li {
        display: inline;
        padding: 2px;
      }  
    } } if options[:default_css]
    # b << (current_page <= 1 ? options[:prev_label] : %Q{<a href="#{url}=#{current_page -1}">#{options[:prev_label]}</a>})
    
    b.ul do
      b.li(:class => current_page <= 1 ? 'previous-off' : 'previous') {
        b << (current_page <= 1 ? options[:prev_label] : %Q{<a href="#{url}=#{current_page -1}">#{options[:prev_label]}</a>})
      }
      [pages[:left], pages[:center], pages[:right]].each do |p|
        p.each do |page_number|
          case page_number
          when String
            b.li(:class=>'more_marker') {|li| li << page_number}
          when current_page
            b.li(page_number, :class=>'current_page')
          else
            b.li { b.a(page_number, :href=>"#{url}=#{page_number}") }
          end
        end
      end
      b.li(:class => current_page >= page_count ? 'next-off' : 'next') {
        b << (current_page >= page_count ? options[:next_label] : %Q{<a href="#{url}=#{current_page +1}">#{options[:next_label]}</a>})
      }
    end
    
    # b << (current_page >= page_count ? options[:next_label] : %Q{<a href="#{url}=#{current_page +1}">#{options[:next_label]}</a>})
  end
end