mirror of
https://github.com/neovim/neovim.git
synced 2025-09-18 17:28:23 +00:00
Merge pull request #17432 from zeertzjq/vim-8.1.2336
vim-patch:8.1.2336,8.2.{4338,4401}: mapping cursor and redrawing patches
This commit is contained in:
@@ -1949,8 +1949,11 @@ static int handle_mapping(int *keylenp, bool *timedout, int *mapdepth)
|
||||
// expression. Also save and restore the command line
|
||||
// for "normal :".
|
||||
if (mp->m_expr) {
|
||||
int save_vgetc_busy = vgetc_busy;
|
||||
const int save_vgetc_busy = vgetc_busy;
|
||||
const bool save_may_garbage_collect = may_garbage_collect;
|
||||
const int save_cursor_row = ui_current_row();
|
||||
const int save_cursor_col = ui_current_col();
|
||||
const int prev_did_emsg = did_emsg;
|
||||
|
||||
vgetc_busy = 0;
|
||||
may_garbage_collect = false;
|
||||
@@ -1960,6 +1963,32 @@ static int handle_mapping(int *keylenp, bool *timedout, int *mapdepth)
|
||||
save_m_str = vim_strsave(mp->m_str);
|
||||
}
|
||||
map_str = eval_map_expr(mp, NUL);
|
||||
|
||||
// The mapping may do anything, but we expect it to take care of
|
||||
// redrawing. Do put the cursor back where it was.
|
||||
ui_cursor_goto(save_cursor_row, save_cursor_col);
|
||||
ui_flush();
|
||||
|
||||
// If an error was displayed and the expression returns an empty
|
||||
// string, generate a <Nop> to allow for a redraw.
|
||||
if (prev_did_emsg != did_emsg && (map_str == NULL || *map_str == NUL)) {
|
||||
char_u buf[4];
|
||||
xfree(map_str);
|
||||
buf[0] = K_SPECIAL;
|
||||
buf[1] = KS_EXTRA;
|
||||
buf[2] = KE_IGNORE;
|
||||
buf[3] = NUL;
|
||||
map_str = vim_strsave(buf);
|
||||
if (State & CMDLINE) {
|
||||
// redraw the command below the error
|
||||
msg_didout = true;
|
||||
if (msg_row < cmdline_row) {
|
||||
msg_row = cmdline_row;
|
||||
}
|
||||
redrawcmd();
|
||||
}
|
||||
}
|
||||
|
||||
vgetc_busy = save_vgetc_busy;
|
||||
may_garbage_collect = save_may_garbage_collect;
|
||||
} else {
|
||||
@@ -3490,6 +3519,7 @@ static void showmap(mapblock_T *mp, bool local)
|
||||
if (p_verbose > 0) {
|
||||
last_set_msg(mp->m_script_ctx);
|
||||
}
|
||||
msg_clr_eos();
|
||||
ui_flush(); // show one line at a time
|
||||
}
|
||||
|
||||
|
@@ -1,6 +1,8 @@
|
||||
" Tests for mappings and abbreviations
|
||||
|
||||
source shared.vim
|
||||
source check.vim
|
||||
source screendump.vim
|
||||
|
||||
func Test_abbreviation()
|
||||
" abbreviation with 0x80 should work
|
||||
@@ -451,6 +453,82 @@ func Test_expr_map_gets_cursor()
|
||||
nunmap !
|
||||
endfunc
|
||||
|
||||
func Test_expr_map_restore_cursor()
|
||||
CheckScreendump
|
||||
|
||||
let lines =<< trim END
|
||||
call setline(1, ['one', 'two', 'three'])
|
||||
2
|
||||
set ls=2
|
||||
hi! link StatusLine ErrorMsg
|
||||
noremap <expr> <C-B> Func()
|
||||
func Func()
|
||||
let g:on = !get(g:, 'on', 0)
|
||||
redraws
|
||||
return ''
|
||||
endfunc
|
||||
func Status()
|
||||
return get(g:, 'on', 0) ? '[on]' : ''
|
||||
endfunc
|
||||
set stl=%{Status()}
|
||||
END
|
||||
call writefile(lines, 'XtestExprMap')
|
||||
let buf = RunVimInTerminal('-S XtestExprMap', #{rows: 10})
|
||||
call term_sendkeys(buf, "\<C-B>")
|
||||
call VerifyScreenDump(buf, 'Test_map_expr_1', {})
|
||||
|
||||
" clean up
|
||||
call StopVimInTerminal(buf)
|
||||
call delete('XtestExprMap')
|
||||
endfunc
|
||||
|
||||
func Test_map_listing()
|
||||
CheckScreendump
|
||||
|
||||
let lines =<< trim END
|
||||
nmap a b
|
||||
END
|
||||
call writefile(lines, 'XtestMapList')
|
||||
let buf = RunVimInTerminal('-S XtestMapList', #{rows: 6})
|
||||
call term_sendkeys(buf, ": nmap a\<CR>")
|
||||
call VerifyScreenDump(buf, 'Test_map_list_1', {})
|
||||
|
||||
" clean up
|
||||
call StopVimInTerminal(buf)
|
||||
call delete('XtestMapList')
|
||||
endfunc
|
||||
|
||||
func Test_expr_map_error()
|
||||
CheckScreendump
|
||||
|
||||
let lines =<< trim END
|
||||
func Func()
|
||||
throw 'test'
|
||||
return ''
|
||||
endfunc
|
||||
|
||||
nnoremap <expr> <F2> Func()
|
||||
cnoremap <expr> <F2> Func()
|
||||
|
||||
call test_override('ui_delay', 10)
|
||||
END
|
||||
call writefile(lines, 'XtestExprMap')
|
||||
let buf = RunVimInTerminal('-S XtestExprMap', #{rows: 10})
|
||||
call term_sendkeys(buf, "\<F2>")
|
||||
call TermWait(buf)
|
||||
call term_sendkeys(buf, "\<CR>")
|
||||
call VerifyScreenDump(buf, 'Test_map_expr_2', {})
|
||||
|
||||
call term_sendkeys(buf, ":abc\<F2>")
|
||||
call VerifyScreenDump(buf, 'Test_map_expr_3', {})
|
||||
call term_sendkeys(buf, "\<Esc>0")
|
||||
call VerifyScreenDump(buf, 'Test_map_expr_4', {})
|
||||
|
||||
" clean up
|
||||
call StopVimInTerminal(buf)
|
||||
call delete('XtestExprMap')
|
||||
endfunc
|
||||
|
||||
" Test for mapping errors
|
||||
func Test_map_error()
|
||||
call assert_fails('unmap', 'E474:')
|
||||
|
Reference in New Issue
Block a user