summaryrefslogtreecommitdiffstats
path: root/src/ex_docmd.c
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2022-03-24 13:08:36 +0000
committerBram Moolenaar <Bram@vim.org>2022-03-24 13:08:36 +0000
commitf4f0525c34d2aa32f214155b0dadcd274ed05dd1 (patch)
treea4665cbfd5ab6d4ee44417ac49c8bf267c083f42 /src/ex_docmd.c
parent454ce6737cadb82886f1fc0eb9e8666cc59ae42b (diff)
patch 8.2.4618: cmdline completion does not recognize single letter commandsv8.2.4618
Problem: Command line completion does not recognize single letter commands. Solution: Use the condition from find_ex_command().
Diffstat (limited to 'src/ex_docmd.c')
-rw-r--r--src/ex_docmd.c65
1 files changed, 40 insertions, 25 deletions
diff --git a/src/ex_docmd.c b/src/ex_docmd.c
index a35924a570..7c2fe3deb7 100644
--- a/src/ex_docmd.c
+++ b/src/ex_docmd.c
@@ -3421,6 +3421,38 @@ skip_option_env_lead(char_u *start)
#endif
/*
+ * Return TRUE and set "*idx" if "p" points to a one letter command.
+ * If not in Vim9 script:
+ * - The 'k' command can directly be followed by any character.
+ * - The 's' command can be followed directly by 'c', 'g', 'i', 'I' or 'r'
+ * but :sre[wind] is another command, as are :scr[iptnames],
+ * :scs[cope], :sim[alt], :sig[ns] and :sil[ent].
+ */
+ static int
+one_letter_cmd(char_u *p, cmdidx_T *idx)
+{
+ if (!in_vim9script())
+ return FALSE;
+ if (*p == 'k')
+ {
+ *idx = CMD_k;
+ return TRUE;
+ }
+ if (p[0] == 's'
+ && ((p[1] == 'c' && (p[2] == NUL || (p[2] != 's' && p[2] != 'r'
+ && (p[3] == NUL || (p[3] != 'i' && p[4] != 'p')))))
+ || p[1] == 'g'
+ || (p[1] == 'i' && p[2] != 'm' && p[2] != 'l' && p[2] != 'g')
+ || p[1] == 'I'
+ || (p[1] == 'r' && p[2] != 'e')))
+ {
+ *idx = CMD_substitute;
+ return TRUE;
+ }
+ return FALSE;
+}
+
+/*
* Find an Ex command by its name, either built-in or user.
* Start of the name can be found at eap->cmd.
* Sets eap->cmdidx and returns a pointer to char after the command name.
@@ -3654,30 +3686,10 @@ find_ex_command(
/*
* Isolate the command and search for it in the command table.
- * Exceptions:
- * - The 'k' command can directly be followed by any character.
- * But it is not used in Vim9 script.
- * - the 's' command can be followed directly by 'c', 'g', 'i', 'I' or 'r'
- * but :sre[wind] is another command, as are :scr[iptnames],
- * :scs[cope], :sim[alt], :sig[ns] and :sil[ent].
- * - the "d" command can directly be followed by 'l' or 'p' flag.
*/
p = eap->cmd;
- if (!vim9 && *p == 'k')
+ if (one_letter_cmd(p, &eap->cmdidx))
{
- eap->cmdidx = CMD_k;
- ++p;
- }
- else if (!vim9
- && p[0] == 's'
- && ((p[1] == 'c' && (p[2] == NUL || (p[2] != 's' && p[2] != 'r'
- && (p[3] == NUL || (p[3] != 'i' && p[4] != 'p')))))
- || p[1] == 'g'
- || (p[1] == 'i' && p[2] != 'm' && p[2] != 'l' && p[2] != 'g')
- || p[1] == 'I'
- || (p[1] == 'r' && p[2] != 'e')))
- {
- eap->cmdidx = CMD_substitute;
++p;
}
else
@@ -3702,6 +3714,8 @@ find_ex_command(
if (p == eap->cmd && vim_strchr((char_u *)"@*!=><&~#}", *p) != NULL)
++p;
len = (int)(p - eap->cmd);
+ // The "d" command can directly be followed by 'l' or 'p' flag, when
+ // not in Vim9 script.
if (!vim9 && *eap->cmd == 'd' && (p[-1] == 'l' || p[-1] == 'p'))
{
// Check for ":dl", ":dell", etc. to ":deletel": that's
@@ -3955,10 +3969,11 @@ excmd_get_cmdidx(char_u *cmd, int len)
{
cmdidx_T idx;
- for (idx = (cmdidx_T)0; (int)idx < (int)CMD_SIZE;
- idx = (cmdidx_T)((int)idx + 1))
- if (STRNCMP(cmdnames[(int)idx].cmd_name, cmd, (size_t)len) == 0)
- break;
+ if (!one_letter_cmd(cmd, &idx))
+ for (idx = (cmdidx_T)0; (int)idx < (int)CMD_SIZE;
+ idx = (cmdidx_T)((int)idx + 1))
+ if (STRNCMP(cmdnames[(int)idx].cmd_name, cmd, (size_t)len) == 0)
+ break;
return idx;
}