-- 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 local function buf_set_option(...) vim.api.nvim_buf_set_option(bufnr, ...) 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 buf_set_option("omnifunc", "v:lua.vim.lsp.omnifunc") if server_capabilities.documentSymbolProvider then buf_set_option("tagfunc", "v:lua.vim.lsp.tagfunc") end if server_capabilities.documentFormattingProvider then buf_set_option("formatexpr", "v:lua.vim.lsp.formatexpr()") end end -- Mappings -- TODO: use functions instead of strings when dropping 0.6 local lsp_keymap = utils.curry_keymap("n", "l", { buffer = bufnr }) lsp_keymap("h", "lua vim.lsp.buf.hover()", { desc = "Display hover" }) lsp_keymap("rn", "lua vim.lsp.buf.rename()", { desc = "Refactor rename" }) lsp_keymap("e", "lua vim.diagnostic.open_float()", { desc = "Open float dialog" }) lsp_keymap("D", "lua vim.lsp.buf.declaration()", { desc = "Go to declaration" }) lsp_keymap("d", "lua vim.lsp.buf.definition()", { desc = "Go to definition" }) lsp_keymap("t", "lua vim.lsp.buf.type_definition()", { desc = "Go to type definition" }) lsp_keymap("i", "lua vim.lsp.buf.implementation()", { desc = "Show implementations" }) lsp_keymap("s", "lua vim.lsp.buf.signature_help()", { desc = "Show signature help" }) lsp_keymap("wa", "lua vim.lsp.buf.add_workspace_folder()", { desc = "Workspace: Add folder" }) lsp_keymap("wr", "lua vim.lsp.buf.remove_workspace_folder()", { desc = "Workspace: Remove folder" }) lsp_keymap( "wl", "lua print(vim.inspect(vim.lsp.buf.list_workspace_folders()))", { desc = "Workspace: List folders" } ) lsp_keymap("r", "lua vim.lsp.buf.references()", { desc = "References" }) lsp_keymap("p", "lua vim.diagnostic.goto_prev()", { desc = "Previous diagnostic" }) lsp_keymap("n", "lua vim.diagnostic.goto_next()", { desc = "Next diagnostic" }) -- Set insert keymap for signature help utils.keymap_set( "i", "", "lua vim.lsp.buf.signature_help()", { buffer = bufnr, desc = "Show signature help" } ) -- Some top level aliases or remaps utils.keymap_set("n", "K", "lua vim.lsp.buf.hover()", { buffer = bufnr, desc = "Display hover" }) utils.keymap_set( "n", "gD", "lua vim.lsp.buf.declaration()", { buffer = bufnr, desc = "Go to declaration" } ) utils.keymap_set( "n", "gd", "lua vim.lsp.buf.definition()", { buffer = bufnr, desc = "Go to definition" } ) utils.keymap_set( "n", "rn", "lua vim.lsp.buf.rename()", { buffer = bufnr, desc = "Refactor rename" } ) utils.keymap_set( "n", "[d", "lua vim.diagnostic.goto_prev()", { buffer = bufnr, desc = "Previous diagnostic" } ) utils.keymap_set( "n", "]d", "lua vim.diagnostic.goto_next()", { buffer = bufnr, desc = "Next diagnostic" } ) -- Older keymaps --[[ buf_set_keymap("n", "gi", "lua vim.lsp.buf.implementation()", opts) buf_set_keymap("n", "wa", "lua vim.lsp.buf.add_workspace_folder()", opts) buf_set_keymap("n", "wr", "lua vim.lsp.buf.remove_workspace_folder()", opts) buf_set_keymap("n", "wl", "lua print(vim.inspect(vim.lsp.buf.list_workspace_folders()))", opts) buf_set_keymap("n", "D", "lua vim.lsp.buf.type_definition()", opts) buf_set_keymap("n", "rn", "lua vim.lsp.buf.rename()", opts) buf_set_keymap("n", "gr", "lua vim.lsp.buf.references()", opts) buf_set_keymap("n", "e", "lua vim.diagnostic.open_float()", opts) buf_set_keymap("n", "q", "lua vim.lsp.diagnostic.set_loclist()", opts) --]] -- Open diagnostic on hold if vim["diagnostic"] ~= nil then -- TODO: When dropping 0.6, use lua aucommand api vim.cmd([[autocmd CursorHold * lua vim.diagnostic.open_float(nil, {focus=false, scope="cursor"})]]) end -- Use IncRename if available if utils.try_require("inc_rename") ~= nil then -- TODO: Should I be using this for calling lua functions from keymaps? 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", "lua vim.lsp.buf.format({async=true})", { desc = "Format code" }) lsp_keymap( "f", "lua vim.lsp.buf.format({async=true})", { 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 lsp_keymap("f", "lua vim.lsp.buf.formatting()", { desc = "Format code" }) lsp_keymap( "f", "lua vim.lsp.buf.range_formatting()", { mode = "v", desc = "Format selected code" } ) if server_capabilities.documentFormattingProvider then vim.cmd([[ augroup lsp_format autocmd! autocmd BufWritePre lua vim.lsp.buf.formatting_sync(nil, 1000) augroup END ]]) end end -- Set autocommands conditional on server_capabilities if server_capabilities.documentHighlightProvider then vim.cmd([[ :highlight link LspReferenceRead MatchParen :highlight link LspReferenceText MatchParen :highlight link LspReferenceWrite MatchParen augroup lsp_document_highlight autocmd! autocmd CursorHold lua vim.lsp.buf.document_highlight() autocmd CursorMoved lua vim.lsp.buf.clear_references() augroup END ]]) end -- Some override some fuzzy finder bindings to use lsp sources if utils.try_require("telescope") ~= nil then -- Replace some Telescope bindings with LSP versions local telescope_keymap = utils.curry_keymap("n", "f", { buffer = bufnr }) if server_capabilities.documentSymbolProvider then telescope_keymap("t", "Telescope lsp_document_symbols", { desc = "Find buffer tags" }) end if server_capabilities.workspaceSymbolProvider then telescope_keymap("T", "Telescope lsp_dynamic_workspace_symbols", { desc = "Find tags" }) end -- Replace some LSP bindings with Telescope ones if server_capabilities.definitionProvider then lsp_keymap("d", "Telescope lsp_definitions", { desc = "Find definition" }) end if server_capabilities.typeDefinitionProvider then lsp_keymap("t", "Telescope lsp_type_definition", { desc = "Find type definition" }) end lsp_keymap("i", "Telescope lsp_implementations", { desc = "Find implementations" }) lsp_keymap("r", "Telescope lsp_references", { desc = "Find references" }) lsp_keymap("A", "Telescope lsp_code_actions", { desc = "Select code actions" }) lsp_keymap("A", "Telescope lsp_range_code_actions", { mode = "v", desc = "Select code actions" }) 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 if lsp_config["util"] and lsp_config.util["available_servers"] then already_setup = lsp_config.util.available_servers() else -- HACK: For lspconfig versions lower than 0.1.4, which is required for nvim <0.7.0 already_setup = lsp_config.available_servers() end 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