Skip to content

Commit 24cc255

Browse files
committed
Fix auto generated header ids so they don't contain e.g. hyperlink destinations (note)
This makes the header ids match the newly added dt ids. Also make sure newlines are preserved in hooks' `.PlainText`. Fixes #13405 Fixes #13410
1 parent a2ca956 commit 24cc255

File tree

6 files changed

+59
-14
lines changed

6 files changed

+59
-14
lines changed

‎hugolib/content_render_hooks_test.go

+28
Original file line numberDiff line numberDiff line change
@@ -350,3 +350,31 @@ Image: ![alt-"<>&](/destination-"<> 'title-"<>&')
350350
})
351351
}
352352
}
353+
354+
// Issue 13410.
355+
func TestRenderHooksMultilineTitlePlainText(t *testing.T) {
356+
t.Parallel()
357+
358+
files := `
359+
-- hugo.toml --
360+
-- content/p1.md --
361+
---
362+
title: "p1"
363+
---
364+
365+
First line.
366+
Second line.
367+
----------------
368+
-- layouts/_default/_markup/render-heading.html --
369+
Plain text: {{ .PlainText }}|Text: {{ .Text }}|
370+
-- layouts/_default/single.html --
371+
Content: {{ .Content}}|
372+
}
373+
`
374+
b := Test(t, files)
375+
376+
b.AssertFileContent("public/p1/index.html",
377+
"Content: Plain text: First line.\nSecond line.|",
378+
"|Text: First line.\nSecond line.||\n",
379+
)
380+
}

‎markup/goldmark/goldmark_integration_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -157,7 +157,7 @@ title: "p1"
157157
b := hugolib.Test(t, files)
158158

159159
b.AssertFileContent("public/p1/index.html",
160-
"<h2 id=\"hello-testhttpsexamplecom\">\n Hello <a href=\"https://example.com\">Test</a>\n\n <a class=\"anchor\" href=\"#hello-testhttpsexamplecom\">#</a>\n</h2>",
160+
"<h2 id=\"hello-test\">\n Hello <a href=\"https://example.com\">Test</a>\n\n <a class=\"anchor\" href=\"#hello-test\">#</a>\n</h2>",
161161
)
162162
}
163163

‎markup/goldmark/internal/extensions/attributes/attributes.go

+14-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
package attributes
22

