*diffchar.txt* Highlight the exact differences, based on characters and words > ____ _ ____ ____ _____ _ _ _____ ____ | | | || || || || | | || _ || _ | | _ || || __|| __|| || | | || | | || | || | | | || || |__ | |__ | __|| |_| || |_| || |_||_ | |_| || || __|| __|| | | || || __ | | || || | | | | |__ | _ || _ || | | | |____| |_||_| |_| |_____||_| |_||_| |_||_| |_| < Last Change: 2025/10/01 Version: 10.0 (on or after vim 9.0 and nvim 0.7.0) Author: Rick Howe (Takumi Ohtani) Copyright: (c) 2014-2025 Rick Howe License: MIT ----------------------------------------------------------------------------- INTRODUCTION *diffchar* This plugin has been developed in order to make diff mode more useful. Vim highlights all the text in between the first and last different characters on a changed line. But this plugin will find the exact differences between them, character by character - so called DiffChar. For example, in diff mode: (`^`:`DiffText`, |#|:|DiffAdd|, |@|:|DiffDelete|) +-------------------------------------------------+ |The `quick brown fox jumps over the lazy` dog. | | `^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^` | |~ | +-------------------------------------------------+ |The `lazy fox jumps over the quick brown` dog. | | `^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^` | |~ | +-------------------------------------------------+ This plugin will exactly show the changed/added/deleted units: +-------------------------------------------------+ |The `quick` |brown| fox jumps over the `lazy` dog. | | `^^^^^` |######| `^^^^` |@| | |~ | +-------------------------------------------------+ |The `lazy` fox jumps over the `quick` |brown| dog. | | `^^^^` |@| `^^^^^` |######| | |~ | +-------------------------------------------------+ Sync with diff mode ~ This plugin will synchronously show/reset the highlights of the exact differences as soon as the diff mode begins/ends. And the exact differences will be kept updated while editing. Note that this plugin does nothing if an "inline" item is set in the 'doffopt' option and its value is other than "simple". On plugin loading, the item is removed only if set by default so that this plugin can work. Diff unit ~ This plugin shows the diffs based on a |g:DiffUnit|. Its default is 'Word1' and it handles a \w\+ word and a \W character as a diff unit. There are other types of word provided and you can also set 'Char' to compare character by character. In addition, you can specify one or more diff unit delimiters, such as comma (','), colon (':'), tab ("\t"), and HTML tag symbols ('<' and '>'), and also specify a custom pattern in the |g:DiffUnit|. Diff matching colors ~ In diff mode, the corresponding |hl-DiffChange| lines are compared between two windows. As a default, all the changed units are highlighted with |hl-DiffText|. You can set |g:DiffColors| to use more than one matching color to make it easy to find the corresponding units between two windows. The number of colors depends on the color scheme. In addition, an added unit is always highlighted with |hl-DiffAdd| and the position of the corresponding deleted unit is shown with bold/underline or a virtual blank column, depending on a |g:DiffDelPosVisible|. Diff pair visible ~ While showing the exact differences, when the cursor is moved on a diff unit, you can see its corresponding unit highlighted with |hl-Cursor|, |hl-TermCursor|, or similar one in another window, based on a |g:DiffPairVisible|. If you change its default, the corresponding unit is echoed in the command line or displayed in a popup/floating window just below the cursor position or at the mouse position. Those options take effect on `:diffupdate` command as well. Jump to next/prev diff unit ~ You can use `]b` or `]e` to jump cursor to start or end position of the next diff unit, and `[b` or `[e` to the start or end position of the previous unit. Get/put a diff unit ~ Like line-based `:diffget`/`:diffput` and `do`/`dp` vim commands, you can use `g` and `p` commands in normal mode to get and put each diff unit, where the cursor is on, between 2 buffers and undo its difference. Those keymaps are configurable in your vimrc and so on. Check diff lines locally ~ When the diff mode begins, this plugin locally checks the |hl-DiffChange| lines in the limited range of the current visible and its upper/lower lines of a window. And each time a cursor is moved on to another range upon scrolling or searching, those diff lines will be checked in that range. Which means, independently of the file size, the number of lines to be checked and then the time consumed are always constant. Tab page individual ~ This plugin works on each tab page individually. You can use a tab page variable (t:), instead of a global one (g:), to specify different options on each tab page. Note that this plugin can not handle more than two diff mode windows in a tab page. If it would happen, to prevent any trouble, all the highlighted units are to be reset in the tab page. Follow 'diffopt' option ~ This plugin supports "icase", "iwhite", "iwhiteall", and "iwhiteeol" in the 'diffopt' option. In addition, when "indent-heuristic" is specified, positioning of the added/deleted diff units is adjusted to reduce the number of diff hunks and make them easier to read. Comparison algorithm ~ To find the exact differences, this plugin uses "An O(NP) Sequence Comparison Algorithm" developed by S.Wu, et al., which always finds an optimum sequence. But it takes time to check a long and dissimilar line. To improve the performance, the algorithm is also implemented in Vim9 script. In addition, if available, this plugin uses a builtin diff function (|diff()| in vim patch-9.1.0071 and Lua |vim.diff()| in nvim 0.6.0) and makes it much faster. See also ~ There are other diff related plugins available: - |spotdiff.vim|: A range and area selectable `:diffthis` to compare partially (https://github.com/rickhowe/spotdiff.vim) - |wrapfiller|: Align each wrapped line virtually between windows (https://github.com/rickhowe/wrapfiller) - |difffilter|: Selectively compare lines as you want in diff mode (https://github.com/rickhowe/difffilter) - |diffunitsyntax|: Highlight word or character based diff units in diff format (https://github.com/rickhowe/diffunitsyntax) ----------------------------------------------------------------------------- OPTIONS *diffchar-options* |g:DiffUnit|, |t:DiffUnit| A type of diff unit 'Char' : any single character 'Word1' : \w\+ word and any \W single character (default) 'Word2' : non-space and space words 'Word3' : \< or \> character class boundaries (set by 'iskeyword') 'word' : see `word` 'WORD' : see `WORD` '[{del}]' : one or more diff unit delimiters (e.g. "[,:\t<>]") '/{pat}/' : a pattern to split into diff units (e.g. '/.\{4}\zs/') |g:DiffColors|, |t:DiffColors| Matching colors for changed units 0 : |hl-DiffText| (default) 1 : |hl-DiffText| + a few (3, 4, ...) highlight groups 2 : |hl-DiffText| + several (7, 8, ...) highlight groups 3 : |hl-DiffText| + many (11, 12, ...) highlight groups 100 : all available highlight groups in random order [{hlg}] : a list of your favorite highlight groups |g:DiffPairVisible|, |t:DiffPairVisible| Visibility of corresponding diff units 0 : disable 1 : highlight with |hl-Cursor| (default) 2 : highlight with |hl-Cursor| + echo in the command line 3 : highlight with |hl-Cursor| + popup/floating window at cursor position 4 : highlight with |hl-Cursor| + popup/floating window at mouse position |g:DiffDelPosVisible|, |t:DiffDelPosVisible| Visibility of the position of deleted units 0 : disable 1 : highlight previous/next chars of a deleted unit in bold/underline (default if inline |virtual-text| is not available) 2 : virtually show a blank column (set by "space" item in 'listchars') with |hl-DiffDelete| (default if inline |virtual-text| is available) ----------------------------------------------------------------------------- KEYMAPS *diffchar-keymaps* JumpDiffCharPrevStart (default: `[b`) Jump cursor to the start position of the previous diff unit JumpDiffCharNextStart (default: `]b`) Jump cursor to the start position of the next diff unit JumpDiffCharPrevEnd (default: `[e`) Jump cursor to the end position of the previous diff unit JumpDiffCharNextEnd (default: `]e`) Jump cursor to the end position of the next diff unit GetDiffCharPair (default: `g`) Get a corresponding diff unit from another buffer to undo difference PutDiffCharPair (default: `p`) Put a corresponding diff unit to another buffer to undo difference ----------------------------------------------------------------------------- CHANGE HISTORY *diffchar-history* 10.0 * Added |g:DiffDelPosVisible| option to show the position of deleted units. * Implemented to remove an "inline" item from the 'diffopt' option on initial loading so that this plugin can work. * Made this plugin available on or after vim 9.0 and nvim 0.7.0. 9.9 * Improved to remain previous diff lines highlighted upon scrolling if possible and not to check them later again. * Implemented a few more functions in Vim9 script to make them faster. 9.8 * Improved to follow "linematch", which is run to realign diff lines when displayed on screen, if specified in the 'diffopt' option (default on nvim 0.11). Accordingly, fixed errors such as E684 and E716 on nvim. * Changed to disable this plugin to avoid a conflict with character/word-wise "inline" diff (patch-9.1.1243), if specified in the 'diffopt' option. * Changed to overwrite nvim default mappings to set plugin specific ones on plugin loading. As of nvim 0.11, `[b` and `]b` keys are duplicated. * Fixed to handle tab and space as a same whitespace character if "iwhite" or "iwhiteall" are specified in the 'diffopt' options. 9.7 * Implemented to use a new builtin |diff()| function (available on patch-9.1.0071) to compare diff units and make it faster in vim. 9.6, 9.61 * Changed to locally but not incrementally check the limited number of the |hl-DiffChange| lines upon scrolling or searching. * Fixed not to use |hl-DiffText| as the first highlight group if [{hlg}] is specified in the |g:DiffColors| option. 9.5 * Improved not to update diff unit highlighting on unchanged diff lines on the DiffUpdated event. * Fixed the E421 error when 'termguicolors' is off on nvim. 9.4 * Implemented to use a builtin Lua |vim.diff()| function to compare diff units and make it faster in nvim. 9.3 * Improved to follow the 'wincolor' option and show the colors accordingly. * Fixed to work on some color scheme in which the "g:colors_name" variable is different from its file name. 9.2 * Fixed the plugin error when 3 or more windows turn to be diff mode. * Fixed the issue which the new diff lines can not be incrementally found upon scrolling on patch-9.0.0913 or later. * Made it faster to find diff lines in a file with many lines folded. 9.1 * Added vim original 'word' and 'WORD' units, one or more unit delimiters, and a custom pattern to split into diff units in |g:DiffUnit| option. * Improved to redraw diff units whenever the ColorScheme event happens when multiple matching colors are being used (0 < |g:DiffColors|). * Changed the category of |g:DiffColors| option as a few, several, and many numbers of matching colors, depending on the loaded color scheme. * Added a list of your favorite highlight groups in |g:DiffColors| option. 9.01 * Fixed to work on a |hl-Diff| highlighting group even if it is linked. 9.0 * Enhanced to make diff units easier to read when "indent-heuristic" is specified in the 'diffopt' option. * Improved to update diff units using `:diffupdate` command when |g:DiffUnit|, |g:DiffColors|, and |g:DiffPairVisible| options are modified. * Updated to check a new WinScrolled event (patch-8.2.4713) to incrementally find scrolled diff lines. * Implemented the comparison algorithm and diff unit highlighting in Vim9 script and made them 10 times faster on patch-8.2.3965 or later. * Made this plugin available on or after patch-8.1.1418 and nvim-0.5.0. vim:tw=78:ts=8:ft=help:norl: