shithub: hugo

Download patch

ref: bec22f8981c251f88594689c65ad7b8822fe1b09
parent: 06d704f243ae01b32f53d9eb947379da255a894c
author: bep <[email protected]>
date: Tue Mar 31 15:12:54 EDT 2015

Add pager size argument to paginator methods

Fixes #1013

--- a/docs/content/extras/pagination.md
+++ b/docs/content/extras/pagination.md
@@ -17,7 +17,7 @@
 
 Pagination can be configured in the site configuration (e.g. `config.toml`):
 
-* `Paginate` (default `10`)
+* `Paginate` (default `10`) (this setting can be overridden in the template)
 * `PaginatePath` (default `page`)
 
 Setting `Paginate` to a positive value will split the list pages for the home page, sections and taxonomies into chunks of that size. But note that the generation of the pagination pages for sections, taxonomies and home page is *lazy* --- the pages will not be created if not referenced by a `.Paginator` (see below).
@@ -34,6 +34,12 @@
 2. Select a sub-set of the pages with the available template functions and ordering options, and pass the slice to `.Paginate`, e.g. `{{ range (.Paginate ( first 50 .Data.Pages.ByTitle )).Pages }}`.
 
 For a given **Node**, it's one of the options above. The `.Paginator` is static and cannot change once created.
+
+
+The global page size setting (`Paginate`) can be overridden by providing a positive integer as the last argument. The examples below will give five items per page:
+
+* `{{ range (.Paginator 5).Pages }}`
+* `{{ $paginator := .Paginate (where .Data.Pages "Type" "post") 5 }}`
 
 ## Build the navigation
 
