handlers.lua 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. -- Set Default Prefix.
  2. -- Note: You can set a prefix per lsp server in the lv-globals.lua file
  3. local M = {}
  4. function M.setup()
  5. local config = { -- your config
  6. virtual_text = lvim.lsp.diagnostics.virtual_text,
  7. signs = lvim.lsp.diagnostics.signs,
  8. underline = lvim.lsp.diagnostics.underline,
  9. update_in_insert = lvim.lsp.diagnostics.update_in_insert,
  10. severity_sort = lvim.lsp.diagnostics.severity_sort,
  11. }
  12. if vim.fn.has "nvim-0.5.1" > 0 then
  13. vim.lsp.handlers["textDocument/publishDiagnostics"] = function(_, result, ctx, _)
  14. local uri = result.uri
  15. local bufnr = vim.uri_to_bufnr(uri)
  16. if not bufnr then
  17. return
  18. end
  19. local diagnostics = result.diagnostics
  20. local ok, vim_diag = pcall(require, "vim.diagnostic")
  21. if ok then
  22. -- FIX: why can't we just use vim.diagnostic.get(buf_id)?
  23. config.signs = true
  24. for i, diagnostic in ipairs(diagnostics) do
  25. local rng = diagnostic.range
  26. diagnostics[i].lnum = rng["start"].line
  27. diagnostics[i].end_lnum = rng["end"].line
  28. diagnostics[i].col = rng["start"].character
  29. diagnostics[i].end_col = rng["end"].character
  30. end
  31. local namespace = vim.lsp.diagnostic.get_namespace(ctx.client_id)
  32. vim_diag.set(namespace, bufnr, diagnostics, config)
  33. if not vim.api.nvim_buf_is_loaded(bufnr) then
  34. return
  35. end
  36. local sign_names = {
  37. "DiagnosticSignError",
  38. "DiagnosticSignWarn",
  39. "DiagnosticSignInfo",
  40. "DiagnosticSignHint",
  41. }
  42. for i, sign in ipairs(lvim.lsp.diagnostics.signs.values) do
  43. vim.fn.sign_define(sign_names[i], { texthl = sign_names[i], text = sign.text, numhl = "" })
  44. end
  45. vim_diag.show(namespace, bufnr, diagnostics, config)
  46. else
  47. vim.lsp.diagnostic.save(diagnostics, bufnr, ctx.client_id)
  48. if not vim.api.nvim_buf_is_loaded(bufnr) then
  49. return
  50. end
  51. vim.lsp.diagnostic.display(diagnostics, bufnr, ctx.client_id, config)
  52. end
  53. end
  54. else
  55. vim.lsp.handlers["textDocument/publishDiagnostics"] = function(_, _, params, client_id, _)
  56. local uri = params.uri
  57. local bufnr = vim.uri_to_bufnr(uri)
  58. if not bufnr then
  59. return
  60. end
  61. local diagnostics = params.diagnostics
  62. vim.lsp.diagnostic.save(diagnostics, bufnr, client_id)
  63. if not vim.api.nvim_buf_is_loaded(bufnr) then
  64. return
  65. end
  66. vim.lsp.diagnostic.display(diagnostics, bufnr, client_id, config)
  67. end
  68. end
  69. vim.lsp.handlers["textDocument/hover"] = vim.lsp.with(vim.lsp.handlers.hover, {
  70. border = lvim.lsp.popup_border,
  71. })
  72. vim.lsp.handlers["textDocument/signatureHelp"] = vim.lsp.with(vim.lsp.handlers.signature_help, {
  73. border = lvim.lsp.popup_border,
  74. })
  75. end
  76. local function split_by_chunk(text, chunkSize)
  77. local s = {}
  78. for i = 1, #text, chunkSize do
  79. s[#s + 1] = text:sub(i, i + chunkSize - 1)
  80. end
  81. return s
  82. end
  83. function M.show_line_diagnostics()
  84. -- TODO: replace all this with vim.diagnostic.show_position_diagnostics()
  85. local diagnostics = vim.lsp.diagnostic.get_line_diagnostics()
  86. local severity_highlight = {
  87. "LspDiagnosticsFloatingError",
  88. "LspDiagnosticsFloatingWarning",
  89. "LspDiagnosticsFloatingInformation",
  90. "LspDiagnosticsFloatingHint",
  91. }
  92. local ok, vim_diag = pcall(require, "vim.diagnostic")
  93. if ok then
  94. local buf_id = vim.api.nvim_win_get_buf(0)
  95. local win_id = vim.api.nvim_get_current_win()
  96. local cursor_position = vim.api.nvim_win_get_cursor(win_id)
  97. severity_highlight = {
  98. "DiagnosticFloatingError",
  99. "DiagnosticFloatingWarn",
  100. "DiagnosticFloatingInfo",
  101. "DiagnosticFloatingHint",
  102. }
  103. diagnostics = vim_diag.get(buf_id, { lnum = cursor_position[1] - 1 })
  104. end
  105. local lines = {}
  106. local max_width = vim.fn.winwidth(0) - 5
  107. local height = #diagnostics
  108. local width = 0
  109. local opts = {}
  110. local close_events = { "CursorMoved", "CursorMovedI", "BufHidden", "InsertCharPre" }
  111. if height == 0 then
  112. return
  113. end
  114. local bufnr = vim.api.nvim_create_buf(false, true)
  115. local diag_message
  116. table.sort(diagnostics, function(a, b)
  117. return a.severity < b.severity
  118. end)
  119. local hash = {}
  120. local diagnostics_no_dupes = {}
  121. for _, v in ipairs(diagnostics) do
  122. if not hash[v["message"]] then
  123. diagnostics_no_dupes[#diagnostics_no_dupes + 1] = v -- you could print here instead of saving to result table if you wanted
  124. hash[v["message"]] = true
  125. end
  126. end
  127. -- print(vim.inspect(diagnostics_no_dupes))
  128. for i, diagnostic in ipairs(diagnostics_no_dupes) do
  129. local source = diagnostic.source
  130. diag_message = diagnostic.message:gsub("[\n\r]", " ")
  131. if source then
  132. if string.find(source, "/") then
  133. source = string.sub(diagnostic.source, string.find(diagnostic.source, "([%w-_]+)$"))
  134. end
  135. diag_message = string.format("%d. %s: %s", i, source, diag_message)
  136. else
  137. diag_message = string.format("%d. %s", i, diag_message)
  138. end
  139. if diagnostic.code then
  140. diag_message = string.format("%s [%s]", diag_message, diagnostic.code)
  141. end
  142. local msgs = split_by_chunk(diag_message, max_width)
  143. for _, diag in ipairs(msgs) do
  144. table.insert(lines, { message = diag, severity = diagnostic.severity })
  145. width = math.max(diag:len(), width)
  146. end
  147. end
  148. height = #lines
  149. opts = vim.lsp.util.make_floating_popup_options(width, height, opts)
  150. opts["style"] = "minimal"
  151. opts["border"] = "rounded"
  152. opts["focusable"] = true
  153. vim.api.nvim_buf_set_option(bufnr, "bufhidden", "wipe")
  154. local winnr = vim.api.nvim_open_win(bufnr, false, opts)
  155. vim.api.nvim_win_set_option(winnr, "winblend", 0)
  156. vim.api.nvim_buf_set_var(bufnr, "lsp_floating_window", winnr)
  157. for i, diag in ipairs(lines) do
  158. vim.api.nvim_buf_set_lines(bufnr, i - 1, i - 1, 0, { diag.message })
  159. vim.api.nvim_buf_add_highlight(bufnr, -1, severity_highlight[diag.severity], i - 1, 0, diag.message:len())
  160. end
  161. vim.api.nvim_command(
  162. "autocmd QuitPre <buffer> ++nested ++once lua pcall(vim.api.nvim_win_close, " .. winnr .. ", true)"
  163. )
  164. vim.lsp.util.close_preview_autocmd(close_events, winnr)
  165. end
  166. return M