mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-20 16:51:48 +00:00 
			
		
		
		
	 ac12dc49cc
			
		
	
	ac12dc49cc
	
	
	
		
			
			Problem:
With these settings, copy/pasting `blockwise-visual` (with `CTRL+V`)
incorrectly pastes as a `linewise` mode because `regtype` is ignored:
    vim.opt.clipboard = 'unnamedplus'
    vim.g.clipboard = 'osc52'
To reproduce: press `CTRL+V` and select some characters press `p` and
observe that it is pasted in `linewise` mode.
Solution:
Enable the [clipboard.vim](https://github.com/neovim/neovim/blob/master/runtime/autoload/provider/clipboard.vim#L281-L283))
cache for function providers, so that `regtype` is maintained for the OSC52
clipboard provider.
		
	
		
			
				
	
	
		
			358 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			VimL
		
	
	
	
	
	
			
		
		
	
	
			358 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			VimL
		
	
	
	
	
	
| " The clipboard provider uses shell commands to communicate with the clipboard.
 | |
| " The provider function will only be registered if a supported command is
 | |
| " available.
 | |
| 
 | |
| if exists('g:loaded_clipboard_provider')
 | |
|   finish
 | |
| endif
 | |
| " Default to 0.  provider#clipboard#Executable() may set 2.
 | |
| " To force a reload:
 | |
| "   :unlet g:loaded_clipboard_provider
 | |
| "   :runtime autoload/provider/clipboard.vim
 | |
| let g:loaded_clipboard_provider = 0
 | |
| 
 | |
| let s:copy = {}
 | |
| let s:paste = {}
 | |
| let s:clipboard = {}
 | |
| 
 | |
| " When caching is enabled, store the jobid of the xclip/xsel process keeping
 | |
| " ownership of the selection, so we know how long the cache is valid.
 | |
| let s:selection = { 'owner': 0, 'data': [], 'stderr_buffered': v:true }
 | |
| 
 | |
| function! s:selection.on_exit(jobid, data, event) abort
 | |
|   " At this point this nvim instance might already have launched
 | |
|   " a new provider instance. Don't drop ownership in this case.
 | |
|   if self.owner == a:jobid
 | |
|     let self.owner = 0
 | |
|   endif
 | |
|   " Don't print if exit code is >= 128 ( exit is 128+SIGNUM if by signal (e.g. 143 on SIGTERM))
 | |
|   if a:data > 0 && a:data < 128
 | |
|     echohl WarningMsg
 | |
|     echomsg 'clipboard: error invoking '.get(self.argv, 0, '?').': '.join(self.stderr)
 | |
|     echohl None
 | |
|   endif
 | |
| endfunction
 | |
| 
 | |
| let s:selections = { '*': s:selection, '+': copy(s:selection) }
 | |
| 
 | |
| function! s:try_cmd(cmd, ...) abort
 | |
|   let out = systemlist(a:cmd, (a:0 ? a:1 : ['']), 1)
 | |
|   if v:shell_error
 | |
|     if !exists('s:did_error_try_cmd')
 | |
|       echohl WarningMsg
 | |
|       echomsg "clipboard: error: ".(len(out) ? out[0] : v:shell_error)
 | |
|       echohl None
 | |
|       let s:did_error_try_cmd = 1
 | |
|     endif
 | |
|     return 0
 | |
|   endif
 | |
|   return out
 | |
| endfunction
 | |
| 
 | |
| " Returns TRUE if `cmd` exits with success, else FALSE.
 | |
| function! s:cmd_ok(cmd) abort
 | |
|   call system(a:cmd)
 | |
|   return v:shell_error == 0
 | |
| endfunction
 | |
| 
 | |
| function! s:split_cmd(cmd) abort
 | |
|   return (type(a:cmd) == v:t_string) ? split(a:cmd, " ") : a:cmd
 | |
| endfunction
 | |
| 
 | |
| function! s:set_osc52() abort
 | |
|   let s:copy['+'] = v:lua.require'vim.ui.clipboard.osc52'.copy('+')
 | |
|   let s:copy['*'] = v:lua.require'vim.ui.clipboard.osc52'.copy('*')
 | |
|   let s:paste['+'] = v:lua.require'vim.ui.clipboard.osc52'.paste('+')
 | |
|   let s:paste['*'] = v:lua.require'vim.ui.clipboard.osc52'.paste('*')
 | |
|   return 'OSC 52'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_pbcopy() abort
 | |
|   let s:copy['+'] = ['pbcopy']
 | |
|   let s:paste['+'] = ['pbpaste']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   let s:cache_enabled = 0
 | |
|   return 'pbcopy'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_wayland() abort
 | |
|   let s:copy['+'] = ['wl-copy', '--type', 'text/plain']
 | |
|   let s:paste['+'] = ['wl-paste', '--no-newline']
 | |
|   let s:copy['*'] = ['wl-copy', '--primary', '--type', 'text/plain']
 | |
|   let s:paste['*'] = ['wl-paste', '--no-newline', '--primary']
 | |
|   return 'wl-copy'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_wayclip() abort
 | |
|   let s:copy['+'] = ['waycopy']
 | |
|   let s:paste['+'] = ['waypaste']
 | |
|   let s:copy['*'] = ['waycopy', '-p']
 | |
|   let s:paste['*'] = ['waypaste', '-p']
 | |
|   return 'wayclip'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_xsel() abort
 | |
|   let s:copy['+'] = ['xsel', '--nodetach', '-i', '-b']
 | |
|   let s:paste['+'] = ['xsel', '-o', '-b']
 | |
|   let s:copy['*'] = ['xsel', '--nodetach', '-i', '-p']
 | |
|   let s:paste['*'] = ['xsel', '-o', '-p']
 | |
|   return 'xsel'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_xclip() abort
 | |
|   let s:copy['+'] = ['xclip', '-quiet', '-i', '-selection', 'clipboard']
 | |
|   let s:paste['+'] = ['xclip', '-o', '-selection', 'clipboard']
 | |
|   let s:copy['*'] = ['xclip', '-quiet', '-i', '-selection', 'primary']
 | |
|   let s:paste['*'] = ['xclip', '-o', '-selection', 'primary']
 | |
|   return 'xclip'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_lemonade() abort
 | |
|   let s:copy['+'] = ['lemonade', 'copy']
 | |
|   let s:paste['+'] = ['lemonade', 'paste']
 | |
|   let s:copy['*'] = ['lemonade', 'copy']
 | |
|   let s:paste['*'] = ['lemonade', 'paste']
 | |
|   return 'lemonade'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_doitclient() abort
 | |
|   let s:copy['+'] = ['doitclient', 'wclip']
 | |
|   let s:paste['+'] = ['doitclient', 'wclip', '-r']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   return 'doitclient'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_win32yank() abort
 | |
|   if has('wsl') && getftype(exepath('win32yank.exe')) == 'link'
 | |
|     let win32yank = resolve(exepath('win32yank.exe'))
 | |
|   else
 | |
|     let win32yank = 'win32yank.exe'
 | |
|   endif
 | |
|   let s:copy['+'] = [win32yank, '-i', '--crlf']
 | |
|   let s:paste['+'] = [win32yank, '-o', '--lf']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   return 'win32yank'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_putclip() abort
 | |
|   let s:copy['+'] = ['putclip']
 | |
|   let s:paste['+'] = ['getclip']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   return 'putclip'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_clip() abort
 | |
|   let s:copy['+'] = ['clip']
 | |
|   let s:paste['+'] = ['powershell', '-NoProfile', '-NoLogo', '-Command', 'Get-Clipboard']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   return 'clip'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_termux() abort
 | |
|   let s:copy['+'] = ['termux-clipboard-set']
 | |
|   let s:paste['+'] = ['termux-clipboard-get']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   return 'termux-clipboard'
 | |
| endfunction
 | |
| 
 | |
| function! s:set_tmux() abort
 | |
|   let tmux_v = v:lua.vim.version.parse(system(['tmux', '-V']))
 | |
|   if !empty(tmux_v) && !v:lua.vim.version.lt(tmux_v, [3,2,0])
 | |
|     let s:copy['+'] = ['tmux', 'load-buffer', '-w', '-']
 | |
|   else
 | |
|     let s:copy['+'] = ['tmux', 'load-buffer', '-']
 | |
|   endif
 | |
|   let s:paste['+'] = ['tmux', 'save-buffer', '-']
 | |
|   let s:copy['*'] = s:copy['+']
 | |
|   let s:paste['*'] = s:paste['+']
 | |
|   return 'tmux'
 | |
| endfunction
 | |
| 
 | |
| let s:cache_enabled = 1
 | |
| let s:err = ''
 | |
| 
 | |
| function! provider#clipboard#Error() abort
 | |
|   return s:err
 | |
| endfunction
 | |
| 
 | |
| function! provider#clipboard#Executable() abort
 | |
|   " Setting g:clipboard to v:false explicitly opts-in to using the "builtin" clipboard providers below
 | |
|   if exists('g:clipboard') && g:clipboard isnot# v:false
 | |
|     if v:t_string ==# type(g:clipboard)
 | |
|       " Handle string form of g:clipboard for all builtin providers
 | |
|       if 'osc52' == g:clipboard
 | |
|         " User opted-in to OSC 52 by manually setting g:clipboard.
 | |
|         return s:set_osc52()
 | |
|       elseif 'pbcopy' == g:clipboard
 | |
|         return s:set_pbcopy()
 | |
|       elseif 'wl-copy' == g:clipboard
 | |
|         return s:set_wayland()
 | |
|       elseif 'wayclip' == g:clipboard
 | |
|         return s:set_wayclip()
 | |
|       elseif 'xsel' == g:clipboard
 | |
|         return s:set_xsel()
 | |
|       elseif 'xclip' == g:clipboard
 | |
|         return s:set_xclip()
 | |
|       elseif 'lemonade' == g:clipboard
 | |
|         return s:set_lemonade()
 | |
|       elseif 'doitclient' == g:clipboard
 | |
|         return s:set_doitclient()
 | |
|       elseif 'win32yank' == g:clipboard
 | |
|         return s:set_win32yank()
 | |
|       elseif 'putclip' == g:clipboard
 | |
|         return s:set_putclip()
 | |
|       elseif 'clip' == g:clipboard
 | |
|         return s:set_clip()
 | |
|       elseif 'termux' == g:clipboard
 | |
|         return s:set_termux()
 | |
|       elseif 'tmux' == g:clipboard
 | |
|         return s:set_tmux()
 | |
|       endif
 | |
|     endif
 | |
| 
 | |
|     if type({}) isnot# type(g:clipboard)
 | |
|           \ || type({}) isnot# type(get(g:clipboard, 'copy', v:null))
 | |
|           \ || type({}) isnot# type(get(g:clipboard, 'paste', v:null))
 | |
|       let s:err = 'clipboard: invalid g:clipboard'
 | |
|       return ''
 | |
|     endif
 | |
| 
 | |
|     let s:copy = {}
 | |
|     let s:copy['+'] = s:split_cmd(get(g:clipboard.copy, '+', v:null))
 | |
|     let s:copy['*'] = s:split_cmd(get(g:clipboard.copy, '*', v:null))
 | |
| 
 | |
|     let s:paste = {}
 | |
|     let s:paste['+'] = s:split_cmd(get(g:clipboard.paste, '+', v:null))
 | |
|     let s:paste['*'] = s:split_cmd(get(g:clipboard.paste, '*', v:null))
 | |
| 
 | |
|     let s:cache_enabled = get(g:clipboard, 'cache_enabled', 0)
 | |
|     return get(g:clipboard, 'name', 'g:clipboard')
 | |
|   elseif has('mac')
 | |
|     return s:set_pbcopy()
 | |
|   elseif !empty($WAYLAND_DISPLAY) && executable('wl-copy') && executable('wl-paste')
 | |
|     return s:set_wayland()
 | |
|   elseif !empty($WAYLAND_DISPLAY) && executable('waycopy') && executable('waypaste')
 | |
|     return s:set_wayclip()
 | |
|   elseif !empty($DISPLAY) && executable('xsel') && s:cmd_ok('xsel -o -b')
 | |
|     return s:set_xsel()
 | |
|   elseif !empty($DISPLAY) && executable('xclip')
 | |
|     return s:set_xclip()
 | |
|   elseif executable('lemonade')
 | |
|     return s:set_lemonade()
 | |
|   elseif executable('doitclient')
 | |
|     return s:set_doitclient()
 | |
|   elseif executable('win32yank.exe')
 | |
|     return s:set_win32yank()
 | |
|   elseif executable('putclip') && executable('getclip')
 | |
|     return s:set_putclip()
 | |
|   elseif executable('clip') && executable('powershell')
 | |
|     return s:set_clip()
 | |
|   elseif executable('termux-clipboard-set')
 | |
|     return s:set_termux()
 | |
|   elseif executable('tmux') && (!empty($TMUX) || 0 == jobwait([jobstart(['tmux', 'list-buffers'])], 2000)[0])
 | |
|     return s:set_tmux()
 | |
|   elseif get(get(g:, 'termfeatures', {}), 'osc52') && &clipboard ==# ''
 | |
|     " Don't use OSC 52 when 'clipboard' is set. It can be slow and cause a lot
 | |
|     " of user prompts. Users can opt-in to it by setting g:clipboard manually.
 | |
|     return s:set_osc52()
 | |
|   endif
 | |
| 
 | |
|   let s:err = 'clipboard: No clipboard tool. :help clipboard'
 | |
|   return ''
 | |
| endfunction
 | |
| 
 | |
| function! s:clipboard.get(reg) abort
 | |
|   if s:selections[a:reg].owner > 0
 | |
|     return s:selections[a:reg].data
 | |
|   end
 | |
| 
 | |
|   let clipboard_data = type(s:paste[a:reg]) == v:t_func ? s:paste[a:reg]() : s:try_cmd(s:paste[a:reg])
 | |
|   if match(&clipboard, '\v(unnamed|unnamedplus)') >= 0
 | |
|         \ && type(clipboard_data) == v:t_list
 | |
|         \ && get(s:selections[a:reg].data, 0, []) ==# clipboard_data
 | |
|     " When system clipboard return is same as our cache return the cache
 | |
|     " as it contains regtype information
 | |
|     return s:selections[a:reg].data
 | |
|   end
 | |
|   return clipboard_data
 | |
| endfunction
 | |
| 
 | |
| function! s:clipboard.set(lines, regtype, reg) abort
 | |
|   if a:reg == '"'
 | |
|     call s:clipboard.set(a:lines,a:regtype,'+')
 | |
|     if s:copy['*'] != s:copy['+']
 | |
|       call s:clipboard.set(a:lines,a:regtype,'*')
 | |
|     end
 | |
|     return 0
 | |
|   end
 | |
| 
 | |
|   if s:cache_enabled == 0 || type(s:copy[a:reg]) == v:t_func
 | |
|     if type(s:copy[a:reg]) == v:t_func
 | |
|       call s:copy[a:reg](a:lines, a:regtype)
 | |
|     else
 | |
|       call s:try_cmd(s:copy[a:reg], a:lines)
 | |
|     endif
 | |
|     "Cache it anyway we can compare it later to get regtype of the yank
 | |
|     let s:selections[a:reg] = copy(s:selection)
 | |
|     let s:selections[a:reg].data = [a:lines, a:regtype]
 | |
|     return 0
 | |
|   end
 | |
| 
 | |
|   if s:selections[a:reg].owner > 0
 | |
|     let prev_job = s:selections[a:reg].owner
 | |
|   end
 | |
|   let s:selections[a:reg] = copy(s:selection)
 | |
|   let selection = s:selections[a:reg]
 | |
|   let selection.data = [a:lines, a:regtype]
 | |
|   let selection.argv = s:copy[a:reg]
 | |
|   let selection.detach = s:cache_enabled
 | |
|   let selection.cwd = "/"
 | |
|   let jobid = jobstart(selection.argv, selection)
 | |
|   if jobid > 0
 | |
|     call jobsend(jobid, a:lines)
 | |
|     call jobclose(jobid, 'stdin')
 | |
|     " xclip does not close stdout when receiving input via stdin
 | |
|     if selection.argv[0] ==# 'xclip'
 | |
|       call jobclose(jobid, 'stdout')
 | |
|     endif
 | |
|     let selection.owner = jobid
 | |
|     let ret = 1
 | |
|   else
 | |
|     echohl WarningMsg
 | |
|     echomsg 'clipboard: failed to execute: '.(s:copy[a:reg])
 | |
|     echohl None
 | |
|     let ret = 1
 | |
|   endif
 | |
| 
 | |
|   " The previous provider instance should exit when the new one takes
 | |
|   " ownership, but kill it to be sure we don't fill up the job table.
 | |
|   if exists('prev_job')
 | |
|     call timer_start(1000, {... ->
 | |
|           \ jobwait([prev_job], 0)[0] == -1
 | |
|           \ && jobstop(prev_job)})
 | |
|   endif
 | |
| 
 | |
|   return ret
 | |
| endfunction
 | |
| 
 | |
| function! provider#clipboard#Call(method, args) abort
 | |
|   if get(s:, 'here', v:false)  " Clipboard provider must not recurse. #7184
 | |
|     return 0
 | |
|   endif
 | |
|   let s:here = v:true
 | |
|   try
 | |
|     return call(s:clipboard[a:method],a:args,s:clipboard)
 | |
|   finally
 | |
|     let s:here = v:false
 | |
|   endtry
 | |
| endfunction
 | |
| 
 | |
| " eval_has_provider() decides based on this variable.
 | |
| let g:loaded_clipboard_provider = empty(provider#clipboard#Executable()) ? 0 : 2
 |