blob: 307b7f656b8204ddc7b8d53fcabb76a49d3a591d [file] [log] [blame]
Bram Moolenaar071d4272004-06-13 20:20:40 +00001" Vim indent file
Bram Moolenaar4399ef42005-02-12 14:29:27 +00002" Language: Python
3" Maintainer: Bram Moolenaar <Bram@vim.org>
Bram Moolenaar071d4272004-06-13 20:20:40 +00004" Original Author: David Bustos <bustos@caltech.edu>
Bram Moolenaard2ea7cf2021-05-30 20:54:13 +02005" Last Change: 2021 May 26
Bram Moolenaar071d4272004-06-13 20:20:40 +00006
7" Only load this indent file when no other was loaded.
8if exists("b:did_indent")
9 finish
10endif
11let b:did_indent = 1
12
13" Some preliminary settings
14setlocal nolisp " Make sure lisp indenting doesn't supersede us
15setlocal autoindent " indentexpr isn't much help otherwise
16
17setlocal indentexpr=GetPythonIndent(v:lnum)
18setlocal indentkeys+=<:>,=elif,=except
19
20" Only define the function once.
21if exists("*GetPythonIndent")
22 finish
23endif
Bram Moolenaar80716072012-05-01 21:14:34 +020024let s:keepcpo= &cpo
25set cpo&vim
Bram Moolenaar071d4272004-06-13 20:20:40 +000026
Bram Moolenaar05159a02005-02-26 23:04:13 +000027" Come here when loading the script the first time.
28
Bram Moolenaar071d4272004-06-13 20:20:40 +000029let s:maxoff = 50 " maximum number of lines to look backwards for ()
30
Bram Moolenaar47e13952020-05-12 22:49:12 +020031" See if the specified line is already user-dedented from the expected value.
32function s:Dedented(lnum, expected)
33 return indent(a:lnum) <= a:expected - shiftwidth()
34endfunction
35
Bram Moolenaar071d4272004-06-13 20:20:40 +000036function GetPythonIndent(lnum)
Bram Moolenaarc9b4b052006-04-30 18:54:39 +000037
Bram Moolenaar071d4272004-06-13 20:20:40 +000038 " If this line is explicitly joined: If the previous line was also joined,
39 " line it up with that one, otherwise add two 'shiftwidth'
40 if getline(a:lnum - 1) =~ '\\$'
41 if a:lnum > 1 && getline(a:lnum - 2) =~ '\\$'
42 return indent(a:lnum - 1)
43 endif
Bram Moolenaar56b45b92013-06-24 22:22:18 +020044 return indent(a:lnum - 1) + (exists("g:pyindent_continue") ? eval(g:pyindent_continue) : (shiftwidth() * 2))
Bram Moolenaar071d4272004-06-13 20:20:40 +000045 endif
46
47 " If the start of the line is in a string don't change the indent.
48 if has('syntax_items')
Bram Moolenaared203462004-06-16 11:19:22 +000049 \ && synIDattr(synID(a:lnum, 1, 1), "name") =~ "String$"
Bram Moolenaar071d4272004-06-13 20:20:40 +000050 return -1
51 endif
52
53 " Search backwards for the previous non-empty line.
54 let plnum = prevnonblank(v:lnum - 1)
55
56 if plnum == 0
57 " This is the first non-empty line, use zero indent.
58 return 0
59 endif
60
Bram Moolenaar071d4272004-06-13 20:20:40 +000061 call cursor(plnum, 1)
Bram Moolenaarf6b40102019-02-22 15:24:03 +010062
63 " Identing inside parentheses can be very slow, regardless of the searchpair()
64 " timeout, so let the user disable this feature if he doesn't need it
65 let disable_parentheses_indenting = get(g:, "pyindent_disable_parentheses_indenting", 0)
66
67 if disable_parentheses_indenting == 1
Bram Moolenaar071d4272004-06-13 20:20:40 +000068 let plindent = indent(plnum)
69 let plnumstart = plnum
Bram Moolenaarf6b40102019-02-22 15:24:03 +010070 else
71 " searchpair() can be slow sometimes, limit the time to 150 msec or what is
72 " put in g:pyindent_searchpair_timeout
73 let searchpair_stopline = 0
74 let searchpair_timeout = get(g:, 'pyindent_searchpair_timeout', 150)
Bram Moolenaar071d4272004-06-13 20:20:40 +000075
Bram Moolenaarf6b40102019-02-22 15:24:03 +010076 " If the previous line is inside parenthesis, use the indent of the starting
77 " line.
78 " Trick: use the non-existing "dummy" variable to break out of the loop when
79 " going too far back.
80 let parlnum = searchpair('(\|{\|\[', '', ')\|}\|\]', 'nbW',
81 \ "line('.') < " . (plnum - s:maxoff) . " ? dummy :"
82 \ . " synIDattr(synID(line('.'), col('.'), 1), 'name')"
83 \ . " =~ '\\(Comment\\|Todo\\|String\\)$'",
84 \ searchpair_stopline, searchpair_timeout)
85 if parlnum > 0
86 let plindent = indent(parlnum)
87 let plnumstart = parlnum
88 else
89 let plindent = indent(plnum)
90 let plnumstart = plnum
91 endif
Bram Moolenaar071d4272004-06-13 20:20:40 +000092
Bram Moolenaarf6b40102019-02-22 15:24:03 +010093 " When inside parenthesis: If at the first line below the parenthesis add
94 " two 'shiftwidth', otherwise same as previous line.
95 " i = (a
96 " + b
97 " + c)
98 call cursor(a:lnum, 1)
99 let p = searchpair('(\|{\|\[', '', ')\|}\|\]', 'bW',
100 \ "line('.') < " . (a:lnum - s:maxoff) . " ? dummy :"
101 \ . " synIDattr(synID(line('.'), col('.'), 1), 'name')"
102 \ . " =~ '\\(Comment\\|Todo\\|String\\)$'",
103 \ searchpair_stopline, searchpair_timeout)
104 if p > 0
105 if p == plnum
106 " When the start is inside parenthesis, only indent one 'shiftwidth'.
107 let pp = searchpair('(\|{\|\[', '', ')\|}\|\]', 'bW',
108 \ "line('.') < " . (a:lnum - s:maxoff) . " ? dummy :"
109 \ . " synIDattr(synID(line('.'), col('.'), 1), 'name')"
110 \ . " =~ '\\(Comment\\|Todo\\|String\\)$'",
111 \ searchpair_stopline, searchpair_timeout)
112 if pp > 0
113 return indent(plnum) + (exists("g:pyindent_nested_paren") ? eval(g:pyindent_nested_paren) : shiftwidth())
114 endif
115 return indent(plnum) + (exists("g:pyindent_open_paren") ? eval(g:pyindent_open_paren) : (shiftwidth() * 2))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000116 endif
Bram Moolenaarf6b40102019-02-22 15:24:03 +0100117 if plnumstart == p
118 return indent(plnum)
119 endif
120 return plindent
Bram Moolenaar071d4272004-06-13 20:20:40 +0000121 endif
Bram Moolenaarf6b40102019-02-22 15:24:03 +0100122
Bram Moolenaar071d4272004-06-13 20:20:40 +0000123 endif
124
125
126 " Get the line and remove a trailing comment.
127 " Use syntax highlighting attributes when possible.
128 let pline = getline(plnum)
129 let pline_len = strlen(pline)
Bram Moolenaar5eb86f92004-07-26 12:53:41 +0000130 if has('syntax_items')
131 " If the last character in the line is a comment, do a binary search for
132 " the start of the comment. synID() is slow, a linear search would take
133 " too long on a long line.
Bram Moolenaar9ba7e172013-07-17 22:37:26 +0200134 if synIDattr(synID(plnum, pline_len, 1), "name") =~ "\\(Comment\\|Todo\\)$"
Bram Moolenaar5eb86f92004-07-26 12:53:41 +0000135 let min = 1
136 let max = pline_len
137 while min < max
138 let col = (min + max) / 2
Bram Moolenaar9ba7e172013-07-17 22:37:26 +0200139 if synIDattr(synID(plnum, col, 1), "name") =~ "\\(Comment\\|Todo\\)$"
Bram Moolenaar5eb86f92004-07-26 12:53:41 +0000140 let max = col
141 else
142 let min = col + 1
143 endif
144 endwhile
Bram Moolenaar5eb86f92004-07-26 12:53:41 +0000145 let pline = strpart(pline, 0, min - 1)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000146 endif
Bram Moolenaar5eb86f92004-07-26 12:53:41 +0000147 else
148 let col = 0
149 while col < pline_len
150 if pline[col] == '#'
151 let pline = strpart(pline, 0, col)
152 break
153 endif
154 let col = col + 1
155 endwhile
156 endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000157
158 " If the previous line ended with a colon, indent this line
159 if pline =~ ':\s*$'
Bram Moolenaar56b45b92013-06-24 22:22:18 +0200160 return plindent + shiftwidth()
Bram Moolenaar071d4272004-06-13 20:20:40 +0000161 endif
162
163 " If the previous line was a stop-execution statement...
Bram Moolenaar9964e462007-05-05 17:54:07 +0000164 if getline(plnum) =~ '^\s*\(break\|continue\|raise\|return\|pass\)\>'
Bram Moolenaar071d4272004-06-13 20:20:40 +0000165 " See if the user has already dedented
Bram Moolenaar47e13952020-05-12 22:49:12 +0200166 if s:Dedented(a:lnum, indent(plnum))
167 " If so, trust the user
168 return -1
Bram Moolenaar071d4272004-06-13 20:20:40 +0000169 endif
Bram Moolenaar47e13952020-05-12 22:49:12 +0200170 " If not, recommend one dedent
171 return indent(plnum) - shiftwidth()
Bram Moolenaar071d4272004-06-13 20:20:40 +0000172 endif
173
174 " If the current line begins with a keyword that lines up with "try"
175 if getline(a:lnum) =~ '^\s*\(except\|finally\)\>'
176 let lnum = a:lnum - 1
177 while lnum >= 1
Bram Moolenaar071d4272004-06-13 20:20:40 +0000178 if getline(lnum) =~ '^\s*\(try\|except\)\>'
179 let ind = indent(lnum)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000180 if ind >= indent(a:lnum)
181 return -1 " indent is already less than this
182 endif
183 return ind " line up with previous try or except
184 endif
185 let lnum = lnum - 1
186 endwhile
Bram Moolenaar071d4272004-06-13 20:20:40 +0000187 return -1 " no matching "try"!
188 endif
189
190 " If the current line begins with a header keyword, dedent
191 if getline(a:lnum) =~ '^\s*\(elif\|else\)\>'
192
193 " Unless the previous line was a one-liner
Bram Moolenaard2ea7cf2021-05-30 20:54:13 +0200194 if getline(plnumstart) =~ '^\s*\(for\|if\|elif\|try\)\>'
Bram Moolenaar071d4272004-06-13 20:20:40 +0000195 return plindent
196 endif
197
198 " Or the user has already dedented
Bram Moolenaar47e13952020-05-12 22:49:12 +0200199 if s:Dedented(a:lnum, plindent)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000200 return -1
201 endif
202
Bram Moolenaar56b45b92013-06-24 22:22:18 +0200203 return plindent - shiftwidth()
Bram Moolenaar071d4272004-06-13 20:20:40 +0000204 endif
205
206 " When after a () construct we probably want to go back to the start line.
207 " a = (b
208 " + c)
209 " here
210 if parlnum > 0
Bram Moolenaar47e13952020-05-12 22:49:12 +0200211 " ...unless the user has already dedented
212 if s:Dedented(a:lnum, plindent)
213 return -1
214 else
215 return plindent
216 endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000217 endif
218
219 return -1
220
221endfunction
222
Bram Moolenaar9a7224b2012-04-30 15:56:52 +0200223let &cpo = s:keepcpo
224unlet s:keepcpo
225
Bram Moolenaar071d4272004-06-13 20:20:40 +0000226" vim:sw=2