LoginSignup
0
0

PyAutoGUIパッケージのlocateOnScreen関数を使った時にOSErrorが発生する原因と回避方法について考えてみた件。(日本語パス問題)

Posted at

実行環境 :computer:

  • Windows 10
  • Python 3.10
    パイソン.jpg

エラーの原因

  PyAutoGUI で画像認識をする時に、pyautogui.locateOnScreen("画像ファイルのパス")というコードを使用するが、その「画像ファイル」までのパス中に 日本語 が含まれていると、 エラー が発生すると思われる。
pyautoguiのlocateOnScreenによるエラーコード.jpg

 「ユーザー」や「パブリック」はエクスプローラー上で日本語表記であるが、パスを取得すると「Users」や「Public」というように英語で表記される。
ユーザー名.jpg
 コードを書く方法ではなく、画像ファイルの命名や配置によってエラーを解消するには、画像ファイルの名前をアルファベットや数字のみにし、ユーザー・フォルダーの直下に画像ファイルを配置する方法がある。ただしユーザー名が日本語だとエラーは解消されないと思われる。

エラーを回避する方法

 画像ファイルの名前に日本語が含まれており、またファイルがデスクトップのフォルダ中などの自由な場所に配置されている状態でも、正しくlocateOnScreen関数が動作するようにする方法について説明していく。注意点としては、画像認識の基となるファイルまでのパスに日本語が含まれてはいけないため、Tempフォルダ内に作成された一時フォルダ内に、さらに日本語が含まれるフォルダを作ったりすると、エラーになると思われる。

重要なモジュール

  • tempfile
    Tempフォルダ.jpg
  • shutil

プログラムの概要 :pencil:

 事前に用意した電卓アプリのスクリーンショット画像を基にPyAutoGUIで画像認識を行い、画面上に電卓アプリが出現した後に、計算式の入力を行う。

手順①
必要なモジュールをインポートする。Python標準ライブラリ(platform, subprocess, time, tempfile, os, shutil)。コマンドラインで「pip install pyautogui」と「pip install opencv_python」を実行してPyAutoGUIとOpenCVを入手してね。
手順②
画像認識に使用する画像ファイルの絶対パスを入力する。(ファイル配置や命名は自由。)
手順③
一時ディレクトリを作成する。(ランダムな文字列がフォルダ名に付与される。)
手順④
複製する画像ファイルの名前を指定する。(日本語が含まれないようにする。)
手順⑤
コピー元となる画像ファイルが存在するか確認し、手順④で決めた名前で画像のコピーを作成する。
手順⑥
電卓アプリを起動し、コピーした画像を基に画面上から電卓アプリを探す。
手順⑦
電卓に入力したいキーを送信する。
手順⑧
コピーした画像ファイルごと、一時フォルダを完全に削除する。

期待通りに動作させるための前提条件

  • 複製する画像ファイルの名前に日本語を使わないこと。
  • ユーザー名に日本語が含まれないこと。
  • コピーした画像ファイルの絶対パス"C:\Users\ユーザー名(英数字)\AppData\Local\Temp\コピーした画像ファイル(英数字)"のどこにも日本語が入っていないこと。

電卓スクリーンショット画像.png

Calculator_AutoGUI改.py
import platform, subprocess, time
import pyautogui

import tempfile
import os
import shutil

# 参照するPNG画像までの絶対パス内に日本語が含まれていると、エラーが発生する。
# 一時的にディレクトリを作成して、エラーを回避してみる。

calculator_png = r"C:\Users\ユーザー名\OneDrive\デスクトップ\任意フォルダ1\任意フォルダ2\任意フォルダ3\電卓スクリーンショット画像.png"



tempdir = tempfile.TemporaryDirectory()

# 生成されたディレクトリ名を確認
print("一時フォルダ:", tempdir.name)

source = calculator_png
name = "calc.png" # 日本語を含まない名前を指定
name = name.replace("/", "-").replace("\\", "-").replace("|", "-").replace("!", "-").replace("<", "-").replace(">", "-").replace("?", "-").replace("*", "-").replace(" ", "-")
# バツ / // \ \\ | ! < > ? *  (全角スペース)
# マル [ ] ( ) { } - _ = ^ ~ ' # $ % & , . + @ ` ; :  (半角スペース)
destination = tempdir.name + "\\" + name
if os.path.isfile(calculator_png):
    calc_png = shutil.copy(source, destination)
else:
    print("画像ファイルのパスが正しくありません。")
    tempdir.cleanup()
    exit()



if platform.system() == 'Windows':
    application = [r'c:\Windows\System32\calc.exe']
elif platform.system() == 'Darwin':
    application = ['open',
                   '/System/Applications/Calculator.app']

subprocess.Popen(application)

coordinate = None
for i in range(10):
    coordinate = pyautogui.locateOnScreen(calc_png,
                                          grayscale=True, confidence=0.8)
    if coordinate is None:
        time.sleep(1)
        print('電卓アプリを探しています・・・')
        continue
    break
if coordinate is None:
    pyautogui.alert('画面上に電卓アプリが見つかりません。用意したスクリーンショット画像が正しくないかもしれません。')
    tempdir.cleanup()
    quit()
print('見つかりました:', coordinate)

if platform.system() == 'Windows':
    pyautogui.write('2*3=', interval=0.3)
else:
    pyautogui.write('6')
print('キー送信完了')

# 最後は下記で一時フォルダを閉じるのを忘れずに。
tempdir.cleanup()
print("一時フォルダの削除完了")

その他

  • エクスプローラー等で対象のファイルやフォルダーを選択した状態(チェックマーク☑が付いた状態)で、Shiftキーを押しながらマウスを右クリックすると、「パスのコピー」という項目が出現する。
    パスのコピー.jpg
  • 一時ディレクトリ(Tempディレクトリ)の場所は、tempfileモジュールの「tempfile.gettempdir()」で確認できる。
  • フォルダ名までのパスを取得するには、osモジュールの「os.path.dirname(ファイルパス文字列)」を使う。
  • ユーザーフォルダ名は5文字で切れる。

参考

[Python] 一時ファイルを作成する(tempfileモジュール)

  • シゴトがはかどるPython自動処理の教科書 (著者:クジラ飛行机)
0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0