Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/chrisgrieser/nvim-rip-substitute
Search and replace in the current buffer with incremental preview, a convenient UI, and modern regex syntax.
https://github.com/chrisgrieser/nvim-rip-substitute
neovim-plugin nvim nvim-plugin replace-text search-and-replace substitution
Last synced: about 6 hours ago
JSON representation
Search and replace in the current buffer with incremental preview, a convenient UI, and modern regex syntax.
- Host: GitHub
- URL: https://github.com/chrisgrieser/nvim-rip-substitute
- Owner: chrisgrieser
- License: mit
- Created: 2024-06-04T19:27:56.000Z (7 months ago)
- Default Branch: main
- Last Pushed: 2025-01-05T21:14:46.000Z (6 days ago)
- Last Synced: 2025-01-06T21:52:50.650Z (5 days ago)
- Topics: neovim-plugin, nvim, nvim-plugin, replace-text, search-and-replace, substitution
- Language: Lua
- Homepage:
- Size: 210 KB
- Stars: 214
- Watchers: 2
- Forks: 7
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
- awesome-neovim - chrisgrieser/nvim-rip-substitute - Search and replace in the current buffer with a modern UI and modern regex flavor. A substitute for Vim's `:substitute` using `ripgrep`. (Search / PHP)
README
# rip-substitute 🪦
Search and replace in the current buffer with incremental preview, a convenient
UI, and modern regex syntax.## Table of Contents
- [Features](#features)
- [Installation](#installation)
- [Configuration](#configuration)
- [Usage](#usage)
- [Advanced](#advanced)
- [Limitations](#limitations)
- [About the developer](#about-the-developer)## Features
- Search and replace in the current buffer using
[ripgrep](https://github.com/BurntSushi/ripgrep).
- Uses **common regex syntax** — no more dealing with arcane vim regex.
- **Incremental preview** of matched strings and replacements, **live count** of
matches.
- Popup window instead of command line. This entails:
+ Syntax highlighting of the regex.
+ Editing with vim motions.
+ No more dealing with delimiters.
- **Sensible defaults**: entire buffer (`%`), all matches in a line
(`/g`), case-sensitive (`/I`).
- Substitute only in a **range**, with visual emphasis of the range
- **History** of previous substitutions.
- **Performant**: In a file with 5000 lines and thousands of matches, still
performs *blazingly fast*.â„¢
- **Regex101 integration**: Open the planned substitution in a pre-configured
[regex101](https://regex101.com/) browser-tab for debugging.
- **Quality-of-Life features**: automatic prefill of the escaped cursorword,
adaptive popup window width, toggle `--fixed-strings`, …
- Syntax comparison:```txt
# all three are equivalent# vim's :substitute
:% s/\(foo\)bar\(\.\)\@!/\1baz/gI# vim's :substitute (very magic mode)
:% s/\v(foo)bar(\.)@!/\1baz/gI# rip-substitute
(foo)bar(?!\.)
$1baz
```## Installation
**Requirements**
- nvim >= 0.10
- [ripgrep](https://github.com/BurntSushi/ripgrep) with `pcre2` support
+ `brew install ripgrep` (already includes `pcre2` by default)
+ `cargo install ripgrep --features pcre2`
- Alternatively, you can also use this plugin without `pcre2` by setting
`regexOptions.pcre2 = false` in the config. However, some features like
lookaheads are not supported then.
- *Optional:* `:TSInstall regex` (adds syntax highlighting)```lua
-- lazy.nvim
{
"chrisgrieser/nvim-rip-substitute",
cmd = "RipSubstitute",
opts = {},
keys = {
{
"fs",
function() require("rip-substitute").sub() end,
mode = { "n", "x" },
desc = " rip substitute",
},
},
},-- packer
use {
"chrisgrieser/nvim-rip-substitute",
config = function()
require("rip-substitute").setup()
end,
}
```## Configuration
```lua
-- default settings
require("rip-substitute").setup {
popupWin = {
title = " rip-substitute",
border = "single",
matchCountHlGroup = "Keyword",
noMatchHlGroup = "ErrorMsg",
position = "bottom", ---@type "top"|"bottom"
hideSearchReplaceLabels = false,
hideKeymapHints = false,
},
prefill = {
normal = "cursorWord", ---@type "cursorWord"|false
visual = "selectionFirstLine", ---@type "selectionFirstLine"|false (does not work with ex-command – see README)
startInReplaceLineIfPrefill = false,
alsoPrefillReplaceLine = false,
},
keymaps = { -- normal mode (if not stated otherwise)
abort = "q",
confirm = "",
insertModeConfirm = "",
prevSubstitutionInHistory = "",
nextSubstitutionInHistory = "",
toggleFixedStrings = "", -- ripgrep's `--fixed-strings`
toggleIgnoreCase = "", -- ripgrep's `--ignore-case`
openAtRegex101 = "R",
showHelp = "?",
},
incrementalPreview = {
matchHlGroup = "IncSearch",
rangeBackdrop = {
enabled = true,
blend = 50, -- between 0 and 100
},
},
regexOptions = {
startWithFixedStringsOn = false,
startWithIgnoreCase = false,
pcre2 = true, -- enables lookarounds and backreferences, but slightly slower
autoBraceSimpleCaptureGroups = true, -- disable if using named capture groups (see README for details)
},
editingBehavior = {
-- Typing `()` in the `search` line, automatically adds `$n` to the `replace` line.
autoCaptureGroups = false,
},
notification = {
onSuccess = true,
icon = "",
},
}
```> [!NOTE]
> A `ripgrep` config file set via `RIPGREP_CONFIG_PATH` is ignored by this
> plugin.## Usage
**lua function**```lua
vim.keymap.set(
{ "n", "x" },
"fs",
function() require("rip-substitute").sub() end,
{ desc = " rip substitute" }
)
```- Normal mode: prefills the *escaped* word under the cursor
- Visual mode: prefills the *escaped* selection
- Visual *line* mode: replacements are only applied to the selected lines
(= the selection is used as range)**Ex-command**
Alternatively, you can use the ex command `:RipSubstitute`, which also
accepts [a range
argument](https://neovim.io/doc/user/cmdline.html#cmdline-ranges). Note that
when using the ex-command, visual mode and visual line mode both pass a range.
To prefill the current selection, you therefore need to use the lua function.```vim
" Substitute in entire file. Prefills the *escaped* word under the cursor.
:RipSubstitute" Substitute in line range of the visual selection.
:'<,'>RipSubstitute" Substitute in given range (in this case: current line to end of file).
:.,$ RipSubstitute
```You can also pass a prefill for the search value, in which case the prefill
is *not* escaped.```vim
:RipSubstitute prefilled_unescaped_string
```> [!TIP]
> Use `showHelp` (default keymap: `?`) to show a notification containing all
> keymaps available in the popup window.## Advanced
**Remember prefill**
The function `require("rip-substitute").rememberCursorWord()` can be used to
save the word under the cursor for the next time `rip-substitute` is called.
(This overrides any other prefill for that run.)One use case for this is to set a prefill for when you intend to run substitute
with a range, since calling `rip-substitute` in visual line is not able to pick
up a prefill.**Filetype**
The popup window uses the filetype `rip-substitute`. This can be useful, for
instance, to disable auto-pairing plugins in the popup window.**`autoBraceSimpleCaptureGroups`**
A gotcha of `ripgrep`'s regex syntax is that it treats `$1a` as the named
capture group "1a" and *not* as the first capture group followed by the
letter "a." (See `ripgrep`'s man page on `--replace` for details.)If `regexOptions.autoBraceSimpleCaptureGroups = true` (the default),
`rip-substitute` automatically changes `$1a` to `${1}a`, to make writing the
regex more intuitive. However, if you regularly use named capture groups, you
may want to disable this setting.## Limitations
- Searching/replacing for line breaks (`\n` or `\r`) is not supported. [See
issue #28](https://github.com/chrisgrieser/nvim-rip-substitute/issues/28).
- This plugin only searches the current buffer. To search and replace in
multiple files via `ripgrep`, use a plugin like
[grug-far.nvim](https://github.com/MagicDuck/grug-far.nvim).## About the developer
In my day job, I am a sociologist studying the social mechanisms underlying the
digital economy. For my PhD project, I investigate the governance of the app
economy and how software ecosystems manage the tension between innovation and
compatibility. If you are interested in this subject, feel free to get in touch.I also occasionally blog about vim: [Nano Tips for Vim](https://nanotipsforvim.prose.sh)
- [Website](https://chris-grieser.de/)
- [Mastodon](https://pkm.social/@pseudometa)
- [ResearchGate](https://www.researchgate.net/profile/Christopher-Grieser)
- [LinkedIn](https://www.linkedin.com/in/christopher-grieser-ba693b17a/)