Chinaunix首页 | 论坛 | 博客
  • 博客访问: 659333
  • 博文数量: 175
  • 博客积分: 2457
  • 博客等级: 大尉
  • 技术积分: 1488
  • 用 户 组: 普通用户
  • 注册时间: 2011-07-13 20:31
文章分类

全部博文(175)

文章存档

2012年(22)

2011年(153)

分类:

2011-12-23 15:19:20

本节所用命令的帮助入口:

 

:help usr_03.txt
:help motion.txt
:help usr_29.txt
:help scroll.txt
:help folding

上一篇文章中我们介绍了一些常用的移动命令,本篇将继续介绍更多的命令,使你在文档中自由穿梭。

 

[ 利用跳转表 ]

 

VIM中,很多命令可以引起跳转,VIM会记住把跳转前光标的位置记录到跳转表中,并提供了一些命令来根据跳转表进行跳转。要知道哪些命令引起跳转,参见“:help jump-motions”。

使用命令“''”(两个单引号)和“``(两个反引号,在键盘上和“~”共用一个键)可以返回到最新的跳转位置。例如,当前光标位于文件中第1234行,然后我使用“G”命令跳转到第4321行;这时如果我按“''”或"``",就会跳回到1234行。

因为这两个命令也属于跳转命令,所以第4321行也被记入跳转表,如果你再次使用这两个命令,就会发现自己又跳回第4321行了。

这两个命令有一点不同,“``”在跳转时会精确到列,而“''”不会回到跳转时光标所在的那一列,而是把光标放在第一个非空白字符上。

如果想回到更老的跳转位置,使用命令“CTRL-O”;与它相对应的,是“CTRL-I”,它跳转到更新的跳转位置(:help CTRL-O:help CTRL-I)。这两个命令前面可以加数字来表示倍数。

使用命令“:jumps”可以查看跳转表(:help :jumps)

 

 [ 使用标记 ]

 

标记(mark)VIM提供的精确定位技术,其功能大概相当于GPS技术,只要你知道标记的名字,就可以使用命令直接跳转到该标记所在的位置。

VIM中的标记都有一个名字,这个名字用单一的字符表示。大写和小写字母(A-Za-z)都可以做为标记的名字,这些标志的位置可以由用户来设置;而数字标记0-9,以及一些标点符号标记,用户不能进行设置,由VIM来自动设置。

我们主要讲述字母标记的使用,对于数字标记和标点符号标记,请自行参阅帮助手册(:help mark-motions)

小写字母标记局限于缓冲区,也就是说,每个缓冲区都可以定义自己的小写字母标记,各缓冲区间的小写字母标记彼此不干扰。如果我在文件A中设置一个标记t,然后在文件B中也可以设置一个标记t。那么在文件A中,可以用“'t”命令跳到文件A的标记t位置 ;在文件B中,可以用“'t”命令跳到文件B的标记t位置。如果文件在缓冲区列表中被删除,小写字母标记就丢失了。

大写字母标记是全局的,它在文件间都有效。如果在文件A中定义一个标记T,那么当使用命令“'T”时,就会跳到文件A的标记T位置,不管你当前处于哪个文件中。

设定一个标记很简单,使用命令“m{a-zA-Z}”就可以了。例如,命令“mt”在把当前光标位置设定为标记t;命令“mT”把当前光标位置设定为标记T(:help m)

要跳转到指定的标记,使用命令“'{a-zA-Z}”或“`{a-zA-Z}”。例如,命令“'t”会跳转到标记t;命令“'T”会跳转到标记T( :help ' )

