从零开始配置 vim(15)——状态栏配置

2022-09-22 17:04:09 浏览数 (1)

vim 下侧有一个状态栏,会显示当前打开的文件等一系列内容,只是我们很少去关注它。而且原生的vim也支持对状态栏进行自定义。这篇文章主要介绍如何自定义状态栏

设置状态栏

我们可以采用 set statusline 来设置状态栏。 例如我们输入 :set statusline=%f - FileType=%y。之后我们可以发现它变成了如下这个样子

statusline 的值是一个格式字符串。上述命令我们使用了 %f 表示当前文件名称 。

从上面的命令中明显感觉到只输出两个内容就已经开始显的比较繁琐了,如果显示的内容多了是不是就更加难以阅读和书写了呢?好在我们还能使用 lua来做设置。上述内容可以翻译为如下的 lua代码

代码语言:javascript复制
vim.o.statusline = "%f - FileType=%y"

我们保存之后发现下方的状态栏显示内容已经发生变化。

控制边距和宽度

如果你写过 c 或者其他编程语言中的输出语句,应该很容易理解如何控制输出的格式,一般使用类似 `%4l`` 这样的语句来控制该项占4个字符宽度。这里的设置也是类似的

代码语言:javascript复制
vim.o.statusline = " f - FileType=%y"

它表示 文件名这项应该站20个字符宽度。最终效果如下所示

默认它的边距是添加在左边的,这样会让左侧空出一大半,会显得比较难看,我们可以使用 -来使空白站位符显示在右侧

代码语言:javascript复制
vim.o.statusline = "%-20f - FileType=%y"

当然我们也可以控制一下输出字符的最大长度,例如使用如下代码

代码语言:javascript复制
vim.o.statusline = "%0.10F - FileType=%y"

使用 %F可以显示文件的全路径。这里我们发现全路径大于10个字符,但是它只是显示了10个字符。使用这个方法可以防止某些超长的字符串破坏了我之前的布局

分割

我们再来介绍一个符号 %=,它表示将 %= 后面的内容全局居右对齐,例如

代码语言:javascript复制
vim.o.statusline = "%f %= FileType=%y"

将得到如下显示内容

更多的关于各个标识符代表的含义可以使用 :help statusline 来查看。

练习

最后我们来做一个小练习,我们希望将状态栏显示为如下内容

代码语言:javascript复制
mode | filename[status] | currentline:totalline                       |encoding|filetype|

其中 mode 代表当前所处模式、filename 代表文件名称、status 代表文件状态(是否可读写、是否保存)、currentline 代表光标当前所在行数、totalline 代表文件一共多少行、encoding 代表文件编码、filetype 代表文件类型

我们通过查阅文档可以知道:

  • 文件名称可以使用 %f 来显示
  • 文件状态可以使用 %m 来显示
  • 当前行可以使用 %l
  • 当前buffer总行数可以使用 %L
  • 文件类型可以使用 %y 来显示

至于当前模式和文件类型我们先不管它,根据这些内容我们可以写下如下代码

代码语言:javascript复制
vim.o.statusline = "mode|%f%m|%l:%L%=encoding|%y"

[ ] 表示对缓冲区所做的修改还没有写入到磁盘中。执行:w写入之后发现它直接消失了

我们可以通过 vim.g.encoding 来获取当前文件的编码方式,因此我们改一下当前代码

代码语言:javascript复制
vim.o.statusline = "mode|%f%m|%l:%L%=" .. vim.g.encoding .. "|%y"

我们发现此时的状态已经改过来了。

最麻烦的是模式,vim 中提供了一个可以获取当前模式的函数 mode 。但是在 lua 接口中我没有找到对应的函数。因此我们仍然采用在 lua 中调用 vimscript 的方式。这次我们使用函数 vim.api.nvim_eval()。它可以执行 vim 命令并将返回执行的结果。

我们可以定义一个函数返回当前所处模式

代码语言:javascript复制
function get_mode()
    local mode = vim.api.nvim_eval([[mode]])
    if mode == "n" then
        return "Normal"
    elseif mode == "v" then
        return "Visual"
    elseif mode == "i" then
        return "Insert"
    else
        return ""
    end
end

这里我们为了演示只返回了 3中模式的字符串,更多模式可以查看vim的帮助文档

我们希望模式改变时对应的就修改 mode 对应的字段,此时我们应该采用自动命令。查看vim的用户手册我们发现,针对模式变化这一事件有一个叫做 ModeChanged 的事件类型,因此我们可以写下一些代码

代码语言:javascript复制
local cmd_statusline = vim.api.nvim_create_autogroup("SET_STATUS_LINE", {clear = true})

vim.api.nvim_create_autocmd({"ModeChanged"}, {
    pattern = "*",
    group = cmd_statusline,
    callback = function()
      local win_id = vim.api.nvim_eval([[win_getid()]]) --获取当前window id
      vim.wo[win_id].statusline = get_mode() .. "|%f%m|%l:%L%=" .. vim.g.encoding .. "|%y" -- 使用 setlocal 针对窗口设置本地化配置
    end
)

后面我们可以对字符串进行一些格式控制,这里就不展开了。我们将这些代码写到 init.lua中,保存后发现它大致效果如下

我们发现就是简单的设置状态栏的工作也是比较麻烦的。而且有时候我们又想它好看,带点颜色什么的。这就更麻烦了。好在有大量的插件可以帮助我们来完成这一工作

lualine 插件

lualine 是一个用 lua 语言开发的 neovim 的状态栏美化插件,可以使用如下代码进行安装

代码语言:javascript复制
use {
  'nvim-lualine/lualine.nvim',
  requires = { 'kyazdani42/nvim-web-devicons', opt = true }
}

我们创建 lualine 的配置文件,加入加载 lualine 的代码

代码语言:javascript复制
require("lualine").setup()

然后在主配置文件中加载该文件

代码语言:javascript复制
require("plugin-config/lualine")

lualine 官方提供了3种主题的配色,我们可以直接在代码中引用,例如我这里引用 evil_lualine 这个配置,然后将 theme改为之前我们安装的 tokyonight主题

代码语言:javascript复制
theme = "tokyonight"

到这里我们已经完成了 状态栏的美化,其实主要靠插件,本篇一大部分写了如何使用原生的statusline 设置主要是自己手工设置比较有意思,而且也可以凑文章字数,其实你不知道statusline 这个也无所谓,很多插件都都对原生的方式进行了大量封装,而且能进行高度的自定义。完全可以满足你各种奇怪的口味。

当然状态栏配置并不只有这一种插件,如果你知道有哪些状态栏的插件也欢迎在评论区留言给出

0 人点赞