Python结合spy++

news2024/12/23 15:34:19

导入必要的库

import win32con
from win32 import win32gui
from win32 import win32clipboard as w
import pyautogui
import sys

1.获取窗口全部属性

def show_window_attr(hwnd):
    """
    显示窗口的属性
    :param hwnd: 窗口句柄(十进制)
    :return: 所有的属性
    WindowName: 窗口标题
    ClassName: 窗口类名
    HwndPy: 窗口句柄(十进制)
    HwndSpy: 窗口句柄(十六进制)
    """
    if not hwnd:
        return
    WindowName = win32gui.GetWindowText(hwnd)
    ClassName = win32gui.GetClassName(hwnd)
    HwndPy = hwnd
    HwndSpy = hex(hwnd)
    return (WindowName, ClassName, HwndPy, HwndSpy)

通过spy++获取的窗口句柄是十六进的,先转化为十进制,可获取全部属性

>>> int(0x31510)
202000
>>> show_window_attr(202000)
('微信', 'WeChatMainWndForPC', 202000, '0x31510')

2.获取全部顶层窗口

def show_top_windows():
    """
    列出所有的顶级窗口及属性
    :return: 全部的顶层窗口及对应属性
    """
    hwndList = []
    win32gui.EnumWindows(
        lambda hwnd, param: param.append(show_window_attr(hwnd)), hwndList
    )
    return hwndList

获取到的是整个桌面所有的窗口

3.模糊查找主窗体

def FindFuzzyTopWindow(FuzzyWindowName=None):
    """
    根据标题模糊查找全部符合条件的主窗体
    :param FuzzyWindowName: 窗口标题部分文字
    :return:
    """
    all_windows = show_top_windows()
    result = []
    for window in all_windows:
        if FuzzyWindowName in window[0]:
            result.append(window)
    return result

有时候我们需要通过“部分名称”来找出主窗体的属性
找出了窗体标题含有“同花顺”的全部窗体

>>> FindFuzzyTopWindow(FuzzyWindowName='同花顺')
[('同花顺(v8.80.80) - 我的板块', 'Afx:400000:b:10003:6:d60ca7',
 134184, '0x20c28'), ('同花顺股灵通', '#32770', 462652, '0x70f3c')]

4.获取目标父窗体下的全部子窗体

def FindSubHandles(pHandle=None, ClassName=None, WinName=None, index=None):
    """
    返回窗体下全部的子窗体,默认主窗体下的窗体
    :param pHandle: 窗口句柄(十进制)
    :param ClassName: 窗口类名,返回特定类名
    :param WinName: 窗口标题,返回特定标题
    :param index: 位置,返回特定位置的窗口
    :return: 包含属性的全部子窗口
    """
    num = 0
    handle = 0
    SubHandlesList = []
    while True:
        # find next handle, return HwndPy
        handle = win32gui.FindWindowEx(pHandle, handle, ClassName, WinName)
        if handle == 0:
            # no more handle
            break
        # get handle attribution
        attr = show_window_attr(handle)
        # append to list
        SubHandlesList.append(tuple(list(attr) + [num]))
        num += 1
    if index is not None:
        return SubHandlesList[index]
    else:
        return SubHandlesList

以刚刚“同花顺”为例,十进制句柄为134184
其他参数在特定场合下会起作用

>>> FindSubHandles(pHandle=134184)
[('', 'Button', 69090, '0x10de2', 0), 
('', 'ToolbarWindow32', 69272, '0x10e98', 1),
 ('', 'msctls_statusbar32', 265490, '0x40d12', 2), 
 ('', 'msctls_statusbar32', 134664, '0x20e08', 3), 
 ('', 'AfxFrameOrView42s', 134212, '0x20c44', 4), 
 ('', 'AfxControlBar42s', 134180, '0x20c24', 5), 
 ('', 'AfxControlBar42s', 134192, '0x20c30', 6), 
 ('功能树', 'AfxControlBar42s', 134194, '0x20c32', 7),
 ('', 'AfxControlBar42s', 134196, '0x20c34', 8), 
 ('HqEmbededTradeContainer', 'Afx:400000:0', 69270, '0x10e96', 9), 
 ('功能树', 'Afx:400000:8:10003:10006e:0', 69320, '0x10ec8', 10), 
 ('', 'Afx:400000:0', 69430, '0x10f36', 11), 
 ('', 'Afx:400000:0', 69432, '0x10f38', 12)]

