blob: 599153b082ee88ee04279e82d3a5566dafa3d8e9 [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 Palya46755e62025-04-27 19:28:06 +02002005 autocmd CmdlineLeavePre : 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
Girish Palya5c3d1e32025-04-22 19:52:16 +02002069 let g:count = 0
2070 autocmd CmdlineLeavePre * let g:count += 1
2071 call feedkeys(":let c = input('? ')\<cr>B\<cr>", "tx")
2072 call assert_equal(2, g:count)
2073 unlet! g:count
2074 unlet! g:log
Girish Palya46755e62025-04-27 19:28:06 +02002075 unlet! g:log2
Girish Palya92f68e22025-04-21 11:12:41 +02002076 bw!
2077endfunc
2078
Girish Palya46755e62025-04-27 19:28:06 +02002079" Ensure :cabbr does not cause a spurious CmdlineLeavePre.
2080func Test_CmdlineLeavePre_cabbr()
2081 CheckFeature terminal
2082 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2083 call assert_equal('running', term_getstatus(buf))
2084 call term_sendkeys(buf, ":let g:a=0\<cr>")
2085 call term_wait(buf, 50)
2086 call term_sendkeys(buf, ":cabbr v v\<cr>")
2087 call term_wait(buf, 50)
2088 call term_sendkeys(buf, ":command! -nargs=* Foo echo\<cr>")
2089 call term_wait(buf, 50)
2090 call term_sendkeys(buf, ":au! CmdlineLeavePre * :let g:a+=1\<cr>")
2091 call term_wait(buf, 50)
2092 call term_sendkeys(buf, ":Foo v\<cr>")
2093 call term_wait(buf, 50)
2094 call term_sendkeys(buf, ":echo g:a\<cr>")
2095 call term_wait(buf, 50)
2096 call WaitForAssert({-> assert_match('^2.*$', term_getline(buf, 3))})
2097 bwipe!
2098endfunc
2099
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002100func Test_Cmdline()
Bram Moolenaar153b7042018-01-31 15:48:32 +01002101 au! CmdlineChanged : let g:text = getcmdline()
2102 let g:text = 0
2103 call feedkeys(":echom 'hello'\<CR>", 'xt')
2104 call assert_equal("echom 'hello'", g:text)
2105 au! CmdlineChanged
2106
2107 au! CmdlineChanged : let g:entered = expand('<afile>')
2108 let g:entered = 0
2109 call feedkeys(":echom 'hello'\<CR>", 'xt')
2110 call assert_equal(':', g:entered)
2111 au! CmdlineChanged
2112
zeertzjq412e0e42023-02-11 10:34:07 +00002113 autocmd CmdlineChanged : let g:log += [getcmdline()]
2114
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002115 let g:log = []
2116 cnoremap <F1> <Cmd>call setcmdline('ls')<CR>
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002117 call feedkeys(":\<F1>", 'xt')
2118 call assert_equal(['ls'], g:log)
Bram Moolenaarbb393d82022-12-09 12:21:50 +00002119 cunmap <F1>
2120
zeertzjqaf9e28a2023-02-06 20:58:09 +00002121 let g:log = []
zeertzjqaf9e28a2023-02-06 20:58:09 +00002122 call feedkeys(":sign \<Tab>\<Tab>\<C-N>\<C-P>\<S-Tab>\<S-Tab>\<Esc>", 'xt')
2123 call assert_equal([
2124 \ 's',
2125 \ 'si',
2126 \ 'sig',
2127 \ 'sign',
2128 \ 'sign ',
2129 \ 'sign define',
2130 \ 'sign jump',
2131 \ 'sign list',
2132 \ 'sign jump',
2133 \ 'sign define',
2134 \ 'sign ',
2135 \ ], g:log)
2136 let g:log = []
2137 set wildmenu wildoptions+=pum
2138 call feedkeys(":sign \<S-Tab>\<PageUp>\<kPageUp>\<kPageDown>\<PageDown>\<Esc>", 'xt')
2139 call assert_equal([
2140 \ 's',
2141 \ 'si',
2142 \ 'sig',
2143 \ 'sign',
2144 \ 'sign ',
2145 \ 'sign unplace',
2146 \ 'sign jump',
2147 \ 'sign define',
2148 \ 'sign undefine',
2149 \ 'sign unplace',
2150 \ ], g:log)
2151 set wildmenu& wildoptions&
zeertzjq412e0e42023-02-11 10:34:07 +00002152
2153 let g:log = []
2154 let @r = 'abc'
2155 call feedkeys(":0\<C-R>r1\<C-R>\<C-O>r2\<C-R>\<C-R>r3\<Esc>", 'xt')
2156 call assert_equal([
2157 \ '0',
2158 \ '0a',
2159 \ '0ab',
2160 \ '0abc',
2161 \ '0abc1',
2162 \ '0abc1abc',
2163 \ '0abc1abc2',
2164 \ '0abc1abc2abc',
2165 \ '0abc1abc2abc3',
2166 \ ], g:log)
2167
zeertzjqaf9e28a2023-02-06 20:58:09 +00002168 unlet g:log
2169 au! CmdlineChanged
2170
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002171 au! CmdlineEnter : let g:entered = expand('<afile>')
2172 au! CmdlineLeave : let g:left = expand('<afile>')
Girish Palya92f68e22025-04-21 11:12:41 +02002173 au! CmdlineLeavePre : let g:leftpre = expand('<afile>')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002174 let g:entered = 0
2175 let g:left = 0
Girish Palya92f68e22025-04-21 11:12:41 +02002176 let g:leftpre = 0
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002177 call feedkeys(":echo 'hello'\<CR>", 'xt')
2178 call assert_equal(':', g:entered)
2179 call assert_equal(':', g:left)
Girish Palya92f68e22025-04-21 11:12:41 +02002180 call assert_equal(':', g:leftpre)
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002181 au! CmdlineEnter
2182 au! CmdlineLeave
Girish Palya92f68e22025-04-21 11:12:41 +02002183 au! CmdlineLeavePre
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002184
Bram Moolenaara4baf5b2018-04-22 13:27:44 +02002185 let save_shellslash = &shellslash
2186 set noshellslash
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002187 au! CmdlineEnter / let g:entered = expand('<afile>')
2188 au! CmdlineLeave / let g:left = expand('<afile>')
Girish Palya92f68e22025-04-21 11:12:41 +02002189 au! CmdlineLeavePre / let g:leftpre = expand('<afile>')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002190 let g:entered = 0
2191 let g:left = 0
Girish Palya92f68e22025-04-21 11:12:41 +02002192 let g:leftpre = 0
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002193 new
2194 call setline(1, 'hello')
2195 call feedkeys("/hello\<CR>", 'xt')
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002196 call assert_equal('/', g:entered)
2197 call assert_equal('/', g:left)
Girish Palya92f68e22025-04-21 11:12:41 +02002198 call assert_equal('/', g:leftpre)
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002199 bwipe!
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002200 au! CmdlineEnter
2201 au! CmdlineLeave
Girish Palya92f68e22025-04-21 11:12:41 +02002202 au! CmdlineLeavePre
Bram Moolenaara4baf5b2018-04-22 13:27:44 +02002203 let &shellslash = save_shellslash
Shougo Matsushitad0952142024-06-20 22:05:16 +02002204
Girish Palya92f68e22025-04-21 11:12:41 +02002205 let g:left = "cancelled"
2206 let g:leftpre = "cancelled"
2207 au! CmdlineLeave : let g:left = "triggered"
2208 au! CmdlineLeavePre : let g:leftpre = "triggered"
2209 call feedkeys(":echo 'hello'\<esc>", 'xt')
2210 call assert_equal('triggered', g:left)
2211 call assert_equal('triggered', g:leftpre)
2212 let g:left = "cancelled"
2213 let g:leftpre = "cancelled"
2214 au! CmdlineLeave : let g:left = "triggered"
2215 call feedkeys(":echo 'hello'\<c-c>", 'xt')
2216 call assert_equal('triggered', g:left)
2217 call assert_equal('triggered', g:leftpre)
2218 au! CmdlineLeave
2219 au! CmdlineLeavePre
2220
zeertzjqbc6f9672024-06-21 07:51:40 +02002221 au! CursorMovedC : let g:pos += [getcmdpos()]
2222 let g:pos = []
zeertzjq81456202024-07-07 20:48:25 +02002223 call feedkeys(":foo bar baz\<C-W>\<C-W>\<C-W>\<Esc>", 'xt')
2224 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 9, 5, 1], g:pos)
2225 let g:pos = []
2226 call feedkeys(":hello\<C-B>\<Esc>", 'xt')
2227 call assert_equal([2, 3, 4, 5, 6, 1], g:pos)
2228 let g:pos = []
2229 call feedkeys(":hello\<C-U>\<Esc>", 'xt')
2230 call assert_equal([2, 3, 4, 5, 6, 1], g:pos)
2231 let g:pos = []
zeertzjqbc6f9672024-06-21 07:51:40 +02002232 call feedkeys(":hello\<Left>\<C-R>=''\<CR>\<Left>\<Right>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002233 call assert_equal([2, 3, 4, 5, 6, 5, 4, 5], g:pos)
zeertzjqbc6f9672024-06-21 07:51:40 +02002234 let g:pos = []
2235 call feedkeys(":12345678\<C-R>=setcmdpos(3)??''\<CR>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002236 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 3], g:pos)
zeertzjqbc6f9672024-06-21 07:51:40 +02002237 let g:pos = []
2238 call feedkeys(":12345678\<C-R>=setcmdpos(3)??''\<CR>\<Left>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002239 call assert_equal([2, 3, 4, 5, 6, 7, 8, 9, 3, 2], g:pos)
Shougo Matsushitad0952142024-06-20 22:05:16 +02002240 au! CursorMovedC
2241
zeertzjqbc6f9672024-06-21 07:51:40 +02002242 " setcmdpos() is no-op inside an autocommand
2243 au! CursorMovedC : let g:pos += [getcmdpos()] | call setcmdpos(1)
2244 let g:pos = []
2245 call feedkeys(":hello\<Left>\<Left>\<Esc>", 'xt')
zeertzjq81456202024-07-07 20:48:25 +02002246 call assert_equal([2, 3, 4, 5, 6, 5, 4], g:pos)
Shougo Matsushitad0952142024-06-20 22:05:16 +02002247 au! CursorMovedC
zeertzjqbc6f9672024-06-21 07:51:40 +02002248
2249 unlet g:entered
2250 unlet g:left
2251 unlet g:pos
Bram Moolenaarfafcf0d2017-10-19 18:35:51 +02002252endfunc
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002253
2254" Test for BufWritePre autocommand that deletes or unloads the buffer.
2255func Test_BufWritePre()
2256 %bwipe
2257 au BufWritePre Xxx1 bunload
2258 au BufWritePre Xxx2 bwipe
2259
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002260 call writefile(['start of Xxx1', 'test', 'end of Xxx1'], 'Xxx1', 'D')
2261 call writefile(['start of Xxx2', 'test', 'end of Xxx2'], 'Xxx2', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002262
2263 edit Xtest
2264 e! Xxx2
2265 bdel Xtest
2266 e Xxx1
2267 " write it, will unload it and give an error msg
Bram Moolenaare2e40752020-09-04 21:18:46 +02002268 call assert_fails('w', 'E203:')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002269 call assert_equal('Xxx2', bufname('%'))
2270 edit Xtest
2271 e! Xxx2
2272 bwipe Xtest
2273 " write it, will delete the buffer and give an error msg
Bram Moolenaare2e40752020-09-04 21:18:46 +02002274 call assert_fails('w', 'E203:')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002275 call assert_equal('Xxx1', bufname('%'))
2276 au! BufWritePre
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002277endfunc
2278
2279" Test for BufUnload autocommand that unloads all the other buffers
2280func Test_bufunload_all()
Bram Moolenaarf08b0eb2021-10-16 13:00:14 +01002281 let g:test_is_flaky = 1
Christian Brabandtee17b6f2023-09-09 11:23:50 +02002282 call writefile(['Test file Xxx1'], 'Xxx1', 'D')
2283 call writefile(['Test file Xxx2'], 'Xxx2', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002284
Bram Moolenaarc79745a2019-05-20 22:12:34 +02002285 let content =<< trim [CODE]
2286 func UnloadAllBufs()
2287 let i = 1
2288 while i <= bufnr('$')
2289 if i != bufnr('%') && bufloaded(i)
2290 exe i . 'bunload'
2291 endif
2292 let i += 1
2293 endwhile
2294 endfunc
2295 au BufUnload * call UnloadAllBufs()
2296 au VimLeave * call writefile(['Test Finished'], 'Xout')
2297 edit Xxx1
2298 split Xxx2
2299 q
2300 [CODE]
2301
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002302 call writefile(content, 'Xbunloadtest', 'D')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002303
2304 call delete('Xout')
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002305 call system(GetVimCommandClean() .. ' -N --not-a-term -S Xbunloadtest')
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002306 call assert_true(filereadable('Xout'))
2307
Bram Moolenaar53f0c962017-10-22 14:23:59 +02002308 call delete('Xout')
2309endfunc
2310
2311" Some tests for buffer-local autocommands
2312func Test_buflocal_autocmd()
2313 let g:bname = ''
2314 edit xx
2315 au BufLeave <buffer> let g:bname = expand("%")
2316 " here, autocommand for xx should trigger.
2317 " but autocommand shall not apply to buffer named <buffer>.
2318 edit somefile
2319 call assert_equal('xx', g:bname)
2320 let g:bname = ''
2321 " here, autocommand shall be auto-deleted
2322 bwipe xx
2323 " autocmd should not trigger
2324 edit xx
2325 call assert_equal('', g:bname)
2326 " autocmd should not trigger
2327 edit somefile
2328 call assert_equal('', g:bname)
2329 enew
2330 unlet g:bname
2331endfunc
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002332
2333" Test for "*Cmd" autocommands
2334func Test_Cmd_Autocmds()
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002335 call writefile(['start of Xxx', "\tabc2", 'end of Xxx'], 'Xxx', 'D')
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002336
2337 enew!
2338 au BufReadCmd XtestA 0r Xxx|$del
2339 edit XtestA " will read text of Xxd instead
2340 call assert_equal('start of Xxx', getline(1))
2341
2342 au BufWriteCmd XtestA call append(line("$"), "write")
2343 write " will append a line to the file
2344 call assert_equal('write', getline('$'))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002345 call assert_fails('read XtestA', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002346 call assert_equal('write', getline(4))
2347
2348 " now we have:
2349 " 1 start of Xxx
2350 " 2 abc2
2351 " 3 end of Xxx
2352 " 4 write
2353
2354 au FileReadCmd XtestB '[r Xxx
2355 2r XtestB " will read Xxx below line 2 instead
2356 call assert_equal('start of Xxx', getline(3))
2357
2358 " now we have:
2359 " 1 start of Xxx
2360 " 2 abc2
2361 " 3 start of Xxx
2362 " 4 abc2
2363 " 5 end of Xxx
2364 " 6 end of Xxx
2365 " 7 write
2366
2367 au FileWriteCmd XtestC '[,']copy $
2368 normal 4GA1
2369 4,5w XtestC " will copy lines 4 and 5 to the end
2370 call assert_equal("\tabc21", getline(8))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002371 call assert_fails('r XtestC', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002372 call assert_equal("end of Xxx", getline(9))
2373
2374 " now we have:
2375 " 1 start of Xxx
2376 " 2 abc2
2377 " 3 start of Xxx
2378 " 4 abc21
2379 " 5 end of Xxx
2380 " 6 end of Xxx
2381 " 7 write
2382 " 8 abc21
2383 " 9 end of Xxx
2384
2385 let g:lines = []
2386 au FileAppendCmd XtestD call extend(g:lines, getline(line("'["), line("']")))
2387 w >>XtestD " will add lines to 'lines'
2388 call assert_equal(9, len(g:lines))
Bram Moolenaare2e40752020-09-04 21:18:46 +02002389 call assert_fails('$r XtestD', 'E484:') " should not read anything
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002390 call assert_equal(9, line('$'))
2391 call assert_equal('end of Xxx', getline('$'))
2392
2393 au BufReadCmd XtestE 0r Xxx|$del
2394 sp XtestE " split window with test.out
2395 call assert_equal('end of Xxx', getline(3))
2396
2397 let g:lines = []
2398 exe "normal 2Goasdf\<Esc>\<C-W>\<C-W>"
2399 au BufWriteCmd XtestE call extend(g:lines, getline(0, '$'))
2400 wall " will write other window to 'lines'
2401 call assert_equal(4, len(g:lines), g:lines)
2402 call assert_equal('asdf', g:lines[2])
2403
2404 au! BufReadCmd
2405 au! BufWriteCmd
2406 au! FileReadCmd
2407 au! FileWriteCmd
2408 au! FileAppendCmd
2409 %bwipe!
Bram Moolenaar430dc5d2017-11-02 21:04:47 +01002410 enew!
2411endfunc
Bram Moolenaaraace2152017-11-05 16:23:10 +01002412
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002413func s:ReadFile()
2414 setl noswapfile nomodified
2415 let filename = resolve(expand("<afile>:p"))
2416 execute 'read' fnameescape(filename)
2417 1d_
2418 exe 'file' fnameescape(filename)
2419 setl buftype=acwrite
2420endfunc
2421
2422func s:WriteFile()
2423 let filename = resolve(expand("<afile>:p"))
2424 setl buftype=
2425 noautocmd execute 'write' fnameescape(filename)
2426 setl buftype=acwrite
2427 setl nomodified
2428endfunc
2429
2430func Test_BufReadCmd()
2431 autocmd BufReadCmd *.test call s:ReadFile()
2432 autocmd BufWriteCmd *.test call s:WriteFile()
2433
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002434 call writefile(['one', 'two', 'three'], 'Xcmd.test', 'D')
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002435 edit Xcmd.test
2436 call assert_match('Xcmd.test" line 1 of 3', execute('file'))
2437 normal! Gofour
2438 write
2439 call assert_equal(['one', 'two', 'three', 'four'], readfile('Xcmd.test'))
2440
2441 bwipe!
Bram Moolenaar0fff4412020-03-29 16:06:29 +02002442 au! BufReadCmd
2443 au! BufWriteCmd
2444endfunc
2445
zeertzjq9c8f9462022-08-30 18:17:15 +01002446func Test_BufWriteCmd()
2447 autocmd BufWriteCmd Xbufwritecmd let g:written = 1
2448 new
2449 file Xbufwritecmd
2450 set buftype=acwrite
Bram Moolenaar6f14da12022-09-07 21:30:44 +01002451 call mkdir('Xbufwritecmd', 'D')
zeertzjq9c8f9462022-08-30 18:17:15 +01002452 write
2453 " BufWriteCmd should be triggered even if a directory has the same name
2454 call assert_equal(1, g:written)
zeertzjq9c8f9462022-08-30 18:17:15 +01002455 unlet g:written
2456 au! BufWriteCmd
2457 bwipe!
2458endfunc
2459
Bram Moolenaaraace2152017-11-05 16:23:10 +01002460func SetChangeMarks(start, end)
Bram Moolenaar97c69432021-01-15 16:45:21 +01002461 exe a:start .. 'mark ['
2462 exe a:end .. 'mark ]'
Bram Moolenaaraace2152017-11-05 16:23:10 +01002463endfunc
2464
2465" Verify the effects of autocmds on '[ and ']
2466func Test_change_mark_in_autocmds()
2467 edit! Xtest
Bram Moolenaar97c69432021-01-15 16:45:21 +01002468 call feedkeys("ia\<CR>b\<CR>c\<CR>d\<C-g>u\<Esc>", 'xtn')
Bram Moolenaaraace2152017-11-05 16:23:10 +01002469
2470 call SetChangeMarks(2, 3)
2471 write
2472 call assert_equal([1, 4], [line("'["), line("']")])
2473
2474 call SetChangeMarks(2, 3)
2475 au BufWritePre * call assert_equal([1, 4], [line("'["), line("']")])
2476 write
2477 au! BufWritePre
2478
Bram Moolenaar14ddd222020-08-05 12:02:40 +02002479 if has('unix')
Bram Moolenaaraace2152017-11-05 16:23:10 +01002480 write XtestFilter
2481 write >> XtestFilter
2482
2483 call SetChangeMarks(2, 3)
2484 " Marks are set to the entire range of the write
2485 au FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
2486 " '[ is adjusted to just before the line that will receive the filtered
2487 " data
2488 au FilterReadPre * call assert_equal([4, 4], [line("'["), line("']")])
2489 " The filtered data is read into the buffer, and the source lines are
2490 " still present, so the range is after the source lines
2491 au FilterReadPost * call assert_equal([5, 12], [line("'["), line("']")])
2492 %!cat XtestFilter
2493 " After the filtered data is read, the original lines are deleted
2494 call assert_equal([1, 8], [line("'["), line("']")])
2495 au! FilterWritePre,FilterReadPre,FilterReadPost
2496 undo
2497
2498 call SetChangeMarks(1, 4)
2499 au FilterWritePre * call assert_equal([2, 3], [line("'["), line("']")])
2500 au FilterReadPre * call assert_equal([3, 3], [line("'["), line("']")])
2501 au FilterReadPost * call assert_equal([4, 11], [line("'["), line("']")])
2502 2,3!cat XtestFilter
2503 call assert_equal([2, 9], [line("'["), line("']")])
2504 au! FilterWritePre,FilterReadPre,FilterReadPost
2505 undo
2506
2507 call delete('XtestFilter')
2508 endif
2509
2510 call SetChangeMarks(1, 4)
2511 au FileWritePre * call assert_equal([2, 3], [line("'["), line("']")])
2512 2,3write Xtest2
2513 au! FileWritePre
2514
2515 call SetChangeMarks(2, 3)
2516 au FileAppendPre * call assert_equal([1, 4], [line("'["), line("']")])
2517 write >> Xtest2
2518 au! FileAppendPre
2519
2520 call SetChangeMarks(1, 4)
2521 au FileAppendPre * call assert_equal([2, 3], [line("'["), line("']")])
2522 2,3write >> Xtest2
2523 au! FileAppendPre
2524
2525 call SetChangeMarks(1, 1)
2526 au FileReadPre * call assert_equal([3, 1], [line("'["), line("']")])
2527 au FileReadPost * call assert_equal([4, 11], [line("'["), line("']")])
2528 3read Xtest2
2529 au! FileReadPre,FileReadPost
2530 undo
2531
2532 call SetChangeMarks(4, 4)
2533 " When the line is 0, it's adjusted to 1
2534 au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
2535 au FileReadPost * call assert_equal([1, 8], [line("'["), line("']")])
2536 0read Xtest2
2537 au! FileReadPre,FileReadPost
2538 undo
2539
2540 call SetChangeMarks(4, 4)
2541 " When the line is 0, it's adjusted to 1
2542 au FileReadPre * call assert_equal([1, 4], [line("'["), line("']")])
2543 au FileReadPost * call assert_equal([2, 9], [line("'["), line("']")])
2544 1read Xtest2
2545 au! FileReadPre,FileReadPost
2546 undo
2547
2548 bwipe!
2549 call delete('Xtest')
2550 call delete('Xtest2')
2551endfunc
2552
2553func Test_Filter_noshelltemp()
Bram Moolenaaraeb313f2020-11-27 19:13:28 +01002554 CheckExecutable cat
Bram Moolenaaraace2152017-11-05 16:23:10 +01002555
2556 enew!
2557 call setline(1, ['a', 'b', 'c', 'd'])
2558
2559 let shelltemp = &shelltemp
2560 set shelltemp
2561
2562 let g:filter_au = 0
2563 au FilterWritePre * let g:filter_au += 1
2564 au FilterReadPre * let g:filter_au += 1
2565 au FilterReadPost * let g:filter_au += 1
2566 %!cat
2567 call assert_equal(3, g:filter_au)
2568
2569 if has('filterpipe')
2570 set noshelltemp
2571
2572 let g:filter_au = 0
2573 au FilterWritePre * let g:filter_au += 1
2574 au FilterReadPre * let g:filter_au += 1
2575 au FilterReadPost * let g:filter_au += 1
2576 %!cat
2577 call assert_equal(0, g:filter_au)
2578 endif
2579
2580 au! FilterWritePre,FilterReadPre,FilterReadPost
2581 let &shelltemp = shelltemp
2582 bwipe!
2583endfunc
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002584
2585func Test_TextYankPost()
2586 enew!
2587 call setline(1, ['foo'])
2588
2589 let g:event = []
2590 au TextYankPost * let g:event = copy(v:event)
2591
2592 call assert_equal({}, v:event)
2593 call assert_fails('let v:event = {}', 'E46:')
2594 call assert_fails('let v:event.mykey = 0', 'E742:')
2595
2596 norm "ayiw
2597 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002598 \ #{regcontents: ['foo'], regname: 'a', operator: 'y',
2599 \ regtype: 'v', visual: v:false, inclusive: v:true},
2600 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002601 norm y_
2602 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002603 \ #{regcontents: ['foo'], regname: '', operator: 'y', regtype: 'V',
2604 \ visual: v:false, inclusive: v:false},
2605 \ g:event)
Bram Moolenaar37d16732020-06-12 22:09:01 +02002606 norm Vy
2607 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002608 \ #{regcontents: ['foo'], regname: '', operator: 'y', regtype: 'V',
2609 \ visual: v:true, inclusive: v:true},
2610 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002611 call feedkeys("\<C-V>y", 'x')
2612 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002613 \ #{regcontents: ['f'], regname: '', operator: 'y', regtype: "\x161",
2614 \ visual: v:true, inclusive: v:true},
2615 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002616 norm "xciwbar
2617 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002618 \ #{regcontents: ['foo'], regname: 'x', operator: 'c', regtype: 'v',
2619 \ visual: v:false, inclusive: v:true},
2620 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002621 norm "bdiw
2622 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002623 \ #{regcontents: ['bar'], regname: 'b', operator: 'd', regtype: 'v',
2624 \ visual: v:false, inclusive: v:true},
2625 \ g:event)
2626
2627 call setline(1, 'foobar')
2628 " exclusive motion
2629 norm $"ay0
2630 call assert_equal(
2631 \ #{regcontents: ['fooba'], regname: 'a', operator: 'y', regtype: 'v',
2632 \ visual: v:false, inclusive: v:false},
2633 \ g:event)
2634 " inclusive motion
2635 norm 0"ay$
2636 call assert_equal(
2637 \ #{regcontents: ['foobar'], regname: 'a', operator: 'y', regtype: 'v',
2638 \ visual: v:false, inclusive: v:true},
2639 \ g:event)
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002640
2641 call assert_equal({}, v:event)
2642
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002643 if has('clipboard_working') && !has('gui_running')
2644 " Test that when the visual selection is automatically copied to clipboard
2645 " register a TextYankPost is emitted
2646 call setline(1, ['foobar'])
2647
2648 let @* = ''
2649 set clipboard=autoselect
2650 exe "norm! ggviw\<Esc>"
2651 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002652 \ #{regcontents: ['foobar'], regname: '*', operator: 'y',
2653 \ regtype: 'v', visual: v:true, inclusive: v:false},
2654 \ g:event)
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002655
2656 let @+ = ''
2657 set clipboard=autoselectplus
2658 exe "norm! ggviw\<Esc>"
2659 call assert_equal(
Bram Moolenaara016eeb2022-04-09 11:37:38 +01002660 \ #{regcontents: ['foobar'], regname: '+', operator: 'y',
2661 \ regtype: 'v', visual: v:true, inclusive: v:false},
2662 \ g:event)
Bram Moolenaarfccbf062020-11-26 20:34:00 +01002663
2664 set clipboard&vim
2665 endif
2666
Bram Moolenaar7e1652c2017-12-16 18:27:02 +01002667 au! TextYankPost
2668 unlet g:event
2669 bwipe!
2670endfunc
Bram Moolenaar9bca8052017-12-18 12:37:55 +01002671
Bram Moolenaar9a046fd2021-01-28 13:47:59 +01002672func Test_autocommand_all_events()
2673 call assert_fails('au * * bwipe', 'E1155:')
2674 call assert_fails('au * x bwipe', 'E1155:')
Bram Moolenaarb6db1462021-12-24 19:24:47 +00002675 call assert_fails('au! * x bwipe', 'E1155:')
Bram Moolenaar4fb921e2017-12-18 15:33:00 +01002676endfunc
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002677
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002678func Test_autocmd_user()
2679 au User MyEvent let s:res = [expand("<afile>"), expand("<amatch>")]
2680 doautocmd User MyEvent
2681 call assert_equal(['MyEvent', 'MyEvent'], s:res)
2682 au! User
2683 unlet s:res
2684endfunc
2685
Bram Moolenaar3b014be2022-11-13 17:53:46 +00002686func Test_autocmd_user_clear_group()
2687 CheckRunVimInTerminal
2688
2689 let lines =<< trim END
2690 autocmd! User
2691 for i in range(1, 999)
2692 exe 'autocmd User ' .. 'Foo' .. i .. ' bar'
2693 endfor
2694 au CmdlineLeave : call timer_start(0, {-> execute('autocmd! User')})
2695 END
2696 call writefile(lines, 'XautoUser', 'D')
2697 let buf = RunVimInTerminal('-S XautoUser', {'rows': 10})
2698
2699 " this was using freed memory
2700 call term_sendkeys(buf, ":autocmd User\<CR>")
2701 call TermWait(buf, 50)
2702 call term_sendkeys(buf, "G")
2703
2704 call StopVimInTerminal(buf)
2705endfunc
2706
Bram Moolenaaref2c3252022-11-25 16:31:51 +00002707func Test_autocmd_CmdlineLeave_unlet()
2708 CheckRunVimInTerminal
2709
2710 let lines =<< trim END
2711 for i in range(1, 999)
2712 exe 'let g:var' .. i '=' i
2713 endfor
2714 au CmdlineLeave : call timer_start(0, {-> execute('unlet g:var990')})
2715 END
2716 call writefile(lines, 'XleaveUnlet', 'D')
2717 let buf = RunVimInTerminal('-S XleaveUnlet', {'rows': 10})
2718
2719 " this was using freed memory
2720 call term_sendkeys(buf, ":let g:\<CR>")
2721 call TermWait(buf, 50)
2722 call term_sendkeys(buf, "G")
2723 call TermWait(buf, 50)
2724 call term_sendkeys(buf, "\<CR>") " for the hit-enter prompt
2725
2726 call StopVimInTerminal(buf)
2727endfunc
2728
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002729function s:Before_test_dirchanged()
2730 augroup test_dirchanged
2731 autocmd!
2732 augroup END
2733 let s:li = []
2734 let s:dir_this = getcwd()
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02002735 let s:dir_foo = s:dir_this . '/Xfoo'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002736 call mkdir(s:dir_foo)
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02002737 let s:dir_bar = s:dir_this . '/Xbar'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002738 call mkdir(s:dir_bar)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002739endfunc
2740
2741function s:After_test_dirchanged()
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002742 call chdir(s:dir_this)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002743 call delete(s:dir_foo, 'd')
2744 call delete(s:dir_bar, 'd')
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002745 augroup test_dirchanged
2746 autocmd!
2747 augroup END
2748endfunc
2749
2750function Test_dirchanged_global()
2751 call s:Before_test_dirchanged()
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002752 autocmd test_dirchanged DirChangedPre global call add(s:li, expand("<amatch>") .. " pre cd " .. v:event.directory)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002753 autocmd test_dirchanged DirChanged global call add(s:li, "cd:")
2754 autocmd test_dirchanged DirChanged global call add(s:li, expand("<afile>"))
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002755 call chdir(s:dir_foo)
Bram Moolenaarf6246f52022-02-11 16:30:12 +00002756 let expected = ["global pre cd " .. s:dir_foo, "cd:", s:dir_foo]
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002757 call assert_equal(expected, s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002758 call chdir(s:dir_foo)
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002759 call assert_equal(expected, s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002760 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002761 call assert_equal(expected, s:li)
Bram Moolenaard8c9d322022-06-12 11:49:16 +01002762
2763 exe 'cd ' .. s:dir_foo
2764 exe 'cd ' .. s:dir_bar
2765 autocmd! test_dirchanged DirChanged global let g:result = expand("<afile>")
2766 cd -
Bram Moolenaardb77c492022-06-12 23:26:50 +01002767 call assert_equal(s:dir_foo, substitute(g:result, '\\', '/', 'g'))
Bram Moolenaard8c9d322022-06-12 11:49:16 +01002768
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002769 call s:After_test_dirchanged()
2770endfunc
2771
2772function Test_dirchanged_local()
2773 call s:Before_test_dirchanged()
2774 autocmd test_dirchanged DirChanged window call add(s:li, "lcd:")
2775 autocmd test_dirchanged DirChanged window call add(s:li, expand("<afile>"))
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002776 call chdir(s:dir_foo)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002777 call assert_equal([], s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002778 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002779 call assert_equal(["lcd:", s:dir_bar], s:li)
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002780 exe 'lcd ' .. fnameescape(s:dir_bar)
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002781 call assert_equal(["lcd:", s:dir_bar], s:li)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002782 call s:After_test_dirchanged()
2783endfunc
2784
2785function Test_dirchanged_auto()
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02002786 CheckOption autochdir
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002787 call s:Before_test_dirchanged()
2788 call test_autochdir()
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002789 autocmd test_dirchanged DirChangedPre auto call add(s:li, "pre cd " .. v:event.directory)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002790 autocmd test_dirchanged DirChanged auto call add(s:li, "auto:")
2791 autocmd test_dirchanged DirChanged auto call add(s:li, expand("<afile>"))
2792 set acd
Bram Moolenaar3503d7c2019-11-09 20:10:17 +01002793 cd ..
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002794 call assert_equal([], s:li)
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01002795 exe 'edit ' . s:dir_foo . '/Xautofile'
Bram Moolenaar2caad3f2018-12-16 15:38:02 +01002796 call assert_equal(s:dir_foo, getcwd())
Bram Moolenaar28e8f732022-02-09 12:58:20 +00002797 let expected = ["pre cd " .. s:dir_foo, "auto:", s:dir_foo]
2798 call assert_equal(expected, s:li)
Bram Moolenaarb7407d32018-02-03 17:36:27 +01002799 set noacd
2800 bwipe!
2801 call s:After_test_dirchanged()
2802endfunc
Bram Moolenaar5a093432018-02-10 18:15:19 +01002803
2804" Test TextChangedI and TextChangedP
2805func Test_ChangedP()
2806 new
2807 call setline(1, ['foo', 'bar', 'foobar'])
2808 call test_override("char_avail", 1)
2809 set complete=. completeopt=menuone
2810
2811 func! TextChangedAutocmd(char)
2812 let g:autocmd .= a:char
2813 endfunc
2814
Christian Brabandtdb3b4462021-10-16 11:58:55 +01002815 " TextChanged will not be triggered, only check that it isn't.
Bram Moolenaar5a093432018-02-10 18:15:19 +01002816 au! TextChanged <buffer> :call TextChangedAutocmd('N')
2817 au! TextChangedI <buffer> :call TextChangedAutocmd('I')
2818 au! TextChangedP <buffer> :call TextChangedAutocmd('P')
2819
2820 call cursor(3, 1)
2821 let g:autocmd = ''
2822 call feedkeys("o\<esc>", 'tnix')
Evgeni Chasnovskid7ae2632023-10-15 09:59:00 +02002823 call assert_equal('I', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002824
2825 let g:autocmd = ''
Christian Brabandt4bca4892023-10-27 19:26:49 +02002826 call feedkeys("Sf", 'tnix')
2827 call assert_equal('II', g:autocmd)
2828
2829 let g:autocmd = ''
Bram Moolenaar5a093432018-02-10 18:15:19 +01002830 call feedkeys("Sf\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002831 call assert_equal('IIP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002832
2833 let g:autocmd = ''
2834 call feedkeys("Sf\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002835 call assert_equal('IIPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002836
2837 let g:autocmd = ''
2838 call feedkeys("Sf\<C-N>\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002839 call assert_equal('IIPPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002840
2841 let g:autocmd = ''
2842 call feedkeys("Sf\<C-N>\<C-N>\<C-N>\<C-N>", 'tnix')
Christian Brabandt4bca4892023-10-27 19:26:49 +02002843 call assert_equal('IIPPPP', g:autocmd)
Bram Moolenaar5a093432018-02-10 18:15:19 +01002844
2845 call assert_equal(['foo', 'bar', 'foobar', 'foo'], getline(1, '$'))
2846 " TODO: how should it handle completeopt=noinsert,noselect?
2847
2848 " CleanUp
2849 call test_override("char_avail", 0)
2850 au! TextChanged
2851 au! TextChangedI
2852 au! TextChangedP
2853 delfu TextChangedAutocmd
2854 unlet! g:autocmd
2855 set complete&vim completeopt&vim
2856
2857 bw!
2858endfunc
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002859
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002860let g:setline_handled = v:false
Bram Moolenaar1e115362019-01-09 23:01:02 +01002861func SetLineOne()
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002862 if !g:setline_handled
2863 call setline(1, "(x)")
2864 let g:setline_handled = v:true
2865 endif
2866endfunc
2867
2868func Test_TextChangedI_with_setline()
2869 new
2870 call test_override('char_avail', 1)
2871 autocmd TextChangedI <buffer> call SetLineOne()
2872 call feedkeys("i(\<CR>\<Esc>", 'tx')
2873 call assert_equal('(', getline(1))
2874 call assert_equal('x)', getline(2))
2875 undo
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002876 call assert_equal('', getline(1))
Bram Moolenaar9fa95062018-08-08 22:08:32 +02002877 call assert_equal('', getline(2))
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002878
Bram Moolenaarca34db32022-01-20 11:17:18 +00002879 call test_override('char_avail', 0)
Bram Moolenaar91d2e782018-08-07 19:05:01 +02002880 bwipe!
2881endfunc
2882
Christian Brabandtc9e79e52024-02-09 19:34:36 +01002883func Test_TextChanged_with_norm()
2884 " For unknown reason this fails on MS-Windows
2885 CheckNotMSWindows
2886 CheckFeature terminal
2887 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2888 call assert_equal('running', term_getstatus(buf))
2889 call term_sendkeys(buf, ":let g:a=0\<cr>")
2890 call term_wait(buf, 50)
2891 call term_sendkeys(buf, ":au! TextChanged * :let g:a+=1\<cr>")
2892 call term_wait(buf, 50)
2893 call term_sendkeys(buf, ":norm! ia\<cr>")
2894 call term_wait(buf, 50)
2895 call term_sendkeys(buf, ":echo g:a\<cr>")
2896 call term_wait(buf, 50)
2897 call WaitForAssert({-> assert_match('^1.*$', term_getline(buf, 3))})
2898 bwipe!
2899endfunc
2900
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002901func Test_Changed_FirstTime()
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02002902 CheckFeature terminal
2903 CheckNotGui
Bram Moolenaar3cdcb092020-03-18 19:18:10 +01002904 " Starting a terminal to run Vim is always considered flaky.
Bram Moolenaar30d53e22020-03-18 21:10:44 +01002905 let g:test_is_flaky = 1
Bram Moolenaar8c5a2782019-08-07 23:07:07 +02002906
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002907 " Prepare file for TextChanged event.
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01002908 call writefile([''], 'Xchanged.txt', 'D')
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002909 let buf = term_start([GetVimProg(), '--clean', '-c', 'set noswapfile'], {'term_rows': 3})
2910 call assert_equal('running', term_getstatus(buf))
Bram Moolenaar1834d372018-03-29 17:40:46 +02002911 " Wait for the ruler (in the status line) to be shown.
Bram Moolenaaraa5df7e2019-02-03 14:53:10 +01002912 " In ConPTY, there is additional character which is drawn up to the width of
2913 " the screen.
2914 if has('conpty')
2915 call WaitForAssert({-> assert_match('\<All.*$', term_getline(buf, 3))})
2916 else
2917 call WaitForAssert({-> assert_match('\<All$', term_getline(buf, 3))})
2918 endif
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002919 " It's only adding autocmd, so that no event occurs.
2920 call term_sendkeys(buf, ":au! TextChanged <buffer> call writefile(['No'], 'Xchanged.txt')\<cr>")
2921 call term_sendkeys(buf, "\<C-\\>\<C-N>:qa!\<cr>")
Bram Moolenaar50182fa2018-04-28 21:34:40 +02002922 call WaitForAssert({-> assert_equal('finished', term_getstatus(buf))})
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002923 call assert_equal([''], readfile('Xchanged.txt'))
2924
2925 " clean up
Bram Moolenaar8c64a362018-03-23 22:39:31 +01002926 bwipe!
2927endfunc
Bram Moolenaar0566e892019-01-24 19:37:40 +01002928
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002929func Test_autocmd_nested()
2930 let g:did_nested = 0
Christian Brabandtfb3f9692024-08-11 20:09:17 +02002931 defer CleanUpTestAuGroup()
2932 augroup testing
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002933 au WinNew * edit somefile
2934 au BufNew * let g:did_nested = 1
2935 augroup END
2936 split
2937 call assert_equal(0, g:did_nested)
2938 close
2939 bwipe! somefile
2940
2941 " old nested argument still works
Christian Brabandtfb3f9692024-08-11 20:09:17 +02002942 augroup testing
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002943 au!
2944 au WinNew * nested edit somefile
2945 au BufNew * let g:did_nested = 1
2946 augroup END
2947 split
2948 call assert_equal(1, g:did_nested)
2949 close
2950 bwipe! somefile
2951
2952 " New ++nested argument works
2953 augroup Testing
2954 au!
2955 au WinNew * ++nested edit somefile
2956 au BufNew * let g:did_nested = 1
2957 augroup END
2958 split
2959 call assert_equal(1, g:did_nested)
2960 close
2961 bwipe! somefile
2962
Bram Moolenaarf0775142022-03-04 20:10:38 +00002963 " nested without ++ does not work in Vim9 script
2964 call assert_fails('vim9cmd au WinNew * nested echo fails', 'E1078:')
2965
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02002966 augroup Testing
2967 au!
2968 augroup END
2969
2970 call assert_fails('au WinNew * ++nested ++nested echo bad', 'E983:')
2971 call assert_fails('au WinNew * nested nested echo bad', 'E983:')
2972endfunc
2973
Bram Moolenaar5fa9f232022-07-23 09:06:48 +01002974func Test_autocmd_nested_cursor_invalid()
2975 set laststatus=0
2976 copen
2977 cclose
2978 call setline(1, ['foo', 'bar', 'baz'])
2979 3
2980 augroup nested_inv
2981 autocmd User foo ++nested copen
2982 autocmd BufAdd * let &laststatus = 2 - &laststatus
2983 augroup END
2984 doautocmd User foo
2985
2986 augroup nested_inv
2987 au!
2988 augroup END
2989 set laststatus&
Bram Moolenaarb03950f2022-07-26 13:47:13 +01002990 cclose
Bram Moolenaar5fa9f232022-07-23 09:06:48 +01002991 bwipe!
2992endfunc
2993
Bram Moolenaar3d6ee8b2022-07-27 15:23:35 +01002994func Test_autocmd_nested_keeps_cursor_pos()
2995 enew
2996 call setline(1, 'foo')
2997 autocmd User foo ++nested normal! $a
2998 autocmd InsertLeave * :
2999 doautocmd User foo
3000 call assert_equal([0, 1, 3, 0], getpos('.'))
3001
3002 bwipe!
3003endfunc
3004
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003005func Test_autocmd_nested_switch_window()
3006 " run this in a separate Vim so that SafeState works
3007 CheckRunVimInTerminal
3008
3009 let lines =<< trim END
3010 vim9script
3011 ['()']->writefile('Xautofile')
3012 autocmd VimEnter * ++nested edit Xautofile | split
3013 autocmd BufReadPost * autocmd SafeState * ++once foldclosed('.')
3014 autocmd WinEnter * matchadd('ErrorMsg', 'pat')
3015 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003016 call writefile(lines, 'Xautoscript', 'D')
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003017 let buf = RunVimInTerminal('-S Xautoscript', {'rows': 10})
3018 call VerifyScreenDump(buf, 'Test_autocmd_nested_switch', {})
3019
3020 call StopVimInTerminal(buf)
3021 call delete('Xautofile')
Bram Moolenaarb03950f2022-07-26 13:47:13 +01003022endfunc
3023
Bram Moolenaareb93f3f2019-04-04 15:04:56 +02003024func Test_autocmd_once()
3025 " Without ++once WinNew triggers twice
3026 let g:did_split = 0
3027 augroup Testing
3028 au WinNew * let g:did_split += 1
3029 augroup END
3030 split
3031 split
3032 call assert_equal(2, g:did_split)
3033 call assert_true(exists('#WinNew'))
3034 close
3035 close
3036
3037 " With ++once WinNew triggers once
3038 let g:did_split = 0
3039 augroup Testing
3040 au!
3041 au WinNew * ++once let g:did_split += 1
3042 augroup END
3043 split
3044 split
3045 call assert_equal(1, g:did_split)
3046 call assert_false(exists('#WinNew'))
3047 close
3048 close
3049
3050 call assert_fails('au WinNew * ++once ++once echo bad', 'E983:')
3051endfunc
3052
Bram Moolenaara68e5952019-04-25 22:22:01 +02003053func Test_autocmd_bufreadpre()
3054 new
3055 let b:bufreadpre = 1
Bram Moolenaarab505b12020-03-23 19:28:44 +01003056 call append(0, range(1000))
Bram Moolenaara68e5952019-04-25 22:22:01 +02003057 w! XAutocmdBufReadPre.txt
3058 autocmd BufReadPre <buffer> :let b:bufreadpre += 1
Bram Moolenaarab505b12020-03-23 19:28:44 +01003059 norm! 500gg
Bram Moolenaara68e5952019-04-25 22:22:01 +02003060 sp
Bram Moolenaarab505b12020-03-23 19:28:44 +01003061 norm! 1000gg
Bram Moolenaara68e5952019-04-25 22:22:01 +02003062 wincmd p
3063 let g:wsv1 = winsaveview()
3064 wincmd p
3065 let g:wsv2 = winsaveview()
3066 " triggers BufReadPre, should not move the cursor in either window
3067 " The topline may change one line in a large window.
3068 edit
3069 call assert_inrange(g:wsv2.topline - 1, g:wsv2.topline + 1, winsaveview().topline)
3070 call assert_equal(g:wsv2.lnum, winsaveview().lnum)
3071 call assert_equal(2, b:bufreadpre)
3072 wincmd p
3073 call assert_equal(g:wsv1.topline, winsaveview().topline)
3074 call assert_equal(g:wsv1.lnum, winsaveview().lnum)
3075 call assert_equal(2, b:bufreadpre)
3076 " Now set the cursor position in an BufReadPre autocommand
3077 " (even though the position will be invalid, this should make Vim reset the
3078 " cursor position in the other window.
3079 wincmd p
3080 set cpo+=g
3081 " won't do anything, but try to set the cursor on an invalid lnum
3082 autocmd BufReadPre <buffer> :norm! 70gg
3083 " triggers BufReadPre, should not move the cursor in either window
3084 e
3085 call assert_equal(1, winsaveview().topline)
3086 call assert_equal(1, winsaveview().lnum)
3087 call assert_equal(3, b:bufreadpre)
3088 wincmd p
3089 call assert_equal(g:wsv1.topline, winsaveview().topline)
3090 call assert_equal(g:wsv1.lnum, winsaveview().lnum)
3091 call assert_equal(3, b:bufreadpre)
3092 close
3093 close
3094 call delete('XAutocmdBufReadPre.txt')
3095 set cpo-=g
3096endfunc
3097
Bram Moolenaar5e66b422019-01-24 21:58:10 +01003098" FileChangedShell tested in test_filechanged.vim
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003099
3100" Tests for the following autocommands:
3101" - FileWritePre writing a compressed file
3102" - FileReadPost reading a compressed file
3103" - BufNewFile reading a file template
3104" - BufReadPre decompressing the file to be read
3105" - FilterReadPre substituting characters in the temp file
3106" - FilterReadPost substituting characters after filtering
3107" - FileReadPre set options for decompression
3108" - FileReadPost decompress the file
3109func Test_ReadWrite_Autocmds()
3110 " Run this test only on Unix-like systems and if gzip is available
Bram Moolenaar6d91bcb2020-08-12 18:50:36 +02003111 CheckUnix
3112 CheckExecutable gzip
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003113
3114 " Make $GZIP empty, "-v" would cause trouble.
3115 let $GZIP = ""
3116
3117 " Use a FileChangedShell autocommand to avoid a prompt for 'Xtestfile.gz'
3118 " being modified outside of Vim (noticed on Solaris).
3119 au FileChangedShell * echo 'caught FileChangedShell'
3120
3121 " Test for the FileReadPost, FileWritePre and FileWritePost autocmds
3122 augroup Test1
3123 au!
3124 au FileWritePre *.gz '[,']!gzip
3125 au FileWritePost *.gz undo
3126 au FileReadPost *.gz '[,']!gzip -d
3127 augroup END
3128
3129 new
3130 set bin
3131 call append(0, [
3132 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3133 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3134 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3135 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3136 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3137 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3138 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3139 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3140 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3141 \ ])
3142 1,9write! Xtestfile.gz
3143 enew! | close
3144
3145 new
3146 " Read and decompress the testfile
3147 0read Xtestfile.gz
3148 call assert_equal([
3149 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3150 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3151 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3152 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3153 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3154 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3155 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3156 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3157 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3158 \ ], getline(1, 9))
3159 enew! | close
3160
3161 augroup Test1
3162 au!
3163 augroup END
3164
3165 " Test for the FileAppendPre and FileAppendPost autocmds
3166 augroup Test2
3167 au!
3168 au BufNewFile *.c read Xtest.c
3169 au FileAppendPre *.out '[,']s/new/NEW/
3170 au FileAppendPost *.out !cat Xtest.c >> test.out
3171 augroup END
3172
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003173 call writefile(['/*', ' * Here is a new .c file', ' */'], 'Xtest.c', 'D')
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003174 new foo.c " should load Xtest.c
3175 call assert_equal(['/*', ' * Here is a new .c file', ' */'], getline(2, 4))
3176 w! >> test.out " append it to the output file
3177
3178 let contents = readfile('test.out')
3179 call assert_equal(' * Here is a NEW .c file', contents[2])
3180 call assert_equal(' * Here is a new .c file', contents[5])
3181
3182 call delete('test.out')
3183 enew! | close
3184 augroup Test2
3185 au!
3186 augroup END
3187
3188 " Test for the BufReadPre and BufReadPost autocmds
3189 augroup Test3
3190 au!
3191 " setup autocommands to decompress before reading and re-compress
3192 " afterwards
3193 au BufReadPre *.gz exe '!gzip -d ' . shellescape(expand("<afile>"))
3194 au BufReadPre *.gz call rename(expand("<afile>:r"), expand("<afile>"))
3195 au BufReadPost *.gz call rename(expand("<afile>"), expand("<afile>:r"))
3196 au BufReadPost *.gz exe '!gzip ' . shellescape(expand("<afile>:r"))
3197 augroup END
3198
3199 e! Xtestfile.gz " Edit compressed file
3200 call assert_equal([
3201 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3202 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3203 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3204 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3205 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3206 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3207 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3208 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3209 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3210 \ ], getline(1, 9))
3211
3212 w! >> test.out " Append it to the output file
3213
3214 augroup Test3
3215 au!
3216 augroup END
3217
3218 " Test for the FilterReadPre and FilterReadPost autocmds.
3219 set shelltemp " need temp files here
3220 augroup Test4
3221 au!
3222 au FilterReadPre *.out call rename(expand("<afile>"), expand("<afile>") . ".t")
3223 au FilterReadPre *.out exe 'silent !sed s/e/E/ ' . shellescape(expand("<afile>")) . ".t >" . shellescape(expand("<afile>"))
3224 au FilterReadPre *.out exe 'silent !rm ' . shellescape(expand("<afile>")) . '.t'
3225 au FilterReadPost *.out '[,']s/x/X/g
3226 augroup END
3227
3228 e! test.out " Edit the output file
3229 1,$!cat
3230 call assert_equal([
3231 \ 'linE 2 AbcdefghijklmnopqrstuvwXyz',
3232 \ 'linE 3 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3233 \ 'linE 4 AbcdefghijklmnopqrstuvwXyz',
3234 \ 'linE 5 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3235 \ 'linE 6 AbcdefghijklmnopqrstuvwXyz',
3236 \ 'linE 7 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3237 \ 'linE 8 AbcdefghijklmnopqrstuvwXyz',
3238 \ 'linE 9 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX',
3239 \ 'linE 10 AbcdefghijklmnopqrstuvwXyz'
3240 \ ], getline(1, 9))
3241 call assert_equal([
3242 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3243 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3244 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3245 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3246 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3247 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3248 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3249 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3250 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3251 \ ], readfile('test.out'))
3252
3253 augroup Test4
3254 au!
3255 augroup END
3256 set shelltemp&vim
3257
3258 " Test for the FileReadPre and FileReadPost autocmds.
3259 augroup Test5
3260 au!
3261 au FileReadPre *.gz exe 'silent !gzip -d ' . shellescape(expand("<afile>"))
3262 au FileReadPre *.gz call rename(expand("<afile>:r"), expand("<afile>"))
3263 au FileReadPost *.gz '[,']s/l/L/
3264 augroup END
3265
3266 new
3267 0r Xtestfile.gz " Read compressed file
3268 call assert_equal([
3269 \ 'Line 2 Abcdefghijklmnopqrstuvwxyz',
3270 \ 'Line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3271 \ 'Line 4 Abcdefghijklmnopqrstuvwxyz',
3272 \ 'Line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3273 \ 'Line 6 Abcdefghijklmnopqrstuvwxyz',
3274 \ 'Line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3275 \ 'Line 8 Abcdefghijklmnopqrstuvwxyz',
3276 \ 'Line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3277 \ 'Line 10 Abcdefghijklmnopqrstuvwxyz'
3278 \ ], getline(1, 9))
3279 call assert_equal([
3280 \ 'line 2 Abcdefghijklmnopqrstuvwxyz',
3281 \ 'line 3 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3282 \ 'line 4 Abcdefghijklmnopqrstuvwxyz',
3283 \ 'line 5 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3284 \ 'line 6 Abcdefghijklmnopqrstuvwxyz',
3285 \ 'line 7 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3286 \ 'line 8 Abcdefghijklmnopqrstuvwxyz',
3287 \ 'line 9 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
3288 \ 'line 10 Abcdefghijklmnopqrstuvwxyz'
3289 \ ], readfile('Xtestfile.gz'))
3290
3291 augroup Test5
3292 au!
3293 augroup END
3294
3295 au! FileChangedShell
3296 call delete('Xtestfile.gz')
Bram Moolenaar69ea5872019-04-25 20:29:00 +02003297 call delete('test.out')
3298endfunc
Bram Moolenaar23b51392019-05-09 21:38:43 +02003299
3300func Test_throw_in_BufWritePre()
3301 new
3302 call setline(1, ['one', 'two', 'three'])
3303 call assert_false(filereadable('Xthefile'))
3304 augroup throwing
3305 au BufWritePre X* throw 'do not write'
3306 augroup END
3307 try
3308 w Xthefile
3309 catch
3310 let caught = 1
3311 endtry
3312 call assert_equal(1, caught)
3313 call assert_false(filereadable('Xthefile'))
3314
3315 bwipe!
3316 au! throwing
3317endfunc
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003318
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003319func Test_autocmd_in_try_block()
Bram Moolenaar6f14da12022-09-07 21:30:44 +01003320 call mkdir('Xintrydir', 'R')
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003321 au BufEnter * let g:fname = expand('%')
3322 try
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01003323 edit Xintrydir/
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003324 endtry
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +01003325 call assert_match('Xintrydir', g:fname)
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003326
3327 unlet g:fname
3328 au! BufEnter
Bram Moolenaar40fa12a2021-09-22 14:18:13 +02003329endfunc
3330
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003331func Test_autocmd_SafeState()
3332 CheckRunVimInTerminal
3333
3334 let lines =<< trim END
3335 let g:safe = 0
3336 let g:again = ''
3337 au SafeState * let g:safe += 1
3338 au SafeStateAgain * let g:again ..= 'x'
3339 func CallTimer()
3340 call timer_start(10, {id -> execute('let g:again ..= "t"')})
3341 endfunc
3342 END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003343 call writefile(lines, 'XSafeState', 'D')
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003344 let buf = RunVimInTerminal('-S XSafeState', #{rows: 6})
3345
Bram Moolenaar8e7d6222020-12-18 19:49:56 +01003346 " Sometimes we loop to handle a K_IGNORE, SafeState may be triggered once or
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003347 " more often.
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003348 call term_sendkeys(buf, ":echo g:safe\<CR>")
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003349 call WaitForAssert({-> assert_match('^\d ', term_getline(buf, 6))}, 1000)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003350
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003351 " SafeStateAgain should be invoked at least three times
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003352 call term_sendkeys(buf, ":echo g:again\<CR>")
Bram Moolenaar8fb1b472020-02-23 16:16:26 +01003353 call WaitForAssert({-> assert_match('^xxx', term_getline(buf, 6))}, 1000)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003354
3355 call term_sendkeys(buf, ":let g:again = ''\<CR>:call CallTimer()\<CR>")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003356 call TermWait(buf, 50)
Bram Moolenaar0f6629a2019-09-22 23:24:13 +02003357 call term_sendkeys(buf, ":\<CR>")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003358 call TermWait(buf, 50)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003359 call term_sendkeys(buf, ":echo g:again\<CR>")
3360 call WaitForAssert({-> assert_match('xtx', term_getline(buf, 6))}, 1000)
3361
3362 call StopVimInTerminal(buf)
Bram Moolenaarcadbe1b2019-09-22 21:50:09 +02003363endfunc
Bram Moolenaar23324a02019-10-01 17:39:04 +02003364
3365func Test_autocmd_CmdWinEnter()
3366 CheckRunVimInTerminal
Bram Moolenaar21829c52021-01-26 22:42:21 +01003367
Bram Moolenaar23324a02019-10-01 17:39:04 +02003368 let lines =<< trim END
Egor Zvorykin125ffd22021-11-17 14:01:14 +00003369 augroup vimHints | au! | augroup END
Bram Moolenaar23324a02019-10-01 17:39:04 +02003370 let b:dummy_var = 'This is a dummy'
3371 autocmd CmdWinEnter * quit
3372 let winnr = winnr('$')
3373 END
Bram Moolenaar1cfb9bb2020-12-22 11:40:45 +01003374 let filename = 'XCmdWinEnter'
Bram Moolenaar23324a02019-10-01 17:39:04 +02003375 call writefile(lines, filename)
3376 let buf = RunVimInTerminal('-S '.filename, #{rows: 6})
3377
3378 call term_sendkeys(buf, "q:")
Bram Moolenaar6a2c5a72020-04-08 21:50:25 +02003379 call TermWait(buf)
Bram Moolenaar23324a02019-10-01 17:39:04 +02003380 call term_sendkeys(buf, ":echo b:dummy_var\<cr>")
Bram Moolenaar353c3512020-03-15 14:19:26 +01003381 call WaitForAssert({-> assert_match('^This is a dummy', term_getline(buf, 6))}, 2000)
Bram Moolenaar23324a02019-10-01 17:39:04 +02003382 call term_sendkeys(buf, ":echo &buftype\<cr>")
3383 call WaitForAssert({-> assert_notmatch('^nofile', term_getline(buf, 6))}, 1000)
3384 call term_sendkeys(buf, ":echo winnr\<cr>")
3385 call WaitForAssert({-> assert_match('^1', term_getline(buf, 6))}, 1000)
3386
3387 " clean up
3388 call StopVimInTerminal(buf)
3389 call delete(filename)
3390endfunc
Bram Moolenaarec66c412019-10-11 21:19:13 +02003391
3392func Test_autocmd_was_using_freed_memory()
Bram Moolenaar5a4c3082019-12-01 15:23:11 +01003393 CheckFeature quickfix
3394
Bram Moolenaarec66c412019-10-11 21:19:13 +02003395 pedit xx
3396 n x
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003397 augroup winenter
3398 au WinEnter * if winnr('$') > 2 | quit | endif
3399 augroup END
Bram Moolenaarec66c412019-10-11 21:19:13 +02003400 split
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003401
3402 augroup winenter
3403 au! WinEnter
3404 augroup END
3405
3406 bwipe xx
3407 bwipe x
3408 pclose
Bram Moolenaarec66c412019-10-11 21:19:13 +02003409endfunc
Bram Moolenaarf4a1d1c2019-11-16 13:50:25 +01003410
3411func Test_BufWrite_lockmarks()
Bram Moolenaarf08b0eb2021-10-16 13:00:14 +01003412 let g:test_is_flaky = 1
Bram Moolenaarf4a1d1c2019-11-16 13:50:25 +01003413 edit! Xtest
3414 call setline(1, ['a', 'b', 'c', 'd'])
3415
3416 " :lockmarks preserves the marks
3417 call SetChangeMarks(2, 3)
3418 lockmarks write
3419 call assert_equal([2, 3], [line("'["), line("']")])
3420
3421 " *WritePre autocmds get the correct line range, but lockmarks preserves the
3422 " original values for the user
3423 augroup lockmarks
3424 au!
3425 au BufWritePre,FilterWritePre * call assert_equal([1, 4], [line("'["), line("']")])
3426 au FileWritePre * call assert_equal([3, 4], [line("'["), line("']")])
3427 augroup END
3428
3429 lockmarks write
3430 call assert_equal([2, 3], [line("'["), line("']")])
3431
3432 if executable('cat')
3433 lockmarks %!cat
3434 call assert_equal([2, 3], [line("'["), line("']")])
3435 endif
3436
3437 lockmarks 3,4write Xtest2
3438 call assert_equal([2, 3], [line("'["), line("']")])
3439
3440 au! lockmarks
3441 augroup! lockmarks
3442 call delete('Xtest')
3443 call delete('Xtest2')
3444endfunc
Bram Moolenaarce6db022020-01-07 20:11:42 +01003445
3446func Test_FileType_spell()
3447 if !isdirectory('/tmp')
3448 throw "Skipped: requires /tmp directory"
3449 endif
3450
3451 " this was crashing with an invalid free()
3452 setglobal spellfile=/tmp/en.utf-8.add
3453 augroup crash
3454 autocmd!
3455 autocmd BufNewFile,BufReadPost crashfile setf somefiletype
3456 autocmd BufNewFile,BufReadPost crashfile set ft=anotherfiletype
3457 autocmd FileType anotherfiletype setlocal spell
3458 augroup END
3459 func! NoCrash() abort
3460 edit /tmp/crashfile
3461 endfunc
3462 call NoCrash()
3463
3464 au! crash
3465 setglobal spellfile=
3466endfunc
Bram Moolenaarbc2b71d2020-02-17 21:33:30 +01003467
Bram Moolenaaref976322022-09-28 11:48:30 +01003468" this was wiping out the current buffer and using freed memory
3469func Test_SpellFileMissing_bwipe()
3470 next 0
3471 au SpellFileMissing 0 bwipe
3472 call assert_fails('set spell spelllang=0', 'E937:')
3473
3474 au! SpellFileMissing
Bram Moolenaar0a60f792022-11-19 21:18:11 +00003475 set nospell spelllang=en
Bram Moolenaaref976322022-09-28 11:48:30 +01003476 bwipe
3477endfunc
3478
Bram Moolenaar406cd902020-02-18 21:54:41 +01003479" Test closing a window or editing another buffer from a FileChangedRO handler
3480" in a readonly buffer
3481func Test_FileChangedRO_winclose()
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003482 call test_override('ui_delay', 10)
3483
Bram Moolenaar406cd902020-02-18 21:54:41 +01003484 augroup FileChangedROTest
3485 au!
3486 autocmd FileChangedRO * quit
3487 augroup END
3488 new
3489 set readonly
3490 call assert_fails('normal i', 'E788:')
3491 close
3492 augroup! FileChangedROTest
3493
3494 augroup FileChangedROTest
3495 au!
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003496 autocmd FileChangedRO * edit Xrofile
Bram Moolenaar406cd902020-02-18 21:54:41 +01003497 augroup END
3498 new
3499 set readonly
3500 call assert_fails('normal i', 'E788:')
3501 close
3502 augroup! FileChangedROTest
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003503 call test_override('ALL', 0)
Bram Moolenaar406cd902020-02-18 21:54:41 +01003504endfunc
3505
Bram Moolenaar0c81d1b2020-02-22 22:45:55 +01003506func LogACmd()
3507 call add(g:logged, line('$'))
3508endfunc
3509
3510func Test_TermChanged()
Bram Moolenaard28e0b32020-02-22 23:08:52 +01003511 CheckNotGui
3512
Bram Moolenaar0c81d1b2020-02-22 22:45:55 +01003513 enew!
3514 tabnew
3515 call setline(1, ['a', 'b', 'c', 'd'])
3516 $
3517 au TermChanged * call LogACmd()
3518 let g:logged = []
3519 let term_save = &term
3520 set term=xterm
3521 call assert_equal([1, 4], g:logged)
3522
3523 au! TermChanged
3524 let &term = term_save
3525 bwipe!
3526endfunc
3527
Bram Moolenaare3284872020-03-19 13:55:03 +01003528" Test for FileReadCmd autocmd
3529func Test_autocmd_FileReadCmd()
3530 func ReadFileCmd()
3531 call append(line('$'), "v:cmdarg = " .. v:cmdarg)
3532 endfunc
3533 augroup FileReadCmdTest
3534 au!
3535 au FileReadCmd Xtest call ReadFileCmd()
3536 augroup END
3537
3538 new
3539 read ++bin Xtest
3540 read ++nobin Xtest
3541 read ++edit Xtest
3542 read ++bad=keep Xtest
3543 read ++bad=drop Xtest
3544 read ++bad=- Xtest
3545 read ++ff=unix Xtest
3546 read ++ff=dos Xtest
3547 read ++ff=mac Xtest
3548 read ++enc=utf-8 Xtest
3549
3550 call assert_equal(['',
3551 \ 'v:cmdarg = ++bin',
3552 \ 'v:cmdarg = ++nobin',
3553 \ 'v:cmdarg = ++edit',
3554 \ 'v:cmdarg = ++bad=keep',
3555 \ 'v:cmdarg = ++bad=drop',
3556 \ 'v:cmdarg = ++bad=-',
3557 \ 'v:cmdarg = ++ff=unix',
3558 \ 'v:cmdarg = ++ff=dos',
3559 \ 'v:cmdarg = ++ff=mac',
3560 \ 'v:cmdarg = ++enc=utf-8'], getline(1, '$'))
3561
Bram Moolenaar23526d22022-12-05 15:50:41 +00003562 bwipe!
Bram Moolenaare3284872020-03-19 13:55:03 +01003563 augroup FileReadCmdTest
3564 au!
3565 augroup END
3566 delfunc ReadFileCmd
3567endfunc
3568
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003569" Test for passing invalid arguments to autocmd
3570func Test_autocmd_invalid_args()
3571 " Additional character after * for event
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003572 call assert_fails('autocmd *a Xinvfile set ff=unix', 'E215:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003573 augroup Test
3574 augroup END
3575 " Invalid autocmd event
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003576 call assert_fails('autocmd Bufabc Xinvfile set ft=vim', 'E216:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003577 " Invalid autocmd event in a autocmd group
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003578 call assert_fails('autocmd Test Bufabc Xinvfile set ft=vim', 'E216:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003579 augroup! Test
3580 " Execute all autocmds
3581 call assert_fails('doautocmd * BufEnter', 'E217:')
3582 call assert_fails('augroup! x1a2b3', 'E367:')
3583 call assert_fails('autocmd BufNew <buffer=999> pwd', 'E680:')
Bram Moolenaar531be472020-09-23 22:38:05 +02003584 call assert_fails('autocmd BufNew \) set ff=unix', 'E55:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003585endfunc
3586
3587" Test for deep nesting of autocmds
3588func Test_autocmd_deep_nesting()
Bram Moolenaar61abe7d2022-08-30 21:46:08 +01003589 autocmd BufEnter Xdeepfile doautocmd BufEnter Xdeepfile
3590 call assert_fails('doautocmd BufEnter Xdeepfile', 'E218:')
3591 autocmd! BufEnter Xdeepfile
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02003592endfunc
3593
Bram Moolenaarbe5ee862020-06-10 20:56:58 +02003594" Tests for SigUSR1 autocmd event, which is only available on posix systems.
3595func Test_autocmd_sigusr1()
3596 CheckUnix
Bram Moolenaar0056ca72022-09-23 21:26:39 +01003597 " FIXME: should this work on MacOS M1?
3598 CheckNotMacM1
Bram Moolenaar62cd26a2020-10-11 20:08:44 +02003599 CheckExecutable /bin/kill
Bram Moolenaarbe5ee862020-06-10 20:56:58 +02003600
3601 let g:sigusr1_passed = 0
3602 au SigUSR1 * let g:sigusr1_passed = 1
3603 call system('/bin/kill -s usr1 ' . getpid())
3604 call WaitForAssert({-> assert_true(g:sigusr1_passed)})
3605
3606 au! SigUSR1
3607 unlet g:sigusr1_passed
3608endfunc
3609
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003610" Test for BufReadPre autocmd deleting the file
3611func Test_BufReadPre_delfile()
3612 augroup TestAuCmd
3613 au!
Bram Moolenaare7cda972022-08-29 11:02:59 +01003614 autocmd BufReadPre XbufreadPre call delete('XbufreadPre')
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003615 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003616 call writefile([], 'XbufreadPre', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +01003617 call assert_fails('new XbufreadPre', 'E200:')
3618 call assert_equal('XbufreadPre', @%)
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003619 call assert_equal(1, &readonly)
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003620
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003621 augroup TestAuCmd
3622 au!
3623 augroup END
3624 close!
3625endfunc
3626
3627" Test for BufReadPre autocmd changing the current buffer
3628func Test_BufReadPre_changebuf()
3629 augroup TestAuCmd
3630 au!
Bram Moolenaare7cda972022-08-29 11:02:59 +01003631 autocmd BufReadPre Xchangebuf edit Xsomeotherfile
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003632 augroup END
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003633 call writefile([], 'Xchangebuf', 'D')
Bram Moolenaare7cda972022-08-29 11:02:59 +01003634 call assert_fails('new Xchangebuf', 'E201:')
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003635 call assert_equal('Xsomeotherfile', @%)
3636 call assert_equal(1, &readonly)
Bram Moolenaare1f3ab72022-09-04 21:29:08 +01003637
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003638 augroup TestAuCmd
3639 au!
3640 augroup END
3641 close!
3642endfunc
3643
3644" Test for BufWipeouti autocmd changing the current buffer when reading a file
3645" in an empty buffer with 'f' flag in 'cpo'
3646func Test_BufDelete_changebuf()
3647 new
3648 augroup TestAuCmd
3649 au!
3650 autocmd BufWipeout * let bufnr = bufadd('somefile') | exe "b " .. bufnr
3651 augroup END
3652 let save_cpo = &cpo
3653 set cpo+=f
Bram Moolenaarb18b4962022-09-02 21:55:50 +01003654 call assert_fails('r Xchangebuf', ['E812:', 'E484:'])
Bram Moolenaarb340bae2020-06-15 19:51:56 +02003655 call assert_equal('somefile', @%)
3656 let &cpo = save_cpo
3657 augroup TestAuCmd
3658 au!
3659 augroup END
3660 close!
3661endfunc
3662
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003663" Test for the temporary internal window used to execute autocmds
3664func Test_autocmd_window()
3665 %bw!
3666 edit one.txt
3667 tabnew two.txt
Bram Moolenaar41cd8032021-03-13 15:47:56 +01003668 vnew three.txt
3669 tabnew four.txt
3670 tabprevious
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003671 let g:blist = []
Bram Moolenaar832adf92020-06-25 19:01:36 +02003672 augroup aucmd_win_test1
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003673 au!
3674 au BufEnter * call add(g:blist, [expand('<afile>'),
3675 \ win_gettype(bufwinnr(expand('<afile>')))])
3676 augroup END
3677
3678 doautoall BufEnter
Bram Moolenaar41cd8032021-03-13 15:47:56 +01003679 call assert_equal([
3680 \ ['one.txt', 'autocmd'],
3681 \ ['two.txt', ''],
3682 \ ['four.txt', 'autocmd'],
3683 \ ['three.txt', ''],
3684 \ ], g:blist)
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003685
Bram Moolenaar832adf92020-06-25 19:01:36 +02003686 augroup aucmd_win_test1
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003687 au!
3688 augroup END
Bram Moolenaar832adf92020-06-25 19:01:36 +02003689 augroup! aucmd_win_test1
3690 %bw!
3691endfunc
3692
3693" Test for trying to close the temporary window used for executing an autocmd
3694func Test_close_autocmd_window()
3695 %bw!
3696 edit one.txt
3697 tabnew two.txt
3698 augroup aucmd_win_test2
3699 au!
3700 au BufEnter * if expand('<afile>') == 'one.txt' | 1close | endif
3701 augroup END
3702
3703 call assert_fails('doautoall BufEnter', 'E813:')
3704
3705 augroup aucmd_win_test2
3706 au!
3707 augroup END
3708 augroup! aucmd_win_test2
Bram Moolenaarcf844172020-06-26 19:44:06 +02003709 %bwipe!
3710endfunc
3711
3712" Test for trying to close the tab that has the temporary window for exeucing
3713" an autocmd.
3714func Test_close_autocmd_tab()
3715 edit one.txt
3716 tabnew two.txt
3717 augroup aucmd_win_test
3718 au!
3719 au BufEnter * if expand('<afile>') == 'one.txt' | tabfirst | tabonly | endif
3720 augroup END
3721
3722 call assert_fails('doautoall BufEnter', 'E813:')
3723
3724 tabonly
3725 augroup aucmd_win_test
3726 au!
3727 augroup END
3728 augroup! aucmd_win_test
3729 %bwipe!
Bram Moolenaar0fe937f2020-06-16 22:42:04 +02003730endfunc
3731
Bram Moolenaarcb1956d2022-01-07 15:45:18 +00003732func Test_Visual_doautoall_redraw()
3733 call setline(1, ['a', 'b'])
Bram Moolenaar94722c52023-01-28 19:19:03 +00003734 new
Bram Moolenaarcb1956d2022-01-07 15:45:18 +00003735 wincmd p
3736 call feedkeys("G\<C-V>", 'txn')
3737 autocmd User Explode ++once redraw
3738 doautoall User Explode
3739 %bwipe!
3740endfunc
3741
Bram Moolenaar6bcb8772021-02-03 21:23:29 +01003742" This was using freed memory.
3743func Test_BufNew_arglocal()
3744 arglocal
3745 au BufNew * arglocal
3746 call assert_fails('drop xx', 'E1156:')
3747
3748 au! BufNew
3749endfunc
3750
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003751func Test_autocmd_closes_window()
3752 au BufNew,BufWinLeave * e %e
3753 file yyy
3754 au BufNew,BufWinLeave * ball
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003755 n xxx
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003756
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003757 %bwipe
Bram Moolenaar8ab37572021-02-03 21:56:59 +01003758 au! BufNew
3759 au! BufWinLeave
3760endfunc
3761
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003762func Test_autocmd_quit_psearch()
3763 sn aa bb
3764 augroup aucmd_win_test
3765 au!
3766 au BufEnter,BufLeave,BufNew,WinEnter,WinLeave,WinNew * if winnr('$') > 1 | q | endif
3767 augroup END
3768 ps /
3769
3770 augroup aucmd_win_test
3771 au!
3772 augroup END
zeertzjq7851c692022-04-21 11:14:01 +01003773 new
3774 pclose
Bram Moolenaar92bb83e2021-02-03 23:04:46 +01003775endfunc
3776
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003777" Fuzzer found some strange combination that caused a crash.
3778func Test_autocmd_normal_mess()
Bram Moolenaardd07c022021-02-07 13:32:46 +01003779 " For unknown reason this hangs on MS-Windows
3780 CheckNotMSWindows
3781
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003782 augroup aucmd_normal_test
3783 au BufLeave,BufWinLeave,BufHidden,BufUnload,BufDelete,BufWipeout * norm 7q/qc
3784 augroup END
zeertzjq67fe77d2025-04-20 10:21:18 +02003785 call assert_fails('o4', 'E1159:')
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003786 silent! H
zeertzjq67fe77d2025-04-20 10:21:18 +02003787 call assert_fails('e xx', 'E1159:')
Bram Moolenaaraad5f9d2021-02-06 17:30:31 +01003788 normal G
3789
3790 augroup aucmd_normal_test
3791 au!
3792 augroup END
3793endfunc
3794
Bram Moolenaar8c6951f2021-02-06 18:08:45 +01003795func Test_autocmd_closing_cmdwin()
Bram Moolenaardd07c022021-02-07 13:32:46 +01003796 " For unknown reason this hangs on MS-Windows
3797 CheckNotMSWindows
3798
Bram Moolenaar8c6951f2021-02-06 18:08:45 +01003799 au BufWinLeave * nested q
3800 call assert_fails("norm 7q?\n", 'E855:')
3801
3802 au! BufWinLeave
3803 new
3804 only
3805endfunc
3806
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003807func Test_autocmd_vimgrep()
3808 augroup aucmd_vimgrep
Charlie Grovesfef44852022-04-19 16:24:12 +01003809 au QuickfixCmdPre,BufNew,BufReadCmd * sb
zeertzjq7851c692022-04-21 11:14:01 +01003810 au QuickfixCmdPre,BufNew,BufReadCmd * q9
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003811 augroup END
Bram Moolenaardd07c022021-02-07 13:32:46 +01003812 call assert_fails('lv ?a? foo', 'E926:')
Bram Moolenaar2c7080b2021-02-06 19:19:42 +01003813
3814 augroup aucmd_vimgrep
3815 au!
3816 augroup END
3817endfunc
3818
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003819func Test_autocmd_with_block()
3820 augroup block_testing
3821 au BufReadPost *.xml {
3822 setlocal matchpairs+=<:>
3823 /<start
3824 }
Bram Moolenaar63b91732021-08-05 20:40:03 +02003825 au CursorHold * {
3826 autocmd BufReadPre * ++once echo 'one' | echo 'two'
3827 g:gotSafeState = 77
3828 }
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003829 augroup END
3830
Ken Takataeccc9272024-09-03 23:01:55 +02003831 let expected = gettext("\n--- Autocommands ---") .. "\nblock_testing BufRead\n *.xml {^@ setlocal matchpairs+=<:>^@ /<start^@ }"
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003832 call assert_equal(expected, execute('au BufReadPost *.xml'))
3833
Bram Moolenaar63b91732021-08-05 20:40:03 +02003834 doautocmd CursorHold
3835 call assert_equal(77, g:gotSafeState)
3836 unlet g:gotSafeState
3837
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003838 augroup block_testing
3839 au!
Bram Moolenaar75ebd2a2022-06-03 17:39:46 +01003840 autocmd CursorHold * {
3841 if true
3842 # comment
3843 && true
3844
3845 && true
3846 g:done = 'yes'
3847 endif
3848 }
3849 augroup END
3850 doautocmd CursorHold
3851 call assert_equal('yes', g:done)
3852
3853 unlet g:done
3854 augroup block_testing
3855 au!
Bram Moolenaar73b8b0a2021-08-01 14:52:32 +02003856 augroup END
3857endfunc
3858
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00003859func Test_closing_autocmd_window()
3860 let lines =<< trim END
3861 edit Xa.txt
3862 tabnew Xb.txt
3863 autocmd BufEnter Xa.txt unhide 1
3864 doautoall BufEnter
3865 END
3866 call v9.CheckScriptFailure(lines, 'E814:')
3867 au! BufEnter
Bram Moolenaar6f2465d2022-03-22 18:13:01 +00003868 bwipe Xa.txt
3869 bwipe Xb.txt
3870endfunc
3871
zeertzjq46bdae02023-09-24 23:16:08 +02003872func Test_switch_window_in_autocmd_window()
3873 edit Xa.txt
3874 tabnew Xb.txt
3875 autocmd BufEnter Xa.txt wincmd w
3876 doautoall BufEnter
3877 au! BufEnter
3878 bwipe Xa.txt
3879 call assert_false(bufexists('Xa.txt'))
3880 bwipe Xb.txt
3881 call assert_false(bufexists('Xb.txt'))
3882endfunc
3883
zeertzjq9d956ee2024-04-07 18:16:10 +02003884" Test that using the autocommand window doesn't change current directory.
3885func Test_autocmd_window_cwd()
3886 let saveddir = getcwd()
3887 call mkdir('Xcwd/a/b/c/d', 'pR')
3888
3889 new Xa.txt
3890 tabnew
3891 new Xb.txt
3892
3893 tabprev
3894 cd Xcwd
3895 call assert_match('/Xcwd$', getcwd())
3896 call assert_match('\[global\] .*/Xcwd$', trim(execute('verbose pwd')))
3897
3898 autocmd BufEnter Xb.txt lcd ./a/b/c/d
3899 doautoall BufEnter
3900 au! BufEnter
3901 call assert_match('/Xcwd$', getcwd())
3902 call assert_match('\[global\] .*/Xcwd$', trim(execute('verbose pwd')))
3903
3904 tabnext
3905 cd ./a
3906 tcd ./b
3907 lcd ./c
3908 call assert_match('/Xcwd/a/b/c$', getcwd())
3909 call assert_match('\[window\] .*/Xcwd/a/b/c$', trim(execute('verbose pwd')))
3910
3911 autocmd BufEnter Xa.txt call assert_match('Xcwd/a/b/c$', getcwd())
3912 doautoall BufEnter
3913 au! BufEnter
3914 call assert_match('/Xcwd/a/b/c$', getcwd())
3915 call assert_match('\[window\] .*/Xcwd/a/b/c$', trim(execute('verbose pwd')))
3916 bwipe!
3917 call assert_match('/Xcwd/a/b$', getcwd())
3918 call assert_match('\[tabpage\] .*/Xcwd/a/b$', trim(execute('verbose pwd')))
3919 bwipe!
3920 call assert_match('/Xcwd/a$', getcwd())
3921 call assert_match('\[global\] .*/Xcwd/a$', trim(execute('verbose pwd')))
3922 bwipe!
3923
3924 call chdir(saveddir)
3925endfunc
3926
Bram Moolenaar347538f2022-03-26 16:28:06 +00003927func Test_bufwipeout_changes_window()
3928 " This should not crash, but we don't have any expectations about what
3929 " happens, changing window in BufWipeout has unpredictable results.
3930 tabedit
3931 let g:window_id = win_getid()
3932 topleft new
3933 setlocal bufhidden=wipe
3934 autocmd BufWipeout <buffer> call win_gotoid(g:window_id)
3935 tabprevious
3936 +tabclose
3937
3938 unlet g:window_id
3939 au! BufWipeout
3940 %bwipe!
3941endfunc
3942
zeertzjq021996f2022-04-10 11:44:04 +01003943func Test_v_event_readonly()
3944 autocmd CompleteChanged * let v:event.width = 0
3945 call assert_fails("normal! i\<C-X>\<C-V>", 'E46:')
3946 au! CompleteChanged
3947
3948 autocmd DirChangedPre * let v:event.directory = ''
3949 call assert_fails('cd .', 'E46:')
3950 au! DirChangedPre
3951
3952 autocmd ModeChanged * let v:event.new_mode = ''
3953 call assert_fails('normal! cc', 'E46:')
3954 au! ModeChanged
3955
3956 autocmd TextYankPost * let v:event.operator = ''
3957 call assert_fails('normal! yy', 'E46:')
3958 au! TextYankPost
3959endfunc
3960
zeertzjqc9e8fd62022-07-26 18:12:38 +01003961" Test for ModeChanged pattern
3962func Test_mode_changes()
3963 let g:index = 0
zeertzjq73916ba2023-04-26 16:50:19 +01003964 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 +01003965 func! TestMode()
3966 call assert_equal(g:mode_seq[g:index], get(v:event, "old_mode"))
3967 call assert_equal(g:mode_seq[g:index + 1], get(v:event, "new_mode"))
3968 call assert_equal(mode(1), get(v:event, "new_mode"))
3969 let g:index += 1
3970 endfunc
3971
3972 au ModeChanged * :call TestMode()
3973 let g:n_to_any = 0
3974 au ModeChanged n:* let g:n_to_any += 1
zeertzjq73916ba2023-04-26 16:50:19 +01003975 call feedkeys("i\<esc>vVca\<CR>\<C-X>\<C-L>\<esc>ggdV\<MouseMove>G", 'tnix')
zeertzjqc9e8fd62022-07-26 18:12:38 +01003976
3977 let g:V_to_v = 0
3978 au ModeChanged V:v let g:V_to_v += 1
3979 call feedkeys("Vv\<C-G>\<esc>", 'tnix')
3980 call assert_equal(len(filter(g:mode_seq[1:], {idx, val -> val == 'n'})), g:n_to_any)
3981 call assert_equal(1, g:V_to_v)
3982 call assert_equal(len(g:mode_seq) - 1, g:index)
3983
3984 let g:n_to_i = 0
3985 au ModeChanged n:i let g:n_to_i += 1
3986 let g:n_to_niI = 0
3987 au ModeChanged i:niI let g:n_to_niI += 1
3988 let g:niI_to_i = 0
3989 au ModeChanged niI:i let g:niI_to_i += 1
3990 let g:nany_to_i = 0
3991 au ModeChanged n*:i let g:nany_to_i += 1
3992 let g:i_to_n = 0
3993 au ModeChanged i:n let g:i_to_n += 1
3994 let g:nori_to_any = 0
3995 au ModeChanged [ni]:* let g:nori_to_any += 1
3996 let g:i_to_any = 0
3997 au ModeChanged i:* let g:i_to_any += 1
3998 let g:index = 0
3999 let g:mode_seq = ['n', 'i', 'niI', 'i', 'n']
4000 call feedkeys("a\<C-O>l\<esc>", 'tnix')
4001 call assert_equal(len(g:mode_seq) - 1, g:index)
4002 call assert_equal(1, g:n_to_i)
4003 call assert_equal(1, g:n_to_niI)
4004 call assert_equal(1, g:niI_to_i)
4005 call assert_equal(2, g:nany_to_i)
4006 call assert_equal(1, g:i_to_n)
4007 call assert_equal(2, g:i_to_any)
4008 call assert_equal(3, g:nori_to_any)
4009
4010 if has('terminal')
4011 let g:mode_seq += ['c', 'n', 't', 'nt', 'c', 'nt', 'n']
4012 call feedkeys(":term\<CR>\<C-W>N:bd!\<CR>", 'tnix')
4013 call assert_equal(len(g:mode_seq) - 1, g:index)
4014 call assert_equal(1, g:n_to_i)
4015 call assert_equal(1, g:n_to_niI)
4016 call assert_equal(1, g:niI_to_i)
4017 call assert_equal(2, g:nany_to_i)
4018 call assert_equal(1, g:i_to_n)
4019 call assert_equal(2, g:i_to_any)
4020 call assert_equal(5, g:nori_to_any)
4021 endif
4022
zeertzjqd1955982022-10-05 11:24:46 +01004023 let g:n_to_c = 0
4024 au ModeChanged n:c let g:n_to_c += 1
4025 let g:c_to_n = 0
4026 au ModeChanged c:n let g:c_to_n += 1
4027 let g:mode_seq += ['c', 'n', 'c', 'n']
4028 call feedkeys("q:\<C-C>\<Esc>", 'tnix')
4029 call assert_equal(len(g:mode_seq) - 1, g:index)
4030 call assert_equal(2, g:n_to_c)
4031 call assert_equal(2, g:c_to_n)
zeertzjqc9e8fd62022-07-26 18:12:38 +01004032
Bram Moolenaar61c4b042022-10-18 15:10:11 +01004033 let g:n_to_v = 0
4034 au ModeChanged n:v let g:n_to_v += 1
4035 let g:v_to_n = 0
4036 au ModeChanged v:n let g:v_to_n += 1
4037 let g:mode_seq += ['v', 'n']
4038 call feedkeys("v\<C-C>", 'tnix')
4039 call assert_equal(len(g:mode_seq) - 1, g:index)
4040 call assert_equal(1, g:n_to_v)
4041 call assert_equal(1, g:v_to_n)
zeertzjqfcaeb3d2023-11-28 20:46:29 +01004042
4043 let g:mode_seq += ['c', 'cr', 'c', 'cr', 'n']
4044 call feedkeys(":\<Insert>\<Insert>\<Insert>\<CR>", 'tnix')
4045 call assert_equal(len(g:mode_seq) - 1, g:index)
Bram Moolenaar61c4b042022-10-18 15:10:11 +01004046
zeertzjqc9e8fd62022-07-26 18:12:38 +01004047 au! ModeChanged
4048 delfunc TestMode
4049 unlet! g:mode_seq
4050 unlet! g:index
4051 unlet! g:n_to_any
4052 unlet! g:V_to_v
4053 unlet! g:n_to_i
4054 unlet! g:n_to_niI
4055 unlet! g:niI_to_i
4056 unlet! g:nany_to_i
4057 unlet! g:i_to_n
4058 unlet! g:nori_to_any
4059 unlet! g:i_to_any
zeertzjqfcaeb3d2023-11-28 20:46:29 +01004060 unlet! g:n_to_c
4061 unlet! g:c_to_n
4062 unlet! g:n_to_v
4063 unlet! g:v_to_n
zeertzjqc9e8fd62022-07-26 18:12:38 +01004064endfunc
4065
4066func Test_recursive_ModeChanged()
4067 au! ModeChanged * norm 0u
4068 sil! norm 
4069 au! ModeChanged
4070endfunc
4071
4072func Test_ModeChanged_starts_visual()
4073 " This was triggering ModeChanged before setting VIsual, causing a crash.
4074 au! ModeChanged * norm 0u
4075 sil! norm 
4076
4077 au! ModeChanged
4078endfunc
Bram Moolenaar347538f2022-03-26 16:28:06 +00004079
Charlie Grovesfef44852022-04-19 16:24:12 +01004080func Test_noname_autocmd()
4081 augroup test_noname_autocmd_group
4082 autocmd!
4083 autocmd BufEnter * call add(s:li, ["BufEnter", expand("<afile>")])
4084 autocmd BufDelete * call add(s:li, ["BufDelete", expand("<afile>")])
4085 autocmd BufLeave * call add(s:li, ["BufLeave", expand("<afile>")])
4086 autocmd BufUnload * call add(s:li, ["BufUnload", expand("<afile>")])
4087 autocmd BufWipeout * call add(s:li, ["BufWipeout", expand("<afile>")])
4088 augroup END
4089
4090 let s:li = []
4091 edit foo
4092 call assert_equal([['BufUnload', ''], ['BufDelete', ''], ['BufWipeout', ''], ['BufEnter', 'foo']], s:li)
4093
4094 au! test_noname_autocmd_group
4095 augroup! test_noname_autocmd_group
4096endfunc
4097
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004098" Test for the autocmd_get() function
4099func Test_autocmd_get()
4100 augroup TestAutoCmdFns
4101 au!
4102 autocmd BufAdd *.vim echo "bufadd-vim"
4103 autocmd BufAdd *.py echo "bufadd-py"
4104 autocmd BufHidden *.vim echo "bufhidden"
4105 augroup END
4106 augroup TestAutoCmdFns2
4107 autocmd BufAdd *.vim echo "bufadd-vim-2"
4108 autocmd BufRead *.a1b2c3 echo "bufadd-vim-2"
4109 augroup END
4110
4111 let l = autocmd_get()
4112 call assert_true(l->len() > 0)
4113
4114 " Test for getting all the autocmds in a group
4115 let expected = [
4116 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4117 \ pattern: '*.vim', nested: v:false, once: v:false,
4118 \ event: 'BufAdd'},
4119 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4120 \ pattern: '*.py', nested: v:false, once: v:false,
4121 \ event: 'BufAdd'},
4122 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4123 \ pattern: '*.vim', nested: v:false,
4124 \ once: v:false, event: 'BufHidden'}]
4125 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4126
4127 " Test for getting autocmds for all the patterns in a group
4128 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns',
4129 \ event: '*'}))
4130
4131 " Test for getting autocmds for an event in a group
4132 let expected = [
4133 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4134 \ pattern: '*.vim', nested: v:false, once: v:false,
4135 \ event: 'BufAdd'},
4136 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4137 \ pattern: '*.py', nested: v:false, once: v:false,
4138 \ event: 'BufAdd'}]
4139 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns',
4140 \ event: 'BufAdd'}))
4141
4142 " Test for getting the autocmds for all the events in a group for particular
4143 " pattern
4144 call assert_equal([{'cmd': 'echo "bufadd-py"', 'group': 'TestAutoCmdFns',
4145 \ 'pattern': '*.py', 'nested': v:false, 'once': v:false,
4146 \ 'event': 'BufAdd'}],
4147 \ autocmd_get(#{group: 'TestAutoCmdFns', event: '*', pattern: '*.py'}))
4148
4149 " Test for getting the autocmds for an events in a group for particular
4150 " pattern
4151 let l = autocmd_get(#{group: 'TestAutoCmdFns', event: 'BufAdd',
4152 \ pattern: '*.vim'})
4153 call assert_equal([
4154 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4155 \ pattern: '*.vim', nested: v:false, once: v:false,
4156 \ event: 'BufAdd'}], l)
4157
4158 " Test for getting the autocmds for a pattern in a group
4159 let l = autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.vim'})
4160 call assert_equal([
4161 \ #{cmd: 'echo "bufadd-vim"', group: 'TestAutoCmdFns',
4162 \ pattern: '*.vim', nested: v:false, once: v:false,
4163 \ event: 'BufAdd'},
4164 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4165 \ pattern: '*.vim', nested: v:false,
4166 \ once: v:false, event: 'BufHidden'}], l)
4167
4168 " Test for getting the autocmds for a pattern in all the groups
4169 let l = autocmd_get(#{pattern: '*.a1b2c3'})
4170 call assert_equal([{'cmd': 'echo "bufadd-vim-2"', 'group': 'TestAutoCmdFns2',
4171 \ 'pattern': '*.a1b2c3', 'nested': v:false, 'once': v:false,
4172 \ 'event': 'BufRead'}], l)
4173
4174 " Test for getting autocmds for a pattern without any autocmds
4175 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4176 \ pattern: '*.abc'}))
4177 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4178 \ event: 'BufAdd', pattern: '*.abc'}))
4179 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns',
4180 \ event: 'BufWipeout'}))
zeertzjq2d1d5c62024-06-09 16:44:33 +02004181
4182 " Test for getting autocmds after removing one inside an autocmd
4183 func CheckAutocmdGet()
4184 augroup TestAutoCmdFns
4185 autocmd! BufAdd *.vim
4186 augroup END
4187
4188 let expected = [
4189 \ #{cmd: 'echo "bufadd-py"', group: 'TestAutoCmdFns',
4190 \ pattern: '*.py', nested: v:false, once: v:false,
4191 \ event: 'BufAdd'},
4192 \ #{cmd: 'echo "bufhidden"', group: 'TestAutoCmdFns',
4193 \ pattern: '*.vim', nested: v:false,
4194 \ once: v:false, event: 'BufHidden'}]
4195
4196 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4197 call assert_equal([expected[0]],
4198 \ autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.py'}))
4199 call assert_equal([expected[1]],
4200 \ autocmd_get(#{group: 'TestAutoCmdFns', pattern: '*.vim'}))
4201 endfunc
4202
4203 autocmd User Xauget call CheckAutocmdGet()
4204 doautocmd User Xauget
4205 autocmd! User Xauget
4206
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004207 call assert_fails("call autocmd_get(#{group: 'abc', event: 'BufAdd'})",
4208 \ 'E367:')
4209 let cmd = "echo autocmd_get(#{group: 'TestAutoCmdFns', event: 'abc'})"
4210 call assert_fails(cmd, 'E216:')
4211 call assert_fails("call autocmd_get(#{group: 'abc'})", 'E367:')
4212 call assert_fails("echo autocmd_get(#{event: 'abc'})", 'E216:')
4213
4214 augroup TestAutoCmdFns
4215 au!
4216 augroup END
4217 call assert_equal([], autocmd_get(#{group: 'TestAutoCmdFns'}))
4218
4219 " Test for nested and once autocmds
4220 augroup TestAutoCmdFns
4221 au!
4222 autocmd VimSuspend * ++nested echo "suspend"
4223 autocmd VimResume * ++once echo "resume"
4224 augroup END
4225
4226 let expected = [
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004227 \ {'cmd': 'echo "resume"', 'group': 'TestAutoCmdFns', 'pattern': '*',
Luuk van Baalb7147f82025-02-08 18:52:39 +01004228 \ 'nested': v:false, 'once': v:true, 'event': 'VimResume'},
4229 \ {'cmd': 'echo "suspend"', 'group': 'TestAutoCmdFns', 'pattern': '*',
4230 \ 'nested': v:true, 'once': v:false, 'event': 'VimSuspend'}]
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004231 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4232
4233 " Test for buffer-local autocmd
4234 augroup TestAutoCmdFns
4235 au!
4236 autocmd TextYankPost <buffer> echo "textyankpost"
4237 augroup END
4238
4239 let expected = [
4240 \ {'cmd': 'echo "textyankpost"', 'group': 'TestAutoCmdFns',
4241 \ 'pattern': '<buffer=' .. bufnr() .. '>', 'nested': v:false,
4242 \ 'once': v:false, 'bufnr': bufnr(), 'event': 'TextYankPost'}]
4243 call assert_equal(expected, autocmd_get(#{group: 'TestAutoCmdFns'}))
4244
4245 augroup TestAutoCmdFns
4246 au!
4247 augroup END
4248 augroup! TestAutoCmdFns
4249 augroup TestAutoCmdFns2
4250 au!
4251 augroup END
4252 augroup! TestAutoCmdFns2
4253
4254 call assert_fails("echo autocmd_get(#{group: []})", 'E730:')
4255 call assert_fails("echo autocmd_get(#{event: {}})", 'E731:')
4256 call assert_fails("echo autocmd_get([])", 'E1206:')
4257endfunc
4258
4259" Test for the autocmd_add() function
4260func Test_autocmd_add()
4261 " Define a single autocmd in a group
4262 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pattern: '*.sh',
4263 \ cmd: 'echo "bufadd"', once: v:true, nested: v:true}])
4264 call assert_equal([#{cmd: 'echo "bufadd"', group: 'TestAcSet',
4265 \ pattern: '*.sh', nested: v:true, once: v:true,
4266 \ event: 'BufAdd'}], autocmd_get(#{group: 'TestAcSet'}))
4267
4268 " Define two autocmds in the same group
4269 call autocmd_delete([#{group: 'TestAcSet'}])
4270 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pattern: '*.sh',
4271 \ cmd: 'echo "bufadd"'},
4272 \ #{group: 'TestAcSet', event: 'BufEnter', pattern: '*.sh',
4273 \ cmd: 'echo "bufenter"'}])
4274 call assert_equal([
4275 \ #{cmd: 'echo "bufadd"', group: 'TestAcSet', pattern: '*.sh',
4276 \ nested: v:false, once: v:false, event: 'BufAdd'},
4277 \ #{cmd: 'echo "bufenter"', group: 'TestAcSet', pattern: '*.sh',
4278 \ nested: v:false, once: v:false, event: 'BufEnter'}],
4279 \ autocmd_get(#{group: 'TestAcSet'}))
4280
4281 " Define a buffer-local autocmd
4282 call autocmd_delete([#{group: 'TestAcSet'}])
4283 call autocmd_add([#{group: 'TestAcSet', event: 'CursorHold',
4284 \ bufnr: bufnr(), cmd: 'echo "cursorhold"'}])
4285 call assert_equal([
4286 \ #{cmd: 'echo "cursorhold"', group: 'TestAcSet',
4287 \ pattern: '<buffer=' .. bufnr() .. '>', nested: v:false,
4288 \ once: v:false, bufnr: bufnr(), event: 'CursorHold'}],
4289 \ autocmd_get(#{group: 'TestAcSet'}))
4290
4291 " Use an invalid buffer number
4292 call autocmd_delete([#{group: 'TestAcSet'}])
4293 call autocmd_add([#{group: 'TestAcSet', event: 'BufEnter',
4294 \ bufnr: -1, cmd: 'echo "bufenter"'}])
4295 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4296 \ cmd: 'echo "bufadd"'}]
4297 call assert_fails("echo autocmd_add(l)", 'E680:')
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004298 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4299 \ pattern: '*.py', cmd: 'echo "bufadd"'}]
4300 call assert_fails("echo autocmd_add(l)", 'E680:')
4301 let l = [#{group: 'TestAcSet', event: 'BufAdd', bufnr: 9999,
4302 \ pattern: ['*.py', '*.c'], cmd: 'echo "bufadd"'}]
4303 call assert_fails("echo autocmd_add(l)", 'E680:')
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004304 let l = [#{group: 'TestAcSet', event: 'BufRead', bufnr: [],
4305 \ cmd: 'echo "bufread"'}]
4306 call assert_fails("echo autocmd_add(l)", 'E745:')
4307 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4308
4309 " Add two commands to the same group, event and pattern
4310 call autocmd_delete([#{group: 'TestAcSet'}])
4311 call autocmd_add([#{group: 'TestAcSet', event: 'BufUnload',
4312 \ pattern: 'abc', cmd: 'echo "cmd1"'}])
4313 call autocmd_add([#{group: 'TestAcSet', event: 'BufUnload',
4314 \ pattern: 'abc', cmd: 'echo "cmd2"'}])
4315 call assert_equal([
4316 \ #{cmd: 'echo "cmd1"', group: 'TestAcSet', pattern: 'abc',
4317 \ nested: v:false, once: v:false, event: 'BufUnload'},
4318 \ #{cmd: 'echo "cmd2"', group: 'TestAcSet', pattern: 'abc',
4319 \ nested: v:false, once: v:false, event: 'BufUnload'}],
4320 \ autocmd_get(#{group: 'TestAcSet'}))
4321
4322 " When adding a new autocmd, if the autocmd 'group' is not specified, then
4323 " the current autocmd group should be used.
4324 call autocmd_delete([#{group: 'TestAcSet'}])
4325 augroup TestAcSet
4326 call autocmd_add([#{event: 'BufHidden', pattern: 'abc', cmd: 'echo "abc"'}])
4327 augroup END
4328 call assert_equal([
4329 \ #{cmd: 'echo "abc"', group: 'TestAcSet', pattern: 'abc',
4330 \ nested: v:false, once: v:false, event: 'BufHidden'}],
4331 \ autocmd_get(#{group: 'TestAcSet'}))
4332
Yegappan Lakshmanan971f6822022-05-24 11:40:11 +01004333 " Test for replacing a cmd for an event in a group
4334 call autocmd_delete([#{group: 'TestAcSet'}])
4335 call autocmd_add([#{replace: v:true, group: 'TestAcSet', event: 'BufEnter',
4336 \ pattern: '*.py', cmd: 'echo "bufenter"'}])
4337 call autocmd_add([#{replace: v:true, group: 'TestAcSet', event: 'BufEnter',
4338 \ pattern: '*.py', cmd: 'echo "bufenter"'}])
4339 call assert_equal([
4340 \ #{cmd: 'echo "bufenter"', group: 'TestAcSet', pattern: '*.py',
4341 \ nested: v:false, once: v:false, event: 'BufEnter'}],
4342 \ autocmd_get(#{group: 'TestAcSet'}))
4343
4344 " Test for adding a command for an unsupported autocmd event
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004345 let l = [#{group: 'TestAcSet', event: 'abc', pattern: '*.sh',
4346 \ cmd: 'echo "bufadd"'}]
4347 call assert_fails('call autocmd_add(l)', 'E216:')
4348
Yegappan Lakshmanane0ff3a72022-05-27 18:05:33 +01004349 " Test for using a list of events and patterns
4350 call autocmd_delete([#{group: 'TestAcSet'}])
4351 let l = [#{group: 'TestAcSet', event: ['BufEnter', 'BufLeave'],
4352 \ pattern: ['*.py', '*.sh'], cmd: 'echo "bufcmds"'}]
4353 call autocmd_add(l)
4354 call assert_equal([
4355 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.py',
4356 \ nested: v:false, once: v:false, event: 'BufEnter'},
4357 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.sh',
4358 \ nested: v:false, once: v:false, event: 'BufEnter'},
4359 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.py',
4360 \ nested: v:false, once: v:false, event: 'BufLeave'},
4361 \ #{cmd: 'echo "bufcmds"', group: 'TestAcSet', pattern: '*.sh',
4362 \ nested: v:false, once: v:false, event: 'BufLeave'}],
4363 \ autocmd_get(#{group: 'TestAcSet'}))
4364
4365 " Test for invalid values for 'event' item
4366 call autocmd_delete([#{group: 'TestAcSet'}])
4367 let l = [#{group: 'TestAcSet', event: test_null_string(),
4368 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4369 call assert_fails('call autocmd_add(l)', 'E928:')
4370 let l = [#{group: 'TestAcSet', event: test_null_list(),
4371 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4372 call assert_fails('call autocmd_add(l)', 'E714:')
4373 let l = [#{group: 'TestAcSet', event: {},
4374 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4375 call assert_fails('call autocmd_add(l)', 'E777:')
4376 let l = [#{group: 'TestAcSet', event: [{}],
4377 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4378 call assert_fails('call autocmd_add(l)', 'E928:')
4379 let l = [#{group: 'TestAcSet', event: [test_null_string()],
4380 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4381 call assert_fails('call autocmd_add(l)', 'E928:')
4382 let l = [#{group: 'TestAcSet', event: 'BufEnter,BufLeave',
4383 \ pattern: '*.py', cmd: 'echo "bufcmds"'}]
4384 call assert_fails('call autocmd_add(l)', 'E216:')
4385 let l = [#{group: 'TestAcSet', event: [],
4386 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4387 call autocmd_add(l)
4388 let l = [#{group: 'TestAcSet', event: [""],
4389 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4390 call assert_fails('call autocmd_add(l)', 'E216:')
4391 let l = [#{group: 'TestAcSet', event: "",
4392 \ pattern: "*.py", cmd: 'echo "bufcmds"'}]
4393 call autocmd_add(l)
4394 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4395
4396 " Test for invalid values for 'pattern' item
4397 let l = [#{group: 'TestAcSet', event: "BufEnter",
4398 \ pattern: test_null_string(), cmd: 'echo "bufcmds"'}]
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004399 call assert_fails('call autocmd_add(l)', 'E928:')
Yegappan Lakshmanane0ff3a72022-05-27 18:05:33 +01004400 let l = [#{group: 'TestAcSet', event: "BufEnter",
4401 \ pattern: test_null_list(), cmd: 'echo "bufcmds"'}]
4402 call assert_fails('call autocmd_add(l)', 'E714:')
4403 let l = [#{group: 'TestAcSet', event: "BufEnter",
4404 \ pattern: {}, cmd: 'echo "bufcmds"'}]
4405 call assert_fails('call autocmd_add(l)', 'E777:')
4406 let l = [#{group: 'TestAcSet', event: "BufEnter",
4407 \ pattern: [{}], cmd: 'echo "bufcmds"'}]
4408 call assert_fails('call autocmd_add(l)', 'E928:')
4409 let l = [#{group: 'TestAcSet', event: "BufEnter",
4410 \ pattern: [test_null_string()], cmd: 'echo "bufcmds"'}]
4411 call assert_fails('call autocmd_add(l)', 'E928:')
4412 let l = [#{group: 'TestAcSet', event: "BufEnter",
4413 \ pattern: [], cmd: 'echo "bufcmds"'}]
4414 call autocmd_add(l)
4415 let l = [#{group: 'TestAcSet', event: "BufEnter",
4416 \ pattern: [""], cmd: 'echo "bufcmds"'}]
4417 call autocmd_add(l)
4418 let l = [#{group: 'TestAcSet', event: "BufEnter",
4419 \ pattern: "", cmd: 'echo "bufcmds"'}]
4420 call autocmd_add(l)
4421 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4422
4423 let l = [#{group: 'TestAcSet', event: 'BufEnter,abc,BufLeave',
4424 \ pattern: '*.py', cmd: 'echo "bufcmds"'}]
4425 call assert_fails('call autocmd_add(l)', 'E216:')
4426
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004427 call assert_fails("call autocmd_add({})", 'E1211:')
4428 call assert_equal(v:false, autocmd_add(test_null_list()))
4429 call assert_true(autocmd_add([[]]))
4430 call assert_true(autocmd_add([test_null_dict()]))
4431
4432 augroup TestAcSet
4433 au!
4434 augroup END
4435
4436 call autocmd_add([#{group: 'TestAcSet'}])
4437 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd'}])
4438 call autocmd_add([#{group: 'TestAcSet', pat: '*.sh'}])
4439 call autocmd_add([#{group: 'TestAcSet', cmd: 'echo "a"'}])
4440 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', pat: '*.sh'}])
4441 call autocmd_add([#{group: 'TestAcSet', event: 'BufAdd', cmd: 'echo "a"'}])
4442 call autocmd_add([#{group: 'TestAcSet', pat: '*.sh', cmd: 'echo "a"'}])
4443 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4444
4445 augroup! TestAcSet
4446endfunc
4447
4448" Test for deleting autocmd events and groups
4449func Test_autocmd_delete()
4450 " Delete an event in an autocmd group
4451 augroup TestAcSet
4452 au!
4453 au BufAdd *.sh echo "bufadd"
4454 au BufEnter *.sh echo "bufenter"
4455 augroup END
4456 call autocmd_delete([#{group: 'TestAcSet', event: 'BufAdd'}])
4457 call assert_equal([#{cmd: 'echo "bufenter"', group: 'TestAcSet',
4458 \ pattern: '*.sh', nested: v:false, once: v:false,
4459 \ event: 'BufEnter'}], autocmd_get(#{group: 'TestAcSet'}))
4460
4461 " Delete all the events in an autocmd group
4462 augroup TestAcSet
4463 au BufAdd *.sh echo "bufadd"
4464 augroup END
4465 call autocmd_delete([#{group: 'TestAcSet', event: '*'}])
4466 call assert_equal([], autocmd_get(#{group: 'TestAcSet'}))
4467
4468 " Delete a non-existing autocmd group
4469 call assert_fails("call autocmd_delete([#{group: 'abc'}])", 'E367:')
4470 " Delete a non-existing autocmd event
4471 let l = [#{group: 'TestAcSet', event: 'abc'}]
4472 call assert_fails("call autocmd_delete(l)", 'E216:')
4473 " Delete a non-existing autocmd pattern
4474 let l = [#{group: 'TestAcSet', event: 'BufAdd', pat: 'abc'}]
4475 call assert_true(autocmd_delete(l))
Yegappan Lakshmanan00e977c2022-06-01 12:31:53 +01004476 " Delete an autocmd for a non-existing buffer
4477 let l = [#{event: '*', bufnr: 9999, cmd: 'echo "x"'}]
4478 call assert_fails('call autocmd_delete(l)', 'E680:')
Yegappan Lakshmanan1755a912022-05-19 10:31:47 +01004479
4480 " Delete an autocmd group
4481 augroup TestAcSet
4482 au!
4483 au BufAdd *.sh echo "bufadd"
4484 au BufEnter *.sh echo "bufenter"
4485 augroup END
4486 call autocmd_delete([#{group: 'TestAcSet'}])
4487 call assert_fails("call autocmd_get(#{group: 'TestAcSet'})", 'E367:')
4488
4489 call assert_true(autocmd_delete([[]]))
4490 call assert_true(autocmd_delete([test_null_dict()]))
4491endfunc
4492
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004493func Test_autocmd_split_dummy()
4494 " Autocommand trying to split a window containing a dummy buffer.
Bram Moolenaar94722c52023-01-28 19:19:03 +00004495 auto BufReadPre * exe "sbuf " .. expand("<abuf>")
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004496 " Avoid the "W11" prompt
4497 au FileChangedShell * let v:fcs_choice = 'reload'
4498 func Xautocmd_changelist()
4499 cal writefile(['Xtestfile2:4:4'], 'Xerr')
4500 edit Xerr
4501 lex 'Xtestfile2:4:4'
4502 endfunc
4503 call Xautocmd_changelist()
Bram Moolenaar53c5c9f2022-10-18 17:25:03 +01004504 " Should get E86, but it doesn't always happen (timing?)
4505 silent! call Xautocmd_changelist()
Bram Moolenaar8f3c3c62022-10-18 17:05:54 +01004506
4507 au! BufReadPre
4508 au! FileChangedShell
4509 delfunc Xautocmd_changelist
4510 bwipe! Xerr
4511 call delete('Xerr')
4512endfunc
4513
Bram Moolenaare76062c2022-11-28 18:51:43 +00004514" This was crashing because there was only one window to execute autocommands
4515" in.
4516func Test_autocmd_nested_setbufvar()
4517 CheckFeature python3
4518
4519 set hidden
4520 edit Xaaa
4521 edit Xbbb
4522 call setline(1, 'bar')
4523 enew
4524 au BufWriteCmd Xbbb ++nested call setbufvar('Xaaa', '&ft', 'foo') | bw! Xaaa
4525 au FileType foo call py3eval('vim.current.buffer.options["cindent"]')
4526 wall
4527
4528 au! BufWriteCmd
4529 au! FileType foo
4530 set nohidden
4531 call delete('Xaaa')
4532 call delete('Xbbb')
4533 %bwipe!
4534endfunc
4535
Christian Brabandt9aee8ec2022-12-16 16:41:23 +00004536func SetupVimTest_shm()
4537 let g:bwe = []
4538 let g:brp = []
4539 set shortmess+=F
zeertzjq657b31f2023-04-15 21:28:02 +01004540 messages clear
Christian Brabandt9aee8ec2022-12-16 16:41:23 +00004541
4542 let dirname='XVimTestSHM'
4543 call mkdir(dirname, 'R')
4544 call writefile(['test'], dirname .. '/1')
4545 call writefile(['test'], dirname .. '/2')
4546 call writefile(['test'], dirname .. '/3')
4547
4548 augroup test
4549 autocmd!
4550 autocmd BufWinEnter * call add(g:bwe, $'BufWinEnter: {expand('<amatch>')}')
4551 autocmd BufReadPost * call add(g:brp, $'BufReadPost: {expand('<amatch>')}')
4552 augroup END
4553
4554 call setqflist([
4555 \ {'filename': dirname .. '/1', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0},
4556 \ {'filename': dirname .. '/2', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0},
4557 \ {'filename': dirname .. '/3', 'lnum': 1, 'col': 1, 'text': 'test', 'vcol': 0}
4558 \ ])
4559 cdo! substitute/test/TEST
4560
4561 " clean up
4562 noa enew!
4563 set shortmess&vim
4564 augroup test
4565 autocmd!
4566 augroup END
4567 augroup! test
4568endfunc
4569
4570func Test_autocmd_shortmess()
4571 CheckNotMSWindows
4572
4573 call SetupVimTest_shm()
4574 let output = execute(':mess')->split('\n')
4575
4576 let info = copy(output)->filter({idx, val -> val =~# '\d of 3'} )
4577 let bytes = copy(output)->filter({idx, val -> val =~# 'bytes'} )
4578
4579 " We test the following here:
4580 " BufReadPost should have been triggered 3 times, once per file
4581 " BufWinEnter should have been triggered 3 times, once per file
4582 " FileInfoMessage should have been shown 3 times, regardless of shm option
4583 " "(x of 3)" message from :cnext has been shown 3 times
4584
4585 call assert_equal(3, g:brp->len())
4586 call assert_equal(3, g:bwe->len())
4587 call assert_equal(3, info->len())
4588 call assert_equal(3, bytes->len())
4589
4590 delfunc SetupVimTest_shm
4591endfunc
Bram Moolenaare76062c2022-11-28 18:51:43 +00004592
Christian Brabandtf0d3d4a2024-02-15 20:15:04 +01004593func Test_autocmd_invalidates_undo_on_textchanged()
4594 CheckRunVimInTerminal
4595 let script =<< trim END
4596 set hidden
4597 " create quickfix list (at least 2 lines to move line)
4598 vimgrep /u/j %
4599
4600 " enter quickfix window
4601 cwindow
4602
4603 " set modifiable
4604 setlocal modifiable
4605
4606 " set autocmd to clear quickfix list
4607
4608 autocmd! TextChanged <buffer> call setqflist([])
4609 " move line
4610 move+1
4611 END
4612 call writefile(script, 'XTest_autocmd_invalidates_undo_on_textchanged', 'D')
4613 let buf = RunVimInTerminal('XTest_autocmd_invalidates_undo_on_textchanged', {'rows': 20})
4614 call term_sendkeys(buf, ":so %\<cr>")
4615 call term_sendkeys(buf, "G")
4616 call WaitForAssert({-> assert_match('^XTest_autocmd_invalidates_undo_on_textchanged\s*$', term_getline(buf, 20))}, 1000)
4617
4618 call StopVimInTerminal(buf)
4619endfunc
4620
Christian Brabandt55f8bba2024-02-28 23:32:00 +01004621func Test_autocmd_creates_new_buffer_on_bufleave()
4622 e a.txt
4623 e b.txt
4624 setlocal bufhidden=wipe
4625 autocmd BufLeave <buffer> diffsplit c.txt
4626 bn
4627 call assert_equal(1, winnr('$'))
4628 call assert_equal('a.txt', bufname('%'))
4629 bw a.txt
4630 bw c.txt
4631endfunc
4632
Colin Kennedye5f22802024-03-26 18:20:16 +01004633" Ensure `expected` was just recently written as a Vim session
4634func s:assert_session_path(expected)
4635 call assert_equal(a:expected, v:this_session)
4636endfunc
4637
4638" Check for `expected` after a session is written to-disk.
4639func s:watch_for_session_path(expected)
4640 execute 'autocmd SessionWritePost * ++once execute "call s:assert_session_path(\"'
4641 \ . a:expected
4642 \ . '\")"'
4643endfunc
4644
4645" Ensure v:this_session gets the full session path, if explicitly stated
4646func Test_explicit_session_absolute_path()
4647 %bwipeout!
4648
4649 let directory = getcwd()
4650
4651 let v:this_session = ""
4652 let name = "some_file.vim"
4653 let expected = fnamemodify(name, ":p")
4654 call s:watch_for_session_path(expected)
4655 execute "mksession! " .. expected
4656
4657 call delete(expected)
4658endfunc
4659
4660" Ensure v:this_session gets the full session path, if explicitly stated
4661func Test_explicit_session_relative_path()
4662 %bwipeout!
4663
4664 let directory = getcwd()
4665
4666 let v:this_session = ""
4667 let name = "some_file.vim"
4668 let expected = fnamemodify(name, ":p")
4669 call s:watch_for_session_path(expected)
4670 execute "mksession! " .. name
4671
4672 call delete(expected)
4673endfunc
4674
4675" Ensure v:this_session gets the full session path, if not specified
4676func Test_implicit_session()
4677 %bwipeout!
4678
4679 let directory = getcwd()
4680
4681 let v:this_session = ""
4682 let expected = fnamemodify("Session.vim", ":p")
4683 call s:watch_for_session_path(expected)
4684 mksession!
4685
4686 call delete(expected)
4687endfunc
4688
Christian Brabandt86032702024-03-31 18:38:09 +02004689" Test TextChangedI and TextChanged
zeertzjqc4226622024-04-03 22:38:07 +02004690func Test_Changed_ChangedI()
zeertzjq8eb75232024-04-01 14:46:20 +02004691 " Run this test in a terminal because it requires running the main loop.
zeertzjqc4226622024-04-03 22:38:07 +02004692 " Don't use CheckRunVimInTerminal as that will skip the test on Windows.
4693 CheckFeature terminal
4694 CheckNotGui
4695 " Starting a terminal to run Vim is always considered flaky.
4696 let g:test_is_flaky = 1
4697
Christian Brabandt86032702024-03-31 18:38:09 +02004698 call writefile(['one', 'two', 'three'], 'XTextChangedI2', 'D')
4699 let before =<< trim END
zeertzjqc4226622024-04-03 22:38:07 +02004700 set ttimeout ttimeoutlen=10
zeertzjq8eb75232024-04-01 14:46:20 +02004701 let [g:autocmd_n, g:autocmd_i] = ['','']
4702
4703 func TextChangedAutocmd(char)
4704 let g:autocmd_{tolower(a:char)} = a:char .. b:changedtick
zeertzjqc4226622024-04-03 22:38:07 +02004705 call writefile([$'{g:autocmd_n},{g:autocmd_i}'], 'XTextChangedI3')
zeertzjq8eb75232024-04-01 14:46:20 +02004706 endfunc
4707
4708 au TextChanged <buffer> :call TextChangedAutocmd('N')
4709 au TextChangedI <buffer> :call TextChangedAutocmd('I')
4710
Christian Brabandt86032702024-03-31 18:38:09 +02004711 nnoremap <CR> o<Esc>
zeertzjq4a653912024-04-04 21:33:36 +02004712 autocmd SafeState * ++once call writefile([''], 'XTextChangedI3')
Christian Brabandt86032702024-03-31 18:38:09 +02004713 END
4714
4715 call writefile(before, 'Xinit', 'D')
zeertzjqc4226622024-04-03 22:38:07 +02004716 let buf = term_start(
4717 \ GetVimCommandCleanTerm() .. '-n -S Xinit XTextChangedI2',
4718 \ {'term_rows': 10})
4719 call assert_equal('running', term_getstatus(buf))
zeertzjq8eb75232024-04-01 14:46:20 +02004720 call WaitForAssert({-> assert_true(filereadable('XTextChangedI3'))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004721 defer delete('XTextChangedI3')
zeertzjq4a653912024-04-04 21:33:36 +02004722 call WaitForAssert({-> assert_equal([''], readfile('XTextChangedI3'))})
Christian Brabandt86032702024-03-31 18:38:09 +02004723
zeertzjqc4226622024-04-03 22:38:07 +02004724 " TextChanged should trigger if a mapping enters and leaves Insert mode.
4725 call term_sendkeys(buf, "\<CR>")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004726 call WaitForAssert({-> assert_equal('N4,', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004727
4728 call term_sendkeys(buf, "i")
4729 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004730 call WaitForAssert({-> assert_equal('N4,', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004731 " TextChangedI should trigger if change is done in Insert mode.
4732 call term_sendkeys(buf, "f")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004733 call WaitForAssert({-> assert_equal('N4,I5', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004734 call term_sendkeys(buf, "o")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004735 call WaitForAssert({-> assert_equal('N4,I6', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004736 call term_sendkeys(buf, "o")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004737 call WaitForAssert({-> assert_equal('N4,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004738 " TextChanged shouldn't trigger when leaving Insert mode and TextChangedI
4739 " has been triggered.
4740 call term_sendkeys(buf, "\<Esc>")
4741 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004742 call WaitForAssert({-> assert_equal('N4,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004743
4744 " TextChanged should trigger if change is done in Normal mode.
4745 call term_sendkeys(buf, "yyp")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004746 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004747
4748 " TextChangedI shouldn't trigger if change isn't done in Insert mode.
4749 call term_sendkeys(buf, "i")
4750 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004751 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004752 call term_sendkeys(buf, "\<Esc>")
4753 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004754 call WaitForAssert({-> assert_equal('N8,I7', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004755
4756 " TextChangedI should trigger if change is a mix of Normal and Insert modes.
4757 func! s:validate_mixed_textchangedi(buf, keys)
4758 let buf = a:buf
4759 call term_sendkeys(buf, "ifoo")
4760 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
4761 call term_sendkeys(buf, "\<Esc>")
4762 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
4763 call term_sendkeys(buf, ":let [g:autocmd_n, g:autocmd_i] = ['', '']\<CR>")
zeertzjqe9ff79a2024-04-05 20:07:39 +02004764 call writefile([], 'XTextChangedI3')
zeertzjqc4226622024-04-03 22:38:07 +02004765 call term_sendkeys(buf, a:keys)
4766 call WaitForAssert({-> assert_match('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004767 call WaitForAssert({-> assert_match('^,I\d\+', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004768 call term_sendkeys(buf, "\<Esc>")
4769 call WaitForAssert({-> assert_notmatch('^-- INSERT --', term_getline(buf, 10))})
zeertzjqe9ff79a2024-04-05 20:07:39 +02004770 call WaitForAssert({-> assert_match('^,I\d\+', readfile('XTextChangedI3')->join("\n"))})
zeertzjqc4226622024-04-03 22:38:07 +02004771 endfunc
4772
4773 call s:validate_mixed_textchangedi(buf, "o")
4774 call s:validate_mixed_textchangedi(buf, "O")
4775 call s:validate_mixed_textchangedi(buf, "ciw")
4776 call s:validate_mixed_textchangedi(buf, "cc")
4777 call s:validate_mixed_textchangedi(buf, "C")
4778 call s:validate_mixed_textchangedi(buf, "s")
4779 call s:validate_mixed_textchangedi(buf, "S")
4780
4781 " clean up
4782 bwipe!
Christian Brabandt86032702024-03-31 18:38:09 +02004783endfunc
4784
zeertzjq5bf6c212024-03-31 18:41:27 +02004785" Test that filetype detection still works when SwapExists autocommand sets
4786" filetype in another buffer.
4787func Test_SwapExists_set_other_buf_filetype()
4788 let lines =<< trim END
4789 set nocompatible directory=.
4790 filetype on
4791
4792 let g:buf = bufnr()
4793 new
4794
4795 func SwapExists()
4796 let v:swapchoice = 'o'
4797 call setbufvar(g:buf, '&filetype', 'text')
4798 endfunc
4799
4800 func SafeState()
4801 edit <script>
4802 redir! > XftSwapExists.out
4803 set readonly? filetype?
4804 redir END
4805 qall!
4806 endfunc
4807
4808 autocmd SwapExists * ++nested call SwapExists()
4809 autocmd SafeState * ++nested ++once call SafeState()
4810 END
4811 call writefile(lines, 'XftSwapExists.vim', 'D')
4812
4813 new XftSwapExists.vim
4814 if RunVim('', '', ' -S XftSwapExists.vim')
4815 call assert_equal(
4816 \ ['', ' readonly', ' filetype=vim'],
4817 \ readfile('XftSwapExists.out'))
4818 call delete('XftSwapExists.out')
4819 endif
4820
4821 bwipe!
4822endfunc
4823
4824" Test that file is not marked as modified when SwapExists autocommand sets
4825" 'modified' in another buffer.
4826func Test_SwapExists_set_other_buf_modified()
4827 let lines =<< trim END
4828 set nocompatible directory=.
4829
4830 let g:buf = bufnr()
4831 new
4832
4833 func SwapExists()
4834 let v:swapchoice = 'o'
4835 call setbufvar(g:buf, '&modified', 1)
4836 endfunc
4837
4838 func SafeState()
4839 edit <script>
4840 redir! > XmodSwapExists.out
4841 set readonly? modified?
4842 redir END
4843 qall!
4844 endfunc
4845
4846 autocmd SwapExists * ++nested call SwapExists()
4847 autocmd SafeState * ++nested ++once call SafeState()
4848 END
4849 call writefile(lines, 'XmodSwapExists.vim', 'D')
4850
4851 new XmodSwapExists.vim
4852 if RunVim('', '', ' -S XmodSwapExists.vim')
4853 call assert_equal(
4854 \ ['', ' readonly', 'nomodified'],
4855 \ readfile('XmodSwapExists.out'))
4856 call delete('XmodSwapExists.out')
4857 endif
4858
4859 bwipe!
4860endfunc
4861
Jaehwang Jungeb80b832024-04-26 18:48:48 +02004862func Test_BufEnter_botline()
4863 set hidden
4864 call writefile(range(10), 'Xxx1', 'D')
4865 call writefile(range(20), 'Xxx2', 'D')
4866 edit Xxx1
4867 edit Xxx2
4868 au BufEnter Xxx1 call assert_true(line('w$') > 1)
4869 edit Xxx1
zeertzjq340643e2024-04-27 11:33:24 +02004870
4871 bwipe! Xxx1
4872 bwipe! Xxx2
Jaehwang Jungeb80b832024-04-26 18:48:48 +02004873 au! BufEnter Xxx1
4874 set hidden&vim
4875endfunc
4876
Shougo Matsushita83678842024-07-11 22:05:12 +02004877func Test_KeyInputPre()
4878 " Consume previous keys
4879 call feedkeys('', 'ntx')
4880
4881 " KeyInputPre can record input keys.
4882 let s:keys = []
4883 au KeyInputPre n call add(s:keys, v:char)
4884
4885 call feedkeys('jkjkjjj', 'ntx')
4886 call assert_equal(
4887 \ ['j', 'k', 'j', 'k', 'j', 'j', 'j'],
4888 \ s:keys)
4889
4890 unlet s:keys
4891 au! KeyInputPre
4892
4893 " KeyInputPre can handle multibyte.
4894 let s:keys = []
4895 au KeyInputPre * call add(s:keys, v:char)
4896 edit Xxx1
4897
4898 call feedkeys("iあ\<ESC>", 'ntx')
4899 call assert_equal(['i', "あ", "\<ESC>"], s:keys)
4900
4901 bwipe! Xxx1
4902 unlet s:keys
4903 au! KeyInputPre
4904
4905 " KeyInputPre can change input keys.
4906 au KeyInputPre i if v:char ==# 'a' | let v:char = 'b' | endif
4907 edit Xxx1
4908
4909 call feedkeys("iaabb\<ESC>", 'ntx')
4910 call assert_equal(getline('.'), 'bbbb')
4911
4912 bwipe! Xxx1
4913 au! KeyInputPre
4914
4915 " KeyInputPre returns multiple characters.
4916 au KeyInputPre i if v:char ==# 'a' | let v:char = 'cccc' | endif
4917 edit Xxx1
4918
4919 call feedkeys("iaabb\<ESC>", 'ntx')
4920 call assert_equal(getline('.'), 'ccbb')
4921
4922 bwipe! Xxx1
4923 au! KeyInputPre
4924
4925 " KeyInputPre can use special keys.
4926 au KeyInputPre i if v:char ==# 'a' | let v:char = "\<Ignore>" | endif
4927 edit Xxx1
4928
4929 call feedkeys("iaabb\<ESC>", 'ntx')
4930 call assert_equal(getline('.'), 'bb')
4931
4932 bwipe! Xxx1
4933 au! KeyInputPre
4934
4935 " Test for v:event.typed
4936 au KeyInputPre n call assert_true(v:event.typed)
4937 call feedkeys('j', 'ntx')
4938
4939 au! KeyInputPre
4940
4941 au KeyInputPre n call assert_false(v:event.typed)
4942 call feedkeys('j', 'nx')
4943
4944 au! KeyInputPre
Shougo Matsushitafcc1b572024-07-17 20:25:22 +02004945
4946 " Test for v:event.typedchar
4947 nnoremap j k
4948 au KeyInputPre n
4949 \ call assert_equal(v:event.typedchar, 'j')
4950 \ | call assert_equal(v:char, 'k')
4951 call feedkeys('j', 'tx')
4952
4953 au! KeyInputPre
Shougo Matsushita83678842024-07-11 22:05:12 +02004954endfunc
4955
Christian Brabandtfb3f9692024-08-11 20:09:17 +02004956" those commands caused null pointer access, see #15464
4957func Test_WinNewPre_crash()
4958 defer CleanUpTestAuGroup()
4959 let _cmdheight=&cmdheight
4960 augroup testing
4961 au!
4962 autocmd WinNewPre * redraw
4963 augroup END
4964 tabnew
4965 tabclose
4966 augroup testing
4967 au!
4968 autocmd WinNewPre * wincmd t
4969 augroup END
4970 tabnew
4971 tabclose
4972 augroup testing
4973 au!
4974 autocmd WinNewPre * wincmd b
4975 augroup END
4976 tabnew
4977 tabclose
4978 augroup testing
4979 au!
4980 autocmd WinNewPre * set cmdheight+=1
4981 augroup END
4982 tabnew
4983 tabclose
4984 let &cmdheight=_cmdheight
4985endfunc
4986
Christian Brabandt84e31752024-09-02 09:59:18 +02004987" The specifics of the turkish locale may
4988" cause that Vim will not treat the GuiEnter autocommand
4989" as case insensitive and instead issues an error
4990func Test_GuiEnter_Turkish_locale()
4991 try
4992 let lng = v:lang
4993 lang tr_TR.UTF-8
4994 let result = execute(':au GuiEnter')
Ken Takataeccc9272024-09-03 23:01:55 +02004995 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02004996 let result = execute(':au GUIENTER')
Ken Takataeccc9272024-09-03 23:01:55 +02004997 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02004998 let result = execute(':au guienter')
Ken Takataeccc9272024-09-03 23:01:55 +02004999 call assert_equal(gettext("\n--- Autocommands ---"), result)
Christian Brabandt84e31752024-09-02 09:59:18 +02005000 exe ":lang" lng
5001 catch /E197:/
5002 " can't use Turkish locale
5003 throw 'Skipped: Turkish locale not available'
5004 endtry
5005endfunc
Christian Brabandtfb3f9692024-08-11 20:09:17 +02005006
Christian Brabandt51b62382024-10-06 17:31:10 +02005007" This was using freed memory
5008func Test_autocmd_BufWinLeave_with_vsp()
5009 new
5010 let fname = 'XXXBufWinLeaveUAF.txt'
5011 let dummy = 'XXXDummy.txt'
5012 call writefile([], fname)
5013 call writefile([], dummy)
5014 defer delete(fname)
5015 defer delete(dummy)
5016 exe "e " fname
5017 vsp
5018 augroup testing
5019 exe "au BufWinLeave " .. fname .. " :e " dummy .. "| vsp " .. fname
5020 augroup END
5021 bw
5022 call CleanUpTestAuGroup()
5023 exe "bw! " .. dummy
5024endfunc
5025
Luuk van Baale15cbc12025-01-04 17:18:08 +01005026func Test_OptionSet_cmdheight()
5027 set mouse=a laststatus=2
5028 au OptionSet cmdheight :let &l:ch = v:option_new
5029
5030 resize -1
5031 call assert_equal(2, &l:ch)
5032 resize +1
5033 call assert_equal(1, &l:ch)
5034
5035 call test_setmouse(&lines - 1, 1)
5036 call feedkeys("\<LeftMouse>", 'xt')
5037 call test_setmouse(&lines - 2, 1)
5038 call feedkeys("\<LeftDrag>", 'xt')
5039 call assert_equal(2, &l:ch)
5040
5041 tabnew | resize +1
5042 call assert_equal(1, &l:ch)
5043 tabfirst
5044 call assert_equal(2, &l:ch)
5045
5046 tabonly
5047 set cmdheight& mouse& laststatus&
5048endfunc
5049
Luuk van Baalb7147f82025-02-08 18:52:39 +01005050func Test_eventignorewin()
5051 defer CleanUpTestAuGroup()
5052 augroup testing
5053 au WinEnter * :call add(g:evs, ["WinEnter", expand("<afile>")])
5054 au WinLeave * :call add(g:evs, ["WinLeave", expand("<afile>")])
5055 au BufWinEnter * :call add(g:evs, ["BufWinEnter", expand("<afile>")])
5056 augroup END
5057
5058 let g:evs = []
5059 set eventignorewin=WinLeave,WinEnter
5060 split foo
5061 call assert_equal([['BufWinEnter', 'foo']], g:evs)
5062 set eventignorewin=all
5063 edit bar
5064 call assert_equal([['BufWinEnter', 'foo']], g:evs)
5065 set eventignorewin=
5066 wincmd w
5067 call assert_equal([['BufWinEnter', 'foo'], ['WinLeave', 'bar']], g:evs)
5068
5069 only!
5070 %bwipe!
5071 set eventignorewin&
5072 unlet g:evs
5073endfunc
5074
5075func Test_WinScrolled_Resized_eiw()
5076 CheckRunVimInTerminal
5077
5078 let lines =<< trim END
5079 call setline(1, ['foo']->repeat(32))
5080 set eventignorewin=WinScrolled,WinResized
5081 split
5082 let [g:afile,g:resized,g:scrolled] = ['none',0,0]
5083 au WinScrolled * let [g:afile,g:scrolled] = [expand('<afile>'),g:scrolled+1]
5084 au WinResized * let [g:afile,g:resized] = [expand('<afile>'),g:resized+1]
5085 END
Christian Brabandtbfc77192025-02-11 20:03:10 +01005086 call writefile(lines, 'Xtest_winscrolled_eiw', 'D')
5087 let buf = RunVimInTerminal('-S Xtest_winscrolled_eiw', {'rows': 10})
Luuk van Baalb7147f82025-02-08 18:52:39 +01005088
5089 " Both windows are ignoring resize events
5090 call term_sendkeys(buf, "\<C-W>-")
5091 call TermWait(buf)
5092 call term_sendkeys(buf, ":echo g:afile g:resized g:scrolled\<CR>")
5093 call WaitForAssert({-> assert_equal('none 0 0', term_getline(buf, 10))}, 1000)
5094
5095 " And scroll events
5096 call term_sendkeys(buf, "Ggg")
5097 call TermWait(buf)
5098 call term_sendkeys(buf, ":echo g:afile g:resized g:scrolled\<CR>")
5099 call WaitForAssert({-> assert_equal('none 0 0', term_getline(buf, 10))}, 1000)
5100
5101 " Un-ignore events in second window, make first window current and resize
5102 call term_sendkeys(buf, ":set eventignorewin=\<CR>\<C-W>w\<C-W>+")
5103 call TermWait(buf)
5104 call term_sendkeys(buf, ":echo win_getid() g:afile g:resized g:scrolled\<CR>")
5105 call WaitForAssert({-> assert_equal('1000 1001 1 1', term_getline(buf, 10))}, 1000)
5106
5107 call StopVimInTerminal(buf)
5108endfunc
5109
Jim Zhou5606ca52025-03-13 21:58:25 +01005110" Test that TabClosedPre and TabClosed are triggered when closing a tab.
5111func Test_autocmd_tabclosedpre()
5112 augroup testing
5113 au TabClosedPre * call add(g:tabpagenr_pre, t:testvar)
5114 au TabClosed * call add(g:tabpagenr_post, t:testvar)
5115 augroup END
5116
5117 " Test 'tabclose' triggering
5118 let g:tabpagenr_pre = []
5119 let g:tabpagenr_post = []
5120 let t:testvar = 1
5121 tabnew
5122 let t:testvar = 2
5123 tabnew
5124 let t:testvar = 3
5125 tabnew
5126 let t:testvar = 4
5127 tabnext
5128 tabclose
5129 tabclose
5130 tabclose
5131 call assert_equal([1, 2, 3], g:tabpagenr_pre)
5132 call assert_equal([2, 3, 4], g:tabpagenr_post)
5133
5134 " Test 'tabclose {count}' triggering
5135 let g:tabpagenr_pre = []
5136 let g:tabpagenr_post = []
5137 let t:testvar = 1
5138 tabnew
5139 let t:testvar = 2
5140 tabnew
5141 let t:testvar = 3
5142 tabclose 2
5143 tabclose 2
5144 call assert_equal([2, 3], g:tabpagenr_pre)
5145 call assert_equal([3, 1], g:tabpagenr_post)
5146
5147 " Test 'tabonly' triggering
5148 let g:tabpagenr_pre = []
5149 let g:tabpagenr_post = []
5150 let t:testvar = 1
5151 tabnew
5152 let t:testvar = 2
5153 tabonly
5154 call assert_equal([1], g:tabpagenr_pre)
5155 call assert_equal([2], g:tabpagenr_post)
5156
5157 " Test 'q' and 'close' triggering (closing the last window in a tab)
5158 let g:tabpagenr_pre = []
5159 let g:tabpagenr_post = []
5160 split
5161 let t:testvar = 1
5162 tabnew
5163 let t:testvar = 2
5164 split
5165 vsplit
5166 tabnew
5167 let t:testvar = 3
5168 tabnext
5169 only
5170 quit
5171 quit
5172 close
5173 close
5174 call assert_equal([1, 2], g:tabpagenr_pre)
5175 call assert_equal([2, 3], g:tabpagenr_post)
5176
5177 func ClearAutomcdAndCreateTabs()
5178 au! TabClosedPre
5179 bw!
5180 e Z
5181 tabonly
5182 tabnew A
5183 tabnew B
5184 tabnew C
5185 endfunc
5186
5187 func GetTabs()
5188 redir => tabsout
5189 tabs
5190 redir END
5191 let tabsout = substitute(tabsout, '\n', '', 'g')
5192 let tabsout = substitute(tabsout, 'Tab page ', '', 'g')
5193 let tabsout = substitute(tabsout, ' ', '', 'g')
5194 return tabsout
5195 endfunc
5196
5197 call CleanUpTestAuGroup()
5198
5199 " Close tab in TabClosedPre autocmd
5200 call ClearAutomcdAndCreateTabs()
5201 au TabClosedPre * tabclose
zeertzjq67fe77d2025-04-20 10:21:18 +02005202 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005203 call ClearAutomcdAndCreateTabs()
5204 au TabClosedPre * tabclose
zeertzjq67fe77d2025-04-20 10:21:18 +02005205 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005206 call ClearAutomcdAndCreateTabs()
5207 au TabClosedPre * tabclose 1
zeertzjq67fe77d2025-04-20 10:21:18 +02005208 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005209
5210 " Close other (all) tabs in TabClosedPre autocmd
5211 call ClearAutomcdAndCreateTabs()
5212 au TabClosedPre * tabonly
zeertzjq67fe77d2025-04-20 10:21:18 +02005213 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005214 call ClearAutomcdAndCreateTabs()
5215 au TabClosedPre * tabonly
zeertzjq67fe77d2025-04-20 10:21:18 +02005216 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005217 call ClearAutomcdAndCreateTabs()
5218 au TabClosedPre * tabclose 4
zeertzjq67fe77d2025-04-20 10:21:18 +02005219 call assert_fails('tabclose 2', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005220
5221 " Open new tabs in TabClosedPre autocmd
5222 call ClearAutomcdAndCreateTabs()
5223 au TabClosedPre * tabnew D
zeertzjq67fe77d2025-04-20 10:21:18 +02005224 call assert_fails('tabclose', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005225 call ClearAutomcdAndCreateTabs()
5226 au TabClosedPre * tabnew D
zeertzjq67fe77d2025-04-20 10:21:18 +02005227 call assert_fails('tabclose 1', 'E1312:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005228
5229 " Moving the tab page in TabClosedPre autocmd
5230 call ClearAutomcdAndCreateTabs()
5231 au TabClosedPre * tabmove 0
5232 tabclose
Jim Zhoubcf66e02025-03-16 20:24:57 +01005233 call assert_equal('1>Z2A3B', GetTabs())
Jim Zhou5606ca52025-03-13 21:58:25 +01005234 call ClearAutomcdAndCreateTabs()
5235 au TabClosedPre * tabmove 0
5236 tabclose 1
5237 call assert_equal('1A2B3>C', GetTabs())
5238 tabonly
5239 call assert_equal('1>C', GetTabs())
5240
5241 " Switching tab page in TabClosedPre autocmd
5242 call ClearAutomcdAndCreateTabs()
5243 au TabClosedPre * tabnext | e Y
5244 tabclose
5245 call assert_equal('1Y2A3>B', GetTabs())
5246 call ClearAutomcdAndCreateTabs()
5247 au TabClosedPre * tabnext | e Y
5248 tabclose 1
5249 call assert_equal('1Y2B3>C', GetTabs())
5250 tabonly
5251 call assert_equal('1>Y', GetTabs())
5252
5253 " Create new windows in TabClosedPre autocmd
5254 call ClearAutomcdAndCreateTabs()
5255 au TabClosedPre * split | e X| vsplit | e Y | split | e Z
zeertzjq67fe77d2025-04-20 10:21:18 +02005256 call assert_fails('tabclose', 'E242:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005257 call ClearAutomcdAndCreateTabs()
5258 au TabClosedPre * new X | new Y | new Z
zeertzjq67fe77d2025-04-20 10:21:18 +02005259 call assert_fails('tabclose 1', 'E242:')
Jim Zhou5606ca52025-03-13 21:58:25 +01005260
Jim Zhoubcf66e02025-03-16 20:24:57 +01005261 " Test directly closing the tab page with ':tabclose'
5262 au!
5263 tabonly
5264 bw!
5265 e Z
5266 au TabClosedPre * mksession!
5267 tabnew A
5268 sp
5269 tabclose
5270 source Session.vim
5271 call assert_equal('1Z2>AA', GetTabs())
5272
5273 " Test directly closing the tab page with ':tabonly'
5274 " Z is closed before A. Hence A overwrites the session.
5275 au!
5276 tabonly
5277 bw!
5278 e Z
5279 au TabClosedPre * mksession!
5280 tabnew A
5281 tabnew B
5282 tabonly
5283 source Session.vim
5284 call assert_equal('1>A2B', GetTabs())
5285
Jim Zhou5606ca52025-03-13 21:58:25 +01005286 " Clean up
Jim Zhoubcf66e02025-03-16 20:24:57 +01005287 call delete('Session.vim')
Jim Zhou5606ca52025-03-13 21:58:25 +01005288 au!
5289 only
5290 tabonly
5291 bw!
5292endfunc
5293
Bram Moolenaarbc2b71d2020-02-17 21:33:30 +01005294" vim: shiftwidth=2 sts=2 expandtab