HOME = os.getenv("HOME") -- Configure the clipboard to access the "+ and "* registers -- N.B. JavaScript copy buttons on the web do not necessarily work as expected vim.opt.clipboard = "unnamedplus,unnamed" -- Spaces indentation vim.opt.expandtab = true -- converts tabs to spaces vim.opt.tabstop = 4 -- tab equals 4 spaces vim.opt.shiftwidth = 4 -- indent size in characters -- Show whitespace (:list) vim.opt.listchars = "eol:¬,tab:>-,trail:~,extends:>,precedes:<,space:·" -- Show line numbers vim.opt.number = true -- Vertically splitting a window (:vsplit) places new window to the right vim.opt.splitright = true -- Highlight cursor line vim.opt.cursorline = true -- Enable folding vim.opt.foldmethod = "syntax" vim.opt.foldlevel = 5 -- Enable 24-bit RGB color in the TUI vim.opt.termguicolors = true -- Minimal number of lines kept above and below the cursor vim.opt.scrolloff = 5 -- Turn off highlight search vim.opt.hlsearch = false -- Temporary file locations vim.opt.backupdir = ".backup/," .. HOME .. "/.backup/,/tmp//" vim.opt.directory = ".swp/," .. HOME .. "/.swp/,/tmp//" -- netrw -- This is the workspace file explorer vim.g.netrw_winsize = 25 -- width of the file explorer vim.g.netrw_liststyle = 3 -- tree style listing -- Sane vim split naviagation (via Gaslight blog) vim.keymap.set("n", "", "j", { noremap = true, desc = 'Go to window below' }) vim.keymap.set("n", "", "k", { noremap = true, desc = 'Go to window above' }) vim.keymap.set("n", "", "h", { noremap = true, desc = 'Go to window to the left' }) -- N.B. This conflicts with the NetRW directory refresh command. Use the alternative `:e .`. -- TODO: This seems not to work for netrw. vim.keymap.set("n", "", "l", { noremap = true, desc = 'Go to window to the right' }) vim.keymap.set("t", "", "j", { noremap = true, desc = 'Go to window below' }) vim.keymap.set("t", "", "k", { noremap = true, desc = 'Go to window above' }) vim.keymap.set("t", "", "h", { noremap = true, desc = 'Go to window to the left' }) vim.keymap.set("t", "", "l", { noremap = true, desc = 'Go to window to the right' }) vim.keymap.set( { "n", "t" }, "z", function() -- This restores the UI to the saved layout 'idelayout' (if it exists) if vim.fn.exists("idelayout") ~= 0 then vim.cmd("exec idelayout") end end, { desc = "Revert window layout" } ) local terminalgroup = vim.api.nvim_create_augroup("TerminalGroup", { clear = true }) vim.api.nvim_create_autocmd( { "TermOpen", "TermEnter" }, { group = terminalgroup, pattern = "*", command = "set nonumber" } ) -- lazy.nvim local lazypath = vim.fn.stdpath("data") .. "/lazy/lazy.nvim" if not vim.loop.fs_stat(lazypath) then vim.fn.system({ "git", "clone", "--filter=blob:none", "https://github.com/folke/lazy.nvim.git", "--branch=stable", -- latest stable release lazypath, }) end vim.opt.rtp:prepend(lazypath) local js_based_languages = { "javascript", "typescript" } require("lazy").setup({ { url = "https://git.theadamcooper.com/adam/dracula-vim.git", branch = "adamc-main", name = "dracula", lazy = false, priority = 1000, config = function() vim.cmd.colorscheme("dracula") end, }, "nvim-lualine/lualine.nvim", "nvim-tree/nvim-web-devicons", { "nvim-treesitter/nvim-treesitter", build = ":TSUpdate" }, "neovim/nvim-lspconfig", { "kylechui/nvim-surround", version = "*", -- Use for stability; omit to use `main` branch for the latest features event = "VeryLazy", config = function() require("nvim-surround").setup({ -- Configuration here, or leave empty to use defaults }) end }, { "folke/which-key.nvim", event = "VeryLazy", init = function() vim.o.timeout = true -- N.B. Setting `timeoutlen` to 0 seems to break the plugin vim.o.timeoutlen = 300 -- 0? 500? 300? end, opts = { window = { border = "single", }, }, }, { "ray-x/lsp_signature.nvim", event = "VeryLazy", opts = {}, config = function(_, opts) require 'lsp_signature'.setup(opts) end }, { "folke/neodev.nvim", opts = {} }, { "rcarriga/nvim-dap-ui", dependencies = { "mfussenegger/nvim-dap", "nvim-neotest/nvim-nio" }, }, { "mfussenegger/nvim-dap", event = "VeryLazy", config = function() local dap = require("dap") -- local Config = require("lazyvim.config") vim.api.nvim_set_hl(0, "DapStoppedLine", { default = true, link = "Visual" }) --[[ for name, sign in pairs(Config.icons.dap) do sign = type(sign) == "table" and sign or { sign } vim.fn.sign_define( "Dap" .. name, { text = sign[1], texthl = sign[2] or "DiagnosticInfo", linehl = sign[3], numhl = sign[3] } ) end --]] for _, language in ipairs({ "javascript", "typescript" }) do dap.configurations[language] = { -- Debug single nodejs files { type = "pwa-node", request = "launch", name = "Launch file", program = "${file}", cwd = vim.fn.getcwd(), sourceMaps = true, }, -- Debug nodejs processes (make sure to add --inspect when you run the process) { type = "pwa-node", request = "attach", name = "Attach", processId = require("dap.utils").pick_process, cwd = vim.fn.getcwd(), sourceMaps = true, }, -- Debug web applications (client side) { type = "pwa-chrome", request = "launch", name = "Launch & Debug Chrome", url = function() local co = coroutine.running() return coroutine.create(function() vim.ui.input({ prompt = "Enter URL: ", default = "http://localhost:3000", }, function(url) if url == nil or url == "" then return else coroutine.resume(co, url) end end) end) end, webRoot = vim.fn.getcwd(), protocol = "inspector", sourceMaps = true, userDataDir = false, }, -- Divider for the launch.json derived configs { name = "----- ↓ launch.json configs ↓ -----", type = "", request = "launch", }, } end end, keys = { { "dO", function() require("dap").step_out() end, desc = "DAP: Step Out", }, { "do", function() require("dap").step_over() end, desc = "DAP: Step Over", }, { "db", function() require("dap").toggle_breakpoint() end, desc = "DAP: Toggle breakpoint", }, { "dc", function() require("dap").continue() end, desc = "DAP: Continue", }, { "di", function() require("dap").step_into() end, desc = "DAP: Step Into", }, { "da", function() if vim.fn.filereadable(".vscode/launch.json") then local dap_vscode = require("dap.ext.vscode") dap_vscode.load_launchjs(nil, { ["pwa-node"] = js_based_languages, ["chrome"] = js_based_languages, ["pwa-chrome"] = js_based_languages, }) end require("dap").continue() end, desc = "DAP: Run with Args", }, }, dependencies = { -- Install the vscode-js-debug adapter { "microsoft/vscode-js-debug", -- After install, build it and rename the dist directory to out build = "npm install --legacy-peer-deps --no-save && npx gulp vsDebugServerBundle && rm -rf out && mv dist out", version = "1.*", }, { "mxsdev/nvim-dap-vscode-js", config = function() ---@diagnostic disable-next-line: missing-fields require("dap-vscode-js").setup({ -- Path of node executable. Defaults to $NODE_PATH, and then "node" -- node_path = "node", -- Path to vscode-js-debug installation. debugger_path = vim.fn.resolve(vim.fn.stdpath("data") .. "/lazy/vscode-js-debug"), -- Command to use to launch the debug server. Takes precedence over "node_path" and "debugger_path" -- debugger_cmd = { "js-debug-adapter" }, -- which adapters to register in nvim-dap adapters = { "chrome", "pwa-node", "pwa-chrome", "pwa-msedge", "pwa-extensionHost", "node-terminal", }, -- Path for file logging -- log_file_path = "(stdpath cache)/dap_vscode_js.log", -- Logging level for output to file. Set to false to disable logging. -- log_file_level = false, -- Logging level for output to console. Set to false to disable console output. -- log_console_level = vim.log.levels.ERROR, }) end, }, { "Joakker/lua-json5", build = "./install.sh", }, }, }, { "leoluz/nvim-dap-go", event = "VeryLazy" }, { "williamboman/mason.nvim", event = "VeryLazy" }, { "williamboman/mason-lspconfig.nvim", event = "VeryLazy" }, { "nvimdev/lspsaga.nvim", event = "VeryLazy" }, { "nvim-telescope/telescope.nvim", event = "VeryLazy", branch = "0.1.x", dependencies = { "nvim-lua/plenary.nvim", { "nvim-telescope/telescope-fzf-native.nvim", build = "cmake -S. -Bbuild -DCMAKE_BUILD_TYPE=Release && cmake --build build --config Release && cmake --install build --prefix build", }, }, }, { "hrsh7th/nvim-cmp", event = "InsertEnter", dependencies = { "hrsh7th/cmp-nvim-lsp", "hrsh7th/cmp-buffer", "hrsh7th/cmp-path", "hrsh7th/cmp-cmdline", "hrsh7th/cmp-vsnip", "hrsh7th/vim-vsnip", "hrsh7th/cmp-nvim-lsp-signature-help", }, }, { "tpope/vim-fugitive", event = "VeryLazy" }, { "lewis6991/gitsigns.nvim", event = "VeryLazy" }, { "pwntester/octo.nvim", requires = { "nvim-lua/plenary.nvim", "nvim-telescope/telescope.nvim", "nvim-tree/nvim-web-devicons", }, config = function() require("octo").setup() end, event = "VeryLazy", }, { "famiu/bufdelete.nvim", event = "VeryLazy" }, }) --[[ mason Mason manages external editor plugins such as LSP servers, DAP servers, linters, and formatters. There are further recommended plugins for better integration. --]] require('mason').setup() require('mason-lspconfig').setup() --[[ lualine Lualine provides the status bar as well as the tabline. --]] require('lualine').setup { options = { theme = 'dracula' }, tabline = { lualine_a = { { 'buffers', mode = 4, }, }, }, } --[[ Telescope Telescope provides lists, pickers, etc. This section includes just the functions bound to keymaps. --]] local builtin = require('telescope.builtin') vim.keymap.set('n', 'ff', builtin.find_files, { desc = 'Telescope: find files' }) vim.keymap.set('n', 'fg', builtin.live_grep, { desc = 'Telescope: live grep' }) vim.keymap.set('n', 'fb', builtin.buffers, { desc = 'Telescope: buffers' }) vim.keymap.set('n', 'fh', builtin.help_tags, { desc = 'Telescope: help tags' }) require "lsp_signature".setup() --[[ nvim-cmp nvim-cmp is a text completion engine. ]] local cmp = require 'cmp' cmp.setup({ sources = { { name = 'nvim_lsp_signature_help' } } }) cmp.setup({ snippet = { -- REQUIRED - you must specify a snippet engine expand = function(args) vim.fn["vsnip#anonymous"](args.body) -- For `vsnip` users. end, }, window = { completion = cmp.config.window.bordered(), documentation = cmp.config.window.bordered(), }, mapping = cmp.mapping.preset.insert({ [''] = cmp.mapping.scroll_docs(-4), [''] = cmp.mapping.scroll_docs(4), [''] = cmp.mapping.complete(), [''] = cmp.mapping.abort(), [''] = cmp.mapping.confirm({ select = true }), -- Accept currently selected item. Set `select` to `false` to only confirm explicitly selected items. }), sources = cmp.config.sources({ { name = 'nvim_lsp' }, { name = 'vsnip' }, -- For vsnip users. }, { { name = 'buffer' }, }) }) -- Set configuration for specific filetype. cmp.setup.filetype('gitcommit', { sources = cmp.config.sources({ { name = 'git' }, -- You can specify the `git` source if [you were installed it](https://github.com/petertriho/cmp-git). }, { { name = 'buffer' }, }) }) -- Use buffer source for `/` and `?` (if you enabled `native_menu`, this won't work anymore). cmp.setup.cmdline({ '/', '?' }, { mapping = cmp.mapping.preset.cmdline(), sources = { { name = 'buffer' } } }) -- Use cmdline & path source for ':' (if you enabled `native_menu`, this won't work anymore). cmp.setup.cmdline(':', { mapping = cmp.mapping.preset.cmdline(), sources = cmp.config.sources({ { name = 'path' } }, { { name = 'cmdline' } }) }) -- Neodev require("neodev").setup({ library = { plugins = { "nvim-dap-ui" }, types = true }, }) --[[ nvim-lspconfig --]] -- Setup language servers. local lua_ls_setup = { settings = { Lua = { runtime = { -- Tell the language server which version of Lua you're using (most likely LuaJIT in the case of Neovim) version = "LuaJIT", }, diagnostics = { -- Get the language server to recognize the `vim` global globals = { "vim" }, }, workspace = { -- Make the server aware of Neovim runtime files library = vim.api.nvim_get_runtime_file("", true), checkThirdParty = false, }, -- Do not send telemetry data containing a randomized but unique identifier telemetry = { enable = false, }, }, }, } local capabilities = require('cmp_nvim_lsp').default_capabilities() local lspconfig = require('lspconfig') lspconfig.gopls.setup { capabilities = capabilities } lspconfig.lua_ls.setup(lua_ls_setup) lspconfig.pyright.setup { capabilities = capabilities } lspconfig.tsserver.setup { capabilities = capabilities } lspconfig.rust_analyzer.setup { -- Server-specific settings. See `:help lspconfig-setup` capabilities = capabilities, settings = { ['rust-analyzer'] = {}, }, } -- Enable (broadcasting) snippet capability for completion local capabilities_html = vim.lsp.protocol.make_client_capabilities() capabilities_html.textDocument.completion.completionItem.snippetSupport = true require 'lspconfig'.html.setup { capabilities = capabilities_html, } -- Global mappings : Diagnostics -- See `:help vim.diagnostic.*` for documentation on any of the below functions vim.keymap.set('n', 'e', vim.diagnostic.open_float, { desc = 'Diagnostic: open float' }) vim.keymap.set('n', '[d', vim.diagnostic.goto_prev, { desc = 'Diagnostic: go to previous' }) vim.keymap.set('n', ']d', vim.diagnostic.goto_next, { desc = 'Diagnostic: go to next' }) vim.keymap.set('n', 'q', vim.diagnostic.setloclist, { desc = 'Diagnostic: set loclist' }) -- Add a border to LSP windows local _border = "single" vim.lsp.handlers["textDocument/hover"] = vim.lsp.with( vim.lsp.handlers.hover, { border = _border } ) -- TODO: Is this necessary? Or is signature help being handled with LSPSaga? --[[ vim.lsp.handlers["textDocument/signatureHelp"] = vim.lsp.with( vim.lsp.handlers.signature_help, { border = _border } ) --]] vim.diagnostic.config { float = { border = _border, max_width = 120 } } -- Use LspAttach autocommand to only map the following keys -- after the language server attaches to the current buffer vim.api.nvim_create_autocmd('LspAttach', { group = vim.api.nvim_create_augroup('UserLspConfig', {}), callback = function(ev) -- Enable completion triggered by vim.bo[ev.buf].omnifunc = 'v:lua.vim.lsp.omnifunc' -- Buffer local mappings. -- See `:help vim.lsp.*` for documentation on any of the below functions local opts = function(desc) return { buffer = ev.buf, desc = desc } end vim.keymap.set('n', 'gD', vim.lsp.buf.declaration, opts('LSP: go to declaration')) vim.keymap.set('n', 'gd', vim.lsp.buf.definition, opts('LSP: go to definition')) vim.keymap.set('n', 'K', vim.lsp.buf.hover, opts('LSP: hover')) vim.keymap.set('n', 'gi', vim.lsp.buf.implementation, opts('LSP: go to implementation')) -- This setting steps on my split navigation setting, so I changed it -- to the probably harmless F9. -- vim.keymap.set('n', '', vim.lsp.buf.signature_help, opts('')) vim.keymap.set('n', '', vim.lsp.buf.signature_help, opts('LSP: signature help')) vim.keymap.set('n', 'wa', vim.lsp.buf.add_workspace_folder, opts('LSP: add workspace folder')) vim.keymap.set('n', 'wr', vim.lsp.buf.remove_workspace_folder, opts('LSP: remove workspace folder')) vim.keymap.set('n', 'wl', function() print(vim.inspect(vim.lsp.buf.list_workspace_folders())) end, opts('LSP: list workspace folder')) vim.keymap.set('n', 'D', vim.lsp.buf.type_definition, opts('LSP: go to type definition')) vim.keymap.set('n', 'rn', vim.lsp.buf.rename, opts('LSP: rename token')) vim.keymap.set({ 'n', 'v' }, 'ca', vim.lsp.buf.code_action, opts('LSP: code action')) vim.keymap.set('n', 'gr', vim.lsp.buf.references, opts('LSP: go to references')) vim.keymap.set('n', 'f', function() vim.lsp.buf.format { async = true } end, opts('LSP: format')) end, }) require('gitsigns').setup({ signs = { add = { text = '│' }, change = { text = '│' }, delete = { text = '_' }, topdelete = { text = '‾' }, changedelete = { text = '~' }, untracked = { text = '┆' }, }, signcolumn = true, -- Toggle with `:Gitsigns toggle_signs` numhl = false, -- Toggle with `:Gitsigns toggle_numhl` linehl = false, -- Toggle with `:Gitsigns toggle_linehl` word_diff = false, -- Toggle with `:Gitsigns toggle_word_diff` watch_gitdir = { follow_files = true }, attach_to_untracked = true, current_line_blame = false, -- Toggle with `:Gitsigns toggle_current_line_blame` current_line_blame_opts = { virt_text = true, virt_text_pos = 'eol', -- 'eol' | 'overlay' | 'right_align' delay = 1000, ignore_whitespace = false, }, current_line_blame_formatter = ', - ', sign_priority = 6, update_debounce = 100, status_formatter = nil, -- Use default max_file_length = 40000, -- Disable if file is longer than this (in lines) preview_config = { -- Options passed to nvim_open_win border = 'single', style = 'minimal', relative = 'cursor', row = 0, col = 1 }, yadm = { enable = false }, on_attach = function(bufnr) local gs = package.loaded.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 return ']c' end vim.schedule(function() gs.next_hunk() end) return '' end, { expr = true, desc = 'GitSigns: go to next hunk' }) map('n', '[c', function() if vim.wo.diff then return '[c' end vim.schedule(function() gs.prev_hunk() end) return '' end, { expr = true, desc = 'GitSigns: go to previous hunk' }) -- Actions map('n', 'hs', gs.stage_hunk, { desc = 'GitSigns: stage hunk' }) map('n', 'hr', gs.reset_hunk, { desc = 'GitSigns: reset hunk' }) map('v', 'hs', function() gs.stage_hunk { vim.fn.line('.'), vim.fn.line('v') } end, { desc = 'GitSigns: stage hunk' }) map('v', 'hr', function() gs.reset_hunk { vim.fn.line('.'), vim.fn.line('v') } end, { desc = 'GitSigns: reset hunk' }) map('n', 'hS', gs.stage_buffer, { desc = 'GitSigns: stage buffer' }) map('n', 'hu', gs.undo_stage_hunk, { desc = 'GitSigns: undo stage hunk' }) map('n', 'hR', gs.reset_buffer, { desc = 'GitSigns: reset_buffer' }) map('n', 'hp', gs.preview_hunk, { desc = 'GitSigns: preview hunk' }) map('n', 'hb', function() gs.blame_line { full = true } end, { desc = 'GitSigns: blame line' }) map('n', 'tb', gs.toggle_current_line_blame, { desc = 'GitSigns: toggle current line blame' }) map('n', 'hd', gs.diffthis, { desc = 'GitSigns: diff this' }) map('n', 'hD', function() gs.diffthis('~') end, { desc = 'GitSigns: diff this' }) map('n', 'td', gs.toggle_deleted, { desc = 'GitSigns: toggle deleted' }) end }) --[[ -- LSPSaga : provides a diverse basket of utilities --]] require('lspsaga').setup({ beacon = { enable = true, frequency = 7, } }) vim.keymap.set('n', 'si', 'Lspsaga incoming_calls') vim.keymap.set('n', 'so', 'Lspsaga outgoing_calls') vim.keymap.set('n', 'ca', 'Lspsaga code_action') vim.keymap.set('n', 'sd', 'Lspsaga peek_definition') vim.keymap.set('n', 'sp', 'Lspsaga peek_type_definition') vim.keymap.set('n', 'sx', 'Lspsaga goto_definition') vim.keymap.set('n', 'sg', 'Lspsaga goto_type_definition') vim.keymap.set('n', '[e', 'Lspsaga diagnostic_jump_prev') vim.keymap.set('n', ']e', 'Lspsaga diagnostic_jump_next') vim.keymap.set('n', 'sK', 'Lspsaga hover_doc') vim.keymap.set('n', 'sm', 'Lspsaga finder imp') vim.keymap.set('n', 'sf', 'Lspsaga finder') vim.keymap.set('n', 'sl', 'Lspsaga outline') vim.keymap.set('n', 'rn', 'Lspsaga rename') vim.keymap.set('n', 'st', 'Lspsaga term_toggle') require('dap-go').setup() local dap, dapui = require("dap"), require("dapui") dapui.setup() dap.listeners.after.event_initialized["dapui_config"] = function() dapui.open() end dap.listeners.before.event_terminated["dapui_config"] = function() dapui.close() end dap.listeners.before.event_exited["dapui_config"] = function() dapui.close() end -- Handlebars: One way to get syntax highlighting in Handlebars files --[[ local syntax_highlighting_tweaks = vim.api.nvim_create_augroup("SyntaxHighlightingTweaks", { clear = true }) vim.api.nvim_create_autocmd( { "BufRead", "BufNewFile" }, { group = syntax_highlighting_tweaks, pattern = "*.hbs", command = "set filetype=html" } ) --]] -- Treesitter require('nvim-treesitter.configs').setup({ modules = {}, highlight = { enable = true, loaded = true, module_path = "nvim-treesitter.highlight", additional_vim_regex_highlighting = true, }, indent = { enable = true, module_path = "nvim-treesitter.indent", }, ensure_installed = { "bash", "comment", "html", "javascript", "json", "lua", "typescript", "yaml" }, sync_install = false, auto_install = false, ignore_install = {}, }) vim.treesitter.language.register("html", "handlebars") -- Tweak GitSigns blame color -- This differentiates the cursorline from the git blame text vim.cmd("highlight GitSignsCurrentLineBlame gui=bold guifg=#339944") vim.cmd("highlight NonText gui=bold guifg=#999999") --[[ Resolve conflict between fugitive and LSPSaga, wherein the latter's breadcrumbs cause a mismatch between the buffer and fugitive's :Git blame window and :Gvdiffsplit window(s). To kill the winbar (the top line where the breadcrumbs and this blame title live), enter `:set winbar&`. --]] local blamegroup = vim.api.nvim_create_augroup("fugitiveSagaBlameConflict", { clear = true }) vim.api.nvim_create_autocmd( 'FileType', { group = blamegroup, pattern = 'fugitiveblame', callback = function() vim.api.nvim_set_option_value('winbar', 'fugitive', { scope = 'local' }) end, } ) local diffgroup = vim.api.nvim_create_augroup("fugitiveSagaDiffConflict", { clear = true }) vim.api.nvim_create_autocmd( 'BufReadCmd', { group = diffgroup, pattern = "fugitive://*", callback = function() vim.api.nvim_set_option_value('winbar', 'fugitive', { scope = 'local' }) end, } ) -- Switch syntax highlighting on vim.cmd("syntax enable")