对比SPY++中的结果完全一致
在这里插入图片描述

5.获取某个父窗口

下面全部的子窗口,遍历所有窗口 这里获取到的是全部层级的子窗口

def ShowAllHandle(pHandle=None, HandleList=[[None]], HandleDict=dict()):
    """
    生成窗口全部对应的关系
    :param pHandle: 目标父窗口
    :param HandleList: 默认为[[None]]
    :param HandleDict: 用于存放对应关系
    :return: 返回目标窗口下全部子父窗口的字典
    """
    sys.setrecursionlimit(1000000)
    if pHandle:
        HandleList[-1][0] = pHandle
        handles = FindSubHandles(HandleList[-1][0][2])
    else:
        handles = FindSubHandles()
    for handle in handles:
        HandleDict[handle] = pHandle
    # 这个根节点已经遍历完,删除
    del HandleList[-1][0]
    # 如果有叶节点,非空,则加入新的叶节点
    if handles:
        HandleList.append(handles)
    # 删除已被清空的根
    HandleList = [HandleGroup for HandleGroup in HandleList if HandleGroup]
    # 如果还有根就继续遍历,否则输出树
    if HandleList:
        return ShowAllHandle(
            pHandle=HandleList[-1][0], HandleList=HandleList, HandleDict=HandleDict
        )
    else:
        return HandleDict

pHandle:这个参数设置为

