blob: 96d0f1abb95debcb70b57b5adcbb95b160b7b33a [file] [log] [blame]
Bram Moolenaarb8060fe2016-01-19 22:29:28 +01001" Test for syntax and syntax iskeyword option
2
Bram Moolenaarb46fecd2019-06-15 17:58:09 +02003source check.vim
4CheckFeature syntax
Bram Moolenaarb8060fe2016-01-19 22:29:28 +01005
Bram Moolenaar4d785892017-06-22 22:00:50 +02006source view_util.vim
Bram Moolenaar6bb2cdf2018-02-24 19:53:53 +01007source screendump.vim
Bram Moolenaar4d785892017-06-22 22:00:50 +02008
Bram Moolenaarb8060fe2016-01-19 22:29:28 +01009func GetSyntaxItem(pat)
10 let c = ''
11 let a = ['a', getreg('a'), getregtype('a')]
12 0
13 redraw!
14 call search(a:pat, 'W')
15 let synid = synID(line('.'), col('.'), 1)
16 while synid == synID(line('.'), col('.'), 1)
17 norm! v"ay
18 " stop at whitespace
19 if @a =~# '\s'
20 break
21 endif
22 let c .= @a
23 norm! l
24 endw
25 call call('setreg', a)
26 0
27 return c
28endfunc
29
30func Test_syn_iskeyword()
31 new
32 call setline(1, [
33 \ 'CREATE TABLE FOOBAR(',
34 \ ' DLTD_BY VARCHAR2(100)',
35 \ ');',
Bram Moolenaar037c54f2019-04-20 23:47:46 +020036 \ ''])
Bram Moolenaarb8060fe2016-01-19 22:29:28 +010037
38 syntax on
39 set ft=sql
40 syn match SYN /C\k\+\>/
41 hi link SYN ErrorMsg
42 call assert_equal('DLTD_BY', GetSyntaxItem('DLTD'))
43 /\<D\k\+\>/:norm! ygn
44 call assert_equal('DLTD_BY', @0)
45 redir @c
46 syn iskeyword
47 redir END
48 call assert_equal("\nsyntax iskeyword not set", @c)
49
50 syn iskeyword @,48-57,_,192-255
51 redir @c
52 syn iskeyword
53 redir END
54 call assert_equal("\nsyntax iskeyword @,48-57,_,192-255", @c)
55
56 setlocal isk-=_
57 call assert_equal('DLTD_BY', GetSyntaxItem('DLTD'))
58 /\<D\k\+\>/:norm! ygn
Bram Moolenaar73b484c2016-12-11 15:11:17 +010059 let b2 = @0
Bram Moolenaarb8060fe2016-01-19 22:29:28 +010060 call assert_equal('DLTD', @0)
61
62 syn iskeyword clear
63 redir @c
64 syn iskeyword
65 redir END
66 call assert_equal("\nsyntax iskeyword not set", @c)
67
68 quit!
69endfunc
Bram Moolenaarc3691332016-04-20 12:49:49 +020070
71func Test_syntax_after_reload()
72 split Xsomefile
73 call setline(1, ['hello', 'there'])
74 w!
75 only!
76 setl filetype=hello
77 au FileType hello let g:gotit = 1
78 call assert_false(exists('g:gotit'))
79 edit other
80 buf Xsomefile
81 call assert_equal('hello', &filetype)
82 call assert_true(exists('g:gotit'))
83 call delete('Xsomefile')
84endfunc
Bram Moolenaar73b484c2016-12-11 15:11:17 +010085
86func Test_syntime()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +020087 CheckFeature profile
Bram Moolenaar73b484c2016-12-11 15:11:17 +010088
89 syntax on
90 syntime on
91 let a = execute('syntime report')
92 call assert_equal("\nNo Syntax items defined for this buffer", a)
93
94 view ../memfile_test.c
95 setfiletype cpp
96 redraw
97 let a = execute('syntime report')
98 call assert_match('^ TOTAL *COUNT *MATCH *SLOWEST *AVERAGE *NAME *PATTERN', a)
99 call assert_match(' \d*\.\d* \+[^0]\d* .* cppRawString ', a)
100 call assert_match(' \d*\.\d* \+[^0]\d* .* cppNumber ', a)
101
102 syntime off
103 syntime clear
104 let a = execute('syntime report')
105 call assert_match('^ TOTAL *COUNT *MATCH *SLOWEST *AVERAGE *NAME *PATTERN', a)
106 call assert_notmatch('.* cppRawString *', a)
107 call assert_notmatch('.* cppNumber*', a)
108 call assert_notmatch('[1-9]', a)
109
110 call assert_fails('syntime abc', 'E475')
111
112 syntax clear
113 let a = execute('syntime report')
114 call assert_equal("\nNo Syntax items defined for this buffer", a)
115
116 bd
117endfunc
118
Bram Moolenaarb513d302018-12-02 14:55:08 +0100119func Test_syntime_completion()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +0200120 CheckFeature profile
Bram Moolenaarb513d302018-12-02 14:55:08 +0100121
122 call feedkeys(":syntime \<C-A>\<C-B>\"\<CR>", 'tx')
123 call assert_equal('"syntime clear off on report', @:)
124endfunc
125
Bram Moolenaar73b484c2016-12-11 15:11:17 +0100126func Test_syntax_list()
127 syntax on
128 let a = execute('syntax list')
129 call assert_equal("\nNo Syntax items defined for this buffer", a)
130
131 view ../memfile_test.c
132 setfiletype c
133
134 let a = execute('syntax list')
135 call assert_match('cInclude*', a)
136 call assert_match('cDefine', a)
137
138 let a = execute('syntax list cDefine')
139 call assert_notmatch('cInclude*', a)
140 call assert_match('cDefine', a)
141 call assert_match(' links to Macro$', a)
142
143 call assert_fails('syntax list ABCD', 'E28:')
144 call assert_fails('syntax list @ABCD', 'E392:')
145
146 syntax clear
147 let a = execute('syntax list')
148 call assert_equal("\nNo Syntax items defined for this buffer", a)
149
Bram Moolenaar08f41572020-04-20 16:50:00 +0200150 syntax keyword Type int containedin=g1 skipwhite skipempty skipnl nextgroup=Abc
151 let exp = "Type xxx containedin=g1 nextgroup=Abc skipnl skipwhite skipempty int"
152 call assert_equal(exp, split(execute("syntax list"), "\n")[1])
153
Bram Moolenaar73b484c2016-12-11 15:11:17 +0100154 bd
155endfunc
156
157func Test_syntax_completion()
158 call feedkeys(":syn \<C-A>\<C-B>\"\<CR>", 'tx')
Bram Moolenaare35a52a2020-05-31 19:48:53 +0200159 call assert_equal('"syn case clear cluster conceal enable foldlevel include iskeyword keyword list manual match off on region reset spell sync', @:)
Bram Moolenaar73b484c2016-12-11 15:11:17 +0100160
161 call feedkeys(":syn case \<C-A>\<C-B>\"\<CR>", 'tx')
162 call assert_equal('"syn case ignore match', @:)
163
Bram Moolenaar2d028392017-01-08 18:28:22 +0100164 call feedkeys(":syn spell \<C-A>\<C-B>\"\<CR>", 'tx')
165 call assert_equal('"syn spell default notoplevel toplevel', @:)
166
167 call feedkeys(":syn sync \<C-A>\<C-B>\"\<CR>", 'tx')
168 call assert_equal('"syn sync ccomment clear fromstart linebreaks= linecont lines= match maxlines= minlines= region', @:)
169
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100170 " Check that clearing "Aap" avoids it showing up before Boolean.
171 hi Aap ctermfg=blue
172 call feedkeys(":syn list \<C-A>\<C-B>\"\<CR>", 'tx')
173 call assert_match('^"syn list Aap Boolean Character ', @:)
174 hi clear Aap
175
Bram Moolenaar73b484c2016-12-11 15:11:17 +0100176 call feedkeys(":syn list \<C-A>\<C-B>\"\<CR>", 'tx')
177 call assert_match('^"syn list Boolean Character ', @:)
178
179 call feedkeys(":syn match \<C-A>\<C-B>\"\<CR>", 'tx')
180 call assert_match('^"syn match Boolean Character ', @:)
181endfunc
Bram Moolenaarde318c52017-01-17 16:27:10 +0100182
Bram Moolenaar297610b2019-12-27 17:20:55 +0100183func Test_echohl_completion()
184 call feedkeys(":echohl no\<C-A>\<C-B>\"\<CR>", 'tx')
185 call assert_equal('"echohl NonText Normal none', @:)
186endfunc
187
Bram Moolenaarde318c52017-01-17 16:27:10 +0100188func Test_syntax_arg_skipped()
189 syn clear
190 syntax case ignore
191 if 0
192 syntax case match
193 endif
194 call assert_match('case ignore', execute('syntax case'))
195
196 syn keyword Foo foo
197 call assert_match('Foo', execute('syntax'))
198 syn clear
199 call assert_match('case match', execute('syntax case'))
200 call assert_notmatch('Foo', execute('syntax'))
201
202 if has('conceal')
203 syn clear
204 syntax conceal on
205 if 0
206 syntax conceal off
207 endif
208 call assert_match('conceal on', execute('syntax conceal'))
209 syn clear
210 call assert_match('conceal off', execute('syntax conceal'))
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100211
212 syntax conceal on
213 syntax conceal off
214 call assert_match('conceal off', execute('syntax conceal'))
Bram Moolenaarde318c52017-01-17 16:27:10 +0100215 endif
216
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100217 syntax region Bar start=/</ end=/>/
Bram Moolenaarde318c52017-01-17 16:27:10 +0100218 if 0
219 syntax region NotTest start=/</ end=/>/ contains=@Spell
220 endif
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100221 call assert_match('Bar', execute('syntax'))
Bram Moolenaarde318c52017-01-17 16:27:10 +0100222 call assert_notmatch('NotTest', execute('syntax'))
223 call assert_notmatch('Spell', execute('syntax'))
224
225 hi Foo ctermfg=blue
226 let a = execute('hi Foo')
227 if 0
228 syntax rest
229 endif
230 call assert_equal(a, execute('hi Foo'))
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100231 hi clear Bar
232 hi clear Foo
Bram Moolenaarde318c52017-01-17 16:27:10 +0100233
234 set ft=tags
235 syn off
236 if 0
237 syntax enable
238 endif
239 call assert_match('No Syntax items defined', execute('syntax'))
240 syntax enable
241 call assert_match('tagComment', execute('syntax'))
242 set ft=
243
244 syn clear
245 if 0
246 syntax include @Spell nothing
247 endif
248 call assert_notmatch('Spell', execute('syntax'))
249
250 syn clear
251 syn iskeyword 48-57,$,_
252 call assert_match('48-57,$,_', execute('syntax iskeyword'))
253 if 0
254 syn clear
255 syn iskeyword clear
256 endif
257 call assert_match('48-57,$,_', execute('syntax iskeyword'))
258 syn iskeyword clear
259 call assert_match('not set', execute('syntax iskeyword'))
260 syn iskeyword 48-57,$,_
261 syn clear
262 call assert_match('not set', execute('syntax iskeyword'))
263
264 syn clear
265 syn keyword Foo foo
266 if 0
267 syn keyword NotAdded bar
268 endif
269 call assert_match('Foo', execute('syntax'))
270 call assert_notmatch('NotAdded', execute('highlight'))
271
272 syn clear
273 syn keyword Foo foo
274 call assert_match('Foo', execute('syntax'))
275 call assert_match('Foo', execute('syntax list'))
276 call assert_notmatch('Foo', execute('if 0 | syntax | endif'))
277 call assert_notmatch('Foo', execute('if 0 | syntax list | endif'))
278
279 syn clear
280 syn match Fopi /asdf/
281 if 0
282 syn match Fopx /asdf/
283 endif
284 call assert_match('Fopi', execute('syntax'))
285 call assert_notmatch('Fopx', execute('syntax'))
286
287 syn clear
288 syn spell toplevel
289 call assert_match('spell toplevel', execute('syntax spell'))
290 if 0
291 syn spell notoplevel
292 endif
293 call assert_match('spell toplevel', execute('syntax spell'))
294 syn spell notoplevel
295 call assert_match('spell notoplevel', execute('syntax spell'))
296 syn spell default
297 call assert_match('spell default', execute('syntax spell'))
298
299 syn clear
300 if 0
301 syntax cluster Spell
302 endif
303 call assert_notmatch('Spell', execute('syntax'))
304
305 syn clear
306 syn keyword Foo foo
307 syn sync ccomment
308 syn sync maxlines=5
309 if 0
310 syn sync maxlines=11
311 endif
312 call assert_match('on C-style comments', execute('syntax sync'))
313 call assert_match('maximal 5 lines', execute('syntax sync'))
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100314 syn sync clear
Bram Moolenaarde318c52017-01-17 16:27:10 +0100315 if 0
316 syn sync ccomment
317 endif
318 call assert_notmatch('on C-style comments', execute('syntax sync'))
319
320 syn clear
321endfunc
322
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200323" Check for an error. Used when multiple errors are thrown and we are checking
324" for an earliest error.
325func AssertFails(cmd, errcode)
326 let save_exception = ''
327 try
328 exe a:cmd
329 catch
330 let save_exception = v:exception
331 endtry
332 call assert_match(a:errcode, save_exception)
333endfunc
334
Bram Moolenaarea588152017-04-10 22:45:30 +0200335func Test_syntax_invalid_arg()
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100336 call assert_fails('syntax case asdf', 'E390:')
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100337 if has('conceal')
338 call assert_fails('syntax conceal asdf', 'E390:')
339 endif
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100340 call assert_fails('syntax spell asdf', 'E390:')
Bram Moolenaarea588152017-04-10 22:45:30 +0200341 call assert_fails('syntax clear @ABCD', 'E391:')
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200342 call assert_fails('syntax include random_file', 'E484:')
343 call assert_fails('syntax include <afile>', 'E495:')
Bram Moolenaarea588152017-04-10 22:45:30 +0200344 call assert_fails('syntax sync x', 'E404:')
345 call assert_fails('syntax keyword Abc a[', 'E789:')
346 call assert_fails('syntax keyword Abc a[bc]d', 'E890:')
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +0200347 call assert_fails('syntax cluster Abc add=A add=', 'E406:')
Bram Moolenaar476a6132020-04-08 19:48:56 +0200348
349 " Test for too many \z\( and unmatched \z\(
350 " Not able to use assert_fails() here because both E50:/E879: and E475:
351 " messages are emitted.
352 set regexpengine=1
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200353 call AssertFails("syntax region MyRegion start='\\z\\(' end='\\*/'", 'E52:')
Bram Moolenaar476a6132020-04-08 19:48:56 +0200354
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200355 let cmd = "syntax region MyRegion start='"
356 let cmd ..= repeat("\\z\\(.\\)", 10) .. "' end='\*/'"
357 call AssertFails(cmd, 'E50:')
Bram Moolenaar476a6132020-04-08 19:48:56 +0200358
359 set regexpengine=2
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200360 call AssertFails("syntax region MyRegion start='\\z\\(' end='\\*/'", 'E54:')
Bram Moolenaar476a6132020-04-08 19:48:56 +0200361
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200362 let cmd = "syntax region MyRegion start='"
363 let cmd ..= repeat("\\z\\(.\\)", 10) .. "' end='\*/'"
364 call AssertFails(cmd, 'E879:')
Bram Moolenaar476a6132020-04-08 19:48:56 +0200365 set regexpengine&
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200366
367 call AssertFails('syntax keyword cMyItem grouphere G1', 'E393:')
368 call AssertFails('syntax sync match Abc grouphere MyItem "abc"', 'E394:')
369 call AssertFails('syn keyword Type contains int', 'E395:')
370 call assert_fails('syntax include @Xxx', 'E397:')
371 call AssertFails('syntax region X start', 'E398:')
372 call assert_fails('syntax region X start="{"', 'E399:')
373 call AssertFails('syntax cluster contains=Abc', 'E400:')
374 call AssertFails("syntax match Character /'.'", 'E401:')
375 call AssertFails("syntax match Character /'.'/a", 'E402:')
376 call assert_fails('syntax sync linecont /pat', 'E404:')
377 call assert_fails('syntax sync linecont', 'E404:')
378 call assert_fails('syntax sync linecont /pat1/ linecont /pat2/', 'E403:')
379 call assert_fails('syntax sync minlines=a', 'E404:')
380 call AssertFails('syntax match ABC /x/ contains=', 'E406:')
381 call AssertFails("syntax match Character contains /'.'/", 'E405:')
382 call AssertFails('syntax match ccFoo "Foo" nextgroup=ALLBUT,F', 'E407:')
383 call AssertFails('syntax region Block start="{" contains=F,ALLBUT', 'E408:')
384 call AssertFails("syntax match Characters contains=a.*x /'.'/", 'E409:')
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100385endfunc
386
387func Test_syn_sync()
388 syntax region HereGroup start=/this/ end=/that/
389 syntax sync match SyncHere grouphere HereGroup "pattern"
390 call assert_match('SyncHere', execute('syntax sync'))
391 syn sync clear
392 call assert_notmatch('SyncHere', execute('syntax sync'))
393 syn clear
394endfunc
395
396func Test_syn_clear()
397 syntax keyword Foo foo
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100398 syntax keyword Bar tar
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100399 call assert_match('Foo', execute('syntax'))
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100400 call assert_match('Bar', execute('syntax'))
Bram Moolenaarc96272e2017-03-26 13:50:09 +0200401 call assert_equal('Foo', synIDattr(hlID("Foo"), "name"))
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100402 syn clear Foo
403 call assert_notmatch('Foo', execute('syntax'))
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100404 call assert_match('Bar', execute('syntax'))
Bram Moolenaarc96272e2017-03-26 13:50:09 +0200405 call assert_equal('Foo', synIDattr(hlID("Foo"), "name"))
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100406 syn clear Foo Bar
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100407 call assert_notmatch('Foo', execute('syntax'))
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100408 call assert_notmatch('Bar', execute('syntax'))
409 hi clear Foo
Bram Moolenaarc96272e2017-03-26 13:50:09 +0200410 call assert_equal('Foo', synIDattr(hlID("Foo"), "name"))
Bram Moolenaard61e8aa2017-01-17 17:44:46 +0100411 hi clear Bar
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200412 call assert_fails('syntax clear invalid_syngroup', 'E28:')
Bram Moolenaar58f60ca2017-01-17 17:19:00 +0100413endfunc
Bram Moolenaar4007ed42017-01-17 18:14:54 +0100414
415func Test_invalid_name()
416 syn clear
417 syn keyword Nop yes
418 call assert_fails("syntax keyword Wr\x17ong bar", 'E669:')
419 syntax keyword @Wrong bar
420 call assert_match('W18:', execute('1messages'))
421 syn clear
422 hi clear Nop
423 hi clear @Wrong
424endfunc
Bram Moolenaarf8ec9982017-04-09 15:41:31 +0200425
426func Test_ownsyntax()
427 new Xfoo
428 call setline(1, '#define FOO')
429 syntax on
430 set filetype=c
431 ownsyntax perl
432 call assert_equal('perlComment', synIDattr(synID(line('.'), col('.'), 1), 'name'))
433 call assert_equal('c', b:current_syntax)
434 call assert_equal('perl', w:current_syntax)
435
436 " A new split window should have the original syntax.
437 split
438 call assert_equal('cDefine', synIDattr(synID(line('.'), col('.'), 1), 'name'))
439 call assert_equal('c', b:current_syntax)
440 call assert_equal(0, exists('w:current_syntax'))
441
442 wincmd x
443 call assert_equal('perlComment', synIDattr(synID(line("."), col("."), 1), "name"))
444
445 syntax off
446 set filetype&
447 %bw!
448endfunc
449
450func Test_ownsyntax_completion()
451 call feedkeys(":ownsyntax java\<C-A>\<C-B>\"\<CR>", 'tx')
Bram Moolenaarea7a08a2019-08-26 22:38:22 +0200452 call assert_equal('"ownsyntax java javacc javascript javascriptreact', @:)
Bram Moolenaarf8ec9982017-04-09 15:41:31 +0200453endfunc
Bram Moolenaarea588152017-04-10 22:45:30 +0200454
455func Test_highlight_invalid_arg()
456 if has('gui_running')
457 call assert_fails('hi XXX guifg=xxx', 'E254:')
458 endif
459 call assert_fails('hi DoesNotExist', 'E411:')
460 call assert_fails('hi link', 'E412:')
461 call assert_fails('hi link a', 'E412:')
462 call assert_fails('hi link a b c', 'E413:')
463 call assert_fails('hi XXX =', 'E415:')
464 call assert_fails('hi XXX cterm', 'E416:')
465 call assert_fails('hi XXX cterm=', 'E417:')
466 call assert_fails('hi XXX cterm=DoesNotExist', 'E418:')
467 call assert_fails('hi XXX ctermfg=DoesNotExist', 'E421:')
468 call assert_fails('hi XXX xxx=White', 'E423:')
469endfunc
470
Bram Moolenaar1615b362017-06-04 21:06:09 +0200471func Test_bg_detection()
Bram Moolenaar8c5a2782019-08-07 23:07:07 +0200472 CheckNotGui
473
Bram Moolenaar1615b362017-06-04 21:06:09 +0200474 " auto-detection of &bg, make sure sure it isn't set anywhere before
475 " this test
476 hi Normal ctermbg=0
477 call assert_equal('dark', &bg)
478 hi Normal ctermbg=4
479 call assert_equal('dark', &bg)
480 hi Normal ctermbg=12
481 call assert_equal('light', &bg)
482 hi Normal ctermbg=15
483 call assert_equal('light', &bg)
484
Bram Moolenaar0b2eef22017-06-27 15:43:49 +0200485 " manually-set &bg takes precedence over auto-detection
Bram Moolenaar1615b362017-06-04 21:06:09 +0200486 set bg=light
487 hi Normal ctermbg=4
488 call assert_equal('light', &bg)
489 set bg=dark
490 hi Normal ctermbg=12
491 call assert_equal('dark', &bg)
Bram Moolenaar6acadda2018-02-24 16:51:32 +0100492
493 hi Normal ctermbg=NONE
Bram Moolenaar1615b362017-06-04 21:06:09 +0200494endfunc
Bram Moolenaar06f1ed22017-06-18 22:41:03 +0200495
496func Test_syntax_hangs()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +0200497 CheckFunction reltimefloat
498 CheckFeature syntax
Bram Moolenaar06f1ed22017-06-18 22:41:03 +0200499
500 " This pattern takes a long time to match, it should timeout.
501 new
502 call setline(1, ['aaa', repeat('abc ', 1000), 'ccc'])
503 let start = reltime()
504 set nolazyredraw redrawtime=101
505 syn match Error /\%#=1a*.*X\@<=b*/
506 redraw
507 let elapsed = reltimefloat(reltime(start))
508 call assert_true(elapsed > 0.1)
509 call assert_true(elapsed < 1.0)
510
511 " second time syntax HL is disabled
512 let start = reltime()
513 redraw
514 let elapsed = reltimefloat(reltime(start))
515 call assert_true(elapsed < 0.1)
516
517 " after CTRL-L the timeout flag is reset
518 let start = reltime()
519 exe "normal \<C-L>"
520 redraw
521 let elapsed = reltimefloat(reltime(start))
522 call assert_true(elapsed > 0.1)
523 call assert_true(elapsed < 1.0)
524
525 set redrawtime&
526 bwipe!
527endfunc
Bram Moolenaar4d785892017-06-22 22:00:50 +0200528
Bram Moolenaar4d785892017-06-22 22:00:50 +0200529func Test_conceal()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +0200530 CheckFeature conceal
Bram Moolenaar4d785892017-06-22 22:00:50 +0200531
532 new
533 call setline(1, ['', '123456'])
534 syn match test23 "23" conceal cchar=X
535 syn match test45 "45" conceal
536
537 set conceallevel=0
538 call assert_equal('123456 ', ScreenLines(2, 7)[0])
Bram Moolenaarcc0750d2017-06-24 22:29:24 +0200539 call assert_equal([[0, '', 0], [0, '', 0], [0, '', 0], [0, '', 0], [0, '', 0], [0, '', 0]], map(range(1, 6), 'synconcealed(2, v:val)'))
Bram Moolenaar4d785892017-06-22 22:00:50 +0200540
541 set conceallevel=1
542 call assert_equal('1X 6 ', ScreenLines(2, 7)[0])
Bram Moolenaarcc0750d2017-06-24 22:29:24 +0200543 call assert_equal([[0, '', 0], [1, 'X', 1], [1, 'X', 1], [1, ' ', 2], [1, ' ', 2], [0, '', 0]], map(range(1, 6), 'synconcealed(2, v:val)'))
Bram Moolenaar4d785892017-06-22 22:00:50 +0200544
545 set conceallevel=1
546 set listchars=conceal:Y
Bram Moolenaarcc0750d2017-06-24 22:29:24 +0200547 call assert_equal([[0, '', 0], [1, 'X', 1], [1, 'X', 1], [1, 'Y', 2], [1, 'Y', 2], [0, '', 0]], map(range(1, 6), 'synconcealed(2, v:val)'))
Bram Moolenaar4d785892017-06-22 22:00:50 +0200548 call assert_equal('1XY6 ', ScreenLines(2, 7)[0])
549
550 set conceallevel=2
551 call assert_match('1X6 ', ScreenLines(2, 7)[0])
Bram Moolenaarcc0750d2017-06-24 22:29:24 +0200552 call assert_equal([[0, '', 0], [1, 'X', 1], [1, 'X', 1], [1, '', 2], [1, '', 2], [0, '', 0]], map(range(1, 6), 'synconcealed(2, v:val)'))
Bram Moolenaar4d785892017-06-22 22:00:50 +0200553
554 set conceallevel=3
555 call assert_match('16 ', ScreenLines(2, 7)[0])
Bram Moolenaarcc0750d2017-06-24 22:29:24 +0200556 call assert_equal([[0, '', 0], [1, '', 1], [1, '', 1], [1, '', 2], [1, '', 2], [0, '', 0]], map(range(1, 6), 'synconcealed(2, v:val)'))
Bram Moolenaar4d785892017-06-22 22:00:50 +0200557
Bram Moolenaarfbf21222020-04-19 18:31:25 +0200558 call AssertFails("syntax match Entity '&amp;' conceal cchar=\<Tab>", 'E844:')
559
Bram Moolenaar4d785892017-06-22 22:00:50 +0200560 syn clear
561 set conceallevel&
562 bw!
563endfunc
Bram Moolenaar0b2eef22017-06-27 15:43:49 +0200564
Bram Moolenaarda650582018-02-20 15:51:40 +0100565func Test_synstack_synIDtrans()
Bram Moolenaar0b2eef22017-06-27 15:43:49 +0200566 new
567 setfiletype c
568 syntax on
569 call setline(1, ' /* A comment with a TODO */')
570
571 call assert_equal([], synstack(1, 1))
572
573 norm f/
Bram Moolenaara74e4942019-08-04 17:35:53 +0200574 eval synstack(line("."), col("."))->map('synIDattr(v:val, "name")')->assert_equal(['cComment', 'cCommentStart'])
575 eval synstack(line("."), col("."))->map('synIDattr(synIDtrans(v:val), "name")')->assert_equal(['Comment', 'Comment'])
Bram Moolenaar0b2eef22017-06-27 15:43:49 +0200576
577 norm fA
578 call assert_equal(['cComment'], map(synstack(line("."), col(".")), 'synIDattr(v:val, "name")'))
579 call assert_equal(['Comment'], map(synstack(line("."), col(".")), 'synIDattr(synIDtrans(v:val), "name")'))
580
581 norm fT
582 call assert_equal(['cComment', 'cTodo'], map(synstack(line("."), col(".")), 'synIDattr(v:val, "name")'))
583 call assert_equal(['Comment', 'Todo'], map(synstack(line("."), col(".")), 'synIDattr(synIDtrans(v:val), "name")'))
584
Bram Moolenaar0e05de42020-03-25 22:23:46 +0100585 call assert_fails("let n=synIDtrans([])", 'E745:')
586
Bram Moolenaar0b2eef22017-06-27 15:43:49 +0200587 syn clear
588 bw!
589endfunc
Bram Moolenaarda650582018-02-20 15:51:40 +0100590
591" Check highlighting for a small piece of C code with a screen dump.
592func Test_syntax_c()
Bram Moolenaar494e9062020-05-31 21:28:02 +0200593 CheckRunVimInTerminal
Bram Moolenaarda650582018-02-20 15:51:40 +0100594 call writefile([
595 \ '/* comment line at the top */',
Bram Moolenaar83e9a1c2019-10-20 14:51:23 +0200596 \ 'int main(int argc, char **argv) { // another comment',
Bram Moolenaarda650582018-02-20 15:51:40 +0100597 \ '#if 0',
598 \ ' int not_used;',
599 \ '#else',
600 \ ' int used;',
601 \ '#endif',
602 \ ' printf("Just an example piece of C code\n");',
603 \ ' return 0x0ff;',
604 \ '}',
Bram Moolenaar82260af2019-10-20 13:16:22 +0200605 \ "\t\t ",
Bram Moolenaarda650582018-02-20 15:51:40 +0100606 \ ' static void',
607 \ 'myFunction(const double count, struct nothing, long there) {',
Bram Moolenaar84590062019-10-18 23:12:20 +0200608 \ "\t// 123: nothing to endif here",
Bram Moolenaar9115c612019-10-16 16:57:06 +0200609 \ "\tfor (int i = 0; i < count; ++i) {",
610 \ "\t break;",
611 \ "\t}",
Bram Moolenaarbbfd1562019-10-19 20:38:15 +0200612 \ "\tNote: asdf",
Bram Moolenaarda650582018-02-20 15:51:40 +0100613 \ '}',
614 \ ], 'Xtest.c')
Bram Moolenaarb7ea7cb2018-02-24 14:38:51 +0100615
616 " This makes the default for 'background' use "dark", check that the
617 " response to t_RB corrects it to "light".
618 let $COLORFGBG = '15;0'
619
Bram Moolenaar83e9a1c2019-10-20 14:51:23 +0200620 let buf = RunVimInTerminal('Xtest.c', {})
Bram Moolenaarbbfd1562019-10-19 20:38:15 +0200621 call term_sendkeys(buf, ":syn keyword Search Note\r")
Bram Moolenaar82260af2019-10-20 13:16:22 +0200622 call term_sendkeys(buf, ":syn match Error /^\\s\\+$/\r")
Bram Moolenaar84590062019-10-18 23:12:20 +0200623 call term_sendkeys(buf, ":set hlsearch\r")
624 call term_sendkeys(buf, "/endif\r")
625 call term_sendkeys(buf, "vjfC")
Bram Moolenaar6bb2cdf2018-02-24 19:53:53 +0100626 call VerifyScreenDump(buf, 'Test_syntax_c_01', {})
Bram Moolenaar84590062019-10-18 23:12:20 +0200627
628 call term_sendkeys(buf, "\<Esc>")
Bram Moolenaarda650582018-02-20 15:51:40 +0100629 call StopVimInTerminal(buf)
630
Bram Moolenaarb7ea7cb2018-02-24 14:38:51 +0100631 let $COLORFGBG = ''
Bram Moolenaarda650582018-02-20 15:51:40 +0100632 call delete('Xtest.c')
633endfun
Bram Moolenaarbcf94422018-06-23 14:21:42 +0200634
635" Using \z() in a region with NFA failing should not crash.
636func Test_syn_wrong_z_one()
637 new
638 call setline(1, ['just some text', 'with foo and bar to match with'])
639 syn region FooBar start="foo\z(.*\)bar" end="\z1"
640 call test_override("nfa_fail", 1)
641 redraw!
642 redraw!
643 call test_override("ALL", 0)
644 bwipe!
645endfunc
Bram Moolenaarc7f1e402019-08-03 13:29:46 +0200646
647func Test_syntax_after_bufdo()
648 call writefile(['/* aaa comment */'], 'Xaaa.c')
649 call writefile(['/* bbb comment */'], 'Xbbb.c')
650 call writefile(['/* ccc comment */'], 'Xccc.c')
651 call writefile(['/* ddd comment */'], 'Xddd.c')
652
653 let bnr = bufnr('%')
654 new Xaaa.c
655 badd Xbbb.c
656 badd Xccc.c
657 badd Xddd.c
658 exe "bwipe " . bnr
659 let l = []
660 bufdo call add(l, bufnr('%'))
661 call assert_equal(4, len(l))
662
663 syntax on
664
665 " This used to only enable syntax HL in the last buffer.
666 bufdo tab split
667 tabrewind
668 for tab in range(1, 4)
669 norm fm
670 call assert_equal(['cComment'], map(synstack(line("."), col(".")), 'synIDattr(v:val, "name")'))
671 tabnext
672 endfor
673
674 bwipe! Xaaa.c
675 bwipe! Xbbb.c
676 bwipe! Xccc.c
677 bwipe! Xddd.c
678 syntax off
679 call delete('Xaaa.c')
680 call delete('Xbbb.c')
681 call delete('Xccc.c')
682 call delete('Xddd.c')
683endfunc
Bram Moolenaar0e05de42020-03-25 22:23:46 +0100684
Bram Moolenaare35a52a2020-05-31 19:48:53 +0200685func Test_syntax_foldlevel()
686 new
687 call setline(1, [
688 \ 'void f(int a)',
689 \ '{',
690 \ ' if (a == 1) {',
691 \ ' a = 0;',
692 \ ' } else if (a == 2) {',
693 \ ' a = 1;',
694 \ ' } else {',
695 \ ' a = 2;',
696 \ ' }',
697 \ ' if (a > 0) {',
698 \ ' if (a == 1) {',
699 \ ' a = 0;',
700 \ ' } /* missing newline */ } /* end of outer if */ else {',
701 \ ' a = 1;',
702 \ ' }',
703 \ ' if (a == 1)',
704 \ ' {',
705 \ ' a = 0;',
706 \ ' }',
707 \ ' else if (a == 2)',
708 \ ' {',
709 \ ' a = 1;',
710 \ ' }',
711 \ ' else',
712 \ ' {',
713 \ ' a = 2;',
714 \ ' }',
715 \ '}',
716 \ ])
717 setfiletype c
718 syntax on
719 set foldmethod=syntax
720
721 call assert_fails('syn foldlevel start start', 'E390')
722 call assert_fails('syn foldlevel not_an_option', 'E390')
723
724 set foldlevel=1
725
726 syn foldlevel start
727 redir @c
728 syn foldlevel
729 redir END
730 call assert_equal("\nsyntax foldlevel start", @c)
731 syn sync fromstart
732 let a = map(range(3,9), 'foldclosed(v:val)')
733 call assert_equal([3,3,3,3,3,3,3], a) " attached cascade folds together
734 let a = map(range(10,15), 'foldclosed(v:val)')
735 call assert_equal([10,10,10,10,10,10], a) " over-attached 'else' hidden
736 let a = map(range(16,27), 'foldclosed(v:val)')
737 let unattached_results = [-1,17,17,17,-1,21,21,21,-1,25,25,25]
738 call assert_equal(unattached_results, a) " unattached cascade folds separately
739
740 syn foldlevel minimum
741 redir @c
742 syn foldlevel
743 redir END
744 call assert_equal("\nsyntax foldlevel minimum", @c)
745 syn sync fromstart
746 let a = map(range(3,9), 'foldclosed(v:val)')
747 call assert_equal([3,3,5,5,7,7,7], a) " attached cascade folds separately
748 let a = map(range(10,15), 'foldclosed(v:val)')
749 call assert_equal([10,10,10,13,13,13], a) " over-attached 'else' visible
750 let a = map(range(16,27), 'foldclosed(v:val)')
751 call assert_equal(unattached_results, a) " unattached cascade folds separately
752
753 set foldlevel=2
754
755 syn foldlevel start
756 syn sync fromstart
757 let a = map(range(11,14), 'foldclosed(v:val)')
758 call assert_equal([11,11,11,-1], a) " over-attached 'else' hidden
759
760 syn foldlevel minimum
761 syn sync fromstart
762 let a = map(range(11,14), 'foldclosed(v:val)')
763 call assert_equal([11,11,-1,-1], a) " over-attached 'else' visible
764
765 quit!
766endfunc
767
Bram Moolenaaradc17a52020-06-06 18:37:51 +0200768func Test_search_syntax_skip()
769 new
770 let lines =<< trim END
771
772 /* This is VIM */
773 Another Text for VIM
774 let a = "VIM"
775 END
776 call setline(1, lines)
777 syntax on
778 syntax match Comment "^/\*.*\*/"
779 syntax match String '".*"'
780
781 " Skip argument using string evaluation.
782 1
783 call search('VIM', 'w', '', 0, 'synIDattr(synID(line("."), col("."), 1), "name") =~? "comment"')
784 call assert_equal('Another Text for VIM', getline('.'))
785 1
786 call search('VIM', 'w', '', 0, 'synIDattr(synID(line("."), col("."), 1), "name") !~? "string"')
787 call assert_equal(' let a = "VIM"', getline('.'))
788
789 " Skip argument using Lambda.
790 1
791 call search('VIM', 'w', '', 0, { -> synIDattr(synID(line("."), col("."), 1), "name") =~? "comment"})
792 call assert_equal('Another Text for VIM', getline('.'))
793
794 1
795 call search('VIM', 'w', '', 0, { -> synIDattr(synID(line("."), col("."), 1), "name") !~? "string"})
796 call assert_equal(' let a = "VIM"', getline('.'))
797
798 " Skip argument using funcref.
799 func InComment()
800 return synIDattr(synID(line("."), col("."), 1), "name") =~? "comment"
801 endfunc
802 func InString()
803 return synIDattr(synID(line("."), col("."), 1), "name") !~? "string"
804 endfunc
805 1
806 call search('VIM', 'w', '', 0, function('InComment'))
807 call assert_equal('Another Text for VIM', getline('.'))
808
809 1
810 call search('VIM', 'w', '', 0, function('InString'))
811 call assert_equal(' let a = "VIM"', getline('.'))
812
813 delfunc InComment
814 delfunc InString
815 bwipe!
816endfunc
817
Bram Moolenaar0e05de42020-03-25 22:23:46 +0100818" vim: shiftwidth=2 sts=2 expandtab