lib/rubocop/cop/mixin/autocorrect_alignment.rb



# encoding: utf-8

module RuboCop
  module Cop
    # This module does auto-correction of nodes that should just be moved to
    # the left or to the right, amount being determined by the instance
    # variable @column_delta.
    module AutocorrectAlignment
      def check_alignment(items, base_column = nil)
        base_column ||= items.first.loc.column unless items.empty?
        prev_line = -1
        items.each do |current|
          if current.loc.line > prev_line && start_of_line?(current.loc)
            @column_delta = base_column - current.loc.column
            add_offense(current, :expression) if @column_delta != 0
          end
          prev_line = current.loc.line
        end
      end

      def start_of_line?(loc)
        loc.expression.source_line[0...loc.column] =~ /^\s*$/
      end

      def autocorrect(arg)
        heredoc_ranges = heredoc_ranges(arg)
        expr = arg.respond_to?(:loc) ? arg.loc.expression : arg

        # We can't use the instance variable inside the lambda. That would
        # just give each lambda the same reference and they would all get
        # the last value of @column_delta. A local variable fixes the
        # problem.
        column_delta = @column_delta

        fail CorrectionNotPossible if block_comment_within?(expr)

        @corrections << lambda do |corrector|
          each_line(expr) do |line_begin_pos|
            range = calculate_range(expr, line_begin_pos, column_delta)
            # We must not change indentation of heredoc stings.
            next if heredoc_ranges.any? { |h| within?(range, h) }

            if column_delta > 0
              unless range.source == "\n"
                corrector.insert_before(range, ' ' * column_delta)
              end
            else
              remove(range, corrector) if range.source =~ /^[ \t]+$/
            end
          end
        end
      end

      private

      def heredoc_ranges(arg)
        return [] unless arg.is_a?(Parser::AST::Node)

        heredoc_ranges = []
        on_node(:dstr, arg) do |n|
          if n.loc.respond_to?(:heredoc_body)
            heredoc_ranges << n.loc.heredoc_body.join(n.loc.heredoc_end)
          end
        end
        heredoc_ranges
      end

      def block_comment_within?(expr)
        processed_source.comments.select { |c| c.document? }.find do |c|
          within?(c.loc.expression, expr)
        end
      end

      def within?(inner, outer)
        inner.begin_pos >= outer.begin_pos && inner.end_pos <= outer.end_pos
      end

      def calculate_range(expr, line_begin_pos, column_delta)
        starts_with_space = expr.source_buffer.source[line_begin_pos] =~ / /
        pos_to_remove = if column_delta > 0 || starts_with_space
                          line_begin_pos
                        else
                          line_begin_pos - column_delta.abs
                        end
        Parser::Source::Range.new(expr.source_buffer, pos_to_remove,
                                  pos_to_remove + column_delta.abs)
      end

      def remove(range, corrector)
        original_stderr = $stderr
        $stderr = StringIO.new # Avoid error messages on console
        corrector.remove(range)
      rescue RuntimeError
        range = Parser::Source::Range.new(range.source_buffer,
                                          range.begin_pos + 1,
                                          range.end_pos + 1)
        retry if range.source =~ /^ +$/
      ensure
        $stderr = original_stderr
      end

      def each_line(expr)
        offset = 0
        expr.source.each_line do |line|
          line_begin_pos = expr.begin_pos + offset
          yield line_begin_pos
          offset += line.length
        end
      end
    end
  end
end