Como substituir e colar texto arrancado no vim sem puxar as linhas excluídas?

39

Por isso, geralmente me pego copiando texto de um ponto para outro enquanto sobrescrito texto antigo onde o novo é colado:

blah1
newtext
blah2
wrong1
blah3
wrong2
blah4

Suponha que eu marque visualmente newtexte yank. Agora eu seleciono wrong1(que pode ser qualquer coisa, não necessariamente apenas uma palavra) e paste o newtext. No entanto, se eu fizer o mesmo agora, wrong2ele será substituído por em wrong1vez de newtext.

Então, como evito que o texto que está no buffer seja trocado pelo texto que estou substituindo no momento?

Editar 1

Embora eu goste das sugestões de re-registro (acho que vou começar a usar mais os registradores, agora que descobri o :discomando), vou fazer uma modificação na resposta de jinfield , porque não uso o modo de troca.

vnoremap p "0p
vnoremap P "0P
vnoremap y "0y
vnoremap d "0d

faz o truque perfeitamente.

Editar 2

Eu fui muito rápido; A solução da romainl é exatamente o que eu estava procurando, sem o hack do Edit 1 .
Na verdade, vnoremap p "_dPé o suficiente!
Então, alterando a resposta aceita.

bitmask
fonte
2
Hey fyi, eu usei isso há muito tempo vnoremap p "_dP mape notei que ele não funciona bem para a última palavra / caractere em uma linha. Eu tenho de volta foi para o vnoremap p "0p, vnoremap P "0Pe set clipboard=unnamed(para OSX)
Kache
vnoremap p "_dPremove o espaço em branco na pasta. O Edit 1 funciona perfeitamente.
Vaclav Kasal

Respostas:

21

Eu tenho esses mapeamentos no meu .vimrc:

" delete without yanking
nnoremap <leader>d "_d
vnoremap <leader>d "_d

" replace currently selected text with default register
" without yanking it
vnoremap <leader>p "_dP

"_é o "registro buraco negro", de acordo com :help "_:

"Ao escrever neste registro, nada acontece. Isso pode ser usado para excluir texto sem afetar os registros normais. Ao ler esse registro, nada é retornado. {Não no Vi}"

romainl
fonte
1
Eu usei isso vnoremap p "_dP mape notei que ele não funciona bem para a última palavra / caractere em uma linha. Eu tenho de volta foi para o vnoremap p "0p, vnoremap P "0Pe set clipboard=unnamed(para OSX)
Kache
vnoremap p "_dPparar de trabalhar para mim no modo de seleção, mas vnoremap <leader>p "_dPfunciona
whitesiroi
1
talvez <leader>seja um espaço reservado que devo substituir por algo, mas isso só funcionou para mim se eu o removesse. O que isso significa aqui?
Hashbrown #
1
@Hashbrown :help mapleader,.
Romainl # 9/16
Advertência pequena: o vnoremap <leader>pmapeamento não funciona corretamente na última linha do buffer, porque assim que você exclui a última linha do buffer, você está na linha uma vez após o último e Pcolará acima dessa linha, em vez de abaixo.
Kal
13

Além do buffer padrão, você pode puxar o texto em buffers nomeados e, em seguida, colocar a partir desses buffers nomeados. Você pode usar até 26 buffers nomeados (um para cada letra). Use aspas duplas e uma letra para acessar um buffer nomeado. Exemplos:

"dyy - Arranque a linha atual no buffer d.
"a7yy- Arranque as próximas sete linhas no buffer a.
"dP- Coloque o conteúdo do buffer d antes do cursor.
"ap- Coloque o conteúdo do buffer depois do cursor

Outra coisa interessante, se você usar uma letra maiúscula em vez de minúscula, ou seja, "Dyya linha atual será anexada ao buffer d em vez de substituí-la. Mais detalhes no livro O`Reilly: http://docstore.mik.ua/orelly/unix/vi/ch04_03.htm

Tradsud
fonte
3
Coisa muito legal. Eu sabia sobre buffers, mas não os conectei com esse problema. Ainda é complicado para "atudo, mas tudo bem.
bitmask
4

