keymappings.lua 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. local M = {}
  2. local Log = require "lvim.core.log"
  3. local generic_opts_any = { noremap = true, silent = true }
  4. local generic_opts = {
  5. insert_mode = generic_opts_any,
  6. normal_mode = generic_opts_any,
  7. visual_mode = generic_opts_any,
  8. visual_block_mode = generic_opts_any,
  9. command_mode = generic_opts_any,
  10. term_mode = { silent = true },
  11. }
  12. local mode_adapters = {
  13. insert_mode = "i",
  14. normal_mode = "n",
  15. term_mode = "t",
  16. visual_mode = "v",
  17. visual_block_mode = "x",
  18. command_mode = "c",
  19. }
  20. -- Append key mappings to lunarvim's defaults for a given mode
  21. -- @param keymaps The table of key mappings containing a list per mode (normal_mode, insert_mode, ..)
  22. function M.append_to_defaults(keymaps)
  23. local default = M.get_defaults()
  24. lvim.keys = lvim.keys or default
  25. for mode, mappings in pairs(keymaps) do
  26. lvim.keys[mode] = lvim.keys[mode] or default[mode]
  27. for k, v in pairs(mappings) do
  28. lvim.keys[mode][k] = v
  29. end
  30. end
  31. end
  32. -- Unsets all keybindings defined in keymaps
  33. -- @param keymaps The table of key mappings containing a list per mode (normal_mode, insert_mode, ..)
  34. function M.clear(keymaps)
  35. local default = M.get_defaults()
  36. for mode, mappings in pairs(keymaps) do
  37. local translated_mode = mode_adapters[mode] and mode_adapters[mode] or mode
  38. for key, _ in pairs(mappings) do
  39. -- some plugins may override default bindings that the user hasn't manually overriden
  40. if default[mode][key] ~= nil or (default[translated_mode] ~= nil and default[translated_mode][key] ~= nil) then
  41. pcall(vim.api.nvim_del_keymap, translated_mode, key)
  42. end
  43. end
  44. end
  45. end
  46. -- Set key mappings individually
  47. -- @param mode The keymap mode, can be one of the keys of mode_adapters
  48. -- @param key The key of keymap
  49. -- @param val Can be form as a mapping or tuple of mapping and user defined opt
  50. function M.set_keymaps(mode, key, val)
  51. local opt = generic_opts[mode] and generic_opts[mode] or generic_opts_any
  52. if type(val) == "table" then
  53. opt = val[2]
  54. val = val[1]
  55. end
  56. if val then
  57. vim.api.nvim_set_keymap(mode, key, val, opt)
  58. else
  59. pcall(vim.api.nvim_del_keymap, mode, key)
  60. end
  61. end
  62. -- Load key mappings for a given mode
  63. -- @param mode The keymap mode, can be one of the keys of mode_adapters
  64. -- @param keymaps The list of key mappings
  65. function M.load_mode(mode, keymaps)
  66. mode = mode_adapters[mode] and mode_adapters[mode] or mode
  67. for k, v in pairs(keymaps) do
  68. M.set_keymaps(mode, k, v)
  69. end
  70. end
  71. -- Load key mappings for all provided modes
  72. -- @param keymaps A list of key mappings for each mode
  73. function M.load(keymaps)
  74. keymaps = keymaps or {}
  75. for mode, mapping in pairs(keymaps) do
  76. M.load_mode(mode, mapping)
  77. end
  78. end
  79. function M.get_defaults()
  80. local keys = {
  81. ---@usage change or add keymappings for insert mode
  82. insert_mode = {
  83. -- 'jk' for quitting insert mode
  84. ["jk"] = "<ESC>",
  85. -- 'kj' for quitting insert mode
  86. ["kj"] = "<ESC>",
  87. -- 'jj' for quitting insert mode
  88. ["jj"] = "<ESC>",
  89. -- Move current line / block with Alt-j/k ala vscode.
  90. ["<A-j>"] = "<Esc>:m .+1<CR>==gi",
  91. -- Move current line / block with Alt-j/k ala vscode.
  92. ["<A-k>"] = "<Esc>:m .-2<CR>==gi",
  93. -- navigation
  94. ["<A-Up>"] = "<C-\\><C-N><C-w>k",
  95. ["<A-Down>"] = "<C-\\><C-N><C-w>j",
  96. ["<A-Left>"] = "<C-\\><C-N><C-w>h",
  97. ["<A-Right>"] = "<C-\\><C-N><C-w>l",
  98. },
  99. ---@usage change or add keymappings for normal mode
  100. normal_mode = {
  101. -- Better window movement
  102. ["<C-h>"] = "<C-w>h",
  103. ["<C-j>"] = "<C-w>j",
  104. ["<C-k>"] = "<C-w>k",
  105. ["<C-l>"] = "<C-w>l",
  106. -- Resize with arrows
  107. ["<C-Up>"] = ":resize -2<CR>",
  108. ["<C-Down>"] = ":resize +2<CR>",
  109. ["<C-Left>"] = ":vertical resize -2<CR>",
  110. ["<C-Right>"] = ":vertical resize +2<CR>",
  111. -- Tab switch buffer
  112. ["<S-l>"] = ":BufferNext<CR>",
  113. ["<S-h>"] = ":BufferPrevious<CR>",
  114. -- Move current line / block with Alt-j/k a la vscode.
  115. ["<A-j>"] = ":m .+1<CR>==",
  116. ["<A-k>"] = ":m .-2<CR>==",
  117. -- QuickFix
  118. ["]q"] = ":cnext<CR>",
  119. ["[q"] = ":cprev<CR>",
  120. ["<C-q>"] = ":call QuickFixToggle()<CR>",
  121. },
  122. ---@usage change or add keymappings for terminal mode
  123. term_mode = {
  124. -- Terminal window navigation
  125. ["<C-h>"] = "<C-\\><C-N><C-w>h",
  126. ["<C-j>"] = "<C-\\><C-N><C-w>j",
  127. ["<C-k>"] = "<C-\\><C-N><C-w>k",
  128. ["<C-l>"] = "<C-\\><C-N><C-w>l",
  129. },
  130. ---@usage change or add keymappings for visual mode
  131. visual_mode = {
  132. -- Better indenting
  133. ["<"] = "<gv",
  134. [">"] = ">gv",
  135. -- ["p"] = '"0p',
  136. -- ["P"] = '"0P',
  137. },
  138. ---@usage change or add keymappings for visual block mode
  139. visual_block_mode = {
  140. -- Move selected line / block of text in visual mode
  141. ["K"] = ":move '<-2<CR>gv-gv",
  142. ["J"] = ":move '>+1<CR>gv-gv",
  143. -- Move current line / block with Alt-j/k ala vscode.
  144. ["<A-j>"] = ":m '>+1<CR>gv-gv",
  145. ["<A-k>"] = ":m '<-2<CR>gv-gv",
  146. },
  147. ---@usage change or add keymappings for command mode
  148. command_mode = {
  149. -- navigate tab completion with <c-j> and <c-k>
  150. -- runs conditionally
  151. ["<C-j>"] = { 'pumvisible() ? "\\<C-n>" : "\\<C-j>"', { expr = true, noremap = true } },
  152. ["<C-k>"] = { 'pumvisible() ? "\\<C-p>" : "\\<C-k>"', { expr = true, noremap = true } },
  153. },
  154. }
  155. if vim.fn.has "mac" == 1 then
  156. keys.normal_mode["<A-Up>"] = keys.normal_mode["<C-Up>"]
  157. keys.normal_mode["<A-Down>"] = keys.normal_mode["<C-Down>"]
  158. keys.normal_mode["<A-Left>"] = keys.normal_mode["<C-Left>"]
  159. keys.normal_mode["<A-Right>"] = keys.normal_mode["<C-Right>"]
  160. Log:debug "Activated mac keymappings"
  161. end
  162. return keys
  163. end
  164. function M.print(mode)
  165. print "List of LunarVim's default keymappings (not including which-key)"
  166. if mode then
  167. print(vim.inspect(lvim.keys[mode]))
  168. else
  169. print(vim.inspect(lvim.keys))
  170. end
  171. end
  172. return M