このページの本文へ

Windows Info 第484回

WindowsのコマンドラインでGrapheme Clusterを正しく扱うには

2025年06月01日 10時00分更新

文● 塩田紳二 編集● ASCII

  • この記事をはてなブックマークに追加
  • 本文印刷

 PowerShell 7で文字列中にZWJ(Zero Width Joiner)などを使った複雑な絵文字を入れると、カーソル位置がずれてしまう。これは、コマンドラインのユニコード文字列のGrapheme Clusterを正しく判定していないからだ。

 Grapheme Clusterとは、人間にとって1文字に見えるコードポイントの連なりである。この件は過去記事(「ユニコードで文字数を数える方法」)に書いた。今回はコマンドラインでの問題を考えてみる。なお評価は、Windows 11 Ver.24H2上のPowerShell Ver.7.5.1でしている。

原因はPSReadLineにある

 では、原因はどこにあるのだろうか? まず、現在のWindowsターミナルは、Ver.1.22(現在の安定版)から、Grapheme Clusterを正しく判定できるようになっている。そして問題は、PSReadLineにある。PSReadLineはヒストリなどの関係で、入力コマンドラインを扱っており、このときコマンドライン文字列を単純にUnicode(UTF-16)文字単位でしか見ていないのだ。

 これは、PSReadLineをオフにしてみるとわかる。以下の画面は、その様子である。

Windows

