diff options
author | zeertzjq <zeertzjq@outlook.com> | 2025-03-04 06:41:41 +0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2025-03-04 06:41:41 +0800 |
commit | e4fb697b69c7307a76d06cf60bb834af2e2cdc5e (patch) | |
tree | c3d4b89e4fddbd39a010cebb65440b6b33b8b93d | |
parent | 948179cb19c75a9e79cdf2c86c441304c5285e81 (diff) | |
parent | e56f7413505c76b7b57ced2552479ad02875bc74 (diff) | |
download | rneovim-e4fb697b69c7307a76d06cf60bb834af2e2cdc5e.tar.gz rneovim-e4fb697b69c7307a76d06cf60bb834af2e2cdc5e.tar.bz2 rneovim-e4fb697b69c7307a76d06cf60bb834af2e2cdc5e.zip |
Merge pull request #32709 from deathbeam/vim-9.1.1166
vim-patch:9.1.{1166,1168}: 'wildmode' noselect
-rw-r--r-- | runtime/doc/news.txt | 1 | ||||
-rw-r--r-- | runtime/doc/options.txt | 11 | ||||
-rw-r--r-- | runtime/lua/vim/_meta/options.lua | 17 | ||||
-rw-r--r-- | src/nvim/cmdexpand.c | 4 | ||||
-rw-r--r-- | src/nvim/cmdexpand.h | 1 | ||||
-rw-r--r-- | src/nvim/ex_getln.c | 15 | ||||
-rw-r--r-- | src/nvim/options.lua | 13 | ||||
-rw-r--r-- | test/old/testdir/gen_opt_test.vim | 1 | ||||
-rw-r--r-- | test/old/testdir/test_cmdline.vim | 61 |
9 files changed, 105 insertions, 19 deletions
diff --git a/runtime/doc/news.txt b/runtime/doc/news.txt index a131934a8e..415a234254 100644 --- a/runtime/doc/news.txt +++ b/runtime/doc/news.txt @@ -329,6 +329,7 @@ OPTIONS • 'completeopt' flag "fuzzy" enables |fuzzy-matching| during |ins-completion|. • 'completeopt' flag "preinsert" highlights text to be inserted. +• 'wildmode' flag "noselect" shows 'wildmenu' without selecting an entry. • 'messagesopt' configures |:messages| and |hit-enter| prompt. • 'tabclose' controls which tab page to focus when closing a tab page. • 'eventignorewin' to persistently ignore events in a window. diff --git a/runtime/doc/options.txt b/runtime/doc/options.txt index a6f5f85d37..a3cef8bacb 100644 --- a/runtime/doc/options.txt +++ b/runtime/doc/options.txt @@ -7050,7 +7050,10 @@ A jump table for the options with a short description can be found at |Q_op|. "lastused" When completing buffer names and more than one buffer matches, sort buffers by time last used (other than the current buffer). - When there is only a single match, it is fully completed in all cases. + "noselect" Do not pre-select first menu item and start 'wildmenu' + if it is enabled. + When there is only a single match, it is fully completed in all cases + except when "noselect" is present. Examples of useful colon-separated values: "longest:full" Like "longest", but also start 'wildmenu' if it is @@ -7073,7 +7076,11 @@ A jump table for the options with a short description can be found at |Q_op|. set wildmode=list,full < List all matches without completing, then each full match >vim set wildmode=longest,list -< Complete longest common string, then list alternatives. +< Complete longest common string, then list alternatives >vim + set wildmode=noselect:full +< Display 'wildmenu' without completing, then each full match >vim + set wildmode=noselect:lastused,full +< Same as above, but sort buffers by time last used. More info here: |cmdline-completion|. *'wildoptions'* *'wop'* diff --git a/runtime/lua/vim/_meta/options.lua b/runtime/lua/vim/_meta/options.lua index 8df174a838..8e6acecc28 100644 --- a/runtime/lua/vim/_meta/options.lua +++ b/runtime/lua/vim/_meta/options.lua @@ -7691,7 +7691,10 @@ vim.go.wmnu = vim.go.wildmenu --- "lastused" When completing buffer names and more than one buffer --- matches, sort buffers by time last used (other than --- the current buffer). ---- When there is only a single match, it is fully completed in all cases. +--- "noselect" Do not pre-select first menu item and start 'wildmenu' +--- if it is enabled. +--- When there is only a single match, it is fully completed in all cases +--- except when "noselect" is present. --- --- Examples of useful colon-separated values: --- "longest:full" Like "longest", but also start 'wildmenu' if it is @@ -7729,7 +7732,17 @@ vim.go.wmnu = vim.go.wildmenu --- ```vim --- set wildmode=longest,list --- ``` ---- Complete longest common string, then list alternatives. +--- Complete longest common string, then list alternatives +--- +--- ```vim +--- set wildmode=noselect:full +--- ``` +--- Display 'wildmenu' without completing, then each full match +--- +--- ```vim +--- set wildmode=noselect:lastused,full +--- ``` +--- Same as above, but sort buffers by time last used. --- More info here: `cmdline-completion`. --- --- @type string diff --git a/src/nvim/cmdexpand.c b/src/nvim/cmdexpand.c index 33ccc834d2..fcfeda5482 100644 --- a/src/nvim/cmdexpand.c +++ b/src/nvim/cmdexpand.c @@ -288,7 +288,7 @@ int nextwild(expand_T *xp, int type, int options, bool escape) p1 = addstar(xp->xp_pattern, xp->xp_pattern_len, xp->xp_context); } // Translate string into pattern and expand it. - const int use_options = (options + const int use_options = ((options & ~WILD_KEEP_SOLE_ITEM) | WILD_HOME_REPLACE | WILD_ADD_SLASH | WILD_SILENT @@ -339,7 +339,7 @@ int nextwild(expand_T *xp, int type, int options, bool escape) if (xp->xp_numfiles <= 0 && p2 == NULL) { beep_flush(); - } else if (xp->xp_numfiles == 1) { + } else if (xp->xp_numfiles == 1 && !(options & WILD_KEEP_SOLE_ITEM)) { // free expanded pattern ExpandOne(xp, NULL, NULL, 0, WILD_FREE); } diff --git a/src/nvim/cmdexpand.h b/src/nvim/cmdexpand.h index 33ff787589..59c734e9a3 100644 --- a/src/nvim/cmdexpand.h +++ b/src/nvim/cmdexpand.h @@ -40,6 +40,7 @@ enum { WILD_NOERROR = 0x800, ///< sets EW_NOERROR WILD_BUFLASTUSED = 0x1000, BUF_DIFF_FILTER = 0x2000, + WILD_KEEP_SOLE_ITEM = 0x4000, }; #ifdef INCLUDE_GENERATED_DECLARATIONS diff --git a/src/nvim/ex_getln.c b/src/nvim/ex_getln.c index fc20748309..240bdae6cb 100644 --- a/src/nvim/ex_getln.c +++ b/src/nvim/ex_getln.c @@ -1082,6 +1082,9 @@ static int command_line_wildchar_complete(CommandLineState *s) if (wim_flags[s->wim_index] & kOptWimFlagLastused) { options |= WILD_BUFLASTUSED; } + if (wim_flags[0] & kOptWimFlagNoselect) { + options |= WILD_KEEP_SOLE_ITEM; + } if (s->xpc.xp_numfiles > 0) { // typed p_wc at least twice // if 'wildmode' contains "list" may still need to list if (s->xpc.xp_numfiles > 1 @@ -1124,19 +1127,20 @@ static int command_line_wildchar_complete(CommandLineState *s) // when more than one match, and 'wildmode' first contains // "list", or no change and 'wildmode' contains "longest,list", // list all matches - if (res == OK && s->xpc.xp_numfiles > 1) { + if (res == OK + && s->xpc.xp_numfiles > ((wim_flags[s->wim_index] & kOptWimFlagNoselect) ? 0 : 1)) { // a "longest" that didn't do anything is skipped (but not // "list:longest") if (wim_flags[0] == kOptWimFlagLongest && ccline.cmdpos == j) { s->wim_index = 1; } if ((wim_flags[s->wim_index] & kOptWimFlagList) - || (p_wmnu && (wim_flags[s->wim_index] & kOptWimFlagFull) != 0)) { + || (p_wmnu && (wim_flags[s->wim_index] & (kOptWimFlagFull|kOptWimFlagNoselect)))) { if (!(wim_flags[0] & kOptWimFlagLongest)) { int p_wmnu_save = p_wmnu; p_wmnu = 0; // remove match - nextwild(&s->xpc, WILD_PREV, 0, s->firstc != '@'); + nextwild(&s->xpc, WILD_PREV, options, s->firstc != '@'); p_wmnu = p_wmnu_save; } @@ -1146,7 +1150,8 @@ static int command_line_wildchar_complete(CommandLineState *s) if (wim_flags[s->wim_index] & kOptWimFlagLongest) { nextwild(&s->xpc, WILD_LONGEST, options, s->firstc != '@'); - } else if (wim_flags[s->wim_index] & kOptWimFlagFull) { + } else if ((wim_flags[s->wim_index] & kOptWimFlagFull) + && !(wim_flags[s->wim_index] & kOptWimFlagNoselect)) { nextwild(&s->xpc, WILD_NEXT, options, s->firstc != '@'); } } else { @@ -2875,6 +2880,8 @@ int check_opt_wim(void) new_wim_flags[idx] |= kOptWimFlagList; } else if (i == 8 && strncmp(p, "lastused", 8) == 0) { new_wim_flags[idx] |= kOptWimFlagLastused; + } else if (i == 8 && strncmp(p, "noselect", 8) == 0) { + new_wim_flags[idx] |= kOptWimFlagNoselect; } else { return FAIL; } diff --git a/src/nvim/options.lua b/src/nvim/options.lua index d1d5a56082..2516d1a970 100644 --- a/src/nvim/options.lua +++ b/src/nvim/options.lua @@ -10020,7 +10020,7 @@ local options = { cb = 'did_set_wildmode', defaults = 'full', -- Keep this in sync with check_opt_wim(). - values = { 'full', 'longest', 'list', 'lastused' }, + values = { 'full', 'longest', 'list', 'lastused', 'noselect' }, flags = true, deny_duplicates = false, desc = [=[ @@ -10042,7 +10042,10 @@ local options = { "lastused" When completing buffer names and more than one buffer matches, sort buffers by time last used (other than the current buffer). - When there is only a single match, it is fully completed in all cases. + "noselect" Do not pre-select first menu item and start 'wildmenu' + if it is enabled. + When there is only a single match, it is fully completed in all cases + except when "noselect" is present. Examples of useful colon-separated values: "longest:full" Like "longest", but also start 'wildmenu' if it is @@ -10065,7 +10068,11 @@ local options = { set wildmode=list,full < List all matches without completing, then each full match >vim set wildmode=longest,list - < Complete longest common string, then list alternatives. + < Complete longest common string, then list alternatives >vim + set wildmode=noselect:full + < Display 'wildmenu' without completing, then each full match >vim + set wildmode=noselect:lastused,full + < Same as above, but sort buffers by time last used. More info here: |cmdline-completion|. ]=], full_name = 'wildmode', diff --git a/test/old/testdir/gen_opt_test.vim b/test/old/testdir/gen_opt_test.vim index 64876ce318..d1d27cf578 100644 --- a/test/old/testdir/gen_opt_test.vim +++ b/test/old/testdir/gen_opt_test.vim @@ -352,6 +352,7 @@ let test_values = { \ 'bs'], \ ['xxx']], \ 'wildmode': [['', 'full', 'longest', 'list', 'lastused', 'list:full', + \ 'noselect', 'noselect,full', 'noselect:lastused,full', \ 'full,longest', 'full,full,full,full'], \ ['xxx', 'a4', 'full,full,full,full,full']], \ 'wildoptions': [['', 'tagfile', 'pum', 'fuzzy'], ['xxx']], diff --git a/test/old/testdir/test_cmdline.vim b/test/old/testdir/test_cmdline.vim index d4ad63d43e..d9496ba2a6 100644 --- a/test/old/testdir/test_cmdline.vim +++ b/test/old/testdir/test_cmdline.vim @@ -2168,22 +2168,58 @@ func Wildmode_tests() call assert_equal('AAA AAAA AAAAA', g:Sline) call assert_equal('"b A', @:) + " When 'wildmenu' is not set, 'noselect' completes first item + set wildmode=noselect + call feedkeys(":MyCmd o\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd oneA', @:) + + " When 'noselect' is present, do not complete first <tab>. + set wildmenu + set wildmode=noselect + call feedkeys(":MyCmd o\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd o', @:) + call feedkeys(":MyCmd o\t\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd o', @:) + call feedkeys(":MyCmd o\t\t\<C-Y>\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd o', @:) + + " When 'full' is present, complete after first <tab>. + set wildmode=noselect,full + call feedkeys(":MyCmd o\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd o', @:) + call feedkeys(":MyCmd o\t\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd oneA', @:) + call feedkeys(":MyCmd o\t\t\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd oneB', @:) + call feedkeys(":MyCmd o\t\t\t\<C-Y>\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd oneB', @:) + + " 'noselect' has no effect when 'longest' is present. + set wildmode=noselect:longest + call feedkeys(":MyCmd o\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd one', @:) + + " Complete 'noselect' value in 'wildmode' option + set wildmode& + call feedkeys(":set wildmode=n\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"set wildmode=noselect', @:) + call feedkeys(":set wildmode=\t\t\t\t\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"set wildmode=noselect', @:) + " when using longest completion match, matches shorter than the argument " should be ignored (happens with :help) set wildmode=longest,full - set wildmenu call feedkeys(":help a*\t\<C-B>\"\<CR>", 'xt') call assert_equal('"help a', @:) " non existing file call feedkeys(":e a1b2y3z4\t\<C-B>\"\<CR>", 'xt') call assert_equal('"e a1b2y3z4', @:) - set wildmenu& " Test for longest file name completion with 'fileignorecase' " On MS-Windows, file names are case insensitive. if has('unix') - call writefile([], 'XTESTfoo') - call writefile([], 'Xtestbar') + call writefile([], 'XTESTfoo', 'D') + call writefile([], 'Xtestbar', 'D') set nofileignorecase call feedkeys(":e XT\<Tab>\<C-B>\"\<CR>", 'xt') call assert_equal('"e XTESTfoo', @:) @@ -2195,10 +2231,23 @@ func Wildmode_tests() call feedkeys(":e Xt\<Tab>\<C-B>\"\<CR>", 'xt') call assert_equal('"e Xtest', @:) set fileignorecase& - call delete('XTESTfoo') - call delete('Xtestbar') endif + " If 'noselect' is present, single item menu should not insert item + func! T(a, c, p) + return "oneA" + endfunc + command! -nargs=1 -complete=custom,T MyCmd + set wildmode=noselect,full + call feedkeys(":MyCmd o\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd o', @:) + call feedkeys(":MyCmd o\t\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd oneA', @:) + " 'nowildmenu' should make 'noselect' ineffective + set nowildmenu + call feedkeys(":MyCmd o\t\<C-B>\"\<CR>", 'xt') + call assert_equal('"MyCmd oneA', @:) + %argdelete delcommand MyCmd delfunc T |