Setting Up a Supercharged Neovim Configuration

12 minute read Published: 2024-10-13

Over the summer, after finally getting around to learning Vim motions (see blog post #9), I quickly fell down the Neovim rabbithole and have been procrastinating work by tinkering away at my configurations ever since! This post will be sharing setup that I have currently landed at to turn my Neovim editor into a supercharged workhorse.

Table of Contents

TL;DR

I think my setup is pretty sweet. If you want to try out my setup without having to do the configuration yourself, you can run these commands to add my setup as a "profile" that you can switch to in Neovim:

  1. Clone the repository and install plugins:

git clone git@github.com:micahkepe/dotfiles ~/.config/micahkepe/dotfiles
  1. Open Neovim with this configuration:

NVIM_APPNAME=micahkepe/dotfiles/nvim nvim

If you're interested in the nerdy details of my setup, keep reading!


What is Neovim Anyway?

Vim (short for Vi Improved) was created by Bram Moolenaar in 1991 as an enhanced version of the Vi editor, which was originally developed in 1976 by Bill Joy for Unix systems. Vim quickly became popular due to its efficiency and ability to perform complex text manipulations with minimal keystrokes.

The original Vi program running with visuals.

The original Vi program running with 'visuals'

However, Vim had its limitations, especially with handling modern development features such as better plugin systems, scripting flexibility, and ease of configuration. Neovim, a fork of Vim, was initiated in 2014 to address these challenges and provide a more modern, extensible framework.

Neovim differentiates itself by embracing the Lua programming language for faster, more customizable configurations. This shift has given rise to a flourishing ecosystem of plugins that are tailored to extend Neovim’s capabilities—from file navigation and autocompletion to even embedding images inside your text editor.

Getting Started with Neovim

If you want to follow along with setup, you can install Neovim by following the instructions below. If you already have Neovim installed, you can skip ahead to the setup.

To install Neovim, check the INSTALL.md file on the Neovim repository for the instructions for your machine.


Neovim logo

The Setup

First off, a confession. I am not using off the shelf Neovim but starting from a very strong base configuration provided by the awesome folks at NVChad. NVChad is not a Neovim distro, but an opinionated starting Neovim configuration that provides a great starting place.

I particularly like NVChad for the following reasons:


NVChad logo

If you are following along, let's first install NVChad before we continue:

For Mac and Linux Users:


git clone https://github.com/NvChad/starter ~/.config/nvim && nvim

Windows Users:

A little more complicated (as usual), check out the install documentation here

Changes to NVChad's Default Settings

Diagnostics Menu

One of the things I changed from NVChad was the diagnostics display. By default, NVChad has the diagnostics displayed inline in red text. One consequence of this is that long diagnostic messages do not wrap at the end of the buffer, making it difficult to read the entire message:

Diagnostics display in red text

NvChad default diagnostics

To make the diagnostics more readable and "Visual-Studio-Code-like", I changed the diagnostics to underline the offending line and display to a floating window at the offending line on a cursor hold:

Diagnostics display in a floating window on the screen

Diagnostics display in a floating window at the bottom of the screen

To do this, I created a separate file for the diagnostics configuration in nvim/lua/configs/diagnostics.lua:


-- nvim/lua/configs/diagnostics.lua
local M = {}

M.setup = function()
  vim.lsp.handlers["textDocument/publishDiagnostics"] = vim.lsp.with(vim.lsp.diagnostic.on_publish_diagnostics, {
    virtual_text = false, -- no inline diagnostics
    underline = true, -- underline problematic code
  })

  -- Set the update time for diagnostics
  vim.o.updatetime = 250

  -- Automatically show diagnostics on cursor hold
  vim.cmd [[
    autocmd CursorHold,CursorHoldI * lua vim.diagnostic.open_float(nil, { focus = false })
  ]]
end

return M

Then to load this configuration, I added the following lines to the top of my nvim/lua/configs/lspconfig.lua:


-- nvim/lua/configs/lspconfig.lua
local diagnostics = require "configs.diagnostics"

-- Diagnostics popup
diagnostics.setup()

Enable Hidden Files in nvim-tree Display

Another change I made was to show hidden files in the nvim-tree file explorer. By default, hidden files are not displayed in the file explorer, which can be a little annoying when you need to access a .gitignore, configuration files, or a .env file, for example.

To do this, I altered the section in the nvim/lua/init.lua file that loads NvChad's defined plugins:


-- nvim/lua/init.lua
-- load plugins
require("lazy").setup({
  {
    "NvChad/NvChad",
    lazy = false,
    branch = "v2.5",
    import = "nvchad.plugins",
    config = function()
      require "options"
      -- Override nvim-tree settings
      local nvim_tree_options = require "nvchad.configs.nvimtree"
      nvim_tree_options.filters.dotfiles = false
      nvim_tree_options.git = { enable = true }
      nvim_tree_options.filters.git_ignored = false
      nvim_tree_options.filters.custom = { "^\\.git$", "DS_Store" }
    end,
  },

  { import = "plugins" },
}, lazy_config)

In the nvim_tree_options.filters.custom table above, I added "^\\.git$" and "DS_Store" regular expressions to not display files from the .git/ folder, and to ignore the .DS_Store files that macOS creates, respectively.


Plugins

I won't be detailing every plugin that I use, but instead the ones that have become staples in my editing experience. All plugins snippets that follow will be using lazy.nvim install configurations. I will break this section into the plugins that come by default from NVChad and the plugins that I have added on.

Note: If you are curious about all of my plugins, you can see them here.


NVChad Plugins

NVChad comes with a carefully curated set of plugins that provide a solid foundation for your Neovim setup. Here are some of the key plugins included:

  1. nvim-tree/nvim-tree.lua: A file explorer tree for Neovim written in Lua. It provides a sleek and customizable way to navigate your project files.

  2. nvim-treesitter/nvim-treesitter: Treesitter configurations and abstraction layer for Neovim. It provides better syntax highlighting and code understanding.

  3. lewis6991/gitsigns.nvim: Git integration for buffers. It shows which lines have been added, removed, or modified in the gutter.

  4. hrsh7th/nvim-cmp: A completion plugin for Neovim coded in Lua. It provides autocompletion functionality similar to what you'd find in modern IDEs.

  5. williamboman/mason.nvim: Portable package manager for Neovim that runs everywhere Neovim runs. Easily install and manage LSP servers, DAP debugger servers, linters, and formatters.

  6. neovim/nvim-lspconfig: A collection of common configurations for Neovim's built-in LSP client.

  7. folke/which-key.nvim: Displays a popup with possible key bindings of the command you started typing. Incredibly useful for discovering and remembering mappings.

  8. nvim-telescope/telescope.nvim: A highly extendable fuzzy finder over lists. It helps you search, filter, find, and pick things in your Neovim setup.

These plugins work together to provide a fully-featured IDE-like experience right out of the box. NVChad's setup ensures that these plugins are configured to work well together, providing a smooth and cohesive editing experience.


Added Plugins "Musts"

In no particular order, here are the plugins that I have added to augment NVChad's base.

gelguy/wilder.nvim

Makes navigating the Neovim command line much faster with fuzzy search capabilities, providing instant suggestions as you type commands.

When executing a command or searching, wilder.nvim offers a real-time list of suggestions based on your command history, file paths, and available commands. This dramatically speeds up workflows that involve repeated or complex command usage.

lazy.nvim:


-- nvim/lua/plugins/wilder.nvim

return {
  {
    "gelguy/wilder.nvim",
    event = "CmdlineEnter",
    build = ":UpdateRemotePlugins",
    dependencies = {
      "nvim-tree/nvim-web-devicons",
    },
    config = function()
      local wilder = require "wilder"
      wilder.setup { modes = { ":", "/", "?" } }
      wilder.set_option("pipeline", {
        wilder.branch(
          wilder.python_file_finder_pipeline {
            file_command = function(_, arg)
              if string.find(arg, ".") ~= nil then
                return { "fd", "-tf", "-H" }
              else
                return { "fd", "-tf" }
              end
            end,
            dir_command = { "fd", "-td" },
            filters = { "fuzzy_filter", "difflib_sorter" },
          },
          wilder.cmdline_pipeline(),
          wilder.python_search_pipeline()
        ),
      })

      wilder.set_option(
        "renderer",
        wilder.popupmenu_renderer {
          highlighter = wilder.basic_highlighter(),
          left = { " ", wilder.popupmenu_devicons() },
          right = { " ", wilder.popupmenu_scrollbar { thumb_char = " " } },
          highlights = {
            default = "WilderMenu",
            accent = wilder.make_hl("WilderAccent", "Pmenu", {
              { a = 1 },
              { a = 1 },
              { foreground = "#f4468f" },
            }),
          },
        }
      )
    end,
  },
}

3rd/image.nvim

Enables image rendering in Neovim, essential for tasks like markdown previews and documentation editing. When editing markdown files or any other file format that supports images, image.nvim renders inline images. This eliminates the need to switch to a browser or other external tool, making Neovim more than just a text editor—it's a media-enabled workspace.

Honestly this one was a pain in the ass to setup but worth it.

I did have to make the switch to the Kitty terminal emulator from iTerm2 as they implement different terminal image protocols (kitty vs. imgcat) and Kitty's protocol has more support as of writing. Additionally, getting the right Luarocks version (5.1) is a little tricky as it is an older version of Luarocks.

Honestly, I needed images in Neovim in order to even consider making the switch from Visual Studio Code, so getting this plugin working was a non-negotiable for me.

Note: This particular configuration of the plugin must be run in Kitty to work. See the backend option if you want to use a different emulator.

lazy.nvim:


-- nvim/lua/plugins/image-nvim.lua

-- For dependencies see
-- `~/github/dotfiles-latest/neovim/nvim-lazyvim/README.md`
--
-- -- Uncomment the following 2 lines if you use the local luarocks installation
-- -- Leave them commented to instead use `luarocks.nvim`
-- -- instead of luarocks.nvim
-- Notice that in the following 2 commands I'm using luaver
-- package.path = package.path
--   .. ";"
--   .. vim.fn.expand("$HOME")
--   .. "/.luaver/luarocks/3.11.0_5.1/share/lua/5.1/magick/?/init.lua"
-- package.path = package.path
--   .. ";"
--   .. vim.fn.expand("$HOME")
--   .. "/.luaver/luarocks/3.11.0_5.1/share/lua/5.1/magick/?.lua"
--
-- -- Here I'm not using luaver, but instead installed lua and luarocks directly through
-- -- homebrew
package.path = package.path .. ";" .. vim.fn.expand "$HOME" .. "/.luarocks/share/lua/5.1/?/init.lua"
package.path = package.path .. ";" .. vim.fn.expand "$HOME" .. "/.luarocks/share/lua/5.1/?.lua"

return {
  {
    -- luarocks.nvim is a Neovim plugin designed to streamline the installation
    -- of luarocks packages directly within Neovim. It simplifies the process
    -- of managing Lua dependencies, ensuring a hassle-free experience for
    -- Neovim users.
    -- https://github.com/vhyrro/luarocks.nvim
    "vhyrro/luarocks.nvim",
    -- this plugin needs to run before anything else
    priority = 1001,
    opts = {
      rocks = { "magick" },
    },
  },
  {
    "3rd/image.nvim",
    lazy = false, -- load on start up
    dependencies = { "luarocks.nvim" },
    config = function()
      require("image").setup {
        backend = "kitty",
        kitty_method = "normal",
        integrations = {
          -- Notice these are the settings for markdown files
          markdown = {
            enabled = true,
            clear_in_insert_mode = false,
            -- Set this to false if you don't want to render images coming from
            -- a URL
            download_remote_images = true,
            -- Change this if you would only like to render the image where the
            -- cursor is at
            -- I set this to true, because if the file has way too many images
            -- it will be laggy and will take time for the initial load
            only_render_image_at_cursor = true,
            -- markdown extensions (ie. quarto) can go here
            filetypes = { "markdown", "vimwiki" },
          },
          neorg = {
            enabled = true,
            clear_in_insert_mode = false,
            download_remote_images = true,
            only_render_image_at_cursor = false,
            filetypes = { "norg" },
          },
          -- This is disabled by default
          -- Detect and render images referenced in HTML files
          -- Make sure you have an html treesitter parser installed
          -- ~/github/dotfiles-latest/neovim/nvim-lazyvim/lua/plugins/treesitter.lua
          html = {
            enabled = false,
          },
          -- This is disabled by default
          -- Detect and render images referenced in CSS files
          -- Make sure you have a css treesitter parser installed
          -- ~/github/dotfiles-latest/neovim/nvim-lazyvim/lua/plugins/treesitter.lua
          css = {
            enabled = true,
          },
        },
        max_width = nil,
        max_height = nil,
        max_width_window_percentage = nil,

        -- This is what I changed to make my images look smaller, like a
        -- thumbnail, the default value is 50
        -- max_height_window_percentage = 20,
        max_height_window_percentage = 40,

        -- toggles images when windows are overlapped
        window_overlap_clear_enabled = false,
        window_overlap_clear_ft_ignore = { "cmp_menu", "cmp_docs", "" },

        -- auto show/hide images when the editor gains/looses focus
        editor_only_render_when_focused = true,

        -- auto show/hide images in the correct tmux window
        -- In the tmux.conf add `set -g visual-activity off`
        tmux_show_only_in_active_window = true,

        -- render image files as images when opened
        hijack_file_patterns = { "*.png", "*.jpg", "*.jpeg", "*.gif", "*.webp", "*.avif" },
      }
    end,
  },
}

rmagatti/autosession

Automatically saves and restores Neovim sessions, allowing you to pick up your work right where you left off. If you close Neovim and reopen it later, autosession will restore your previous session (open files, split windows, etc.). This is incredibly useful when you are juggling multiple projects and want to resume exactly where you left off.

lazy.nvim:


-- automatically creates a Vim session when Neovim opens for saving work
return {
  "rmagatti/auto-session",
  lazy = false,
  config = function()
    local auto_session = require "auto-session"
    auto_session.setup {
      auto_restore = false,
      suppressed_dirs = { "~/", "~/Dev/", "~/Downloads", "~/Documents", "~/Desktop/" },
    }
  end,
}

Then I define these mappings to save and restore sessions:


-- nvim/lua/mappings.lua

-- Autosession mappings
map("n", "<leader>ws", "<cmd>SessionSave<CR>", { desc = "Save session for auto session root dir" })
map("n", "<leader>wr", "<cmd>SessionRestore<CR>", { desc = "Restore session for cwd" })

christoomey/vim-tmux-navigator

Enables smooth navigation between Neovim windows and tmux panes, making it feel like a unified experience. If you’re using tmux and Neovim together,vim-tmux-navigator allows you to navigate between tmux panes and Neovim splits using the same keybindings (like <C-h>,<C-j>, <C-k>, and <C-l>). This makes my workflow more fluid as I don't have to think about whether I'm moving inside Neovim or across Tmux. Not to mention I find this bindings much less awkward than the default <C-w> Vim windown motions.

The true value of this plugins comes when you are working in various tmux panes running a combination of Neovim, a shell, and other tools. Without thinking I can navigate between these panes with ease.

lazy.nvim:


  {
    "christoomey/vim-tmux-navigator",
    lazy = false, -- load on start up to immediately enable
    cmd = {
      "TmuxNavigateLeft",
      "TmuxNavigateDown",
      "TmuxNavigateUp",
      "TmuxNavigateRight",
      "TmuxNavigatePrevious",
    },
    keys = {
      { "<C-h>", "<cmd><C-U>TmuxNavigateLeft<cr>", desc = "Navigate left" },
      { "<C-j>", "<cmd><C-U>TmuxNavigateDown<cr>", desc = "Navigate down" },
      { "<C-k>", "<cmd><C-U>TmuxNavigateUp<cr>", desc = "Navigate up" },
      { "<C-l>", "<cmd><C-U>TmuxNavigateRight<cr>", desc = "Navigate right" },
      { "<C-\\>", "<cmd><C-U>TmuxNavigatePrevious<cr>", desc = "Navigate previous" },
    },
  },

Note: If using NVChad, you will need to override its mapping of these keys to use the Tmux navigation commands instead:


-- nvim/lua/options.lua
require "nvchad.mappings"

local map = vim.keymap.set

-- override nvchad mappings for window navigation so I can use them for
-- vim-tmux-navigator
map("n", "<C-h>", "<cmd>TmuxNavigateLeft<cr>", { desc = "Navigate left" })
map("n", "<C-j>", "<cmd>TmuxNavigateDown<cr>", { desc = "Navigate down" })
map("n", "<C-k>", "<cmd>TmuxNavigateUp<cr>", { desc = "Navigate up" })
map("n", "<C-l>", "<cmd>TmuxNavigateRight<cr>", { desc = "Navigate right" })
map("n", "<C-\\>", "<cmd>TmuxNavigatePrevious<cr>", { desc = "Navigate previous" })

If using tmux, add the following to your ~/.tmux.conf:


# ~/.tmux.conf
# Smart pane switching with awareness of Vim splits.
# See: https://github.com/christoomey/vim-tmux-navigator
is_vim="ps -o state= -o comm= -t '#{pane_tty}' \
    | grep -iqE '^[^TXZ ]+ +(\\S+\\/)?g?(view|l?n?vim?x?|fzf)(diff)?$'"
bind -n 'C-h' if-shell "$is_vim" "send-keys C-h" "select-pane -L"
bind -n 'C-j' if-shell "$is_vim" "send-keys C-j" "select-pane -D"
bind -n 'C-k' if-shell "$is_vim" "send-keys C-k" "select-pane -U"
bind -n 'C-l' if-shell "$is_vim" "send-keys C-l" "select-pane -R"
bind -n 'C-\' if-shell "$is_vim" "send-keys C-\\" "select-pane -l"
tmux_version='$(tmux -V | sed -En "s/^tmux ([0-9]+(.[0-9]+)?).*/\1/p")'
if-shell -b '[ "$(echo "$tmux_version < 3.0" | bc)" = 1 ]' \
    "bind-key -n 'C-\\' if-shell \"$is_vim\" 'send-keys C-\\'  'select-pane -l'"
if-shell -b '[ "$(echo "$tmux_version >= 3.0" | bc)" = 1 ]' \
    "bind-key -n 'C-\\' if-shell \"$is_vim\" 'send-keys C-\\\\'  'select-pane -l'"

bind-key -T copy-mode-vi 'C-h' select-pane -L
bind-key -T copy-mode-vi 'C-j' select-pane -D
bind-key -T copy-mode-vi 'C-k' select-pane -U
bind-key -T copy-mode-vi 'C-l' select-pane -R
bind-key -T copy-mode-vi 'C-\' select-pane -l

# Enable passthrough for panes
set -g allow-passthrough on

pocco81/auto-save.nvim

Automatically saves your files when you leave insert mode. This plugin is a lifesaver for those who forget to save their work frequently. It ensures that your changes are always saved, even if you forget to do so manually. It is also highly customizable, allowing you to configure the save interval and other settings to suit your workflow.

lazy.nvim:


-- nvim/lua/plugins/init.lua
{
"pocco81/auto-save.nvim",
lazy = false,
config = function()
  require("auto-save").setup {}
end,
},

stevearc/dressing.nvim

Dressing.nvim plugin in action

This plugins provides way better styling to the default vim.ui interfaces. The biggest improvement in my opinion is over the vim.input interface that is used for renaming files in nvim-tree and renaming variable definitions using NvChad's renamer. This is another highly customizable plugin that allows you to style the UI to your liking.

lazy.nvim:


-- nvim/lua/plugins/init.lua
{ "stevearc/dressing.nvim", event = "VeryLazy" },

Key Mappings

Similarly, I have a ton of mappings that I have defined so I will just be going over the ones that I can't live without. Additionally, I will not be detailing the mappings that come with the NVChad base configuration (NVChad users can view these by doing <leader>ch to pull up the mappings cheat sheet).

Quality of Life Remaps


-- nvim/lua/mappings.lua
require "nvchad.mappings" -- NVChad-defined mappings

local map = vim.keymap.set

-- Quickly exit Normal mode with `jk`
-- This makes the transition from typing to command mode faster and more comfortable for the fingers.
-- UPDATE: I disabled this and instead just use Karabiner to map capslock to ESC
-- map("i", "jk", "<ESC>")

-- Map <C-s> to save
-- This is a common shortcut in many editors, bringing familiarity from other tools like VSCode.
map({ "n", "i", "v" }, "<C-s>", "<cmd> w <cr>")

"Reveal in Finder" Dupe

One of the things that I was missing coming from Visual Studio Code was the option in the file explorer to "Reveal in Finder" so I created a simple Vim user command that allows me to open a file under in my cursor in a Finder window either from the open file buffer or from the file explorer sidebar.


-- Open file in default viewer
vim.api.nvim_create_user_command("OpenFileInViewer", function()
  local current_file = vim.fn.expand "%:p"
  vim.fn.system('open "' .. current_file .. '"')
end, {})

-- Map <leader>sv to open file in standard viewer
-- NOTE: if cursor is on the file name Nvim Tree, you can simply press `sv` to open the file in the default viewer
-- since the leader key is already used by Nvim Tree.
map("n", "<leader>sv", ":OpenFileInViewer<CR>", { noremap = true, silent = true, desc = "Open file in default viewer" })

Window Management

These mappings manage window splits and layout efficiently in Neovim. The desc key provides a description of the mapping for use in command help or plugins like which-key.nvim.


map("n", "<leader>v", "<C-w>v", { desc = "Split window vertically" }) -- split window vertically
map("n", "<leader>sd", "<cmd>close<CR>", { desc = "Close current split" }) -- close current split window
map("n", "<leader>se", "<C-w>=", { desc = "Equalize splits" }) -- equalize split layouts
-- can't use <leader>sh since that is used by LSPs to signature help
map("n", "<leader>s", "<cmd>split<CR>", { desc = "Split window horizontally" }) -- split window horizobtally

Visually Appealing Scrolling and Searching

These mappings keep the cursor and search result centered in the window while scrolling or jumping to search results.


-- maintain visual context on page navigation and searching
map("n", "<C-d>", "<C-d>zz") -- Keeps cursor centered when going down the page
map("n", "<C-u>", "<C-u>zz") -- Keeps cursor centered when going up the page
map("n", "n", "nzzzv") -- Keeps the search result in the center after jumping to next result
map("n", "N", "Nzzzv") -- Keeps the search result in the center after jumping to previous result

GitSigns Mappings

These are the mappings I use for GitSigns, which I believe are fairly standard. gitsigns works in "hunks" of code, hence all of these mappings begin with an h after the leader.


map("n", "<leader>hn", "<cmd>lua require'gitsigns'.next_hunk()<CR>", { desc = "Next hunk" })
map("n", "<leader>hp", "<cmd>lua require'gitsigns'.prev_hunk()<CR>", { desc = "Previous hunk" })
map("n", "<leader>hs", "<cmd>lua require'gitsigns'.stage_hunk()<CR>", { desc = "Stage hunk" })
map("n", "<leader>hu", "<cmd>lua require'gitsigns'.undo_stage_hunk()<CR>", { desc = "Undo stage hunk" })
map("n", "<leader>hr", "<cmd>lua require'gitsigns'.reset_hunk()<CR>", { desc = "Reset hunk" })
map("n", "<leader>hR", "<cmd>lua require'gitsigns'.reset_buffer()<CR>", { desc = "Reset buffer" })
map("n", "<leader>hp", "<cmd>lua require'gitsigns'.preview_hunk()<CR>", { desc = "Preview hunk" })
map("n", "<leader>hb", "<cmd>lua require'gitsigns'.blame_line()<CR>", { desc = "Blame line" })
map("n", "<leader>hS", "<cmd>lua require'gitsigns'.stage_buffer()<CR>", { desc = "Stage buffer" })
map("n", "<leader>hU", "<cmd>lua require'gitsigns'.reset_buffer_index()<CR>", { desc = "Reset buffer index" })

Options

These options provide a basic, yet highly functional configuration for Neovim that enhances usability by enabling features like clipboard integration and better search behavior.


-- nvim/lua/options.lua

require "nvchad.options"

local o = vim.o
local opt = vim.opt

-- Cursorline
o.cursorlineopt = "both" -- to enable cursorline!

-- Make line numbers defaults
o.number = true

-- Search and replace
opt.ignorecase = true -- ignore case letters when searching
opt.smartcase = true -- match case if capital letter is present
opt.incsearch = true -- show search matches as you type
opt.inccommand = "split" -- show live preview of substitute commands

-- disable swapfile
opt.swapfile = false

-- clipboard
opt.clipboard:append "unnamedplus"

-- text wrapping
o.wrap = false

Conclusion

While transitioning from VSCode to Neovim wasn’t without its challenges— especially getting used to Vim motions and configuring the plugins to my liking—it has ultimately made my workflow faster and more enjoyable. I particularly appreciated how customizable Neovim is; I can tailor it exactly to my needs, unlike most GUI-based editors.

It took me about a month to feel completely comfortable using Neovim, but once the muscle memory kicks in, you start to realize the immense productivity boost. If you’re on the fence, I highly recommend giving Neovim a chance for at least a few weeks.

If you have any questions about my setup, want to share your own tips and tricks, or just want to chat about Neovim, feel free to write in the comments below!

References