fzf.vim

by junegunn

junegunn / fzf.vim

fzf :heart: vim

5.8K Stars 390 Forks Last release: Not found 406 Commits 0 Releases

Available items

No Items, yet!

The developer of this repository has not created any items for sale yet. Need a bug fixed? Help with integration? A different license? Create a request here:

fzf :heart: vim

Things you can do with fzf and Vim.

Rationale

fzf in itself is not a Vim plugin, and the official repository only provides the basic wrapper function for Vim and it's up to the users to write their own Vim commands with it. However, I've learned that many users of fzf are not familiar with Vimscript and are looking for the "default" implementation of the features they can find in the alternative Vim plugins.

This repository is a bundle of fzf-based commands and mappings extracted from my .vimrc to address such needs. They are not designed to be flexible or configurable, and there's no guarantee of backward-compatibility.

Why you should use fzf on Vim

Because you can and you love fzf.

fzf runs asynchronously and can be orders of magnitude faster than similar Vim plugins. However, the benefit may not be noticeable if the size of the input is small, which is the case for many of the commands provided here. Nevertheless I wrote them anyway since it's really easy to implement custom selector with fzf.

Installation

fzf.vim depends on the basic Vim plugin of the main fzf repository, which means you need to set up both "fzf" and "fzf.vim" on Vim. To learn more about fzf/Vim integration, see README-VIM.

Using vim-plug