33
import (
4+
"strings"
5+
46
"github.com/gohugoio/hugo/markup/goldmark/goldmark_config"
57
"github.com/gohugoio/hugo/markup/goldmark/internal/render"
68
"github.com/yuin/goldmark"
@@ -181,12 +183,17 @@ func (a *transformer) generateAutoID(n ast.Node, reader text.Reader, pc parser.C
181183
}
182184

183185
// Markdown settext headers can have multiple lines, use the last line for the ID.
184-
func textHeadingID(node *ast.Heading, reader text.Reader) []byte {
185-
var line []byte
186-
lastIndex := node.Lines().Len() - 1
187-
if lastIndex > -1 {
188-
lastLine := node.Lines().At(lastIndex)
189-
line = lastLine.Value(reader.Source())
186+
func textHeadingID(n *ast.Heading, reader text.Reader) []byte {
187+
text := render.TextPlain(n, reader.Source())
188+
if n.Lines().Len() > 1 {
189+
190+
// For multiline headings, Goldmark's extension for headings returns the last line.
191+
// We have a slightly different approach, but in most cases the end result should be the same.
192+
// Instead of looking at the text segments in Lines (see #13405 for issues with that),
193+
// we split the text above and use the last line.
194+
parts := strings.Split(text, "\n")
195+
text = parts[len(parts)-1]
190196
}
191-
return line
197+
198+
return []byte(text)
192199
}

‎markup/goldmark/internal/extensions/attributes/attributes_integration_test.go

+6-3
Original file line numberDiff line numberDiff line change
@@ -47,10 +47,12 @@ foo [something](/a/b/) bar
4747
Ā ā Ă ă Ą ą Ć ć Ĉ ĉ Ċ ċ Č č Ď
4848
: Testing accents.
4949
50-
Mutiline set text header
50+
Multiline set text header
5151
Second line
5252
---------------
5353
54+
## Example [hyperlink](https://example.com/) in a header
55+
5456
-- layouts/_default/single.html --
5557
{{ .Content }}|Identifiers: {{ .Fragments.Identifiers }}|
5658
`
@@ -68,7 +70,8 @@ Second line
6870
`<dt id="my-title-1">My Title</dt>`,
6971
`<dt id="term">良善天父</dt>`,
7072
`<dt id="a-a-a-a-a-a-c-c-c-c-c-c-c-c-d">Ā ā Ă ă Ą ą Ć ć Ĉ ĉ Ċ ċ Č č Ď</dt>`,
71-
`<h2 id="second-line">Mutiline set text header`,
72-
"|Identifiers: [a-a-a-a-a-a-c-c-c-c-c-c-c-c-d base-name base-name-1 foo-something-bar foobar my-title my-title-1 second-line term title-with-id title-with-id]|",
73+
`<h2 id="second-line">`,
74+
`<h2 id="example-hyperlink-in-a-header">`,
75+
"|Identifiers: [a-a-a-a-a-a-c-c-c-c-c-c-c-c-d base-name base-name-1 example-hyperlink-in-a-header foo-something-bar foobar my-title my-title-1 second-line term title-with-id title-with-id]|",
7376
)
7477
}

‎markup/goldmark/internal/render/context.go

+8-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// Copyright 2024 The Hugo Authors. All rights reserved.
1+
// Copyright 2025 The Hugo Authors. All rights reserved.
22
//
33
// Licensed under the Apache License, Version 2.0 (the "License");
44
// you may not use this file except in compliance with the License.
@@ -20,6 +20,7 @@ import (
2020
"sync"
2121

2222
bp "github.com/gohugoio/hugo/bufferpool"
23+
east "github.com/yuin/goldmark-emoji/ast"
2324

2425
htext "github.com/gohugoio/hugo/common/text"
2526
"github.com/gohugoio/hugo/tpl"
@@ -282,6 +283,7 @@ func textPlainTo(c ast.Node, source []byte, buf *bytes.Buffer) {
282283
if c == nil {
283284
return
284285
}
286+
285287
switch c := c.(type) {
286288
case *ast.RawHTML:
287289
s := strings.TrimSpace(tpl.StripHTML(string(c.Segments.Value(source))))
@@ -290,6 +292,11 @@ func textPlainTo(c ast.Node, source []byte, buf *bytes.Buffer) {
290292
buf.Write(c.Value)
291293
case *ast.Text:
292294
buf.Write(c.Segment.Value(source))
295+
if c.HardLineBreak() || c.SoftLineBreak() {
296+
buf.WriteByte('\n')
297+
}
298+
case *east.Emoji:
299+
buf.WriteString(string(c.ShortName))
293300
default:
294301
textPlainTo(c.FirstChild(), source, buf)
295302
}

‎markup/goldmark/toc_integration_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -239,12 +239,12 @@ title: p7 (emoji)
239239

240240
// image
241241
b.AssertFileContent("public/p3/index.html", `
242-
<li><a href="#an-image-kittenajpg">An image <img src="a.jpg" alt="kitten" /></a></li>
242+
<li><a href="#an-image-kitten">An image <img src="a.jpg" alt="kitten" /></a></li>
243243
`)
244244

245245
// raw html
246246
b.AssertFileContent("public/p4/index.html", `
247-
<li><a href="#some-spanrawspan-html">Some <span>raw</span> HTML</a></li>
247+
<li><a href="#some-raw-html">Some <span>raw</span> HTML</a></li>
248248
`)
249249

250250
// typographer

0 commit comments

Comments
 (0)