Last active
April 6, 2026 23:34
-
-
Save wilyJ80/6f9be00c8db9355b600f2fa4381ec1ed to your computer and use it in GitHub Desktop.
Neovim Config
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| -- Personal preferences | |
| vim.opt.tabstop = 4 | |
| vim.opt.shiftwidth = 4 | |
| vim.opt.softtabstop = 4 | |
| -- Other custom required stuff | |
| vim.opt.termguicolors = true | |
| -- Set <space> as the leader key | |
| -- See `:help mapleader` | |
| -- NOTE: Must happen before plugins are loaded (otherwise wrong leader will be used) | |
| vim.g.mapleader = " " | |
| vim.g.maplocalleader = " " | |
| -- [[ Setting options ]] | |
| -- See `:help vim.o` | |
| -- NOTE: You can change these options as you wish! | |
| -- For more options, you can see `:help option-list` | |
| -- Make line numbers default | |
| vim.o.number = true | |
| -- Enable mouse mode, can be useful for resizing splits for example! | |
| vim.o.mouse = "a" | |
| -- Don't show the mode, since it's already in the status line | |
| vim.o.showmode = false | |
| -- Sync clipboard between OS and Neovim. | |
| -- Schedule the setting after `UiEnter` because it can increase startup-time. | |
| -- Remove this option if you want your OS clipboard to remain independent. | |
| -- See `:help 'clipboard'` | |
| vim.schedule(function() | |
| vim.o.clipboard = "unnamedplus" | |
| end) | |
| -- Enable break indent | |
| vim.o.breakindent = true | |
| -- Enable undo/redo changes even after closing and reopening a file | |
| vim.o.undofile = true | |
| -- Case-insensitive searching UNLESS \C or one or more capital letters in the search term | |
| vim.o.ignorecase = true | |
| vim.o.smartcase = true | |
| -- Keep signcolumn on by default | |
| vim.o.signcolumn = "yes" | |
| -- Decrease update time | |
| vim.o.updatetime = 250 | |
| -- Decrease mapped sequence wait time | |
| vim.o.timeoutlen = 300 | |
| -- Configure how new splits should be opened | |
| vim.o.splitright = true | |
| vim.o.splitbelow = true | |
| -- Sets how neovim will display certain whitespace characters in the editor. | |
| -- See `:help 'list'` | |
| -- and `:help 'listchars'` | |
| -- | |
| -- Notice listchars is set using `vim.opt` instead of `vim.o`. | |
| -- It is very similar to `vim.o` but offers an interface for conveniently interacting with tables. | |
| -- See `:help lua-options` | |
| -- and `:help lua-guide-options` | |
| vim.o.list = true | |
| vim.opt.listchars = { tab = "Β» ", trail = "Β·", nbsp = "β£" } | |
| -- Preview substitutions live, as you type! | |
| vim.o.inccommand = "split" | |
| -- Show which line your cursor is on | |
| vim.o.cursorline = true | |
| -- if performing an operation that would fail due to unsaved changes in the buffer (like `:q`), | |
| -- instead raise a dialog asking if you wish to save the current file(s) | |
| -- See `:help 'confirm'` | |
| vim.o.confirm = true | |
| -- [[ Basic Keymaps ]] | |
| -- See `:help vim.keymap.set()` | |
| -- Clear highlights on search when pressing <Esc> in normal mode | |
| -- See `:help hlsearch` | |
| vim.keymap.set("n", "<Esc>", "<cmd>nohlsearch<CR>") | |
| -- Diagnostic Config & Keymaps | |
| -- See :help vim.diagnostic.Opts | |
| vim.diagnostic.config({ | |
| update_in_insert = false, | |
| severity_sort = true, | |
| float = { border = "rounded", source = "if_many" }, | |
| underline = { severity = { min = vim.diagnostic.severity.WARN } }, | |
| -- Can switch between these as you prefer | |
| virtual_text = true, -- Text shows up at the end of the line | |
| virtual_lines = false, -- Text shows up underneath the line, with virtual lines | |
| -- Auto open the float, so you can easily read the errors when jumping with `[d` and `]d` | |
| jump = { float = true }, | |
| }) | |
| vim.keymap.set("n", "<leader>q", vim.diagnostic.setloclist, { desc = "Open diagnostic [Q]uickfix list" }) | |
| -- Exit terminal mode in the builtin terminal with a shortcut that is a bit easier | |
| -- for people to discover. Otherwise, you normally need to press <C-\><C-n>, which | |
| -- is not what someone will guess without a bit more experience. | |
| -- | |
| -- NOTE: This won't work in all terminal emulators/tmux/etc. Try your own mapping | |
| -- or just use <C-\><C-n> to exit terminal mode | |
| vim.keymap.set("t", "<Esc><Esc>", "<C-\\><C-n>", { desc = "Exit terminal mode" }) | |
| -- Keybinds to make split navigation easier. | |
| -- Use CTRL+<hjkl> to switch between windows | |
| -- | |
| -- See `:help wincmd` for a list of all window commands | |
| vim.keymap.set("n", "<C-h>", "<C-w><C-h>", { desc = "Move focus to the left window" }) | |
| vim.keymap.set("n", "<C-l>", "<C-w><C-l>", { desc = "Move focus to the right window" }) | |
| vim.keymap.set("n", "<C-j>", "<C-w><C-j>", { desc = "Move focus to the lower window" }) | |
| vim.keymap.set("n", "<C-k>", "<C-w><C-k>", { desc = "Move focus to the upper window" }) | |
| -- [[ Basic Autocommands ]] | |
| -- See `:help lua-guide-autocommands` | |
| -- Highlight when yanking (copying) text | |
| -- Try it with `yap` in normal mode | |
| -- See `:help vim.hl.on_yank()` | |
| vim.api.nvim_create_autocmd("TextYankPost", { | |
| desc = "Highlight when yanking (copying) text", | |
| group = vim.api.nvim_create_augroup("kickstart-highlight-yank", { clear = true }), | |
| callback = function() | |
| vim.hl.on_yank() | |
| end, | |
| }) | |
| -- [[ Install `lazy.nvim` plugin manager ]] | |
| -- See `:help lazy.nvim.txt` or https://github.com/folke/lazy.nvim for more info | |
| local lazypath = vim.fn.stdpath("data") .. "/lazy/lazy.nvim" | |
| if not (vim.uv or vim.loop).fs_stat(lazypath) then | |
| local lazyrepo = "https://github.com/folke/lazy.nvim.git" | |
| local out = vim.fn.system({ "git", "clone", "--filter=blob:none", "--branch=stable", lazyrepo, lazypath }) | |
| if vim.v.shell_error ~= 0 then | |
| error("Error cloning lazy.nvim:\n" .. out) | |
| end | |
| end | |
| ---@type vim.Option | |
| local rtp = vim.opt.rtp | |
| rtp:prepend(lazypath) | |
| -- [[ Configure and install plugins ]] | |
| -- | |
| -- To check the current status of your plugins, run | |
| -- :Lazy | |
| -- | |
| -- You can press `?` in this menu for help. Use `:q` to close the window | |
| -- | |
| -- To update plugins you can run | |
| -- :Lazy update | |
| -- | |
| -- NOTE: Here is where you install your plugins. | |
| require("lazy").setup({ | |
| -- KICKSTART BUILTIN | |
| { | |
| "windwp/nvim-autopairs", | |
| event = "InsertEnter", | |
| opts = {}, | |
| }, | |
| { | |
| "lewis6991/gitsigns.nvim", | |
| ---@module 'gitsigns' | |
| ---@type Gitsigns.Config | |
| ---@diagnostic disable-next-line: missing-fields | |
| opts = { | |
| on_attach = function(bufnr) | |
| local gitsigns = require("gitsigns") | |
| local function map(mode, l, r, opts) | |
| opts = opts or {} | |
| opts.buffer = bufnr | |
| vim.keymap.set(mode, l, r, opts) | |
| end | |
| -- Navigation | |
| map("n", "]c", function() | |
| if vim.wo.diff then | |
| vim.cmd.normal({ "]c", bang = true }) | |
| else | |
| gitsigns.nav_hunk("next") | |
| end | |
| end, { desc = "Jump to next git [c]hange" }) | |
| map("n", "[c", function() | |
| if vim.wo.diff then | |
| vim.cmd.normal({ "[c", bang = true }) | |
| else | |
| gitsigns.nav_hunk("prev") | |
| end | |
| end, { desc = "Jump to previous git [c]hange" }) | |
| -- Actions | |
| -- visual mode | |
| map("v", "<leader>hs", function() | |
| gitsigns.stage_hunk({ vim.fn.line("."), vim.fn.line("v") }) | |
| end, { desc = "git [s]tage hunk" }) | |
| map("v", "<leader>hr", function() | |
| gitsigns.reset_hunk({ vim.fn.line("."), vim.fn.line("v") }) | |
| end, { desc = "git [r]eset hunk" }) | |
| -- normal mode | |
| map("n", "<leader>hs", gitsigns.stage_hunk, { desc = "git [s]tage hunk" }) | |
| map("n", "<leader>hr", gitsigns.reset_hunk, { desc = "git [r]eset hunk" }) | |
| map("n", "<leader>hS", gitsigns.stage_buffer, { desc = "git [S]tage buffer" }) | |
| map("n", "<leader>hR", gitsigns.reset_buffer, { desc = "git [R]eset buffer" }) | |
| map("n", "<leader>hp", gitsigns.preview_hunk, { desc = "git [p]review hunk" }) | |
| map("n", "<leader>hi", gitsigns.preview_hunk_inline, { desc = "git preview hunk [i]nline" }) | |
| map("n", "<leader>hb", function() | |
| gitsigns.blame_line({ full = true }) | |
| end, { desc = "git [b]lame line" }) | |
| map("n", "<leader>hd", gitsigns.diffthis, { desc = "git [d]iff against index" }) | |
| map("n", "<leader>hD", function() | |
| gitsigns.diffthis("@") | |
| end, { desc = "git [D]iff against last commit" }) | |
| map("n", "<leader>hQ", function() | |
| gitsigns.setqflist("all") | |
| end) | |
| map("n", "<leader>hq", gitsigns.setqflist) | |
| -- Toggles | |
| map("n", "<leader>tb", gitsigns.toggle_current_line_blame, { desc = "[T]oggle git show [b]lame line" }) | |
| map("n", "<leader>tw", gitsigns.toggle_word_diff) | |
| -- Text object | |
| map({ "o", "x" }, "ih", gitsigns.select_hunk) | |
| end, | |
| }, | |
| }, | |
| -- NOTE: Plugins can be added via a link or github org/name. To run setup automatically, use `opts = {}` | |
| { "NMAC427/guess-indent.nvim", opts = {} }, | |
| -- Alternatively, use `config = function() ... end` for full control over the configuration. | |
| -- If you prefer to call `setup` explicitly, use: | |
| -- { | |
| -- 'lewis6991/gitsigns.nvim', | |
| -- config = function() | |
| -- require('gitsigns').setup({ | |
| -- -- Your gitsigns configuration here | |
| -- }) | |
| -- end, | |
| -- } | |
| -- | |
| -- Here is a more advanced example where we pass configuration | |
| -- options to `gitsigns.nvim`. | |
| -- | |
| -- See `:help gitsigns` to understand what the configuration keys do | |
| { -- Adds git related signs to the gutter, as well as utilities for managing changes | |
| "lewis6991/gitsigns.nvim", | |
| ---@module 'gitsigns' | |
| ---@type Gitsigns.Config | |
| ---@diagnostic disable-next-line: missing-fields | |
| opts = { | |
| signs = { | |
| add = { text = "+" }, ---@diagnostic disable-line: missing-fields | |
| change = { text = "~" }, ---@diagnostic disable-line: missing-fields | |
| delete = { text = "_" }, ---@diagnostic disable-line: missing-fields | |
| topdelete = { text = "βΎ" }, ---@diagnostic disable-line: missing-fields | |
| changedelete = { text = "~" }, ---@diagnostic disable-line: missing-fields | |
| }, | |
| }, | |
| }, | |
| { | |
| "nvim-neo-tree/neo-tree.nvim", | |
| version = "*", | |
| dependencies = { | |
| "nvim-lua/plenary.nvim", | |
| "nvim-tree/nvim-web-devicons", -- not strictly required, but recommended | |
| "MunifTanjim/nui.nvim", | |
| }, | |
| lazy = false, | |
| keys = { | |
| { "\\", ":Neotree reveal<CR>", desc = "NeoTree reveal", silent = true }, | |
| }, | |
| opts = { | |
| filesystem = { | |
| window = { | |
| mappings = { | |
| ["\\"] = "close_window", | |
| }, | |
| }, | |
| }, | |
| }, | |
| }, | |
| { "akinsho/toggleterm.nvim", opts = { | |
| open_mapping = [[<F12>]], | |
| direction = "float", | |
| } }, | |
| { | |
| "norcalli/nvim-colorizer.lua", | |
| config = function() | |
| require("colorizer").setup({ | |
| "*", -- highlight all filetypes, or pick specific ones | |
| }, { | |
| RGB = true, -- #RGB hex codes | |
| RRGGBB = true, -- #RRGGBB hex codes | |
| names = true, -- "Blue", "red", etc. | |
| RRGGBBAA = true, -- #RRGGBBAA | |
| rgb_fn = true, -- rgb() and rgba() functions | |
| hsl_fn = true, -- hsl() and hsla() functions | |
| css = true, -- Enable all CSS features: rgb_fn, hsl_fn, names, etc. | |
| css_fn = true, -- Enable all CSS *functions*: rgb_fn, hsl_fn, etc. | |
| }) | |
| end, | |
| }, | |
| -- NOTE: Plugins can also be configured to run Lua code when they are loaded. | |
| -- | |
| -- This is often very useful to both group configuration, as well as handle | |
| -- lazy loading plugins that don't need to be loaded immediately at startup. | |
| -- | |
| -- For example, in the following configuration, we use: | |
| -- event = 'VimEnter' | |
| -- | |
| -- which loads which-key before all the UI elements are loaded. Events can be | |
| -- normal autocommands events (`:help autocmd-events`). | |
| -- | |
| -- Then, because we use the `opts` key (recommended), the configuration runs | |
| -- after the plugin has been loaded as `require(MODULE).setup(opts)`. | |
| { -- Useful plugin to show you pending keybinds. | |
| "folke/which-key.nvim", | |
| event = "VimEnter", | |
| ---@module 'which-key' | |
| ---@type wk.Opts | |
| ---@diagnostic disable-next-line: missing-fields | |
| opts = { | |
| -- delay between pressing a key and opening which-key (milliseconds) | |
| delay = 0, | |
| icons = { mappings = vim.g.have_nerd_font }, | |
| -- Document existing key chains | |
| spec = { | |
| { "<leader>s", group = "[S]earch", mode = { "n", "v" } }, | |
| { "<leader>t", group = "[T]oggle" }, | |
| { "<leader>h", group = "Git [H]unk", mode = { "n", "v" } }, -- Enable gitsigns recommended keymaps first | |
| { "gr", group = "LSP Actions", mode = { "n" } }, | |
| }, | |
| }, | |
| }, | |
| -- NOTE: Plugins can specify dependencies. | |
| -- | |
| -- The dependencies are proper plugin specifications as well - anything | |
| -- you do for a plugin at the top level, you can do for a dependency. | |
| -- | |
| -- Use the `dependencies` key to specify the dependencies of a particular plugin | |
| { -- Fuzzy Finder (files, lsp, etc) | |
| "nvim-telescope/telescope.nvim", | |
| -- By default, Telescope is included and acts as your picker for everything. | |
| -- If you would like to switch to a different picker (like snacks, or fzf-lua) | |
| -- you can disable the Telescope plugin by setting enabled to false and enable | |
| -- your replacement picker by requiring it explicitly (e.g. 'custom.plugins.snacks') | |
| -- Note: If you customize your config for yourself, | |
| -- itβs best to remove the Telescope plugin config entirely | |
| -- instead of just disabling it here, to keep your config clean. | |
| enabled = true, | |
| event = "VimEnter", | |
| dependencies = { | |
| "nvim-lua/plenary.nvim", | |
| { -- If encountering errors, see telescope-fzf-native README for installation instructions | |
| "nvim-telescope/telescope-fzf-native.nvim", | |
| -- `build` is used to run some command when the plugin is installed/updated. | |
| -- This is only run then, not every time Neovim starts up. | |
| build = "make", | |
| -- `cond` is a condition used to determine whether this plugin should be | |
| -- installed and loaded. | |
| cond = function() | |
| return vim.fn.executable("make") == 1 | |
| end, | |
| }, | |
| { "nvim-telescope/telescope-ui-select.nvim" }, | |
| -- Useful for getting pretty icons, but requires a Nerd Font. | |
| { "nvim-tree/nvim-web-devicons", enabled = vim.g.have_nerd_font }, | |
| }, | |
| config = function() | |
| -- Telescope is a fuzzy finder that comes with a lot of different things that | |
| -- it can fuzzy find! It's more than just a "file finder", it can search | |
| -- many different aspects of Neovim, your workspace, LSP, and more! | |
| -- | |
| -- The easiest way to use Telescope, is to start by doing something like: | |
| -- :Telescope help_tags | |
| -- | |
| -- After running this command, a window will open up and you're able to | |
| -- type in the prompt window. You'll see a list of `help_tags` options and | |
| -- a corresponding preview of the help. | |
| -- | |
| -- Two important keymaps to use while in Telescope are: | |
| -- - Insert mode: <c-/> | |
| -- - Normal mode: ? | |
| -- | |
| -- This opens a window that shows you all of the keymaps for the current | |
| -- Telescope picker. This is really useful to discover what Telescope can | |
| -- do as well as how to actually do it! | |
| -- [[ Configure Telescope ]] | |
| -- See `:help telescope` and `:help telescope.setup()` | |
| require("telescope").setup({ | |
| -- You can put your default mappings / updates / etc. in here | |
| -- All the info you're looking for is in `:help telescope.setup()` | |
| -- | |
| -- defaults = { | |
| -- mappings = { | |
| -- i = { ['<c-enter>'] = 'to_fuzzy_refine' }, | |
| -- }, | |
| -- }, | |
| -- pickers = {} | |
| extensions = { | |
| ["ui-select"] = { require("telescope.themes").get_dropdown() }, | |
| }, | |
| }) | |
| -- Enable Telescope extensions if they are installed | |
| pcall(require("telescope").load_extension, "fzf") | |
| pcall(require("telescope").load_extension, "ui-select") | |
| -- See `:help telescope.builtin` | |
| local builtin = require("telescope.builtin") | |
| vim.keymap.set("n", "<leader>sh", builtin.help_tags, { desc = "[S]earch [H]elp" }) | |
| vim.keymap.set("n", "<leader>sk", builtin.keymaps, { desc = "[S]earch [K]eymaps" }) | |
| vim.keymap.set("n", "<leader>sf", builtin.find_files, { desc = "[S]earch [F]iles" }) | |
| vim.keymap.set("n", "<leader>ss", builtin.builtin, { desc = "[S]earch [S]elect Telescope" }) | |
| vim.keymap.set({ "n", "v" }, "<leader>sw", builtin.grep_string, { desc = "[S]earch current [W]ord" }) | |
| vim.keymap.set("n", "<leader>sg", builtin.live_grep, { desc = "[S]earch by [G]rep" }) | |
| vim.keymap.set("n", "<leader>sd", builtin.diagnostics, { desc = "[S]earch [D]iagnostics" }) | |
| vim.keymap.set("n", "<leader>sr", builtin.resume, { desc = "[S]earch [R]esume" }) | |
| vim.keymap.set("n", "<leader>s.", builtin.oldfiles, { desc = '[S]earch Recent Files ("." for repeat)' }) | |
| vim.keymap.set("n", "<leader>sc", builtin.commands, { desc = "[S]earch [C]ommands" }) | |
| vim.keymap.set("n", "<leader><leader>", builtin.buffers, { desc = "[ ] Find existing buffers" }) | |
| -- This runs on LSP attach per buffer (see main LSP attach function in 'neovim/nvim-lspconfig' config for more info, | |
| -- it is better explained there). This allows easily switching between pickers if you prefer using something else! | |
| vim.api.nvim_create_autocmd("LspAttach", { | |
| group = vim.api.nvim_create_augroup("telescope-lsp-attach", { clear = true }), | |
| callback = function(event) | |
| local buf = event.buf | |
| -- Find references for the word under your cursor. | |
| vim.keymap.set("n", "grr", builtin.lsp_references, { buffer = buf, desc = "[G]oto [R]eferences" }) | |
| -- Jump to the implementation of the word under your cursor. | |
| -- Useful when your language has ways of declaring types without an actual implementation. | |
| vim.keymap.set( | |
| "n", | |
| "gri", | |
| builtin.lsp_implementations, | |
| { buffer = buf, desc = "[G]oto [I]mplementation" } | |
| ) | |
| -- Jump to the definition of the word under your cursor. | |
| -- This is where a variable was first declared, or where a function is defined, etc. | |
| -- To jump back, press <C-t>. | |
| vim.keymap.set("n", "grd", builtin.lsp_definitions, { buffer = buf, desc = "[G]oto [D]efinition" }) | |
| -- Fuzzy find all the symbols in your current document. | |
| -- Symbols are things like variables, functions, types, etc. | |
| vim.keymap.set( | |
| "n", | |
| "gO", | |
| builtin.lsp_document_symbols, | |
| { buffer = buf, desc = "Open Document Symbols" } | |
| ) | |
| -- Fuzzy find all the symbols in your current workspace. | |
| -- Similar to document symbols, except searches over your entire project. | |
| vim.keymap.set( | |
| "n", | |
| "gW", | |
| builtin.lsp_dynamic_workspace_symbols, | |
| { buffer = buf, desc = "Open Workspace Symbols" } | |
| ) | |
| -- Jump to the type of the word under your cursor. | |
| -- Useful when you're not sure what type a variable is and you want to see | |
| -- the definition of its *type*, not where it was *defined*. | |
| vim.keymap.set( | |
| "n", | |
| "grt", | |
| builtin.lsp_type_definitions, | |
| { buffer = buf, desc = "[G]oto [T]ype Definition" } | |
| ) | |
| end, | |
| }) | |
| -- Override default behavior and theme when searching | |
| vim.keymap.set("n", "<leader>/", function() | |
| -- You can pass additional configuration to Telescope to change the theme, layout, etc. | |
| builtin.current_buffer_fuzzy_find(require("telescope.themes").get_dropdown({ | |
| winblend = 10, | |
| previewer = false, | |
| })) | |
| end, { desc = "[/] Fuzzily search in current buffer" }) | |
| -- It's also possible to pass additional configuration options. | |
| -- See `:help telescope.builtin.live_grep()` for information about particular keys | |
| vim.keymap.set("n", "<leader>s/", function() | |
| builtin.live_grep({ | |
| grep_open_files = true, | |
| prompt_title = "Live Grep in Open Files", | |
| }) | |
| end, { desc = "[S]earch [/] in Open Files" }) | |
| -- Shortcut for searching your Neovim configuration files | |
| vim.keymap.set("n", "<leader>sn", function() | |
| builtin.find_files({ cwd = vim.fn.stdpath("config") }) | |
| end, { desc = "[S]earch [N]eovim files" }) | |
| end, | |
| }, | |
| -- LSP Plugins | |
| { | |
| -- Main LSP Configuration | |
| "neovim/nvim-lspconfig", | |
| dependencies = { | |
| -- Automatically install LSPs and related tools to stdpath for Neovim | |
| -- Mason must be loaded before its dependents so we need to set it up here. | |
| -- NOTE: `opts = {}` is the same as calling `require('mason').setup({})` | |
| { | |
| "mason-org/mason.nvim", | |
| ---@module 'mason.settings' | |
| ---@type MasonSettings | |
| ---@diagnostic disable-next-line: missing-fields | |
| opts = {}, | |
| }, | |
| -- Maps LSP server names between nvim-lspconfig and Mason package names. | |
| "mason-org/mason-lspconfig.nvim", | |
| "WhoIsSethDaniel/mason-tool-installer.nvim", | |
| -- Useful status updates for LSP. | |
| { "j-hui/fidget.nvim", opts = {} }, | |
| }, | |
| config = function() | |
| -- Brief aside: **What is LSP?** | |
| -- | |
| -- LSP is an initialism you've probably heard, but might not understand what it is. | |
| -- | |
| -- LSP stands for Language Server Protocol. It's a protocol that helps editors | |
| -- and language tooling communicate in a standardized fashion. | |
| -- | |
| -- In general, you have a "server" which is some tool built to understand a particular | |
| -- language (such as `gopls`, `lua_ls`, `rust_analyzer`, etc.). These Language Servers | |
| -- (sometimes called LSP servers, but that's kind of like ATM Machine) are standalone | |
| -- processes that communicate with some "client" - in this case, Neovim! | |
| -- | |
| -- LSP provides Neovim with features like: | |
| -- - Go to definition | |
| -- - Find references | |
| -- - Autocompletion | |
| -- - Symbol Search | |
| -- - and more! | |
| -- | |
| -- Thus, Language Servers are external tools that must be installed separately from | |
| -- Neovim. This is where `mason` and related plugins come into play. | |
| -- | |
| -- If you're wondering about lsp vs treesitter, you can check out the wonderfully | |
| -- and elegantly composed help section, `:help lsp-vs-treesitter` | |
| -- This function gets run when an LSP attaches to a particular buffer. | |
| -- That is to say, every time a new file is opened that is associated with | |
| -- an lsp (for example, opening `main.rs` is associated with `rust_analyzer`) this | |
| -- function will be executed to configure the current buffer | |
| vim.api.nvim_create_autocmd("LspAttach", { | |
| group = vim.api.nvim_create_augroup("kickstart-lsp-attach", { clear = true }), | |
| callback = function(event) | |
| -- NOTE: Remember that Lua is a real programming language, and as such it is possible | |
| -- to define small helper and utility functions so you don't have to repeat yourself. | |
| -- | |
| -- In this case, we create a function that lets us more easily define mappings specific | |
| -- for LSP related items. It sets the mode, buffer and description for us each time. | |
| local map = function(keys, func, desc, mode) | |
| mode = mode or "n" | |
| vim.keymap.set(mode, keys, func, { buffer = event.buf, desc = "LSP: " .. desc }) | |
| end | |
| -- Rename the variable under your cursor. | |
| -- Most Language Servers support renaming across files, etc. | |
| map("grn", vim.lsp.buf.rename, "[R]e[n]ame") | |
| -- Execute a code action, usually your cursor needs to be on top of an error | |
| -- or a suggestion from your LSP for this to activate. | |
| map("gra", vim.lsp.buf.code_action, "[G]oto Code [A]ction", { "n", "x" }) | |
| -- WARN: This is not Goto Definition, this is Goto Declaration. | |
| -- For example, in C this would take you to the header. | |
| map("grD", vim.lsp.buf.declaration, "[G]oto [D]eclaration") | |
| -- The following two autocommands are used to highlight references of the | |
| -- word under your cursor when your cursor rests there for a little while. | |
| -- See `:help CursorHold` for information about when this is executed | |
| -- | |
| -- When you move your cursor, the highlights will be cleared (the second autocommand). | |
| local client = vim.lsp.get_client_by_id(event.data.client_id) | |
| if client and client:supports_method("textDocument/documentHighlight", event.buf) then | |
| local highlight_augroup = | |
| vim.api.nvim_create_augroup("kickstart-lsp-highlight", { clear = false }) | |
| vim.api.nvim_create_autocmd({ "CursorHold", "CursorHoldI" }, { | |
| buffer = event.buf, | |
| group = highlight_augroup, | |
| callback = vim.lsp.buf.document_highlight, | |
| }) | |
| vim.api.nvim_create_autocmd({ "CursorMoved", "CursorMovedI" }, { | |
| buffer = event.buf, | |
| group = highlight_augroup, | |
| callback = vim.lsp.buf.clear_references, | |
| }) | |
| vim.api.nvim_create_autocmd("LspDetach", { | |
| group = vim.api.nvim_create_augroup("kickstart-lsp-detach", { clear = true }), | |
| callback = function(event2) | |
| vim.lsp.buf.clear_references() | |
| vim.api.nvim_clear_autocmds({ group = "kickstart-lsp-highlight", buffer = event2.buf }) | |
| end, | |
| }) | |
| end | |
| -- The following code creates a keymap to toggle inlay hints in your | |
| -- code, if the language server you are using supports them | |
| -- | |
| -- This may be unwanted, since they displace some of your code | |
| if client and client:supports_method("textDocument/inlayHint", event.buf) then | |
| map("<leader>th", function() | |
| vim.lsp.inlay_hint.enable(not vim.lsp.inlay_hint.is_enabled({ bufnr = event.buf })) | |
| end, "[T]oggle Inlay [H]ints") | |
| end | |
| end, | |
| }) | |
| -- Enable the following language servers | |
| -- Feel free to add/remove any LSPs that you want here. They will automatically be installed. | |
| -- See `:help lsp-config` for information about keys and how to configure | |
| ---@type table<string, vim.lsp.Config> | |
| local servers = { | |
| -- clangd = {}, | |
| -- gopls = {}, | |
| -- pyright = {}, | |
| -- rust_analyzer = {}, | |
| -- | |
| -- Some languages (like typescript) have entire language plugins that can be useful: | |
| -- https://github.com/pmizio/typescript-tools.nvim | |
| -- | |
| -- But for many setups, the LSP (`ts_ls`) will work just fine | |
| -- ts_ls = {}, | |
| -- stylua = {}, -- Used to format Lua code | |
| -- Special Lua Config, as recommended by neovim help docs | |
| lua_ls = { | |
| on_init = function(client) | |
| if client.workspace_folders then | |
| local path = client.workspace_folders[1].name | |
| if | |
| path ~= vim.fn.stdpath("config") | |
| and (vim.uv.fs_stat(path .. "/.luarc.json") or vim.uv.fs_stat(path .. "/.luarc.jsonc")) | |
| then | |
| return | |
| end | |
| end | |
| client.config.settings.Lua = vim.tbl_deep_extend("force", client.config.settings.Lua, { | |
| runtime = { | |
| version = "LuaJIT", | |
| path = { "lua/?.lua", "lua/?/init.lua" }, | |
| }, | |
| workspace = { | |
| checkThirdParty = false, | |
| -- NOTE: this is a lot slower and will cause issues when working on your own configuration. | |
| -- See https://github.com/neovim/nvim-lspconfig/issues/3189 | |
| library = vim.tbl_extend("force", vim.api.nvim_get_runtime_file("", true), { | |
| "${3rd}/luv/library", | |
| "${3rd}/busted/library", | |
| }), | |
| }, | |
| }) | |
| end, | |
| settings = { | |
| Lua = {}, | |
| }, | |
| }, | |
| } | |
| -- Ensure the servers and tools above are installed | |
| -- | |
| -- To check the current status of installed tools and/or manually install | |
| -- other tools, you can run | |
| -- :Mason | |
| -- | |
| -- You can press `g?` for help in this menu. | |
| local ensure_installed = vim.tbl_keys(servers or {}) | |
| vim.list_extend(ensure_installed, { | |
| -- You can add other tools here that you want Mason to install | |
| }) | |
| require("mason-tool-installer").setup({ ensure_installed = ensure_installed }) | |
| require("mason-lspconfig").setup({ | |
| handlers = { | |
| function(server_name) | |
| local server = servers[server_name] or {} | |
| -- Pass blink.cmp capabilities to the LSP so autocompletion works correctly | |
| server.capabilities = vim.tbl_deep_extend( | |
| "force", | |
| {}, | |
| require("blink.cmp").get_lsp_capabilities(), | |
| server.capabilities or {} | |
| ) | |
| -- This correctly passes your config to lspconfig, injecting standard defaults (like filetypes) | |
| require("lspconfig")[server_name].setup(server) | |
| end, | |
| }, | |
| }) | |
| end, | |
| }, | |
| { -- Autoformat | |
| "stevearc/conform.nvim", | |
| event = { "BufWritePre" }, | |
| cmd = { "ConformInfo" }, | |
| keys = { | |
| { | |
| "<leader>f", | |
| function() | |
| require("conform").format({ async = true, lsp_format = "fallback" }) | |
| end, | |
| mode = "", | |
| desc = "[F]ormat buffer", | |
| }, | |
| }, | |
| ---@module 'conform' | |
| ---@type conform.setupOpts | |
| opts = { | |
| notify_on_error = false, | |
| format_on_save = function(bufnr) | |
| -- Disable "format_on_save lsp_fallback" for languages that don't | |
| -- have a well standardized coding style. You can add additional | |
| -- languages here or re-enable it for the disabled ones. | |
| local disable_filetypes = { c = true, cpp = true } | |
| if disable_filetypes[vim.bo[bufnr].filetype] then | |
| return nil | |
| else | |
| return { | |
| timeout_ms = 500, | |
| lsp_format = "fallback", | |
| } | |
| end | |
| end, | |
| formatters_by_ft = { | |
| lua = { "stylua" }, | |
| -- Conform can also run multiple formatters sequentially | |
| -- python = { "isort", "black" }, | |
| -- | |
| -- You can use 'stop_after_first' to run the first available formatter from the list | |
| -- javascript = { "prettierd", "prettier", stop_after_first = true }, | |
| }, | |
| }, | |
| }, | |
| { -- Autocompletion | |
| "saghen/blink.cmp", | |
| event = "VimEnter", | |
| version = "1.*", | |
| dependencies = { | |
| -- Snippet Engine | |
| { | |
| "L3MON4D3/LuaSnip", | |
| version = "2.*", | |
| build = (function() | |
| -- Build Step is needed for regex support in snippets. | |
| -- This step is not supported in many windows environments. | |
| -- Remove the below condition to re-enable on windows. | |
| if vim.fn.has("win32") == 1 or vim.fn.executable("make") == 0 then | |
| return | |
| end | |
| return "make install_jsregexp" | |
| end)(), | |
| dependencies = { | |
| -- `friendly-snippets` contains a variety of premade snippets. | |
| -- See the README about individual language/framework/plugin snippets: | |
| -- https://github.com/rafamadriz/friendly-snippets | |
| { | |
| "rafamadriz/friendly-snippets", | |
| config = function() | |
| require("luasnip.loaders.from_vscode").lazy_load() | |
| end, | |
| }, | |
| }, | |
| opts = {}, | |
| }, | |
| }, | |
| ---@module 'blink.cmp' | |
| ---@type blink.cmp.Config | |
| opts = { | |
| keymap = { | |
| -- 'default' (recommended) for mappings similar to built-in completions | |
| -- <c-y> to accept ([y]es) the completion. | |
| -- This will auto-import if your LSP supports it. | |
| -- This will expand snippets if the LSP sent a snippet. | |
| -- 'super-tab' for tab to accept | |
| -- 'enter' for enter to accept | |
| -- 'none' for no mappings | |
| -- | |
| -- For an understanding of why the 'default' preset is recommended, | |
| -- you will need to read `:help ins-completion` | |
| -- | |
| -- No, but seriously. Please read `:help ins-completion`, it is really good! | |
| -- | |
| -- All presets have the following mappings: | |
| -- <tab>/<s-tab>: move to right/left of your snippet expansion | |
| -- <c-space>: Open menu or open docs if already open | |
| -- <c-n>/<c-p> or <up>/<down>: Select next/previous item | |
| -- <c-e>: Hide menu | |
| -- <c-k>: Toggle signature help | |
| -- | |
| -- See :h blink-cmp-config-keymap for defining your own keymap | |
| preset = "default", | |
| -- For more advanced Luasnip keymaps (e.g. selecting choice nodes, expansion) see: | |
| -- https://github.com/L3MON4D3/LuaSnip?tab=readme-ov-file#keymaps | |
| }, | |
| appearance = { | |
| -- 'mono' (default) for 'Nerd Font Mono' or 'normal' for 'Nerd Font' | |
| -- Adjusts spacing to ensure icons are aligned | |
| nerd_font_variant = "mono", | |
| }, | |
| completion = { | |
| -- By default, you may press `<c-space>` to show the documentation. | |
| -- Optionally, set `auto_show = true` to show the documentation after a delay. | |
| documentation = { auto_show = false, auto_show_delay_ms = 500 }, | |
| }, | |
| sources = { | |
| default = { "lsp", "path", "snippets" }, | |
| }, | |
| snippets = { preset = "luasnip" }, | |
| -- Blink.cmp includes an optional, recommended rust fuzzy matcher, | |
| -- which automatically downloads a prebuilt binary when enabled. | |
| -- | |
| -- By default, we use the Lua implementation instead, but you may enable | |
| -- the rust implementation via `'prefer_rust_with_warning'` | |
| -- | |
| -- See :h blink-cmp-config-fuzzy for more information | |
| fuzzy = { implementation = "lua" }, | |
| -- Shows a signature help window while you type arguments for a function | |
| signature = { enabled = true }, | |
| }, | |
| }, | |
| -- Highlight todo, notes, etc in comments | |
| { | |
| "folke/todo-comments.nvim", | |
| event = "VimEnter", | |
| dependencies = { "nvim-lua/plenary.nvim" }, | |
| ---@module 'todo-comments' | |
| ---@type TodoOptions | |
| ---@diagnostic disable-next-line: missing-fields | |
| opts = { signs = false }, | |
| }, | |
| { -- Collection of various small independent plugins/modules | |
| "nvim-mini/mini.nvim", | |
| config = function() | |
| -- Better Around/Inside textobjects | |
| -- | |
| -- Examples: | |
| -- - va) - [V]isually select [A]round [)]paren | |
| -- - yinq - [Y]ank [I]nside [N]ext [Q]uote | |
| -- - ci' - [C]hange [I]nside [']quote | |
| require("mini.ai").setup({ n_lines = 500 }) | |
| -- Add/delete/replace surroundings (brackets, quotes, etc.) | |
| -- | |
| -- - saiw) - [S]urround [A]dd [I]nner [W]ord [)]Paren | |
| -- - sd' - [S]urround [D]elete [']quotes | |
| -- - sr)' - [S]urround [R]eplace [)] ['] | |
| require("mini.surround").setup() | |
| -- Simple and easy statusline. | |
| -- You could remove this setup call if you don't like it, | |
| -- and try some other statusline plugin | |
| local statusline = require("mini.statusline") | |
| -- set use_icons to true if you have a Nerd Font | |
| statusline.setup({ use_icons = vim.g.have_nerd_font }) | |
| -- You can configure sections in the statusline by overriding their | |
| -- default behavior. For example, here we set the section for | |
| -- cursor location to LINE:COLUMN | |
| ---@diagnostic disable-next-line: duplicate-set-field | |
| statusline.section_location = function() | |
| return "%2l:%-2v" | |
| end | |
| -- ... and there is more! | |
| -- Check out: https://github.com/nvim-mini/mini.nvim | |
| end, | |
| }, | |
| { -- Highlight, edit, and navigate code | |
| "nvim-treesitter/nvim-treesitter", | |
| lazy = false, | |
| build = ":TSUpdate", | |
| branch = "main", | |
| -- [[ Configure Treesitter ]] See `:help nvim-treesitter-intro` | |
| config = function() | |
| -- ensure basic parser are installed | |
| -- local parsers = { 'bash', 'c', 'diff', 'html', 'lua', 'luadoc', 'markdown', 'markdown_inline', 'query', 'vim', 'vimdoc' } | |
| -- require('nvim-treesitter').install(parsers) | |
| ---@param buf integer | |
| ---@param language string | |
| local function treesitter_try_attach(buf, language) | |
| -- check if parser exists and load it | |
| if not vim.treesitter.language.add(language) then | |
| return | |
| end | |
| -- enables syntax highlighting and other treesitter features | |
| vim.treesitter.start(buf, language) | |
| -- enables treesitter based folds | |
| -- for more info on folds see `:help folds` | |
| -- vim.wo.foldexpr = 'v:lua.vim.treesitter.foldexpr()' | |
| -- vim.wo.foldmethod = 'expr' | |
| -- enables treesitter based indentation | |
| vim.bo.indentexpr = "v:lua.require'nvim-treesitter'.indentexpr()" | |
| end | |
| local available_parsers = require("nvim-treesitter").get_available() | |
| vim.api.nvim_create_autocmd("FileType", { | |
| callback = function(args) | |
| local buf, filetype = args.buf, args.match | |
| local language = vim.treesitter.language.get_lang(filetype) | |
| if not language then | |
| return | |
| end | |
| local installed_parsers = require("nvim-treesitter").get_installed("parsers") | |
| if vim.tbl_contains(installed_parsers, language) then | |
| -- enable the parser if it is installed | |
| treesitter_try_attach(buf, language) | |
| elseif vim.tbl_contains(available_parsers, language) then | |
| -- if a parser is available in `nvim-treesitter` auto install it, and enable it after the installation is done | |
| require("nvim-treesitter").install(language):await(function() | |
| treesitter_try_attach(buf, language) | |
| end) | |
| else | |
| -- try to enable treesitter features in case the parser exists but is not available from `nvim-treesitter` | |
| treesitter_try_attach(buf, language) | |
| end | |
| end, | |
| }) | |
| end, | |
| }, | |
| -- The following comments only work if you have downloaded the kickstart repo, not just copy pasted the | |
| -- init.lua. If you want these files, they are in the repository, so you can just download them and | |
| -- place them in the correct locations. | |
| -- NOTE: Next step on your Neovim journey: Add/Configure additional plugins for Kickstart | |
| -- | |
| -- Here are some example plugins that I've included in the Kickstart repository. | |
| -- Uncomment any of the lines below to enable them (you will need to restart nvim). | |
| -- | |
| -- require 'kickstart.plugins.debug', | |
| -- require 'kickstart.plugins.indent_line', | |
| -- require 'kickstart.plugins.lint', | |
| -- require 'kickstart.plugins.autopairs', | |
| -- require 'kickstart.plugins.neo-tree', | |
| -- require 'kickstart.plugins.gitsigns', -- adds gitsigns recommended keymaps | |
| -- NOTE: The import below can automatically add your own plugins, configuration, etc from `lua/custom/plugins/*.lua` | |
| -- This is the easiest way to modularize your config. | |
| -- | |
| -- Uncomment the following line and add your plugins to `lua/custom/plugins/*.lua` to get going. | |
| -- { import = 'custom.plugins' }, | |
| -- | |
| -- For additional information with loading, sourcing and examples see `:help lazy.nvim-π-plugin-spec` | |
| -- Or use telescope! | |
| -- In normal mode type `<space>sh` then write `lazy.nvim-plugin` | |
| -- you can continue same window with `<space>sr` which resumes last telescope search | |
| }, { ---@diagnostic disable-line: missing-fields | |
| ui = { | |
| -- If you are using a Nerd Font: set icons to an empty table which will use the | |
| -- default lazy.nvim defined Nerd Font icons, otherwise define a unicode icons table | |
| icons = vim.g.have_nerd_font and {} or { | |
| cmd = "β", | |
| config = "π ", | |
| event = "π ", | |
| ft = "π", | |
| init = "β", | |
| keys = "π", | |
| plugin = "π", | |
| runtime = "π»", | |
| require = "π", | |
| source = "π", | |
| start = "π", | |
| task = "π", | |
| lazy = "π€ ", | |
| }, | |
| }, | |
| }) | |
| -- The line beneath this is called `modeline`. See `:help modeline` |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment