当前位置: 首页 > news >正文

在C++中实现一个能够捕获弹窗、检查内容并在满足条件时点击按钮的程序;使用python的方案

在C++中实现一个能够捕获弹窗、检查内容并在满足条件时点击按钮的程序是相当复杂的,因为C++本身并不直接提供高级的GUI自动化功能。通常,这样的任务会使用Windows API(如User32.dll中的函数)或者一些第三方库(如UIAutomationClient.dll提供的UI Automation API)来完成。

下面是一个使用Windows API和UI Automation的简单示例,它展示了如何查找包含特定文本的窗口,并尝试点击其中的按钮。但是,请注意,这个示例并不完整,并且可能需要根据你的具体需求进行大量的修改和扩展。此外,由于UI Automation API相对复杂,下面的代码只是提供了一个起点。

首先,你需要确保你的开发环境包含了UI Automation的相关头文件和库。这通常意味着你需要有一个较新的Windows SDK,并且你的项目需要链接到UIAutomationClient.lib。

 #include <windows.h>
#include <uiautomation.h>
#include <uiautomationclient.h>
#include <iostream>
#include <comdef.h>
#include <comutil.h>#pragma comment(lib, "UIAutomationClient.lib")// 用于转换HRESULT到std::string的辅助函数
std::string HRESULTToString(HRESULT hr) {char* messageBuffer = nullptr;FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,nullptr, hr, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),(LPSTR)&messageBuffer, 0, nullptr);std::string message(messageBuffer, SysStringLen(messageBuffer));// 释放由FormatMessage分配的缓冲区LocalFree(messageBuffer);return message;
}int main() {// 初始化COM库CoInitialize(nullptr);// 创建一个UI Automation客户端IUIAutomation* pAutomation = nullptr;HRESULT hr = CoCreateInstance(CLSID_CUIAutomation, nullptr, CLSCTX_INPROC_SERVER, IID_IUIAutomation, reinterpret_cast<void**>(&pAutomation));if (FAILED(hr)) {std::cerr << "Failed to create UI Automation client: " << HRESULTToString(hr) << std::endl;CoUninitialize();return 1;}// 获取根元素(桌面)IUIAutomationElement* pRootElement = nullptr;hr = pAutomation->GetRootElement(&pRootElement);if (FAILED(hr)) {std::cerr << "Failed to get root element: " << HRESULTToString(hr) << std::endl;pAutomation->Release();CoUninitialize();return 1;}// 在这里,你应该编写代码来遍历窗口树,查找包含特定文本的窗口,// 并使用UI Automation的属性条件和模式来查找和点击按钮。// 这通常涉及到使用IUIAutomationCondition、IUIAutomationTreeWalker和IUIAutomationInvokePattern等接口。// 由于篇幅限制,这里不会展示完整的实现。// ...(省略了查找窗口和按钮的代码)// 清理if (pRootElement) {pRootElement->Release();}if (pAutomation) {pAutomation->Release();}CoUninitialize();return 0;
}

在上面的代码中,我们初始化了COM库,创建了一个UI Automation客户端,并获取了桌面的根元素。但是,实际的窗口查找和按钮点击逻辑被省略了,因为这需要相当复杂的代码来处理。

为了完成这个任务,你需要:

  1. 使用IUIAutomationTreeWalker接口遍历UI元素树。

  2. 使用IUIAutomationCondition接口创建条件来过滤包含特定文本的窗口或按钮。

  3. 使用IUIAutomationPropertyCondition或IUIAutomationAndCondition等条件来组合搜索条件。

  4. 检查每个元素的属性(如UIA_NamePropertyId)以确定它是否包含你感兴趣的文本。

  5. 如果找到了匹配的窗口,则进一步搜索其中的按钮。

  6. 使用IUIAutomationInvokePattern接口来点击按钮。

由于这是一个高级主题,并且涉及到大量的UI Automation API调用和错误处理,因此建议你查阅UI Automation的官方文档,并可能需要编写相当多的代码来实现你的需求。如果你不熟悉UI Automation API,那么这可能是一个相当大的学习曲线。

方案二

捕获弹窗,并点击第一个按键,失焦点

在 Visual Studio (VS) 上编写一个 Windows 程序来捕获当前桌面的最前面窗口,并检查其内部是否有特定的按钮(取消、继续、忽略),然后模拟鼠标点击忽略按钮,这需要使用 Windows API。以下是一个使用 C++ 和 Windows API 的示例代码:

创建一个新的 Windows 桌面应用程序项目。
在项目中包含必要的头文件,并链接到 User32.lib 库。
编写以下代码来实现所需功能:

