blob: d5d516ce24f08fe6ff5211eeaf84121187742852 [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"
Girish Palya612f63b2025-04-28 18:29:52 +02002005 autocmd CmdlineLeave : let g:log2 = "CmdlineLeave"
Girish Palya92f68e22025-04-21 11:12:41 +02002006 new
2007 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002008 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002009 nnoremap <F1> <Cmd>echo "hello"<CR>
2010 call feedkeys("\<F1>", 'x')
2011 call assert_equal('', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002012 call assert_equal('', g:log2)
Girish Palya92f68e22025-04-21 11:12:41 +02002013 nunmap <F1>
Girish Palya46755e62025-04-27 19:28:06 +02002014
Girish Palya92f68e22025-04-21 11:12:41 +02002015 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002016 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002017 nnoremap <F1> :echo "hello"<CR>
2018 call feedkeys("\<F1>", 'x')
2019 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002020 call assert_equal('CmdlineLeave', g:log2)
Girish Palya92f68e22025-04-21 11:12:41 +02002021 nunmap <F1>
Girish Palya46755e62025-04-27 19:28:06 +02002022
Girish Palya92f68e22025-04-21 11:12:41 +02002023 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002024 let g:log2 = ''
2025 call feedkeys(":\<bs>", "tx")
2026 call assert_equal('CmdlineLeavePre', g:log)
2027 call assert_equal('CmdlineLeave', g:log2)
2028
2029 let g:log = ''
2030 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002031 split
2032 call assert_equal('', g:log)
2033 call feedkeys(":echo hello", "tx")
2034 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002035 call assert_equal('CmdlineLeave', g:log2)
2036
Girish Palya92f68e22025-04-21 11:12:41 +02002037 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002038 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002039 close
2040 call assert_equal('', g:log)
2041 call feedkeys(":echo hello", "tx")
2042 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002043 call assert_equal('CmdlineLeave', g:log2)
2044
Girish Palya92f68e22025-04-21 11:12:41 +02002045 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002046 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002047 tabnew
2048 call assert_equal('', g:log)
2049 call feedkeys(":echo hello", "tx")
2050 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002051 call assert_equal('CmdlineLeave', g:log2)
2052
Girish Palya92f68e22025-04-21 11:12:41 +02002053 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002054 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002055 split
2056 call assert_equal('', g:log)
2057 call feedkeys(":echo hello", "tx")
2058 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002059 call assert_equal('CmdlineLeave', g:log2)
2060
Girish Palya92f68e22025-04-21 11:12:41 +02002061 let g:log = ''
Girish Palya46755e62025-04-27 19:28:06 +02002062 let g:log2 = ''
Girish Palya92f68e22025-04-21 11:12:41 +02002063 tabclose
2064 call assert_equal('', g:log)
2065 call feedkeys(":echo hello", "tx")
2066 call assert_equal('CmdlineLeavePre', g:log)
Girish Palya46755e62025-04-27 19:28:06 +02002067 call assert_equal('CmdlineLeave', g:log2)
2068
zeertzjq92403692025-04-28 18:04:00 +02002069 autocmd CmdlineLeavePre * let g:cmdline += [getcmdline()]
2070
2071 for end_keys in ["\<CR>", "\<NL>", "\<kEnter>", "\<C-C>", "\<Esc>",
2072 \ "\<C-\>\<C-N>", "\<C-\>\<C-G>"]
2073 let g:cmdline = []
2074 let g:log = ''
2075 let g:log2 = ''
2076 call assert_equal('', g:log)
2077 let keys = $':echo "hello"{end_keys}'
2078 let msg = keytrans(keys)
2079 call feedkeys(keys, "tx")
2080 call assert_equal(['echo "hello"'], g:cmdline, msg)
2081 call assert_equal('CmdlineLeavePre', g:log, msg)
2082 call assert_equal('CmdlineLeave', g:log2, msg)
2083 endfor
2084
2085 let g:cmdline = []
2086 call feedkeys(":let c = input('? ')\<cr>ABCDE\<cr>", "tx")
2087 call assert_equal(["let c = input('? ')", 'ABCDE'], g:cmdline)
2088
2089 au! CmdlineLeavePre
2090 unlet! g:cmdline
Girish Palya5c3d1e32025-04-22 19:52:16 +02002091 unlet! g:log
Girish Palya46755e62025-04-27 19:28:06 +02002092 unlet! g:log2
Girish Palya92f68e22025-04-21 11:12:41 +02002093 bw!
2094endfunc
2095
Girish Palya46755e62025-04-27 19:28:06 +02002096" Ensure :cabbr does not cause a spurious CmdlineLeavePre.
2097func Test_CmdlineLeavePre_cabbr()
Girish Palya6220bba2025-04-28 18:00:40 +02002098 " For unknown reason this fails intermittently on MS-Windows
2099 CheckNotMSWindows
Girish Palya46755e62025-04-27 19:28:06 +02002100 CheckFeature terminal
2101 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2102 call assert_equal('running', term_getstatus(buf))
2103 call term_sendkeys(buf, ":let g:a=0\<cr>")
2104 call term_wait(buf, 50)
2105 call term_sendkeys(buf, ":cabbr v v\<cr>")
2106 call term_wait(buf, 50)
2107 call term_sendkeys(buf, ":command! -nargs=* Foo echo\<cr>")
2108 call term_wait(buf, 50)
2109 call term_sendkeys(buf, ":au! CmdlineLeavePre * :let g:a+=1\<cr>")
2110 call term_wait(buf, 50)
2111 call term_sendkeys(buf, ":Foo v\<cr>")
2112 call term_wait(buf, 50)
2113 call term_sendkeys(buf, ":echo g:a\<cr>")
2114 call term_wait(buf, 50)
2115 call WaitForAssert({-> assert_match('^2.*$', term_getline(buf, 3))})
2116 bwipe!
2117endfunc
2118
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002119func Test_Cmdline()
Bram Moolenaar153b7042018-01-31 15:48:32 +01002120 au! CmdlineChanged : let g:text = getcmdline()
2121 let g:text = 0
2122 call feedkeys(":echom 'hello'\<CR>", 'xt')
2123 call assert_equal("echom 'hello'", g:text)
2124 au! CmdlineChanged
2125
2126 au! CmdlineChanged : let g:entered = expand('<afile>')
2127 let g:entered = 0
2128 call feedkeys(":echom 'hello'\<CR>", 'xt')
2129 call assert_equal(':', g:entered)
2130 au! CmdlineChanged
2131
zeertzjq412e0e42023-02-11 10:34:07 +00002132 autocmd CmdlineChanged : let g:log += [getcmdline()]
2133
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002134 let g:log = []
2135 cnoremap <F1> <Cmd>call setcmdline('ls')<CR>
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002136 call feedkeys(":\<F1>", 'xt')
2137 call assert_equal(['ls'], g:log)
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002138 cunmap <F1>
2139
zeertzjqaf9e28a2023-02-06 20:58:09 +00002140 let g:log = []
zeertzjqaf9e28a2023-02-06 20:58:09 +00002141 call feedkeys(":sign \<Tab>\<Tab>\<C-N>\<C-P>\<S-Tab>\<S-Tab>\<Esc>", 'xt')
2142 call assert_equal([
2143 \ 's',
2144 \ 'si',
2145 \ 'sig',
2146 \ 'sign',
2147 \ 'sign ',
2148 \ 'sign define',
2149 \ 'sign jump',
2150 \ 'sign list',
2151 \ 'sign jump',
2152 \ 'sign define',
2153 \ 'sign ',
2154 \ ], g:log)
2155 let g:log = []
2156 set wildmenu wildoptions+=pum
2157 call feedkeys(":sign \<S-Tab>\<PageUp>\<kPageUp>\<kPageDown>\<PageDown>\<Esc>", 'xt')
2158 call assert_equal([
2159 \ 's',
2160 \ 'si',
2161 \ 'sig',
2162 \ 'sign',
2163 \ 'sign ',
2164 \ 'sign unplace',
2165 \ 'sign jump',
2166 \ 'sign define',
2167 \ 'sign undefine',
2168 \ 'sign unplace',
2169 \ ], g:log)
2170 set wildmenu& wildoptions&
zeertzjq412e0e42023-02-11 10:34:07 +00002171
2172 let g:log = []
2173 let @r = 'abc'
2174 call feedkeys(":0\<C-R>r1\<C-R>\<C-O>r2\<C-R>\<C-R>r3\<Esc>", 'xt')
2175 call assert_equal([
2176 \ '0',
2177 \ '0a',
2178 \ '0ab',
2179 \ '0abc',
2180 \ '0abc1',
2181 \ '0abc1abc',
2182 \ '0abc1abc2',
2183 \ '0abc1abc2abc',
2184 \ '0abc1abc2abc3',
2185 \ ], g:log)
2186
zeertzjqaf9e28a2023-02-06 20:58:09 +00002187 unlet g:log
2188 au! CmdlineChanged
2189
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002190 au! CmdlineEnter : let g:entered = expand('<afile>')
2191 au! CmdlineLeave : let g:left = expand('<afile>')
Girish Palya92f68e22025-04-21 11:12:41 +02002192 au! CmdlineLeavePre : let g:leftpre = expand('<afile>')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002193 let g:entered = 0
2194 let g:left = 0
Girish Palya92f68e22025-04-21 11:12:41 +02002195 let g:leftpre = 0
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002196 call feedkeys(":echo 'hello'\<CR>", 'xt')
2197 call assert_equal(':', g:entered)
2198 call assert_equal(':', g:left)
Girish Palya92f68e22025-04-21 11:12:41 +02002199 call assert_equal(':', g:leftpre)
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002200 au! CmdlineEnter
2201 au! CmdlineLeave
Girish Palya92f68e22025-04-21 11:12:41 +02002202 au! CmdlineLeavePre
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002203
Bram Moolenaara4baf5b2018-04-22 13:27:44 +02002204 let save_shellslash = &shellslash
2205 set noshellslash
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002206 au! CmdlineEnter / let g:entered = expand('<afile>')
2207 au! CmdlineLeave / let g:left = expand('<afile>')
Girish Palya92f68e22025-04-21 11:12:41 +02002208 au! CmdlineLeavePre / let g:leftpre = expand('<afile>')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002209 let g:entered = 0
2210 let g:left = 0
Girish Palya92f68e22025-04-21 11:12:41 +02002211 let g:leftpre = 0
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002212 new
2213 call setline(1, 'hello')
2214 call feedkeys("/hello\<CR>", 'xt')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002215 call assert_equal('/', g:entered)
2216 call assert_equal('/', g:left)
Girish Palya92f68e22025-04-21 11:12:41 +02002217 call assert_equal('/', g:leftpre)
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002218 bwipe!
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002219 au! CmdlineEnter
2220 au! CmdlineLeave
Girish Palya92f68e22025-04-21 11:12:41 +02002221 au! CmdlineLeavePre
Bram Moolenaara4baf5b2018-04-22 13:27:44 +02002222 let &shellslash = save_shellslash
Shougo Matsushitad0952142024-06-20 22:05:16 +02002223
Girish Palya92f68e22025-04-21 11:12:41 +02002224 let g:left = "cancelled"
2225 let g:leftpre = "cancelled"
2226 au! CmdlineLeave : let g:left = "triggered"
2227 au! CmdlineLeavePre : let g:leftpre = "triggered"
2228 call feedkeys(":echo 'hello'\<esc>", 'xt')
2229 call assert_equal('triggered', g:left)
2230 call assert_equal('triggered', g:leftpre)
2231 let g:left = "cancelled"
2232 let g:leftpre = "cancelled"
2233 au! CmdlineLeave : let g:left = "triggered"
2234 call feedkeys(":echo 'hello'\<c-c>", 'xt')
2235 call assert_equal('triggered', g:left)
2236 call assert_equal('triggered', g:leftpre)
2237 au! CmdlineLeave
2238 au! CmdlineLeavePre
2239
zeertzjqbc6f9672024-06-21 07:51:40 +02002240 au! CursorMovedC : let g:pos += [getcmdpos()]
2241 let g:pos = []
zeertzjq81456202024-07-07 20:48:25 +02002242 call feedkeys(":foo bar baz\<C-W>\<C-W>\<C-W>\<Esc>", 'xt')
2243 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 9, 5, 1], g:pos)
2244 let g:pos = []
2245 call feedkeys(":hello\<C-B>\<Esc>", 'xt')
2246 call assert_equal([2, 3, 4, 5, 6, 1], g:pos)
2247 let g:pos = []
2248 call feedkeys(":hello\<C-U>\<Esc>", 'xt')
2249 call assert_equal([2, 3, 4, 5, 6, 1], g:pos)
2250 let g:pos = []
zeertzjqbc6f9672024-06-21 07:51:40 +02002251 call feedkeys(":hello\<Left>\<C-R>=''\<CR>\<Left>\<Right>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002252 call assert_equal([2, 3, 4, 5, 6, 5, 4, 5], g:pos)
zeertzjqbc6f9672024-06-21 07:51:40 +02002253 let g:pos = []
2254 call feedkeys(":12345678\<C-R>=setcmdpos(3)??''\<CR>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002255 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 3], g:pos)
zeertzjqbc6f9672024-06-21 07:51:40 +02002256 let g:pos = []
2257 call feedkeys(":12345678\<C-R>=setcmdpos(3)??''\<CR>\<Left>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002258 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 3, 2], g:pos)
Shougo Matsushitad0952142024-06-20 22:05:16 +02002259 au! CursorMovedC
2260
zeertzjqbc6f9672024-06-21 07:51:40 +02002261 " setcmdpos() is no-op inside an autocommand
2262 au! CursorMovedC : let g:pos += [getcmdpos()] | call setcmdpos(1)
2263 let g:pos = []
2264 call feedkeys(":hello\<Left>\<Left>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002265 call assert_equal([2, 3, 4, 5, 6, 5, 4], g:pos)
Shougo Matsushitad0952142024-06-20 22:05:16 +02002266 au! CursorMovedC
zeertzjqbc6f9672024-06-21 07:51:40 +02002267
2268 unlet g:entered
2269 unlet g:left
2270 unlet g:pos
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002271endfunc
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002272
2273" Test for BufWritePre autocommand that deletes or unloads the buffer.
2274func Test_BufWritePre()
2275 %bwipe
2276 au BufWritePre Xxx1 bunload
2277 au BufWritePre Xxx2 bwipe
2278
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002279 call writefile(['start of Xxx1', 'test', 'end of Xxx1'], 'Xxx1', 'D')
2280 call writefile(['start of Xxx2', 'test', 'end of Xxx2'], 'Xxx2', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002281
2282 edit Xtest
2283 e! Xxx2
2284 bdel Xtest
2285 e Xxx1
2286 " write it, will unload it and give an error msg
Bram Moolenaare2e40752020-09-04 21:18:46 +02002287 call assert_fails('w', 'E203:')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002288 call assert_equal('Xxx2', bufname('%'))
2289 edit Xtest
2290 e! Xxx2
2291 bwipe Xtest
2292 " write it, will delete the buffer and give an error msg
Bram Moolenaare2e40752020-09-04 21:18:46 +02002293 call assert_fails('w', 'E203:')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002294 call assert_equal('Xxx1', bufname('%'))
2295 au! BufWritePre
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002296endfunc
2297
2298" Test for BufUnload autocommand that unloads all the other buffers
2299func Test_bufunload_all()
Bram Moolenaarf08b0eb2021-10-16 13:00:14 +01002300 let g:test_is_flaky = 1
Christian Brabandtee17b6f2023-09-09 11:23:50 +02002301 call writefile(['Test file Xxx1'], 'Xxx1', 'D')
2302 call writefile(['Test file Xxx2'], 'Xxx2', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002303
Bram Moolenaarc79745a2019-05-20 22:12:34 +02002304 let content =<< trim [CODE]
2305 func UnloadAllBufs()
2306 let i = 1
2307 while i <= bufnr('$')
2308 if i != bufnr('%') && bufloaded(i)
2309 exe i . 'bunload'
2310 endif
2311 let i += 1
2312 endwhile
2313 endfunc
2314 au BufUnload * call UnloadAllBufs()
2315 au VimLeave * call writefile(['Test Finished'], 'Xout')
2316 edit Xxx1
2317 split Xxx2
2318 q
2319 [CODE]
2320
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002321 call writefile(content, 'Xbunloadtest', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002322
2323 call delete('Xout')
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002324 call system(GetVimCommandClean() .. ' -N --not-a-term -S Xbunloadtest')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002325 call assert_true(filereadable('Xout'))
2326
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002327 call delete('Xout')
2328endfunc
2329
2330" Some tests for buffer-local autocommands
2331func Test_buflocal_autocmd()
2332 let g:bname = ''
2333 edit xx
2334 au BufLeave <buffer> let g:bname = expand("%")
2335 " here, autocommand for xx should trigger.
2336 " but autocommand shall not apply to buffer named <buffer>.
2337 edit somefile
2338 call assert_equal('xx', g:bname)
2339 let g:bname = ''
2340 " here, autocommand shall be auto-deleted
2341 bwipe xx
2342 " autocmd should not trigger
2343 edit xx
2344 call assert_equal('', g:bname)
2345 " autocmd should not trigger
2346 edit somefile
2347 call assert_equal('', g:bname)
2348 enew
2349 unlet g:bname
2350endfunc
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002351
2352" Test for "*Cmd" autocommands
2353func Test_Cmd_Autocmds()
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002354 call writefile(['start of Xxx', "\tabc2", 'end of Xxx'], 'Xxx', 'D')
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002355
2356 enew!
2357 au BufReadCmd XtestA 0r Xxx|$del
2358 edit XtestA " will read text of Xxd instead
2359 call assert_equal('start of Xxx', getline(1))
2360
2361 au BufWriteCmd XtestA call append(line("$"), "write")
2362 write " will append a line to the file
2363 call assert_equal('write', getline('$'))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002364 call assert_fails('read XtestA', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002365 call assert_equal('write', getline(4))
2366
2367 " now we have:
2368 " 1 start of Xxx
2369 " 2 abc2
2370 " 3 end of Xxx
2371 " 4 write
2372
2373 au FileReadCmd XtestB '[r Xxx
2374 2r XtestB " will read Xxx below line 2 instead
2375 call assert_equal('start of Xxx', getline(3))
2376
2377 " now we have:
2378 " 1 start of Xxx
2379 " 2 abc2
2380 " 3 start of Xxx
2381 " 4 abc2
2382 " 5 end of Xxx
2383 " 6 end of Xxx
2384 " 7 write
2385
2386 au FileWriteCmd XtestC '[,']copy $
2387 normal 4GA1
2388 4,5w XtestC " will copy lines 4 and 5 to the end
2389 call assert_equal("\tabc21", getline(8))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002390 call assert_fails('r XtestC', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002391 call assert_equal("end of Xxx", getline(9))
2392
2393 " now we have:
2394 " 1 start of Xxx
2395 " 2 abc2
2396 " 3 start of Xxx
2397 " 4 abc21
2398 " 5 end of Xxx
2399 " 6 end of Xxx
2400 " 7 write
2401 " 8 abc21
2402 " 9 end of Xxx
2403
2404 let g:lines = []
2405 au FileAppendCmd XtestD call extend(g:lines, getline(line("'["), line("']")))
2406 w >>XtestD " will add lines to 'lines'
2407 call assert_equal(9, len(g:lines))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002408 call assert_fails('$r XtestD', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002409 call assert_equal(9, line('$'))
2410 call assert_equal('end of Xxx', getline('$'))
2411
2412 au BufReadCmd XtestE 0r Xxx|$del
2413 sp XtestE " split window with test.out
2414 call assert_equal('end of Xxx', getline(3))
2415
2416 let g:lines = []
2417 exe "normal 2Goasdf\<Esc>\<C-W>\<C-W>"
2418 au BufWriteCmd XtestE call extend(g:lines, getline(0, '$'))
2419 wall " will write other window to 'lines'
2420 call assert_equal(4, len(g:lines), g:lines)
2421 call assert_equal('asdf', g:lines[2])
2422
2423 au! BufReadCmd
2424 au! BufWriteCmd
2425 au! FileReadCmd
2426 au! FileWriteCmd
2427 au! FileAppendCmd
2428 %bwipe!
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002429 enew!
2430endfunc
Bram Moolenaaraace2152017-11-05 16:23:10 +01002431
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002432func s:ReadFile()
2433 setl noswapfile nomodified
2434 let filename = resolve(expand("<afile>:p"))
2435 execute 'read' fnameescape(filename)
2436 1d_
2437 exe 'file' fnameescape(filename)
2438 setl buftype=acwrite
2439endfunc
2440
2441func s:WriteFile()
2442 let filename = resolve(expand("<afile>:p"))
2443 setl buftype=
2444 noautocmd execute 'write' fnameescape(filename)
2445 setl buftype=acwrite
2446 setl nomodified
2447endfunc
2448
2449func Test_BufReadCmd()
2450 autocmd BufReadCmd *.test call s:ReadFile()
2451 autocmd BufWriteCmd *.test call s:WriteFile()
2452
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002453 call writefile(['one', 'two', 'three'], 'Xcmd.test', 'D')
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002454 edit Xcmd.test
2455 call assert_match('Xcmd.test" line 1 of 3', execute('file'))
2456 normal! Gofour
2457 write
2458 call assert_equal(['one', 'two', 'three', 'four'], readfile('Xcmd.test'))
2459
2460 bwipe!
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002461 au! BufReadCmd
2462 au! BufWriteCmd
2463endfunc
2464
zeertzjq9c8f9462022-08-30 18:17:15 +01002465func Test_BufWriteCmd()
2466 autocmd BufWriteCmd Xbufwritecmd let g:written = 1
2467 new
2468 file Xbufwritecmd
2469 set buftype=acwrite
Bram Moolenaar6f14da12022-09-07 21:30:44 +01002470 call mkdir('Xbufwritecmd', 'D')
zeertzjq9c8f9462022-08-30 18:17:15 +01002471 write
2472 " BufWriteCmd should be triggered even if a directory has the same name
2473 call assert_equal(1, g:written)
zeertzjq9c8f9462022-08-30 18:17:15 +01002474 unlet g:written
2475 au! BufWriteCmd
2476 bwipe!
2477endfunc
2478
Bram Moolenaaraace2152017-11-05 16:23:10 +01002479func SetChangeMarks(start, end)
Bram Moolenaar97c69432021-01-15 16:45:21 +01002480 exe a:start .. 'mark ['
2481 exe a:end .. 'mark ]'
Bram Moolenaaraace2152017-11-05 16:23:10 +01002482endfunc
2483
2484" Verify the effects of autocmds on '[ and ']
2485func Test_change_mark_in_autocmds()
2486 edit! Xtest
Bram Moolenaar97c69432021-01-15 16:45:21 +01002487 call feedkeys("ia\<CR>b\<CR>c\<CR>d\<C-g>u\<Esc>", 'xtn')
Bram Moolenaaraace2152017-11-05 16:23:10 +01002488
2489 call SetChangeMarks(2, 3)
2490 write
2491 call assert_equal([1, 4], [line("'["), line("']")])
2492
2493 call SetChangeMarks(2, 3)
2494 au BufWritePre * call assert_equal([1, 4], [line("'["), line("']")])
2495 write
2496 au! BufWritePre
2497
Bram Moolenaar14ddd222020-08-05 12:02:40 +02002498 if has('unix')
Bram Moolenaaraace2152017-11-05 16:23:10 +01002499 write XtestFilter
2500 write >> XtestFilter
2501
2502 call SetChangeMarks(2, 3)
2503 " Marks are set to the entire range of the write
2504 au FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
2505 " '[ is adjusted to just before the line that will receive the filtered
2506 " data
2507 au FilterReadPre * call assert_equal([4, 4], [line("'["), line("']")])
2508 " The filtered data is read into the buffer, and the source lines are
2509 " still present, so the range is after the source lines
2510 au FilterReadPost * call assert_equal([5, 12], [line("'["), line("']")])
2511 %!cat XtestFilter
2512 " After the filtered data is read, the original lines are deleted
2513 call assert_equal([1, 8], [line("'["), line("']")])
2514 au! FilterWritePre,FilterReadPre,FilterReadPost
2515 undo
2516
2517 call SetChangeMarks(1, 4)
2518 au FilterWritePre * call assert_equal([2, 3], [line("'["), line("']")])
2519 au FilterReadPre * call assert_equal([3, 3], [line("'["), line("']")])
2520 au FilterReadPost * call assert_equal([4, 11], [line("'["), line("']")])
2521 2,3!cat XtestFilter
2522 call assert_equal([2, 9], [line("'["), line("']")])
2523 au! FilterWritePre,FilterReadPre,FilterReadPost
2524 undo
2525
2526 call delete('XtestFilter')
2527 endif
2528
2529 call SetChangeMarks(1, 4)
2530 au FileWritePre * call assert_equal([2, 3], [line("'["), line("']")])
2531 2,3write Xtest2
2532 au! FileWritePre
2533
2534 call SetChangeMarks(2, 3)
2535 au FileAppendPre * call assert_equal([1, 4], [line("'["), line("']")])
2536 write >> Xtest2
2537 au! FileAppendPre
2538
2539 call SetChangeMarks(1, 4)
2540 au FileAppendPre * call assert_equal([2, 3], [line("'["), line("']")])
2541 2,3write >> Xtest2
2542 au! FileAppendPre
2543
2544 call SetChangeMarks(1, 1)
2545 au FileReadPre * call assert_equal([3, 1], [line("'["), line("']")])
2546 au FileReadPost * call assert_equal([4, 11], [line("'["), line("']")])
2547 3read Xtest2
2548 au! FileReadPre,FileReadPost
2549 undo
2550
2551 call SetChangeMarks(4, 4)
2552 " When the line is 0, it's adjusted to 1
2553 au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
2554 au FileReadPost * call assert_equal([1, 8], [line("'["), line("']")])
2555 0read Xtest2
2556 au! FileReadPre,FileReadPost
2557 undo
2558
2559 call SetChangeMarks(4, 4)
2560 " When the line is 0, it's adjusted to 1
2561 au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
2562 au FileReadPost * call assert_equal([2, 9], [line("'["), line("']")])
2563 1read Xtest2
2564 au! FileReadPre,FileReadPost
2565 undo
2566
2567 bwipe!
2568 call delete('Xtest')
2569 call delete('Xtest2')
2570endfunc
2571
2572func Test_Filter_noshelltemp()
Bram Moolenaaraeb313f2020-11-27 19:13:28 +01002573 CheckExecutable cat
Bram Moolenaaraace2152017-11-05 16:23:10 +01002574
2575 enew!
2576 call setline(1, ['a', 'b', 'c', 'd'])
2577
2578 let shelltemp = &shelltemp
2579 set shelltemp
2580
2581 let g:filter_au = 0
2582 au FilterWritePre * let g:filter_au += 1
2583 au FilterReadPre * let g:filter_au += 1
2584 au FilterReadPost * let g:filter_au += 1
2585 %!cat
2586 call assert_equal(3, g:filter_au)
2587
2588 if has('filterpipe')
2589 set noshelltemp
2590
2591 let g:filter_au = 0
2592 au FilterWritePre * let g:filter_au += 1
2593 au FilterReadPre * let g:filter_au += 1
2594 au FilterReadPost * let g:filter_au += 1
2595 %!cat
2596 call assert_equal(0, g:filter_au)
2597 endif
2598
2599 au! FilterWritePre,FilterReadPre,FilterReadPost
2600 let &shelltemp = shelltemp
2601 bwipe!
2602endfunc
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002603
2604func Test_TextYankPost()
2605 enew!
2606 call setline(1, ['foo'])
2607
2608 let g:event = []
2609 au TextYankPost * let g:event = copy(v:event)
2610
2611 call assert_equal({}, v:event)
2612 call assert_fails('let v:event = {}', 'E46:')
2613 call assert_fails('let v:event.mykey = 0', 'E742:')
2614
2615 norm "ayiw
2616 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002617 \ #{regcontents: ['foo'], regname: 'a', operator: 'y',
2618 \ regtype: 'v', visual: v:false, inclusive: v:true},
2619 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002620 norm y_
2621 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002622 \ #{regcontents: ['foo'], regname: '', operator: 'y', regtype: 'V',
2623 \ visual: v:false, inclusive: v:false},
2624 \ g:event)
Bram Moolenaar37d16732020-06-12 22:09:01 +02002625 norm Vy
2626 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002627 \ #{regcontents: ['foo'], regname: '', operator: 'y', regtype: 'V',
2628 \ visual: v:true, inclusive: v:true},
2629 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002630 call feedkeys("\<C-V>y", 'x')
2631 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002632 \ #{regcontents: ['f'], regname: '', operator: 'y', regtype: "\x161",
2633 \ visual: v:true, inclusive: v:true},
2634 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002635 norm "xciwbar
2636 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002637 \ #{regcontents: ['foo'], regname: 'x', operator: 'c', regtype: 'v',
2638 \ visual: v:false, inclusive: v:true},
2639 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002640 norm "bdiw
2641 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002642 \ #{regcontents: ['bar'], regname: 'b', operator: 'd', regtype: 'v',
2643 \ visual: v:false, inclusive: v:true},
2644 \ g:event)
2645
2646 call setline(1, 'foobar')
2647 " exclusive motion
2648 norm $"ay0
2649 call assert_equal(
2650 \ #{regcontents: ['fooba'], regname: 'a', operator: 'y', regtype: 'v',
2651 \ visual: v:false, inclusive: v:false},
2652 \ g:event)
2653 " inclusive motion
2654 norm 0"ay$
2655 call assert_equal(
2656 \ #{regcontents: ['foobar'], regname: 'a', operator: 'y', regtype: 'v',
2657 \ visual: v:false, inclusive: v:true},
2658 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002659
2660 call assert_equal({}, v:event)
2661
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002662 if has('clipboard_working') && !has('gui_running')
2663 " Test that when the visual selection is automatically copied to clipboard
2664 " register a TextYankPost is emitted
2665 call setline(1, ['foobar'])
2666
2667 let @* = ''
2668 set clipboard=autoselect
2669 exe "norm! ggviw\<Esc>"
2670 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002671 \ #{regcontents: ['foobar'], regname: '*', operator: 'y',
2672 \ regtype: 'v', visual: v:true, inclusive: v:false},
2673 \ g:event)
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002674
2675 let @+ = ''
2676 set clipboard=autoselectplus
2677 exe "norm! ggviw\<Esc>"
2678 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002679 \ #{regcontents: ['foobar'], regname: '+', operator: 'y',
2680 \ regtype: 'v', visual: v:true, inclusive: v:false},
2681 \ g:event)
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002682
2683 set clipboard&vim
2684 endif
2685
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002686 au! TextYankPost
2687 unlet g:event
2688 bwipe!
2689endfunc
Bram Moolenaar9bca8052017-12-18 12:37:55 +01002690
Bram Moolenaar9a046fd2021-01-28 13:47:59 +01002691func Test_autocommand_all_events()
2692 call assert_fails('au * * bwipe', 'E1155:')
2693 call assert_fails('au * x bwipe', 'E1155:')
Bram Moolenaarb6db1462021-12-24 19:24:47 +00002694 call assert_fails('au! * x bwipe', 'E1155:')
Bram Moolenaar4fb921e2017-12-18 15:33:00 +01002695endfunc
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002696
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002697func Test_autocmd_user()
2698 au User MyEvent let s:res = [expand("<afile>"), expand("<amatch>")]
2699 doautocmd User MyEvent
2700 call assert_equal(['MyEvent', 'MyEvent'], s:res)
2701 au! User
2702 unlet s:res
2703endfunc
2704
Bram Moolenaar3b014be2022-11-13 17:53:46 +00002705func Test_autocmd_user_clear_group()
2706 CheckRunVimInTerminal
2707
2708 let lines =<< trim END
2709 autocmd! User
2710 for i in range(1, 999)
2711 exe 'autocmd User ' .. 'Foo' .. i .. ' bar'
2712 endfor
2713 au CmdlineLeave : call timer_start(0, {-> execute('autocmd! User')})
2714 END
2715 call writefile(lines, 'XautoUser', 'D')
2716 let buf = RunVimInTerminal('-S XautoUser', {'rows': 10})
2717
2718 " this was using freed memory
2719 call term_sendkeys(buf, ":autocmd User\<CR>")
2720 call TermWait(buf, 50)
2721 call term_sendkeys(buf, "G")
2722
2723 call StopVimInTerminal(buf)
2724endfunc
2725
Bram Moolenaaref2c3252022-11-25 16:31:51 +00002726func Test_autocmd_CmdlineLeave_unlet()
2727 CheckRunVimInTerminal
2728
2729 let lines =<< trim END
2730 for i in range(1, 999)
2731 exe 'let g:var' .. i '=' i
2732 endfor
2733 au CmdlineLeave : call timer_start(0, {-> execute('unlet g:var990')})
2734 END
2735 call writefile(lines, 'XleaveUnlet', 'D')
2736 let buf = RunVimInTerminal('-S XleaveUnlet', {'rows': 10})
2737
2738 " this was using freed memory
2739 call term_sendkeys(buf, ":let g:\<CR>")
2740 call TermWait(buf, 50)
2741 call term_sendkeys(buf, "G")
2742 call TermWait(buf, 50)
2743 call term_sendkeys(buf, "\<CR>") " for the hit-enter prompt
2744
2745 call StopVimInTerminal(buf)
2746endfunc
2747
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002748function s:Before_test_dirchanged()
2749 augroup test_dirchanged
2750 autocmd!
2751 augroup END
2752 let s:li = []
2753 let s:dir_this = getcwd()
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02002754 let s:dir_foo = s:dir_this . '/Xfoo'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002755 call mkdir(s:dir_foo)
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02002756 let s:dir_bar = s:dir_this . '/Xbar'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002757 call mkdir(s:dir_bar)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002758endfunc
2759
2760function s:After_test_dirchanged()
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002761 call chdir(s:dir_this)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002762 call delete(s:dir_foo, 'd')
2763 call delete(s:dir_bar, 'd')
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002764 augroup test_dirchanged
2765 autocmd!
2766 augroup END
2767endfunc
2768
2769function Test_dirchanged_global()
2770 call s:Before_test_dirchanged()
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002771 autocmd test_dirchanged DirChangedPre global call add(s:li, expand("<amatch>") .. " pre cd " .. v:event.directory)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002772 autocmd test_dirchanged DirChanged global call add(s:li, "cd:")
2773 autocmd test_dirchanged DirChanged global call add(s:li, expand("<afile>"))
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002774 call chdir(s:dir_foo)
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002775 let expected = ["global pre cd " .. s:dir_foo, "cd:", s:dir_foo]
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002776 call assert_equal(expected, s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002777 call chdir(s:dir_foo)
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002778 call assert_equal(expected, s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002779 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002780 call assert_equal(expected, s:li)
Bram Moolenaard8c9d322022-06-12 11:49:16 +01002781
2782 exe 'cd ' .. s:dir_foo
2783 exe 'cd ' .. s:dir_bar
2784 autocmd! test_dirchanged DirChanged global let g:result = expand("<afile>")
2785 cd -
Bram Moolenaardb77c492022-06-12 23:26:50 +01002786 call assert_equal(s:dir_foo, substitute(g:result, '\\', '/', 'g'))
Bram Moolenaard8c9d322022-06-12 11:49:16 +01002787
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002788 call s:After_test_dirchanged()
2789endfunc
2790
2791function Test_dirchanged_local()
2792 call s:Before_test_dirchanged()
2793 autocmd test_dirchanged DirChanged window call add(s:li, "lcd:")
2794 autocmd test_dirchanged DirChanged window call add(s:li, expand("<afile>"))
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002795 call chdir(s:dir_foo)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002796 call assert_equal([], s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002797 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002798 call assert_equal(["lcd:", s:dir_bar], s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002799 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002800 call assert_equal(["lcd:", s:dir_bar], s:li)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002801 call s:After_test_dirchanged()
2802endfunc
2803
2804function Test_dirchanged_auto()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02002805 CheckOption autochdir
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002806 call s:Before_test_dirchanged()
2807 call test_autochdir()
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002808 autocmd test_dirchanged DirChangedPre auto call add(s:li, "pre cd " .. v:event.directory)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002809 autocmd test_dirchanged DirChanged auto call add(s:li, "auto:")
2810 autocmd test_dirchanged DirChanged auto call add(s:li, expand("<afile>"))
2811 set acd
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002812 cd ..
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002813 call assert_equal([], s:li)
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01002814 exe 'edit ' . s:dir_foo . '/Xautofile'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002815 call assert_equal(s:dir_foo, getcwd())
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002816 let expected = ["pre cd " .. s:dir_foo, "auto:", s:dir_foo]
2817 call assert_equal(expected, s:li)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002818 set noacd
2819 bwipe!
2820 call s:After_test_dirchanged()
2821endfunc
Bram Moolenaar5a093432018-02-10 18:15:19 +01002822
2823" Test TextChangedI and TextChangedP
2824func Test_ChangedP()
2825 new
2826 call setline(1, ['foo', 'bar', 'foobar'])
2827 call test_override("char_avail", 1)
2828 set complete=. completeopt=menuone
2829
2830 func! TextChangedAutocmd(char)
2831 let g:autocmd .= a:char
2832 endfunc
2833
Christian Brabandtdb3b4462021-10-16 11:58:55 +01002834 " TextChanged will not be triggered, only check that it isn't.
Bram Moolenaar5a093432018-02-10 18:15:19 +01002835 au! TextChanged <buffer> :call TextChangedAutocmd('N')
2836 au! TextChangedI <buffer> :call TextChangedAutocmd('I')
2837 au! TextChangedP <buffer> :call TextChangedAutocmd('P')
2838
2839 call cursor(3, 1)
2840 let g:autocmd = ''
2841 call feedkeys("o\<esc>", 'tnix')
Evgeni Chasnovskid7ae2632023-10-15 09:59:00 +02002842 call assert_equal('I', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002843
2844 let g:autocmd = ''
Christian Brabandt4bca4892023-10-27 19:26:49 +02002845 call feedkeys("Sf", 'tnix')
2846 call assert_equal('II', g:autocmd)
2847
2848 let g:autocmd = ''
Bram Moolenaar5a093432018-02-10 18:15:19 +01002849 call feedkeys("Sf\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002850 call assert_equal('IIP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002851
2852 let g:autocmd = ''
2853 call feedkeys("Sf\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002854 call assert_equal('IIPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002855
2856 let g:autocmd = ''
2857 call feedkeys("Sf\<C-N>\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002858 call assert_equal('IIPPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002859
2860 let g:autocmd = ''
2861 call feedkeys("Sf\<C-N>\<C-N>\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002862 call assert_equal('IIPPPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002863
2864 call assert_equal(['foo', 'bar', 'foobar', 'foo'], getline(1, '$'))
2865 " TODO: how should it handle completeopt=noinsert,noselect?
2866
2867 " CleanUp
2868 call test_override("char_avail", 0)
2869 au! TextChanged
2870 au! TextChangedI
2871 au! TextChangedP
2872 delfu TextChangedAutocmd
2873 unlet! g:autocmd
2874 set complete&vim completeopt&vim
2875
2876 bw!
2877endfunc
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002878
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002879let g:setline_handled = v:false
Bram Moolenaar1e115362019-01-09 23:01:02 +01002880func SetLineOne()
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002881 if !g:setline_handled
2882 call setline(1, "(x)")
2883 let g:setline_handled = v:true
2884 endif
2885endfunc
2886
2887func Test_TextChangedI_with_setline()
2888 new
2889 call test_override('char_avail', 1)
2890 autocmd TextChangedI <buffer> call SetLineOne()
2891 call feedkeys("i(\<CR>\<Esc>", 'tx')
2892 call assert_equal('(', getline(1))
2893 call assert_equal('x)', getline(2))
2894 undo
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002895 call assert_equal('', getline(1))
Bram Moolenaar9fa95062018-08-08 22:08:32 +02002896 call assert_equal('', getline(2))
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002897
Bram Moolenaarca34db32022-01-20 11:17:18 +00002898 call test_override('char_avail', 0)
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002899 bwipe!
2900endfunc
2901
Christian Brabandtc9e79e52024-02-09 19:34:36 +01002902func Test_TextChanged_with_norm()
2903 " For unknown reason this fails on MS-Windows
2904 CheckNotMSWindows
2905 CheckFeature terminal
2906 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2907 call assert_equal('running', term_getstatus(buf))
2908 call term_sendkeys(buf, ":let g:a=0\<cr>")
2909 call term_wait(buf, 50)
2910 call term_sendkeys(buf, ":au! TextChanged * :let g:a+=1\<cr>")
2911 call term_wait(buf, 50)
2912 call term_sendkeys(buf, ":norm! ia\<cr>")
2913 call term_wait(buf, 50)
2914 call term_sendkeys(buf, ":echo g:a\<cr>")
2915 call term_wait(buf, 50)
2916 call WaitForAssert({-> assert_match('^1.*$', term_getline(buf, 3))})
2917 bwipe!
2918endfunc
2919
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002920func Test_Changed_FirstTime()
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02002921 CheckFeature terminal
2922 CheckNotGui
Bram Moolenaar3cdcb092020-03-18 19:18:10 +01002923 " Starting a terminal to run Vim is always considered flaky.
Bram Moolenaar30d53e22020-03-18 21:10:44 +01002924 let g:test_is_flaky = 1
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02002925
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002926 " Prepare file for TextChanged event.
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002927 call writefile([''], 'Xchanged.txt', 'D')
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002928 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2929 call assert_equal('running', term_getstatus(buf))
Bram Moolenaar1834d372018-03-29 17:40:46 +02002930 " Wait for the ruler (in the status line) to be shown.
Bram Moolenaaraa5df7e2019-02-03 14:53:10 +01002931 " In ConPTY, there is additional character which is drawn up to the width of
2932 " the screen.
2933 if has('conpty')
2934 call WaitForAssert({-> assert_match('\<All.*$', term_getline(buf, 3))})
2935 else
2936 call WaitForAssert({-> assert_match('\<All$', term_getline(buf, 3))})
2937 endif
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002938 " It's only adding autocmd, so that no event occurs.
2939 call term_sendkeys(buf, ":au! TextChanged <buffer> call writefile(['No'], 'Xchanged.txt')\<cr>")
2940 call term_sendkeys(buf, "\<C-\\>\<C-N>:qa!\<cr>")
Bram Moolenaar50182fa2018-04-28 21:34:40 +02002941 call WaitForAssert({-> assert_equal('finished', term_getstatus(buf))})
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002942 call assert_equal([''], readfile('Xchanged.txt'))
2943
2944 " clean up
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002945 bwipe!
2946endfunc
Bram Moolenaar0566e892019-01-24 19:37:40 +01002947
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002948func Test_autocmd_nested()
2949 let g:did_nested = 0
Christian Brabandtfb3f9692024-08-11 20:09:17 +02002950 defer CleanUpTestAuGroup()
2951 augroup testing
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002952 au WinNew * edit somefile
2953 au BufNew * let g:did_nested = 1
2954 augroup END
2955 split
2956 call assert_equal(0, g:did_nested)
2957 close
2958 bwipe! somefile
2959
2960 " old nested argument still works
Christian Brabandtfb3f9692024-08-11 20:09:17 +02002961 augroup testing
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002962 au!
2963 au WinNew * nested edit somefile
2964 au BufNew * let g:did_nested = 1
2965 augroup END
2966 split
2967 call assert_equal(1, g:did_nested)
2968 close
2969 bwipe! somefile
2970
2971 " New ++nested argument works
2972 augroup Testing
2973 au!
2974 au WinNew * ++nested edit somefile
2975 au BufNew * let g:did_nested = 1
2976 augroup END
2977 split
2978 call assert_equal(1, g:did_nested)
2979 close
2980 bwipe! somefile
2981
Bram Moolenaarf0775142022-03-04 20:10:38 +00002982 " nested without ++ does not work in Vim9 script
2983 call assert_fails('vim9cmd au WinNew * nested echo fails', 'E1078:')
2984
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002985 augroup Testing
2986 au!
2987 augroup END
2988
2989 call assert_fails('au WinNew * ++nested ++nested echo bad', 'E983:')
2990 call assert_fails('au WinNew * nested nested echo bad', 'E983:')
2991endfunc
2992
Bram Moolenaar5fa9f232022-07-23 09:06:48 +01002993func Test_autocmd_nested_cursor_invalid()
2994 set laststatus=0
2995 copen
2996 cclose
2997 call setline(1, ['foo', 'bar', 'baz'])
2998 3
2999 augroup nested_inv
3000 autocmd User foo ++nested copen
3001 autocmd BufAdd * let &laststatus = 2 - &laststatus
3002 augroup END
3003 doautocmd User foo
3004
3005 augroup nested_inv
3006 au!
3007 augroup END
3008 set laststatus&
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003009 cclose
Bram Moolenaar5fa9f232022-07-23 09:06:48 +01003010 bwipe!
3011endfunc
3012
Bram Moolenaar3d6ee8b2022-07-27 15:23:35 +01003013func Test_autocmd_nested_keeps_cursor_pos()
3014 enew
3015 call setline(1, 'foo')
3016 autocmd User foo ++nested normal! $a
3017 autocmd InsertLeave * :
3018 doautocmd User foo
3019 call assert_equal([0, 1, 3, 0], getpos('.'))
3020
3021 bwipe!
3022endfunc
3023
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003024func Test_autocmd_nested_switch_window()
3025 " run this in a separate Vim so that SafeState works
3026 CheckRunVimInTerminal
3027
3028 let lines =<< trim END
3029 vim9script
3030 ['()']->writefile('Xautofile')
3031 autocmd VimEnter * ++nested edit Xautofile | split
3032 autocmd BufReadPost * autocmd SafeState * ++once foldclosed('.')
3033 autocmd WinEnter * matchadd('ErrorMsg', 'pat')
3034 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003035 call writefile(lines, 'Xautoscript', 'D')
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003036 let buf = RunVimInTerminal('-S Xautoscript', {'rows': 10})
3037 call VerifyScreenDump(buf, 'Test_autocmd_nested_switch', {})
3038
3039 call StopVimInTerminal(buf)
3040 call delete('Xautofile')
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003041endfunc
3042
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02003043func Test_autocmd_once()
3044 " Without ++once WinNew triggers twice
3045 let g:did_split = 0
3046 augroup Testing
3047 au WinNew * let g:did_split += 1
3048 augroup END
3049 split
3050 split
3051 call assert_equal(2, g:did_split)
3052 call assert_true(exists('#WinNew'))
3053 close
3054 close
3055
3056 " With ++once WinNew triggers once
3057 let g:did_split = 0
3058 augroup Testing
3059 au!
3060 au WinNew * ++once let g:did_split += 1
3061 augroup END
3062 split
3063 split
3064 call assert_equal(1, g:did_split)
3065 call assert_false(exists('#WinNew'))
3066 close
3067 close
3068
3069 call assert_fails('au WinNew * ++once ++once echo bad', 'E983:')
3070endfunc
3071
Bram Moolenaara68e5952019-04-25 22:22:01 +02003072func Test_autocmd_bufreadpre()
3073 new
3074 let b:bufreadpre = 1
Bram Moolenaarab505b12020-03-23 19:28:44 +01003075 call append(0, range(1000))
Bram Moolenaara68e5952019-04-25 22:22:01 +02003076 w! XAutocmdBufReadPre.txt
3077 autocmd BufReadPre <buffer> :let b:bufreadpre += 1
Bram Moolenaarab505b12020-03-23 19:28:44 +01003078 norm! 500gg
Bram Moolenaara68e5952019-04-25 22:22:01 +02003079 sp
Bram Moolenaarab505b12020-03-23 19:28:44 +01003080 norm! 1000gg
Bram Moolenaara68e5952019-04-25 22:22:01 +02003081 wincmd p
3082 let g:wsv1 = winsaveview()
3083 wincmd p
3084 let g:wsv2 = winsaveview()
3085 " triggers BufReadPre, should not move the cursor in either window
3086 " The topline may change one line in a large window.
3087 edit
3088 call assert_inrange(g:wsv2.topline - 1, g:wsv2.topline + 1, winsaveview().topline)
3089 call assert_equal(g:wsv2.lnum, winsaveview().lnum)
3090 call assert_equal(2, b:bufreadpre)
3091 wincmd p
3092 call assert_equal(g:wsv1.topline, winsaveview().topline)
3093 call assert_equal(g:wsv1.lnum, winsaveview().lnum)
3094 call assert_equal(2, b:bufreadpre)
3095 " Now set the cursor position in an BufReadPre autocommand
3096 " (even though the position will be invalid, this should make Vim reset the
3097 " cursor position in the other window.
3098 wincmd p
3099 set cpo+=g
3100 " won't do anything, but try to set the cursor on an invalid lnum
3101 autocmd BufReadPre <buffer> :norm! 70gg
3102 " triggers BufReadPre, should not move the cursor in either window
3103 e
3104 call assert_equal(1, winsaveview().topline)
3105 call assert_equal(1, winsaveview().lnum)
3106 call assert_equal(3, b:bufreadpre)
3107 wincmd p
3108 call assert_equal(g:wsv1.topline, winsaveview().topline)
3109 call assert_equal(g:wsv1.lnum, winsaveview().lnum)
3110 call assert_equal(3, b:bufreadpre)
3111 close
3112 close
3113 call delete('XAutocmdBufReadPre.txt')
3114 set cpo-=g
3115endfunc
3116
Bram Moolenaar5e66b422019-01-24 21:58:10 +01003117" FileChangedShell tested in test_filechanged.vim
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003118
3119" Tests for the following autocommands:
3120" - FileWritePre writing a compressed file
3121" - FileReadPost reading a compressed file
3122" - BufNewFile reading a file template
3123" - BufReadPre decompressing the file to be read
3124" - FilterReadPre substituting characters in the temp file
3125" - FilterReadPost substituting characters after filtering
3126" - FileReadPre set options for decompression
3127" - FileReadPost decompress the file
3128func Test_ReadWrite_Autocmds()
3129 " Run this test only on Unix-like systems and if gzip is available
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02003130 CheckUnix
3131 CheckExecutable gzip
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003132
3133 " Make $GZIP empty, "-v" would cause trouble.
3134 let $GZIP = ""
3135
3136 " Use a FileChangedShell autocommand to avoid a prompt for 'Xtestfile.gz'
3137 " being modified outside of Vim (noticed on Solaris).
3138 au FileChangedShell * echo 'caught FileChangedShell'
3139
3140 " Test for the FileReadPost, FileWritePre and FileWritePost autocmds
3141 augroup Test1
3142 au!
3143 au FileWritePre *.gz '[,']!gzip
3144 au FileWritePost *.gz undo
3145 au FileReadPost *.gz '[,']!gzip -d
3146 augroup END
3147
3148 new
3149 set bin
3150 call append(0, [
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 \ ])
3161 1,9write! Xtestfile.gz
3162 enew! | close
3163
3164 new
3165 " Read and decompress the testfile
3166 0read Xtestfile.gz
3167 call assert_equal([
3168 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3169 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3170 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3171 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3172 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3173 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3174 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3175 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3176 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3177 \ ], getline(1, 9))
3178 enew! | close
3179
3180 augroup Test1
3181 au!
3182 augroup END
3183
3184 " Test for the FileAppendPre and FileAppendPost autocmds
3185 augroup Test2
3186 au!
3187 au BufNewFile *.c read Xtest.c
3188 au FileAppendPre *.out '[,']s/new/NEW/
3189 au FileAppendPost *.out !cat Xtest.c >> test.out
3190 augroup END
3191
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003192 call writefile(['/*', ' * Here is a new .c file', ' */'], 'Xtest.c', 'D')
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003193 new foo.c " should load Xtest.c
3194 call assert_equal(['/*', ' * Here is a new .c file', ' */'], getline(2, 4))
3195 w! >> test.out " append it to the output file
3196
3197 let contents = readfile('test.out')
3198 call assert_equal(' * Here is a NEW .c file', contents[2])
3199 call assert_equal(' * Here is a new .c file', contents[5])
3200
3201 call delete('test.out')
3202 enew! | close
3203 augroup Test2
3204 au!
3205 augroup END
3206
3207 " Test for the BufReadPre and BufReadPost autocmds
3208 augroup Test3
3209 au!
3210 " setup autocommands to decompress before reading and re-compress
3211 " afterwards
3212 au BufReadPre *.gz exe '!gzip -d ' . shellescape(expand("<afile>"))
3213 au BufReadPre *.gz call rename(expand("<afile>:r"), expand("<afile>"))
3214 au BufReadPost *.gz call rename(expand("<afile>"), expand("<afile>:r"))
3215 au BufReadPost *.gz exe '!gzip ' . shellescape(expand("<afile>:r"))
3216 augroup END
3217
3218 e! Xtestfile.gz " Edit compressed file
3219 call assert_equal([
3220 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3221 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3222 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3223 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3224 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3225 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3226 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3227 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3228 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3229 \ ], getline(1, 9))
3230
3231 w! >> test.out " Append it to the output file
3232
3233 augroup Test3
3234 au!
3235 augroup END
3236
3237 " Test for the FilterReadPre and FilterReadPost autocmds.
3238 set shelltemp " need temp files here
3239 augroup Test4
3240 au!
3241 au FilterReadPre *.out call rename(expand("<afile>"), expand("<afile>") . ".t")
3242 au FilterReadPre *.out exe 'silent !sed s/e/E/ ' . shellescape(expand("<afile>")) . ".t >" . shellescape(expand("<afile>"))
3243 au FilterReadPre *.out exe 'silent !rm ' . shellescape(expand("<afile>")) . '.t'
3244 au FilterReadPost *.out '[,']s/x/X/g
3245 augroup END
3246
3247 e! test.out " Edit the output file
3248 1,$!cat
3249 call assert_equal([
3250 \ 'linE 2 AbcdefghijklmnopqrstuvwXyz',
3251 \ 'linE 3 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3252 \ 'linE 4 AbcdefghijklmnopqrstuvwXyz',
3253 \ 'linE 5 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3254 \ 'linE 6 AbcdefghijklmnopqrstuvwXyz',
3255 \ 'linE 7 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3256 \ 'linE 8 AbcdefghijklmnopqrstuvwXyz',
3257 \ 'linE 9 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3258 \ 'linE 10 AbcdefghijklmnopqrstuvwXyz'
3259 \ ], getline(1, 9))
3260 call assert_equal([
3261 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3262 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3263 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3264 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3265 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3266 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3267 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3268 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3269 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3270 \ ], readfile('test.out'))
3271
3272 augroup Test4
3273 au!
3274 augroup END
3275 set shelltemp&vim
3276
3277 " Test for the FileReadPre and FileReadPost autocmds.
3278 augroup Test5
3279 au!
3280 au FileReadPre *.gz exe 'silent !gzip -d ' . shellescape(expand("<afile>"))
3281 au FileReadPre *.gz call rename(expand("<afile>:r"), expand("<afile>"))
3282 au FileReadPost *.gz '[,']s/l/L/
3283 augroup END
3284
3285 new
3286 0r Xtestfile.gz " Read compressed file
3287 call assert_equal([
3288 \ 'Line 2 Abcdefghijklmnopqrstuvwxyz',
3289 \ 'Line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3290 \ 'Line 4 Abcdefghijklmnopqrstuvwxyz',
3291 \ 'Line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3292 \ 'Line 6 Abcdefghijklmnopqrstuvwxyz',
3293 \ 'Line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3294 \ 'Line 8 Abcdefghijklmnopqrstuvwxyz',
3295 \ 'Line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3296 \ 'Line 10 Abcdefghijklmnopqrstuvwxyz'
3297 \ ], getline(1, 9))
3298 call assert_equal([
3299 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3300 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3301 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3302 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3303 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3304 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3305 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3306 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3307 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3308 \ ], readfile('Xtestfile.gz'))
3309
3310 augroup Test5
3311 au!
3312 augroup END
3313
3314 au! FileChangedShell
3315 call delete('Xtestfile.gz')
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003316 call delete('test.out')
3317endfunc
Bram Moolenaar23b51392019-05-09 21:38:43 +02003318
3319func Test_throw_in_BufWritePre()
3320 new
3321 call setline(1, ['one', 'two', 'three'])
3322 call assert_false(filereadable('Xthefile'))
3323 augroup throwing
3324 au BufWritePre X* throw 'do not write'
3325 augroup END
3326 try
3327 w Xthefile
3328 catch
3329 let caught = 1
3330 endtry
3331 call assert_equal(1, caught)
3332 call assert_false(filereadable('Xthefile'))
3333
3334 bwipe!
3335 au! throwing
3336endfunc
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003337
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003338func Test_autocmd_in_try_block()
Bram Moolenaar6f14da12022-09-07 21:30:44 +01003339 call mkdir('Xintrydir', 'R')
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003340 au BufEnter * let g:fname = expand('%')
3341 try
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01003342 edit Xintrydir/
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003343 endtry
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01003344 call assert_match('Xintrydir', g:fname)
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003345
3346 unlet g:fname
3347 au! BufEnter
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003348endfunc
3349
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003350func Test_autocmd_SafeState()
3351 CheckRunVimInTerminal
3352
3353 let lines =<< trim END
3354 let g:safe = 0
3355 let g:again = ''
3356 au SafeState * let g:safe += 1
3357 au SafeStateAgain * let g:again ..= 'x'
3358 func CallTimer()
3359 call timer_start(10, {id -> execute('let g:again ..= "t"')})
3360 endfunc
3361 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003362 call writefile(lines, 'XSafeState', 'D')
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003363 let buf = RunVimInTerminal('-S XSafeState', #{rows: 6})
3364
Bram Moolenaar8e7d6222020-12-18 19:49:56 +01003365 " Sometimes we loop to handle a K_IGNORE, SafeState may be triggered once or
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003366 " more often.
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003367 call term_sendkeys(buf, ":echo g:safe\<CR>")
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003368 call WaitForAssert({-> assert_match('^\d ', term_getline(buf, 6))}, 1000)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003369
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003370 " SafeStateAgain should be invoked at least three times
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003371 call term_sendkeys(buf, ":echo g:again\<CR>")
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003372 call WaitForAssert({-> assert_match('^xxx', term_getline(buf, 6))}, 1000)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003373
3374 call term_sendkeys(buf, ":let g:again = ''\<CR>:call CallTimer()\<CR>")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003375 call TermWait(buf, 50)
Bram Moolenaar0f6629a2019-09-22 23:24:13 +02003376 call term_sendkeys(buf, ":\<CR>")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003377 call TermWait(buf, 50)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003378 call term_sendkeys(buf, ":echo g:again\<CR>")
3379 call WaitForAssert({-> assert_match('xtx', term_getline(buf, 6))}, 1000)
3380
3381 call StopVimInTerminal(buf)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003382endfunc
Bram Moolenaar23324a02019-10-01 17:39:04 +02003383
3384func Test_autocmd_CmdWinEnter()
3385 CheckRunVimInTerminal
Bram Moolenaar21829c52021-01-26 22:42:21 +01003386
Bram Moolenaar23324a02019-10-01 17:39:04 +02003387 let lines =<< trim END
Egor Zvorykin125ffd22021-11-17 14:01:14 +00003388 augroup vimHints | au! | augroup END
Bram Moolenaar23324a02019-10-01 17:39:04 +02003389 let b:dummy_var = 'This is a dummy'
3390 autocmd CmdWinEnter * quit
3391 let winnr = winnr('$')
3392 END
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01003393 let filename = 'XCmdWinEnter'
Bram Moolenaar23324a02019-10-01 17:39:04 +02003394 call writefile(lines, filename)
3395 let buf = RunVimInTerminal('-S '.filename, #{rows: 6})
3396
3397 call term_sendkeys(buf, "q:")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003398 call TermWait(buf)
Bram Moolenaar23324a02019-10-01 17:39:04 +02003399 call term_sendkeys(buf, ":echo b:dummy_var\<cr>")
Bram Moolenaar353c3512020-03-15 14:19:26 +01003400 call WaitForAssert({-> assert_match('^This is a dummy', term_getline(buf, 6))}, 2000)
Bram Moolenaar23324a02019-10-01 17:39:04 +02003401 call term_sendkeys(buf, ":echo &buftype\<cr>")
3402 call WaitForAssert({-> assert_notmatch('^nofile', term_getline(buf, 6))}, 1000)
3403 call term_sendkeys(buf, ":echo winnr\<cr>")
3404 call WaitForAssert({-> assert_match('^1', term_getline(buf, 6))}, 1000)
3405
3406 " clean up
3407 call StopVimInTerminal(buf)
3408 call delete(filename)
3409endfunc
Bram Moolenaarec66c412019-10-11 21:19:13 +02003410
3411func Test_autocmd_was_using_freed_memory()
Bram Moolenaar5a4c3082019-12-01 15:23:11 +01003412 CheckFeature quickfix
3413
Bram Moolenaarec66c412019-10-11 21:19:13 +02003414 pedit xx
3415 n x
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003416 augroup winenter
3417 au WinEnter * if winnr('$') > 2 | quit | endif
3418 augroup END
Bram Moolenaarec66c412019-10-11 21:19:13 +02003419 split
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003420
3421 augroup winenter
3422 au! WinEnter
3423 augroup END
3424
3425 bwipe xx
3426 bwipe x
3427 pclose
Bram Moolenaarec66c412019-10-11 21:19:13 +02003428endfunc
Bram Moolenaarf4a1d1c2019-11-16 13:50:25 +01003429
3430func Test_BufWrite_lockmarks()
Bram Moolenaarf08b0eb2021-10-16 13:00:14 +01003431 let g:test_is_flaky = 1
Bram Moolenaarf4a1d1c2019-11-16 13:50:25 +01003432 edit! Xtest
3433 call setline(1, ['a', 'b', 'c', 'd'])
3434
3435 " :lockmarks preserves the marks
3436 call SetChangeMarks(2, 3)
3437 lockmarks write
3438 call assert_equal([2, 3], [line("'["), line("']")])
3439
3440 " *WritePre autocmds get the correct line range, but lockmarks preserves the
3441 " original values for the user
3442 augroup lockmarks
3443 au!
3444 au BufWritePre,FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
3445 au FileWritePre * call assert_equal([3, 4], [line("'["), line("']")])
3446 augroup END
3447
3448 lockmarks write
3449 call assert_equal([2, 3], [line("'["), line("']")])
3450
3451 if executable('cat')
3452 lockmarks %!cat
3453 call assert_equal([2, 3], [line("'["), line("']")])
3454 endif
3455
3456 lockmarks 3,4write Xtest2
3457 call assert_equal([2, 3], [line("'["), line("']")])
3458
3459 au! lockmarks
3460 augroup! lockmarks
3461 call delete('Xtest')
3462 call delete('Xtest2')
3463endfunc
Bram Moolenaarce6db022020-01-07 20:11:42 +01003464
3465func Test_FileType_spell()
3466 if !isdirectory('/tmp')
3467 throw "Skipped: requires /tmp directory"
3468 endif
3469
3470 " this was crashing with an invalid free()
3471 setglobal spellfile=/tmp/en.utf-8.add
3472 augroup crash
3473 autocmd!
3474 autocmd BufNewFile,BufReadPost crashfile setf somefiletype
3475 autocmd BufNewFile,BufReadPost crashfile set ft=anotherfiletype
3476 autocmd FileType anotherfiletype setlocal spell
3477 augroup END
3478 func! NoCrash() abort
3479 edit /tmp/crashfile
3480 endfunc
3481 call NoCrash()
3482
3483 au! crash
3484 setglobal spellfile=
3485endfunc
Bram Moolenaarbc2b71d2020-02-17 21:33:30 +01003486
Bram Moolenaaref976322022-09-28 11:48:30 +01003487" this was wiping out the current buffer and using freed memory
3488func Test_SpellFileMissing_bwipe()
3489 next 0
3490 au SpellFileMissing 0 bwipe
3491 call assert_fails('set spell spelllang=0', 'E937:')
3492
3493 au! SpellFileMissing
Bram Moolenaar0a60f792022-11-19 21:18:11 +00003494 set nospell spelllang=en
Bram Moolenaaref976322022-09-28 11:48:30 +01003495 bwipe
3496endfunc
3497
Bram Moolenaar406cd902020-02-18 21:54:41 +01003498" Test closing a window or editing another buffer from a FileChangedRO handler
3499" in a readonly buffer
3500func Test_FileChangedRO_winclose()
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003501 call test_override('ui_delay', 10)
3502
Bram Moolenaar406cd902020-02-18 21:54:41 +01003503 augroup FileChangedROTest
3504 au!
3505 autocmd FileChangedRO * quit
3506 augroup END
3507 new
3508 set readonly
3509 call assert_fails('normal i', 'E788:')
3510 close
3511 augroup! FileChangedROTest
3512
3513 augroup FileChangedROTest
3514 au!
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003515 autocmd FileChangedRO * edit Xrofile
Bram Moolenaar406cd902020-02-18 21:54:41 +01003516 augroup END
3517 new
3518 set readonly
3519 call assert_fails('normal i', 'E788:')
3520 close
3521 augroup! FileChangedROTest
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003522 call test_override('ALL', 0)
Bram Moolenaar406cd902020-02-18 21:54:41 +01003523endfunc
3524
Bram Moolenaar0c81d1b2020-02-22 22:45:55 +01003525func LogACmd()
3526 call add(g:logged, line('$'))
3527endfunc
3528
3529func Test_TermChanged()
Bram Moolenaard28e0b32020-02-22 23:08:52 +01003530 CheckNotGui
3531
Bram Moolenaar0c81d1b2020-02-22 22:45:55 +01003532 enew!
3533 tabnew
3534 call setline(1, ['a', 'b', 'c', 'd'])
3535 $
3536 au TermChanged * call LogACmd()
3537 let g:logged = []
3538 let term_save = &term
3539 set term=xterm
3540 call assert_equal([1, 4], g:logged)
3541
3542 au! TermChanged
3543 let &term = term_save
3544 bwipe!
3545endfunc
3546
Bram Moolenaare3284872020-03-19 13:55:03 +01003547" Test for FileReadCmd autocmd
3548func Test_autocmd_FileReadCmd()
3549 func ReadFileCmd()
3550 call append(line('$'), "v:cmdarg = " .. v:cmdarg)
3551 endfunc
3552 augroup FileReadCmdTest
3553 au!
3554 au FileReadCmd Xtest call ReadFileCmd()
3555 augroup END
3556
3557 new
3558 read ++bin Xtest
3559 read ++nobin Xtest
3560 read ++edit Xtest
3561 read ++bad=keep Xtest
3562 read ++bad=drop Xtest
3563 read ++bad=- Xtest
3564 read ++ff=unix Xtest
3565 read ++ff=dos Xtest
3566 read ++ff=mac Xtest
3567 read ++enc=utf-8 Xtest
3568
3569 call assert_equal(['',
3570 \ 'v:cmdarg = ++bin',
3571 \ 'v:cmdarg = ++nobin',
3572 \ 'v:cmdarg = ++edit',
3573 \ 'v:cmdarg = ++bad=keep',
3574 \ 'v:cmdarg = ++bad=drop',
3575 \ 'v:cmdarg = ++bad=-',
3576 \ 'v:cmdarg = ++ff=unix',
3577 \ 'v:cmdarg = ++ff=dos',
3578 \ 'v:cmdarg = ++ff=mac',
3579 \ 'v:cmdarg = ++enc=utf-8'], getline(1, '$'))
3580
Bram Moolenaar23526d22022-12-05 15:50:41 +00003581 bwipe!
Bram Moolenaare3284872020-03-19 13:55:03 +01003582 augroup FileReadCmdTest
3583 au!
3584 augroup END
3585 delfunc ReadFileCmd
3586endfunc
3587
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003588" Test for passing invalid arguments to autocmd
3589func Test_autocmd_invalid_args()
3590 " Additional character after * for event
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003591 call assert_fails('autocmd *a Xinvfile set ff=unix', 'E215:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003592 augroup Test
3593 augroup END
3594 " Invalid autocmd event
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003595 call assert_fails('autocmd Bufabc Xinvfile set ft=vim', 'E216:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003596 " Invalid autocmd event in a autocmd group
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003597 call assert_fails('autocmd Test Bufabc Xinvfile set ft=vim', 'E216:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003598 augroup! Test
3599 " Execute all autocmds
3600 call assert_fails('doautocmd * BufEnter', 'E217:')
3601 call assert_fails('augroup! x1a2b3', 'E367:')
3602 call assert_fails('autocmd BufNew <buffer=999> pwd', 'E680:')
Bram Moolenaar531be472020-09-23 22:38:05 +02003603 call assert_fails('autocmd BufNew \) set ff=unix', 'E55:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003604endfunc
3605
3606" Test for deep nesting of autocmds
3607func Test_autocmd_deep_nesting()
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003608 autocmd BufEnter Xdeepfile doautocmd BufEnter Xdeepfile
3609 call assert_fails('doautocmd BufEnter Xdeepfile', 'E218:')
3610 autocmd! BufEnter Xdeepfile
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003611endfunc
3612
Bram Moolenaarbe5ee862020-06-10 20:56:58 +02003613" Tests for SigUSR1 autocmd event, which is only available on posix systems.
3614func Test_autocmd_sigusr1()
3615 CheckUnix
Bram Moolenaar0056ca72022-09-23 21:26:39 +01003616 " FIXME: should this work on MacOS M1?
3617 CheckNotMacM1
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003618 CheckExecutable /bin/kill
Bram Moolenaarbe5ee862020-06-10 20:56:58 +02003619
3620 let g:sigusr1_passed = 0
3621 au SigUSR1 * let g:sigusr1_passed = 1
3622 call system('/bin/kill -s usr1 ' . getpid())
3623 call WaitForAssert({-> assert_true(g:sigusr1_passed)})
3624
3625 au! SigUSR1
3626 unlet g:sigusr1_passed
3627endfunc
3628
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003629" Test for BufReadPre autocmd deleting the file
3630func Test_BufReadPre_delfile()
3631 augroup TestAuCmd
3632 au!
Bram Moolenaare7cda972022-08-29 11:02:59 +01003633 autocmd BufReadPre XbufreadPre call delete('XbufreadPre')
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003634 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003635 call writefile([], 'XbufreadPre', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +01003636 call assert_fails('new XbufreadPre', 'E200:')
3637 call assert_equal('XbufreadPre', @%)
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003638 call assert_equal(1, &readonly)
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003639
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003640 augroup TestAuCmd
3641 au!
3642 augroup END
3643 close!
3644endfunc
3645
3646" Test for BufReadPre autocmd changing the current buffer
3647func Test_BufReadPre_changebuf()
3648 augroup TestAuCmd
3649 au!
Bram Moolenaare7cda972022-08-29 11:02:59 +01003650 autocmd BufReadPre Xchangebuf edit Xsomeotherfile
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003651 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003652 call writefile([], 'Xchangebuf', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +01003653 call assert_fails('new Xchangebuf', 'E201:')
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003654 call assert_equal('Xsomeotherfile', @%)
3655 call assert_equal(1, &readonly)
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003656
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003657 augroup TestAuCmd
3658 au!
3659 augroup END
3660 close!
3661endfunc
3662
3663" Test for BufWipeouti autocmd changing the current buffer when reading a file
3664" in an empty buffer with 'f' flag in 'cpo'
3665func Test_BufDelete_changebuf()
3666 new
3667 augroup TestAuCmd
3668 au!
3669 autocmd BufWipeout * let bufnr = bufadd('somefile') | exe "b " .. bufnr
3670 augroup END
3671 let save_cpo = &cpo
3672 set cpo+=f
Bram Moolenaarb18b4962022-09-02 21:55:50 +01003673 call assert_fails('r Xchangebuf', ['E812:', 'E484:'])
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003674 call assert_equal('somefile', @%)
3675 let &cpo = save_cpo
3676 augroup TestAuCmd
3677 au!
3678 augroup END
3679 close!
3680endfunc
3681
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003682" Test for the temporary internal window used to execute autocmds
3683func Test_autocmd_window()
3684 %bw!
3685 edit one.txt
3686 tabnew two.txt
Bram Moolenaar41cd8032021-03-13 15:47:56 +01003687 vnew three.txt
3688 tabnew four.txt
3689 tabprevious
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003690 let g:blist = []
Bram Moolenaar832adf92020-06-25 19:01:36 +02003691 augroup aucmd_win_test1
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003692 au!
3693 au BufEnter * call add(g:blist, [expand('<afile>'),
3694 \ win_gettype(bufwinnr(expand('<afile>')))])
3695 augroup END
3696
3697 doautoall BufEnter
Bram Moolenaar41cd8032021-03-13 15:47:56 +01003698 call assert_equal([
3699 \ ['one.txt', 'autocmd'],
3700 \ ['two.txt', ''],
3701 \ ['four.txt', 'autocmd'],
3702 \ ['three.txt', ''],
3703 \ ], g:blist)
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003704
Bram Moolenaar832adf92020-06-25 19:01:36 +02003705 augroup aucmd_win_test1
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003706 au!
3707 augroup END
Bram Moolenaar832adf92020-06-25 19:01:36 +02003708 augroup! aucmd_win_test1
3709 %bw!
3710endfunc
3711
3712" Test for trying to close the temporary window used for executing an autocmd
3713func Test_close_autocmd_window()
3714 %bw!
3715 edit one.txt
3716 tabnew two.txt
3717 augroup aucmd_win_test2
3718 au!
3719 au BufEnter * if expand('<afile>') == 'one.txt' | 1close | endif
3720 augroup END
3721
3722 call assert_fails('doautoall BufEnter', 'E813:')
3723
3724 augroup aucmd_win_test2
3725 au!
3726 augroup END
3727 augroup! aucmd_win_test2
Bram Moolenaarcf844172020-06-26 19:44:06 +02003728 %bwipe!
3729endfunc
3730
3731" Test for trying to close the tab that has the temporary window for exeucing
3732" an autocmd.
3733func Test_close_autocmd_tab()
3734 edit one.txt
3735 tabnew two.txt
3736 augroup aucmd_win_test
3737 au!
3738 au BufEnter * if expand('<afile>') == 'one.txt' | tabfirst | tabonly | endif
3739 augroup END
3740
3741 call assert_fails('doautoall BufEnter', 'E813:')
3742
3743 tabonly
3744 augroup aucmd_win_test
3745 au!
3746 augroup END
3747 augroup! aucmd_win_test
3748 %bwipe!
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003749endfunc
3750
Bram Moolenaarcb1956d2022-01-07 15:45:18 +00003751func Test_Visual_doautoall_redraw()
3752 call setline(1, ['a', 'b'])
Bram Moolenaar94722c52023-01-28 19:19:03 +00003753 new
Bram Moolenaarcb1956d2022-01-07 15:45:18 +00003754 wincmd p
3755 call feedkeys("G\<C-V>", 'txn')
3756 autocmd User Explode ++once redraw
3757 doautoall User Explode
3758 %bwipe!
3759endfunc
3760
Bram Moolenaar6bcb8772021-02-03 21:23:29 +01003761" This was using freed memory.
3762func Test_BufNew_arglocal()
3763 arglocal
3764 au BufNew * arglocal
3765 call assert_fails('drop xx', 'E1156:')
3766
3767 au! BufNew
3768endfunc
3769
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003770func Test_autocmd_closes_window()
3771 au BufNew,BufWinLeave * e %e
3772 file yyy
3773 au BufNew,BufWinLeave * ball
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003774 n xxx
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003775
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003776 %bwipe
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003777 au! BufNew
3778 au! BufWinLeave
3779endfunc
3780
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003781func Test_autocmd_quit_psearch()
3782 sn aa bb
3783 augroup aucmd_win_test
3784 au!
3785 au BufEnter,BufLeave,BufNew,WinEnter,WinLeave,WinNew * if winnr('$') > 1 | q | endif
3786 augroup END
3787 ps /
3788
3789 augroup aucmd_win_test
3790 au!
3791 augroup END
zeertzjq7851c692022-04-21 11:14:01 +01003792 new
3793 pclose
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003794endfunc
3795
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003796" Fuzzer found some strange combination that caused a crash.
3797func Test_autocmd_normal_mess()
Bram Moolenaardd07c022021-02-07 13:32:46 +01003798 " For unknown reason this hangs on MS-Windows
3799 CheckNotMSWindows
3800
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003801 augroup aucmd_normal_test
3802 au BufLeave,BufWinLeave,BufHidden,BufUnload,BufDelete,BufWipeout * norm 7q/qc
3803 augroup END
zeertzjq67fe77d2025-04-20 10:21:18 +02003804 call assert_fails('o4', 'E1159:')
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003805 silent! H
zeertzjq67fe77d2025-04-20 10:21:18 +02003806 call assert_fails('e xx', 'E1159:')
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003807 normal G
3808
3809 augroup aucmd_normal_test
3810 au!
3811 augroup END
3812endfunc
3813
Bram Moolenaar8c6951f2021-02-06 18:08:45 +01003814func Test_autocmd_closing_cmdwin()
Bram Moolenaardd07c022021-02-07 13:32:46 +01003815 " For unknown reason this hangs on MS-Windows
3816 CheckNotMSWindows
3817
Bram Moolenaar8c6951f2021-02-06 18:08:45 +01003818 au BufWinLeave * nested q
3819 call assert_fails("norm 7q?\n", 'E855:')
3820
3821 au! BufWinLeave
3822 new
3823 only
3824endfunc
3825
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003826func Test_autocmd_vimgrep()
3827 augroup aucmd_vimgrep
Charlie Grovesfef44852022-04-19 16:24:12 +01003828 au QuickfixCmdPre,BufNew,BufReadCmd * sb
zeertzjq7851c692022-04-21 11:14:01 +01003829 au QuickfixCmdPre,BufNew,BufReadCmd * q9
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003830 augroup END
Bram Moolenaardd07c022021-02-07 13:32:46 +01003831 call assert_fails('lv ?a? foo', 'E926:')
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003832
3833 augroup aucmd_vimgrep
3834 au!
3835 augroup END
3836endfunc
3837
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003838func Test_autocmd_with_block()
3839 augroup block_testing
3840 au BufReadPost *.xml {
3841 setlocal matchpairs+=<:>
3842 /<start
3843 }
Bram Moolenaar63b91732021-08-05 20:40:03 +02003844 au CursorHold * {
3845 autocmd BufReadPre * ++once echo 'one' | echo 'two'
3846 g:gotSafeState = 77
3847 }
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003848 augroup END
3849
Ken Takataeccc9272024-09-03 23:01:55 +02003850 let expected = gettext("\n--- Autocommands ---") .. "\nblock_testing BufRead\n *.xml {^@ setlocal matchpairs+=<:>^@ /<start^@ }"
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003851 call assert_equal(expected, execute('au BufReadPost *.xml'))
3852
Bram Moolenaar63b91732021-08-05 20:40:03 +02003853 doautocmd CursorHold
3854 call assert_equal(77, g:gotSafeState)
3855 unlet g:gotSafeState
3856
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003857 augroup block_testing
3858 au!
Bram Moolenaar75ebd2a2022-06-03 17:39:46 +01003859 autocmd CursorHold * {
3860 if true
3861 # comment
3862 && true
3863
3864 && true
3865 g:done = 'yes'
3866 endif
3867 }
3868 augroup END
3869 doautocmd CursorHold
3870 call assert_equal('yes', g:done)
3871
3872 unlet g:done
3873 augroup block_testing
3874 au!
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003875 augroup END
3876endfunc
3877
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00003878func Test_closing_autocmd_window()
3879 let lines =<< trim END
3880 edit Xa.txt
3881 tabnew Xb.txt
3882 autocmd BufEnter Xa.txt unhide 1
3883 doautoall BufEnter
3884 END
3885 call v9.CheckScriptFailure(lines, 'E814:')
3886 au! BufEnter
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00003887 bwipe Xa.txt
3888 bwipe Xb.txt
3889endfunc
3890
zeertzjq46bdae02023-09-24 23:16:08 +02003891func Test_switch_window_in_autocmd_window()
3892 edit Xa.txt
3893 tabnew Xb.txt
3894 autocmd BufEnter Xa.txt wincmd w
3895 doautoall BufEnter
3896 au! BufEnter
3897 bwipe Xa.txt
3898 call assert_false(bufexists('Xa.txt'))
3899 bwipe Xb.txt
3900 call assert_false(bufexists('Xb.txt'))
3901endfunc
3902
zeertzjq9d956ee2024-04-07 18:16:10 +02003903" Test that using the autocommand window doesn't change current directory.
3904func Test_autocmd_window_cwd()
3905 let saveddir = getcwd()
3906 call mkdir('Xcwd/a/b/c/d', 'pR')
3907
3908 new Xa.txt
3909 tabnew
3910 new Xb.txt
3911
3912 tabprev
3913 cd Xcwd
3914 call assert_match('/Xcwd$', getcwd())
3915 call assert_match('\[global\] .*/Xcwd$', trim(execute('verbose pwd')))
3916
3917 autocmd BufEnter Xb.txt lcd ./a/b/c/d
3918 doautoall BufEnter
3919 au! BufEnter
3920 call assert_match('/Xcwd$', getcwd())
3921 call assert_match('\[global\] .*/Xcwd$', trim(execute('verbose pwd')))
3922
3923 tabnext
3924 cd ./a
3925 tcd ./b
3926 lcd ./c
3927 call assert_match('/Xcwd/a/b/c$', getcwd())
3928 call assert_match('\[window\] .*/Xcwd/a/b/c$', trim(execute('verbose pwd')))
3929
3930 autocmd BufEnter Xa.txt call assert_match('Xcwd/a/b/c$', getcwd())
3931 doautoall BufEnter
3932 au! BufEnter
3933 call assert_match('/Xcwd/a/b/c$', getcwd())
3934 call assert_match('\[window\] .*/Xcwd/a/b/c$', trim(execute('verbose pwd')))
3935 bwipe!
3936 call assert_match('/Xcwd/a/b$', getcwd())
3937 call assert_match('\[tabpage\] .*/Xcwd/a/b$', trim(execute('verbose pwd')))
3938 bwipe!
3939 call assert_match('/Xcwd/a$', getcwd())
3940 call assert_match('\[global\] .*/Xcwd/a$', trim(execute('verbose pwd')))
3941 bwipe!
3942
3943 call chdir(saveddir)
3944endfunc
3945
Bram Moolenaar347538f2022-03-26 16:28:06 +00003946func Test_bufwipeout_changes_window()
3947 " This should not crash, but we don't have any expectations about what
3948 " happens, changing window in BufWipeout has unpredictable results.
3949 tabedit
3950 let g:window_id = win_getid()
3951 topleft new
3952 setlocal bufhidden=wipe
3953 autocmd BufWipeout <buffer> call win_gotoid(g:window_id)
3954 tabprevious
3955 +tabclose
3956
3957 unlet g:window_id
3958 au! BufWipeout
3959 %bwipe!
3960endfunc
3961
zeertzjq021996f2022-04-10 11:44:04 +01003962func Test_v_event_readonly()
3963 autocmd CompleteChanged * let v:event.width = 0
3964 call assert_fails("normal! i\<C-X>\<C-V>", 'E46:')
3965 au! CompleteChanged
3966
3967 autocmd DirChangedPre * let v:event.directory = ''
3968 call assert_fails('cd .', 'E46:')
3969 au! DirChangedPre
3970
3971 autocmd ModeChanged * let v:event.new_mode = ''
3972 call assert_fails('normal! cc', 'E46:')
3973 au! ModeChanged
3974
3975 autocmd TextYankPost * let v:event.operator = ''
3976 call assert_fails('normal! yy', 'E46:')
3977 au! TextYankPost
3978endfunc
3979
zeertzjqc9e8fd62022-07-26 18:12:38 +01003980" Test for ModeChanged pattern
3981func Test_mode_changes()
3982 let g:index = 0
zeertzjq73916ba2023-04-26 16:50:19 +01003983 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 +01003984 func! TestMode()
3985 call assert_equal(g:mode_seq[g:index], get(v:event, "old_mode"))
3986 call assert_equal(g:mode_seq[g:index + 1], get(v:event, "new_mode"))
3987 call assert_equal(mode(1), get(v:event, "new_mode"))
3988 let g:index += 1
3989 endfunc
3990
3991 au ModeChanged * :call TestMode()
3992 let g:n_to_any = 0
3993 au ModeChanged n:* let g:n_to_any += 1
zeertzjq73916ba2023-04-26 16:50:19 +01003994 call feedkeys("i\<esc>vVca\<CR>\<C-X>\<C-L>\<esc>ggdV\<MouseMove>G", 'tnix')
zeertzjqc9e8fd62022-07-26 18:12:38 +01003995
3996 let g:V_to_v = 0
3997 au ModeChanged V:v let g:V_to_v += 1
3998 call feedkeys("Vv\<C-G>\<esc>", 'tnix')
3999 call assert_equal(len(filter(g:mode_seq[1:], {idx, val -> val == 'n'})), g:n_to_any)
4000 call assert_equal(1, g:V_to_v)
4001 call assert_equal(len(g:mode_seq) - 1, g:index)
4002
4003 let g:n_to_i = 0
4004 au ModeChanged n:i let g:n_to_i += 1
4005 let g:n_to_niI = 0
4006 au ModeChanged i:niI let g:n_to_niI += 1
4007 let g:niI_to_i = 0
4008 au ModeChanged niI:i let g:niI_to_i += 1
4009 let g:nany_to_i = 0
4010 au ModeChanged n*:i let g:nany_to_i += 1
4011 let g:i_to_n = 0
4012 au ModeChanged i:n let g:i_to_n += 1
4013 let g:nori_to_any = 0
4014 au ModeChanged [ni]:* let g:nori_to_any += 1
4015 let g:i_to_any = 0
4016 au ModeChanged i:* let g:i_to_any += 1
4017 let g:index = 0
4018 let g:mode_seq = ['n', 'i', 'niI', 'i', 'n']
4019 call feedkeys("a\<C-O>l\<esc>", 'tnix')
4020 call assert_equal(len(g:mode_seq) - 1, g:index)
4021 call assert_equal(1, g:n_to_i)
4022 call assert_equal(1, g:n_to_niI)
4023 call assert_equal(1, g:niI_to_i)
4024 call assert_equal(2, g:nany_to_i)
4025 call assert_equal(1, g:i_to_n)
4026 call assert_equal(2, g:i_to_any)
4027 call assert_equal(3, g:nori_to_any)
4028
4029 if has('terminal')
4030 let g:mode_seq += ['c', 'n', 't', 'nt', 'c', 'nt', 'n']
4031 call feedkeys(":term\<CR>\<C-W>N:bd!\<CR>", 'tnix')
4032 call assert_equal(len(g:mode_seq) - 1, g:index)
4033 call assert_equal(1, g:n_to_i)
4034 call assert_equal(1, g:n_to_niI)
4035 call assert_equal(1, g:niI_to_i)
4036 call assert_equal(2, g:nany_to_i)
4037 call assert_equal(1, g:i_to_n)
4038 call assert_equal(2, g:i_to_any)
4039 call assert_equal(5, g:nori_to_any)
4040 endif
4041
zeertzjqd1955982022-10-05 11:24:46 +01004042 let g:n_to_c = 0
4043 au ModeChanged n:c let g:n_to_c += 1
4044 let g:c_to_n = 0
4045 au ModeChanged c:n let g:c_to_n += 1
4046 let g:mode_seq += ['c', 'n', 'c', 'n']
4047 call feedkeys("q:\<C-C>\<Esc>", 'tnix')
4048 call assert_equal(len(g:mode_seq) - 1, g:index)
4049 call assert_equal(2, g:n_to_c)
4050 call assert_equal(2, g:c_to_n)
zeertzjqc9e8fd62022-07-26 18:12:38 +01004051
Bram Moolenaar61c4b042022-10-18 15:10:11 +01004052 let g:n_to_v = 0
4053 au ModeChanged n:v let g:n_to_v += 1
4054 let g:v_to_n = 0
4055 au ModeChanged v:n let g:v_to_n += 1
4056 let g:mode_seq += ['v', 'n']
4057 call feedkeys("v\<C-C>", 'tnix')
4058 call assert_equal(len(g:mode_seq) - 1, g:index)
4059 call assert_equal(1, g:n_to_v)
4060 call assert_equal(1, g:v_to_n)
zeertzjqfcaeb3d2023-11-28 20:46:29 +01004061
4062 let g:mode_seq += ['c', 'cr', 'c', 'cr', 'n']
4063 call feedkeys(":\<Insert>\<Insert>\<Insert>\<CR>", 'tnix')
4064 call assert_equal(len(g:mode_seq) - 1, g:index)
Bram Moolenaar61c4b042022-10-18 15:10:11 +01004065
zeertzjqc9e8fd62022-07-26 18:12:38 +01004066 au! ModeChanged
4067 delfunc TestMode
4068 unlet! g:mode_seq
4069 unlet! g:index
4070 unlet! g:n_to_any
4071 unlet! g:V_to_v
4072 unlet! g:n_to_i
4073 unlet! g:n_to_niI
4074 unlet! g:niI_to_i
4075 unlet! g:nany_to_i
4076 unlet! g:i_to_n
4077 unlet! g:nori_to_any
4078 unlet! g:i_to_any
zeertzjqfcaeb3d2023-11-28 20:46:29 +01004079 unlet! g:n_to_c
4080 unlet! g:c_to_n
4081 unlet! g:n_to_v
4082 unlet! g:v_to_n
zeertzjqc9e8fd62022-07-26 18:12:38 +01004083endfunc
4084
4085func Test_recursive_ModeChanged()
4086 au! ModeChanged * norm 0u
4087 sil! norm 
4088 au! ModeChanged
4089endfunc
4090
4091func Test_ModeChanged_starts_visual()
4092 " This was triggering ModeChanged before setting VIsual, causing a crash.
4093 au! ModeChanged * norm 0u
4094 sil! norm 
4095
4096 au! ModeChanged
4097endfunc
Bram Moolenaar347538f2022-03-26 16:28:06 +00004098
Charlie Grovesfef44852022-04-19 16:24:12 +01004099func Test_noname_autocmd()
4100 augroup test_noname_autocmd_group
4101 autocmd!
4102 autocmd BufEnter * call add(s:li, ["BufEnter", expand("<afile>")])
4103 autocmd BufDelete * call add(s:li, ["BufDelete", expand("<afile>")])
4104 autocmd BufLeave * call add(s:li, ["BufLeave", expand("<afile>")])
4105 autocmd BufUnload * call add(s:li, ["BufUnload", expand("<afile>")])
4106 autocmd BufWipeout * call add(s:li, ["BufWipeout", expand("<afile>")])
4107 augroup END
4108
4109 let s:li = []
4110 edit foo
4111 call assert_equal([['BufUnload', ''], ['BufDelete', ''], ['BufWipeout', ''], ['BufEnter', 'foo']], s:li)
4112
4113 au! test_noname_autocmd_group
4114 augroup! test_noname_autocmd_group
4115endfunc
4116
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004117" Test for the autocmd_get() function
4118func Test_autocmd_get()
4119 augroup TestAutoCmdFns
4120 au!
4121 autocmd BufAdd *.vim echo "bufadd-vim"
4122 autocmd BufAdd *.py echo "bufadd-py"
4123 autocmd BufHidden *.vim echo "bufhidden"
4124 augroup END
4125 augroup TestAutoCmdFns2
4126 autocmd BufAdd *.vim echo "bufadd-vim-2"
4127 autocmd BufRead *.a1b2c3 echo "bufadd-vim-2"
4128 augroup END
4129
4130 let l = autocmd_get()
4131 call assert_true(l->len() > 0)
4132
4133 " Test for getting all the autocmds in a group
4134 let expected = [
4135 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4136 \ pattern: '*.vim', nested: v:false, once: v:false,
4137 \ event: 'BufAdd'},
4138 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4139 \ pattern: '*.py', nested: v:false, once: v:false,
4140 \ event: 'BufAdd'},
4141 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4142 \ pattern: '*.vim', nested: v:false,
4143 \ once: v:false, event: 'BufHidden'}]
4144 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4145
4146 " Test for getting autocmds for all the patterns in a group
4147 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns',
4148 \ event: '*'}))
4149
4150 " Test for getting autocmds for an event in a group
4151 let expected = [
4152 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4153 \ pattern: '*.vim', nested: v:false, once: v:false,
4154 \ event: 'BufAdd'},
4155 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4156 \ pattern: '*.py', nested: v:false, once: v:false,
4157 \ event: 'BufAdd'}]
4158 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns',
4159 \ event: 'BufAdd'}))
4160
4161 " Test for getting the autocmds for all the events in a group for particular
4162 " pattern
4163 call assert_equal([{'cmd': 'echo "bufadd-py"', 'group': 'TestAutoCmdFns',
4164 \ 'pattern': '*.py', 'nested': v:false, 'once': v:false,
4165 \ 'event': 'BufAdd'}],
4166 \ autocmd_get(#{group: 'TestAutoCmdFns', event: '*', pattern: '*.py'}))
4167
4168 " Test for getting the autocmds for an events in a group for particular
4169 " pattern
4170 let l = autocmd_get(#{group: 'TestAutoCmdFns', event: 'BufAdd',
4171 \ pattern: '*.vim'})
4172 call assert_equal([
4173 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4174 \ pattern: '*.vim', nested: v:false, once: v:false,
4175 \ event: 'BufAdd'}], l)
4176
4177 " Test for getting the autocmds for a pattern in a group
4178 let l = autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.vim'})
4179 call assert_equal([
4180 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4181 \ pattern: '*.vim', nested: v:false, once: v:false,
4182 \ event: 'BufAdd'},
4183 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4184 \ pattern: '*.vim', nested: v:false,
4185 \ once: v:false, event: 'BufHidden'}], l)
4186
4187 " Test for getting the autocmds for a pattern in all the groups
4188 let l = autocmd_get(#{pattern: '*.a1b2c3'})
4189 call assert_equal([{'cmd': 'echo "bufadd-vim-2"', 'group': 'TestAutoCmdFns2',
4190 \ 'pattern': '*.a1b2c3', 'nested': v:false, 'once': v:false,
4191 \ 'event': 'BufRead'}], l)
4192
4193 " Test for getting autocmds for a pattern without any autocmds
4194 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4195 \ pattern: '*.abc'}))
4196 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4197 \ event: 'BufAdd', pattern: '*.abc'}))
4198 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4199 \ event: 'BufWipeout'}))
zeertzjq2d1d5c62024-06-09 16:44:33 +02004200
4201 " Test for getting autocmds after removing one inside an autocmd
4202 func CheckAutocmdGet()
4203 augroup TestAutoCmdFns
4204 autocmd! BufAdd *.vim
4205 augroup END
4206
4207 let expected = [
4208 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4209 \ pattern: '*.py', nested: v:false, once: v:false,
4210 \ event: 'BufAdd'},
4211 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4212 \ pattern: '*.vim', nested: v:false,
4213 \ once: v:false, event: 'BufHidden'}]
4214
4215 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4216 call assert_equal([expected[0]],
4217 \ autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.py'}))
4218 call assert_equal([expected[1]],
4219 \ autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.vim'}))
4220 endfunc
4221
4222 autocmd User Xauget call CheckAutocmdGet()
4223 doautocmd User Xauget
4224 autocmd! User Xauget
4225
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004226 call assert_fails("call autocmd_get(#{group: 'abc', event: 'BufAdd'})",
4227 \ 'E367:')
4228 let cmd = "echo autocmd_get(#{group: 'TestAutoCmdFns', event: 'abc'})"
4229 call assert_fails(cmd, 'E216:')
4230 call assert_fails("call autocmd_get(#{group: 'abc'})", 'E367:')
4231 call assert_fails("echo autocmd_get(#{event: 'abc'})", 'E216:')
4232
4233 augroup TestAutoCmdFns
4234 au!
4235 augroup END
4236 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns'}))
4237
4238 " Test for nested and once autocmds
4239 augroup TestAutoCmdFns
4240 au!
4241 autocmd VimSuspend * ++nested echo "suspend"
4242 autocmd VimResume * ++once echo "resume"
4243 augroup END
4244
4245 let expected = [
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004246 \ {'cmd': 'echo "resume"', 'group': 'TestAutoCmdFns', 'pattern': '*',
Luuk van Baalb7147f82025-02-08 18:52:39 +01004247 \ 'nested': v:false, 'once': v:true, 'event': 'VimResume'},
4248 \ {'cmd': 'echo "suspend"', 'group': 'TestAutoCmdFns', 'pattern': '*',
4249 \ 'nested': v:true, 'once': v:false, 'event': 'VimSuspend'}]
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004250 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4251
4252 " Test for buffer-local autocmd
4253 augroup TestAutoCmdFns
4254 au!
4255 autocmd TextYankPost <buffer> echo "textyankpost"
4256 augroup END
4257
4258 let expected = [
4259 \ {'cmd': 'echo "textyankpost"', 'group': 'TestAutoCmdFns',
4260 \ 'pattern': '<buffer=' .. bufnr() .. '>', 'nested': v:false,
4261 \ 'once': v:false, 'bufnr': bufnr(), 'event': 'TextYankPost'}]
4262 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4263
4264 augroup TestAutoCmdFns
4265 au!
4266 augroup END
4267 augroup! TestAutoCmdFns
4268 augroup TestAutoCmdFns2
4269 au!
4270 augroup END
4271 augroup! TestAutoCmdFns2
4272
4273 call assert_fails("echo autocmd_get(#{group: []})", 'E730:')
4274 call assert_fails("echo autocmd_get(#{event: {}})", 'E731:')
4275 call assert_fails("echo autocmd_get([])", 'E1206:')
4276endfunc
4277
4278" Test for the autocmd_add() function
4279func Test_autocmd_add()
4280 " Define a single autocmd in a group
4281 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pattern: '*.sh',
4282 \ cmd: 'echo "bufadd"', once: v:true, nested: v:true}])
4283 call assert_equal([#{cmd: 'echo "bufadd"', group: 'TestAcSet',
4284 \ pattern: '*.sh', nested: v:true, once: v:true,
4285 \ event: 'BufAdd'}], autocmd_get(#{group: 'TestAcSet'}))
4286
4287 " Define two autocmds in the same group
4288 call autocmd_delete([#{group: 'TestAcSet'}])
4289 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pattern: '*.sh',
4290 \ cmd: 'echo "bufadd"'},
4291 \ #{group: 'TestAcSet', event: 'BufEnter', pattern: '*.sh',
4292 \ cmd: 'echo "bufenter"'}])
4293 call assert_equal([
4294 \ #{cmd: 'echo "bufadd"', group: 'TestAcSet', pattern: '*.sh',
4295 \ nested: v:false, once: v:false, event: 'BufAdd'},
4296 \ #{cmd: 'echo "bufenter"', group: 'TestAcSet', pattern: '*.sh',
4297 \ nested: v:false, once: v:false, event: 'BufEnter'}],
4298 \ autocmd_get(#{group: 'TestAcSet'}))
4299
4300 " Define a buffer-local autocmd
4301 call autocmd_delete([#{group: 'TestAcSet'}])
4302 call autocmd_add([#{group: 'TestAcSet', event: 'CursorHold',
4303 \ bufnr: bufnr(), cmd: 'echo "cursorhold"'}])
4304 call assert_equal([
4305 \ #{cmd: 'echo "cursorhold"', group: 'TestAcSet',
4306 \ pattern: '<buffer=' .. bufnr() .. '>', nested: v:false,
4307 \ once: v:false, bufnr: bufnr(), event: 'CursorHold'}],
4308 \ autocmd_get(#{group: 'TestAcSet'}))
4309
4310 " Use an invalid buffer number
4311 call autocmd_delete([#{group: 'TestAcSet'}])
4312 call autocmd_add([#{group: 'TestAcSet', event: 'BufEnter',
4313 \ bufnr: -1, cmd: 'echo "bufenter"'}])
4314 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4315 \ cmd: 'echo "bufadd"'}]
4316 call assert_fails("echo autocmd_add(l)", 'E680:')
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004317 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4318 \ pattern: '*.py', cmd: 'echo "bufadd"'}]
4319 call assert_fails("echo autocmd_add(l)", 'E680:')
4320 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4321 \ pattern: ['*.py', '*.c'], cmd: 'echo "bufadd"'}]
4322 call assert_fails("echo autocmd_add(l)", 'E680:')
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004323 let l = [#{group: 'TestAcSet', event: 'BufRead', bufnr: [],
4324 \ cmd: 'echo "bufread"'}]
4325 call assert_fails("echo autocmd_add(l)", 'E745:')
4326 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4327
4328 " Add two commands to the same group, event and pattern
4329 call autocmd_delete([#{group: 'TestAcSet'}])
4330 call autocmd_add([#{group: 'TestAcSet', event: 'BufUnload',
4331 \ pattern: 'abc', cmd: 'echo "cmd1"'}])
4332 call autocmd_add([#{group: 'TestAcSet', event: 'BufUnload',
4333 \ pattern: 'abc', cmd: 'echo "cmd2"'}])
4334 call assert_equal([
4335 \ #{cmd: 'echo "cmd1"', group: 'TestAcSet', pattern: 'abc',
4336 \ nested: v:false, once: v:false, event: 'BufUnload'},
4337 \ #{cmd: 'echo "cmd2"', group: 'TestAcSet', pattern: 'abc',
4338 \ nested: v:false, once: v:false, event: 'BufUnload'}],
4339 \ autocmd_get(#{group: 'TestAcSet'}))
4340
4341 " When adding a new autocmd, if the autocmd 'group' is not specified, then
4342 " the current autocmd group should be used.
4343 call autocmd_delete([#{group: 'TestAcSet'}])
4344 augroup TestAcSet
4345 call autocmd_add([#{event: 'BufHidden', pattern: 'abc', cmd: 'echo "abc"'}])
4346 augroup END
4347 call assert_equal([
4348 \ #{cmd: 'echo "abc"', group: 'TestAcSet', pattern: 'abc',
4349 \ nested: v:false, once: v:false, event: 'BufHidden'}],
4350 \ autocmd_get(#{group: 'TestAcSet'}))
4351
Yegappan Lakshmanan971f6822022-05-24 11:40:11 +01004352 " Test for replacing a cmd for an event in a group
4353 call autocmd_delete([#{group: 'TestAcSet'}])
4354 call autocmd_add([#{replace: v:true, group: 'TestAcSet', event: 'BufEnter',
4355 \ pattern: '*.py', cmd: 'echo "bufenter"'}])
4356 call autocmd_add([#{replace: v:true, group: 'TestAcSet', event: 'BufEnter',
4357 \ pattern: '*.py', cmd: 'echo "bufenter"'}])
4358 call assert_equal([
4359 \ #{cmd: 'echo "bufenter"', group: 'TestAcSet', pattern: '*.py',
4360 \ nested: v:false, once: v:false, event: 'BufEnter'}],
4361 \ autocmd_get(#{group: 'TestAcSet'}))
4362
4363 " Test for adding a command for an unsupported autocmd event
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004364 let l = [#{group: 'TestAcSet', event: 'abc', pattern: '*.sh',
4365 \ cmd: 'echo "bufadd"'}]
4366 call assert_fails('call autocmd_add(l)', 'E216:')
4367
Yegappan Lakshmanane0ff3a72022-05-27 18:05:33 +01004368 " Test for using a list of events and patterns
4369 call autocmd_delete([#{group: 'TestAcSet'}])
4370 let l = [#{group: 'TestAcSet', event: ['BufEnter', 'BufLeave'],
4371 \ pattern: ['*.py', '*.sh'], cmd: 'echo "bufcmds"'}]
4372 call autocmd_add(l)
4373 call assert_equal([
4374 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.py',
4375 \ nested: v:false, once: v:false, event: 'BufEnter'},
4376 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.sh',
4377 \ nested: v:false, once: v:false, event: 'BufEnter'},
4378 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.py',
4379 \ nested: v:false, once: v:false, event: 'BufLeave'},
4380 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.sh',
4381 \ nested: v:false, once: v:false, event: 'BufLeave'}],
4382 \ autocmd_get(#{group: 'TestAcSet'}))
4383
4384 " Test for invalid values for 'event' item
4385 call autocmd_delete([#{group: 'TestAcSet'}])
4386 let l = [#{group: 'TestAcSet', event: test_null_string(),
4387 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4388 call assert_fails('call autocmd_add(l)', 'E928:')
4389 let l = [#{group: 'TestAcSet', event: test_null_list(),
4390 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4391 call assert_fails('call autocmd_add(l)', 'E714:')
4392 let l = [#{group: 'TestAcSet', event: {},
4393 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4394 call assert_fails('call autocmd_add(l)', 'E777:')
4395 let l = [#{group: 'TestAcSet', event: [{}],
4396 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4397 call assert_fails('call autocmd_add(l)', 'E928:')
4398 let l = [#{group: 'TestAcSet', event: [test_null_string()],
4399 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4400 call assert_fails('call autocmd_add(l)', 'E928:')
4401 let l = [#{group: 'TestAcSet', event: 'BufEnter,BufLeave',
4402 \ pattern: '*.py', cmd: 'echo "bufcmds"'}]
4403 call assert_fails('call autocmd_add(l)', 'E216:')
4404 let l = [#{group: 'TestAcSet', event: [],
4405 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4406 call autocmd_add(l)
4407 let l = [#{group: 'TestAcSet', event: [""],
4408 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4409 call assert_fails('call autocmd_add(l)', 'E216:')
4410 let l = [#{group: 'TestAcSet', event: "",
4411 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4412 call autocmd_add(l)
4413 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4414
4415 " Test for invalid values for 'pattern' item
4416 let l = [#{group: 'TestAcSet', event: "BufEnter",
4417 \ pattern: test_null_string(), cmd: 'echo "bufcmds"'}]
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004418 call assert_fails('call autocmd_add(l)', 'E928:')
Yegappan Lakshmanane0ff3a72022-05-27 18:05:33 +01004419 let l = [#{group: 'TestAcSet', event: "BufEnter",
4420 \ pattern: test_null_list(), cmd: 'echo "bufcmds"'}]
4421 call assert_fails('call autocmd_add(l)', 'E714:')
4422 let l = [#{group: 'TestAcSet', event: "BufEnter",
4423 \ pattern: {}, cmd: 'echo "bufcmds"'}]
4424 call assert_fails('call autocmd_add(l)', 'E777:')
4425 let l = [#{group: 'TestAcSet', event: "BufEnter",
4426 \ pattern: [{}], cmd: 'echo "bufcmds"'}]
4427 call assert_fails('call autocmd_add(l)', 'E928:')
4428 let l = [#{group: 'TestAcSet', event: "BufEnter",
4429 \ pattern: [test_null_string()], cmd: 'echo "bufcmds"'}]
4430 call assert_fails('call autocmd_add(l)', 'E928:')
4431 let l = [#{group: 'TestAcSet', event: "BufEnter",
4432 \ pattern: [], cmd: 'echo "bufcmds"'}]
4433 call autocmd_add(l)
4434 let l = [#{group: 'TestAcSet', event: "BufEnter",
4435 \ pattern: [""], cmd: 'echo "bufcmds"'}]
4436 call autocmd_add(l)
4437 let l = [#{group: 'TestAcSet', event: "BufEnter",
4438 \ pattern: "", cmd: 'echo "bufcmds"'}]
4439 call autocmd_add(l)
4440 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4441
4442 let l = [#{group: 'TestAcSet', event: 'BufEnter,abc,BufLeave',
4443 \ pattern: '*.py', cmd: 'echo "bufcmds"'}]
4444 call assert_fails('call autocmd_add(l)', 'E216:')
4445
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004446 call assert_fails("call autocmd_add({})", 'E1211:')
4447 call assert_equal(v:false, autocmd_add(test_null_list()))
4448 call assert_true(autocmd_add([[]]))
4449 call assert_true(autocmd_add([test_null_dict()]))
4450
4451 augroup TestAcSet
4452 au!
4453 augroup END
4454
4455 call autocmd_add([#{group: 'TestAcSet'}])
4456 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd'}])
4457 call autocmd_add([#{group: 'TestAcSet', pat: '*.sh'}])
4458 call autocmd_add([#{group: 'TestAcSet', cmd: 'echo "a"'}])
4459 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pat: '*.sh'}])
4460 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', cmd: 'echo "a"'}])
4461 call autocmd_add([#{group: 'TestAcSet', pat: '*.sh', cmd: 'echo "a"'}])
4462 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4463
4464 augroup! TestAcSet
4465endfunc
4466
4467" Test for deleting autocmd events and groups
4468func Test_autocmd_delete()
4469 " Delete an event in an autocmd group
4470 augroup TestAcSet
4471 au!
4472 au BufAdd *.sh echo "bufadd"
4473 au BufEnter *.sh echo "bufenter"
4474 augroup END
4475 call autocmd_delete([#{group: 'TestAcSet', event: 'BufAdd'}])
4476 call assert_equal([#{cmd: 'echo "bufenter"', group: 'TestAcSet',
4477 \ pattern: '*.sh', nested: v:false, once: v:false,
4478 \ event: 'BufEnter'}], autocmd_get(#{group: 'TestAcSet'}))
4479
4480 " Delete all the events in an autocmd group
4481 augroup TestAcSet
4482 au BufAdd *.sh echo "bufadd"
4483 augroup END
4484 call autocmd_delete([#{group: 'TestAcSet', event: '*'}])
4485 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4486
4487 " Delete a non-existing autocmd group
4488 call assert_fails("call autocmd_delete([#{group: 'abc'}])", 'E367:')
4489 " Delete a non-existing autocmd event
4490 let l = [#{group: 'TestAcSet', event: 'abc'}]
4491 call assert_fails("call autocmd_delete(l)", 'E216:')
4492 " Delete a non-existing autocmd pattern
4493 let l = [#{group: 'TestAcSet', event: 'BufAdd', pat: 'abc'}]
4494 call assert_true(autocmd_delete(l))
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004495 " Delete an autocmd for a non-existing buffer
4496 let l = [#{event: '*', bufnr: 9999, cmd: 'echo "x"'}]
4497 call assert_fails('call autocmd_delete(l)', 'E680:')
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004498
4499 " Delete an autocmd group
4500 augroup TestAcSet
4501 au!
4502 au BufAdd *.sh echo "bufadd"
4503 au BufEnter *.sh echo "bufenter"
4504 augroup END
4505 call autocmd_delete([#{group: 'TestAcSet'}])
4506 call assert_fails("call autocmd_get(#{group: 'TestAcSet'})", 'E367:')
4507
4508 call assert_true(autocmd_delete([[]]))
4509 call assert_true(autocmd_delete([test_null_dict()]))
4510endfunc
4511
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004512func Test_autocmd_split_dummy()
4513 " Autocommand trying to split a window containing a dummy buffer.
Bram Moolenaar94722c52023-01-28 19:19:03 +00004514 auto BufReadPre * exe "sbuf " .. expand("<abuf>")
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004515 " Avoid the "W11" prompt
4516 au FileChangedShell * let v:fcs_choice = 'reload'
4517 func Xautocmd_changelist()
4518 cal writefile(['Xtestfile2:4:4'], 'Xerr')
4519 edit Xerr
4520 lex 'Xtestfile2:4:4'
4521 endfunc
4522 call Xautocmd_changelist()
Bram Moolenaar53c5c9f2022-10-18 17:25:03 +01004523 " Should get E86, but it doesn't always happen (timing?)
4524 silent! call Xautocmd_changelist()
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004525
4526 au! BufReadPre
4527 au! FileChangedShell
4528 delfunc Xautocmd_changelist
4529 bwipe! Xerr
4530 call delete('Xerr')
4531endfunc
4532
Bram Moolenaare76062c2022-11-28 18:51:43 +00004533" This was crashing because there was only one window to execute autocommands
4534" in.
4535func Test_autocmd_nested_setbufvar()
4536 CheckFeature python3
4537
4538 set hidden
4539 edit Xaaa
4540 edit Xbbb
4541 call setline(1, 'bar')
4542 enew
4543 au BufWriteCmd Xbbb ++nested call setbufvar('Xaaa', '&ft', 'foo') | bw! Xaaa
4544 au FileType foo call py3eval('vim.current.buffer.options["cindent"]')
4545 wall
4546
4547 au! BufWriteCmd
4548 au! FileType foo
4549 set nohidden
4550 call delete('Xaaa')
4551 call delete('Xbbb')
4552 %bwipe!
4553endfunc
4554
Christian Brabandt9aee8ec2022-12-16 16:41:23 +00004555func SetupVimTest_shm()
4556 let g:bwe = []
4557 let g:brp = []
4558 set shortmess+=F
zeertzjq657b31f2023-04-15 21:28:02 +01004559 messages clear
Christian Brabandt9aee8ec2022-12-16 16:41:23 +00004560
4561 let dirname='XVimTestSHM'
4562 call mkdir(dirname, 'R')
4563 call writefile(['test'], dirname .. '/1')
4564 call writefile(['test'], dirname .. '/2')
4565 call writefile(['test'], dirname .. '/3')
4566
4567 augroup test
4568 autocmd!
4569 autocmd BufWinEnter * call add(g:bwe, $'BufWinEnter: {expand('<amatch>')}')
4570 autocmd BufReadPost * call add(g:brp, $'BufReadPost: {expand('<amatch>')}')
4571 augroup END
4572
4573 call setqflist([
4574 \ {'filename': dirname .. '/1', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0},
4575 \ {'filename': dirname .. '/2', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0},
4576 \ {'filename': dirname .. '/3', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0}
4577 \ ])
4578 cdo! substitute/test/TEST
4579
4580 " clean up
4581 noa enew!
4582 set shortmess&vim
4583 augroup test
4584 autocmd!
4585 augroup END
4586 augroup! test
4587endfunc
4588
4589func Test_autocmd_shortmess()
4590 CheckNotMSWindows
4591
4592 call SetupVimTest_shm()
4593 let output = execute(':mess')->split('\n')
4594
4595 let info = copy(output)->filter({idx, val -> val =~# '\d of 3'} )
4596 let bytes = copy(output)->filter({idx, val -> val =~# 'bytes'} )
4597
4598 " We test the following here:
4599 " BufReadPost should have been triggered 3 times, once per file
4600 " BufWinEnter should have been triggered 3 times, once per file
4601 " FileInfoMessage should have been shown 3 times, regardless of shm option
4602 " "(x of 3)" message from :cnext has been shown 3 times
4603
4604 call assert_equal(3, g:brp->len())
4605 call assert_equal(3, g:bwe->len())
4606 call assert_equal(3, info->len())
4607 call assert_equal(3, bytes->len())
4608
4609 delfunc SetupVimTest_shm
4610endfunc
Bram Moolenaare76062c2022-11-28 18:51:43 +00004611
Christian Brabandtf0d3d4a2024-02-15 20:15:04 +01004612func Test_autocmd_invalidates_undo_on_textchanged()
4613 CheckRunVimInTerminal
4614 let script =<< trim END
4615 set hidden
4616 " create quickfix list (at least 2 lines to move line)
4617 vimgrep /u/j %
4618
4619 " enter quickfix window
4620 cwindow
4621
4622 " set modifiable
4623 setlocal modifiable
4624
4625 " set autocmd to clear quickfix list
4626
4627 autocmd! TextChanged <buffer> call setqflist([])
4628 " move line
4629 move+1
4630 END
4631 call writefile(script, 'XTest_autocmd_invalidates_undo_on_textchanged', 'D')
4632 let buf = RunVimInTerminal('XTest_autocmd_invalidates_undo_on_textchanged', {'rows': 20})
4633 call term_sendkeys(buf, ":so %\<cr>")
4634 call term_sendkeys(buf, "G")
4635 call WaitForAssert({-> assert_match('^XTest_autocmd_invalidates_undo_on_textchanged\s*$', term_getline(buf, 20))}, 1000)
4636
4637 call StopVimInTerminal(buf)
4638endfunc
4639
Christian Brabandt55f8bba2024-02-28 23:32:00 +01004640func Test_autocmd_creates_new_buffer_on_bufleave()
4641 e a.txt
4642 e b.txt
4643 setlocal bufhidden=wipe
4644 autocmd BufLeave <buffer> diffsplit c.txt
4645 bn
4646 call assert_equal(1, winnr('$'))
4647 call assert_equal('a.txt', bufname('%'))
4648 bw a.txt
4649 bw c.txt
4650endfunc
4651
Colin Kennedye5f22802024-03-26 18:20:16 +01004652" Ensure `expected` was just recently written as a Vim session
4653func s:assert_session_path(expected)
4654 call assert_equal(a:expected, v:this_session)
4655endfunc
4656
4657" Check for `expected` after a session is written to-disk.
4658func s:watch_for_session_path(expected)
4659 execute 'autocmd SessionWritePost * ++once execute "call s:assert_session_path(\"'
4660 \ . a:expected
4661 \ . '\")"'
4662endfunc
4663
4664" Ensure v:this_session gets the full session path, if explicitly stated
4665func Test_explicit_session_absolute_path()
4666 %bwipeout!
4667
4668 let directory = getcwd()
4669
4670 let v:this_session = ""
4671 let name = "some_file.vim"
4672 let expected = fnamemodify(name, ":p")
4673 call s:watch_for_session_path(expected)
4674 execute "mksession! " .. expected
4675
4676 call delete(expected)
4677endfunc
4678
4679" Ensure v:this_session gets the full session path, if explicitly stated
4680func Test_explicit_session_relative_path()
4681 %bwipeout!
4682
4683 let directory = getcwd()
4684
4685 let v:this_session = ""
4686 let name = "some_file.vim"
4687 let expected = fnamemodify(name, ":p")
4688 call s:watch_for_session_path(expected)
4689 execute "mksession! " .. name
4690
4691 call delete(expected)
4692endfunc
4693
4694" Ensure v:this_session gets the full session path, if not specified
4695func Test_implicit_session()
4696 %bwipeout!
4697
4698 let directory = getcwd()
4699
4700 let v:this_session = ""
4701 let expected = fnamemodify("Session.vim", ":p")
4702 call s:watch_for_session_path(expected)
4703 mksession!
4704
4705 call delete(expected)
4706endfunc
4707
Christian Brabandt86032702024-03-31 18:38:09 +02004708" Test TextChangedI and TextChanged
zeertzjqc4226622024-04-03 22:38:07 +02004709func Test_Changed_ChangedI()
zeertzjq8eb75232024-04-01 14:46:20 +02004710 " Run this test in a terminal because it requires running the main loop.
zeertzjqc4226622024-04-03 22:38:07 +02004711 " Don't use CheckRunVimInTerminal as that will skip the test on Windows.
4712 CheckFeature terminal
4713 CheckNotGui
4714 " Starting a terminal to run Vim is always considered flaky.
4715 let g:test_is_flaky = 1
4716
Christian Brabandt86032702024-03-31 18:38:09 +02004717 call writefile(['one', 'two', 'three'], 'XTextChangedI2', 'D')
4718 let before =<< trim END
zeertzjqc4226622024-04-03 22:38:07 +02004719 set ttimeout ttimeoutlen=10
zeertzjq8eb75232024-04-01 14:46:20 +02004720 let [g:autocmd_n, g:autocmd_i] = ['','']
4721
4722 func TextChangedAutocmd(char)
4723 let g:autocmd_{tolower(a:char)} = a:char .. b:changedtick
zeertzjqc4226622024-04-03 22:38:07 +02004724 call writefile([$'{g:autocmd_n},{g:autocmd_i}'], 'XTextChangedI3')
zeertzjq8eb75232024-04-01 14:46:20 +02004725 endfunc
4726
4727 au TextChanged <buffer> :call TextChangedAutocmd('N')
4728 au TextChangedI <buffer> :call TextChangedAutocmd('I')
4729
Christian Brabandt86032702024-03-31 18:38:09 +02004730 nnoremap <CR> o<Esc>
zeertzjq4a653912024-04-04 21:33:36 +02004731 autocmd SafeState * ++once call writefile([''], 'XTextChangedI3')
Christian Brabandt86032702024-03-31 18:38:09 +02004732 END
4733
4734 call writefile(before, 'Xinit', 'D')
zeertzjqc4226622024-04-03 22:38:07 +02004735 let buf = term_start(
4736 \ GetVimCommandCleanTerm() .. '-n -S Xinit XTextChangedI2',
4737 \ {'term_rows': 10})
4738 call assert_equal('running', term_getstatus(buf))
zeertzjq8eb75232024-04-01 14:46:20 +02004739 call WaitForAssert({-> assert_true(filereadable('XTextChangedI3'))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004740 defer delete('XTextChangedI3')
zeertzjq4a653912024-04-04 21:33:36 +02004741 call WaitForAssert({-> assert_equal([''], readfile('XTextChangedI3'))})
Christian Brabandt86032702024-03-31 18:38:09 +02004742
zeertzjqc4226622024-04-03 22:38:07 +02004743 " TextChanged should trigger if a mapping enters and leaves Insert mode.
4744 call term_sendkeys(buf, "\<CR>")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004745 call WaitForAssert({-> assert_equal('N4,', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004746
4747 call term_sendkeys(buf, "i")
4748 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004749 call WaitForAssert({-> assert_equal('N4,', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004750 " TextChangedI should trigger if change is done in Insert mode.
4751 call term_sendkeys(buf, "f")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004752 call WaitForAssert({-> assert_equal('N4,I5', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004753 call term_sendkeys(buf, "o")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004754 call WaitForAssert({-> assert_equal('N4,I6', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004755 call term_sendkeys(buf, "o")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004756 call WaitForAssert({-> assert_equal('N4,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004757 " TextChanged shouldn't trigger when leaving Insert mode and TextChangedI
4758 " has been triggered.
4759 call term_sendkeys(buf, "\<Esc>")
4760 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004761 call WaitForAssert({-> assert_equal('N4,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004762
4763 " TextChanged should trigger if change is done in Normal mode.
4764 call term_sendkeys(buf, "yyp")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004765 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004766
4767 " TextChangedI shouldn't trigger if change isn't done in Insert mode.
4768 call term_sendkeys(buf, "i")
4769 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004770 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004771 call term_sendkeys(buf, "\<Esc>")
4772 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004773 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004774
4775 " TextChangedI should trigger if change is a mix of Normal and Insert modes.
4776 func! s:validate_mixed_textchangedi(buf, keys)
4777 let buf = a:buf
4778 call term_sendkeys(buf, "ifoo")
4779 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
4780 call term_sendkeys(buf, "\<Esc>")
4781 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
4782 call term_sendkeys(buf, ":let [g:autocmd_n, g:autocmd_i] = ['', '']\<CR>")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004783 call writefile([], 'XTextChangedI3')
zeertzjqc4226622024-04-03 22:38:07 +02004784 call term_sendkeys(buf, a:keys)
4785 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004786 call WaitForAssert({-> assert_match('^,I\d\+', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004787 call term_sendkeys(buf, "\<Esc>")
4788 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004789 call WaitForAssert({-> assert_match('^,I\d\+', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004790 endfunc
4791
4792 call s:validate_mixed_textchangedi(buf, "o")
4793 call s:validate_mixed_textchangedi(buf, "O")
4794 call s:validate_mixed_textchangedi(buf, "ciw")
4795 call s:validate_mixed_textchangedi(buf, "cc")
4796 call s:validate_mixed_textchangedi(buf, "C")
4797 call s:validate_mixed_textchangedi(buf, "s")
4798 call s:validate_mixed_textchangedi(buf, "S")
4799
4800 " clean up
4801 bwipe!
Christian Brabandt86032702024-03-31 18:38:09 +02004802endfunc
4803
zeertzjq5bf6c212024-03-31 18:41:27 +02004804" Test that filetype detection still works when SwapExists autocommand sets
4805" filetype in another buffer.
4806func Test_SwapExists_set_other_buf_filetype()
4807 let lines =<< trim END
4808 set nocompatible directory=.
4809 filetype on
4810
4811 let g:buf = bufnr()
4812 new
4813
4814 func SwapExists()
4815 let v:swapchoice = 'o'
4816 call setbufvar(g:buf, '&filetype', 'text')
4817 endfunc
4818
4819 func SafeState()
4820 edit <script>
4821 redir! > XftSwapExists.out
4822 set readonly? filetype?
4823 redir END
4824 qall!
4825 endfunc
4826
4827 autocmd SwapExists * ++nested call SwapExists()
4828 autocmd SafeState * ++nested ++once call SafeState()
4829 END
4830 call writefile(lines, 'XftSwapExists.vim', 'D')
4831
4832 new XftSwapExists.vim
4833 if RunVim('', '', ' -S XftSwapExists.vim')
4834 call assert_equal(
4835 \ ['', ' readonly', ' filetype=vim'],
4836 \ readfile('XftSwapExists.out'))
4837 call delete('XftSwapExists.out')
4838 endif
4839
4840 bwipe!
4841endfunc
4842
4843" Test that file is not marked as modified when SwapExists autocommand sets
4844" 'modified' in another buffer.
4845func Test_SwapExists_set_other_buf_modified()
4846 let lines =<< trim END
4847 set nocompatible directory=.
4848
4849 let g:buf = bufnr()
4850 new
4851
4852 func SwapExists()
4853 let v:swapchoice = 'o'
4854 call setbufvar(g:buf, '&modified', 1)
4855 endfunc
4856
4857 func SafeState()
4858 edit <script>
4859 redir! > XmodSwapExists.out
4860 set readonly? modified?
4861 redir END
4862 qall!
4863 endfunc
4864
4865 autocmd SwapExists * ++nested call SwapExists()
4866 autocmd SafeState * ++nested ++once call SafeState()
4867 END
4868 call writefile(lines, 'XmodSwapExists.vim', 'D')
4869
4870 new XmodSwapExists.vim
4871 if RunVim('', '', ' -S XmodSwapExists.vim')
4872 call assert_equal(
4873 \ ['', ' readonly', 'nomodified'],
4874 \ readfile('XmodSwapExists.out'))
4875 call delete('XmodSwapExists.out')
4876 endif
4877
4878 bwipe!
4879endfunc
4880
Jaehwang Jungeb80b832024-04-26 18:48:48 +02004881func Test_BufEnter_botline()
4882 set hidden
4883 call writefile(range(10), 'Xxx1', 'D')
4884 call writefile(range(20), 'Xxx2', 'D')
4885 edit Xxx1
4886 edit Xxx2
4887 au BufEnter Xxx1 call assert_true(line('w$') > 1)
4888 edit Xxx1
zeertzjq340643e2024-04-27 11:33:24 +02004889
4890 bwipe! Xxx1
4891 bwipe! Xxx2
Jaehwang Jungeb80b832024-04-26 18:48:48 +02004892 au! BufEnter Xxx1
4893 set hidden&vim
4894endfunc
4895
Shougo Matsushita83678842024-07-11 22:05:12 +02004896func Test_KeyInputPre()
4897 " Consume previous keys
4898 call feedkeys('', 'ntx')
4899
4900 " KeyInputPre can record input keys.
4901 let s:keys = []
4902 au KeyInputPre n call add(s:keys, v:char)
4903
4904 call feedkeys('jkjkjjj', 'ntx')
4905 call assert_equal(
4906 \ ['j', 'k', 'j', 'k', 'j', 'j', 'j'],
4907 \ s:keys)
4908
4909 unlet s:keys
4910 au! KeyInputPre
4911
4912 " KeyInputPre can handle multibyte.
4913 let s:keys = []
4914 au KeyInputPre * call add(s:keys, v:char)
4915 edit Xxx1
4916
4917 call feedkeys("iあ\<ESC>", 'ntx')
4918 call assert_equal(['i', "あ", "\<ESC>"], s:keys)
4919
4920 bwipe! Xxx1
4921 unlet s:keys
4922 au! KeyInputPre
4923
4924 " KeyInputPre can change input keys.
4925 au KeyInputPre i if v:char ==# 'a' | let v:char = 'b' | endif
4926 edit Xxx1
4927
4928 call feedkeys("iaabb\<ESC>", 'ntx')
4929 call assert_equal(getline('.'), 'bbbb')
4930
4931 bwipe! Xxx1
4932 au! KeyInputPre
4933
4934 " KeyInputPre returns multiple characters.
4935 au KeyInputPre i if v:char ==# 'a' | let v:char = 'cccc' | endif
4936 edit Xxx1
4937
4938 call feedkeys("iaabb\<ESC>", 'ntx')
4939 call assert_equal(getline('.'), 'ccbb')
4940
4941 bwipe! Xxx1
4942 au! KeyInputPre
4943
4944 " KeyInputPre can use special keys.
4945 au KeyInputPre i if v:char ==# 'a' | let v:char = "\<Ignore>" | endif
4946 edit Xxx1
4947
4948 call feedkeys("iaabb\<ESC>", 'ntx')
4949 call assert_equal(getline('.'), 'bb')
4950
4951 bwipe! Xxx1
4952 au! KeyInputPre
4953
4954 " Test for v:event.typed
4955 au KeyInputPre n call assert_true(v:event.typed)
4956 call feedkeys('j', 'ntx')
4957
4958 au! KeyInputPre
4959
4960 au KeyInputPre n call assert_false(v:event.typed)
4961 call feedkeys('j', 'nx')
4962
4963 au! KeyInputPre
Shougo Matsushitafcc1b572024-07-17 20:25:22 +02004964
4965 " Test for v:event.typedchar
4966 nnoremap j k
4967 au KeyInputPre n
4968 \ call assert_equal(v:event.typedchar, 'j')
4969 \ | call assert_equal(v:char, 'k')
4970 call feedkeys('j', 'tx')
4971
4972 au! KeyInputPre
Shougo Matsushita83678842024-07-11 22:05:12 +02004973endfunc
4974
Christian Brabandtfb3f9692024-08-11 20:09:17 +02004975" those commands caused null pointer access, see #15464
4976func Test_WinNewPre_crash()
4977 defer CleanUpTestAuGroup()
4978 let _cmdheight=&cmdheight
4979 augroup testing
4980 au!
4981 autocmd WinNewPre * redraw
4982 augroup END
4983 tabnew
4984 tabclose
4985 augroup testing
4986 au!
4987 autocmd WinNewPre * wincmd t
4988 augroup END
4989 tabnew
4990 tabclose
4991 augroup testing
4992 au!
4993 autocmd WinNewPre * wincmd b
4994 augroup END
4995 tabnew
4996 tabclose
4997 augroup testing
4998 au!
4999 autocmd WinNewPre * set cmdheight+=1
5000 augroup END
5001 tabnew
5002 tabclose
5003 let &cmdheight=_cmdheight
5004endfunc
5005
Christian Brabandt84e31752024-09-02 09:59:18 +02005006" The specifics of the turkish locale may
5007" cause that Vim will not treat the GuiEnter autocommand
5008" as case insensitive and instead issues an error
5009func Test_GuiEnter_Turkish_locale()
5010 try
5011 let lng = v:lang
5012 lang tr_TR.UTF-8
5013 let result = execute(':au GuiEnter')
Ken Takataeccc9272024-09-03 23:01:55 +02005014 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02005015 let result = execute(':au GUIENTER')
Ken Takataeccc9272024-09-03 23:01:55 +02005016 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02005017 let result = execute(':au guienter')
Ken Takataeccc9272024-09-03 23:01:55 +02005018 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02005019 exe ":lang" lng
5020 catch /E197:/
5021 " can't use Turkish locale
5022 throw 'Skipped: Turkish locale not available'
5023 endtry
5024endfunc
Christian Brabandtfb3f9692024-08-11 20:09:17 +02005025
Christian Brabandt51b62382024-10-06 17:31:10 +02005026" This was using freed memory
5027func Test_autocmd_BufWinLeave_with_vsp()
5028 new
5029 let fname = 'XXXBufWinLeaveUAF.txt'
5030 let dummy = 'XXXDummy.txt'
5031 call writefile([], fname)
5032 call writefile([], dummy)
5033 defer delete(fname)
5034 defer delete(dummy)
5035 exe "e " fname
5036 vsp
5037 augroup testing
5038 exe "au BufWinLeave " .. fname .. " :e " dummy .. "| vsp " .. fname
5039 augroup END
5040 bw
5041 call CleanUpTestAuGroup()
5042 exe "bw! " .. dummy
5043endfunc
5044
Luuk van Baale15cbc12025-01-04 17:18:08 +01005045func Test_OptionSet_cmdheight()
5046 set mouse=a laststatus=2
5047 au OptionSet cmdheight :let &l:ch = v:option_new
5048
5049 resize -1
5050 call assert_equal(2, &l:ch)
5051 resize +1
5052 call assert_equal(1, &l:ch)
5053
5054 call test_setmouse(&lines - 1, 1)
5055 call feedkeys("\<LeftMouse>", 'xt')
5056 call test_setmouse(&lines - 2, 1)
5057 call feedkeys("\<LeftDrag>", 'xt')
5058 call assert_equal(2, &l:ch)
5059
5060 tabnew | resize +1
5061 call assert_equal(1, &l:ch)
5062 tabfirst
5063 call assert_equal(2, &l:ch)
5064
5065 tabonly
5066 set cmdheight& mouse& laststatus&
5067endfunc
5068
Luuk van Baalb7147f82025-02-08 18:52:39 +01005069func Test_eventignorewin()
5070 defer CleanUpTestAuGroup()
5071 augroup testing
5072 au WinEnter * :call add(g:evs, ["WinEnter", expand("<afile>")])
5073 au WinLeave * :call add(g:evs, ["WinLeave", expand("<afile>")])
5074 au BufWinEnter * :call add(g:evs, ["BufWinEnter", expand("<afile>")])
5075 augroup END
5076
5077 let g:evs = []
5078 set eventignorewin=WinLeave,WinEnter
5079 split foo
5080 call assert_equal([['BufWinEnter', 'foo']], g:evs)
5081 set eventignorewin=all
5082 edit bar
5083 call assert_equal([['BufWinEnter', 'foo']], g:evs)
5084 set eventignorewin=
5085 wincmd w
5086 call assert_equal([['BufWinEnter', 'foo'], ['WinLeave', 'bar']], g:evs)
5087
5088 only!
5089 %bwipe!
5090 set eventignorewin&
5091 unlet g:evs
5092endfunc
5093
5094func Test_WinScrolled_Resized_eiw()
5095 CheckRunVimInTerminal
5096
5097 let lines =<< trim END
5098 call setline(1, ['foo']->repeat(32))
5099 set eventignorewin=WinScrolled,WinResized
5100 split
5101 let [g:afile,g:resized,g:scrolled] = ['none',0,0]
5102 au WinScrolled * let [g:afile,g:scrolled] = [expand('<afile>'),g:scrolled+1]
5103 au WinResized * let [g:afile,g:resized] = [expand('<afile>'),g:resized+1]
5104 END
Christian Brabandtbfc77192025-02-11 20:03:10 +01005105 call writefile(lines, 'Xtest_winscrolled_eiw', 'D')
5106 let buf = RunVimInTerminal('-S Xtest_winscrolled_eiw', {'rows': 10})
Luuk van Baalb7147f82025-02-08 18:52:39 +01005107
5108 " Both windows are ignoring resize events
5109 call term_sendkeys(buf, "\<C-W>-")
5110 call TermWait(buf)
5111 call term_sendkeys(buf, ":echo g:afile g:resized g:scrolled\<CR>")
5112 call WaitForAssert({-> assert_equal('none 0 0', term_getline(buf, 10))}, 1000)
5113
5114 " And scroll events
5115 call term_sendkeys(buf, "Ggg")
5116 call TermWait(buf)
5117 call term_sendkeys(buf, ":echo g:afile g:resized g:scrolled\<CR>")
5118 call WaitForAssert({-> assert_equal('none 0 0', term_getline(buf, 10))}, 1000)
5119
5120 " Un-ignore events in second window, make first window current and resize
5121 call term_sendkeys(buf, ":set eventignorewin=\<CR>\<C-W>w\<C-W>+")
5122 call TermWait(buf)
5123 call term_sendkeys(buf, ":echo win_getid() g:afile g:resized g:scrolled\<CR>")
5124 call WaitForAssert({-> assert_equal('1000 1001 1 1', term_getline(buf, 10))}, 1000)
5125
5126 call StopVimInTerminal(buf)
5127endfunc
5128
Jim Zhou5606ca52025-03-13 21:58:25 +01005129" Test that TabClosedPre and TabClosed are triggered when closing a tab.
5130func Test_autocmd_tabclosedpre()
5131 augroup testing
5132 au TabClosedPre * call add(g:tabpagenr_pre, t:testvar)
5133 au TabClosed * call add(g:tabpagenr_post, t:testvar)
5134 augroup END
5135
5136 " Test 'tabclose' triggering
5137 let g:tabpagenr_pre = []
5138 let g:tabpagenr_post = []
5139 let t:testvar = 1
5140 tabnew
5141 let t:testvar = 2
5142 tabnew
5143 let t:testvar = 3
5144 tabnew
5145 let t:testvar = 4
5146 tabnext
5147 tabclose
5148 tabclose
5149 tabclose
5150 call assert_equal([1, 2, 3], g:tabpagenr_pre)
5151 call assert_equal([2, 3, 4], g:tabpagenr_post)
5152
5153 " Test 'tabclose {count}' triggering
5154 let g:tabpagenr_pre = []
5155 let g:tabpagenr_post = []
5156 let t:testvar = 1
5157 tabnew
5158 let t:testvar = 2
5159 tabnew
5160 let t:testvar = 3
5161 tabclose 2
5162 tabclose 2
5163 call assert_equal([2, 3], g:tabpagenr_pre)
5164 call assert_equal([3, 1], g:tabpagenr_post)
5165
5166 " Test 'tabonly' triggering
5167 let g:tabpagenr_pre = []
5168 let g:tabpagenr_post = []
5169 let t:testvar = 1
5170 tabnew
5171 let t:testvar = 2
5172 tabonly
5173 call assert_equal([1], g:tabpagenr_pre)
5174 call assert_equal([2], g:tabpagenr_post)
5175
5176 " Test 'q' and 'close' triggering (closing the last window in a tab)
5177 let g:tabpagenr_pre = []
5178 let g:tabpagenr_post = []
5179 split
5180 let t:testvar = 1
5181 tabnew
5182 let t:testvar = 2
5183 split
5184 vsplit
5185 tabnew
5186 let t:testvar = 3
5187 tabnext
5188 only
5189 quit
5190 quit
5191 close
5192 close
5193 call assert_equal([1, 2], g:tabpagenr_pre)
5194 call assert_equal([2, 3], g:tabpagenr_post)
5195
5196 func ClearAutomcdAndCreateTabs()
5197 au! TabClosedPre
5198 bw!
5199 e Z
5200 tabonly
5201 tabnew A
5202 tabnew B
5203 tabnew C
5204 endfunc
5205
5206 func GetTabs()
5207 redir => tabsout
5208 tabs
5209 redir END
5210 let tabsout = substitute(tabsout, '\n', '', 'g')
5211 let tabsout = substitute(tabsout, 'Tab page ', '', 'g')
5212 let tabsout = substitute(tabsout, ' ', '', 'g')
5213 return tabsout
5214 endfunc
5215
5216 call CleanUpTestAuGroup()
5217
5218 " Close tab in TabClosedPre autocmd
5219 call ClearAutomcdAndCreateTabs()
5220 au TabClosedPre * tabclose
zeertzjq67fe77d2025-04-20 10:21:18 +02005221 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005222 call ClearAutomcdAndCreateTabs()
5223 au TabClosedPre * tabclose
zeertzjq67fe77d2025-04-20 10:21:18 +02005224 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005225 call ClearAutomcdAndCreateTabs()
5226 au TabClosedPre * tabclose 1
zeertzjq67fe77d2025-04-20 10:21:18 +02005227 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005228
5229 " Close other (all) tabs in TabClosedPre autocmd
5230 call ClearAutomcdAndCreateTabs()
5231 au TabClosedPre * tabonly
zeertzjq67fe77d2025-04-20 10:21:18 +02005232 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005233 call ClearAutomcdAndCreateTabs()
5234 au TabClosedPre * tabonly
zeertzjq67fe77d2025-04-20 10:21:18 +02005235 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005236 call ClearAutomcdAndCreateTabs()
5237 au TabClosedPre * tabclose 4
zeertzjq67fe77d2025-04-20 10:21:18 +02005238 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005239
5240 " Open new tabs in TabClosedPre autocmd
5241 call ClearAutomcdAndCreateTabs()
5242 au TabClosedPre * tabnew D
zeertzjq67fe77d2025-04-20 10:21:18 +02005243 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005244 call ClearAutomcdAndCreateTabs()
5245 au TabClosedPre * tabnew D
zeertzjq67fe77d2025-04-20 10:21:18 +02005246 call assert_fails('tabclose 1', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005247
5248 " Moving the tab page in TabClosedPre autocmd
5249 call ClearAutomcdAndCreateTabs()
5250 au TabClosedPre * tabmove 0
5251 tabclose
Jim Zhoubcf66e02025-03-16 20:24:57 +01005252 call assert_equal('1>Z2A3B', GetTabs())
Jim Zhou5606ca52025-03-13 21:58:25 +01005253 call ClearAutomcdAndCreateTabs()
5254 au TabClosedPre * tabmove 0
5255 tabclose 1
5256 call assert_equal('1A2B3>C', GetTabs())
5257 tabonly
5258 call assert_equal('1>C', GetTabs())
5259
5260 " Switching tab page in TabClosedPre autocmd
5261 call ClearAutomcdAndCreateTabs()
5262 au TabClosedPre * tabnext | e Y
5263 tabclose
5264 call assert_equal('1Y2A3>B', GetTabs())
5265 call ClearAutomcdAndCreateTabs()
5266 au TabClosedPre * tabnext | e Y
5267 tabclose 1
5268 call assert_equal('1Y2B3>C', GetTabs())
5269 tabonly
5270 call assert_equal('1>Y', GetTabs())
5271
5272 " Create new windows in TabClosedPre autocmd
5273 call ClearAutomcdAndCreateTabs()
5274 au TabClosedPre * split | e X| vsplit | e Y | split | e Z
zeertzjq67fe77d2025-04-20 10:21:18 +02005275 call assert_fails('tabclose', 'E242:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005276 call ClearAutomcdAndCreateTabs()
5277 au TabClosedPre * new X | new Y | new Z
zeertzjq67fe77d2025-04-20 10:21:18 +02005278 call assert_fails('tabclose 1', 'E242:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005279
Jim Zhoubcf66e02025-03-16 20:24:57 +01005280 " Test directly closing the tab page with ':tabclose'
5281 au!
5282 tabonly
5283 bw!
5284 e Z
5285 au TabClosedPre * mksession!
5286 tabnew A
5287 sp
5288 tabclose
5289 source Session.vim
5290 call assert_equal('1Z2>AA', GetTabs())
5291
5292 " Test directly closing the tab page with ':tabonly'
5293 " Z is closed before A. Hence A overwrites the session.
5294 au!
5295 tabonly
5296 bw!
5297 e Z
5298 au TabClosedPre * mksession!
5299 tabnew A
5300 tabnew B
5301 tabonly
5302 source Session.vim
5303 call assert_equal('1>A2B', GetTabs())
5304
Jim Zhou5606ca52025-03-13 21:58:25 +01005305 " Clean up
Jim Zhoubcf66e02025-03-16 20:24:57 +01005306 call delete('Session.vim')
Jim Zhou5606ca52025-03-13 21:58:25 +01005307 au!
5308 only
5309 tabonly
5310 bw!
5311endfunc
5312
Bram Moolenaarbc2b71d2020-02-17 21:33:30 +01005313" vim: shiftwidth=2 sts=2 expandtab