![]() # Add AnyBrackets text object for Vim mode ## Overview This PR introduces a new text object `AnyBrackets` that allows operations on the closest matching pair of brackets, regardless of the bracket type. This enhances the editing experience by reducing the need to identify specific bracket types before performing text operations. By default, this feature is NOT mapped to any key in vim.json. However, it can be enabled manually, and the recommended key for mapping is b: If you want to add it to your zed keymap config you need to add the following config: ```json { "context": "vim_operator == a || vim_operator == i || vim_operator == cs", "bindings": { "b": "vim::AnyBrackets" } } ``` ## Features - New text object that works with parentheses `()`, square brackets `[]`, curly braces `{}`, they are also know as round brackets, square brackets and curly brackets in english. - Automatically finds the closest matching pair of any bracket type - Works with all standard Vim operators (delete, change, yank) - Supports both "inside" and "around" variants (`i` and `a`) ## Usage Examples ```vim # Delete inside the closest brackets di( # Works on (), [] or {} depending on which is closest # Change around the closest brackets ca[ # Works on (), [] or {} depending on which is closest # Visual select inside the closest brackets vi{ # Works on (), [] or {} depending on which is closest ``` # References: - Based on the popular plugin https://github.com/echasnovski/mini.ai # Important Notes This PR also fixes a bug with nested quotes on AnyQuotes, now it works fine with any type of quotes or brackets. Please take a look at the new tests to understand the expected behavior. Release Notes: - vim: Add `ab`/`ib` "AnyBrackets" text objects that are the smallest of `a(`, `a[` or `a{` or `i(`, `i[` or `i{` - vim: Fix aq/iq "AnyQuotes" text objects when they are nested |
||
---|---|---|
.. | ||
src | ||
test_data | ||
Cargo.toml | ||
LICENSE-GPL | ||
README.md |
This contains the code for Zed's Vim emulation mode.
Vim mode in Zed is supposed to primarily "do what you expect": it mostly tries to copy vim exactly, but will use Zed-specific functionality when available to make things smoother. This means Zed will never be 100% vim compatible, but should be 100% vim familiar!
The backlog is maintained in the #vim
channel notes.
Testing against Neovim
If you are making a change to make Zed's behavior more closely match vim/nvim, you can create a test using the NeovimBackedTestContext
.
For example, the following test checks that Zed and Neovim have the same behavior when running *
in visual mode:
#[gpui::test]
async fn test_visual_star_hash(cx: &mut gpui::TestAppContext) {
let mut cx = NeovimBackedTestContext::new(cx).await;
cx.set_shared_state("ˇa.c. abcd a.c. abcd").await;
cx.simulate_shared_keystrokes(["v", "3", "l", "*"]).await;
cx.assert_shared_state("a.c. abcd ˇa.c. abcd").await;
}
To keep CI runs fast, by default the neovim tests use a cached JSON file that records what neovim did (see crates/vim/test_data), but while developing this test you'll need to run it with the neovim flag enabled:
cargo test -p vim --features neovim test_visual_star_hash
This will run your keystrokes against a headless neovim and cache the results in the test_data directory.
Testing zed-only behavior
Zed does more than vim/neovim in their default modes. The VimTestContext
can be used instead. This lets you test integration with the language server and other parts of zed's UI that don't have a NeoVim equivalent.