Vim 语法定义syntax详解

回忆

语法文件

  • vi /usr/share/vim/vim81/syntax/java.vim

图片描述

  • 我们可以看见这个java语法文件的:
    • 作者
    • github地址
    • 修改时间
  • 然后就是大量的syn定义

图片描述

  • 什么是syn呢?

syntax语法

  • h syn

图片描述

  • syntax就是语法
  • :syntax list可以列出
    • syntax item 语法项
    • syntax group 语法组
    • syntax cluster 集群
  • 我们来试试

syntax list 语法列表

  • :syntax list

图片描述

  • 语法项目都是用help打头的
  • 为什么呢?
  • q跳出去
  • :set filetype?
  • 可以知道目前的文件是help文件
  • :q跳回到java.vim
  • :syntax list

图片描述

  • 得到的语法项都是和vim有关
  • q跳出去
  • :e oeasy.java
  • :syntax list

图片描述

  • 为什么打开不同的文件就会有不同的语法项列表呢?

语法文件

  • 语法列表是由语法文件来决定的
  • 语法文件位于语法文件夹中
    • /usr/share/vim/vim81/syntax/

图片描述

  • help的语法文件
    • /usr/share/vim/vim81/syntax/help.vim
  • vim的语法文件
    • /usr/share/vim/vim81/syntax/vim.vim
  • java的语法文件
    • /usr/share/vim/vim81/syntax/java
  • 我们继续来看java的语法文件

关键字keyword

  • /\<public\>

图片描述

  • syntax keyword javaScopeDecl public protected private abstract
    • syntax 语法
    • keyword 关键字
    • javaScopeDecl java的Scope可见性Declaration描述
    • public protected private abstract 具体的关键字
  • vi oeasy.java
  • :colorscheme murphy
  • :hi javaScopeDecl ctermfg=brown

图片描述

  • 试验成功
  • 我们可以自定义的一个属于自己的关键字么?

自定义关键字

  • :h syntax

图片描述

  • 这里是定义syntax语法项的地方
  • 第一种语法项就是keyword
  • 我们去定义一个
  • sudo vi /usr/share/vim/vim81/syntax/java.vim

图片描述

  • 把System定义为oeasy这个类型的关键字
  • 然后保存退出
  • 重新打开oeasy.java
  • 颜色模式设置为murphy
  • :hi oeasy ctermfg=yellow

图片描述

  • 试验成功
  • 但是public对应的这个javaScope
  • 又是如何与StorageClass挂上关系的呢?

层层关联

  • 回到syntax中的java.vim
  • /\<javaScopeDecl\> 9个匹配

图片描述

  • 这里有个hi def link

图片描述

  • 就是对于高亮hilight组的链接

图片描述

  • 把javaScopeDecl、javaClassDecl、javaMethodDecl链接到了javaStorageClass上
  • 这样,:hi javaStorageClass cterm=white就可以用到三组上面
  • 但是上图最上的一句话又是什么意思呢?
  • hi def link javaStorageClass StorageClass

组名通用化

  • :e /usr/share/vim/vim81/syntax/c.vim

图片描述

  • 我们可以看到不同语言有不同的语言的StorageClass
    • java的叫做javaStorageClass
    • c的叫做cStorageClass
    • rust叫做rustStorageClass
  • 不管他们原来的名字如何,都映射到StorageClass
  • 这样murphy、morning、blue等颜色方案只要设置StorageClass的高亮
  • 就可以映射到不同语言的不同组里面了
  • 像StorageClass这样的通用组名还有很多

图片描述

  • 如果我们做一门叫做oeasy的语言
  • 就需要把关键字定义到oeasyStorageClass中
  • 然后把oeasyStorageClass映射到StorageClass上
  • 只要颜色方案中有关于StorageClass的定义
  • 我们相关关键字就可以显示出来
  • 除了keyword之外,还有什么样的高亮item的定义呢?

match 匹配模式

图片描述

  • match顾名思义就是匹配模式
  • 比如syntax match javaComment /#.*/
    • javaComment是高亮项名称
    • /#.*/
      • /是模式匹配的开始和结束
      • #代表#
      • 如果是#开头的行,^#
      • 如果是#开头的单词,\<#
      • .任意字符
      • *代表任意字符.的数量是0到任意多个
  • /match

图片描述

  • javaBraces 对应的是左右大括号
  • "[{}]"
    • ""里面是pattern
    • []里面是或的关系
    • {}是左或右大括号

图片描述

  • 匹配import 或者 import static

图片描述

  • 这里怎么引号外面还有东西呢?

偏移量 offset

图片描述

  • "\.\s*\<class\>"
    • \.开头有任意字符
    • \s*然后有0到任意多个空格或tab
    • 然后是独立单词class

图片描述

  • ms的意思是Match Start匹配头
  • ms=s+1是什么意思

图片描述

  • s是start
  • ms=s+1
  • 就是把\s*前面的\.去掉开始定义匹配的开始
  • 我们再看一个

图片描述

  • match 的思路是匹配一个模式
  • 匹配了就进入组
  • 除了match直接匹配之外
  • 还有什么定义高亮组的方法么?

region 区域

