}
ps.codeowners = owners
+ mi, err := s.h.markdownTitleInfoForPage(ps)
+ if err != nil {
+ return nil, errors.Wrap(err, "failed to load MarkdownTitle data")
+ }
+ ps.markdownTitleInfo = mi
+
r, err := content()
if err != nil {
return nil, err
gitInfo *gitInfo
codeownerInfo *codeownerInfo
+ markdownTitleInfo *markdownTitleInfo
+
// As loaded from the /data dirs
data map[string]any
// Loads the Git info and CODEOWNERS for all the pages if enabled.
gitInfo *lazy.Init
+ markdownTitleInfo *lazy.Init
+
// Maps page translations.
translations *lazy.Init
}
h.data.Reset()
h.layouts.Reset()
h.gitInfo.Reset()
+ h.markdownTitleInfo.Reset()
h.translations.Reset()
}
return h.codeownerInfo.forPage(p), nil
}
+
+func (h *HugoSites) markdownTitleInfoForPage(p page.Page) (*MarkdownTitleInfo, error) {
+ if _, err := h.init.markdownTitleInfo.Do(); err != nil {
+ return nil, err
+ }
+
+ if h.markdownTitleInfo == nil {
+ return nil, nil
+ }
+
+ return h.markdownTitleInfo.forPage(p), nil
+}
+
func (h *HugoSites) siteInfos() page.Sites {
infos := make(page.Sites, len(h.Sites))
for i, site := range h.Sites {
data: lazy.New(),
layouts: lazy.New(),
gitInfo: lazy.New(),
+ markdownTitleInfo: lazy.New(),
translations: lazy.New(),
},
}
return nil, nil
})
+ h.init.markdownTitleInfo.Add(func() (any, error) {
+ err := h.loadMarkdownTitleInfo()
+ if err != nil {
+ return nil, errors.Wrap(err, "failed to load MarkdownTitle info")
+ }
+ return nil, nil
+ })
+
for _, s := range sites {
s.h = h
}
return nil
}
+func (h *HugoSites) loadMarkdownTitleInfo() error {
+ if h.Cfg.GetBool("enableMarkdownTitleInfo") {
+ mi, err := newMarkdownTitleInfo(h.Cfg)
+ if err != nil {
+ h.Log.Errorln("Failed to read MarkdownTitle:", err)
+ } else {
+ h.markdownTitleInfo = mi
+ }
+ }
+ return nil
+}
+
func (l configLoader) applyDeps(cfg deps.DepsCfg, sites ...*Site) error {
if cfg.TemplateProvider == nil {
cfg.TemplateProvider = tplimpl.DefaultTemplateProvider
--- /dev/null
+package hugolib
+
+import (
+ "path/filepath"
+ "strings"
+ "os"
+
+ "github.com/yuin/goldmark"
+ "github.com/yuin/goldmark/text"
+ "github.com/yuin/goldmark/ast"
+
+ "github.com/gohugoio/hugo/config"
+ "github.com/gohugoio/hugo/resources/page"
+)
+
+type MarkdownTitleInfo struct {
+ Title string `json:"title"` // Title
+}
+
+type MarkdownTitleMap map[string]*MarkdownTitleInfo
+
+type markdownTitleInfo struct {
+ contentDir string
+ Map MarkdownTitleMap
+}
+
+func findTitle(n ast.Node, source []byte) (string, bool) {
+ ret := ""
+ if n.Kind() == ast.KindHeading {
+ return string(n.Text(source)[:]), true
+ }
+ for c := n.FirstChild(); c != nil; c = c.NextSibling() {
+ ret, res := findTitle(c, source)
+ if res {
+ return ret, true
+ }
+ }
+
+ return ret, false
+}
+
+func (m *markdownTitleInfo) forPage(p page.Page) *MarkdownTitleInfo {
+ markdown := goldmark.New()
+ name := strings.TrimPrefix(filepath.ToSlash(p.File().Filename()), m.contentDir)
+ // name = strings.TrimPrefix(name, "/")
+
+ source, err := os.ReadFile(name)
+ if err != nil {
+ return nil
+ }
+
+ // Parse file and return title
+ n := markdown.Parser().Parse(text.NewReader([]byte(source)))
+ title, res := findTitle(n, source)
+ if !res {
+ return &MarkdownTitleInfo{Title: ""}
+ }
+
+ return &MarkdownTitleInfo{Title: title}
+}
+
+func newMarkdownTitleInfo(cfg config.Provider) (*markdownTitleInfo, error) {
+ contentDir := cfg.GetString("contentDir")
+
+ return &markdownTitleInfo{contentDir: contentDir}, nil
+}
return p.codeowners
}
+func (p *pageState) MarkdownTitleInfo() *MarkdownTitleInfo {
+ return p.markdownTitleInfo
+}
+
// GetTerms gets the terms defined on this page in the given taxonomy.
// The pages returned will be ordered according to the front matter.
func (p *pageState) GetTerms(taxonomy string) page.Pages {
gitInfo *gitmap.GitInfo
codeowners []string
+ markdownTitleInfo *MarkdownTitleInfo
+
// Positional navigation
posNextPrev *nextPrev
posNextPrevSection *nextPrev