Skip to content

Commit 797b82c

Browse files
authored
Merge pull request #91 from Vimjas/maxoff-by-type
Intermediate fix for indenting huge dicts
2 parents e1b71a2 + 5e95516 commit 797b82c

File tree

2 files changed

+88
-20
lines changed

2 files changed

+88
-20
lines changed

indent/python.vim

+24-20
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,6 @@ if !exists('g:python_pep8_indent_multiline_string')
3434
let g:python_pep8_indent_multiline_string = 0
3535
endif
3636

37-
let s:maxoff = 50
3837
let s:block_rules = {
3938
\ '^\s*elif\>': ['if', 'elif'],
4039
\ '^\s*except\>': ['try', 'except'],
@@ -43,7 +42,13 @@ let s:block_rules = {
4342
let s:block_rules_multiple = {
4443
\ '^\s*else\>': ['if', 'elif', 'for', 'try', 'except'],
4544
\ }
46-
let s:paren_pairs = ['()', '{}', '[]']
45+
" Pairs to look for when searching for opening parenthesis.
46+
" The value is the maximum offset in lines.
47+
let s:paren_pairs = {'()': 50, '[]': 100, '{}': 1000}
48+
49+
" Maximum offset when looking for multiline statements (in round parenthesis).
50+
let s:maxoff_multiline_statement = 50
51+
4752
if &filetype ==# 'pyrex' || &filetype ==# 'cython'
4853
let b:control_statement = '\v^\s*(class|def|if|while|with|for|except|cdef|cpdef)>'
4954
else
@@ -105,37 +110,36 @@ function! s:find_opening_paren(...)
105110
return ret
106111
endif
107112

108-
let stopline = max([0, line('.') - s:maxoff])
109-
110113
" Return if cursor is in a comment.
111114
exe 'if' s:skip_search '| return [0, 0] | endif'
112115

113-
let positions = []
114-
for p in s:paren_pairs
115-
call add(positions, searchpairpos(
116-
\ '\V'.p[0], '', '\V'.p[1], 'bnW', s:skip_special_chars, stopline))
116+
let nearest = [0, 0]
117+
for [p, maxoff] in items(s:paren_pairs)
118+
let stopline = max([0, line('.') - maxoff, nearest[0]])
119+
let next = searchpairpos(
120+
\ '\V'.p[0], '', '\V'.p[1], 'bnW', s:skip_special_chars, stopline)
121+
if next[0] && (next[0] > nearest[0] || (next[0] == nearest[0] && next[1] > nearest[1]))
122+
let nearest = next
123+
endif
117124
endfor
118-
119-
" Remove empty matches and return the type with the closest match
120-
call filter(positions, 'v:val[0]')
121-
call sort(positions, 's:pair_sort')
122-
123-
return get(positions, -1, [0, 0])
125+
return nearest
124126
endfunction
125127

126-
" Find the start of a multi-line statement
128+
" Find the start of a multi-line statement (based on surrounding parens).
127129
function! s:find_start_of_multiline_statement(lnum)
128130
let lnum = a:lnum
129131
while lnum > 0
132+
" XXX: not tested?!
130133
if getline(lnum - 1) =~# '\\$'
131134
let lnum = prevnonblank(lnum - 1)
132135
else
133-
let [paren_lnum, _] = s:find_opening_paren(lnum)
134-
if paren_lnum < 1
135-
return lnum
136-
else
137-
let lnum = paren_lnum
136+
call cursor(lnum, 1)
137+
let stopline = max([1, lnum - s:maxoff_multiline_statement])
138+
let pos = searchpairpos('\V(', '', '\V)', 'bnW', s:skip_special_chars, stopline)
139+
if pos[0]
140+
return pos[0]
138141
endif
142+
return lnum
139143
endif
140144
endwhile
141145
endfunction

spec/indent/indent_spec.rb

+64
Original file line numberDiff line numberDiff line change
@@ -587,3 +587,67 @@
587587
end
588588
end
589589
end
590+
591+
describe "Handles far away opening parens" do
592+
before { vim.feedkeys '\<ESC>ggdGifrom foo import (' }
593+
594+
it "indents by one level" do
595+
vim.feedkeys '\<CR>'
596+
proposed_indent.should == shiftwidth
597+
end
598+
599+
it "indents by one level for 10 lines" do
600+
vim.command('set paste | exe "norm 9o" | set nopaste')
601+
vim.feedkeys '\<Esc>o'
602+
indent.should == shiftwidth
603+
end
604+
605+
it "indents by one level for 50 lines" do
606+
vim.command('set paste | exe "norm 49o" | set nopaste')
607+
vim.feedkeys '\<Esc>o'
608+
indent.should == shiftwidth
609+
end
610+
end
611+
612+
describe "Handles far away opening square brackets" do
613+
before { vim.feedkeys '\<ESC>ggdGibar = [' }
614+
615+
it "indents by one level" do
616+
vim.feedkeys '\<CR>'
617+
proposed_indent.should == shiftwidth
618+
end
619+
620+
it "indents by one level for 10 lines" do
621+
vim.command('set paste | exe "norm 9o" | set nopaste')
622+
vim.feedkeys '\<Esc>o'
623+
indent.should == shiftwidth
624+
end
625+
626+
it "indents by one level for 100 lines" do
627+
vim.command('set paste | exe "norm 99o" | set nopaste')
628+
vim.feedkeys '\<Esc>o'
629+
indent.should == shiftwidth
630+
end
631+
end
632+
633+
describe "Handles far away opening curly brackets" do
634+
before { vim.feedkeys '\<ESC>ggdGijson = {' }
635+
636+
it "indents by one level" do
637+
vim.feedkeys '\<CR>'
638+
vim.feedkeys '\<Esc>o'
639+
proposed_indent.should == shiftwidth
640+
end
641+
642+
it "indents by one level for 10 lines" do
643+
vim.command('set paste | exe "norm 9o" | set nopaste')
644+
vim.feedkeys '\<Esc>o'
645+
indent.should == shiftwidth
646+
end
647+
648+
it "indents by one level for 1000 lines" do
649+
vim.command('set paste | exe "norm 999o" | set nopaste')
650+
vim.feedkeys '\<Esc>o'
651+
indent.should == shiftwidth
652+
end
653+
end

0 commit comments

Comments
 (0)