--- a/hugolib/pagination.go
+++ b/hugolib/pagination.go
@@ -16,6 +16,7 @@
 import (
 	"errors"
 	"fmt"
+	"github.com/spf13/cast"
 	"github.com/spf13/hugo/helpers"
 	"github.com/spf13/viper"
 	"html/template"
@@ -54,8 +55,8 @@
 
 // Url is deprecated. Will be removed in 0.15.
 func (p *pager) Url() template.HTML {
-    helpers.Deprecated("Paginator", ".Url", ".URL")
-    return p.URL()
+	helpers.Deprecated("Paginator", ".Url", ".URL")
+	return p.URL()
 }
 
 // Pages returns the elements on this page.
@@ -139,8 +140,14 @@
 
 // Paginator gets this Node's paginator if it's already created.
 // If it's not, one will be created with all pages in Data["Pages"].
-func (n *Node) Paginator() (*pager, error) {
+func (n *Node) Paginator(options ...interface{}) (*pager, error) {
 
+	pagerSize, err := resolvePagerSize(options...)
+
+	if err != nil {
+		return nil, err
+	}
+
 	var initError error
 
 	n.paginatorInit.Do(func() {
@@ -148,7 +155,7 @@
 			return
 		}
 
-		pagers, err := paginatePages(n.Data["Pages"], n.URL)
+		pagers, err := paginatePages(n.Data["Pages"], pagerSize, n.URL)
 
 		if err != nil {
 			initError = err
@@ -170,12 +177,12 @@
 }
 
 // Paginator on Page isn't supported, calling this yields an error.
-func (p *Page) Paginator() (*pager, error) {
+func (p *Page) Paginator(options ...interface{}) (*pager, error) {
 	return nil, errors.New("Paginators not supported for content pages.")
 }
 
 // Paginate on Page isn't supported, calling this yields an error.
-func (p *Page) Paginate(seq interface{}) (*pager, error) {
+func (p *Page) Paginate(seq interface{}, options ...interface{}) (*pager, error) {
 	return nil, errors.New("Paginators not supported for content pages.")
 }
 
@@ -182,8 +189,14 @@
 // Paginate gets this Node's paginator if it's already created.
 // If it's not, one will be created with the qiven sequence.
 // Note that repeated calls will return the same result, even if the sequence is different.
-func (n *Node) Paginate(seq interface{}) (*pager, error) {
+func (n *Node) Paginate(seq interface{}, options ...interface{}) (*pager, error) {
 
+	pagerSize, err := resolvePagerSize(options...)
+
+	if err != nil {
+		return nil, err
+	}
+
 	var initError error
 
 	n.paginatorInit.Do(func() {
@@ -190,7 +203,7 @@
 		if n.paginator != nil {
 			return
 		}
-		pagers, err := paginatePages(seq, n.URL)
+		pagers, err := paginatePages(seq, pagerSize, n.URL)
 
 		if err != nil {
 			initError = err
@@ -211,12 +224,30 @@
 	return n.paginator, nil
 }
 
-func paginatePages(seq interface{}, section string) (pagers, error) {
-	paginateSize := viper.GetInt("paginate")
+func resolvePagerSize(options ...interface{}) (int, error) {
+	if len(options) == 0 {
+		return viper.GetInt("paginate"), nil
+	}
 
-	if paginateSize <= 0 {
+	if len(options) > 1 {
+		return -1, errors.New("too many arguments, 'pager size' is currently the only option")
+	}
+
+	pas, err := cast.ToIntE(options[0])
+
+	if err != nil || pas <= 0 {
+		return -1, errors.New(("'pager size' must be a positive integer"))
+	}
+
+	return pas, nil
+}
+
+func paginatePages(seq interface{}, pagerSize int, section string) (pagers, error) {
+
+	if pagerSize <= 0 {
 		return nil, errors.New("'paginate' configuration setting must be positive to paginate")
 	}
+
 	var pages Pages
 	switch seq.(type) {
 	case Pages:
@@ -232,7 +263,7 @@
 	}
 
 	urlFactory := newPaginationURLFactory(section)
-	paginator, _ := newPaginator(pages, paginateSize, urlFactory)
+	paginator, _ := newPaginator(pages, pagerSize, urlFactory)
 	pagers := paginator.Pagers()
 
 	return pagers, nil
--- a/hugolib/pagination_test.go
+++ b/hugolib/pagination_test.go
@@ -44,6 +44,7 @@
 
 	first := paginatorPages[0]
 	assert.Equal(t, "page/1/", first.URL())
+	assert.Equal(t, first.URL(), first.Url())
 	assert.Equal(t, first, first.First())
 	assert.True(t, first.HasNext())
 	assert.Equal(t, paginatorPages[1], first.Next())
@@ -110,7 +111,18 @@
 }
 
 func TestPaginator(t *testing.T) {
-	viper.Set("paginate", 5)
+	for _, useViper := range []bool{false, true} {
+		doTestPaginator(t, useViper)
+	}
+}
+
+func doTestPaginator(t *testing.T, useViper bool) {
+	pagerSize := 5
+	if useViper {
+		viper.Set("paginate", pagerSize)
+	} else {
+		viper.Set("paginate", -1)
+	}
 	pages := createTestPages(12)
 	s := &Site{}
 	n1 := s.newHomeNode()
@@ -117,8 +129,15 @@
 	n2 := s.newHomeNode()
 	n1.Data["Pages"] = pages
 
-	paginator1, err := n1.Paginator()
+	var paginator1 *pager
+	var err error
 
+	if useViper {
+		paginator1, err = n1.Paginator()
+	} else {
+		paginator1, err = n1.Paginator(pagerSize)
+	}
+
 	assert.Nil(t, err)
 	assert.NotNil(t, paginator1)
 	assert.Equal(t, 3, paginator1.TotalPages())
@@ -146,14 +165,34 @@
 }
 
 func TestPaginate(t *testing.T) {
-	viper.Set("paginate", 5)
+	for _, useViper := range []bool{false, true} {
+		doTestPaginate(t, useViper)
+	}
+}
+
+func doTestPaginate(t *testing.T, useViper bool) {
+
+	pagerSize := 5
+	if useViper {
+		viper.Set("paginate", pagerSize)
+	} else {
+		viper.Set("paginate", -1)
+	}
+
 	pages := createTestPages(6)
 	s := &Site{}
 	n1 := s.newHomeNode()
 	n2 := s.newHomeNode()
 
-	paginator1, err := n1.Paginate(pages)
+	var paginator1 *pager
+	var err error
 
+	if useViper {
+		paginator1, err = n1.Paginate(pages)
+	} else {
+		paginator1, err = n1.Paginate(pages, pagerSize)
+	}
+
 	assert.Nil(t, err)
 	assert.NotNil(t, paginator1)
 	assert.Equal(t, 2, paginator1.TotalPages())
@@ -172,6 +211,17 @@
 	assert.NotNil(t, err)
 }
 
+func TestInvalidOptions(t *testing.T) {
+	s := &Site{}
+	n1 := s.newHomeNode()
+	_, err := n1.Paginate(createTestPages(1), 1, 2)
+	assert.NotNil(t, err)
+	_, err = n1.Paginator(1, 2)
+	assert.NotNil(t, err)
+	_, err = n1.Paginator(-1)
+	assert.NotNil(t, err)
+}
+
 func TestPaginateWithNegativePaginate(t *testing.T) {
 	viper.Set("paginate", -1)
 	s := &Site{}
@@ -180,13 +230,12 @@
 }
 
 func TestPaginatePages(t *testing.T) {
-	viper.Set("paginate", 11)
 	for i, seq := range []interface{}{createTestPages(11), WeightedPages{}, PageGroup{}, &Pages{}} {
-		v, err := paginatePages(seq, "t")
+		v, err := paginatePages(seq, 11, "t")
 		assert.NotNil(t, v, "Val %d", i)
 		assert.Nil(t, err, "Err %d", i)
 	}
-	_, err := paginatePages(Site{}, "t")
+	_, err := paginatePages(Site{}, 11, "t")
 	assert.NotNil(t, err)
 
 }