You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

286 lines
10 KiB

9 months ago
9 months ago
  1. return {
  2. "neovim/nvim-lspconfig",
  3. event = { "BufReadPre", "BufNewFile" },
  4. dependencies = {
  5. { "antosha417/nvim-lsp-file-operations", config = true },
  6. { "williamboman/mason.nvim", config = true },
  7. { "williamboman/mason-lspconfig.nvim" },
  8. { "WhoIsSethDaniel/mason-tool-installer.nvim" },
  9. { "j-hui/fidget.nvim", opts = {} },
  10. { "folke/neodev.nvim", opts = {} },
  11. },
  12. config = function()
  13. local lspconfig = require("lspconfig")
  14. local util = require("lspconfig.util")
  15. local keymap = vim.keymap
  16. vim.api.nvim_create_autocmd("LspAttach", {
  17. group = vim.api.nvim_create_augroup("lsp-attach", { clear = true }),
  18. callback = function(event)
  19. local opts = { noremap = true, silent = true }
  20. opts.desc = "Show LSP references"
  21. keymap.set("n", "gr", "<cmd>Telescope lsp_references<CR>", opts)
  22. opts.desc = "Go to declaration"
  23. keymap.set("n", "gD", vim.lsp.buf.declaration, opts)
  24. opts.desc = "Show LSP definitions"
  25. keymap.set("n", "gd", "<cmd>Telescope lsp_definitions<CR>", opts)
  26. opts.desc = "Show LSP implementations"
  27. keymap.set("n", "gi", "<cmd>Telescope lsp_implementations<CR>", opts)
  28. opts.desc = "Show LSP type definitions"
  29. keymap.set("n", "gt", "<cmd>Telescope lsp_type_definitions<CR>", opts)
  30. opts.desc = "See available code actions"
  31. keymap.set({ "n", "v" }, "<leader>ca", vim.lsp.buf.code_action, opts)
  32. opts.desc = "Smart rename"
  33. keymap.set("n", "<leader>rn", vim.lsp.buf.rename, opts)
  34. opts.desc = "Show buffer diagnostics"
  35. keymap.set("n", "<leader>D", "<cmd>Telescope diagnostics bufnr=0<CR>", opts)
  36. opts.desc = "Show line diagnostics"
  37. keymap.set("n", "<leader>d", vim.diagnostic.open_float, opts)
  38. opts.desc = "Go to previous diagnostic"
  39. keymap.set("n", "[d", vim.diagnostic.goto_prev, opts)
  40. opts.desc = "Go to next diagnostic"
  41. keymap.set("n", "]d", vim.diagnostic.goto_next, opts)
  42. opts.desc = "Go to previous diagnostic (error only)"
  43. keymap.set("n", "[e", function()
  44. vim.diagnostic.goto_prev({ severity = vim.diagnostic.severity.ERROR })
  45. end, opts)
  46. opts.desc = "Go to next diagnostic (error only)"
  47. keymap.set("n", "]e", function()
  48. vim.diagnostic.goto_next({ severity = vim.diagnostic.severity.ERROR })
  49. end, opts)
  50. opts.desc = "Show documentation for what is under cursor"
  51. keymap.set("n", "K", vim.lsp.buf.hover, opts)
  52. local client = vim.lsp.get_client_by_id(event.data.client_id)
  53. if client and client.server_capabilities.documentHighlightProvider then
  54. vim.api.nvim_create_autocmd("LspDetach", {
  55. group = vim.api.nvim_create_augroup("lsp-detach", { clear = true }),
  56. callback = function(event2)
  57. vim.lsp.buf.clear_references()
  58. end,
  59. })
  60. end
  61. end,
  62. })
  63. local capabilities = vim.lsp.protocol.make_client_capabilities()
  64. capabilities = vim.tbl_deep_extend(
  65. "force",
  66. capabilities,
  67. -- require('blink.cmp').get_lsp_capabilities(),
  68. require('cmp_nvim_lsp').default_capabilities()
  69. )
  70. local base_path = ""
  71. if (vim.loop.fs_stat("/usr/lib/node_modules")) then
  72. base_path = "/usr/lib/node_modules"
  73. elseif (vim.loop.fs_stat("/usr/local/lib/node_modules")) then
  74. base_path = "/usr/local/lib/node_modules"
  75. elseif (vim.loop.fs_stat("/opt/homebrew/lib/node_modules")) then
  76. base_path = "/opt/homebrew/lib/node_modules"
  77. end
  78. local function get_typescript_server_path(root_dir)
  79. local global_ts = base_path .. "/typescript/lib"
  80. local found_ts = ""
  81. local function check_dir(path)
  82. found_ts = util.path.join(path, "node_modules", "typescript", "lib")
  83. if vim.loop.fs_stat(found_ts) then
  84. return path
  85. end
  86. end
  87. if util.search_ancestors(root_dir .. '/frontend/node_modules', check_dir) then
  88. return found_ts
  89. end
  90. if util.search_ancestors(root_dir .. '/node_modules', check_dir) then
  91. return found_ts
  92. end
  93. return global_ts
  94. end
  95. local path = util.path
  96. local function organize_imports()
  97. local params = {
  98. command = "_typescript.organizeImports",
  99. arguments = { vim.api.nvim_buf_get_name(0) },
  100. title = ""
  101. }
  102. vim.lsp.buf.execute_command(params)
  103. end
  104. local servers = {
  105. ts_ls = {
  106. init_options = {
  107. plugins = {
  108. {
  109. name = "@vue/typescript-plugin",
  110. location = base_path .. "/@vue/typescript-plugin",
  111. languages = { "javascript", "typescript", "vue", "react" },
  112. },
  113. },
  114. },
  115. filetypes = {
  116. "javascript",
  117. "typescript",
  118. "vue",
  119. "typescriptreact",
  120. },
  121. commands = {
  122. OrganizeImports = {
  123. organize_imports,
  124. description = "Organize Imports"
  125. }
  126. },
  127. on_new_config = function(new_config, new_root_dir)
  128. if get_typescript_server_path then
  129. new_config.init_options.typescript = new_config.init_options.typescript or {}
  130. new_config.init_options.typescript.tsdk = get_typescript_server_path(new_root_dir)
  131. else
  132. vim.notify("get_typescript_server_path is not defined", vim.log.levels.ERROR)
  133. end
  134. end,
  135. },
  136. -- volar = {
  137. -- filetypes = {
  138. -- "javascript",
  139. -- "typescript",
  140. -- "vue",
  141. -- },
  142. -- on_new_config = function(new_config, new_root_dir)
  143. -- new_config.init_options.typescript.tsdk = get_typescript_server_path(new_root_dir)
  144. -- end,
  145. -- },
  146. cssls = {},
  147. intelephense = {
  148. root_dir = function(pattern)
  149. ---@diagnostic disable-next-line: undefined-field
  150. local cwd = vim.loop.cwd()
  151. local root = util.root_pattern("composer.json")(pattern)
  152. -- prefer cwd if root is a descendant
  153. return util.path.is_descendant(cwd, root) and cwd or root
  154. end,
  155. init_options = {
  156. licenceKey = vim.fn.expand("$HOME/.local/share/nvim/intelephense-licence.txt"),
  157. },
  158. settings = {
  159. intelephense = {
  160. format = {
  161. enable = true,
  162. sortUseStatements = false,
  163. },
  164. },
  165. },
  166. },
  167. gopls = {
  168. cmd = { "gopls" },
  169. filetypes = { "go", "gomod", "gowork", "gotmpl" },
  170. root_dir = util.root_pattern("go.work", "go.mod", ".git"),
  171. settings = {
  172. gopls = {
  173. completeUnimported = true,
  174. usePlaceholders = true,
  175. analyses = {
  176. unusedparams = true,
  177. },
  178. },
  179. },
  180. },
  181. lua_ls = {
  182. settings = { -- custom settings for lua
  183. Lua = {
  184. -- make the language server recognize "vim" global
  185. diagnostics = {
  186. globals = { "vim" },
  187. },
  188. workspace = {
  189. -- make language server aware of runtime files
  190. library = {
  191. [vim.fn.expand("$VIMRUNTIME/lua")] = true,
  192. [vim.fn.stdpath("config") .. "/lua"] = true,
  193. },
  194. },
  195. },
  196. },
  197. },
  198. dartls = {
  199. cmd = { "dart", "language-server", "--protocol=lsp" },
  200. },
  201. rust_analyzer = {
  202. diagnostics = {
  203. enable = false,
  204. },
  205. },
  206. pyright = {
  207. -- before_init = function(_, config)
  208. -- config.settings.python.pythonpath = get_python_path(config.root_dir)
  209. -- end
  210. },
  211. yamlls = {
  212. settings = {
  213. yaml = {
  214. keyOrdering = false,
  215. },
  216. },
  217. },
  218. }
  219. require("mason").setup()
  220. local ensure_installed = vim.tbl_keys(servers or {})
  221. vim.list_extend(ensure_installed, {
  222. "stylua",
  223. "prettier",
  224. "prettierd",
  225. "eslint",
  226. "eslint_d",
  227. "jsonlint",
  228. "markdownlint",
  229. "phpcbf",
  230. "phpcs",
  231. "golangci-lint",
  232. "hadolint",
  233. "gofumpt",
  234. "goimports",
  235. })
  236. require("mason-tool-installer").setup({
  237. ensure_installed = ensure_installed,
  238. run_on_start = false,
  239. })
  240. require("mason-lspconfig").setup()
  241. for server_name, server in pairs(servers) do
  242. server.capabilities = vim.tbl_deep_extend("force", {}, capabilities, server.capabilities or {})
  243. lspconfig[server_name].setup(server)
  244. end
  245. end,
  246. }