b.WithTemplates("index.html", `
{{ $p := site.GetPage "p1.md" }}
+{{ $p2 := site.GetPage "p2.md" }}
P1: {{ $p.Content }}
+P2: {{ $p2.Content }}
`, "index.xml", `
`,
"_default/_markup/render-link.html", `html-link: {{ .Destination | safeURL }}|`,
"_default/_markup/render-link.rss.xml", `xml-link: {{ .Destination | safeURL }}|`,
+ "_default/_markup/render-heading.html", `html-heading: {{ .Text }}|`,
+ "_default/_markup/render-heading.rss.xml", `xml-heading: {{ .Text }}|`,
)
b.WithContent("p1.md", `---
---
P1. [I'm an inline-style link](https://www.gohugo.io)
+# Heading in p1
`, "p2.md", `---
title: "p2"
---
P1. [I'm an inline-style link](https://www.bep.is)
+# Heading in p2
`,
"p3.md", `---
b.Build(BuildCfg{})
- b.AssertFileContent("public/index.html", "P1: <p>P1. html-link: https://www.gohugo.io|</p>")
+ b.AssertFileContent("public/index.html", `
+P1: <p>P1. html-link: https://www.gohugo.io|</p>
+html-heading: Heading in p1|
+html-heading: Heading in p2|
+`)
b.AssertFileContent("public/index.xml", `
P2: <p>P1. xml-link: https://www.bep.is|</p>
P3: <p>P3. xml-link: https://www.example.org|</p>
+xml-heading: Heading in p2|
`)
}
return nil
}
-func (p *pageState) createRenderHooks(f output.Format) (*hooks.Renderers, error) {
+func (p *pageState) createRenderHooks(f output.Format) (hooks.Renderers, error) {
layoutDescriptor := p.getLayoutDescriptor()
layoutDescriptor.RenderingHook = true
layoutDescriptor.LayoutOverride = false
layoutDescriptor.Kind = "render-link"
templ, templFound, err := p.s.Tmpl().LookupLayout(layoutDescriptor, f)
if err != nil {
- return nil, err
+ return renderers, err
}
if templFound {
renderers.LinkRenderer = hookRenderer{
layoutDescriptor.Kind = "render-image"
templ, templFound, err = p.s.Tmpl().LookupLayout(layoutDescriptor, f)
if err != nil {
- return nil, err
+ return renderers, err
}
if templFound {
renderers.ImageRenderer = hookRenderer{
layoutDescriptor.Kind = "render-heading"
templ, templFound, err = p.s.Tmpl().LookupLayout(layoutDescriptor, f)
if err != nil {
- return nil, err
+ return renderers, err
}
if templFound {
renderers.HeadingRenderer = hookRenderer{
}
}
- return &renderers, nil
+ return renderers, nil
}
func (p *pageState) getLayoutDescriptor() output.LayoutDescriptor {
h, err := ps.createRenderHooks(o.f)
if err != nil {
initErr = err
- }
- if h == nil {
return
}
-
o.cp.renderHooks.hooks = h
+
+ if !o.cp.renderHooksHaveVariants || h.IsZero() {
+ // Check if there is a different render hooks template
+ // for any of the other page output formats.
+ // If not, we can reuse this.
+ for _, po := range ps.pageOutputs {
+ if po.f.Name != o.f.Name {
+ h2, err := ps.createRenderHooks(po.f)
+ if err != nil {
+ initErr = err
+ return
+ }
+
+ if h2.IsZero() {
+ continue
+ }
+
+ if o.cp.renderHooks.hooks.IsZero() {
+ o.cp.renderHooks.hooks = h2
+ }
+
+ o.cp.renderHooksHaveVariants = !h2.Eq(o.cp.renderHooks.hooks)
+
+ if o.cp.renderHooksHaveVariants {
+ break
+ }
+
+ }
+ }
+ }
})
return initErr
}
type renderHooks struct {
- hooks *hooks.Renderers
+ hooks hooks.Renderers
init sync.Once
}
type RenderContext struct {
Src []byte
RenderTOC bool
- RenderHooks *hooks.Renderers
+ RenderHooks hooks.Renderers
}
var FeatureRenderHooks = identity.NewPathIdentity("markup", "renderingHooks")
package hooks
import (
+ "fmt"
"io"
+ "strings"
"github.com/gohugoio/hugo/identity"
)
HeadingRenderer HeadingRenderer
}
-func (r *Renderers) Eq(other interface{}) bool {
- ro, ok := other.(*Renderers)
+func (r Renderers) Eq(other interface{}) bool {
+ ro, ok := other.(Renderers)
if !ok {
return false
}
- if r == nil || ro == nil {
- return r == nil
+
+ if r.IsZero() || ro.IsZero() {
+ return r.IsZero() && ro.IsZero()
}
- if r.ImageRenderer.GetIdentity() != ro.ImageRenderer.GetIdentity() {
+ var b1, b2 bool
+ b1, b2 = r.ImageRenderer == nil, ro.ImageRenderer == nil
+ if (b1 || b2) && (b1 != b2) {
+ return false
+ }
+ if !b1 && r.ImageRenderer.GetIdentity() != ro.ImageRenderer.GetIdentity() {
return false
}
- if r.LinkRenderer.GetIdentity() != ro.LinkRenderer.GetIdentity() {
+ b1, b2 = r.LinkRenderer == nil, ro.LinkRenderer == nil
+ if (b1 || b2) && (b1 != b2) {
+ return false
+ }
+ if !b1 && r.LinkRenderer.GetIdentity() != ro.LinkRenderer.GetIdentity() {
return false
}
- if r.HeadingRenderer.GetIdentity() != ro.HeadingRenderer.GetIdentity() {
+ b1, b2 = r.HeadingRenderer == nil, ro.HeadingRenderer == nil
+ if (b1 || b2) && (b1 != b2) {
+ return false
+ }
+ if !b1 && r.HeadingRenderer.GetIdentity() != ro.HeadingRenderer.GetIdentity() {
return false
}
return true
}
+
+func (r Renderers) IsZero() bool {
+ return r.HeadingRenderer == nil && r.LinkRenderer == nil && r.ImageRenderer == nil
+}
+
+func (r Renderers) String() string {
+ if r.IsZero() {
+ return "<zero>"
+ }
+
+ var sb strings.Builder
+
+ if r.LinkRenderer != nil {
+ sb.WriteString(fmt.Sprintf("LinkRenderer<%s>|", r.LinkRenderer.GetIdentity()))
+ }
+ if r.HeadingRenderer != nil {
+ sb.WriteString(fmt.Sprintf("HeadingRenderer<%s>|", r.HeadingRenderer.GetIdentity()))
+ }
+ if r.ImageRenderer != nil {
+ sb.WriteString(fmt.Sprintf("ImageRenderer<%s>|", r.ImageRenderer.GetIdentity()))
+ }
+
+ return sb.String()
+}
func (r *hookedRenderer) renderImage(w util.BufWriter, source []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
n := node.(*ast.Image)
- var h *hooks.Renderers
+ var h hooks.Renderers
ctx, ok := w.(*renderContext)
if ok {
h = ctx.RenderContext().RenderHooks
- ok = h != nil && h.ImageRenderer != nil
+ ok = h.ImageRenderer != nil
}
if !ok {
func (r *hookedRenderer) renderLink(w util.BufWriter, source []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
n := node.(*ast.Link)
- var h *hooks.Renderers
+ var h hooks.Renderers
ctx, ok := w.(*renderContext)
if ok {
h = ctx.RenderContext().RenderHooks
- ok = h != nil && h.LinkRenderer != nil
+ ok = h.LinkRenderer != nil
}
if !ok {
func (r *hookedRenderer) renderHeading(w util.BufWriter, source []byte, node ast.Node, entering bool) (ast.WalkStatus, error) {
n := node.(*ast.Heading)
- var h *hooks.Renderers
+ var h hooks.Renderers
ctx, ok := w.(*renderContext)
if ok {
h = ctx.RenderContext().RenderHooks
- ok = h != nil && h.HeadingRenderer != nil
+ ok = h.HeadingRenderer != nil
}
if !ok {