I have been using vim-airline to customize my Nvim statusline ever since I first started using Neovim. It works great. Recently, I started a repo to create a minimal configuration without external plugins for both Vim and Neovim. During the process, I learned how to build the Vim statusline from scratch.

The primary source of my reference is the Vim help file on statusline (see :h 'statusline'). It is well written and should give your enough knowledge on building a simple statusline.

The Vim statusline consists of a number of items with special format to indicate what you want to show and how they should be displayed, e.g., width, color, alignment, etc.

# Statusline item format

Usually, the statusline consists of multiple printf style % items for showing various info about current file, e.g., %F is used to show the full path of current file. The complete format for items are as follows:

%-0{minWidth}.{maxWidth}{item}


- means to align the item to the left instead of the default right alignment. 0 is the leading zeros for items which return numeric numbers and is overridden by -. minWidth and maxWidth decide the min and max length of the item to be shown. All items are optional except {item}.

# Good convention

Since you may use several items to build your statusline, it is best to concatenate the various components to build the statusline (also see :h set+=) instead of using a single long strings to set up statusline, for example:

set statusline=
" show full file path
set statusline+=%F
" show current line number
set statusline+=%l


# Show the result of functions or an expression

Vim provides some default items to show info about current file, but they may not be enough. You may want to show the result of a function or show strings based on some conditions. Vim use %{} to evaluate functions or expressions and show the result in statusline.

To show the current mode in statusline, you can use the following setting:

let g:currentmode={
\ 'n'  : 'NORMAL ',
\ 'v'  : 'VISUAL ',
\ 'V'  : 'V·Line ',
\ '' : 'V·Block ',
\ 'i'  : 'INSERT ',
\ 'R'  : 'R ',
\ 'Rv' : 'V·Replace ',
\ 'c'  : 'Command ',
\}

set statusline=
set statusline+=\ %{toupper(g:currentmode[mode()])}


Based on which mode you are in, the statusline will change automatically.

To show a modified sign [+] based on whether current file has been modified, you can use the following setting:

set statusline+=%{&modified?'[+]':''}


# Using group

Group is used to set width and alignment for several elements in a whole. Start a group with ( and end it with %), i.e., (group_element%). For example, to show the file format, you may use the following setting:

set statusline+=%-7([%{&fileformat}]%)


The above group has min width 7 and is left aligned.

# Using highlight groups

Ever wondering how to show different colors for different parts of the statusline? It is simple: use highlight groups.

There are two slightly different ways to use highlight.

## Use a highlight group name

The first way is to use highlight group’s name. You can check what highlight groups are available using :highlight command. To use highlight group Warnings, you can use the following setting:

set statusline+=%#Warnings#
set statusline+=%{&bomb?'[BOM]':''}


The highlight group will take effect from the point it is used in statusline. To switch a highlight, you just need to provide another highlight group at your desired position.

## Use special user defined highlight groups

Vim also allows you to define 9 special highlight groups named from User1 to User9 and refer to them in statusline with a special syntax like the following:

# use highlight group User2
set statusline+=%2*


First, you need to define the needed user highlight group:

hi User1 ctermfg=0 ctermbg=114
hi User2 ctermfg=114 ctermbg=0


Then you can use them in statusline settings. To return to the default Statusline colors, you can use:

set statusline+=%*


# Common errors to avoid

The most common error is fail to escape some special characters. Since space | have special meanings (see also :h option-backslash), you need to use backslash to escape them to include a literal one. For example:

set statusline+=%{&filetype!=#''?&filetype.'\ ':'none\ '}


The above setting will show the current filetype followed a space or none followed by space if filetype is empty.

% is also a special character. To include a literal % char, you need to use %%. For example, to show the percentage of current cursor position in the buffer, use:

set statusline+=%2p%%