mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-26 04:17:01 +00:00 
			
		
		
		
	 df646572c5
			
		
	
	df646572c5
	
	
	
		
			
			Co-authored-by: Raphael <glephunter@gmail.com> Co-authored-by: smjonas <jonas.strittmatter@gmx.de> Co-authored-by: zeertzjq <zeertzjq@outlook.com>
		
			
				
	
	
		
			1169 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			1169 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| local helpers = require('test.functional.helpers')(after_each)
 | |
| 
 | |
| local bufmeths = helpers.bufmeths
 | |
| local clear = helpers.clear
 | |
| local command = helpers.command
 | |
| local curbufmeths = helpers.curbufmeths
 | |
| local eq, neq = helpers.eq, helpers.neq
 | |
| local exec_lua = helpers.exec_lua
 | |
| local exec = helpers.exec
 | |
| local feed = helpers.feed
 | |
| local funcs = helpers.funcs
 | |
| local meths = helpers.meths
 | |
| local source = helpers.source
 | |
| local pcall_err = helpers.pcall_err
 | |
| 
 | |
| local shallowcopy = helpers.shallowcopy
 | |
| local sleep = helpers.sleep
 | |
| 
 | |
| local sid_api_client = -9
 | |
| local sid_lua = -8
 | |
| 
 | |
