Files
outline.nvim/lua/outline/providers/nvim-lsp.lua
2023-12-27 22:14:12 +08:00

194 lines
5.5 KiB
Lua

local cfg = require('outline.config')
local jsx = require('outline.providers.jsx')
local lsp_utils = require('outline.utils.lsp')
local l = vim.lsp
local M = {
name = 'lsp',
---@type lsp.client
client = nil,
}
local request_timeout = 2000
function M.get_status()
if not M.client then
return { 'No clients' }
end
return { 'client: ' .. M.client.name }
end
local function get_appropriate_client(bufnr, capability)
local clients = l.get_active_clients({ bufnr = bufnr })
local use_client
for _, client in ipairs(clients) do
if cfg.is_client_blacklisted(client) then
goto continue
else
if client.server_capabilities[capability] then
use_client = client
M.client = client
break
end
end
::continue::
end
return use_client
end
---@return boolean
function M.supports_buffer(bufnr)
local client = get_appropriate_client(bufnr, 'documentSymbolProvider')
if not client then
return false
end
return true
end
---@param response outline.ProviderSymbol[]
---@return outline.ProviderSymbol[]
local function postprocess_symbols(response)
local symbols = lsp_utils.flatten_response(response)
local jsx_symbols = jsx.get_symbols()
if #jsx_symbols > 0 then
return lsp_utils.merge_symbols(symbols, jsx_symbols)
else
return symbols
end
end
---@param on_symbols fun(symbols?:outline.ProviderSymbol[], opts?:table)
---@param opts table
function M.request_symbols(on_symbols, opts)
local params = {
textDocument = l.util.make_text_document_params(),
}
l.buf_request_all(0, 'textDocument/documentSymbol', params, function(response)
response = postprocess_symbols(response)
on_symbols(response, opts)
end)
end
-- No good way to update outline when LSP action complete for now
---@param sidebar outline.Sidebar
---@return boolean success
function M.code_actions(sidebar)
local client = get_appropriate_client(sidebar.code.buf, 'codeActionProvider')
if not client then
return false
end
-- NOTE: Unfortunately the code_action function provided by neovim does a
-- lot, yet it doesn't let us filter clients. Since handling of code_actions
-- is beyond the scope of outline.nvim itself, we will not respect
-- blacklist_clients for code actions for now. Code actions feature would not
-- actually be included if I were to write this plugin from scratch. However
-- we still keep it because many people are moving here from
-- symbols-outline.nvim, which happened to implement this feature.
sidebar:wrap_goto_location(function()
l.buf.code_action()
end)
return true
end
---@see rename_symbol
---@param sidebar outline.Sidebar
---@param node outline.FlatSymbol
---@return boolean success
local function legacy_rename(sidebar, client, node)
-- Using fn.input so it's synchronous
local new_name = vim.fn.input({ prompt = 'New Name: ', default = node.name })
if not new_name or new_name == '' or new_name == node.name then
return true
end
local params = {
textDocument = { uri = 'file://' .. vim.api.nvim_buf_get_name(sidebar.code.buf) },
position = { line = node.line, character = node.character },
bufnr = sidebar.code.buf,
newName = new_name,
}
local status, err =
client.request_sync('textDocument/rename', params, request_timeout, sidebar.code.buf)
if status == nil or status.err or err or status.result == nil then
return false
end
l.util.apply_workspace_edit(status.result, client.offset_encoding)
node.name = new_name
sidebar:_update_lines(false)
return true
end
---Synchronously request rename from LSP
---@param sidebar outline.Sidebar
---@return boolean success
function M.rename_symbol(sidebar)
local client = get_appropriate_client(sidebar.code.buf, 'renameProvider')
if not client then
return false
end
local node = sidebar:_current_node()
if not node then
return false
end
if _G._outline_nvim_has[8] then
sidebar:wrap_goto_location(function()
-- Options table with filter key only added in nvim-0.8
-- Use vim.lsp's function because it has better support.
l.buf.rename(nil, { filter = function (client)
return not cfg.is_client_blacklisted(client)
end })
end)
return true
else
return legacy_rename(sidebar, client, node)
end
end
---Synchronously request and show hover info from LSP
---@param sidebar outline.Sidebar
---@return boolean success
function M.show_hover(sidebar)
local client = get_appropriate_client(sidebar.code.buf, 'hoverProvider')
if not client then
return false
end
local node = sidebar:_current_node()
if not node then
return false
end
local params = {
textDocument = { uri = vim.uri_from_bufnr(sidebar.code.buf) },
position = { line = node.line, character = node.character },
bufnr = sidebar.code.buf,
}
local status, err = client.request_sync('textDocument/hover', params, request_timeout)
if status == nil or status.err or err or not status.result or not status.result.contents then
return false
end
local md_lines = l.util.convert_input_to_markdown_lines(status.result.contents)
md_lines = l.util.trim_empty_lines(md_lines)
if vim.tbl_isempty(md_lines) then
-- Request was successful, but there is no hover content
return true
end
local code_width = vim.api.nvim_win_get_width(sidebar.code.win)
local bufnr, winnr = l.util.open_floating_preview(md_lines, 'markdown', {
border = cfg.o.preview_window.border,
width = code_width,
})
vim.api.nvim_win_set_option(winnr, 'winhighlight', cfg.o.preview_window.winhl)
return true
end
return M