单引号和反引号的区别和上面所讲的一样,“`”在跳转时会精确到列,而“'”不会回到跳转时光标所在的那一列,而是把光标放在第一个非空白字符上。

标记也可以被删除,使用命令“:delmarks”可以删除指定标记。命令“:marks”列出所有的标记。

 

关于标记,有两个非常有用的插件,一个是ShowMarks,另外一个叫marks browser

ShowMarks是我最常用的插件之一,它使用VIM提供的sign功能以及高亮功能显示出标记的位置。这样,你在设定了一个标记后,它就会在你的VIM窗口中显示出标记的名字,并高亮这一行。

这个插件在这里下载,下载后,在你的$HOME/.vim目录把它解压:

 

在我的vimrc中,对ShowMarks进行了如下配置:

   """"""""""""""""""""""""""""""
   " showmarks setting
   """"""""""""""""""""""""""""""
   " Enable ShowMarks
   let showmarks_enable = 1
   " Show which marks
   let showmarks_include = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
   " Ignore help, quickfix, non-modifiable buffers
   let showmarks_ignore_type = "hqm"
   " Hilight lower & upper marks
   let showmarks_hlline_lower = 1
   let showmarks_hlline_upper = 1

首先,使能showmarks插件,然后定义showmarks只显示全部的大写标记和小写,并高亮这两种标记;对文件类型为help, quickfix, 和不可修改的缓冲区,不显示标记的位置。

你可以定义自己的颜色来高亮标记所在的行,下面是我的定义,我把它放在我自己的colorscheme文件中:

" For showmarks plugin
hi ShowMarksHLl ctermbg=Yellow   ctermfg=Black  guibg=#FFDB72    guifg=Black
hi ShowMarksHLu ctermbg=Magenta  ctermfg=Black  guibg=#FFB3FF    guifg=Black

ShowMarks插件中已经定义了一些快捷键:


mt   - 打开/关闭ShowMarks插件
mo   - 强制打开ShowMarks插件
mh   - 清除当前行的标记
ma   - 清除当前缓冲区中所有的标记
mm   - 在当前行打一个标记,使用下一个可用的标记名

我最常使用的是“mm”和“mh”,用起来非常方便。

VIM 7.0中,如果大写的标记被定义了,那么函数line()无论在哪个缓冲区里都会返回该标记的行号,导致showmarks在每个缓冲区里都会把这个大写标记显示出来。因此我为这个插件打了个补丁来修正此问题。
VIM 7.0
中也可以真正的删除一个mark标记,所以也改了showmarks插件的删除标记功能。原来的功能在删除标记时,并未真正删除它,只是把这个标记指向缓冲区的第一行;现在则是真正删除此标记。

补丁的内容如下:

--- easwy/showmarks.vim    2007-03-13 10:15:07.000000000 +0800
+++ plugin/showmarks.vim    2007-03-23 09:35:01.000000000 +0800
@@ -144,6 +144,25 @@
 hi default ShowMarksHLo ctermfg=darkblue ctermbg=blue cterm=bold guifg=blue guibg=lightblue gui=bold
 hi default ShowMarksHLm ctermfg=darkblue ctermbg=blue cterm=bold guifg=blue guibg=lightblue gui=bold
 
+" Function: GetMarkLine()
+" Authors: Easwy Yang
+" Description: This function will return the line number where the mark
+" placed. In VIM 7.0 and later, function line() always returns line number but
+" not 0 in case an uppercase mark or number mark is placed. However, in VIM 6,
+" it only returns 0 when the uppercase mark isn't placed in current file.
+fun! s:GetMarkLine(mark)
+    if v:version < 700
+        let lnum = line(a:mark)
+    else
+        let pos = getpos(a:mark)
+        let lnum = pos[1]
+        if pos[0] && bufnr("%") != pos[0]
+            let lnum = 0
+        endif
+    endif
+    return lnum
+endf
+
 " Function: IncludeMarks()
 " Description: This function returns the list of marks (in priority order) to
 " show in this buffer.  Each buffer, if not already set, inherits the global
@@ -354,7 +373,8 @@
         let c = strpart(s:IncludeMarks(), n, 1)
         let nm = s:NameOfMark(c)
         let id = n + (s:maxmarks * winbufnr(0))
-        let ln = line("'".c)
+        "let ln = line("'".c)
+        let ln = s:GetMarkLine("'".c)
 
         if ln == 0 && (exists('b:placed_'.nm) && b:placed_{nm} != ln)
             exe 'sign unplace '.id.' buffer='.winbufnr(0)
@@ -393,11 +413,18 @@
     let s:maxmarks = strlen(s:IncludeMarks())
     while n < s:maxmarks
         let c = strpart(s:IncludeMarks(), n, 1)
-        if c =~# '[a-zA-Z]' && ln == line("'".c)
+        "if c =~# '[a-zA-Z]' && ln == line("'".c)
+        if c =~# '[a-zA-Z]' && ln == s:GetMarkLine("'".c)
             let nm = s:NameOfMark(c)
             let id = n + (s:maxmarks * winbufnr(0))
             exe 'sign unplace '.id.' buffer='.winbufnr(0)
-            exe '1 mark '.c
+            " Easwy, we can really remove marks in VIM 7.0 and later
+            if v:version >= 700
+                exe 'delm '.c
+            else
+                exe '1 mark '.c
+            endif
+            " Easwy, end
             let b:placed_{nm} = 1
         endif
         let n = n + 1
@@ -417,7 +444,13 @@
             let nm = s:NameOfMark(c)
             let id = n + (s:maxmarks * winbufnr(0))
             exe 'sign unplace '.id.' buffer='.winbufnr(0)
-            exe '1 mark '.c
+            " Easwy, we can really remove marks in VIM 7.0 and later
+            if v:version >= 700
+                exe 'delm '.c
+            else
+                exe '1 mark '.c
+            endif
+            " Easwy, end
             let b:placed_{nm} = 1
         endif
         let n = n + 1
@@ -466,7 +499,8 @@
     while n < s:maxmarks
         let c = strpart(s:IncludeMarks(), n, 1)
         if c =~# '[a-z]'
-            if line("'".c) <= 1
+            "if line("'".c) <= 1
+            if s:GetMarkLine("'".c) <= 1
                 " Found an unused [a-z] mark; we're done.
                 let next_mark = n
                 break

用法:

1. 保存该patch到某一目录,例如:/tmp/showmarks.vim.patch

2. cd到你的.vim目录:cd ~/.vim

3. 运行命令:cat /tmp/showmarks.vim.patch | patch -p0

 

Marks Browser可以显示出当前缓冲区中定义的小写标记的位置,在你无法对应上标记的名字和其位置时,非常有用。这个插件在此下载:

下载后把它放到你的$HOME/.vim/plugin目录即可,我为其定义了一个快捷键:

   """"""""""""""""""""""""""""""
   " markbrowser setting
   """"""""""""""""""""""""""""""
   nmap mk :MarksBrowser

这样,直接使用“,mk”就可以打开Mark Browser窗口了。

下图显示这两个插件工作时的效果。我在文件中定义了三个标记,一个大写标记A,两个小写标记at。最上面的窗口是Mark Browser窗口,主编辑窗口中的高亮行及sign标记是ShowMarks插件放置的。

 


[ 折行 ]

 

在文件比较大时,在文件中移动也许会比较费力。这个时候,你可以根据自己的需要把暂时不会访问的文本折叠起来,既减少了对空间的占用,移动速度也会快很多。

VIM提供了多种方法来进行折叠,既可以手动折叠,也可以根据缩进、语法,或使用表达式来进行折叠。

程序文件一般都具有良好的结构,所以根据语法进行折叠是一个不错的选择。

 

要启用折叠,首先要使能'foldenable'选项,这个选项是局部于窗口的选项,因此可以为每个窗口定义不同的折叠。

接下来,设置'foldmethod'选项,对于程序,我们可以选择根据语法高亮进行折叠。需注意的,要根据语法高亮进行折叠,必须打开文件类型检测和语法高亮功能,请参见我前面的文章。

下面是我的vimrc中的设置,它使用了自动命令,如果发现文件类型为ccpp,就启用折叠功能,并按语法进行折叠:

autocmd FileType c,cpp  setl fdm=syntax | setl fen

注意,VIM的很多命令、选项名都有简写形式,在帮助手册中可以看到简写形式,也可以按简写形式来help,例如,要查看'foldmethod'选项的帮助,可以只输入“:help 'fdm'”。

 

折叠后的效果见下图:

 

图中以黑色背景显示的行就是被折叠起来的行,VIM会显示这个fold中被折叠了多少行,以及起始行的内容。留意一下左下方的“__Tag_List__”窗口,在这个窗口中也存在着折叠,我把macro, typedef, variable几项折叠起来了,而把function的折叠打开。从该窗口最左边的折叠栏(:help fold-foldcolumn)也可以看出不同:被折叠的文本前显示了"+",打开的折叠前显示的是"|"。

折叠的背景色及显示文字等都可以修改,参阅帮助手册(:help folding)

 

下面的命令用来打开和关闭折叠:

zo – 打开光标下的折叠
zO – 循环打开光标下的折叠,也就是说,如果存在多级折叠,每一级都会被打开
zc – 关闭光标下的折叠
zC – 循环关闭光标下的折叠

更多的命令,请参阅手册。

VIM提供了一些命令在折叠间快速移动:

[z – 到当前打开折叠的开始
]z – 到当前打开折叠的结束
zj – 向下移动到下一个折叠的开始处
zk – 向上移动到上一个折叠的结束处

我通常不喜欢把文本折叠起来,因为我更喜欢一目了然的看到全部文本。你可以根据自己的喜好来决定是否启用折叠。

多说一点,手动创建的折叠是可以保存在session文件,这样下次进入VIM时可以载入之前创建的折叠,参见:help 'sessionoptions'

 

[ 在程序中移动 ]

 

VIM的作者是一个程序员,这就不难理解为什么VIM提供了众多在程序中移动的命令。这里面既包括我们前面的文章中介绍过的利用tag文件及cscope在标签间跳转,也包括众多在函数、注释、预处理指令、程序段,及其它程序元素中移动的命令。

本文不再详细介绍这些命令,作为程序员,一定要熟读usr_29.txt!这些命令,可以帮助你在程序中得心应手的移动。
 

在这里介绍两个插件,增强了在程序中移动的功能,一个是a.vim,另外一个是matchit

a.vim的功能非常简单,它帮助你在源文件和头文件间进行切换,这个简单的功能,却非常实用,至少它为我节省了很多时间。

在此下载a.vim插件:

       

然后把它放到你的.vim/plugin目录就可以了。

假设你正在浏览C语言的源文件,这时想修改它对应的头文件,只需要输入“:A”命令,就切换到头文件了(需要源文件和头文件在同一目录中)。a.vim插件还定义了其它一些命令和快捷键,参见它的帮助。
 

VIM中,"%"命令跳转到与当前项目相匹配的项目。例如,当光标位置在“{”时,按下%,光标就跳转到对应的“}( :help %)

VIM提供的%命令,只能在括号,或者C注释的开始和结束( /*  */),或者C编译预处理指令间进行跳转。对于其它程序结构,例如HTML%命令不能从标记,跳转到对应的标记。

Matchit插件则扩展了%命令的功能,使%命令可以对其它程序语言的开始和结束标记间进行跳转。在此处下载matchit插件:

       

把这个插件放到你的.vim/plugin目录,你就可以用%在各种开始/结束标记间跳转了,目前,它可以支持Ada, ASP with VBS, Csh, DTD, Essbase, Fortran, HTML, JSP (same as HTML), LaTeX, Lua, Pascal, SGML, Shell, Tcsh, Vim, XML等语言。

 

[ 插入模式下的移动 ]

 

上面介绍的移动命令,都是在normal模式下使用的,如果想在insert模式下移动,阅读:help ins-special-special

你真的需要在插入模式下移动吗?我几乎不会!通常我会先按ESC返回Normal模式,然后再移动,当你习惯了以后,你会发现效率会更高。

 

[ 小结 ]

 

你会发现,本文的内容,和usr_03.txt帮助文档很相似。是的,只要你学会了usr_03.txt中列出的命令,你就掌握了最常用最实用的VIM移动命令。

如果你想了解更多的移动命令,请通篇阅读motion.txt,记住你最有可能用到的那些键。当你的手指能够不假思索的使用这些命令后,你在VIM中就能做到指随意动、移动如飞了。

 

 [参考文档]

1. VIM帮助文件

2.

阅读(560) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~