#include <windows.h>
#include <string>
#include <vector>
#include <tlhelp32.h> // For CreateToolhelp32Snapshot// Function to find a window by its class name and window name
HWND FindWindowByClassNameAndTitle(const std::wstring& className, const std::wstring& windowTitle) {return FindWindowW(className.c_str(), windowTitle.c_str());
}// Function to enumerate all buttons in a window and check for specific button texts
bool CheckForButtons(HWND hwnd, const std::vector<std::wstring>& buttonTexts) {std::vector<HWND> buttons;// Enumerate all child windows of the specified windowEnumChildWindows(hwnd, [](HWND hwndChild, LPARAM lParam) -> BOOL {char className[256];GetClassNameA(hwndChild, className, sizeof(className));// Check if the child window is a buttonif (strcmp(className, "Button") == 0) {buttons.push_back(hwndChild);}return TRUE; // Continue enumeration}, 0);// Check if all specified button texts are foundfor (const auto& text : buttonTexts) {bool found = false;for (HWND button : buttons) {wchar_t buttonText[256];GetWindowTextW(button, buttonText, sizeof(buttonText) / sizeof(wchar_t));if (wcscmp(buttonText, text.c_str()) == 0) {found = true;break;}}if (!found) {return false; // One of the button texts was not found}}return true; // All button texts were found
}// Function to simulate a mouse click on a specified button
void SimulateMouseClick(HWND hwndButton) {// Set the cursor position to the center of the buttonRECT rect;GetWindowRect(hwndButton, &rect);POINT ptCenter;ptCenter.x = (rect.left + rect.right) / 2;ptCenter.y = (rect.top + rect.bottom) / 2;ScreenToClient(hwndButton, &ptCenter);// Simulate mouse events (move, left button down, left button up)SetCursorPos(ptCenter.x, ptCenter.y);mouse_event(MOUSEEVENTF_MOVE | MOUSEEVENTF_LEFTDOWN, ptCenter.x, ptCenter.y, 0, 0);Sleep(50); // Small delay to ensure the button is pressedmouse_event(MOUSEEVENTF_LEFTUP, ptCenter.x, ptCenter.y, 0, 0);
}int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) {// Find the foreground window (the window that is currently active)HWND hwndForeground = GetForegroundWindow();// Specify the button texts to check forstd::vector<std::wstring> buttonTexts = { L"取消", L"继续", L"忽略" };// Check if the foreground window has the specified buttonsif (CheckForButtons(hwndForeground, buttonTexts)) {// Enumerate all child windows again to find the "忽略" buttonEnumChildWindows(hwndForeground, [](HWND hwndChild, LPARAM lParam) -> BOOL {wchar_t buttonText[256];GetWindowTextW(hwndChild, buttonText, sizeof(buttonText) / sizeof(wchar_t));if (wcscmp(buttonText, L"忽略") == 0) {// Simulate a mouse click on the "忽略" buttonSimulateMouseClick(hwndChild);return FALSE; // Stop enumeration after finding the button}return TRUE; // Continue enumeration}, 0);}return 0;
}

注意:

这个示例代码使用了 EnumChildWindows 函数来枚举窗口的子窗口,并检查它们是否是按钮以及按钮的文本。
SimulateMouseClick 函数用于模拟鼠标点击事件。它首先将光标移动到按钮的中心位置,然后模拟鼠标按下和释放事件。
由于 Windows API 的复杂性,这个示例代码可能无法在所有情况下都正常工作。特别是,如果目标窗口使用了自定义的按钮控件或特殊的 UI 框架,可能需要更复杂的逻辑来识别按钮。
在实际使用中,请确保您的程序有足够的权限来访问其他程序的窗口和控件。在某些情况下,可能需要提升程序的权限(例如,以管理员身份运行)。
这个示例代码仅用于演示目的,并不包含错误处理或健壮性检查。在实际应用中,您应该添加适当的错误处理和边界检查来确保程序的稳定性和可靠性。

User32动态库的功能主要是管理Windows用户界面的元素,例如窗口、消息、菜单、对话框等。它提供了大量的API函数,允许开发者在Windows应用程序中创建、显示和操作这些元素。
当开发者需要创建图形用户界面(GUI)应用程序,特别是涉及到窗口管理、事件处理、用户输入等方面时,就必须要使用User32动态库。它是Windows GUI编程不可或缺的一部分。

使用python的方案