Ao usar putno modo visual, o texto que você está substituindo wrong1é substituído pelo conteúdo do registro 'sem nome'.

Na verdade, isso funciona 'colocando' o registro após a seleção e excluindo a seleção. O problema é que essa exclusão agora está armazenada no unnamedregistro e será usada para a próxima putação.

A solução, de acordo com :h v_p, é puxar para um registro nomeado, como "0ycolar e colar o "0ptempo que você precisar. Pode ser útil mapear <leader>ye <leader>pusar um registro nomeado, se isso é algo que você faz com freqüência.

:map <leader>y "0y
:map <leader>p "0p

para obter mais ajuda, consulte:

:help v_p
:help map
jinfield
fonte
Essa solução parece mais útil, até que algo inteligente apareça no próprio vim.
Yugal Jindle
4

"0É importante saber colar do registro, mas muitas vezes você deseja substituir muitas vezes. Se você fizer uma ação repetível, poderá usar o .operador, conforme mencionado por garyjohn. É explicado no wiki do vim :

yiw     yank inner word (copy word under cursor, say "first". Same as above).
...     Move the cursor to another word (say "second").
ciw<C-r>0   select "second", then replace it with "first" If you are at the start of the word then cw<C-r>0 is sufficient.
...     Move the cursor to another word (say "third").
.   select "third", then replace it with "first". 
Jerph
fonte
3

Quando você puxa o texto para o registro sem nome *, uma cópia também é colocada no registro 0. Cada vez que você substitui o texto selecionado, basta colar do registro 0. Vejo

:help registers

Além disso, se você estiver substituindo um número de palavras pela mesma palavra, basta ir para o início da palavra a ser substituída e digitar .. Isso repetirá a última operação de edição. Vejo

:help single-repeat

* Os locais de armazenamento nos quais você puxa e coloca são chamados de registradores. Um buffer é o que você edita, geralmente uma cópia de um arquivo do disco.

garyjohn
fonte
1

Eu preciso disso com tanta frequência que escrevi um plugin para isso: ReplaceWithRegister .

Este plug-in oferece um grcomando dois-em-um que substitui o texto coberto por {motion}, linha (s) inteira (s) ou a seleção atual pelo conteúdo de um registro; o texto antigo é excluído no registro do buraco negro, ou seja, foi removido.

Ingo Karkat
fonte
1

