如何利用Python和win32编程避免重复性体力劳动(一)——开始、FindWindow和FindWindowEx
2013-12-23 15:06
861 查看
本博客正逐步迁移至OrangeCube四次元
请移步至这里以获得更加的排版和阅读体验,谢谢您
本系列文章假设各位看官对python是足够熟悉的,但却不太了解win32编程。
嘛。。其实我也没学过win32编程,脸请各位看官随意招呼。
需求:最近因为做课题,要把800个FaceGen软件生成的三维面孔保存成图片,以后不排除每一张面孔还要生成某个特质上连续变化的图片。FaceGen以抽取面孔的特征向量来构建面孔,所以保存的文件相当精简,只需要300字节就能无损保存面孔的全部信息。一般的三维工具包是没有办法解析FaceGen软件的fg格式的,而他们自己的SDK又死贵死贵的,3K刀貌似。。嘛,自己动手,丰衣足食。
分析:要做的事情其实很简单,fg文件已经都准备好,我只需要用FaceGen打开它的fg文件,然后直接Save to Image就好了。鉴于一系列原因,不是很想用按键精灵之类的东西。何况利用pywin32可以很方便的利用win32的一些接口,何必再去模拟操作。
其次,为了方面查找目标窗口的句柄,可以下载一个微软自家的Spy++,这玩意儿满大街都是。有了它,还能很方便的查看窗体的消息。
句柄是一个32位整数,在windows中标记对象用,类似一个dict中的key,详情参看这篇文章。
消息是windows应用的重要部分,用来告诉窗体“发生了什么”,比如给一个按钮发送BN_CLICKED这么个消息,按钮就知道“哦,我被点了”,才能执行相应的下一步操作。本文将大量使用消息机制。详情参看这篇文章。
FindWindow(lpClassName=None, lpWindowName=None):
描述:自顶层窗口(也就是桌面)开始搜索条件匹配的窗体,并返回这个窗体的句柄。不搜索子窗口、不区分大小写。找不到就返回0
参数:
lpClassName:字符型,是窗体的类名,这个可以在Spy++里找到。
lpWindowName:字符型,是窗口名,也就是标题栏上你能看见的那个标题。
说明:这个函数我们仅能用来找主窗口。
FindWindowEx(hwndParent=0, hwndChildAfter=0, lpszClass=None, lpszWindow=None);
描述:搜索类名和窗体名匹配的窗体,并返回这个窗体的句柄。不区分大小写,找不到就返回0。
参数:
hwndParent:若不为0,则搜索句柄为hwndParent窗体的子窗体。
hwndChildAfter:若不为0,则按照z-index的顺序从hwndChildAfter向后开始搜索子窗体,否则从第一个子窗体开始搜索。
lpClassName:字符型,是窗体的类名,这个可以在Spy++里找到。
lpWindowName:字符型,是窗口名,也就是标题栏上你能看见的那个标题。
说明:找到了主窗口以后就靠它来定位子窗体啦。
有了这两个函数,我们就可以写出可以定义到任意一个窗体句柄的函数啦:
这样在后续的调用中,我们就能使用我们定义的finde_subHandle来方便地找到某个特定的Edit窗体控件。
比如我们定义一个FaceGenWindow的类:
实体化以后就可以很方便地在类的方法中调用find_subHandle函数来找到FaceGenWindow的子窗体了,比如某个Edit控件:
另外,python中找回来的句柄都是十进制整型,Spy++里显示的都是十六进制整型,这个要注意下,调试的时候用十六进制输出句柄,如下:
请移步至这里以获得更加的排版和阅读体验,谢谢您
本系列文章假设各位看官对python是足够熟悉的,但却不太了解win32编程。
嘛。。其实我也没学过win32编程,脸请各位看官随意招呼。
需求:最近因为做课题,要把800个FaceGen软件生成的三维面孔保存成图片,以后不排除每一张面孔还要生成某个特质上连续变化的图片。FaceGen以抽取面孔的特征向量来构建面孔,所以保存的文件相当精简,只需要300字节就能无损保存面孔的全部信息。一般的三维工具包是没有办法解析FaceGen软件的fg格式的,而他们自己的SDK又死贵死贵的,3K刀貌似。。嘛,自己动手,丰衣足食。
分析:要做的事情其实很简单,fg文件已经都准备好,我只需要用FaceGen打开它的fg文件,然后直接Save to Image就好了。鉴于一系列原因,不是很想用按键精灵之类的东西。何况利用pywin32可以很方便的利用win32的一些接口,何必再去模拟操作。
Part 0: 开始之前?
首先,来这里安装一个Pywin32吧,Pywin32是一个Python库,为python提供访问Windows API的扩展,提供了齐全的windows常量、接口、线程以及COM机制等等。安装后自带一个pythonwin的IDE。或者也可以来这边下载一个ActivePython,整合了pywin32和一些其他的库以及一大堆的支持文档,他们的文档查起来是非常方便的。当然了,WIN32的一些相关函数在MSDN上也能直接找到。其次,为了方面查找目标窗口的句柄,可以下载一个微软自家的Spy++,这玩意儿满大街都是。有了它,还能很方便的查看窗体的消息。
句柄是一个32位整数,在windows中标记对象用,类似一个dict中的key,详情参看这篇文章。
消息是windows应用的重要部分,用来告诉窗体“发生了什么”,比如给一个按钮发送BN_CLICKED这么个消息,按钮就知道“哦,我被点了”,才能执行相应的下一步操作。本文将大量使用消息机制。详情参看这篇文章。
Part 1: 查找窗体句柄
貌似在win32编程的世界里,包括窗口到文本框的所有控件就是窗体,所有的窗体都有独立的句柄。要操作任意一个窗体,你都需要找到这个窗体的句柄,这里,我们就可以用到FindWindow函数和FindWindowEx函数。在pywin32中,他们都属于win32gui的模块。FindWindow(lpClassName=None, lpWindowName=None):
描述:自顶层窗口(也就是桌面)开始搜索条件匹配的窗体,并返回这个窗体的句柄。不搜索子窗口、不区分大小写。找不到就返回0
参数:
lpClassName:字符型,是窗体的类名,这个可以在Spy++里找到。
lpWindowName:字符型,是窗口名,也就是标题栏上你能看见的那个标题。
说明:这个函数我们仅能用来找主窗口。
FindWindowEx(hwndParent=0, hwndChildAfter=0, lpszClass=None, lpszWindow=None);
描述:搜索类名和窗体名匹配的窗体,并返回这个窗体的句柄。不区分大小写,找不到就返回0。
参数:
hwndParent:若不为0,则搜索句柄为hwndParent窗体的子窗体。
hwndChildAfter:若不为0,则按照z-index的顺序从hwndChildAfter向后开始搜索子窗体,否则从第一个子窗体开始搜索。
lpClassName:字符型,是窗体的类名,这个可以在Spy++里找到。
lpWindowName:字符型,是窗口名,也就是标题栏上你能看见的那个标题。
说明:找到了主窗口以后就靠它来定位子窗体啦。
有了这两个函数,我们就可以写出可以定义到任意一个窗体句柄的函数啦:
def find_idxSubHandle(pHandle, winClass, index=0): """ 已知子窗口的窗体类名 寻找第index号个同类型的兄弟窗口 """ assert type(index) == int and index >= 0 handle = win32gui.FindWindowEx(pHandle, 0, winClass, None) while index > 0: handle = win32gui.FindWindowEx(pHandle, handle, winClass, None) index -= 1 return handle def find_subHandle(pHandle, winClassList): """ 递归寻找子窗口的句柄 pHandle是祖父窗口的句柄 winClassList是各个子窗口的class列表,父辈的list-index小于子辈 """ assert type(winClassList) == list if len(winClassList) == 1: return find_idxSubHandle(pHandle, winClassList[0][0], winClassList[0][1]) else: pHandle = find_idxSubHandle(pHandle, winClassList[0][0], winClassList[0][1]) return find_subHandle(pHandle, winClassList[1:])
这样在后续的调用中,我们就能使用我们定义的finde_subHandle来方便地找到某个特定的Edit窗体控件。
比如我们定义一个FaceGenWindow的类:
class FaceGenWindow(object): def __init__(self, fgFilePath=None): self.Mhandle = win32gui.FindWindow("FaceGenMainWinClass", None) print "FaceGen initialization compeleted"
实体化以后就可以很方便地在类的方法中调用find_subHandle函数来找到FaceGenWindow的子窗体了,比如某个Edit控件:
handle = find_subHandle(self.Mhandle, [("ComboBoxEx32", 1), ("ComboBox", 0), ("Edit", 0)])
另外,python中找回来的句柄都是十进制整型,Spy++里显示的都是十六进制整型,这个要注意下,调试的时候用十六进制输出句柄,如下:
print "%x" % (handle)
相关文章推荐
- 如何利用Python和win32编程避免重复性体力劳动(一)——开始、FindWindow和FindWindowEx
- 如何利用Python和win32编程避免重复性体力劳动(三)——文本框操作:WM_GETTEXT 和WM_SETTEXT
- 如何利用Python和win32编程避免重复性体力劳动(四)——下拉列表操作:CB_SETCURSEL、CBN_SELENDOK和CBN_SELCHANGE
- 如何利用Python和win32编程避免重复性体力劳动(三)——文本框操作:WM_GETTEXT 和WM_SETTEXT
- 如何利用Python和win32编程避免重复性体力劳动(二)——菜单操作:GetMenu,GetSubMenu,GetMenuItemID以及wParam的HIWORD&LOWORD
- 如何利用Python和win32编程避免重复性体力劳动(二)——菜单操作:GetMenu,GetSubMenu,GetMenuItemID以及wParam的HIWORD&LOWORD
- 如何利用Python和win32编程避免重复性体力劳动(四)——下拉列表操作:CB_SETCURSEL、CBN_SELENDOK和CBN_SELCHANGE
- 如何安装 wxPython 开始 python GUI 编程
- 利用Python和win32编程范例——按需定制一个按键精灵
- 利用Python和win32编程范例——按需定制一个按键精灵
- 如何利用Python标准库编程
- 利用Python如何生成便签图片详解
- Python高级编程-如何判断字符串a是否是以字符串b开头或结尾?
- win32下利用python操作printer
- 零基础如何开始学习 Python?看完这篇从小白变大牛!
- JAVA中如何利用socket进行网络编程
- 我们如何从VC++开始“编程”?
- Java 如何有效地避免OOM:善于利用软引用和弱引用
- python 读文件 如何从第二行开始
- CUDA GPU编程如何避免Bank conflict