在Windows上实现一个能够捕获弹窗、检查内容并在满足条件时点击按钮的程序,可以使用Python结合一些库来实现,例如pywinauto和pygetwindow。这些库允许你与Windows GUI进行交互。

以下是一个基本的示例程序,它展示了如何实现这个功能。请注意,这只是一个起点,你可能需要根据你的具体需求调整代码。

首先,你需要安装所需的库:

【bash】
pip install pywinauto pygetwindow

然后,你可以使用以下代码:

 import time
import pywinauto
from pywinauto import Application, Desktop
from pygetwindow import getAllWindows, getWindowsWithTitledef find_and_click_button(window_title, button_text_substring):# 尝试连接到已经打开的窗口try:app = Application().connect(title=window_title)except pywinauto.findwindows.ElementNotFoundError:return False  # 窗口未找到# 获取窗口window = app[window_title]# 等待窗口完全加载(可能需要根据实际情况调整等待时间)time.sleep(1)# 查找按钮并点击try:buttons = window.children(title=f"*{button_text_substring}*")if buttons:buttons[0].click()  # 点击第一个找到的按钮return Trueexcept Exception as e:print(f"Error clicking button: {e}")return Falsedef monitor_popups():while True:# 获取所有窗口windows = getAllWindows()# 遍历窗口以查找包含"helloworld"的弹窗for window in windows:if "helloworld" in window.title.lower():print(f"Found window with title: {window.title}")# 尝试点击包含"ignore"的按钮success = find_and_click_button(window.title, "ignore")if success:print(f"Clicked on a button containing 'ignore' in window: {window.title}")# 等待一段时间再检查(避免过于频繁地检查)time.sleep(5)if __name__ == "__main__":monitor_popups()

注意事项:

  1. 权限:确保你的脚本有足够的权限去操作其他窗口。在某些情况下,可能需要以管理员身份运行脚本。

  2. 窗口加载时间:在尝试与窗口交互之前,确保窗口已经完全加载。可能需要添加适当的等待时间(使用time.sleep())。

  3. 窗口和按钮的查找:pywinauto和pygetwindow使用窗口的标题来查找窗口。如果弹窗的标题不是固定的,或者按钮的文本不是固定的,你可能需要调整查找逻辑。

  4. 错误处理:示例代码中的错误处理相对简单。在实际应用中,你可能需要更详细的错误处理逻辑。

  5. 依赖:确保你的环境中安装了所有必要的库,并且它们的版本与你的代码兼容。

  6. 安全性:自动化GUI交互可能会带来安全风险,特别是在处理敏感信息或执行敏感操作时。务必谨慎使用。

这个示例提供了一个基本的框架,你可能需要根据你的具体需求进行进一步的定制和扩展。

相关文章:

在C++中实现一个能够捕获弹窗、检查内容并在满足条件时点击按钮的程序;使用python的方案

在C中实现一个能够捕获弹窗、检查内容并在满足条件时点击按钮的程序是相当复杂的&#xff0c;因为C本身并不直接提供高级的GUI自动化功能。通常&#xff0c;这样的任务会使用Windows API&#xff08;如User32.dll中的函数&#xff09;或者一些第三方库&#xff08;如UIAutomati…...

《Vue3实战教程》26:Vue3Transition

如果您有疑问&#xff0c;请观看视频教程《Vue3实战教程》...

【架构设计(一)】常见的Java架构模式

常见的 Java 架构模式解析 在 Java 开发领域&#xff0c;选择合适的架构模式对于构建高效、可维护且能满足业务需求的软件系统至关重要。本文将深入探讨几种常见的 Java架构模式&#xff0c;包括单体架构与微服务架构、分层架构与微服务架构的对比&#xff0c;以及事件驱动架构…...

自定义有序Map

package cn.ziqirj.common.utils;import lombok.Getter; import lombok.Setter;import java.util.ArrayList; import java.util.List;/*** 模拟Map集合&#xff0c;key不可重复&#xff0c;按插入顺序排序* author zhangji** param <T>*/ public class CustomOrderlyMap&…...

Jenkins(持续集成与自动化部署)

Jenkins 是一个开源软件项目&#xff0c;是基于Java开发的一种持续集成工具。 官网&#xff1a;https://www.jenkins.io/ GitLab安装使用 安装前提&#xff1a;内存至少需要4G 官方网站&#xff1a;https://about.gitlab.com/ 安装文档&#xff1a;https://docs.gitlab.c…...

redis7基础篇2 redis的哨兵模式2