图片描述

  • 最重要的有三个属性
    • start开头
    • end结尾
    • skip 跳过的

图片描述

  • start=+"+
    • +是分隔符号相当于/
    • pattern是"
    • 以"为开头
  • end=+"+"
    • 同上
    • 以"为结尾
  • skip=+\"+
    • 分割符还是+
    • 中间如果有\"就跳过
    • 不作为结尾
  • echo "i say \"oeasy\"" #echo sth
  • 匹配"i say \"oeasy\""

region带偏移量

图片描述

  • 高亮开始
    • hs highlight start 高亮开始位置
    • e 是end 指的是start开始模式/\*的结束位置
    • e+1指的是\*后面一个字符的位置
    • hs=e+1 高亮开始于开始模式末尾后一个字符
  • 高亮结束
    • he highlight end 高亮结束位置
    • s 是 start 指的是end结束模式/\*的开始位置
    • s-1 指的是\*/前面一个字符的位置
    • he=s-1 高亮结束于结束模式开头前一个字符

图片描述

  • 匹配模式match的是mmm...mmm的位置
  • 匹配高亮的region是rrr对应的位置
  • 除了region之外,还有什么高亮组的定义方法么?

嵌入的高亮组

图片描述

  • xComment
    • x语言的注释
    • %开始的都是注释
    • 注释里面可以包含TODO
  • xTODO
    • x语言的TODO
    • TODO高亮组只能存在于注释里面
    • 不在注释里面的不是TODO高亮组

递归嵌入

图片描述

  • xBlock
    • x语言的Block块
    • 大括号包起来的都是Block
    • Block可以自身递归嵌套

      统一截行 keepend

图片描述

  • xComment
    • x语言的注释,从%到结尾$
  • xPreProc
    • x语言的预处理,从#到结尾$
    • 可以包含xComment
  • 蓝色部分把第二行也认为是预处理
  • 因为第一行的结尾回车给了xComment
  • xPreProc就把第二行的结尾回车当作自己的了
  • 为了避免这个情况
  • keepend

图片描述

排除元素ALLBUT

图片描述

  • contains里面可以放具体的高亮组
  • 也可以放所有东西
  • 如果要排除某些东西怎么办呢?

图片描述

下一组nextgroup

图片描述

  • 这三个东西必须挨着
  • 在一起
  • 各自才能成立

匹配组 matchgroup

图片描述

  • 第一种模式定义了一个组
    • 包括括号和括号里面的内容
  • 第二种模式生成了两个组
    • xInside
    • xParen
    • 可以分别上色

透明transparent

  • 如果你想要while和for循环的条件有不同颜色
  • 可以这样试试

图片描述

  • for、while各有一套
  • cCondNest本身是透明的
  • 跟着上一层高亮组走

行内oneline

图片描述

  • oneline就是一行
  • 保证这个匹配的模式必须在一行以内
  • 有一行那有没有多行呢?

多行

图片描述

  • 多行里面包含了xLineContinue
  • xLineContinue包含了\\$
  • 换行符被包含在了高亮组里
  • 于是高亮组可以换行了

集群cluster

图片描述

  • contains里面的东西是重复的
  • 可以定一个集群cluster

图片描述

  • 然后contains里面就用集群

图片描述

  • 集群本质上是高亮组的列表
  • 这个列表可以添加和删除

图片描述

更多语法

  • 比如你打开一个cpp文件
  • 文件缓冲被解析为cpp的高亮组
  • 但是我同时想用c的高亮组
  • 怎么办?

图片描述

  • 什么是runtime呢?
    • 运行时环境
  • syntax/c.vim
    • 在运行时环境里面找syntax/c.vim
  • 运行时环境是什么?
  • :echo $VIMRUNTIME
    • /usr/share/vim/vim81
  • 这是一切的来源
    • 语法文件的来源
    • 高亮颜色模式的来源
    • 缩进的来源
  • 如果出现了问题
  • 可以去看看这里是不是正确

总结

  • 这次我们彻底研究了vim高亮的原理
  • 各种语法项syntax item
    • 关键字keyword
    • 匹配模式match
    • 区域region
  • 定义好了之后还可以设置链接成组
    • hi def link javaComment Comment
  • 然后就可以在颜色文件中具体配色了
    • 比如murphy中对于Comment的定义
    • hi Comment ctermfg=yellow
  • 不同的语言可以有不同的语法定义
  • 其实,vim也是门语言
  • vimscript也是可以编程的
  • 怎么玩呢?🤔
  • 下次再说!

教程来源于Github,感谢overmind1980大佬的无私奉献,致敬!

技术教程推荐

雷蓓蓓的项目管理实战课 -〔雷蓓蓓〕

Kafka核心源码解读 -〔胡夕〕

技术面试官识人手册 -〔熊燚(四火)〕

朱涛 · Kotlin编程第一课 -〔朱涛〕

徐昊 · TDD项目实战70讲 -〔徐昊〕

大厂设计进阶实战课 -〔小乔〕

快手 · 移动端音视频开发实战 -〔展晓凯〕

Vue 3 企业级项目实战课 -〔杨文坚〕

互联网人的数字化企业生存指南 -〔沈欣〕