はじめに
PowerShellはポリシーの都合でファイルをダブルクリックして実行という操作で実行するのは少し面倒です。
そこで、任意のファイルでスクリプト実行する方法であるエクスプローラーの「送る」で、PowerShellスクリプトを実行できるか調べてみました。
送られたファイルに日時の文字列をつけてコピーするスクリプトをサンプルとして作成したのでまとめておきます。
2019/01/30 更新
括弧を含むパスの場合の問題点と対策をページ下に追記しました。
「送る」機能について
エクスプローラーの右クリックで表示される「送る」の動作の説明をしておくと、
- SendToのフォルダにショートカットを登録すると「送る」の項目に表示される
- 送られたファイルのパスは実行時のパラメータとして渡される
という2つの動作になっています。
ショートカットの作成処理とファイルを受け取る側の2つに分けて解説します。
「送る」のショートカット作成
こちらの記事でスタートアップにショートカットを作成していますが、これとほぼ同じです。
sendtoのパスを取得して、ショートカットを作成します。
# 日本語のショートカットを作成する場合は UTF-8 with BOM で保存すること.
function create_sendto_shortcut($script_path, $name){
# sendtoのパスを取得.
$sendto_path = [Environment]::GetFolderPath([Environment+SpecialFolder]::SendTo)
$sendto_shortcut_path = [System.IO.Path]::Combine($sendto_path, $name + ".lnk" )
# ショートカットの有無確認.
if(![System.IO.File]::Exists($sendto_shortcut_path)){
# ショートカット生成.
$WshShell = New-Object -ComObject WScript.Shell
$ShortCut = $WshShell.CreateShortcut($sendto_shortcut_path)
$ShortCut.TargetPath = "powershell.exe"
$ShortCut.Arguments = "-NoProfile -ExecutionPolicy Unrestricted " + $script_path
# $ShortCut.WindowStyle = 7 # 最小化する場合はコメント外す
$ShortCut.Save()
}
}
# 実行するps1ファイルのフルパス.
$current_path = Split-Path -Parent $MyInvocation.MyCommand.Path
$sendto_script = [System.IO.Path]::Combine($current_path, "sendto.ps1")
create_sendto_shortcut $sendto_script "日時追加コピー"
送られたファイルを受け取る
PowerShellでは実行時のパラメータは$Argsにパラメータが格納されているそうなので、こちらを使います。
先頭パラメータが $Args[0]に格納されているはずなのですが、動作を確認したところ、スペースを含むパスの場合は別のパラメータと認識されてしまうので、連結させる必要がありました。
# パラメータ配列をパス別に再構築.
$files_args = New-Object System.Collections.ArrayList
foreach($arg in $Args){
if([System.IO.Path]::IsPathRooted($arg)){
$files_args.Add($arg) | Out-Null
}
else{
# ファイルパスの先頭でない文字列は前の文字列の後にスペースで結合.
$files_args[$files_args.Count-1] = $files_args[$files_args.Count-1] + " " + $arg
}
}
# 日時文字列.
$datetime = (Get-Date).ToString("yyyyMMddHHmmss")
foreach($file_path in $files_args){
Write-Output ("file_path:"+$file_path)
if(Test-Path $file_path){
# パスの分解.
$folder = [System.IO.Path]::GetDirectoryName($file_path)
$file = [System.IO.Path]::GetFileNameWithoutExtension($file_path)
$ext = [System.IO.Path]::GetExtension($file_path)
# コピー先パスの作成.
$new_file_path = [System.IO.Path]::Combine($folder, $file + "_" + $datetime + $ext )
# ファイルコピー.
Copy-Item $file_path $new_file_path -Recurse
}
}
動作確認
ファイルドロップに応用
ショートカットにファイルをドロップする操作は、「送る」と同じ動作になります。
デスクトップにショートカットを置くことでドロップ用に使う場合は以下のようにします。
# 日本語のショートカットを作成する場合は UTF-8 with BOM で保存すること.
function create_desktop_shortcut($script_path, $name){
# Desktopのパスを取得.
$desktop_path = [Environment]::GetFolderPath([Environment+SpecialFolder]::Desktop)
$desktop_shortcut_path = [System.IO.Path]::Combine($desktop_path, $name + ".lnk" )
# ショートカットの有無確認.
if(![System.IO.File]::Exists($desktop_shortcut_path)){
# ショートカット生成.
$WshShell = New-Object -ComObject WScript.Shell
$ShortCut = $WshShell.CreateShortcut($desktop_shortcut_path)
$ShortCut.TargetPath = "powershell.exe"
$ShortCut.Arguments = "-NoProfile -ExecutionPolicy Unrestricted " + $script_path
# $ShortCut.WindowStyle = 7 # 最小化する場合はコメント外す
$ShortCut.Save()
}
}
# 実行するps1ファイルのフルパス.
$current_path = Split-Path -Parent $MyInvocation.MyCommand.Path
$sendto_script = [System.IO.Path]::Combine($current_path, "sendto.ps1")
create_desktop_shortcut $sendto_script "日時追加コピー"
動作確認
おわりに
「送る」やファイルドロップでのファイル受け渡しは、ユーザービリティ向上が大きいので覚えておいて損はない小技だと思います。
致命的な問題がありました(2018/12/27追記)
括弧()を含むファイル名だと送るときに、PowerShell側がファイル名文字列と認識せずにエラーになっていることに気づきました。
エスケープすれば動作するのですが、ショートカットから渡す引数の文字列が編集できないので、回避するのは難しそうです。
回避案:コンテキストメニューに登録
次の案としては、コンテキストメニューに登録して、引数のファイル名をエスケープぜずに渡す方法です。
管理者権限がなくても登録可能なHKCU:\Software\Classes*\shell\ のレジストリに追加します。
regファイルで追加してもいいのですが、せっかくなのでPowerShellで登録します。
# 実行するps1ファイルのフルパス.
$current_path = Split-Path -Parent $MyInvocation.MyCommand.Path
$sendto_script = [System.IO.Path]::Combine($current_path, "sendto.ps1")
# 登録するコマンド. ※%1をシングルクォートで括る
$context_command = "powershell.exe -NoProfile -ExecutionPolicy Unrestricted " + $sendto_script + " '%1'"
# レジストリ登録. HKCUは管理者権限不要.
New-Item HKCU:\Software\Classes\*\shell\日時追加コピー
Set-ItemProperty HKCU:\Software\Classes\*\shell\日時追加コピー -name "(default)" -value "日時追加コピー"
New-Item HKCU:\Software\Classes\*\shell\日時追加コピー\command
Set-ItemProperty HKCU:\Software\Classes\*\shell\日時追加コピー\command -name "(default)" -value $context_command
この方法だと最小化できないのでウィンドウが表示されてしまうのが課題です。
batファイルを利用した対策(2019/01/30追記)
名前に括弧が含まれるファイルを送ることができない問題をbatファイルを経由することで解決しました。
括弧が含まれるとエラーになる例
powershell -NoProfile -ExecutionPolicy Unrestricted .\sendto.ps1 .\aa(b).txt
実行結果
b : 用語 'b' は、コマンドレット、関数、スクリプト ファイル、または操作可能なプログラムの名前として認識されません。名前
が正しく記述されていることを確認し、パスが含まれている場合はそのパスが正しいことを確認してから、再試行してください。
発生場所 行:1 文字:71
+ ... ell -NoProfile -ExecutionPolicy Unrestricted .\sendto.ps1 .\aa(b).txt
+ ~
+ CategoryInfo : ObjectNotFound: (b:String) [], CommandNotFoundException
+ FullyQualifiedErrorId : CommandNotFoundException
この問題に対してbatファイルでシングルクォートを付けることで回避します。
余計なファイルが増えるし、経由するので遅くなりますが、これ以上の方法が思い浮かばなかった。
対策版のコード
batファイルに渡されたパラメータpathを'path'の形式にします。
複数パラメータ渡された場合はスペース区切りの1つの文字列にします。
この文字列をps1ファイルのパラメータとして渡します
@echo off
set ARGS_PARAM=
:loop
if "%~f1" == "" goto end
set ARGS_PARAM=%ARGS_PARAM% '%~f1'
shift
goto loop
:end
powershell -NoProfile -ExecutionPolicy Unrestricted ".\sendto.ps1" %ARGS_PARAM%
sendto.bat内で相対パスでps1を呼ぶために、ショートカットを作るときに作業フォルダーを指定しておきます。
function create_sendto_shortcut($script_path,$link_name,$work_path){
# sendtoのパスを取得.
$sendto_path = [Environment]::GetFolderPath([Environment+SpecialFolder]::SendTo)
$sendto_shortcut_path = [System.IO.Path]::Combine($sendto_path, $link_name + ".lnk" )
# ショートカットの有無確認.
$is_shortcut = (Test-Path $sendto_shortcut_path)
if(!$is_shortcut){
# ショートカット生成.
$WshShell = New-Object -ComObject WScript.Shell
$ShortCut = $WshShell.CreateShortcut($sendto_shortcut_path)
$ShortCut.TargetPath = $script_path
$ShortCut.WorkingDirectory = $work_path
# $ShortCut.WindowStyle = 7 # 最小化する場合はコメント外す
$ShortCut.Save()
}
}
# 実行するbatファイルパス.
$current_path = Split-Path -Parent $MyInvocation.MyCommand.Path
$sendto_script = [System.IO.Path]::Combine($current_path, "sendto.bat")
create_sendto_shortcut $sendto_script "日時追加コピー" $current_path
$Argsには必ず'で括られたパスの配列が渡される前提で実装します。
$files_args = $Args
# 日時文字列.
$datetime = (Get-Date).ToString("yyyyMMddHHmmss")
foreach($file_path in $files_args){
Write-Output ("file_path:"+$file_path)
if(Test-Path $file_path){
# パスの分解.
$folder = [System.IO.Path]::GetDirectoryName($file_path)
$file = [System.IO.Path]::GetFileNameWithoutExtension($file_path)
$ext = [System.IO.Path]::GetExtension($file_path)
# コピー先パスの作成.
$new_file_path = [System.IO.Path]::Combine($folder, $file + "_" + $datetime + $ext )
# ファイルコピー.
Copy-Item $file_path $new_file_path -Recurse
}
}
これで括弧を含むファイル名を送られても処理することが可能になりました。
私PowerShellだけど…シリーズ
私PowerShellだけど、君のタスクトレイで暮らしたい
私powershellだけどタスクトレイの片隅でアイを叫ぶ
私PowerShellだけど子を持つ親になるのはいろいろ大変そう
私PowerShellだけどあなたにトーストを届けたい(プログレスバー付)
私Powershellだけど日付とサイズでログを切り替えたい(log4net)
私PowerShellだけどスクリプトだけでサービス登録したい
私PowerShellだけどスクリーンショットを撮影したい
私PowerShellだけどマウスカーソル付きのスクリーンショットを撮影したい