目录 一 哨兵模式 1.1 redis的哨兵模式作用 1.2 redis的哨兵模式架构 1.3 redis的哨兵模式参数说明 二 redis的哨兵模式搭建 2.1 redis的主从复制模式 2.2 redis的sentinel配置文件 2.3 redis的实例节点和sentinel节点启动 3.3 redis的哨兵模式原理 3.3.1 redis的哨兵…...

windows终端conda activate命令行不显示环境名

问题&#xff1a; 始终不显示环境名 解决 首先需要配置conda的环境变量 确保conda --version能显示版本 然后对cmd进行初始化&#xff0c;如果用的是vscode中的终端&#xff0c;那需要对powershell进行初始化 Windows CMD conda init cmd.exeWindows PowerShell conda …...

SpringBoot 2.6 集成es 7.17

引言 在现代应用开发中&#xff0c;Elasticsearch作为一个强大的搜索引擎和分析引擎&#xff0c;已经成为许多项目不可或缺的一部分。Spring Boot作为Java生态中最受欢迎的微服务框架之一&#xff0c;其对Elasticsearch的支持自然也是开发者关注的焦点。本文将详细介绍如何在S…...

加固服务器有什么用?

为什么越来越多的企业和个人都在加固他们的服务器&#xff1f;加固服务器不仅可以保护数据安全&#xff0c;还能提升整体系统的稳定性和可靠性。下面是聚名网的一些介绍。 加固服务器的首要目的就是提高安全性。随着网络攻击手段的不断演变&#xff0c;黑客和恶意软件的威胁也…...

Personal APP

1、Matlab 2023b https://www.bilibili.com/opus/887246540317392920 https://blog.csdn.net/qq_25719943/article/details/138096918 https://www.jokerdown.com/22886.html 2、 3、...

探索最新的编程技术趋势:AI 编程助手和未来的编程方式

随着技术的飞速发展&#xff0c;编程技术领域在近年来经历了深刻的变革。从人工智能到低代码开发工具&#xff0c;新的技术趋势不断涌现&#xff0c;不仅大幅提高了开发效率&#xff0c;也重新定义了开发者的角色和工作方式。本篇博客将探讨几项当前最值得关注的编程技术&#…...

Android:文件管理:打开文件意图

三步走&#xff1a; 一、先在AndroidManifest.xml声明provider&#xff1a; <providerandroid:name"androidx.core.content.FileProvider"android:authorities"${applicationId}.FileProvider"android:exported"false"android:grantUriPermi…...

从纯虚类到普通类:提升C++ ABI兼容性的策略

在C编程中&#xff0c;纯虚类&#xff08;也被称为抽象类&#xff09;通常用于定义接口&#xff0c;而普通类则包含具体的实现。然而&#xff0c;在某些情况下&#xff0c;将纯虚类转换为普通类并提供默认实现&#xff0c;可以显著提升应用程序二进制接口&#xff08;ABI&#…...

QT中如何限制 限制QLineEdit只能输入字母,或数字,或某个范围内数字等限制约束?

在 Qt 中&#xff0c;你可以通过多种方式来限制 QLineEdit 只能输入特定类型的字符&#xff0c;如字母、数字或某个范围内的数字。以下是一些常见的方法&#xff1a; 1. 使用输入验证器&#xff08;QIntValidator, QDoubleValidator, QRegExpValidator&#xff09; Qt 提供了…...

Tailwind CSS 使用简介

参考网站安装 - Tailwind CSS 中文网 号称是开始使用 Tailwind CSS 通过 npm 安装 tailwindcss&#xff0c;并创建你的 tailwind.config.js 文件。 npm install -D tailwindcss npx tailwindcss init 在 tailwind.config.js 文件中添加所有模板文件的路径。 /** type {im…...

iOS 逆向学习 - iOS Architecture Cocoa Touch Layer

iOS 逆向学习 - iOS Architecture Cocoa Touch Layer 一、Cocoa Touch Layer 简介二、Cocoa Touch Layer 的核心功能1. UIKit2. Event Handling&#xff08;事件处理&#xff09;3. Multitasking&#xff08;多任务处理&#xff09;4. Push Notifications&#xff08;推送通知&…...

C语言实现库函数strlen

size_t是 unsigned int fgets会读入\n&#xff0c;用strcspn函数除去 assert判读指针是否为空指针&#xff0c;使用前要引头文件<assert.h> #include <stdio.h> #include <assert.h> size_t mystrlen(const char* str) {assert(str);size_t count 0;while …...

050_小驰私房菜_MTK Camera debug, data rate 、mipi_pixel_rate 确认

