在中文windows下使用pywinauto进行窗口操作

这两天开始接触pywinauto,听说百度的自动化QA也用这个模块,于是来了兴趣,但网上的教程很少,而且基本上都是拿官方的notepad来说,首先中文菜单的支持是问题,其次各种操作也没有写清楚,阅读官方的文档,发现这个东西使用起来还真是非常方便,下面我也以notepad为例来说明一下它的简单操作。

安装

1. pywinauto  https://sourceforge.net/project/showfiles.php?group_id=157379

2. Sendkeys http://www.rutherfurd.net/python/sendkeys/index.html

3.ctypes (如果你是python2.3或者2.4)

检测你是否安装正确

>>> from pywinauto import application
>>> app = application.Application.start("notepad.exe")
>>> app.notepad.TypeKeys("%FX")

都安装好了以后,我们来正式进入pywinauto的世界

一、启动程序

1
2
from pywinauto import application
app = application.Application.start('notepad.exe')

start() 函数里也可以接路径+程序名

使用spy++lite查看notepad的信息

1

里面的窗口类名与标题文本相关重要,以后的查找窗口基本上都要用的到

现在我们来点击“帮助->关于记事本”操作

1
app.Notepad.MenuSelect('帮助->关于记事本'.decode('gb2312'))

这里的app是你刚才实例的对象,Notepad是类名,可以从spy++lite中看到,MenuSelect方法可以自动检索Notepad上的菜单选项,

decode(‘gb2312’)方法是把中文强制转换为unicode编码,对于非英文的操作系统都是要转换的,后面还有更简单的方法

二、查找“关于记事本”的窗口

还是使用spy++lite来查看“关于记事本”的信息

2

窗口类名:#32770

标题文字:关于“记事本”

官方法文档中有以下两个方法

1. 通过top_dlg = app.top_window_() 来获得最上面的window,但是官方并不推荐这种方式,目前来说这个“关于记事本”是最上面,但是也不能保证在测试的进程当中有什么意外的进程跑到了上面,一旦有新的进程,那么得到的就是一个错误的对象

2.通过find_dlg = app.window_(title_re = ‘’, class_name = ‘’) 方法获得,这也是为什么我上面说标题文本与窗口类名非常重要的原因,title_re和 class_name这两个可以单独使用也可以一块使用,因为有时没有标题文本,也有时一个窗口类名有多个对象,比如“Edit”有时当一个对话框中有多个输入框时会有多个Edit类名,对于“关于记事本”我们可以通过以下代码获得

1
about_dlg = app.window_(title_re = u"关于", class_name = "#32770")

中文要进行unicode编码,这里也可以通过decode(‘gb2312’)方法实现,但是不如输入一个U省事~ 呵呵

我们print一下得到的about_dlg

<pywinauto.application.WindowSpecification object at 0x01F0A530>

说明我们得到的是一个application.WindowSpecification对象

三、在”关于记事本”窗口上找到“确定”按钮(button)

在pywinauto中,对话框下面的是controller,button,checkbox,textbox等都是controller

我们可以使用print_control_identifiers() 方法来打印出该窗口中所有的controller

1
about_dlg.print_control_identifiers()

会得到以下的输出

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
Control Identifiers:
Static - ''   (L312, T265, R738, B267)
 '' '0' '1' 'Static' 'Static0' 'Static1'
Static - ''   (L308, T280, R340, B313)
 '2' 'Static2'
Static - 'Microsoft Windows'   (L350, T280, R695, B295)
 'Microsoft Windows' 'Microsoft WindowsStatic' 'Static3'
Static - 'u7248u672c 6.1 (u5185u90e8u7248u672c 7601: Service Pack 1)'   (L350, T295, R748, B310)
 'Static4' 'u7248u672c 6.1 (u5185u90e8u7248u672c 7601: Service Pack 1)' 'u7248u672c 6.1 (u5185u90e8u7248u672c 7601: Service Pack 1)Static'
