mirror of
https://github.com/neovim/neovim.git
synced 2025-02-25 18:55:25 -06:00
vim-patch:9.0.2031: TextChangedI may be triggered by non-insert mode change (#25656)
Problem: `TextChangedI` can trigger on entering Insert mode if there
was previously a change not in Insert mode.
Solution: Make it trigger only when text is actually changed in Insert
mode.
closes: vim/vim#13265
closes: vim/vim#13338
d7ae263af8
Co-authored-by: Evgeni Chasnovski <evgeni.chasnovski@gmail.com>
This commit is contained in:
parent
e5a424d78e
commit
a350fb2976
@ -143,6 +143,9 @@ static void insert_enter(InsertState *s)
|
|||||||
update_Insstart_orig = true;
|
update_Insstart_orig = true;
|
||||||
|
|
||||||
ins_compl_clear(); // clear stuff for CTRL-X mode
|
ins_compl_clear(); // clear stuff for CTRL-X mode
|
||||||
|
// Reset Changedtick_i, so that TextChangedI will only be triggered for stuff
|
||||||
|
// from insert mode
|
||||||
|
curbuf->b_last_changedtick_i = buf_get_changedtick(curbuf);
|
||||||
|
|
||||||
// Trigger InsertEnter autocommands. Do not do this for "r<CR>" or "grx".
|
// Trigger InsertEnter autocommands. Do not do this for "r<CR>" or "grx".
|
||||||
if (s->cmdchar != 'r' && s->cmdchar != 'v') {
|
if (s->cmdchar != 'r' && s->cmdchar != 'v') {
|
||||||
@ -356,6 +359,7 @@ static void insert_enter(InsertState *s)
|
|||||||
ins_apply_autocmds(EVENT_INSERTLEAVE);
|
ins_apply_autocmds(EVENT_INSERTLEAVE);
|
||||||
}
|
}
|
||||||
did_cursorhold = false;
|
did_cursorhold = false;
|
||||||
|
curbuf->b_last_changedtick = buf_get_changedtick(curbuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int insert_check(VimState *state)
|
static int insert_check(VimState *state)
|
||||||
|
@ -25,14 +25,15 @@ it('TextChangedI and TextChangedP autocommands', function()
|
|||||||
feed('o')
|
feed('o')
|
||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
|
-- TextChangedI triggers only if text is actually changed in Insert mode
|
||||||
|
eq('', eval('g:autocmd'))
|
||||||
|
|
||||||
|
command([[let g:autocmd = '']])
|
||||||
|
feed('S')
|
||||||
|
poke_eventloop()
|
||||||
|
feed('f')
|
||||||
|
poke_eventloop()
|
||||||
eq('I', eval('g:autocmd'))
|
eq('I', eval('g:autocmd'))
|
||||||
|
|
||||||
command([[let g:autocmd = '']])
|
|
||||||
feed('S')
|
|
||||||
poke_eventloop()
|
|
||||||
feed('f')
|
|
||||||
poke_eventloop()
|
|
||||||
eq('II', eval('g:autocmd'))
|
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
|
|
||||||
command([[let g:autocmd = '']])
|
command([[let g:autocmd = '']])
|
||||||
@ -42,7 +43,7 @@ it('TextChangedI and TextChangedP autocommands', function()
|
|||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
feed('<C-N>')
|
feed('<C-N>')
|
||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
eq('IIP', eval('g:autocmd'))
|
eq('IP', eval('g:autocmd'))
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
|
|
||||||
command([[let g:autocmd = '']])
|
command([[let g:autocmd = '']])
|
||||||
@ -54,7 +55,7 @@ it('TextChangedI and TextChangedP autocommands', function()
|
|||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
feed('<C-N>')
|
feed('<C-N>')
|
||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
eq('IIPP', eval('g:autocmd'))
|
eq('IPP', eval('g:autocmd'))
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
|
|
||||||
command([[let g:autocmd = '']])
|
command([[let g:autocmd = '']])
|
||||||
@ -68,7 +69,7 @@ it('TextChangedI and TextChangedP autocommands', function()
|
|||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
feed('<C-N>')
|
feed('<C-N>')
|
||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
eq('IIPPP', eval('g:autocmd'))
|
eq('IPPP', eval('g:autocmd'))
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
|
|
||||||
command([[let g:autocmd = '']])
|
command([[let g:autocmd = '']])
|
||||||
@ -83,7 +84,7 @@ it('TextChangedI and TextChangedP autocommands', function()
|
|||||||
feed('<C-N>')
|
feed('<C-N>')
|
||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
feed('<C-N>')
|
feed('<C-N>')
|
||||||
eq('IIPPPP', eval('g:autocmd'))
|
eq('IPPPP', eval('g:autocmd'))
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
|
|
||||||
eq({'foo', 'bar', 'foobar', 'foo'}, eval('getline(1, "$")'))
|
eq({'foo', 'bar', 'foobar', 'foo'}, eval('getline(1, "$")'))
|
||||||
@ -117,7 +118,7 @@ it('TextChangedI with setline()', function()
|
|||||||
end)
|
end)
|
||||||
|
|
||||||
-- oldtest: Test_Changed_ChangedI()
|
-- oldtest: Test_Changed_ChangedI()
|
||||||
it('TextChanged is triggerd after TextChangedI', function()
|
it('TextChangedI and TextChanged', function()
|
||||||
exec([[
|
exec([[
|
||||||
let [g:autocmd_i, g:autocmd_n] = ['','']
|
let [g:autocmd_i, g:autocmd_n] = ['','']
|
||||||
|
|
||||||
@ -141,10 +142,20 @@ it('TextChanged is triggerd after TextChangedI', function()
|
|||||||
feed('o')
|
feed('o')
|
||||||
poke_eventloop()
|
poke_eventloop()
|
||||||
feed('<esc>')
|
feed('<esc>')
|
||||||
eq('N5', eval('g:autocmd_n'))
|
eq('', eval('g:autocmd_n'))
|
||||||
eq('I5', eval('g:autocmd_i'))
|
eq('I5', eval('g:autocmd_i'))
|
||||||
|
|
||||||
command([[call feedkeys("yyp", 'tnix')]])
|
command([[call feedkeys("yyp", 'tnix')]])
|
||||||
eq('N6', eval('g:autocmd_n'))
|
eq('N6', eval('g:autocmd_n'))
|
||||||
eq('I5', eval('g:autocmd_i'))
|
eq('I5', eval('g:autocmd_i'))
|
||||||
|
|
||||||
|
-- TextChangedI should only trigger if change was done in Insert mode
|
||||||
|
command([[let g:autocmd_i = '']])
|
||||||
|
command([[call feedkeys("yypi\<esc>", 'tnix')]])
|
||||||
|
eq('', eval('g:autocmd_i'))
|
||||||
|
|
||||||
|
-- TextChanged should only trigger if change was done in Normal mode
|
||||||
|
command([[let g:autocmd_n = '']])
|
||||||
|
command([[call feedkeys("ibar\<esc>", 'tnix')]])
|
||||||
|
eq('', eval('g:autocmd_n'))
|
||||||
end)
|
end)
|
||||||
|
@ -2476,27 +2476,28 @@ func Test_ChangedP()
|
|||||||
call cursor(3, 1)
|
call cursor(3, 1)
|
||||||
let g:autocmd = ''
|
let g:autocmd = ''
|
||||||
call feedkeys("o\<esc>", 'tnix')
|
call feedkeys("o\<esc>", 'tnix')
|
||||||
call assert_equal('I', g:autocmd)
|
" `TextChangedI` triggers only if text is actually changed in Insert mode
|
||||||
|
call assert_equal('', g:autocmd)
|
||||||
|
|
||||||
let g:autocmd = ''
|
let g:autocmd = ''
|
||||||
call feedkeys("Sf", 'tnix')
|
call feedkeys("Sf", 'tnix')
|
||||||
call assert_equal('II', g:autocmd)
|
call assert_equal('I', g:autocmd)
|
||||||
|
|
||||||
let g:autocmd = ''
|
let g:autocmd = ''
|
||||||
call feedkeys("Sf\<C-N>", 'tnix')
|
call feedkeys("Sf\<C-N>", 'tnix')
|
||||||
call assert_equal('IIP', g:autocmd)
|
call assert_equal('IP', g:autocmd)
|
||||||
|
|
||||||
let g:autocmd = ''
|
let g:autocmd = ''
|
||||||
call feedkeys("Sf\<C-N>\<C-N>", 'tnix')
|
call feedkeys("Sf\<C-N>\<C-N>", 'tnix')
|
||||||
call assert_equal('IIPP', g:autocmd)
|
call assert_equal('IPP', g:autocmd)
|
||||||
|
|
||||||
let g:autocmd = ''
|
let g:autocmd = ''
|
||||||
call feedkeys("Sf\<C-N>\<C-N>\<C-N>", 'tnix')
|
call feedkeys("Sf\<C-N>\<C-N>\<C-N>", 'tnix')
|
||||||
call assert_equal('IIPPP', g:autocmd)
|
call assert_equal('IPPP', g:autocmd)
|
||||||
|
|
||||||
let g:autocmd = ''
|
let g:autocmd = ''
|
||||||
call feedkeys("Sf\<C-N>\<C-N>\<C-N>\<C-N>", 'tnix')
|
call feedkeys("Sf\<C-N>\<C-N>\<C-N>\<C-N>", 'tnix')
|
||||||
call assert_equal('IIPPPP', g:autocmd)
|
call assert_equal('IPPPP', g:autocmd)
|
||||||
|
|
||||||
call assert_equal(['foo', 'bar', 'foobar', 'foo'], getline(1, '$'))
|
call assert_equal(['foo', 'bar', 'foobar', 'foo'], getline(1, '$'))
|
||||||
" TODO: how should it handle completeopt=noinsert,noselect?
|
" TODO: how should it handle completeopt=noinsert,noselect?
|
||||||
@ -3478,6 +3479,16 @@ func Test_Changed_ChangedI()
|
|||||||
" call assert_equal('N4', g:autocmd_n)
|
" call assert_equal('N4', g:autocmd_n)
|
||||||
call assert_equal('I3', g:autocmd_i)
|
call assert_equal('I3', g:autocmd_i)
|
||||||
|
|
||||||
|
" TextChangedI should only trigger if change was done in Insert mode
|
||||||
|
let g:autocmd_i = ''
|
||||||
|
call feedkeys("yypi\<esc>", 'tnix')
|
||||||
|
call assert_equal('', g:autocmd_i)
|
||||||
|
|
||||||
|
" TextChanged should only trigger if change was done in Normal mode
|
||||||
|
let g:autocmd_n = ''
|
||||||
|
call feedkeys("ibar\<esc>", 'tnix')
|
||||||
|
call assert_equal('', g:autocmd_n)
|
||||||
|
|
||||||
" CleanUp
|
" CleanUp
|
||||||
call test_override("char_avail", 0)
|
call test_override("char_avail", 0)
|
||||||
au! TextChanged <buffer>
|
au! TextChanged <buffer>
|
||||||
|
Loading…
Reference in New Issue
Block a user