mipi_pixel_rate = data rate * 4 / 10 (4 是表示4lane,10表示raw数据是10bit) mipi_pixel_rate 信息,我们可以通过 sentest命令打印看到: 下面的信息我们可以看到,mipi_pixel_rate = 501.357739Mpps,mipi rate = 10000000,是对应的我们驱动文件里面配置写的mipi_pixel_r…...

(六)vForm 动态表单(数据量大,下拉选卡顿问题)

系列文章目录 (一)vForm 动态表单设计器之使用 (二)vForm 动态表单设计器之下拉、选择 (三)vForm 动态表单解决下拉框无数据显示id问题 (四)vForm 动态表单自定义组件、属性 (五)vForm 动态表单文件上传、下载 文章目录 目录 前言 一、组件改造 1.添加分页所需参…...

【mybatis-plus问题集锦系列】mybatis使用xml配置文件实现数据的基础增删改查

简单的数据查询&#xff0c;我们可以在mapper接口里面去实现&#xff0c;但是如果是复杂的查询&#xff0c;我们就可以使用xml配置文件去做&#xff0c; 官网链接xml配置文件 实现效果 实现代码 根据mapper接口的包结构&#xff0c;在resources包里面新建同名同结构的xml文件…...

Java 语言特性(面试系列1)

一、面向对象编程 1. 封装&#xff08;Encapsulation&#xff09; 定义&#xff1a;将数据&#xff08;属性&#xff09;和操作数据的方法绑定在一起&#xff0c;通过访问控制符&#xff08;private、protected、public&#xff09;隐藏内部实现细节。示例&#xff1a; public …...

简易版抽奖活动的设计技术方案

1.前言 本技术方案旨在设计一套完整且可靠的抽奖活动逻辑,确保抽奖活动能够公平、公正、公开地进行,同时满足高并发访问、数据安全存储与高效处理等需求,为用户提供流畅的抽奖体验,助力业务顺利开展。本方案将涵盖抽奖活动的整体架构设计、核心流程逻辑、关键功能实现以及…...

汽车生产虚拟实训中的技能提升与生产优化​

在制造业蓬勃发展的大背景下&#xff0c;虚拟教学实训宛如一颗璀璨的新星&#xff0c;正发挥着不可或缺且日益凸显的关键作用&#xff0c;源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例&#xff0c;汽车生产线上各类…...

测试markdown--肇兴

day1&#xff1a; 1、去程&#xff1a;7:04 --11:32高铁 高铁右转上售票大厅2楼&#xff0c;穿过候车厅下一楼&#xff0c;上大巴车 &#xffe5;10/人 **2、到达&#xff1a;**12点多到达寨子&#xff0c;买门票&#xff0c;美团/抖音&#xff1a;&#xffe5;78人 3、中饭&a…...

[10-3]软件I2C读写MPU6050 江协科技学习笔记(16个知识点)

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16...

.Net Framework 4/C# 关键字(非常用,持续更新...)

一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...

Device Mapper 机制

Device Mapper 机制详解 Device Mapper&#xff08;简称 DM&#xff09;是 Linux 内核中的一套通用块设备映射框架&#xff0c;为 LVM、加密磁盘、RAID 等提供底层支持。本文将详细介绍 Device Mapper 的原理、实现、内核配置、常用工具、操作测试流程&#xff0c;并配以详细的…...

视频行为标注工具BehaviLabel(源码+使用介绍+Windows.Exe版本)

前言&#xff1a; 最近在做行为检测相关的模型&#xff0c;用的是时空图卷积网络&#xff08;STGCN&#xff09;&#xff0c;但原有kinetic-400数据集数据质量较低&#xff0c;需要进行细粒度的标注&#xff0c;同时粗略搜了下已有开源工具基本都集中于图像分割这块&#xff0c…...

Java + Spring Boot + Mybatis 实现批量插入

在 Java 中使用 Spring Boot 和 MyBatis 实现批量插入可以通过以下步骤完成。这里提供两种常用方法&#xff1a;使用 MyBatis 的 <foreach> 标签和批处理模式&#xff08;ExecutorType.BATCH&#xff09;。 方法一&#xff1a;使用 XML 的 <foreach> 标签&#xff…...

Selenium常用函数介绍

目录 一&#xff0c;元素定位 1.1 cssSeector 1.2 xpath 二&#xff0c;操作测试对象 三&#xff0c;窗口 3.1 案例 3.2 窗口切换 3.3 窗口大小 3.4 屏幕截图 3.5 关闭窗口 四&#xff0c;弹窗 五&#xff0c;等待 六&#xff0c;导航 七&#xff0c;文件上传 …...