Make Nvim Blazingly Fast

How to Track Down the Lag

First, try to reproduce the freeze or lag. Once you can consistently trigger the sluggishness, start disabling your custom plugins, autocmds, and options until you pinpoint the culprit.

But what if nothing seems to work? What if the lag only appears after hours of coding? Then my friend, profile.nvim is your new best buddy. When you notice lag consistently, fire up profile.nvim to capture everything nvim executes. Just keep the recording short—long enough to catch what happens between your keystrokes.

Next, pass the generated profile.json to a GUI like Perfetto UI. There, you can see which functions run when you press j (or any key) and how long they take, ultimately revealing the sneaky offender.

Perfetto UI

EDIT (March 27, 2025): snacks.nvim includes a profiler tool, which I prefer over profile.nvim. snacks profiler

Tweaking options

Disabling built-in syntax highlighting

I set syntax to manual so that I can still use nvim’s built-in syntax for filetypes that lack treesitter support. This configuration lets you manually enable syntax for those filetypes with an autocmd:

vim.cmd.syntax("manual")

Whenever a filetype doesn’t support treesitter, you can enable its syntax using the following autocmd:

vim.api.nvim_create_autocmd("FileType", {
  pattern = { "gitsendemail", "conf", "editorconfig", "qf", "checkhealth", "less" },
  callback = function(event)
    vim.bo[event.buf].syntax = vim.bo[event.buf].filetype
  end,
})

Setting synmaxcol

The synmaxcol option limits how much of a long horizontal line is highlighted by nvim’s syntax engine. This setting is especially useful when opening files with extremely long lines, as it prevents nvim from freezing while attempting to highlight the entire line:

vim.opt.synmaxcol = 500

Managing Signs

Try to avoid enabling signs where possible, as they can add performance overhead—especially when used with custom status column plugins like statuscol.nvim. For instance, I disable diagnostic signs since I often have numerous diagnostics (hints and warnings) on screen. Similarly, with plugins like gitsigns.nvim, showing signs for new files can also impact performance.

Disabling Inlay Hints in Insert Mode

Inlay hints execute on every keystroke. If your language server protocol (LSP) is performing poorly, these frequent updates can block nvim. To mitigate this issue, disable inlay hints when entering insert mode and re-enable them upon leaving:

vim.api.nvim_create_autocmd("InsertEnter", {
  pattern = "*",
  callback = function(event)
    vim.schedule(function()
      vim.lsp.inlay_hint.enable(false, { bufnr = event.buf })
    end)
  end,
})
vim.api.nvim_create_autocmd("InsertLeave", {
  group = utils.augroup("enable_inlay_hints"),
  pattern = "*",
  callback = function(event)
    vim.schedule(function()
      vim.lsp.inlay_hint.enable(true, { bufnr = event.buf })
    end)
  end,
})

LSP

Since I spend most of my time juggling React and Vue, here’s a quick tip: if you’re using the eslint LSP, configure it to run only on save—because ain’t nobody got time for constant linting. Also, consider ditching ts_ls in favor of vtsls; your editor (and your sanity) will thank you.

-- eslint LSP config
{
  settings = {
    run = "onSave",
  }
}

Some LSPs can return a large number of completions, such as vtsls. This may cause your completion plugin (e.g., blink.cmp, nvim-cmp) to lag. To prevent this, you should limit the number of completions returned by the LSP. For example, to restrict the number of completions provided by vtsls, you can modify its configuration as follows:

-- vtsls LSP config
{
  settings = {
    vtsls = {
      experimental = {
        completion = {
          enableServerSideFuzzyMatch = true,
          entriesLimit = 20,
        },
      },
    },
  },
}

Tweaking Plugins

nvim-treesitter

Disable nvim-treesitter highlighting for files that are either too large or detected as minified. This prevents performance issues when processing such files. Also, disable additional_vim_regex_highlighting:

highlight = {
  enable = true,
  additional_vim_regex_highlighting = false,
  disable = function(_, bufnr)
    -- Return true if the buffer is a minified file or if its size is too large.
  end,
},

You can view my implementation of the treesitter.highlight.disable function here and here.

nvim-tree.lua

Be sure to check out the “Performance Tips” on its wiki page. If you have filesystem_watchers enabled, make sure to ignore directories with a large number of files, such as node_modules. Additionally, disable the modified option since it can cause lag in large codebases:

{
  filesystem_watchers = {
    enable = true,
    -- I personally ignore the `.git` and `.direnv` directories since I use https://direnv.net
    ignore_dirs = { "node_modules", ".direnv", ".git" },
  },
  modified = {
    enabled = false
  }
}

snacks.nvim

I highly recommend replacing certain plugins with their Snacks counterparts, as Snacks is blazingly fast, for example: