Module: Jekyll::Convertible

Included in:
Layout, Page
Defined in:
lib/jekyll/convertible.rb

Instance Method Summary collapse

Instance Method Details

#[](property) ⇒ Object

Accessor for data properties by Liquid.

property - The String name of the property to retrieve.

Returns the String value or nil if the property isn’t included.



292
293
294
295
296
297
298
# File 'lib/jekyll/convertible.rb', line 292

def [](property)
  if self.class::ATTRIBUTES_FOR_LIQUID.include?(property)
    send(property)
  else
    data[property]
  end
end

#asset_file?Boolean

Determine whether the document is an asset file. Asset files include CoffeeScript files and Sass/SCSS files.

Returns true if the extname belongs to the set of extensions

that asset files use.

Returns:

  • (Boolean)


156
157
158
# File 'lib/jekyll/convertible.rb', line 156

def asset_file?
  sass_file? || coffeescript_file?
end

#coffeescript_file?Boolean

Determine whether the document is a CoffeeScript file.

Returns true if extname == .coffee, false otherwise.

Returns:

  • (Boolean)


170
171
172
# File 'lib/jekyll/convertible.rb', line 170

def coffeescript_file?
  '.coffee'.eql?(ext)
end

#convertersObject

Determine which converter to use based on this convertible’s extension.

Returns the Converter instance.



101
102
103
# File 'lib/jekyll/convertible.rb', line 101

def converters
  @converters ||= site.converters.select { |c| c.matches(ext) }.sort
end

#do_layout(payload, layouts) ⇒ Object

Add any necessary layouts to this convertible document.

payload - The site payload Drop or Hash. layouts - A Hash of => “layout”.

Returns nothing.



247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
# File 'lib/jekyll/convertible.rb', line 247

def do_layout(payload, layouts)
  Jekyll.logger.debug "Rendering:", self.relative_path

  Jekyll.logger.debug "Pre-Render Hooks:", self.relative_path
  Jekyll::Hooks.trigger hook_owner, :pre_render, self, payload
  info = { :filters => [Jekyll::Filters], :registers => { :site => site, :page => payload["page"] } }

  # render and transform content (this becomes the final content of the object)
  payload["highlighter_prefix"] = converters.first.highlighter_prefix
  payload["highlighter_suffix"] = converters.first.highlighter_suffix

  if render_with_liquid?
    Jekyll.logger.debug "Rendering Liquid:", self.relative_path
    self.content = render_liquid(content, payload, info, path)
  end
  Jekyll.logger.debug "Rendering Markup:", self.relative_path
  self.content = transform

  # output keeps track of what will finally be written
  self.output = content

  render_all_layouts(layouts, payload, info) if place_in_layout?
  Jekyll.logger.debug "Post-Render Hooks:", self.relative_path
  Jekyll::Hooks.trigger hook_owner, :post_render, self
end

#hook_ownerObject

returns the owner symbol for hook triggering



145
146
147
148
149
# File 'lib/jekyll/convertible.rb', line 145

def hook_owner
  if is_a?(Page)
    :pages
  end
end

#invalid_layout?(layout) ⇒ Boolean

Checks if the layout specified in the document actually exists

layout - the layout to check

Returns true if the layout is invalid, false if otherwise

Returns:

  • (Boolean)


193
194
195
# File 'lib/jekyll/convertible.rb', line 193

def invalid_layout?(layout)
  !data["layout"].nil? && layout.nil? && !(self.is_a? Jekyll::Excerpt)
end

#output_extObject

Determine the extension depending on content_type.

Returns the String extension for the output file.

e.g. ".html" for an HTML output file.


93
94
95
# File 'lib/jekyll/convertible.rb', line 93

def output_ext
  Jekyll::Renderer.new(site, self).output_ext
end

#place_in_layout?Boolean

Determine whether the file should be placed into layouts.

Returns false if the document is an asset file.

Returns:

  • (Boolean)


184
185
186
# File 'lib/jekyll/convertible.rb', line 184

def place_in_layout?
  !asset_file?
end

#published?Boolean

Whether the file is published or not, as indicated in YAML front-matter

Returns:

  • (Boolean)


27
28
29
# File 'lib/jekyll/convertible.rb', line 27

def published?
  !(data.key?('published') && data['published'] == false)
end

#read_yaml(base, name, opts = {}) ⇒ Object

Read the YAML frontmatter.

base - The String path to the dir containing the file. name - The String filename of the file. opts - optional parameter to File.read, default at site configs

Returns nothing.



38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
# File 'lib/jekyll/convertible.rb', line 38

def read_yaml(base, name, opts = {})
  filename = File.join(base, name)

  begin
    self.content = File.read(site.in_source_dir(base, name),
                             Utils.merged_file_read_opts(site, opts))
    if content =~ /\A(---\s*\n.*?\n?)^((---|\.\.\.)\s*$\n?)/m
      self.content = $POSTMATCH
      self.data = SafeYAML.load(Regexp.last_match(1))
    end
  rescue SyntaxError => e
    Jekyll.logger.warn "YAML Exception reading #{filename}: #{e.message}"
  rescue Exception => e
    Jekyll.logger.warn "Error reading file #{filename}: #{e.message}"
  end

  self.data ||= {}

  validate_data! filename
  validate_permalink! filename

  self.data
