2
0
mirror of https://github.com/mhinz/neovim-remote synced 2024-11-11 13:10:34 +00:00
Go to file
2018-02-28 20:29:22 +01:00
images Remove logo 2017-04-27 11:43:22 +02:00
nvr Make -c/-cc/--remote-expr read from stdin on - 2017-12-20 17:20:19 +01:00
tests Tests: add first pytest 2017-05-25 00:13:26 +02:00
.gitignore Add .gitignore 2016-06-05 13:27:55 +02:00
.travis.yml CI: never notify me on success 2017-10-24 13:19:23 +02:00
INSTALLATION.md README: improve installation section 2018-02-28 20:13:05 +01:00
LICENSE Add LICENSE 2015-12-04 15:58:37 +01:00
Makefile Makefile: ditch twine for uploading 2017-07-26 01:52:02 +02:00
README.md README: add theory and first steps 2018-02-28 20:29:22 +01:00
setup.py Make -c/-cc/--remote-expr read from stdin on - 2017-12-20 17:20:19 +01:00

neovim-remote

Build status Wheel? PyPI version Supported Python versions License

This package provides an executable called nvr which solves these cases:

  • Controlling nvim processes from the shell. E.g. opening files in another terminal window.
  • Opening files from within :terminal without starting a nested nvim process.

Installation

pip3 install neovim-remote

If you encounter any issues, e.g. permission denied errors or you can't find the nvr executable, read INSTALLATION.md.

Theory

Nvim always starts a server. Get its address via :echo $NVIM_LISTEN_ADDRESS or :echo v:servername. Or specify an address at startup: NVIM_LISTEN_ADDRESS=/tmp/nvimsocket nvim.

nvr will use $NVIM_LISTEN_ADDRESS or any address given to it via --servername.

If the targeted address does not exist, nvr starts a new process by running "nvim". You can change the command by setting $NVR_CMD. (This requires forking, so it won't work on Windows.)

First steps

Start a nvim process (which acts as a server) in one shell:

NVIM_LISTEN_ADDRESS=/tmp/nvimsocket nvim

And do this in another shell:

# nvr uses /tmp/nvimsocket by default, so we're good.

# Open two files:
nvr --remote file1 file2

# Send keys to the current buffer:
nvr --remote-send 'iabc<esc>'
# Enter insert mode, insert 'abc', and go back to normal mode again.

# Evaluate any VimL expression, e.g. get the current buffer:
nvr --remote-expr 'bufname("")'
README.md

See nvr -h for all options.

nvr -h ``` nvr -h usage: /Users/mhi/.pyenv/versions/3.6.4/bin/nvr [arguments]

Remote control Neovim processes.

If no process is found, a new one will be started.

$ nvr --remote-send 'iabc<cr><esc>'
$ nvr --remote-expr 'map([1,2,3], "v:val + 1")'

Any arguments not consumed by options will be fed to --remote-silent:

$ nvr --remote-silent file1 file2
$ nvr file1 file2

All --remote options take optional commands. Exception: --remote-expr, --remote-send.

$ nvr +10 file
$ nvr +'echomsg "foo" | echomsg "bar"' file
$ nvr --remote-tab-wait +'set bufhidden=delete' file

Open files in a new window from a terminal buffer:

$ nvr -cc split file1 file2

Use nvr from git to edit commit messages:

$ git config --global core.editor 'nvr --remote-wait-silent'

optional arguments: -h, --help show this help message and exit --remote [ [ ...]] Use :edit to open files. If no process is found, throw an error and start a new one. --remote-wait [ [ ...]] Like --remote, but block until all buffers opened by this option get deleted or the process exits. --remote-silent [ [ ...]] Like --remote, but throw no error if no process is found. --remote-wait-silent [ [ ...]] Combines --remote-wait and --remote-silent. --remote-tab [ [ ...]] Like --remote, but use :tabedit. --remote-tab-wait [ [ ...]] Like --remote-wait, but use :tabedit. --remote-tab-silent [ [ ...]] Like --remote-silent, but use :tabedit. --remote-tab-wait-silent [ [ ...]] Like --remote-wait-silent, but use :tabedit. --remote-send Send key presses. --remote-expr Evaluate expression and print result in shell. --servername Set the address to be used. This overrides the default "/tmp/nvimsocket" and $NVIM_LISTEN_ADDRESS. --serverlist Print the TCPv4 and Unix domain socket addresses of all nvim processes. -cc Execute a command before every other option. -c Execute a command after every other option. -l Change to previous window via ":wincmd p". -o [ ...] Open files via ":split". -O [ ...] Open files via ":vsplit". -p [ ...] Open files via ":tabedit". -q Read errorfile into quickfix list and display first error. -s Silence "no server found" message. -t Jump to file and position of given tag. --nostart If no process is found, do not start a new one.

Happy hacking!

</details>

## Use case

Imagine Neovim is set as your default editor: `EDITOR=nvim`.

Now run `git commit`. In a regular shell, a new nvim process starts. In a
terminal buffer (`:terminal`), a new nvim process starts as well. Now you have
one nvim nested within another. You don't want that. Put this in your vimrc:

```vim
if has('nvim')
  let $VISUAL = 'nvr -cc split --remote-wait'
endif

That way, you get a new window for entering the commit message instead of a nested nvim process.

Alternatively, you can make git always using nvr. In a regular shell, nvr will create a new nvim process. In a terminal buffer, nvr will open a new buffer.

$ git config --global core.editor 'nvr --remote-wait-silent'

Demos

(Click on the GIFs to watch them full-size.)

Using nvr from another shell: Demo 1

Using nvr from within :terminal: Demo 2

FAQ

  • How to open directories?

    :e /tmp opens a directory view via netrw. Netrw works by hooking into certain events, BufEnter in this case (see :au FileExplorer for all of them).

    Unfortunately Neovim's API doesn't trigger any autocmds on its own, so simply nvr /tmp won't work. Meanwhile you can work around it like this:

      $ nvr /tmp -c 'doautocmd BufEnter'
    
  • Reading from stdin?

    Yes! E.g. echo "foo\nbar" | nvr -o - and cat file | nvr --remote - work just as you would expect them to work.

  • Exit code?

    If you use a recent enough Neovim, nvr will use the same exit code as the linked nvim.

    E.g. nvr --remote-wait <file> and then :cquit in the linked nvim will make nvr return with 1.

  • How to send a message to all waiting clients?

    If you open a buffer with any of the wait options, that buffer will get a variable b:nvr. The variable contains a list of channels wheres each channel is a waiting nvr client.

    Currently nvr only understands the Exit message. You could use it to disconnect all waiting nvr clients at once:

    command! DisconnectClients
        \  if exists('b:nvr')
        \|   for client in b:nvr
        \|     silent! call rpcnotify(client, 'Exit', 1)
        \|   endfor
        \| endif