# ---------------------------------------------------------------------------- # Frozen-string-literal: true # Copyright: 2012 - 2016 - MIT License # Encoding: utf-8 # ---------------------------------------------------------------------------- require "fastimage" module Jekyll module Assets module Liquid class Tag < ::Liquid::Tag require_relative "tag/proxies" require_relative "tag/proxied_asset" require_relative "tag/defaults" require_relative "tag/parser" attr_reader :args # -------------------------------------------------------------------- class << self public :new end # -------------------------------------------------------------------- class AssetNotFoundError < StandardError def initialize(asset) super "Could not find the asset `#{asset}'" end end # -------------------------------------------------------------------- # Tags that we allow our users to use. # -------------------------------------------------------------------- AcceptableTags = %W( img image javascript asset_source stylesheet asset_path style asset css js ).freeze # -------------------------------------------------------------------- # The HTML version of every tag that we accept. # -------------------------------------------------------------------- Tags = { "css" => %(), "js" => %(), "img" => %() }.freeze # -------------------------------------------------------------------- # Allows us to normalize tags so we can simplify logic. # -------------------------------------------------------------------- Alias = { "image" => "img", "stylesheet" => "css", "javascript" => "js", "style" => "css" }.freeze # -------------------------------------------------------------------- def initialize(tag, args, tokens) tag = tag.to_s @tokens = tokens @tag = from_alias(tag) @args = Parser.new(args, @tag) @og_tag = tag super end # -------------------------------------------------------------------- # NOTE: We only attach to the regenerator if you are using digested # assets, otherwise we forego any association with it so that we keep # your builds ultra fast, this is ideal in dev. Disable digests and # let us process independent so the entire site isn't regenerated # because of a single asset change. # -------------------------------------------------------------------- def render(context) site = context.registers[:site] page = context.registers.fetch(:page, {}) args = @args.parse_liquid(context) sprockets = site.sprockets page = page["path"] asset = find_asset(args, sprockets) add_as_jekyll_dependency(site, sprockets, page, asset) process_tag(args, sprockets, asset) rescue => e capture_and_out_error( site, e ) end # -------------------------------------------------------------------- private def from_alias(tag) Alias.key?(tag) ? Alias[tag] : tag end # -------------------------------------------------------------------- private def process_tag(args, sprockets, asset) sprockets.manifest.add(asset) unless @tag == "asset_source" Defaults.set_defaults_for!(@tag, args ||= {}, asset, sprockets) build_html(args, sprockets, asset) end # -------------------------------------------------------------------- private def build_html(args, sprockets, asset, path = get_path(sprockets, asset)) return path if @tag == "asset_path" return asset.to_s if @tag == "asset" || @tag == "asset_source" data = args.key?(:data) && args[:data].key?(:uri) ? asset.data_uri : path format(Tags[@tag], data, args.to_html) end # -------------------------------------------------------------------- private def get_path(sprockets, asset) sprockets.prefix_path( sprockets.digest?? asset.digest_path : asset.logical_path ) end # -------------------------------------------------------------------- private def add_as_jekyll_dependency(site, sprockets, page, asset) if page && sprockets.digest? site.regenerator.add_dependency( site.in_source_dir(page), site.in_source_dir(asset.logical_path) ) end end # -------------------------------------------------------------------- private def find_asset(args, sprockets) out = _find_asset(args[:file], args[:sprockets] ||= {}, sprockets ) if !out raise( AssetNotFoundError, args[ :file ] ) else out.liquid_tags << self !args.proxies?? out : ProxiedAsset.new( out, args, sprockets, self ) end end # -------------------------------------------------------------------- private def _find_asset(file, args, sprockets) if args.empty? then sprockets.manifest.find(file).first elsif args.size == 1 && args.key?(:accept) if File.extname(file) == "" file = file + _ext_for(args[ :accept ]) end sprockets.manifest.find(file).find do |asset| asset.content_type == args[ :accept ] end else sprockets.find_asset( file, args ) end end # -------------------------------------------------------------------- private def _ext_for(type) out = Sprockets.mime_exts.select do |k, v| v == type end out.keys \ .first end # -------------------------------------------------------------------- # There is no guarantee that Jekyll will pass on the error for some # reason (unless you are just booting up) so we capture that error and # always output it, it can lead to some double errors but I would # rather there be a double error than no error. # -------------------------------------------------------------------- private def capture_and_out_error(site, error) if error.is_a?(Sass::SyntaxError) file = error.sass_filename.gsub(/#{Regexp.escape(site.source)}\//, "") Jekyll.logger.error(%(Error in #{file}:#{error.sass_line} #{ error })) else Jekyll.logger.error( "", error.to_s ) end raise error end end end end end # ---------------------------------------------------------------------------- Jekyll::Assets::Liquid::Tag::AcceptableTags.each do |tag| Liquid::Template.register_tag tag, Jekyll::Assets::Liquid::Tag end