end

#render_all_layouts(layouts, payload, info) ⇒ Object

Recursively render layouts

layouts - a list of the layouts payload - the payload for Liquid info - the info for Liquid

Returns nothing



204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
# File 'lib/jekyll/convertible.rb', line 204

def render_all_layouts(layouts, payload, info)
  # recursively render layouts
  layout = layouts[data["layout"]]

  Jekyll.logger.warn("Build Warning:", "Layout '#{data["layout"]}' requested in #{path} does not exist.") if invalid_layout? layout

  used = Set.new([layout])

  # Reset the payload layout data to ensure it starts fresh for each page.
  payload["layout"] = nil

  while layout
    Jekyll.logger.debug "Rendering Layout:", path
    payload["content"] = output
    payload["layout"]  = Utils.deep_merge_hashes(layout.data, payload["layout"] || {})

    self.output = render_liquid(layout.content,
                                     payload,
                                     info,
                                     File.join(site.config['layouts_dir'], layout.name))

    # Add layout to dependency tree
    site.regenerator.add_dependency(
      site.in_source_dir(path),
      site.in_source_dir(layout.path)
    )

    if layout = layouts[layout.data["layout"]]
      if used.include?(layout)
        layout = nil # avoid recursive chain
      else
        used << layout
      end
    end
  end
end

#render_liquid(content, payload, info, path) ⇒ Object

Render Liquid in the content

content - the raw Liquid content to render payload - the payload for Liquid info - the info for Liquid

Returns the converted content



112
113
114
115
116
117
118
119
120
# File 'lib/jekyll/convertible.rb', line 112

def render_liquid(content, payload, info, path)
  site.liquid_renderer.file(path).parse(content).render!(payload, info)
rescue Tags::IncludeTagError => e
  Jekyll.logger.error "Liquid Exception:", "#{e.message} in #{e.path}, included in #{path || self.path}"
  raise e
rescue Exception => e
  Jekyll.logger.error "Liquid Exception:", "#{e.message} in #{path || self.path}"
  raise e
end

#render_with_liquid?Boolean

Determine whether the file should be rendered with Liquid.

Always returns true.

Returns:

  • (Boolean)


177
178
179
# File 'lib/jekyll/convertible.rb', line 177

def render_with_liquid?
  true
end

#sass_file?Boolean

Determine whether the document is a Sass file.

Returns true if extname == .sass or .scss, false otherwise.

Returns:

  • (Boolean)


163
164
165
# File 'lib/jekyll/convertible.rb', line 163

def sass_file?
  %w(.sass .scss).include?(ext)
end

#to_liquid(attrs = nil) ⇒ Object

Convert this Convertible’s data to a Hash suitable for use by Liquid.

Returns the Hash representation of this Convertible.



125
126
127
128
129
130
131
132
# File 'lib/jekyll/convertible.rb', line 125

def to_liquid(attrs = nil)
  further_data = Hash[(attrs || self.class::ATTRIBUTES_FOR_LIQUID).map do |attribute|
    [attribute, send(attribute)]
  end]

  defaults = site.frontmatter_defaults.all(relative_path, type)
  Utils.deep_merge_hashes defaults, Utils.deep_merge_hashes(data, further_data)
end

#to_sObject

Returns the contents as a String.



22
23
24
# File 'lib/jekyll/convertible.rb', line 22

def to_s
  content || ''
end

#transformObject

Transform the contents based on the content type.

Returns the transformed contents.



77
78
79
80
81
82
83
84
85
86
87
# File 'lib/jekyll/convertible.rb', line 77

def transform
  converters.reduce(content) do |output, converter|
    begin
      converter.convert output
    rescue => e
      Jekyll.logger.error "Conversion error:", "#{converter.class} encountered an error while converting '#{path}':"
      Jekyll.logger.error("", e.to_s)
      raise e
    end
  end
end

#typeObject

The type of a document,

i.e., its classname downcase'd and to_sym'd.

Returns the type of self.



138
139
140
141
142
# File 'lib/jekyll/convertible.rb', line 138

def type
  if is_a?(Page)
    :pages
  end
end

#validate_data!(filename) ⇒ Object



62
63
64
65
66
# File 'lib/jekyll/convertible.rb', line 62

def validate_data!(filename)
  unless self.data.is_a?(Hash)
    raise Errors::InvalidYAMLFrontMatterError, "Invalid YAML front matter in #{filename}"
  end
end

#validate_permalink!(filename) ⇒ Object



68
69
70
71
72
# File 'lib/jekyll/convertible.rb', line 68

def validate_permalink!(filename)
  if self.data['permalink'] && self.data['permalink'].size == 0
    raise Errors::InvalidPermalinkError, "Invalid permalink in #{filename}"
  end
end

#write(dest) ⇒ Object

Write the generated page file to the destination directory.

dest - The String path to the destination dir.

Returns nothing.



278
279
280
281
282
283
284
285
# File 'lib/jekyll/convertible.rb', line 278

def write(dest)
  path = destination(dest)
  FileUtils.mkdir_p(File.dirname(path))
  File.open(path, 'wb') do |f|
    f.write(output)
  end
  Jekyll::Hooks.trigger hook_owner, :post_write, self
end