Static - 'u7248u6743u6240u6709 xa9 2009 Microsoft Corporationu3002u4fddu7559u6240u6709u6743u5229u3002'   (L350, T310, R710, B325)
 'Static5' 'u7248u6743u6240u6709 xa9 2009 Microsoft Corporationu3002u4fddu7559u6240u6709u6743u5229u3002' 'u7248u6743u6240u6709 xa9 2009 Microsoft Corporationu3002u4fddu7559u6240u6709u6743u5229u3002Static'
Static - 'Windows 7 u65d7u8230u7248 u64cdu4f5cu7cfbu7edfu53cau5176u7528u6237u754cu9762u53d7u7f8eu56fdu548cu5176u4ed6u56fdu5bb6/u5730u533au7684u5546u6807u6cd5u548cu5176u4ed6u5f85u9881u5e03u6216u5df2u9881u5e03u7684u77e5u8bc6u4ea7u6743u6cd5u4fddu62a4u3002'   (L350, T325, R710, B385)
 'Static6' 'Windows 7 u65d7u8230u7248 u64cdu4f5cu7cfbu7edfu53cau5176u7528u6237u754cu9762u53d7u7f8eu56fdu548cu5176u4ed6u56fdu5bb6/u5730u533au7684u5546u6807u6cd5u548cu5176u4ed6u5f85u9881u5e03u6216u5df2u9881u5e03u7684u77e5u8bc6u4ea7u6743u6cd5u4fddu62a4u3002' 'Windows 7 u65d7u8230u7248 u64cdu4f5cu7cfbu7edfu53cau5176u7528u6237u754cu9762u53d7u7f8eu56fdu548cu5176u4ed6u56fdu5bb6/u5730u533au7684u5546u6807u6cd5u548cu5176u4ed6u5f85u9881u5e03u6216u5df2u9881u5e03u7684u77e5u8bc6u4ea7u6743u6cd5u4fddu62a4u3002Static' 'Windows 7 u65d7u8230u7248 u64cdu4f5cu7cfbu7edfu53cau5176u7528u6237u754cu9762u53d7u7f8eu56fdu548cu5176u4ed6u56fdu5bb6/u5730u533au7684u5546u6807u6cd5u548cu5176u4ed6u5f85u9881u5e03u6216u5df2u9881u5e03u7684u77e5u8bc6u4ea7u6743u6cd5u4fddu62a4u3002Static0' 'Windows 7 u65d7u8230u7248 u64cdu4f5cu7cfbu7edfu53cau5176u7528u6237u754cu9762u53d7u7f8eu56fdu548cu5176u4ed6u56fdu5bb6/u5730u533au7684u5546u6807u6cd5u548cu5176u4ed6u5f85u9881u5e03u6216u5df2u9881u5e03u7684u77e5u8bc6u4ea7u6743u6cd5u4fddu62a4u3002Static1'
Static - ''   (L350, T385, R665, B415)
 'Static7' 'Windows 7 u65d7u8230u7248 u64cdu4f5cu7cfbu7edfu53cau5176u7528u6237u754cu9762u53d7u7f8eu56fdu548cu5176u4ed6u56fdu5bb6/u5730u533au7684u5546u6807u6cd5u548cu5176u4ed6u5f85u9881u5e03u6216u5df2u9881u5e03u7684u77e5u8bc6u4ea7u6743u6cd5u4fddu62a4u3002Static2'
SysLink - 'u6839u636e <A>Microsoft u8f6fu4ef6u8bb8u53efu6761u6b3e</A>uff0cu672cu4ea7u54c1u4f7fu7528u6743u5c5eu4e8e:'   (L350, T415, R665, B445)
 'SysLink' 'u6839u636e <A>Microsoft u8f6fu4ef6u8bb8u53efu6761u6b3e</A>uff0cu672cu4ea7u54c1u4f7fu7528u6743u5c5eu4e8e:' 'u6839u636e <A>Microsoft u8f6fu4ef6u8bb8u53efu6761u6b3e</A>uff0cu672cu4ea7u54c1u4f7fu7528u6743u5c5eu4e8e:SysLink'
