Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/bennypowers/nvim-regexplainer

Describe the regexp under the cursor
https://github.com/bennypowers/nvim-regexplainer

lua neovim neovim-plugin nvim nvim-plugin regex regexp regular-expression tree-sitter

Last synced: 2 days ago
JSON representation

Describe the regexp under the cursor

Awesome Lists containing this project

README

        

![Am Yisrael Chai - עם ישראל חי](https://bennypowers.dev/assets/flag.am.yisrael.chai.png)

# nvim-regexplainer

![Lua][made-with-lua]
![GitHub Workflow Status][build-status]
[![Number of users on dotfyle][dotfyle-badge]][dotfyle]

Describe the regular expression under the cursor.

https://user-images.githubusercontent.com/1466420/156946492-a05600dc-0a5b-49e6-9ad2-417a403909a8.mov

Heavily inspired by the venerable [atom-regexp-railroad][atom-regexp-railroad].

> 👉 **NOTE**: Requires Neovim 0.7 👈

## 🚚 Installation

```lua
use { 'bennypowers/nvim-regexplainer',
config = function() require'regexplainer'.setup() end,
requires = {
'nvim-treesitter/nvim-treesitter',
'MunifTanjim/nui.nvim',
} }
```

You need to install `regex` with `nvim-treesitter`, as well as the grammar for
whichever host language you're using. So for example if you wish to use
Regexplainer with TypeScript sources, you need to do this:

```vimscript
:TSInstall regex typescript
```

## 🤔 Config

```lua
-- defaults
require'regexplainer'.setup {
-- 'narrative'
mode = 'narrative', -- TODO: 'ascii', 'graphical'

-- automatically show the explainer when the cursor enters a regexp
auto = false,

-- filetypes (i.e. extensions) in which to run the autocommand
filetypes = {
'html',
'js',
'cjs',
'mjs',
'ts',
'jsx',
'tsx',
'cjsx',
'mjsx',
},

-- Whether to log debug messages
debug = false,

-- 'split', 'popup'
display = 'popup',

mappings = {
toggle = 'gR',
-- examples, not defaults:
-- show = 'gS',
-- hide = 'gH',
-- show_split = 'gP',
-- show_popup = 'gU',
},

narrative = {
indendation_string = '> ', -- default ' '
},
}
```

### `display`

Regexplainer offers a small variety of display modes to suit your preferences.

#### Split Window

Set to `split` to display the explainer in a window below the editor.
The window will be reused, and has the filetype `Regexplainer`

#### Popup Below Cursor

Set to `popup` (the default) to display the explainer in a popup below the
cursor. When the cursor moves, the popup closes. if `auto` is set, the popup
will automatically display whenever the cursor moves inside a regular expression
You can call `show` with your own display type to override your config

```lua
require'regexplainer'.show { display = 'split' }
```

Or use the commands `RegexplainerShowSplit` or `RegexplainerShowPopup`.
`RegexplainerHide` and `RegexplainerToggle` are also available.

You can customize the popup window by specifying `options.popup.border`,
which is a table of [popup options from nui][popup-options].
Any options specified for `options.popup` will also override the defaults.

```lua
require'regexplainer'.show {
display = 'popup',
popup = {
border = {
padding = { 1, 2 },
style = 'solid',
},
},
}
```

You could use this to, for example, set a different border based on the state of
your editor.

### Render Options

`narrative.indendation_string` can be a function taking the current component and
returning an indendation indicator string. For example, to show the capture group on each line:

```lua
narrative = {
indentation_string = function(component)
return component.capture_depth .. '> '
end
},
```

Input:

```javascript
/zero(one(two(three)))/;
```

Output:

```markdown
`zero`
capture group 1:
1> `one`
1> capture group 2:
1> 2> `two`
1> 2> capture group 3:
1> 2> 3> `three`
```

## Yank
You can yank the regexplanation into any register with the `yank` function. The
default register is `"`. This can be useful if you'd like to share the
explanation of a regexp with your teammates, or if you'd like to report a
mistake in regexplainer. The argument to `yank` is either a string (the register
to yank to) or a table with `register: string` and options to `show` (e.g. `mode
= 'narrative', narrative = {}`, etc.).

For example, to copy the regexplanation to your system clipboard, use either of
these:

```lua
require'regexplainer'.yank'+'
```

```lua
require'regexplainer'.yank { register = '+' }
```

You can also use the command `RegexplainerYank`

```vim
:RegexplainerYank +
```

## 🗃️ TODO list
- [ ] Display Regexp [railroad diagrams][railroad-diagrams] using ASCII-art
- [ ] Display Regexp [railroad diagrams][railroad-diagrams] via
[hologram][hologram] and [kitty image protocol][kitty], maybe with a sixel
fallback
- [ ] online documentation
- [x] some unit tests or something, i guess

[made-with-lua]: https://img.shields.io/badge/Made%20with%20Lua-blueviolet.svg?style=for-the-badge&logo=lua
[build-status]: https://img.shields.io/github/actions/workflow/status/bennypowers/nvim-regexplainer/main.yml?branch=main&label=tests&style=for-the-badge
[atom-regexp-railroad]: https://github.com/klorenz/atom-regex-railroad-diagrams/
[popup-options]: https://github.com/MunifTanjim/nui.nvim/tree/main/lua/nui/popup#border
[railroad-diagrams]: https://github.com/tabatkins/railroad-diagrams/
[hologram]: https://github.com/edluffy/hologram.nvim
[kitty]: https://sw.kovidgoyal.net/kitty/graphics-protocol/
[dotfyle]: https://dotfyle.com/plugins/bennypowers/nvim-regexplainer
[dotfyle-badge]: https://dotfyle.com/plugins/bennypowers/nvim-regexplainer/shield?style=for-the-badge