1071d4279SBram Moolenaar" Vim indent file 2071d4279SBram Moolenaar" Language: Python 3071d4279SBram Moolenaar" Maintainer: Bram Moolenaar <[email protected]> 4071d4279SBram Moolenaar" Original Author: David Bustos <[email protected]> 5*6e649224SBram Moolenaar" Last Change: 2021 Sep 26 6071d4279SBram Moolenaar 7071d4279SBram Moolenaar" Only load this indent file when no other was loaded. 8071d4279SBram Moolenaarif exists("b:did_indent") 9071d4279SBram Moolenaar finish 10071d4279SBram Moolenaarendif 11071d4279SBram Moolenaarlet b:did_indent = 1 12071d4279SBram Moolenaar 13071d4279SBram Moolenaar" Some preliminary settings 14071d4279SBram Moolenaarsetlocal nolisp " Make sure lisp indenting doesn't supersede us 15071d4279SBram Moolenaarsetlocal autoindent " indentexpr isn't much help otherwise 16071d4279SBram Moolenaar 17071d4279SBram Moolenaarsetlocal indentexpr=GetPythonIndent(v:lnum) 18071d4279SBram Moolenaarsetlocal indentkeys+=<:>,=elif,=except 19071d4279SBram Moolenaar 20*6e649224SBram Moolenaarlet b:undo_indent = "setl ai< inde< indk< lisp<" 21*6e649224SBram Moolenaar 22071d4279SBram Moolenaar" Only define the function once. 23071d4279SBram Moolenaarif exists("*GetPythonIndent") 24071d4279SBram Moolenaar finish 25071d4279SBram Moolenaarendif 268071607aSBram Moolenaarlet s:keepcpo= &cpo 278071607aSBram Moolenaarset cpo&vim 28071d4279SBram Moolenaar 2905159a0cSBram Moolenaar" Come here when loading the script the first time. 3005159a0cSBram Moolenaar 31071d4279SBram Moolenaarlet s:maxoff = 50 " maximum number of lines to look backwards for () 32071d4279SBram Moolenaar 3347e13953SBram Moolenaar" See if the specified line is already user-dedented from the expected value. 3447e13953SBram Moolenaarfunction s:Dedented(lnum, expected) 3547e13953SBram Moolenaar return indent(a:lnum) <= a:expected - shiftwidth() 3647e13953SBram Moolenaarendfunction 3747e13953SBram Moolenaar 38071d4279SBram Moolenaarfunction GetPythonIndent(lnum) 3905159a0cSBram Moolenaar 40071d4279SBram Moolenaar " If this line is explicitly joined: If the previous line was also joined, 41071d4279SBram Moolenaar " line it up with that one, otherwise add two 'shiftwidth' 42071d4279SBram Moolenaar if getline(a:lnum - 1) =~ '\\$' 43071d4279SBram Moolenaar if a:lnum > 1 && getline(a:lnum - 2) =~ '\\$' 44071d4279SBram Moolenaar return indent(a:lnum - 1) 45071d4279SBram Moolenaar endif 4656b45b9bSBram Moolenaar return indent(a:lnum - 1) + (exists("g:pyindent_continue") ? eval(g:pyindent_continue) : (shiftwidth() * 2)) 47071d4279SBram Moolenaar endif 48071d4279SBram Moolenaar 49071d4279SBram Moolenaar " If the start of the line is in a string don't change the indent. 50071d4279SBram Moolenaar if has('syntax_items') 51ed20346fSBram Moolenaar \ && synIDattr(synID(a:lnum, 1, 1), "name") =~ "String$" 52071d4279SBram Moolenaar return -1 53071d4279SBram Moolenaar endif 54071d4279SBram Moolenaar 55071d4279SBram Moolenaar " Search backwards for the previous non-empty line. 56071d4279SBram Moolenaar let plnum = prevnonblank(v:lnum - 1) 57071d4279SBram Moolenaar 58071d4279SBram Moolenaar if plnum == 0 59071d4279SBram Moolenaar " This is the first non-empty line, use zero indent. 60071d4279SBram Moolenaar return 0 61071d4279SBram Moolenaar endif 62071d4279SBram Moolenaar 63f6b40109SBram Moolenaar call cursor(plnum, 1) 64f6b40109SBram Moolenaar 65f6b40109SBram Moolenaar " Identing inside parentheses can be very slow, regardless of the searchpair() 66f6b40109SBram Moolenaar " timeout, so let the user disable this feature if he doesn't need it 67f6b40109SBram Moolenaar let disable_parentheses_indenting = get(g:, "pyindent_disable_parentheses_indenting", 0) 68f6b40109SBram Moolenaar 69f6b40109SBram Moolenaar if disable_parentheses_indenting == 1 70f6b40109SBram Moolenaar let plindent = indent(plnum) 71f6b40109SBram Moolenaar let plnumstart = plnum 72f6b40109SBram Moolenaar else 73f6b40109SBram Moolenaar " searchpair() can be slow sometimes, limit the time to 150 msec or what is 742c64ca18SBram Moolenaar " put in g:pyindent_searchpair_timeout 752c64ca18SBram Moolenaar let searchpair_stopline = 0 762c64ca18SBram Moolenaar let searchpair_timeout = get(g:, 'pyindent_searchpair_timeout', 150) 772c64ca18SBram Moolenaar 78071d4279SBram Moolenaar " If the previous line is inside parenthesis, use the indent of the starting 79071d4279SBram Moolenaar " line. 80071d4279SBram Moolenaar " Trick: use the non-existing "dummy" variable to break out of the loop when 81071d4279SBram Moolenaar " going too far back. 82910f66f9SBram Moolenaar let parlnum = searchpair('(\|{\|\[', '', ')\|}\|\]', 'nbW', 83071d4279SBram Moolenaar \ "line('.') < " . (plnum - s:maxoff) . " ? dummy :" 84071d4279SBram Moolenaar \ . " synIDattr(synID(line('.'), col('.'), 1), 'name')" 852c64ca18SBram Moolenaar \ . " =~ '\\(Comment\\|Todo\\|String\\)$'", 862c64ca18SBram Moolenaar \ searchpair_stopline, searchpair_timeout) 87071d4279SBram Moolenaar if parlnum > 0 88071d4279SBram Moolenaar let plindent = indent(parlnum) 89071d4279SBram Moolenaar let plnumstart = parlnum 90071d4279SBram Moolenaar else 91071d4279SBram Moolenaar let plindent = indent(plnum) 92071d4279SBram Moolenaar let plnumstart = plnum 93071d4279SBram Moolenaar endif 94071d4279SBram Moolenaar 95071d4279SBram Moolenaar " When inside parenthesis: If at the first line below the parenthesis add 96071d4279SBram Moolenaar " two 'shiftwidth', otherwise same as previous line. 97071d4279SBram Moolenaar " i = (a 98071d4279SBram Moolenaar " + b 99071d4279SBram Moolenaar " + c) 100071d4279SBram Moolenaar call cursor(a:lnum, 1) 101910f66f9SBram Moolenaar let p = searchpair('(\|{\|\[', '', ')\|}\|\]', 'bW', 102071d4279SBram Moolenaar \ "line('.') < " . (a:lnum - s:maxoff) . " ? dummy :" 103071d4279SBram Moolenaar \ . " synIDattr(synID(line('.'), col('.'), 1), 'name')" 1042c64ca18SBram Moolenaar \ . " =~ '\\(Comment\\|Todo\\|String\\)$'", 1052c64ca18SBram Moolenaar \ searchpair_stopline, searchpair_timeout) 106071d4279SBram Moolenaar if p > 0 107071d4279SBram Moolenaar if p == plnum 108071d4279SBram Moolenaar " When the start is inside parenthesis, only indent one 'shiftwidth'. 109910f66f9SBram Moolenaar let pp = searchpair('(\|{\|\[', '', ')\|}\|\]', 'bW', 110071d4279SBram Moolenaar \ "line('.') < " . (a:lnum - s:maxoff) . " ? dummy :" 111071d4279SBram Moolenaar \ . " synIDattr(synID(line('.'), col('.'), 1), 'name')" 1122c64ca18SBram Moolenaar \ . " =~ '\\(Comment\\|Todo\\|String\\)$'", 1132c64ca18SBram Moolenaar \ searchpair_stopline, searchpair_timeout) 114071d4279SBram Moolenaar if pp > 0 11556b45b9bSBram Moolenaar return indent(plnum) + (exists("g:pyindent_nested_paren") ? eval(g:pyindent_nested_paren) : shiftwidth()) 116071d4279SBram Moolenaar endif 11756b45b9bSBram Moolenaar return indent(plnum) + (exists("g:pyindent_open_paren") ? eval(g:pyindent_open_paren) : (shiftwidth() * 2)) 118071d4279SBram Moolenaar endif 119071d4279SBram Moolenaar if plnumstart == p 120071d4279SBram Moolenaar return indent(plnum) 121071d4279SBram Moolenaar endif 122071d4279SBram Moolenaar return plindent 123071d4279SBram Moolenaar endif 124071d4279SBram Moolenaar 125f6b40109SBram Moolenaar endif 126f6b40109SBram Moolenaar 127071d4279SBram Moolenaar 128071d4279SBram Moolenaar " Get the line and remove a trailing comment. 129071d4279SBram Moolenaar " Use syntax highlighting attributes when possible. 130071d4279SBram Moolenaar let pline = getline(plnum) 131071d4279SBram Moolenaar let pline_len = strlen(pline) 1325eb86f91SBram Moolenaar if has('syntax_items') 1335eb86f91SBram Moolenaar " If the last character in the line is a comment, do a binary search for 1345eb86f91SBram Moolenaar " the start of the comment. synID() is slow, a linear search would take 1355eb86f91SBram Moolenaar " too long on a long line. 1369ba7e17dSBram Moolenaar if synIDattr(synID(plnum, pline_len, 1), "name") =~ "\\(Comment\\|Todo\\)$" 1375eb86f91SBram Moolenaar let min = 1 1385eb86f91SBram Moolenaar let max = pline_len 1395eb86f91SBram Moolenaar while min < max 1405eb86f91SBram Moolenaar let col = (min + max) / 2 1419ba7e17dSBram Moolenaar if synIDattr(synID(plnum, col, 1), "name") =~ "\\(Comment\\|Todo\\)$" 1425eb86f91SBram Moolenaar let max = col 1435eb86f91SBram Moolenaar else 1445eb86f91SBram Moolenaar let min = col + 1 1455eb86f91SBram Moolenaar endif 1465eb86f91SBram Moolenaar endwhile 1475eb86f91SBram Moolenaar let pline = strpart(pline, 0, min - 1) 1485eb86f91SBram Moolenaar endif 1495eb86f91SBram Moolenaar else 150071d4279SBram Moolenaar let col = 0 151071d4279SBram Moolenaar while col < pline_len 1525eb86f91SBram Moolenaar if pline[col] == '#' 153071d4279SBram Moolenaar let pline = strpart(pline, 0, col) 154071d4279SBram Moolenaar break 155071d4279SBram Moolenaar endif 156071d4279SBram Moolenaar let col = col + 1 157071d4279SBram Moolenaar endwhile 1585eb86f91SBram Moolenaar endif 159071d4279SBram Moolenaar 160071d4279SBram Moolenaar " If the previous line ended with a colon, indent this line 161071d4279SBram Moolenaar if pline =~ ':\s*$' 16256b45b9bSBram Moolenaar return plindent + shiftwidth() 163071d4279SBram Moolenaar endif 164071d4279SBram Moolenaar 165071d4279SBram Moolenaar " If the previous line was a stop-execution statement... 1669964e468SBram Moolenaar if getline(plnum) =~ '^\s*\(break\|continue\|raise\|return\|pass\)\>' 167071d4279SBram Moolenaar " See if the user has already dedented 16847e13953SBram Moolenaar if s:Dedented(a:lnum, indent(plnum)) 16947e13953SBram Moolenaar " If so, trust the user 17047e13953SBram Moolenaar return -1 17147e13953SBram Moolenaar endif 172071d4279SBram Moolenaar " If not, recommend one dedent 17356b45b9bSBram Moolenaar return indent(plnum) - shiftwidth() 174071d4279SBram Moolenaar endif 175071d4279SBram Moolenaar 176071d4279SBram Moolenaar " If the current line begins with a keyword that lines up with "try" 177071d4279SBram Moolenaar if getline(a:lnum) =~ '^\s*\(except\|finally\)\>' 178071d4279SBram Moolenaar let lnum = a:lnum - 1 179071d4279SBram Moolenaar while lnum >= 1 180071d4279SBram Moolenaar if getline(lnum) =~ '^\s*\(try\|except\)\>' 181071d4279SBram Moolenaar let ind = indent(lnum) 182071d4279SBram Moolenaar if ind >= indent(a:lnum) 183071d4279SBram Moolenaar return -1 " indent is already less than this 184071d4279SBram Moolenaar endif 185071d4279SBram Moolenaar return ind " line up with previous try or except 186071d4279SBram Moolenaar endif 187071d4279SBram Moolenaar let lnum = lnum - 1 188071d4279SBram Moolenaar endwhile 189071d4279SBram Moolenaar return -1 " no matching "try"! 190071d4279SBram Moolenaar endif 191071d4279SBram Moolenaar 192071d4279SBram Moolenaar " If the current line begins with a header keyword, dedent 193071d4279SBram Moolenaar if getline(a:lnum) =~ '^\s*\(elif\|else\)\>' 194071d4279SBram Moolenaar 195071d4279SBram Moolenaar " Unless the previous line was a one-liner 196d2ea7cf1SBram Moolenaar if getline(plnumstart) =~ '^\s*\(for\|if\|elif\|try\)\>' 197071d4279SBram Moolenaar return plindent 198071d4279SBram Moolenaar endif 199071d4279SBram Moolenaar 200071d4279SBram Moolenaar " Or the user has already dedented 20147e13953SBram Moolenaar if s:Dedented(a:lnum, plindent) 202071d4279SBram Moolenaar return -1 203071d4279SBram Moolenaar endif 204071d4279SBram Moolenaar 20556b45b9bSBram Moolenaar return plindent - shiftwidth() 206071d4279SBram Moolenaar endif 207071d4279SBram Moolenaar 208071d4279SBram Moolenaar " When after a () construct we probably want to go back to the start line. 209071d4279SBram Moolenaar " a = (b 210071d4279SBram Moolenaar " + c) 211071d4279SBram Moolenaar " here 212071d4279SBram Moolenaar if parlnum > 0 21347e13953SBram Moolenaar " ...unless the user has already dedented 21447e13953SBram Moolenaar if s:Dedented(a:lnum, plindent) 21547e13953SBram Moolenaar return -1 21647e13953SBram Moolenaar else 217071d4279SBram Moolenaar return plindent 218071d4279SBram Moolenaar endif 21947e13953SBram Moolenaar endif 220071d4279SBram Moolenaar 221071d4279SBram Moolenaar return -1 222071d4279SBram Moolenaar 223071d4279SBram Moolenaarendfunction 224071d4279SBram Moolenaar 2259a7224b5SBram Moolenaarlet &cpo = s:keepcpo 2269a7224b5SBram Moolenaarunlet s:keepcpo 2279a7224b5SBram Moolenaar 228071d4279SBram Moolenaar" vim:sw=2 229