上ではPSReadLineが有効だと、Grapheme Clusterが正しく認識されず、カーソル位置がずれてしまっている。下ではremove-moduleコマンドを使いPSReadLineをオフにしたので、Grapheme Clusterが正しく認識され、ダブルクオートの直後にカーソルが来るようになった

 複雑な絵文字をコピー&ペーストでコマンドラインに貼り付けると、カーソル位置がずれてしまう。これは、PSReadLineがGrapheme Clusterではなく、Unicode文字(UTF-16)単位でしか扱っていないからだ。PSReadLineはGitHub(https://github.com/PowerShell/PSReadLine)にソースコードがある。今のところ、この部分がすぐに修正される様子はなさそうなので、当面は必要に応じてPSReadLineをオン/オフして使うしかないだろう。

 PSReadLineをオフにすると、コマンドラインでは、同じ複雑な絵文字をダブルクオートの中に入れてもカーソル位置がずれることはなくなる。これは、PowerShellがGrapheme Clusterを正しく認識しているからだ。

 こうした複雑な絵文字や異体字セレクタ(異体字シーケンス)などを含む文字列をコマンドの引数にする場合、一時的にPSReadLineを停止した方が編集処理が簡単になる。

 ただし、PSReadLineを停止してしまうと、履歴が「組み込みヒストリ」になってしまい、補完方法などの変更ができなくなる。また、コマンドラインの色分けも行われない。このあたりに関しても過去記事(「Windowsでのコマンドラインのヒストリ機能」)を参照してほしい。

PSReadLineをオン/オフする

 PSReadLineは、モジュールとして読み込まれているため、これを削除することで動作を停止できる。具体的には、以下のようにすることでPSReadLineを停止できる。

remove-Module PSReadLine

 再開させるには、PSReadLineモジュールをインポートする。

import-module PSReadLine

 なお、セッションの途中でPSReadLineを停止し、その後上記のコマンドで再開した場合でも、履歴やPSReadLineOptionの設定は、停止前のものが残っている。なので、気軽に停止してもかまわない。再開すれば元の状態に戻る。

 ただし、停止中に実行したコマンドに関しては、組み込み履歴側に記録される。PSReadLineモジュールが組み込まれているかどうかは、以下のコマンドで調べられる。

get-module PSReadLine

 PSReadLineが組み込まれていればモジュールのプロパティが表示される。スクリプトなどからは、以下のようにして判定ができる。

if ((Get-Module PSReadline) -ne $null){ "PSReadLine is Exist"} else {"No PSReadLine"}

 PSReadLineがオフの場合、モジュールが存在しないため、Get-Moduleコマンドはヌル($null)を返す。

オン/オフの頻度が高いなら、もう一工夫

 文字列に絵文字や異体字セレクタを多用し、これを編集する頻度も高いようなら、毎回コマンドを打ち込むのも面倒なので、関数を定義し、短いエイリアスを定義しておく。

function Enable-PSReadline(){
    import-module -Name PSReadLine
}
function Disable-PSReadline(){
    remove-module -Name PSReadLine
}
set-alias -name epsrl -value Enable-PSReadline
set-alias -name dpsrl -value Disable-PSReadline

 ここでは、エイリアスとして「epsrl」(有効化、Enable PSReadLine)と「dpsrl」(無効化、Disable PSReadLine)を定義した。最初から短い名前の関数にしないのは、Get-Commandからコマンドを探しやすくするためだ。

 PowerShellでは、「Verb "-" Noun」形式のコマンドをfunctionで定義すると、Get-Commandが動詞(Verb)、名詞(Noun)で検索が可能になる。このようにすることで、コマンドの綴りを忘れても、対象となるPSReadLineというキーワードを使って、「Get-Command -Noun psread*」などとして検索できるわけだ。

 また、PowerShellのプロンプトでPSReadLineのオン/オフを表示できれば、混乱も少ないだろう。それには、Prompt関数を書き換え、Get-Moduleコマンドで、PSReadLineの状態を判定してプロンプト文字列を切り替える(リスト02#%PROMPT%#)。PSReadLineがオンならば、プロンプトの右端は「>」となり、オフならば「:」になる。

function prompt() {
    $promptString+="PS$($PSVersionTable.PSVersion.Major).$($PSVersionTable.PSVersion.Minor)"
    if ((Get-Module -Name PSReadline).Name -eq "PSReadLine") {
        $promptString+="> "
    } else {
        $promptString+=": "
    }
    return $promptString
}

 これらは、PowerShellのプロファイルに記述しておく。PowerShellのプロファイルに関しては過去記事(「WindowsのPowerShellのプロファイルを設定する」)を参照してほしい。

組み込み履歴のキーボードショートカット

 前述のように、PSReadLineが停止すると、組み込み履歴機能が有効になり、PowerShell標準のコマンドライン編集キーに切り替わる。

Windows

 これは、PSReadLineとはちょっと異なる割り当てになっている。キーボードショートカットの詳細に関しては、Microsoftのページ「about_Line_Editing - PowerShell」(https://learn.microsoft.com/ja-jp/powershell/module/microsoft.powershell.core/about/about_line_editing?view=powershell-7.5)を参考にしてほしい。また、組み込み履歴では、以下の4つのコマンドが利用できる。

Add-History:履歴を追加
Clear-History:履歴をクリア
Get-History:履歴リスト
Invoke-History:履歴を実行

 ただし、履歴リストを使うならGet-HistoryよりF7キーの方が便利だ。

 PSReadLineは便利だが、Grapheme Clusterに対応していないところが、ちょっとイタイ。もちろん、絵文字や異体字セレクタを使わないのであれば、問題はないのだが、逆に多用している場合、履歴の編集などが面倒になる。

カテゴリートップへ

この連載の記事

ASCII倶楽部

注目ニュース

  • 角川アスキー総合研究所

プレミアム実機レビュー

ピックアップ
1
KIOXIA(キオクシア) 旧東芝メモリ microSD 128GB UHS-I Class10 (最大読出速度100MB/s) Nintendo Switch動作確認済 国内サポート正規品 メーカー保証5年 KLMEA128G
KIOXIA(キオクシア) 旧東芝メモリ microSD 128GB UHS-I Class10 (最大読出速度100MB/s) Nintendo Switch動作確認済 国内サポート正規品 メーカー保証5年 KLMEA128G
¥2,280
2
Anker PowerLine III Flow USB-C & USB-C ケーブル Anker絡まないケーブル 240W 結束バンド付き USB PD対応 シリコン素材採用 iPhone 17 / 16 / 15 / Galaxy iPad Pro MacBook Pro/Air 各種対応 (1.8m ミッドナイトブラック)
Anker PowerLine III Flow USB-C & USB-C ケーブル Anker絡まないケーブル 240W 結束バンド付き USB PD対応 シリコン素材採用 iPhone 17 / 16 / 15 / Galaxy iPad Pro MacBook Pro/Air 各種対応 (1.8m ミッドナイトブラック)
¥1,890
3
Anker USB Type C ケーブル PowerLine USB-C & USB-A 3.0 ケーブル iPhone 17 / 16 / 15 /Xperia/Galaxy/LG/iPad Pro/MacBook その他 Android 等 USB-C機器対応 テレワーク リモート 在宅勤務 0.9m ホワイト
Anker USB Type C ケーブル PowerLine USB-C & USB-A 3.0 ケーブル iPhone 17 / 16 / 15 /Xperia/Galaxy/LG/iPad Pro/MacBook その他 Android 等 USB-C機器対応 テレワーク リモート 在宅勤務 0.9m ホワイト
¥740
4
UGREEN USB Type Cケーブル PD対応 100W/5A 超急速充電 USB C ナイロン編み 断線防止 iphone17/16/15シリーズ/iPad/MacBook Pro/Galaxy S24/Matebook/iPad/Xperia等USB-C各種対応(1m, ブラック)
UGREEN USB Type Cケーブル PD対応 100W/5A 超急速充電 USB C ナイロン編み 断線防止 iphone17/16/15シリーズ/iPad/MacBook Pro/Galaxy S24/Matebook/iPad/Xperia等USB-C各種対応(1m, ブラック)
¥743
5
Anker iPhone充電ケーブル PowerLine II ライトニングケーブル MFi認証 超高耐久 iPhone 14 / 14 Pro Max / 14 Plus / 13 / 13 Pro / 12 / 11 / X/XS/XR / 8 Plus 各種対応 (0.9m ホワイト)
Anker iPhone充電ケーブル PowerLine II ライトニングケーブル MFi認証 超高耐久 iPhone 14 / 14 Pro Max / 14 Plus / 13 / 13 Pro / 12 / 11 / X/XS/XR / 8 Plus 各種対応 (0.9m ホワイト)
¥990
6
KIOXIA(キオクシア)【日本製】USBフラッシュメモリ 32GB USB2.0 国内サポート正規品 KLU202A032GW
KIOXIA(キオクシア)【日本製】USBフラッシュメモリ 32GB USB2.0 国内サポート正規品 KLU202A032GW
¥980
7
KIOXIA(キオクシア)【日本製】SDカード 64GB SDXC UHS-I Class10 読出速度100MB/s 国内正規品 メーカー保証5年 KLNEA064G
KIOXIA(キオクシア)【日本製】SDカード 64GB SDXC UHS-I Class10 読出速度100MB/s 国内正規品 メーカー保証5年 KLNEA064G
¥1,180
8
エルパ(ELPA) 扉付タップラン 電源タップ 延長コード 125V 3m 3個口 ホワイト WBT-N3030B(W)
エルパ(ELPA) 扉付タップラン 電源タップ 延長コード 125V 3m 3個口 ホワイト WBT-N3030B(W)
¥652
9
NIMASO ガラスフィルム iPad 第11世代(A16) 2025用/iPad 10.9インチ 第10世代 2022用 衝撃吸収 強化 ガラス 保護フィルム 指紋防止 ガイド枠付き NTB22I574
NIMASO ガラスフィルム iPad 第11世代(A16) 2025用/iPad 10.9インチ 第10世代 2022用 衝撃吸収 強化 ガラス 保護フィルム 指紋防止 ガイド枠付き NTB22I574
¥1,359
10
バッファロー マウス 無線 ワイヤレス 5ボタン 【戻る/進むボタン搭載】 小型 軽量 節電モデル 最大584日使用可能 BlueLED ブラック BSMBW315BK
バッファロー マウス 無線 ワイヤレス 5ボタン 【戻る/進むボタン搭載】 小型 軽量 節電モデル 最大584日使用可能 BlueLED ブラック BSMBW315BK
¥1,040

Amazonのアソシエイトとして、ASCII.jpは適格販売により収入を得ています。

デジタル用語辞典

ASCII.jpメール デジタルMac/iPodマガジン