init.lua 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. -- file/buffer/etc picker
  2. require('telescope').setup({
  3. defaults = {
  4. mappings = {
  5. i = {
  6. ['jj'] = 'close',
  7. ['<C-j>'] = 'move_selection_next',
  8. ['<C-k>'] = 'move_selection_previous',
  9. },
  10. },
  11. layout_config = {
  12. prompt_position = 'top',
  13. },
  14. sorting_strategy = 'ascending',
  15. },
  16. })
  17. -- use native sorter for better performance
  18. require('telescope').load_extension('fzf')
  19. -- shows added/removed/changed lines
  20. require('gitsigns').setup()
  21. require('mini.statusline').setup({
  22. content = {
  23. -- copy-pasted from default, we just want to remove the icon
  24. active = function()
  25. local mode, mode_hl = MiniStatusline.section_mode({ trunc_width = 120 })
  26. local git = MiniStatusline.section_git({ trunc_width = 75, icon = '' })
  27. local diagnostics = MiniStatusline.section_diagnostics({ trunc_width = 75, icon = '' })
  28. local filename = MiniStatusline.section_filename({ trunc_width = 140 })
  29. local fileinfo = MiniStatusline.section_fileinfo({ trunc_width = 120 })
  30. local location = MiniStatusline.section_location({ trunc_width = 75 })
  31. return MiniStatusline.combine_groups({
  32. { hl = mode_hl, strings = { mode } },
  33. { hl = 'MiniStatuslineDevinfo', strings = { git, diagnostics } },
  34. '%<', -- Mark general truncate point
  35. { hl = 'MiniStatuslineFilename', strings = { filename } },
  36. '%=', -- End left alignment
  37. { hl = 'MiniStatuslineFileinfo', strings = { fileinfo } },
  38. { hl = mode_hl, strings = { location } },
  39. })
  40. end
  41. },
  42. })
  43. -- delete buffer while preserving layout
  44. require('mini.bufremove').setup()
  45. -- shows a line indicating the current indentation scope
  46. require('mini.indentscope').setup()
  47. -- LSP completion and function signature display
  48. require('mini.completion').setup({
  49. delay = {
  50. -- disable autocomplete
  51. completion = 100000000,
  52. info = 100,
  53. signature = 50,
  54. },
  55. lsp_completion = {
  56. source_func = 'omnifunc',
  57. auto_setup = false,
  58. },
  59. });
  60. -- Use Treesitter for syntax highlighting
  61. require('nvim-treesitter.configs').setup({
  62. highlight = {
  63. enable = true,
  64. },
  65. })
  66. -- ------ Mappings ------
  67. local opts = { noremap=true, silent=true }
  68. vim.api.nvim_set_keymap('n', '<space>e', '<cmd>lua vim.diagnostic.open_float()<CR>', opts)
  69. vim.api.nvim_set_keymap('n', '[d', '<cmd>lua vim.diagnostic.goto_prev()<CR>', opts)
  70. vim.api.nvim_set_keymap('n', ']d', '<cmd>lua vim.diagnostic.goto_next()<CR>', opts)
  71. vim.api.nvim_set_keymap('n', '<space>q', '<cmd>lua vim.diagnostic.setqflist({ severity = { min = vim.diagnostic.severity.WARN } })<CR>', opts)
  72. vim.api.nvim_set_keymap('n', '<Leader>q', '<cmd>lua MiniBufremove.delete()<CR>', opts)
  73. vim.api.nvim_set_keymap('n', '<C-P>', '<cmd>Telescope find_files<CR>', opts)
  74. vim.api.nvim_set_keymap('n', '<C-O>', '<cmd>Telescope buffers<CR>', opts)
  75. vim.api.nvim_set_keymap('n', '<Leader>ff', '<cmd>Telescope live_grep<CR>', opts)
  76. -- Allow pressing enter to autocomplete
  77. vim.api.nvim_set_keymap('i', '<CR>', 'pumvisible() ? "\\<C-y>" : "\\<CR>"', { noremap = true, expr = true })
  78. function on_attach(client, bufnr)
  79. -- Enable completion triggered by <c-x><c-o>
  80. vim.api.nvim_buf_set_option(bufnr, 'omnifunc', 'v:lua.MiniCompletion.completefunc_lsp')
  81. -- LSP-specific mappings
  82. vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gD', '<cmd>lua vim.lsp.buf.declaration()<CR>', opts)
  83. vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gd', '<cmd>lua vim.lsp.buf.definition()<CR>', opts)
  84. vim.api.nvim_buf_set_keymap(bufnr, 'n', 'K', '<cmd>lua vim.lsp.buf.hover()<CR>', opts)
  85. vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gi', '<cmd>lua vim.lsp.buf.implementation()<CR>', opts)
  86. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<C-k>', '<cmd>lua vim.lsp.buf.signature_help()<CR>', opts)
  87. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>wa', '<cmd>lua vim.lsp.buf.add_workspace_folder()<CR>', opts)
  88. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>wr', '<cmd>lua vim.lsp.buf.remove_workspace_folder()<CR>', opts)
  89. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>wl', '<cmd>lua print(vim.inspect(vim.lsp.buf.list_workspace_folders()))<CR>', opts)
  90. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>D', '<cmd>lua vim.lsp.buf.type_definition()<CR>', opts)
  91. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>rn', '<cmd>lua vim.lsp.buf.rename()<CR>', opts)
  92. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>ca', '<cmd>lua vim.lsp.buf.code_action()<CR>', opts)
  93. vim.api.nvim_buf_set_keymap(bufnr, 'n', 'gr', '<cmd>Telescope lsp_references<CR>', opts)
  94. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<space>f', '<cmd>lua vim.lsp.buf.formatting()<CR>', opts)
  95. vim.api.nvim_buf_set_keymap(bufnr, 'n', '<Leader>fs', '<cmd>Telescope lsp_document_symbols<CR>', opts)
  96. end
  97. -- ------ LSP settings ------
  98. -- only show virtual text for WARN and higher
  99. vim.diagnostic.config({
  100. virtual_text = { severity = { min = vim.diagnostic.severity.WARN } }
  101. })
  102. local nvim_lsp = require('lspconfig')
  103. local null_ls = require('null-ls')
  104. local null_ls_sources = {}
  105. -- enable LS / null-ls sources based on executable presence
  106. if vim.fn.executable("node_modules/.bin/eslint") == 1 then
  107. table.insert(null_ls_sources, null_ls.builtins.formatting.eslint_d)
  108. table.insert(null_ls_sources, null_ls.builtins.diagnostics.eslint_d)
  109. table.insert(null_ls_sources, null_ls.builtins.code_actions.eslint_d)
  110. end
  111. if vim.fn.executable("gofmt") == 1 then
  112. table.insert(null_ls_sources, null_ls.builtins.formatting.gofmt)
  113. end
  114. if vim.fn.executable("shellcheck") == 1 then
  115. table.insert(null_ls_sources, null_ls.builtins.diagnostics.shellcheck)
  116. table.insert(null_ls_sources, null_ls.builtins.code_actions.shellcheck)
  117. end
  118. if vim.fn.executable("deno") == 1 then
  119. nvim_lsp.denols.setup({
  120. on_attach = on_attach,
  121. });
  122. table.insert(null_ls_sources, null_ls.builtins.formatting.deno_fmt);
  123. end
  124. if vim.fn.executable("node_modules/.bin/tsc") == 1 then
  125. -- this provides auto-import on completion, among other things
  126. local ts_utils = require("nvim-lsp-ts-utils")
  127. nvim_lsp.tsserver.setup({
  128. init_options = ts_utils.init_options,
  129. flags = {
  130. debounce_text_changes = 150,
  131. },
  132. on_attach = function(client, bufnr)
  133. -- mark tsserver as not having formatting available as we rely on
  134. -- null-ls/eslint for that and having both available makes nvim ask us
  135. -- which LS to use everytime we format
  136. client.resolved_capabilities.document_formatting = false
  137. client.resolved_capabilities.document_range_formatting = false
  138. -- settings here are buffer-local so has to be run on_attach
  139. ts_utils.setup({
  140. debug = false,
  141. disable_commands = false,
  142. enable_import_on_completion = true,
  143. -- import all
  144. import_all_timeout = 5000, -- ms
  145. -- lower numbers = higher priority
  146. import_all_priorities = {
  147. same_file = 1, -- add to existing import statement
  148. local_files = 2, -- git files or files with relative path markers
  149. buffer_content = 3, -- loaded buffer content
  150. buffers = 4, -- loaded buffer names
  151. },
  152. import_all_scan_buffers = 100,
  153. import_all_select_source = false,
  154. -- filter diagnostics
  155. filter_out_diagnostics_by_severity = {},
  156. filter_out_diagnostics_by_code = {},
  157. -- inlay hints
  158. auto_inlay_hints = false,
  159. inlay_hints_highlight = "Comment",
  160. -- update imports on file move
  161. update_imports_on_move = false,
  162. require_confirmation_on_move = false,
  163. watch_dir = nil,
  164. })
  165. -- required to fix code action ranges and filter diagnostics
  166. ts_utils.setup_client(client)
  167. on_attach(client, bufnr)
  168. end,
  169. })
  170. end
  171. null_ls.setup({
  172. sources = null_ls_sources,
  173. on_attach = function(client, bufnr)
  174. -- format on save
  175. if client.resolved_capabilities.document_formatting then
  176. vim.cmd([[
  177. augroup LspFormatting
  178. autocmd! * <buffer>
  179. autocmd BufWritePre <buffer> lua vim.lsp.buf.formatting_sync()
  180. augroup END
  181. ]])
  182. end
  183. end,
  184. });