sds/haml-lint

View on GitHub
lib/haml_lint/haml_visitor.rb

Summary

Maintainability
A
45 mins
Test Coverage
# frozen_string_literal: true

module HamlLint
  # Provides an interface which when included allows a class to visit nodes in
  # the parse tree of a HAML document.
  module HamlVisitor
    def visit(node)
      # Keep track of whether this block was consumed by the visitor. This
      # allows us to visit all nodes by default, but can override the behavior
      # by specifying `yield false` in a visit method, indicating that no
      # further visiting should occur for the current node's children.
      block_called = false

      block = ->(descend = :children) do
        block_called = true
        visit_children(node) if descend == :children
      end

      disabled = node.disabled?(self)

      safe_send("visit_#{node_name(node)}", node, &block) unless disabled

      # Visit all children by default unless the block was invoked (indicating
      # the user intends to not recurse further, or wanted full control over
      # when the children were visited).
      visit_children(node) unless block_called

      safe_send("after_visit_#{node_name(node)}", node, &block) unless disabled
    end

    def visit_children(parent)
      parent.children.each { |node| visit(node) }
    end

    private

    def node_name(node)
      node.type
    end

    def safe_send(name, *args, &block)
      send(name, *args, &block) if respond_to?(name, true)
    end
  end
end