Add Page.GetTerms
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 19 Feb 2020 08:16:27 +0000 (09:16 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 19 Feb 2020 13:52:23 +0000 (14:52 +0100)
Fixes #6905

docs/content/en/templates/taxonomy-templates.md
hugolib/content_map_page.go
hugolib/page.go
hugolib/site_benchmark_new_test.go
hugolib/taxonomy_test.go
resources/page/page.go
resources/page/page_nop.go
resources/page/testhelpers_test.go

index b82a5175c8a6f64b7c66307c8f87c4e1236cbe78..bef2d32264a72142d7bf4e0cd395b4ffc319ce22 100644 (file)
@@ -216,6 +216,18 @@ Because we are leveraging the front matter system to define taxonomies for conte
 
 ### Example: List Tags in a Single Page Template
 
+{{< new-in "0.65.0" >}}
+
+```go-html-template
+<ul>
+    {{ range (.GetTerms "tags") }}
+        <li><a href="{{ .Permalink }}">{{ .LinkTitle }}</a></li>
+   {{ end }}
+</ul>
+```
+
+Before Hugo 0.65.0 you needed to do something like this:
+
 ```go-html-template
 {{ $taxo := "tags" }} <!-- Use the plural form here -->
 <ul id="{{ $taxo }}">
index 06d1310ec81f88698a9282065fbeaecd4608505a..5075095eaf0571258e1a68c5e3cef1bea9581051 100644 (file)
@@ -599,7 +599,7 @@ func (m *pageMap) attachPageToViews(s string, b *contentNode) {
 
                        if s == "/" {
                                // To avoid getting an empty key.
-                               s = "home"
+                               s = page.KindHome
                        }
                        key := cleanTreeKey(path.Join(viewName.plural, termKey, s))
                        m.taxonomyEntries.Insert(key, bv)
index 1384d7293c7f387430d3ee43ee52dd1c6bc39493..12129d4ad9f5d1e0febca343af1ed19ba60e7120 100644 (file)
@@ -131,6 +131,34 @@ func (p *pageState) GitInfo() *gitmap.GitInfo {
        return p.gitInfo
 }
 
+// GetTerms gets the terms defined on this page in the given taxonomy.
+func (p *pageState) GetTerms(taxonomy string) page.Pages {
+       taxonomy = strings.ToLower(taxonomy)
+       m := p.s.pageMap
+       prefix := cleanTreeKey(taxonomy)
+
+       var self string
+       if p.IsHome() {
+               // TODO(bep) make this less magical, see taxonomyEntries.Insert.
+               self = "/" + page.KindHome
+       } else {
+               self = p.treeRef.key
+       }
+
+       var pas page.Pages
+
+       m.taxonomies.WalkPrefixListable(prefix, func(s string, n *contentNode) bool {
+               if _, found := m.taxonomyEntries.Get(s + self); found {
+                       pas = append(pas, n.p)
+               }
+               return false
+       })
+
+       page.SortByDefault(pas)
+
+       return pas
+}
+
 func (p *pageState) MarshalJSON() ([]byte, error) {
        return page.MarshalPageToJSON(p)
 }
index 4ffeaa8a17e08368ef12f1d38c3347f381420069..1f16c97e55f18835c6cc632f7a2fa63a91098def 100644 (file)
@@ -378,14 +378,10 @@ contentDir="content/sv"
                {"List terms", func(b testing.TB) *sitesBuilder {
 
                        pageTemplateTemplate := `
-{{ $taxo := "categories" }}
 <ul>
-    {{ range .Param $taxo }}
-        {{ $name := . }}
-        {{ with $.Site.GetPage (printf "/%s/%s" $taxo ($name | urlize)) }}
-            <li><a href="{{ .Permalink }}">{{ $name }}</a></li>
-        {{ end }}
-    {{ end }}
+    {{ range (.GetTerms "categories") }}
+        <li><a href="{{ .Permalink }}">{{ .LinkTitle }}</a></li>
+   {{ end }}
 </ul>
 `
 
index d91e7699f3808f875815c8cd46954132d4fe1132..913773da6a8999124efa9a65e9f775651570a484 100644 (file)
@@ -542,25 +542,41 @@ func TestTaxonomiesPageCollections(t *testing.T) {
        t.Parallel()
 
        b := newTestSitesBuilder(t)
-       b.WithContent("p1.md", `---
+       b.WithContent(
+               "_index.md", `---
+title: "Home Sweet Home"
+categories: [ "dogs", "gorillas"]
+---
+`,
+               "section/_index.md", `---
+title: "Section"
+categories: [ "cats", "dogs", "birds"]
+---
+`,
+               "section/p1.md", `---
 title: "Page1"
 categories: ["funny", "cats"]
 ---
-`, "p2.md", `---
+`, "section/p2.md", `---
 title: "Page2"
 categories: ["funny"]
 ---
 `)
 
        b.WithTemplatesAdded("index.html", `
+{{ $home := site.Home }}
+{{ $section := site.GetPage "section" }}
 {{ $categories := site.GetPage "categories" }}
 {{ $funny := site.GetPage "categories/funny" }}
 {{ $cats := site.GetPage "categories/cats" }}
+{{ $p1 := site.GetPage "section/p1" }}
 
 Categories Pages: {{ range $categories.Pages}}{{.RelPermalink }}|{{ end }}:END
 Funny Pages: {{ range $funny.Pages}}{{.RelPermalink }}|{{ end }}:END
 Cats Pages: {{ range $cats.Pages}}{{.RelPermalink }}|{{ end }}:END
-
+P1 Terms: {{ range $p1.GetTerms "categories" }}{{.RelPermalink }}|{{ end }}:END
+Section Terms: {{ range $section.GetTerms "categories" }}{{.RelPermalink }}|{{ end }}:END
+Home Terms: {{ range $home.GetTerms "categories" }}{{.RelPermalink }}|{{ end }}:END
 `)
 
        b.Build(BuildCfg{})
@@ -575,12 +591,15 @@ Cats Pages: {{ range $cats.Pages}}{{.RelPermalink }}|{{ end }}:END
        b.Assert(funny.Parent(), qt.Equals, cat)
 
        b.AssertFileContent("public/index.html", `
-Categories Pages: /categories/cats/|/categories/funny/|:END
-Funny Pages: /p1/|/p2/|:END
-Cats Pages: /p1/|:END
+ Categories Pages: /categories/birds/|/categories/cats/|/categories/dogs/|/categories/funny/|/categories/gorillas/|:END
+ Funny Pages: /section/p1/|/section/p2/|:END
+ Cats Pages: /section/p1/|/section/|:END
+ P1 Terms: /categories/cats/|/categories/funny/|:END
+ Section Terms: /categories/birds/|/categories/cats/|/categories/dogs/|:END
+ Home Terms: /categories/dogs/|/categories/gorillas/|:END
 `)
 
-       b.AssertFileContent("public/categories/funny/index.xml", `<link>http://example.com/p1/</link>`)
+       b.AssertFileContent("public/categories/funny/index.xml", `<link>http://example.com/section/p1/</link>`)
        b.AssertFileContent("public/categories/index.xml", `<link>http://example.com/categories/funny/</link>`)
 
 }
