local nvim_lsp = require('lspconfig') local completion = require('completion') local lsp_status = require('lsp-status') -- Taken from https://www.reddit.com/r/neovim/comments/gyb077/nvimlsp_peek_defination_javascript_ttserver/ function preview_location(location, context, before_context) -- location may be LocationLink or Location (more useful for the former) context = context or 10 before_context = before_context or 5 local uri = location.targetUri or location.uri if uri == nil then return end local bufnr = vim.uri_to_bufnr(uri) if not vim.api.nvim_buf_is_loaded(bufnr) then vim.fn.bufload(bufnr) end local range = location.targetRange or location.range local contents = vim.api.nvim_buf_get_lines(bufnr, range.start.line - before_context, range["end"].line + 1 + context, false) local filetype = vim.api.nvim_buf_get_option(bufnr, "filetype") return vim.lsp.util.open_floating_preview(contents, filetype) end function preview_location_callback(_, method, result) local context = 10 if result == nil or vim.tbl_isempty(result) then print("No location found: " .. method) return nil end if vim.tbl_islist(result) then floating_buf, floating_win = preview_location(result[1], context) else floating_buf, floating_win = preview_location(result, context) end end function peek_definition() if vim.tbl_contains(vim.api.nvim_list_wins(), floating_win) then vim.api.nvim_set_current_win(floating_win) else local params = vim.lsp.util.make_position_params() return vim.lsp.buf_request(0, "textDocument/definition", params, preview_location_callback) end end local on_attach = function(client, bufnr) vim.api.nvim_buf_set_option(bufnr, 'omnifunc', 'v:lua.vim.lsp.omnifunc') completion.on_attach(client, bufnr) lsp_status.on_attach(client, bufnr) if client.config.flags then client.config.flags.allow_incremental_sync = true end local opts = { noremap=true, silent=true } vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gd', 'lua vim.lsp.buf.definition()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'K', 'lua vim.lsp.buf.hover()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gr', 'lua vim.lsp.buf.references()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'ga', 'lua vim.lsp.buf.code_action()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gi', 'lua vim.lsp.buf.implementation()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gs', 'lua vim.lsp.buf.signature_help()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gt', 'lua vim.lsp.buf.type_definition()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'g0', 'lua vim.lsp.buf.document_symbol()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gW', 'lua vim.lsp.buf.workspace_symbol()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'de', 'lua vim.lsp.buf.declaration()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'rn', 'lua vim.lsp.buf.rename()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'pd', 'lua peek_definition()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', '[d', 'lua vim.lsp.diagnostic.goto_prev { wrap = false }', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', ']d', 'lua vim.lsp.diagnostic.goto_next { wrap = false }', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', '[D', 'lua vim.lsp.diagnostic.goto_prev()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', ']D', 'lua vim.lsp.diagnostic.goto_next()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'do', 'lua vim.lsp.diagnostic.set_loclist()', opts) vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gD', 'lua vim.lsp.diagnostic.show_line_diagnostics()', opts) end local servers = { 'hls', 'rust_analyzer' } for _, lsp in ipairs(servers) do lsp_status.register_progress() lsp_status.config({ status_symbol = '', indicator_errors = 'e', indicator_warnings = 'w', indicator_info = 'i', indicator_hint = 'h', indicator_ok = '✔️', spinner_frames = { '⣾', '⣽', '⣻', '⢿', '⡿', '⣟', '⣯', '⣷' }, }) nvim_lsp[lsp].setup { on_attach = on_attach, capabilities = lsp_status.capabilities } end