summaryrefslogtreecommitdiffhomepage
path: root/vhosts/blog/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
diff options
context:
space:
mode:
Diffstat (limited to 'vhosts/blog/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html')
-rw-r--r--vhosts/blog/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html244
1 files changed, 0 insertions, 244 deletions
diff --git a/vhosts/blog/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html b/vhosts/blog/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
deleted file mode 100644
index fcdbec23..00000000
--- a/vhosts/blog/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
+++ /dev/null
@@ -1,244 +0,0 @@
-<!DOCTYPE html>
-<html lang="ja-JP">
- <head>
- <meta charset="UTF-8">
- <meta name="viewport" content="width=device-width, initial-scale=1.0">
- <meta name="author" content="nsfisis">
- <meta name="copyright" content="&copy; 2021 nsfisis">
- <meta name="description" content="Vim で選択した行の順番を入れ替える方法。">
- <meta name="keywords" content="Vim">
- <meta property="og:type" content="article">
- <meta property="og:title" content="Vimで選択した行の順番を入れ替える|REPL: Rest-Eat-Program Loop">
- <meta property="og:description" content="Vim で選択した行の順番を入れ替える方法。">
- <meta property="og:site_name" content="REPL: Rest-Eat-Program Loop">
- <meta property="og:locale" content="ja_JP">
- <link rel="icon" type="image/svg+xml" href="/favicon.svg">
- <title>Vimで選択した行の順番を入れ替える|REPL: Rest-Eat-Program Loop</title>
- <link rel="stylesheet" href="/style.css?h=9513229b52eb2041b99ba1b959305633">
- </head>
- <body class="single">
- <header class="header">
- <div class="site-logo">
- <a href="/">REPL: Rest-Eat-Program Loop</a>
- </div>
- <nav class="nav">
- <ul>
- <li>
- <a href="/about/">About</a>
- </li>
- <li>
- <a href="/posts/">Posts</a>
- </li>
- <li>
- <a href="/slides/">Slides</a>
- </li>
- <li>
- <a href="/tags/">Tags</a>
- </li>
- </ul>
- </nav>
- </header>
- <main class="main">
- <article class="post-single">
- <header class="post-header">
- <h1 class="post-title">Vimで選択した行の順番を入れ替える</h1>
- <ul class="post-tags">
- <li class="tag">
- <a href="/tags/vim/">Vim</a>
- </li>
- </ul>
- </header>
- <div class="post-content">
- <section id="changelog">
- <h2><a href="#changelog">更新履歴</a></h2>
- <ol>
- <li class="revision">
- <time datetime="2021-10-02">2021-10-02</time>: Qiita から移植
- </li>
- </ol>
- </section>
- <div class="admonition">
- <div class="admonition-label">
- NOTE
- </div>
- <div class="admonition-content">
- <p>
- この記事は Qiita から移植してきたものです。元 URL: <a href="https://qiita.com/nsfisis/items/4fefb361d9a693803520" rel="noreferrer" target="_blank">https://qiita.com/nsfisis/items/4fefb361d9a693803520</a>
- </p>
- </div>
- </div>
- <section id="section--tl-dr">
- <h2><a href="#section--tl-dr">TL; DR</a></h2>
- <div class="codeblock">
- <pre class="shiki github-light" style="background-color:#f5f5f5;color:#24292e" tabindex="0"><code><span class="line"><span style="color:#6A737D">" License: Public Domain</span></span>
-<span class="line"></span>
-<span class="line"><span style="color:#D73A49">command!</span><span style="color:#24292E"> -bar -</span><span style="color:#D73A49">range=</span><span style="color:#24292E">%</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> Reverse</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> keeppatterns &#x3C;line1>,&#x3C;line2>g</span><span style="color:#032F62">/^/</span><span style="color:#24292E">m&#x3C;line1>-</span><span style="color:#005CC5">1</span></span></code></pre>
- </div>
- </section>
- <section id="section--version">
- <h2><a href="#section--version">バージョン情報</a></h2>
- <p>
- <code>:version</code> の一部
- </p>
- <blockquote>
- <p>
- VIM - Vi IMproved 8.2 (2019 Dec 12, compiled Jan 26 2020 11:30:30) macOS version Included patches: 1-148 Huge version without GUI.
- </p>
- </blockquote>
- </section>
- <section id="section--existing-solution">
- <h2><a href="#section--existing-solution">よく紹介されている手法</a></h2>
- <section id="section--existing-solution--external-commands">
- <h3><a href="#section--existing-solution--external-commands"><code>tac</code> / <code>tail</code></a></h3>
- <p>
- <code>tac</code> や <code>tail -r</code> などの外部コマンドを <code>!</code> を使って呼び出し、置き換える。
- </p>
- <blockquote>
- <p>
- :h v_!
- </p>
- </blockquote>
- <p>
- <code>tac</code> コマンドや <code>tail</code> の <code>-r</code> オプションは環境によって利用できないことがあり、複数の環境を行き来する場合に採用しづらい
- </p>
- </section>
- <section id="section--existing-solution--global-command">
- <h3><a href="#section--existing-solution--global-command"><code>:g/^/m0</code></a></h3>
- <p>
- こちらは外部コマンドに頼らず、Vim の機能のみを使う。<code>g</code> は <code>:global</code> コマンドの、<code>m</code> は <code>:move</code> コマンドの略
- </p>
- <p>
- <code>:global</code> コマンドは <code>:[range]global/{pattern}/[command]</code> のように使い、<code>[range]</code> で指定された範囲の行のうち、<code>{pattern}</code> で指定された検索パターンにマッチする行に対して、順番に <code>[command]</code> で指定された Ex コマンドを呼び出す。
- </p>
- <blockquote>
- <p>
- :h :global
- </p>
- </blockquote>
- <p>
- <code>:move</code> コマンドは <code>[range]:move {address}</code> のように使い、<code>[range]</code> で指定された範囲の行を <code>{address}</code> で指定された位置に移動させる。
- </p>
- <blockquote>
- <p>
- :h :move
- </p>
- </blockquote>
- <p>
- <code>:g/^/m0</code> のように組み合わせると、「すべての行を1行ずつ 0行目(1行目の上)に動かす」という動きをする。これは確かに行の入れ替えになっている。
- </p>
- <p>
- なお、<code>:g/^/m0</code> は全ての行を入れ替えるが、<code>:N,Mg/^/mN-1</code> とすることで N行目から M行目を処理範囲とするよう拡張できる。手でこれを入力するわけにはいかないので、次のようなコマンドを用意する。
- </p>
- <div class="codeblock">
- <pre class="shiki github-light" style="background-color:#f5f5f5;color:#24292e" tabindex="0"><code><span class="line"><span style="color:#D73A49">command!</span><span style="color:#24292E"> -bar -</span><span style="color:#D73A49">range=</span><span style="color:#24292E">%</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> Reverse</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> &#x3C;line1>,&#x3C;line2>g</span><span style="color:#032F62">/^/</span><span style="color:#24292E">m&#x3C;line1>-</span><span style="color:#005CC5">1</span></span></code></pre>
- </div>
- <p>
- これは望みの動作をするが、実際に実行してみると全行がハイライトされてしまう。次節で詳細を述べる。
- </p>
- </section>
- </section>
- <section id="section--problem-of-global-command">
- <h2><a href="#section--problem-of-global-command"><code>:g/^/m0</code> の問題点</a></h2>
- <p>
- <code>:global</code> コマンドは各行に対してマッチングを行う際、現在の検索パターンを上書きしてしまう。<code>^</code> は行の先頭にマッチするため、結果として全ての行がハイライトされてしまう。<code>&apos;hlsearch&apos;</code> オプションを無効にしている場合その限りではないが、その場合でも直前の検索パターンが失われてしまうと <code>n</code> コマンドなどの際に不便である。
- </p>
- <blockquote>
- <p>
- :h @/
- </p>
- </blockquote>
- </section>
- <section id="section--solution">
- <h2><a href="#section--solution">解決策</a></h2>
- <div class="admonition" editat="2020-09-28" operation="追記">
- <div class="admonition-label">
- 2020-09-28 追記
- </div>
- <div class="admonition-content">
- <p>
- より簡潔な方法を見つけたので次節に追記した。
- </p>
- </div>
- </div>
- <p>
- 前述した <code>:Reverse</code> コマンドの定義を少し変えて、次のようにする:
- </p>
- <div class="codeblock">
- <pre class="shiki github-light" style="background-color:#f5f5f5;color:#24292e" tabindex="0"><code><span class="line"><span style="color:#D73A49">function!</span><span style="color:#6F42C1"> s:reverse_lines</span><span style="color:#24292E">(from, </span><span style="color:#005CC5">to</span><span style="color:#24292E">) </span><span style="color:#D73A49">abort</span></span>
-<span class="line"><span style="color:#005CC5"> execute</span><span style="color:#6F42C1"> printf</span><span style="color:#24292E">(</span><span style="color:#032F62">"%d,%dg/^/m%d"</span><span style="color:#24292E">, a:from, a:to, a:from</span><span style="color:#D73A49"> - </span><span style="color:#005CC5">1</span><span style="color:#24292E">)</span></span>
-<span class="line"><span style="color:#D73A49">endfunction</span></span>
-<span class="line"></span>
-<span class="line"><span style="color:#D73A49">command!</span><span style="color:#24292E"> -bar -</span><span style="color:#D73A49">range=</span><span style="color:#24292E">%</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> Reverse</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#D73A49"> call</span><span style="color:#24292E"> &#x3C;</span><span style="color:#005CC5">SID</span><span style="color:#24292E">></span><span style="color:#6F42C1">reverse_lines</span><span style="color:#24292E">(&#x3C;line1>, &#x3C;line2>)</span></span></code></pre>
- </div>
- <p>
- 実行しているコマンドが変わったわけではないが、関数呼び出しを経由するようにした。これだけで前述の問題が解決する。
- </p>
- <p>
- この理由は、ユーザー定義関数を実行する際は検索パターンが一度保存され、実行が終了したあと復元されるため。結果として検索パターンが <code>^</code> で上書きされることがなくなる。
- </p>
- <p>
- Vim のヘルプから該当箇所を引用する (強調は筆者による)。
- </p>
- <blockquote>
- <p>
- :h autocmd-searchpat
- </p>
- <p>
- <strong>Autocommands do not change the current search patterns.</strong> Vim saves the current search patterns before executing autocommands then restores them after the autocommands finish. This means that autocommands do not affect the strings highlighted with the ‘hlsearch’ option.
- </p>
- </blockquote>
- <p>
- これは autocommand の実行に関しての記述だが、これと同じことがユーザー定義関数の実行時にも適用される。このことは <code>:nohlsearch</code> のヘルプにある。同じく該当箇所を引用する (強調は筆者による)。
- </p>
- <blockquote>
- <p>
- :h :nohlsearch
- </p>
- <p>
- (略) This command doesn’t work in an autocommand, because the highlighting state is saved and restored when executing autocommands |autocmd-searchpat|. <strong>Same thing for when invoking a user function.</strong>
- </p>
- </blockquote>
- <p>
- この仕様により、<code>:g/^/m0</code> の呼び出しをユーザー定義関数に切り出すことで上述の問題を解決できる。
- </p>
- </section>
- <section id="section--solution-revised">
- <h2><a href="#section--solution-revised">解決策 (改訂版)</a></h2>
- <div class="admonition" editat="2020-09-28" operation="追記">
- <div class="admonition-label">
- 2020-09-28 追記
- </div>
- <div class="admonition-content">
- <p>
- より簡潔な方法を見つけたため追記する。
- </p>
- </div>
- </div>
- <div class="codeblock">
- <pre class="shiki github-light" style="background-color:#f5f5f5;color:#24292e" tabindex="0"><code><span class="line"><span style="color:#D73A49">command!</span><span style="color:#24292E"> -bar -</span><span style="color:#D73A49">range=</span><span style="color:#24292E">%</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> Reverse</span></span>
-<span class="line"><span style="color:#D73A49"> \</span><span style="color:#24292E"> keeppatterns &#x3C;line1>,&#x3C;line2>g</span><span style="color:#032F62">/^/</span><span style="color:#24292E">m&#x3C;line1>-</span><span style="color:#005CC5">1</span></span></code></pre>
- </div>
- <p>
- まさにこのための Exコマンド、<code>:keeppatterns</code> が存在する。<code>:keeppatterns {command}</code> のように使い、読んで字の如く、後ろに続く Exコマンドを「現在の検索パターンを保ったまま」実行する。はるかに分かりやすく意図を表現できる。
- </p>
- <blockquote>
- <p>
- :h :keeppatterns
- </p>
- </blockquote>
- </section>
- </div>
- </article>
- </main>
- <footer class="footer">
- &copy; 2021 nsfisis
- </footer>
- </body>
-</html>