lib/haml/compiler/tag_compiler.rb
# frozen_string_literal: true require 'haml/util' require 'haml/attribute_compiler' require 'haml/string_splitter' module Haml class Compiler class TagCompiler def initialize(identity, options) @autoclose = options[:autoclose] @identity = identity @attribute_compiler = AttributeCompiler.new(identity, options) end def compile(node, &block) attrs = @attribute_compiler.compile(node) contents = compile_contents(node, &block) [:html, :tag, node.value[:name], attrs, contents] end private def compile_contents(node, &block) case when !node.children.empty? yield(node) when node.value[:value].nil? && self_closing?(node) nil when node.value[:parse] return compile_interpolated_plain(node) if node.value[:escape_interpolation] if Ripper.respond_to?(:lex) # No Ripper.lex in truffleruby return delegate_optimization(node) if RubyExpression.string_literal?(node.value[:value]) return delegate_optimization(node) if Temple::StaticAnalyzer.static?(node.value[:value]) end var = @identity.generate [:multi, [:code, "#{var} = (#{node.value[:value]}"], [:newline], [:code, ')'], [:escape, node.value[:escape_html], [:dynamic, var]] ] else [:static, node.value[:value]] end end # :dynamic is optimized in other filters: StringSplitter or StaticAnalyzer def delegate_optimization(node) [:multi, [:escape, node.value[:escape_html], [:dynamic, node.value[:value]]], [:newline], ] end # We should handle interpolation here to escape only interpolated values. def compile_interpolated_plain(node) temple = [:multi] StringSplitter.compile(node.value[:value]).each do |type, value| case type when :static temple << [:static, value] when :dynamic temple << [:escape, node.value[:escape_interpolation], [:dynamic, value]] end end temple << [:newline] end def self_closing?(node) return true if @autoclose && @autoclose.include?(node.value[:name]) node.value[:self_closing] end end end end