Plug 'junegunn/fzf', { 'do': { -> fzf#install() } }
Plug 'junegunn/fzf.vim'

fzf#install()
makes sure that you have the latest binary, but it's optional, so you can omit it if you use a plugin manager that doesn't support hooks.

Dependencies

Commands

| Command | List | | --- | --- | |

:Files [PATH]
| Files (runs
$FZF_DEFAULT_COMMAND
if defined) | |
:GFiles [OPTS]
| Git files (
git ls-files
) | |
:GFiles?
| Git files (
git status
) | |
:Buffers
| Open buffers | |
:Colors
| Color schemes | |
:Ag [PATTERN]
| ag search result (
ALT-A
to select all,
ALT-D
to deselect all) | |
:Rg [PATTERN]
| rg search result (
ALT-A
to select all,
ALT-D
to deselect all) | |
:Lines [QUERY]
| Lines in loaded buffers | |
:BLines [QUERY]
| Lines in the current buffer | |
:Tags [QUERY]
| Tags in the project (
ctags -R
) | |
:BTags [QUERY]
| Tags in the current buffer | |
:Marks
| Marks | |
:Windows
| Windows | |
:Locate PATTERN
|
locate
command output | |
:History
|
v:oldfiles
and open buffers | |
:History:
| Command history | |
:History/
| Search history | |
:Snippets
| Snippets (UltiSnips) | |
:Commits
| Git commits (requires fugitive.vim) | |
:BCommits
| Git commits for the current buffer | |
:Commands
| Commands | |
:Maps
| Normal mode mappings | |
:Helptags
| Help tags 1 | |
:Filetypes
| File types
  • Most commands support
    CTRL-T
    /
    CTRL-X
    /
    CTRL-V
    key bindings to open in a new tab, a new split, or in a new vertical split
  • Bang-versions of the commands (e.g.
    Ag!
    ) will open fzf in fullscreen
  • You can set
    g:fzf_command_prefix
    to give the same prefix to the commands
    • e.g.
      let g:fzf_command_prefix = 'Fzf'
      and you have
      FzfFiles
      , etc.

(1:

Helptags
will shadow the command of the same name from pathogen. But its functionality is still available via
call
pathogen#helptags()
. )

Customization

Global options

Every command in fzf.vim internally calls

fzf#wrap
function of the main repository which supports a set of global option variables. So please read through README-VIM to learn more about them.

Preview window

Some commands will show the preview window on the right. You can customize the behavior with

g:fzf_preview_window
. Here are some examples:
" This is the default option:
"   - Preview window on the right with 50% width
"   - CTRL-/ will toggle preview window.
" - Note that this array is passed as arguments to fzf#vim#with_preview function.
" - To learn more about preview window options, see `--preview-window` section of `man fzf`.
let g:fzf_preview_window = ['right:50%', 'ctrl-/']

" Preview window on the upper side of the window with 40% height, " hidden by default, ctrl-/ to toggle let g:fzf_preview_window = ['up:40%:hidden', 'ctrl-/']

" Empty value to disable preview window altogether let g:fzf_preview_window = []

Command-local options

A few commands in fzf.vim can be customized with global option variables shown below.

" [Buffers] Jump to the existing window if possible
let g:fzf_buffers_jump = 1

" [[B]Commits] Customize the options used by 'git log': let g:fzf_commits_log_options = '--graph --color=always --format="%C(auto)%h%d %s %C(black)%C(bold)%cr"'

" [Tags] Command to generate tags file let g:fzf_tags_command = 'ctags -R'

" [Commands] --expect expression for directly executing the command let g:fzf_commands_expect = 'alt-enter,ctrl-x'

Advanced customization

Vim functions

Each command in fzf.vim is backed by a Vim function. You can override a command or define a variation of it by calling its corresponding function.

| Command | Vim function | | --- | --- | |

Files
|
fzf#vim#files(dir, [spec dict], [fullscreen bool])
| |
GFiles
|
fzf#vim#gitfiles(git_options, [spec dict], [fullscreen bool])
| |
GFiles?
|
fzf#vim#gitfiles('?', [spec dict], [fullscreen bool])
| |
Buffers
|
fzf#vim#buffers([spec dict], [fullscreen bool])
| |
Colors
|
fzf#vim#colors([spec dict], [fullscreen bool])
| |
Rg
|
fzf#vim#grep(command, [has_column bool], [spec dict], [fullscreen bool])
| | ... | ... |

(We can see that the last two optional arguments of each function are identical. They are directly passed to

fzf#wrap
function. If you haven't read README-VIM already, please read it before proceeding.)

Example: Customizing
Files
command

This is the default definition of

Files
command:
command! -bang -nargs=? -complete=dir Files call fzf#vim#files(, 0)

Let's say you want to a variation of it called

ProjectFiles
that only searches inside
~/projects
directory. Then you can do it like this:
command! -bang ProjectFiles call fzf#vim#files('~/projects', 0)

Or, if you want to override the command with different fzf options, just pass a custom spec to the function.

command! -bang -nargs=? -complete=dir Files
    \ call fzf#vim#files(, {'options': ['--layout=reverse', '--info=inline']}, 0)

Want a preview window?

command! -bang -nargs=? -complete=dir Files
    \ call fzf#vim#files(, {'options': ['--layout=reverse', '--info=inline', '--preview', 'cat {}']}, 0)

It kind of works, but you probably want a nicer previewer program than

cat
. fzf.vim ships a versatile preview script you can readily use. It internally executes bat for syntax highlighting, so make sure to install it.
command! -bang -nargs=? -complete=dir Files
    \ call fzf#vim#files(, {'options': ['--layout=reverse', '--info=inline', '--preview', '~/.vim/plugged/fzf.vim/bin/preview.sh {}']}, 0)

However, it's not ideal to hard-code the path to the script which can be different in different circumstances. So in order to make it easier to set up the previewer, fzf.vim provides

fzf#vim#with_preview
helper function. Similarly to
fzf#wrap
, it takes a spec dictionary and returns a copy of it with additional preview options.
command! -bang -nargs=? -complete=dir Files
    \ call fzf#vim#files(, fzf#vim#with_preview({'options': ['--layout=reverse', '--info=inline']}), 0)

You can just omit the spec argument if you only want the previewer.

command! -bang -nargs=? -complete=dir Files
    \ call fzf#vim#files(, fzf#vim#with_preview(), 0)

Example:
git grep
wrapper

The following example implements

GGrep
command that works similarly to predefined
Ag
or
Rg
using
fzf#vim#grep
.
  • The second argument to
    fzf#vim#grep
    is 0 (false), because
    git grep
    does not print column numbers.
  • We set the base directory to git root by setting
    dir
    attribute in spec dictionary.
  • The preview script supports
    grep
    format (
    FILE_PATH:LINE_NO:...
    ), so we can just wrap the spec with
    fzf#vim#with_preview
    as before to enable previewer.
command! -bang -nargs=* GGrep
  \ call fzf#vim#grep(
  \   'git grep --line-number -- '.shellescape(), 0,
  \   fzf#vim#with_preview({'dir': systemlist('git rev-parse --show-toplevel')[0]}), 0)

Example:
Rg
command with preview window

You can see the definition of

Rg
command with
:command Rg
. With the information, you can redefine it with the preview window enabled. In this case, we're only interested in setting up the preview window, so we will omit the spec argument to
fzf#vim#preview
.
command! -bang -nargs=* Rg
  \ call fzf#vim#grep(
  \   'rg --column --line-number --no-heading --color=always --smart-case -- '.shellescape(), 1,
  \   fzf#vim#with_preview(), 0)

Example: Advanced ripgrep integration

In the default implementation of

Rg
, ripgrep process starts only once with the initial query (e.g.
:Rg foo
) and fzf filters the output of the process.

This is okay in most cases because fzf is quite performant even with millions of lines, but we can make fzf completely delegate its search responsibliity to ripgrep process by making it restart ripgrep whenever the query string is updated. In this scenario, fzf becomes a simple selector interface rather than a "fuzzy finder".

  • We will name the new command all-uppercase
    RG
    so we can still access the default version.
  • --bind 'change:reload:rg ... {q}'
    will make fzf restart ripgrep process whenever the query string, denoted by
    {q}
    , is changed.
  • With
    --phony
    option, fzf will no longer perform search. The query string you type on fzf prompt is only used for restarting ripgrep process.
  • Also note that we enabled previewer with
    fzf#vim#with_preview
    .
function! RipgrepFzf(query, fullscreen)
  let command_fmt = 'rg --column --line-number --no-heading --color=always --smart-case -- %s || true'
  let initial_command = printf(command_fmt, shellescape(a:query))
  let reload_command = printf(command_fmt, '{q}')
  let spec = {'options': ['--phony', '--query', a:query, '--bind', 'change:reload:'.reload_command]}
  call fzf#vim#grep(initial_command, 1, fzf#vim#with_preview(spec), a:fullscreen)
endfunction

command! -nargs=* -bang RG call RipgrepFzf(, 0)

Mappings

| Mapping | Description | | --- | --- | |

(fzf-maps-n)
| Normal mode mappings | |
(fzf-maps-i)
| Insert mode mappings | |
(fzf-maps-x)
| Visual mode mappings | |
(fzf-maps-o)
| Operator-pending mappings | |
(fzf-complete-word)
|
cat /usr/share/dict/words
| |
(fzf-complete-path)
| Path completion using
find
(file + dir) | |
(fzf-complete-file)
| File completion using
find
| |
(fzf-complete-line)
| Line completion (all open buffers) | |
(fzf-complete-buffer-line)
| Line completion (current buffer only) |
" Mapping selecting mappings
nmap  (fzf-maps-n)
xmap  (fzf-maps-x)
omap  (fzf-maps-o)

" Insert mode completion imap (fzf-complete-word) imap (fzf-complete-path) imap (fzf-complete-line)

Completion functions

| Function | Description | | --- | --- | |

fzf#vim#complete#path(command, [spec])
| Path completion | |
fzf#vim#complete#word([spec])
| Word completion | |
fzf#vim#complete#line([spec])
| Line completion (all open buffers) | |
fzf#vim#complete#buffer_line([spec])
| Line completion (current buffer only) |
" Path completion with custom source command
inoremap   fzf#vim#complete#path('fd')
inoremap   fzf#vim#complete#path('rg --files')

" Word completion with custom spec with popup layout option inoremap fzf#vim#complete#word({'window': { 'width': 0.2, 'height': 0.9, 'xoffset': 1 }})

Custom completion

fzf#vim#complete
is a helper function for creating custom fuzzy completion using fzf. If the first parameter is a command string or a Vim list, it will be used as the source.
" Replace the default dictionary completion with fzf-based fuzzy completion
inoremap   fzf#vim#complete('cat /usr/share/dict/words')

For advanced uses, you can pass an options dictionary to the function. The set of options is pretty much identical to that for

fzf#run
only with the following exceptions:
  • reducer
    (funcref)
    • Reducer transforms the output lines of fzf into a single string value
  • prefix
    (string or funcref; default:
    \k*$
    )
    • Regular expression pattern to extract the completion prefix
    • Or a function to extract completion prefix
  • Both
    source
    and
    options
    can be given as funcrefs that take the completion prefix as the argument and return the final value
  • sink
    or
    sink*
    are ignored
" Global line completion (not just open buffers. ripgrep required.)
inoremap   fzf#vim#complete(fzf#wrap({
  \ 'prefix': '^.*$',
  \ 'source': 'rg -n ^ --color always',
  \ 'options': '--ansi --delimiter : --nth 3..',
  \ 'reducer': { lines -> join(split(lines[0], ':\zs')[2:], '') }}))

Reducer example

function! s:make_sentence(lines)
  return substitute(join(a:lines), '^.', '\=toupper(submatch(0))', '').'.'
endfunction

inoremap fzf#vim#complete({ \ 'source': 'cat /usr/share/dict/words', \ 'reducer': function('make_sentence'), \ 'options': '--multi --reverse --margin 15%,0', \ 'left': 20})

Status line of terminal buffer

When fzf starts in a terminal buffer (see fzf/README-VIM.md), you may want to customize the statusline of the containing buffer.

Hide statusline

autocmd! FileType fzf set laststatus=0 noshowmode noruler
  \| autocmd BufLeave  set laststatus=2 showmode ruler

Custom statusline

function! s:fzf_statusline()
  " Override statusline as you like
  highlight fzf1 ctermfg=161 ctermbg=251
  highlight fzf2 ctermfg=23 ctermbg=251
  highlight fzf3 ctermfg=237 ctermbg=251
  setlocal statusline=%#fzf1#\ >\ %#fzf2#fz%#fzf3#f
endfunction

autocmd! User FzfStatusLine call fzf_statusline()

License

MIT

We use cookies. If you continue to browse the site, you agree to the use of cookies. For more information on our use of cookies please see our Privacy Policy.