削除した行をヤンクせずに、vimでヤンクしたテキストを置換ペーストする方法は?


39

だから私は通常、新しいテキストが貼り付けられている古いテキストを上書きしながら、あるポイントから別のポイントにテキストをコピーしていることに気付きます:

blah1
newtext
blah2
wrong1
blah3
wrong2
blah4

私が視覚的なマークnewtexty付けて、それを承認したとします。今、私は選択しwrong1(これは、必ずしも単なる単語ではなく、何でも可能です)、をp味わいますnewtext。ただし、今と同じwrong2ことをすると、のwrong1代わりにに置き換えられnewtextます。

それでは、バッファにあるテキストを、現在上書きしているテキストと交換しないようにするにはどうすればよいですか?

編集1

私はレジスターの提案が非常に好きですが(:disコマンドを発見したので、レジスターの使用を開始すると思います)、スワップモードを使用しないため、jinfieldの答えを修正します。

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

完全にトリックを行います。

編集2

私は速すぎました。romainlのソリューションは、Edit 1のハックなしで、まさに私が探していたものです。
実際にvnoremap p "_dPは、十分です!
だから、受け入れられた答えを変える。


2
ねえちなみに、私は長いこと使ってきたvnoremap p "_dP map、と私はそれが行の最後の単語/文字ではうまく機能しないことに気付きました。私はに行って戻ってきたvnoremap p "0pvnoremap P "0Pset clipboard=unnamed(OSX用)
Kache

vnoremap p "_dPペーストの空白を削除します。編集1は完全に機能します。
Vaclav Kasal

回答:


21

私の.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

"_によると、「ブラックホールレジスタ」:help "_です。

「このレジスタに書き込むとき、何も起こりません。これは、通常のレジスタに影響を与えずにテキストを削除するために使用できます。このレジスタから読み取るとき、何も返されません。{Viにはない}」


1
私はそれを使用vnoremap p "_dP mapし、行の最後の単語/文字に対してうまく機能しないことに気付きました。私はに行って戻ってきましたがvnoremap p "0pvnoremap P "0P、およびset clipboard=unnamed(OSX用)
Kache

vnoremap p "_dP選択モードでvnoremap <leader>p "_dPは動作しなくなりますが、動作します
-whitesiroi

1
多分<leader>私は何かに置き換えるために知っているはずのプレースホルダーですが、これは私がそれを削除した場合にのみ私のために働いた。ここでどういう意味ですか?
ハッシュブラウン

1
@ハッシュブラウン、:help mapleader
ロメイン

小さな警告:vnoremap <leader>pバッファーの最後の行ではマッピングが正しく機能しません。バッファーの最後の行を削除するとすぐに最後の1行目になり、Pその行の上に貼り付けられるためです。以下の代わりに。
カル

13

標準バッファに加えて、テキストを名前付きバッファにヤンクしてから、それらの名前付きバッファから入れることができます。使用できる名前付きバッファは最大26個あります(各文字に1つ)。名前付きバッファにアクセスするには、二重引用符と文字を使用します。例:

"dyy -現在の行をバッファdにヤンクします。
"a7yy-次の7行をバッファーaにヤンクします。
"dP-バッファーdの内容をカーソルの前に置きます。
"ap-バッファの内容をカーソルの後に置く

もう1つのクールなことは、小文字ではなく大文字を使用する場合です。つまり"Dyy、現在の行はバッファーdに置き換えられずに追加されます。O`Reilly本の詳細:http ://docstore.mik.ua/orelly/unix/vi/ch04_03.htm


3
とてもクールなこと。バッファについては知っていましたが、この問題に接続しませんでした。それはまだ"aすべてにとって面倒ですが、大丈夫です。
ビットマスク

4

使用するとき putビジュアルモードで、置換するテキストwrong1が、「未使用」レジスタの内容で上書きされます。

これは実際には、選択後にレジスタを「配置」してから選択を削除することで機能します。問題は、この削除がunnamedレジスタに保存され、次の使用に使用されることですputアクションにです。

によると:h v_p、解決策は、などの名前付きレジスタにヤンクし、必要な回数だけ"0y貼り付け"0pます。頻繁に行う場合は、名前付きレジスタをマップ<leader>y<leader>pて使用すると役立つ場合があります。

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

詳細については、以下を参照してください。

:help v_p
:help map

この解決策は、vim自体から何か巧妙なものが現れるまで、最も使いやすいようです。
ユーガルジンドル

4

"0登録簿からの貼り付けは知っておくことが重要ですが、多くの場合、何度も交換する必要があります。繰り返し可能なアクションにする.と、garyjohnが暗示しているように、演算子を使用できます。vim wikiで説明されています

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". 

3

テキストを名前のないレジスタ*にヤンクすると、コピーもレジスタ0に配置されます。選択したテキストを置き換えるたびに、0レジスタから貼り付けることができます。見る

:help registers

また、複数の単語を同じ単語に置き換える場合は、置き換える単語の先頭に移動してを入力するだけ.です。これにより、最後の編集操作が繰り返されます。見る

:help single-repeat

*ヤンクしたり入れたりする保管場所はレジスタと呼ばれます。バッファは編集するもので、通常はディスクからのファイルのコピーです。


1

頻繁にこれが必要なので、そのためのプラグインを作成しました:ReplaceWithRegister

このプラグインは、gr{motion}、行全体、または現在の選択で覆われているテキストをレジスターの内容で置き換えるツーインワンコマンドを提供します。古いテキストはブラックホールレジスタに削除されます。つまり、削除されます。


1

vnoremap p "_dP(私も試しましたxc)のようなものが行の始まりと終わりに問題があるため、vnoremap p :<C-U>let @p = @+<CR>gvp:let @+ = @p<CR>これを行うことになりました:既存のプラグインよりも簡単であることがわかりました(これset clipboard=unnamedplusはすぐに動作しませんでした)。だからそれは何ですか:

  • コマンドモードに切り替えます
  • 範囲を削除(C-U
  • バックアップ+レジスタ(unnamedplusのために、選択肢がある"*、あなたの構成に応じて)へp
  • 選択を回復して貼り付ける
  • もう一度コマンドモードに切り替える
  • レジスタを回復

パーフェクト!これは、私にとって期待どおりに機能した最初のオプションです。ありがとう!
ジャミスチャールズ

1

これは、WindowsスタイルのControl + X cut / Control + C copy / Control + V paste / Control + S save / Control + F find / Control + H replace動作に使用するものです(.vimrcから文字通りコピーされます)。

smartpaste()関数には、探しているもの、つまり、選択したものを同時にヤンクせずに強調表示されたテキストに貼り付ける方法が含まれている必要があります。

" 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

-1

tl; dr-vnoremap p "_c *

:ここでは私の完全なマッピングのリストである
"修正レジスタのコピー/貼り付け
DD nnoremapは" DD *
nnoremap Dを"* D
D vnoremap" Dの
nnoremap dの"_d
nnoremap DD" _dd
vnoremapさd "_d
nnoremapの" _s
vnoremap S「_s
nnoremap c "_c
vnoremap c" _c
nnoremap x "_x
vnoremap x" _x
vnoremap p "_c

"新しい行に貼り付けます
nnoremap、p op
nnoremap、P Op

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.