Como algo como vnoremap p "_dP(eu também tentei xou c) tem problemas com o início e o fim da linha, acabei fazendo o seguinte: o vnoremap p :<C-U>let @p = @+<CR>gvp:let @+ = @p<CR>que achei mais simples do que os plugins existentes (que também não funcionavam set clipboard=unnamedplusde fábrica). Então, o que ele faz é:

  • alternar para o modo de comando
  • remover alcance ( C-U)
  • +registro de backup (devido ao unnamedplus, as alternativas são "e *dependendo da sua configuração) parap
  • recuperar seleção e colar
  • alternar para o modo de comando novamente
  • recuperar registro
Sebastian Blask
fonte
Perfeito! Esta é a primeira dessas opções que funcionou exatamente como o esperado para mim. Obrigado!
Jamis Charles 15/04
1

Isso é o que eu uso (literalmente copiado do meu .vimrc) para o estilo do Windows Control + X cut / Control + C copy / Control + V colar / Control + S save / Control + S save / Control + F find / Control + H substituir o comportamento.

A função smartpaste () deve conter o que você procura, ou seja, uma maneira de colar sobre o texto realçado sem simultaneamente puxar o que foi selecionado.

" Windows common keyboard shortcuts and pasting behavior {{{

" Uncomment to enable debugging.
" Check debug output with :messages
let s:debug_smart_cut = 0
let s:debug_smart_copy = 0
let s:debug_smart_paste = 0

function! SmartCut()
    execute 'normal! gv"+c'

    if visualmode() != "\<C-v>" " If not Visual-Block mode
        " Trim the last \r\n | \n | \r character in the '+' buffer
        " NOTE: This messes up Visual-Block pasting.
        let @+ = substitute(@+,'\(\r\?\n\|\r\)$','','g')
    endif

    if exists("s:debug_smart_cut") && s:debug_smart_cut
        echomsg "SmartCut '+' buffer: " . @+
    endif
endfunction

function! SmartCopy()
    execute 'normal! gv"+y'

    if visualmode() != "\<C-v>" " If not Visual-Block mode
        " Trim the last \r\n | \n | \r character in the '+' buffer
        " NOTE: This messes up Visual-Block pasting.
        let @+ = substitute(@+,'\(\r\?\n\|\r\)$','','g')
    endif

    if exists("s:debug_smart_copy") && s:debug_smart_copy
        echomsg "SmartCopy '+' buffer: " . @+
    endif
endfunction

" Delete to black hole register before pasting. This function is a smarter version of "_d"+P or "_dp to handle special cases better.
" SOURCE: http://stackoverflow.com/questions/12625722/vim-toggling-buffer-overwrite-behavior-when-deleting
function! SmartPaste()
    let mode = 'gv'

    let delete = '"_d'

    let reg = '"+'

    " See :help '> for more information. Hint: if you select some text and press ':' you will see :'<,'>
    " SOURCE: http://superuser.com/questions/723621/how-can-i-check-if-the-cursor-is-at-the-end-of-a-line
    " SOURCE: http://stackoverflow.com/questions/7262536/vim-count-lines-in-selected-range
    " SOURCE: https://git.zug.fr/config/vim/blob/master/init.vim
    " SOURCE: https://git.zug.fr/config/vim/blob/master/after/plugin/zzzmappings.vim
    let currentColumn = col(".")
    let currentLine = line(".")
    let lastVisibleLetterColumn = col("$") - 1
    let lastLineOfBuffer = line("$")
    let selectionEndLine = line("'>")
    let selectionEndLineLength = len(getline(selectionEndLine))
    let nextLineLength = len(getline(currentLine + 1))
    let selectionStartColumn = col("'<")
    let selectionEndColumn = col("'>")

    " If selection does not include or go beyond the last visible character of the line (by also selecting the invisible EOL character)
    if selectionEndColumn < selectionEndLineLength
        let cmd = 'P'

        if exists("s:debug_smart_paste") && s:debug_smart_paste
            echomsg "SmartPaste special case #1"
        endif

    " If attempting to paste on a blank last line
    elseif selectionEndLineLength == 0 && selectionEndLine == lastLineOfBuffer
        let cmd = 'P'

        if exists("s:debug_smart_paste") && s:debug_smart_paste
            echomsg "SmartPaste special case #2"
        endif

    " If selection ends after the last visible character of the line (by also selecting the invisible EOL character) and next line is not blank and not the last line
    elseif selectionEndColumn > selectionEndLineLength && nextLineLength > 0 && selectionEndLine != lastLineOfBuffer
        let cmd = 'P'

        if exists("s:debug_smart_paste") && s:debug_smart_paste
            echomsg "SmartPaste special case #3"
        endif

    " If selection ends after the last visible character of the line (by also selecting the invisible EOL character), or the line is visually selected (Shift + V), and next line is the last line
    elseif selectionEndColumn > selectionEndLineLength && selectionEndLine == lastLineOfBuffer
        " SOURCE:  http://vim.wikia.com/wiki/Quickly_adding_and_deleting_empty_lines

        " Fixes bug where if the last line is fully selected (Shift + V) and a paste occurs, that the paste appears to insert after the first character of the line above it because the delete operation [which occurs before the paste]
        " is causing the caret to go up a line, and then 'p' cmd causes the paste to occur after the caret, thereby pasting after the first letter of that line.
        " However this but does not occur if there's a blank line underneath the selected line, prior to deleting it, as the cursor goes down after the delete in that situation.
        call append(selectionEndLine, "")

        let cmd = 'p'

        if exists("s:debug_smart_paste") && s:debug_smart_paste
            echomsg "SmartPaste special case #4"
        endif

    else
        let cmd = 'p'

        if exists("s:debug_smart_paste") && s:debug_smart_paste
            echomsg "SmartPaste default case"
        endif
    endif

    if exists("s:debug_smart_paste") && s:debug_smart_paste
        echomsg "SmartPaste debug info:"
        echomsg "    currentColumn: " . currentColumn
        echomsg "    currentLine: " . currentLine
        echomsg "    lastVisibleLetterColumn: " . lastVisibleLetterColumn
        echomsg "    lastLineOfBuffer: " . lastLineOfBuffer
        echomsg "    selectionEndLine: " . selectionEndLine
        echomsg "    selectionEndLineLength: " . selectionEndLineLength
        echomsg "    nextLineLength: " . nextLineLength
        echomsg "    selectionStartColumn: " . selectionStartColumn
        echomsg "    selectionEndColumn: " . selectionEndColumn
        echomsg "    cmd: " . cmd
        echo [getpos("'<")[1:2], getpos("'>")[1:2]]
        echo "visualmode(): " . visualmode()
        echo "mode(): " . mode()
    endif

    if visualmode() != "\<C-v>" " If not Visual-Block mode
        " Trim the last \r\n | \n | \r character in the '+' buffer
        " NOTE: This messes up Visual-Block pasting.
        let @+ = substitute(@+,'\(\r\?\n\|\r\)$','','g')
    endif

    try
        execute 'normal! ' . mode . delete . reg . cmd
    catch /E353:\ Nothing\ in\ register\ +/
    endtry

    " Move caret one position to right
    call cursor(0, col(".") + 1)
endfunction

" p or P delete to black hole register before pasting
" NOTE: <C-u> removes the '<,'> visual-selection from the command line. See :h c_CTRL-u
vnoremap <silent> p :<C-u>call SmartPaste()<CR>
vnoremap <silent> P :<C-u>call SmartPaste()<CR>

" MiddleMouse delete to black hole register before pasting
nnoremap <MiddleMouse> "+p " Changes default behavior from 'P' mode to 'p' mode for normal mode middle-mouse pasting
" NOTE: <C-u> removes the '<,'> visual-selection from the command line. See :h c_CTRL-u
vnoremap <silent> <MiddleMouse> :<C-u>call SmartPaste()<CR>
inoremap <MiddleMouse> <C-r><C-o>+

" Disable weird multi-click things you can do with middle mouse button
" SOURCE: http://vim.wikia.com/wiki/Mouse_wheel_for_scroll_only_-_disable_middle_button_paste
noremap <2-MiddleMouse> <Nop>
inoremap <2-MiddleMouse> <Nop>
noremap <3-MiddleMouse> <Nop>
inoremap <3-MiddleMouse> <Nop>
noremap <4-MiddleMouse> <Nop>
inoremap <4-MiddleMouse> <Nop>

if os != "mac" " NOTE: MacVim provides Command+C|X|V|A|S and undo/redo support and also can Command+C|V to the command line by default
    " SOURCE: https://opensource.apple.com/source/vim/vim-62.41.2/runtime/macmap.vim.auto.html
    " NOTE: Only copy and paste are possible in the command line from what i can tell.
    "       Their is no undo for text typed in the command line and you cannot paste text onto a selection of text to replace it.
    cnoremap <C-c> <C-y>
    cnoremap <C-v> <C-r>+
    " TODO: Is their a select-all for the command line???

    " Cut, copy, and paste support for visual and insert mode (not for normal mode)
    " SOURCE: http://superuser.com/questions/10588/how-to-make-cut-copy-paste-in-gvim-on-ubuntu-work-with-ctrlx-ctrlc-ctrlv
    " NOTE: <C-u> removes the '<,'> visual-selection from the command line. See :h c_CTRL-u
    vnoremap <silent> <C-x> :<C-u>call SmartCut()<CR>
    vnoremap <silent> <C-c> :<C-u>call SmartCopy()<CR>
    vnoremap <silent> <C-v> :<C-u>call SmartPaste()<CR>
    inoremap <C-v> <C-r><C-o>+

    " Select-all support for normal, visual, and insert mode
    " http://vim.wikia.com/wiki/Using_standard_editor_shortcuts_in_Vim
    nnoremap <C-a> ggVG
    vnoremap <C-a> ggVG
    inoremap <C-a> <Esc>ggVG

    " Save file support for normal, visual, and insert mode
    " SOURCE: http://vim.wikia.com/wiki/Map_Ctrl-S_to_save_current_or_new_files
    " If the current buffer has never been saved, it will have no name,
    " call the file browser to save it, otherwise just save it.
    command -nargs=0 -bar Update if &modified |
                                \    if empty(bufname('%')) |
                                \        browse confirm write |
                                \    else |
                                \        confirm write |
                                \    endif |
                                \endif
    nnoremap <silent> <C-s> :update<CR>
    " NOTE: <C-u> removes the '<,'> visual-selection from the command line. See :h c_CTRL-u
    vnoremap <silent> <C-s> :<C-u>update<CR>V
    " NOTE: <C-o> executes a normal-mode command without leaving insert mode. See :help ins-special-special
    "inoremap <silent> <C-s> <C-o>:update<CR>
    "
    " <C-o> doesn't seem to work while also using the "Open the OmniCompletion menu as you type" code while the menu is visible.
    " Doing "call feedkeys("\<C-x>\<C-o>", "n")" to perform omni completion seems to be the issue.
    " However doing "call feedkeys("\<C-x>\<C-i>", "n")" to perform keywork completion seems to work without issue.
    "
    " Workaround will exit insert mode to execute the command and then enter insert mode.
    inoremap <silent> <C-s> <Esc>:update<CR>I

    " Undo and redo support for normal, visual, and insert mode
    nnoremap <C-z> <Esc>u
    nnoremap <C-y> <Esc><C-r>

    " NOTE: <C-u> removes the '<,'> visual-selection from the command line. See :h c_CTRL-u
    vnoremap <C-z> :<C-u>uV
    vnoremap <C-y> :<C-u><C-r>V

    inoremap <C-z> <Esc>uI
    inoremap <C-y> <Esc><C-r>I

    function! Find()
        let wordUnderCursor = expand('<cword>')
        if len(wordUnderCursor) > 0
            execute 'promptfind ' . wordUnderCursor
        else
            execute 'promptfind'
        endif
    endfunction

    function! Replace()
        let wordUnderCursor = expand('<cword>')
        if len(wordUnderCursor) > 0
            execute 'promptrepl ' . wordUnderCursor
        else
            execute 'promptrepl'
        endif
    endfunction

    " Find and Find/Replace support for normal, visual, and insert mode
    nnoremap <C-f> :call Find()<CR>
    nnoremap <C-h> :call Replace()<CR>

    " NOTE: <C-u> removes the '<,'> visual-selection from the command line. See :h c_CTRL-u
    vnoremap <C-f> :<C-u>call Find()<CR>
    vnoremap <C-h> :<C-u>call Replace()<CR>

    " NOTE: <C-o> executes a normal-mode command without leaving insert mode. See :help ins-special-special
    inoremap <C-f> <C-o>:call Find()<CR>
    inoremap <C-h> <C-o>:call Replace()<CR>
endif

" }}} Windows common keyboard shortcuts and pasting behavior
FocusedWolf
fonte
-1

tl; dr - vnoremap p "_c *

Aqui está uma lista dos meus mapeamentos completos:
"Corrigir cópia / colar
registro nnoremap DD" * dd
nnoremap D "* d
vnoremap D" d
nnoremap d "_d
nnoremap dd" _dd
vnoremap d "_d
nnoremap s" _s vnoremap s "
_s
nnoremap c "_c
vnoremap c" _c
nnoremap x "_x
vnoremap x" _x
vnoremap p "_c

"Colar na nova linha
nnoremap, p op
nnmapmap, P Op

Kawerte
fonte