python操纵windows自动化-pywinauto

  • Post author:
  • Post category:python


想在Windows系统里实现一些程序的点击操作

如果用类似按键精灵的包是pyautogui,也可以针对一个图片进行点击,但都不是很精准;

通过pywinauto可以智能找到相关程序、按钮和菜单,实现启动、关闭、点击,输入文字等功能。



一、pywinauto

特点:通过进程、菜单、按钮等直接控制



(一)入门



1. 安装

安装的时候,遇到了一些插件需要再装

ModuleNotFoundError: No module named ‘win32api’

提示错误,需要安装pypiwin32这个插件:

pip install pypiwin32

又提示错误

ModuleNotFoundError: No module named ‘six’

装完提示,需要comtypes

完成安装



2. 判断程序类型backend

程序的backend有如下两种:

  • win32:MFC, VB6, VCL, simple WinForms,旧应用

    探测方式:Spy++
  • UIA,WinForms, WPF, Store apps, Qt5, browsers,很多都是这类

    探测方式:系统自带工具inspect,目前已升级为Accessibility Insights

    官网下载https://accessibilityinsights.io/

还有一种py_inspect支持以上两种类型

只有当以上工具都探测不到的时候,再考虑用鼠标和键盘



3. 建立入口

  • Application 可以在一个程序的多个实例间切换,每个实例都是一个进程
  • Desktop组件,利用suprocess.Popen 单个程序实例由多进程组成,例如win10的计算器
from pywinauto.application import Application
app = Application(backend="uia").start('notepad.exe')

# describe the window inside Notepad.exe process
dlg_spec = app.UntitledNotepad
# wait till the window is really open
actionable_dlg = dlg_spec.wait('visible')
from subprocess import Popen
from pywinauto import Desktop

Popen('calc.exe', shell=True)
dlg = Desktop(backend="uia").Calculator
dlg.wait('visible')

最终都是timeout,可能我的电脑没有计算器

这两个都是特定的程序类型,后面不用使用具体程序名称。



4. windows窗口规格

>>> dlg_spec = app.window(title='Untitled - Notepad')

>>> dlg_spec
<pywinauto.application.WindowSpecification object at 0x0568B790>

>>> dlg_spec.wrapper_object()
<pywinauto.controls.win32_controls.DialogWrapper object at 0x05639B70>

窗口规则是给后续高级操作用的,有更细节的操作,或在程序还没开启时用

其中wrapper是包装器,我运行报错

另外以下两个语言是一样的效果,相当于是简化了语言

dlg_spec.wrapper_object().minimize() # while debugging
dlg_spec.minimize() # in production
# can be multi-level
app.window(title_re='.* - Notepad$').window(class_name='Edit')

# can combine criteria 组合条件
dlg = Desktop(backend="uia").Calculator
dlg.window(auto_id='num8Button', control_type='Button')



5. 属性解析魔术

空格、逗号、大小写,都可以通过最佳匹配来查找

app.UntitledNotepad
# is equivalent to
app.window(best_match='UntitledNotepad')

一些unicode和特色字符,可以像字典一样

app['Untitled - Notepad']
# is the same as
app.window(best_match='Untitled - Notepad')



6. 如何知道魔术的属性名称

  • By title (window text, name) 按标题
app.Properties.OK.click()
  • By title and control type 按标题和控制类型
app.Properties.OKButton.click()
  • By control type and number 按控制类型和编号
app.Properties.Button3.click() 

(Button0和Button1是同一个按钮,Button2才是下一个)

  • By top-left label and control type 左上标签和控制类型
app.OpenDialog.FileNameEdit.set_text("")
  • By control type and item text 控制类型和项目文本

    app.Properties.TabControlSharing.select(“General”)

不是所有同时可用,针对特定的对话框,可使用以下命令检查名称,通过最佳匹配名称,列出所有的子项:

app.UntitledNotepad.print_control_identifiers()
Control Identifiers:
Notepad - '无标题 - 记事本'    (L251, T293, R642, B681)
['无标题 - 记事本Notepad', 'Notepad', '无标题 - 记事本']
child_window(title="无标题 - 记事本", class_name="Notepad")
   | 
   | Edit - ''    (L259, T344, R634, B651)
   | ['Edit', '无标题 - 记事本Edit'



版权声明:本文为weixin_42665546原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。