Vim技能修炼教程(3) - 语法高亮进阶

时间:2022-06-19 13:59:46

语法高亮进阶

首先我们复习一下上节学到的三个命令:
* syntax match用于定义正则表达式和规则的对应
* highlight default定义配色方案
* highlight link将正则规则和配色方案对应起来

但是,定义好了规则,如何让它自动生效呢?我们需要识别文件类型。

识别文件类型

我们创建一个新插件,建立一个ftdetect目录,然后为这个类型创建一个vim文件,比如之前logcat的例子,我们就建立一个logcat.vim。
例子我们还是找github上的:https://github.com/serpent7776/vim-logcat/blob/master/ftdetect/logcat.vim

au BufNewFile,BufRead logcat set filetype=logcat

au是autocmd的缩写,BufNewFile和BufRead是触发自动命令的事件。
BufNewFile是创建一个新文件,BufRead是将文件读入一个新缓冲区时触发。
具体事件可以参看:help autocmd
logcat是文件名匹配的模式。当符合这种模式之后,将filetype设成logcat,就激活了logcat.vim中配置的高亮。

光比较文件名还是有点low的,还可以通过文件内容来识别.我们来看个复杂一点的例子:https://github.com/thinca/vim-logcat/blob/master/ftdetect/logcat.vim
例:

autocmd BufReadPost,BufNewFile *.logcat setlocal filetype=logcat
autocmd BufReadPost,BufNewFile * if getline(1) =~# '^-\{9} beginning of.*$'
\ | setfiletype logcat
\ | endif

更高级的写法,可以将文件内容识别封装成函数:
例子来源于:https://github.com/gburca/vim-logcat/blob/master/ftdetect/logcat.vim

fun! s:DetectLogcat()
" Detect from the 2nd line. The 1st line could be:
"
-------- beginning of system
if line('$') > 1 && getline(2) =~# '.*[F|E|W|I|D|V]/\S*\s*(.*\d'
set filetype=logcat
endif
endfun

au BufNewFile,BufRead *.lc set filetype=logcat
au BufNewFile,BufRead *.logcat set filetype=logcat
au BufNewFile,BufRead Boot-*_Set-*_Stream-*.txt set filetype=logcat
au BufNewFile,BufRead * call s:DetectLogcat()

如果自动命令没生效,我们也可以通过手动的方式来设置:set filetype=文件类型。filetype太常用了,也可以简写成ft.

vim插件的目录结构

第一讲我们是通过vundle插件来管理插件路径的。在古老的年代里,插件要么写在.vimrc里,要么放在~/.vim/下面。后来,可以通过指定runtimepath,简写为rtp来指定。

每个runtimepath指定的目录都可以包含下面的文件和子目录:
* filetype.vim 根据文件名来判断文件类型
* scripts.vim 根据内容还判断文件类型
* autoload/ 每次启动vim时都自动加载的脚本。相当于是对.vimrc中的autocmd命令的扩展,需要做autocmd的,都可以写进这个目录。详情请参见:help autoload-functions
* colors/ 配色方案,详情请参见:help :colorscheme
* compiler/ 跟各种语言编译器打交道的脚本,后面会讲,详情可参见:help :comp。可以运行下:comp命令试试,会列出一系列的编译器相关的vim文件,可以打开几个看看。
* doc/ 文档。这个需要强调一下,vim是以文档完备而著称的。详情参见:help write-local-help
* ftplugin/ 只处理本缓冲区的file type处理的plugin
* indent/ 处理格式缩进的脚本。详情请看:help :indent-expression
* keymap/ 有点类似于输入法的意思,将一串英文字符转化成多语言的字符,比如中文
* lang/ 菜单文字的翻译
* menu.vim 图形模式下的菜单
* pack/ 插件包。详情可参见:help :packadd
* plugin/ 通用的plugin
* print/ 用于postscript打印的插件
* spell/ 用于拼写检查的插件
* syntax/ 语法高亮,详情请参见:help mysyntaxfile
* tutor/ 教程

对于我们的语法高亮的插件来说,我们可以建立doc, ftdetect和syntax三个目录,分别放文档,类型判断脚本和语法高亮脚本。

关键字高亮

我们上次讲的syn match其实才是高亮的核心武器。不过,针对于一些比较简单的情况,比如语言中的关键字,就可以更简单一些,只要全字匹配就好。

这时候可以用到syntax keyword命令。

我们来看个llvm的例子:

syn keyword llvmType void half float double x86_fp80 fp128 ppc_fp128
syn keyword llvmType label metadata x86_mmx
syn keyword llvmType type label opaque
syn match llvmType /\<i\d\+\>/

关键字定义了之后,还是要通过hi def link命令跟预定义的配色方案对应起来。

例:

hi def link llvmType Type
hi def link llvmStatement Statement

常用的预定义的配色方案有:
* Comment: 注释
* Identifier:标识符
* Statement:语句
* PreProc:预处理语句
* Type: 数据类型,如int, long, double
* Special: 特殊符号
* Underlined: 带下划线的符号,如超文本链接
* Ignore: 留空
* Error:错误信息
* Todo: TODO, FIXME之类的信息

实际操作练习

下面我们通过一个最简单的例子来检验一下前面的学习成果。

首先,我们在~/.vim/下建立一个ftdetect目录,随便起个文件名字,比如就叫test.vim吧,先做类型匹配:

 autocmd BufNewFile,BufRead *.test set filetype=test

第二步,我们定义几个类型关键字。在~/.vim/下建立syntax目录,还叫test.vim:

if exists("b:current_syntax")
finish
endif
syntax keyword TestType int long short byte

highlight default link TestType Type

let b:current_syntax = "test"

最后一步,检验一下我们的成果,随便写几句用类型的语名吧,叫xxx.test之类的

int x = 10;
long y = 20;

可以看到,int和long已经被作为关键字被识别出来了。

小作业:在github上建立一个插件,用vunble来下载您的这个插件,实现语法高亮的功能

扩展阅读:
* :help group-name,除了上面介绍的常用预定义组之外,这里有更详细的说明

不过,关键字对于很多情况是不够用的,我们还是要回到上一讲的三步曲上写正则表达式。