This is the series 3 of my blog posts about some nifty techniques of using Nvim. For other series, see

How do I use last search pattern in substitute command?

As is often the case, when we want to replace some patterns, we may not be able to write the correct regular expression in one pass and have to refine it many times. I often search first to make sure that the search pattern is correct. Then I manually type the search pattern again in substitute command to replace it with the text I want. This is acceptable when the pattern is simple to type. It becomes unacceptable once the search patterns are complicated.

To use the last search pattern, you can keep the search part empty in substitute command, e.g., %s//REPLACE/g, which will replace all occurrences of last search pattern with REPLACE. Alternatively, you could use <C-R>/ to input your last search pattern, that is, first press Ctrl+ R and then press /1.


Where Can I Find A list of the Builtin Functions?

Sometimes, to write a Vim script, we may need to find if we can employ a builtin function. A list of builtin function can be found here.

How do I write the output of a command or builtin function to a buffer?

A least three ways are possible (take function strftime() and command :scriptnames (:h :scriptnames) for an example).

  1. Use <C-R>= in insert mode. In insert mode, press Ctrl+R followed by =, then type the function (e.g., strftime('%c')) or use execute() to run a command (e.g, execute('scriptnames')). The output will inserted right after the current cursor.

  2. Use :put= in command mode. In command mode, first input put=, then input the function or execute() command. The output will be put below the current cursor line.

  3. Use redir. We first use redir to redirect the output to a Vim register and then paste the content of the register to current buffer. For functions, use the following

    :redir @a | echo strftime('%c') | redir END<CR>

    For commands, use:

    :redir @a | silent scriptnames |redir END<CR>

This will redirect the output of function strftime() or command scriptnames to register a. To paste the content of register a, use "ap in normal mode.

you can also directly redirect the output to current file:

:redir >> % | silent scriptnames|redir END|edit

, which may be more convenient sometimes.

Using <C-R>= in insert mode is not good. It will write the output line by line, thus are very slow if your output has hundreds or thousands of the lines. The line indentation may also break if you use this method.


Why can’t I search some unicode characters?

For example, given the following text:

ๅ…่ดน	ๅ…่ดน ๐Ÿ†“๏ธ๏ธ

if you move the cursor to the emoji and use ga, the output is:

<๐Ÿ†“> 127379, Hex 0001f193, Octal 370623 < ๏ธ> 65039, Hex fe0f, Octal 177017 < ๏ธ> 65039, Hex fe0f, Octal 177017

It shows that this character is actually made up of three unicode characters! This is called variant form and the unicode character U+FE0F is called a variant selector . There are other forms multi-Unicode-char characters, for example, combining characters, where the character is made up of a base character and one or more decoration characters. In Vim, the main character is called base character and the other character that follows is called composing character.

If you use search the base character or the composing character using their unicode code points, you will find disappointedly that no pattern are found.

It turns out the way we search the characters are wrong. To search the base character, you should follow it by \%C (in no-magic mode) or %C (in magic mode). For example, to search the above base character in the emoji, use

  • Magic mode: \v%U1f193%C
  • No-Magic Mode: \%U1f193\%C

To search the composing character, it is more complicated. You shoud type /<Ctrl-V>ufe0f:

  • <Ctrl-v>u: init unicode input
  • fe0f: the code point of the composing character

Then press Enter to search.


How do I clear message on status line?

If you set the option laststatus to 2, after searching a word using *, you will find that the text is not cleared automatically. To clear the text , you can press Ctrl+L .


How do I use a variable when calling shell command inside Neovim?

You can use :execute, which will evalute the expr followed before executing it. Suppose that you have defined a download URL variable URL, in order to download it using wget inside Neovim, you can use the following command:

:execute '!wget ' . URL

Note the space inside '!wget ' because we use . to concate the arguments behind execute to form an expression. Without the space, the command you are going to run is !wget{the_content_of_URL}, which is apparently wrong.


How do I remove the trailing newline when calling shell command using system() inside Vim?

When I wanted to assign the output of an external shell command to a Vim variable using system() function, I found that a trailing newline character was present in the returned string, which breaks a lot of things. For example,

let python_path = system('which python')

the variable python_path will contain an annoying newline. To remove this newline character, you can use various methods:

  • substitute(): you can replace the trailing newline with an empty string to remove it:

    substitube(python_path, '\n$', '', 'g')
  • string indexing: you can also utlize string indexing (see :h expr-[:]) if you are sure that thre is only one newline at the end of the string:

    " remove last byte from the string
    let new_path=python_path[:-2]


  1. see :h c_CTRL-R for more information. โŽ