2
2
" Language: Javascript
3
3
" Maintainer: vim-javascript community
4
4
" URL: https://github.com/pangloss/vim-javascript
5
- " Last Change: August 20 , 2016
5
+ " Last Change: September 4 , 2016
6
6
7
7
" Only load this indent file when no other was loaded.
8
8
if exists (' b:did_indent' )
38
38
endif
39
39
40
40
let s: line_pre = ' ^\s*\%(\%(\%(\/\*.\{-}\)\=\*\+\/\s*\)\=\)\@>'
41
- let s: expr_case = s: line_pre . ' \%(\%(case\>.\+\)\|default\)\s*:'
41
+ let s: expr_case = s: line_pre . ' \%(\%(case\>.\+\)\|default\)\s*:\C '
42
42
" Regex of syntax group names that are or delimit string or are comments.
43
43
let s: syng_strcom = ' \%(s\%(tring\|pecial\)\|comment\|regex\|doc\|template\)'
44
44
@@ -47,13 +47,20 @@ let s:syng_comment = '\%(comment\|doc\)'
47
47
48
48
" Expression used to check whether we should skip a match with searchpair().
49
49
let s: skip_expr = " synIDattr(synID(line('.'),col('.'),0),'name') =~? '" .s: syng_strcom ." '"
50
+ function s: skip_func (lnum)
51
+ if ! s: free || getline (line (' .' )) = ~ ' ['' /"\\]' || search (' `' ,' nW' ,a: lnum ) || search (' \*\/' ,' nW' ,a: lnum )
52
+ let s: free = ! eval (s: skip_expr )
53
+ endif
54
+ let s: looksyn = s: free ? line (' .' ) : s: looksyn
55
+ return ! s: free
56
+ endfunction
50
57
51
58
if has (' reltime' )
52
- function s: GetPair (start ,end ,flags,time)
53
- return searchpair (a: start ,' ' ,a: end ,a: flags ,s: skip_expr ,max ([prevnonblank (v: lnum ) - 2000 ,0 ]),a: time )
59
+ function s: GetPair (start ,end ,flags,skip , time)
60
+ return searchpair (a: start ,' ' ,a: end ,a: flags ,a: skip ,max ([prevnonblank (v: lnum ) - 2000 ,0 ]),a: time )
54
61
endfunction
55
62
else
56
- function s: GetPair (start ,end ,flags,n )
63
+ function s: GetPair (start ,end ,flags,... )
57
64
return searchpair (a: start ,' ' ,a: end ,a: flags ,0 ,max ([prevnonblank (v: lnum ) - 2000 ,0 ]))
58
65
endfunction
59
66
endif
@@ -62,29 +69,28 @@ let s:line_term = '\s*\%(\%(\/\%(\%(\*.\{-}\*\/\)\|\%(\*\+\)\)\)\s*\)\=$'
62
69
63
70
" configurable regexes that define continuation lines, not including (, {, or [.
64
71
if ! exists (' g:javascript_opfirst' )
65
- let g: javascript_opfirst = ' \%([<>,: ?^%|*&]\|\([-/. +]\)\1\@!\|=>\@!\|in\%(stanceof\)\=\>\)'
72
+ let g: javascript_opfirst = ' \%([<>,?^%|*&]\|\/[^/*]\|\ ([-.: +]\)\1\@!\|=>\@!\|in\%(stanceof\)\=\>\)'
66
73
endif
67
74
if ! exists (' g:javascript_continuation' )
68
- let g: javascript_continuation = ' \%([<=,.?/*: ^%|&]\|+\@<!+\|-\@<!-\|=\@<!>\|\<in\%(stanceof\)\=\)'
75
+ let g: javascript_continuation = ' \%([<=,.?/*^%|&: ]\|+\@<!+\|-\@<!-\|=\@<!>\|\<in\%(stanceof\)\=\)'
69
76
endif
70
77
71
78
let g: javascript_opfirst = s: line_pre . g: javascript_opfirst
72
79
let g: javascript_continuation .= s: line_term
73
80
74
- function s: OneScope (lnum,text, add )
81
+ function s: OneScope (lnum,text)
75
82
return a: text = ~# ' \%(\<else\|\<do\|=>\)' . s: line_term ? ' no b' :
76
- \ ((a: add && a: text = ~ s: line_pre . ' $' && search (' \%' . s: PrevCodeLine (a: lnum - 1 ) . ' l.)' . s: line_term )) ||
77
- \ cursor (a: lnum , match (a: text , ' )' . s: line_term )) > -1 ) &&
78
- \ s: GetPair (' (' , ' )' , ' cbW' , 100 ) > 0 && search (' \C\l\+\_s*\%#' ,' bW' ) &&
79
- \ (a: add || ((expand (' <cword>' ) !=# ' while' || ! s: GetPair (' \C\<do\>' , ' \C\<while\>' ,' nbW' ,100 )) &&
80
- \ expand (' cword' ) !=# ' each' || search (' \C\<for\_s\+\%#' ,' nbW' ))) ? expand (' <cword>' ) : ' '
83
+ \ cursor (a: lnum , match (' ' . a: text , ' )' . s: line_term )) > -1 &&
84
+ \ s: GetPair (' (' , ' )' , ' bW' , s: skip_expr , 100 ) > 0 && search (' \C\l\+\_s*\%#' ,' bW' ) &&
85
+ \ (expand (' <cword>' ) !=# ' while' || s: GetPair (' \C\<do\>' , ' \C\<while\>' ,' nbW' ,s: skip_expr ,100 ) <= 0 ) &&
86
+ \ (expand (' <cword>' ) !=# ' each' || search (' \C\<for\_s\+\%#' ,' nbW' )) ? expand (' <cword>' ) : ' '
81
87
endfunction
82
88
83
89
" https://github.com/sweet-js/sweet.js/wiki/design#give-lookbehind-to-the-reader
84
90
function s: IsBlock ()
85
91
return getline (line (' .' ))[col (' .' )-1 ] == ' {' && ! search (
86
- \ ' \C\%(\<return\s*\|\%([-=~!<*+,.?^%|&\[(]\|=\@<!>\|\*\@<!\/\|\<\%(var\|const\|let\|yield\|delete\|void\|t\%(ypeof\|hrow\)\|new\|\< in\%(stanceof\)\=\)\)\_s*\)\%#' ,' bnW' ) &&
87
- \ (! search (' : \_s*\%#' ,' bW ' ) || ( ! s: GetPair (' [({[] ' ,' [])}] ' , ' bW' , 200 ) || s: IsBlock () ))
92
+ \ ' \C\%(\<return\s*\|\%([-=~!<*+,.?^%|&\[(]\|=\@<!>\|\*\@<!\/\|\<\%(var\|const\|let\|import\|export\%(\_s\+default\)\=\| yield\|delete\|void\|t\%(ypeof\|hrow\)\|new\|in\%(stanceof\)\=\)\)\_s*\)\%#' ,' bnW' ) &&
93
+ \ (search (s: expr_case . ' \_s*\%#' ,' nbW ' ) || ! search (' [{:]\_s*\%# ' ,' bW' ) || s: IsBlock ())
88
94
endfunction
89
95
90
96
" Auxiliary Functions {{{2
@@ -109,9 +115,12 @@ function s:Balanced(lnum)
109
115
if synIDattr (synID (a: lnum ,pos + 1 ,0 ),' name' ) !~? s: syng_strcom
110
116
let idx = stridx (' (){}[]' , l: line [pos])
111
117
if idx % 2 == 0
112
- let open_{idx} = open_{idx} + 1
118
+ let open_{idx} += 1
113
119
else
114
- let open_{idx - 1 } = open_{idx - 1 } - 1
120
+ let open_{idx - 1 } -= 1
121
+ if open_{idx - 1 } < 0
122
+ return 0
123
+ endif
115
124
endif
116
125
endif
117
126
let pos = match (l: line , ' [][(){}]' , pos + 1 )
@@ -129,7 +138,7 @@ function GetJavascriptIndent()
129
138
let syns = synIDattr (synID (v: lnum , 1 , 0 ), ' name' )
130
139
131
140
" start with strings,comments,etc.{{{2
132
- if (l: line !~ ' ^['' "` ]' && syns = ~? ' \%(string\|template\)' ) ||
141
+ if (l: line !~ ' ^['' "]' && syns = ~? ' \%(string\|template\)' ) ||
133
142
\ (l: line !~ ' ^\s*[/*]' && syns = ~? s: syng_comment )
134
143
return -1
135
144
endif
@@ -152,36 +161,37 @@ function GetJavascriptIndent()
152
161
153
162
" the containing paren, bracket, curly. Memoize, last lineNr either has the
154
163
" same scope or starts a new one, unless if it closed a scope.
164
+ let [s: looksyn ,s: free ] = [v: lnum - 1 ,1 ]
155
165
call cursor (v: lnum ,1 )
156
- if b: js_cache [0 ] >= l : lnum && b: js_cache [0 ] < v : lnum && b: js_cache [ 0 ] &&
166
+ if b: js_cache [0 ] < v : lnum && b: js_cache [0 ] >= l : lnum &&
157
167
\ (b: js_cache [0 ] > l: lnum || s: Balanced (l: lnum ) > 0 )
158
168
let num = b: js_cache [1 ]
159
169
elseif syns != ' ' && l: line [0 ] = ~ ' \s'
160
170
let pattern = syns = ~? ' block' ? [' {' ,' }' ] : syns = ~? ' jsparen' ? [' (' ,' )' ] :
161
171
\ syns = ~? ' jsbracket' ? [' \[' ,' \]' ] : [' [({[]' ,' [])}]' ]
162
- let num = s: GetPair (pattern[0 ],pattern[1 ],' bW' ,2000 )
172
+ let num = s: GetPair (pattern[0 ],pattern[1 ],' bW' ,' s:skip_func(s:looksyn) ' , 2000 )
163
173
else
164
- let num = s: GetPair (' [({[]' ,' [])}]' ,' bW' ,2000 )
174
+ let num = s: GetPair (' [({[]' ,' [])}]' ,' bW' ,' s:skip_func(s:looksyn) ' , 2000 )
165
175
endif
176
+
166
177
let b: js_cache = [v: lnum ,num,line (' .' ) == v: lnum ? b: js_cache [2 ] : col (' .' )]
167
178
168
179
if l: line = ~ s: line_pre . ' [])}]'
169
180
return indent (num)
170
181
endif
171
182
183
+ let pline = substitute (substitute (getline (l: lnum ),s: expr_case ,' \=repeat(" ",strlen(submatch(0)))' ,' ' ), ' \%(:\@<!\/\/.*\)$' , ' ' ,' ' )
172
184
call cursor (b: js_cache [1 ],b: js_cache [2 ])
173
-
174
- let swcase = getline (l: lnum ) = ~# s: expr_case
175
- let pline = swcase ? getline (l: lnum ) : substitute (getline (l: lnum ), ' \%(:\@<!\/\/.*\)$' , ' ' ,' ' )
176
- let inb = num == 0 || num < l: lnum && ((l: line !~ s: line_pre . ' ,' && pline !~ ' ,' . s: line_term ) || s: IsBlock ())
177
- let switch_offset = num == 0 || s: OneScope (num, strpart (getline (num),0 ,b: js_cache [2 ] - 1 ),1 ) !=# ' switch' ? 0 :
178
- \ &cino !~ ' :' || ! has (' float' ) ? s: sw () :
185
+ let switch_offset = num <= 0 || ! (search (' )\_s*\%#' ,' bW' ) &&
186
+ \ s: GetPair (' (' , ' )' , ' bW' , s: skip_expr , 100 ) > 0 && search (' \C\<switch\_s*\%#' ,' bW' )) ? 0 :
187
+ \ &cino !~ ' :' || ! has (' float' ) ? s: sw () :
179
188
\ float2nr (str2float (matchstr (&cino ,' .*:\zs[-0-9.]*' )) * (&cino = ~# ' .*:[^,]*s' ? s: sw () : 1 ))
180
189
181
190
" most significant, find the indent amount
182
- if inb && ! swcase && ((l: line = ~# g: javascript_opfirst || pline = ~# g: javascript_continuation ) ||
183
- \ num < l: lnum && s: OneScope (l: lnum ,pline,0 ) = ~# ' \<\%(for\|each\|if\|let\|no\sb\|w\%(hile\|ith\)\)\>' &&
184
- \ l: line !~ s: line_pre . ' {' )
191
+ let isOp = l: line = ~# g: javascript_opfirst || pline = ~# g: javascript_continuation
192
+ if isOp && (num <= 0 || cursor (b: js_cache [1 ],b: js_cache [2 ]) || s: IsBlock ()) ||
193
+ \ s: OneScope (l: lnum ,pline) = ~# ' \<\%(for\|each\|if\|let\|no\sb\|w\%(hile\|ith\)\)\>' &&
194
+ \ l: line !~ s: line_pre . ' {'
185
195
return (num > 0 ? indent (num) : - s: sw ()) + (s: sw () * 2 ) + switch_offset
186
196
elseif num > 0
187
197
return indent (num) + s: sw () + switch_offset
0 commit comments