('同花顺(v8.80.80) - 我的板块', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28')

调用函数,得到一个全部对应关系的字典

ShowAllHandle(FindFuzzyTopWindow(FuzzyWindowName='同花顺')[0])

6.找到特定窗口的路径

因为句柄在不同电脑上的结果是不同的,所以我们要找到这个唯一的路

def FindHandlePath(TargetHandle, num):
    """
    寻找特定窗口的寻找路径
    :param TargetHandle: 窗口句柄(十六进制)
    :param num: 窗口所属index 在spy++内查看
    :return:
    ParentWindow:顶层窗口
    TargetPath:路径的index
    """
    AllPath = ShowAllHandle(pHandle=None, HandleList=[[None]], HandleDict=dict())
    key = tuple(list(show_window_attr(int(TargetHandle))) + [num])
    handlepath = [key]
    while True:
        key = AllPath[key]
        if not key:
            handlepath = handlepath[::-1]
            ParentWindow = handlepath[0]
            TargetPath = [(i[-1]) for i in handlepath[1:]]
            return ParentWindow, TargetPath
        handlepath.append(key)

逻辑是找到全部层级的对应关系,然后反向搜索
以“同花顺”的“买入”按钮为例,通过SPY++查找
在这里插入图片描述

我们得到其十六进制句柄为“0x40D98”,同时index为6
在这里插入图片描述
TargetHandle=0x40D98, num=6
获取路径如下

>>> FindHandlePath(TargetHandle=0x40D98, num=6)
(('同花顺(v8.80.80) - 我的板块', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28', 308), 
[5, 0, 6])

主窗体:同花顺(v8.80.80) - 我的板块
5:第6个子窗体
0:第1个子窗体
6:第7个子窗体

7.根据路径来查找某个特定窗口的句柄

def FindTargetHandle(pHandle, WindowList):
    """
    递归寻找子窗口的句柄
    :param pHandle: 祖父窗口的完整句柄 (WindowName, ClassName, HwndPy, HwndSpy)
    :param WindowList: 子窗口列表
    :return: 目标窗口的完整属性
    """
    for i in range(len(WindowList)):
        pHandle = FindSubHandles(pHandle[2], index=WindowList[i])
    return pHandle

现在,我们已经有了某个窗体的查找路径,通过上述函数来查找其句柄
我们的路径是:(('同花顺(v8.80.80) - 我的板块', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28', 308), [5, 0, 6])
调用函数

>>> FindTargetHandle(('同花顺(v8.80.80) - 我的板块', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28', 308), [5, 0, 6])
('', 'Button', 265624, '0x40d98', 6)

获得到了名为“Button”的窗体,对比我们刚刚的结果,是正确的。

8.根据句柄定位窗体

这个按钮的十进制句柄为:265624
调用函数GetWindowRect

>>> x,y,m,n = win32gui.GetWindowRect(265624)
>>> pyautogui.moveTo((x+m)/2, (y+n)/2)

此时鼠标会自动移动到这个窗体上

参考链接:
· https://www.jb51.net/article/205053.htm

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1583657.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Severt

severt是让我们自己写一些类,然后把这些类给加载Tomcat中,后续Tomcat收到HTTP请求(来自于浏览器),就会执行到咱们上面写的代码.从而通过这些代码,完成一定的业务逻辑. 创建项目 此处创建的是一种新的项目的形式称为Maven项目,Maven是Java 中的一个的构建…

MyBatis 等类似的 XML 映射文件中,当传入的参数为空字符串时,<if> 标签可能会导致 SQL 语句中的条件判断出现意外结果。

问题 传入的参数为空字符串,但还是根据参数查询了。 原因 在 XML 中使用 标签进行条件判断时,需要明确理解其行为。在 MyBatis 等类似的 XML 映射文件中, 标签通常用于动态拼接 SQL 语句的条件部分。当传入的参数 riskLevel 为空字符串时…

【linux】基础IO(四)

在上一篇基础IO中我们主要讲述了文件再磁盘中的存储,当然我们说的也都只是预备知识,为这一篇的文件系统进行铺垫。 目录 搭文件系统的架子:填补细节:inode:datablock[]: 更上层的理解: 搭文件系统的架子&a…

竞赛升温,量子革命待发

人工智能已经吸引了全球企业巨头和政界领袖的广泛关注。同时,一场激烈的全球竞赛正在展开,目标是开发被许多专家视为下一个领域革命性技术的量子计算。 量子计算机利用量子物理学的原理,有潜力推动包括药物研究、股票市场分析以及数据加密在内…

unable to find a medium containing a live file system解决办法!

背景: 用Ventoy制作U盘系统安装盘,只需要把ISO镜像拷进去就可以,可以放多少个镜像取决于U盘的大小,无需重复制作。Ventoy 将U盘的第一个分区默认格式化为exFAT文件系统来存放ISO文件。 但是,今天鲲鹏920平台安装银河…

外包干了15天,技术倒退明显

先说情况,大专毕业,18年通过校招进入湖南某软件公司,干了接近6年的功能测试,今年年初,感觉自己不能够在这样下去了,长时间呆在一个舒适的环境会让一个人堕落! 而我已经在一个企业干了四年的功能…

H.265视频直播点播录像EasyPlayer.js流媒体播放器用户常见问题及解答

EasyPlayer属于一款高效、精炼、稳定且免费的流媒体播放器,可支持多种流媒体协议播放,无须安装任何插件,起播快、延迟低、兼容性强,使用非常便捷。 今天我们来汇总下用户常见的几个问题及解答。 1、EasyPlayer.js播放多路H.265视…

番外篇 | YOLOv8改进之引入YOLOv9的ADown模块 | 替换YOLOv8卷积

前言:Hello大家好,我是小哥谈。YOLOv9是一种目标检测算法,而ADown模块是YOLOv9中的一个重要组成部分。ADown模块主要用于特征提取和下采样操作,以便在后续的检测任务中更好地捕捉目标的特征。具体来说,ADown模块是YOLOv9中的一个卷积块,由一系列卷积层和池化层组成。它的…

类和对象—初阶

目录 1.面向过程和面向对象初步认识 2.类的引入 3.类的定义 4.类的访问限定符及封装 4.1 访问限定符 【面试题】 4.2 封装 【面试题】 5.类的作用域 6.类的实例化 7.类对象模型 7.1 如何计算类对象的大小 7.2 类对象的存储方式 7.3 结构体内存对齐规则 【面试题】…

微服务项目sc2024通用Base工程

1. cloud-provider-payment8001 2.pom文件 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"ht…

106. 跑步锻炼(结果填空)

public class Main { public static void main(String[] args) { int startYear 2000; int startMonth 1; int startDay 1; // 周六 int endYear 2020; int endMonth 10; int endDay 1; // 周四 int totalDistance 0; // 计算开始日期到结束日期之间的每一天 …

【漏洞复现】深澜计费管理系统任意文件读取漏洞

0x01 产品简介 深澜计费管理系统是一套完善的、领先的具有复杂生物型特征的弹性认证计费系统。其主要由以下几个模块组成&#xff1a;AAA认证计费平台、系统运营维护管理平台、用户及策略管理平台、用户自助服务平台、智能客户端模块、消息推送模块以及数据统计模块。该系统为…

蓝桥杯真题 字串简写 前缀和

&#x1f468;‍&#x1f3eb; Acwing 字串简写 输入 4 abababdb a b输出 6&#x1f496; 前缀和 import java.util.Scanner;public class Main {static int N (int) 5e5 10;static int[] l new int[N];// l[i] 表示 i 以及 i 左边包含多少个字符 apublic static void …

前端:自制年历

详细思路可以看我的另一篇文章《前端&#xff1a;自制月历》&#xff0c;基本思路一致&#xff0c;只是元素布局略有差异 ①获取起始位startnew Date(moment().format(yyyy-01-01)).getDay() ②获取总的格子数numMath.ceil(365/7)*7,这里用365或者366计算结果都是一样的371 …

雪花飘,购物抛物线,进度条等四个案列,带入走进 CSS transition

前言 今天从四个案例&#xff0c;我们一起走进 CSS Transition。 源码 以及 在线演示地址 源码地址&#xff1a; 四个案例&#xff0c; CSS Transition 源码 在线演示地址&#xff1a;(兼容移动端) 贝塞尔曲线运动进度条雪花飘飘效果购物车抛物线效果 案例演示 内置贝塞…

魔法阵-蓝桥每日真题

0魔法阵 - 蓝桥云课 (lanqiao.cn) #include <iostream> #include <queue> #include <vector> #include <cstring> #include <algorithm>using namespace std;#define x first #define y second const int N 1010; const int inf 1e4; vector&…

基于Springboot中小企业设备管理系统设计与实现(论文+源码)_kaic

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&a…

java国产化云HIS基层医院系统源码 SaaS模式

目录 ​ 云HIS开发环境 功能模块介绍&#xff1a; 1、门诊模块 2、住院模块 3、药房、药库模块 ​编辑 4、电子病历模块 5、统计报表模块 6、系统管理模块 系统优势 云his之电子病历子系统功能 云 his 系统是运用云计算、大数据、物联网等新兴信息技术&#xff0c;按…

PID c++算法学习和实现

原理图&#xff1a; &#xff08;1&#xff09;位置式PID 是1&#xff1a;当前系统的实际位置&#xff0c;与你想要达到的预期位置的偏差&#xff0c; 2&#xff1a;进行PID控制,误差会一直累加&#xff0c;会使当前输出与过去的所有输入相关&#xff0c;输入uk出错&#xff…

自己写的组件中使用v-model双向绑定

这里的时间选择表单是我写的一个组件&#xff0c;我想用v-model获取到实时的ref值。 代码&#xff1a; //父组件<TimePickerModal v-model:value"time" label-text"计划客面时间" /> const time ref(2024-04-09 15:20:00);//子组件<template>…