Rewrite keymap.lua (#452)
* Move feedkeys related codes to `cmp.utils.feedkeys` * Improve mapping handling * Fix test * fmt
This commit is contained in:
@@ -1,6 +1,7 @@
|
|||||||
local debug = require('cmp.utils.debug')
|
local debug = require('cmp.utils.debug')
|
||||||
local char = require('cmp.utils.char')
|
local char = require('cmp.utils.char')
|
||||||
local pattern = require('cmp.utils.pattern')
|
local pattern = require('cmp.utils.pattern')
|
||||||
|
local feedkeys = require('cmp.utils.feedkeys')
|
||||||
local async = require('cmp.utils.async')
|
local async = require('cmp.utils.async')
|
||||||
local keymap = require('cmp.utils.keymap')
|
local keymap = require('cmp.utils.keymap')
|
||||||
local context = require('cmp.context')
|
local context = require('cmp.context')
|
||||||
@@ -214,9 +215,9 @@ core.autoindent = function(self, trigger_event, callback)
|
|||||||
local release = self:suspend()
|
local release = self:suspend()
|
||||||
vim.schedule(function()
|
vim.schedule(function()
|
||||||
if cursor_before_line == api.get_cursor_before_line() then
|
if cursor_before_line == api.get_cursor_before_line() then
|
||||||
keymap.feedkeys(keymap.t('<Cmd>setlocal cindent<CR>'), 'n')
|
feedkeys.call(keymap.t('<Cmd>setlocal cindent<CR>'), 'n')
|
||||||
keymap.feedkeys(keymap.t('<C-f>'), 'n')
|
feedkeys.call(keymap.t('<C-f>'), 'n')
|
||||||
keymap.feedkeys(keymap.t('<Cmd>setlocal %scindent<CR>'):format(vim.bo.cindent and '' or 'no'), 'n', function()
|
feedkeys.call(keymap.t('<Cmd>setlocal %scindent<CR>'):format(vim.bo.cindent and '' or 'no'), 'n', function()
|
||||||
release()
|
release()
|
||||||
callback()
|
callback()
|
||||||
end)
|
end)
|
||||||
@@ -318,12 +319,12 @@ core.confirm = function(self, e, option, callback)
|
|||||||
local confirm = {}
|
local confirm = {}
|
||||||
table.insert(confirm, keymap.backspace(ctx.cursor.character - misc.to_utfindex(e.context.cursor_before_line, e:get_offset())))
|
table.insert(confirm, keymap.backspace(ctx.cursor.character - misc.to_utfindex(e.context.cursor_before_line, e:get_offset())))
|
||||||
table.insert(confirm, e:get_word())
|
table.insert(confirm, e:get_word())
|
||||||
keymap.feedkeys(table.concat(confirm, ''), 'nt', function()
|
feedkeys.call(table.concat(confirm, ''), 'nt', function()
|
||||||
-- Restore to the requested state.
|
-- Restore to the requested state.
|
||||||
local restore = {}
|
local restore = {}
|
||||||
table.insert(restore, keymap.backspace(vim.str_utfindex(e:get_word())))
|
table.insert(restore, keymap.backspace(vim.str_utfindex(e:get_word())))
|
||||||
table.insert(restore, string.sub(e.context.cursor_before_line, e:get_offset()))
|
table.insert(restore, string.sub(e.context.cursor_before_line, e:get_offset()))
|
||||||
keymap.feedkeys(table.concat(restore, ''), 'n', function()
|
feedkeys.call(table.concat(restore, ''), 'n', function()
|
||||||
--@see https://github.com/microsoft/vscode/blob/main/src/vs/editor/contrib/suggest/suggestController.ts#L334
|
--@see https://github.com/microsoft/vscode/blob/main/src/vs/editor/contrib/suggest/suggestController.ts#L334
|
||||||
if #(misc.safe(e:get_completion_item().additionalTextEdits) or {}) == 0 then
|
if #(misc.safe(e:get_completion_item().additionalTextEdits) or {}) == 0 then
|
||||||
local pre = context.new()
|
local pre = context.new()
|
||||||
@@ -384,7 +385,7 @@ core.confirm = function(self, e, option, callback)
|
|||||||
table.insert(keys, completion_item.textEdit.newText)
|
table.insert(keys, completion_item.textEdit.newText)
|
||||||
end
|
end
|
||||||
|
|
||||||
keymap.feedkeys(table.concat(keys, ''), 'n', function()
|
feedkeys.call(table.concat(keys, ''), 'n', function()
|
||||||
if is_snippet then
|
if is_snippet then
|
||||||
-- remove snippet prefix without changing `dot` register.
|
-- remove snippet prefix without changing `dot` register.
|
||||||
local snippet_ctx = context.new()
|
local snippet_ctx = context.new()
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
local core = require('cmp.core')
|
local core = require('cmp.core')
|
||||||
local source = require('cmp.source')
|
local source = require('cmp.source')
|
||||||
local config = require('cmp.config')
|
local config = require('cmp.config')
|
||||||
|
local feedkeys = require('cmp.utils.feedkeys')
|
||||||
local autocmd = require('cmp.utils.autocmd')
|
local autocmd = require('cmp.utils.autocmd')
|
||||||
local keymap = require('cmp.utils.keymap')
|
local keymap = require('cmp.utils.keymap')
|
||||||
local misc = require('cmp.utils.misc')
|
local misc = require('cmp.utils.misc')
|
||||||
@@ -117,9 +118,9 @@ cmp.select_next_item = function(option)
|
|||||||
return true
|
return true
|
||||||
elseif vim.fn.pumvisible() == 1 then
|
elseif vim.fn.pumvisible() == 1 then
|
||||||
if (option.behavior or cmp.SelectBehavior.Insert) == cmp.SelectBehavior.Insert then
|
if (option.behavior or cmp.SelectBehavior.Insert) == cmp.SelectBehavior.Insert then
|
||||||
keymap.feedkeys(keymap.t('<C-n>'), 'n')
|
feedkeys.call(keymap.t('<C-n>'), 'n')
|
||||||
else
|
else
|
||||||
keymap.feedkeys(keymap.t('<Down>'), 'n')
|
feedkeys.call(keymap.t('<Down>'), 'n')
|
||||||
end
|
end
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
@@ -142,9 +143,9 @@ cmp.select_prev_item = function(option)
|
|||||||
return true
|
return true
|
||||||
elseif vim.fn.pumvisible() == 1 then
|
elseif vim.fn.pumvisible() == 1 then
|
||||||
if (option.behavior or cmp.SelectBehavior.Insert) == cmp.SelectBehavior.Insert then
|
if (option.behavior or cmp.SelectBehavior.Insert) == cmp.SelectBehavior.Insert then
|
||||||
keymap.feedkeys(keymap.t('<C-p>'), 'n')
|
feedkeys.call(keymap.t('<C-p>'), 'n')
|
||||||
else
|
else
|
||||||
keymap.feedkeys(keymap.t('<Up>'), 'n')
|
feedkeys.call(keymap.t('<Up>'), 'n')
|
||||||
end
|
end
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
@@ -182,7 +183,7 @@ cmp.confirm = function(option, callback)
|
|||||||
return true
|
return true
|
||||||
else
|
else
|
||||||
if vim.fn.complete_info({ 'selected' }).selected ~= -1 then
|
if vim.fn.complete_info({ 'selected' }).selected ~= -1 then
|
||||||
keymap.feedkeys(keymap.t('<C-y>'), 'n')
|
feedkeys.call(keymap.t('<C-y>'), 'n')
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
return false
|
return false
|
||||||
|
|||||||
106
lua/cmp/utils/feedkeys.lua
Normal file
106
lua/cmp/utils/feedkeys.lua
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
local keymap = require('cmp.utils.keymap')
|
||||||
|
local misc = require('cmp.utils.misc')
|
||||||
|
|
||||||
|
local feedkeys = {}
|
||||||
|
|
||||||
|
feedkeys.call = setmetatable({
|
||||||
|
callbacks = {},
|
||||||
|
}, {
|
||||||
|
__call = function(self, keys, mode, callback)
|
||||||
|
if vim.fn.reg_recording() ~= '' then
|
||||||
|
return feedkeys.call_macro(keys, mode, callback)
|
||||||
|
end
|
||||||
|
|
||||||
|
local is_insert = string.match(mode, 'i') ~= nil
|
||||||
|
|
||||||
|
local queue = {}
|
||||||
|
if #keys > 0 then
|
||||||
|
table.insert(queue, { keymap.t('<Cmd>set backspace=start<CR>'), 'n' })
|
||||||
|
table.insert(queue, { keymap.t('<Cmd>set eventignore=all<CR>'), 'n' })
|
||||||
|
table.insert(queue, { keys, string.gsub(mode, '[it]', ''), true })
|
||||||
|
table.insert(queue, { keymap.t('<Cmd>set backspace=%s<CR>'):format(vim.o.backspace or ''), 'n' })
|
||||||
|
table.insert(queue, { keymap.t('<Cmd>set eventignore=%s<CR>'):format(vim.o.eventignore or ''), 'n' })
|
||||||
|
end
|
||||||
|
if #keys > 0 or callback then
|
||||||
|
local id = misc.id('cmp.utils.feedkeys.call')
|
||||||
|
self.callbacks[id] = function()
|
||||||
|
if callback then
|
||||||
|
callback()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
table.insert(queue, { keymap.t('<Cmd>call v:lua.cmp.utils.feedkeys.call.run(%s)<CR>'):format(id), 'n', true })
|
||||||
|
end
|
||||||
|
|
||||||
|
if is_insert then
|
||||||
|
for i = #queue, 1, -1 do
|
||||||
|
vim.api.nvim_feedkeys(queue[i][1], queue[i][2] .. 'i', queue[i][3])
|
||||||
|
end
|
||||||
|
else
|
||||||
|
for i = 1, #queue do
|
||||||
|
vim.api.nvim_feedkeys(queue[i][1], queue[i][2], queue[i][3])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
misc.set(_G, { 'cmp', 'utils', 'feedkeys', 'call', 'run' }, function(id)
|
||||||
|
if feedkeys.call.callbacks[id] then
|
||||||
|
feedkeys.call.callbacks[id]()
|
||||||
|
feedkeys.call.callbacks[id] = nil
|
||||||
|
end
|
||||||
|
return ''
|
||||||
|
end)
|
||||||
|
|
||||||
|
feedkeys.call_macro = setmetatable({
|
||||||
|
queue = {},
|
||||||
|
current = nil,
|
||||||
|
timer = vim.loop.new_timer(),
|
||||||
|
running = false,
|
||||||
|
}, {
|
||||||
|
__call = function(self, keys, mode, callback)
|
||||||
|
local is_insert = string.match(mode, 'i') ~= nil
|
||||||
|
table.insert(self.queue, is_insert and 1 or #self.queue + 1, {
|
||||||
|
keys = keys,
|
||||||
|
mode = mode,
|
||||||
|
callback = callback,
|
||||||
|
})
|
||||||
|
|
||||||
|
if not self.running then
|
||||||
|
self.running = true
|
||||||
|
local consume
|
||||||
|
consume = vim.schedule_wrap(function()
|
||||||
|
if vim.fn.getchar(1) == 0 then
|
||||||
|
if self.current then
|
||||||
|
vim.cmd(('set backspace=%s'):format(self.current.backspace or ''))
|
||||||
|
vim.cmd(('set eventignore=%s'):format(self.current.eventignore or ''))
|
||||||
|
if self.current.callback then
|
||||||
|
self.current.callback()
|
||||||
|
end
|
||||||
|
self.current = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
local current = table.remove(self.queue, 1)
|
||||||
|
if current then
|
||||||
|
self.current = {
|
||||||
|
keys = current.keys,
|
||||||
|
callback = current.callback,
|
||||||
|
backspace = vim.o.backspace,
|
||||||
|
eventignore = vim.o.eventignore,
|
||||||
|
}
|
||||||
|
vim.api.nvim_feedkeys(keymap.t('<Cmd>set backspace=start<CR>'), 'n', true)
|
||||||
|
vim.api.nvim_feedkeys(keymap.t('<Cmd>set eventignore=all<CR>'), 'n', true)
|
||||||
|
vim.api.nvim_feedkeys(current.keys, string.gsub(current.mode, '[i]', ''), true) -- 'i' flag is manually resolved.
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if #self.queue ~= 0 or self.current then
|
||||||
|
vim.defer_fn(consume, 1)
|
||||||
|
else
|
||||||
|
self.running = false
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
vim.defer_fn(consume, 1)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
|
||||||
|
return feedkeys
|
||||||
23
lua/cmp/utils/feedkeys_spec.lua
Normal file
23
lua/cmp/utils/feedkeys_spec.lua
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
local spec = require('cmp.utils.spec')
|
||||||
|
local keymap = require('cmp.utils.keymap')
|
||||||
|
|
||||||
|
local feedkeys = require('cmp.utils.feedkeys')
|
||||||
|
|
||||||
|
describe('feedkeys', function()
|
||||||
|
before_each(spec.before)
|
||||||
|
|
||||||
|
it('dot-repeat', function()
|
||||||
|
feedkeys.call(keymap.t('iaiueo<Esc>'), 'nx')
|
||||||
|
assert.are.equal(vim.fn.getreg('.'), keymap.t('aiueo'))
|
||||||
|
end)
|
||||||
|
it('macro', function()
|
||||||
|
vim.fn.setreg('q', '')
|
||||||
|
vim.cmd([[normal! qq]])
|
||||||
|
feedkeys.call(keymap.t('iaiueo'), 'nt')
|
||||||
|
feedkeys.call(keymap.t('<Esc>'), 'nt', function()
|
||||||
|
vim.cmd([[normal! q]])
|
||||||
|
assert.are.equal(vim.fn.getreg('q'), keymap.t('iaiueo<Esc>'))
|
||||||
|
print(vim.fn.getreg('q'))
|
||||||
|
end)
|
||||||
|
end)
|
||||||
|
end)
|
||||||
@@ -98,154 +98,33 @@ keymap.equals = function(a, b)
|
|||||||
return keymap.t(a) == keymap.t(b)
|
return keymap.t(a) == keymap.t(b)
|
||||||
end
|
end
|
||||||
|
|
||||||
---Feedkeys with callback
|
|
||||||
---@param keys string
|
|
||||||
---@param mode string
|
|
||||||
---@param callback function
|
|
||||||
keymap.feedkeys = setmetatable({
|
|
||||||
callbacks = {},
|
|
||||||
}, {
|
|
||||||
__call = function(self, keys, mode, callback)
|
|
||||||
if vim.fn.reg_recording() ~= '' then
|
|
||||||
return keymap.feedkeys_macro_safe(keys, mode, callback)
|
|
||||||
end
|
|
||||||
|
|
||||||
local is_insert = string.match(mode, 'i') ~= nil
|
|
||||||
|
|
||||||
local queue = {}
|
|
||||||
if #keys > 0 then
|
|
||||||
table.insert(queue, { keymap.t('<Cmd>set backspace=start<CR>'), 'n' })
|
|
||||||
table.insert(queue, { keymap.t('<Cmd>set eventignore=all<CR>'), 'n' })
|
|
||||||
table.insert(queue, { keys, string.gsub(mode, '[it]', ''), true })
|
|
||||||
table.insert(queue, { keymap.t('<Cmd>set backspace=%s<CR>'):format(vim.o.backspace or ''), 'n' })
|
|
||||||
table.insert(queue, { keymap.t('<Cmd>set eventignore=%s<CR>'):format(vim.o.eventignore or ''), 'n' })
|
|
||||||
end
|
|
||||||
if #keys > 0 or callback then
|
|
||||||
local id = misc.id('cmp.utils.keymap.feedkeys')
|
|
||||||
self.callbacks[id] = function()
|
|
||||||
if callback then
|
|
||||||
callback()
|
|
||||||
end
|
|
||||||
end
|
|
||||||
table.insert(queue, { keymap.t('<Cmd>call v:lua.cmp.utils.keymap.feedkeys.run(%s)<CR>'):format(id), 'n', true })
|
|
||||||
end
|
|
||||||
|
|
||||||
if is_insert then
|
|
||||||
for i = #queue, 1, -1 do
|
|
||||||
vim.api.nvim_feedkeys(queue[i][1], queue[i][2] .. 'i', queue[i][3])
|
|
||||||
end
|
|
||||||
else
|
|
||||||
for i = 1, #queue do
|
|
||||||
vim.api.nvim_feedkeys(queue[i][1], queue[i][2], queue[i][3])
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end,
|
|
||||||
})
|
|
||||||
misc.set(_G, { 'cmp', 'utils', 'keymap', 'feedkeys', 'run' }, function(id)
|
|
||||||
if keymap.feedkeys.callbacks[id] then
|
|
||||||
keymap.feedkeys.callbacks[id]()
|
|
||||||
keymap.feedkeys.callbacks[id] = nil
|
|
||||||
end
|
|
||||||
return ''
|
|
||||||
end)
|
|
||||||
|
|
||||||
---Macro safe feedkeys.
|
|
||||||
---@param keys string
|
|
||||||
---@param mode string
|
|
||||||
---@param callback function
|
|
||||||
keymap.feedkeys_macro_safe = setmetatable({
|
|
||||||
queue = {},
|
|
||||||
current = nil,
|
|
||||||
timer = vim.loop.new_timer(),
|
|
||||||
running = false,
|
|
||||||
}, {
|
|
||||||
__call = function(self, keys, mode, callback)
|
|
||||||
local is_insert = string.match(mode, 'i') ~= nil
|
|
||||||
table.insert(self.queue, is_insert and 1 or #self.queue + 1, {
|
|
||||||
keys = keys,
|
|
||||||
mode = mode,
|
|
||||||
callback = callback,
|
|
||||||
})
|
|
||||||
|
|
||||||
if not self.running then
|
|
||||||
self.running = true
|
|
||||||
local consume
|
|
||||||
consume = vim.schedule_wrap(function()
|
|
||||||
if vim.fn.getchar(1) == 0 then
|
|
||||||
if self.current then
|
|
||||||
vim.cmd(('set backspace=%s'):format(self.current.backspace or ''))
|
|
||||||
vim.cmd(('set eventignore=%s'):format(self.current.eventignore or ''))
|
|
||||||
if self.current.callback then
|
|
||||||
self.current.callback()
|
|
||||||
end
|
|
||||||
self.current = nil
|
|
||||||
end
|
|
||||||
|
|
||||||
local current = table.remove(self.queue, 1)
|
|
||||||
if current then
|
|
||||||
self.current = {
|
|
||||||
keys = current.keys,
|
|
||||||
callback = current.callback,
|
|
||||||
backspace = vim.o.backspace,
|
|
||||||
eventignore = vim.o.eventignore,
|
|
||||||
}
|
|
||||||
vim.api.nvim_feedkeys(keymap.t('<Cmd>set backspace=start<CR>'), 'n', true)
|
|
||||||
vim.api.nvim_feedkeys(keymap.t('<Cmd>set eventignore=all<CR>'), 'n', true)
|
|
||||||
vim.api.nvim_feedkeys(current.keys, string.gsub(current.mode, '[i]', ''), true) -- 'i' flag is manually resolved.
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
if #self.queue ~= 0 or self.current then
|
|
||||||
vim.defer_fn(consume, 1)
|
|
||||||
else
|
|
||||||
self.running = false
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
vim.defer_fn(consume, 1)
|
|
||||||
end
|
|
||||||
end,
|
|
||||||
})
|
|
||||||
|
|
||||||
---Register keypress handler.
|
---Register keypress handler.
|
||||||
keymap.listen = setmetatable({
|
keymap.listen = setmetatable({
|
||||||
cache = cache.new(),
|
cache = cache.new(),
|
||||||
}, {
|
}, {
|
||||||
__call = function(self, mode, keys_or_chars, callback)
|
__call = function(self, mode, keys_or_chars, callback)
|
||||||
local keys = keymap.normalize(keymap.to_keymap(keys_or_chars))
|
local keys = keymap.normalize(keymap.to_keymap(keys_or_chars))
|
||||||
local bufnr = vim.api.nvim_get_current_buf()
|
local existing = keymap.get_mapping(mode, keys)
|
||||||
local existing = keymap.find_map_by_lhs(mode, keys)
|
if not existing then
|
||||||
|
return
|
||||||
local cur_definition = self.cache:get({ 'definition', self.cache:get({ 'id', mode, bufnr, keys }) or '' })
|
|
||||||
if cur_definition then
|
|
||||||
local same = true
|
|
||||||
same = same and existing
|
|
||||||
same = same and cur_definition.existing.lhs == existing.lhs
|
|
||||||
same = same and cur_definition.existing.rhs == existing.rhs
|
|
||||||
same = same and cur_definition.existing.expr == existing.expr
|
|
||||||
same = same and cur_definition.existing.noremap == existing.noremap
|
|
||||||
same = same and cur_definition.existing.script == existing.script
|
|
||||||
if not existing or same then
|
|
||||||
return
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
local bufnr = existing.buffer and vim.api.nvim_get_current_buf() or '*'
|
||||||
self.cache:set({ 'id', mode, bufnr, keys }, misc.id('cmp.utils.keymap.listen'))
|
self.cache:set({ 'id', mode, bufnr, keys }, misc.id('cmp.utils.keymap.listen'))
|
||||||
|
|
||||||
existing = existing or {
|
|
||||||
lhs = keys,
|
|
||||||
rhs = keys,
|
|
||||||
expr = 0,
|
|
||||||
script = 0,
|
|
||||||
noremap = 1,
|
|
||||||
nowait = 0,
|
|
||||||
silent = 1,
|
|
||||||
}
|
|
||||||
|
|
||||||
local fallback = keymap.evacuate(mode, keys)
|
local fallback = keymap.evacuate(mode, keys)
|
||||||
vim.api.nvim_buf_set_keymap(0, mode, keys, ('<Cmd>call v:lua.cmp.utils.keymap.listen.run(%s)<CR>'):format(self.cache:get({ 'id', mode, bufnr, keys })), {
|
if existing.buffer then
|
||||||
expr = false,
|
vim.api.nvim_buf_set_keymap(0, mode, keys, ('<Cmd>call v:lua.cmp.utils.keymap.listen.run(%s)<CR>'):format(self.cache:get({ 'id', mode, bufnr, keys })), {
|
||||||
noremap = true,
|
expr = false,
|
||||||
silent = true,
|
noremap = true,
|
||||||
})
|
silent = true,
|
||||||
|
})
|
||||||
|
else
|
||||||
|
vim.api.nvim_set_keymap(mode, keys, ('<Cmd>call v:lua.cmp.utils.keymap.listen.run(%s)<CR>'):format(self.cache:get({ 'id', mode, bufnr, keys })), {
|
||||||
|
expr = false,
|
||||||
|
noremap = true,
|
||||||
|
silent = true,
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
self.cache:set({ 'definition', self.cache:get({ 'id', mode, bufnr, keys }) }, {
|
self.cache:set({ 'definition', self.cache:get({ 'id', mode, bufnr, keys }) }, {
|
||||||
keys = keys,
|
keys = keys,
|
||||||
@@ -271,43 +150,97 @@ misc.set(_G, { 'cmp', 'utils', 'keymap', 'listen', 'run' }, function(id)
|
|||||||
return keymap.t('<Ignore>')
|
return keymap.t('<Ignore>')
|
||||||
end)
|
end)
|
||||||
|
|
||||||
|
---Get mapping
|
||||||
|
---@param mode string
|
||||||
|
---@param lhs string
|
||||||
|
---@return table
|
||||||
|
keymap.get_mapping = function(mode, lhs)
|
||||||
|
lhs = keymap.normalize(lhs)
|
||||||
|
|
||||||
|
for _, map in ipairs(vim.api.nvim_buf_get_keymap(0, mode)) do
|
||||||
|
if keymap.equals(map.lhs, lhs) then
|
||||||
|
if string.match(map.rhs, vim.pesc('v:lua.cmp.utils.keymap.listen.run')) then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
return {
|
||||||
|
lhs = map.lhs,
|
||||||
|
rhs = map.rhs,
|
||||||
|
expr = map.expr == 1,
|
||||||
|
noremap = map.noremap == 1,
|
||||||
|
script = map.script == 1,
|
||||||
|
silent = map.silent == 1,
|
||||||
|
nowait = map.nowait == 1,
|
||||||
|
buffer = true,
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
for _, map in ipairs(vim.api.nvim_get_keymap(mode)) do
|
||||||
|
if keymap.equals(map.lhs, lhs) then
|
||||||
|
if string.match(map.rhs, vim.pesc('v:lua.cmp.utils.keymap.listen.run')) then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
return {
|
||||||
|
lhs = map.lhs,
|
||||||
|
rhs = map.rhs,
|
||||||
|
expr = map.expr == 1,
|
||||||
|
noremap = map.noremap == 1,
|
||||||
|
script = map.script == 1,
|
||||||
|
silent = map.silent == 1,
|
||||||
|
nowait = map.nowait == 1,
|
||||||
|
buffer = false,
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return {
|
||||||
|
lhs = lhs,
|
||||||
|
rhs = lhs,
|
||||||
|
expr = false,
|
||||||
|
noremap = true,
|
||||||
|
script = false,
|
||||||
|
silent = false,
|
||||||
|
nowait = false,
|
||||||
|
buffer = false,
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
---Evacuate existing key mapping
|
---Evacuate existing key mapping
|
||||||
---@param mode string
|
---@param mode string
|
||||||
---@param lhs string
|
---@param lhs string
|
||||||
---@return { keys: string, mode: string }
|
---@return { keys: string, mode: string }
|
||||||
keymap.evacuate = function(mode, lhs)
|
keymap.evacuate = function(mode, lhs)
|
||||||
local map = keymap.find_map_by_lhs(mode, lhs)
|
local map = keymap.get_mapping(mode, lhs)
|
||||||
if not map then
|
if not map then
|
||||||
return { keys = lhs, mode = 'itn' }
|
return { keys = lhs, mode = 'itn' }
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Keep existing mapping as <Plug> mapping. We escape fisrt recursive key sequence. See `:help recursive_mapping`)
|
-- Keep existing mapping as <Plug> mapping. We escape fisrt recursive key sequence. See `:help recursive_mapping`)
|
||||||
local rhs = map.rhs
|
local rhs = map.rhs
|
||||||
if map.noremap == 0 and map.expr == 1 then
|
if not map.noremap and map.expr then
|
||||||
-- remap & expr mapping should evacuate as <Plug> mapping with solving recursive mapping.
|
-- remap & expr mapping should evacuate as <Plug> mapping with solving recursive mapping.
|
||||||
rhs = string.format('v:lua.cmp.utils.keymap.evacuate.expr("%s", "%s", "%s")', mode, str.escape(keymap.escape(lhs), { '"' }), str.escape(keymap.escape(rhs), { '"' }))
|
rhs = string.format('v:lua.cmp.utils.keymap.evacuate.expr("%s", "%s", "%s")', mode, str.escape(keymap.escape(lhs), { '"' }), str.escape(keymap.escape(rhs), { '"' }))
|
||||||
elseif map.noremap ~= 0 and map.expr == 1 then
|
elseif map.noremap and map.expr then
|
||||||
-- noremap & expr mapping should always evacuate as <Plug> mapping.
|
-- noremap & expr mapping should always evacuate as <Plug> mapping.
|
||||||
rhs = rhs
|
rhs = rhs
|
||||||
elseif map.script == 1 then
|
elseif map.script then
|
||||||
-- script mapping should always evacuate as <Plug> mapping.
|
-- script mapping should always evacuate as <Plug> mapping.
|
||||||
rhs = rhs
|
rhs = rhs
|
||||||
elseif map.noremap == 0 then
|
elseif not map.noremap then
|
||||||
-- remap & non-expr mapping should be checked if recursive or not.
|
-- remap & non-expr mapping should be checked if recursive or not.
|
||||||
rhs = keymap.recursive(mode, lhs, rhs)
|
rhs = keymap.recursive(mode, lhs, rhs)
|
||||||
if rhs == map.rhs or map.noremap ~= 0 then
|
if rhs == map.rhs or map.noremap then
|
||||||
return { keys = rhs, mode = 'it' .. (map.noremap == 1 and 'n' or '') }
|
return { keys = rhs, mode = 'it' .. (map.noremap and 'n' or '') }
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
-- noremap & non-expr mapping doesn't need to evacuate.
|
-- noremap & non-expr mapping doesn't need to evacuate.
|
||||||
return { keys = rhs, mode = 'it' .. (map.noremap == 1 and 'n' or '') }
|
return { keys = rhs, mode = 'it' .. (map.noremap and 'n' or '') }
|
||||||
end
|
end
|
||||||
|
|
||||||
local fallback = ('<Plug>(cmp-utils-keymap-evacuate-rhs:%s)'):format(map.lhs)
|
local fallback = ('<Plug>(cmp-utils-keymap-evacuate-rhs:%s)'):format(map.lhs)
|
||||||
vim.api.nvim_buf_set_keymap(0, mode, fallback, rhs, {
|
vim.api.nvim_buf_set_keymap(0, mode, fallback, rhs, {
|
||||||
expr = map.expr ~= 0,
|
expr = map.expr,
|
||||||
noremap = map.noremap ~= 0,
|
noremap = map.noremap,
|
||||||
script = map.script ~= 0,
|
script = map.script,
|
||||||
silent = mode ~= 'c', -- I can't understand but it solves the #427 (wilder.nvim's mapping does not work if silent=true in cmdline mode...)
|
silent = mode ~= 'c', -- I can't understand but it solves the #427 (wilder.nvim's mapping does not work if silent=true in cmdline mode...)
|
||||||
})
|
})
|
||||||
return { keys = fallback, mode = 'it' }
|
return { keys = fallback, mode = 'it' }
|
||||||
@@ -335,45 +268,4 @@ keymap.recursive = function(mode, lhs, rhs)
|
|||||||
return new_rhs
|
return new_rhs
|
||||||
end
|
end
|
||||||
|
|
||||||
---Get specific key mapping
|
|
||||||
---@param mode string
|
|
||||||
---@param lhs string
|
|
||||||
---@return table
|
|
||||||
keymap.find_map_by_lhs = function(mode, lhs)
|
|
||||||
for _, map in ipairs(vim.api.nvim_buf_get_keymap(0, mode)) do
|
|
||||||
if keymap.equals(map.lhs, lhs) then
|
|
||||||
if string.match(map.rhs, vim.pesc('v:lua.cmp.utils.keymap.listen.run')) then
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
return map
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
for _, map in ipairs(vim.api.nvim_get_keymap(mode)) do
|
|
||||||
if keymap.equals(map.lhs, lhs) then
|
|
||||||
if string.match(map.rhs, vim.pesc('v:lua.cmp.utils.keymap.listen.run')) then
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
return map
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
keymap.spec = function()
|
|
||||||
vim.fn.setreg('q', '')
|
|
||||||
vim.cmd([[normal! qq]])
|
|
||||||
vim.schedule(function()
|
|
||||||
keymap.feedkeys('i', 'nt', function()
|
|
||||||
keymap.feedkeys(keymap.t('foo2'), 'n')
|
|
||||||
keymap.feedkeys(keymap.t('bar2'), 'nt')
|
|
||||||
keymap.feedkeys(keymap.t('baz2'), 'n', function()
|
|
||||||
vim.cmd([[normal! q]])
|
|
||||||
end)
|
|
||||||
keymap.feedkeys(keymap.t('baz1'), 'ni')
|
|
||||||
keymap.feedkeys(keymap.t('bar1'), 'nti')
|
|
||||||
keymap.feedkeys(keymap.t('foo1'), 'ni')
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
|
|
||||||
return keymap
|
return keymap
|
||||||
|
|||||||
@@ -17,23 +17,6 @@ describe('keymap', function()
|
|||||||
assert.are.equal(keymap.escape('<LT>C-d>'), '<LT>C-d>')
|
assert.are.equal(keymap.escape('<LT>C-d>'), '<LT>C-d>')
|
||||||
end)
|
end)
|
||||||
|
|
||||||
describe('feedkeys', function()
|
|
||||||
it('dot-repeat', function()
|
|
||||||
keymap.feedkeys(keymap.t('iaiueo<Esc>'), 'nx')
|
|
||||||
assert.are.equal(vim.fn.getreg('.'), keymap.t('aiueo'))
|
|
||||||
end)
|
|
||||||
it('macro', function()
|
|
||||||
vim.fn.setreg('q', '')
|
|
||||||
vim.cmd([[normal! qq]])
|
|
||||||
keymap.feedkeys(keymap.t('iaiueo'), 'nt')
|
|
||||||
keymap.feedkeys(keymap.t('<Esc>'), 'nt', function()
|
|
||||||
vim.cmd([[normal! q]])
|
|
||||||
assert.are.equal(vim.fn.getreg('q'), keymap.t('iaiueo<Esc>'))
|
|
||||||
print(vim.fn.getreg('q'))
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
|
|
||||||
describe('evacuate', function()
|
describe('evacuate', function()
|
||||||
before_each(spec.before)
|
before_each(spec.before)
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
local event = require('cmp.utils.event')
|
local event = require('cmp.utils.event')
|
||||||
local autocmd = require('cmp.utils.autocmd')
|
local autocmd = require('cmp.utils.autocmd')
|
||||||
|
local feedkeys = require('cmp.utils.feedkeys')
|
||||||
local window = require('cmp.utils.window')
|
local window = require('cmp.utils.window')
|
||||||
local config = require('cmp.config')
|
local config = require('cmp.config')
|
||||||
local types = require('cmp.types')
|
local types = require('cmp.types')
|
||||||
@@ -183,7 +184,7 @@ custom_entries_view.abort = function(self)
|
|||||||
if self.prefix then
|
if self.prefix then
|
||||||
self:_insert(self.prefix)
|
self:_insert(self.prefix)
|
||||||
end
|
end
|
||||||
keymap.feedkeys('', 'n', function()
|
feedkeys.call('', 'n', function()
|
||||||
self:close()
|
self:close()
|
||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
@@ -306,10 +307,10 @@ custom_entries_view._insert = function(self, word)
|
|||||||
vim.api.nvim_feedkeys(keymap.backspace(length) .. word, 'int', true)
|
vim.api.nvim_feedkeys(keymap.backspace(length) .. word, 'int', true)
|
||||||
else
|
else
|
||||||
local release = require('cmp').core:suspend()
|
local release = require('cmp').core:suspend()
|
||||||
keymap.feedkeys('', 'n', function()
|
feedkeys.call('', 'n', function()
|
||||||
local cursor = api.get_cursor()
|
local cursor = api.get_cursor()
|
||||||
local length = vim.str_utfindex(string.sub(api.get_current_line(), self.offset, cursor[2]))
|
local length = vim.str_utfindex(string.sub(api.get_current_line(), self.offset, cursor[2]))
|
||||||
keymap.feedkeys(
|
feedkeys.call(
|
||||||
keymap.backspace(length) .. word,
|
keymap.backspace(length) .. word,
|
||||||
'int',
|
'int',
|
||||||
vim.schedule_wrap(function()
|
vim.schedule_wrap(function()
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
local event = require('cmp.utils.event')
|
local event = require('cmp.utils.event')
|
||||||
local autocmd = require('cmp.utils.autocmd')
|
local autocmd = require('cmp.utils.autocmd')
|
||||||
local keymap = require('cmp.utils.keymap')
|
local keymap = require('cmp.utils.keymap')
|
||||||
|
local feedkeys = require('cmp.utils.feedkeys')
|
||||||
local types = require('cmp.types')
|
local types = require('cmp.types')
|
||||||
local config = require('cmp.config')
|
local config = require('cmp.config')
|
||||||
local api = require('cmp.utils.api')
|
local api = require('cmp.utils.api')
|
||||||
@@ -118,9 +119,9 @@ end
|
|||||||
native_entries_view.select_next_item = function(self, option)
|
native_entries_view.select_next_item = function(self, option)
|
||||||
if self:visible() then
|
if self:visible() then
|
||||||
if (option.behavior or types.cmp.SelectBehavior.Insert) == types.cmp.SelectBehavior.Insert then
|
if (option.behavior or types.cmp.SelectBehavior.Insert) == types.cmp.SelectBehavior.Insert then
|
||||||
keymap.feedkeys(keymap.t('<C-n>'), 'n')
|
feedkeys.call(keymap.t('<C-n>'), 'n')
|
||||||
else
|
else
|
||||||
keymap.feedkeys(keymap.t('<Down>'), 'n')
|
feedkeys.call(keymap.t('<Down>'), 'n')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -128,9 +129,9 @@ end
|
|||||||
native_entries_view.select_prev_item = function(self, option)
|
native_entries_view.select_prev_item = function(self, option)
|
||||||
if self:visible() then
|
if self:visible() then
|
||||||
if (option.behavior or types.cmp.SelectBehavior.Insert) == types.cmp.SelectBehavior.Insert then
|
if (option.behavior or types.cmp.SelectBehavior.Insert) == types.cmp.SelectBehavior.Insert then
|
||||||
keymap.feedkeys(keymap.t('<C-p>'), 'n')
|
feedkeys.call(keymap.t('<C-p>'), 'n')
|
||||||
else
|
else
|
||||||
keymap.feedkeys(keymap.t('<Up>'), 'n')
|
feedkeys.call(keymap.t('<Up>'), 'n')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
Reference in New Issue
Block a user