1
0
mirror of https://github.com/danog/gojekyll.git synced 2025-01-23 04:51:15 +01:00
gojekyll/pages/page.go

145 lines
3.2 KiB
Go
Raw Normal View History

2017-06-22 11:02:32 -04:00
package pages
import (
"bytes"
2017-07-03 09:37:14 -04:00
"fmt"
"io"
"io/ioutil"
2017-07-14 10:59:32 -04:00
"sync"
2017-07-03 09:37:14 -04:00
"time"
"github.com/osteele/gojekyll/frontmatter"
"github.com/osteele/gojekyll/templates"
2017-07-05 11:35:20 -04:00
"github.com/osteele/liquid/evaluator"
)
2017-07-07 12:24:00 -04:00
// Page is a document with frontmatter.
type Page interface {
Document
// Content asks a page to compute its content.
// This has the side effect of causing the content to subsequently appear in the drop.
2017-07-10 13:23:51 -04:00
Content() ([]byte, error)
2017-07-08 19:57:41 -04:00
SetContent(content []byte)
FrontMatter() map[string]interface{}
2017-07-07 12:24:00 -04:00
// PostDate returns the date computed from the filename or frontmatter.
// It is an uncaught error to call this on a page that is not a Post.
// TODO Should posts have their own interface?
PostDate() time.Time
2017-07-13 12:28:02 -04:00
Categories() []string
Tags() []string
2017-07-07 12:24:00 -04:00
}
2017-07-02 13:46:05 -04:00
type page struct {
file
2017-07-14 10:59:32 -04:00
sync.Mutex
firstLine int
raw []byte
content *[]byte
}
// Static is in the File interface.
2017-07-02 13:46:05 -04:00
func (p *page) Static() bool { return false }
2017-07-08 19:57:41 -04:00
func makePage(filename string, f file) (*page, error) {
b, err := ioutil.ReadFile(filename)
2017-06-17 00:16:59 -04:00
if err != nil {
2017-06-22 10:37:31 -04:00
return nil, err
2017-06-17 00:16:59 -04:00
}
lineNo := 1
frontMatter, err := frontmatter.Read(&b, &lineNo)
2017-06-15 09:07:06 -04:00
if err != nil {
2017-06-22 10:37:31 -04:00
return nil, err
2017-06-15 09:07:06 -04:00
}
f.frontMatter = templates.MergeVariableMaps(f.frontMatter, frontMatter)
2017-07-08 19:57:41 -04:00
p := page{
file: f,
firstLine: lineNo,
raw: b,
2017-07-08 19:57:41 -04:00
}
if err = p.setPermalink(); err != nil {
return nil, err
}
return &p, nil
}
2017-07-13 12:28:02 -04:00
func (p *page) FrontMatter() map[string]interface{} {
return p.frontMatter
}
// Categories is in the Page interface
func (p *page) Categories() []string {
return frontmatter.FrontMatter(p.frontMatter).SortedStringArray("categories")
}
// Tags is in the Page interface
func (p *page) Tags() []string {
return frontmatter.FrontMatter(p.frontMatter).SortedStringArray("tags")
}
// TemplateContext returns the local variables for template evaluation
2017-07-10 13:23:51 -04:00
func (p *page) TemplateContext() map[string]interface{} {
2017-06-30 23:56:29 -04:00
return map[string]interface{}{
2017-07-03 09:37:14 -04:00
"page": p,
2017-07-10 13:23:51 -04:00
"site": p.site,
}
}
2017-06-15 09:07:06 -04:00
2017-07-03 09:37:14 -04:00
// PostDate is part of the Page interface.
// FIXME move this back to Page interface, or re-work this entirely.
func (f *file) PostDate() time.Time {
switch value := f.frontMatter["date"].(type) {
2017-07-03 09:37:14 -04:00
case time.Time:
return value
case string:
t, err := evaluator.ParseDate(value)
2017-07-03 09:37:14 -04:00
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.
2017-07-10 13:23:51 -04:00
func (p *page) Write(w io.Writer) error {
content, err := p.Content()
if err != nil {
return err
}
2017-07-04 17:13:47 -04:00
layout, ok := p.frontMatter["layout"].(string)
if ok && layout != "" {
2017-07-10 13:23:51 -04:00
rp := p.site.RenderingPipeline()
content, err = rp.ApplyLayout(layout, content, p.TemplateContext())
if err != nil {
return err
}
}
2017-07-08 14:38:37 -04:00
_, err = w.Write(content)
return err
2017-06-15 09:07:06 -04:00
}
2017-06-22 12:56:08 -04:00
2017-07-01 09:35:54 -04:00
// Content computes the page content.
2017-07-10 13:23:51 -04:00
func (p *page) Content() ([]byte, error) {
2017-07-14 10:59:32 -04:00
content := p.maybeContent(false)
if content != nil {
return content, nil
2017-07-09 16:09:03 -04:00
}
2017-07-10 13:23:51 -04:00
pipe := p.site.RenderingPipeline()
2017-07-09 16:09:03 -04:00
buf := new(bytes.Buffer)
2017-07-10 13:23:51 -04:00
b, err := pipe.Render(buf, p.raw, p.filename, p.firstLine, p.TemplateContext())
2017-07-09 16:09:03 -04:00
if err != nil {
return nil, err
2017-06-22 12:56:08 -04:00
}
2017-07-14 10:59:32 -04:00
p.SetContent(b)
2017-07-09 16:09:03 -04:00
return b, nil
2017-06-22 12:56:08 -04:00
}
2017-07-08 19:57:41 -04:00
2017-07-14 10:59:32 -04:00
// retains its argument
2017-07-08 19:57:41 -04:00
func (p *page) SetContent(content []byte) {
2017-07-14 10:59:32 -04:00
p.Lock()
defer p.Unlock()
2017-07-08 19:57:41 -04:00
p.content = &content
}