Skip to content

Escape from insert mode, terminal mode & more without delay

License

Notifications You must be signed in to change notification settings

TheBlob42/houdini.nvim

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

47 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

🧙 Houdini

Escape insert mode, terminal mode and more with a simple two character mapping

Motivation

Escaping from insert mode back to normal mode is the one action VIM users are (probably) doing most often. But ESC is hard to reach and CTRL-\ CTRL-N is too cumbersome. So Vimmers came up with a little trick to make their lives a little easier

imap jk <ESC>

But behold as this comes with a slight input delay

typing with delay

Neovim is waiting for timeoutlen if you type the second key of your keymapping or if it should actually insert the character instead

houdini removes this delay

typing without delay

But why stop in insert mode?

Compared to other alternatives houdini does also work for other modes that you might want to escape easily using the same mapping and without any visible typing delay

  • insert mode
  • terminal mode
  • command line mode
  • operator mode
  • visual mode
  • select mode
  • (virtual) replace mode
  • ex mode

Installation

Requires at least Neovim version 0.10.0

Install it with your favorite plugin manager and call the setup function

lazy.nvim

{
    'TheBlob42/houdini.nvim',
    config = function()
        require('houdini').setup()
    end
}

Configuration

Call the setup function with your own configuration table to overwrite the defaults

-- default settings
require('houdini').setup {
    mappings = { 'jk' },
    timeout = vim.o.timeoutlen,
    check_modified = true,
    excluded_filetypes = {},
    escape_sequences = {
        ['i']    = '<BS><BS><ESC>',
        ['ic']   = '<BS><BS><ESC>',
        ['ix']   = '<BS><BS><ESC>',
        ['R']    = '<BS><BS><RIGHT><ESC>',
        ['Rc']   = '<BS><BS><ESC>',
        ['Rx']   = '<BS><BS><ESC>',
        ['Rv']   = '<BS><BS><RIGHT><ESC>',
        ['Rvc']  = '<BS><BS><ESC>',
        ['Rvx']  = '<BS><BS><ESC>',
        ['v']    = escape_and_undo,
        ['vs']   = escape_and_undo,
        ['V']    = escape_and_undo,
        ['Vs']   = escape_and_undo,
        ['^V']   = escape_and_undo,
        ['^Vs']  = escape_and_undo,
        ['no']   = escape_and_undo,
        ['nov']  = escape_and_undo,
        ['noV']  = escape_and_undo,
        ['no^V'] = escape_and_undo,
        ['s']  = '<BS><BS><ESC>:u! | call histdel("cmd", -1) | echo ""<CR>',
        ['S']  = '<BS><BS><ESC>:u! | call histdel("cmd", -1) | echo ""<CR>',
        ['^S'] = '<BS><BS><ESC>:u! | call histdel("cmd", -1) | echo ""<CR>',
        ['t'] = '<BS><BS><C-\\><C-n>',
        ['c'] = '<BS><BS><C-c>',
    },
}

Since control characters can not be rendered correctly we depict them here with a ^ prefix
Check lua/houdini.lua for the accurate values that are being used

mappings

A list of all two character mappings that you would like to use for "escaping"

timeout

The time (ms) in which both keys need to be pressed successively to trigger the escape action

check_modified

By default Neovim will always mark a buffer as modified after you've escaped insert mode via houdini. This is because inserting and deleting characters is considered a "change", even if you did not add/delete/modify any text

With this option enabled houdini will compare the changes made after leaving insert mode and suppress the modified status by Neovim, so that it works more like pressing <ESC>

This feature will not intervene in the following cases:

  • the buffer was modified already before entering insert mode
  • you changed some text while entering insert mode using for example S, C or o

excluded_filetypes

A list of filetypes where houdini is not active

escape_sequences

The escape sequences which are used to escape a certain mode

Check :help mode() for a detailed explanation of all the available mode options
See the default configuration for all cases supported "out of the box" (and how its done)

You can provide your own sequence as a string or even use a function for more customization

require('houdini').setup {
    mappings = { 'jk', 'AA', 'II' },
    escape_sequences = {
        ['i'] = function(char_one, char_two, pos, tick)
            local seq = char_one..char_two

            if seq == 'AA' then
                -- jump to the end of the line in insert mode
                return '<BS><BS><End>'
            end

            if seq == 'II' then
                -- jump to the beginning of the line in insert mode
                return '<BS><BS><Home>'
            end

            -- you can also deactivate houdini for certain
            -- occasions by simply returning an empty string
            if vim.opt.filetype:get() == 'TelescopePrompt' then
                return ''
            end

            return '<BS><BS><ESC>'
        end,
        -- set the sequence of a specific mode to `false`
        -- in order to completely disable houdini for this mode
        ['c'] = false,
    },
}

For some more inspiration about custom functions being used for escape sequences also check out the M.escape_and_undo function which is used for the visual and operator modes by default. It shows an example usage of the pos and tick parameters that are passed to any escape function. See also the corresponding help text for some additional information (:h houdini-config-escape-sequences-escape-and-undo)

Alternatives

There are quite a few other plugins with a very similar scope:

They all use the InserCharPre autocommand event to implement their logic which limits their functionality to insert mode only. For houdini we add a custom function via vim.on_key that handles everything. This brings several advantages:

There is also better-escape.nvim which just had a recent rewrite that switched to vim.on_key as well, so the functionality should now be a lot closer with houdini (I have not checked the exact details)

Troubleshooting

General

Most of the work performed by houdini happens inside a custom function registered via vim.on_key. Whenever an error occurs in there this custom function will be removed from being triggered anymore. So in case houdini is suddenly not working anymore it is mostly because of an error. In these cases check :messages for error logs and open an issue to start the investigation

Macros with :normal

In case you encounter issues while executing macros via :normal you might need to update your Neovim version to at least v0.10.0-dev-1902+g184f84234 (20.12.2023) see here for more information

Lightspeed

There is a known issue with lightspeed.nvim which blocks the first <BS> after a jump and changing text. This conflicts with the default escape sequence for insert mode. Fortunately there is a simple workaround to mitigate this problem, see here for more information