aboutsummaryrefslogtreecommitdiffhomepage
path: root/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
diff options
context:
space:
mode:
Diffstat (limited to 'public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html')
-rw-r--r--public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html497
1 files changed, 207 insertions, 290 deletions
diff --git a/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html b/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
index 8b94185..14beb7f 100644
--- a/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
+++ b/public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html
@@ -4,18 +4,14 @@
<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; nsfisis">
+ <meta name="copyright" content="&copy; 2021 nsfisis">
<meta name="description" content="Vim で選択した行の順番を入れ替える方法。">
<meta name="keywords" content="Vim">
<link rel="icon" type="image/svg+xml" href="/favicon.svg">
<title>Vimで選択した行の順番を入れ替える | REPL: Rest-Eat-Program Loop</title>
-
- <link rel="stylesheet" href="/hl.css?208c52e3b7c9db1cad782c5d30b4698f">
-
- <link rel="stylesheet" href="/style.css?779b1a3debcaeba619f6fe500e93d525">
-
- <link rel="stylesheet" href="/custom.css?a649ea3528d4b626fb636505d94c1144">
-
+ <link rel="stylesheet" href="/hl.css?h=208c52e3b7c9db1cad782c5d30b4698f">
+ <link rel="stylesheet" href="/style.css?h=779b1a3debcaeba619f6fe500e93d525">
+ <link rel="stylesheet" href="/custom.css?h=a649ea3528d4b626fb636505d94c1144">
</head>
<body class="single">
<header class="header">
@@ -29,299 +25,220 @@
<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>
-
+ <ul class="post-tags">
+ <li class="tag">
+ <a href="/tags/vim">Vim</a>
+ </li>
+ </ul>
</header>
<div class="post-content">
<section>
<h2 id="changelog">更新履歴</h2>
<ol>
+ <li class="revision">
+ <time datetime="2021-10-02">2021-10-02</time>: Qiita から移植
+ </li>
+ </ol>
+ </section>
+ <p>
+ この記事は Qiita から移植してきたものです。 元 URL:<a xl:href="https://qiita.com/nsfisis/items/4fefb361d9a693803520">https://qiita.com/nsfisis/items/4fefb361d9a693803520</a>
+ </p>
+
+ <p>
+ <hr>
+ </hr>
+ </p>
+
+ <section id="section--_バージョン情報">
+ <h2><a href="#section--_バージョン情報">バージョン情報</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--_よく紹介されている手法">
+ <h2><a href="#section--_よく紹介されている手法">よく紹介されている手法</a></h2>
+ <section id="section--_tac_tail">
+ <h3><a href="#section--_tac_tail"><code>tac</code>/<code>tail</code></a></h3>
+ <p>
+ <code>tac</code>や<code>tail -r</code>などの外部コマンドを<code>!</code>を使って呼び出し、置き換える。
+ </p>
- <li class="revision">
- <time datetime="2021-10-02">2021-10-02</time>: Qiita から移植
- </li>
+ <blockquote>
+ <p>
+ :h v_!
+ </p>
+ </blockquote>
- </ol>
+ <p>
+ <code>tac</code>コマンドや<code>tail</code>の<code>-r</code>オプションは環境によって利用できないことがあり、複数の環境を行き来する場合に採用しづらい
+ </p>
+ </section>
+
+ <section id="section--_gm0">
+ <h3><a href="#section--_gm0"><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>
+
+ <pre language="vim" linenumbering="unnumbered">
+ <code>command! -bar -range=%
+\ Reverse
+\ &lt;line1&gt;,&lt;line2&gt;g/^/m&lt;line1&gt;-1</code>
+ </pre>
+
+ <p>
+ これは望みの動作をするが、実際に実行してみると全行がハイライトされてしまう。次節で詳細を述べる。
+ </p>
+ </section>
</section>
- <div id="preamble">
-<div class="sectionbody">
-<div class="paragraph">
-<p>この記事は Qiita から移植してきたものです。 元 URL:
-<a href="https://qiita.com/nsfisis/items/4fefb361d9a693803520" class="bare">https://qiita.com/nsfisis/items/4fefb361d9a693803520</a></p>
-</div>
-<hr>
-</div>
-</div>
-<section class="section-1">
- <h2 id="" class="section-header">
-
- バージョン情報
-
- </h2>
- <div class="section-body">
- <div class="paragraph">
-<p><code>:version</code> の一部</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<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>
-</div>
-</blockquote>
-</div>
- </div>
-</section>
-<section class="section-1">
- <h2 id="" class="section-header">
-
- よく紹介されている手法
-
- </h2>
- <div class="section-body">
- <section class="section-2">
- <h3 id="" class="section-header">
-
- <code>tac</code> / <code>tail</code>
-
- </h3>
- <div class="section-body">
- <div class="paragraph">
-<p><code>tac</code> や <code>tail -r</code> などの外部コマンドを <code>!</code>
-を使って呼び出し、置き換える。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h v_!</p>
-</div>
-</blockquote>
-</div>
-<div class="paragraph">
-<p><code>tac</code> コマンドや <code>tail</code> の <code>-r</code>
-オプションは環境によって利用できないことがあり、複数の環境を行き来する場合に採用しづらい</p>
-</div>
- </div>
-</section>
-<section class="section-2">
- <h3 id="" class="section-header">
-
- <code>:g/^/m0</code>
-
- </h3>
- <div class="section-body">
- <div class="paragraph">
-<p>こちらは外部コマンドに頼らず、Vim の機能のみを使う。<code>g</code> は <code>:global</code>
-コマンドの、<code>m</code> は <code>:move</code> コマンドの略</p>
-</div>
-<div class="paragraph">
-<p><code>:global</code> コマンドは <code>:[range]global/{pattern}/[command]</code>
-のように使い、<code>[range]</code> で指定された範囲の行のうち、<code>{pattern}</code>
-で指定された検索パターンにマッチする行に対して、順番に <code>[command]</code>
-で指定された Ex コマンドを呼び出す。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h :global</p>
-</div>
-</blockquote>
-</div>
-<div class="paragraph">
-<p><code>:move</code> コマンドは <code>[range]:move {address}</code> のように使い、<code>[range]</code>
-で指定された範囲の行を <code>{address}</code> で指定された位置に移動させる。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h :move</p>
-</div>
-</blockquote>
-</div>
-<div class="paragraph">
-<p><code>:g/^/m0</code> のように組み合わせると、「すべての行を1行ずつ
-0行目(1行目の上)に動かす」という動きをする。これは確かに行の入れ替えになっている。</p>
-</div>
-<div class="paragraph">
-<p>なお、<code>:g/^/m0</code> は全ての行を入れ替えるが、<code>:N,Mg/^/mN-1</code> とすることで
-N行目から
-M行目を処理範囲とするよう拡張できる。手でこれを入力するわけにはいかないので、次のようなコマンドを用意する。</p>
-</div>
-<div id="source." class="listingblock">
-<div class="content">
-<pre class="rouge highlight"><code data-lang="vim">command<span class="p">!</span> <span class="p">-</span>bar <span class="p">-</span><span class="nb">range</span><span class="p">=</span>%
-<span class="se"> \</span> Reverse
-<span class="se"> \</span> <span class="p">&lt;</span>line1<span class="p">&gt;,&lt;</span>line2<span class="p">&gt;</span><span class="k">g</span><span class="sr">/^/</span><span class="k">m</span><span class="p">&lt;</span>line1<span class="p">&gt;</span><span class="m">-1</span></code></pre>
-</div>
-</div>
-<div class="paragraph">
-<p>これは望みの動作をするが、実際に実行してみると全行がハイライトされてしまう。次節で詳細を述べる。</p>
-</div>
- </div>
-</section>
- </div>
-</section>
-<section class="section-1">
- <h2 id="" class="section-header">
-
- <code>:g/^/m0</code> の問題点
-
- </h2>
- <div class="section-body">
- <div class="paragraph">
-<p><code>:global</code>
-コマンドは各行に対してマッチングを行う際、現在の検索パターンを上書きしてしまう。<code>^</code>
-は行の先頭にマッチするため、結果として全ての行がハイライトされてしまう。<code>'hlsearch'</code>
-オプションを無効にしている場合その限りではないが、その場合でも直前の検索パターンが失われてしまうと
-<code>n</code> コマンドなどの際に不便である。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h @/</p>
-</div>
-</blockquote>
-</div>
- </div>
-</section>
-<section class="section-1">
- <h2 id="" class="section-header">
-
- 解決策
-
- </h2>
- <div class="section-body">
- <div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>[2020/9/28追記] より簡潔な方法を見つけたので次節に追記した</p>
-</div>
-</blockquote>
-</div>
-<div class="paragraph">
-<p>前述した <code>:Reverse</code> コマンドの定義を少し変えて、次のようにする:</p>
-</div>
-<div id="source." class="listingblock">
-<div class="content">
-<pre class="rouge highlight"><code data-lang="vim"><span class="k">function</span><span class="p">!</span> <span class="nv">s:reverse_lines</span><span class="p">(</span>from<span class="p">,</span> <span class="k">to</span><span class="p">)</span> abort
- <span class="nb">execute</span> <span class="nb">printf</span><span class="p">(</span><span class="s2">"%d,%dg/^/m%d"</span><span class="p">,</span> <span class="nv">a:from</span><span class="p">,</span> <span class="nv">a:to</span><span class="p">,</span> <span class="nv">a:from</span> <span class="p">-</span> <span class="m">1</span><span class="p">)</span>
-<span class="k">endfunction</span>
+
+ <section id="section--_gm0_の問題点">
+ <h2><a href="#section--_gm0_の問題点"><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--_解決策">
+ <h2><a href="#section--_解決策">解決策</a></h2>
+ <blockquote>
+ <p>
+ [2020/9/28追記] より簡潔な方法を見つけたので次節に追記した
+ </p>
+ </blockquote>
+
+ <p>
+ 前述した<code>:Reverse</code>コマンドの定義を少し変えて、次のようにする:
+ </p>
+
+ <pre language="vim" linenumbering="unnumbered">
+ <code>function! s:reverse_lines(from, to) abort
+ execute printf(&quot;%d,%dg/^/m%d&quot;, a:from, a:to, a:from - 1)
+ endfunction
-command<span class="p">!</span> <span class="p">-</span>bar <span class="p">-</span><span class="nb">range</span><span class="p">=</span>%
-<span class="se"> \</span> Reverse
-<span class="se"> \</span> <span class="k">call</span> <span class="p">&lt;</span>SID<span class="p">&gt;</span>reverse_lines<span class="p">(&lt;</span>line1<span class="p">&gt;,</span> <span class="p">&lt;</span>line2<span class="p">&gt;)</span></code></pre>
-</div>
-</div>
-<div class="paragraph">
-<p>実行しているコマンドが変わったわけではないが、関数呼び出しを経由するようにした。これだけで前述の問題が解決する。</p>
-</div>
-<div class="paragraph">
-<p>この理由は、ユーザー定義関数を実行する際は検索パターンが一度保存され、実行が終了したあと復元されるため。結果として検索パターンが
-<code>^</code> で上書きされることがなくなる。</p>
-</div>
-<div class="paragraph">
-<p>Vim のヘルプから該当箇所を引用する (強調は筆者による)。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h autocmd-searchpat</p>
-</div>
-<div class="paragraph">
-<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>
-</div>
-</blockquote>
-</div>
-<div class="paragraph">
-<p>これは autocommand
-の実行に関しての記述だが、これと同じことがユーザー定義関数の実行時にも適用される。このことは
-<code>:nohlsearch</code> のヘルプにある。同じく該当箇所を引用する
-(強調は筆者による)。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h :nohlsearch</p>
-</div>
-<div class="paragraph">
-<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>
-</div>
-</blockquote>
-</div>
-<div class="paragraph">
-<p>この仕様により、<code>:g/^/m0</code>
-の呼び出しをユーザー定義関数に切り出すことで上述の問題を解決できる。</p>
-</div>
- </div>
-</section>
-<section class="section-1">
- <h2 id="" class="section-header">
-
- 解決策 (改訂版)
-
- </h2>
- <div class="section-body">
- <div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>[2020/9/28追記] より簡潔な方法を見つけたため追記する</p>
-</div>
-</blockquote>
-</div>
-<div id="source." class="listingblock">
-<div class="content">
-<pre class="rouge highlight"><code data-lang="vim">command<span class="p">!</span> <span class="p">-</span>bar <span class="p">-</span><span class="nb">range</span><span class="p">=</span>%
-<span class="se"> \</span> Reverse
-<span class="se"> \</span> <span class="k">keeppatterns</span> <span class="p">&lt;</span>line1<span class="p">&gt;,&lt;</span>line2<span class="p">&gt;</span><span class="k">g</span><span class="sr">/^/</span><span class="k">m</span><span class="p">&lt;</span>line1<span class="p">&gt;</span><span class="m">-1</span></code></pre>
-</div>
-</div>
-<div class="paragraph">
-<p>まさにこのための Exコマンド、<code>:keeppatterns</code>
-が存在する。<code>:keeppatterns {command}</code>
-のように使い、読んで字の如く、後ろに続く
-Exコマンドを「現在の検索パターンを保ったまま」実行する。はるかに分かりやすく意図を表現できる。</p>
-</div>
-<div class="quoteblock">
-<blockquote>
-<div class="paragraph">
-<p>:h :keeppatterns</p>
-</div>
-</blockquote>
-</div>
- </div>
-</section>
-<section class="section-1">
- <h2 id="" class="section-header">
-
- コピペ用再掲
-
- </h2>
- <div class="section-body">
- <div id="source." class="listingblock">
-<div class="content">
-<pre class="rouge highlight"><code data-lang="vim"><span class="c">" License: Public Domain</span>
+ command! -bar -range=%
+ \ Reverse
+ \ call &lt;SID&gt;reverse_lines(&lt;line1&gt;, &lt;line2&gt;)</code>
+ </pre>
+
+ <p>
+ 実行しているコマンドが変わったわけではないが、関数呼び出しを経由するようにした。これだけで前述の問題が解決する。
+ </p>
+
+ <p>
+ この理由は、ユーザー定義関数を実行する際は検索パターンが一度保存され、実行が終了したあと復元されるため。結果として検索パターンが<code>^</code>で上書きされることがなくなる。
+ </p>
+
+ <p>
+ Vim のヘルプから該当箇所を引用する (強調は筆者による)。
+ </p>
+
+ <blockquote>
+ <p>
+ :h autocmd-searchpat
+ </p>
+
+ <p>
+ <em role="strong">Autocommands do not change the current search patterns.</em>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&apos; 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|.<em role="strong">Same thing for when invoking a user function.</em>
+ </p>
+ </blockquote>
+
+ <p>
+ この仕様により、<code>:g/^/m0</code>の呼び出しをユーザー定義関数に切り出すことで上述の問題を解決できる。
+ </p>
+ </section>
+
+ <section id="section--_解決策_改訂版">
+ <h2><a href="#section--_解決策_改訂版">解決策 (改訂版)</a></h2>
+ <blockquote>
+ <p>
+ [2020/9/28追記] より簡潔な方法を見つけたため追記する
+ </p>
+ </blockquote>
+
+ <pre language="vim" linenumbering="unnumbered">
+ <code>command! -bar -range=%
+ \ Reverse
+ \ keeppatterns &lt;line1&gt;,&lt;line2&gt;g/^/m&lt;line1&gt;-1</code>
+ </pre>
+
+ <p>
+ まさにこのための Exコマンド、<code>:keeppatterns</code>が存在する。<code>:keeppatterns {command}</code>のように使い、読んで字の如く、後ろに続く Exコマンドを「現在の検索パターンを保ったまま」実行する。はるかに分かりやすく意図を表現できる。
+ </p>
+
+ <blockquote>
+ <p>
+ :h :keeppatterns
+ </p>
+ </blockquote>
+ </section>
+
+ <section id="section--_コピペ用再掲">
+ <h2><a href="#section--_コピペ用再掲">コピペ用再掲</a></h2>
+ <pre language="vim" linenumbering="unnumbered">
+ <code>&quot; License: Public Domain
-command<span class="p">!</span> <span class="p">-</span>bar <span class="p">-</span><span class="nb">range</span><span class="p">=</span>%
-<span class="se"> \</span> Reverse
-<span class="se"> \</span> <span class="k">keeppatterns</span> <span class="p">&lt;</span>line1<span class="p">&gt;,&lt;</span>line2<span class="p">&gt;</span><span class="k">g</span><span class="sr">/^/</span><span class="k">m</span><span class="p">&lt;</span>line1<span class="p">&gt;</span><span class="m">-1</span></code></pre>
-</div>
-</div>
- </div>
-</section>
+ command! -bar -range=%
+ \ Reverse
+ \ keeppatterns &lt;line1&gt;,&lt;line2&gt;g/^/m&lt;line1&gt;-1</code>
+ </pre>
+ </section>
</div>
-
</article>
</main>
<footer class="footer">