| describe('nvim_get_keymap', function()
 | |
|   before_each(clear)
 | |
| 
 | |
|   -- Basic mapping and table to be used to describe results
 | |
|   local foo_bar_string = 'nnoremap foo bar'
 | |
|   local foo_bar_map_table = {
 | |
|     lhs='foo',
 | |
|     lhsraw='foo',
 | |
|     script=0,
 | |
|     silent=0,
 | |
|     rhs='bar',
 | |
|     expr=0,
 | |
|     sid=0,
 | |
|     buffer=0,
 | |
|     nowait=0,
 | |
|     mode='n',
 | |
|     noremap=1,
 | |
|     lnum=0,
 | |
|   }
 | |
| 
 | |
|   it('returns empty list when no map', function()
 | |
|     eq({}, meths.get_keymap('n'))
 | |
|   end)
 | |
| 
 | |
|   it('returns list of all applicable mappings', function()
 | |
|     command(foo_bar_string)
 | |
|     -- Only one mapping available
 | |
|     -- Should be the same as the dictionary we supplied earlier
 | |
|     -- and the dictionary you would get from maparg
 | |
|     -- since this is a global map, and not script local
 | |
|     eq({foo_bar_map_table}, meths.get_keymap('n'))
 | |
|     eq({funcs.maparg('foo', 'n', false, true)},
 | |
|       meths.get_keymap('n')
 | |
|     )
 | |
| 
 | |
|     -- Add another mapping
 | |
|     command('nnoremap foo_longer bar_longer')
 | |
|     local foolong_bar_map_table = shallowcopy(foo_bar_map_table)
 | |
|     foolong_bar_map_table['lhs'] = 'foo_longer'
 | |
|     foolong_bar_map_table['lhsraw'] = 'foo_longer'
 | |
|     foolong_bar_map_table['rhs'] = 'bar_longer'
 | |
| 
 | |
|     eq({foolong_bar_map_table, foo_bar_map_table},
 | |
|       meths.get_keymap('n')
 | |
|     )
 | |
| 
 | |
|     -- Remove a mapping
 | |
|     command('unmap foo_longer')
 | |
|     eq({foo_bar_map_table},
 | |
|       meths.get_keymap('n')
 | |
|     )
 | |
|   end)
 | |
| 
 | |
|   it('works for other modes', function()
 | |
|     -- Add two mappings, one in insert and one normal
 | |
|     -- We'll only check the insert mode one
 | |
|     command('nnoremap not_going to_check')
 | |
| 
 | |
|     command('inoremap foo bar')
 | |
|     -- The table will be the same except for the mode
 | |
|     local insert_table = shallowcopy(foo_bar_map_table)
 | |
|     insert_table['mode'] = 'i'
 | |
| 
 | |
|     eq({insert_table}, meths.get_keymap('i'))
 | |
|   end)
 | |
| 
 | |
|   it('considers scope', function()
 | |
|     -- change the map slightly
 | |
|     command('nnoremap foo_longer bar_longer')
 | |
|     local foolong_bar_map_table = shallowcopy(foo_bar_map_table)
 | |
|     foolong_bar_map_table['lhs'] = 'foo_longer'
 | |
|     foolong_bar_map_table['lhsraw'] = 'foo_longer'
 | |
|     foolong_bar_map_table['rhs'] = 'bar_longer'
 | |
| 
 | |
|     local buffer_table = shallowcopy(foo_bar_map_table)
 | |
|     buffer_table['buffer'] = 1
 | |
| 
 | |
|     command('nnoremap <buffer> foo bar')
 | |
| 
 | |
|     -- The buffer mapping should not show up
 | |
|     eq({foolong_bar_map_table}, meths.get_keymap('n'))
 | |
|     eq({buffer_table}, curbufmeths.get_keymap('n'))
 | |
|   end)
 | |
| 
 | |
|   it('considers scope for overlapping maps', function()
 | |
|     command('nnoremap foo bar')
 | |
| 
 | |
|     local buffer_table = shallowcopy(foo_bar_map_table)
 | |
|     buffer_table['buffer'] = 1
 | |
| 
 | |
|     command('nnoremap <buffer> foo bar')
 | |
| 
 | |
|     eq({foo_bar_map_table}, meths.get_keymap('n'))
 | |
|     eq({buffer_table}, curbufmeths.get_keymap('n'))
 | |
|   end)
 | |
| 
 | |
|   it('can retrieve mapping for different buffers', function()
 | |
|     local original_buffer = curbufmeths.get_number()
 | |
|     -- Place something in each of the buffers to make sure they stick around
 | |
|     -- and set hidden so we can leave them
 | |
|     command('set hidden')
 | |
|     command('new')
 | |
|     command('normal! ihello 2')
 | |
|     command('new')
 | |
|     command('normal! ihello 3')
 | |
| 
 | |
|     local final_buffer = curbufmeths.get_number()
 | |
| 
 | |
|     command('nnoremap <buffer> foo bar')
 | |
|     -- Final buffer will have buffer mappings
 | |
|     local buffer_table = shallowcopy(foo_bar_map_table)
 | |
|     buffer_table['buffer'] = final_buffer
 | |
|     eq({buffer_table}, meths.buf_get_keymap(final_buffer, 'n'))
 | |
|     eq({buffer_table}, meths.buf_get_keymap(0, 'n'))
 | |
| 
 | |
|     command('buffer ' .. original_buffer)
 | |
|     eq(original_buffer, curbufmeths.get_number())
 | |
|     -- Original buffer won't have any mappings
 | |
|     eq({}, meths.get_keymap('n'))
 | |
|     eq({}, curbufmeths.get_keymap('n'))
 | |
|     eq({buffer_table}, meths.buf_get_keymap(final_buffer, 'n'))
 | |
|   end)
 | |
| 
 | |
|   -- Test toggle switches for basic options
 | |
|   -- @param  option  The key represented in the `maparg()` result dict
 | |
|   local function global_and_buffer_test(map,
 | |
|                                         option,
 | |
|                                         option_token,
 | |
|                                         global_on_result,
 | |
|                                         buffer_on_result,
 | |
|                                         global_off_result,
 | |
|                                         buffer_off_result,
 | |
|                                         new_windows)
 | |
| 
 | |
|     local function make_new_windows(number_of_windows)
 | |
|       if new_windows == nil then
 | |
|         return nil
 | |
|       end
 | |
| 
 | |
|       for _=1,number_of_windows do
 | |
|         command('new')
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     local mode = string.sub(map, 1,1)
 | |
|     -- Don't run this for the <buffer> mapping, since it doesn't make sense
 | |
|     if option_token ~= '<buffer>' then
 | |
|       it(string.format( 'returns %d for the key "%s" when %s is used globally with %s (%s)',
 | |
|           global_on_result, option, option_token, map, mode), function()
 | |
|         make_new_windows(new_windows)
 | |
|         command(map .. ' ' .. option_token .. ' foo bar')
 | |
|         local result = meths.get_keymap(mode)[1][option]
 | |
|         eq(global_on_result, result)
 | |
|       end)
 | |
|     end
 | |
| 
 | |
|     it(string.format('returns %d for the key "%s" when %s is used for buffers with %s (%s)',
 | |
|         buffer_on_result, option, option_token, map, mode), function()
 | |
|       make_new_windows(new_windows)
 | |
|       command(map .. ' <buffer> ' .. option_token .. ' foo bar')
 | |
|       local result = curbufmeths.get_keymap(mode)[1][option]
 | |
|       eq(buffer_on_result, result)
 | |
|     end)
 | |
| 
 | |
|     -- Don't run these for the <buffer> mapping, since it doesn't make sense
 | |
|     if option_token ~= '<buffer>' then
 | |
|       it(string.format('returns %d for the key "%s" when %s is not used globally with %s (%s)',
 | |
|           global_off_result, option, option_token, map, mode), function()
 | |
|         make_new_windows(new_windows)
 | |
|         command(map .. ' baz bat')
 | |
|         local result = meths.get_keymap(mode)[1][option]
 | |
|         eq(global_off_result, result)
 | |
|       end)
 | |
| 
 | |
|       it(string.format('returns %d for the key "%s" when %s is not used for buffers with %s (%s)',
 | |
|           buffer_off_result, option, option_token, map, mode), function()
 | |
|         make_new_windows(new_windows)
 | |
|         command(map .. ' <buffer> foo bar')
 | |
| 
 | |
|         local result = curbufmeths.get_keymap(mode)[1][option]
 | |
|         eq(buffer_off_result, result)
 | |
|       end)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   -- Standard modes and returns the same values in the dictionary as maparg()
 | |
|   local mode_list = {'nnoremap', 'nmap', 'imap', 'inoremap', 'cnoremap'}
 | |
|   for mode in pairs(mode_list) do
 | |
|     global_and_buffer_test(mode_list[mode], 'silent', '<silent>', 1, 1, 0, 0)
 | |
|     global_and_buffer_test(mode_list[mode], 'nowait', '<nowait>', 1, 1, 0, 0)
 | |
|     global_and_buffer_test(mode_list[mode], 'expr', '<expr>', 1, 1, 0, 0)
 | |
|   end
 | |
| 
 | |
|   -- noremap will now be 2 if script was used, which is not the same as maparg()
 | |
|   global_and_buffer_test('nmap', 'noremap', '<script>', 2, 2, 0, 0)
 | |
|   global_and_buffer_test('nnoremap', 'noremap', '<script>', 2, 2, 1, 1)
 | |
| 
 | |
|   -- buffer will return the buffer ID, which is not the same as maparg()
 | |
|   -- Three of these tests won't run
 | |
|   global_and_buffer_test('nnoremap', 'buffer', '<buffer>', nil, 3, nil, nil, 2)
 | |
| 
 | |
|   it('returns script numbers for global maps', function()
 | |
|     source([[
 | |
|       function! s:maparg_test_function() abort
 | |
|         return 'testing'
 | |
|       endfunction
 | |
| 
 | |
|       nnoremap fizz :call <SID>maparg_test_function()<CR>
 | |
|     ]])
 | |
|     local sid_result = meths.get_keymap('n')[1]['sid']
 | |
|     eq(1, sid_result)
 | |
|     eq('testing', meths.call_function('<SNR>' .. sid_result .. '_maparg_test_function', {}))
 | |
|   end)
 | |
| 
 | |
|   it('returns script numbers for buffer maps', function()
 | |
|     source([[
 | |
|       function! s:maparg_test_function() abort
 | |
|         return 'testing'
 | |
|       endfunction
 | |
| 
 | |
|       nnoremap <buffer> fizz :call <SID>maparg_test_function()<CR>
 | |
|     ]])
 | |
|     local sid_result = curbufmeths.get_keymap('n')[1]['sid']
 | |
|     eq(1, sid_result)
 | |
|     eq('testing', meths.call_function('<SNR>' .. sid_result .. '_maparg_test_function', {}))
 | |
|   end)
 | |
| 
 | |
|   it('works with <F12> and others', function()
 | |
|     command('nnoremap <F12> :let g:maparg_test_var = 1<CR>')
 | |
|     eq('<F12>', meths.get_keymap('n')[1]['lhs'])
 | |
|     eq(':let g:maparg_test_var = 1<CR>', meths.get_keymap('n')[1]['rhs'])
 | |
|   end)
 | |
| 
 | |
|   it('works correctly despite various &cpo settings', function()
 | |
|     local cpo_table = {
 | |
|       script=0,
 | |
|       silent=0,
 | |
|       expr=0,
 | |
|       sid=0,
 | |
|       buffer=0,
 | |
|       nowait=0,
 | |
|       noremap=1,
 | |
|       lnum=0,
 | |
|     }
 | |
|     local function cpomap(lhs, rhs, mode)
 | |
|       local ret = shallowcopy(cpo_table)
 | |
|       ret.lhs = lhs
 | |
|       ret.rhs = rhs
 | |
|       ret.mode = mode
 | |
|       return ret
 | |
|     end
 | |
| 
 | |
|     command('set cpo+=B')
 | |
|     command('nnoremap \\<C-a><C-a><LT>C-a>\\  \\<C-b><C-b><LT>C-b>\\')
 | |
|     command('nnoremap <special> \\<C-c><C-c><LT>C-c>\\  \\<C-d><C-d><LT>C-d>\\')
 | |
| 
 | |
|     command('set cpo+=B')
 | |
|     command('xnoremap \\<C-a><C-a><LT>C-a>\\  \\<C-b><C-b><LT>C-b>\\')
 | |
|     command('xnoremap <special> \\<C-c><C-c><LT>C-c>\\  \\<C-d><C-d><LT>C-d>\\')
 | |
| 
 | |
|     command('set cpo-=B')
 | |
|     command('snoremap \\<C-a><C-a><LT>C-a>\\  \\<C-b><C-b><LT>C-b>\\')
 | |
|     command('snoremap <special> \\<C-c><C-c><LT>C-c>\\  \\<C-d><C-d><LT>C-d>\\')
 | |
| 
 | |
|     command('set cpo-=B')
 | |
|     command('onoremap \\<C-a><C-a><LT>C-a>\\  \\<C-b><C-b><LT>C-b>\\')
 | |
|     command('onoremap <special> \\<C-c><C-c><LT>C-c>\\  \\<C-d><C-d><LT>C-d>\\')
 | |
| 
 | |
|     -- wrapper around get_keymap() that drops "lhsraw" and "lhsrawalt" which are hard to check
 | |
|     local function get_keymap_noraw(...)
 | |
|       local ret = meths.get_keymap(...)
 | |
|       for _, item in ipairs(ret) do
 | |
|         item.lhsraw = nil
 | |
|         item.lhsrawalt = nil
 | |
|       end
 | |
|       return ret
 | |
|     end
 | |
| 
 | |
|     for _, cmd in ipairs({
 | |
|       'set cpo-=B',
 | |
|       'set cpo+=B',
 | |
|     }) do
 | |
|       command(cmd)
 | |
|       eq({cpomap('\\<C-C><C-C><lt>C-c>\\', '\\<C-D><C-D><lt>C-d>\\', 'n'),
 | |
|           cpomap('\\<C-A><C-A><lt>C-a>\\', '\\<C-B><C-B><lt>C-b>\\', 'n')},
 | |
|          get_keymap_noraw('n'))
 | |
|       eq({cpomap('\\<C-C><C-C><lt>C-c>\\', '\\<C-D><C-D><lt>C-d>\\', 'x'),
 | |
|           cpomap('\\<C-A><C-A><lt>C-a>\\', '\\<C-B><C-B><lt>C-b>\\', 'x')},
 | |
|          get_keymap_noraw('x'))
 | |
|       eq({cpomap('<lt>C-c><C-C><lt>C-c> ', '<lt>C-d><C-D><lt>C-d>', 's'),
 | |
|           cpomap('<lt>C-a><C-A><lt>C-a> ', '<lt>C-b><C-B><lt>C-b>', 's')},
 | |
|          get_keymap_noraw('s'))
 | |
|       eq({cpomap('<lt>C-c><C-C><lt>C-c> ', '<lt>C-d><C-D><lt>C-d>', 'o'),
 | |
|           cpomap('<lt>C-a><C-A><lt>C-a> ', '<lt>C-b><C-B><lt>C-b>', 'o')},
 | |
|          get_keymap_noraw('o'))
 | |
|     end
 | |
|   end)
 | |
| 
 | |
|   it('always uses space for space and bar for bar', function()
 | |
|     local space_table = {
 | |
|       lhs='|   |',
 | |
|       lhsraw='|   |',
 | |
|       rhs='|    |',
 | |
|       mode='n',
 | |
|       script=0,
 | |
|       silent=0,
 | |
|       expr=0,
 | |
|       sid=0,
 | |
|       buffer=0,
 | |
|       nowait=0,
 | |
|       noremap=1,
 | |
|       lnum=0,
 | |
|     }
 | |
|     command('nnoremap \\|<Char-0x20><Char-32><Space><Bar> \\|<Char-0x20><Char-32><Space> <Bar>')
 | |
|     eq({space_table}, meths.get_keymap('n'))
 | |
|   end)
 | |
| 
 | |
|   it('can handle lua mappings', function()
 | |
|     eq(0, exec_lua([[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_set_keymap('n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]]))
 | |
| 
 | |
|     feed('asdf\n')
 | |
|     eq(1, exec_lua([[return GlobalCount]]))
 | |
| 
 | |
|     eq(2, exec_lua([[
 | |
|       vim.api.nvim_get_keymap('n')[1].callback()
 | |
|       return GlobalCount
 | |
|     ]]))
 | |
| 
 | |
|     exec([[
 | |
|       call nvim_get_keymap('n')[0].callback()
 | |
|     ]])
 | |
|     eq(3, exec_lua([[return GlobalCount]]))
 | |
| 
 | |
|     local mapargs = meths.get_keymap('n')
 | |
|     mapargs[1].callback = nil
 | |
|     eq({
 | |
|       lhs='asdf',
 | |
|       lhsraw='asdf',
 | |
|       script=0,
 | |
|       silent=0,
 | |
|       expr=0,
 | |
|       sid=sid_lua,
 | |
|       buffer=0,
 | |
|       nowait=0,
 | |
|       mode='n',
 | |
|       noremap=0,
 | |
|       lnum=0,
 | |
|     }, mapargs[1])
 | |
|   end)
 | |
| 
 | |
|   it ('can handle map descriptions', function()
 | |
|     meths.set_keymap('n', 'lhs', 'rhs', {desc="map description"})
 | |
|     eq({
 | |
|       lhs='lhs',
 | |
|       lhsraw='lhs',
 | |
|       rhs='rhs',
 | |
|       script=0,
 | |
|       silent=0,
 | |
|       expr=0,
 | |
|       sid=sid_api_client,
 | |
|       buffer=0,
 | |
|       nowait=0,
 | |
|       mode='n',
 | |
|       noremap=0,
 | |
|       lnum=0,
 | |
|       desc='map description'
 | |
|     }, meths.get_keymap('n')[1])
 | |
|   end)
 | |
| end)
 | |
| 
 | |
| describe('nvim_set_keymap, nvim_del_keymap', function()
 | |
|   before_each(clear)
 | |
| 
 | |
|   -- `generate_expected` is truthy: for generating an expected output for
 | |
|   -- maparg(), which does not accept "!" (though it returns "!" in its output
 | |
|   -- if getting a mapping set with |:map!|).
 | |
|   local function normalize_mapmode(mode, generate_expected)
 | |
|     if not generate_expected and mode == '!' then
 | |
|       -- Cannot retrieve mapmode-ic mappings with "!", but can with "i" or "c".
 | |
|       mode = 'i'
 | |
|     elseif mode == '' then
 | |
|       mode = generate_expected and ' ' or mode
 | |
|     end
 | |
|     return mode
 | |
|   end
 | |
| 
 | |
|   -- Generate a mapargs dict, for comparison against the mapping that was
 | |
|   -- actually set
 | |
|   local function generate_mapargs(mode, lhs, rhs, opts)
 | |
|     if not opts then
 | |
|       opts = {}
 | |
|     end
 | |
| 
 | |
|     local to_return = {}
 | |
|     to_return.mode = normalize_mapmode(mode, true)
 | |
|     to_return.noremap = not opts.noremap and 0 or 1
 | |
|     to_return.lhs = lhs
 | |
|     to_return.rhs = rhs
 | |
|     to_return.script = 0
 | |
|     to_return.silent = not opts.silent and 0 or 1
 | |
|     to_return.nowait = not opts.nowait and 0 or 1
 | |
|     to_return.expr = not opts.expr and 0 or 1
 | |
|     to_return.sid = not opts.sid and sid_api_client or opts.sid
 | |
|     to_return.buffer = not opts.buffer and 0 or opts.buffer
 | |
|     to_return.lnum = not opts.lnum and 0 or opts.lnum
 | |
|     to_return.desc = opts.desc
 | |
| 
 | |
|     return to_return
 | |
|   end
 | |
| 
 | |
|   -- Gets a maparg() dict from Nvim, if one exists.
 | |
|   local function get_mapargs(mode, lhs)
 | |
|     local mapargs = funcs.maparg(lhs, normalize_mapmode(mode), false, true)
 | |
|     -- drop "lhsraw" and "lhsrawalt" which are hard to check
 | |
|     mapargs.lhsraw = nil
 | |
|     mapargs.lhsrawalt = nil
 | |
|     return mapargs
 | |
|   end
 | |
| 
 | |
|   it('error on empty LHS', function()
 | |
|     -- escape parentheses in lua string, else comparison fails erroneously
 | |
|     eq('Invalid (empty) LHS', pcall_err(meths.set_keymap, '', '', 'rhs', {}))
 | |
|     eq('Invalid (empty) LHS', pcall_err(meths.set_keymap, '', '', '', {}))
 | |
|     eq('Invalid (empty) LHS', pcall_err(meths.del_keymap, '', ''))
 | |
|   end)
 | |
| 
 | |
|   it('error if LHS longer than MAXMAPLEN', function()
 | |
|     -- assume MAXMAPLEN of 50 chars, as declared in vim.h
 | |
|     local MAXMAPLEN = 50
 | |
|     local lhs = ''
 | |
|     for i=1,MAXMAPLEN do
 | |
|       lhs = lhs..(i % 10)
 | |
|     end
 | |
| 
 | |
|     -- exactly 50 chars should be fine
 | |
|     meths.set_keymap('', lhs, 'rhs', {})
 | |
| 
 | |
|     -- del_keymap should unmap successfully
 | |
|     meths.del_keymap('', lhs)
 | |
|     eq({}, get_mapargs('', lhs))
 | |
| 
 | |
|     -- 51 chars should produce an error
 | |
|     lhs = lhs..'1'
 | |
|     eq('LHS exceeds maximum map length: '..lhs,
 | |
|       pcall_err(meths.set_keymap, '', lhs, 'rhs', {}))
 | |
|     eq('LHS exceeds maximum map length: '..lhs,
 | |
|       pcall_err(meths.del_keymap, '', lhs))
 | |
|   end)
 | |
| 
 | |
|   it('does not throw errors when rhs is longer than MAXMAPLEN', function()
 | |
|     local MAXMAPLEN = 50
 | |
|     local rhs = ''
 | |
|     for i=1,MAXMAPLEN do
 | |
|       rhs = rhs..(i % 10)
 | |
|     end
 | |
|     rhs = rhs..'1'
 | |
|     meths.set_keymap('', 'lhs', rhs, {})
 | |
|     eq(generate_mapargs('', 'lhs', rhs),
 | |
|        get_mapargs('', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('throws errors when given too-long mode shortnames', function()
 | |
|     eq('Shortname is too long: map',
 | |
|       pcall_err(meths.set_keymap, 'map', 'lhs', 'rhs', {}))
 | |
| 
 | |
|     eq('Shortname is too long: vmap',
 | |
|       pcall_err(meths.set_keymap, 'vmap', 'lhs', 'rhs', {}))
 | |
| 
 | |
|     eq('Shortname is too long: xnoremap',
 | |
|       pcall_err(meths.set_keymap, 'xnoremap', 'lhs', 'rhs', {}))
 | |
| 
 | |
|     eq('Shortname is too long: map', pcall_err(meths.del_keymap, 'map', 'lhs'))
 | |
|     eq('Shortname is too long: vmap', pcall_err(meths.del_keymap, 'vmap', 'lhs'))
 | |
|     eq('Shortname is too long: xnoremap', pcall_err(meths.del_keymap, 'xnoremap', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('error on invalid mode shortname', function()
 | |
|     eq('Invalid mode shortname: " "',
 | |
|       pcall_err(meths.set_keymap, ' ', 'lhs', 'rhs', {}))
 | |
|     eq('Invalid mode shortname: "m"',
 | |
|       pcall_err(meths.set_keymap, 'm', 'lhs', 'rhs', {}))
 | |
|     eq('Invalid mode shortname: "?"',
 | |
|       pcall_err(meths.set_keymap, '?', 'lhs', 'rhs', {}))
 | |
|     eq('Invalid mode shortname: "y"',
 | |
|       pcall_err(meths.set_keymap, 'y', 'lhs', 'rhs', {}))
 | |
|     eq('Invalid mode shortname: "p"',
 | |
|       pcall_err(meths.set_keymap, 'p', 'lhs', 'rhs', {}))
 | |
|     eq('Invalid mode shortname: "?"', pcall_err(meths.del_keymap, '?', 'lhs'))
 | |
|     eq('Invalid mode shortname: "y"', pcall_err(meths.del_keymap, 'y', 'lhs'))
 | |
|     eq('Invalid mode shortname: "p"', pcall_err(meths.del_keymap, 'p', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('error on invalid optnames', function()
 | |
|     eq("Invalid key: 'silentt'",
 | |
|       pcall_err(meths.set_keymap, 'n', 'lhs', 'rhs', {silentt = true}))
 | |
|     eq("Invalid key: 'sidd'",
 | |
|       pcall_err(meths.set_keymap, 'n', 'lhs', 'rhs', {sidd = false}))
 | |
|     eq("Invalid key: 'nowaiT'",
 | |
|       pcall_err(meths.set_keymap, 'n', 'lhs', 'rhs', {nowaiT = false}))
 | |
|   end)
 | |
| 
 | |
|   it('error on <buffer> option key', function()
 | |
|     eq("Invalid key: 'buffer'",
 | |
|       pcall_err(meths.set_keymap, 'n', 'lhs', 'rhs', {buffer = true}))
 | |
|   end)
 | |
| 
 | |
|   it('error when "replace_keycodes" is used without "expr"', function()
 | |
|     eq('"replace_keycodes" requires "expr"',
 | |
|       pcall_err(meths.set_keymap, 'n', 'lhs', 'rhs', {replace_keycodes = true}))
 | |
|   end)
 | |
| 
 | |
|   local optnames = {'nowait', 'silent', 'script', 'expr', 'unique'}
 | |
|   for _, opt in ipairs(optnames) do
 | |
|     -- note: need '%' to escape hyphens, which have special meaning in lua
 | |
|     it('throws an error when given non-boolean value for '..opt, function()
 | |
|       local opts = {}
 | |
|       opts[opt] = 'fooo'
 | |
|       eq(opt..' is not a boolean',
 | |
|         pcall_err(meths.set_keymap, 'n', 'lhs', 'rhs', opts))
 | |
|     end)
 | |
|   end
 | |
| 
 | |
|   -- Perform tests of basic functionality
 | |
|   it('sets ordinary mappings', function()
 | |
|     meths.set_keymap('n', 'lhs', 'rhs', {})
 | |
|     eq(generate_mapargs('n', 'lhs', 'rhs'), get_mapargs('n', 'lhs'))
 | |
| 
 | |
|     meths.set_keymap('v', 'lhs', 'rhs', {})
 | |
|     eq(generate_mapargs('v', 'lhs', 'rhs'), get_mapargs('v', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('does not throw when LHS or RHS have leading/trailing whitespace', function()
 | |
|     meths.set_keymap('n', '   lhs', 'rhs', {})
 | |
|     eq(generate_mapargs('n', '<Space><Space><Space>lhs', 'rhs'),
 | |
|        get_mapargs('n', '   lhs'))
 | |
| 
 | |
|     meths.set_keymap('n', 'lhs    ', 'rhs', {})
 | |
|     eq(generate_mapargs('n', 'lhs<Space><Space><Space><Space>', 'rhs'),
 | |
|        get_mapargs('n', 'lhs    '))
 | |
| 
 | |
|     meths.set_keymap('v', ' lhs  ', '\trhs\t\f', {})
 | |
|     eq(generate_mapargs('v', '<Space>lhs<Space><Space>', '\trhs\t\f'),
 | |
|        get_mapargs('v', ' lhs  '))
 | |
|   end)
 | |
| 
 | |
|   it('can set noremap mappings', function()
 | |
|     meths.set_keymap('x', 'lhs', 'rhs', {noremap = true})
 | |
|     eq(generate_mapargs('x', 'lhs', 'rhs', {noremap = true}),
 | |
|        get_mapargs('x', 'lhs'))
 | |
| 
 | |
|     meths.set_keymap('t', 'lhs', 'rhs', {noremap = true})
 | |
|     eq(generate_mapargs('t', 'lhs', 'rhs', {noremap = true}),
 | |
|        get_mapargs('t', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('can unmap mappings', function()
 | |
|     meths.set_keymap('v', 'lhs', 'rhs', {})
 | |
|     meths.del_keymap('v', 'lhs')
 | |
|     eq({}, get_mapargs('v', 'lhs'))
 | |
| 
 | |
|     meths.set_keymap('t', 'lhs', 'rhs', {noremap = true})
 | |
|     meths.del_keymap('t', 'lhs')
 | |
|     eq({}, get_mapargs('t', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   -- Test some edge cases
 | |
|   it('"!" and empty string are synonyms for mapmode-nvo', function()
 | |
|     local nvo_shortnames = {'', '!'}
 | |
|     for _, name in ipairs(nvo_shortnames) do
 | |
|       meths.set_keymap(name, 'lhs', 'rhs', {})
 | |
|       meths.del_keymap(name, 'lhs')
 | |
|       eq({}, get_mapargs(name, 'lhs'))
 | |
|     end
 | |
|   end)
 | |
| 
 | |
|   local special_chars = {'<C-U>', '<S-Left>', '<F12><F2><Tab>', '<Space><Tab>'}
 | |
|   for _, lhs in ipairs(special_chars) do
 | |
|     for _, rhs in ipairs(special_chars) do
 | |
|       local mapmode = '!'
 | |
|       it('can set mappings with special characters, lhs: '..lhs..', rhs: '..rhs,
 | |
|           function()
 | |
|         meths.set_keymap(mapmode, lhs, rhs, {})
 | |
|         eq(generate_mapargs(mapmode, lhs, rhs), get_mapargs(mapmode, lhs))
 | |
|       end)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   it('can set mappings containing literal keycodes', function()
 | |
|     meths.set_keymap('n', '\n\r\n', 'rhs', {})
 | |
|     local expected = generate_mapargs('n', '<NL><CR><NL>', 'rhs')
 | |
|     eq(expected, get_mapargs('n', '<NL><CR><NL>'))
 | |
|   end)
 | |
| 
 | |
|   it('can set mappings whose RHS is a <Nop>', function()
 | |
|     meths.set_keymap('i', 'lhs', '<Nop>', {})
 | |
|     command('normal ilhs')
 | |
|     eq({''}, curbufmeths.get_lines(0, -1, 0))  -- imap to <Nop> does nothing
 | |
|     eq(generate_mapargs('i', 'lhs', '<Nop>', {}),
 | |
|        get_mapargs('i', 'lhs'))
 | |
| 
 | |
|     -- also test for case insensitivity
 | |
|     meths.set_keymap('i', 'lhs', '<nOp>', {})
 | |
|     command('normal ilhs')
 | |
|     eq({''}, curbufmeths.get_lines(0, -1, 0))
 | |
|     -- note: RHS in returned mapargs() dict reflects the original RHS
 | |
|     -- provided by the user
 | |
|     eq(generate_mapargs('i', 'lhs', '<nOp>', {}),
 | |
|        get_mapargs('i', 'lhs'))
 | |
| 
 | |
|     meths.set_keymap('i', 'lhs', '<NOP>', {})
 | |
|     command('normal ilhs')
 | |
|     eq({''}, curbufmeths.get_lines(0, -1, 0))
 | |
|     eq(generate_mapargs('i', 'lhs', '<NOP>', {}),
 | |
|        get_mapargs('i', 'lhs'))
 | |
| 
 | |
|     -- a single ^V in RHS is also <Nop> (see :h map-empty-rhs)
 | |
|     meths.set_keymap('i', 'lhs', '\022', {})
 | |
|     command('normal ilhs')
 | |
|     eq({''}, curbufmeths.get_lines(0, -1, 0))
 | |
|     eq(generate_mapargs('i', 'lhs', '\022', {}),
 | |
|        get_mapargs('i', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('treats an empty RHS in a mapping like a <Nop>', function()
 | |
|     meths.set_keymap('i', 'lhs', '', {})
 | |
|     command('normal ilhs')
 | |
|     eq({''}, curbufmeths.get_lines(0, -1, 0))
 | |
|     eq(generate_mapargs('i', 'lhs', '', {}),
 | |
|        get_mapargs('i', 'lhs'))
 | |
|   end)
 | |
| 
 | |
|   it('can set and unset <M-">', function()
 | |
|     -- Taken from the legacy test: test_mapping.vim. Exposes a bug in which
 | |
|     -- replace_termcodes changes the length of the mapping's LHS, but
 | |
|     -- do_map continues to use the *old* length of LHS.
 | |
|     meths.set_keymap('i', '<M-">', 'foo', {})
 | |
|     meths.del_keymap('i', '<M-">')
 | |
|     eq({}, get_mapargs('i', '<M-">'))
 | |
|   end)
 | |
| 
 | |
|   it('interprets control sequences in expr-quotes correctly when called '
 | |
|      ..'inside vim', function()
 | |
|     command([[call nvim_set_keymap('i', "\<space>", "\<tab>", {})]])
 | |
|     eq(generate_mapargs('i', '<Space>', '\t', {sid=0}),
 | |
|        get_mapargs('i', '<Space>'))
 | |
|     feed('i ')
 | |
|     eq({'\t'}, curbufmeths.get_lines(0, -1, 0))
 | |
|   end)
 | |
| 
 | |
|   it('throws appropriate error messages when setting <unique> maps', function()
 | |
|     meths.set_keymap('l', 'lhs', 'rhs', {})
 | |
|     eq('E227: mapping already exists for lhs',
 | |
|       pcall_err(meths.set_keymap, 'l', 'lhs', 'rhs', {unique = true}))
 | |
|     -- different mapmode, no error should be thrown
 | |
|     meths.set_keymap('t', 'lhs', 'rhs', {unique = true})
 | |
|   end)
 | |
| 
 | |
|   it('can set <expr> mappings whose RHS change dynamically', function()
 | |
|     meths.exec([[
 | |
|         function! FlipFlop() abort
 | |
|           if !exists('g:flip') | let g:flip = 0 | endif
 | |
|           let g:flip = !g:flip
 | |
|           return g:flip
 | |
|         endfunction
 | |
|         ]], true)
 | |
|     eq(1, meths.call_function('FlipFlop', {}))
 | |
|     eq(0, meths.call_function('FlipFlop', {}))
 | |
|     eq(1, meths.call_function('FlipFlop', {}))
 | |
|     eq(0, meths.call_function('FlipFlop', {}))
 | |
| 
 | |
|     meths.set_keymap('i', 'lhs', 'FlipFlop()', {expr = true})
 | |
|     command('normal ilhs')
 | |
|     eq({'1'}, curbufmeths.get_lines(0, -1, 0))
 | |
| 
 | |
|     command('normal! ggVGd')
 | |
| 
 | |
|     command('normal ilhs')
 | |
|     eq({'0'}, curbufmeths.get_lines(0, -1, 0))
 | |
|   end)
 | |
| 
 | |
|   it('can set mappings that do trigger other mappings', function()
 | |
|     meths.set_keymap('i', 'mhs', 'rhs', {})
 | |
|     meths.set_keymap('i', 'lhs', 'mhs', {})
 | |
| 
 | |
|     command('normal imhs')
 | |
|     eq({'rhs'}, curbufmeths.get_lines(0, -1, 0))
 | |
| 
 | |
|     command('normal! ggVGd')
 | |
| 
 | |
|     command('normal ilhs')
 | |
|     eq({'rhs'}, curbufmeths.get_lines(0, -1, 0))
 | |
|   end)
 | |
| 
 | |
|   it("can set noremap mappings that don't trigger other mappings", function()
 | |
|     meths.set_keymap('i', 'mhs', 'rhs', {})
 | |
|     meths.set_keymap('i', 'lhs', 'mhs', {noremap = true})
 | |
| 
 | |
|     command('normal imhs')
 | |
|     eq({'rhs'}, curbufmeths.get_lines(0, -1, 0))
 | |
| 
 | |
|     command('normal! ggVGd')
 | |
| 
 | |
|     command('normal ilhs')  -- shouldn't trigger mhs-to-rhs mapping
 | |
|     eq({'mhs'}, curbufmeths.get_lines(0, -1, 0))
 | |
|   end)
 | |
| 
 | |
|   it("can set nowait mappings that fire without waiting", function()
 | |
|     meths.set_keymap('i', '123456', 'longer',  {})
 | |
|     meths.set_keymap('i', '123',    'shorter', {nowait = true})
 | |
| 
 | |
|     -- feed keys one at a time; if all keys arrive atomically, the longer
 | |
|     -- mapping will trigger
 | |
|     local keys = 'i123456'
 | |
|     for c in string.gmatch(keys, '.') do
 | |
|       feed(c)
 | |
|       sleep(5)
 | |
|     end
 | |
|     eq({'shorter456'}, curbufmeths.get_lines(0, -1, 0))
 | |
|   end)
 | |
| 
 | |
|   -- Perform exhaustive tests of basic functionality
 | |
|   local mapmodes = {'n', 'v', 'x', 's', 'o', '!', 'i', 'l', 'c', 't', ''}
 | |
|   for _, mapmode in ipairs(mapmodes) do
 | |
|     it('can set/unset normal mappings in mapmode '..mapmode, function()
 | |
|       meths.set_keymap(mapmode, 'lhs', 'rhs', {})
 | |
|       eq(generate_mapargs(mapmode, 'lhs', 'rhs'),
 | |
|          get_mapargs(mapmode, 'lhs'))
 | |
| 
 | |
|       -- some mapmodes (like 'o') will prevent other mapmodes (like '!') from
 | |
|       -- taking effect, so unmap after each mapping
 | |
|       meths.del_keymap(mapmode, 'lhs')
 | |
|       eq({}, get_mapargs(mapmode, 'lhs'))
 | |
|     end)
 | |
|   end
 | |
| 
 | |
|   for _, mapmode in ipairs(mapmodes) do
 | |
|     it('can set/unset noremap mappings using mapmode '..mapmode, function()
 | |
|       meths.set_keymap(mapmode, 'lhs', 'rhs', {noremap = true})
 | |
|       eq(generate_mapargs(mapmode, 'lhs', 'rhs', {noremap = true}),
 | |
|          get_mapargs(mapmode, 'lhs'))
 | |
| 
 | |
|       meths.del_keymap(mapmode, 'lhs')
 | |
|       eq({}, get_mapargs(mapmode, 'lhs'))
 | |
|     end)
 | |
|   end
 | |
| 
 | |
|   -- Test map-arguments, using optnames from above
 | |
|   -- remove some map arguments that are harder to test, or were already tested
 | |
|   optnames = {'nowait', 'silent', 'expr', 'noremap'}
 | |
|   for _, mapmode in ipairs(mapmodes) do
 | |
|     local printable_mode = normalize_mapmode(mapmode)
 | |
| 
 | |
|     -- Test with single mappings
 | |
|     for _, maparg in ipairs(optnames) do
 | |
|       it('can set/unset '..printable_mode..'-mappings with maparg: '..maparg,
 | |
|           function()
 | |
|         meths.set_keymap(mapmode, 'lhs', 'rhs', {[maparg] = true})
 | |
|         eq(generate_mapargs(mapmode, 'lhs', 'rhs', {[maparg] = true}),
 | |
|            get_mapargs(mapmode, 'lhs'))
 | |
|         meths.del_keymap(mapmode, 'lhs')
 | |
|         eq({}, get_mapargs(mapmode, 'lhs'))
 | |
|       end)
 | |
|       it ('can set/unset '..printable_mode..'-mode mappings with maparg '..
 | |
|           maparg..', whose value is false', function()
 | |
|         meths.set_keymap(mapmode, 'lhs', 'rhs', {[maparg] = false})
 | |
|         eq(generate_mapargs(mapmode, 'lhs', 'rhs'),
 | |
|            get_mapargs(mapmode, 'lhs'))
 | |
|         meths.del_keymap(mapmode, 'lhs')
 | |
|         eq({}, get_mapargs(mapmode, 'lhs'))
 | |
|       end)
 | |
|     end
 | |
| 
 | |
|     -- Test with triplets of mappings, one of which is false
 | |
|     for i = 1, (#optnames - 2) do
 | |
|       local opt1, opt2, opt3 = optnames[i], optnames[i + 1], optnames[i + 2]
 | |
|       it('can set/unset '..printable_mode..'-mode mappings with mapargs '..
 | |
|           opt1..', '..opt2..', '..opt3, function()
 | |
|         local opts = {[opt1] = true, [opt2] = false, [opt3] = true}
 | |
|         meths.set_keymap(mapmode, 'lhs', 'rhs', opts)
 | |
|         eq(generate_mapargs(mapmode, 'lhs', 'rhs', opts),
 | |
|            get_mapargs(mapmode, 'lhs'))
 | |
|         meths.del_keymap(mapmode, 'lhs')
 | |
|         eq({}, get_mapargs(mapmode, 'lhs'))
 | |
|       end)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   it('can make lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_set_keymap ('n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|   end)
 | |
| 
 | |
|   it (':map command shows lua mapping correctly', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_set_keymap ('n', 'asdf', '', {callback = function() print('jkl;') end })
 | |
|     ]]
 | |
|     assert.truthy(string.match(exec_lua[[return vim.api.nvim_exec(':nmap asdf', true)]],
 | |
|                   "^\nn  asdf          <Lua %d+>"))
 | |
|   end)
 | |
| 
 | |
|   it ('mapcheck() returns lua mapping correctly', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_set_keymap ('n', 'asdf', '', {callback = function() print('jkl;') end })
 | |
|     ]]
 | |
|     assert.truthy(string.match(funcs.mapcheck('asdf', 'n'),
 | |
|                   "^<Lua %d+>"))
 | |
|   end)
 | |
| 
 | |
|   it ('maparg() returns lua mapping correctly', function()
 | |
|     eq(0, exec_lua([[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_set_keymap('n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]]))
 | |
| 
 | |
|     assert.truthy(string.match(funcs.maparg('asdf', 'n'), "^<Lua %d+>"))
 | |
| 
 | |
|     local mapargs = funcs.maparg('asdf', 'n', false, true)
 | |
|     mapargs.callback = nil
 | |
|     mapargs.lhsraw = nil
 | |
|     mapargs.lhsrawalt = nil
 | |
|     eq(generate_mapargs('n', 'asdf', nil, {sid=sid_lua}), mapargs)
 | |
| 
 | |
|     eq(1, exec_lua([[
 | |
|       vim.fn.maparg('asdf', 'n', false, true).callback()
 | |
|       return GlobalCount
 | |
|     ]]))
 | |
| 
 | |
|     exec([[
 | |
|       call maparg('asdf', 'n', v:false, v:true).callback()
 | |
|     ]])
 | |
|     eq(2, exec_lua([[return GlobalCount]]))
 | |
|   end)
 | |
| 
 | |
|   it('can make lua expr mappings replacing keycodes', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_set_keymap ('n', 'aa', '', {callback = function() return '<Insert>π<C-V><M-π>foo<lt><Esc>' end, expr = true, replace_keycodes = true })
 | |
|     ]]
 | |
| 
 | |
|     feed('aa')
 | |
| 
 | |
|     eq({'π<M-π>foo<'}, meths.buf_get_lines(0, 0, -1, false))
 | |
|   end)
 | |
| 
 | |
|   it('can make lua expr mappings without replacing keycodes', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_set_keymap ('i', 'aa', '', {callback = function() return '<space>' end, expr = true })
 | |
|     ]]
 | |
| 
 | |
|     feed('iaa<esc>')
 | |
| 
 | |
|     eq({'<space>'}, meths.buf_get_lines(0, 0, -1, false))
 | |
|   end)
 | |
| 
 | |
|   it('lua expr mapping returning nil is equivalent to returning an empty string', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_set_keymap ('i', 'aa', '', {callback = function() return nil end, expr = true })
 | |
|     ]]
 | |
| 
 | |
|     feed('iaa<esc>')
 | |
| 
 | |
|     eq({''}, meths.buf_get_lines(0, 0, -1, false))
 | |
|   end)
 | |
| 
 | |
|   it('does not reset pum in lua mapping', function()
 | |
|     eq(0, exec_lua [[
 | |
|       VisibleCount = 0
 | |
|       vim.api.nvim_set_keymap ('i', '<F2>', '', {callback = function() VisibleCount = VisibleCount + vim.fn.pumvisible() end})
 | |
|       return VisibleCount
 | |
|     ]])
 | |
|     feed('i<C-X><C-V><F2><F2><esc>')
 | |
|     eq(2, exec_lua[[return VisibleCount]])
 | |
|   end)
 | |
| 
 | |
|   it('can overwrite lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_set_keymap ('n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_set_keymap ('n', 'asdf', '', {callback = function() GlobalCount = GlobalCount - 1 end })
 | |
|     ]]
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(0, exec_lua[[return GlobalCount]])
 | |
|   end)
 | |
| 
 | |
|   it('can unmap lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_set_keymap ('n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_del_keymap('n', 'asdf' )
 | |
|     ]]
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
|     eq('\nNo mapping found', helpers.exec_capture('nmap asdf'))
 | |
|   end)
 | |
| 
 | |
|   it('no double-free when unmapping simplifiable lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_set_keymap('n', '<C-I>', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('<C-I>\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_del_keymap('n', '<C-I>')
 | |
|     ]]
 | |
| 
 | |
|     feed('<C-I>\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
|     eq('\nNo mapping found', helpers.exec_capture('nmap <C-I>'))
 | |
|   end)
 | |
| 
 | |
|   it('can set descriptions on mappings', function()
 | |
|     meths.set_keymap('n', 'lhs', 'rhs', {desc="map description"})
 | |
|     eq(generate_mapargs('n', 'lhs', 'rhs', {desc="map description"}), get_mapargs('n', 'lhs'))
 | |
|     eq("\nn  lhs           rhs\n                 map description",
 | |
|        helpers.exec_capture("nmap lhs"))
 | |
|   end)
 | |
| end)
 | |
| 
 | |
| describe('nvim_buf_set_keymap, nvim_buf_del_keymap', function()
 | |
|   before_each(clear)
 | |
| 
 | |
|   -- nvim_set_keymap is implemented as a wrapped call to nvim_buf_set_keymap,
 | |
|   -- so its tests also effectively test nvim_buf_set_keymap
 | |
| 
 | |
|   -- here, we mainly test for buffer specificity and other special cases
 | |
| 
 | |
|   -- switch to the given buffer, abandoning any changes in the current buffer
 | |
|   local function switch_to_buf(bufnr)
 | |
|     command(bufnr..'buffer!')
 | |
|   end
 | |
| 
 | |
|   -- `set hidden`, then create two buffers and return their bufnr's
 | |
|   -- If start_from_first is truthy, the first buffer will be open when
 | |
|   -- the function returns; if falsy, the second buffer will be open.
 | |
|   local function make_two_buffers(start_from_first)
 | |
|     command('set hidden')
 | |
| 
 | |
|     local first_buf = meths.call_function('bufnr', {'%'})
 | |
|     command('new')
 | |
|     local second_buf = meths.call_function('bufnr', {'%'})
 | |
|     neq(second_buf, first_buf)  -- sanity check
 | |
| 
 | |
|     if start_from_first then
 | |
|       switch_to_buf(first_buf)
 | |
|     end
 | |
| 
 | |
|     return first_buf, second_buf
 | |
|   end
 | |
| 
 | |
|   it('rejects negative bufnr values', function()
 | |
|     eq('Wrong type for argument 1 when calling nvim_buf_set_keymap, expecting Buffer',
 | |
|       pcall_err(bufmeths.set_keymap, -1, '', 'lhs', 'rhs', {}))
 | |
|   end)
 | |
| 
 | |
|   it('can set mappings active in the current buffer but not others', function()
 | |
|     local first, second = make_two_buffers(true)
 | |
| 
 | |
|     bufmeths.set_keymap(0, '', 'lhs', 'irhs<Esc>', {})
 | |
|     command('normal lhs')
 | |
|     eq({'rhs'}, bufmeths.get_lines(0, 0, 1, 1))
 | |
| 
 | |
|     -- mapping should have no effect in new buffer
 | |
|     switch_to_buf(second)
 | |
|     command('normal lhs')
 | |
|     eq({''}, bufmeths.get_lines(0, 0, 1, 1))
 | |
| 
 | |
|     -- mapping should remain active in old buffer
 | |
|     switch_to_buf(first)
 | |
|     command('normal ^lhs')
 | |
|     eq({'rhsrhs'}, bufmeths.get_lines(0, 0, 1, 1))
 | |
|   end)
 | |
| 
 | |
|   it('can set local mappings in buffer other than current', function()
 | |
|     local first = make_two_buffers(false)
 | |
|     bufmeths.set_keymap(first, '', 'lhs', 'irhs<Esc>', {})
 | |
| 
 | |
|     -- shouldn't do anything
 | |
|     command('normal lhs')
 | |
|     eq({''}, bufmeths.get_lines(0, 0, 1, 1))
 | |
| 
 | |
|     -- should take effect
 | |
|     switch_to_buf(first)
 | |
|     command('normal lhs')
 | |
|     eq({'rhs'}, bufmeths.get_lines(0, 0, 1, 1))
 | |
|   end)
 | |
| 
 | |
|   it('can disable mappings made in another buffer, inside that buffer', function()
 | |
|     local first = make_two_buffers(false)
 | |
|     bufmeths.set_keymap(first, '', 'lhs', 'irhs<Esc>', {})
 | |
|     bufmeths.del_keymap(first, '', 'lhs')
 | |
|     switch_to_buf(first)
 | |
| 
 | |
|     -- shouldn't do anything
 | |
|     command('normal lhs')
 | |
|     eq({''}, bufmeths.get_lines(0, 0, 1, 1))
 | |
|   end)
 | |
| 
 | |
|   it("can't disable mappings given wrong buffer handle", function()
 | |
|     local first, second = make_two_buffers(false)
 | |
|     bufmeths.set_keymap(first, '', 'lhs', 'irhs<Esc>', {})
 | |
|     eq('E31: No such mapping',
 | |
|       pcall_err(bufmeths.del_keymap, second, '', 'lhs'))
 | |
| 
 | |
|     -- should still work
 | |
|     switch_to_buf(first)
 | |
|     command('normal lhs')
 | |
|     eq({'rhs'}, bufmeths.get_lines(0, 0, 1, 1))
 | |
|   end)
 | |
| 
 | |
|   it("does not crash when setting keymap in a non-existing buffer #13541", function()
 | |
|     pcall_err(bufmeths.set_keymap, 100, '', 'lsh', 'irhs<Esc>', {})
 | |
|     helpers.assert_alive()
 | |
|   end)
 | |
| 
 | |
|   it('can make lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_buf_set_keymap (0, 'n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
|   end)
 | |
| 
 | |
|   it('can make lua expr mappings replacing keycodes', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_buf_set_keymap (0, 'n', 'aa', '', {callback = function() return '<Insert>π<C-V><M-π>foo<lt><Esc>' end, expr = true, replace_keycodes = true })
 | |
|     ]]
 | |
| 
 | |
|     feed('aa')
 | |
| 
 | |
|     eq({'π<M-π>foo<'}, meths.buf_get_lines(0, 0, -1, false))
 | |
|   end)
 | |
| 
 | |
|   it('can make lua expr mappings without replacing keycodes', function()
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_buf_set_keymap (0, 'i', 'aa', '', {callback = function() return '<space>' end, expr = true })
 | |
|     ]]
 | |
| 
 | |
|     feed('iaa<esc>')
 | |
| 
 | |
|     eq({'<space>'}, meths.buf_get_lines(0, 0, -1, false))
 | |
|   end)
 | |
| 
 | |
| 
 | |
|   it('can overwrite lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_buf_set_keymap (0, 'n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_buf_set_keymap (0, 'n', 'asdf', '', {callback = function() GlobalCount = GlobalCount - 1 end })
 | |
|     ]]
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(0, exec_lua[[return GlobalCount]])
 | |
|   end)
 | |
| 
 | |
|   it('can unmap lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_buf_set_keymap (0, 'n', 'asdf', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_buf_del_keymap(0, 'n', 'asdf' )
 | |
|     ]]
 | |
| 
 | |
|     feed('asdf\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
|     eq('\nNo mapping found', helpers.exec_capture('nmap asdf'))
 | |
|   end)
 | |
| 
 | |
|   it('no double-free when unmapping simplifiable lua mappings', function()
 | |
|     eq(0, exec_lua [[
 | |
|       GlobalCount = 0
 | |
|       vim.api.nvim_buf_set_keymap(0, 'n', '<C-I>', '', {callback = function() GlobalCount = GlobalCount + 1 end })
 | |
|       return GlobalCount
 | |
|     ]])
 | |
| 
 | |
|     feed('<C-I>\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
| 
 | |
|     exec_lua [[
 | |
|       vim.api.nvim_buf_del_keymap(0, 'n', '<C-I>')
 | |
|     ]]
 | |
| 
 | |
|     feed('<C-I>\n')
 | |
| 
 | |
|     eq(1, exec_lua[[return GlobalCount]])
 | |
|     eq('\nNo mapping found', helpers.exec_capture('nmap <C-I>'))
 | |
|   end)
 | |
| end)
 |