index c096cb726f04f8dec06a3aa65532dbf20dea0261..1225f43d0e5603cb68a235159e0ae5f228d03ee8 100644 (file)
@@ -252,6 +252,10 @@ type PageWithoutContent interface {
        maps.Scratcher
        RelatedKeywordsProvider
 
+       // GetTerms gets the terms of a given taxonomy,
+       // e.g. GetTerms("categories")
+       GetTerms(taxonomy string) Pages
+
        DeprecatedWarningPageMethods
 }
 
index 16663ab398c2ec57cb30668e22bd3a40614ac789..ccfbf525f6f2830365038c9dcf65da00e8afe0fc 100644 (file)
@@ -174,6 +174,10 @@ func (p *nopPage) GetParam(key string) interface{} {
        return nil
 }
 
+func (p *nopPage) GetTerms(taxonomy string) Pages {
+       return nil
+}
+
 func (p *nopPage) GitInfo() *gitmap.GitInfo {
        return nil
 }
index dd28fa2cbe55caf0d5f60cc3d71d9dffc997b7c6..9c8605dad0a32803ccacc7e9c5ba04043b92935d 100644 (file)
@@ -222,6 +222,10 @@ func (p *testPage) GetParam(key string) interface{} {
        panic("not implemented")
 }
 
+func (p *testPage) GetTerms(taxonomy string) Pages {
+       panic("not implemented")
+}
+
 func (p *testPage) GetRelatedDocsHandler() *RelatedDocsHandler {
        return relatedDocsHandler
 }