aboutsummaryrefslogtreecommitdiffhomepage
path: root/public/posts/2021-10-02/vim-swap-order-of-selected-lines
diff options
context:
space:
mode:
Diffstat (limited to 'public/posts/2021-10-02/vim-swap-order-of-selected-lines')
-rw-r--r--public/posts/2021-10-02/vim-swap-order-of-selected-lines/index.html118
1 files changed, 59 insertions, 59 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 ada0dbd..48b57ac 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
@@ -54,115 +54,115 @@
</ol>
</section>
<p>
- この記事は Qiita から移植してきたものです。 元 URL:<a href="https://qiita.com/nsfisis/items/4fefb361d9a693803520">https://qiita.com/nsfisis/items/4fefb361d9a693803520</a>
+ この記事は Qiita から移植してきたものです。 元 URL: <a 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>
+ <h2><a href="#section--_バージョン情報">バージョン情報</a></h2>
<p>
- <code>:version</code>の一部
+ <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>
+ <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>
+ <h3><a href="#section--_tac_tail"><code>tac</code> / <code>tail</code></a></h3>
<p>
- <code>tac</code>や<code>tail -r</code>などの外部コマンドを<code>!</code>を使って呼び出し、置き換える。
+ <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>オプションは環境によって利用できないことがあり、複数の環境を行き来する場合に採用しづらい
+ <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>
+ <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>コマンドの略
+ こちらは外部コマンドに頼らず、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 コマンドを呼び出す。
+ <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>で指定された位置に移動させる。
+ <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行目の上)に動かす」という動きをする。これは確かに行の入れ替えになっている。
+ <code>:g/^/m0</code> のように組み合わせると、「すべての行を1行ずつ 0行目(1行目の上)に動かす」という動きをする。これは確かに行の入れ替えになっている。
</p>
-
+
<p>
- なお、<code>:g/^/m0</code>は全ての行を入れ替えるが、<code>:N,Mg/^/mN-1</code>とすることで N行目から M行目を処理範囲とするよう拡張できる。手でこれを入力するわけにはいかないので、次のようなコマンドを用意する。
+ なお、<code>:g/^/m0</code> は全ての行を入れ替えるが、<code>:N,Mg/^/mN-1</code> とすることで N行目から M行目を処理範囲とするよう拡張できる。手でこれを入力するわけにはいかないので、次のようなコマンドを用意する。
</p>
-
+
<pre class="highlight" language="vim" linenumbering="unnumbered"><code class="highlight">command! -bar -<span class="hljs-built_in">range</span>=%
\ Reverse
\ <span class="hljs-symbol">&lt;line1&gt;</span>,<span class="hljs-symbol">&lt;line2&gt;</span>g/^/<span class="hljs-keyword">m</span><span class="hljs-symbol">&lt;line1&gt;</span>-<span class="hljs-number">1</span></code></pre>
-
+
<p>
これは望みの動作をするが、実際に実行してみると全行がハイライトされてしまう。次節で詳細を述べる。
</p>
</section>
</section>
-
+
<section id="section--_gm0_の問題点">
- <h2><a href="#section--_gm0_の問題点"><code>:g/^/m0</code>の問題点</a></h2>
+ <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>コマンドなどの際に不便である。
+ <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>
+ <h2><a href="#section--_解決策">解決策</a></h2>
<blockquote>
<p>
[2020/9/28追記] より簡潔な方法を見つけたので次節に追記した
</p>
</blockquote>
-
+
<p>
- 前述した<code>:Reverse</code>コマンドの定義を少し変えて、次のようにする:
+ 前述した <code>:Reverse</code> コマンドの定義を少し変えて、次のようにする:
</p>
-
+
<pre class="highlight" language="vim" linenumbering="unnumbered"><code class="highlight"><span class="hljs-keyword">function!</span> <span class="hljs-title">s</span>:reverse_lines<span class="hljs-params">(from, to)</span> abort
<span class="hljs-keyword">execute</span> <span class="hljs-built_in">printf</span>(<span class="hljs-string">&quot;%d,%dg/^/m%d&quot;</span>, <span class="hljs-variable">a:from</span>, <span class="hljs-variable">a:to</span>, <span class="hljs-variable">a:from</span> - <span class="hljs-number">1</span>)
<span class="hljs-keyword">endfunction</span>
@@ -170,73 +170,73 @@
command! -bar -<span class="hljs-built_in">range</span>=%
\ Reverse
\ <span class="hljs-keyword">call</span> <span class="hljs-symbol">&lt;SID&gt;</span>reverse_lines(<span class="hljs-symbol">&lt;line1&gt;</span>, <span class="hljs-symbol">&lt;line2&gt;</span>)</code></pre>
-
+
<p>
実行しているコマンドが変わったわけではないが、関数呼び出しを経由するようにした。これだけで前述の問題が解決する。
</p>
-
+
<p>
- この理由は、ユーザー定義関数を実行する際は検索パターンが一度保存され、実行が終了したあと復元されるため。結果として検索パターンが<code>^</code>で上書きされることがなくなる。
+ この理由は、ユーザー定義関数を実行する際は検索パターンが一度保存され、実行が終了したあと復元されるため。結果として検索パターンが <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.
+ <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>のヘルプにある。同じく該当箇所を引用する (強調は筆者による)。
+ これは 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>
+ (略) 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>の呼び出しをユーザー定義関数に切り出すことで上述の問題を解決できる。
+ この仕様により、<code>:g/^/m0</code> の呼び出しをユーザー定義関数に切り出すことで上述の問題を解決できる。
</p>
</section>
-
+
<section id="section--_解決策_改訂版">
- <h2><a href="#section--_解決策_改訂版">解決策 (改訂版)</a></h2>
+ <h2><a href="#section--_解決策_改訂版">解決策 (改訂版)</a></h2>
<blockquote>
<p>
[2020/9/28追記] より簡潔な方法を見つけたため追記する
</p>
</blockquote>
-
+
<pre class="highlight" language="vim" linenumbering="unnumbered"><code class="highlight">command! -bar -<span class="hljs-built_in">range</span>=%
\ Reverse
\ keeppatterns <span class="hljs-symbol">&lt;line1&gt;</span>,<span class="hljs-symbol">&lt;line2&gt;</span>g/^/<span class="hljs-keyword">m</span><span class="hljs-symbol">&lt;line1&gt;</span>-<span class="hljs-number">1</span></code></pre>
-
+
<p>
- まさにこのための Exコマンド、<code>:keeppatterns</code>が存在する。<code>:keeppatterns {command}</code>のように使い、読んで字の如く、後ろに続く Exコマンドを「現在の検索パターンを保ったまま」実行する。はるかに分かりやすく意図を表現できる。
+ まさにこのための 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>
+ <h2><a href="#section--_コピペ用再掲">コピペ用再掲</a></h2>
<pre class="highlight" language="vim" linenumbering="unnumbered"><code class="highlight"><span class="hljs-comment">&quot; License: Public Domain</span>
command! -bar -<span class="hljs-built_in">range</span>=%