Skip to content

Latest commit

 

History

History
585 lines (426 loc) · 20 KB

README.md

File metadata and controls

585 lines (426 loc) · 20 KB

Neovim's config

I am back to Neovim 😋 Home sweet home 🏠 Neovim is simply beautiful! My setup is mainly for R and Julia and it might help you with migrating to Neovim.

Neovim 0.5 is really exciting, with its native LSP, treesitter and Lua API.

I just spent about one month relentlessly setting up and hacking my Neovim so I can use it as soon as possible. Pretty much setup most things the way I wanted. Now I try to feature-freeze, refactor, cleanup and pay off technical debts. Feel free to use it.


Summary of setups

  • Neovim's built-in terminal emulator to replace my Tmux
  • For session management, I use abduco. Attach/detach Neovim sessions and/or specific process
  • Theme minimalist.nvim
  • Vim-like keybindings for plot (not related to Neovim)
  • LSP
  • Terminal
    • Terminal buffer management neoterm
    • CLI and Neovim integration (e.g. open file from terminal in neovim) with nvr and neomux
      • open my_file
      • ff: call ff command in CLI and use telescope.nvim to preview and open file in Neovim
      • Send command or copy registers from CLI to Neovim, or vice verse
    • Use nvim as git editor for commit/rebase/etc
    • Floating terminal window management toggleterm
  • REPL
    • I customize my own REPL for sending to terminal and highlighting the codes sent e.g.
      • Send word under cursor
      • Send str, names and help ? for word under the cursor
      • Send line/paragraph/selection/motion
      • Send language-specific (R and Julia) blocks/code-blocks (e.g. R's function/for loop, Julia's function/struct/for loop)
      • Send chain (e.g. R's chain %>%)
      • Send partial chain. I stole the idea from MilesMcBain/breakerofchains.
      • R REPL auto-suggestion
      • fzf-like history search for R REPL. For Julia, use JLFzf.jl.
  • RMarkdown
    • Go to prev/next chunk and highlight chunk
    • Send chunk to REPL and highlight chunk
    • Chunk object (e.g. change/delete/select) chunk with if, af, etc
  • Git
    • Syntax-highlighting pager delta
      • e.g. <leader>gb preview diff for current file, <leader>gd preview diff
    • Terminal UI lazygit with delta integration
    • Magit clone for Neovim neogit
    • Git decorations gitsigns
  • fzf-like find/filter/preview telescope.nvim
  • Search & replace panel nvim-spectre
  • orgmode-like note-taking apps neorg
  • Motion (This is real good!) lightspeed
  • Autopairs auto-pairs

Other R-related setup


My <escape> for both text and terminal buffers is capslock, which is mapped to <f10> in Neovim and remapped by OS-level tools AutoHotKey. So whenever you see <f10>, it's <escape>. And whenever you see comment AHKREMAP, which stands for AutoHotKey remap, it means the keybind is remapped using OS-level tools because some keybinds are not recognized by Neovim/terminal.


Installation

code
git clone git@github.com:kar9222/nvim.git
make init
# install and configure other tools (e.g. rg, fd)...

Gallery 🎨

Overview

nvim_transparent

Search & replace

search_and_replace

LSP

Go to symbol (document/workspace)

NOTE You can

  • Filter symbols by class (e.g. function, variable) with <c-l> and
  • Query as usual

go_to_symbol

Go to definition & references

references

Symbols outline

outline

Signature

signature

Diagnostics

diagnostics

Completion

completion

REPL

I customize my own REPL for sending to terminal and highlighting the codes sent e.g.

  • Send word under cursor
  • Send str, names and help ? for word under the cursor
  • Send line/paragraph/selection/motion
  • Send language-specific (R and Julia) blocks/code-blocks (e.g. R's function/for loop, Julia's function/struct/for loop)
  • Send chain (e.g. R's chain %>%)
  • Send partial chain. I stole the idea from MilesMcBain/breakerofchains.
  • fzf-like history search for R REPL. For Julia, use JLFzf.jl.

R REPL

r_repl

RMarkdown

  • Go to prev/next chunk and highlight chunk
  • Send chunk to REPL and highlight chunk
  • Chunk object (e.g. change/delete/select) chunk with if, af, etc

rmarkdown

R REPL auto-suggestion

Start typing, suggestions will automatically shown inline, press keybind (e.g. <m-l>) to complete the sentence, or continue typing to trigger other suggestions, or simply ignore it.

autosuggestion

R REPL fzf-like history search

Interactive R history search with Neovim's floating window. Great for single-line history search and can be used together with native {radian} REPL's ctrl+r multi-line history search.

Details

Call bash script send_r_history and Neovim's floating window search_history defined in toggle_float.lua with either

  • radian REPL keybind mapped to Python (radian is written in Python):
    • import os ; os.system('nvr --nostart -c "lua search_history(\\"send_r_history\\", _hist_opts)"')
  • or Neovim's terminal mode keybind e.g. vim.api.nvim_set_keymap('t', ..., '<cmd>lua search_history("send_r_history", _hist_opts)<CR>', ...)

I use radian REPL keybind, so that it's process-specific and doesn't clash with other keybinds. For Neovim's keybind, to make it process-specific or terminal-specific, you have to identify the process stored in terminal buffer's variable vim.b.term_title, for example, in Lua:

  • if string.match(vim.b.term_title, 'radian') then vim.api.nvim_set_keymap(...) end

Bash script send_r_history

#! /bin/bash

# R history: Remove duplicates, commented lines, blank lines, starting `+` symbol
# NOTE Edit the path ~/.radian_history (default) if necessary
tac ~/.radian_history                    | \
    awk '!a[$0]++'                       | \
    sed -e '/^#/d  ;  /^$/d  ;  s/^+//g' | \
    fzf --exact --no-sort

r_repl_fzf_history_search

Terminal buffer management

I use multiple terminals on the right windows, with named terminal buffers on the statusline (R, Julia and shell). Use neoterm to manage it, bind Tprev and Tnext for navigating between multiple terminals.

terminal_buffer_management

abduco session management

abduco
#! /bin/bash

# Usage:
# NOTE `:qa` exits nvim and hence it kills the session. Hence only use it if the intention is to kill the session. To quit without killing the session, for example
# - detach session
# - close the terminal window.

# unset NVIM_LISTEN_ADDRESS  # TODO Need this for SSH?
session=`basename $PWD`

# If the session isn't exist, force creation of session when there is an already terminated session of the same name, after showing its exit status.
# And attach to the session.
# Detach key is <c-/>
abduco -e ^_ -fA $session nvim

CLI & Neovim integration with nvr and neomux

CLI is still the best when it matters. You can call Neovim from CLI, for example,

  • open my_file
  • ff: call ff command in CLI and use telescope.nvim to preview and open file in Neovim
  • ge: simiar as ff, but for live grep

See both telescope.lua and below for details. Once you get the hang of it, you can use similar logic for any other CLI-Neovim integration. Please share them with me! 😄

Details: General CLI-Neovim integration and to call Neovim functions from CLI
if [ -n "$NVIM_LISTEN_ADDRESS" ]; then
    alias nvim=nvr -cc split --remote-wait +'set bufhidden=wipe'
fi

# For example, file explorer on the left is the first window.
# To open file in the middle window, it's the second window. Adjust as needed.
export MIDDLE_WIN_NR=2  

export EDITOR="nvr -cc '${MIDDLE_WIN_NR}wincmd w' --remote-wait +'set bufhidden=wipe'"
export VISUAL="nvr -cc '${MIDDLE_WIN_NR}wincmd w' --remote-wait +'set bufhidden=wipe'"

open() {  # Open neovim middle window
    out=$1
    if [[ $out != "" ]]; then
        vw $MIDDLE_WIN_NR $out
    fi
}

copen() {  # Create and open file 
  mkdir -p $(dirname $1)
  touch $1
  open $1
}

# Save directory for telescope's find_files_custom_dir and find_files_live_grep
save_dir() { echo $PWD | vc 'z' }  # Save directory in register `z` 
ff() {  # Find files in custom directory
    save_dir
    nvr -c 'lua find_files_custom_dir()'
    nvr -c 'startinsert'
}
aff() {  # Find files (inc. hidden) in custom directory
    save_dir
    nvr -c 'lua find_files_hidden_custom_dir()'
    nvr -c 'startinsert'
}
iff() {  # Find files (inc. hidden & ignored) in custom directory
    save_dir
    nvr -c 'lua find_files_hidden_ignore_custom_dir()'
    nvr -c 'startinsert'
}
ge() {  # Live grep
    save_dir
    nvr -c 'lua live_grep_custom_dir()'
    nvr -c 'startinsert'
}

open_file_in_custom_dir

Git

Syntax-highlighting pager delta

Use delta in terminal buffer in floating window. See telescope.lua and delta.lua for details.

  • <leader>gb preview diff for current file
  • <leader>gd preview diff
  • q quit
  • <c-s> stage current buffer
  • <c-u> unstage current buffer
  • <c-s-s> stage unstage
  • <c-u-u> unstage stage
  • It's pager (default to less), hence all the less navigation commands work: d, u, f, b, g, G, search / ?, n, N, etc

git_delta

Terminal UI lazygit with delta integration

lazygit

Magit clone for Neovim neogit

neogit

Git decorations gitsigns

gitsigns

Vim-like keybindings for plot (not related to Neovim)

For plot, I use httpgd

  • Focus/minimize plot
  • Toggle position at top right window (re-position right terminals to bottom)
  • Increase/decrease window size
  • Move plot window left/right/up/down
Plot
#ifWinActive ahk_group plot  ; TODO Clashes key with vscode.ahk

plot__toggle_activate_minimized:
^!p::
winGet, state, minMax, R Plot
if (state == -1)  ; Minimized
    winActivate, R Plot
else  ; Not minimized
    winMinimize, R Plot
; TODO Bug: Window is activated but not focus e.g. keybindings not working. Hacky workaround: call `msgbox` with minimal timeout so that it disappear almost instantly. When it disappears, R plot windows is focused.
msgbox,,,, 0.001
return

; Init and restore R plot window, hacky solution to enlarge/restore shell pane, etc TODO Optimize
plot_toggle := 0
plot__toggle_init_restore:
+!p::
    setKeyDelay, 100
    toggle := ! toggle

    ; Init plot
    if (toggle = 1) {
        winActivate ahk_exe WindowsTerminal.exe
        winWaitActive ahk_exe WindowsTerminal.exe
        send ^+!{f8}  ; Call nvim's plot_init

        ; R plot window: set `alwaysOnTop` and position to top-right corner
        winActivate R Plot
        winWaitActive R Plot
        winSet, alwaysOnTop, , R Plot
        winMove, R Plot, , 1044, -32, 881, 597  ; NOTE Also move and resize to default size. Same as win_pos___standard
        send !{tab}
    }
    ; Restore plot
    else {
        winActivate ahk_exe WindowsTerminal.exe
        winWaitActive ahk_exe WindowsTerminal.exe
        send ^+!{f7}  ; Call nvim's plot_restore

        ; R plot window: set `alwaysOnTop` and restore default position
        winActivate R Plot
        winWaitActive R Plot
        winSet, alwaysOnTop, , R Plot
        winMove, R Plot, , 523.5, 166  ; Same as from .Rprofile
        winMinimize, R Plot
    }
    ; Restore default
    setKeyDelay, -1
return


; _ Navigation, window management ---------------

#ifWinActive R Plot

plot__minimize:
p::
winMinimize A
return

plot__increase_size:
i::
; Increase width and height by 25%
; Move upper-left: decrease x and y by 25%
winActivate R Plot
winGetPos, x, y, width, height, A
winMove, A, , .75*x, .75*y, 1.25*width, 1.25*height
return

plot__decrease_size:
e::
; TODO Correct? Very slightly diff compared to default size. Test by `plot__increase_size`, `plot__decrease_size`, `get_info_of_active_window`
; Decrease width and height by 20% (1/1.25 == .8)
; Move lower-right: increase x and y by 33% (1/.75 == 1.33)
winActivate R Plot
winGetPos, x, y, width, height, A
winMove, A, , 1.33*x, 1.33*y, .8*width, .8*height
return

plot__move_win_up:
+k::
winGetPos, , y, , , A
winMove, A, , , y-200
return

plot__move_win_dn:
+j::
winGetPos, , y, , , A
winMove, A, , , y+200
return

plot__move_win_left:
+h::
winGetPos, x, , , , A
winMove, A, , x-200
return

plot__move_win_right:
+l::
winGetPos, x, , , , A
winMove, A, , x+200
return

plot

Other great stuffs

Rust's implementation of process/system monitor

CLI file manager joshuto

cli_file_manager

Other R-related stuffs

R terminal output

See My {colorout}

colorout

Custom {skimr} summary stats

See Custom {skimr} summary stats

skimr

Use R help system with Vimium

See Use R help system with Vimium

r_help_1

r_help_2

R theme package

See {theme} for R packages: {ggplot2}, {plotly}, {shiny}, {rmarkdown}, {ggdist}, {ggdag}, etc.

theme

Summary of opinions for setup

In my config, you will realize I love using

  • Tabs for multiple workspace (like virtual desktop environment), e.g. for
    • terminal UI lazygit
    • Neogit
    • splitting windows for different files and/or terminals
    • basically anything
  • Floating terminal so that I can focus it in the center of my screen, with the option of re-positioning it to the right of the window. See toggleterm.lua

REPL: There are many REPL plugins out there like vim-slime, but they lacks some features which I use. Hence, I customize them myself.


telescope.nvim: Alternative to fzf, telescope is native to Neovim, hence it has better integration/keybinds/etc and much more easily customize/hack.


Neovim's built-in terminal emulator to replace my Tmux

  • Pros
    • Better overall integration and easier to hack
    • Better keybindings integration
    • Use everything (e.g. floating window/terminal) in full screen inside neovim
  • Cons
    • Redraw issues
    • Minor issues I can live with

For completion, I am happy with nvim-cmp and some of its features. coq_nvim is the fastest but for me, it's 'too fast', poping out too frequently. I just love old-school typing and manually trigger completion when I need it. I also played with it and feel free to use my config coq_nvim.lua. I might use it in the future.

Authors

This repo is mainly for sharing my setup. Thank you open source.

Some idea and codes are stolen from others lol. (I have no idea how to credit them, please let me know if I have done anything wrong in this regard, miss out your names, or remove your name, thanks!). I have included their repo and name in the file e.g. see vim/repl.vim.

Hence, I would like to credit them. They are also the authors of this repo

Internal notes

  • feat(terminal): close shell terminals automatically #15440

  • Should I move all the gists to this repo so other people can easier refer to them?

  • textwidth for rmd.vim and pandoc.vim

  • nvim-cmp's highlights

  • Nvim v0.6.0 potential break: See change log, pull #15585, etc

  • Document paste_markdown_link

  • lazygit issue #1698

  • Add docs for toggle_term_tab (lazydocker) and btm

  • packer package snapshot (#370)

  • Potential break for update of plugins

  • https://github.com/kyazdani42/nvim-tree.lua/commit/0e7856fd8de18edaaee8fa0c2f2c9432c973aef5

  • toggleterm: terminal_mappings pull #149

  • (Feat) lightspeed: Bidirectional search, cross-window motions

  • Update julia-vim (changes from 9 Feb 22 onwards)

  • telescope:

    • wrap_results pull #1735
    • LSP ignore_symbols pull #1745
    • LSP filter diagnostics pull 1610
  • nvim-cmp: THROTTLE_TIME

  • Luasnip: Deprecated autosnippets aeea587

  • Explore rcarriga/nvim-notify

  • nvim --remote. Potentially replace nvr

  • Explore highlight: support for blend in nvim_set_hl (#17516) (b5bf487)

  • Explore ggandor/leap.nvim and ggandor/flit.nvim

  • Explore toggleterm pull #245 with new, default feature persist_mode = true

  • Explore cmp pull #901 and #1026

  • Explore nvim-cmp pull #1067 and #1003

  • Explore gitsigns commit f6c95960, fe6b09b, 88d7aae

  • Use glepnir/lspsaga.nvim

  • Explore set winbar and set laststatus=3

  • Explore feat(runtime)!: enable filetype.lua by default #19216

  • Explore toggleterm commit d358641, 42438b3, c8574d7, 01a84bc, 81ea9f7

  • Explore nvimtree commit 69a07d1

  • Explore spectre commit 17b7c84

  • Explore telescope commit 84d53df

  • Fix HOTFIX tag

  • Explore nvimtree Open At Startup

  • Explore: See lspconfig-single-file-support

  • TODO_URGENT NvimTree bugs (6da7467)

    • See view.preserve_window_proportions
  • See commented-out a.nvim_command("wincmd " .. move_to) removes the error

function M.reposition_window()
  local move_to = move_tbl[M.View.side]
  -- a.nvim_command("wincmd " .. move_to)
  M.resize()
end