vim match-up
match-up is a drop-in replacement for the vim plugin matchit.vim.
match-up aims to enhance all of matchit's features, fix a number of its
deficiencies and bugs, and add a few totally new features. It also
replaces the standard plugin matchparen, allowing all of matchit's words
to be highlighted along with the matchpairs ((){}[]).
A major goal of this project is to keep a modern and modular code base. Contributions are welcome!
Screenshot
Table of contents
Overview
See detailed feature documentation for more information. This plugin:
-
Extends vim's
%motion to language-specific words. The following vim file type plugins currently provide support for match-up:abaqus, ada, aspvbs, c, clojure, cobol, config, context, csc, csh, dtd, dtrace, eiffel, eruby, falcon, fortran, framescript, haml, hamster, hog, html, ishd, j, jsp, kconfig, liquid, lua, make, matlab, mf, mp, ocaml, pascal, pdf, perl, php, plaintex, postscr, ruby, sh, spec, sql, tex, vb, verilog, vhdl, vim, xhtml, xml, zimbu, zsh
Note: match-up uses the same
b:match_wordsas matchit. -
Adds motions
g%,[%,]%, andz%. -
Combines these motions into convenient text objects
i%anda%. -
Highlights symbols and words under the cursor which
%can work on, and highlights matching symbols and words. Now you can easily tell where%will jump to.
Installation
If you use vim-plug, then add the following line to your vimrc file:
Plug 'andymass/vim-matchup'and then use :PlugInstall. Or, you can use any other plugin manager such as
vundle,
dein,
neobundle, or
pathogen.
match-up should automatically disable matchit and matchparen, but if you are still having trouble, try placing this near the top of your vimrc:
let g:loaded_matchit = 1See Interoperability for more information about working together with other plugins.
Features
| feature | match-up | matchit | matchparen | |
|---|---|---|---|---|
| (a.1) | jump between matching words | |||
| (a.2) | jump to open & close words | |||
| (a.3) | jump inside | |||
| (b.1) | full set of text objects | |||
| (c.1) | highlight (), [], & {} |
|||
| (c.2) | highlight all matches | |||
| (c.3) | display matches off-screen | |||
| (d.1) | parallel transmutation | |||
| (e.1) | modern, modular coding style | |||
| (e.2) | actively developed |
Legend:
Detailed feature documentation
What do we mean by open, close, mid? This depends on the specific file
type and is configured through the variable b:match_words. Here are a
couple examples:
vim-script
if l:x == 1
call one()
elseif l:x == 2
call two()
else
call three()
endifFor the vim-script language, match-up understands the words if,
else, elseif, endif and that they form a sequential construct. The
"open" word is if, the "close" word is endif, and the "mid"
words are else and elseif. The if/endif pair is called an
"open-to-close" block and the if/else, else/elsif, and
elseif/endif are called "any" blocks.
C, C++
#if 0
#else
#endif
void some_func() {
if (true) {
one();
} else if (false && false) {
two();
} else {
three();
}
}Since in C and C++, blocks are delimited using braces ({ & }),
match-up will recognize { as the open word and } as the close word.
It will ignore the if and else if because they are not defined in
vim's C file type plugin.
On the other hand, match-up will recognize the #if, #else, #endif
preprocessor directives.
(a.1) jump between matching words
%go forwards to next matching word. If at a close word, cycle back to the corresponding open word.{count}%forwards{count}times. Requires{count} <= g:matchup_motion_override_Npercent. For larger{count},{count}%goes to the{count}percentage in the file.g%go backwards to[count]th previous matching word. If at an open word, cycle around to the corresponding close word.
(a.2) jump to open and close words
[%go to[count]th previous outer open word. Allows navigation to the start of blocks surrounding the cursor. This is similar to vim's built-in[(and[{and is an exclusive motion.]%go to[count]th next surrounding close word. This is an exclusive motion.
(a.3) jump inside
z%go to inside[count]th nearest inner contained block. This is an exclusive motion when used with operators, except it eats whitespace. For example, whereβis the cursor position,
β call somefunction(param1, param2)dz% produces
param1, param2)but in
β call somefunction( param1, param2)dz% also produces
param1, param2)(b.1) full set of text objects
-
i%the inside of an any block -
1i%the inside of an open-to-close block -
{count}i%If count is greater than 1, the inside of the{count}th surrounding open-to-close block -
a%an any block. -
1a%an open-to-close block. Includes mids but does not include open and close words. -
{count}a%if{count}is greater than 1, the{count}th surrounding open-to-close block.
See here for some examples and important special cases.
(c.1) highlight (), [], and {}
match-up emulates vim's matchparen to highlight the symbols contained
in the matchpairs setting.
(c.2) highlight all matches
To disable match highlighting, let g:matchup_matchparen_enabled = 0.
If this option is set before the plugin is loaded, it will not disable
the built-in matchparen plugin. See here for
other related options.
(c.3) display matches off screen
If a open or close which would have been highlighted is on a line positioned outside the current window, the match is shown in the status line. If both the open and close match are off-screen, the close match is preferred.
(d.1) parallel transmutation
To enable, use let g:matchup_transmute_enabled = 1.
In insert mode, after changing text inside a word, matching words will be changed in parallel. As an example,
<pre>
text
</pre>Changing pre to div and leaving insert mode will produce:
<div>
text
</div>Note: this currently only works for match words which define a backref
relation like \1.
Parallel transmutation requires the matchparen module to be enabled.
Inclusive and exclusive motions
In vim, character motions following operators (such as d for delete
and c for change) are either inclusive or exclusive. This means
they either include the ending position or not. Here, "ending position"
means the line and column closest to the end of the buffer of the region
swept over by the motion. match-up is designed so that d]% inside a set
of parenthesis behaves exactly like d]), except generalized to words.
Put differently, forward exclusive motions will not include the close
word. In this example, where β is the cursor position,
if βx | continue | endifpressing d]% will produce (cursor on the e)
if endifTo include the close word, use either dv]% or vd]%. This is also
compatible with vim's d]) and d]}.
Operators over backward exclusive motions will instead exclude the position the cursor was on before the operator was invoked. For example, in
if βx | continue | endifpressing d[% will produce
βx | continue | endifThis is compatible with vim's d[( and d[{.
Unlike ]%, % is an inclusive motion. As a special case for the
d (delete) operator, if d% leaves behind lines white-space, they will
be deleted also. In effect, it will be operating line-wise. As an
example, pressing d% will leave behind nothing.
β(
)
To operate character-wise in this situation, use dv% or vd%.
This is vim compatible with the built-in d% on matchpairs.
Line-wise operator/text-object combinations
Normally, the text objects i% and a% work character-wise. However,
there are some special cases. For certain operators combined with i%,
under certain conditions, match-up will effectively operate line-wise
instead. For example, in
if condition
βcall one()
call two()
endifpressing di% will produce
if condition
endifeven though deleting condition would be suggested by the object i%.
The intention is to make operators more useful in some cases. The
following rules apply:
- The operator must be listed in
g:matchup_text_obj_linewise_operators. By default this isdandy(e.g.,di%andya%). - The outer block must span multiple lines.
- The open and close delimiters must be more than one character long. In
particular,
di%involving a(...)block will not be subject to these special rules.
To prevent this behavior for a particular sequence dvi% or vdi%.
To disable this entirely, remove the operator from the following variable,
let g:matchup_text_obj_linewise_operators = [ 'y' ]Note: unlike vim's built-in i), ab, etc., i% does not make an
existing visual mode character-wise.
A second special case involves da%. In this example,
if condition
βcall one()
call two()
endifpressing da% will delete all four lines and leave no white-space. This
is vim compatible with da(, dab, etc.
Options
To disable the plugin entirely,
let g:matchup_enabled = 0default: 1
To disable a particular module,
let g:matchup_matchparen_enabled = 0
let g:matchup_motion_enabled = 0
let g:matchup_text_obj_enabled = 0defaults: 1
To enable the experimental transmute module,
let g:matchup_transmute_enabled = 1default: 0
To configure the number of lines to search in either direction while using
motions and text objects. Does not apply to match highlighting
(see g:matchup_matchparen_stopline instead).
let g:matchup_delim_stopline = 1500default: 1500
Variables
match-up understands the following variables from matchit.
b:match_wordsb:match_skipb:match_ignorecase
These are set in the respective ftplugin files. They may not exist for
every file type. To support a new file type, create a file
after/ftplugin/{filetype}.vim which sets them appropriately.
Module matchparen
The matchparen module can be disabled on a per-buffer basis
let b:matchup_matchparen_enabled = 0default: 1
If this module is disabled on a particular buffer, match-up will still
fall-back to the vim standard plugin matchit, which will highlight
matchpairs such as (), [], & {}. To disable this,
let b:matchup_matchparen_fallback = 0default: 1
A common usage is to automatically disable matchparen for particular file types;
augroup matchup_matchparen_disable_ft
autocmd!
autocmd FileType tex let [b:matchup_matchparen_fallback,
\ b:matchup_matchparen_enabled] = [0, 0]
augroup ENDWhether to highlight known words even if there is no match:
let g:matchup_matchparen_singleton = 1default: 0
Whether to replace the statusline for off-screen matches:
let g:matchup_matchparen_status_offscreen = 0If a match is off of the screen, the line belonging to that match will be displayed syntax-highlighted in the status line along with the line number (if line numbers are enabled). If the match is above the screen border, an additional Ξ symbol will be shown to indicate that the matching line is really above the cursor line.
default: 1
The number of lines to search in either direction while highlighting matches. Set this conservatively since high values may cause performance issues.
let g:matchup_matchparen_stopline = 400 " for match highlighting onlydefault: 400
highlighting timeouts
Adjust timeouts in milliseconds for matchparen highlighting:
let g:matchup_matchparen_timeout = 300
let g:matchup_matchparen_insert_timeout = 60default: 300, 60
deferred highlighting
Deferred highlighting improves cursor movement performance (for example,
when using hjkl) by delaying highlighting for a short time and waiting
to see if the cursor continues moving;
let g:matchup_matchparen_deferred = 1default: 0 (disabled)
Note: this feature is only available if your vim version has timers and
the function timer_pause, version 7.4.2180 and after. For neovim, this
will only work in nvim-0.2.1 and after.
Adjust delays in milliseconds for deferred highlighting:
let g:matchup_matchparen_deferred_show_delay = 50
let g:matchup_matchparen_deferred_hide_delay = 700default: 50, 700
Note: these delays cannot be changed dynamically and should be configured before the plugin loads (e.g., in your vimrc).
highlight surrounding
To highlight the surrounding delimiters until the cursor moves, use a map such as the following
nmap <silent> <F7> <plug>(matchup-hi-surround)There is no default map for this feature.
Module motion
In vim, {count}% goes to the {count} percentage in the file.
match-up overrides this motion for small {count} (by default, anything
less than 7). To allow {count}% for {count} up to 11,
g:matchup_motion_override_Npercent = 11To disable this feature, and restore vim's default {count}%,
g:matchup_motion_override_Npercent = 0default: 6
If enabled, cursor will land on the end of mid and close words while
moving downwards (%/]%). While moving upwards (g%, [%) the cursor
will land on the beginning. To disable,
let g:matchup_motion_cursor_end = 0default: 1
Module text_obj
Modify the set of operators which may operate line-wise
let g:matchup_text_obj_linewise_operators' = ['d', 'y']default: ['d', 'y']
Module transmute
Options planned.
FAQ
-
match-up doesn't work
This plugin requires at least vim 7.4. It should work in vim 7.4.898 but at least vim 7.4.1689 is better. I recommend using the most recent version of vim if possible.
If you have issues, please tell me your vim version and error messages. Try updating vim and see if the problem persists.
-
Why does jumping not work for construct X in language Y?
Please open a new issue
-
Highlighting is not correct for construct X
match-up uses matchit's filetype-specific data, which may not give enough information to create proper highlights. To fix this, you may need to modify
b:match_words.For help, please open a new issue and be a specific as possible.
-
I'm having performance problems
match-up aims to be as fast as possible, but highlighting matching words can be intensive and may be slow on less powerful machines. There are a few things you can try to improve performance:
- Update to a recent version of vim. Newer versions are faster, more extensively tested, and better supported by match-up.
- Try deferred highlighting, which delays highlighting until the cursor is stationary to improve cursor movement performance.
- Lower the highlighting timeouts. Note that if highlighting takes longer than the timeout, highlighting will not be attempted again until the cursor moves.
If are having any other performance issues, please open a new issue and report the output of
:MatchupShowTimes. -
Why is there a weird entry on the status line?
This is a feature which helps you see matches that are outside of the vim screen, similar to some IDEs. If you wish to disable it, use
let g:matchup_matchparen_status_offscreen = 0
-
Matching does not work when lines are too far apart.
The number of search lines is limited for performance reasons. You may increase the limits with the following options:
let g:matchup_delim_stopline = 1500 " generally let g:matchup_matchparen_stopline = 400 " for match highlighting only
-
The maps
1i%and1a%are difficult to press.You may use the following maps
I%andA%for convenience:function! s:matchup_convenience_maps() xnoremap <sid>(std-I) I xnoremap <sid>(std-A) A xmap <expr> I mode()=='<c-v>'?'<sid>(std-I)':(v:count?'':'1').'i' xmap <expr> A mode()=='<c-v>'?'<sid>(std-A)':(v:count?'':'1').'a' for l:v in ['', 'v', 'V', '<c-v>'] execute 'omap <expr>' l:v.'I%' "(v:count?'':'1').'".l:v."i%'" execute 'omap <expr>' l:v.'A%' "(v:count?'':'1').'".l:v."a%'" endfor endfunction call s:matchup_convenience_maps()
Note: this is not compatible with the plugin targets.vim.
-
How can I contribute?
Read the contribution guidelines and issue template. Be as precise and detailed as possible when submitting issues and pull requests.
Interoperability
vimtex, for LaTeX documents
By default, match-up will be disabled automatically for tex files when vimtex is detected. To enable match-up for tex files, use
let g:matchup_override_vimtex = 1match-up's matching engine is more advanced than vimtex's and supports
middle delimiters such as \middle| and \else. The exact set of
delimiters recognized may differ between the two plugins. For example,
the mappings da% and dad will not always match, particularly if you
have customized vimtex's delimiters.
Matchit
matchit.vim should not be loaded. If it is loaded, it must be loaded after match-up (in this case, matchit.vim will be disabled). Note that some plugins, such as vim-sensible, load matchit.vim so these must also be initialized after match-up.
Matchparen emulation
match-up loads matchparen if it is not already loaded. Ordinarily, match-up
disables matchparen's highlighting and emulates it to highlight the symbol
contained in the 'matchpairs' option (by default (), [], and {}). If match-up
is disabled per-buffer using b:matchup_matchparen_enabled, match-up will use
matchparen instead of its own highlighting. See b:matchup_matchparen_fallback
for more information.
Acknowledgments
Origins
match-up was originally based on @lervag's
vimtex. The concept and style of this plugin and its development are
heavily influenced by vimtex.
Other inspirations
- matchit
- matchparen
- MatchTag
- MatchTagAlways
- vim-endwise
- auto-pairs
- delimitMate
- splitjoin.vim
- vim-surround
- vim-sandwich
Development
Reporting problems
This is a brand new plugin and there are likely to be many bugs. Thorough issue reports are encouraged. Please read the issue template first. Be as precise and detailed as possible when submitting issues.
Feature requests are also welcome.
Contributing
Please read the contribution guidelines before contributing.
Todo list
- complete parallel transmutation in an efficient way.
- add screenshots and animations
- support for fenced code possible?
- add file type
quirksmodule - investigate whether
&selection/&virtualeditoptions are important - can match-up be integrated with vim-surround?
- support python (like python_match.vim)

Formed in 2009, the Archive Team (not to be confused with the archive.org Archive-It Team) is a rogue archivist collective dedicated to saving copies of rapidly dying or deleted websites for the sake of history and digital heritage. The group is 100% composed of volunteers and interested parties, and has expanded into a large amount of related projects for saving online and digital history.


