summaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
authoraycabta <aycabta@gmail.com>2019-06-21 21:31:56 +0900
committeraycabta <aycabta@gmail.com>2019-06-21 21:35:25 +0900
commit50841eca43d2840df54804392afa67bca5f53c0f (patch)
tree17512c79bcd7cce5774cfd77ff0be446f56f1965 /lib
parent38ccb8f74753c6c3d3e47ab612691bd01148eb90 (diff)
Support irregular auto indent
v = if true 3 end # this "end" is auto-indented correctly
Diffstat (limited to 'lib')
-rw-r--r--lib/irb/ruby-lex.rb32
1 files changed, 24 insertions, 8 deletions
diff --git a/lib/irb/ruby-lex.rb b/lib/irb/ruby-lex.rb
index e5c8b14b59..b158028f7a 100644
--- a/lib/irb/ruby-lex.rb
+++ b/lib/irb/ruby-lex.rb
@@ -92,9 +92,9 @@ class RubyLex
last_line = lines[line_index]&.byteslice(0, byte_pointer)
code += last_line if last_line
@tokens = Ripper.lex(code)
- indent, close_token = process_nesting_level(check_closing: true)
- if close_token
- indent * 2
+ indent, corresponding_token_depth = process_nesting_level(check_closing: true)
+ if corresponding_token_depth
+ corresponding_token_depth
else
nil
end
@@ -290,32 +290,48 @@ class RubyLex
end
def process_nesting_level(check_closing: false)
- close_token = false
+ corresponding_token_depth = nil
+ is_first_spaces_of_line = true
+ spaces_of_nest = []
+ spaces_at_line_head = 0
indent = @tokens.inject(0) { |indent, t|
- close_token = false
+ corresponding_token_depth = nil
+ case t[1]
+ when :on_ignored_nl, :on_nl
+ spaces_at_line_head = nil
+ is_first_spaces_of_line = true
+ when :on_sp
+ spaces_at_line_head = t[2].count(' ') if is_first_spaces_of_line
+ is_first_spaces_of_line = false
+ else
+ is_first_spaces_of_line = false
+ end
case t[1]
when :on_lbracket, :on_lbrace, :on_lparen
indent += 1
+ spaces_of_nest.push(spaces_at_line_head)
when :on_rbracket, :on_rbrace, :on_rparen
indent -= 1
- close_token = true
+ corresponding_token_depth = spaces_of_nest.pop
when :on_kw
case t[2]
when 'def', 'do', 'case', 'for', 'begin', 'class', 'module'
indent += 1
+ spaces_of_nest.push(spaces_at_line_head)
when 'if', 'unless', 'while', 'until'
# postfix if/unless/while/until/rescue must be Ripper::EXPR_LABEL
indent += 1 unless t[3].allbits?(Ripper::EXPR_LABEL)
+ spaces_of_nest.push(spaces_at_line_head)
when 'end'
indent -= 1
- close_token = true
+ corresponding_token_depth = spaces_of_nest.pop
end
end
# percent literals are not indented
indent
}
if check_closing
- [indent, close_token]
+ [indent, corresponding_token_depth]
else
indent
end