feat(lsp): migrate from blink.cmp to built-in completion
This commit is contained in:
@@ -1,4 +1,3 @@
|
|||||||
local lsp = require("lsp")
|
|
||||||
local util = require("util")
|
local util = require("util")
|
||||||
|
|
||||||
---@type vim.lsp.Config
|
---@type vim.lsp.Config
|
||||||
@@ -9,8 +8,6 @@ return {
|
|||||||
"zsh",
|
"zsh",
|
||||||
},
|
},
|
||||||
on_attach = function(client, bufnr)
|
on_attach = function(client, bufnr)
|
||||||
lsp.on_attach(client, bufnr)
|
|
||||||
|
|
||||||
vim.keymap.set("n", "<leader>lf", function()
|
vim.keymap.set("n", "<leader>lf", function()
|
||||||
util.format({
|
util.format({
|
||||||
buf = bufnr,
|
buf = bufnr,
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
local Linter = require("linter")
|
local Linter = require("linter")
|
||||||
local lsp = require("lsp")
|
|
||||||
|
|
||||||
---@type vim.lsp.Config
|
---@type vim.lsp.Config
|
||||||
return {
|
return {
|
||||||
@@ -14,8 +13,6 @@ return {
|
|||||||
},
|
},
|
||||||
single_file_support = true,
|
single_file_support = true,
|
||||||
on_attach = function(client, bufnr)
|
on_attach = function(client, bufnr)
|
||||||
lsp.on_attach(client, bufnr)
|
|
||||||
|
|
||||||
Linter.add(bufnr, {
|
Linter.add(bufnr, {
|
||||||
cmd = {
|
cmd = {
|
||||||
"clang-tidy",
|
"clang-tidy",
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
local lsp = require("lsp")
|
|
||||||
local util = require("util")
|
local util = require("util")
|
||||||
|
|
||||||
---@type vim.lsp.Config
|
---@type vim.lsp.Config
|
||||||
@@ -13,8 +12,6 @@ return {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
on_attach = function(client, bufnr)
|
on_attach = function(client, bufnr)
|
||||||
lsp.on_attach(client, bufnr)
|
|
||||||
|
|
||||||
vim.keymap.set("n", "<leader>lf", function()
|
vim.keymap.set("n", "<leader>lf", function()
|
||||||
util.format({
|
util.format({
|
||||||
buf = bufnr,
|
buf = bufnr,
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
local Linter = require("linter")
|
local Linter = require("linter")
|
||||||
local lsp = require("lsp")
|
|
||||||
local util = require("util")
|
local util = require("util")
|
||||||
|
|
||||||
---@type vim.lsp.Config
|
---@type vim.lsp.Config
|
||||||
@@ -13,8 +12,6 @@ return {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
on_attach = function(client, bufnr)
|
on_attach = function(client, bufnr)
|
||||||
lsp.on_attach(client, bufnr)
|
|
||||||
|
|
||||||
Linter.add(bufnr, {
|
Linter.add(bufnr, {
|
||||||
cmd = {
|
cmd = {
|
||||||
"phpcs",
|
"phpcs",
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ return {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
on_attach = function(client, bufnr)
|
on_attach = function(client, bufnr)
|
||||||
require("lsp").on_attach(client, bufnr)
|
|
||||||
local util = require("util")
|
local util = require("util")
|
||||||
|
|
||||||
vim.keymap.set("n", "<leader>lf", function()
|
vim.keymap.set("n", "<leader>lf", function()
|
||||||
|
|||||||
@@ -1,11 +1,8 @@
|
|||||||
local lsp = require("lsp")
|
|
||||||
local util = require("util")
|
local util = require("util")
|
||||||
|
|
||||||
---@type vim.lsp.Config
|
---@type vim.lsp.Config
|
||||||
return {
|
return {
|
||||||
on_attach = function(client, bufnr)
|
on_attach = function(client, bufnr)
|
||||||
lsp.on_attach(client, bufnr)
|
|
||||||
|
|
||||||
vim.keymap.set("n", "<leader>lf", function()
|
vim.keymap.set("n", "<leader>lf", function()
|
||||||
vim.lsp.buf.format()
|
vim.lsp.buf.format()
|
||||||
util.format({
|
util.format({
|
||||||
|
|||||||
@@ -25,10 +25,6 @@ local ts = require("ts")
|
|||||||
require("pack").setup({
|
require("pack").setup({
|
||||||
"https://github.com/navarasu/onedark.nvim",
|
"https://github.com/navarasu/onedark.nvim",
|
||||||
"https://github.com/ibhagwan/fzf-lua",
|
"https://github.com/ibhagwan/fzf-lua",
|
||||||
{
|
|
||||||
"https://github.com/saghen/blink.cmp",
|
|
||||||
version = vim.version.range("^1"),
|
|
||||||
},
|
|
||||||
"https://github.com/neovim/nvim-lspconfig",
|
"https://github.com/neovim/nvim-lspconfig",
|
||||||
"https://github.com/mason-org/mason.nvim",
|
"https://github.com/mason-org/mason.nvim",
|
||||||
"https://github.com/owallb/mason-auto-install.nvim",
|
"https://github.com/owallb/mason-auto-install.nvim",
|
||||||
|
|||||||
@@ -26,14 +26,14 @@ vim.opt.foldlevel = 99
|
|||||||
vim.opt.foldlevelstart = 99
|
vim.opt.foldlevelstart = 99
|
||||||
vim.opt.foldmethod = "indent"
|
vim.opt.foldmethod = "indent"
|
||||||
vim.opt.foldignore = ""
|
vim.opt.foldignore = ""
|
||||||
---@diagnostic disable-next-line: undefined-field
|
vim.opt.completeopt = {
|
||||||
vim.opt.completeopt:append({
|
|
||||||
"menu",
|
"menu",
|
||||||
"menuone",
|
"menuone",
|
||||||
"preview",
|
"popup",
|
||||||
"noinsert",
|
"noinsert",
|
||||||
"noselect",
|
"noselect",
|
||||||
})
|
}
|
||||||
|
vim.opt.complete = { "o" }
|
||||||
-- set nowrap
|
-- set nowrap
|
||||||
vim.opt.matchpairs:append({ "<:>" })
|
vim.opt.matchpairs:append({ "<:>" })
|
||||||
-- Only relevant with wrap enabled (default)
|
-- Only relevant with wrap enabled (default)
|
||||||
|
|||||||
+221
-18
@@ -1,8 +1,12 @@
|
|||||||
---@type fun(name: string, cfg: vim.lsp.Config)
|
|
||||||
vim.lsp.config = vim.lsp.config
|
|
||||||
|
|
||||||
local log = require("log")
|
local log = require("log")
|
||||||
|
|
||||||
|
local WORD_CHARS =
|
||||||
|
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"
|
||||||
|
|
||||||
|
local function fence(ft, text)
|
||||||
|
return string.format("```%s\n%s\n```", ft, text)
|
||||||
|
end
|
||||||
|
|
||||||
local M = {}
|
local M = {}
|
||||||
|
|
||||||
M.diagnostic_signs = {
|
M.diagnostic_signs = {
|
||||||
@@ -45,6 +49,58 @@ function M.on_attach(client, buf)
|
|||||||
client.settings
|
client.settings
|
||||||
) or client.settings
|
) or client.settings
|
||||||
|
|
||||||
|
if
|
||||||
|
client:supports_method(vim.lsp.protocol.Methods.textDocument_completion)
|
||||||
|
then
|
||||||
|
local provider = client.server_capabilities.completionProvider
|
||||||
|
provider.triggerCharacters = provider.triggerCharacters or {}
|
||||||
|
if not provider._word_chars_added then
|
||||||
|
provider._word_chars_added = true
|
||||||
|
for c in WORD_CHARS:gmatch(".") do
|
||||||
|
table.insert(provider.triggerCharacters, c)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local group = vim.api.nvim_create_augroup(
|
||||||
|
"lsp_completion_" .. buf,
|
||||||
|
{ clear = true }
|
||||||
|
)
|
||||||
|
vim.api.nvim_create_autocmd("TextChangedI", {
|
||||||
|
buffer = buf,
|
||||||
|
group = group,
|
||||||
|
callback = function()
|
||||||
|
if vim.fn.pumvisible() ~= 0 then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
local col = vim.fn.col(".") - 1
|
||||||
|
if col <= 0 then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
local char = vim.api.nvim_get_current_line():sub(col, col)
|
||||||
|
if char:match("[%w_]") then
|
||||||
|
vim.lsp.completion.get()
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
vim.lsp.completion.enable(true, client.id, buf, {
|
||||||
|
autotrigger = true,
|
||||||
|
convert = function(item)
|
||||||
|
local signature = vim.tbl_get(
|
||||||
|
item,
|
||||||
|
"labelDetails",
|
||||||
|
"description"
|
||||||
|
) or item.detail or ""
|
||||||
|
return {
|
||||||
|
abbr = item.label:match("[^(]+") or item.label,
|
||||||
|
menu = "",
|
||||||
|
kind = "",
|
||||||
|
info = signature ~= ""
|
||||||
|
and fence(vim.bo[buf].filetype, signature)
|
||||||
|
or " ",
|
||||||
|
}
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
vim.api.nvim_create_autocmd("LspProgress", {
|
vim.api.nvim_create_autocmd("LspProgress", {
|
||||||
buffer = buf,
|
buffer = buf,
|
||||||
callback = function(ev)
|
callback = function(ev)
|
||||||
@@ -72,7 +128,7 @@ function M.setup()
|
|||||||
border = "rounded",
|
border = "rounded",
|
||||||
focusable = true,
|
focusable = true,
|
||||||
format = function(diagnostic)
|
format = function(diagnostic)
|
||||||
return string.format("%s", diagnostic.message)
|
return diagnostic.message
|
||||||
end,
|
end,
|
||||||
width = 80,
|
width = 80,
|
||||||
},
|
},
|
||||||
@@ -109,23 +165,170 @@ function M.setup()
|
|||||||
})
|
})
|
||||||
|
|
||||||
local capabilities = vim.lsp.protocol.make_client_capabilities()
|
local capabilities = vim.lsp.protocol.make_client_capabilities()
|
||||||
capabilities = vim.tbl_deep_extend(
|
capabilities.textDocument.completion.completionItem.snippetSupport = false
|
||||||
"force",
|
|
||||||
capabilities,
|
|
||||||
require("blink.cmp").get_lsp_capabilities({
|
|
||||||
textDocument = {
|
|
||||||
completion = {
|
|
||||||
completionItem = {
|
|
||||||
snippetSupport = false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, false)
|
|
||||||
)
|
|
||||||
vim.lsp.config("*", {
|
vim.lsp.config("*", {
|
||||||
capabilities = capabilities,
|
capabilities = capabilities,
|
||||||
on_attach = M.on_attach,
|
|
||||||
})
|
})
|
||||||
|
|
||||||
|
vim.api.nvim_create_autocmd("LspAttach", {
|
||||||
|
callback = function(ev)
|
||||||
|
local client = vim.lsp.get_client_by_id(ev.data.client_id)
|
||||||
|
if client then
|
||||||
|
M.on_attach(client, ev.buf)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
|
||||||
|
local function style_popup(winid, bufnr, width)
|
||||||
|
if
|
||||||
|
not winid
|
||||||
|
or winid <= 0
|
||||||
|
or not vim.api.nvim_win_is_valid(winid)
|
||||||
|
or vim.api.nvim_win_get_config(winid).relative == ""
|
||||||
|
then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
local cfg = { border = "rounded" }
|
||||||
|
if width then
|
||||||
|
cfg.width = math.min(width, 80)
|
||||||
|
end
|
||||||
|
if bufnr and vim.api.nvim_buf_is_valid(bufnr) then
|
||||||
|
vim.wo[winid].wrap = true
|
||||||
|
vim.wo[winid].linebreak = true
|
||||||
|
vim.wo[winid].conceallevel = 2
|
||||||
|
pcall(vim.treesitter.start, bufnr, "markdown")
|
||||||
|
end
|
||||||
|
pcall(vim.api.nvim_win_set_config, winid, cfg)
|
||||||
|
end
|
||||||
|
|
||||||
|
vim.api.nvim_create_autocmd("CompleteChanged", {
|
||||||
|
callback = function(ev)
|
||||||
|
local cinfo = vim.fn.complete_info({
|
||||||
|
"selected",
|
||||||
|
"preview_winid",
|
||||||
|
"preview_bufnr",
|
||||||
|
})
|
||||||
|
style_popup(cinfo.preview_winid, cinfo.preview_bufnr)
|
||||||
|
|
||||||
|
local completed = vim.v.event.completed_item or {}
|
||||||
|
local lsp_item = vim.tbl_get(
|
||||||
|
completed,
|
||||||
|
"user_data",
|
||||||
|
"nvim",
|
||||||
|
"lsp",
|
||||||
|
"completion_item"
|
||||||
|
)
|
||||||
|
local client_id =
|
||||||
|
vim.tbl_get(completed, "user_data", "nvim", "lsp", "client_id")
|
||||||
|
if not lsp_item or not client_id then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local client = vim.lsp.get_client_by_id(client_id)
|
||||||
|
if
|
||||||
|
not client
|
||||||
|
or not client:supports_method(
|
||||||
|
vim.lsp.protocol.Methods.completionItem_resolve
|
||||||
|
)
|
||||||
|
then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local selected = cinfo.selected
|
||||||
|
local word = completed.word
|
||||||
|
client:request(
|
||||||
|
vim.lsp.protocol.Methods.completionItem_resolve,
|
||||||
|
lsp_item,
|
||||||
|
function(err, result)
|
||||||
|
if err or not result then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local cur = vim.fn.complete_info({
|
||||||
|
"selected",
|
||||||
|
"completed",
|
||||||
|
"preview_winid",
|
||||||
|
})
|
||||||
|
if
|
||||||
|
cur.selected ~= selected
|
||||||
|
or (vim.tbl_get(cur, "completed", "word") or "")
|
||||||
|
~= word
|
||||||
|
then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local signature = vim.tbl_get(
|
||||||
|
result,
|
||||||
|
"labelDetails",
|
||||||
|
"description"
|
||||||
|
) or result.detail or ""
|
||||||
|
local doc = result.documentation
|
||||||
|
if type(doc) == "table" then
|
||||||
|
doc = doc.value
|
||||||
|
end
|
||||||
|
doc = doc or ""
|
||||||
|
|
||||||
|
local ft = vim.bo[ev.buf].filetype
|
||||||
|
local code_parts = {}
|
||||||
|
if result.additionalTextEdits then
|
||||||
|
for _, edit in ipairs(result.additionalTextEdits) do
|
||||||
|
local text = (edit.newText or ""):gsub("%s+$", "")
|
||||||
|
if text ~= "" then
|
||||||
|
table.insert(code_parts, fence(ft, text))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if signature ~= "" then
|
||||||
|
table.insert(code_parts, fence(ft, signature))
|
||||||
|
end
|
||||||
|
local sections = {}
|
||||||
|
if #code_parts > 0 then
|
||||||
|
table.insert(sections, table.concat(code_parts, "\n\n"))
|
||||||
|
end
|
||||||
|
if doc ~= "" then
|
||||||
|
table.insert(sections, doc)
|
||||||
|
end
|
||||||
|
if #sections == 0 then
|
||||||
|
if
|
||||||
|
cur.preview_winid
|
||||||
|
and cur.preview_winid > 0
|
||||||
|
and vim.api.nvim_win_is_valid(cur.preview_winid)
|
||||||
|
then
|
||||||
|
pcall(
|
||||||
|
vim.api.nvim_win_close,
|
||||||
|
cur.preview_winid,
|
||||||
|
true
|
||||||
|
)
|
||||||
|
end
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local max_w = 0
|
||||||
|
for _, s in ipairs(sections) do
|
||||||
|
for _, line in
|
||||||
|
ipairs(vim.split(s, "\n", { plain = true }))
|
||||||
|
do
|
||||||
|
max_w =
|
||||||
|
math.max(max_w, vim.fn.strdisplaywidth(line))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
local sep = "\n"
|
||||||
|
.. string.rep("─", math.min(max_w, 80))
|
||||||
|
.. "\n"
|
||||||
|
local combined = table.concat(sections, sep)
|
||||||
|
|
||||||
|
local windata = vim.api.nvim__complete_set(selected, {
|
||||||
|
info = combined,
|
||||||
|
})
|
||||||
|
if windata then
|
||||||
|
style_popup(windata.winid, windata.bufnr, max_w)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
ev.buf
|
||||||
|
)
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
|
||||||
vim.lsp.log.set_level(vim.log.levels.WARN)
|
vim.lsp.log.set_level(vim.log.levels.WARN)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,88 +0,0 @@
|
|||||||
require("blink.cmp").setup({
|
|
||||||
completion = {
|
|
||||||
documentation = {
|
|
||||||
auto_show = true,
|
|
||||||
window = {
|
|
||||||
winhighlight = "",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
ghost_text = {
|
|
||||||
enabled = true,
|
|
||||||
},
|
|
||||||
list = {
|
|
||||||
selection = {
|
|
||||||
preselect = false,
|
|
||||||
auto_insert = false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
menu = {
|
|
||||||
border = "none",
|
|
||||||
draw = {
|
|
||||||
align_to = "simple_label",
|
|
||||||
columns = {
|
|
||||||
{ "simple_label" },
|
|
||||||
{ "kind_icon", "label_description", gap = 1 },
|
|
||||||
},
|
|
||||||
components = {
|
|
||||||
simple_label = {
|
|
||||||
width = { fill = true, max = 60 },
|
|
||||||
text = function(ctx)
|
|
||||||
return ctx.label
|
|
||||||
end,
|
|
||||||
highlight = function(ctx)
|
|
||||||
local highlights = {
|
|
||||||
{
|
|
||||||
0,
|
|
||||||
#ctx.label,
|
|
||||||
group = ctx.deprecated
|
|
||||||
and "BlinkCmpLabelDeprecated"
|
|
||||||
or "BlinkCmpLabel",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
for _, idx in ipairs(ctx.label_matched_indices) do
|
|
||||||
table.insert(highlights, {
|
|
||||||
idx,
|
|
||||||
idx + 1,
|
|
||||||
group = "BlinkCmpLabelMatch",
|
|
||||||
})
|
|
||||||
end
|
|
||||||
|
|
||||||
return highlights
|
|
||||||
end,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fuzzy = {
|
|
||||||
implementation = "prefer_rust_with_warning",
|
|
||||||
},
|
|
||||||
signature = {
|
|
||||||
enabled = true,
|
|
||||||
window = {
|
|
||||||
winhighlight = "",
|
|
||||||
show_documentation = false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
sources = {
|
|
||||||
default = {
|
|
||||||
"lsp",
|
|
||||||
"path",
|
|
||||||
},
|
|
||||||
transform_items = function(_, items)
|
|
||||||
return vim.tbl_filter(function(item)
|
|
||||||
return item.kind
|
|
||||||
~= require("blink.cmp.types").CompletionItemKind.Snippet
|
|
||||||
end, items)
|
|
||||||
end,
|
|
||||||
},
|
|
||||||
keymap = {
|
|
||||||
preset = "none",
|
|
||||||
["<Tab>"] = { "insert_next", "fallback" },
|
|
||||||
["<S-Tab>"] = { "insert_prev", "fallback" },
|
|
||||||
["<CR>"] = { "accept", "fallback" },
|
|
||||||
["<C-k>"] = { "show_signature", "hide_signature", "fallback" },
|
|
||||||
["<C-n>"] = { "snippet_forward", "fallback" },
|
|
||||||
["<C-p>"] = { "snippet_backward", "fallback" },
|
|
||||||
},
|
|
||||||
})
|
|
||||||
Reference in New Issue
Block a user