diff options
author | zeertzjq <zeertzjq@outlook.com> | 2023-04-27 00:23:39 +0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-04-27 00:23:39 +0800 |
commit | efae71819a99b1cf9cce024b520b62bb455229f9 (patch) | |
tree | 606619758ac55c4e28f7d8c52df7b33513bf64a9 | |
parent | 9f0762f1fec2aa23df592dda70124e3cbdb703b7 (diff) | |
parent | a35bca21125980d407bdb830e7c52d95a629de76 (diff) | |
download | rneovim-efae71819a99b1cf9cce024b520b62bb455229f9.tar.gz rneovim-efae71819a99b1cf9cce024b520b62bb455229f9.tar.bz2 rneovim-efae71819a99b1cf9cce024b520b62bb455229f9.zip |
Merge pull request #23288 from MunifTanjim/issue-22263
fix(normal): fix repeated trigger modechanged for scheduled callback
-rw-r--r-- | src/nvim/normal.c | 14 | ||||
-rw-r--r-- | test/functional/autocmd/modechanged_spec.lua | 39 | ||||
-rw-r--r-- | test/old/testdir/test_autocmd.vim | 4 |
3 files changed, 46 insertions, 11 deletions
diff --git a/src/nvim/normal.c b/src/nvim/normal.c index 2bafeccba4..635d177ecc 100644 --- a/src/nvim/normal.c +++ b/src/nvim/normal.c @@ -956,10 +956,12 @@ normal_end: set_reg_var(get_default_register_name()); } - // Reset finish_op, in case it was set s->c = finish_op; - finish_op = false; - may_trigger_modechanged(); + if (s->oa.op_type == OP_NOP) { + // Reset finish_op, in case it was set + finish_op = false; + may_trigger_modechanged(); + } // Redraw the cursor with another shape, if we were in Operator-pending // mode or did a replace command. if (s->c || s->ca.cmdchar == 'r' @@ -1729,9 +1731,9 @@ void prep_redo_num2(int regname, long num1, int cmd1, int cmd2, long num2, int c } } -/// check for operator active and clear it +/// Check for operator active and clear it. /// -/// @return true if operator was active +/// Beep and return true if an operator was active. static bool checkclearop(oparg_T *oap) { if (oap->op_type == OP_NOP) { @@ -1743,7 +1745,7 @@ static bool checkclearop(oparg_T *oap) /// Check for operator or Visual active. Clear active operator. /// -/// @return true if operator or Visual was active. +/// Beep and return true if an operator or Visual was active. static bool checkclearopq(oparg_T *oap) { if (oap->op_type == OP_NOP && !VIsual_active) { diff --git a/test/functional/autocmd/modechanged_spec.lua b/test/functional/autocmd/modechanged_spec.lua index be5a291ac9..69a722a0e9 100644 --- a/test/functional/autocmd/modechanged_spec.lua +++ b/test/functional/autocmd/modechanged_spec.lua @@ -1,17 +1,19 @@ local helpers = require('test.functional.helpers')(after_each) local clear, eval, eq = helpers.clear, helpers.eval, helpers.eq local feed, command = helpers.feed, helpers.command +local exec_lua = helpers.exec_lua describe('ModeChanged', function() before_each(function() clear() + end) + + it('picks up terminal mode changes', function() command('let g:count = 0') command('au ModeChanged * let g:event = copy(v:event)') command('au ModeChanged * let g:count += 1') - end) - it('picks up terminal mode changes', function() - command("term") + command('term') feed('i') eq({ old_mode = 'nt', @@ -28,4 +30,35 @@ describe('ModeChanged', function() -- v:event is cleared after the autocommand is done eq({}, eval('v:event')) end) + + it('does not repeatedly trigger for scheduled callback', function() + exec_lua([[ + vim.g.s_count = 0 + vim.g.s_mode = "" + vim.g.t_count = 0 + vim.g.t_mode = "" + vim.api.nvim_create_autocmd("ModeChanged", { + callback = function() + vim.g.s_count = vim.g.s_count + 1 + vim.g.s_mode = vim.api.nvim_get_mode().mode + vim.schedule(function() + vim.g.t_count = vim.g.t_count + 1 + vim.g.t_mode = vim.api.nvim_get_mode().mode + end) + end, + }) + ]]) + + feed('d') + eq(1, eval('g:s_count')) + eq('no', eval('g:s_mode')) + eq(1, eval('g:t_count')) + eq('no', eval('g:t_mode')) + + feed('<Esc>') + eq(2, eval('g:s_count')) + eq('n', eval('g:s_mode')) + eq(2, eval('g:t_count')) + eq('n', eval('g:t_mode')) + end) end) diff --git a/test/old/testdir/test_autocmd.vim b/test/old/testdir/test_autocmd.vim index decfec4763..c44988321f 100644 --- a/test/old/testdir/test_autocmd.vim +++ b/test/old/testdir/test_autocmd.vim @@ -3453,7 +3453,7 @@ endfunc " Test for ModeChanged pattern func Test_mode_changes() let g:index = 0 - let g:mode_seq = ['n', 'i', 'n', 'v', 'V', 'i', 'ix', 'i', 'ic', 'i', 'n', 'no', 'n', 'V', 'v', 's', 'n'] + let g:mode_seq = ['n', 'i', 'n', 'v', 'V', 'i', 'ix', 'i', 'ic', 'i', 'n', 'no', 'noV', 'n', 'V', 'v', 's', 'n'] func! TestMode() call assert_equal(g:mode_seq[g:index], get(v:event, "old_mode")) call assert_equal(g:mode_seq[g:index + 1], get(v:event, "new_mode")) @@ -3464,7 +3464,7 @@ func Test_mode_changes() au ModeChanged * :call TestMode() let g:n_to_any = 0 au ModeChanged n:* let g:n_to_any += 1 - call feedkeys("i\<esc>vVca\<CR>\<C-X>\<C-L>\<esc>ggdG", 'tnix') + call feedkeys("i\<esc>vVca\<CR>\<C-X>\<C-L>\<esc>ggdV\<MouseMove>G", 'tnix') let g:V_to_v = 0 au ModeChanged V:v let g:V_to_v += 1 |