summaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
authoraycabta <aycabta@gmail.com>2019-06-25 22:07:32 +0900
committeraycabta <aycabta@gmail.com>2019-06-25 22:07:32 +0900
commit57e1a69ea38d30bc249553e5fe15295ae0f5af81 (patch)
tree141fd64f0a925788ce259e0a3b19b39add7813c7 /lib
parent0b57f9b25d43c2efec8ac17c0c03ce9fa461937a (diff)
Treat auto indent with newline correctly
Diffstat (limited to 'lib')
-rw-r--r--lib/irb/ruby-lex.rb52
1 files changed, 42 insertions, 10 deletions
diff --git a/lib/irb/ruby-lex.rb b/lib/irb/ruby-lex.rb
index cb1538044e..27029e5266 100644
--- a/lib/irb/ruby-lex.rb
+++ b/lib/irb/ruby-lex.rb
@@ -77,16 +77,9 @@ class RubyLex
if is_newline
md = lines[line_index - 1].match(/(\A +)/)
prev_spaces = md.nil? ? 0 : md[1].count(' ')
- indent_list = []
- code = ''
- lines.each_with_index { |l, i|
- code << l + "\n"
- @tokens = Ripper.lex(code)
- indent_list << process_nesting_level
- }
- prev_indent = (line_index - 1).zero? ? 0 : indent_list[line_index - 2]
- indent = indent_list[line_index - 1]
- prev_spaces + (indent - prev_indent) * 2
+ @tokens = Ripper.lex(lines[0..line_index].join("\n"))
+ depth_difference = check_newline_depth_difference
+ prev_spaces + depth_difference * 2
else
code = line_index.zero? ? '' : lines[0..(line_index - 1)].map{ |l| l + "\n" }.join
last_line = lines[line_index]&.byteslice(0, byte_pointer)
@@ -313,6 +306,45 @@ class RubyLex
indent
end
+ def check_newline_depth_difference
+ depth_difference = 0
+ $stderr.puts ?= * 100
+ $stderr.puts @tokens.inspect
+ @tokens.each_with_index do |t, index|
+ case t[1]
+ when :on_ignored_nl, :on_nl
+ if index != (@tokens.size - 1)
+ $stderr.puts "nl"
+ depth_difference = 0
+ end
+ next
+ when :on_sp
+ next
+ end
+ case t[1]
+ when :on_lbracket, :on_lbrace, :on_lparen
+ depth_difference += 1
+ when :on_rbracket, :on_rbrace, :on_rparen
+ depth_difference -= 1
+ when :on_kw
+ case t[2]
+ when 'def', 'do', 'case', 'for', 'begin', 'class', 'module'
+ depth_difference += 1
+ when 'if', 'unless', 'while', 'until'
+ # postfix if/unless/while/until/rescue must be Ripper::EXPR_LABEL
+ unless t[3].allbits?(Ripper::EXPR_LABEL)
+ $stderr.puts "if"
+ depth_difference += 1
+ end
+ when 'else', 'rescue', 'ensure', 'when', 'in'
+ $stderr.puts "else"
+ depth_difference += 1
+ end
+ end
+ end
+ depth_difference
+ end
+
def check_corresponding_token_depth
corresponding_token_depth = nil
is_first_spaces_of_line = true