Static - 'kevin'   (L365, T445, R680, B460)
 'Static8' 'kevin' 'kevinStatic' 'kevinStatic0' 'kevinStatic1'
Static - ''   (L365, T460, R680, B475)
 'Static9' 'kevinStatic2'
Button - 'u786eu5b9a'   (L672, T503, R747, B524)
 'Button' 'u786eu5b9a' 'u786eu5b9aButton'

static,SysLink,button等是它类型,后面接的是title,都是unicode的,这里面就有没有title的controller,再后面的(L,T,R,B)是这个控件的位置,分别对应着左上右下

在”关于记事本”窗口上找到“确定”按钮,可以通过app.window_()方法,传入的参数可以是title,也可以是class_name,所以我说这两个值相当重要,一直在用,这里的title支持正则表达式,非常方便

在app上先找到about_dlg,然后再about_dlg上找确定button

app.window_(title_re = u'关于“记事本”').window_(title_re = u'确定'),然后通过Click()方法来单击这个button

另外一种方法也是官方推荐的在非英文系统下的方法

1
2
OK = u'确定'
about_dlg[OK].Click()

这个的意思就是在about_dlg下找到u’确定’,看起来比上面要简练好理解,理解了这种方式,接下来还有更简单的,都不用找about_dlg

直接 app[u'关于“记事本”'][u'确定'].Click()

四、在记事本里写点东西

这个其实在校验pywinauto的时候已经做过了全用TypeKeys函数,但是这里如果要输入中文还是要u一下

1
app.notepad.TypeKeys(u"杨彦星")

五、一个比较恶心的问题

在MenuSelect函数中不支持正则,完全是全文匹配,如我输入

dig = app.Notepad.MenuSelect("编辑->替换".decode('gb2312')) 是找不到对象的

必须要

dig = app.Notepad.MenuSelect("编辑(E)->替换(R)".decode('gb2312')) 这样才行,得把(R) (E)写上才行,但是奇怪的是上面的“帮助->关于记事本”就不用输入,所以说是一个挺恶心的问题,我也不知道这是为什么……

最后把上面的函数合并一下,跑下来应该会很快

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
#! /usr/bin/env python
#coding=gbk
 
 
import time
from pywinauto import application
app = application.Application.start('notepad.exe')
app.Notepad.MenuSelect('帮助->关于记事本'.decode('gb2312'))
time.sleep(.5)
 
#这里有两种方法可以进行定位“关于记事本”的对话框
#top_dlg = app.top_window_() 不推荐这种方式,因为可能得到的并不是你想要的
about_dlg = app.window_(title_re = u"关于", class_name = "#32770")#这里可以进行正则匹配title
#about_dlg.print_control_identifiers()
app.window_(title_re = u'关于“记事本”').window_(title_re = u'确定').Click()
app.Notepad.MenuSelect('帮助->关于记事本'.decode('gb2312'))
time.sleep(.5) #停0.5s 否则你都看不出来它是否弹出来了!
ABOUT = u'关于“记事本”'
OK = u'确定'
#about_dlg[OK].Click()
#app[ABOUT][OK].Click()
app[u'关于“记事本”'][u'确定'].Click()
 
app.Notepad.TypeKeys(u"杨彦星")
dig = app.Notepad.MenuSelect("编辑(E)->替换(R)".decode('gb2312'))
Replace = u'替换'
Cancle = u'取消'
time.sleep(.5)
app[Replace][Cancle].Click()
dialogs = app.windows_()

转自:

http://my.oschina.net/yangyanxing/blog/167042

原文地址:https://www.cnblogs.com/testlife007/p/4710599.html