1
0
mirror of https://github.com/danog/gojekyll.git synced 2025-01-23 10:41:12 +01:00
gojekyll/pages/page.go
2017-07-05 11:35:20 -04:00

106 lines
2.2 KiB
Go

package pages
import (
"bytes"
"fmt"
"io"
"io/ioutil"
"time"
"github.com/osteele/gojekyll/templates"
"github.com/osteele/liquid/evaluator"
)
type page struct {
file
raw []byte
content *[]byte
}
// Static is in the File interface.
func (p *page) Static() bool { return false }
func newPage(filename string, f file) (*page, error) {
b, err := ioutil.ReadFile(filename)
if err != nil {
return nil, err
}
frontMatter, err := templates.ReadFrontMatter(&b)
if err != nil {
return nil, err
}
f.frontMatter = templates.MergeVariableMaps(f.frontMatter, frontMatter)
return &page{
file: f,
raw: b,
}, nil
}
// TemplateContext returns the local variables for template evaluation
func (p *page) TemplateContext(rc RenderingContext) map[string]interface{} {
return map[string]interface{}{
"page": p,
"site": rc.Site(),
}
}
// // Categories is part of the Page interface.
// func (p *page) Categories() []string {
// return []string{}
// }
// Tags is part of the Page interface.
func (p *page) Tags() []string {
return []string{}
}
// PostDate is part of the Page interface.
func (p *page) PostDate() time.Time {
switch value := p.frontMatter["date"].(type) {
case time.Time:
return value
case string:
t, err := evaluator.ParseTime(value)
if err == nil {
return t
}
default:
panic(fmt.Sprintf("expected a date %v", value))
}
panic("read posts should have set this")
}
// Write applies Liquid and Markdown, as appropriate.
func (p *page) Write(w io.Writer, rc RenderingContext) error {
rp := rc.RenderingPipeline()
b, err := rp.Render(w, p.raw, p.filename, p.TemplateContext(rc))
if err != nil {
return err
}
layout, ok := p.frontMatter["layout"].(string)
if ok && layout != "" {
b, err = rp.ApplyLayout(layout, b, p.TemplateContext(rc))
if err != nil {
return err
}
}
_, err = w.Write(b)
return err
}
// Content computes the page content.
func (p *page) Content(rc RenderingContext) ([]byte, error) {
if p.content == nil {
// TODO DRY w/ Page.Write
rp := rc.RenderingPipeline()
buf := new(bytes.Buffer)
b, err := rp.Render(buf, p.raw, p.filename, p.TemplateContext(rc))
if err != nil {
return nil, err
}
p.content = &b
}
return *p.content, nil
}