2017-03-06 18:16:31 +00:00
// Copyright 2017 The Hugo Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package hugolib
import (
2017-03-22 08:54:56 +00:00
"fmt"
2017-03-19 14:25:32 +00:00
"html/template"
2017-03-21 23:25:55 +00:00
"strings"
2017-03-09 18:19:29 +00:00
"sync"
2017-06-13 16:42:45 +00:00
"github.com/gohugoio/hugo/media"
2017-03-22 10:03:42 +00:00
2017-06-13 16:42:45 +00:00
"github.com/gohugoio/hugo/helpers"
"github.com/gohugoio/hugo/output"
2017-03-06 18:16:31 +00:00
)
// PageOutput represents one of potentially many output formats of a given
// Page.
type PageOutput struct {
* Page
2017-03-09 18:19:29 +00:00
// Pagination
paginator * Pager
paginatorInit sync . Once
// Keep this to create URL/path variations, i.e. paginators.
targetPathDescriptor targetPathDescriptor
2017-03-16 07:32:14 +00:00
outputFormat output . Format
2017-03-06 18:16:31 +00:00
}
2017-03-09 18:19:29 +00:00
func ( p * PageOutput ) targetPath ( addends ... string ) ( string , error ) {
2017-03-16 07:32:14 +00:00
tp , err := p . createTargetPath ( p . outputFormat , addends ... )
2017-03-09 18:19:29 +00:00
if err != nil {
return "" , err
}
return tp , nil
}
2017-03-16 07:32:14 +00:00
func newPageOutput ( p * Page , createCopy bool , f output . Format ) ( * PageOutput , error ) {
2017-03-25 17:28:38 +00:00
// TODO(bep) This is only needed for tests and we should get rid of it.
2017-03-17 15:35:09 +00:00
if p . targetPathDescriptorPrototype == nil {
if err := p . initTargetPathDescriptor ( ) ; err != nil {
return nil , err
}
if err := p . initURLs ( ) ; err != nil {
return nil , err
}
}
2017-03-07 08:55:17 +00:00
if createCopy {
p = p . copy ( )
}
2017-03-09 18:19:29 +00:00
2017-03-16 07:32:14 +00:00
td , err := p . createTargetPathDescriptor ( f )
2017-03-09 18:19:29 +00:00
if err != nil {
return nil , err
}
return & PageOutput {
Page : p ,
2017-03-16 07:32:14 +00:00
outputFormat : f ,
2017-03-09 18:19:29 +00:00
targetPathDescriptor : td ,
} , nil
2017-03-06 18:16:31 +00:00
}
// copy creates a copy of this PageOutput with the lazy sync.Once vars reset
// so they will be evaluated again, for word count calculations etc.
2017-04-12 16:11:37 +00:00
func ( p * PageOutput ) copyWithFormat ( f output . Format ) ( * PageOutput , error ) {
c , err := newPageOutput ( p . Page , true , f )
2017-03-09 18:19:29 +00:00
if err != nil {
2017-04-12 16:11:37 +00:00
return nil , err
2017-03-09 18:19:29 +00:00
}
2017-04-12 11:00:44 +00:00
c . paginator = p . paginator
2017-04-12 16:11:37 +00:00
return c , nil
}
func ( p * PageOutput ) copy ( ) ( * PageOutput , error ) {
return p . copyWithFormat ( p . outputFormat )
2017-03-06 18:16:31 +00:00
}
2017-03-19 14:25:32 +00:00
2017-03-26 17:34:30 +00:00
func ( p * PageOutput ) layouts ( layouts ... string ) ( [ ] string , error ) {
2017-03-25 17:28:38 +00:00
if len ( layouts ) == 0 && p . selfLayout != "" {
2017-03-26 17:34:30 +00:00
return [ ] string { p . selfLayout } , nil
2017-03-19 14:25:32 +00:00
}
layoutOverride := ""
if len ( layouts ) > 0 {
layoutOverride = layouts [ 0 ]
}
return p . s . layoutHandler . For (
p . layoutDescriptor ,
layoutOverride ,
p . outputFormat )
}
func ( p * PageOutput ) Render ( layout ... string ) template . HTML {
2017-03-26 16:51:12 +00:00
if ! p . checkRender ( ) {
2017-03-29 06:08:45 +00:00
return ""
2017-03-26 16:51:12 +00:00
}
2017-03-26 17:34:30 +00:00
l , err := p . layouts ( layout ... )
if err != nil {
helpers . DistinctErrorLog . Printf ( "in .Render: Failed to resolve layout %q for page %q" , layout , p . pathOrTitle ( ) )
2017-03-27 18:43:49 +00:00
return ""
2017-03-26 17:34:30 +00:00
}
2017-03-27 18:43:49 +00:00
for _ , layout := range l {
templ := p . s . Tmpl . Lookup ( layout )
if templ == nil {
// This is legacy from when we had only one output format and
// HTML templates only. Some have references to layouts without suffix.
// We default to good old HTML.
templ = p . s . Tmpl . Lookup ( layout + ".html" )
}
if templ != nil {
res , err := templ . ExecuteToString ( p )
if err != nil {
helpers . DistinctErrorLog . Printf ( "in .Render: Failed to execute template %q for page %q" , layout , p . pathOrTitle ( ) )
return template . HTML ( "" )
}
return template . HTML ( res )
}
}
return ""
2017-03-19 14:25:32 +00:00
}
func ( p * Page ) Render ( layout ... string ) template . HTML {
2017-03-26 16:51:12 +00:00
if ! p . checkRender ( ) {
2017-03-29 06:08:45 +00:00
return ""
2017-03-26 16:51:12 +00:00
}
2017-03-26 09:45:12 +00:00
p . pageOutputInit . Do ( func ( ) {
2017-04-12 16:11:37 +00:00
if p . mainPageOutput != nil {
return
}
2017-03-26 09:45:12 +00:00
// If Render is called in a range loop, the page output isn't available.
// So, create one.
outFormat := p . outputFormats [ 0 ]
pageOutput , err := newPageOutput ( p , true , outFormat )
if err != nil {
2017-03-26 17:34:30 +00:00
p . s . Log . ERROR . Printf ( "Failed to create output page for type %q for page %q: %s" , outFormat . Name , p . pathOrTitle ( ) , err )
2017-03-26 09:45:12 +00:00
return
}
p . mainPageOutput = pageOutput
} )
2017-03-19 14:25:32 +00:00
return p . mainPageOutput . Render ( layout ... )
}
2017-03-21 23:25:55 +00:00
2017-03-26 16:51:12 +00:00
// We may fix this in the future, but the layout handling in Render isn't built
// for list pages.
func ( p * Page ) checkRender ( ) bool {
if p . Kind != KindPage {
2017-03-29 06:08:45 +00:00
helpers . DistinctWarnLog . Printf ( ".Render only available for regular pages, not for of kind %q. You probably meant .Site.RegularPages and not.Site.Pages." , p . Kind )
2017-03-26 16:51:12 +00:00
return false
}
return true
}
2017-03-21 23:25:55 +00:00
// OutputFormats holds a list of the relevant output formats for a given resource.
type OutputFormats [ ] * OutputFormat
2017-08-02 12:25:05 +00:00
// OutputFormat links to a representation of a resource.
2017-03-21 23:25:55 +00:00
type OutputFormat struct {
2017-03-22 08:54:56 +00:00
// Rel constains a value that can be used to construct a rel link.
// This is value is fetched from the output format definition.
// Note that for pages with only one output format,
// this method will always return "canonical".
// As an example, the AMP output format will, by default, return "amphtml".
//
// See:
// https://www.ampproject.org/docs/guides/deploy/discovery
//
// Most other output formats will have "alternate" as value for this.
Rel string
// It may be tempting to export this, but let us hold on to that horse for a while.
2017-03-21 23:25:55 +00:00
f output . Format
2017-03-22 10:03:42 +00:00
2017-03-21 23:25:55 +00:00
p * Page
}
2017-03-22 08:54:56 +00:00
// Name returns this OutputFormat's name, i.e. HTML, AMP, JSON etc.
func ( o OutputFormat ) Name ( ) string {
return o . f . Name
}
2017-03-22 10:03:42 +00:00
// MediaType returns this OutputFormat's MediaType (MIME type).
func ( o OutputFormat ) MediaType ( ) media . Type {
return o . f . MediaType
}
2017-03-21 23:25:55 +00:00
// OutputFormats gives the output formats for this Page.
func ( p * Page ) OutputFormats ( ) OutputFormats {
var o OutputFormats
for _ , f := range p . outputFormats {
2017-03-24 10:25:25 +00:00
o = append ( o , newOutputFormat ( p , f ) )
2017-03-21 23:25:55 +00:00
}
return o
}
2017-03-24 10:25:25 +00:00
func newOutputFormat ( p * Page , f output . Format ) * OutputFormat {
rel := f . Rel
isCanonical := len ( p . outputFormats ) == 1
if isCanonical {
rel = "canonical"
}
return & OutputFormat { Rel : rel , f : f , p : p }
}
2017-08-02 12:25:05 +00:00
// AlternativeOutputFormats gives the alternative output formats for this PageOutput.
2017-04-08 09:15:28 +00:00
// Note that we use the term "alternative" and not "alternate" here, as it
// does not necessarily replace the other format, it is an alternative representation.
2017-03-22 08:54:56 +00:00
func ( p * PageOutput ) AlternativeOutputFormats ( ) ( OutputFormats , error ) {
var o OutputFormats
for _ , of := range p . OutputFormats ( ) {
2017-04-08 09:15:28 +00:00
if of . f . NotAlternative || of . f == p . outputFormat {
2017-03-22 08:54:56 +00:00
continue
}
o = append ( o , of )
}
return o , nil
}
// AlternativeOutputFormats is only available on the top level rendering
// entry point, and not inside range loops on the Page collections.
// This method is just here to inform users of that restriction.
func ( p * Page ) AlternativeOutputFormats ( ) ( OutputFormats , error ) {
return nil , fmt . Errorf ( "AlternativeOutputFormats only available from the top level template context for page %q" , p . Path ( ) )
}
2017-03-21 23:25:55 +00:00
// Get gets a OutputFormat given its name, i.e. json, html etc.
// It returns nil if not found.
func ( o OutputFormats ) Get ( name string ) * OutputFormat {
for _ , f := range o {
2017-04-04 16:14:41 +00:00
if strings . EqualFold ( f . f . Name , name ) {
2017-03-21 23:25:55 +00:00
return f
}
}
return nil
}
// Permalink returns the absolute permalink to this output format.
func ( o * OutputFormat ) Permalink ( ) string {
rel := o . p . createRelativePermalinkForOutputFormat ( o . f )
2017-03-23 19:05:10 +00:00
perm , _ := o . p . s . permalinkForOutputFormat ( rel , o . f )
return perm
2017-03-21 23:25:55 +00:00
}
2017-08-02 12:25:05 +00:00
// RelPermalink returns the relative permalink to this output format.
2017-03-21 23:25:55 +00:00
func ( o * OutputFormat ) RelPermalink ( ) string {
rel := o . p . createRelativePermalinkForOutputFormat ( o . f )
return o . p . s . PathSpec . PrependBasePath ( rel )
}