blob: 6f83b9caeae971235d9b8246c341847497c4b7e9 [file] [log] [blame]
Bram Moolenaar14735512016-03-26 21:00:08 +01001" Tests for autocommands
2
Bram Moolenaar8c64a362018-03-23 22:39:31 +01003source shared.vim
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02004source check.vim
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02005source term_util.vim
LemonBoy09371822022-04-08 15:18:45 +01006source screendump.vim
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00007import './vim9.vim' as v9
Bram Moolenaar8c64a362018-03-23 22:39:31 +01008
Bram Moolenaar1e115362019-01-09 23:01:02 +01009func s:cleanup_buffers() abort
Bram Moolenaarb3435b02016-09-29 20:54:59 +020010 for bnr in range(1, bufnr('$'))
11 if bufloaded(bnr) && bufnr('%') != bnr
12 execute 'bd! ' . bnr
13 endif
14 endfor
Bram Moolenaar04f62f82017-07-19 18:18:39 +020015endfunc
Bram Moolenaarb3435b02016-09-29 20:54:59 +020016
Christian Brabandtfb3f9692024-08-11 20:09:17 +020017func CleanUpTestAuGroup()
18 augroup testing
19 au!
20 augroup END
21 augroup! testing
22endfunc
23
Bram Moolenaar14735512016-03-26 21:00:08 +010024func Test_vim_did_enter()
25 call assert_false(v:vim_did_enter)
26
27 " This script will never reach the main loop, can't check if v:vim_did_enter
28 " becomes one.
29endfunc
Bram Moolenaar40b1b542016-04-20 20:18:23 +020030
Bram Moolenaar75911162020-07-21 19:44:47 +020031" Test for the CursorHold autocmd
32func Test_CursorHold_autocmd()
33 CheckRunVimInTerminal
Bram Moolenaare1f3ab72022-09-04 21:29:08 +010034 call writefile(['one', 'two', 'three'], 'XoneTwoThree', 'D')
Bram Moolenaar75911162020-07-21 19:44:47 +020035 let before =<< trim END
36 set updatetime=10
Bram Moolenaare7cda972022-08-29 11:02:59 +010037 au CursorHold * call writefile([line('.')], 'XCHoutput', 'a')
Bram Moolenaar75911162020-07-21 19:44:47 +020038 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +010039 call writefile(before, 'XCHinit', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +010040 let buf = RunVimInTerminal('-S XCHinit XoneTwoThree', {})
Bram Moolenaar17f67542020-08-20 18:29:13 +020041 call term_sendkeys(buf, "G")
Bram Moolenaar62cd26a2020-10-11 20:08:44 +020042 call term_wait(buf, 50)
Bram Moolenaar75911162020-07-21 19:44:47 +020043 call term_sendkeys(buf, "gg")
44 call term_wait(buf)
Bram Moolenaare7cda972022-08-29 11:02:59 +010045 call WaitForAssert({-> assert_equal(['1'], readfile('XCHoutput')[-1:-1])})
Bram Moolenaar75911162020-07-21 19:44:47 +020046 call term_sendkeys(buf, "j")
47 call term_wait(buf)
Bram Moolenaare7cda972022-08-29 11:02:59 +010048 call WaitForAssert({-> assert_equal(['1', '2'], readfile('XCHoutput')[-2:-1])})
Bram Moolenaar75911162020-07-21 19:44:47 +020049 call term_sendkeys(buf, "j")
50 call term_wait(buf)
Bram Moolenaare7cda972022-08-29 11:02:59 +010051 call WaitForAssert({-> assert_equal(['1', '2', '3'], readfile('XCHoutput')[-3:-1])})
Bram Moolenaar75911162020-07-21 19:44:47 +020052 call StopVimInTerminal(buf)
53
Bram Moolenaare7cda972022-08-29 11:02:59 +010054 call delete('XCHoutput')
Bram Moolenaar75911162020-07-21 19:44:47 +020055endfunc
56
Bram Moolenaarc67e8922016-05-24 16:07:40 +020057if has('timers')
Bram Moolenaar97b00752019-05-12 13:07:14 +020058
Bram Moolenaarc67e8922016-05-24 16:07:40 +020059 func ExitInsertMode(id)
60 call feedkeys("\<Esc>")
61 endfunc
62
63 func Test_cursorhold_insert()
zeertzjq657b31f2023-04-15 21:28:02 +010064 " depends on timing
65 let g:test_is_flaky = 1
66
Bram Moolenaarf18c4db2016-09-08 22:10:06 +020067 " Need to move the cursor.
68 call feedkeys("ggG", "xt")
69
Bram Moolenaarc67e8922016-05-24 16:07:40 +020070 let g:triggered = 0
71 au CursorHoldI * let g:triggered += 1
72 set updatetime=20
Bram Moolenaar92bb83e2021-02-03 23:04:46 +010073 call timer_start(200, 'ExitInsertMode')
Bram Moolenaarc67e8922016-05-24 16:07:40 +020074 call feedkeys('a', 'x!')
Bram Moolenaar3b014be2022-11-13 17:53:46 +000075 sleep 30m
Bram Moolenaarc67e8922016-05-24 16:07:40 +020076 call assert_equal(1, g:triggered)
Bram Moolenaar26d98212019-01-27 22:32:55 +010077 unlet g:triggered
78 au! CursorHoldI
79 set updatetime&
80 endfunc
81
82 func Test_cursorhold_insert_with_timer_interrupt()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +020083 CheckFeature job
Bram Moolenaar26d98212019-01-27 22:32:55 +010084 " Need to move the cursor.
85 call feedkeys("ggG", "xt")
86
87 " Confirm the timer invoked in exit_cb of the job doesn't disturb
88 " CursorHoldI event.
89 let g:triggered = 0
90 au CursorHoldI * let g:triggered += 1
Bram Moolenaar62cd26a2020-10-11 20:08:44 +020091 set updatetime=100
Milly4f5681d2024-10-20 11:06:00 +020092 call job_start(has('win32') ? 'cmd /D /c echo:' : 'echo',
Bram Moolenaar62cd26a2020-10-11 20:08:44 +020093 \ {'exit_cb': {-> timer_start(200, 'ExitInsertMode')}})
Bram Moolenaar26d98212019-01-27 22:32:55 +010094 call feedkeys('a', 'x!')
95 call assert_equal(1, g:triggered)
96 unlet g:triggered
Bram Moolenaare99e8442016-07-26 20:43:40 +020097 au! CursorHoldI
Bram Moolenaaraeac9002016-09-06 22:15:08 +020098 set updatetime&
Bram Moolenaarc67e8922016-05-24 16:07:40 +020099 endfunc
100
101 func Test_cursorhold_insert_ctrl_x()
102 let g:triggered = 0
103 au CursorHoldI * let g:triggered += 1
104 set updatetime=20
105 call timer_start(100, 'ExitInsertMode')
106 " CursorHoldI does not trigger after CTRL-X
107 call feedkeys("a\<C-X>", 'x!')
108 call assert_equal(0, g:triggered)
Bram Moolenaar26d98212019-01-27 22:32:55 +0100109 unlet g:triggered
Bram Moolenaare99e8442016-07-26 20:43:40 +0200110 au! CursorHoldI
Bram Moolenaaraeac9002016-09-06 22:15:08 +0200111 set updatetime&
Bram Moolenaarc67e8922016-05-24 16:07:40 +0200112 endfunc
Bram Moolenaar97b00752019-05-12 13:07:14 +0200113
Bram Moolenaar5a9357d2021-10-03 16:22:05 +0100114 func Test_cursorhold_insert_ctrl_g_U()
115 au CursorHoldI * :
116 set updatetime=20
117 new
118 call timer_start(100, { -> feedkeys("\<Left>foo\<Esc>", 't') })
119 call feedkeys("i()\<C-g>U", 'tx!')
120 sleep 200m
121 call assert_equal('(foo)', getline(1))
122 undo
123 call assert_equal('', getline(1))
124
125 bwipe!
126 au! CursorHoldI
127 set updatetime&
128 endfunc
129
Bram Moolenaar97b00752019-05-12 13:07:14 +0200130 func Test_OptionSet_modeline()
131 call test_override('starting', 1)
132 au! OptionSet
133 augroup set_tabstop
134 au OptionSet tabstop call timer_start(1, {-> execute("echo 'Handler called'", "")})
135 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +0100136 call writefile(['vim: set ts=7 sw=5 :', 'something'], 'XoptionsetModeline', 'D')
Bram Moolenaar97b00752019-05-12 13:07:14 +0200137 set modeline
138 let v:errmsg = ''
139 call assert_fails('split XoptionsetModeline', 'E12:')
140 call assert_equal(7, &ts)
141 call assert_equal('', v:errmsg)
142
143 augroup set_tabstop
144 au!
145 augroup END
146 bwipe!
147 set ts&
Bram Moolenaar97b00752019-05-12 13:07:14 +0200148 call test_override('starting', 0)
149 endfunc
150
151endif "has('timers')
Bram Moolenaar40b1b542016-04-20 20:18:23 +0200152
Bram Moolenaar04f62f82017-07-19 18:18:39 +0200153func Test_bufunload()
Bram Moolenaarc67e8922016-05-24 16:07:40 +0200154 augroup test_bufunload_group
155 autocmd!
156 autocmd BufUnload * call add(s:li, "bufunload")
157 autocmd BufDelete * call add(s:li, "bufdelete")
158 autocmd BufWipeout * call add(s:li, "bufwipeout")
159 augroup END
Bram Moolenaar40b1b542016-04-20 20:18:23 +0200160
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +0100161 let s:li = []
Bram Moolenaarc67e8922016-05-24 16:07:40 +0200162 new
163 setlocal bufhidden=
164 bunload
165 call assert_equal(["bufunload", "bufdelete"], s:li)
Bram Moolenaar40b1b542016-04-20 20:18:23 +0200166
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +0100167 let s:li = []
Bram Moolenaarc67e8922016-05-24 16:07:40 +0200168 new
169 setlocal bufhidden=delete
170 bunload
171 call assert_equal(["bufunload", "bufdelete"], s:li)
172
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +0100173 let s:li = []
Bram Moolenaarc67e8922016-05-24 16:07:40 +0200174 new
175 setlocal bufhidden=unload
176 bwipeout
177 call assert_equal(["bufunload", "bufdelete", "bufwipeout"], s:li)
178
Bram Moolenaare99e8442016-07-26 20:43:40 +0200179 au! test_bufunload_group
Bram Moolenaarc67e8922016-05-24 16:07:40 +0200180 augroup! test_bufunload_group
Bram Moolenaar40b1b542016-04-20 20:18:23 +0200181endfunc
Bram Moolenaar30445cb2016-07-09 15:21:02 +0200182
183" SEGV occurs in older versions. (At least 7.4.2005 or older)
Bram Moolenaar04f62f82017-07-19 18:18:39 +0200184func Test_autocmd_bufunload_with_tabnext()
Bram Moolenaar30445cb2016-07-09 15:21:02 +0200185 tabedit
186 tabfirst
187
188 augroup test_autocmd_bufunload_with_tabnext_group
189 autocmd!
190 autocmd BufUnload <buffer> tabnext
191 augroup END
192
193 quit
194 call assert_equal(2, tabpagenr('$'))
195
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200196 autocmd! test_autocmd_bufunload_with_tabnext_group
Bram Moolenaar30445cb2016-07-09 15:21:02 +0200197 augroup! test_autocmd_bufunload_with_tabnext_group
198 tablast
199 quit
200endfunc
Bram Moolenaarc917da42016-07-19 22:31:36 +0200201
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100202func Test_argdelete_in_next()
203 au BufNew,BufEnter,BufLeave,BufWinEnter * argdel
204 call assert_fails('next a b', 'E1156:')
205 au! BufNew,BufEnter,BufLeave,BufWinEnter *
206endfunc
207
Bram Moolenaar04f62f82017-07-19 18:18:39 +0200208func Test_autocmd_bufwinleave_with_tabfirst()
Bram Moolenaarf9e687e2016-09-04 21:33:09 +0200209 tabedit
210 augroup sample
211 autocmd!
212 autocmd BufWinLeave <buffer> tabfirst
213 augroup END
214 call setline(1, ['a', 'b', 'c'])
215 edit! a.txt
Bram Moolenaarf18c4db2016-09-08 22:10:06 +0200216 tabclose
Bram Moolenaarf9e687e2016-09-04 21:33:09 +0200217endfunc
218
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200219" SEGV occurs in older versions. (At least 7.4.2321 or older)
Bram Moolenaar04f62f82017-07-19 18:18:39 +0200220func Test_autocmd_bufunload_avoiding_SEGV_01()
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200221 split aa.txt
222 let lastbuf = bufnr('$')
223
224 augroup test_autocmd_bufunload
225 autocmd!
226 exe 'autocmd BufUnload <buffer> ' . (lastbuf + 1) . 'bwipeout!'
227 augroup END
228
Bram Moolenaar28ee8922020-10-28 20:20:00 +0100229 call assert_fails('edit bb.txt', 'E937:')
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200230
231 autocmd! test_autocmd_bufunload
232 augroup! test_autocmd_bufunload
233 bwipe! aa.txt
234 bwipe! bb.txt
235endfunc
236
237" SEGV occurs in older versions. (At least 7.4.2321 or older)
Bram Moolenaar04f62f82017-07-19 18:18:39 +0200238func Test_autocmd_bufunload_avoiding_SEGV_02()
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200239 setlocal buftype=nowrite
240 let lastbuf = bufnr('$')
241
242 augroup test_autocmd_bufunload
243 autocmd!
244 exe 'autocmd BufUnload <buffer> ' . (lastbuf + 1) . 'bwipeout!'
245 augroup END
246
247 normal! i1
248 call assert_fails('edit a.txt', 'E517:')
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200249
250 autocmd! test_autocmd_bufunload
251 augroup! test_autocmd_bufunload
252 bwipe! a.txt
253endfunc
254
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +0100255func Test_autocmd_dummy_wipeout()
256 " prepare files
Bram Moolenaare1f3ab72022-09-04 21:29:08 +0100257 call writefile([''], 'Xdummywipetest1.txt', 'D')
258 call writefile([''], 'Xdummywipetest2.txt', 'D')
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +0100259 augroup test_bufunload_group
260 autocmd!
261 autocmd BufUnload * call add(s:li, "bufunload")
262 autocmd BufDelete * call add(s:li, "bufdelete")
263 autocmd BufWipeout * call add(s:li, "bufwipeout")
264 augroup END
265
266 let s:li = []
267 split Xdummywipetest1.txt
268 silent! vimgrep /notmatched/ Xdummywipetest*
269 call assert_equal(["bufunload", "bufwipeout"], s:li)
270
271 bwipeout
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +0100272 au! test_bufunload_group
273 augroup! test_bufunload_group
274endfunc
275
Bram Moolenaarc917da42016-07-19 22:31:36 +0200276func Test_win_tab_autocmd()
277 let g:record = []
278
Christian Brabandtfb3f9692024-08-11 20:09:17 +0200279 defer CleanUpTestAuGroup()
Bram Moolenaarc917da42016-07-19 22:31:36 +0200280 augroup testing
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100281 au WinNewPre * call add(g:record, 'WinNewPre')
Bram Moolenaarc917da42016-07-19 22:31:36 +0200282 au WinNew * call add(g:record, 'WinNew')
naohiro ono23beefe2021-11-13 12:38:49 +0000283 au WinClosed * call add(g:record, 'WinClosed')
Bram Moolenaar94722c52023-01-28 19:19:03 +0000284 au WinEnter * call add(g:record, 'WinEnter')
285 au WinLeave * call add(g:record, 'WinLeave')
Bram Moolenaarc917da42016-07-19 22:31:36 +0200286 au TabNew * call add(g:record, 'TabNew')
Bram Moolenaar12c11d52016-07-19 23:13:03 +0200287 au TabClosed * call add(g:record, 'TabClosed')
Bram Moolenaarc917da42016-07-19 22:31:36 +0200288 au TabEnter * call add(g:record, 'TabEnter')
289 au TabLeave * call add(g:record, 'TabLeave')
290 augroup END
291
292 split
293 tabnew
294 close
295 close
296
297 call assert_equal([
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100298 \ 'WinNewPre', 'WinLeave', 'WinNew', 'WinEnter',
Christian Brabandtfb3f9692024-08-11 20:09:17 +0200299 \ 'WinLeave', 'TabLeave', 'WinNew', 'WinEnter', 'TabNew', 'TabEnter',
naohiro ono23beefe2021-11-13 12:38:49 +0000300 \ 'WinLeave', 'TabLeave', 'WinClosed', 'TabClosed', 'WinEnter', 'TabEnter',
301 \ 'WinLeave', 'WinClosed', 'WinEnter'
Bram Moolenaarc917da42016-07-19 22:31:36 +0200302 \ ], g:record)
303
Bram Moolenaar12c11d52016-07-19 23:13:03 +0200304 let g:record = []
305 tabnew somefile
306 tabnext
307 bwipe somefile
308
309 call assert_equal([
Christian Brabandtfb3f9692024-08-11 20:09:17 +0200310 \ 'WinLeave', 'TabLeave', 'WinNew', 'WinEnter', 'TabNew', 'TabEnter',
Bram Moolenaar12c11d52016-07-19 23:13:03 +0200311 \ 'WinLeave', 'TabLeave', 'WinEnter', 'TabEnter',
naohiro ono23beefe2021-11-13 12:38:49 +0000312 \ 'WinClosed', 'TabClosed'
Bram Moolenaar12c11d52016-07-19 23:13:03 +0200313 \ ], g:record)
314
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100315 let g:record = []
316 copen
317 help
318 tabnext
319 vnew
320
321 call assert_equal([
322 \ 'WinNewPre', 'WinLeave', 'WinNew', 'WinEnter',
323 \ 'WinNewPre', 'WinLeave', 'WinNew', 'WinEnter',
324 \ 'WinNewPre', 'WinLeave', 'WinNew', 'WinEnter'
325 \ ], g:record)
326
Bram Moolenaarc917da42016-07-19 22:31:36 +0200327 unlet g:record
328endfunc
Bram Moolenaare99e8442016-07-26 20:43:40 +0200329
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100330func Test_WinNewPre()
331 " Test that the old window layout can be accessed before a new window is created.
332 let g:layouts_pre = []
333 let g:layouts_post = []
334 augroup testing
335 au WinNewPre * call add(g:layouts_pre, winlayout())
336 au WinNew * call add(g:layouts_post, winlayout())
337 augroup END
Christian Brabandtfb3f9692024-08-11 20:09:17 +0200338 defer CleanUpTestAuGroup()
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100339 split
340 call assert_notequal(g:layouts_pre[0], g:layouts_post[0])
341 split
342 call assert_equal(g:layouts_pre[1], g:layouts_post[0])
343 call assert_notequal(g:layouts_pre[1], g:layouts_post[1])
Christian Brabandtfb3f9692024-08-11 20:09:17 +0200344 " not triggered for tabnew
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100345 tabnew
Christian Brabandtfb3f9692024-08-11 20:09:17 +0200346 call assert_equal(2, len(g:layouts_pre))
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100347 unlet g:layouts_pre
348 unlet g:layouts_post
349
350 " Test modifying window layout during WinNewPre throws.
351 let g:caught = 0
352 augroup testing
353 au!
354 au WinNewPre * split
355 augroup END
356 try
357 vnew
358 catch
359 let g:caught += 1
360 endtry
361 augroup testing
362 au!
363 au WinNewPre * tabnew
364 augroup END
365 try
366 vnew
367 catch
368 let g:caught += 1
369 endtry
370 augroup testing
371 au!
372 au WinNewPre * close
373 augroup END
374 try
375 vnew
376 catch
377 let g:caught += 1
378 endtry
379 augroup testing
380 au!
381 au WinNewPre * tabclose
382 augroup END
383 try
384 vnew
385 catch
386 let g:caught += 1
387 endtry
388 call assert_equal(4, g:caught)
Sergey Vlasov1f47db72024-01-25 23:07:00 +0100389 unlet g:caught
390endfunc
391
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000392func Test_WinResized()
393 CheckRunVimInTerminal
394
395 let lines =<< trim END
396 set scrolloff=0
397 call setline(1, ['111', '222'])
398 vnew
399 call setline(1, ['aaa', 'bbb'])
400 new
401 call setline(1, ['foo', 'bar'])
402
403 let g:resized = 0
404 au WinResized * let g:resized += 1
405
406 func WriteResizedEvent()
407 call writefile([json_encode(v:event)], 'XresizeEvent')
408 endfunc
409 au WinResized * call WriteResizedEvent()
410 END
411 call writefile(lines, 'Xtest_winresized', 'D')
412 let buf = RunVimInTerminal('-S Xtest_winresized', {'rows': 10})
413
414 " redraw now to avoid a redraw after the :echo command
415 call term_sendkeys(buf, ":redraw!\<CR>")
416 call TermWait(buf)
417
418 call term_sendkeys(buf, ":echo g:resized\<CR>")
419 call WaitForAssert({-> assert_match('^0$', term_getline(buf, 10))}, 1000)
420
421 " increase window height, two windows will be reported
422 call term_sendkeys(buf, "\<C-W>+")
423 call TermWait(buf)
424 call term_sendkeys(buf, ":echo g:resized\<CR>")
425 call WaitForAssert({-> assert_match('^1$', term_getline(buf, 10))}, 1000)
426
427 let event = readfile('XresizeEvent')[0]->json_decode()
428 call assert_equal({
429 \ 'windows': [1002, 1001],
430 \ }, event)
431
432 " increase window width, three windows will be reported
433 call term_sendkeys(buf, "\<C-W>>")
434 call TermWait(buf)
435 call term_sendkeys(buf, ":echo g:resized\<CR>")
436 call WaitForAssert({-> assert_match('^2$', term_getline(buf, 10))}, 1000)
437
438 let event = readfile('XresizeEvent')[0]->json_decode()
439 call assert_equal({
440 \ 'windows': [1002, 1001, 1000],
441 \ }, event)
442
443 call delete('XresizeEvent')
444 call StopVimInTerminal(buf)
445endfunc
446
LemonBoy09371822022-04-08 15:18:45 +0100447func Test_WinScrolled()
448 CheckRunVimInTerminal
449
450 let lines =<< trim END
zeertzjqd58862d2022-04-12 11:32:48 +0100451 set nowrap scrolloff=0
452 for ii in range(1, 18)
453 call setline(ii, repeat(nr2char(96 + ii), ii * 2))
454 endfor
455 let win_id = win_getid()
456 let g:matched = v:false
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000457 func WriteScrollEvent()
458 call writefile([json_encode(v:event)], 'XscrollEvent')
459 endfunc
zeertzjqd58862d2022-04-12 11:32:48 +0100460 execute 'au WinScrolled' win_id 'let g:matched = v:true'
461 let g:scrolled = 0
462 au WinScrolled * let g:scrolled += 1
463 au WinScrolled * let g:amatch = str2nr(expand('<amatch>'))
464 au WinScrolled * let g:afile = str2nr(expand('<afile>'))
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000465 au WinScrolled * call WriteScrollEvent()
LemonBoy09371822022-04-08 15:18:45 +0100466 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +0100467 call writefile(lines, 'Xtest_winscrolled', 'D')
LemonBoy09371822022-04-08 15:18:45 +0100468 let buf = RunVimInTerminal('-S Xtest_winscrolled', {'rows': 6})
469
470 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
471 call WaitForAssert({-> assert_match('^0 ', term_getline(buf, 6))}, 1000)
472
473 " Scroll left/right in Normal mode.
474 call term_sendkeys(buf, "zlzh:echo g:scrolled\<CR>")
475 call WaitForAssert({-> assert_match('^2 ', term_getline(buf, 6))}, 1000)
476
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000477 let event = readfile('XscrollEvent')[0]->json_decode()
478 call assert_equal({
zeertzjq3fc84dc2022-12-07 09:17:59 +0000479 \ 'all': {'leftcol': 1, 'topline': 0, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
480 \ '1000': {'leftcol': -1, 'topline': 0, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0}
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000481 \ }, event)
482
LemonBoy09371822022-04-08 15:18:45 +0100483 " Scroll up/down in Normal mode.
484 call term_sendkeys(buf, "\<c-e>\<c-y>:echo g:scrolled\<CR>")
485 call WaitForAssert({-> assert_match('^4 ', term_getline(buf, 6))}, 1000)
486
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000487 let event = readfile('XscrollEvent')[0]->json_decode()
488 call assert_equal({
zeertzjq3fc84dc2022-12-07 09:17:59 +0000489 \ 'all': {'leftcol': 0, 'topline': 1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
490 \ '1000': {'leftcol': 0, 'topline': -1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0}
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000491 \ }, event)
492
LemonBoy09371822022-04-08 15:18:45 +0100493 " Scroll up/down in Insert mode.
494 call term_sendkeys(buf, "Mi\<c-x>\<c-e>\<Esc>i\<c-x>\<c-y>\<Esc>")
495 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
496 call WaitForAssert({-> assert_match('^6 ', term_getline(buf, 6))}, 1000)
497
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000498 let event = readfile('XscrollEvent')[0]->json_decode()
499 call assert_equal({
zeertzjq3fc84dc2022-12-07 09:17:59 +0000500 \ 'all': {'leftcol': 0, 'topline': 1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
501 \ '1000': {'leftcol': 0, 'topline': -1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0}
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000502 \ }, event)
503
LemonBoy09371822022-04-08 15:18:45 +0100504 " Scroll the window horizontally to focus the last letter of the third line
505 " containing only six characters. Moving to the previous and shorter lines
506 " should trigger another autocommand as Vim has to make them visible.
507 call term_sendkeys(buf, "5zl2k")
508 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
509 call WaitForAssert({-> assert_match('^8 ', term_getline(buf, 6))}, 1000)
510
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000511 let event = readfile('XscrollEvent')[0]->json_decode()
512 call assert_equal({
zeertzjq3fc84dc2022-12-07 09:17:59 +0000513 \ 'all': {'leftcol': 5, 'topline': 0, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
514 \ '1000': {'leftcol': -5, 'topline': 0, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0}
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000515 \ }, event)
516
LemonBoy09371822022-04-08 15:18:45 +0100517 " Ensure the command was triggered for the specified window ID.
518 call term_sendkeys(buf, ":echo g:matched\<CR>")
519 call WaitForAssert({-> assert_match('^v:true ', term_getline(buf, 6))}, 1000)
520
521 " Ensure the expansion of <amatch> and <afile> matches the window ID.
522 call term_sendkeys(buf, ":echo g:amatch == win_id && g:afile == win_id\<CR>")
523 call WaitForAssert({-> assert_match('^v:true ', term_getline(buf, 6))}, 1000)
524
Bram Moolenaar35fc61c2022-11-22 12:40:50 +0000525 call delete('XscrollEvent')
LemonBoy09371822022-04-08 15:18:45 +0100526 call StopVimInTerminal(buf)
LemonBoy09371822022-04-08 15:18:45 +0100527endfunc
528
LemonBoy66e13ae2022-04-21 22:52:11 +0100529func Test_WinScrolled_mouse()
530 CheckRunVimInTerminal
531
532 let lines =<< trim END
533 set nowrap scrolloff=0
534 set mouse=a term=xterm ttymouse=sgr mousetime=200 clipboard=
535 call setline(1, ['foo']->repeat(32))
536 split
537 let g:scrolled = 0
538 au WinScrolled * let g:scrolled += 1
539 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +0100540 call writefile(lines, 'Xtest_winscrolled_mouse', 'D')
LemonBoy66e13ae2022-04-21 22:52:11 +0100541 let buf = RunVimInTerminal('-S Xtest_winscrolled_mouse', {'rows': 10})
542
543 " With the upper split focused, send a scroll-down event to the unfocused one.
544 call test_setmouse(7, 1)
545 call term_sendkeys(buf, "\<ScrollWheelDown>")
546 call TermWait(buf)
547 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
548 call WaitForAssert({-> assert_match('^1', term_getline(buf, 10))}, 1000)
549
550 " Again, but this time while we're in insert mode.
551 call term_sendkeys(buf, "i\<ScrollWheelDown>\<Esc>")
552 call TermWait(buf)
553 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
554 call WaitForAssert({-> assert_match('^2', term_getline(buf, 10))}, 1000)
555
556 call StopVimInTerminal(buf)
LemonBoy66e13ae2022-04-21 22:52:11 +0100557endfunc
558
zeertzjqd58862d2022-04-12 11:32:48 +0100559func Test_WinScrolled_close_curwin()
560 CheckRunVimInTerminal
561
562 let lines =<< trim END
563 set nowrap scrolloff=0
564 call setline(1, ['aaa', 'bbb'])
565 vsplit
566 au WinScrolled * close
567 au VimLeave * call writefile(['123456'], 'Xtestout')
568 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +0100569 call writefile(lines, 'Xtest_winscrolled_close_curwin', 'D')
zeertzjqd58862d2022-04-12 11:32:48 +0100570 let buf = RunVimInTerminal('-S Xtest_winscrolled_close_curwin', {'rows': 6})
571
572 " This was using freed memory
573 call term_sendkeys(buf, "\<C-E>")
574 call TermWait(buf)
575 call StopVimInTerminal(buf)
576
Bram Moolenaar0a60f792022-11-19 21:18:11 +0000577 " check the startup script finished to the end
zeertzjqd58862d2022-04-12 11:32:48 +0100578 call assert_equal(['123456'], readfile('Xtestout'))
zeertzjqd58862d2022-04-12 11:32:48 +0100579 call delete('Xtestout')
580endfunc
581
Bram Moolenaar0a60f792022-11-19 21:18:11 +0000582func Test_WinScrolled_once_only()
583 CheckRunVimInTerminal
584
585 let lines =<< trim END
586 set cmdheight=2
587 call setline(1, ['aaa', 'bbb'])
588 let trigger_count = 0
589 func ShowInfo(id)
590 echo g:trigger_count g:winid winlayout()
591 endfunc
592
593 vsplit
594 split
595 " use a timer to show the info after a redraw
596 au WinScrolled * let trigger_count += 1 | let winid = expand('<amatch>') | call timer_start(100, 'ShowInfo')
597 wincmd j
598 wincmd l
599 END
600 call writefile(lines, 'Xtest_winscrolled_once', 'D')
601 let buf = RunVimInTerminal('-S Xtest_winscrolled_once', #{rows: 10, cols: 60, statusoff: 2})
602
603 call term_sendkeys(buf, "\<C-E>")
604 call VerifyScreenDump(buf, 'Test_winscrolled_once_only_1', {})
605
606 call StopVimInTerminal(buf)
607endfunc
608
Bram Moolenaar29967732022-11-20 12:11:45 +0000609" Check that WinScrolled is not triggered immediately when defined and there
610" are split windows.
611func Test_WinScrolled_not_when_defined()
612 CheckRunVimInTerminal
613
614 let lines =<< trim END
615 call setline(1, ['aaa', 'bbb'])
616 echo 'nothing happened'
617 func ShowTriggered(id)
618 echo 'triggered'
619 endfunc
620 END
621 call writefile(lines, 'Xtest_winscrolled_not', 'D')
622 let buf = RunVimInTerminal('-S Xtest_winscrolled_not', #{rows: 10, cols: 60, statusoff: 2})
623 call term_sendkeys(buf, ":split\<CR>")
624 call TermWait(buf)
625 " use a timer to show the message after redrawing
626 call term_sendkeys(buf, ":au WinScrolled * call timer_start(100, 'ShowTriggered')\<CR>")
627 call VerifyScreenDump(buf, 'Test_winscrolled_not_when_defined_1', {})
628
629 call term_sendkeys(buf, "\<C-E>")
630 call VerifyScreenDump(buf, 'Test_winscrolled_not_when_defined_2', {})
631
632 call StopVimInTerminal(buf)
633endfunc
634
zeertzjq670ab032022-08-28 19:16:15 +0100635func Test_WinScrolled_long_wrapped()
636 CheckRunVimInTerminal
637
638 let lines =<< trim END
639 set scrolloff=0
640 let height = winheight(0)
641 let width = winwidth(0)
642 let g:scrolled = 0
643 au WinScrolled * let g:scrolled += 1
644 call setline(1, repeat('foo', height * width))
645 call cursor(1, height * width)
646 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +0100647 call writefile(lines, 'Xtest_winscrolled_long_wrapped', 'D')
zeertzjq670ab032022-08-28 19:16:15 +0100648 let buf = RunVimInTerminal('-S Xtest_winscrolled_long_wrapped', {'rows': 6})
649
650 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
651 call WaitForAssert({-> assert_match('^0 ', term_getline(buf, 6))}, 1000)
652
653 call term_sendkeys(buf, 'gj')
654 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
655 call WaitForAssert({-> assert_match('^1 ', term_getline(buf, 6))}, 1000)
656
657 call term_sendkeys(buf, '0')
658 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
659 call WaitForAssert({-> assert_match('^2 ', term_getline(buf, 6))}, 1000)
660
661 call term_sendkeys(buf, '$')
662 call term_sendkeys(buf, ":echo g:scrolled\<CR>")
663 call WaitForAssert({-> assert_match('^3 ', term_getline(buf, 6))}, 1000)
Bram Moolenaar23526d22022-12-05 15:50:41 +0000664
665 call StopVimInTerminal(buf)
zeertzjq670ab032022-08-28 19:16:15 +0100666endfunc
667
zeertzjq3fc84dc2022-12-07 09:17:59 +0000668func Test_WinScrolled_diff()
669 CheckRunVimInTerminal
670
671 let lines =<< trim END
672 set diffopt+=foldcolumn:0
673 call setline(1, ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i'])
674 vnew
675 call setline(1, ['d', 'e', 'f', 'g', 'h', 'i'])
676 windo diffthis
677 func WriteScrollEvent()
678 call writefile([json_encode(v:event)], 'XscrollEvent')
679 endfunc
680 au WinScrolled * call WriteScrollEvent()
681 END
682 call writefile(lines, 'Xtest_winscrolled_diff', 'D')
683 let buf = RunVimInTerminal('-S Xtest_winscrolled_diff', {'rows': 8})
684
685 call term_sendkeys(buf, "\<C-E>")
686 call WaitForAssert({-> assert_match('^d', term_getline(buf, 3))}, 1000)
687
688 let event = readfile('XscrollEvent')[0]->json_decode()
689 call assert_equal({
690 \ 'all': {'leftcol': 0, 'topline': 1, 'topfill': 1, 'width': 0, 'height': 0, 'skipcol': 0},
691 \ '1000': {'leftcol': 0, 'topline': 1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
692 \ '1001': {'leftcol': 0, 'topline': 0, 'topfill': -1, 'width': 0, 'height': 0, 'skipcol': 0}
693 \ }, event)
694
695 call term_sendkeys(buf, "2\<C-E>")
696 call WaitForAssert({-> assert_match('^f', term_getline(buf, 3))}, 1000)
697
698 let event = readfile('XscrollEvent')[0]->json_decode()
699 call assert_equal({
700 \ 'all': {'leftcol': 0, 'topline': 2, 'topfill': 2, 'width': 0, 'height': 0, 'skipcol': 0},
701 \ '1000': {'leftcol': 0, 'topline': 2, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
702 \ '1001': {'leftcol': 0, 'topline': 0, 'topfill': -2, 'width': 0, 'height': 0, 'skipcol': 0}
703 \ }, event)
704
705 call term_sendkeys(buf, "\<C-E>")
706 call WaitForAssert({-> assert_match('^g', term_getline(buf, 3))}, 1000)
707
708 let event = readfile('XscrollEvent')[0]->json_decode()
709 call assert_equal({
710 \ 'all': {'leftcol': 0, 'topline': 2, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
711 \ '1000': {'leftcol': 0, 'topline': 1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
712 \ '1001': {'leftcol': 0, 'topline': 1, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0}
713 \ }, event)
714
715 call term_sendkeys(buf, "2\<C-Y>")
716 call WaitForAssert({-> assert_match('^e', term_getline(buf, 3))}, 1000)
717
718 let event = readfile('XscrollEvent')[0]->json_decode()
719 call assert_equal({
720 \ 'all': {'leftcol': 0, 'topline': 3, 'topfill': 1, 'width': 0, 'height': 0, 'skipcol': 0},
721 \ '1000': {'leftcol': 0, 'topline': -2, 'topfill': 0, 'width': 0, 'height': 0, 'skipcol': 0},
722 \ '1001': {'leftcol': 0, 'topline': -1, 'topfill': 1, 'width': 0, 'height': 0, 'skipcol': 0}
723 \ }, event)
724
725 call StopVimInTerminal(buf)
Dominique Pelle541c87c2023-01-17 21:20:44 +0000726 call delete('XscrollEvent')
zeertzjq3fc84dc2022-12-07 09:17:59 +0000727endfunc
728
naohiro ono23beefe2021-11-13 12:38:49 +0000729func Test_WinClosed()
730 " Test that the pattern is matched against the closed window's ID, and both
731 " <amatch> and <afile> are set to it.
732 new
733 let winid = win_getid()
734 let g:matched = v:false
735 augroup test-WinClosed
736 autocmd!
737 execute 'autocmd WinClosed' winid 'let g:matched = v:true'
738 autocmd WinClosed * let g:amatch = str2nr(expand('<amatch>'))
739 autocmd WinClosed * let g:afile = str2nr(expand('<afile>'))
740 augroup END
741 close
742 call assert_true(g:matched)
743 call assert_equal(winid, g:amatch)
744 call assert_equal(winid, g:afile)
745
746 " Test that WinClosed is non-recursive.
747 new
748 new
749 call assert_equal(3, winnr('$'))
750 let g:triggered = 0
751 augroup test-WinClosed
752 autocmd!
753 autocmd WinClosed * let g:triggered += 1
754 autocmd WinClosed * 2 wincmd c
755 augroup END
756 close
757 call assert_equal(1, winnr('$'))
758 call assert_equal(1, g:triggered)
759
760 autocmd! test-WinClosed
761 augroup! test-WinClosed
762 unlet g:matched
763 unlet g:amatch
764 unlet g:afile
765 unlet g:triggered
766endfunc
767
Bram Moolenaarc947b9a2022-04-06 17:59:21 +0100768func Test_WinClosed_throws()
769 vnew
770 let bnr = bufnr()
771 call assert_equal(1, bufloaded(bnr))
772 augroup test-WinClosed
773 autocmd WinClosed * throw 'foo'
774 augroup END
775 try
776 close
777 catch /.*/
778 endtry
779 call assert_equal(0, bufloaded(bnr))
780
781 autocmd! test-WinClosed
782 augroup! test-WinClosed
783endfunc
784
zeertzjq6a069402022-04-07 14:08:29 +0100785func Test_WinClosed_throws_with_tabs()
786 tabnew
787 let bnr = bufnr()
788 call assert_equal(1, bufloaded(bnr))
789 augroup test-WinClosed
790 autocmd WinClosed * throw 'foo'
791 augroup END
792 try
793 close
794 catch /.*/
795 endtry
796 call assert_equal(0, bufloaded(bnr))
797
798 autocmd! test-WinClosed
799 augroup! test-WinClosed
800endfunc
801
zeertzjq62de54b2022-09-22 18:08:37 +0100802" This used to trigger WinClosed twice for the same window, and the window's
803" buffer was NULL in the second autocommand.
804func Test_WinClosed_switch_tab()
805 edit Xa
806 split Xb
807 split Xc
808 tab split
809 new
810 augroup test-WinClosed
811 autocmd WinClosed * tabprev | bwipe!
812 augroup END
813 close
814 " Check that the tabline has been fully removed
815 call assert_equal([1, 1], win_screenpos(0))
816
817 autocmd! test-WinClosed
818 augroup! test-WinClosed
819 %bwipe!
820endfunc
821
zeertzjqb2ec0da2024-03-09 15:39:27 +0100822" This used to trigger WinClosed twice for the same window, and the window's
823" buffer was NULL in the second autocommand.
824func Test_WinClosed_BufUnload_close_other()
825 tabnew
826 let g:tab = tabpagenr()
827 let g:buf = bufnr()
828 new
829 setlocal bufhidden=wipe
830 augroup test-WinClosed
831 autocmd BufUnload * ++once exe g:buf .. 'bwipe!'
832 autocmd WinClosed * call tabpagebuflist(g:tab)
833 augroup END
834 close
835
836 unlet g:tab
837 unlet g:buf
838 autocmd! test-WinClosed
839 augroup! test-WinClosed
840 %bwipe!
841endfunc
842
Bram Moolenaare99e8442016-07-26 20:43:40 +0200843func s:AddAnAutocmd()
844 augroup vimBarTest
845 au BufReadCmd * echo 'hello'
846 augroup END
847 call assert_equal(3, len(split(execute('au vimBarTest'), "\n")))
848endfunc
849
850func Test_early_bar()
851 " test that a bar is recognized before the {event}
852 call s:AddAnAutocmd()
Bram Moolenaarb8e642f2021-11-20 10:38:25 +0000853 augroup vimBarTest | au! | let done = 77 | augroup END
Bram Moolenaare99e8442016-07-26 20:43:40 +0200854 call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
Bram Moolenaarb8e642f2021-11-20 10:38:25 +0000855 call assert_equal(77, done)
Bram Moolenaare99e8442016-07-26 20:43:40 +0200856
857 call s:AddAnAutocmd()
Bram Moolenaarb8e642f2021-11-20 10:38:25 +0000858 augroup vimBarTest| au!| let done = 88 | augroup END
Bram Moolenaare99e8442016-07-26 20:43:40 +0200859 call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
Bram Moolenaarb8e642f2021-11-20 10:38:25 +0000860 call assert_equal(88, done)
Bram Moolenaare99e8442016-07-26 20:43:40 +0200861
862 " test that a bar is recognized after the {event}
863 call s:AddAnAutocmd()
Bram Moolenaarb8e642f2021-11-20 10:38:25 +0000864 augroup vimBarTest| au!BufReadCmd| let done = 99 | augroup END
Bram Moolenaare99e8442016-07-26 20:43:40 +0200865 call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
Bram Moolenaarb8e642f2021-11-20 10:38:25 +0000866 call assert_equal(99, done)
Bram Moolenaare99e8442016-07-26 20:43:40 +0200867
868 " test that a bar is recognized after the {group}
869 call s:AddAnAutocmd()
870 au! vimBarTest|echo 'hello'
871 call assert_equal(1, len(split(execute('au vimBarTest'), "\n")))
872endfunc
Bram Moolenaarf2c4c392016-07-29 20:50:24 +0200873
Bram Moolenaar5c809082016-09-01 16:21:48 +0200874func RemoveGroup()
875 autocmd! StartOK
876 augroup! StartOK
877endfunc
878
Bram Moolenaarf2c4c392016-07-29 20:50:24 +0200879func Test_augroup_warning()
880 augroup TheWarning
881 au VimEnter * echo 'entering'
882 augroup END
Bram Moolenaar5dc4e2f2020-11-25 14:15:12 +0100883 call assert_match("TheWarning.*VimEnter", execute('au VimEnter'))
Bram Moolenaarf2c4c392016-07-29 20:50:24 +0200884 redir => res
885 augroup! TheWarning
886 redir END
Bram Moolenaar5dc4e2f2020-11-25 14:15:12 +0100887 call assert_match("W19:", res)
888 call assert_match("-Deleted-.*VimEnter", execute('au VimEnter'))
Bram Moolenaarf2c4c392016-07-29 20:50:24 +0200889
890 " check "Another" does not take the pace of the deleted entry
891 augroup Another
892 augroup END
Bram Moolenaar5dc4e2f2020-11-25 14:15:12 +0100893 call assert_match("-Deleted-.*VimEnter", execute('au VimEnter'))
Bram Moolenaaraeac9002016-09-06 22:15:08 +0200894 augroup! Another
Bram Moolenaar5c809082016-09-01 16:21:48 +0200895
896 " no warning for postpone aucmd delete
897 augroup StartOK
898 au VimEnter * call RemoveGroup()
899 augroup END
Bram Moolenaar5dc4e2f2020-11-25 14:15:12 +0100900 call assert_match("StartOK.*VimEnter", execute('au VimEnter'))
Bram Moolenaar5c809082016-09-01 16:21:48 +0200901 redir => res
902 doautocmd VimEnter
903 redir END
Bram Moolenaar5dc4e2f2020-11-25 14:15:12 +0100904 call assert_notmatch("W19:", res)
Bram Moolenaarde653f02016-09-03 16:59:06 +0200905 au! VimEnter
Bram Moolenaarad48e6c2020-04-21 22:19:45 +0200906
907 call assert_fails('augroup!', 'E471:')
Bram Moolenaarf2c4c392016-07-29 20:50:24 +0200908endfunc
Bram Moolenaarb62cc362016-09-03 16:43:53 +0200909
Bram Moolenaar8d84ff12017-10-26 16:42:16 +0200910func Test_BufReadCmdHelp()
911 " This used to cause access to free memory
912 au BufReadCmd * e +h
913 help
914
Bram Moolenaar8d84ff12017-10-26 16:42:16 +0200915 au! BufReadCmd
916endfunc
917
918func Test_BufReadCmdHelpJump()
919 " This used to cause access to free memory
920 au BufReadCmd * e +h{
Bram Moolenaarcf1ba352017-10-27 00:55:04 +0200921 " } to fix highlighting
922 call assert_fails('help', 'E434:')
Bram Moolenaar8d84ff12017-10-26 16:42:16 +0200923
Bram Moolenaar8d84ff12017-10-26 16:42:16 +0200924 au! BufReadCmd
925endfunc
926
zeertzjq93f72cc2022-08-26 15:34:52 +0100927" BufReadCmd is triggered for a "nofile" buffer. Check all values.
Bram Moolenaarb1d2c812022-08-26 11:55:01 +0100928func Test_BufReadCmdNofile()
zeertzjq93f72cc2022-08-26 15:34:52 +0100929 for val in ['nofile',
930 \ 'nowrite',
931 \ 'acwrite',
932 \ 'quickfix',
933 \ 'help',
934 \ 'terminal',
935 \ 'prompt',
936 \ 'popup',
937 \ ]
938 new somefile
939 exe 'set buftype=' .. val
940 au BufReadCmd somefile call setline(1, 'triggered')
941 edit
942 call assert_equal('triggered', getline(1))
Bram Moolenaarb1d2c812022-08-26 11:55:01 +0100943
zeertzjq93f72cc2022-08-26 15:34:52 +0100944 au! BufReadCmd
945 bwipe!
946 endfor
Bram Moolenaarb1d2c812022-08-26 11:55:01 +0100947endfunc
948
Bram Moolenaarb62cc362016-09-03 16:43:53 +0200949func Test_augroup_deleted()
Bram Moolenaarde653f02016-09-03 16:59:06 +0200950 " This caused a crash before E936 was introduced
Bram Moolenaarb62cc362016-09-03 16:43:53 +0200951 augroup x
Bram Moolenaarde653f02016-09-03 16:59:06 +0200952 call assert_fails('augroup! x', 'E936:')
953 au VimEnter * echo
954 augroup end
Bram Moolenaarb62cc362016-09-03 16:43:53 +0200955 augroup! x
Bram Moolenaar5dc4e2f2020-11-25 14:15:12 +0100956 call assert_match("-Deleted-.*VimEnter", execute('au VimEnter'))
Bram Moolenaarde653f02016-09-03 16:59:06 +0200957 au! VimEnter
Bram Moolenaarb62cc362016-09-03 16:43:53 +0200958endfunc
959
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200960" Tests for autocommands on :close command.
961" This used to be in test13.
962func Test_three_windows()
Bram Moolenaarb3435b02016-09-29 20:54:59 +0200963 " Clean up buffers, because in some cases this function fails.
964 call s:cleanup_buffers()
965
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200966 " Write three files and open them, each in a window.
967 " Then go to next window, with autocommand that deletes the previous one.
968 " Do this twice, writing the file.
969 e! Xtestje1
970 call setline(1, 'testje1')
971 w
972 sp Xtestje2
973 call setline(1, 'testje2')
974 w
975 sp Xtestje3
976 call setline(1, 'testje3')
977 w
978 wincmd w
979 au WinLeave Xtestje2 bwipe
980 wincmd w
981 call assert_equal('Xtestje1', expand('%'))
982
983 au WinLeave Xtestje1 bwipe Xtestje3
984 close
985 call assert_equal('Xtestje1', expand('%'))
986
987 " Test deleting the buffer on a Unload event. If this goes wrong there
988 " will be the ATTENTION prompt.
989 e Xtestje1
990 au!
991 au! BufUnload Xtestje1 bwipe
992 call assert_fails('e Xtestje3', 'E937:')
993 call assert_equal('Xtestje3', expand('%'))
994
995 e Xtestje2
996 sp Xtestje1
997 call assert_fails('e', 'E937:')
Bram Moolenaara997b452018-04-17 23:24:06 +0200998 call assert_equal('Xtestje1', expand('%'))
Bram Moolenaare0ab94e2016-09-04 19:50:54 +0200999
1000 " Test changing buffers in a BufWipeout autocommand. If this goes wrong
1001 " there are ml_line errors and/or a Crash.
1002 au!
1003 only
1004 e Xanother
1005 e Xtestje1
1006 bwipe Xtestje2
1007 bwipe Xtestje3
1008 au BufWipeout Xtestje1 buf Xtestje1
1009 bwipe
1010 call assert_equal('Xanother', expand('%'))
1011
1012 only
1013 help
1014 wincmd w
1015 1quit
1016 call assert_equal('Xanother', expand('%'))
1017
1018 au!
Bram Moolenaar4520d442017-03-19 16:09:46 +01001019 enew
Bram Moolenaare0ab94e2016-09-04 19:50:54 +02001020 call delete('Xtestje1')
1021 call delete('Xtestje2')
1022 call delete('Xtestje3')
1023endfunc
Bram Moolenaare13b9af2017-01-13 22:01:02 +01001024
1025func Test_BufEnter()
1026 au! BufEnter
1027 au Bufenter * let val = val . '+'
1028 let g:val = ''
1029 split NewFile
1030 call assert_equal('+', g:val)
1031 bwipe!
1032 call assert_equal('++', g:val)
1033
1034 " Also get BufEnter when editing a directory
Bram Moolenaar6f14da12022-09-07 21:30:44 +01001035 call mkdir('Xbufenterdir', 'D')
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01001036 split Xbufenterdir
Bram Moolenaare13b9af2017-01-13 22:01:02 +01001037 call assert_equal('+++', g:val)
Bram Moolenaare94260f2017-03-21 15:50:12 +01001038
1039 " On MS-Windows we can't edit the directory, make sure we wipe the right
1040 " buffer.
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01001041 bwipe! Xbufenterdir
Bram Moolenaare13b9af2017-01-13 22:01:02 +01001042 au! BufEnter
Bram Moolenaara9b5b852022-08-26 13:16:20 +01001043
1044 " Editing a "nofile" buffer doesn't read the file but does trigger BufEnter
zeertzjq93f72cc2022-08-26 15:34:52 +01001045 " for historic reasons. Also test other 'buftype' values.
1046 for val in ['nofile',
1047 \ 'nowrite',
1048 \ 'acwrite',
1049 \ 'quickfix',
1050 \ 'help',
1051 \ 'terminal',
1052 \ 'prompt',
1053 \ 'popup',
1054 \ ]
1055 new somefile
1056 exe 'set buftype=' .. val
1057 au BufEnter somefile call setline(1, 'some text')
1058 edit
1059 call assert_equal('some text', getline(1))
1060 bwipe!
1061 au! BufEnter
1062 endfor
Bram Moolenaar9fda8152022-11-19 13:14:10 +00001063
1064 new
1065 new
1066 autocmd BufEnter * ++once close
1067 call assert_fails('close', 'E1312:')
1068
1069 au! BufEnter
1070 only
Bram Moolenaare13b9af2017-01-13 22:01:02 +01001071endfunc
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001072
1073" Closing a window might cause an endless loop
1074" E814 for older Vims
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001075func Test_autocmd_bufwipe_in_SessLoadPost()
Bram Moolenaar1d68d9b2017-10-13 22:33:32 +02001076 edit Xtest
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001077 tabnew
Bram Moolenaar1d68d9b2017-10-13 22:33:32 +02001078 file Xsomething
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001079 set noswapfile
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001080 mksession!
1081
Bram Moolenaarc79745a2019-05-20 22:12:34 +02001082 let content =<< trim [CODE]
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02001083 call test_override('ui_delay', 10)
Bram Moolenaarc79745a2019-05-20 22:12:34 +02001084 set nocp noswapfile
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001085 let v:swapchoice = "e"
Bram Moolenaarc79745a2019-05-20 22:12:34 +02001086 augroup test_autocmd_sessionload
1087 autocmd!
1088 autocmd SessionLoadPost * exe bufnr("Xsomething") . "bw!"
1089 augroup END
1090
1091 func WriteErrors()
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001092 call writefile([execute("messages")], "XerrorsBwipe")
Bram Moolenaarc79745a2019-05-20 22:12:34 +02001093 endfunc
1094 au VimLeave * call WriteErrors()
1095 [CODE]
1096
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001097 call writefile(content, 'Xvimrc', 'D')
Bram Moolenaar93344c22019-08-14 21:12:05 +02001098 call system(GetVimCommand('Xvimrc') .. ' --not-a-term --noplugins -S Session.vim -c cq')
Bram Moolenaarae04a602022-09-09 15:08:10 +01001099 sleep 100m
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001100 let errors = join(readfile('XerrorsBwipe'))
Bram Moolenaare2e40752020-09-04 21:18:46 +02001101 call assert_match('E814:', errors)
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001102
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001103 set swapfile
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001104 for file in ['Session.vim', 'XerrorsBwipe']
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001105 call delete(file)
1106 endfor
1107endfunc
1108
Bram Moolenaar797e63b2021-01-15 16:22:52 +01001109" Using :blast and :ball for many events caused a crash, because b_nwindows was
1110" not incremented correctly.
1111func Test_autocmd_blast_badd()
1112 let content =<< trim [CODE]
1113 au BufNew,BufAdd,BufWinEnter,BufEnter,BufLeave,BufWinLeave,BufUnload,VimEnter foo* blast
1114 edit foo1
1115 au BufNew,BufAdd,BufWinEnter,BufEnter,BufLeave,BufWinLeave,BufUnload,VimEnter foo* ball
1116 edit foo2
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001117 call writefile(['OK'], 'XerrorsBlast')
Bram Moolenaar797e63b2021-01-15 16:22:52 +01001118 qall
1119 [CODE]
1120
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001121 call writefile(content, 'XblastBall', 'D')
Bram Moolenaar797e63b2021-01-15 16:22:52 +01001122 call system(GetVimCommand() .. ' --clean -S XblastBall')
Bram Moolenaarae04a602022-09-09 15:08:10 +01001123 sleep 100m
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001124 call assert_match('OK', readfile('XerrorsBlast')->join())
Bram Moolenaar797e63b2021-01-15 16:22:52 +01001125
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001126 call delete('XerrorsBlast')
Bram Moolenaar797e63b2021-01-15 16:22:52 +01001127endfunc
1128
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001129" SEGV occurs in older versions.
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001130func Test_autocmd_bufwipe_in_SessLoadPost2()
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001131 tabnew
1132 set noswapfile
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001133 mksession!
1134
Bram Moolenaarc79745a2019-05-20 22:12:34 +02001135 let content =<< trim [CODE]
1136 set nocp noswapfile
1137 function! DeleteInactiveBufs()
1138 tabfirst
1139 let tabblist = []
1140 for i in range(1, tabpagenr(''$''))
1141 call extend(tabblist, tabpagebuflist(i))
1142 endfor
1143 for b in range(1, bufnr(''$''))
1144 if bufexists(b) && buflisted(b) && (index(tabblist, b) == -1 || bufname(b) =~# ''^$'')
1145 exec ''bwipeout '' . b
1146 endif
1147 endfor
1148 echomsg "SessionLoadPost DONE"
1149 endfunction
1150 au SessionLoadPost * call DeleteInactiveBufs()
1151
1152 func WriteErrors()
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001153 call writefile([execute("messages")], "XerrorsPost")
Bram Moolenaarc79745a2019-05-20 22:12:34 +02001154 endfunc
1155 au VimLeave * call WriteErrors()
1156 [CODE]
1157
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001158 call writefile(content, 'Xvimrc', 'D')
Bram Moolenaar93344c22019-08-14 21:12:05 +02001159 call system(GetVimCommand('Xvimrc') .. ' --not-a-term --noplugins -S Session.vim -c cq')
Bram Moolenaarae04a602022-09-09 15:08:10 +01001160 sleep 100m
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001161 let errors = join(readfile('XerrorsPost'))
Bram Moolenaare94260f2017-03-21 15:50:12 +01001162 " This probably only ever matches on unix.
1163 call assert_notmatch('Caught deadly signal SEGV', errors)
1164 call assert_match('SessionLoadPost DONE', errors)
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001165
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001166 set swapfile
Bram Moolenaarbe9fc5b2022-09-09 17:09:35 +01001167 for file in ['Session.vim', 'XerrorsPost']
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01001168 call delete(file)
1169 endfor
1170endfunc
Bram Moolenaarfaf29d72017-07-09 11:07:16 +02001171
1172func Test_empty_doau()
1173 doau \|
1174endfunc
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001175
1176func s:AutoCommandOptionSet(match)
Bram Moolenaard7c96872019-06-15 17:12:48 +02001177 let template = "Option: <%s>, OldVal: <%s>, OldValLocal: <%s>, OldValGlobal: <%s>, NewVal: <%s>, Scope: <%s>, Command: <%s>\n"
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001178 let item = remove(g:options, 0)
Bram Moolenaard7c96872019-06-15 17:12:48 +02001179 let expected = printf(template, item[0], item[1], item[2], item[3], item[4], item[5], item[6])
1180 let actual = printf(template, a:match, v:option_old, v:option_oldlocal, v:option_oldglobal, v:option_new, v:option_type, v:option_command)
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001181 let g:opt = [expected, actual]
1182 "call assert_equal(expected, actual)
1183endfunc
1184
1185func Test_OptionSet()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02001186 CheckOption autochdir
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001187
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001188 badd test_autocmd.vim
1189
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001190 call test_override('starting', 1)
1191 set nocp
1192 au OptionSet * :call s:AutoCommandOptionSet(expand("<amatch>"))
1193
1194 " 1: Setting number option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001195 let g:options = [['number', 0, 0, 0, 1, 'global', 'set']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001196 set nu
1197 call assert_equal([], g:options)
1198 call assert_equal(g:opt[0], g:opt[1])
1199
1200 " 2: Setting local number option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001201 let g:options = [['number', 1, 1, '', 0, 'local', 'setlocal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001202 setlocal nonu
1203 call assert_equal([], g:options)
1204 call assert_equal(g:opt[0], g:opt[1])
1205
1206 " 3: Setting global number option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001207 let g:options = [['number', 1, '', 1, 0, 'global', 'setglobal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001208 setglobal nonu
1209 call assert_equal([], g:options)
1210 call assert_equal(g:opt[0], g:opt[1])
1211
1212 " 4: Setting local autoindent option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001213 let g:options = [['autoindent', 0, 0, '', 1, 'local', 'setlocal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001214 setlocal ai
1215 call assert_equal([], g:options)
1216 call assert_equal(g:opt[0], g:opt[1])
1217
1218 " 5: Setting global autoindent option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001219 let g:options = [['autoindent', 0, '', 0, 1, 'global', 'setglobal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001220 setglobal ai
1221 call assert_equal([], g:options)
1222 call assert_equal(g:opt[0], g:opt[1])
1223
1224 " 6: Setting global autoindent option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001225 let g:options = [['autoindent', 1, 1, 1, 0, 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001226 set ai!
1227 call assert_equal([], g:options)
1228 call assert_equal(g:opt[0], g:opt[1])
1229
1230 " 6a: Setting global autoindent option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001231 let g:options = [['autoindent', 1, 1, 0, 0, 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001232 noa setlocal ai
1233 noa setglobal noai
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001234 set ai!
1235 call assert_equal([], g:options)
1236 call assert_equal(g:opt[0], g:opt[1])
1237
1238 " Should not print anything, use :noa
1239 " 7: don't trigger OptionSet"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001240 let g:options = [['invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001241 noa set nonu
Bram Moolenaard7c96872019-06-15 17:12:48 +02001242 call assert_equal([['invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid']], g:options)
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001243 call assert_equal(g:opt[0], g:opt[1])
1244
1245 " 8: Setting several global list and number option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001246 let g:options = [['list', 0, 0, 0, 1, 'global', 'set'], ['number', 0, 0, 0, 1, 'global', 'set']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001247 set list nu
1248 call assert_equal([], g:options)
1249 call assert_equal(g:opt[0], g:opt[1])
1250
1251 " 9: don't trigger OptionSet"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001252 let g:options = [['invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid'], ['invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001253 noa set nolist nonu
Bram Moolenaard7c96872019-06-15 17:12:48 +02001254 call assert_equal([['invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid'], ['invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid', 'invalid']], g:options)
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001255 call assert_equal(g:opt[0], g:opt[1])
1256
1257 " 10: Setting global acd"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001258 let g:options = [['autochdir', 0, 0, '', 1, 'local', 'setlocal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001259 setlocal acd
1260 call assert_equal([], g:options)
1261 call assert_equal(g:opt[0], g:opt[1])
1262
1263 " 11: Setting global autoread (also sets local value)"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001264 let g:options = [['autoread', 0, 0, 0, 1, 'global', 'set']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001265 set ar
1266 call assert_equal([], g:options)
1267 call assert_equal(g:opt[0], g:opt[1])
1268
1269 " 12: Setting local autoread"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001270 let g:options = [['autoread', 1, 1, '', 1, 'local', 'setlocal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001271 setlocal ar
1272 call assert_equal([], g:options)
1273 call assert_equal(g:opt[0], g:opt[1])
1274
1275 " 13: Setting global autoread"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001276 let g:options = [['autoread', 1, '', 1, 0, 'global', 'setglobal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001277 setglobal invar
1278 call assert_equal([], g:options)
1279 call assert_equal(g:opt[0], g:opt[1])
1280
1281 " 14: Setting option backspace through :let"
Luca Saccarola959ef612024-12-01 16:25:53 +01001282 let g:options = [['backspace', 'indent,eol,start', 'indent,eol,start', 'indent,eol,start', '', 'global', 'set']]
1283 let &bs = ''
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001284 call assert_equal([], g:options)
1285 call assert_equal(g:opt[0], g:opt[1])
1286
1287 " 15: Setting option backspace through setbufvar()"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001288 let g:options = [['backup', 0, 0, '', 1, 'local', 'setlocal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001289 " try twice, first time, shouldn't trigger because option name is invalid,
1290 " second time, it should trigger
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001291 let bnum = bufnr('%')
Bram Moolenaare2e40752020-09-04 21:18:46 +02001292 call assert_fails("call setbufvar(bnum, '&l:bk', 1)", 'E355:')
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001293 " should trigger, use correct option name
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001294 call setbufvar(bnum, '&backup', 1)
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001295 call assert_equal([], g:options)
1296 call assert_equal(g:opt[0], g:opt[1])
1297
1298 " 16: Setting number option using setwinvar"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001299 let g:options = [['number', 0, 0, '', 1, 'local', 'setlocal']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001300 call setwinvar(0, '&number', 1)
1301 call assert_equal([], g:options)
1302 call assert_equal(g:opt[0], g:opt[1])
1303
1304 " 17: Setting key option, shouldn't trigger"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001305 let g:options = [['key', 'invalid', 'invalid1', 'invalid2', 'invalid3', 'invalid4', 'invalid5']]
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001306 setlocal key=blah
1307 setlocal key=
Bram Moolenaard7c96872019-06-15 17:12:48 +02001308 call assert_equal([['key', 'invalid', 'invalid1', 'invalid2', 'invalid3', 'invalid4', 'invalid5']], g:options)
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001309 call assert_equal(g:opt[0], g:opt[1])
1310
Bram Moolenaard7c96872019-06-15 17:12:48 +02001311
1312 " 18a: Setting string global option"
1313 let oldval = &backupext
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001314 let g:options = [['backupext', oldval, oldval, oldval, 'foo', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001315 set backupext=foo
1316 call assert_equal([], g:options)
1317 call assert_equal(g:opt[0], g:opt[1])
1318
1319 " 18b: Resetting string global option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001320 let g:options = [['backupext', 'foo', 'foo', 'foo', oldval, 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001321 set backupext&
1322 call assert_equal([], g:options)
1323 call assert_equal(g:opt[0], g:opt[1])
1324
1325 " 18c: Setting global string global option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001326 let g:options = [['backupext', oldval, '', oldval, 'bar', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001327 setglobal backupext=bar
1328 call assert_equal([], g:options)
1329 call assert_equal(g:opt[0], g:opt[1])
1330
1331 " 18d: Setting local string global option"
1332 " As this is a global option this sets the global value even though
1333 " :setlocal is used!
1334 noa set backupext& " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001335 let g:options = [['backupext', oldval, oldval, '', 'baz', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001336 setlocal backupext=baz
1337 call assert_equal([], g:options)
1338 call assert_equal(g:opt[0], g:opt[1])
1339
1340 " 18e: Setting again string global option"
1341 noa setglobal backupext=ext_global " Reset global and local value (without triggering autocmd)
1342 noa setlocal backupext=ext_local " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001343 let g:options = [['backupext', 'ext_local', 'ext_local', 'ext_local', 'fuu', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001344 set backupext=fuu
1345 call assert_equal([], g:options)
1346 call assert_equal(g:opt[0], g:opt[1])
1347
1348
zeertzjqb811de52021-10-21 10:50:44 +01001349 " 19a: Setting string global-local (to buffer) option"
Bram Moolenaar8efa0262017-08-20 15:47:20 +02001350 let oldval = &tags
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001351 let g:options = [['tags', oldval, oldval, oldval, 'tagpath', 'global', 'set']]
Bram Moolenaar8efa0262017-08-20 15:47:20 +02001352 set tags=tagpath
1353 call assert_equal([], g:options)
1354 call assert_equal(g:opt[0], g:opt[1])
1355
zeertzjqb811de52021-10-21 10:50:44 +01001356 " 19b: Resetting string global-local (to buffer) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001357 let g:options = [['tags', 'tagpath', 'tagpath', 'tagpath', oldval, 'global', 'set']]
Bram Moolenaar8efa0262017-08-20 15:47:20 +02001358 set tags&
1359 call assert_equal([], g:options)
1360 call assert_equal(g:opt[0], g:opt[1])
1361
zeertzjqb811de52021-10-21 10:50:44 +01001362 " 19c: Setting global string global-local (to buffer) option "
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001363 let g:options = [['tags', oldval, '', oldval, 'tagpath1', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001364 setglobal tags=tagpath1
1365 call assert_equal([], g:options)
1366 call assert_equal(g:opt[0], g:opt[1])
1367
zeertzjqb811de52021-10-21 10:50:44 +01001368 " 19d: Setting local string global-local (to buffer) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001369 let g:options = [['tags', 'tagpath1', 'tagpath1', '', 'tagpath2', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001370 setlocal tags=tagpath2
1371 call assert_equal([], g:options)
1372 call assert_equal(g:opt[0], g:opt[1])
1373
zeertzjqb811de52021-10-21 10:50:44 +01001374 " 19e: Setting again string global-local (to buffer) option"
1375 " Note: v:option_old is the old global value for global-local string options
Bram Moolenaard7c96872019-06-15 17:12:48 +02001376 " but the old local value for all other kinds of options.
1377 noa setglobal tags=tag_global " Reset global and local value (without triggering autocmd)
1378 noa setlocal tags=tag_local
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001379 let g:options = [['tags', 'tag_global', 'tag_local', 'tag_global', 'tagpath', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001380 set tags=tagpath
1381 call assert_equal([], g:options)
1382 call assert_equal(g:opt[0], g:opt[1])
1383
zeertzjqb811de52021-10-21 10:50:44 +01001384 " 19f: Setting string global-local (to buffer) option to an empty string"
1385 " Note: v:option_old is the old global value for global-local string options
Bram Moolenaard7c96872019-06-15 17:12:48 +02001386 " but the old local value for all other kinds of options.
1387 noa set tags=tag_global " Reset global and local value (without triggering autocmd)
1388 noa setlocal tags= " empty string
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001389 let g:options = [['tags', 'tag_global', '', 'tag_global', 'tagpath', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001390 set tags=tagpath
1391 call assert_equal([], g:options)
1392 call assert_equal(g:opt[0], g:opt[1])
1393
1394
1395 " 20a: Setting string local (to buffer) option"
1396 let oldval = &spelllang
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001397 let g:options = [['spelllang', oldval, oldval, oldval, 'elvish,klingon', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001398 set spelllang=elvish,klingon
1399 call assert_equal([], g:options)
1400 call assert_equal(g:opt[0], g:opt[1])
1401
1402 " 20b: Resetting string local (to buffer) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001403 let g:options = [['spelllang', 'elvish,klingon', 'elvish,klingon', 'elvish,klingon', oldval, 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001404 set spelllang&
1405 call assert_equal([], g:options)
1406 call assert_equal(g:opt[0], g:opt[1])
1407
1408 " 20c: Setting global string local (to buffer) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001409 let g:options = [['spelllang', oldval, '', oldval, 'elvish', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001410 setglobal spelllang=elvish
1411 call assert_equal([], g:options)
1412 call assert_equal(g:opt[0], g:opt[1])
1413
1414 " 20d: Setting local string local (to buffer) option"
1415 noa set spelllang& " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001416 let g:options = [['spelllang', oldval, oldval, '', 'klingon', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001417 setlocal spelllang=klingon
1418 call assert_equal([], g:options)
1419 call assert_equal(g:opt[0], g:opt[1])
1420
1421 " 20e: Setting again string local (to buffer) option"
zeertzjqb811de52021-10-21 10:50:44 +01001422 " Note: v:option_old is the old global value for global-local string options
Bram Moolenaard7c96872019-06-15 17:12:48 +02001423 " but the old local value for all other kinds of options.
1424 noa setglobal spelllang=spellglobal " Reset global and local value (without triggering autocmd)
1425 noa setlocal spelllang=spelllocal
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001426 let g:options = [['spelllang', 'spelllocal', 'spelllocal', 'spellglobal', 'foo', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001427 set spelllang=foo
1428 call assert_equal([], g:options)
1429 call assert_equal(g:opt[0], g:opt[1])
1430
1431
zeertzjqb811de52021-10-21 10:50:44 +01001432 " 21a: Setting string global-local (to window) option"
Bram Moolenaard7c96872019-06-15 17:12:48 +02001433 let oldval = &statusline
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001434 let g:options = [['statusline', oldval, oldval, oldval, 'foo', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001435 set statusline=foo
1436 call assert_equal([], g:options)
1437 call assert_equal(g:opt[0], g:opt[1])
1438
zeertzjqb811de52021-10-21 10:50:44 +01001439 " 21b: Resetting string global-local (to window) option"
1440 " Note: v:option_old is the old global value for global-local string options
Bram Moolenaard7c96872019-06-15 17:12:48 +02001441 " but the old local value for all other kinds of options.
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001442 let g:options = [['statusline', 'foo', 'foo', 'foo', oldval, 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001443 set statusline&
1444 call assert_equal([], g:options)
1445 call assert_equal(g:opt[0], g:opt[1])
1446
zeertzjqb811de52021-10-21 10:50:44 +01001447 " 21c: Setting global string global-local (to window) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001448 let g:options = [['statusline', oldval, '', oldval, 'bar', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001449 setglobal statusline=bar
1450 call assert_equal([], g:options)
1451 call assert_equal(g:opt[0], g:opt[1])
1452
zeertzjqb811de52021-10-21 10:50:44 +01001453 " 21d: Setting local string global-local (to window) option"
Bram Moolenaard7c96872019-06-15 17:12:48 +02001454 noa set statusline& " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001455 let g:options = [['statusline', oldval, oldval, '', 'baz', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001456 setlocal statusline=baz
1457 call assert_equal([], g:options)
1458 call assert_equal(g:opt[0], g:opt[1])
1459
zeertzjqb811de52021-10-21 10:50:44 +01001460 " 21e: Setting again string global-local (to window) option"
1461 " Note: v:option_old is the old global value for global-local string options
Bram Moolenaard7c96872019-06-15 17:12:48 +02001462 " but the old local value for all other kinds of options.
1463 noa setglobal statusline=bar " Reset global and local value (without triggering autocmd)
1464 noa setlocal statusline=baz
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001465 let g:options = [['statusline', 'bar', 'baz', 'bar', 'foo', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001466 set statusline=foo
1467 call assert_equal([], g:options)
1468 call assert_equal(g:opt[0], g:opt[1])
1469
1470
1471 " 22a: Setting string local (to window) option"
1472 let oldval = &foldignore
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001473 let g:options = [['foldignore', oldval, oldval, oldval, 'fo', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001474 set foldignore=fo
1475 call assert_equal([], g:options)
1476 call assert_equal(g:opt[0], g:opt[1])
1477
1478 " 22b: Resetting string local (to window) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001479 let g:options = [['foldignore', 'fo', 'fo', 'fo', oldval, 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001480 set foldignore&
1481 call assert_equal([], g:options)
1482 call assert_equal(g:opt[0], g:opt[1])
1483
1484 " 22c: Setting global string local (to window) option"
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001485 let g:options = [['foldignore', oldval, '', oldval, 'bar', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001486 setglobal foldignore=bar
1487 call assert_equal([], g:options)
1488 call assert_equal(g:opt[0], g:opt[1])
1489
1490 " 22d: Setting local string local (to window) option"
1491 noa set foldignore& " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001492 let g:options = [['foldignore', oldval, oldval, '', 'baz', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001493 setlocal foldignore=baz
1494 call assert_equal([], g:options)
1495 call assert_equal(g:opt[0], g:opt[1])
1496
1497 " 22e: Setting again string local (to window) option"
1498 noa setglobal foldignore=glob " Reset global and local value (without triggering autocmd)
1499 noa setlocal foldignore=loc
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001500 let g:options = [['foldignore', 'loc', 'loc', 'glob', 'fo', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001501 set foldignore=fo
1502 call assert_equal([], g:options)
1503 call assert_equal(g:opt[0], g:opt[1])
1504
1505
zeertzjqb811de52021-10-21 10:50:44 +01001506 " 23a: Setting global number global option"
Bram Moolenaard7c96872019-06-15 17:12:48 +02001507 noa setglobal cmdheight=8 " Reset global and local value (without triggering autocmd)
1508 noa setlocal cmdheight=1 " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001509 let g:options = [['cmdheight', '1', '', '1', '2', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001510 setglobal cmdheight=2
1511 call assert_equal([], g:options)
1512 call assert_equal(g:opt[0], g:opt[1])
1513
1514 " 23b: Setting local number global option"
1515 noa setglobal cmdheight=8 " Reset global and local value (without triggering autocmd)
1516 noa setlocal cmdheight=1 " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001517 let g:options = [['cmdheight', '1', '1', '', '2', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001518 setlocal cmdheight=2
1519 call assert_equal([], g:options)
1520 call assert_equal(g:opt[0], g:opt[1])
1521
1522 " 23c: Setting again number global option"
1523 noa setglobal cmdheight=8 " Reset global and local value (without triggering autocmd)
1524 noa setlocal cmdheight=1 " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001525 let g:options = [['cmdheight', '1', '1', '1', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001526 set cmdheight=2
1527 call assert_equal([], g:options)
1528 call assert_equal(g:opt[0], g:opt[1])
1529
1530 " 23d: Setting again number global option"
1531 noa set cmdheight=8 " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001532 let g:options = [['cmdheight', '8', '8', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001533 set cmdheight=2
1534 call assert_equal([], g:options)
1535 call assert_equal(g:opt[0], g:opt[1])
1536
1537
1538 " 24a: Setting global number global-local (to buffer) option"
1539 noa setglobal undolevels=8 " Reset global and local value (without triggering autocmd)
1540 noa setlocal undolevels=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001541 let g:options = [['undolevels', '8', '', '8', '2', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001542 setglobal undolevels=2
1543 call assert_equal([], g:options)
1544 call assert_equal(g:opt[0], g:opt[1])
1545
1546 " 24b: Setting local number global-local (to buffer) option"
1547 noa setglobal undolevels=8 " Reset global and local value (without triggering autocmd)
1548 noa setlocal undolevels=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001549 let g:options = [['undolevels', '1', '1', '', '2', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001550 setlocal undolevels=2
1551 call assert_equal([], g:options)
1552 call assert_equal(g:opt[0], g:opt[1])
1553
1554 " 24c: Setting again number global-local (to buffer) option"
1555 noa setglobal undolevels=8 " Reset global and local value (without triggering autocmd)
1556 noa setlocal undolevels=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001557 let g:options = [['undolevels', '1', '1', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001558 set undolevels=2
1559 call assert_equal([], g:options)
1560 call assert_equal(g:opt[0], g:opt[1])
1561
1562 " 24d: Setting again global number global-local (to buffer) option"
1563 noa set undolevels=8 " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001564 let g:options = [['undolevels', '8', '8', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001565 set undolevels=2
1566 call assert_equal([], g:options)
1567 call assert_equal(g:opt[0], g:opt[1])
1568
1569
1570 " 25a: Setting global number local (to buffer) option"
1571 noa setglobal wrapmargin=8 " Reset global and local value (without triggering autocmd)
1572 noa setlocal wrapmargin=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001573 let g:options = [['wrapmargin', '8', '', '8', '2', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001574 setglobal wrapmargin=2
1575 call assert_equal([], g:options)
1576 call assert_equal(g:opt[0], g:opt[1])
1577
1578 " 25b: Setting local number local (to buffer) option"
1579 noa setglobal wrapmargin=8 " Reset global and local value (without triggering autocmd)
1580 noa setlocal wrapmargin=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001581 let g:options = [['wrapmargin', '1', '1', '', '2', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001582 setlocal wrapmargin=2
1583 call assert_equal([], g:options)
1584 call assert_equal(g:opt[0], g:opt[1])
1585
1586 " 25c: Setting again number local (to buffer) option"
1587 noa setglobal wrapmargin=8 " Reset global and local value (without triggering autocmd)
1588 noa setlocal wrapmargin=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001589 let g:options = [['wrapmargin', '1', '1', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001590 set wrapmargin=2
1591 call assert_equal([], g:options)
1592 call assert_equal(g:opt[0], g:opt[1])
1593
1594 " 25d: Setting again global number local (to buffer) option"
1595 noa set wrapmargin=8 " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001596 let g:options = [['wrapmargin', '8', '8', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001597 set wrapmargin=2
1598 call assert_equal([], g:options)
1599 call assert_equal(g:opt[0], g:opt[1])
1600
1601
1602 " 26: Setting number global-local (to window) option.
1603 " Such option does currently not exist.
1604
1605
1606 " 27a: Setting global number local (to window) option"
1607 noa setglobal foldcolumn=8 " Reset global and local value (without triggering autocmd)
1608 noa setlocal foldcolumn=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001609 let g:options = [['foldcolumn', '8', '', '8', '2', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001610 setglobal foldcolumn=2
1611 call assert_equal([], g:options)
1612 call assert_equal(g:opt[0], g:opt[1])
1613
1614 " 27b: Setting local number local (to window) option"
1615 noa setglobal foldcolumn=8 " Reset global and local value (without triggering autocmd)
1616 noa setlocal foldcolumn=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001617 let g:options = [['foldcolumn', '1', '1', '', '2', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001618 setlocal foldcolumn=2
1619 call assert_equal([], g:options)
1620 call assert_equal(g:opt[0], g:opt[1])
1621
1622 " 27c: Setting again number local (to window) option"
1623 noa setglobal foldcolumn=8 " Reset global and local value (without triggering autocmd)
1624 noa setlocal foldcolumn=1
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001625 let g:options = [['foldcolumn', '1', '1', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001626 set foldcolumn=2
1627 call assert_equal([], g:options)
1628 call assert_equal(g:opt[0], g:opt[1])
1629
zeertzjqb811de52021-10-21 10:50:44 +01001630 " 27d: Setting again global number local (to window) option"
Bram Moolenaard7c96872019-06-15 17:12:48 +02001631 noa set foldcolumn=8 " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001632 let g:options = [['foldcolumn', '8', '8', '8', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001633 set foldcolumn=2
1634 call assert_equal([], g:options)
1635 call assert_equal(g:opt[0], g:opt[1])
1636
1637
1638 " 28a: Setting global boolean global option"
1639 noa setglobal nowrapscan " Reset global and local value (without triggering autocmd)
1640 noa setlocal wrapscan " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001641 let g:options = [['wrapscan', '1', '', '1', '0', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001642 setglobal nowrapscan
1643 call assert_equal([], g:options)
1644 call assert_equal(g:opt[0], g:opt[1])
1645
1646 " 28b: Setting local boolean global option"
1647 noa setglobal nowrapscan " Reset global and local value (without triggering autocmd)
1648 noa setlocal wrapscan " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001649 let g:options = [['wrapscan', '1', '1', '', '0', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001650 setlocal nowrapscan
1651 call assert_equal([], g:options)
1652 call assert_equal(g:opt[0], g:opt[1])
1653
1654 " 28c: Setting again boolean global option"
1655 noa setglobal nowrapscan " Reset global and local value (without triggering autocmd)
1656 noa setlocal wrapscan " Sets the global(!) value!
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001657 let g:options = [['wrapscan', '1', '1', '1', '0', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001658 set nowrapscan
1659 call assert_equal([], g:options)
1660 call assert_equal(g:opt[0], g:opt[1])
1661
1662 " 28d: Setting again global boolean global option"
1663 noa set nowrapscan " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001664 let g:options = [['wrapscan', '0', '0', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001665 set wrapscan
1666 call assert_equal([], g:options)
1667 call assert_equal(g:opt[0], g:opt[1])
1668
1669
1670 " 29a: Setting global boolean global-local (to buffer) option"
1671 noa setglobal noautoread " Reset global and local value (without triggering autocmd)
1672 noa setlocal autoread
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001673 let g:options = [['autoread', '0', '', '0', '1', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001674 setglobal autoread
1675 call assert_equal([], g:options)
1676 call assert_equal(g:opt[0], g:opt[1])
1677
1678 " 29b: Setting local boolean global-local (to buffer) option"
1679 noa setglobal noautoread " Reset global and local value (without triggering autocmd)
1680 noa setlocal autoread
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001681 let g:options = [['autoread', '1', '1', '', '0', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001682 setlocal noautoread
1683 call assert_equal([], g:options)
1684 call assert_equal(g:opt[0], g:opt[1])
1685
1686 " 29c: Setting again boolean global-local (to buffer) option"
1687 noa setglobal noautoread " Reset global and local value (without triggering autocmd)
1688 noa setlocal autoread
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001689 let g:options = [['autoread', '1', '1', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001690 set autoread
1691 call assert_equal([], g:options)
1692 call assert_equal(g:opt[0], g:opt[1])
1693
1694 " 29d: Setting again global boolean global-local (to buffer) option"
1695 noa set noautoread " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001696 let g:options = [['autoread', '0', '0', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001697 set autoread
1698 call assert_equal([], g:options)
1699 call assert_equal(g:opt[0], g:opt[1])
1700
1701
1702 " 30a: Setting global boolean local (to buffer) option"
1703 noa setglobal nocindent " Reset global and local value (without triggering autocmd)
1704 noa setlocal cindent
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001705 let g:options = [['cindent', '0', '', '0', '1', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001706 setglobal cindent
1707 call assert_equal([], g:options)
1708 call assert_equal(g:opt[0], g:opt[1])
1709
1710 " 30b: Setting local boolean local (to buffer) option"
1711 noa setglobal nocindent " Reset global and local value (without triggering autocmd)
1712 noa setlocal cindent
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001713 let g:options = [['cindent', '1', '1', '', '0', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001714 setlocal nocindent
1715 call assert_equal([], g:options)
1716 call assert_equal(g:opt[0], g:opt[1])
1717
1718 " 30c: Setting again boolean local (to buffer) option"
1719 noa setglobal nocindent " Reset global and local value (without triggering autocmd)
1720 noa setlocal cindent
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001721 let g:options = [['cindent', '1', '1', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001722 set cindent
1723 call assert_equal([], g:options)
1724 call assert_equal(g:opt[0], g:opt[1])
1725
1726 " 30d: Setting again global boolean local (to buffer) option"
1727 noa set nocindent " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001728 let g:options = [['cindent', '0', '0', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001729 set cindent
1730 call assert_equal([], g:options)
1731 call assert_equal(g:opt[0], g:opt[1])
1732
1733
1734 " 31: Setting boolean global-local (to window) option
1735 " Currently no such option exists.
1736
1737
1738 " 32a: Setting global boolean local (to window) option"
1739 noa setglobal nocursorcolumn " Reset global and local value (without triggering autocmd)
1740 noa setlocal cursorcolumn
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001741 let g:options = [['cursorcolumn', '0', '', '0', '1', 'global', 'setglobal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001742 setglobal cursorcolumn
1743 call assert_equal([], g:options)
1744 call assert_equal(g:opt[0], g:opt[1])
1745
1746 " 32b: Setting local boolean local (to window) option"
1747 noa setglobal nocursorcolumn " Reset global and local value (without triggering autocmd)
1748 noa setlocal cursorcolumn
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001749 let g:options = [['cursorcolumn', '1', '1', '', '0', 'local', 'setlocal']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001750 setlocal nocursorcolumn
1751 call assert_equal([], g:options)
1752 call assert_equal(g:opt[0], g:opt[1])
1753
1754 " 32c: Setting again boolean local (to window) option"
1755 noa setglobal nocursorcolumn " Reset global and local value (without triggering autocmd)
1756 noa setlocal cursorcolumn
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001757 let g:options = [['cursorcolumn', '1', '1', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001758 set cursorcolumn
1759 call assert_equal([], g:options)
1760 call assert_equal(g:opt[0], g:opt[1])
1761
1762 " 32d: Setting again global boolean local (to window) option"
1763 noa set nocursorcolumn " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001764 let g:options = [['cursorcolumn', '0', '0', '0', '1', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001765 set cursorcolumn
1766 call assert_equal([], g:options)
1767 call assert_equal(g:opt[0], g:opt[1])
1768
1769
Bram Moolenaar1bc353b2019-09-01 14:45:28 +02001770 " 33: Test autocommands when an option value is converted internally.
Bram Moolenaard7c96872019-06-15 17:12:48 +02001771 noa set backspace=1 " Reset global and local value (without triggering autocmd)
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001772 let g:options = [['backspace', 'indent,eol', 'indent,eol', 'indent,eol', '2', 'global', 'set']]
Bram Moolenaard7c96872019-06-15 17:12:48 +02001773 set backspace=2
1774 call assert_equal([], g:options)
1775 call assert_equal(g:opt[0], g:opt[1])
1776
1777
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001778 " Cleanup
1779 au! OptionSet
Bram Moolenaar0331faf2019-06-15 18:40:37 +02001780 " set tags&
Bram Moolenaard7c96872019-06-15 17:12:48 +02001781 for opt in ['nu', 'ai', 'acd', 'ar', 'bs', 'backup', 'cul', 'cp', 'backupext', 'tags', 'spelllang', 'statusline', 'foldignore', 'cmdheight', 'undolevels', 'wrapmargin', 'foldcolumn', 'wrapscan', 'autoread', 'cindent', 'cursorcolumn']
Bram Moolenaar91d2e782018-08-07 19:05:01 +02001782 exe printf(":set %s&vim", opt)
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001783 endfor
1784 call test_override('starting', 0)
1785 delfunc! AutoCommandOptionSet
1786endfunc
1787
1788func Test_OptionSet_diffmode()
1789 call test_override('starting', 1)
Bram Moolenaar26d98212019-01-27 22:32:55 +01001790 " 18: Changing an option when entering diff mode
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001791 new
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01001792 au OptionSet diff :let &l:cul = v:option_new
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001793
1794 call setline(1, ['buffer 1', 'line2', 'line3', 'line4'])
1795 call assert_equal(0, &l:cul)
1796 diffthis
1797 call assert_equal(1, &l:cul)
1798
1799 vnew
1800 call setline(1, ['buffer 2', 'line 2', 'line 3', 'line4'])
1801 call assert_equal(0, &l:cul)
1802 diffthis
1803 call assert_equal(1, &l:cul)
1804
1805 diffoff
1806 call assert_equal(0, &l:cul)
1807 call assert_equal(1, getwinvar(2, '&l:cul'))
1808 bw!
1809
1810 call assert_equal(1, &l:cul)
1811 diffoff!
1812 call assert_equal(0, &l:cul)
1813 call assert_equal(0, getwinvar(1, '&l:cul'))
1814 bw!
1815
1816 " Cleanup
1817 au! OptionSet
1818 call test_override('starting', 0)
1819endfunc
1820
1821func Test_OptionSet_diffmode_close()
1822 call test_override('starting', 1)
1823 " 19: Try to close the current window when entering diff mode
1824 " should not segfault
1825 new
1826 au OptionSet diff close
1827
1828 call setline(1, ['buffer 1', 'line2', 'line3', 'line4'])
Bram Moolenaare2e40752020-09-04 21:18:46 +02001829 call assert_fails(':diffthis', 'E788:')
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001830 call assert_equal(1, &diff)
1831 vnew
1832 call setline(1, ['buffer 2', 'line 2', 'line 3', 'line4'])
Bram Moolenaare2e40752020-09-04 21:18:46 +02001833 call assert_fails(':diffthis', 'E788:')
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001834 call assert_equal(1, &diff)
Bram Moolenaara9aa86f2019-11-10 21:25:45 +01001835 set diffopt-=closeoff
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001836 bw!
Bram Moolenaare2e40752020-09-04 21:18:46 +02001837 call assert_fails(':diffoff!', 'E788:')
Bram Moolenaar04f62f82017-07-19 18:18:39 +02001838 bw!
1839
1840 " Cleanup
1841 au! OptionSet
1842 call test_override('starting', 0)
1843 "delfunc! AutoCommandOptionSet
1844endfunc
Bram Moolenaar4a137b42017-08-04 22:37:11 +02001845
1846" Test for Bufleave autocommand that deletes the buffer we are about to edit.
1847func Test_BufleaveWithDelete()
Bram Moolenaare7cda972022-08-29 11:02:59 +01001848 new | edit XbufLeave1
Bram Moolenaar4a137b42017-08-04 22:37:11 +02001849
1850 augroup test_bufleavewithdelete
1851 autocmd!
Bram Moolenaare7cda972022-08-29 11:02:59 +01001852 autocmd BufLeave XbufLeave1 bwipe XbufLeave2
Bram Moolenaar4a137b42017-08-04 22:37:11 +02001853 augroup END
1854
Bram Moolenaare7cda972022-08-29 11:02:59 +01001855 call assert_fails('edit XbufLeave2', 'E143:')
1856 call assert_equal('XbufLeave1', bufname('%'))
Bram Moolenaar4a137b42017-08-04 22:37:11 +02001857
Bram Moolenaare7cda972022-08-29 11:02:59 +01001858 autocmd! test_bufleavewithdelete BufLeave XbufLeave1
Bram Moolenaar4a137b42017-08-04 22:37:11 +02001859 augroup! test_bufleavewithdelete
1860
1861 new
Bram Moolenaare7cda972022-08-29 11:02:59 +01001862 bwipe! XbufLeave1
Bram Moolenaar4a137b42017-08-04 22:37:11 +02001863endfunc
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001864
1865" Test for autocommand that changes the buffer list, when doing ":ball".
1866func Test_Acmd_BufAll()
1867 enew!
1868 %bwipe!
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001869 call writefile(['Test file Xxx1'], 'Xxx1', 'D')
1870 call writefile(['Test file Xxx2'], 'Xxx2', 'D')
1871 call writefile(['Test file Xxx3'], 'Xxx3', 'D')
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001872
1873 " Add three files to the buffer list
1874 split Xxx1
1875 close
1876 split Xxx2
1877 close
1878 split Xxx3
1879 close
1880
1881 " Wipe the buffer when the buffer is opened
1882 au BufReadPost Xxx2 bwipe
1883
1884 call append(0, 'Test file Xxx4')
1885 ball
1886
1887 call assert_equal(2, winnr('$'))
1888 call assert_equal('Xxx1', bufname(winbufnr(winnr('$'))))
1889 wincmd t
1890
1891 au! BufReadPost
1892 %bwipe!
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001893 enew! | only
1894endfunc
1895
1896" Test for autocommand that changes current buffer on BufEnter event.
1897" Check if modelines are interpreted for the correct buffer.
1898func Test_Acmd_BufEnter()
1899 %bwipe!
1900 call writefile(['start of test file Xxx1',
1901 \ "\<Tab>this is a test",
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001902 \ 'end of test file Xxx1'], 'Xxx1', 'D')
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001903 call writefile(['start of test file Xxx2',
1904 \ 'vim: set noai :',
1905 \ "\<Tab>this is a test",
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001906 \ 'end of test file Xxx2'], 'Xxx2', 'D')
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001907
1908 au BufEnter Xxx2 brew
1909 set ai modeline modelines=3
1910 edit Xxx1
1911 " edit Xxx2, autocmd will do :brew
1912 edit Xxx2
1913 exe "normal G?this is a\<CR>"
1914 " Append text with autoindent to this file
1915 normal othis should be auto-indented
1916 call assert_equal("\<Tab>this should be auto-indented", getline('.'))
1917 call assert_equal(3, line('.'))
1918 " Remove autocmd and edit Xxx2 again
1919 au! BufEnter Xxx2
1920 buf! Xxx2
1921 exe "normal G?this is a\<CR>"
1922 " append text without autoindent to Xxx
1923 normal othis should be in column 1
1924 call assert_equal("this should be in column 1", getline('.'))
1925 call assert_equal(4, line('.'))
1926
1927 %bwipe!
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001928 set ai&vim modeline&vim modelines&vim
1929endfunc
1930
1931" Test for issue #57
1932" do not move cursor on <c-o> when autoindent is set
1933func Test_ai_CTRL_O()
1934 enew!
1935 set ai
1936 let save_fo = &fo
1937 set fo+=r
1938 exe "normal o# abcdef\<Esc>2hi\<CR>\<C-O>d0\<Esc>"
1939 exe "normal o# abcdef\<Esc>2hi\<C-O>d0\<Esc>"
1940 call assert_equal(['# abc', 'def', 'def'], getline(2, 4))
1941
1942 set ai&vim
1943 let &fo = save_fo
1944 enew!
1945endfunc
1946
1947" Test for autocommand that deletes the current buffer on BufLeave event.
1948" Also test deleting the last buffer, should give a new, empty buffer.
1949func Test_BufLeave_Wipe()
1950 %bwipe!
1951 let content = ['start of test file Xxx',
1952 \ 'this is a test',
1953 \ 'end of test file Xxx']
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01001954 call writefile(content, 'Xxx1', 'D')
1955 call writefile(content, 'Xxx2', 'D')
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001956
1957 au BufLeave Xxx2 bwipe
1958 edit Xxx1
1959 split Xxx2
1960 " delete buffer Xxx2, we should be back to Xxx1
1961 bwipe
1962 call assert_equal('Xxx1', bufname('%'))
1963 call assert_equal(1, winnr('$'))
1964
1965 " Create an alternate buffer
1966 %write! test.out
1967 call assert_equal('test.out', bufname('#'))
1968 " delete alternate buffer
1969 bwipe test.out
1970 call assert_equal('Xxx1', bufname('%'))
1971 call assert_equal('', bufname('#'))
1972
1973 au BufLeave Xxx1 bwipe
1974 " delete current buffer, get an empty one
1975 bwipe!
1976 call assert_equal(1, line('$'))
1977 call assert_equal('', bufname('%'))
Bram Moolenaarb2c87502017-10-14 21:15:58 +02001978 let g:bufinfo = getbufinfo()
1979 call assert_equal(1, len(g:bufinfo))
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001980
Bram Moolenaar53f0c962017-10-22 14:23:59 +02001981 call delete('test.out')
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001982 %bwipe
1983 au! BufLeave
Bram Moolenaarb2c87502017-10-14 21:15:58 +02001984
1985 " check that bufinfo doesn't contain a pointer to freed memory
1986 call test_garbagecollect_now()
Bram Moolenaar4a6fcf82017-10-12 21:29:22 +02001987endfunc
Bram Moolenaar87ffb5c2017-10-19 12:37:42 +02001988
1989func Test_QuitPre()
1990 edit Xfoo
1991 let winid = win_getid(winnr())
1992 split Xbar
1993 au! QuitPre * let g:afile = expand('<afile>')
1994 " Close the other window, <afile> should be correct.
1995 exe win_id2win(winid) . 'q'
1996 call assert_equal('Xfoo', g:afile)
LemonBoy66e13ae2022-04-21 22:52:11 +01001997
Bram Moolenaar87ffb5c2017-10-19 12:37:42 +02001998 unlet g:afile
1999 bwipe Xfoo
2000 bwipe Xbar
2001endfunc
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002002
Girish Palya92f68e22025-04-21 11:12:41 +02002003func Test_Cmdline_Trigger()
2004 autocmd CmdlineLeavePre : let g:log = "CmdlineLeavePre"
2005 new
2006 let g:log = ''
2007 nnoremap <F1> <Cmd>echo "hello"<CR>
2008 call feedkeys("\<F1>", 'x')
2009 call assert_equal('', g:log)
2010 nunmap <F1>
2011 let g:log = ''
2012 nnoremap <F1> :echo "hello"<CR>
2013 call feedkeys("\<F1>", 'x')
2014 call assert_equal('CmdlineLeavePre', g:log)
2015 nunmap <F1>
2016 let g:log = ''
2017 split
2018 call assert_equal('', g:log)
2019 call feedkeys(":echo hello", "tx")
2020 call assert_equal('CmdlineLeavePre', g:log)
2021 let g:log = ''
2022 close
2023 call assert_equal('', g:log)
2024 call feedkeys(":echo hello", "tx")
2025 call assert_equal('CmdlineLeavePre', g:log)
2026 let g:log = ''
2027 tabnew
2028 call assert_equal('', g:log)
2029 call feedkeys(":echo hello", "tx")
2030 call assert_equal('CmdlineLeavePre', g:log)
2031 let g:log = ''
2032 split
2033 call assert_equal('', g:log)
2034 call feedkeys(":echo hello", "tx")
2035 call assert_equal('CmdlineLeavePre', g:log)
2036 let g:log = ''
2037 tabclose
2038 call assert_equal('', g:log)
2039 call feedkeys(":echo hello", "tx")
2040 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya5c3d1e32025-04-22 19:52:16 +02002041 let g:count = 0
2042 autocmd CmdlineLeavePre * let g:count += 1
2043 call feedkeys(":let c = input('? ')\<cr>B\<cr>", "tx")
2044 call assert_equal(2, g:count)
2045 unlet! g:count
2046 unlet! g:log
Girish Palya92f68e22025-04-21 11:12:41 +02002047 bw!
2048endfunc
2049
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002050func Test_Cmdline()
Bram Moolenaar153b7042018-01-31 15:48:32 +01002051 au! CmdlineChanged : let g:text = getcmdline()
2052 let g:text = 0
2053 call feedkeys(":echom 'hello'\<CR>", 'xt')
2054 call assert_equal("echom 'hello'", g:text)
2055 au! CmdlineChanged
2056
2057 au! CmdlineChanged : let g:entered = expand('<afile>')
2058 let g:entered = 0
2059 call feedkeys(":echom 'hello'\<CR>", 'xt')
2060 call assert_equal(':', g:entered)
2061 au! CmdlineChanged
2062
zeertzjq412e0e42023-02-11 10:34:07 +00002063 autocmd CmdlineChanged : let g:log += [getcmdline()]
2064
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002065 let g:log = []
2066 cnoremap <F1> <Cmd>call setcmdline('ls')<CR>
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002067 call feedkeys(":\<F1>", 'xt')
2068 call assert_equal(['ls'], g:log)
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002069 cunmap <F1>
2070
zeertzjqaf9e28a2023-02-06 20:58:09 +00002071 let g:log = []
zeertzjqaf9e28a2023-02-06 20:58:09 +00002072 call feedkeys(":sign \<Tab>\<Tab>\<C-N>\<C-P>\<S-Tab>\<S-Tab>\<Esc>", 'xt')
2073 call assert_equal([
2074 \ 's',
2075 \ 'si',
2076 \ 'sig',
2077 \ 'sign',
2078 \ 'sign ',
2079 \ 'sign define',
2080 \ 'sign jump',
2081 \ 'sign list',
2082 \ 'sign jump',
2083 \ 'sign define',
2084 \ 'sign ',
2085 \ ], g:log)
2086 let g:log = []
2087 set wildmenu wildoptions+=pum
2088 call feedkeys(":sign \<S-Tab>\<PageUp>\<kPageUp>\<kPageDown>\<PageDown>\<Esc>", 'xt')
2089 call assert_equal([
2090 \ 's',
2091 \ 'si',
2092 \ 'sig',
2093 \ 'sign',
2094 \ 'sign ',
2095 \ 'sign unplace',
2096 \ 'sign jump',
2097 \ 'sign define',
2098 \ 'sign undefine',
2099 \ 'sign unplace',
2100 \ ], g:log)
2101 set wildmenu& wildoptions&
zeertzjq412e0e42023-02-11 10:34:07 +00002102
2103 let g:log = []
2104 let @r = 'abc'
2105 call feedkeys(":0\<C-R>r1\<C-R>\<C-O>r2\<C-R>\<C-R>r3\<Esc>", 'xt')
2106 call assert_equal([
2107 \ '0',
2108 \ '0a',
2109 \ '0ab',
2110 \ '0abc',
2111 \ '0abc1',
2112 \ '0abc1abc',
2113 \ '0abc1abc2',
2114 \ '0abc1abc2abc',
2115 \ '0abc1abc2abc3',
2116 \ ], g:log)
2117
zeertzjqaf9e28a2023-02-06 20:58:09 +00002118 unlet g:log
2119 au! CmdlineChanged
2120
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002121 au! CmdlineEnter : let g:entered = expand('<afile>')
2122 au! CmdlineLeave : let g:left = expand('<afile>')
Girish Palya92f68e22025-04-21 11:12:41 +02002123 au! CmdlineLeavePre : let g:leftpre = expand('<afile>')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002124 let g:entered = 0
2125 let g:left = 0
Girish Palya92f68e22025-04-21 11:12:41 +02002126 let g:leftpre = 0
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002127 call feedkeys(":echo 'hello'\<CR>", 'xt')
2128 call assert_equal(':', g:entered)
2129 call assert_equal(':', g:left)
Girish Palya92f68e22025-04-21 11:12:41 +02002130 call assert_equal(':', g:leftpre)
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002131 au! CmdlineEnter
2132 au! CmdlineLeave
Girish Palya92f68e22025-04-21 11:12:41 +02002133 au! CmdlineLeavePre
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002134
Bram Moolenaara4baf5b2018-04-22 13:27:44 +02002135 let save_shellslash = &shellslash
2136 set noshellslash
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002137 au! CmdlineEnter / let g:entered = expand('<afile>')
2138 au! CmdlineLeave / let g:left = expand('<afile>')
Girish Palya92f68e22025-04-21 11:12:41 +02002139 au! CmdlineLeavePre / let g:leftpre = expand('<afile>')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002140 let g:entered = 0
2141 let g:left = 0
Girish Palya92f68e22025-04-21 11:12:41 +02002142 let g:leftpre = 0
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002143 new
2144 call setline(1, 'hello')
2145 call feedkeys("/hello\<CR>", 'xt')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002146 call assert_equal('/', g:entered)
2147 call assert_equal('/', g:left)
Girish Palya92f68e22025-04-21 11:12:41 +02002148 call assert_equal('/', g:leftpre)
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002149 bwipe!
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002150 au! CmdlineEnter
2151 au! CmdlineLeave
Girish Palya92f68e22025-04-21 11:12:41 +02002152 au! CmdlineLeavePre
Bram Moolenaara4baf5b2018-04-22 13:27:44 +02002153 let &shellslash = save_shellslash
Shougo Matsushitad0952142024-06-20 22:05:16 +02002154
Girish Palya92f68e22025-04-21 11:12:41 +02002155 let g:left = "cancelled"
2156 let g:leftpre = "cancelled"
2157 au! CmdlineLeave : let g:left = "triggered"
2158 au! CmdlineLeavePre : let g:leftpre = "triggered"
2159 call feedkeys(":echo 'hello'\<esc>", 'xt')
2160 call assert_equal('triggered', g:left)
2161 call assert_equal('triggered', g:leftpre)
2162 let g:left = "cancelled"
2163 let g:leftpre = "cancelled"
2164 au! CmdlineLeave : let g:left = "triggered"
2165 call feedkeys(":echo 'hello'\<c-c>", 'xt')
2166 call assert_equal('triggered', g:left)
2167 call assert_equal('triggered', g:leftpre)
2168 au! CmdlineLeave
2169 au! CmdlineLeavePre
2170
zeertzjqbc6f9672024-06-21 07:51:40 +02002171 au! CursorMovedC : let g:pos += [getcmdpos()]
2172 let g:pos = []
zeertzjq81456202024-07-07 20:48:25 +02002173 call feedkeys(":foo bar baz\<C-W>\<C-W>\<C-W>\<Esc>", 'xt')
2174 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 9, 5, 1], g:pos)
2175 let g:pos = []
2176 call feedkeys(":hello\<C-B>\<Esc>", 'xt')
2177 call assert_equal([2, 3, 4, 5, 6, 1], g:pos)
2178 let g:pos = []
2179 call feedkeys(":hello\<C-U>\<Esc>", 'xt')
2180 call assert_equal([2, 3, 4, 5, 6, 1], g:pos)
2181 let g:pos = []
zeertzjqbc6f9672024-06-21 07:51:40 +02002182 call feedkeys(":hello\<Left>\<C-R>=''\<CR>\<Left>\<Right>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002183 call assert_equal([2, 3, 4, 5, 6, 5, 4, 5], g:pos)
zeertzjqbc6f9672024-06-21 07:51:40 +02002184 let g:pos = []
2185 call feedkeys(":12345678\<C-R>=setcmdpos(3)??''\<CR>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002186 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 3], g:pos)
zeertzjqbc6f9672024-06-21 07:51:40 +02002187 let g:pos = []
2188 call feedkeys(":12345678\<C-R>=setcmdpos(3)??''\<CR>\<Left>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002189 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 3, 2], g:pos)
Shougo Matsushitad0952142024-06-20 22:05:16 +02002190 au! CursorMovedC
2191
zeertzjqbc6f9672024-06-21 07:51:40 +02002192 " setcmdpos() is no-op inside an autocommand
2193 au! CursorMovedC : let g:pos += [getcmdpos()] | call setcmdpos(1)
2194 let g:pos = []
2195 call feedkeys(":hello\<Left>\<Left>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002196 call assert_equal([2, 3, 4, 5, 6, 5, 4], g:pos)
Shougo Matsushitad0952142024-06-20 22:05:16 +02002197 au! CursorMovedC
zeertzjqbc6f9672024-06-21 07:51:40 +02002198
2199 unlet g:entered
2200 unlet g:left
2201 unlet g:pos
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002202endfunc
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002203
2204" Test for BufWritePre autocommand that deletes or unloads the buffer.
2205func Test_BufWritePre()
2206 %bwipe
2207 au BufWritePre Xxx1 bunload
2208 au BufWritePre Xxx2 bwipe
2209
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002210 call writefile(['start of Xxx1', 'test', 'end of Xxx1'], 'Xxx1', 'D')
2211 call writefile(['start of Xxx2', 'test', 'end of Xxx2'], 'Xxx2', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002212
2213 edit Xtest
2214 e! Xxx2
2215 bdel Xtest
2216 e Xxx1
2217 " write it, will unload it and give an error msg
Bram Moolenaare2e40752020-09-04 21:18:46 +02002218 call assert_fails('w', 'E203:')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002219 call assert_equal('Xxx2', bufname('%'))
2220 edit Xtest
2221 e! Xxx2
2222 bwipe Xtest
2223 " write it, will delete the buffer and give an error msg
Bram Moolenaare2e40752020-09-04 21:18:46 +02002224 call assert_fails('w', 'E203:')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002225 call assert_equal('Xxx1', bufname('%'))
2226 au! BufWritePre
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002227endfunc
2228
2229" Test for BufUnload autocommand that unloads all the other buffers
2230func Test_bufunload_all()
Bram Moolenaarf08b0eb2021-10-16 13:00:14 +01002231 let g:test_is_flaky = 1
Christian Brabandtee17b6f2023-09-09 11:23:50 +02002232 call writefile(['Test file Xxx1'], 'Xxx1', 'D')
2233 call writefile(['Test file Xxx2'], 'Xxx2', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002234
Bram Moolenaarc79745a2019-05-20 22:12:34 +02002235 let content =<< trim [CODE]
2236 func UnloadAllBufs()
2237 let i = 1
2238 while i <= bufnr('$')
2239 if i != bufnr('%') && bufloaded(i)
2240 exe i . 'bunload'
2241 endif
2242 let i += 1
2243 endwhile
2244 endfunc
2245 au BufUnload * call UnloadAllBufs()
2246 au VimLeave * call writefile(['Test Finished'], 'Xout')
2247 edit Xxx1
2248 split Xxx2
2249 q
2250 [CODE]
2251
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002252 call writefile(content, 'Xbunloadtest', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002253
2254 call delete('Xout')
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002255 call system(GetVimCommandClean() .. ' -N --not-a-term -S Xbunloadtest')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002256 call assert_true(filereadable('Xout'))
2257
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002258 call delete('Xout')
2259endfunc
2260
2261" Some tests for buffer-local autocommands
2262func Test_buflocal_autocmd()
2263 let g:bname = ''
2264 edit xx
2265 au BufLeave <buffer> let g:bname = expand("%")
2266 " here, autocommand for xx should trigger.
2267 " but autocommand shall not apply to buffer named <buffer>.
2268 edit somefile
2269 call assert_equal('xx', g:bname)
2270 let g:bname = ''
2271 " here, autocommand shall be auto-deleted
2272 bwipe xx
2273 " autocmd should not trigger
2274 edit xx
2275 call assert_equal('', g:bname)
2276 " autocmd should not trigger
2277 edit somefile
2278 call assert_equal('', g:bname)
2279 enew
2280 unlet g:bname
2281endfunc
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002282
2283" Test for "*Cmd" autocommands
2284func Test_Cmd_Autocmds()
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002285 call writefile(['start of Xxx', "\tabc2", 'end of Xxx'], 'Xxx', 'D')
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002286
2287 enew!
2288 au BufReadCmd XtestA 0r Xxx|$del
2289 edit XtestA " will read text of Xxd instead
2290 call assert_equal('start of Xxx', getline(1))
2291
2292 au BufWriteCmd XtestA call append(line("$"), "write")
2293 write " will append a line to the file
2294 call assert_equal('write', getline('$'))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002295 call assert_fails('read XtestA', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002296 call assert_equal('write', getline(4))
2297
2298 " now we have:
2299 " 1 start of Xxx
2300 " 2 abc2
2301 " 3 end of Xxx
2302 " 4 write
2303
2304 au FileReadCmd XtestB '[r Xxx
2305 2r XtestB " will read Xxx below line 2 instead
2306 call assert_equal('start of Xxx', getline(3))
2307
2308 " now we have:
2309 " 1 start of Xxx
2310 " 2 abc2
2311 " 3 start of Xxx
2312 " 4 abc2
2313 " 5 end of Xxx
2314 " 6 end of Xxx
2315 " 7 write
2316
2317 au FileWriteCmd XtestC '[,']copy $
2318 normal 4GA1
2319 4,5w XtestC " will copy lines 4 and 5 to the end
2320 call assert_equal("\tabc21", getline(8))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002321 call assert_fails('r XtestC', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002322 call assert_equal("end of Xxx", getline(9))
2323
2324 " now we have:
2325 " 1 start of Xxx
2326 " 2 abc2
2327 " 3 start of Xxx
2328 " 4 abc21
2329 " 5 end of Xxx
2330 " 6 end of Xxx
2331 " 7 write
2332 " 8 abc21
2333 " 9 end of Xxx
2334
2335 let g:lines = []
2336 au FileAppendCmd XtestD call extend(g:lines, getline(line("'["), line("']")))
2337 w >>XtestD " will add lines to 'lines'
2338 call assert_equal(9, len(g:lines))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002339 call assert_fails('$r XtestD', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002340 call assert_equal(9, line('$'))
2341 call assert_equal('end of Xxx', getline('$'))
2342
2343 au BufReadCmd XtestE 0r Xxx|$del
2344 sp XtestE " split window with test.out
2345 call assert_equal('end of Xxx', getline(3))
2346
2347 let g:lines = []
2348 exe "normal 2Goasdf\<Esc>\<C-W>\<C-W>"
2349 au BufWriteCmd XtestE call extend(g:lines, getline(0, '$'))
2350 wall " will write other window to 'lines'
2351 call assert_equal(4, len(g:lines), g:lines)
2352 call assert_equal('asdf', g:lines[2])
2353
2354 au! BufReadCmd
2355 au! BufWriteCmd
2356 au! FileReadCmd
2357 au! FileWriteCmd
2358 au! FileAppendCmd
2359 %bwipe!
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002360 enew!
2361endfunc
Bram Moolenaaraace2152017-11-05 16:23:10 +01002362
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002363func s:ReadFile()
2364 setl noswapfile nomodified
2365 let filename = resolve(expand("<afile>:p"))
2366 execute 'read' fnameescape(filename)
2367 1d_
2368 exe 'file' fnameescape(filename)
2369 setl buftype=acwrite
2370endfunc
2371
2372func s:WriteFile()
2373 let filename = resolve(expand("<afile>:p"))
2374 setl buftype=
2375 noautocmd execute 'write' fnameescape(filename)
2376 setl buftype=acwrite
2377 setl nomodified
2378endfunc
2379
2380func Test_BufReadCmd()
2381 autocmd BufReadCmd *.test call s:ReadFile()
2382 autocmd BufWriteCmd *.test call s:WriteFile()
2383
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002384 call writefile(['one', 'two', 'three'], 'Xcmd.test', 'D')
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002385 edit Xcmd.test
2386 call assert_match('Xcmd.test" line 1 of 3', execute('file'))
2387 normal! Gofour
2388 write
2389 call assert_equal(['one', 'two', 'three', 'four'], readfile('Xcmd.test'))
2390
2391 bwipe!
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002392 au! BufReadCmd
2393 au! BufWriteCmd
2394endfunc
2395
zeertzjq9c8f9462022-08-30 18:17:15 +01002396func Test_BufWriteCmd()
2397 autocmd BufWriteCmd Xbufwritecmd let g:written = 1
2398 new
2399 file Xbufwritecmd
2400 set buftype=acwrite
Bram Moolenaar6f14da12022-09-07 21:30:44 +01002401 call mkdir('Xbufwritecmd', 'D')
zeertzjq9c8f9462022-08-30 18:17:15 +01002402 write
2403 " BufWriteCmd should be triggered even if a directory has the same name
2404 call assert_equal(1, g:written)
zeertzjq9c8f9462022-08-30 18:17:15 +01002405 unlet g:written
2406 au! BufWriteCmd
2407 bwipe!
2408endfunc
2409
Bram Moolenaaraace2152017-11-05 16:23:10 +01002410func SetChangeMarks(start, end)
Bram Moolenaar97c69432021-01-15 16:45:21 +01002411 exe a:start .. 'mark ['
2412 exe a:end .. 'mark ]'
Bram Moolenaaraace2152017-11-05 16:23:10 +01002413endfunc
2414
2415" Verify the effects of autocmds on '[ and ']
2416func Test_change_mark_in_autocmds()
2417 edit! Xtest
Bram Moolenaar97c69432021-01-15 16:45:21 +01002418 call feedkeys("ia\<CR>b\<CR>c\<CR>d\<C-g>u\<Esc>", 'xtn')
Bram Moolenaaraace2152017-11-05 16:23:10 +01002419
2420 call SetChangeMarks(2, 3)
2421 write
2422 call assert_equal([1, 4], [line("'["), line("']")])
2423
2424 call SetChangeMarks(2, 3)
2425 au BufWritePre * call assert_equal([1, 4], [line("'["), line("']")])
2426 write
2427 au! BufWritePre
2428
Bram Moolenaar14ddd222020-08-05 12:02:40 +02002429 if has('unix')
Bram Moolenaaraace2152017-11-05 16:23:10 +01002430 write XtestFilter
2431 write >> XtestFilter
2432
2433 call SetChangeMarks(2, 3)
2434 " Marks are set to the entire range of the write
2435 au FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
2436 " '[ is adjusted to just before the line that will receive the filtered
2437 " data
2438 au FilterReadPre * call assert_equal([4, 4], [line("'["), line("']")])
2439 " The filtered data is read into the buffer, and the source lines are
2440 " still present, so the range is after the source lines
2441 au FilterReadPost * call assert_equal([5, 12], [line("'["), line("']")])
2442 %!cat XtestFilter
2443 " After the filtered data is read, the original lines are deleted
2444 call assert_equal([1, 8], [line("'["), line("']")])
2445 au! FilterWritePre,FilterReadPre,FilterReadPost
2446 undo
2447
2448 call SetChangeMarks(1, 4)
2449 au FilterWritePre * call assert_equal([2, 3], [line("'["), line("']")])
2450 au FilterReadPre * call assert_equal([3, 3], [line("'["), line("']")])
2451 au FilterReadPost * call assert_equal([4, 11], [line("'["), line("']")])
2452 2,3!cat XtestFilter
2453 call assert_equal([2, 9], [line("'["), line("']")])
2454 au! FilterWritePre,FilterReadPre,FilterReadPost
2455 undo
2456
2457 call delete('XtestFilter')
2458 endif
2459
2460 call SetChangeMarks(1, 4)
2461 au FileWritePre * call assert_equal([2, 3], [line("'["), line("']")])
2462 2,3write Xtest2
2463 au! FileWritePre
2464
2465 call SetChangeMarks(2, 3)
2466 au FileAppendPre * call assert_equal([1, 4], [line("'["), line("']")])
2467 write >> Xtest2
2468 au! FileAppendPre
2469
2470 call SetChangeMarks(1, 4)
2471 au FileAppendPre * call assert_equal([2, 3], [line("'["), line("']")])
2472 2,3write >> Xtest2
2473 au! FileAppendPre
2474
2475 call SetChangeMarks(1, 1)
2476 au FileReadPre * call assert_equal([3, 1], [line("'["), line("']")])
2477 au FileReadPost * call assert_equal([4, 11], [line("'["), line("']")])
2478 3read Xtest2
2479 au! FileReadPre,FileReadPost
2480 undo
2481
2482 call SetChangeMarks(4, 4)
2483 " When the line is 0, it's adjusted to 1
2484 au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
2485 au FileReadPost * call assert_equal([1, 8], [line("'["), line("']")])
2486 0read Xtest2
2487 au! FileReadPre,FileReadPost
2488 undo
2489
2490 call SetChangeMarks(4, 4)
2491 " When the line is 0, it's adjusted to 1
2492 au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
2493 au FileReadPost * call assert_equal([2, 9], [line("'["), line("']")])
2494 1read Xtest2
2495 au! FileReadPre,FileReadPost
2496 undo
2497
2498 bwipe!
2499 call delete('Xtest')
2500 call delete('Xtest2')
2501endfunc
2502
2503func Test_Filter_noshelltemp()
Bram Moolenaaraeb313f2020-11-27 19:13:28 +01002504 CheckExecutable cat
Bram Moolenaaraace2152017-11-05 16:23:10 +01002505
2506 enew!
2507 call setline(1, ['a', 'b', 'c', 'd'])
2508
2509 let shelltemp = &shelltemp
2510 set shelltemp
2511
2512 let g:filter_au = 0
2513 au FilterWritePre * let g:filter_au += 1
2514 au FilterReadPre * let g:filter_au += 1
2515 au FilterReadPost * let g:filter_au += 1
2516 %!cat
2517 call assert_equal(3, g:filter_au)
2518
2519 if has('filterpipe')
2520 set noshelltemp
2521
2522 let g:filter_au = 0
2523 au FilterWritePre * let g:filter_au += 1
2524 au FilterReadPre * let g:filter_au += 1
2525 au FilterReadPost * let g:filter_au += 1
2526 %!cat
2527 call assert_equal(0, g:filter_au)
2528 endif
2529
2530 au! FilterWritePre,FilterReadPre,FilterReadPost
2531 let &shelltemp = shelltemp
2532 bwipe!
2533endfunc
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002534
2535func Test_TextYankPost()
2536 enew!
2537 call setline(1, ['foo'])
2538
2539 let g:event = []
2540 au TextYankPost * let g:event = copy(v:event)
2541
2542 call assert_equal({}, v:event)
2543 call assert_fails('let v:event = {}', 'E46:')
2544 call assert_fails('let v:event.mykey = 0', 'E742:')
2545
2546 norm "ayiw
2547 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002548 \ #{regcontents: ['foo'], regname: 'a', operator: 'y',
2549 \ regtype: 'v', visual: v:false, inclusive: v:true},
2550 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002551 norm y_
2552 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002553 \ #{regcontents: ['foo'], regname: '', operator: 'y', regtype: 'V',
2554 \ visual: v:false, inclusive: v:false},
2555 \ g:event)
Bram Moolenaar37d16732020-06-12 22:09:01 +02002556 norm Vy
2557 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002558 \ #{regcontents: ['foo'], regname: '', operator: 'y', regtype: 'V',
2559 \ visual: v:true, inclusive: v:true},
2560 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002561 call feedkeys("\<C-V>y", 'x')
2562 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002563 \ #{regcontents: ['f'], regname: '', operator: 'y', regtype: "\x161",
2564 \ visual: v:true, inclusive: v:true},
2565 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002566 norm "xciwbar
2567 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002568 \ #{regcontents: ['foo'], regname: 'x', operator: 'c', regtype: 'v',
2569 \ visual: v:false, inclusive: v:true},
2570 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002571 norm "bdiw
2572 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002573 \ #{regcontents: ['bar'], regname: 'b', operator: 'd', regtype: 'v',
2574 \ visual: v:false, inclusive: v:true},
2575 \ g:event)
2576
2577 call setline(1, 'foobar')
2578 " exclusive motion
2579 norm $"ay0
2580 call assert_equal(
2581 \ #{regcontents: ['fooba'], regname: 'a', operator: 'y', regtype: 'v',
2582 \ visual: v:false, inclusive: v:false},
2583 \ g:event)
2584 " inclusive motion
2585 norm 0"ay$
2586 call assert_equal(
2587 \ #{regcontents: ['foobar'], regname: 'a', operator: 'y', regtype: 'v',
2588 \ visual: v:false, inclusive: v:true},
2589 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002590
2591 call assert_equal({}, v:event)
2592
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002593 if has('clipboard_working') && !has('gui_running')
2594 " Test that when the visual selection is automatically copied to clipboard
2595 " register a TextYankPost is emitted
2596 call setline(1, ['foobar'])
2597
2598 let @* = ''
2599 set clipboard=autoselect
2600 exe "norm! ggviw\<Esc>"
2601 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002602 \ #{regcontents: ['foobar'], regname: '*', operator: 'y',
2603 \ regtype: 'v', visual: v:true, inclusive: v:false},
2604 \ g:event)
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002605
2606 let @+ = ''
2607 set clipboard=autoselectplus
2608 exe "norm! ggviw\<Esc>"
2609 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002610 \ #{regcontents: ['foobar'], regname: '+', operator: 'y',
2611 \ regtype: 'v', visual: v:true, inclusive: v:false},
2612 \ g:event)
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002613
2614 set clipboard&vim
2615 endif
2616
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002617 au! TextYankPost
2618 unlet g:event
2619 bwipe!
2620endfunc
Bram Moolenaar9bca8052017-12-18 12:37:55 +01002621
Bram Moolenaar9a046fd2021-01-28 13:47:59 +01002622func Test_autocommand_all_events()
2623 call assert_fails('au * * bwipe', 'E1155:')
2624 call assert_fails('au * x bwipe', 'E1155:')
Bram Moolenaarb6db1462021-12-24 19:24:47 +00002625 call assert_fails('au! * x bwipe', 'E1155:')
Bram Moolenaar4fb921e2017-12-18 15:33:00 +01002626endfunc
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002627
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002628func Test_autocmd_user()
2629 au User MyEvent let s:res = [expand("<afile>"), expand("<amatch>")]
2630 doautocmd User MyEvent
2631 call assert_equal(['MyEvent', 'MyEvent'], s:res)
2632 au! User
2633 unlet s:res
2634endfunc
2635
Bram Moolenaar3b014be2022-11-13 17:53:46 +00002636func Test_autocmd_user_clear_group()
2637 CheckRunVimInTerminal
2638
2639 let lines =<< trim END
2640 autocmd! User
2641 for i in range(1, 999)
2642 exe 'autocmd User ' .. 'Foo' .. i .. ' bar'
2643 endfor
2644 au CmdlineLeave : call timer_start(0, {-> execute('autocmd! User')})
2645 END
2646 call writefile(lines, 'XautoUser', 'D')
2647 let buf = RunVimInTerminal('-S XautoUser', {'rows': 10})
2648
2649 " this was using freed memory
2650 call term_sendkeys(buf, ":autocmd User\<CR>")
2651 call TermWait(buf, 50)
2652 call term_sendkeys(buf, "G")
2653
2654 call StopVimInTerminal(buf)
2655endfunc
2656
Bram Moolenaaref2c3252022-11-25 16:31:51 +00002657func Test_autocmd_CmdlineLeave_unlet()
2658 CheckRunVimInTerminal
2659
2660 let lines =<< trim END
2661 for i in range(1, 999)
2662 exe 'let g:var' .. i '=' i
2663 endfor
2664 au CmdlineLeave : call timer_start(0, {-> execute('unlet g:var990')})
2665 END
2666 call writefile(lines, 'XleaveUnlet', 'D')
2667 let buf = RunVimInTerminal('-S XleaveUnlet', {'rows': 10})
2668
2669 " this was using freed memory
2670 call term_sendkeys(buf, ":let g:\<CR>")
2671 call TermWait(buf, 50)
2672 call term_sendkeys(buf, "G")
2673 call TermWait(buf, 50)
2674 call term_sendkeys(buf, "\<CR>") " for the hit-enter prompt
2675
2676 call StopVimInTerminal(buf)
2677endfunc
2678
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002679function s:Before_test_dirchanged()
2680 augroup test_dirchanged
2681 autocmd!
2682 augroup END
2683 let s:li = []
2684 let s:dir_this = getcwd()
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02002685 let s:dir_foo = s:dir_this . '/Xfoo'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002686 call mkdir(s:dir_foo)
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02002687 let s:dir_bar = s:dir_this . '/Xbar'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002688 call mkdir(s:dir_bar)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002689endfunc
2690
2691function s:After_test_dirchanged()
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002692 call chdir(s:dir_this)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002693 call delete(s:dir_foo, 'd')
2694 call delete(s:dir_bar, 'd')
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002695 augroup test_dirchanged
2696 autocmd!
2697 augroup END
2698endfunc
2699
2700function Test_dirchanged_global()
2701 call s:Before_test_dirchanged()
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002702 autocmd test_dirchanged DirChangedPre global call add(s:li, expand("<amatch>") .. " pre cd " .. v:event.directory)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002703 autocmd test_dirchanged DirChanged global call add(s:li, "cd:")
2704 autocmd test_dirchanged DirChanged global call add(s:li, expand("<afile>"))
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002705 call chdir(s:dir_foo)
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002706 let expected = ["global pre cd " .. s:dir_foo, "cd:", s:dir_foo]
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002707 call assert_equal(expected, s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002708 call chdir(s:dir_foo)
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002709 call assert_equal(expected, s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002710 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002711 call assert_equal(expected, s:li)
Bram Moolenaard8c9d322022-06-12 11:49:16 +01002712
2713 exe 'cd ' .. s:dir_foo
2714 exe 'cd ' .. s:dir_bar
2715 autocmd! test_dirchanged DirChanged global let g:result = expand("<afile>")
2716 cd -
Bram Moolenaardb77c492022-06-12 23:26:50 +01002717 call assert_equal(s:dir_foo, substitute(g:result, '\\', '/', 'g'))
Bram Moolenaard8c9d322022-06-12 11:49:16 +01002718
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002719 call s:After_test_dirchanged()
2720endfunc
2721
2722function Test_dirchanged_local()
2723 call s:Before_test_dirchanged()
2724 autocmd test_dirchanged DirChanged window call add(s:li, "lcd:")
2725 autocmd test_dirchanged DirChanged window call add(s:li, expand("<afile>"))
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002726 call chdir(s:dir_foo)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002727 call assert_equal([], s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002728 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002729 call assert_equal(["lcd:", s:dir_bar], s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002730 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002731 call assert_equal(["lcd:", s:dir_bar], s:li)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002732 call s:After_test_dirchanged()
2733endfunc
2734
2735function Test_dirchanged_auto()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02002736 CheckOption autochdir
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002737 call s:Before_test_dirchanged()
2738 call test_autochdir()
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002739 autocmd test_dirchanged DirChangedPre auto call add(s:li, "pre cd " .. v:event.directory)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002740 autocmd test_dirchanged DirChanged auto call add(s:li, "auto:")
2741 autocmd test_dirchanged DirChanged auto call add(s:li, expand("<afile>"))
2742 set acd
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002743 cd ..
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002744 call assert_equal([], s:li)
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01002745 exe 'edit ' . s:dir_foo . '/Xautofile'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002746 call assert_equal(s:dir_foo, getcwd())
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002747 let expected = ["pre cd " .. s:dir_foo, "auto:", s:dir_foo]
2748 call assert_equal(expected, s:li)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002749 set noacd
2750 bwipe!
2751 call s:After_test_dirchanged()
2752endfunc
Bram Moolenaar5a093432018-02-10 18:15:19 +01002753
2754" Test TextChangedI and TextChangedP
2755func Test_ChangedP()
2756 new
2757 call setline(1, ['foo', 'bar', 'foobar'])
2758 call test_override("char_avail", 1)
2759 set complete=. completeopt=menuone
2760
2761 func! TextChangedAutocmd(char)
2762 let g:autocmd .= a:char
2763 endfunc
2764
Christian Brabandtdb3b4462021-10-16 11:58:55 +01002765 " TextChanged will not be triggered, only check that it isn't.
Bram Moolenaar5a093432018-02-10 18:15:19 +01002766 au! TextChanged <buffer> :call TextChangedAutocmd('N')
2767 au! TextChangedI <buffer> :call TextChangedAutocmd('I')
2768 au! TextChangedP <buffer> :call TextChangedAutocmd('P')
2769
2770 call cursor(3, 1)
2771 let g:autocmd = ''
2772 call feedkeys("o\<esc>", 'tnix')
Evgeni Chasnovskid7ae2632023-10-15 09:59:00 +02002773 call assert_equal('I', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002774
2775 let g:autocmd = ''
Christian Brabandt4bca4892023-10-27 19:26:49 +02002776 call feedkeys("Sf", 'tnix')
2777 call assert_equal('II', g:autocmd)
2778
2779 let g:autocmd = ''
Bram Moolenaar5a093432018-02-10 18:15:19 +01002780 call feedkeys("Sf\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002781 call assert_equal('IIP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002782
2783 let g:autocmd = ''
2784 call feedkeys("Sf\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002785 call assert_equal('IIPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002786
2787 let g:autocmd = ''
2788 call feedkeys("Sf\<C-N>\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002789 call assert_equal('IIPPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002790
2791 let g:autocmd = ''
2792 call feedkeys("Sf\<C-N>\<C-N>\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002793 call assert_equal('IIPPPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002794
2795 call assert_equal(['foo', 'bar', 'foobar', 'foo'], getline(1, '$'))
2796 " TODO: how should it handle completeopt=noinsert,noselect?
2797
2798 " CleanUp
2799 call test_override("char_avail", 0)
2800 au! TextChanged
2801 au! TextChangedI
2802 au! TextChangedP
2803 delfu TextChangedAutocmd
2804 unlet! g:autocmd
2805 set complete&vim completeopt&vim
2806
2807 bw!
2808endfunc
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002809
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002810let g:setline_handled = v:false
Bram Moolenaar1e115362019-01-09 23:01:02 +01002811func SetLineOne()
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002812 if !g:setline_handled
2813 call setline(1, "(x)")
2814 let g:setline_handled = v:true
2815 endif
2816endfunc
2817
2818func Test_TextChangedI_with_setline()
2819 new
2820 call test_override('char_avail', 1)
2821 autocmd TextChangedI <buffer> call SetLineOne()
2822 call feedkeys("i(\<CR>\<Esc>", 'tx')
2823 call assert_equal('(', getline(1))
2824 call assert_equal('x)', getline(2))
2825 undo
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002826 call assert_equal('', getline(1))
Bram Moolenaar9fa95062018-08-08 22:08:32 +02002827 call assert_equal('', getline(2))
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002828
Bram Moolenaarca34db32022-01-20 11:17:18 +00002829 call test_override('char_avail', 0)
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002830 bwipe!
2831endfunc
2832
Christian Brabandtc9e79e52024-02-09 19:34:36 +01002833func Test_TextChanged_with_norm()
2834 " For unknown reason this fails on MS-Windows
2835 CheckNotMSWindows
2836 CheckFeature terminal
2837 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2838 call assert_equal('running', term_getstatus(buf))
2839 call term_sendkeys(buf, ":let g:a=0\<cr>")
2840 call term_wait(buf, 50)
2841 call term_sendkeys(buf, ":au! TextChanged * :let g:a+=1\<cr>")
2842 call term_wait(buf, 50)
2843 call term_sendkeys(buf, ":norm! ia\<cr>")
2844 call term_wait(buf, 50)
2845 call term_sendkeys(buf, ":echo g:a\<cr>")
2846 call term_wait(buf, 50)
2847 call WaitForAssert({-> assert_match('^1.*$', term_getline(buf, 3))})
2848 bwipe!
2849endfunc
2850
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002851func Test_Changed_FirstTime()
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02002852 CheckFeature terminal
2853 CheckNotGui
Bram Moolenaar3cdcb092020-03-18 19:18:10 +01002854 " Starting a terminal to run Vim is always considered flaky.
Bram Moolenaar30d53e22020-03-18 21:10:44 +01002855 let g:test_is_flaky = 1
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02002856
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002857 " Prepare file for TextChanged event.
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002858 call writefile([''], 'Xchanged.txt', 'D')
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002859 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2860 call assert_equal('running', term_getstatus(buf))
Bram Moolenaar1834d372018-03-29 17:40:46 +02002861 " Wait for the ruler (in the status line) to be shown.
Bram Moolenaaraa5df7e2019-02-03 14:53:10 +01002862 " In ConPTY, there is additional character which is drawn up to the width of
2863 " the screen.
2864 if has('conpty')
2865 call WaitForAssert({-> assert_match('\<All.*$', term_getline(buf, 3))})
2866 else
2867 call WaitForAssert({-> assert_match('\<All$', term_getline(buf, 3))})
2868 endif
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002869 " It's only adding autocmd, so that no event occurs.
2870 call term_sendkeys(buf, ":au! TextChanged <buffer> call writefile(['No'], 'Xchanged.txt')\<cr>")
2871 call term_sendkeys(buf, "\<C-\\>\<C-N>:qa!\<cr>")
Bram Moolenaar50182fa2018-04-28 21:34:40 +02002872 call WaitForAssert({-> assert_equal('finished', term_getstatus(buf))})
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002873 call assert_equal([''], readfile('Xchanged.txt'))
2874
2875 " clean up
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002876 bwipe!
2877endfunc
Bram Moolenaar0566e892019-01-24 19:37:40 +01002878
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002879func Test_autocmd_nested()
2880 let g:did_nested = 0
Christian Brabandtfb3f9692024-08-11 20:09:17 +02002881 defer CleanUpTestAuGroup()
2882 augroup testing
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002883 au WinNew * edit somefile
2884 au BufNew * let g:did_nested = 1
2885 augroup END
2886 split
2887 call assert_equal(0, g:did_nested)
2888 close
2889 bwipe! somefile
2890
2891 " old nested argument still works
Christian Brabandtfb3f9692024-08-11 20:09:17 +02002892 augroup testing
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002893 au!
2894 au WinNew * nested edit somefile
2895 au BufNew * let g:did_nested = 1
2896 augroup END
2897 split
2898 call assert_equal(1, g:did_nested)
2899 close
2900 bwipe! somefile
2901
2902 " New ++nested argument works
2903 augroup Testing
2904 au!
2905 au WinNew * ++nested edit somefile
2906 au BufNew * let g:did_nested = 1
2907 augroup END
2908 split
2909 call assert_equal(1, g:did_nested)
2910 close
2911 bwipe! somefile
2912
Bram Moolenaarf0775142022-03-04 20:10:38 +00002913 " nested without ++ does not work in Vim9 script
2914 call assert_fails('vim9cmd au WinNew * nested echo fails', 'E1078:')
2915
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002916 augroup Testing
2917 au!
2918 augroup END
2919
2920 call assert_fails('au WinNew * ++nested ++nested echo bad', 'E983:')
2921 call assert_fails('au WinNew * nested nested echo bad', 'E983:')
2922endfunc
2923
Bram Moolenaar5fa9f232022-07-23 09:06:48 +01002924func Test_autocmd_nested_cursor_invalid()
2925 set laststatus=0
2926 copen
2927 cclose
2928 call setline(1, ['foo', 'bar', 'baz'])
2929 3
2930 augroup nested_inv
2931 autocmd User foo ++nested copen
2932 autocmd BufAdd * let &laststatus = 2 - &laststatus
2933 augroup END
2934 doautocmd User foo
2935
2936 augroup nested_inv
2937 au!
2938 augroup END
2939 set laststatus&
Bram Moolenaarb03950f2022-07-26 13:47:13 +01002940 cclose
Bram Moolenaar5fa9f232022-07-23 09:06:48 +01002941 bwipe!
2942endfunc
2943
Bram Moolenaar3d6ee8b2022-07-27 15:23:35 +01002944func Test_autocmd_nested_keeps_cursor_pos()
2945 enew
2946 call setline(1, 'foo')
2947 autocmd User foo ++nested normal! $a
2948 autocmd InsertLeave * :
2949 doautocmd User foo
2950 call assert_equal([0, 1, 3, 0], getpos('.'))
2951
2952 bwipe!
2953endfunc
2954
Bram Moolenaarb03950f2022-07-26 13:47:13 +01002955func Test_autocmd_nested_switch_window()
2956 " run this in a separate Vim so that SafeState works
2957 CheckRunVimInTerminal
2958
2959 let lines =<< trim END
2960 vim9script
2961 ['()']->writefile('Xautofile')
2962 autocmd VimEnter * ++nested edit Xautofile | split
2963 autocmd BufReadPost * autocmd SafeState * ++once foldclosed('.')
2964 autocmd WinEnter * matchadd('ErrorMsg', 'pat')
2965 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002966 call writefile(lines, 'Xautoscript', 'D')
Bram Moolenaarb03950f2022-07-26 13:47:13 +01002967 let buf = RunVimInTerminal('-S Xautoscript', {'rows': 10})
2968 call VerifyScreenDump(buf, 'Test_autocmd_nested_switch', {})
2969
2970 call StopVimInTerminal(buf)
2971 call delete('Xautofile')
Bram Moolenaarb03950f2022-07-26 13:47:13 +01002972endfunc
2973
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002974func Test_autocmd_once()
2975 " Without ++once WinNew triggers twice
2976 let g:did_split = 0
2977 augroup Testing
2978 au WinNew * let g:did_split += 1
2979 augroup END
2980 split
2981 split
2982 call assert_equal(2, g:did_split)
2983 call assert_true(exists('#WinNew'))
2984 close
2985 close
2986
2987 " With ++once WinNew triggers once
2988 let g:did_split = 0
2989 augroup Testing
2990 au!
2991 au WinNew * ++once let g:did_split += 1
2992 augroup END
2993 split
2994 split
2995 call assert_equal(1, g:did_split)
2996 call assert_false(exists('#WinNew'))
2997 close
2998 close
2999
3000 call assert_fails('au WinNew * ++once ++once echo bad', 'E983:')
3001endfunc
3002
Bram Moolenaara68e5952019-04-25 22:22:01 +02003003func Test_autocmd_bufreadpre()
3004 new
3005 let b:bufreadpre = 1
Bram Moolenaarab505b12020-03-23 19:28:44 +01003006 call append(0, range(1000))
Bram Moolenaara68e5952019-04-25 22:22:01 +02003007 w! XAutocmdBufReadPre.txt
3008 autocmd BufReadPre <buffer> :let b:bufreadpre += 1
Bram Moolenaarab505b12020-03-23 19:28:44 +01003009 norm! 500gg
Bram Moolenaara68e5952019-04-25 22:22:01 +02003010 sp
Bram Moolenaarab505b12020-03-23 19:28:44 +01003011 norm! 1000gg
Bram Moolenaara68e5952019-04-25 22:22:01 +02003012 wincmd p
3013 let g:wsv1 = winsaveview()
3014 wincmd p
3015 let g:wsv2 = winsaveview()
3016 " triggers BufReadPre, should not move the cursor in either window
3017 " The topline may change one line in a large window.
3018 edit
3019 call assert_inrange(g:wsv2.topline - 1, g:wsv2.topline + 1, winsaveview().topline)
3020 call assert_equal(g:wsv2.lnum, winsaveview().lnum)
3021 call assert_equal(2, b:bufreadpre)
3022 wincmd p
3023 call assert_equal(g:wsv1.topline, winsaveview().topline)
3024 call assert_equal(g:wsv1.lnum, winsaveview().lnum)
3025 call assert_equal(2, b:bufreadpre)
3026 " Now set the cursor position in an BufReadPre autocommand
3027 " (even though the position will be invalid, this should make Vim reset the
3028 " cursor position in the other window.
3029 wincmd p
3030 set cpo+=g
3031 " won't do anything, but try to set the cursor on an invalid lnum
3032 autocmd BufReadPre <buffer> :norm! 70gg
3033 " triggers BufReadPre, should not move the cursor in either window
3034 e
3035 call assert_equal(1, winsaveview().topline)
3036 call assert_equal(1, winsaveview().lnum)
3037 call assert_equal(3, b:bufreadpre)
3038 wincmd p
3039 call assert_equal(g:wsv1.topline, winsaveview().topline)
3040 call assert_equal(g:wsv1.lnum, winsaveview().lnum)
3041 call assert_equal(3, b:bufreadpre)
3042 close
3043 close
3044 call delete('XAutocmdBufReadPre.txt')
3045 set cpo-=g
3046endfunc
3047
Bram Moolenaar5e66b422019-01-24 21:58:10 +01003048" FileChangedShell tested in test_filechanged.vim
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003049
3050" Tests for the following autocommands:
3051" - FileWritePre writing a compressed file
3052" - FileReadPost reading a compressed file
3053" - BufNewFile reading a file template
3054" - BufReadPre decompressing the file to be read
3055" - FilterReadPre substituting characters in the temp file
3056" - FilterReadPost substituting characters after filtering
3057" - FileReadPre set options for decompression
3058" - FileReadPost decompress the file
3059func Test_ReadWrite_Autocmds()
3060 " Run this test only on Unix-like systems and if gzip is available
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02003061 CheckUnix
3062 CheckExecutable gzip
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003063
3064 " Make $GZIP empty, "-v" would cause trouble.
3065 let $GZIP = ""
3066
3067 " Use a FileChangedShell autocommand to avoid a prompt for 'Xtestfile.gz'
3068 " being modified outside of Vim (noticed on Solaris).
3069 au FileChangedShell * echo 'caught FileChangedShell'
3070
3071 " Test for the FileReadPost, FileWritePre and FileWritePost autocmds
3072 augroup Test1
3073 au!
3074 au FileWritePre *.gz '[,']!gzip
3075 au FileWritePost *.gz undo
3076 au FileReadPost *.gz '[,']!gzip -d
3077 augroup END
3078
3079 new
3080 set bin
3081 call append(0, [
3082 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3083 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3084 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3085 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3086 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3087 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3088 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3089 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3090 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3091 \ ])
3092 1,9write! Xtestfile.gz
3093 enew! | close
3094
3095 new
3096 " Read and decompress the testfile
3097 0read Xtestfile.gz
3098 call assert_equal([
3099 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3100 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3101 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3102 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3103 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3104 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3105 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3106 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3107 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3108 \ ], getline(1, 9))
3109 enew! | close
3110
3111 augroup Test1
3112 au!
3113 augroup END
3114
3115 " Test for the FileAppendPre and FileAppendPost autocmds
3116 augroup Test2
3117 au!
3118 au BufNewFile *.c read Xtest.c
3119 au FileAppendPre *.out '[,']s/new/NEW/
3120 au FileAppendPost *.out !cat Xtest.c >> test.out
3121 augroup END
3122
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003123 call writefile(['/*', ' * Here is a new .c file', ' */'], 'Xtest.c', 'D')
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003124 new foo.c " should load Xtest.c
3125 call assert_equal(['/*', ' * Here is a new .c file', ' */'], getline(2, 4))
3126 w! >> test.out " append it to the output file
3127
3128 let contents = readfile('test.out')
3129 call assert_equal(' * Here is a NEW .c file', contents[2])
3130 call assert_equal(' * Here is a new .c file', contents[5])
3131
3132 call delete('test.out')
3133 enew! | close
3134 augroup Test2
3135 au!
3136 augroup END
3137
3138 " Test for the BufReadPre and BufReadPost autocmds
3139 augroup Test3
3140 au!
3141 " setup autocommands to decompress before reading and re-compress
3142 " afterwards
3143 au BufReadPre *.gz exe '!gzip -d ' . shellescape(expand("<afile>"))
3144 au BufReadPre *.gz call rename(expand("<afile>:r"), expand("<afile>"))
3145 au BufReadPost *.gz call rename(expand("<afile>"), expand("<afile>:r"))
3146 au BufReadPost *.gz exe '!gzip ' . shellescape(expand("<afile>:r"))
3147 augroup END
3148
3149 e! Xtestfile.gz " Edit compressed file
3150 call assert_equal([
3151 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3152 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3153 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3154 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3155 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3156 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3157 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3158 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3159 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3160 \ ], getline(1, 9))
3161
3162 w! >> test.out " Append it to the output file
3163
3164 augroup Test3
3165 au!
3166 augroup END
3167
3168 " Test for the FilterReadPre and FilterReadPost autocmds.
3169 set shelltemp " need temp files here
3170 augroup Test4
3171 au!
3172 au FilterReadPre *.out call rename(expand("<afile>"), expand("<afile>") . ".t")
3173 au FilterReadPre *.out exe 'silent !sed s/e/E/ ' . shellescape(expand("<afile>")) . ".t >" . shellescape(expand("<afile>"))
3174 au FilterReadPre *.out exe 'silent !rm ' . shellescape(expand("<afile>")) . '.t'
3175 au FilterReadPost *.out '[,']s/x/X/g
3176 augroup END
3177
3178 e! test.out " Edit the output file
3179 1,$!cat
3180 call assert_equal([
3181 \ 'linE 2 AbcdefghijklmnopqrstuvwXyz',
3182 \ 'linE 3 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3183 \ 'linE 4 AbcdefghijklmnopqrstuvwXyz',
3184 \ 'linE 5 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3185 \ 'linE 6 AbcdefghijklmnopqrstuvwXyz',
3186 \ 'linE 7 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3187 \ 'linE 8 AbcdefghijklmnopqrstuvwXyz',
3188 \ 'linE 9 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3189 \ 'linE 10 AbcdefghijklmnopqrstuvwXyz'
3190 \ ], getline(1, 9))
3191 call assert_equal([
3192 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3193 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3194 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3195 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3196 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3197 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3198 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3199 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3200 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3201 \ ], readfile('test.out'))
3202
3203 augroup Test4
3204 au!
3205 augroup END
3206 set shelltemp&vim
3207
3208 " Test for the FileReadPre and FileReadPost autocmds.
3209 augroup Test5
3210 au!
3211 au FileReadPre *.gz exe 'silent !gzip -d ' . shellescape(expand("<afile>"))
3212 au FileReadPre *.gz call rename(expand("<afile>:r"), expand("<afile>"))
3213 au FileReadPost *.gz '[,']s/l/L/
3214 augroup END
3215
3216 new
3217 0r Xtestfile.gz " Read compressed file
3218 call assert_equal([
3219 \ 'Line 2 Abcdefghijklmnopqrstuvwxyz',
3220 \ 'Line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3221 \ 'Line 4 Abcdefghijklmnopqrstuvwxyz',
3222 \ 'Line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3223 \ 'Line 6 Abcdefghijklmnopqrstuvwxyz',
3224 \ 'Line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3225 \ 'Line 8 Abcdefghijklmnopqrstuvwxyz',
3226 \ 'Line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3227 \ 'Line 10 Abcdefghijklmnopqrstuvwxyz'
3228 \ ], getline(1, 9))
3229 call assert_equal([
3230 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3231 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3232 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3233 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3234 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3235 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3236 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3237 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3238 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3239 \ ], readfile('Xtestfile.gz'))
3240
3241 augroup Test5
3242 au!
3243 augroup END
3244
3245 au! FileChangedShell
3246 call delete('Xtestfile.gz')
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003247 call delete('test.out')
3248endfunc
Bram Moolenaar23b51392019-05-09 21:38:43 +02003249
3250func Test_throw_in_BufWritePre()
3251 new
3252 call setline(1, ['one', 'two', 'three'])
3253 call assert_false(filereadable('Xthefile'))
3254 augroup throwing
3255 au BufWritePre X* throw 'do not write'
3256 augroup END
3257 try
3258 w Xthefile
3259 catch
3260 let caught = 1
3261 endtry
3262 call assert_equal(1, caught)
3263 call assert_false(filereadable('Xthefile'))
3264
3265 bwipe!
3266 au! throwing
3267endfunc
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003268
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003269func Test_autocmd_in_try_block()
Bram Moolenaar6f14da12022-09-07 21:30:44 +01003270 call mkdir('Xintrydir', 'R')
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003271 au BufEnter * let g:fname = expand('%')
3272 try
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01003273 edit Xintrydir/
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003274 endtry
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01003275 call assert_match('Xintrydir', g:fname)
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003276
3277 unlet g:fname
3278 au! BufEnter
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003279endfunc
3280
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003281func Test_autocmd_SafeState()
3282 CheckRunVimInTerminal
3283
3284 let lines =<< trim END
3285 let g:safe = 0
3286 let g:again = ''
3287 au SafeState * let g:safe += 1
3288 au SafeStateAgain * let g:again ..= 'x'
3289 func CallTimer()
3290 call timer_start(10, {id -> execute('let g:again ..= "t"')})
3291 endfunc
3292 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003293 call writefile(lines, 'XSafeState', 'D')
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003294 let buf = RunVimInTerminal('-S XSafeState', #{rows: 6})
3295
Bram Moolenaar8e7d6222020-12-18 19:49:56 +01003296 " Sometimes we loop to handle a K_IGNORE, SafeState may be triggered once or
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003297 " more often.
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003298 call term_sendkeys(buf, ":echo g:safe\<CR>")
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003299 call WaitForAssert({-> assert_match('^\d ', term_getline(buf, 6))}, 1000)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003300
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003301 " SafeStateAgain should be invoked at least three times
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003302 call term_sendkeys(buf, ":echo g:again\<CR>")
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003303 call WaitForAssert({-> assert_match('^xxx', term_getline(buf, 6))}, 1000)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003304
3305 call term_sendkeys(buf, ":let g:again = ''\<CR>:call CallTimer()\<CR>")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003306 call TermWait(buf, 50)
Bram Moolenaar0f6629a2019-09-22 23:24:13 +02003307 call term_sendkeys(buf, ":\<CR>")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003308 call TermWait(buf, 50)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003309 call term_sendkeys(buf, ":echo g:again\<CR>")
3310 call WaitForAssert({-> assert_match('xtx', term_getline(buf, 6))}, 1000)
3311
3312 call StopVimInTerminal(buf)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003313endfunc
Bram Moolenaar23324a02019-10-01 17:39:04 +02003314
3315func Test_autocmd_CmdWinEnter()
3316 CheckRunVimInTerminal
Bram Moolenaar21829c52021-01-26 22:42:21 +01003317
Bram Moolenaar23324a02019-10-01 17:39:04 +02003318 let lines =<< trim END
Egor Zvorykin125ffd22021-11-17 14:01:14 +00003319 augroup vimHints | au! | augroup END
Bram Moolenaar23324a02019-10-01 17:39:04 +02003320 let b:dummy_var = 'This is a dummy'
3321 autocmd CmdWinEnter * quit
3322 let winnr = winnr('$')
3323 END
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01003324 let filename = 'XCmdWinEnter'
Bram Moolenaar23324a02019-10-01 17:39:04 +02003325 call writefile(lines, filename)
3326 let buf = RunVimInTerminal('-S '.filename, #{rows: 6})
3327
3328 call term_sendkeys(buf, "q:")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003329 call TermWait(buf)
Bram Moolenaar23324a02019-10-01 17:39:04 +02003330 call term_sendkeys(buf, ":echo b:dummy_var\<cr>")
Bram Moolenaar353c3512020-03-15 14:19:26 +01003331 call WaitForAssert({-> assert_match('^This is a dummy', term_getline(buf, 6))}, 2000)
Bram Moolenaar23324a02019-10-01 17:39:04 +02003332 call term_sendkeys(buf, ":echo &buftype\<cr>")
3333 call WaitForAssert({-> assert_notmatch('^nofile', term_getline(buf, 6))}, 1000)
3334 call term_sendkeys(buf, ":echo winnr\<cr>")
3335 call WaitForAssert({-> assert_match('^1', term_getline(buf, 6))}, 1000)
3336
3337 " clean up
3338 call StopVimInTerminal(buf)
3339 call delete(filename)
3340endfunc
Bram Moolenaarec66c412019-10-11 21:19:13 +02003341
3342func Test_autocmd_was_using_freed_memory()
Bram Moolenaar5a4c3082019-12-01 15:23:11 +01003343 CheckFeature quickfix
3344
Bram Moolenaarec66c412019-10-11 21:19:13 +02003345 pedit xx
3346 n x
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003347 augroup winenter
3348 au WinEnter * if winnr('$') > 2 | quit | endif
3349 augroup END
Bram Moolenaarec66c412019-10-11 21:19:13 +02003350 split
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003351
3352 augroup winenter
3353 au! WinEnter
3354 augroup END
3355
3356 bwipe xx
3357 bwipe x
3358 pclose
Bram Moolenaarec66c412019-10-11 21:19:13 +02003359endfunc
Bram Moolenaarf4a1d1c2019-11-16 13:50:25 +01003360
3361func Test_BufWrite_lockmarks()
Bram Moolenaarf08b0eb2021-10-16 13:00:14 +01003362 let g:test_is_flaky = 1
Bram Moolenaarf4a1d1c2019-11-16 13:50:25 +01003363 edit! Xtest
3364 call setline(1, ['a', 'b', 'c', 'd'])
3365
3366 " :lockmarks preserves the marks
3367 call SetChangeMarks(2, 3)
3368 lockmarks write
3369 call assert_equal([2, 3], [line("'["), line("']")])
3370
3371 " *WritePre autocmds get the correct line range, but lockmarks preserves the
3372 " original values for the user
3373 augroup lockmarks
3374 au!
3375 au BufWritePre,FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
3376 au FileWritePre * call assert_equal([3, 4], [line("'["), line("']")])
3377 augroup END
3378
3379 lockmarks write
3380 call assert_equal([2, 3], [line("'["), line("']")])
3381
3382 if executable('cat')
3383 lockmarks %!cat
3384 call assert_equal([2, 3], [line("'["), line("']")])
3385 endif
3386
3387 lockmarks 3,4write Xtest2
3388 call assert_equal([2, 3], [line("'["), line("']")])
3389
3390 au! lockmarks
3391 augroup! lockmarks
3392 call delete('Xtest')
3393 call delete('Xtest2')
3394endfunc
Bram Moolenaarce6db022020-01-07 20:11:42 +01003395
3396func Test_FileType_spell()
3397 if !isdirectory('/tmp')
3398 throw "Skipped: requires /tmp directory"
3399 endif
3400
3401 " this was crashing with an invalid free()
3402 setglobal spellfile=/tmp/en.utf-8.add
3403 augroup crash
3404 autocmd!
3405 autocmd BufNewFile,BufReadPost crashfile setf somefiletype
3406 autocmd BufNewFile,BufReadPost crashfile set ft=anotherfiletype
3407 autocmd FileType anotherfiletype setlocal spell
3408 augroup END
3409 func! NoCrash() abort
3410 edit /tmp/crashfile
3411 endfunc
3412 call NoCrash()
3413
3414 au! crash
3415 setglobal spellfile=
3416endfunc
Bram Moolenaarbc2b71d2020-02-17 21:33:30 +01003417
Bram Moolenaaref976322022-09-28 11:48:30 +01003418" this was wiping out the current buffer and using freed memory
3419func Test_SpellFileMissing_bwipe()
3420 next 0
3421 au SpellFileMissing 0 bwipe
3422 call assert_fails('set spell spelllang=0', 'E937:')
3423
3424 au! SpellFileMissing
Bram Moolenaar0a60f792022-11-19 21:18:11 +00003425 set nospell spelllang=en
Bram Moolenaaref976322022-09-28 11:48:30 +01003426 bwipe
3427endfunc
3428
Bram Moolenaar406cd902020-02-18 21:54:41 +01003429" Test closing a window or editing another buffer from a FileChangedRO handler
3430" in a readonly buffer
3431func Test_FileChangedRO_winclose()
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003432 call test_override('ui_delay', 10)
3433
Bram Moolenaar406cd902020-02-18 21:54:41 +01003434 augroup FileChangedROTest
3435 au!
3436 autocmd FileChangedRO * quit
3437 augroup END
3438 new
3439 set readonly
3440 call assert_fails('normal i', 'E788:')
3441 close
3442 augroup! FileChangedROTest
3443
3444 augroup FileChangedROTest
3445 au!
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003446 autocmd FileChangedRO * edit Xrofile
Bram Moolenaar406cd902020-02-18 21:54:41 +01003447 augroup END
3448 new
3449 set readonly
3450 call assert_fails('normal i', 'E788:')
3451 close
3452 augroup! FileChangedROTest
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003453 call test_override('ALL', 0)
Bram Moolenaar406cd902020-02-18 21:54:41 +01003454endfunc
3455
Bram Moolenaar0c81d1b2020-02-22 22:45:55 +01003456func LogACmd()
3457 call add(g:logged, line('$'))
3458endfunc
3459
3460func Test_TermChanged()
Bram Moolenaard28e0b32020-02-22 23:08:52 +01003461 CheckNotGui
3462
Bram Moolenaar0c81d1b2020-02-22 22:45:55 +01003463 enew!
3464 tabnew
3465 call setline(1, ['a', 'b', 'c', 'd'])
3466 $
3467 au TermChanged * call LogACmd()
3468 let g:logged = []
3469 let term_save = &term
3470 set term=xterm
3471 call assert_equal([1, 4], g:logged)
3472
3473 au! TermChanged
3474 let &term = term_save
3475 bwipe!
3476endfunc
3477
Bram Moolenaare3284872020-03-19 13:55:03 +01003478" Test for FileReadCmd autocmd
3479func Test_autocmd_FileReadCmd()
3480 func ReadFileCmd()
3481 call append(line('$'), "v:cmdarg = " .. v:cmdarg)
3482 endfunc
3483 augroup FileReadCmdTest
3484 au!
3485 au FileReadCmd Xtest call ReadFileCmd()
3486 augroup END
3487
3488 new
3489 read ++bin Xtest
3490 read ++nobin Xtest
3491 read ++edit Xtest
3492 read ++bad=keep Xtest
3493 read ++bad=drop Xtest
3494 read ++bad=- Xtest
3495 read ++ff=unix Xtest
3496 read ++ff=dos Xtest
3497 read ++ff=mac Xtest
3498 read ++enc=utf-8 Xtest
3499
3500 call assert_equal(['',
3501 \ 'v:cmdarg = ++bin',
3502 \ 'v:cmdarg = ++nobin',
3503 \ 'v:cmdarg = ++edit',
3504 \ 'v:cmdarg = ++bad=keep',
3505 \ 'v:cmdarg = ++bad=drop',
3506 \ 'v:cmdarg = ++bad=-',
3507 \ 'v:cmdarg = ++ff=unix',
3508 \ 'v:cmdarg = ++ff=dos',
3509 \ 'v:cmdarg = ++ff=mac',
3510 \ 'v:cmdarg = ++enc=utf-8'], getline(1, '$'))
3511
Bram Moolenaar23526d22022-12-05 15:50:41 +00003512 bwipe!
Bram Moolenaare3284872020-03-19 13:55:03 +01003513 augroup FileReadCmdTest
3514 au!
3515 augroup END
3516 delfunc ReadFileCmd
3517endfunc
3518
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003519" Test for passing invalid arguments to autocmd
3520func Test_autocmd_invalid_args()
3521 " Additional character after * for event
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003522 call assert_fails('autocmd *a Xinvfile set ff=unix', 'E215:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003523 augroup Test
3524 augroup END
3525 " Invalid autocmd event
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003526 call assert_fails('autocmd Bufabc Xinvfile set ft=vim', 'E216:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003527 " Invalid autocmd event in a autocmd group
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003528 call assert_fails('autocmd Test Bufabc Xinvfile set ft=vim', 'E216:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003529 augroup! Test
3530 " Execute all autocmds
3531 call assert_fails('doautocmd * BufEnter', 'E217:')
3532 call assert_fails('augroup! x1a2b3', 'E367:')
3533 call assert_fails('autocmd BufNew <buffer=999> pwd', 'E680:')
Bram Moolenaar531be472020-09-23 22:38:05 +02003534 call assert_fails('autocmd BufNew \) set ff=unix', 'E55:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003535endfunc
3536
3537" Test for deep nesting of autocmds
3538func Test_autocmd_deep_nesting()
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003539 autocmd BufEnter Xdeepfile doautocmd BufEnter Xdeepfile
3540 call assert_fails('doautocmd BufEnter Xdeepfile', 'E218:')
3541 autocmd! BufEnter Xdeepfile
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003542endfunc
3543
Bram Moolenaarbe5ee862020-06-10 20:56:58 +02003544" Tests for SigUSR1 autocmd event, which is only available on posix systems.
3545func Test_autocmd_sigusr1()
3546 CheckUnix
Bram Moolenaar0056ca72022-09-23 21:26:39 +01003547 " FIXME: should this work on MacOS M1?
3548 CheckNotMacM1
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003549 CheckExecutable /bin/kill
Bram Moolenaarbe5ee862020-06-10 20:56:58 +02003550
3551 let g:sigusr1_passed = 0
3552 au SigUSR1 * let g:sigusr1_passed = 1
3553 call system('/bin/kill -s usr1 ' . getpid())
3554 call WaitForAssert({-> assert_true(g:sigusr1_passed)})
3555
3556 au! SigUSR1
3557 unlet g:sigusr1_passed
3558endfunc
3559
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003560" Test for BufReadPre autocmd deleting the file
3561func Test_BufReadPre_delfile()
3562 augroup TestAuCmd
3563 au!
Bram Moolenaare7cda972022-08-29 11:02:59 +01003564 autocmd BufReadPre XbufreadPre call delete('XbufreadPre')
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003565 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003566 call writefile([], 'XbufreadPre', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +01003567 call assert_fails('new XbufreadPre', 'E200:')
3568 call assert_equal('XbufreadPre', @%)
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003569 call assert_equal(1, &readonly)
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003570
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003571 augroup TestAuCmd
3572 au!
3573 augroup END
3574 close!
3575endfunc
3576
3577" Test for BufReadPre autocmd changing the current buffer
3578func Test_BufReadPre_changebuf()
3579 augroup TestAuCmd
3580 au!
Bram Moolenaare7cda972022-08-29 11:02:59 +01003581 autocmd BufReadPre Xchangebuf edit Xsomeotherfile
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003582 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003583 call writefile([], 'Xchangebuf', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +01003584 call assert_fails('new Xchangebuf', 'E201:')
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003585 call assert_equal('Xsomeotherfile', @%)
3586 call assert_equal(1, &readonly)
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003587
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003588 augroup TestAuCmd
3589 au!
3590 augroup END
3591 close!
3592endfunc
3593
3594" Test for BufWipeouti autocmd changing the current buffer when reading a file
3595" in an empty buffer with 'f' flag in 'cpo'
3596func Test_BufDelete_changebuf()
3597 new
3598 augroup TestAuCmd
3599 au!
3600 autocmd BufWipeout * let bufnr = bufadd('somefile') | exe "b " .. bufnr
3601 augroup END
3602 let save_cpo = &cpo
3603 set cpo+=f
Bram Moolenaarb18b4962022-09-02 21:55:50 +01003604 call assert_fails('r Xchangebuf', ['E812:', 'E484:'])
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003605 call assert_equal('somefile', @%)
3606 let &cpo = save_cpo
3607 augroup TestAuCmd
3608 au!
3609 augroup END
3610 close!
3611endfunc
3612
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003613" Test for the temporary internal window used to execute autocmds
3614func Test_autocmd_window()
3615 %bw!
3616 edit one.txt
3617 tabnew two.txt
Bram Moolenaar41cd8032021-03-13 15:47:56 +01003618 vnew three.txt
3619 tabnew four.txt
3620 tabprevious
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003621 let g:blist = []
Bram Moolenaar832adf92020-06-25 19:01:36 +02003622 augroup aucmd_win_test1
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003623 au!
3624 au BufEnter * call add(g:blist, [expand('<afile>'),
3625 \ win_gettype(bufwinnr(expand('<afile>')))])
3626 augroup END
3627
3628 doautoall BufEnter
Bram Moolenaar41cd8032021-03-13 15:47:56 +01003629 call assert_equal([
3630 \ ['one.txt', 'autocmd'],
3631 \ ['two.txt', ''],
3632 \ ['four.txt', 'autocmd'],
3633 \ ['three.txt', ''],
3634 \ ], g:blist)
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003635
Bram Moolenaar832adf92020-06-25 19:01:36 +02003636 augroup aucmd_win_test1
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003637 au!
3638 augroup END
Bram Moolenaar832adf92020-06-25 19:01:36 +02003639 augroup! aucmd_win_test1
3640 %bw!
3641endfunc
3642
3643" Test for trying to close the temporary window used for executing an autocmd
3644func Test_close_autocmd_window()
3645 %bw!
3646 edit one.txt
3647 tabnew two.txt
3648 augroup aucmd_win_test2
3649 au!
3650 au BufEnter * if expand('<afile>') == 'one.txt' | 1close | endif
3651 augroup END
3652
3653 call assert_fails('doautoall BufEnter', 'E813:')
3654
3655 augroup aucmd_win_test2
3656 au!
3657 augroup END
3658 augroup! aucmd_win_test2
Bram Moolenaarcf844172020-06-26 19:44:06 +02003659 %bwipe!
3660endfunc
3661
3662" Test for trying to close the tab that has the temporary window for exeucing
3663" an autocmd.
3664func Test_close_autocmd_tab()
3665 edit one.txt
3666 tabnew two.txt
3667 augroup aucmd_win_test
3668 au!
3669 au BufEnter * if expand('<afile>') == 'one.txt' | tabfirst | tabonly | endif
3670 augroup END
3671
3672 call assert_fails('doautoall BufEnter', 'E813:')
3673
3674 tabonly
3675 augroup aucmd_win_test
3676 au!
3677 augroup END
3678 augroup! aucmd_win_test
3679 %bwipe!
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003680endfunc
3681
Bram Moolenaarcb1956d2022-01-07 15:45:18 +00003682func Test_Visual_doautoall_redraw()
3683 call setline(1, ['a', 'b'])
Bram Moolenaar94722c52023-01-28 19:19:03 +00003684 new
Bram Moolenaarcb1956d2022-01-07 15:45:18 +00003685 wincmd p
3686 call feedkeys("G\<C-V>", 'txn')
3687 autocmd User Explode ++once redraw
3688 doautoall User Explode
3689 %bwipe!
3690endfunc
3691
Bram Moolenaar6bcb8772021-02-03 21:23:29 +01003692" This was using freed memory.
3693func Test_BufNew_arglocal()
3694 arglocal
3695 au BufNew * arglocal
3696 call assert_fails('drop xx', 'E1156:')
3697
3698 au! BufNew
3699endfunc
3700
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003701func Test_autocmd_closes_window()
3702 au BufNew,BufWinLeave * e %e
3703 file yyy
3704 au BufNew,BufWinLeave * ball
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003705 n xxx
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003706
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003707 %bwipe
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003708 au! BufNew
3709 au! BufWinLeave
3710endfunc
3711
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003712func Test_autocmd_quit_psearch()
3713 sn aa bb
3714 augroup aucmd_win_test
3715 au!
3716 au BufEnter,BufLeave,BufNew,WinEnter,WinLeave,WinNew * if winnr('$') > 1 | q | endif
3717 augroup END
3718 ps /
3719
3720 augroup aucmd_win_test
3721 au!
3722 augroup END
zeertzjq7851c692022-04-21 11:14:01 +01003723 new
3724 pclose
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003725endfunc
3726
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003727" Fuzzer found some strange combination that caused a crash.
3728func Test_autocmd_normal_mess()
Bram Moolenaardd07c022021-02-07 13:32:46 +01003729 " For unknown reason this hangs on MS-Windows
3730 CheckNotMSWindows
3731
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003732 augroup aucmd_normal_test
3733 au BufLeave,BufWinLeave,BufHidden,BufUnload,BufDelete,BufWipeout * norm 7q/qc
3734 augroup END
zeertzjq67fe77d2025-04-20 10:21:18 +02003735 call assert_fails('o4', 'E1159:')
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003736 silent! H
zeertzjq67fe77d2025-04-20 10:21:18 +02003737 call assert_fails('e xx', 'E1159:')
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003738 normal G
3739
3740 augroup aucmd_normal_test
3741 au!
3742 augroup END
3743endfunc
3744
Bram Moolenaar8c6951f2021-02-06 18:08:45 +01003745func Test_autocmd_closing_cmdwin()
Bram Moolenaardd07c022021-02-07 13:32:46 +01003746 " For unknown reason this hangs on MS-Windows
3747 CheckNotMSWindows
3748
Bram Moolenaar8c6951f2021-02-06 18:08:45 +01003749 au BufWinLeave * nested q
3750 call assert_fails("norm 7q?\n", 'E855:')
3751
3752 au! BufWinLeave
3753 new
3754 only
3755endfunc
3756
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003757func Test_autocmd_vimgrep()
3758 augroup aucmd_vimgrep
Charlie Grovesfef44852022-04-19 16:24:12 +01003759 au QuickfixCmdPre,BufNew,BufReadCmd * sb
zeertzjq7851c692022-04-21 11:14:01 +01003760 au QuickfixCmdPre,BufNew,BufReadCmd * q9
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003761 augroup END
Bram Moolenaardd07c022021-02-07 13:32:46 +01003762 call assert_fails('lv ?a? foo', 'E926:')
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003763
3764 augroup aucmd_vimgrep
3765 au!
3766 augroup END
3767endfunc
3768
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003769func Test_autocmd_with_block()
3770 augroup block_testing
3771 au BufReadPost *.xml {
3772 setlocal matchpairs+=<:>
3773 /<start
3774 }
Bram Moolenaar63b91732021-08-05 20:40:03 +02003775 au CursorHold * {
3776 autocmd BufReadPre * ++once echo 'one' | echo 'two'
3777 g:gotSafeState = 77
3778 }
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003779 augroup END
3780
Ken Takataeccc9272024-09-03 23:01:55 +02003781 let expected = gettext("\n--- Autocommands ---") .. "\nblock_testing BufRead\n *.xml {^@ setlocal matchpairs+=<:>^@ /<start^@ }"
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003782 call assert_equal(expected, execute('au BufReadPost *.xml'))
3783
Bram Moolenaar63b91732021-08-05 20:40:03 +02003784 doautocmd CursorHold
3785 call assert_equal(77, g:gotSafeState)
3786 unlet g:gotSafeState
3787
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003788 augroup block_testing
3789 au!
Bram Moolenaar75ebd2a2022-06-03 17:39:46 +01003790 autocmd CursorHold * {
3791 if true
3792 # comment
3793 && true
3794
3795 && true
3796 g:done = 'yes'
3797 endif
3798 }
3799 augroup END
3800 doautocmd CursorHold
3801 call assert_equal('yes', g:done)
3802
3803 unlet g:done
3804 augroup block_testing
3805 au!
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003806 augroup END
3807endfunc
3808
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00003809func Test_closing_autocmd_window()
3810 let lines =<< trim END
3811 edit Xa.txt
3812 tabnew Xb.txt
3813 autocmd BufEnter Xa.txt unhide 1
3814 doautoall BufEnter
3815 END
3816 call v9.CheckScriptFailure(lines, 'E814:')
3817 au! BufEnter
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00003818 bwipe Xa.txt
3819 bwipe Xb.txt
3820endfunc
3821
zeertzjq46bdae02023-09-24 23:16:08 +02003822func Test_switch_window_in_autocmd_window()
3823 edit Xa.txt
3824 tabnew Xb.txt
3825 autocmd BufEnter Xa.txt wincmd w
3826 doautoall BufEnter
3827 au! BufEnter
3828 bwipe Xa.txt
3829 call assert_false(bufexists('Xa.txt'))
3830 bwipe Xb.txt
3831 call assert_false(bufexists('Xb.txt'))
3832endfunc
3833
zeertzjq9d956ee2024-04-07 18:16:10 +02003834" Test that using the autocommand window doesn't change current directory.
3835func Test_autocmd_window_cwd()
3836 let saveddir = getcwd()
3837 call mkdir('Xcwd/a/b/c/d', 'pR')
3838
3839 new Xa.txt
3840 tabnew
3841 new Xb.txt
3842
3843 tabprev
3844 cd Xcwd
3845 call assert_match('/Xcwd$', getcwd())
3846 call assert_match('\[global\] .*/Xcwd$', trim(execute('verbose pwd')))
3847
3848 autocmd BufEnter Xb.txt lcd ./a/b/c/d
3849 doautoall BufEnter
3850 au! BufEnter
3851 call assert_match('/Xcwd$', getcwd())
3852 call assert_match('\[global\] .*/Xcwd$', trim(execute('verbose pwd')))
3853
3854 tabnext
3855 cd ./a
3856 tcd ./b
3857 lcd ./c
3858 call assert_match('/Xcwd/a/b/c$', getcwd())
3859 call assert_match('\[window\] .*/Xcwd/a/b/c$', trim(execute('verbose pwd')))
3860
3861 autocmd BufEnter Xa.txt call assert_match('Xcwd/a/b/c$', getcwd())
3862 doautoall BufEnter
3863 au! BufEnter
3864 call assert_match('/Xcwd/a/b/c$', getcwd())
3865 call assert_match('\[window\] .*/Xcwd/a/b/c$', trim(execute('verbose pwd')))
3866 bwipe!
3867 call assert_match('/Xcwd/a/b$', getcwd())
3868 call assert_match('\[tabpage\] .*/Xcwd/a/b$', trim(execute('verbose pwd')))
3869 bwipe!
3870 call assert_match('/Xcwd/a$', getcwd())
3871 call assert_match('\[global\] .*/Xcwd/a$', trim(execute('verbose pwd')))
3872 bwipe!
3873
3874 call chdir(saveddir)
3875endfunc
3876
Bram Moolenaar347538f2022-03-26 16:28:06 +00003877func Test_bufwipeout_changes_window()
3878 " This should not crash, but we don't have any expectations about what
3879 " happens, changing window in BufWipeout has unpredictable results.
3880 tabedit
3881 let g:window_id = win_getid()
3882 topleft new
3883 setlocal bufhidden=wipe
3884 autocmd BufWipeout <buffer> call win_gotoid(g:window_id)
3885 tabprevious
3886 +tabclose
3887
3888 unlet g:window_id
3889 au! BufWipeout
3890 %bwipe!
3891endfunc
3892
zeertzjq021996f2022-04-10 11:44:04 +01003893func Test_v_event_readonly()
3894 autocmd CompleteChanged * let v:event.width = 0
3895 call assert_fails("normal! i\<C-X>\<C-V>", 'E46:')
3896 au! CompleteChanged
3897
3898 autocmd DirChangedPre * let v:event.directory = ''
3899 call assert_fails('cd .', 'E46:')
3900 au! DirChangedPre
3901
3902 autocmd ModeChanged * let v:event.new_mode = ''
3903 call assert_fails('normal! cc', 'E46:')
3904 au! ModeChanged
3905
3906 autocmd TextYankPost * let v:event.operator = ''
3907 call assert_fails('normal! yy', 'E46:')
3908 au! TextYankPost
3909endfunc
3910
zeertzjqc9e8fd62022-07-26 18:12:38 +01003911" Test for ModeChanged pattern
3912func Test_mode_changes()
3913 let g:index = 0
zeertzjq73916ba2023-04-26 16:50:19 +01003914 let g:mode_seq = ['n', 'i', 'n', 'v', 'V', 'i', 'ix', 'i', 'ic', 'i', 'n', 'no', 'noV', 'n', 'V', 'v', 's', 'n']
zeertzjqc9e8fd62022-07-26 18:12:38 +01003915 func! TestMode()
3916 call assert_equal(g:mode_seq[g:index], get(v:event, "old_mode"))
3917 call assert_equal(g:mode_seq[g:index + 1], get(v:event, "new_mode"))
3918 call assert_equal(mode(1), get(v:event, "new_mode"))
3919 let g:index += 1
3920 endfunc
3921
3922 au ModeChanged * :call TestMode()
3923 let g:n_to_any = 0
3924 au ModeChanged n:* let g:n_to_any += 1
zeertzjq73916ba2023-04-26 16:50:19 +01003925 call feedkeys("i\<esc>vVca\<CR>\<C-X>\<C-L>\<esc>ggdV\<MouseMove>G", 'tnix')
zeertzjqc9e8fd62022-07-26 18:12:38 +01003926
3927 let g:V_to_v = 0
3928 au ModeChanged V:v let g:V_to_v += 1
3929 call feedkeys("Vv\<C-G>\<esc>", 'tnix')
3930 call assert_equal(len(filter(g:mode_seq[1:], {idx, val -> val == 'n'})), g:n_to_any)
3931 call assert_equal(1, g:V_to_v)
3932 call assert_equal(len(g:mode_seq) - 1, g:index)
3933
3934 let g:n_to_i = 0
3935 au ModeChanged n:i let g:n_to_i += 1
3936 let g:n_to_niI = 0
3937 au ModeChanged i:niI let g:n_to_niI += 1
3938 let g:niI_to_i = 0
3939 au ModeChanged niI:i let g:niI_to_i += 1
3940 let g:nany_to_i = 0
3941 au ModeChanged n*:i let g:nany_to_i += 1
3942 let g:i_to_n = 0
3943 au ModeChanged i:n let g:i_to_n += 1
3944 let g:nori_to_any = 0
3945 au ModeChanged [ni]:* let g:nori_to_any += 1
3946 let g:i_to_any = 0
3947 au ModeChanged i:* let g:i_to_any += 1
3948 let g:index = 0
3949 let g:mode_seq = ['n', 'i', 'niI', 'i', 'n']
3950 call feedkeys("a\<C-O>l\<esc>", 'tnix')
3951 call assert_equal(len(g:mode_seq) - 1, g:index)
3952 call assert_equal(1, g:n_to_i)
3953 call assert_equal(1, g:n_to_niI)
3954 call assert_equal(1, g:niI_to_i)
3955 call assert_equal(2, g:nany_to_i)
3956 call assert_equal(1, g:i_to_n)
3957 call assert_equal(2, g:i_to_any)
3958 call assert_equal(3, g:nori_to_any)
3959
3960 if has('terminal')
3961 let g:mode_seq += ['c', 'n', 't', 'nt', 'c', 'nt', 'n']
3962 call feedkeys(":term\<CR>\<C-W>N:bd!\<CR>", 'tnix')
3963 call assert_equal(len(g:mode_seq) - 1, g:index)
3964 call assert_equal(1, g:n_to_i)
3965 call assert_equal(1, g:n_to_niI)
3966 call assert_equal(1, g:niI_to_i)
3967 call assert_equal(2, g:nany_to_i)
3968 call assert_equal(1, g:i_to_n)
3969 call assert_equal(2, g:i_to_any)
3970 call assert_equal(5, g:nori_to_any)
3971 endif
3972
zeertzjqd1955982022-10-05 11:24:46 +01003973 let g:n_to_c = 0
3974 au ModeChanged n:c let g:n_to_c += 1
3975 let g:c_to_n = 0
3976 au ModeChanged c:n let g:c_to_n += 1
3977 let g:mode_seq += ['c', 'n', 'c', 'n']
3978 call feedkeys("q:\<C-C>\<Esc>", 'tnix')
3979 call assert_equal(len(g:mode_seq) - 1, g:index)
3980 call assert_equal(2, g:n_to_c)
3981 call assert_equal(2, g:c_to_n)
zeertzjqc9e8fd62022-07-26 18:12:38 +01003982
Bram Moolenaar61c4b042022-10-18 15:10:11 +01003983 let g:n_to_v = 0
3984 au ModeChanged n:v let g:n_to_v += 1
3985 let g:v_to_n = 0
3986 au ModeChanged v:n let g:v_to_n += 1
3987 let g:mode_seq += ['v', 'n']
3988 call feedkeys("v\<C-C>", 'tnix')
3989 call assert_equal(len(g:mode_seq) - 1, g:index)
3990 call assert_equal(1, g:n_to_v)
3991 call assert_equal(1, g:v_to_n)
zeertzjqfcaeb3d2023-11-28 20:46:29 +01003992
3993 let g:mode_seq += ['c', 'cr', 'c', 'cr', 'n']
3994 call feedkeys(":\<Insert>\<Insert>\<Insert>\<CR>", 'tnix')
3995 call assert_equal(len(g:mode_seq) - 1, g:index)
Bram Moolenaar61c4b042022-10-18 15:10:11 +01003996
zeertzjqc9e8fd62022-07-26 18:12:38 +01003997 au! ModeChanged
3998 delfunc TestMode
3999 unlet! g:mode_seq
4000 unlet! g:index
4001 unlet! g:n_to_any
4002 unlet! g:V_to_v
4003 unlet! g:n_to_i
4004 unlet! g:n_to_niI
4005 unlet! g:niI_to_i
4006 unlet! g:nany_to_i
4007 unlet! g:i_to_n
4008 unlet! g:nori_to_any
4009 unlet! g:i_to_any
zeertzjqfcaeb3d2023-11-28 20:46:29 +01004010 unlet! g:n_to_c
4011 unlet! g:c_to_n
4012 unlet! g:n_to_v
4013 unlet! g:v_to_n
zeertzjqc9e8fd62022-07-26 18:12:38 +01004014endfunc
4015
4016func Test_recursive_ModeChanged()
4017 au! ModeChanged * norm 0u
4018 sil! norm 
4019 au! ModeChanged
4020endfunc
4021
4022func Test_ModeChanged_starts_visual()
4023 " This was triggering ModeChanged before setting VIsual, causing a crash.
4024 au! ModeChanged * norm 0u
4025 sil! norm 
4026
4027 au! ModeChanged
4028endfunc
Bram Moolenaar347538f2022-03-26 16:28:06 +00004029
Charlie Grovesfef44852022-04-19 16:24:12 +01004030func Test_noname_autocmd()
4031 augroup test_noname_autocmd_group
4032 autocmd!
4033 autocmd BufEnter * call add(s:li, ["BufEnter", expand("<afile>")])
4034 autocmd BufDelete * call add(s:li, ["BufDelete", expand("<afile>")])
4035 autocmd BufLeave * call add(s:li, ["BufLeave", expand("<afile>")])
4036 autocmd BufUnload * call add(s:li, ["BufUnload", expand("<afile>")])
4037 autocmd BufWipeout * call add(s:li, ["BufWipeout", expand("<afile>")])
4038 augroup END
4039
4040 let s:li = []
4041 edit foo
4042 call assert_equal([['BufUnload', ''], ['BufDelete', ''], ['BufWipeout', ''], ['BufEnter', 'foo']], s:li)
4043
4044 au! test_noname_autocmd_group
4045 augroup! test_noname_autocmd_group
4046endfunc
4047
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004048" Test for the autocmd_get() function
4049func Test_autocmd_get()
4050 augroup TestAutoCmdFns
4051 au!
4052 autocmd BufAdd *.vim echo "bufadd-vim"
4053 autocmd BufAdd *.py echo "bufadd-py"
4054 autocmd BufHidden *.vim echo "bufhidden"
4055 augroup END
4056 augroup TestAutoCmdFns2
4057 autocmd BufAdd *.vim echo "bufadd-vim-2"
4058 autocmd BufRead *.a1b2c3 echo "bufadd-vim-2"
4059 augroup END
4060
4061 let l = autocmd_get()
4062 call assert_true(l->len() > 0)
4063
4064 " Test for getting all the autocmds in a group
4065 let expected = [
4066 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4067 \ pattern: '*.vim', nested: v:false, once: v:false,
4068 \ event: 'BufAdd'},
4069 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4070 \ pattern: '*.py', nested: v:false, once: v:false,
4071 \ event: 'BufAdd'},
4072 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4073 \ pattern: '*.vim', nested: v:false,
4074 \ once: v:false, event: 'BufHidden'}]
4075 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4076
4077 " Test for getting autocmds for all the patterns in a group
4078 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns',
4079 \ event: '*'}))
4080
4081 " Test for getting autocmds for an event in a group
4082 let expected = [
4083 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4084 \ pattern: '*.vim', nested: v:false, once: v:false,
4085 \ event: 'BufAdd'},
4086 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4087 \ pattern: '*.py', nested: v:false, once: v:false,
4088 \ event: 'BufAdd'}]
4089 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns',
4090 \ event: 'BufAdd'}))
4091
4092 " Test for getting the autocmds for all the events in a group for particular
4093 " pattern
4094 call assert_equal([{'cmd': 'echo "bufadd-py"', 'group': 'TestAutoCmdFns',
4095 \ 'pattern': '*.py', 'nested': v:false, 'once': v:false,
4096 \ 'event': 'BufAdd'}],
4097 \ autocmd_get(#{group: 'TestAutoCmdFns', event: '*', pattern: '*.py'}))
4098
4099 " Test for getting the autocmds for an events in a group for particular
4100 " pattern
4101 let l = autocmd_get(#{group: 'TestAutoCmdFns', event: 'BufAdd',
4102 \ pattern: '*.vim'})
4103 call assert_equal([
4104 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4105 \ pattern: '*.vim', nested: v:false, once: v:false,
4106 \ event: 'BufAdd'}], l)
4107
4108 " Test for getting the autocmds for a pattern in a group
4109 let l = autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.vim'})
4110 call assert_equal([
4111 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4112 \ pattern: '*.vim', nested: v:false, once: v:false,
4113 \ event: 'BufAdd'},
4114 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4115 \ pattern: '*.vim', nested: v:false,
4116 \ once: v:false, event: 'BufHidden'}], l)
4117
4118 " Test for getting the autocmds for a pattern in all the groups
4119 let l = autocmd_get(#{pattern: '*.a1b2c3'})
4120 call assert_equal([{'cmd': 'echo "bufadd-vim-2"', 'group': 'TestAutoCmdFns2',
4121 \ 'pattern': '*.a1b2c3', 'nested': v:false, 'once': v:false,
4122 \ 'event': 'BufRead'}], l)
4123
4124 " Test for getting autocmds for a pattern without any autocmds
4125 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4126 \ pattern: '*.abc'}))
4127 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4128 \ event: 'BufAdd', pattern: '*.abc'}))
4129 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4130 \ event: 'BufWipeout'}))
zeertzjq2d1d5c62024-06-09 16:44:33 +02004131
4132 " Test for getting autocmds after removing one inside an autocmd
4133 func CheckAutocmdGet()
4134 augroup TestAutoCmdFns
4135 autocmd! BufAdd *.vim
4136 augroup END
4137
4138 let expected = [
4139 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4140 \ pattern: '*.py', nested: v:false, once: v:false,
4141 \ event: 'BufAdd'},
4142 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4143 \ pattern: '*.vim', nested: v:false,
4144 \ once: v:false, event: 'BufHidden'}]
4145
4146 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4147 call assert_equal([expected[0]],
4148 \ autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.py'}))
4149 call assert_equal([expected[1]],
4150 \ autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.vim'}))
4151 endfunc
4152
4153 autocmd User Xauget call CheckAutocmdGet()
4154 doautocmd User Xauget
4155 autocmd! User Xauget
4156
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004157 call assert_fails("call autocmd_get(#{group: 'abc', event: 'BufAdd'})",
4158 \ 'E367:')
4159 let cmd = "echo autocmd_get(#{group: 'TestAutoCmdFns', event: 'abc'})"
4160 call assert_fails(cmd, 'E216:')
4161 call assert_fails("call autocmd_get(#{group: 'abc'})", 'E367:')
4162 call assert_fails("echo autocmd_get(#{event: 'abc'})", 'E216:')
4163
4164 augroup TestAutoCmdFns
4165 au!
4166 augroup END
4167 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns'}))
4168
4169 " Test for nested and once autocmds
4170 augroup TestAutoCmdFns
4171 au!
4172 autocmd VimSuspend * ++nested echo "suspend"
4173 autocmd VimResume * ++once echo "resume"
4174 augroup END
4175
4176 let expected = [
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004177 \ {'cmd': 'echo "resume"', 'group': 'TestAutoCmdFns', 'pattern': '*',
Luuk van Baalb7147f82025-02-08 18:52:39 +01004178 \ 'nested': v:false, 'once': v:true, 'event': 'VimResume'},
4179 \ {'cmd': 'echo "suspend"', 'group': 'TestAutoCmdFns', 'pattern': '*',
4180 \ 'nested': v:true, 'once': v:false, 'event': 'VimSuspend'}]
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004181 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4182
4183 " Test for buffer-local autocmd
4184 augroup TestAutoCmdFns
4185 au!
4186 autocmd TextYankPost <buffer> echo "textyankpost"
4187 augroup END
4188
4189 let expected = [
4190 \ {'cmd': 'echo "textyankpost"', 'group': 'TestAutoCmdFns',
4191 \ 'pattern': '<buffer=' .. bufnr() .. '>', 'nested': v:false,
4192 \ 'once': v:false, 'bufnr': bufnr(), 'event': 'TextYankPost'}]
4193 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4194
4195 augroup TestAutoCmdFns
4196 au!
4197 augroup END
4198 augroup! TestAutoCmdFns
4199 augroup TestAutoCmdFns2
4200 au!
4201 augroup END
4202 augroup! TestAutoCmdFns2
4203
4204 call assert_fails("echo autocmd_get(#{group: []})", 'E730:')
4205 call assert_fails("echo autocmd_get(#{event: {}})", 'E731:')
4206 call assert_fails("echo autocmd_get([])", 'E1206:')
4207endfunc
4208
4209" Test for the autocmd_add() function
4210func Test_autocmd_add()
4211 " Define a single autocmd in a group
4212 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pattern: '*.sh',
4213 \ cmd: 'echo "bufadd"', once: v:true, nested: v:true}])
4214 call assert_equal([#{cmd: 'echo "bufadd"', group: 'TestAcSet',
4215 \ pattern: '*.sh', nested: v:true, once: v:true,
4216 \ event: 'BufAdd'}], autocmd_get(#{group: 'TestAcSet'}))
4217
4218 " Define two autocmds in the same group
4219 call autocmd_delete([#{group: 'TestAcSet'}])
4220 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pattern: '*.sh',
4221 \ cmd: 'echo "bufadd"'},
4222 \ #{group: 'TestAcSet', event: 'BufEnter', pattern: '*.sh',
4223 \ cmd: 'echo "bufenter"'}])
4224 call assert_equal([
4225 \ #{cmd: 'echo "bufadd"', group: 'TestAcSet', pattern: '*.sh',
4226 \ nested: v:false, once: v:false, event: 'BufAdd'},
4227 \ #{cmd: 'echo "bufenter"', group: 'TestAcSet', pattern: '*.sh',
4228 \ nested: v:false, once: v:false, event: 'BufEnter'}],
4229 \ autocmd_get(#{group: 'TestAcSet'}))
4230
4231 " Define a buffer-local autocmd
4232 call autocmd_delete([#{group: 'TestAcSet'}])
4233 call autocmd_add([#{group: 'TestAcSet', event: 'CursorHold',
4234 \ bufnr: bufnr(), cmd: 'echo "cursorhold"'}])
4235 call assert_equal([
4236 \ #{cmd: 'echo "cursorhold"', group: 'TestAcSet',
4237 \ pattern: '<buffer=' .. bufnr() .. '>', nested: v:false,
4238 \ once: v:false, bufnr: bufnr(), event: 'CursorHold'}],
4239 \ autocmd_get(#{group: 'TestAcSet'}))
4240
4241 " Use an invalid buffer number
4242 call autocmd_delete([#{group: 'TestAcSet'}])
4243 call autocmd_add([#{group: 'TestAcSet', event: 'BufEnter',
4244 \ bufnr: -1, cmd: 'echo "bufenter"'}])
4245 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4246 \ cmd: 'echo "bufadd"'}]
4247 call assert_fails("echo autocmd_add(l)", 'E680:')
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004248 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4249 \ pattern: '*.py', cmd: 'echo "bufadd"'}]
4250 call assert_fails("echo autocmd_add(l)", 'E680:')
4251 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4252 \ pattern: ['*.py', '*.c'], cmd: 'echo "bufadd"'}]
4253 call assert_fails("echo autocmd_add(l)", 'E680:')
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004254 let l = [#{group: 'TestAcSet', event: 'BufRead', bufnr: [],
4255 \ cmd: 'echo "bufread"'}]
4256 call assert_fails("echo autocmd_add(l)", 'E745:')
4257 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4258
4259 " Add two commands to the same group, event and pattern
4260 call autocmd_delete([#{group: 'TestAcSet'}])
4261 call autocmd_add([#{group: 'TestAcSet', event: 'BufUnload',
4262 \ pattern: 'abc', cmd: 'echo "cmd1"'}])
4263 call autocmd_add([#{group: 'TestAcSet', event: 'BufUnload',
4264 \ pattern: 'abc', cmd: 'echo "cmd2"'}])
4265 call assert_equal([
4266 \ #{cmd: 'echo "cmd1"', group: 'TestAcSet', pattern: 'abc',
4267 \ nested: v:false, once: v:false, event: 'BufUnload'},
4268 \ #{cmd: 'echo "cmd2"', group: 'TestAcSet', pattern: 'abc',
4269 \ nested: v:false, once: v:false, event: 'BufUnload'}],
4270 \ autocmd_get(#{group: 'TestAcSet'}))
4271
4272 " When adding a new autocmd, if the autocmd 'group' is not specified, then
4273 " the current autocmd group should be used.
4274 call autocmd_delete([#{group: 'TestAcSet'}])
4275 augroup TestAcSet
4276 call autocmd_add([#{event: 'BufHidden', pattern: 'abc', cmd: 'echo "abc"'}])
4277 augroup END
4278 call assert_equal([
4279 \ #{cmd: 'echo "abc"', group: 'TestAcSet', pattern: 'abc',
4280 \ nested: v:false, once: v:false, event: 'BufHidden'}],
4281 \ autocmd_get(#{group: 'TestAcSet'}))
4282
Yegappan Lakshmanan971f6822022-05-24 11:40:11 +01004283 " Test for replacing a cmd for an event in a group
4284 call autocmd_delete([#{group: 'TestAcSet'}])
4285 call autocmd_add([#{replace: v:true, group: 'TestAcSet', event: 'BufEnter',
4286 \ pattern: '*.py', cmd: 'echo "bufenter"'}])
4287 call autocmd_add([#{replace: v:true, group: 'TestAcSet', event: 'BufEnter',
4288 \ pattern: '*.py', cmd: 'echo "bufenter"'}])
4289 call assert_equal([
4290 \ #{cmd: 'echo "bufenter"', group: 'TestAcSet', pattern: '*.py',
4291 \ nested: v:false, once: v:false, event: 'BufEnter'}],
4292 \ autocmd_get(#{group: 'TestAcSet'}))
4293
4294 " Test for adding a command for an unsupported autocmd event
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004295 let l = [#{group: 'TestAcSet', event: 'abc', pattern: '*.sh',
4296 \ cmd: 'echo "bufadd"'}]
4297 call assert_fails('call autocmd_add(l)', 'E216:')
4298
Yegappan Lakshmanane0ff3a72022-05-27 18:05:33 +01004299 " Test for using a list of events and patterns
4300 call autocmd_delete([#{group: 'TestAcSet'}])
4301 let l = [#{group: 'TestAcSet', event: ['BufEnter', 'BufLeave'],
4302 \ pattern: ['*.py', '*.sh'], cmd: 'echo "bufcmds"'}]
4303 call autocmd_add(l)
4304 call assert_equal([
4305 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.py',
4306 \ nested: v:false, once: v:false, event: 'BufEnter'},
4307 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.sh',
4308 \ nested: v:false, once: v:false, event: 'BufEnter'},
4309 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.py',
4310 \ nested: v:false, once: v:false, event: 'BufLeave'},
4311 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.sh',
4312 \ nested: v:false, once: v:false, event: 'BufLeave'}],
4313 \ autocmd_get(#{group: 'TestAcSet'}))
4314
4315 " Test for invalid values for 'event' item
4316 call autocmd_delete([#{group: 'TestAcSet'}])
4317 let l = [#{group: 'TestAcSet', event: test_null_string(),
4318 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4319 call assert_fails('call autocmd_add(l)', 'E928:')
4320 let l = [#{group: 'TestAcSet', event: test_null_list(),
4321 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4322 call assert_fails('call autocmd_add(l)', 'E714:')
4323 let l = [#{group: 'TestAcSet', event: {},
4324 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4325 call assert_fails('call autocmd_add(l)', 'E777:')
4326 let l = [#{group: 'TestAcSet', event: [{}],
4327 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4328 call assert_fails('call autocmd_add(l)', 'E928:')
4329 let l = [#{group: 'TestAcSet', event: [test_null_string()],
4330 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4331 call assert_fails('call autocmd_add(l)', 'E928:')
4332 let l = [#{group: 'TestAcSet', event: 'BufEnter,BufLeave',
4333 \ pattern: '*.py', cmd: 'echo "bufcmds"'}]
4334 call assert_fails('call autocmd_add(l)', 'E216:')
4335 let l = [#{group: 'TestAcSet', event: [],
4336 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4337 call autocmd_add(l)
4338 let l = [#{group: 'TestAcSet', event: [""],
4339 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4340 call assert_fails('call autocmd_add(l)', 'E216:')
4341 let l = [#{group: 'TestAcSet', event: "",
4342 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4343 call autocmd_add(l)
4344 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4345
4346 " Test for invalid values for 'pattern' item
4347 let l = [#{group: 'TestAcSet', event: "BufEnter",
4348 \ pattern: test_null_string(), cmd: 'echo "bufcmds"'}]
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004349 call assert_fails('call autocmd_add(l)', 'E928:')
Yegappan Lakshmanane0ff3a72022-05-27 18:05:33 +01004350 let l = [#{group: 'TestAcSet', event: "BufEnter",
4351 \ pattern: test_null_list(), cmd: 'echo "bufcmds"'}]
4352 call assert_fails('call autocmd_add(l)', 'E714:')
4353 let l = [#{group: 'TestAcSet', event: "BufEnter",
4354 \ pattern: {}, cmd: 'echo "bufcmds"'}]
4355 call assert_fails('call autocmd_add(l)', 'E777:')
4356 let l = [#{group: 'TestAcSet', event: "BufEnter",
4357 \ pattern: [{}], cmd: 'echo "bufcmds"'}]
4358 call assert_fails('call autocmd_add(l)', 'E928:')
4359 let l = [#{group: 'TestAcSet', event: "BufEnter",
4360 \ pattern: [test_null_string()], cmd: 'echo "bufcmds"'}]
4361 call assert_fails('call autocmd_add(l)', 'E928:')
4362 let l = [#{group: 'TestAcSet', event: "BufEnter",
4363 \ pattern: [], cmd: 'echo "bufcmds"'}]
4364 call autocmd_add(l)
4365 let l = [#{group: 'TestAcSet', event: "BufEnter",
4366 \ pattern: [""], cmd: 'echo "bufcmds"'}]
4367 call autocmd_add(l)
4368 let l = [#{group: 'TestAcSet', event: "BufEnter",
4369 \ pattern: "", cmd: 'echo "bufcmds"'}]
4370 call autocmd_add(l)
4371 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4372
4373 let l = [#{group: 'TestAcSet', event: 'BufEnter,abc,BufLeave',
4374 \ pattern: '*.py', cmd: 'echo "bufcmds"'}]
4375 call assert_fails('call autocmd_add(l)', 'E216:')
4376
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004377 call assert_fails("call autocmd_add({})", 'E1211:')
4378 call assert_equal(v:false, autocmd_add(test_null_list()))
4379 call assert_true(autocmd_add([[]]))
4380 call assert_true(autocmd_add([test_null_dict()]))
4381
4382 augroup TestAcSet
4383 au!
4384 augroup END
4385
4386 call autocmd_add([#{group: 'TestAcSet'}])
4387 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd'}])
4388 call autocmd_add([#{group: 'TestAcSet', pat: '*.sh'}])
4389 call autocmd_add([#{group: 'TestAcSet', cmd: 'echo "a"'}])
4390 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pat: '*.sh'}])
4391 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', cmd: 'echo "a"'}])
4392 call autocmd_add([#{group: 'TestAcSet', pat: '*.sh', cmd: 'echo "a"'}])
4393 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4394
4395 augroup! TestAcSet
4396endfunc
4397
4398" Test for deleting autocmd events and groups
4399func Test_autocmd_delete()
4400 " Delete an event in an autocmd group
4401 augroup TestAcSet
4402 au!
4403 au BufAdd *.sh echo "bufadd"
4404 au BufEnter *.sh echo "bufenter"
4405 augroup END
4406 call autocmd_delete([#{group: 'TestAcSet', event: 'BufAdd'}])
4407 call assert_equal([#{cmd: 'echo "bufenter"', group: 'TestAcSet',
4408 \ pattern: '*.sh', nested: v:false, once: v:false,
4409 \ event: 'BufEnter'}], autocmd_get(#{group: 'TestAcSet'}))
4410
4411 " Delete all the events in an autocmd group
4412 augroup TestAcSet
4413 au BufAdd *.sh echo "bufadd"
4414 augroup END
4415 call autocmd_delete([#{group: 'TestAcSet', event: '*'}])
4416 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4417
4418 " Delete a non-existing autocmd group
4419 call assert_fails("call autocmd_delete([#{group: 'abc'}])", 'E367:')
4420 " Delete a non-existing autocmd event
4421 let l = [#{group: 'TestAcSet', event: 'abc'}]
4422 call assert_fails("call autocmd_delete(l)", 'E216:')
4423 " Delete a non-existing autocmd pattern
4424 let l = [#{group: 'TestAcSet', event: 'BufAdd', pat: 'abc'}]
4425 call assert_true(autocmd_delete(l))
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004426 " Delete an autocmd for a non-existing buffer
4427 let l = [#{event: '*', bufnr: 9999, cmd: 'echo "x"'}]
4428 call assert_fails('call autocmd_delete(l)', 'E680:')
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004429
4430 " Delete an autocmd group
4431 augroup TestAcSet
4432 au!
4433 au BufAdd *.sh echo "bufadd"
4434 au BufEnter *.sh echo "bufenter"
4435 augroup END
4436 call autocmd_delete([#{group: 'TestAcSet'}])
4437 call assert_fails("call autocmd_get(#{group: 'TestAcSet'})", 'E367:')
4438
4439 call assert_true(autocmd_delete([[]]))
4440 call assert_true(autocmd_delete([test_null_dict()]))
4441endfunc
4442
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004443func Test_autocmd_split_dummy()
4444 " Autocommand trying to split a window containing a dummy buffer.
Bram Moolenaar94722c52023-01-28 19:19:03 +00004445 auto BufReadPre * exe "sbuf " .. expand("<abuf>")
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004446 " Avoid the "W11" prompt
4447 au FileChangedShell * let v:fcs_choice = 'reload'
4448 func Xautocmd_changelist()
4449 cal writefile(['Xtestfile2:4:4'], 'Xerr')
4450 edit Xerr
4451 lex 'Xtestfile2:4:4'
4452 endfunc
4453 call Xautocmd_changelist()
Bram Moolenaar53c5c9f2022-10-18 17:25:03 +01004454 " Should get E86, but it doesn't always happen (timing?)
4455 silent! call Xautocmd_changelist()
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004456
4457 au! BufReadPre
4458 au! FileChangedShell
4459 delfunc Xautocmd_changelist
4460 bwipe! Xerr
4461 call delete('Xerr')
4462endfunc
4463
Bram Moolenaare76062c2022-11-28 18:51:43 +00004464" This was crashing because there was only one window to execute autocommands
4465" in.
4466func Test_autocmd_nested_setbufvar()
4467 CheckFeature python3
4468
4469 set hidden
4470 edit Xaaa
4471 edit Xbbb
4472 call setline(1, 'bar')
4473 enew
4474 au BufWriteCmd Xbbb ++nested call setbufvar('Xaaa', '&ft', 'foo') | bw! Xaaa
4475 au FileType foo call py3eval('vim.current.buffer.options["cindent"]')
4476 wall
4477
4478 au! BufWriteCmd
4479 au! FileType foo
4480 set nohidden
4481 call delete('Xaaa')
4482 call delete('Xbbb')
4483 %bwipe!
4484endfunc
4485
Christian Brabandt9aee8ec2022-12-16 16:41:23 +00004486func SetupVimTest_shm()
4487 let g:bwe = []
4488 let g:brp = []
4489 set shortmess+=F
zeertzjq657b31f2023-04-15 21:28:02 +01004490 messages clear
Christian Brabandt9aee8ec2022-12-16 16:41:23 +00004491
4492 let dirname='XVimTestSHM'
4493 call mkdir(dirname, 'R')
4494 call writefile(['test'], dirname .. '/1')
4495 call writefile(['test'], dirname .. '/2')
4496 call writefile(['test'], dirname .. '/3')
4497
4498 augroup test
4499 autocmd!
4500 autocmd BufWinEnter * call add(g:bwe, $'BufWinEnter: {expand('<amatch>')}')
4501 autocmd BufReadPost * call add(g:brp, $'BufReadPost: {expand('<amatch>')}')
4502 augroup END
4503
4504 call setqflist([
4505 \ {'filename': dirname .. '/1', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0},
4506 \ {'filename': dirname .. '/2', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0},
4507 \ {'filename': dirname .. '/3', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0}
4508 \ ])
4509 cdo! substitute/test/TEST
4510
4511 " clean up
4512 noa enew!
4513 set shortmess&vim
4514 augroup test
4515 autocmd!
4516 augroup END
4517 augroup! test
4518endfunc
4519
4520func Test_autocmd_shortmess()
4521 CheckNotMSWindows
4522
4523 call SetupVimTest_shm()
4524 let output = execute(':mess')->split('\n')
4525
4526 let info = copy(output)->filter({idx, val -> val =~# '\d of 3'} )
4527 let bytes = copy(output)->filter({idx, val -> val =~# 'bytes'} )
4528
4529 " We test the following here:
4530 " BufReadPost should have been triggered 3 times, once per file
4531 " BufWinEnter should have been triggered 3 times, once per file
4532 " FileInfoMessage should have been shown 3 times, regardless of shm option
4533 " "(x of 3)" message from :cnext has been shown 3 times
4534
4535 call assert_equal(3, g:brp->len())
4536 call assert_equal(3, g:bwe->len())
4537 call assert_equal(3, info->len())
4538 call assert_equal(3, bytes->len())
4539
4540 delfunc SetupVimTest_shm
4541endfunc
Bram Moolenaare76062c2022-11-28 18:51:43 +00004542
Christian Brabandtf0d3d4a2024-02-15 20:15:04 +01004543func Test_autocmd_invalidates_undo_on_textchanged()
4544 CheckRunVimInTerminal
4545 let script =<< trim END
4546 set hidden
4547 " create quickfix list (at least 2 lines to move line)
4548 vimgrep /u/j %
4549
4550 " enter quickfix window
4551 cwindow
4552
4553 " set modifiable
4554 setlocal modifiable
4555
4556 " set autocmd to clear quickfix list
4557
4558 autocmd! TextChanged <buffer> call setqflist([])
4559 " move line
4560 move+1
4561 END
4562 call writefile(script, 'XTest_autocmd_invalidates_undo_on_textchanged', 'D')
4563 let buf = RunVimInTerminal('XTest_autocmd_invalidates_undo_on_textchanged', {'rows': 20})
4564 call term_sendkeys(buf, ":so %\<cr>")
4565 call term_sendkeys(buf, "G")
4566 call WaitForAssert({-> assert_match('^XTest_autocmd_invalidates_undo_on_textchanged\s*$', term_getline(buf, 20))}, 1000)
4567
4568 call StopVimInTerminal(buf)
4569endfunc
4570
Christian Brabandt55f8bba2024-02-28 23:32:00 +01004571func Test_autocmd_creates_new_buffer_on_bufleave()
4572 e a.txt
4573 e b.txt
4574 setlocal bufhidden=wipe
4575 autocmd BufLeave <buffer> diffsplit c.txt
4576 bn
4577 call assert_equal(1, winnr('$'))
4578 call assert_equal('a.txt', bufname('%'))
4579 bw a.txt
4580 bw c.txt
4581endfunc
4582
Colin Kennedye5f22802024-03-26 18:20:16 +01004583" Ensure `expected` was just recently written as a Vim session
4584func s:assert_session_path(expected)
4585 call assert_equal(a:expected, v:this_session)
4586endfunc
4587
4588" Check for `expected` after a session is written to-disk.
4589func s:watch_for_session_path(expected)
4590 execute 'autocmd SessionWritePost * ++once execute "call s:assert_session_path(\"'
4591 \ . a:expected
4592 \ . '\")"'
4593endfunc
4594
4595" Ensure v:this_session gets the full session path, if explicitly stated
4596func Test_explicit_session_absolute_path()
4597 %bwipeout!
4598
4599 let directory = getcwd()
4600
4601 let v:this_session = ""
4602 let name = "some_file.vim"
4603 let expected = fnamemodify(name, ":p")
4604 call s:watch_for_session_path(expected)
4605 execute "mksession! " .. expected
4606
4607 call delete(expected)
4608endfunc
4609
4610" Ensure v:this_session gets the full session path, if explicitly stated
4611func Test_explicit_session_relative_path()
4612 %bwipeout!
4613
4614 let directory = getcwd()
4615
4616 let v:this_session = ""
4617 let name = "some_file.vim"
4618 let expected = fnamemodify(name, ":p")
4619 call s:watch_for_session_path(expected)
4620 execute "mksession! " .. name
4621
4622 call delete(expected)
4623endfunc
4624
4625" Ensure v:this_session gets the full session path, if not specified
4626func Test_implicit_session()
4627 %bwipeout!
4628
4629 let directory = getcwd()
4630
4631 let v:this_session = ""
4632 let expected = fnamemodify("Session.vim", ":p")
4633 call s:watch_for_session_path(expected)
4634 mksession!
4635
4636 call delete(expected)
4637endfunc
4638
Christian Brabandt86032702024-03-31 18:38:09 +02004639" Test TextChangedI and TextChanged
zeertzjqc4226622024-04-03 22:38:07 +02004640func Test_Changed_ChangedI()
zeertzjq8eb75232024-04-01 14:46:20 +02004641 " Run this test in a terminal because it requires running the main loop.
zeertzjqc4226622024-04-03 22:38:07 +02004642 " Don't use CheckRunVimInTerminal as that will skip the test on Windows.
4643 CheckFeature terminal
4644 CheckNotGui
4645 " Starting a terminal to run Vim is always considered flaky.
4646 let g:test_is_flaky = 1
4647
Christian Brabandt86032702024-03-31 18:38:09 +02004648 call writefile(['one', 'two', 'three'], 'XTextChangedI2', 'D')
4649 let before =<< trim END
zeertzjqc4226622024-04-03 22:38:07 +02004650 set ttimeout ttimeoutlen=10
zeertzjq8eb75232024-04-01 14:46:20 +02004651 let [g:autocmd_n, g:autocmd_i] = ['','']
4652
4653 func TextChangedAutocmd(char)
4654 let g:autocmd_{tolower(a:char)} = a:char .. b:changedtick
zeertzjqc4226622024-04-03 22:38:07 +02004655 call writefile([$'{g:autocmd_n},{g:autocmd_i}'], 'XTextChangedI3')
zeertzjq8eb75232024-04-01 14:46:20 +02004656 endfunc
4657
4658 au TextChanged <buffer> :call TextChangedAutocmd('N')
4659 au TextChangedI <buffer> :call TextChangedAutocmd('I')
4660
Christian Brabandt86032702024-03-31 18:38:09 +02004661 nnoremap <CR> o<Esc>
zeertzjq4a653912024-04-04 21:33:36 +02004662 autocmd SafeState * ++once call writefile([''], 'XTextChangedI3')
Christian Brabandt86032702024-03-31 18:38:09 +02004663 END
4664
4665 call writefile(before, 'Xinit', 'D')
zeertzjqc4226622024-04-03 22:38:07 +02004666 let buf = term_start(
4667 \ GetVimCommandCleanTerm() .. '-n -S Xinit XTextChangedI2',
4668 \ {'term_rows': 10})
4669 call assert_equal('running', term_getstatus(buf))
zeertzjq8eb75232024-04-01 14:46:20 +02004670 call WaitForAssert({-> assert_true(filereadable('XTextChangedI3'))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004671 defer delete('XTextChangedI3')
zeertzjq4a653912024-04-04 21:33:36 +02004672 call WaitForAssert({-> assert_equal([''], readfile('XTextChangedI3'))})
Christian Brabandt86032702024-03-31 18:38:09 +02004673
zeertzjqc4226622024-04-03 22:38:07 +02004674 " TextChanged should trigger if a mapping enters and leaves Insert mode.
4675 call term_sendkeys(buf, "\<CR>")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004676 call WaitForAssert({-> assert_equal('N4,', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004677
4678 call term_sendkeys(buf, "i")
4679 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004680 call WaitForAssert({-> assert_equal('N4,', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004681 " TextChangedI should trigger if change is done in Insert mode.
4682 call term_sendkeys(buf, "f")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004683 call WaitForAssert({-> assert_equal('N4,I5', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004684 call term_sendkeys(buf, "o")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004685 call WaitForAssert({-> assert_equal('N4,I6', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004686 call term_sendkeys(buf, "o")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004687 call WaitForAssert({-> assert_equal('N4,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004688 " TextChanged shouldn't trigger when leaving Insert mode and TextChangedI
4689 " has been triggered.
4690 call term_sendkeys(buf, "\<Esc>")
4691 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004692 call WaitForAssert({-> assert_equal('N4,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004693
4694 " TextChanged should trigger if change is done in Normal mode.
4695 call term_sendkeys(buf, "yyp")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004696 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004697
4698 " TextChangedI shouldn't trigger if change isn't done in Insert mode.
4699 call term_sendkeys(buf, "i")
4700 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004701 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004702 call term_sendkeys(buf, "\<Esc>")
4703 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004704 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004705
4706 " TextChangedI should trigger if change is a mix of Normal and Insert modes.
4707 func! s:validate_mixed_textchangedi(buf, keys)
4708 let buf = a:buf
4709 call term_sendkeys(buf, "ifoo")
4710 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
4711 call term_sendkeys(buf, "\<Esc>")
4712 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
4713 call term_sendkeys(buf, ":let [g:autocmd_n, g:autocmd_i] = ['', '']\<CR>")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004714 call writefile([], 'XTextChangedI3')
zeertzjqc4226622024-04-03 22:38:07 +02004715 call term_sendkeys(buf, a:keys)
4716 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004717 call WaitForAssert({-> assert_match('^,I\d\+', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004718 call term_sendkeys(buf, "\<Esc>")
4719 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004720 call WaitForAssert({-> assert_match('^,I\d\+', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004721 endfunc
4722
4723 call s:validate_mixed_textchangedi(buf, "o")
4724 call s:validate_mixed_textchangedi(buf, "O")
4725 call s:validate_mixed_textchangedi(buf, "ciw")
4726 call s:validate_mixed_textchangedi(buf, "cc")
4727 call s:validate_mixed_textchangedi(buf, "C")
4728 call s:validate_mixed_textchangedi(buf, "s")
4729 call s:validate_mixed_textchangedi(buf, "S")
4730
4731 " clean up
4732 bwipe!
Christian Brabandt86032702024-03-31 18:38:09 +02004733endfunc
4734
zeertzjq5bf6c212024-03-31 18:41:27 +02004735" Test that filetype detection still works when SwapExists autocommand sets
4736" filetype in another buffer.
4737func Test_SwapExists_set_other_buf_filetype()
4738 let lines =<< trim END
4739 set nocompatible directory=.
4740 filetype on
4741
4742 let g:buf = bufnr()
4743 new
4744
4745 func SwapExists()
4746 let v:swapchoice = 'o'
4747 call setbufvar(g:buf, '&filetype', 'text')
4748 endfunc
4749
4750 func SafeState()
4751 edit <script>
4752 redir! > XftSwapExists.out
4753 set readonly? filetype?
4754 redir END
4755 qall!
4756 endfunc
4757
4758 autocmd SwapExists * ++nested call SwapExists()
4759 autocmd SafeState * ++nested ++once call SafeState()
4760 END
4761 call writefile(lines, 'XftSwapExists.vim', 'D')
4762
4763 new XftSwapExists.vim
4764 if RunVim('', '', ' -S XftSwapExists.vim')
4765 call assert_equal(
4766 \ ['', ' readonly', ' filetype=vim'],
4767 \ readfile('XftSwapExists.out'))
4768 call delete('XftSwapExists.out')
4769 endif
4770
4771 bwipe!
4772endfunc
4773
4774" Test that file is not marked as modified when SwapExists autocommand sets
4775" 'modified' in another buffer.
4776func Test_SwapExists_set_other_buf_modified()
4777 let lines =<< trim END
4778 set nocompatible directory=.
4779
4780 let g:buf = bufnr()
4781 new
4782
4783 func SwapExists()
4784 let v:swapchoice = 'o'
4785 call setbufvar(g:buf, '&modified', 1)
4786 endfunc
4787
4788 func SafeState()
4789 edit <script>
4790 redir! > XmodSwapExists.out
4791 set readonly? modified?
4792 redir END
4793 qall!
4794 endfunc
4795
4796 autocmd SwapExists * ++nested call SwapExists()
4797 autocmd SafeState * ++nested ++once call SafeState()
4798 END
4799 call writefile(lines, 'XmodSwapExists.vim', 'D')
4800
4801 new XmodSwapExists.vim
4802 if RunVim('', '', ' -S XmodSwapExists.vim')
4803 call assert_equal(
4804 \ ['', ' readonly', 'nomodified'],
4805 \ readfile('XmodSwapExists.out'))
4806 call delete('XmodSwapExists.out')
4807 endif
4808
4809 bwipe!
4810endfunc
4811
Jaehwang Jungeb80b832024-04-26 18:48:48 +02004812func Test_BufEnter_botline()
4813 set hidden
4814 call writefile(range(10), 'Xxx1', 'D')
4815 call writefile(range(20), 'Xxx2', 'D')
4816 edit Xxx1
4817 edit Xxx2
4818 au BufEnter Xxx1 call assert_true(line('w$') > 1)
4819 edit Xxx1
zeertzjq340643e2024-04-27 11:33:24 +02004820
4821 bwipe! Xxx1
4822 bwipe! Xxx2
Jaehwang Jungeb80b832024-04-26 18:48:48 +02004823 au! BufEnter Xxx1
4824 set hidden&vim
4825endfunc
4826
Shougo Matsushita83678842024-07-11 22:05:12 +02004827func Test_KeyInputPre()
4828 " Consume previous keys
4829 call feedkeys('', 'ntx')
4830
4831 " KeyInputPre can record input keys.
4832 let s:keys = []
4833 au KeyInputPre n call add(s:keys, v:char)
4834
4835 call feedkeys('jkjkjjj', 'ntx')
4836 call assert_equal(
4837 \ ['j', 'k', 'j', 'k', 'j', 'j', 'j'],
4838 \ s:keys)
4839
4840 unlet s:keys
4841 au! KeyInputPre
4842
4843 " KeyInputPre can handle multibyte.
4844 let s:keys = []
4845 au KeyInputPre * call add(s:keys, v:char)
4846 edit Xxx1
4847
4848 call feedkeys("iあ\<ESC>", 'ntx')
4849 call assert_equal(['i', "あ", "\<ESC>"], s:keys)
4850
4851 bwipe! Xxx1
4852 unlet s:keys
4853 au! KeyInputPre
4854
4855 " KeyInputPre can change input keys.
4856 au KeyInputPre i if v:char ==# 'a' | let v:char = 'b' | endif
4857 edit Xxx1
4858
4859 call feedkeys("iaabb\<ESC>", 'ntx')
4860 call assert_equal(getline('.'), 'bbbb')
4861
4862 bwipe! Xxx1
4863 au! KeyInputPre
4864
4865 " KeyInputPre returns multiple characters.
4866 au KeyInputPre i if v:char ==# 'a' | let v:char = 'cccc' | endif
4867 edit Xxx1
4868
4869 call feedkeys("iaabb\<ESC>", 'ntx')
4870 call assert_equal(getline('.'), 'ccbb')
4871
4872 bwipe! Xxx1
4873 au! KeyInputPre
4874
4875 " KeyInputPre can use special keys.
4876 au KeyInputPre i if v:char ==# 'a' | let v:char = "\<Ignore>" | endif
4877 edit Xxx1
4878
4879 call feedkeys("iaabb\<ESC>", 'ntx')
4880 call assert_equal(getline('.'), 'bb')
4881
4882 bwipe! Xxx1
4883 au! KeyInputPre
4884
4885 " Test for v:event.typed
4886 au KeyInputPre n call assert_true(v:event.typed)
4887 call feedkeys('j', 'ntx')
4888
4889 au! KeyInputPre
4890
4891 au KeyInputPre n call assert_false(v:event.typed)
4892 call feedkeys('j', 'nx')
4893
4894 au! KeyInputPre
Shougo Matsushitafcc1b572024-07-17 20:25:22 +02004895
4896 " Test for v:event.typedchar
4897 nnoremap j k
4898 au KeyInputPre n
4899 \ call assert_equal(v:event.typedchar, 'j')
4900 \ | call assert_equal(v:char, 'k')
4901 call feedkeys('j', 'tx')
4902
4903 au! KeyInputPre
Shougo Matsushita83678842024-07-11 22:05:12 +02004904endfunc
4905
Christian Brabandtfb3f9692024-08-11 20:09:17 +02004906" those commands caused null pointer access, see #15464
4907func Test_WinNewPre_crash()
4908 defer CleanUpTestAuGroup()
4909 let _cmdheight=&cmdheight
4910 augroup testing
4911 au!
4912 autocmd WinNewPre * redraw
4913 augroup END
4914 tabnew
4915 tabclose
4916 augroup testing
4917 au!
4918 autocmd WinNewPre * wincmd t
4919 augroup END
4920 tabnew
4921 tabclose
4922 augroup testing
4923 au!
4924 autocmd WinNewPre * wincmd b
4925 augroup END
4926 tabnew
4927 tabclose
4928 augroup testing
4929 au!
4930 autocmd WinNewPre * set cmdheight+=1
4931 augroup END
4932 tabnew
4933 tabclose
4934 let &cmdheight=_cmdheight
4935endfunc
4936
Christian Brabandt84e31752024-09-02 09:59:18 +02004937" The specifics of the turkish locale may
4938" cause that Vim will not treat the GuiEnter autocommand
4939" as case insensitive and instead issues an error
4940func Test_GuiEnter_Turkish_locale()
4941 try
4942 let lng = v:lang
4943 lang tr_TR.UTF-8
4944 let result = execute(':au GuiEnter')
Ken Takataeccc9272024-09-03 23:01:55 +02004945 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02004946 let result = execute(':au GUIENTER')
Ken Takataeccc9272024-09-03 23:01:55 +02004947 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02004948 let result = execute(':au guienter')
Ken Takataeccc9272024-09-03 23:01:55 +02004949 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02004950 exe ":lang" lng
4951 catch /E197:/
4952 " can't use Turkish locale
4953 throw 'Skipped: Turkish locale not available'
4954 endtry
4955endfunc
Christian Brabandtfb3f9692024-08-11 20:09:17 +02004956
Christian Brabandt51b62382024-10-06 17:31:10 +02004957" This was using freed memory
4958func Test_autocmd_BufWinLeave_with_vsp()
4959 new
4960 let fname = 'XXXBufWinLeaveUAF.txt'
4961 let dummy = 'XXXDummy.txt'
4962 call writefile([], fname)
4963 call writefile([], dummy)
4964 defer delete(fname)
4965 defer delete(dummy)
4966 exe "e " fname
4967 vsp
4968 augroup testing
4969 exe "au BufWinLeave " .. fname .. " :e " dummy .. "| vsp " .. fname
4970 augroup END
4971 bw
4972 call CleanUpTestAuGroup()
4973 exe "bw! " .. dummy
4974endfunc
4975
Luuk van Baale15cbc12025-01-04 17:18:08 +01004976func Test_OptionSet_cmdheight()
4977 set mouse=a laststatus=2
4978 au OptionSet cmdheight :let &l:ch = v:option_new
4979
4980 resize -1
4981 call assert_equal(2, &l:ch)
4982 resize +1
4983 call assert_equal(1, &l:ch)
4984
4985 call test_setmouse(&lines - 1, 1)
4986 call feedkeys("\<LeftMouse>", 'xt')
4987 call test_setmouse(&lines - 2, 1)
4988 call feedkeys("\<LeftDrag>", 'xt')
4989 call assert_equal(2, &l:ch)
4990
4991 tabnew | resize +1
4992 call assert_equal(1, &l:ch)
4993 tabfirst
4994 call assert_equal(2, &l:ch)
4995
4996 tabonly
4997 set cmdheight& mouse& laststatus&
4998endfunc
4999
Luuk van Baalb7147f82025-02-08 18:52:39 +01005000func Test_eventignorewin()
5001 defer CleanUpTestAuGroup()
5002 augroup testing
5003 au WinEnter * :call add(g:evs, ["WinEnter", expand("<afile>")])
5004 au WinLeave * :call add(g:evs, ["WinLeave", expand("<afile>")])
5005 au BufWinEnter * :call add(g:evs, ["BufWinEnter", expand("<afile>")])
5006 augroup END
5007
5008 let g:evs = []
5009 set eventignorewin=WinLeave,WinEnter
5010 split foo
5011 call assert_equal([['BufWinEnter', 'foo']], g:evs)
5012 set eventignorewin=all
5013 edit bar
5014 call assert_equal([['BufWinEnter', 'foo']], g:evs)
5015 set eventignorewin=
5016 wincmd w
5017 call assert_equal([['BufWinEnter', 'foo'], ['WinLeave', 'bar']], g:evs)
5018
5019 only!
5020 %bwipe!
5021 set eventignorewin&
5022 unlet g:evs
5023endfunc
5024
5025func Test_WinScrolled_Resized_eiw()
5026 CheckRunVimInTerminal
5027
5028 let lines =<< trim END
5029 call setline(1, ['foo']->repeat(32))
5030 set eventignorewin=WinScrolled,WinResized
5031 split
5032 let [g:afile,g:resized,g:scrolled] = ['none',0,0]
5033 au WinScrolled * let [g:afile,g:scrolled] = [expand('<afile>'),g:scrolled+1]
5034 au WinResized * let [g:afile,g:resized] = [expand('<afile>'),g:resized+1]
5035 END
Christian Brabandtbfc77192025-02-11 20:03:10 +01005036 call writefile(lines, 'Xtest_winscrolled_eiw', 'D')
5037 let buf = RunVimInTerminal('-S Xtest_winscrolled_eiw', {'rows': 10})
Luuk van Baalb7147f82025-02-08 18:52:39 +01005038
5039 " Both windows are ignoring resize events
5040 call term_sendkeys(buf, "\<C-W>-")
5041 call TermWait(buf)
5042 call term_sendkeys(buf, ":echo g:afile g:resized g:scrolled\<CR>")
5043 call WaitForAssert({-> assert_equal('none 0 0', term_getline(buf, 10))}, 1000)
5044
5045 " And scroll events
5046 call term_sendkeys(buf, "Ggg")
5047 call TermWait(buf)
5048 call term_sendkeys(buf, ":echo g:afile g:resized g:scrolled\<CR>")
5049 call WaitForAssert({-> assert_equal('none 0 0', term_getline(buf, 10))}, 1000)
5050
5051 " Un-ignore events in second window, make first window current and resize
5052 call term_sendkeys(buf, ":set eventignorewin=\<CR>\<C-W>w\<C-W>+")
5053 call TermWait(buf)
5054 call term_sendkeys(buf, ":echo win_getid() g:afile g:resized g:scrolled\<CR>")
5055 call WaitForAssert({-> assert_equal('1000 1001 1 1', term_getline(buf, 10))}, 1000)
5056
5057 call StopVimInTerminal(buf)
5058endfunc
5059
Jim Zhou5606ca52025-03-13 21:58:25 +01005060" Test that TabClosedPre and TabClosed are triggered when closing a tab.
5061func Test_autocmd_tabclosedpre()
5062 augroup testing
5063 au TabClosedPre * call add(g:tabpagenr_pre, t:testvar)
5064 au TabClosed * call add(g:tabpagenr_post, t:testvar)
5065 augroup END
5066
5067 " Test 'tabclose' triggering
5068 let g:tabpagenr_pre = []
5069 let g:tabpagenr_post = []
5070 let t:testvar = 1
5071 tabnew
5072 let t:testvar = 2
5073 tabnew
5074 let t:testvar = 3
5075 tabnew
5076 let t:testvar = 4
5077 tabnext
5078 tabclose
5079 tabclose
5080 tabclose
5081 call assert_equal([1, 2, 3], g:tabpagenr_pre)
5082 call assert_equal([2, 3, 4], g:tabpagenr_post)
5083
5084 " Test 'tabclose {count}' triggering
5085 let g:tabpagenr_pre = []
5086 let g:tabpagenr_post = []
5087 let t:testvar = 1
5088 tabnew
5089 let t:testvar = 2
5090 tabnew
5091 let t:testvar = 3
5092 tabclose 2
5093 tabclose 2
5094 call assert_equal([2, 3], g:tabpagenr_pre)
5095 call assert_equal([3, 1], g:tabpagenr_post)
5096
5097 " Test 'tabonly' triggering
5098 let g:tabpagenr_pre = []
5099 let g:tabpagenr_post = []
5100 let t:testvar = 1
5101 tabnew
5102 let t:testvar = 2
5103 tabonly
5104 call assert_equal([1], g:tabpagenr_pre)
5105 call assert_equal([2], g:tabpagenr_post)
5106
5107 " Test 'q' and 'close' triggering (closing the last window in a tab)
5108 let g:tabpagenr_pre = []
5109 let g:tabpagenr_post = []
5110 split
5111 let t:testvar = 1
5112 tabnew
5113 let t:testvar = 2
5114 split
5115 vsplit
5116 tabnew
5117 let t:testvar = 3
5118 tabnext
5119 only
5120 quit
5121 quit
5122 close
5123 close
5124 call assert_equal([1, 2], g:tabpagenr_pre)
5125 call assert_equal([2, 3], g:tabpagenr_post)
5126
5127 func ClearAutomcdAndCreateTabs()
5128 au! TabClosedPre
5129 bw!
5130 e Z
5131 tabonly
5132 tabnew A
5133 tabnew B
5134 tabnew C
5135 endfunc
5136
5137 func GetTabs()
5138 redir => tabsout
5139 tabs
5140 redir END
5141 let tabsout = substitute(tabsout, '\n', '', 'g')
5142 let tabsout = substitute(tabsout, 'Tab page ', '', 'g')
5143 let tabsout = substitute(tabsout, ' ', '', 'g')
5144 return tabsout
5145 endfunc
5146
5147 call CleanUpTestAuGroup()
5148
5149 " Close tab in TabClosedPre autocmd
5150 call ClearAutomcdAndCreateTabs()
5151 au TabClosedPre * tabclose
zeertzjq67fe77d2025-04-20 10:21:18 +02005152 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005153 call ClearAutomcdAndCreateTabs()
5154 au TabClosedPre * tabclose
zeertzjq67fe77d2025-04-20 10:21:18 +02005155 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005156 call ClearAutomcdAndCreateTabs()
5157 au TabClosedPre * tabclose 1
zeertzjq67fe77d2025-04-20 10:21:18 +02005158 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005159
5160 " Close other (all) tabs in TabClosedPre autocmd
5161 call ClearAutomcdAndCreateTabs()
5162 au TabClosedPre * tabonly
zeertzjq67fe77d2025-04-20 10:21:18 +02005163 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005164 call ClearAutomcdAndCreateTabs()
5165 au TabClosedPre * tabonly
zeertzjq67fe77d2025-04-20 10:21:18 +02005166 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005167 call ClearAutomcdAndCreateTabs()
5168 au TabClosedPre * tabclose 4
zeertzjq67fe77d2025-04-20 10:21:18 +02005169 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005170
5171 " Open new tabs in TabClosedPre autocmd
5172 call ClearAutomcdAndCreateTabs()
5173 au TabClosedPre * tabnew D
zeertzjq67fe77d2025-04-20 10:21:18 +02005174 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005175 call ClearAutomcdAndCreateTabs()
5176 au TabClosedPre * tabnew D
zeertzjq67fe77d2025-04-20 10:21:18 +02005177 call assert_fails('tabclose 1', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005178
5179 " Moving the tab page in TabClosedPre autocmd
5180 call ClearAutomcdAndCreateTabs()
5181 au TabClosedPre * tabmove 0
5182 tabclose
Jim Zhoubcf66e02025-03-16 20:24:57 +01005183 call assert_equal('1>Z2A3B', GetTabs())
Jim Zhou5606ca52025-03-13 21:58:25 +01005184 call ClearAutomcdAndCreateTabs()
5185 au TabClosedPre * tabmove 0
5186 tabclose 1
5187 call assert_equal('1A2B3>C', GetTabs())
5188 tabonly
5189 call assert_equal('1>C', GetTabs())
5190
5191 " Switching tab page in TabClosedPre autocmd
5192 call ClearAutomcdAndCreateTabs()
5193 au TabClosedPre * tabnext | e Y
5194 tabclose
5195 call assert_equal('1Y2A3>B', GetTabs())
5196 call ClearAutomcdAndCreateTabs()
5197 au TabClosedPre * tabnext | e Y
5198 tabclose 1
5199 call assert_equal('1Y2B3>C', GetTabs())
5200 tabonly
5201 call assert_equal('1>Y', GetTabs())
5202
5203 " Create new windows in TabClosedPre autocmd
5204 call ClearAutomcdAndCreateTabs()
5205 au TabClosedPre * split | e X| vsplit | e Y | split | e Z
zeertzjq67fe77d2025-04-20 10:21:18 +02005206 call assert_fails('tabclose', 'E242:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005207 call ClearAutomcdAndCreateTabs()
5208 au TabClosedPre * new X | new Y | new Z
zeertzjq67fe77d2025-04-20 10:21:18 +02005209 call assert_fails('tabclose 1', 'E242:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005210
Jim Zhoubcf66e02025-03-16 20:24:57 +01005211 " Test directly closing the tab page with ':tabclose'
5212 au!
5213 tabonly
5214 bw!
5215 e Z
5216 au TabClosedPre * mksession!
5217 tabnew A
5218 sp
5219 tabclose
5220 source Session.vim
5221 call assert_equal('1Z2>AA', GetTabs())
5222
5223 " Test directly closing the tab page with ':tabonly'
5224 " Z is closed before A. Hence A overwrites the session.
5225 au!
5226 tabonly
5227 bw!
5228 e Z
5229 au TabClosedPre * mksession!
5230 tabnew A
5231 tabnew B
5232 tabonly
5233 source Session.vim
5234 call assert_equal('1>A2B', GetTabs())
5235
Jim Zhou5606ca52025-03-13 21:58:25 +01005236 " Clean up
Jim Zhoubcf66e02025-03-16 20:24:57 +01005237 call delete('Session.vim')
Jim Zhou5606ca52025-03-13 21:58:25 +01005238 au!
5239 only
5240 tabonly
5241 bw!
5242endfunc
5243
Bram Moolenaarbc2b71d2020-02-17 21:33:30 +01005244" vim: shiftwidth=2 sts=2 expandtab