-- luacheck: globals packer_plugins local M = {} local utils = require("utils") function M.config_lsp_ui() local border = { { "┌", "FloatBorder" }, { "─", "FloatBorder" }, { "┐", "FloatBorder" }, { "│", "FloatBorder" }, { "┘", "FloatBorder" }, { "─", "FloatBorder" }, { "└", "FloatBorder" }, { "│", "FloatBorder" }, } local orig_util_open_floating_preview = vim.lsp.util.open_floating_preview function vim.lsp.util.open_floating_preview(contents, syntax, opts, ...) opts = opts or {} opts.border = opts.border or border return orig_util_open_floating_preview(contents, syntax, opts, ...) end -- Diagnostics signs for type, icon in pairs(utils.diagnostic_signs) do local hl = "DiagnosticSign" .. type vim.fn.sign_define(hl, { text = icon, texthl = hl, numhl = hl }) end utils.try_require("trouble", function(trouble) trouble.setup({ fold_open = "▼", fold_closed = "▶", icons = false, use_diagnostic_signs = true, }) end) end local function get_server_capabilities(client) -- HACK: Support for <0.8 if client.server_capabilities ~= nil then return client.server_capabilities end local capabilities = client.resolved_capabilities if capabilities.documentSymbolProvider == nil then capabilities.documentSymbolProvider = capabilities.goto_definition end if capabilities.documentFormattingProvider == nil then capabilities.documentFormattingProvider = capabilities.document_formatting end if capabilities.documentRangeFormattingProvider == nil then capabilities.documentRangeFormattingProvider = capabilities.document_range_formatting end if capabilities.documentHighlightProvider == nil then capabilities.documentHighlightProvider = capabilities.document_highlight end if capabilities.workspaceSymbolProvider == nil then -- Not sure what the legacy version of this is capabilities.workspaceSymbolProvider = capabilities.goto_definition end return capabilities end local function get_default_attach(override_capabilities) return function(client, bufnr) -- Allow overriding capabilities to avoid duplicate lsps with capabilities -- Using custom method to extract for <0.8 support local server_capabilities = get_server_capabilities(client) if override_capabilities ~= nil then server_capabilities = vim.tbl_extend("force", server_capabilities, override_capabilities or {}) end -- Set built in features to use lsp functions (automatic in nvim-0.8) -- HACK: Support for <0.8 if vim.fn.has("nvim-0.8") ~= 1 then vim.api.nvim_buf_set_option(bufnr, "omnifunc", "v:lua.vim.lsp.omnifunc") if server_capabilities.documentSymbolProvider then vim.api.nvim_buf_set_option(bufnr, "tagfunc", "v:lua.vim.lsp.tagfunc") end if server_capabilities.documentFormattingProvider then vim.api.nvim_buf_set_option(bufnr, "formatexpr", "v:lua.vim.lsp.formatexpr()") end end -- Mappings local lsp_keymap = utils.curry_keymap("n", "l", { buffer = bufnr, group_desc = "LSP" }) lsp_keymap("h", vim.lsp.buf.hover, { desc = "Display hover" }) lsp_keymap("rn", vim.lsp.buf.rename, { desc = "Refactor rename" }) lsp_keymap("e", vim.diagnostic.open_float, { desc = "Open float dialog" }) lsp_keymap("D", vim.lsp.buf.declaration, { desc = "Go to declaration" }) lsp_keymap("d", vim.lsp.buf.definition, { desc = "Go to definition" }) lsp_keymap("t", vim.lsp.buf.type_definition, { desc = "Go to type definition" }) lsp_keymap("i", vim.lsp.buf.implementation, { desc = "Show implementations" }) lsp_keymap("s", vim.lsp.buf.signature_help, { desc = "Show signature help" }) lsp_keymap("wa", vim.lsp.buf.add_workspace_folder, { desc = "Workspace: Add folder" }) lsp_keymap("wr", vim.lsp.buf.remove_workspace_folder, { desc = "Workspace: Remove folder" }) lsp_keymap("wl", function() print(vim.inspect(vim.lsp.buf.list_workspace_folders())) end, { desc = "Workspace: List folders" }) lsp_keymap("r", vim.lsp.buf.references, { desc = "References" }) lsp_keymap("p", vim.diagnostic.goto_prev, { desc = "Previous diagnostic" }) lsp_keymap("n", vim.diagnostic.goto_next, { desc = "Next diagnostic" }) if server_capabilities.codeActionProvider then lsp_keymap("A", vim.lsp.buf.code_action, { desc = "Select code actions" }) lsp_keymap("A", vim.lsp.buf.code_action, { mode = "v", desc = "Select code actions" }) end -- Set insert keymap for signature help utils.keymap_set("i", "", vim.lsp.buf.signature_help, { buffer = bufnr, desc = "Show signature help" }) -- Some top level aliases or remaps utils.keymap_set("n", "K", vim.lsp.buf.hover, { buffer = bufnr, desc = "Display hover" }) utils.keymap_set("n", "gD", vim.lsp.buf.declaration, { buffer = bufnr, desc = "Go to declaration" }) utils.keymap_set("n", "gd", vim.lsp.buf.definition, { buffer = bufnr, desc = "Go to definition" }) utils.keymap_set("n", "rn", vim.lsp.buf.rename, { buffer = bufnr, desc = "Refactor rename" }) utils.keymap_set("n", "[d", vim.diagnostic.goto_prev, { buffer = bufnr, desc = "Previous diagnostic" }) utils.keymap_set("n", "]d", vim.diagnostic.goto_next, { buffer = bufnr, desc = "Next diagnostic" }) -- Open diagnostic on hold vim.api.nvim_create_autocmd({ "CursorHold" }, { pattern = "", callback = function() vim.diagnostic.open_float(nil, { focus = false, scope = "cursor" }) end, group = vim.api.nvim_create_augroup("diagnostic_float", { clear = true }), desc = "Open float dialog on hold", }) -- Use IncRename if available if utils.try_require("inc_rename") ~= nil then vim.keymap.set("n", "rn", function() return ":IncRename " .. vim.fn.expand("") end, { expr = true, buffer = true, desc = "Rename current symbol" }) end -- Set some keybinds conditional on server capabilities if vim.fn.has("nvim-0.8") == 1 then lsp_keymap("f", function() vim.lsp.buf.format({ async = true }) end, { desc = "Format code" }) lsp_keymap("f", function() vim.lsp.buf.format({ async = true }) end, { mode = "v", desc = "Format selected code" }) if server_capabilities.documentFormattingProvider then vim.api.nvim_create_autocmd({ "BufWritePre" }, { pattern = { "*.rs", "*.go", "*.sh", "*.lua" }, callback = function() vim.lsp.buf.format({ async = false, timeout_ms = 1000 }) end, group = vim.api.nvim_create_augroup("lsp_format", { clear = true }), desc = "Auto format code on save", }) end else -- HACK: Support for <0.8 with older formatting lsp_keymap("f", vim.lsp.buf.formatting, { desc = "Format code" }) lsp_keymap("f", vim.lsp.buf.range_formatting, { mode = "v", desc = "Format selected code" }) if server_capabilities.documentFormattingProvider then vim.api.nvim_create_autocmd({ "BufWritePre" }, { pattern = { "*.rs", "*.go", "*.sh", "*.lua" }, callback = function() vim.lsp.buf.formatting_sync(nil, 1000) end, group = vim.api.nvim_create_augroup("lsp_format", { clear = true }), desc = "Auto format code on save", }) end end -- Set autocommands conditional on server_capabilities if server_capabilities.documentHighlightProvider then vim.api.nvim_set_hl(0, "LspReferenceRead", { link = "MatchParen" }) vim.api.nvim_set_hl(0, "LspReferenceText", { link = "MatchParen" }) vim.api.nvim_set_hl(0, "LspReferenceWrite", { link = "MatchParen" }) local hl_group = vim.api.nvim_create_augroup("lsp_document_highlight", { clear = true }) vim.api.nvim_create_autocmd( { "CursorHold" }, { pattern = "", callback = vim.lsp.buf.document_highlight, group = hl_group } ) vim.api.nvim_create_autocmd( { "CursorMoved" }, { pattern = "", callback = vim.lsp.buf.clear_references, group = hl_group } ) end -- Some override some fuzzy finder bindings to use lsp sources utils.try_require("telescope.builtin", function(telescope_builtin) -- Replace some Telescope bindings with LSP versions local telescope_keymap = utils.curry_keymap("n", "f", { buffer = bufnr, group_desc = "Finder" }) if server_capabilities.documentSymbolProvider then telescope_keymap("t", telescope_builtin.lsp_document_symbols, { desc = "Find buffer tags" }) end if server_capabilities.workspaceSymbolProvider then telescope_keymap("T", telescope_builtin.lsp_dynamic_workspace_symbols, { desc = "Find tags" }) end -- Replace some LSP bindings with Telescope ones if server_capabilities.definitionProvider then lsp_keymap("d", telescope_builtin.lsp_definitions, { desc = "Find definition" }) end if server_capabilities.typeDefinitionProvider then lsp_keymap("t", telescope_builtin.lsp_type_definitions, { desc = "Find type definition" }) end lsp_keymap("i", telescope_builtin.lsp_implementations, { desc = "Find implementations" }) lsp_keymap("r", telescope_builtin.lsp_references, { desc = "Find references" }) end) -- Attach navic for statusline location if server_capabilities.documentSymbolProvider then utils.try_require("nvim-navic", function(navic) navic.attach(client, bufnr) end) end end end local function merged_capabilities() -- Maybe update capabilities local capabilities = nil utils.try_require("cmp-nvim-lsp", function(cmp_nvim_lsp) capabilities = cmp_nvim_lsp.default_capabilities() end) return capabilities end function M.config_lsp() utils.try_require("lspconfig", function(lsp_config) local capabilities = merged_capabilities() local default_attach = get_default_attach() local default_setup = { capabilities = capabilities, on_attach = default_attach } -- Configure each server lsp_config.gopls.setup(default_setup) lsp_config.pyright.setup(default_setup) lsp_config.bashls.setup({ capabilities = capabilities, on_attach = default_attach, settings = { bashIde = { -- Disable shellcheck linting because we have it enabled in null-ls -- Some machines I use aren't configured with npm so bashls cannot -- be relied on as the sole source of shellcheck linting. shellcheckPath = "", }, }, }) -- Set up rust analyzer (preferred) or rls -- TODO: Remove rls and all configuration for it when all machines are up to date if vim.fn.executable("rust-analyzer") == 1 then -- Prefer rust-tools, if present utils.try_require("rust-tools", function(rust_tools) rust_tools.setup({ server = { capabilities = capabilities, on_attach = function(client, bufnr) default_attach(client, bufnr) -- TODO: override some bindings from rust-tools -- Eg. rust_tools.hover_actions.hover_actions or rt.code_action_group.code_action_group end, }, }) end, function() lsp_config.rust_analyzer.setup({ capabilities = capabilities, on_attach = default_attach, settings = { ["rust-analyzer"] = { cargo = { features = "all", }, }, }, }) end) else lsp_config.rls.setup({ capabilities = capabilities, on_attach = default_attach, }) end -- Configure neovim dev for when sumneko_lua is installed utils.try_require("neodev", function(neodev) local config = {} if utils.can_require("dapui") then config.plugins = { "nvim-dap-ui" } config.types = true end neodev.setup(config) end) -- Auto setup mason installed servers utils.try_require("mason-lspconfig", function(mason_lspconfig) -- Get list of servers that are installed but not set up local already_setup = lsp_config.util.available_servers() local needs_setup = vim.tbl_filter(function(server) return not vim.tbl_contains(already_setup, server) end, mason_lspconfig.get_installed_servers()) -- Setup each server with default config vim.tbl_map(function(server) if server == "lua_ls" then -- Disable formatting with lua_ls because I use luacheck local config = vim.tbl_extend("force", default_setup, { settings = { Lua = { format = { enable = false, }, }, }, }) lsp_config[server].setup(config) else lsp_config[server].setup(default_setup) end end, needs_setup) end) -- Config null-ls after lsps so we can disable for languages that have language servers require("plugins.null-ls").configure(default_setup) end) end function M.config_lsp_intaller() utils.try_require("mason", function(mason) mason.setup() end) utils.try_require("mason-lspconfig", function(mason_lspconfig) mason_lspconfig.setup() end) end return M