blob: 70a71ea9324c98f4dbd0406a24399f4a449a5ac0 [file] [log] [blame]
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001" Test various aspects of the Vim9 script language.
2
3source check.vim
Bram Moolenaarad304702020-09-06 18:22:53 +02004source term_util.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02005source view_util.vim
Bram Moolenaar62aec932022-01-29 21:45:34 +00006import './vim9.vim' as v9
Bram Moolenaar47e7d702020-07-05 18:18:42 +02007source screendump.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02008
9func Test_def_basic()
10 def SomeFunc(): string
11 return 'yes'
12 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +020013 call SomeFunc()->assert_equal('yes')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020014endfunc
15
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020016func Test_compiling_error()
17 " use a terminal to see the whole error message
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020018 CheckRunVimInTerminal
19
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020020 call TestCompilingError()
Bram Moolenaare8c46602021-04-05 22:27:37 +020021 call TestCompilingErrorInTry()
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020022endfunc
23
24def TestCompilingError()
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020025 var lines =<< trim END
26 vim9script
27 def Fails()
28 echo nothing
29 enddef
30 defcompile
31 END
Bram Moolenaare8c46602021-04-05 22:27:37 +020032 writefile(lines, 'XTest_compile_error')
Bram Moolenaar62aec932022-01-29 21:45:34 +000033 var buf = g:RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010034 {rows: 10, wait_for_ruler: 0})
Bram Moolenaar62aec932022-01-29 21:45:34 +000035 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
36 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020037
38 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000039 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020040 delete('XTest_compile_error')
41enddef
42
43def TestCompilingErrorInTry()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +010044 var dir = 'Xcompdir/autoload'
Bram Moolenaare8c46602021-04-05 22:27:37 +020045 mkdir(dir, 'p')
46
47 var lines =<< trim END
48 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +000049 export def OnlyCompiled()
Bram Moolenaare8c46602021-04-05 22:27:37 +020050 g:runtime = 'yes'
51 invalid
52 enddef
53 END
54 writefile(lines, dir .. '/script.vim')
55
56 lines =<< trim END
57 vim9script
58 todo
59 try
60 script#OnlyCompiled()
61 catch /nothing/
62 endtry
63 END
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +010064 lines[1] = 'set rtp=' .. getcwd() .. '/Xcompdir'
Bram Moolenaare8c46602021-04-05 22:27:37 +020065 writefile(lines, 'XTest_compile_error')
66
Bram Moolenaar62aec932022-01-29 21:45:34 +000067 var buf = g:RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
68 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
69 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaare8c46602021-04-05 22:27:37 +020070
71 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000072 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020073 delete('XTest_compile_error')
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +010074 delete('Xcompdir', 'rf')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020075enddef
76
Bram Moolenaarad6d9cc2022-08-08 21:43:11 +010077def Test_comment_error()
78 v9.CheckDefFailure(['#{ comment'], 'E1170:')
79enddef
80
Bram Moolenaarb55d6182021-06-08 22:01:53 +020081def Test_compile_error_in_called_function()
82 var lines =<< trim END
83 vim9script
84 var n: number
85 def Foo()
86 &hls = n
87 enddef
88 def Bar()
89 Foo()
90 enddef
91 silent! Foo()
92 Bar()
93 END
Bram Moolenaar62aec932022-01-29 21:45:34 +000094 v9.CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
Bram Moolenaarb55d6182021-06-08 22:01:53 +020095enddef
96
Bram Moolenaar22f17a22021-06-21 20:48:58 +020097def Test_wrong_function_name()
98 var lines =<< trim END
99 vim9script
100 func _Foo()
101 echo 'foo'
102 endfunc
103 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000104 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200105
106 lines =<< trim END
107 vim9script
108 def _Foo()
109 echo 'foo'
110 enddef
111 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000112 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaardea5ab02022-02-23 22:12:02 +0000113
114 lines =<< trim END
115 vim9script
116 var Object = {}
117 function Object.Method()
118 endfunction
119 END
120 v9.CheckScriptFailure(lines, 'E1182:')
121
122 lines =<< trim END
123 vim9script
124 var Object = {}
125 def Object.Method()
126 enddef
127 END
128 v9.CheckScriptFailure(lines, 'E1182:')
129
130 lines =<< trim END
131 vim9script
132 g:Object = {}
133 function g:Object.Method()
134 endfunction
135 END
136 v9.CheckScriptFailure(lines, 'E1182:')
137
138 lines =<< trim END
139 let s:Object = {}
140 def Define()
141 function s:Object.Method()
142 endfunction
143 enddef
144 defcompile
145 END
146 v9.CheckScriptFailure(lines, 'E1182:')
147 delfunc g:Define
148
149 lines =<< trim END
150 let s:Object = {}
151 def Define()
152 def Object.Method()
153 enddef
154 enddef
155 defcompile
156 END
157 v9.CheckScriptFailure(lines, 'E1182:')
158 delfunc g:Define
159
160 lines =<< trim END
161 let g:Object = {}
162 def Define()
163 function g:Object.Method()
164 endfunction
165 enddef
166 defcompile
167 END
168 v9.CheckScriptFailure(lines, 'E1182:')
169 delfunc g:Define
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200170enddef
171
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200172def Test_autoload_name_mismatch()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100173 var dir = 'Xnamedir/autoload'
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200174 mkdir(dir, 'p')
175
176 var lines =<< trim END
177 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000178 export def NoFunction()
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200179 # comment
180 g:runtime = 'yes'
181 enddef
182 END
183 writefile(lines, dir .. '/script.vim')
184
185 var save_rtp = &rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100186 exe 'set rtp=' .. getcwd() .. '/Xnamedir'
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200187 lines =<< trim END
188 call script#Function()
189 END
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000190 v9.CheckScriptFailure(lines, 'E117:', 1)
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200191
192 &rtp = save_rtp
Dominique Pellef2403952022-08-30 18:42:16 +0100193 delete('Xnamedir', 'rf')
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200194enddef
195
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200196def Test_autoload_names()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100197 var dir = 'Xandir/autoload'
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200198 mkdir(dir, 'p')
199
200 var lines =<< trim END
201 func foobar#function()
202 return 'yes'
203 endfunc
204 let foobar#var = 'no'
205 END
206 writefile(lines, dir .. '/foobar.vim')
207
208 var save_rtp = &rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100209 exe 'set rtp=' .. getcwd() .. '/Xandir'
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200210
211 lines =<< trim END
212 assert_equal('yes', foobar#function())
213 var Function = foobar#function
214 assert_equal('yes', Function())
215
216 assert_equal('no', foobar#var)
217 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000218 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200219
220 &rtp = save_rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100221 delete('Xandir', 'rf')
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200222enddef
223
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200224def Test_autoload_error_in_script()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100225 var dir = 'Xaedir/autoload'
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200226 mkdir(dir, 'p')
227
228 var lines =<< trim END
229 func scripterror#function()
230 let g:called_function = 'yes'
231 endfunc
232 let 0 = 1
233 END
234 writefile(lines, dir .. '/scripterror.vim')
235
236 var save_rtp = &rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100237 exe 'set rtp=' .. getcwd() .. '/Xaedir'
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200238
239 g:called_function = 'no'
240 # The error in the autoload script cannot be checked with assert_fails(), use
241 # CheckDefSuccess() instead of CheckDefFailure()
242 try
Bram Moolenaar62aec932022-01-29 21:45:34 +0000243 v9.CheckDefSuccess(['scripterror#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200244 catch
245 assert_match('E121: Undefined variable: 0', v:exception)
246 endtry
247 assert_equal('no', g:called_function)
248
249 lines =<< trim END
250 func scriptcaught#function()
251 let g:called_function = 'yes'
252 endfunc
253 try
254 let 0 = 1
255 catch
256 let g:caught = v:exception
257 endtry
258 END
259 writefile(lines, dir .. '/scriptcaught.vim')
260
261 g:called_function = 'no'
Bram Moolenaar62aec932022-01-29 21:45:34 +0000262 v9.CheckDefSuccess(['scriptcaught#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200263 assert_match('E121: Undefined variable: 0', g:caught)
264 assert_equal('yes', g:called_function)
265
266 &rtp = save_rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100267 delete('Xaedir', 'rf')
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200268enddef
269
Bram Moolenaar62aec932022-01-29 21:45:34 +0000270def s:CallRecursive(n: number): number
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100271 return CallRecursive(n + 1)
272enddef
273
Bram Moolenaar62aec932022-01-29 21:45:34 +0000274def s:CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100275 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100276enddef
277
278def Test_funcdepth_error()
279 set maxfuncdepth=10
280
281 var caught = false
282 try
283 CallRecursive(1)
284 catch /E132:/
285 caught = true
286 endtry
287 assert_true(caught)
288
289 caught = false
290 try
291 CallMapRecursive([1])
292 catch /E132:/
293 caught = true
294 endtry
295 assert_true(caught)
296
297 set maxfuncdepth&
298enddef
299
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100300def Test_endfunc_enddef()
301 var lines =<< trim END
302 def Test()
303 echo 'test'
304 endfunc
305 enddef
306 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000307 v9.CheckScriptFailure(lines, 'E1151:', 3)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100308
309 lines =<< trim END
310 def Test()
311 func Nested()
312 echo 'test'
313 enddef
314 enddef
315 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000316 v9.CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100317
318 lines =<< trim END
319 def Ok()
320 echo 'hello'
321 enddef | echo 'there'
322 def Bad()
323 echo 'hello'
324 enddef there
325 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000326 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100327enddef
328
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100329def Test_missing_endfunc_enddef()
330 var lines =<< trim END
331 vim9script
332 def Test()
333 echo 'test'
334 endef
335 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000336 v9.CheckScriptFailure(lines, 'E1057:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100337
338 lines =<< trim END
339 vim9script
340 func Some()
341 echo 'test'
342 enfffunc
343 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000344 v9.CheckScriptFailure(lines, 'E126:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100345enddef
346
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100347def Test_white_space_before_paren()
348 var lines =<< trim END
349 vim9script
350 def Test ()
351 echo 'test'
352 enddef
353 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000354 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100355
356 lines =<< trim END
357 vim9script
358 func Test ()
359 echo 'test'
360 endfunc
361 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000362 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100363
364 lines =<< trim END
365 def Test ()
366 echo 'test'
367 enddef
368 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000369 v9.CheckScriptFailure(lines, 'E1068:', 1)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100370
371 lines =<< trim END
372 func Test ()
373 echo 'test'
374 endfunc
375 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000376 v9.CheckScriptSuccess(lines)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100377enddef
378
Bram Moolenaar832ea892021-01-08 21:55:26 +0100379def Test_enddef_dict_key()
380 var d = {
381 enddef: 'x',
382 endfunc: 'y',
383 }
384 assert_equal({enddef: 'x', endfunc: 'y'}, d)
385enddef
386
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200387def ReturnString(): string
388 return 'string'
389enddef
390
391def ReturnNumber(): number
392 return 123
393enddef
394
395let g:notNumber = 'string'
396
397def ReturnGlobal(): number
398 return g:notNumber
399enddef
400
401def Test_return_something()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000402 g:ReturnString()->assert_equal('string')
403 g:ReturnNumber()->assert_equal(123)
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000404 assert_fails('g:ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaaref7aadb2022-01-18 18:46:07 +0000405
406 var lines =<< trim END
407 vim9script
408
409 def Msg()
410 echomsg 'in Msg()...'
411 enddef
412
413 def Func()
414 return Msg()
415 enddef
416 defcompile
417 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000418 v9.CheckScriptFailure(lines, 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200419enddef
420
Bram Moolenaare32e5162021-01-21 20:21:29 +0100421def Test_check_argument_type()
422 var lines =<< trim END
423 vim9script
424 def Val(a: number, b: number): number
425 return 0
426 enddef
427 def Func()
428 var x: any = true
429 Val(0, x)
430 enddef
431 disass Func
432 Func()
433 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000434 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
Bram Moolenaare32e5162021-01-21 20:21:29 +0100435enddef
436
Bram Moolenaarefd88552020-06-18 20:50:10 +0200437def Test_missing_return()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000438 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200439 ' if g:cond',
440 ' echo "no return"',
441 ' else',
442 ' return 0',
Bram Moolenaar2984ed32022-08-20 14:51:17 +0100443 ' endif',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200444 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000445 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200446 ' if g:cond',
447 ' return 1',
448 ' else',
449 ' echo "no return"',
Bram Moolenaar2984ed32022-08-20 14:51:17 +0100450 ' endif',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200451 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000452 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200453 ' if g:cond',
454 ' return 1',
455 ' else',
456 ' return 2',
Bram Moolenaar2984ed32022-08-20 14:51:17 +0100457 ' endif',
458 ' return 3',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200459 'enddef'], 'E1095:')
460enddef
461
Bram Moolenaar403dc312020-10-17 19:29:51 +0200462def Test_return_bool()
463 var lines =<< trim END
464 vim9script
465 def MenuFilter(id: number, key: string): bool
466 return popup_filter_menu(id, key)
467 enddef
468 def YesnoFilter(id: number, key: string): bool
469 return popup_filter_yesno(id, key)
470 enddef
471 defcompile
472 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000473 v9.CheckScriptSuccess(lines)
Bram Moolenaar403dc312020-10-17 19:29:51 +0200474enddef
475
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200476let s:nothing = 0
477def ReturnNothing()
478 s:nothing = 1
479 if true
480 return
481 endif
482 s:nothing = 2
483enddef
484
485def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000486 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200487 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200488enddef
489
Bram Moolenaar648ea762021-01-15 19:04:32 +0100490def Test_return_invalid()
491 var lines =<< trim END
492 vim9script
493 def Func(): invalid
494 return xxx
495 enddef
496 defcompile
497 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000498 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100499
500 lines =<< trim END
501 vim9script
502 def Test(Fun: func(number): number): list<number>
503 return map([1, 2, 3], (_, i) => Fun(i))
504 enddef
505 defcompile
506 def Inc(nr: number): nr
507 return nr + 2
508 enddef
509 echo Test(Inc)
510 END
511 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000512 v9.CheckScriptFailure(lines, 'E1010:')
513 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100514enddef
515
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200516def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000517 # This used to fail but now the actual list type is checked, and since it has
518 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200519 var lines =<< trim END
520 vim9script
521 def Func(): list<string>
522 var l: list<any>
523 l->add('string')
524 return l
525 enddef
526 echo Func()
527 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000528 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000529
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200530 lines =<< trim END
531 vim9script
532 def Func(): list<string>
533 var l: list<any>
534 l += ['string']
535 return l
536 enddef
537 echo Func()
538 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000539 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200540enddef
541
Bram Moolenaar1a572e92022-03-15 12:28:10 +0000542def Test_return_any_two_types()
543 var lines =<< trim END
544 vim9script
545
546 def G(Fn: func(string): any)
547 g:result = Fn("hello")
548 enddef
549
550 def F(a: number, b: string): any
551 echo b
552 if a > 0
553 return 1
554 else
555 return []
556 endif
557 enddef
558
559 G(function(F, [1]))
560 END
561 v9.CheckScriptSuccess(lines)
562 assert_equal(1, g:result)
563 unlet g:result
564enddef
565
Bram Moolenaar62aec932022-01-29 21:45:34 +0000566func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200567 let g:counter += 1
568endfunc
569
570def Test_call_ufunc_count()
571 g:counter = 1
572 Increment()
573 Increment()
574 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200575 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200576 g:counter->assert_equal(4)
577 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200578 unlet g:counter
579enddef
580
Bram Moolenaar1983f1a2022-02-28 20:55:02 +0000581def Test_call_ufunc_failure()
582 var lines =<< trim END
583 vim9script
584 def Tryit()
585 g:Global(1, 2, 3)
586 enddef
587
588 func g:Global(a, b, c)
589 echo a:a a:b a:c
590 endfunc
591
592 defcompile
593
594 func! g:Global(a, b)
595 echo a:a a:b
596 endfunc
597 Tryit()
598 END
599 v9.CheckScriptFailure(lines, 'E118: Too many arguments for function: Global')
600 delfunc g:Global
601
602 lines =<< trim END
603 vim9script
604
605 g:Ref = function('len')
606 def Tryit()
607 g:Ref('x')
608 enddef
609
610 defcompile
611
612 g:Ref = function('add')
613 Tryit()
614 END
615 v9.CheckScriptFailure(lines, 'E119: Not enough arguments for function: add')
616 unlet g:Ref
617enddef
618
Bram Moolenaar62aec932022-01-29 21:45:34 +0000619def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200620 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200621 for s in rest
622 res ..= ',' .. s
623 endfor
624 return res
625enddef
626
627def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200628 MyVarargs('one')->assert_equal('one')
629 MyVarargs('one', 'two')->assert_equal('one,two')
630 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200631enddef
632
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200633def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000634 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200635enddef
636
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200637def MyDefaultArgs(name = 'string'): string
638 return name
639enddef
640
Bram Moolenaar62aec932022-01-29 21:45:34 +0000641def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200642 return second ? name : 'none'
643enddef
644
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200645
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200646def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000647 g:MyDefaultArgs()->assert_equal('string')
648 g:MyDefaultArgs(v:none)->assert_equal('string')
649 g:MyDefaultArgs('one')->assert_equal('one')
650 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200651
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200652 MyDefaultSecond('test')->assert_equal('test')
653 MyDefaultSecond('test', true)->assert_equal('test')
654 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200655
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200656 var lines =<< trim END
657 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
658 return name .. aa .. bb
659 enddef
660
661 MyDefaultThird('->')->assert_equal('->aabb')
662 MyDefaultThird('->', v:none)->assert_equal('->aabb')
663 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
664 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
665 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
666 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
667 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200668
669 def DefArg(mandatory: any, optional = mandatory): string
670 return mandatory .. optional
671 enddef
672 DefArg(1234)->assert_equal('12341234')
673 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200674 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000675 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200676
Bram Moolenaar62aec932022-01-29 21:45:34 +0000677 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100678 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000679 v9.CheckScriptFailure(['def Func(arg: number = "text")', 'enddef', 'defcompile'], 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100680 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000681 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100682
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200683 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100684 vim9script
685 def Func(a = b == 0 ? 1 : 2, b = 0)
686 enddef
687 defcompile
688 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000689 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000690
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000691 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000692 lines =<< trim END
693 vim9script
694 var a: any
695 def Func(arg: string = a)
696 echo arg
697 enddef
698 defcompile
699 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000700 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
701 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000702
703 # using global variable does not require type cast
704 lines =<< trim END
705 vim9script
706 def Func(arg: string = g:str)
707 echo arg
708 enddef
709 g:str = 'works'
710 Func()
711 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000712 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200713enddef
714
Bram Moolenaar62aec932022-01-29 21:45:34 +0000715def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100716 a: number, #comment
717 b: bool, # comment
718 c: string) #comment
719 assert_equal(4, a)
720 assert_equal(true, b)
721 assert_equal('yes', c)
722enddef
723
724def Test_func_with_comments()
725 FuncWithComment(4, true, 'yes')
726
727 var lines =<< trim END
728 def Func(# comment
729 arg: string)
730 enddef
731 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000732 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100733
734 lines =<< trim END
735 def Func(
736 arg: string# comment
737 )
738 enddef
739 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000740 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100741
742 lines =<< trim END
743 def Func(
744 arg: string
745 )# comment
746 enddef
747 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000748 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100749enddef
750
Bram Moolenaar04b12692020-05-04 23:24:44 +0200751def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000752 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200753 return 'nested ' .. arg
754 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000755 NestedDef(':def')->assert_equal('nested :def')
756
757 func NestedFunc(arg)
758 return 'nested ' .. a:arg
759 endfunc
760 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200761
Bram Moolenaar62aec932022-01-29 21:45:34 +0000762 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
763 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200764
Bram Moolenaar62aec932022-01-29 21:45:34 +0000765 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
766 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200767
Bram Moolenaar54021752020-12-06 18:50:36 +0100768 var lines =<< trim END
769 def Outer()
770 def Inner()
771 # comment
772 enddef
773 def Inner()
774 enddef
775 enddef
776 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000777 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100778
779 lines =<< trim END
780 def Outer()
781 def Inner()
782 # comment
783 enddef
784 def! Inner()
785 enddef
786 enddef
787 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000788 v9.CheckDefFailure(lines, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100789
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000790 lines =<< trim END
791 vim9script
792 def Outer()
793 def Inner()
794 g:result = 'ok'
795 enddef
796 Inner()
797 enddef
798 Outer()
799 Inner()
800 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000801 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000802 assert_equal('ok', g:result)
803 unlet g:result
804
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000805 lines =<< trim END
806 vim9script
807 def Outer()
808 def _Inner()
809 echo 'bad'
810 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000811 _Inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000812 enddef
813 defcompile
814 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000815 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000816
817 lines =<< trim END
818 vim9script
819 def Outer()
820 def g:inner()
821 echo 'bad'
822 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000823 g:inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000824 enddef
825 defcompile
826 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000827 v9.CheckScriptFailure(lines, 'E1267:')
828
829 lines =<< trim END
830 vim9script
831 def g:_Func()
832 echo 'bad'
833 enddef
834 END
835 v9.CheckScriptFailure(lines, 'E1267:')
836
837 lines =<< trim END
838 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +0000839 def _Func()
Bram Moolenaar3787f262022-02-07 21:54:01 +0000840 echo 'bad'
841 enddef
842 END
843 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000844
Bram Moolenaar54021752020-12-06 18:50:36 +0100845 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100846 lines =<< trim END
847 vim9script
848 var thecount = 0
849 if true
850 def Test(): number
851 def TheFunc(): number
852 thecount += 1
853 return thecount
854 enddef
855 return TheFunc()
856 enddef
857 endif
858 defcompile
859 assert_equal(1, Test())
860 assert_equal(2, Test())
861 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000862 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100863
864 # also works when "thecount" is inside the "if" block
865 lines =<< trim END
866 vim9script
867 if true
868 var thecount = 0
869 def Test(): number
870 def TheFunc(): number
871 thecount += 1
872 return thecount
873 enddef
874 return TheFunc()
875 enddef
876 endif
877 defcompile
878 assert_equal(1, Test())
879 assert_equal(2, Test())
880 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000881 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200882
Bram Moolenaara915fa02022-03-23 11:29:15 +0000883 # nested function with recursive call
884 lines =<< trim END
885 vim9script
886
887 def MyFunc(): number
888 def Fib(n: number): number
889 if n < 2
890 return 1
891 endif
892 return Fib(n - 2) + Fib(n - 1)
893 enddef
894
895 return Fib(5)
896 enddef
897
898 assert_equal(8, MyFunc())
899 END
900 v9.CheckScriptSuccess(lines)
901
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200902 lines =<< trim END
903 vim9script
904 def Outer()
905 def Inner()
906 echo 'hello'
907 enddef burp
908 enddef
909 defcompile
910 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000911 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200912enddef
913
Bram Moolenaar1889f492022-08-16 19:34:44 +0100914def Test_nested_function_fails()
915 var lines =<< trim END
916 def T()
917 def Func(g: string):string
918 enddef
919 Func()
920 enddef
921 silent! defcompile
922 END
923 v9.CheckScriptFailure(lines, 'E1069:')
924enddef
925
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100926def Test_not_nested_function()
927 echo printf('%d',
928 function('len')('xxx'))
929enddef
930
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200931func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200932 call MyDefaultArgs()->assert_equal('string')
933 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200934 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200935endfunc
936
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200937def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200938 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200939 vim9script
940 def Outer()
941 def g:Inner(): string
942 return 'inner'
943 enddef
944 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200945 defcompile
946 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200947 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200948 delfunc g:Inner
949 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200950 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200951 delfunc g:Inner
952 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200953 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200954 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200955 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000956 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200957
958 lines =<< trim END
959 vim9script
960 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000961 func g:Inner()
962 return 'inner'
963 endfunc
964 enddef
965 defcompile
966 Outer()
967 g:Inner()->assert_equal('inner')
968 delfunc g:Inner
969 Outer()
970 g:Inner()->assert_equal('inner')
971 delfunc g:Inner
972 Outer()
973 g:Inner()->assert_equal('inner')
974 delfunc g:Inner
975 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000976 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000977
978 lines =<< trim END
979 vim9script
980 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200981 def g:Inner(): string
982 return 'inner'
983 enddef
984 enddef
985 defcompile
986 Outer()
987 Outer()
988 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000989 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100990 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200991
992 lines =<< trim END
993 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100994 def Outer()
995 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100996 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100997 enddef
998 g:Inner()
999 enddef
1000 Outer()
1001 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001002 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +01001003 delfunc g:Inner
1004
1005 lines =<< trim END
1006 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +02001007 def Func()
1008 echo 'script'
1009 enddef
1010 def Outer()
1011 def Func()
1012 echo 'inner'
1013 enddef
1014 enddef
1015 defcompile
1016 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001017 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +00001018
1019 lines =<< trim END
1020 vim9script
1021 def Func()
1022 echo 'script'
1023 enddef
1024 def Func()
1025 echo 'script'
1026 enddef
1027 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001028 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001029enddef
1030
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001031def DefListAll()
1032 def
1033enddef
1034
1035def DefListOne()
1036 def DefListOne
1037enddef
1038
1039def DefListMatches()
1040 def /DefList
1041enddef
1042
1043def Test_nested_def_list()
1044 var funcs = split(execute('call DefListAll()'), "\n")
1045 assert_true(len(funcs) > 10)
1046 assert_true(funcs->index('def DefListAll()') >= 0)
1047
1048 funcs = split(execute('call DefListOne()'), "\n")
1049 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
1050
1051 funcs = split(execute('call DefListMatches()'), "\n")
1052 assert_true(len(funcs) >= 3)
1053 assert_true(funcs->index('def DefListAll()') >= 0)
1054 assert_true(funcs->index('def DefListOne()') >= 0)
1055 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +01001056
1057 var lines =<< trim END
1058 vim9script
1059 def Func()
1060 def +Func+
1061 enddef
1062 defcompile
1063 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001064 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001065enddef
1066
Bram Moolenaare08be092022-02-17 13:08:26 +00001067def Test_global_function_not_found()
1068 var lines =<< trim END
1069 g:Ref = 123
1070 call g:Ref()
1071 END
1072 v9.CheckDefExecAndScriptFailure(lines, ['E117:', 'E1085:'], 2)
1073enddef
1074
Bram Moolenaar333894b2020-08-01 18:53:07 +02001075def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001076 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +02001077 vim9script
1078 def g:Func(): string
1079 return 'global'
1080 enddef
1081 def Func(): string
1082 return 'local'
1083 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001084 g:Func()->assert_equal('global')
1085 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001086 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +02001087 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001088 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001089
1090 lines =<< trim END
1091 vim9script
1092 def g:Funcy()
1093 echo 'funcy'
1094 enddef
Bram Moolenaara749a422022-02-12 19:52:25 +00001095 Funcy()
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001096 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001097 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +02001098enddef
1099
Bram Moolenaar0f769812020-09-12 18:32:34 +02001100def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001101 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +02001102 vim9script
1103 def g:Gfunc(): string
1104 return 'global'
1105 enddef
1106 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001107 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001108 return Gfunc('testing')
1109 enddef
1110 g:Gfunc()->assert_equal('global')
1111 AnotherFunc()->assert_equal(7)
1112 delfunc g:Gfunc
1113 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001114 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001115
1116 lines =<< trim END
1117 vim9script
1118 def g:Func(): string
1119 return 'global'
1120 enddef
1121 def AnotherFunc()
1122 g:Func = function('len')
1123 enddef
1124 AnotherFunc()
1125 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001126 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001127 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001128
Bram Moolenaar62aec932022-01-29 21:45:34 +00001129 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +02001130 lines =<< trim END
1131 vim9script
1132 def g:Func(): string
1133 return 'global'
1134 enddef
1135 def AnotherFunc(): string
1136 return Func()
1137 enddef
1138 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001139 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001140 v9.CheckScriptFailure(lines, 'E117:')
1141 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001142
1143 lines =<< trim END
1144 vim9script
1145 def g:Func(): string
1146 return 'global'
1147 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001148 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001149 delfunc g:Func
1150 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001151 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +00001152
1153 # This does not shadow "i" which is visible only inside the for loop
1154 lines =<< trim END
1155 vim9script
1156
1157 def Foo(i: number)
1158 echo i
1159 enddef
1160
1161 for i in range(3)
1162 # Foo() is compiled here
1163 Foo(i)
1164 endfor
1165 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001166 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001167enddef
1168
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001169func TakesOneArg(arg)
1170 echo a:arg
1171endfunc
1172
1173def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001174 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
1175 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
1176 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
1177 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001178
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001179 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001180 vim9script
1181 def Func(s: string)
1182 echo s
1183 enddef
1184 Func([])
1185 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001186 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001187
Bram Moolenaar9a015112021-12-31 14:06:45 +00001188 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001189 lines =<< trim END
1190 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001191 var name = 'piet'
1192 def FuncOne(name: string)
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001193 echo name
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001194 enddef
1195 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001196 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001197
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001198 # same, inside the same block
1199 lines =<< trim END
1200 vim9script
1201 if true
1202 var name = 'piet'
1203 def FuncOne(name: string)
1204 echo name
1205 enddef
1206 endif
1207 END
1208 v9.CheckScriptFailure(lines, 'E1168:')
1209
1210 # variable in other block is OK
1211 lines =<< trim END
1212 vim9script
1213 if true
1214 var name = 'piet'
1215 endif
1216 def FuncOne(name: string)
1217 echo name
1218 enddef
1219 END
1220 v9.CheckScriptSuccess(lines)
1221
Bram Moolenaardce24412022-02-08 20:35:30 +00001222 # with another variable in another block
1223 lines =<< trim END
1224 vim9script
1225 if true
1226 var name = 'piet'
1227 # define a function so that the variable isn't cleared
1228 def GetItem(): string
1229 return item
1230 enddef
1231 endif
1232 if true
1233 var name = 'peter'
1234 def FuncOne(name: string)
1235 echo name
1236 enddef
1237 endif
1238 END
1239 v9.CheckScriptFailure(lines, 'E1168:')
1240
1241 # only variable in another block is OK
1242 lines =<< trim END
1243 vim9script
1244 if true
1245 var name = 'piet'
1246 # define a function so that the variable isn't cleared
1247 def GetItem(): string
1248 return item
1249 enddef
1250 endif
1251 if true
1252 def FuncOne(name: string)
1253 echo name
1254 enddef
1255 endif
1256 END
1257 v9.CheckScriptSuccess(lines)
1258
Bram Moolenaar9a015112021-12-31 14:06:45 +00001259 # argument name declared later is only found when compiling
1260 lines =<< trim END
1261 vim9script
1262 def FuncOne(name: string)
1263 echo nr
1264 enddef
1265 var name = 'piet'
1266 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001267 v9.CheckScriptSuccess(lines)
1268 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001269
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001270 lines =<< trim END
1271 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001272 def FuncOne(nr: number)
1273 echo nr
1274 enddef
1275 def FuncTwo()
1276 FuncOne()
1277 enddef
1278 defcompile
1279 END
1280 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001281 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001282 try
1283 source Xscript
1284 catch
1285 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
1286 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1287 didCatch = true
1288 endtry
1289 assert_true(didCatch)
1290
1291 lines =<< trim END
1292 vim9script
1293 def FuncOne(nr: number)
1294 echo nr
1295 enddef
1296 def FuncTwo()
1297 FuncOne(1, 2)
1298 enddef
1299 defcompile
1300 END
1301 writefile(lines, 'Xscript')
1302 didCatch = false
1303 try
1304 source Xscript
1305 catch
1306 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1307 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1308 didCatch = true
1309 endtry
1310 assert_true(didCatch)
1311
1312 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001313enddef
1314
Bram Moolenaar50824712020-12-20 21:10:17 +01001315def Test_call_funcref_wrong_args()
1316 var head =<< trim END
1317 vim9script
1318 def Func3(a1: string, a2: number, a3: list<number>)
1319 echo a1 .. a2 .. a3[0]
1320 enddef
1321 def Testme()
1322 var funcMap: dict<func> = {func: Func3}
1323 END
1324 var tail =<< trim END
1325 enddef
1326 Testme()
1327 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001328 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001329
Bram Moolenaar62aec932022-01-29 21:45:34 +00001330 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1331 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001332
1333 var lines =<< trim END
1334 vim9script
1335 var Ref: func(number): any
1336 Ref = (j) => !j
1337 echo Ref(false)
1338 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001339 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001340
1341 lines =<< trim END
1342 vim9script
1343 var Ref: func(number): any
1344 Ref = (j) => !j
1345 call Ref(false)
1346 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001347 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001348enddef
1349
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001350def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001351 var lines =<< trim END
1352 var Callback = (..._) => 'anything'
1353 assert_equal('anything', Callback())
1354 assert_equal('anything', Callback(1))
1355 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001356
1357 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001358 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001359 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001360
Bram Moolenaar62aec932022-01-29 21:45:34 +00001361 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001362 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001363
Bram Moolenaar2a389082021-04-09 20:24:31 +02001364 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001365 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001366 echo Ref(1, 'x')
1367 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001368 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001369
1370 lines =<< trim END
1371 var Ref: func(job, string, number)
1372 Ref = (x, y) => 0
1373 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001374 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001375
1376 lines =<< trim END
1377 var Ref: func(job, string)
1378 Ref = (x, y, z) => 0
1379 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001380 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001381
1382 lines =<< trim END
1383 var one = 1
1384 var l = [1, 2, 3]
1385 echo map(l, (one) => one)
1386 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001387 v9.CheckDefFailure(lines, 'E1167:')
1388 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001389
1390 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001391 var Ref: func(any, ?any): bool
1392 Ref = (_, y = 1) => false
1393 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001394 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001395
1396 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001397 var a = 0
1398 var b = (a == 0 ? 1 : 2)
1399 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001400 var txt = 'a'
1401 b = (txt =~ 'x' ? 1 : 2)
1402 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001403 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001404 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001405
1406 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001407 def ShadowLocal()
1408 var one = 1
1409 var l = [1, 2, 3]
1410 echo map(l, (one) => one)
1411 enddef
1412 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001413 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001414
1415 lines =<< trim END
1416 def Shadowarg(one: number)
1417 var l = [1, 2, 3]
1418 echo map(l, (one) => one)
1419 enddef
1420 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001421 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001422
1423 lines =<< trim END
1424 echo ((a) => a)('aa', 'bb')
1425 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001426 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001427
1428 lines =<< trim END
1429 echo 'aa'->((a) => a)('bb')
1430 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001431 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1432 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001433enddef
1434
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001435def Test_lambda_line_nr()
1436 var lines =<< trim END
1437 vim9script
1438 # comment
1439 # comment
1440 var id = timer_start(1'000, (_) => 0)
1441 var out = execute('verbose ' .. timer_info(id)[0].callback
1442 ->string()
1443 ->substitute("('\\|')", ' ', 'g'))
1444 assert_match('Last set from .* line 4', out)
1445 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001446 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001447enddef
1448
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001449def FilterWithCond(x: string, Cond: func(string): bool): bool
1450 return Cond(x)
1451enddef
1452
Bram Moolenaar0346b792021-01-31 22:18:29 +01001453def Test_lambda_return_type()
1454 var lines =<< trim END
1455 var Ref = (): => 123
1456 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001457 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001458
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001459 # no space before the return type
1460 lines =<< trim END
1461 var Ref = (x):number => x + 1
1462 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001463 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001464
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001465 # this works
1466 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001467 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001468 endfor
1469
1470 # this fails
1471 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001472 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001473 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001474 v9.CheckDefAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected func(string): bool but got func(any): string', 1)
Bram Moolenaara9931532021-06-12 15:58:16 +02001475
1476 lines =<< trim END
1477 var Lambda1 = (x) => {
1478 return x
1479 }
1480 assert_equal('asdf', Lambda1('asdf'))
1481 var Lambda2 = (x): string => {
1482 return x
1483 }
1484 assert_equal('foo', Lambda2('foo'))
1485 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001486 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001487
1488 lines =<< trim END
1489 var Lambda = (x): string => {
1490 return x
1491 }
1492 echo Lambda(['foo'])
1493 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001494 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001495enddef
1496
Bram Moolenaar709664c2020-12-12 14:33:41 +01001497def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001498 v9.CheckDefSuccess([
Bram Moolenaar2984ed32022-08-20 14:51:17 +01001499 'var x: any = "aaa"',
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001500 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001501enddef
1502
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001503def Test_pass_legacy_lambda_to_def_func()
1504 var lines =<< trim END
1505 vim9script
1506 func Foo()
1507 eval s:Bar({x -> 0})
1508 endfunc
1509 def Bar(y: any)
1510 enddef
1511 Foo()
1512 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001513 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001514
1515 lines =<< trim END
1516 vim9script
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001517 def g:TestFunc(F: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001518 enddef
1519 legacy call g:TestFunc({-> 0})
1520 delfunc g:TestFunc
1521
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001522 def g:TestFunc(F: func(number))
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001523 enddef
1524 legacy call g:TestFunc({nr -> 0})
1525 delfunc g:TestFunc
1526 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001527 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001528enddef
1529
Bram Moolenaar844fb642021-10-23 13:32:30 +01001530def Test_lambda_in_reduce_line_break()
1531 # this was using freed memory
1532 var lines =<< trim END
1533 vim9script
1534 const result: dict<number> =
1535 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1536 ->reduce((acc, val) => {
1537 if has_key(acc, val)
1538 acc[val] += 1
1539 return acc
1540 else
1541 acc[val] = 1
1542 return acc
1543 endif
1544 }, {})
1545 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1546 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001547 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001548enddef
1549
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001550def Test_set_opfunc_to_lambda()
1551 var lines =<< trim END
1552 vim9script
1553 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1554 def CountSpaces(type = ''): string
1555 if type == ''
1556 &operatorfunc = (t) => CountSpaces(t)
1557 return 'g@'
1558 endif
1559 normal! '[V']y
1560 g:result = getreg('"')->count(' ')
1561 return ''
1562 enddef
1563 new
1564 'a b c d e'->setline(1)
1565 feedkeys("\<F4>", 'x')
1566 assert_equal(4, g:result)
1567 bwipe!
1568 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001569 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001570enddef
1571
Bram Moolenaaref082e12021-12-12 21:02:03 +00001572def Test_set_opfunc_to_global_function()
1573 var lines =<< trim END
1574 vim9script
1575 def g:CountSpaces(type = ''): string
1576 normal! '[V']y
1577 g:result = getreg('"')->count(' ')
1578 return ''
1579 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001580 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001581 &operatorfunc = g:CountSpaces
1582 new
1583 'a b c d e'->setline(1)
1584 feedkeys("g@_", 'x')
1585 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001586
1587 &operatorfunc = ''
1588 g:result = 0
1589 # global function works in :def function
1590 def Func()
1591 &operatorfunc = g:CountSpaces
1592 enddef
1593 Func()
1594 feedkeys("g@_", 'x')
1595 assert_equal(4, g:result)
1596
Bram Moolenaaref082e12021-12-12 21:02:03 +00001597 bwipe!
1598 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001599 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001600 &operatorfunc = ''
1601enddef
1602
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001603def Test_use_script_func_name_with_prefix()
1604 var lines =<< trim END
1605 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +00001606 func g:Getit()
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001607 return 'it'
1608 endfunc
Bram Moolenaara749a422022-02-12 19:52:25 +00001609 var Fn = g:Getit
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001610 assert_equal('it', Fn())
1611 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001612 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001613enddef
1614
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001615def Test_lambda_type_allocated()
1616 # Check that unreferencing a partial using a lambda can use the variable type
1617 # after the lambda has been freed and does not leak memory.
1618 var lines =<< trim END
1619 vim9script
1620
1621 func MyomniFunc1(val, findstart, base)
1622 return a:findstart ? 0 : []
1623 endfunc
1624
1625 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1626 &omnifunc = Lambda
1627 Lambda = (a, b) => MyomniFunc1(20, a, b)
1628 &omnifunc = string(Lambda)
1629 Lambda = (a, b) => strlen(a)
1630 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001631 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001632enddef
1633
Bram Moolenaara7583c42022-05-07 21:14:05 +01001634def Test_define_lambda_in_execute()
1635 var lines =<< trim [CODE]
1636 vim9script
1637
1638 def BuildFuncMultiLine(): func
1639 var x =<< trim END
1640 g:SomeRandomFunc = (d: dict<any>) => {
1641 return d.k1 + d.k2
1642 }
1643 END
1644 execute(x)
1645 return g:SomeRandomFunc
1646 enddef
1647 var ResultPlus = BuildFuncMultiLine()
1648 assert_equal(7, ResultPlus({k1: 3, k2: 4}))
1649 [CODE]
1650 v9.CheckScriptSuccess(lines)
1651 unlet g:SomeRandomFunc
1652enddef
1653
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001654" Default arg and varargs
1655def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001656 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001657 for s in rest
1658 res ..= ',' .. s
1659 endfor
1660 return res
1661enddef
1662
1663def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001664 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1665 g:MyDefVarargs('one')->assert_equal('one,foo')
1666 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1667 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1668 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001669 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001670 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001671 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001672
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001673 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001674 vim9script
1675 def Func(...l: list<string>)
1676 echo l
1677 enddef
1678 Func('a', 'b', 'c')
1679 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001680 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001681
1682 lines =<< trim END
1683 vim9script
1684 def Func(...l: list<string>)
1685 echo l
1686 enddef
1687 Func()
1688 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001689 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001690
1691 lines =<< trim END
1692 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001693 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001694 echo l
1695 enddef
1696 Func(0)
1697 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001698 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001699
1700 lines =<< trim END
1701 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001702 def Func(...l: any)
1703 echo l
1704 enddef
1705 Func(0)
1706 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001707 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001708
1709 lines =<< trim END
1710 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001711 def Func(..._l: list<string>)
1712 echo _l
1713 enddef
1714 Func('a', 'b', 'c')
1715 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001716 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001717
1718 lines =<< trim END
1719 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001720 def Func(...l: list<string>)
1721 echo l
1722 enddef
1723 Func(1, 2, 3)
1724 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001725 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001726
1727 lines =<< trim END
1728 vim9script
1729 def Func(...l: list<string>)
1730 echo l
1731 enddef
1732 Func('a', 9)
1733 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001734 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001735
1736 lines =<< trim END
1737 vim9script
1738 def Func(...l: list<string>)
1739 echo l
1740 enddef
1741 Func(1, 'a')
1742 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001743 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001744
1745 lines =<< trim END
1746 vim9script
1747 def Func( # some comment
1748 ...l = []
1749 )
1750 echo l
1751 enddef
1752 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001753 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001754
1755 lines =<< trim END
1756 vim9script
1757 def DoIt()
1758 g:Later('')
1759 enddef
1760 defcompile
1761 def g:Later(...l: list<number>)
1762 enddef
1763 DoIt()
1764 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001765 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001766enddef
1767
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001768let s:value = ''
1769
1770def FuncOneDefArg(opt = 'text')
1771 s:value = opt
1772enddef
1773
1774def FuncTwoDefArg(nr = 123, opt = 'text'): string
1775 return nr .. opt
1776enddef
1777
1778def FuncVarargs(...arg: list<string>): string
1779 return join(arg, ',')
1780enddef
1781
1782def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001783 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001784 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001785 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001786 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001787 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001788 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001789
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001790 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001791 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001792 RefDef2Arg()->assert_equal('123text')
1793 RefDef2Arg(99)->assert_equal('99text')
1794 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001795
Bram Moolenaar62aec932022-01-29 21:45:34 +00001796 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1797 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001798
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001799 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001800 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001801 RefVarargs()->assert_equal('')
1802 RefVarargs('one')->assert_equal('one')
1803 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001804
Bram Moolenaar62aec932022-01-29 21:45:34 +00001805 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1806 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001807enddef
1808
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001809" Only varargs
1810def MyVarargsOnly(...args: list<string>): string
1811 return join(args, ',')
1812enddef
1813
1814def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001815 g:MyVarargsOnly()->assert_equal('')
1816 g:MyVarargsOnly('one')->assert_equal('one')
1817 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1818 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1819 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001820enddef
1821
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001822def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001823 var lines =<< trim END
1824 def Func(x: number)
1825 var x = 234
1826 enddef
1827 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001828 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001829
1830 lines =<< trim END
1831 def Func(Ref: number)
1832 def Ref()
1833 enddef
1834 enddef
1835 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001836 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001837enddef
1838
Bram Moolenaar62aec932022-01-29 21:45:34 +00001839def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001840 arg['key'] = 'value'
1841enddef
1842
Bram Moolenaar62aec932022-01-29 21:45:34 +00001843def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001844 arg[0] = 'value'
1845enddef
1846
1847def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001848 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001849 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001850 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001851 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001852 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001853 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001854 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001855
Bram Moolenaar62aec932022-01-29 21:45:34 +00001856 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001857 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001858enddef
1859
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001860" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001861def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001862 return firstline .. lastline
1863enddef
1864
1865def Test_argument_names()
1866 assert_equal('OK', WithReservedNames('O', 'K'))
1867enddef
1868
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001869def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001870 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001871 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001872enddef
1873
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001874func DefinedLater(arg)
1875 return a:arg
1876endfunc
1877
1878def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001879 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001880 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001881 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001882
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001883 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001884 vim9script
1885 def RetNumber(): number
1886 return 123
1887 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001888 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001889 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001890 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001891 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001892
1893 lines =<< trim END
1894 vim9script
1895 def RetNumber(): number
1896 return 123
1897 enddef
1898 def Bar(F: func: number): number
1899 return F()
1900 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001901 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001902 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001903 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001904 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001905
1906 lines =<< trim END
1907 vim9script
1908 def UseNumber(nr: number)
1909 echo nr
1910 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001911 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001912 Funcref(123)
1913 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001914 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001915
1916 lines =<< trim END
1917 vim9script
1918 def UseNumber(nr: number)
1919 echo nr
1920 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001921 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001922 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001923 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001924
1925 lines =<< trim END
1926 vim9script
1927 def EchoNr(nr = 34)
1928 g:echo = nr
1929 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001930 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001931 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001932 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001933 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001934 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001935 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001936 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001937
1938 lines =<< trim END
1939 vim9script
1940 def EchoList(...l: list<number>)
1941 g:echo = l
1942 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001943 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001944 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001945 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001946 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001947 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001948 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001949 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001950
1951 lines =<< trim END
1952 vim9script
1953 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1954 g:optarg = opt
1955 g:listarg = l
1956 return nr
1957 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001958 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001959 Funcref(10)->assert_equal(10)
1960 g:optarg->assert_equal(12)
1961 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001962
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001963 Funcref(11, 22)->assert_equal(11)
1964 g:optarg->assert_equal(22)
1965 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001966
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001967 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1968 g:optarg->assert_equal(18)
1969 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001970 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001971 v9.CheckScriptSuccess(lines)
Kota Kato948a3892022-08-16 16:09:59 +01001972
1973 lines =<< trim END
1974 function s:func(num)
1975 return a:num * 2
1976 endfunction
1977
1978 def s:CallFuncref()
1979 var Funcref = function('s:func')
1980 Funcref(3)->assert_equal(6)
1981 enddef
1982 call s:CallFuncref()
1983 END
1984 v9.CheckScriptSuccess(lines)
1985
1986 lines =<< trim END
1987 function s:func(num)
1988 return a:num * 2
1989 endfunction
1990
1991 def s:CallFuncref()
1992 var Funcref = function(s:func)
1993 Funcref(3)->assert_equal(6)
1994 enddef
1995 call s:CallFuncref()
1996 END
1997 v9.CheckScriptSuccess(lines)
1998
1999 lines =<< trim END
2000 function s:func(num)
2001 return a:num * 2
2002 endfunction
2003
2004 def s:CallFuncref()
2005 var Funcref = s:func
2006 Funcref(3)->assert_equal(6)
2007 enddef
2008 call s:CallFuncref()
2009 END
2010 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02002011enddef
2012
2013let SomeFunc = function('len')
2014let NotAFunc = 'text'
2015
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002016def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002017 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002018 var Ref1: func(bool): string
2019 var Ref2: func(bool): number
2020 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002021 Ref3 = g:cond ? Ref1 : Ref2
2022
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002023 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002024 var Refa1: func(bool): number
2025 var Refa2: func(bool, number): number
2026 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002027 Refa3 = g:cond ? Refa1 : Refa2
2028
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002029 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002030 var Refb1: func(bool, string): number
2031 var Refb2: func(string, number): number
2032 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002033 Refb3 = g:cond ? Refb1 : Refb2
2034enddef
2035
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002036def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02002037 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002038enddef
2039
2040def DefinedEvenLater(arg: string): string
2041 return arg
2042enddef
2043
2044def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002045 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002046 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002047enddef
2048
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002049def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002050 var lines =<< trim END
2051 vim9script
2052 # Define an outer function
2053 def FirstFunction()
2054 # Define an inner function
2055 def SecondFunction()
2056 # the function has a body, a double free is detected.
2057 AAAAA
2058
2059 # enddef followed by | or } followed by # one or more characters
2060 enddef|BBBB
2061 enddef
2062
2063 # Compile all functions
2064 defcompile
2065 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002066 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002067enddef
2068
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002069def Test_nested_function_with_args_split()
2070 var lines =<< trim END
2071 vim9script
2072 def FirstFunction()
2073 def SecondFunction(
2074 )
2075 # had a double free if the right parenthesis of the nested function is
2076 # on the next line
2077
2078 enddef|BBBB
2079 enddef
2080 # Compile all functions
2081 defcompile
2082 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002083 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00002084
2085 lines =<< trim END
2086 vim9script
2087 def FirstFunction()
2088 func SecondFunction()
2089 endfunc|BBBB
2090 enddef
2091 defcompile
2092 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002093 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002094enddef
2095
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002096def Test_error_in_function_args()
2097 var lines =<< trim END
2098 def FirstFunction()
2099 def SecondFunction(J =
2100 # Nois
2101 # one
2102
2103 enddef|BBBB
2104 enddef
2105 # Compile all functions
2106 defcompile
2107 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002108 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002109enddef
2110
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002111def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002112 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002113 'def Func(): number',
2114 'return "a"',
2115 'enddef',
2116 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002117 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002118 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002119 'def Func(): string',
2120 'return 1',
2121 'enddef',
2122 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002123 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002124 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002125 'def Func(): void',
2126 'return "a"',
2127 'enddef',
2128 'defcompile'],
2129 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002130 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002131 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002132 'def Func()',
2133 'return "a"',
2134 'enddef',
2135 'defcompile'],
2136 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002137 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002138
Bram Moolenaar62aec932022-01-29 21:45:34 +00002139 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002140 'def Func(): number',
2141 'return',
2142 'enddef',
2143 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002144 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002145
Bram Moolenaar62aec932022-01-29 21:45:34 +00002146 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002147 'def Func():number',
2148 'return 123',
2149 'enddef',
2150 'defcompile'], 'E1069:')
2151 delfunc! g:Func
2152
Bram Moolenaar62aec932022-01-29 21:45:34 +00002153 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002154 'def Func() :number',
2155 'return 123',
2156 'enddef',
2157 'defcompile'], 'E1059:')
2158 delfunc! g:Func
2159
Bram Moolenaar62aec932022-01-29 21:45:34 +00002160 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002161 'def Func() : number',
2162 'return 123',
2163 'enddef',
2164 'defcompile'], 'E1059:')
2165 delfunc! g:Func
2166
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002167 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008: Missing <type> after list')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002168 delfunc! g:Func
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002169 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008: Missing <type> after dict')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002170 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002171 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002172 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002173
Bram Moolenaar62aec932022-01-29 21:45:34 +00002174 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002175 'vim9script',
2176 'def FuncB()',
2177 ' return 123',
2178 'enddef',
2179 'def FuncA()',
2180 ' FuncB()',
2181 'enddef',
2182 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002183enddef
2184
2185def Test_arg_type_wrong()
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002186 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type> after list')
Bram Moolenaar62aec932022-01-29 21:45:34 +00002187 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
2188 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
2189 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
2190 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
2191 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002192enddef
2193
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002194def Test_white_space_before_comma()
2195 var lines =<< trim END
2196 vim9script
2197 def Func(a: number , b: number)
2198 enddef
2199 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002200 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02002201 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002202enddef
2203
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002204def Test_white_space_after_comma()
2205 var lines =<< trim END
2206 vim9script
2207 def Func(a: number,b: number)
2208 enddef
2209 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002210 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002211
2212 # OK in legacy function
2213 lines =<< trim END
2214 vim9script
2215 func Func(a,b)
2216 endfunc
2217 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002218 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002219enddef
2220
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002221def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002222 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002223 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002224 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002225 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002226 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002227 enddef
2228 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002229 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002230
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002231 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002232 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002233 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002234
Bram Moolenaar67979662020-06-20 22:50:47 +02002235 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002236 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002237 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002238
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002239 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002240 def ListFunc(arg: list<number>)
2241 listvar = arg
2242 enddef
2243 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002244 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002245
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002246 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002247 def DictFunc(arg: dict<number>)
2248 dictvar = arg
2249 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01002250 {a: 1, b: 2}->DictFunc()
2251 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002252 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002253 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002254 enddef
2255 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002256 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002257
Bram Moolenaare0de1712020-12-02 17:36:54 +01002258 {a: 3, b: 4}->DictFunc()
2259 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002260
2261 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002262 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002263 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002264 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02002265
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002266 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02002267 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002268 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002269 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02002270 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002271 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002272
2273 def UseString()
2274 'xyork'->MyFunc()
2275 enddef
2276 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002277 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002278
Bram Moolenaar10409562020-07-29 20:00:38 +02002279 def UseString2()
2280 "knife"->MyFunc()
2281 enddef
2282 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002283 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02002284
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002285 # prepending a colon makes it a mark
2286 new
2287 setline(1, ['aaa', 'bbb', 'ccc'])
2288 normal! 3Gmt1G
2289 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002290 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002291 bwipe!
2292
Bram Moolenaare6b53242020-07-01 17:28:33 +02002293 MyFunc(
2294 'continued'
2295 )
2296 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002297 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002298 )
2299
2300 call MyFunc(
2301 'more'
2302 ..
2303 'lines'
2304 )
2305 assert_equal(
2306 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002307 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002308 END
2309 writefile(lines, 'Xcall.vim')
2310 source Xcall.vim
2311 delete('Xcall.vim')
2312enddef
2313
2314def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002315 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002316 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002317 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002318 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002319 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002320 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002321 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002322 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002323 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002324enddef
2325
Bram Moolenaar65b95452020-07-19 14:03:09 +02002326def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002327 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002328 vim9script
2329 def MyFunc(arg: string)
2330 echo arg
2331 enddef
2332 MyFunc(1234)
2333 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002334 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002335enddef
2336
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002337def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002338 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002339 vim9script
2340 const var = ''
2341 def MyFunc(arg: string)
2342 var = 'asdf'
2343 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002344 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002345 END
2346 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002347 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002348 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002349
2350 lines =<< trim END
2351 const g:Aconst = 77
2352 def Change()
2353 # comment
2354 g:Aconst = 99
2355 enddef
2356 call Change()
2357 unlet g:Aconst
2358 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002359 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002360enddef
2361
2362" Test that inside :function a Python function can be defined, :def is not
2363" recognized.
2364func Test_function_python()
2365 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002366 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002367 execute py "<< EOF"
2368def do_something():
2369 return 1
2370EOF
2371endfunc
2372
2373def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002374 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002375 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002376 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002377 echo 'hello'
2378 enddef
2379
2380 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002381 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002382 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002383 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002384
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002385 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002386 CallGoneSoon()
2387 END
2388 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002389 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2390 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002391
2392 delete('XToDelFunc')
2393enddef
2394
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002395func Test_free_dict_while_in_funcstack()
2396 " relies on the sleep command
2397 CheckUnix
2398 call Run_Test_free_dict_while_in_funcstack()
2399endfunc
2400
2401def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002402 # this was freeing the TermRun() default argument dictionary while it was
2403 # still referenced in a funcstack_T
2404 var lines =<< trim END
2405 vim9script
2406
2407 &updatetime = 400
2408 def TermRun(_ = {})
2409 def Post()
2410 enddef
2411 def Exec()
2412 term_start('sleep 1', {
2413 term_finish: 'close',
2414 exit_cb: (_, _) => Post(),
2415 })
2416 enddef
2417 Exec()
2418 enddef
2419 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2420 timer_start(100, (_) => feedkeys("\<F4>"))
2421 timer_start(1000, (_) => feedkeys("\<F4>"))
2422 sleep 1500m
2423 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002424 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002425 nunmap <F4>
2426 set updatetime&
2427enddef
2428
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002429def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002430 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002431 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002432 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002433 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002434 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002435 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002436 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002437 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002438 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002439
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002440 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002441 g:Func1()->assert_equal('Func1')
2442 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002443
2444 delfunc! Func0
2445 delfunc! Func1
2446 delfunc! Func2
2447enddef
2448
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002449def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002450 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002451 vim9script
2452 func Func(arg)
2453 echo a:arg
2454 endfunc
2455 Func('text')
2456 END
2457 writefile(lines, 'XVim9Func')
2458 so XVim9Func
2459
2460 delete('XVim9Func')
2461enddef
2462
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002463let s:funcResult = 0
2464
2465def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002466 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002467enddef
2468
2469def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002470 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002471 return 1234
2472enddef
2473
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002474def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002475 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002476 return 'text'
2477enddef
2478
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002479def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002480 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002481enddef
2482
2483def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002484 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002485 return arg
2486enddef
2487
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002488def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002489 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002490enddef
2491
Bram Moolenaar62aec932022-01-29 21:45:34 +00002492def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002493 return arg
2494enddef
2495
Bram Moolenaar62aec932022-01-29 21:45:34 +00002496def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002497 return arg
2498enddef
2499
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002500def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002501 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002502 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002503 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002504 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002505 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002506
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002507 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002508 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002509 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002510 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002511 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002512
Bram Moolenaar53900992020-08-22 19:02:02 +02002513 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002514 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002515 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002516 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002517
Bram Moolenaar53900992020-08-22 19:02:02 +02002518 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002519 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002520 Ref2()->assert_equal(1234)
2521 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002522
Bram Moolenaar53900992020-08-22 19:02:02 +02002523 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002524 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002525 Ref2(13)->assert_equal(13)
2526 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002527enddef
2528
Bram Moolenaar9978d472020-07-05 16:01:56 +02002529def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002530 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002531 for n in repeat([1], 3)
2532 res += n
2533 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002534 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002535
2536 res = 0
Bakudankun375141e2022-09-09 18:46:47 +01002537 for n in repeat(0z01, 3)->blob2list()
2538 res += n
2539 endfor
2540 res->assert_equal(3)
2541
2542 res = 0
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002543 for n in add([1, 2], 3)
2544 res += n
2545 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002546 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002547enddef
2548
Bram Moolenaar846178a2020-07-05 17:04:13 +02002549def Test_argv_return_type()
2550 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002551 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002552 for name in argv()
2553 res ..= name
2554 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002555 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002556enddef
2557
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002558def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002559 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002560 RefVoid = g:FuncNoArgNoRet
2561 RefVoid = g:FuncOneArgNoRet
2562 v9.CheckDefFailure(['var RefVoid: func: void', 'RefVoid = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...) but got func(): number')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002563 v9.CheckDefFailure(['var RefVoid: func: void', 'RefVoid = g:FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...) but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002564
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002565 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002566 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002567 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002568 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2569 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func(number)')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002570
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002571 var RefAnyNoArgs: func: any = RefAny
2572
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002573 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002574 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002575 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002576 v9.CheckDefFailure(['var RefNr: func: number', 'RefNr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): number but got func()')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002577 v9.CheckDefFailure(['var RefNr: func: number', 'RefNr = g:FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...): number but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002578
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002579 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002580 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002581 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002582 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2583 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...): string but got func(): number')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002584enddef
2585
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002586def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002587 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002588
Bram Moolenaar62aec932022-01-29 21:45:34 +00002589 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2590 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002591 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(number): number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00002592 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2593 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2594 v9.CheckDefFailure(['var Ref1: func(...bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(...bool) but got func(bool, number)')
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002595
Bram Moolenaar62aec932022-01-29 21:45:34 +00002596 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2597 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2598 v9.CheckDefFailure(['var RefWrong: func(bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool)'], 'E1005:')
2599 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002600enddef
2601
Bram Moolenaar89228602020-04-05 22:14:54 +02002602def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002603 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002604 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002605 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002606
2607 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002608 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002609
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002610 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002611 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002612 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002613
Bram Moolenaar62aec932022-01-29 21:45:34 +00002614 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002615enddef
2616
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002617def Test_func_common_type()
2618 def FuncOne(n: number): number
2619 return n
2620 enddef
2621 def FuncTwo(s: string): number
2622 return len(s)
2623 enddef
2624 def FuncThree(n: number, s: string): number
2625 return n + len(s)
2626 enddef
2627 var list = [FuncOne, FuncTwo, FuncThree]
2628 assert_equal(8, list[0](8))
2629 assert_equal(4, list[1]('word'))
2630 assert_equal(7, list[2](3, 'word'))
2631enddef
2632
Bram Moolenaar62aec932022-01-29 21:45:34 +00002633def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002634 arg1: string,
2635 arg2 = 1234,
2636 ...rest: list<string>
2637 ): string
2638 return arg1 .. arg2 .. join(rest, '-')
2639enddef
2640
Bram Moolenaar2c330432020-04-13 14:41:35 +02002641def MultiLineComment(
2642 arg1: string, # comment
2643 arg2 = 1234, # comment
2644 ...rest: list<string> # comment
2645 ): string # comment
2646 return arg1 .. arg2 .. join(rest, '-')
2647enddef
2648
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002649def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002650 MultiLine('text')->assert_equal('text1234')
2651 MultiLine('text', 777)->assert_equal('text777')
2652 MultiLine('text', 777, 'one')->assert_equal('text777one')
2653 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002654enddef
2655
Bram Moolenaar23e03252020-04-12 22:22:31 +02002656func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002657 call s:MultiLine('text')->assert_equal('text1234')
2658 call s:MultiLine('text', 777)->assert_equal('text777')
2659 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2660 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002661endfunc
2662
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002663
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002664" When using CheckScriptFailure() for the below test, E1010 is generated instead
2665" of E1056.
2666func Test_E1056_1059()
2667 let caught_1056 = 0
2668 try
2669 def F():
2670 return 1
2671 enddef
2672 catch /E1056:/
2673 let caught_1056 = 1
2674 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002675 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002676
2677 let caught_1059 = 0
2678 try
2679 def F5(items : list)
2680 echo 'a'
2681 enddef
2682 catch /E1059:/
2683 let caught_1059 = 1
2684 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002685 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002686endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002687
Bram Moolenaar015f4262020-05-05 21:25:22 +02002688func DelMe()
2689 echo 'DelMe'
2690endfunc
2691
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002692def Test_error_reporting()
2693 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002694 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002695 " comment
2696 def Func()
2697 # comment
2698 # comment
2699 invalid
2700 enddef
2701 defcompile
2702 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002703 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002704 try
2705 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002706 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002707 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002708 v:exception->assert_match('Invalid command: invalid')
2709 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002710 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002711 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002712
2713 # comment lines after the start of the function
2714 lines =<< trim END
2715 " comment
2716 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002717 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002718 # comment
2719 # comment
2720 invalid
2721 enddef
2722 defcompile
2723 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002724 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002725 try
2726 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002727 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002728 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002729 v:exception->assert_match('Invalid command: invalid')
2730 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002731 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002732 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002733
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002734 lines =<< trim END
2735 vim9script
2736 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002737 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002738 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002739 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002740 enddef
2741 defcompile
2742 Func()
2743 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002744 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002745 try
2746 source Xdef
2747 assert_report('should have failed')
2748 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002749 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002750 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002751 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002752
Bram Moolenaar08052222020-09-14 17:04:31 +02002753 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002754enddef
2755
Bram Moolenaar015f4262020-05-05 21:25:22 +02002756def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002757 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002758 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002759 'delfunc g:DelMe',
2760 'echo RefMe()'], 'E117:')
2761enddef
2762
2763def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002764 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002765 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002766 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002767enddef
2768
Bram Moolenaar62aec932022-01-29 21:45:34 +00002769def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002770 return Ref('more')
2771enddef
2772
2773def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002774 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002775 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002776enddef
2777
Bram Moolenaar62aec932022-01-29 21:45:34 +00002778def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002779 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002780 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002781enddef
2782
2783def Test_closure_ref_after_return()
2784 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002785 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002786 unlet g:Ref
2787enddef
2788
Bram Moolenaar62aec932022-01-29 21:45:34 +00002789def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002790 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002791 g:Extend = (s) => local->add(s)
2792 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002793enddef
2794
2795def Test_closure_two_refs()
2796 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002797 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002798 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002799 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002800 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002801 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002802
2803 unlet g:Extend
2804 unlet g:Read
2805enddef
2806
Bram Moolenaar62aec932022-01-29 21:45:34 +00002807def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002808 return join(Ref(), ' ')
2809enddef
2810
Bram Moolenaar62aec932022-01-29 21:45:34 +00002811def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002812 Ref(add)
2813enddef
2814
2815def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002816 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002817 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002818 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002819 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002820 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002821 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002822
2823 unlet g:Extend
2824 unlet g:Read
2825enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002826
Bram Moolenaar62aec932022-01-29 21:45:34 +00002827def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002828 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002829 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002830enddef
2831
Bram Moolenaar62aec932022-01-29 21:45:34 +00002832def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002833 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002834 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002835enddef
2836
2837def Test_closure_using_argument()
2838 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002839 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002840
2841 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002842 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002843
2844 unlet g:UseArg
2845 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002846
2847 var lines =<< trim END
2848 vim9script
2849 def Test(Fun: func(number): number): list<number>
2850 return map([1, 2, 3], (_, i) => Fun(i))
2851 enddef
2852 def Inc(nr: number): number
2853 return nr + 2
2854 enddef
2855 assert_equal([3, 4, 5], Test(Inc))
2856 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002857 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002858enddef
2859
Bram Moolenaar62aec932022-01-29 21:45:34 +00002860def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002861 var local = 'a'
2862
2863 def Append(arg: string)
2864 local ..= arg
2865 enddef
2866 g:Append = Append
2867
2868 def Get(): string
2869 return local
2870 enddef
2871 g:Get = Get
2872enddef
2873
2874def Test_closure_append_get()
2875 MakeGetAndAppendRefs()
2876 g:Get()->assert_equal('a')
2877 g:Append('-b')
2878 g:Get()->assert_equal('a-b')
2879 g:Append('-c')
2880 g:Get()->assert_equal('a-b-c')
2881
2882 unlet g:Append
2883 unlet g:Get
2884enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002885
Bram Moolenaar04b12692020-05-04 23:24:44 +02002886def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002887 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002888 def Closure(arg: string): string
2889 return local .. arg
2890 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002891 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002892enddef
2893
Bram Moolenaar62aec932022-01-29 21:45:34 +00002894func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002895 return a:Ref('some')
2896endfunc
2897
2898def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002899 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002900 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002901 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002902enddef
2903
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002904def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002905 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002906 vim9script
2907 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002908 var name = 0
2909 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002910 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002911 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002912 enddef
2913 Func()
2914 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002915 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002916enddef
2917
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002918def Test_nested_closure_used()
2919 var lines =<< trim END
2920 vim9script
2921 def Func()
2922 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002923 var Closure = () => x
2924 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002925 enddef
2926 Func()
2927 assert_equal('hello', g:Myclosure())
2928 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002929 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002930enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002931
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002932def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002933 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002934 vim9script
2935 def FuncA()
2936 FuncB(0)
2937 enddef
2938 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002939 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002940 enddef
2941 FuncA()
2942 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002943 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002944enddef
2945
Bram Moolenaar6de22962022-09-09 21:35:36 +01002946def Run_Test_closure_in_for_loop_fails()
2947 var lines =<< trim END
2948 vim9script
2949 for n in [0]
2950 timer_start(10, (_) => {
2951 echo n
2952 })
2953 endfor
2954 END
2955 writefile(lines, 'XTest_closure_fails', 'D')
2956
2957 # Check that an error shows
2958 var buf = g:RunVimInTerminal('-S XTest_closure_fails', {'rows': 6})
2959 g:VerifyScreenDump(buf, 'Test_vim9_closure_fails', {})
2960
2961 # clean up
2962 g:StopVimInTerminal(buf)
2963enddef
2964
2965func Test_closure_in_for_loop_fails()
2966 CheckScreendump
2967 call Run_Test_closure_in_for_loop_fails()
2968endfunc
2969
Bram Moolenaarf112f302020-12-20 17:47:52 +01002970def Test_global_closure()
2971 var lines =<< trim END
2972 vim9script
2973 def ReverseEveryNLines(n: number, line1: number, line2: number)
2974 var mods = 'sil keepj keepp lockm '
2975 var range = ':' .. line1 .. ',' .. line2
2976 def g:Offset(): number
2977 var offset = (line('.') - line1 + 1) % n
2978 return offset != 0 ? offset : n
2979 enddef
2980 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2981 enddef
2982
2983 new
2984 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2985 ReverseEveryNLines(3, 1, 9)
2986 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002987 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01002988 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2989 assert_equal(expected, getline(1, 9))
2990 bwipe!
2991enddef
2992
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002993def Test_global_closure_called_directly()
2994 var lines =<< trim END
2995 vim9script
2996 def Outer()
2997 var x = 1
2998 def g:Inner()
2999 var y = x
3000 x += 1
3001 assert_equal(1, y)
3002 enddef
3003 g:Inner()
3004 assert_equal(2, x)
3005 enddef
3006 Outer()
3007 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003008 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01003009 delfunc g:Inner
3010enddef
3011
Bram Moolenaar69c76172021-12-02 16:38:52 +00003012def Test_closure_called_from_legacy()
3013 var lines =<< trim END
3014 vim9script
3015 def Func()
3016 var outer = 'foo'
3017 var F = () => {
3018 outer = 'bar'
3019 }
3020 execute printf('call %s()', string(F))
3021 enddef
3022 Func()
3023 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003024 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00003025enddef
3026
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01003027def Test_failure_in_called_function()
3028 # this was using the frame index as the return value
3029 var lines =<< trim END
3030 vim9script
3031 au TerminalWinOpen * eval [][0]
3032 def PopupTerm(a: any)
3033 # make sure typvals on stack are string
3034 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
3035 FireEvent()
3036 enddef
3037 def FireEvent()
3038 do TerminalWinOpen
3039 enddef
3040 # use try/catch to make eval fail
3041 try
3042 call PopupTerm(0)
3043 catch
3044 endtry
3045 au! TerminalWinOpen
3046 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003047 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01003048enddef
3049
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003050def Test_nested_lambda()
3051 var lines =<< trim END
3052 vim9script
3053 def Func()
3054 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003055 var Lambda1 = () => 7
3056 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003057 var res = Lambda2()
3058 assert_equal([7, 4], res)
3059 enddef
3060 Func()
3061 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003062 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003063enddef
3064
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02003065def Test_double_nested_lambda()
3066 var lines =<< trim END
3067 vim9script
3068 def F(head: string): func(string): func(string): string
3069 return (sep: string): func(string): string => ((tail: string): string => {
3070 return head .. sep .. tail
3071 })
3072 enddef
3073 assert_equal('hello-there', F('hello')('-')('there'))
3074 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003075 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02003076enddef
3077
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003078def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003079 var lines =<< trim END
3080 vim9script
3081 def F(text: string): func(string): func(string): string
3082 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003083 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003084 })
3085 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003086 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003087 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003088 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02003089
3090 lines =<< trim END
3091 vim9script
3092 echo range(4)->mapnew((_, v) => {
3093 return range(v) ->mapnew((_, s) => {
3094 return string(s)
3095 })
3096 })
3097 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003098 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003099
3100 lines =<< trim END
3101 vim9script
3102
Bram Moolenaara749a422022-02-12 19:52:25 +00003103 def Func()
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003104 range(10)
3105 ->mapnew((_, _) => ({
3106 key: range(10)->mapnew((_, _) => {
3107 return ' '
3108 }),
3109 }))
3110 enddef
3111
3112 defcomp
3113 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003114 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003115enddef
3116
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003117def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003118 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003119 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003120enddef
3121
3122def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00003123 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003124enddef
3125
Bram Moolenaar21dc8f12022-03-16 17:54:17 +00003126def Test_compiling_referenced_func_no_shadow()
3127 var lines =<< trim END
3128 vim9script
3129
3130 def InitializeReply(lspserver: dict<any>)
3131 enddef
3132
3133 def ProcessReply(lspserver: dict<any>)
3134 var lsp_reply_handlers: dict<func> =
3135 { 'initialize': InitializeReply }
3136 lsp_reply_handlers['initialize'](lspserver)
3137 enddef
3138
3139 call ProcessReply({})
3140 END
3141 v9.CheckScriptSuccess(lines)
3142enddef
3143
Bram Moolenaar62aec932022-01-29 21:45:34 +00003144def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003145 var path: string = empty(dir)
3146 \ ? 'empty'
3147 \ : 'full'
3148 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003149enddef
3150
3151def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003152 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003153enddef
3154
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003155def Test_script_var_in_lambda()
3156 var lines =<< trim END
3157 vim9script
3158 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003159 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003160 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003161 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003162enddef
3163
Bram Moolenaar62aec932022-01-29 21:45:34 +00003164def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003165 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003166 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003167 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003168 ->reverse()
3169 return x
3170enddef
3171
3172def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003173 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01003174
3175 var lines =<< trim END
3176 vim9script
3177 var res = [{n: 1, m: 2, s: 'xxx'}]
3178 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
3179 v.n,
3180 v.m,
3181 substitute(v.s, '.*', 'yyy', '')
3182 ))
3183 assert_equal(['1:2:yyy'], res)
3184 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003185 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003186enddef
3187
Bram Moolenaarb6571982021-01-08 22:24:19 +01003188def Test_list_lambda()
3189 timer_start(1000, (_) => 0)
3190 var body = execute(timer_info()[0].callback
3191 ->string()
3192 ->substitute("('", ' ', '')
3193 ->substitute("')", '', '')
3194 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02003195 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01003196enddef
3197
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003198def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02003199 var lines =<< trim END
3200 vim9script
3201 var flist: list<func>
3202 for i in range(10)
3203 var inloop = i
3204 flist[i] = () => inloop
3205 endfor
3206 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003207 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003208
3209 lines =<< trim END
3210 vim9script
3211 if true
3212 var outloop = 5
3213 var flist: list<func>
3214 for i in range(10)
3215 flist[i] = () => outloop
3216 endfor
3217 endif
3218 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003219 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003220
3221 lines =<< trim END
3222 vim9script
3223 if true
3224 var outloop = 5
3225 endif
3226 var flist: list<func>
3227 for i in range(10)
3228 flist[i] = () => outloop
3229 endfor
3230 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003231 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003232
3233 lines =<< trim END
3234 vim9script
3235 for i in range(10)
3236 var Ref = () => 0
3237 endfor
3238 assert_equal(0, ((i) => 0)(0))
3239 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003240 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003241enddef
3242
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003243def Test_legacy_lambda()
3244 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003245
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003246 var lines =<< trim END
3247 echo {x -> 'hello ' .. x}('foo')
3248 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003249 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003250
3251 lines =<< trim END
3252 vim9script
3253 def Func()
3254 echo (() => 'no error')()
3255 enddef
3256 legacy call s:Func()
3257 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003258 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003259enddef
3260
Bram Moolenaarce024c32021-06-26 13:00:49 +02003261def Test_legacy()
3262 var lines =<< trim END
3263 vim9script
3264 func g:LegacyFunction()
3265 let g:legacyvar = 1
3266 endfunc
3267 def Testit()
3268 legacy call g:LegacyFunction()
3269 enddef
3270 Testit()
3271 assert_equal(1, g:legacyvar)
3272 unlet g:legacyvar
3273 delfunc g:LegacyFunction
3274 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003275 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02003276enddef
3277
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003278def Test_legacy_errors()
3279 for cmd in ['if', 'elseif', 'else', 'endif',
3280 'for', 'endfor', 'continue', 'break',
3281 'while', 'endwhile',
3282 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00003283 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003284 endfor
3285enddef
3286
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003287def Test_call_legacy_with_dict()
3288 var lines =<< trim END
3289 vim9script
3290 func Legacy() dict
3291 let g:result = self.value
3292 endfunc
3293 def TestDirect()
3294 var d = {value: 'yes', func: Legacy}
3295 d.func()
3296 enddef
3297 TestDirect()
3298 assert_equal('yes', g:result)
3299 unlet g:result
3300
3301 def TestIndirect()
3302 var d = {value: 'foo', func: Legacy}
3303 var Fi = d.func
3304 Fi()
3305 enddef
3306 TestIndirect()
3307 assert_equal('foo', g:result)
3308 unlet g:result
3309
3310 var d = {value: 'bar', func: Legacy}
3311 d.func()
3312 assert_equal('bar', g:result)
3313 unlet g:result
3314 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003315 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003316enddef
3317
Bram Moolenaar62aec932022-01-29 21:45:34 +00003318def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01003319 # closure nested inside another closure using argument
3320 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
3321 return ['x', 'y', 'a', 'x2', 'c']->Filter()
3322enddef
3323
3324def Test_nested_closure_using_argument()
3325 assert_equal(['x', 'x2'], DoFilterThis('x'))
3326enddef
3327
Bram Moolenaar0186e582021-01-10 18:33:11 +01003328def Test_triple_nested_closure()
3329 var what = 'x'
3330 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
3331 var Filter = (l) => filter(l, (_, v) => Match(v, what))
3332 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
3333enddef
3334
Bram Moolenaar8f510af2020-07-05 18:48:23 +02003335func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003336 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003337 call Run_Test_silent_echo()
3338endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003339
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003340def Run_Test_silent_echo()
3341 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003342 vim9script
3343 def EchoNothing()
3344 silent echo ''
3345 enddef
3346 defcompile
3347 END
Bram Moolenaar6de22962022-09-09 21:35:36 +01003348 writefile(lines, 'XTest_silent_echo', 'D')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003349
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003350 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003351 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003352 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003353 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003354
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003355 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003356 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003357enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003358
Bram Moolenaar171fb922020-10-28 16:54:47 +01003359def SilentlyError()
3360 execute('silent! invalid')
3361 g:did_it = 'yes'
3362enddef
3363
Bram Moolenaar62aec932022-01-29 21:45:34 +00003364func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003365 silent! invalid
3366endfunc
3367
3368def SilentlyUserError()
3369 UserError()
3370 g:did_it = 'yes'
3371enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003372
3373" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003374func Test_ignore_silent_error()
3375 let g:did_it = 'no'
3376 call SilentlyError()
3377 call assert_equal('yes', g:did_it)
3378
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003379 let g:did_it = 'no'
3380 call SilentlyUserError()
3381 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003382
3383 unlet g:did_it
3384endfunc
3385
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003386def Test_ignore_silent_error_in_filter()
3387 var lines =<< trim END
3388 vim9script
3389 def Filter(winid: number, key: string): bool
3390 if key == 'o'
3391 silent! eval [][0]
3392 return true
3393 endif
3394 return popup_filter_menu(winid, key)
3395 enddef
3396
Bram Moolenaare0de1712020-12-02 17:36:54 +01003397 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003398 feedkeys("o\r", 'xnt')
3399 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003400 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003401enddef
3402
Bram Moolenaar62aec932022-01-29 21:45:34 +00003403def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003404 if n < 2
3405 return n
3406 else
3407 return Fibonacci(n - 1) + Fibonacci(n - 2)
3408 endif
3409enddef
3410
Bram Moolenaar985116a2020-07-12 17:31:09 +02003411def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003412 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003413enddef
3414
Bram Moolenaar62aec932022-01-29 21:45:34 +00003415def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003416 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003417 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003418 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003419 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003420 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003421enddef
3422
3423def Test_closure_in_map()
3424 mkdir('XclosureDir/tdir', 'p')
3425 writefile(['111'], 'XclosureDir/file1')
3426 writefile(['222'], 'XclosureDir/file2')
3427 writefile(['333'], 'XclosureDir/tdir/file3')
3428
Bram Moolenaare0de1712020-12-02 17:36:54 +01003429 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003430
3431 delete('XclosureDir', 'rf')
3432enddef
3433
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003434def Test_invalid_function_name()
3435 var lines =<< trim END
3436 vim9script
3437 def s: list<string>
3438 END
Bram Moolenaara749a422022-02-12 19:52:25 +00003439 v9.CheckScriptFailure(lines, 'E1268:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003440
3441 lines =<< trim END
3442 vim9script
3443 def g: list<string>
3444 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003445 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003446
3447 lines =<< trim END
3448 vim9script
3449 def <SID>: list<string>
3450 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003451 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003452
3453 lines =<< trim END
3454 vim9script
3455 def F list<string>
3456 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003457 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003458enddef
3459
Bram Moolenaara90afb92020-07-15 22:38:56 +02003460def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003461 var lines =<< trim END
3462 var Xsetlist: func
3463 Xsetlist = function('setloclist', [0])
3464 Xsetlist([], ' ', {title: 'test'})
3465 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003466
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003467 Xsetlist = function('setloclist', [0, [], ' '])
3468 Xsetlist({title: 'test'})
3469 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003470
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003471 Xsetlist = function('setqflist')
3472 Xsetlist([], ' ', {title: 'test'})
3473 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003474
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003475 Xsetlist = function('setqflist', [[], ' '])
3476 Xsetlist({title: 'test'})
3477 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02003478
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003479 var Len: func: number = function('len', ['word'])
3480 assert_equal(4, Len())
3481
3482 var RepeatFunc = function('repeat', ['o'])
3483 assert_equal('ooooo', RepeatFunc(5))
3484 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003485 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003486
3487 lines =<< trim END
3488 vim9script
3489 def Foo(Parser: any)
3490 enddef
3491 var Expr: func(dict<any>): dict<any>
3492 const Call = Foo(Expr)
3493 END
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +00003494 v9.CheckScriptFailure(lines, 'E1031:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003495enddef
3496
Bram Moolenaarcd1cda22022-02-16 21:48:25 +00003497def Test_partial_double_nested()
3498 var idx = 123
3499 var Get = () => idx
3500 var Ref = function(Get, [])
3501 var RefRef = function(Ref, [])
3502 assert_equal(123, RefRef())
3503enddef
3504
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003505def Test_partial_null_function()
3506 var lines =<< trim END
3507 var d: dict<func> = {f: null_function}
3508 var Ref = d.f
Bram Moolenaared0c62e2022-03-08 19:43:55 +00003509 assert_equal('func(...): unknown', typename(Ref))
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003510 END
3511 v9.CheckDefAndScriptSuccess(lines)
3512enddef
3513
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003514" Using "idx" from a legacy global function does not work.
3515" This caused a crash when called from legacy context.
3516func Test_partial_call_fails()
3517 let lines =<< trim END
3518 vim9script
3519
3520 var l = ['a', 'b', 'c']
3521 def Iter(container: any): any
3522 var idx = -1
3523 var obj = {state: container}
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003524 def g:NextItem__(self: dict<any>): any
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003525 ++idx
3526 return self.state[idx]
3527 enddef
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003528 obj.__next__ = function('g:NextItem__', [obj])
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003529 return obj
3530 enddef
3531
3532 var it = Iter(l)
3533 echo it.__next__()
3534 END
3535 call writefile(lines, 'XpartialCall')
3536 try
3537 source XpartialCall
3538 catch /E1248:/
3539 endtry
3540 call delete('XpartialCall')
3541endfunc
3542
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003543def Test_cmd_modifier()
3544 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003545 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003546enddef
3547
3548def Test_restore_modifiers()
3549 # check that when compiling a :def function command modifiers are not messed
3550 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003551 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003552 vim9script
3553 set eventignore=
3554 autocmd QuickFixCmdPost * copen
3555 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003556 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003557 enddef
3558 func Func()
3559 noautocmd call s:AutocmdsDisabled()
3560 let g:ei_after = &eventignore
3561 endfunc
3562 Func()
3563 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003564 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003565 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003566enddef
3567
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003568def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003569 eval 1 + 2
3570 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003571 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003572 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003573enddef
3574
3575def StackBot()
3576 # throw an error
3577 eval [][0]
3578enddef
3579
3580def Test_callstack_def()
3581 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003582 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003583 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003584 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003585 endtry
3586enddef
3587
Bram Moolenaare8211a32020-10-09 22:04:29 +02003588" Re-using spot for variable used in block
3589def Test_block_scoped_var()
3590 var lines =<< trim END
3591 vim9script
3592 def Func()
3593 var x = ['a', 'b', 'c']
3594 if 1
3595 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003596 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003597 endif
3598 var z = x
3599 assert_equal(['x', 'x', 'x'], z)
3600 enddef
3601 Func()
3602 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003603 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003604enddef
3605
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003606def Test_reset_did_emsg()
3607 var lines =<< trim END
3608 @s = 'blah'
3609 au BufWinLeave * #
3610 def Func()
3611 var winid = popup_create('popup', {})
3612 exe '*s'
3613 popup_close(winid)
3614 enddef
3615 Func()
3616 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003617 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003618 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003619enddef
3620
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003621def Test_did_emsg_reset()
3622 # executing an autocommand resets did_emsg, this should not result in a
3623 # builtin function considered failing
3624 var lines =<< trim END
3625 vim9script
3626 au BufWinLeave * #
3627 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003628 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003629 eval [][0]
3630 enddef
3631 nno <F3> <cmd>call <sid>Func()<cr>
3632 feedkeys("\<F3>\e", 'xt')
3633 END
3634 writefile(lines, 'XemsgReset')
3635 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
3636 delete('XemsgReset')
3637 nunmap <F3>
3638 au! BufWinLeave
3639enddef
3640
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003641def Test_abort_with_silent_call()
3642 var lines =<< trim END
3643 vim9script
3644 g:result = 'none'
3645 def Func()
3646 g:result += 3
3647 g:result = 'yes'
3648 enddef
3649 # error is silenced, but function aborts on error
3650 silent! Func()
3651 assert_equal('none', g:result)
3652 unlet g:result
3653 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003654 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003655enddef
3656
Bram Moolenaarf665e972020-12-05 19:17:16 +01003657def Test_continues_with_silent_error()
3658 var lines =<< trim END
3659 vim9script
3660 g:result = 'none'
3661 def Func()
3662 silent! g:result += 3
3663 g:result = 'yes'
3664 enddef
3665 # error is silenced, function does not abort
3666 Func()
3667 assert_equal('yes', g:result)
3668 unlet g:result
3669 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003670 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003671enddef
3672
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003673def Test_abort_even_with_silent()
3674 var lines =<< trim END
3675 vim9script
3676 g:result = 'none'
3677 def Func()
3678 eval {-> ''}() .. '' .. {}['X']
3679 g:result = 'yes'
3680 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003681 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003682 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003683 unlet g:result
3684 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003685 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003686enddef
3687
Bram Moolenaarf665e972020-12-05 19:17:16 +01003688def Test_cmdmod_silent_restored()
3689 var lines =<< trim END
3690 vim9script
3691 def Func()
3692 g:result = 'none'
3693 silent! g:result += 3
3694 g:result = 'none'
3695 g:result += 3
3696 enddef
3697 Func()
3698 END
3699 # can't use CheckScriptFailure, it ignores the :silent!
3700 var fname = 'Xdefsilent'
3701 writefile(lines, fname)
3702 var caught = 'no'
3703 try
3704 exe 'source ' .. fname
3705 catch /E1030:/
3706 caught = 'yes'
3707 assert_match('Func, line 4', v:throwpoint)
3708 endtry
3709 assert_equal('yes', caught)
3710 delete(fname)
3711enddef
3712
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003713def Test_cmdmod_silent_nested()
3714 var lines =<< trim END
3715 vim9script
3716 var result = ''
3717
3718 def Error()
3719 result ..= 'Eb'
3720 eval [][0]
3721 result ..= 'Ea'
3722 enddef
3723
3724 def Crash()
3725 result ..= 'Cb'
3726 sil! Error()
3727 result ..= 'Ca'
3728 enddef
3729
3730 Crash()
3731 assert_equal('CbEbEaCa', result)
3732 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003733 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003734enddef
3735
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003736def Test_dict_member_with_silent()
3737 var lines =<< trim END
3738 vim9script
3739 g:result = 'none'
3740 var d: dict<any>
3741 def Func()
3742 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003743 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003744 catch
3745 endtry
3746 enddef
3747 silent! Func()
3748 assert_equal('0', g:result)
3749 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003750 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003751 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003752enddef
3753
Bram Moolenaarf9041332021-01-21 19:41:16 +01003754def Test_skip_cmds_with_silent()
3755 var lines =<< trim END
3756 vim9script
3757
3758 def Func(b: bool)
3759 Crash()
3760 enddef
3761
3762 def Crash()
3763 sil! :/not found/d _
3764 sil! :/not found/put _
3765 enddef
3766
3767 Func(true)
3768 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003769 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003770enddef
3771
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003772def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003773 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003774 def g:Opfunc(_: any): string
3775 setline(1, 'ASDF')
3776 return ''
3777 enddef
3778 new
3779 setline(1, 'asdf')
3780 feedkeys("\<F3>$", 'x')
3781 assert_equal('ASDF', getline(1))
3782
3783 bwipe!
3784 nunmap <F3>
3785enddef
3786
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003787func Test_opfunc_error()
3788 CheckScreendump
3789 call Run_Test_opfunc_error()
3790endfunc
3791
3792def Run_Test_opfunc_error()
3793 # test that the error from Opfunc() is displayed right away
3794 var lines =<< trim END
3795 vim9script
3796
3797 def Opfunc(type: string)
3798 try
3799 eval [][0]
3800 catch /nothing/ # error not caught
3801 endtry
3802 enddef
3803 &operatorfunc = Opfunc
3804 nnoremap <expr> l <SID>L()
3805 def L(): string
3806 return 'l'
3807 enddef
3808 'x'->repeat(10)->setline(1)
3809 feedkeys('g@l', 'n')
3810 feedkeys('llll')
3811 END
3812 call writefile(lines, 'XTest_opfunc_error')
3813
Bram Moolenaar62aec932022-01-29 21:45:34 +00003814 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3815 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
Bram Moolenaarec892232022-05-06 17:53:06 +01003816 g:WaitForAssert(() => assert_match('E684: List index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003817
3818 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003819 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003820 delete('XTest_opfunc_error')
3821enddef
3822
Bram Moolenaar077a4232020-12-22 18:33:27 +01003823" this was crashing on exit
3824def Test_nested_lambda_in_closure()
3825 var lines =<< trim END
3826 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003827 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003828 def Outer()
3829 def g:Inner()
3830 echo map([1, 2, 3], {_, v -> v + 1})
3831 enddef
3832 g:Inner()
3833 enddef
3834 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003835 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003836 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003837 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003838 return
3839 endif
3840 assert_equal(['Done'], readfile('XnestedDone'))
3841 delete('XnestedDone')
3842enddef
3843
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003844def Test_nested_closure_funcref()
3845 var lines =<< trim END
3846 vim9script
3847 def Func()
3848 var n: number
3849 def Nested()
3850 ++n
3851 enddef
3852 Nested()
3853 g:result_one = n
3854 var Ref = function(Nested)
3855 Ref()
3856 g:result_two = n
3857 enddef
3858 Func()
3859 END
3860 v9.CheckScriptSuccess(lines)
3861 assert_equal(1, g:result_one)
3862 assert_equal(2, g:result_two)
3863 unlet g:result_one g:result_two
3864enddef
3865
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003866def Test_nested_closure_in_dict()
3867 var lines =<< trim END
3868 vim9script
3869 def Func(): dict<any>
3870 var n: number
3871 def Inc(): number
3872 ++n
3873 return n
3874 enddef
3875 return {inc: function(Inc)}
3876 enddef
3877 disas Func
3878 var d = Func()
3879 assert_equal(1, d.inc())
3880 assert_equal(2, d.inc())
3881 END
3882 v9.CheckScriptSuccess(lines)
3883enddef
3884
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003885def Test_script_local_other_script()
3886 var lines =<< trim END
3887 function LegacyJob()
3888 let FuncRef = function('s:close_cb')
3889 endfunction
3890 function s:close_cb(...)
3891 endfunction
3892 END
3893 lines->writefile('Xlegacy.vim')
3894 source Xlegacy.vim
3895 g:LegacyJob()
3896 g:LegacyJob()
3897 g:LegacyJob()
3898
3899 delfunc g:LegacyJob
3900 delete('Xlegacy.vim')
3901enddef
3902
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003903def Test_check_func_arg_types()
3904 var lines =<< trim END
3905 vim9script
3906 def F1(x: string): string
3907 return x
3908 enddef
3909
3910 def F2(x: number): number
3911 return x + 1
3912 enddef
3913
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003914 def G(Fg: func): dict<func>
3915 return {f: Fg}
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003916 enddef
3917
3918 def H(d: dict<func>): string
3919 return d.f('a')
3920 enddef
3921 END
3922
Bram Moolenaar62aec932022-01-29 21:45:34 +00003923 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3924 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003925
3926 v9.CheckScriptFailure(lines + ['def SomeFunc(ff: func)', 'enddef'], 'E704:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003927enddef
3928
Bram Moolenaarbadf04f2022-03-12 21:28:22 +00003929def Test_call_func_with_null()
3930 var lines =<< trim END
3931 def Fstring(v: string)
3932 assert_equal(null_string, v)
3933 enddef
3934 Fstring(null_string)
3935 def Fblob(v: blob)
3936 assert_equal(null_blob, v)
3937 enddef
3938 Fblob(null_blob)
3939 def Flist(v: list<number>)
3940 assert_equal(null_list, v)
3941 enddef
3942 Flist(null_list)
3943 def Fdict(v: dict<number>)
3944 assert_equal(null_dict, v)
3945 enddef
3946 Fdict(null_dict)
3947 def Ffunc(Fv: func(number): number)
3948 assert_equal(null_function, Fv)
3949 enddef
3950 Ffunc(null_function)
3951 if has('channel')
3952 def Fchannel(v: channel)
3953 assert_equal(null_channel, v)
3954 enddef
3955 Fchannel(null_channel)
3956 def Fjob(v: job)
3957 assert_equal(null_job, v)
3958 enddef
3959 Fjob(null_job)
3960 endif
3961 END
3962 v9.CheckDefAndScriptSuccess(lines)
3963enddef
3964
3965def Test_null_default_argument()
3966 var lines =<< trim END
3967 def Fstring(v: string = null_string)
3968 assert_equal(null_string, v)
3969 enddef
3970 Fstring()
3971 def Fblob(v: blob = null_blob)
3972 assert_equal(null_blob, v)
3973 enddef
3974 Fblob()
3975 def Flist(v: list<number> = null_list)
3976 assert_equal(null_list, v)
3977 enddef
3978 Flist()
3979 def Fdict(v: dict<number> = null_dict)
3980 assert_equal(null_dict, v)
3981 enddef
3982 Fdict()
3983 def Ffunc(Fv: func(number): number = null_function)
3984 assert_equal(null_function, Fv)
3985 enddef
3986 Ffunc()
3987 if has('channel')
3988 def Fchannel(v: channel = null_channel)
3989 assert_equal(null_channel, v)
3990 enddef
3991 Fchannel()
3992 def Fjob(v: job = null_job)
3993 assert_equal(null_job, v)
3994 enddef
3995 Fjob()
3996 endif
3997 END
3998 v9.CheckDefAndScriptSuccess(lines)
3999enddef
4000
4001def Test_null_return()
4002 var lines =<< trim END
4003 def Fstring(): string
4004 return null_string
4005 enddef
4006 assert_equal(null_string, Fstring())
4007 def Fblob(): blob
4008 return null_blob
4009 enddef
4010 assert_equal(null_blob, Fblob())
4011 def Flist(): list<number>
4012 return null_list
4013 enddef
4014 assert_equal(null_list, Flist())
4015 def Fdict(): dict<number>
4016 return null_dict
4017 enddef
4018 assert_equal(null_dict, Fdict())
4019 def Ffunc(): func(number): number
4020 return null_function
4021 enddef
4022 assert_equal(null_function, Ffunc())
4023 if has('channel')
4024 def Fchannel(): channel
4025 return null_channel
4026 enddef
4027 assert_equal(null_channel, Fchannel())
4028 def Fjob(): job
4029 return null_job
4030 enddef
4031 assert_equal(null_job, Fjob())
4032 endif
4033 END
4034 v9.CheckDefAndScriptSuccess(lines)
4035enddef
4036
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004037def Test_list_any_type_checked()
4038 var lines =<< trim END
4039 vim9script
4040 def Foo()
4041 --decl--
4042 Bar(l)
4043 enddef
4044 def Bar(ll: list<dict<any>>)
4045 enddef
4046 Foo()
4047 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004048 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004049 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004050 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004051 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004052 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004053
4054 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00004055 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004056enddef
4057
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004058def Test_compile_error()
4059 var lines =<< trim END
4060 def g:Broken()
4061 echo 'a' + {}
4062 enddef
4063 call g:Broken()
4064 END
4065 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00004066 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004067
4068 # Second call won't try compiling again
4069 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02004070 delfunc g:Broken
4071
4072 # No error when compiling with :silent!
4073 lines =<< trim END
4074 def g:Broken()
4075 echo 'a' + []
4076 enddef
4077 silent! defcompile
4078 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004079 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02004080
4081 # Calling the function won't try compiling again
4082 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
4083 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004084enddef
4085
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004086def Test_ignored_argument()
4087 var lines =<< trim END
4088 vim9script
4089 def Ignore(_, _): string
4090 return 'yes'
4091 enddef
4092 assert_equal('yes', Ignore(1, 2))
4093
4094 func Ok(_)
4095 return a:_
4096 endfunc
4097 assert_equal('ok', Ok('ok'))
4098
4099 func Oktoo()
4100 let _ = 'too'
4101 return _
4102 endfunc
4103 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02004104
4105 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004106 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004107 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004108
4109 lines =<< trim END
4110 def Ignore(_: string): string
4111 return _
4112 enddef
4113 defcompile
4114 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004115 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004116
4117 lines =<< trim END
4118 var _ = 1
4119 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004120 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02004121
4122 lines =<< trim END
4123 var x = _
4124 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004125 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004126enddef
4127
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004128def Test_too_many_arguments()
4129 var lines =<< trim END
4130 echo [0, 1, 2]->map(() => 123)
4131 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004132 v9.CheckDefAndScriptFailure(lines, ['E176:', 'E1106: 2 arguments too many'], 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004133
4134 lines =<< trim END
4135 echo [0, 1, 2]->map((_) => 123)
4136 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004137 v9.CheckDefAndScriptFailure(lines, ['E176', 'E1106: One argument too many'], 1)
Bram Moolenaar31d99482022-05-26 22:24:43 +01004138
4139 lines =<< trim END
4140 vim9script
4141 def OneArgument(arg: string)
4142 echo arg
4143 enddef
4144 var Ref = OneArgument
4145 Ref('a', 'b')
4146 END
4147 v9.CheckScriptFailure(lines, 'E118:')
4148enddef
4149
4150def Test_funcref_with_base()
4151 var lines =<< trim END
4152 vim9script
4153 def TwoArguments(str: string, nr: number)
4154 echo str nr
4155 enddef
4156 var Ref = TwoArguments
4157 Ref('a', 12)
4158 'b'->Ref(34)
4159 END
4160 v9.CheckScriptSuccess(lines)
4161
4162 lines =<< trim END
4163 vim9script
4164 def TwoArguments(str: string, nr: number)
4165 echo str nr
4166 enddef
4167 var Ref = TwoArguments
4168 'a'->Ref('b')
4169 END
4170 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string', 6)
4171
4172 lines =<< trim END
4173 vim9script
4174 def TwoArguments(str: string, nr: number)
4175 echo str nr
4176 enddef
4177 var Ref = TwoArguments
4178 123->Ref(456)
4179 END
4180 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
4181
4182 lines =<< trim END
4183 vim9script
4184 def TwoArguments(nr: number, str: string)
4185 echo str nr
4186 enddef
4187 var Ref = TwoArguments
4188 123->Ref('b')
4189 def AndNowCompiled()
4190 456->Ref('x')
4191 enddef
4192 AndNowCompiled()
4193 END
4194 v9.CheckScriptSuccess(lines)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004195enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01004196
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004197def Test_closing_brace_at_start_of_line()
4198 var lines =<< trim END
4199 def Func()
4200 enddef
4201 Func(
4202 )
4203 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004204 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004205enddef
4206
Bram Moolenaar62aec932022-01-29 21:45:34 +00004207func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02004208 let g:mydict = {}
4209 func g:mydict.afunc()
4210 let g:result = self.key
4211 endfunc
4212endfunc
4213
4214def Test_numbered_function_reference()
4215 CreateMydict()
4216 var output = execute('legacy func g:mydict.afunc')
4217 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
4218 execute 'function(' .. funcName .. ', [], {key: 42})()'
4219 # check that the function still exists
4220 assert_equal(output, execute('legacy func g:mydict.afunc'))
4221 unlet g:mydict
4222enddef
4223
Bram Moolenaard3a11782022-01-05 16:50:40 +00004224def Test_go_beyond_end_of_cmd()
4225 # this was reading the byte after the end of the line
4226 var lines =<< trim END
4227 def F()
4228 cal
4229 enddef
4230 defcompile
4231 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004232 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00004233enddef
4234
Yegappan Lakshmanan7c7e19c2022-04-09 11:09:07 +01004235" Test for memory allocation failure when defining a new lambda
4236func Test_lambda_allocation_failure()
4237 new
4238 let lines =<< trim END
4239 vim9script
4240 g:Xlambda = (x): number => {
4241 return x + 1
4242 }
4243 END
4244 call setline(1, lines)
4245 call test_alloc_fail(GetAllocId('get_func'), 0, 0)
4246 call assert_fails('source', 'E342:')
4247 call assert_false(exists('g:Xlambda'))
4248 bw!
4249endfunc
4250
Bram Moolenaarbce69d62022-05-22 13:45:52 +01004251def Test_multiple_funcref()
4252 # This was using a NULL pointer
4253 var lines =<< trim END
4254 vim9script
4255 def A(F: func, ...args: list<any>): func
4256 return funcref(F, args)
4257 enddef
4258
4259 def B(F: func): func
4260 return funcref(A, [F])
4261 enddef
4262
4263 def Test(n: number)
4264 enddef
4265
4266 const X = B(Test)
4267 X(1)
4268 END
4269 v9.CheckScriptSuccess(lines)
4270
4271 # slightly different case
4272 lines =<< trim END
4273 vim9script
4274
4275 def A(F: func, ...args: list<any>): any
4276 return call(F, args)
4277 enddef
4278
4279 def B(F: func): func
4280 return funcref(A, [F])
4281 enddef
4282
4283 def Test(n: number)
4284 enddef
4285
4286 const X = B(Test)
4287 X(1)
4288 END
4289 v9.CheckScriptSuccess(lines)
4290enddef
4291
Bram Moolenaarbd683e32022-07-18 17:49:03 +01004292def Test_cexpr_errmsg_line_number()
4293 var lines =<< trim END
4294 vim9script
4295 def Func()
4296 var qfl = {}
4297 cexpr qfl
4298 enddef
4299 Func()
4300 END
4301 v9.CheckScriptFailure(lines, 'E777', 2)
4302enddef
4303
Bram Moolenaar1d84f762022-09-03 21:35:53 +01004304def AddDefer(s: string)
4305 g:deferred->extend([s])
4306enddef
4307
4308def DeferTwo()
4309 g:deferred->extend(['in Two'])
4310 for n in range(3)
4311 defer g:AddDefer('two' .. n)
4312 endfor
4313 g:deferred->extend(['end Two'])
4314enddef
4315
4316def DeferOne()
4317 g:deferred->extend(['in One'])
4318 defer g:AddDefer('one')
4319 g:DeferTwo()
4320 g:deferred->extend(['end One'])
4321
4322 writefile(['text'], 'XdeferFile')
4323 defer delete('XdeferFile')
4324enddef
4325
4326def Test_defer()
4327 g:deferred = []
4328 g:DeferOne()
4329 assert_equal(['in One', 'in Two', 'end Two', 'two2', 'two1', 'two0', 'end One', 'one'], g:deferred)
4330 unlet g:deferred
4331 assert_equal('', glob('XdeferFile'))
4332enddef
4333
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004334" The following messes up syntax highlight, keep near the end.
Bram Moolenaar20677332021-06-06 17:02:53 +02004335if has('python3')
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004336 def Test_python3_command()
4337 py3 import vim
Bram Moolenaarf5288c52022-02-15 21:33:29 +00004338 py3 vim.command("g:done = 'yes'")
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004339 assert_equal('yes', g:done)
4340 unlet g:done
4341 enddef
4342
Bram Moolenaar20677332021-06-06 17:02:53 +02004343 def Test_python3_heredoc()
4344 py3 << trim EOF
4345 import vim
4346 vim.vars['didit'] = 'yes'
4347 EOF
4348 assert_equal('yes', g:didit)
4349
4350 python3 << trim EOF
4351 import vim
4352 vim.vars['didit'] = 'again'
4353 EOF
4354 assert_equal('again', g:didit)
4355 enddef
4356endif
4357
Bram Moolenaar20677332021-06-06 17:02:53 +02004358if has('lua')
4359 def Test_lua_heredoc()
4360 g:d = {}
4361 lua << trim EOF
4362 x = vim.eval('g:d')
4363 x['key'] = 'val'
4364 EOF
4365 assert_equal('val', g:d.key)
4366 enddef
Bram Moolenaarefd73ae2022-03-20 18:51:00 +00004367
4368 def Test_lua_heredoc_fails()
4369 var lines = [
4370 'vim9script',
4371 'def ExeLua()',
4372 'lua << trim EOLUA',
4373 "x = vim.eval('g:nodict')",
4374 'EOLUA',
4375 'enddef',
4376 'ExeLua()',
4377 ]
4378 v9.CheckScriptFailure(lines, 'E121: Undefined variable: g:nodict')
4379 enddef
Bram Moolenaar20677332021-06-06 17:02:53 +02004380endif
4381
Bram Moolenaard881d152022-05-13 13:50:36 +01004382if has('perl')
4383 def Test_perl_heredoc_nested()
4384 var lines =<< trim END
4385 vim9script
4386 def F(): string
4387 def G(): string
4388 perl << EOF
4389 EOF
4390 return 'done'
4391 enddef
4392 return G()
4393 enddef
4394 assert_equal('done', F())
4395 END
4396 v9.CheckScriptSuccess(lines)
4397 enddef
4398endif
4399
Bram Moolenaarf7779c62020-05-03 15:38:16 +02004400
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02004401" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker