*insert.txt* For Vim version 7.1. 最近更新: 2007年5月 VIM 参考手册 by Bram Moolenaar 译者: Willis,tocer http://vimcdoc.sf.net *Insert* *Insert-mode* 插入和替换文本 *mode-ins-repl* 本文件主要讨论插入和替换模式。最后讨论一些其它方式插入文本的命令。 最常用的命令的总览可以在用户手册第 24 章 |usr_24.txt| 找到。 1. 特殊键 |ins-special-keys| 2. 特殊的特殊键 |ins-special-special| 3. 'textwidth' 和 'wrapmargin' 选项 |ins-textwidth| 4. 'expandtab'、'smarttab' 和 'softtabstop' 选项 |ins-expandtab| 5. 替换模式 |Replace-mode| 6. 虚拟替换模式 |Virtual-Replace-mode| 7. 插入模式补全 |ins-completion| 8. 插入模式命令 |inserting| 9. Ex 插入命令 |inserting-ex| 10. 插入文件 |inserting-file| 关于如何移动光标到没有字符的位置,另见 'virtualedit'。对编辑表格有用。
1. 特殊键 *ins-special-keys* 在插入和替换模式里,以下字符有特殊含义;其它字符被直接插入。要插入这些特殊字符 到缓冲区里,在前面加上CTRL-V
。要插入<Nul>
字符,使用 "CTRL-V
CTRL-@" 或者 "CTRL-V
000"。在有的系统上,你必须使用 "CTRL-V
003" 来插入CTRL-C
。注意: 如果CTRL-V
被映射,你也许会经常使用CTRL-Q
来代替 |i_CTRL-Q|。 如果插入时你在特殊的语言模式下工作,参见 'langmap' 选项 |'langmap'| 了解如何避 免反复进出这些模式。 如果置位了 'insertmode',<Esc>
和一些其它的键有另外的含义。见 |'insertmode'|。字符 动作
*i_CTRL-[* *i_<Esc>*<Esc>
或 CTRL-[ 结束插入或替换模式,回到普通模式。结束缩写。 注意: 如果你很难在键盘上敲上<Esc>
键,训练自己使用 CTRL-[。 *i_CTRL-C*CTRL-C
退出插入模式,回到普通模式。不检查缩写。不激活 |InsertLeave| 自动命令事件。 *i_CTRL-@* CTRL-@ 插入最近插入的文本,并停止插入 {Vi: 仅当敲入第一个字符时,而且 只限于前 128 个字符} *i_CTRL-A*CTRL-A
插入最近插入的文本。{Vi 无此功能}
*i_CTRL-H* *i_<BS>* *i_BS*<BS>
或CTRL-H
删除光标前的字符 (关于连接行,见 |i_backspacing|)。 如果你的<BS>
键不正确,见 |:fixdel|。{Vi: 不删除自动的缩进}
*i_<Del>* *i_DEL*<Del>
删除光标下的字符。如果光标在行尾,并且 'backspace' 选项包括 "eol",删除<EOL>
;下一行就此附加于当前行之后。 如果你的<Del>
键不正确,见 |:fixdel|。{Vi: 不删除自动的缩进}
{Vi 无此功能}
*i_CTRL-W*CTRL-W
删除光标前的单词 (关于连接行,见 |i_backspacing|)。关于单词的 定义,见 |word-motions| 关于 "单词动作" 的定义。 *i_CTRL-U*CTRL-U
删除光标所有输入的字符 (关于连接行,见 |i_backspacing|)。 *i_CTRL-I* *i_<Tab>* *i_Tab*<Tab>
或CTRL-I
插入制表。如果打开 'expandtab' 选项,等价数目的空格被插入 (使 用CTRL-V
<Tab>
避免这种扩展: 如果CTRL-V
被映射,可以使用CTRL-Q
<Tab>
。|i_CTRL-Q|)。另见 'smarttab' 选项和 |ins-expandtab|。 *i_CTRL-J* *i_<NL>*<NL>
或CTRL-J
开始新行。 *i_CTRL-M* *i_<CR>*<CR>
或CTRL-M
开始新行。 *i_CTRL-K*CTRL-K
{char1}
[char2] 输入二合字母 (见 |digraphs|)。当{char1}
为特殊字符时,该键的 键码以<>
形式插入。例如字符串 "<S-Space>
" 可以这样输入:<C-K>
<S-Space>
(两个键)。两个键都不考虑映射。{Vi 无此功能}
CTRL-N
查找下一个关键字 (见 |i_CTRL-N|)。{Vi 无此功能}
CTRL-P
查找上一个关键字 (见 |i_CTRL-P|)。{Vi 无此功能}
CTRL-R
{0-9a-z"%#*+:.-=} *i_CTRL-R
* 插入寄存器内容。在输入CTRL-R
和第二个字符之间,'"' 会显示出 来,以提示你需要输入寄存器的名字。文本插入方式和直接输入相同, 但不使用映射和缩写。如果设置了 'textwidth'、'formatoptions' 或 'autoindent',插入的结果会受到影响。这和使用 "p" 命令和用鼠标 粘贴文本不同。 特殊寄存器: '"' 无名寄存器,包含最近删除或抽出的文本 '%' 当前文件名 '#' 轮换文件名 '*' 剪贴板内容 (X11: 主选择) '+' 剪贴板内容 '/' 最近的搜索模式 ':' 最近的命令行 '.' 最近插入的文本 '-' 最近的行内 (少于一行) 删除 '=' 表达式寄存器;你会被提示输入一个表达式 (见 |expression|) 注意 0x80 (十进制 128) 用于特殊键。例如,你可 以这样移动光标向上:CTRL-R
="\<Up>
" 用CTRL-R
CTRL-R
可以按本义插入文本。 如果结果是 |List|,里面的项目被看作行,之间以 换行符连接。 关于寄存器见 |registers|。{Vi 无此功能}
CTRL-R
CTRL-R
{0-9a-z"%#*+/:.-=} *i_CTRL-R_CTRL
-R* 插入寄存器内容。和单个CTRL-R
类似,但是文本按本义插入,而不是 像键盘输入那样。这意味着如果寄存器包含<BS>
这样的字符,结果会 不同。例如,如果寄存器包含 "ab^Hc":
CTRL-R
a 产生 "ac"。'textwidth'、'formatoptions' 等等选项仍然适用。如果你连这些都 想避免,使用 "
CTRL-R
CTRL-R
a 产生 "ab^Hc"。<C-R>
<C-O>
r",见下。 '.' 寄存器 (最近插入的文本) 依然如同键盘输入那样的方式插入。{Vi 无此功能}
CTRL-R
CTRL-O
{0-9a-z"%#*+/:.-=} *i_CTRL-R_CTRL
-O* 按本义插入寄存器内容,并且不进行自动缩进。和鼠标粘贴文本相同 |<MiddleMouse>|。 不会替换字符! '.' 寄存器 (最近插入的文本) 依然如同键盘输入那样的方式插入。{Vi 无此功能}
CTRL-R
CTRL-P
{0-9a-z"%#*+/:.-=} *i_CTRL-R_CTRL
-P* 按本义插入寄存器内容,修正缩进,和 |[<MiddleMouse>| 类似。 不会替换字符! '.' 寄存器 (最近插入的文本) 依然如同键盘输入那样的方式插入。{Vi 无此功能}
*i_CTRL-T*CTRL-T
在当前行开始处插入一个 shiftwidth 的缩进。缩进总是取整到 'shiftwidth' 的倍数 (这是 vi 兼容的)。{Vi: 只有在缩进内部才能用}
*i_CTRL-D*CTRL-D
在当前行开始处删除一个 shiftwidth 的缩进。缩进总是取整到 'shiftwidth' 的倍数 (这是 vi 兼容的)。{Vi:
*i_0_CTRL-D* 0CTRL-D
只有在使用自动缩进之后才有效}CTRL-D
删除所有当前行的缩进。{Vi:
*i_^_CTRL-D* ^CTRL-D
只有在使用自动缩进之后才有效}CTRL-D
删除当前行的所有缩进。缩进在下一行上恢复。这可以用于插入卷标。{Vi:
*i_CTRL-V*CTRL-D
只有在使用自动缩进之后才有效}CTRL-V
如果下一个是非数字,按本义插入。对特殊键而言,插入其终端代码。 不然,输入的是字符的十、八或十六进制值。|i_CTRL-V_digit|。CTRL-V
之后紧接着输入的字符不经过映射。{Vi: 没有十进制字节输入}
注意: 当CTRL-V
被映射时 (例如,用来粘贴文本),你可能经常需要 使用CTRL-Q
来代替。|i_CTRL-Q|。 *i_CTRL-Q*CTRL-Q
等同于CTRL-V
。 注意: 有的终端连接会吃掉CTRL-Q
,导致该快捷键无效。在 GUI 版本 里就不会出现这样的问题。CTRL-X
进入CTRL-X
模式,一个子模式。那里你可以给出命令来补全单词或者 滚动窗口。见 |i_CTRL-X| 和 |ins-completion|。{Vi 无此功能}
*i_CTRL-E*CTRL-E
插入光标下面的字符。{Vi 无此功能}
*i_CTRL-Y*CTRL-Y
插入光标上面的字符。{Vi 无此功能}
注意CTRL-E
和CTRL-Y
不使用 'textwidth',从而可以从长行里复制 字符。 *i_CTRL-_*CTRL-_
切换语言,如下: - 在从右到左的窗口里,切换 revins 和 nohkmap,因为在这种情况 下英语的输入可能就是倒过来的。 - 在非从右到左的窗口里,切换 revins 和 hkmap。因为希伯来语等 语种可能是倒过来输入的。CTRL-_
移动光标到输入文本的尾部。 该命令只有在 'allowrevins' 选项置位的时候才有效。 请参考 |rileft.txt|,那里可以了解到更多有关从右到左模式的信 息。{Vi 无此功能}
只有在编译时加入 |+rightleft| 特性才有效。 *i_CTRL-^* CTRL-^ 切换语言字符输入的使用方式。 如果定义了语言映射 |:lmap|: - 如果 'iminsert' 为 1 (使用 langmap 映射),变成 0 (不使用 langmap 映射)。 - 如果 'iminsert' 为其它值,变成 1,这样打开了 langmap 映射。 如果没有定义语言映射: - 如果 'iminsert' 为 2 (使用输入方法 (Input Method)),变成 0 (不使用输入方法)。 - 如果 'iminsert' 为其它值,变成 2,从而打开输入方法。 如果 "b:keymap_name" 变量的值设为 1,'keymap' 选项或者 "<lang>
" 出现在状态行上。语言映射通常用来输入不同于键盘上能直 接产生的字符。'keymap' 选项用来安装若干完整的映射表。{Vi 无此 功能} *i_CTRL-]* CTRL-] 切换缩写,不插入字符。{Vi 无此功能}
*i_<Insert>*<Insert>
切换插入和替换模式。{Vi 无此功能}
*i_backspacing*<BS>
、CTRL-W
和CTRL-U
的效果决定于 'backspace' 选项 (除非置位了 'revins')。这 时一个逗号分隔的项目列表:项目 动作
indent 允许退格删除自动缩进 eol 允许退格删除换行符 (连接行) start 允许退格删除插入开始之前的位置;CTRL-W
和CTRL-U
在开始位置停止 如果 'backspace' 为空,则使用 Vi 兼容的退格方式。不能退格删除自动缩进、回到第 一列之前、或者超过插入开始的地方。 为了后向兼容起见,取值 "0"、"1" 和 "2" 也是允许的,见 |'backspace'|。 如果 'backspace' 选项的确包含 "eol",光标在第一列,而使用了这三个键中的一个, 当前行会和上一行连接。这实际上删除了光标之前的<EOL>
。{Vi: 不会跨行,不会删除插入开始位置之前的内容}
*i_CTRL-V_digit* 使用CTRL-V
,字符的十、八、十六进制值可以直接输入。这样你可以输入任何字符,除 了换行符 (<NL>
,其值为 10)。有五个方法可以输入字符值:第一个字符 模式 最大字符数 最大值
(无) 十进制 3 255 o 或 O 八进制 3 377 (255) x 或 X 十六进制 2 ff (255) u 十六进制 4 ffff (65535) U 十六进制 8 7fffffff (2147483647) 通常你会输入最大数目的字符数。这样,要输入空格 (值为 32),你需要键入<C-V>
032。你可以忽略开头的零,这时,数字之后的字符必须不能再是数字。其它模式下 也一样: 一旦你输入在该模式下不合法的字符,那么这之前的值就会被使用,而 "非法" 的这个字符以正常的方式继续处理。 如果你输入的值为 10,在文件中最后会以 0 出现。10 是<NL>
,内部被用来代表<Nul>
字符。在写入文件时,<NL>
字符被翻译成<Nul>
。而在每行的最后写入<NL>
。所以,如 果你想在文件中插入<NL>
字符,你需要使用换行。 *i_CTRL-X* *insert_expand*CTRL-X
进入一个子模式,那里可以使用若干命令。绝大多数命令执行关键字补全;见 |ins-completion|。只有在 Vim 编译时加入 |+insert_expand| 特性才能使用这些功 能。 有两个命令可以在不退出插入模式的前提下上下滚动窗口: *i_CTRL-X_CTRL-E*CTRL-X
CTRL-E
窗口滚动上移一行。 补全时看这里:|complete_CTRL-E| *i_CTRL-X_CTRL-Y*CTRL-X
CTRL-Y
窗口滚动下移一行。 补全时看这里:|complete_CTRL-Y| 按了CTRL-X
以后,每个CTRL-E
(CTRL-Y
) 滚动窗口上 (下) 移一行,除非这使得光标 不得不离开当前文件中所在的位置。一旦按了另外一个键,CTRL-X
模式就会退出,而回 到插入模式下解释该键。
2. 特殊的特殊键 *ins-special-special* 一些的键是特殊的。它们停止当前的插入,做一些事情,然后重新插入。这意味着你可以 不脱离插入模式的情况下做一些事情。这适合于经常使用插入模式的用户,就像编辑器没 有单独的普通模式一样。这时,也可以设置 'backspace' 选项为 "indent,eol,start" 还有置位 'insertmode' 选项。如果你想给功能键映射到一个命令,你可以使用CTRL-O
。 这些键前后的改动 (插入或者删除字符) 可以分别撤销。只有最后的改动可以重做,而其 行为和 "i" 命令相当。字符 动作
<Up>
光标上移一行 *i_<Up>*<Down>
光标下移一行 *i_<Down>*CTRL-G
<Up>
光标上移一行,到插入开始时所在的列 *i_CTRL-G_<Up>*CTRL-G
k 光标上移一行,到插入开始时所在的列 *i_CTRL-G_k*CTRL-G
CTRL-K
光标上移一行,到插入开始时所在的列 *i_CTRL-G_CTRL-K*CTRL-G
<Down>
光标下移一行,到插入开始时所在的列 *i_CTRL-G_<Down>*CTRL-G
j 光标下移一行,到插入开始时所在的列 *i_CTRL-G_j*CTRL-G
CTRL-J
光标下移一行,到插入开始时所在的列 *i_CTRL-G_CTRL-J*<Left>
光标左移一个字符 *i_<Left>*<Right>
光标右移一个字符 *i_<Right>*<S-Left>
光标反向一个单词 (像 "b" 命令那样) *i_<S-Left>*<C-Left>
光标反向一个单词 (像 "b" 命令那样) *i_<C-Left>*<S-Right>
光标正向一个单词 (像 "w" 命令那样) *i_<S-Right>*<C-Right>
光标正向一个单词 (像 "w" 命令那样) *i_<C-Right>*<Home>
光标移到该行第一个字符 *i_<Home>*<End>
光标移到该行最后一个字符 *i_<End>*<C-Home>
光标移到该文件第一个字符 *i_<C-Home>*<C-End>
光标移到该文件最后一个字符 *i_<C-End>*<LeftMouse>
光标移动鼠标点击处 *i_<LeftMouse>*<S-Up>
上翻窗口一页 *i_<S-Up>*<PageUp>
上翻窗口一页 *i_<PageUp>*<S-Down>
下翻窗口一页 *i_<S-Down>*<PageDown>
下翻窗口一页 *i_<PageDown>*<MouseDown>
向下滚动三行 *i_<MouseDown>*<S-MouseDown>
向下滚动一个整页 *i_<S-MouseDown>*<MouseUp>
向上滚动三行 *i_<MouseUp>*<S-MouseUp>
向上滚动一个整页 *i_<S-MouseUp>*CTRL-O
执行命令,然后返回到插入模式 *i_CTRL-O* CTRL-\CTRL-O
类似于CTRL-O
,但不移动光标 *i_CTRL-\_CTRL-O*CTRL-L
置位 'insertmode' 时: 转到普通模式 *i_CTRL-L*CTRL-G
u 打断撤销序列,开始新的改变 *i_CTRL-G_u*
注意: 如果光标键把你带出插入模式,查查 'noesckeys' 选项。CTRL-O
命令有时有副作用: 如果光标在行尾之后,它会先被移动该行最后一个字符上。 在映射里,通常更好的方法是使用<Esc>
(先在文本中放一个 "x",<Esc>
这时总会把 光标放到它的上面)。或者使用 CTRL-\CTRL-O
,不过这时要注意光标可能移到行尾之外 的位置。 不是在所有的终端上都能用 Shift + 光标键。 另外一个副作用是 "i" 或 "a" 命令之前指定的计数会被忽略。这是因为要实现CTRL-O
之后的命令的重复执行太复杂了。 一个使用CTRL-G
u 的例子:
:inoremap
<C-H>
<C-G>
u<C-H>
它重定义退格键开始新的撤销序列。你可以撤销退格键的效果,而不会改变你之前输入的 内容,就像
CTRL-O
u 那样。CTRL-O
的使用分割撤销: 之前输入的文本和之后的被分别撤销。如果不想如此 (比如用 在映射里),你可以用CTRL-R
= |i_CTRL-R|。例如,要调用函数::imap
<F2>
<C-R>
=MyFunc()<CR>
如果正确设置 'whichwrap' 选项,在一行的第一个/最后一个字符上按
<Left>
和<Right>
键使得光标回绕到上一行/下一行。CTRL-G
j 和CTRL-G
k 命令可以用来在某一列前插入文本。例如:int i;
int j;
把光标定位在第一个 "int" 上,输入 "istatic<C-G>
j "。结果是:static int i;
int j;
要把相同的文本插入在每行的某列之前,使用可视列块命令 "I" |v_b_I|。
3. 'textwidth' 和 'wrapmargin' 选项 *ins-textwidth* 'textwidth' 选项可以用来在行变得很长之前自动断行。设置 'textwidth' 选项为希望 的最大行长。如果你输入更多字符 (不是空格或者制表),最后一个单词会放在一个新行 上 (除非这是该行唯一一个单词)。如果你设置 'textwidth' 为 0,关闭该特性。 'wrapmargin' 选项做的事情基本相同。区别在于 'textwidth' 是一个固定的宽度,而 'wrapmargin' 根据屏幕的宽度设置。如果设置 'wrapmargin',这等价于 'textwidth' 设为 (columns - 'wrapmargin'),其中 columns 是屏幕的宽度。 如果同时设置 'textwidth' 和 'wrapmargin',使用 'textwidth'。 如果你并不真的想断开行,而只是想文本行在合适的位置回绕,见 'linebreak' 选项。 文本行只有在插入模式下或者附加到行后的时候才会自动断开。在替换模式下,只要行的 长度没有变,就不会断行。 长行在你输入一个出现在边界之后的非空白字符的时候断开。什么时候断行的限制还可以 通过在 'formatoptions' 选项增加如下字母决定: "l" 断行只有在插入开始时文本行的长度不超过 'textwidth' 才会发生。 "v" 只有在当前插入命令中输入的空白字符上才会断行。这是和 Vi 最兼容的行为。 "lv" 断行只有在插入开始时文本行的长度不超过 'textwidth' 并且在当前插入命令中输 入的空白字符上才会发生。和 "l" 唯一的不同在超过 'textwidth' 边界之后输入 非空白字符的时候。 通常使用内部函数来决定哪里断行。如果你想使用不同的方法,设置 'formatexpr' 选项 为一个表达式,它处理换行的行为。 如果你想排版文本块,可以使用 "gq" 操作符。输入 "gq" 和可以移动光标到块尾的移动 命令。在许多情况下,命令 "gq}" 会做你想要做的事情 (排版直到段落尾部)。或者,你 可以使用 "gqap"。它会排版整个段落,和当前光标所在的位置无关。或者,你可以使用 可视模式: 敲击 "v",移动到块尾,然后输入 "gq"。另见 |gq|。
4. 'expandtab'、'smarttab' 和 'softtabstop' 选项 *ins-expandtab* 如果打开 'expandtab' 选项,空格可以用来填充制表键的空白位置。如果你需要输入一 个真正的<Tab>
,先输入CTRL-V
(用CTRL-Q
如果CTRL-V
被映射的话 |i_CTRL-Q|)。 缺省 'expandtab' 选项是关闭的。注意 在替换模式里,一个字符被多个空格字符所代 替。结果是,行内的字符数会增加。退格键只会一次删一个空格键。原来的字符只有在一 个空格 (最后一个) 上退格才能得回来{Vi 没有 'expandtab' 选项}
*ins-smarttab* 当 'smarttab' 选项打开时,<Tab>
在行首插入 'shiftwidth' 个位置,而在其它地方插 入 'tabstop' 个。这意味着经常会插入空格而不是<Tab>
字符。当 'smarttab' 关闭 时,<Tab>
总是插入 'tabstop' 个位置,而 'shiftwidth' 只有在 ">>" 和类似的命令 中才会用到。{Vi 无此功能}
*ins-softtabstop* 如果 'softtabstop' 选项不为零,<Tab>
插入 'softtabstop' 个位置,而过去用来删除 空格的<BS>
,现在会删除 'softtabstop' 个位置。感觉上, 'tabstop' 被设成了 'softtabstop' 的值,但实际上一个真正的<Tab>
字符还是占据 'tabstop' 个位置。从 而,你的文件在别的应用程序里看起来还是正确的。 如果 'softtabstop' 不为零,<BS>
会试图删除尽量多的空白,以便能够回到往前 'softtabstop' 的位置,除非前面一个插入的字符正好就是一个空格,这时它只会删除光 标前的那个字符。否则,你不一定总能删除光标前的一个字符。你需要先删除 'softabstop' 个字符,然后再输入额外的空格,以到达你想要的地方。
5. 替换模式 *Replace* *Replace-mode* *mode-replace* 在普通模式里输入 "R" 命令进入替换模式。 在替换模式里,行内的单个字符在你输入字符的时候被删除。如果没有字符可以删了 (在 行尾),输入的字符被附加 (和插入模式一样)。这样,一行内的字符数保持不变,直到你 到达行尾为止。如果输入了<NL>
,插入一个换行符,但不会删除任何字符。 要小心<Tab>
字符。如果你输入一个正常的打印字符在它上面,字符数仍然一样,但是 列数看起来少了。 如果你在替换模式下删除字符 (用<BS>
、CTRL-W
或CTRL-U
),实际发生的事是你删除了 改变。被替换的字符被复原了。如果你的输入超过已有的部分,新增的字符被删除了。实 际上,这可以看作是一次一个字符的撤销。 如果打开了 'expandtab' 选项,<Tab>
会用多个空格替换一个字符。结果是,行内的字 符数增加了。退格键只能一次删一个空格。原来的字符只有在一个空格 (最后一个) 上退 格才能得回来{Vi 没有 'expandtab' 选项}
6. 虚拟替换模式 *vreplace-mode* *Virtual-Replace-mode* 在普通模式里输入 "gR" 命令进入虚拟替换模式。{仅当编译时加入 +vreplace 特性才会有效}
{Vi 没有虚拟替换模式}
虚拟替换模式和替换模式类似,但不是替换文件里的实际字符,而是替换屏幕的领地。这 样,文件里的字符看起来不会移动。 所以,如果你输入了<Tab>
,它会替换多个普通的字符,而如果你在<Tab>
上输入字 母,它可能什么都没有代替,因为<Tab>
还是会占据相同的位置。 输入<NL>
不是导致文件后面的字符看起来在移动。结果是,当前行的后面部分被<NL>
所替换 (也就是,它们被删除),而替换继续在下一行进行。新行_不_会被插入,除非你 到达文件尾部之后。 输入CTRL-T
和CTRL-D
会看到有趣的效果。光标前面的字符向一边移动,跟平常一样, 但是光标后面的字符保持不动。CTRL-T
会隐藏一些被移动字符遮盖的旧行,但是CTRL-D
会重新让它们显现出来。 和替换模式一样,使用<BS>
等会恢复被替换的字符。即使和 'smartindent'、CTRL-T
和CTRL-D
、'expandtab'、'smarttab'、'softtabstop' 等一起使用的效果也是如此。 在 'list' 模式下,虚拟替换模式的行为和不在 'list' 模式下一样,除非 'cpoptions' 里设置了 "L"。 注意 唯一不在光标位置上的字符看起来在移动的可能是在 'list' 模式下,偶尔也会在 置位 'wrap' 的时候出现 (这时行改变长度,使得比屏幕宽度更窄或者更宽),难得也会 在输入 CTRL 字符的时候。CTRL 字符占据两个屏幕位置。如果用两个普通字符替换,第 一个会被插入,而第二个会替换 CTRL 字符。 该模式对编辑<Tab>
分隔表格列的时候很有用,因为输入新的数据的时候同时还能保持 所有的列对齐。
7. 插入模式补全 *ins-completion* 在插入和替换模式下,有若干命令可以补全输入的部分关键字或者行。这可以用于使用复 杂关键字的场合 (例如,函数名里有大写字母或者下划线)。 如果编译时关闭了 |+insert_expand| 特性,这些功能都不可用。 补全可以是针对: 1. 整行 |i_CTRL-X_CTRL-L| 2. 当前文件内的关键字 |i_CTRL-X_CTRL-N| 3. 'dictionary' 的关键字 |i_CTRL-X_CTRL-K| 4. 'thesaurus' 的关键字,同义词风格 |i_CTRL-X_CTRL-T| 5. 当前和头文件内的关键字 |i_CTRL-X_CTRL-I| 6. 标签 |i_CTRL-X_CTRL-]| 7. 文件名 |i_CTRL-X_CTRL-F| 8. 定义或宏 |i_CTRL-X_CTRL-D| 9. Vim 命令 |i_CTRL-X_CTRL-V| 10. 用户定义的补全 |i_CTRL-X_CTRL-U| 11. 全能 (omni) 补全 |i_CTRL-X_CTRL-O| 12. 拼写建议 |i_CTRL-X_s| 13. 'complete' 的关键字 |i_CTRL-N| 所有这些 (除了 2 以外 (译者注: 原文如此)) 都通过CTRL-X
模式完成。这是插入和替 换模式的一个子模式。你可以键入CTRL-X
和一个CTRL-X
命令进入CTRL-X
模式。要退 出,输入不合法的CTRL-X
模式的命令。合法的键包括CTRL-X
命令本身,CTRL-N
(下一 个) 和CTRL-P
(前一个)。 如果你想调整匹配的大小写,参见 'infercase' 选项。 *complete_CTRL-E* 如果补全处于激活状态,可以用CTRL-E
来停止补全并回到原来录入的文字。CTRL-E
本 身不会被插入。 *complete_CTRL-Y* 如果已经弹出菜单,可以使用CTRL-Y
停止补全并接受当前的选择项。CTRL-Y
本身不会 被插入。键入空格、回车或者其他不可显示字符将离开补全模式并插入键入的字符。 弹出菜单显示时,有一些特殊键可用,见 |popupmenu-keys|。 注意:CTRL-X
模式下合法的键不经过映射。这使得 ":map ^F ^X^F" 能够工作 (其中 ^F 是CTRL-F
而 ^X 是CTRL-X
)。能够使得CTRL-X
模式退出的键 (任何不是合法CTRL-X
模式命令的键) 则经过映射。 另外,通过 'complete' 的补全也使用映射。 注意: 补全激活时,不能递归使用插入模式。以某种方式调用 ":normal i.." 的映射将 产生 E523 错误。 建议使用以下映射来使得输入补全命令简单一点 (不过它们可能屏蔽其它的命令)::inoremap ^] ^X^]
:inoremap ^F ^X^F
:inoremap ^D ^X^D
:inoremap ^L ^X^L
一个特例是,执行寄存器插入的
CTRL-R
(见 |i_CTRL-R|) 不会退出CTRL-X
模式。这主 要是为了允许通过使用 '=' 寄存器来调用若干函数来决定下一个操作。如果该寄存器的 内容 (或者 '=' 寄存器计算的结果) 不是合法的CTRL-X
模式键,那么就会退出CTRL-X
模式,如同键盘输入这些内容一样。 例如,下面的程序会如此映射<Tab>
: 如果当前行只有空白,就插入<Tab>
,不然就开始 或继续CTRL-N
补全操作:
function! CleverTab()
if strpart( getline('.'), 0, col('.')-1 ) =~ '^\s*$'
return "\
<Tab>
"else
return "\
<C-N>
"endfunction
inoremap
<Tab>
<C-R>
=CleverTab()<CR>
补全整行 *compl-whole-line* *i_CTRL-X_CTRL-L*
CTRL-X
CTRL-L
反向搜索和当前行光标前字符序列完全相同的行。忽略缩进。 找到的行插入在光标的前面。 'complete' 选项用来决定匹配在哪个缓冲区里搜索,已载入 和未载入的缓冲区都被使用。CTRL-L
或CTRL-P
反向搜索前一个匹配行。替换上一次匹配的行。CTRL-N
正向搜索下一个匹配行。替换上一次匹配的行。CTRL-X
CTRL-L
在扩展一行以后,你可以通过接着输入CTRL-X
CTRL-L
得到 紧接着匹配行之后的行,直到见到两个CTRL-X
为止。 补全当前文件内的关键字 *compl-current* *i_CTRL-X_CTRL-P* *i_CTRL-X_CTRL-N*CTRL-X
CTRL-N
正向搜索以光标前面的关键字开始的单词。找到的关键字插入 在光标的前面。CTRL-X
CTRL-P
反向搜索以光标前面的关键字开始的单词。找到的关键字插入 在光标的前面。CTRL-N
正向搜索下一个匹配的关键字。替换上一次匹配的关键字。CTRL-P
反向搜索前一个匹配的关键字。替换上一次匹配的关键字。CTRL-X
CTRL-N
或CTRL-X
CTRL-P
继续使用CTRL-X
CTRL-N
或CTRL-X
CTRL-P
会复制上次本类 型补全在其它上下文里扩展的单词之后紧跟的单词,直到见到 两个CTRL-X
为止。 如果在光标的前面有一个关键字 (由字母字符和 'iskeyword' 指定的字符组成的名字), 它的前面再加上 "\<" (含义: 单词开始) 就被用作搜索模式。否则 "\<\k\k" 被用作搜 索模式 (任何包含至少两个字符的关键字的开始)。 在替换模式里,替换的字符数目决定于匹配字符串的长度。这和直接在替换模式下键盘输 入经过替换的字符串类似。 如果光标前面不是一个合法的关键字字符,则匹配任何至少有两个字符的关键字。 例如,要得到: printf("(%g, %g, %g)", vector[0], vector[1], vector[2]); 只需输入: printf("(%g, %g, %g)", vector[0], ^P[1], ^P[2]); 搜索会在文件末尾回绕,这里不使用 'wrapscan' 的值。 相同的补全内容多次重复会被跳过;这样每次CTRL-N
和CTRL-P
都会插入不同的匹配 (除非只有一个匹配的关键字)。 永远不会得到单个字符的匹配,因为它们通常不是你真想要的。 例如,要得到: printf("name = %s\n", name); 或者: printf("name = %s\n", n^P); 甚至: printf("name = %s\n", ^P); '\n' 中的 'n' 被跳过。 在扩展完一个词后,你可以使用CTRL-X
CTRL-P
或CTRL-X
CTRL-N
得到紧跟在扩展词 之后的单词。这些序列搜索刚刚扩展的文本,并且继续扩展之,使之包括另外一个词。这 可以用于你需要重复一系列复杂的单词的场合。尽管CTRL-P
和CTRL-N
只找至少两个字 符的字符串,CTRL-X
CTRL-P
和CTRL-X
CTRL-N
可以用来扩展只有一个字符的单词。 例如,要得到: México 你可以输入: M^N^P^X^P^X^PCTRL-N
开始一个扩展,而CTRL-P
回到单个字符 "M",然后后面的两个CTRL-X
CTRL-P
分别得到 "é" 和 ";xico"。 如果上次的扩展因为超过 'textwidth' 被分割,则只会使用当前行的文本。 如果匹配在行尾,那么下一行的第一个单词会被插入,而且显示消息 "word from next line"。如果该词被接受,那么下个CTRL-X
CTRL-P
或者CTRL-X
CTRL-N
会搜索那些以 这个单词开始的行。 补全 'dictionary' 的关键字 *compl-dictionary* *i_CTRL-X_CTRL-K*CTRL-X
CTRL-K
根据 'dictionary' 选项给出的文件搜索光标前关键字开始的 单词。这和CTRL-N
类似,只不过搜索的是字典文件,而不是 当前文件。找到的关键字插入在光标之前。这可能很慢,因为 在第一个匹配用到之前,所有的匹配都会被找到。缺省, 'dictionary' 选项为空。 要得到哪里能找单词列表的建议,见 'dictionary' 选项。CTRL-K
或CTRL-N
正向搜索下一个匹配的关键字。替换上一次匹配的关键字。CTRL-P
反向搜索前一个匹配的关键字。替换上一次匹配的关键字。 *i_CTRL-X_CTRL-T*CTRL-X
CTRL-T
和CTRL-X
CTRL-K
类似,但稍有不同。它使用 'thesaurus' 选项,而不是 'dictionary'。如果匹配在同义词字典里找 到,同一行里其余单词也在匹配里列出,即使它们并不真的匹 配。这样一个单词可以被完全替换。 举一个例子,假想 'thesaurus' 文件有一行形如:angry furious mad enraged
把光标放在字母 "ang" 之后并输入CTRL-X
CTRL-T
会匹配单 词 "angry";继续按会把单词改为 "furious"、"mad" 等等。 其它的使用包括两种语言之间的翻译,或者用关键字给 API 函数归类。CTRL-T
或CTRL-N
正向搜索下一个匹配的关键字。替换上一次匹配的关键字。CTRL-P
反向搜索前一个匹配的关键字。替换上一次匹配的关键字。 补全当前和头文件内的关键字 *compl-keyword* 'include' 选项指定如何找到含有头文件名字的行。'path' 选项用来搜索头文件。 *i_CTRL-X_CTRL-I*CTRL-X
CTRL-I
搜索当前和头文件里第一个以光标前面的字母序列开始的关键 字。找到的关键字插入在光标的前面。CTRL-N
正向搜索下一个匹配的关键字。替换上一次匹配的关键字。 注意:CTRL-I
和<Tab>
相同,而这可能会在成功的补全之后 输入,因此不使用CTRL-I
来搜索下一个匹配。CTRL-P
反向搜索前一个匹配的关键字。替换上一次匹配的关键字。CTRL-X
CTRL-I
继续使用CTRL-X
CTRL-I
会复制上次本类型补全在其它上下 文里扩展的单词之后紧跟的单词,直到见到两个CTRL-X
为 止。 补全标签 *compl-tag* *i_CTRL-X_CTRL-]*CTRL-X
CTRL-] 搜索第一个以光标前面的字母序列开始的标签。匹配的标签插 在光标前面。标签名可以包含字母字符和由 'iskeyword' 决 定的字符 (和关键字相同)。另见 |CTRL-]|。 'showfulltag' 选项可以用来增加标签定义前后的上下文。 CTRL-] 或CTRL-N
正向搜索下一个匹配的标签。替换上一次匹配的标签。CTRL-P
反向搜索前一个匹配的标签。替换上一次匹配的标签。 补全文件名 *compl-filename* *i_CTRL-X_CTRL-F*CTRL-X
CTRL-F
搜索第一个以光标前面的字母序列开始的文件。匹配的文件插 在光标前面。标签名可以包含字母字符和由 'isfname' 决 定的字符 (和关键字相同)。注意,(目前) 这里不使用 'path' 选项。CTRL-F
或CTRL-N
正向搜索下一个匹配的文件名。替换上一次匹配的文件名。CTRL-P
反向搜索前一个匹配的文件名。替换上一次匹配的文件名。 补全定义或宏 *compl-define* 'define' 选项用来指定包含定义的行。'include' 选项用来指定包含头文件名的行。 'path' 选项用来搜索头文件。 *i_CTRL-X_CTRL-D*CTRL-X
CTRL-D
搜索当前和头文件里第一个以光标前面的字母序列开始的 定义 (或宏)。找到的定义名插入在光标的前面。CTRL-D
或CTRL-N
正向搜索下一个匹配的定义。替换上一次匹配的定义。CTRL-P
反向搜索前一个匹配的定义。替换上一次匹配的定义。CTRL-X
CTRL-D
继续使用CTRL-X
CTRL-D
会复制上次本类型补全在其它上下 文里扩展的单词之后紧跟的单词,直到见到两个CTRL-X
为 止。 补全 Vim 命令 *compl-vim* 这里,补全是上下文敏感的,和命令行上的情况相似。它既能补全 Ex 命令,又能补全它 的参数。可用于编写 Vim 脚本。 *i_CTRL-X_CTRL-V*CTRL-X
CTRL-V
猜测光标前的项目的条目,并找到第一个匹配。 注意: 如果CTRL-V
被映射,你通常可以用CTRL-Q
来代替 |i_CTRL-Q|。CTRL-V
或CTRL-N
正向搜索下一个匹配。替换上一次匹配。CTRL-P
反向搜索前一个匹配。替换上一次匹配。CTRL-X
CTRL-V
继续使用CTRL-X
CTRL-V
和CTRL-V
一样。这允许映射键来 执行 Vim 命令补全,例如::imap
<Tab>
<C-X>
<C-V>
用户定义补全 *compl-function* 命令补全可以由用户通过 'completefunc' 选项自定义一个函数来完成。下面说明如何调 用此函数,并提供示例 |complete-functions|。 *i_CTRL-X_CTRL-U*
CTRL-X
CTRL-U
猜测光标前面项目的类型,并寻找它的第一个匹配。CTRL-U
或CTRL-N
正向搜索下一个匹配。替换上一次匹配。CTRL-P
反向搜索前一个匹配。替换上一次匹配。 全能 (omni) 补全 *compl-omni* 命令补全可以由用户通过 'omnifunc' 选项自定义一个函数来完成。这通常用于特定文件 类型的补全。 下面说明如何调用此函数,并提供示例 |complete-functions|。 关于特定文件类型的说明,见 |compl-omni-filetypes|。 将来会有更多补全脚本,欢迎查阅 www.vim.org。目前已经有了 C++ 的首个版本。 *i_CTRL-X_CTRL-O*CTRL-X
CTRL-O
猜测光标前面项目的类型,并寻找它的第一个匹配。CTRL-O
或CTRL-N
正向搜索下一个匹配。替换上一次匹配。CTRL-P
反向搜索前一个匹配。替换上一次匹配。 拼写建议 *compl-spelling* 定位光标所在或之前的单词,然后提出正确拼写的单词作为建议进行替代。如果该行里有 一个错误拼写的单词在光标之前或之下,移动光标到它后面。否则,使用刚刚在光标之前 的那个单词来提出建议,即使该单词没有拼写错误。NOTE:
很多 Unix 终端上,CTRL-S
暂停显示。这时用 's' 可以代替。如果显示暂停,输 入CTRL-Q
会继续显示。 *i_CTRL-X_CTRL-S* *i_CTRL-X_s*CTRL-X
CTRL-S
或CTRL-X
s 定位光标之前的单词,并寻找它的第一个拼写建议。CTRL-S
或CTRL-N
正向搜索下一个建议。替换上一次的建议。 注意 这里不能用 's'。CTRL-P
反向搜索前一个建议。替换上一次的建议。 从不同的来源补全关键字 *compl-generic* *i_CTRL-N*CTRL-N
在 'complete' 选项给出的地方搜索下一个以光标前面的关键 字开始的单词。找到的关键字名插入在光标的前面。 *i_CTRL-P*CTRL-P
在 'complete' 选项给出的地方搜索上一个以光标前面的关键 字开始的单词。找到的关键字名插入在光标的前面。CTRL-N
正向搜索下一个匹配的关键字。替换上一次匹配的关键字。CTRL-P
反向搜索前一个匹配的关键字。替换上一次匹配的关键字。CTRL-X
CTRL-N
或CTRL-X
CTRL-P
继续使用CTRL-X
CTRL-N
或CTRL-X
CTRL-P
会复制上次本类 型补全在其它上下文里扩展的单词之后紧跟的单词,直到见到 两个CTRL-X
为止。 寻 找 补 全 的 函 数 *complete-functions* 这里指 'completefunc' 和 'omnifunc'。 函数被调用两次,使用不同的方式: - 首先,调用函数以寻找补全文本的开始位置。 - 然后,调用函数以寻找实际的匹配。 第一次调用时,参数是: a:findstart 1 a:base 空 函数必须返回补全开始位置的列数,这个数字必须在零到光标所在列 "col('.')" 之间。 过程应该检查光标之前的字符,并包含那些可能成为补全项一部分的字符。该列到光标列 之间的文本将来会被匹配结果替换。如果补全无法进行,返回 -1。 第二次调用时,参数是: a:findstart 0 a:base 补全必须匹配的文本;即第一次调用定位的文本 (可以为空) 函数必须返回匹配单词的列表。这些匹配通常包含 "a:base" 文本。如果没有匹配,返回 空列表。 *complete-items* 每个列表项可以是字符串或者字典类型。如果是字符串,直接用作补全文本。如果是字 典,可以包含以下各项: word 需要插入的文本,必需 abbr "word" 的缩写;如果非空,菜单里使用它而不是 "word" menu 用于弹出菜单的补充文本,在 "word" 或 "abbr" 之后显示 info 关于补全项的更多信息,能够在预览窗口显示 kind 代表补全类型的单个字母 icase 如果非零,比较项目是否等同时忽略大小写;如果省略就假定 为零,这时可以同时加入只有大小写有差异的匹配项 dup 如果非零,那么即使和此匹配包含相同单词的匹配项已经存在 也无妨。 除了 'icase' (译者注: 还有 'dup') 以外,其它各项必须是字符串。如果有一项不合要 求,报错,而列表的其余项目也不再使用。你可以在返回列表中混用字符串和字典项目。 "menu" 项目用于弹出菜单且可能被截短,所以它应该尽量简短。"info" 项目可以稍长。 如果在 'completeopt' 中包含 "preview",预览窗口会显示该项信息。关闭弹出菜单后, "info" 项目将保留显示,这对录入函数参数很有用。用单个空格设置 "info" 可以清除 预览窗口现存的文本。 "kind" 项目用单一字母表示补全类型。用它可以指定补全的不同显示方式 (不同颜色或 者图标)。目前,可用如下类型: v 变量 f 函数或方法 m 结构或类成员 t typedef d #define 或宏 如果搜索匹配耗时较长,可以调用|complete_add()|向总列表中增加每个匹配。不要在返 回的列表里包含这些匹配!搜索匹配的同时,时不时地调用 |complete_check()| 来使得 用户仍然可以按键。如果该函数返回非零,搜索停止。 该函数可以移动光标,结束后光标会恢复。为了安全原因,不能在 |modeline| 或 |sandbox| 中设置此选项。 补全月份名的示例:fun! CompleteMonths(findstart, base)
if a:findstart
" 定位单词的开始处
let line = getline('.')
let start = col('.') - 1
while start > 0 && line[start - 1] =~ '\a'
let start -= 1
endwhile
return start
else
" 寻找匹配 "a:base" 的月份
let res = []
for m in split("Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec")
if m =~ '^' . a:base
call add(res, m)
endif
endfor
return res
endif
endfun
set completefunc=CompleteMonths
功能同上,但是现在假设搜索比较慢:fun! CompleteMonths(findstart, base)
if a:findstart
" 定位单词的开始处
let line = getline('.')
let start = col('.') - 1
while start > 0 && line[start - 1] =~ '\a'
let start -= 1
endwhile
return start
else
" 寻找匹配 "a:base" 的月份
for m in split("Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec")
if m =~ '^' . a:base
call complete_add(m)
endif
sleep 300m " simulate searching for next match
if complete_check()
break
endif
endfor
return []
endif
endfun
set completefunc=CompleteMonths
插 入 补 全 弹 出 菜 单 *ins-completion-menu* *popupmenu-completion* Vim 可以用更简单的弹出菜单来显示匹配。 当下面条件符合时使用弹出菜单: - 'completeopt' 选项包含 "menu" 或 "menuone"。 - 显示终端至少支持 8 色。 - 至少有两条匹配项。如果使用 "menuone",一条匹配也可以。 选项 'pumheight' 用于设置最大高度。默认值是使用全部有效空间。 有三个状态: 1. 插入了完整的匹配,例如在CTRL-N
或CTRL-P
之后。 2. 用光标键选择其它匹配项。此时不插入该匹配项,只在弹出菜单中高亮选中的条目。 3. 只插入了部分匹配文本,并且已经输入字符或者使用了退格键,这时匹配项列表根据 光标前的内容进行调整。 开始时你通常处于状态一并插入第一个匹配。如果 'completeopt' 包含了 "longest" 而 且有多个匹配项,那么开始于状态三。 如果选择其它匹配项,例如键入CTRL-N
或CTRL-P
,就进入了状态一。这不会改变匹配 项列表。 如果退回到原文,就会处于状态三。要立即进入该状态,可以使用快捷键映射方法,该映 射在补全开始后立即使用CTRL-P
::imap
*popupmenu-keys* 状态一下,这些键有特别的含义:<F7>
<C-N>
<C-P>
<BS>
和CTRL-H
删除一个字符,查找光标前单词的匹配项。这会减少列表中匹配项的 数目,常常到只有一个项目,然后切换到状态二。 其它非特殊字符: 停止补全,不改变匹配,然后插入输入的字符。 状态二和状态三下,这些键有特别的含义:<BS>
和CTRL-H
删除一个字符,并查找光标前变短的单词的匹配项。这可能会发现更 多的匹配项。CTRL-L
从当前匹配项中增加一个字符,可能会减少匹配项的数量。 任何可显示的非空白字符: 加入该字符,减少匹配项的数量。 全部三个状态中,可以使用这些键:CTRL-Y
是 (Yes): 接受当前选择的匹配项并停止补全。CTRL-E
结束 (End) 补全,回退到选择匹配前原有的内容 (原先输入的或者 最长的公共字符串)。<PageUp>
反向若干项选择一个匹配项,但不插入。<PageDown>
正向若干项选择一个匹配项,但不插入。<Up>
选择前一个匹配,同CTRL-P
,但不插入。<Down>
选择下一个匹配,同CTRL-N
,但不插入。<Space>
或<Tab>
停止补全,不改变匹配,插入键入的该字符<Enter>
键的行为取决于你现在所处的状态: 状态一: 使用现有的文本,然后插入换行符。 状态二: 插入当前选择项。 状态三: 使用现有的文本,然后插入换行符。 换句话说: 如果你只使用光标键在匹配项列表中选择其它条目,按<Enter>
键将插入该 匹配。如果键入其它字符,按<Enter>
键将插入换行符。 下面的高亮组能够改变菜单颜色: Pmenu 普通项 |hl-Pmenu| PmenuSel 选中项 |hl-PmenuSel| PmenuSbar 滚动条 |hl-PmenuSbar| PmenuThumb 滚动条拇指 (thumb) |hl-PmenuThumb| 显示弹出窗口时,没有专门的映射。但你可以使用插入模式映射并检查 |pumvisible()| 函数的返回值以进行不同的处理。例如::inoremap
<Down>
<C-R>
=pumvisible() ? "\<lt>
C-N>" : "\<lt>
Down>"<CR>
映射中用
<expr>
可以在键入某字符或者满足某些条件时弹出菜单。例如,键入 '.':inoremap
<expr>
. MayComplete()func MayComplete()
if (can complete)
return ".\
<C-X>
\<C-O>
"endif
return '.'
endfunc
详见 |:map-<expr>|。 特 定 文 件 类 型 全 能 补 全 的 补 充 说 明 *compl-omni-filetypes* 文件类型
{filetype}
使用的是 'runtimepath' 的 autoload/{filetype}
complete.vim 文件。比如对于 "java",就是文件 autoload/javacomplete.vim。 C *ft-c-omni* C 代码补全需要标签文件。你应该使用 Exuberant ctags 软件,因为它会加入补全所需 要的额外信息。你可以在这里找到它: http://ctags.sourceforge.net/ 推荐使用 5.6 或以后版本。 对于 5.5.4 版本,你应该打上增加 "typename:" 字段的补丁: ftp://ftp.vim.org/pub/vim/unstable/patches/ctags-5.5.4.patch 可以在这里找到 MS-Windows 上已经编译好的可执行版本: http://georgevreilly.com/vim/ctags.html 如果你想补全系统函数,可以用 ctags 生成包含所有系统头文件的标签文件:% ctags -R -f ~/.vim/systags /usr/include /usr/local/include
在 vimrc 文件中,把这个标签文件增加到 'tags' 选项中:set tags+=~/.vim/systags
如果在不包含 "." 或 "=" 的名字后面用
CTRL-X
CTRL-O
,会直接从标签文件中补全。这 适用于任何标识符,包括函数名。如果你想补全标签文件没有的局部变量名,用CTRL-P
代替。 如果在包含 "." 或 "=" 的名字后面用CTRL-X
CTRL-O
,Vim 会试图识别变量类型并指出 它所含的成员。这意味着只会列出该变量的有效成员。 如果成员名字已经完整,CTRL-X
CTRL-O
会为复合类型加上 "." 或 "->"。 Vim 没有包含 C 编译器,它只能识别使用明确格式的声明。预处理器指令可能会引起混 淆。如果在多个位置里定义了同样的结构名,所有可能的结构成员都会被包括。 CSS *ft-css-omni* 遵循 CSS 2.1 标准来补全属性和相应的值。 HTML *ft-html-omni* XHTML *ft-xhtml-omni*CTRL-X
CTRL-O
能够补全 (X)HTML 文件的各种元素。它是为了支持编写 XHTML 1.0 Strict 文件而设计的,但也可用于其它 HTML 版本。特性: - "<" 之后,根据上下文补全标签名 (在标签内部,不给出 div 标签建议);'/>
' 表明 空标签 - 在标签中,补全合适的属性 (不包括标签的 width 属性);同时显示属性类型;'*' 表 明必需的属性 - 如果属性只有有限的几个可能值,用它们来补全 - 补全实体 (entity) 名 - 根据<style>
标签和包含的 CSS 文件里提取的数据,补全 "class" 和 "id" 属性值 - 对 "style" 属性值或在 "style" 标签内部补全时,切换到 |ft-css-omni| 补全 - 对事件属性值或在 "script" 标签内部补全时,切换到 |ft-javascript-omni| 补全 - "</" 之后,CTRL-X
CTRL-O
会关闭最近打开的标签 备注: 如果是第一次使用,补全菜单的显示会有少许延迟--这点时间用于数据文件载入。 备注: 补全可能会在错误格式的文档中失效。在这种情况下,请尝试运行 |:make| 命令 检查格式问题。 HTML 类型 *html-flavor* 默认的 HTML 补全机制依赖于文件类型: HTML 文件使用 HTML 4.01 Transitional 标准 ('filetype' 是 'html'),XHTML 使用 XHTML 1.0 Strict 标准 ('filetype' 是 'xhtml')。 如果在任何标签的外部进行补全,你可以选择 DOCTYPE,然后载入合适的数据文件,并用 于后面所有的补全操作。 关于数据文件格式更多的信息见 |xml-omni-datafile|。vim 的线上站点 (|www|) 可以 找到一些数据文件。 注意 b:html_omni_flavor 可以指向任何使用 XML 数据的文件。这就使混合 PHP (|ft-php-omni|) 和任何 XML 方言的补全成为可能 (假设你有相应的数据文件)。如果该 变量没有设置,使用 XHTML 1.0 Strict 标准。 JAVASCRIPT *ft-javascript-omni* 补全绝大部分 JavaScript 语言和 DOM 元素。 补全: - 变量 - 函数名;显示函数参数 - 函数参数 - 变量属性,试图检测变量类型 - 根据上下文补全 DOM 对象和属性 - 语言中的关键字 补全机制可以用于单独的 JavaScript 文件 (&ft==javascript)、(X)HTML 的<script>
标签内部和事件的属性值 (包含对外部文件的扫描)。 DOM 兼容性 当前 (2006 年初) 有两种主要浏览器 - MS Internet Explorer 和 Mozilla Firefox。 这两个软件市场占有率达到 90% 以上。理论上,W3C 组织 (http://www.w3c.org) 建立 标准,但是这些标准并没有完全遵守和实现。IE FF W3C 全能补全
+/- +/- + +
+ + - +
+ - - -
- + - -
不管浏览器的实现进度如何,补全插件总是把标准定义的元素放入建议列表。两个主要浏 览器都实现但标准没有覆盖的元素也作为建议。而其它的元素则不出现于建议列表。 PHP *ft-php-omni* PHP 代码的补全需要标签文件才能对外部文件的数据和类进行补全。应该使用 Exuberant ctags 5.5.4 版本或更新的版本。在这里可以找到它: http://ctags.sourceforge.net/ 脚本对以下项目进行补全: - $ 变量名之后 - 如果变量声明为对象,加上 "->",如果标签文件有效,显示类名 - 在 "->" 之后,只补全给定类中的函数和变量名。为了查找类的位置和内容,需要标 签文件。因为 PHP 不是强类型的语言,用户可以使用 @var 标签来声明类:
/* @var $myVar myClass */
$myVar->
不过,要找到 myClass 的内容,仍然需要标签文件。 - 带有附加信息的函数名: - 如果是内建函数,列出可能的参数,在 | 之后列出函数返回的数据类型 - 如果是用户函数,列出参数和函数定义的文件名 (如果不是当前文件) - 常量名 - 在 "new" 声明之后的类名 注意: 如果第一次调用补全功能,Vim 会把所有需要的数据载入内存。这可能需要几秒 钟。下次补全时,就几乎感觉不到延迟了。 脚本检测光标是否在<?php ?>
标签内。如果不是,会自动切换到 HTML/CSS/JavaScript 补全。注意: 和原始 HTML 文件不同,标签补全 (也仅对标签补全而言) 和上下文无关。 RUBY *ft-ruby-omni* Ruby 代码的补全需要 Vim 编译时带 |+ruby| 特性。 Ruby 补全会因需分析你的缓冲区以提供补全列表。它会从 'require' 载入的和当前缓冲 区定义的模块里提取补全。CTRL-X
CTRL-O
提供的补全是上下文相关的:上 下 文 提 供 的 补 全
1. 不在类定义中 类名、常量和全局变量 2. 类定义中 这个类所定义的方法或常量 3. '.'、'::' 或者 ':' 之后 被解除参照的对象所适用的方法 4. ':' 或者 ':foo' 之后 符号名 ('foo' 起始的那些) 备注: - Vim 会载入/执行程序代码,以便提供补全。这可能会导致部分代码被执行。这一点也 许值得关注。缺省行为不再打开这一功能。如果需要此特性,加上let g:rubycomplete_buffer_loading = 1
- 在第 1 点中,Vim 可以解析整个缓冲区以获得用作补全结果的类名列表。默认关闭此 功能。要使其生效,在 vimrc 里加入let g:rubycomplete_classes_in_global = 1
- 在第 2 点中,不支持匿名类。 - 在第 3 点中,Vim 会试图判断对象所支持的方法。 - Vim 可以检测和载入 Rails 环境并用于 rails 项目的文件。默认关闭此特性。要使 其生效,在 vimrc 里加入let g:rubycomplete_rails = 1
SYNTAX *ft-syntax-omni* Vim 能够对将近 500 种语言进行语法高亮。高亮支持的一部分是需要知道构成语言的关 键字。许多文件类型已经有专门的补全脚本,而对其它的文件类型而言,syntaxcomplete 脚本可以提供基本的补全。实现的方法是用 Vim 知道如何色彩高亮的那些文本来构造全 能补全列表。它适用于任何文件类型,并可以提供基本但和语言相关的补全机制。 要打开语法打开补全:setlocal omnifunc=syntaxcomplete#Complete
你可以把下列语句放到 vimrc 中 (要在任何 ":filetype" 命令之后),使其自动生效
if has("autocmd") && exists("+omnifunc")
autocmd Filetype *
\ if &omnifunc == "" |
\ setlocal omnifunc=syntaxcomplete#Complete |
\ endif
endif
只有在针对特定文件类型的插件不存在的情况下,上述语句才对脚本设置补全操作。 每个文件类型可能有很多语法项目。此插件允许你定制从列表里包含或排除哪些语法组。 让我们看看 PHP 文件类型如何处理。 如果你正在编辑一个 index.php 文件,运行如下命令:
:syntax list
首先你将看到有许多不同的语法组。PHP 语言可以包含来自不同语言的元素,比如 HTML、 JavaScript 和许多其它语言。这种情况下,syntax 插件只包含由文件类型 "php" 开头的语法组。例如,缺省 PHP 包含这些语法组: phpEnvVar、phpIntVar、 phpFunctions。 PHP 语言可以进行语法高亮的项目非常多,而这些项目在全能补全列表里都会出现。有些 人可能觉得现这个列表不实用或者只对某些项感兴趣。 (如有必要) 有两种方法可以裁剪这个列表。如果不想显示特定语法组,在 vimrc 中增加 下列语句:
let g:omni_syntax_group_exclude_php = 'phpCoreConstant,phpConstant'
该列表可以加入多个语法组,以逗号分隔。这个变量的基本形式是:
let g:omni_syntax_group_exclude_
{filetype}
= '逗号,分隔的,列表'为了完整起见,还有相反的形式。在 vimrc 里建立下面的变量可以只包含 phpFunctions 和 phpMethods 语法组的项目:
let g:omni_syntax_group_include_php = 'phpFunctions,phpMethods'
可以建立任意多个这些变量,只要变量名尾部的文件类型不同就行了。 此插件使用 isKeyword 选项来决定用于语法项目的单词边界。例如,Scheme 语言的补全 应该包含 "-",call-with-output-file。取决于你的文件类型,此方法未必能提供你期 待的单词。设置 g:omni_syntax_use_iskeyword 选项为 0 会强制语法插件在单词字符上 断开。在 vimrc 可以加如下行进行控制:
let g:omni_syntax_use_iskeyword = 0
SQL *ft-sql-omni* SQL 语言的补全包括语句、函数和关键字。还可以动态地补全表、过程、视图和列的列 表,此时数据直接从数据库里提取。详细的指令和教程见 |omni-sql-completion|。 SQL 补全插件可以和其它补全插件一起使用。例如,PHP 文件类型有它自己的补全插件。 因为 PHP 常用来生成访问数据库的动态网站,也可以同时打开 SQL 补全插件。这样就可 以同时补全 PHP 代码和 SQL 代码。 XML *ft-xml-omni* Vim 7 为 XML 文件中提供上下文相关的补全机制。它依赖于特殊的数据文件 |xml-omni-datafile| 和两个命令: |:XMLns| 和 |:XMLent|。特性如下: - "<" 之后,根据上下文补全标签名 - 标签内部补全合适的属性 - 如果属性只有有限的几个可能值,用它们来补全 - 补全实体 (entity) 名 (|xml-omni-datafile| 里的定义加上当前文件 "<!ENTITY" 的 声明) - "</" 之后,
CTRL-X
CTRL-O
会关闭最后打开的标签 XML 数据文件的格式 *xml-omni-datafile* XML 数据文件保存在 'runtimepath' 下的 "autoload/xml" 目录中。Vim 发布在 "$VIMRUNTIME/autoload/xml" 目录下提供了示例数据文件。这些文件名有特别含义,命 令里使用会使用这些名字。文件名应该唯一,否则以后会产生冲突。例如,xhtml10s.vim 代表 XHTML 1.0 Strict 标准的数据文件。 每个文件包含一个名字形如 g:xmldata_xhtml10s 的变量。它由两个部分组成: 1. "g:xmldata_" 通用前缀,所有数据文件都是如此 2. "xhtml10s" 文件名,描述 XML 的方言;会用作 |:XMLns| 命令的参数 第二部分必须和文件名完全一样。 该变量为字典 |Dictionary| 类型。键是标签名,而值是两个元素的 |List|。列表中第 一个元素也是列表,包含可能的子元素名称,第二个元素是字典 |Dictionary|,键是属 性名,而值是属性的可能值。例如:
let g:xmldata_crippled = {
\ "vimxmlentities": ["amp", "lt", "gt", "apos", "quot"],
\ 'vimxmlroot': ['tag1'],
\ 'tag1':
\ [ ['childoftag1a', 'childoftag1b'], {'attroftag1a': [],
\ 'attroftag1b': ['valueofattr1', 'valueofattr2']}],
\ 'childoftag1a':
\ [ [],
{'attrofchild': ['attrofchild']}
],\ 'childoftag1b':
\ [ ['childoftag1a'],
{'attrofchild': []}
],\ "vimxmltaginfo": {
\ 'tag1': ['Menu info', 'Long information visible in preview window']},
\ 'vimxmlattrinfo': {
\ 'attrofchild': ['Menu info', 'Long information visible in preview window']}}
该例应放到 "autoload/xml/crippled.vim" 文件中。可用于编写下述文件:
<tag1 attroftag1b="valueofattr1">
<childoftag1a attrofchild>
& <
</childoftag1a>
<childoftag1b attrofchild="5">
<childoftag1a>
> ' "
</childoftag1a>
</childoftag1b>
</tag1>
从该例中,我们可以看到四种特殊元素: 1. "vimxmlentities" - 特殊键,包含此 XML 方言的所有实体的列表。 2. 如果这个包含属性可能值的列表只有一个元素,而该元素和属性名一样,那么该属性 被看作逻辑值,插入时使用 'attrname' 而不是 'attrname="'。 3. "vimxmltaginfo" - 特殊键,包含键为标签名、值为两元素列表的字典。值列表包含 附加的菜单信息和长描述。 4. "vimxmlattrinfo" - 特殊键,包含键为属性名、值为两元素列表的字典。值列表包含 附加的菜单信息和长描述。 注意: 数据文件里的标签名_必须_不能包含命名空间的描述。示例见 xsl.vim。 注意: 所有的数据和函数都作为全局变量/函数可以在任何地方访问,所以它们可以用于 个人编辑用的函数。 DTD -> Vim *dtd2vim* |www| 上有个脚本 |dtd2vim|,能够解析 DTD 并为 Vim XML 全能补全建立 XML 数据文 件。 dtd2vim: http://www.vim.org/scripts/script.php?script_id=1462 查看文件开始部分的详细用例。 该脚本需要 Perl 和: perlSGML: http://savannah.nongnu.org/projects/perlsgml 命令 :XMLns
{name}
[{namespace}
] *:XMLns* Vim 需要知道要使用的数据文件和命名空间。|:XMLns| 命令可以载入数据文件并把数据 连接到合适的命名空间。第一个 (必需的) 参数是数据名 (xhtml10s、xsl)。第二个参数 是命名空间编码 (h,xsl)。如果不使用第二个参数,那么将使用默认值--不声明命名空 间。例如在 .xsl 文件中使用 XML 补全:
:XMLns xhtml10s
:XMLns xsl xsl
:XMLent
{name}
*:XMLent* 缺省,根据默认命名空间的数据文件补全实体 (entity) 。如果没有默认命名空间,应该 用 XMLent 命令:
:XMLent xhtml10s
用法 在下述情况下 (在前一部分的声明之后,| 代表当前光标位置):
<|
补全合适的 XHTML 标签。而:
<xsl:|
补全合适的 XSL 标签。 由 |autoload| 机制提供的 xmlcomplete.vim 脚本定义了函数 xmlcomplete#GetLastOpenTag(),在 XML 文件中,这个函数可用于取得最后打开的标签 名 (下例必须先定义 b:unaryTagsStack):
:echo xmlcomplete#GetLastOpenTag("b:unaryTagsStack")
8. 插入模式命令 *inserting* 下列命令可以用来在缓冲区里插入新的文本。它们都可以撤销,也可以通过 "." 命令重 复。 *a* a 在光标后附加文本[count]
次。如果光标在空行的第一列, 启动插入模式。但在置位了 'virtualedit' 以后就不是! *A* A 在行尾附加文本[count]
次。<insert>
或 *i* *insert* *<Insert>* i 在光标前插入文本[count]
次。在插入模式里使用CTRL-O
的时候,|i_CTRL-O| 不支持计数。 *I* I 在本行第一个非空白字符之前插入文本[count]
次。 如果 'cpoptions' 里有 'H' 标志位而本行只有空白,在最后 一个空白前插入。 *gI* gI 在第一列插入文本[count]
次。{Vi 无此功能}
*gi* gi 在当前缓冲区最近一次插入模式停止的位置继续插入文本。 该位置记在 |'^| 位置标记里。如果标记在行末之后,和 "`^i" 有所差异。 该位置在插入/删除行时会自动修正。但_不_在插入/删除字符 时被修正。 使用 |:keepjumps| 命令修饰符时,不改变 |'^| 位置标记。{Vi 无此功能}
*o* o 在光标下方开启新行,并插入文本,重复[count]
次。{Vi: 清空[count]
个屏幕行} 如果 'cpoptions' 里有 '#' 标志位,忽略计数。 *O* O 在光标上方开启新行,并插入文本,重复[count]
次。{Vi: 清空[count]
个屏幕行} 如果 'cpoptions' 里有 '#' 标志位,忽略计数。 这些命令用以开始插入文本。你可以用<Esc>
退出插入模式。关于插入模式里的其它特 殊字符,见 |mode-ins-repl|。[count]
的效果只有在退出插入模式以后才会发生。 如果打开 'autoindent',新行的缩进从上一行得到。打开 'smartindent' 或 'cindent' 时,行的缩进根据 C 程序的要求自动调整。 'textwidth' 可以设置一行的最大宽度。如果一行过长,在添加字符时会自动添加换行 符。
9. Ex 插入命令 *inserting-ex* *:a* *:append* :{range}
a[ppend][!] 在指定行下方添加若干行。如果没有给出{range}
,文本会在 当前行之后插入。 加入 [!] 切换此命令执行时的 'autoindent'。 *:i* *:in* *:insert* :{range}
i[nsert][!] 在指定行上方添加若干行。如果没有给出{range}
,文本会在 当前行之前插入。 加入 [!] 切换此命令执行时的 'autoindent'。 这两个命令会继续要求行,直到你输入了只包含 "." 的一行。小心反斜杠开始的行,见 |line-continuation|。 如果这些命令和 |:global| 或 |:vglobal| 一起使用,那么从命令之后的文本里获得文 本行。这些行以反斜杠转义的 NL 分隔::global/abc/insert\
one line\
another line
不再需要最后的 "." 。 注意: ":append" 和 ":insert" 在 ":if" 和 ":endif"、":for" 和 ":endfor" 还有 ":while" 和 ":endwhile" 之间不能很好的工作。 *:start* *:startinsert* :star[tinsert][!] 在执行完本命令后,启动插入模式。和普通模式下输入 "i" 类似。如果包含 !,和 "A" 类似,附加到行后。否则,就从 光标当前位置开始插入。 注意 在函数或者脚本里使用本命令时,插入只会在函数和脚 本结束的时候才会开始。 此命令不能在 |:normal| 里使用。{Vi 无此功能}
{仅当编译时带 +ex_extra 特性时才可用}
*:stopi* *:stopinsert* :stopi[nsert] 尽快停止插入模式。和在插入模式时输入<Esc>
类似。可以 用在自动命令里。示例::au BufEnter scratch stopinsert
*replacing-ex* *:startreplace* :startr[eplace][!] 在执行完本命令后,启动替换模式。和普通模式下输入 "R" 类似。如果包含 !,和 "$R" 类似 (也就是,从行尾开始替换 模式)。否则,从光标当前位置开始替换。 注意 在函数或者脚本里使用本命令时,替换只会在函数和脚 本结束的时候才会开始。{Vi 无此功能}
{仅当编译时带 +ex_extra 特性时才可用}
*:startgreplace* :startg[replace][!] 和 |:startreplace| 完全类似,用虚拟替换模式,和使用 |gR| 类似。{Vi 无此功能}
{仅当编译时带 +ex_extra 特性时才可用}
10. 插入文件 *inserting-file* *:r* *:re* *:read* :r[ead] [++opt] [name] 在光标下方插入文件 [name] (缺省: 当前文件)。 |++opt| 说明 [++opt] 可能的取值。 :{range}
r[ead] [++opt] [name] 在指定行下方插入文件 [name] (缺省: 当前文件)。 |++opt| 说明 [++opt] 可能的取值。 *:r!* *:read!* :[range]
r[ead] !{cmd}
执行{cmd}
并把它的标准输出插入到光标下方。临时文件会 建立来保存命令输出的结果,并被读到缓冲区里。 'shellredir' 用来保存命令的输出结果,它可以设置是否包 含标准错误的输出。{cmd}
的执行和 ":!{cmd}
" 类似,任何 的 '!' 会被替换成以前的命令 |:!|。 这些命令插入文件的内容,或者命令的输出结果到缓冲区里。两者都可以撤销。但不能用 "." 命令重复。它们是基于行工作的,插入从光标所在行或指定行的下方开始。要在第一 行之上插入文本,使用命令 ":0r{name}
"。 在 ":read" 命令之后,光标留在第一个新行的第一个非空白处。和 Ex 模式不一样。那 里光标留在最后一个新行上 (对不起,那是为了和 Vi 兼容)。 如果文件名字通过 ":r" 给出,它成为轮换文件。这可以用来,比如说,你想编辑那个文 件的时候: ":e! #"。该特性可以通过删除 'cpoptions' 选项里的 'a' 标志位来关闭。 [++opt] 参数里,有一个是 ":read" 专用的: ++edit 参数。当 ":read" 命令就像编辑 文件一样把文件读入到缓冲区时,这个参数很有用。在空缓冲区上使用如下命令::read ++edit filename
效果是 'fileformat'、'fileencoding'、'bomb' 等选项根据 "filename" 的检测进行设 置。注意 会留下一行空行,你也许想把它删掉。 *file-read* 'fileformat' 选项设置文件的<EOL>
风格:'fileformat' 字符 名称
"dos"<CR>
<NL>
或<NL>
DOS 格式 "unix"<NL>
Unix 格式 "mac"<CR>
Mac 格式 以前使用 'textmode'。现在已经废弃了。 如果 'fileformat' 为 "dos",在<NL>
之前的<CR>
被忽略,而在文件尾部的CTRL-Z
被忽略。 如果 'fileformat' 为 "mac",文件里的<NL>
被内部表示为<CR>
。这是为了避免和用 来表示<NUL>
的<NL>
引起混淆。见 |CR-used-for-NL|。 如果 'fileformats' 选项不为空,Vim 试图识别<EOL>
的类型 (见 |file-formats|)。 不过,'fileformat' 选项的值不会被改变,检测到的格式只会在读入文件时使用。 'fileencodings' 与此情形类似。 在非 MS-DOS、Win32 和 OS/2 系统上,消息 "[dos format]" 会在读入 DOS 格式的文件 时给出,以提醒你发生了不寻常的事情。 在 Macintosh、MS-DOS、Win32 和 OS/2 系统上,消息 "[unix format]" 会在读入 Unix 格式的文件时给出。 在非 Macintosh 的系统上,消息 "[Mac format]" 会在读入 Mac 格式的文件时给出。 关于如何使用 ":r !" 的一个例子::r !uuencode binfile binfile
该命令读入 "binfile",用 uuencode 进行编码,并读入当前缓冲区。可以用于编辑包含 附带的二进制的文件的 e-mail。 *read-messages* 在读入文件时,Vim 会显示消息,显示读入文件的相关信息。以下的表格给出一些项目的 解释。其它的项目都不言自明。使用长格式还是短格式取决于 'shortmess' 选项的设 置。长 短 含义
[readonly]{RO}
文件被写保护 [fifo/socket] 使用流 [fifo] 使用 fifo 流 [socket] 使用套接字 (socket) 流 [CR missing] 使用 "dos" 'fileformat' 读入文件的时候 出现没有前导的 CR 的 NL [NL found] 使用 "mac" 'fileformat' 读入文件的时候 出现 NL (可能是 "unix" 格式) [long lines split] 至少一行以上被分割 [NOT converted] 期待从 'fileencoding' 到 'encoding' 的 转换但是做不到 [converted] 从 'fileencoding' 到 'encoding' 的转换 完成 [crypted] 文件被解密 [READ ERRORS] 不是文件所有部分都被成功读入 vim:tw=78:ts=8:ft=help:norl:
Generated by vim2html on 2008年 03月 27日 星期四 17:04:45 CST