Implement support for base templates

This commit is contained in:
adnano 2021-05-09 19:28:09 -04:00
parent 4736a6580e
commit 86ab6169a5
3 changed files with 107 additions and 34 deletions

View file

@ -48,18 +48,13 @@ func LoadConfig(path string) (*Config, error) {
return nil, err return nil, err
} }
return c, nil
}
// LoadTemplates loads templates from the provided path.
func (c *Config) LoadTemplates(path string) error {
// Site contains site metadata passed to templates // Site contains site metadata passed to templates
type Site struct { type Site struct {
Title string Title string
URLs []string URLs []string
} }
// Load templates // Initialize templates
c.templates = NewTemplates() c.templates = NewTemplates()
c.templates.Funcs(map[string]interface{}{ c.templates.Funcs(map[string]interface{}{
"site": func() Site { "site": func() Site {
@ -83,5 +78,10 @@ func (c *Config) LoadTemplates(path string) error {
return template.HTML(s) return template.HTML(s)
}, },
}) })
return c.templates.Load(path)
return c, nil
}
func (c *Config) LoadTemplates(path string, exts []string) error {
return c.templates.Load(path, exts)
} }

View file

@ -56,7 +56,14 @@ func build() {
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
} }
if err := cfg.LoadTemplates("templates"); err != nil {
templateExts := []string{}
for _, task := range cfg.Tasks {
if task.TemplateExt != "" {
templateExts = append(templateExts, task.TemplateExt)
}
}
if err := cfg.LoadTemplates("templates", templateExts); err != nil {
log.Fatal(err) log.Fatal(err)
} }

View file

@ -3,6 +3,7 @@ package main
import ( import (
htemplate "html/template" htemplate "html/template"
"io" "io"
"io/fs"
"io/ioutil" "io/ioutil"
"os" "os"
pathpkg "path" pathpkg "path"
@ -35,42 +36,107 @@ func (t *Templates) Funcs(funcs map[string]interface{}) {
t.funcs = funcs t.funcs = funcs
} }
// LoadTemplate loads a template from the provided path and content. // LoadTemplate loads a template from the provided filenames.
func (t *Templates) LoadTemplate(path string, content []byte) { func (t *Templates) LoadTemplate(name string, filenames ...string) error {
tmpl := template.New(path) if pathpkg.Ext(name) == ".html" {
tmpl.Funcs(t.funcs) return t.loadHTMLTemplate(name, filenames...)
template.Must(tmpl.Parse(string(content))) }
t.tmpls[path] = tmpl return t.loadTextTemplate(name, filenames...)
} }
func (t *Templates) LoadHTMLTemplate(path string, content []byte) { func (t *Templates) loadTextTemplate(name string, filenames ...string) error {
tmpl := htemplate.New(path) tmpl := template.New(name).Funcs(t.funcs)
tmpl.Funcs(htemplate.FuncMap(t.funcs)) for i := range filenames {
htemplate.Must(tmpl.Parse(string(content))) b, err := ioutil.ReadFile(filenames[i])
t.tmpls[path] = tmpl
}
// Load loads templates from the provided directory
func (t *Templates) Load(dir string) error {
return filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
if err != nil { if err != nil {
return err return err
} }
if info.Mode().IsRegular() { if _, err := tmpl.Parse(string(b)); err != nil {
b, err := ioutil.ReadFile(path) return err
if err != nil { }
return err }
t.tmpls[name] = tmpl
return nil
}
func (t *Templates) loadHTMLTemplate(name string, filenames ...string) error {
tmpl := htemplate.New(name).Funcs(t.funcs)
for i := range filenames {
b, err := ioutil.ReadFile(filenames[i])
if err != nil {
return err
}
if _, err := tmpl.Parse(string(b)); err != nil {
return err
}
}
t.tmpls[name] = tmpl
return nil
}
// Load loads templates from the provided directory.
func (t *Templates) Load(dir string, exts []string) error {
err := filepath.WalkDir(dir, func(path string, d fs.DirEntry, err error) error {
if err != nil {
return err
}
if d.IsDir() {
name := d.Name()
if strings.HasPrefix(name, "_") && name != "_default" {
return fs.SkipDir
} }
// Remove directory from beginning of path
path = strings.TrimPrefix(path, dir) // Load atom.xml template
if pathpkg.Ext(path) == ".html" { atom := pathpkg.Join(path, "atom.xml")
t.LoadHTMLTemplate(path, b) if _, err := os.Stat(atom); err == nil {
} else { name := strings.TrimPrefix(atom, dir)
t.LoadTemplate(path, b) t.loadHTMLTemplate(name, atom)
}
// Load page templates
for _, ext := range exts {
for _, name := range []string{"index" + ext, "page" + ext} {
filename := pathpkg.Join(path, name)
if _, err := os.Stat(filename); err != nil {
// Template does not exist
continue
}
filenames := []string{filename}
base := pathpkg.Join(path, "base"+ext)
if _, err := os.Stat(base); err == nil {
filenames = append(filenames, base)
}
name := strings.TrimPrefix(filename, dir)
if err := t.LoadTemplate(name, filenames...); err != nil {
return err
}
}
} }
} }
return nil return nil
}) })
if err != nil && !os.IsNotExist(err) {
return err
}
// Load partial templates
partials := pathpkg.Join(dir, "_partials")
err = filepath.WalkDir(partials, func(path string, d fs.DirEntry, err error) error {
if err != nil {
return err
}
if d.Type().IsRegular() {
name := strings.TrimPrefix(path, dir)
t.LoadTemplate(name, path)
}
return nil
})
if err != nil && !os.IsNotExist(err) {
return err
}
return nil
} }
// FindTemplate returns the template for the given path. // FindTemplate returns the template for the given path.