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

C++/Qt 集成 AutoHotkey

C++/Qt 集成 AutoHotkey

    • 前言
      • AutoHotkey 介绍
    • 方案一:子进程启动
      • 编写AutoHotkey脚本
      • 准备 AutoHotkey 运行环境
      • 编写 C++/Qt 代码
    • 方案二:显式动态链接
      • 方案探索
      • 编译动态链接库
      • 集成到C++工程
      • 关于AutoHotkeyDll.dll中的函数原型
    • 总结

前言

上一篇介绍了AutoHotkey的基本情况和使用,其功能丰富易用,于是搬出我们的老朋友 C++/Qt,将AutoHotkey 集成到 C++/Qt开发环境,使其为我们所用。

AutoHotkey 介绍

上一篇链接: C++ AutoHotkey 开源项目介绍

 

方案一:子进程启动

上一篇我们介绍了AutoHotkey可以通过命令行启动,于是自然而然的想到了在我们的工程中使用命令行启动AutoHotkey脚本;

 

编写AutoHotkey脚本

编写一个简单的ahk脚本,如:

MsgBox "hello world!"

 

准备 AutoHotkey 运行环境

为了方便测试,将需要的依赖和ahk脚本放在了同一路径下,如图:

在这里插入图片描述
 

编写 C++/Qt 代码

创建一个Qt命令行工程用于集成测试,编写代码如下:

#include <QCoreApplication>
#include <QProcess>
#include <QDebug>int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);auto executeTerminalCommand = [=](const QString &command){QProcess process;process.start("cmd", QStringList() << "/c" << command);process.waitForStarted();process.waitForFinished();QString error = QString::fromLocal8Bit( process.readAllStandardError());if(!error.isEmpty()){qDebug() << "error:" + error;}QString result = QString::fromLocal8Bit( process.readAllStandardOutput() );return result;};executeTerminalCommand("D:/test/AutoHotkey.exe D:/test/hello.ahk");return a.exec();
}

以上代码使用QProcess启动cmd,命令行启动了ahk脚本,运行结果如图:
在这里插入图片描述

这种方式简单易用,没什么心智负担,但如果你的软件需要分发给其他用户使用,你就必须要打包AutoHotkey.exe 和ahk脚本文件,一眼就被别人看穿的感觉并不愉快,于是我们继续探索其它方案。

 

方案二:显式动态链接

方案探索

在AutoHotkey项目的README中,看到如下内容:
在这里插入图片描述
字面意思不太易于理解,进入 README-LIB.md 链接中继续寻找,发现其并不是所预期的编译为动态库供其它软件链接,而是在ahk中启用另一个ahk以在V2版本的ahk中启用V1版本功能等等,但文档中发现了有趣的线索:
在这里插入图片描述
进入 HotKeyIt/ahkdll 链接,查看项目简介:
在这里插入图片描述
说明了“为其它编程语言、脚本语言打开了AutoHotkey的世界”,正是我们的需求,话不多说,开始验证;

 

编译动态链接库

下载ahkdll源码到本地,解压后如图:
在这里插入图片描述
可以看出是visual studio的工程,不折腾直接使用vs构建;vs打开工程文件(这里我使用的是vs2019),配置切换为releaseDll、x64,右键AutoHotkey项目,点击生成:

在这里插入图片描述
在这里插入图片描述

在项目的bin/x64w 下,找到了AutoHotkeyDLL.dll:
在这里插入图片描述
 

集成到C++工程

妥善安置 AutoHotkeyDll.dll,编写代码如下(注意动态库路径):

#include <Windows.h>
#include <winuser.h>
#include <iostream>
#include <libloaderapi.h>
#include <comdef.h>using namespace std;LPTSTR strToLPTSTR(string str)
{_bstr_t bstr(str.c_str());return (LPTSTR)bstr;
}int main()
{typedef BOOL (*AhkReady)(void);typedef BOOL (*AhkExec)(LPTSTR script);typedef UINT_PTR (*AhkDll)(LPTSTR script,LPTSTR p1,LPTSTR p2);HINSTANCE handle = LoadLibrary(L"D:\\test\\AutoHotkeyDLL.dll");AhkDll ahkdll = (AhkDll)GetProcAddress(handle, "ahkdll");AhkReady ahkReady = (AhkReady)GetProcAddress(handle, "ahkReady");AhkExec ahkExec = (AhkExec)GetProcAddress(handle, "ahkExec");LPTSTR empty = strToLPTSTR("");ahkdll(empty, empty, empty);ahkReady();string script = "MsgBox \"hello world!\"";ahkExec(strToLPTSTR(script));return 0 ;
}

以上代码显式链接AutoHotkeyDll.dll,定义函数指针调用函数,在程序中以字符串的形式运行ahk脚本,运行结果如图:
在这里插入图片描述
 

关于AutoHotkeyDll.dll中的函数原型

函数原型来自于 ahkdll 项目中 source 文件夹下的 exports.h,如 ahkExec 函数:
在这里插入图片描述

 

总结

两种方案都可方便的将AutoHotkey集成到其它编程语言中,命令行启动、显式动态链接并不是 c++ 或 Qt 的特性,利用AutoHotkey丰富易用的功能及多年积累下的生态,我们可以开发出更多有用的功能。

相关文章:

C++/Qt 集成 AutoHotkey

C/Qt 集成 AutoHotkey 前言AutoHotkey 介绍 方案一&#xff1a;子进程启动编写AutoHotkey脚本准备 AutoHotkey 运行环境编写 C/Qt 代码 方案二&#xff1a;显式动态链接方案探索编译动态链接库集成到C工程关于AutoHotkeyDll.dll中的函数原型 总结 前言 上一篇介绍了AutoHotkey…...

OpenMV学习第一步安装IDE_2024.09.20

用360浏览器访问星瞳科技官网&#xff0c;一直提示访问不了。后面换了IE浏览器就可以访问。第一个坑。...

RK3568平台(基础篇)万用表的使用

一.万用表的通断判断 万用表两个笔头的插法:黑笔头是插在COM的孔里面,红色笔头可以插在其他的三个孔里面,20A和mA分别用来测电流,另外一个孔可以用来测其他(电压 电阻)。 以下这个三角符号(像wifi一样的)可以用来测通断: 使用万用表的红笔和黑笔进行短接,这时候两端…...

more、less 命令:阅读文本

一、命令简介 ​more​ 和 less​ 都是用于查看文本文件内容的命令&#xff0c;但它们在显示方式和功能上有一些区别。 简单的说 less​ 是 more​ 的升级版&#xff1a;增加了搜索、跳转等功能。所以优先使用 less​&#xff0c;可以不用 more ​了。 ‍ 二、命令参数 基…...

【笔记】1.3 塑性变形

一、塑性变形的方式 DDWs&#xff08;Dislocation-Dipole Walls&#xff0c;位错偶极墙&#xff09;&#xff1a;指由两个位错构成的结构&#xff0c;它们以一种特定的方式排列在一起&#xff0c;形成一个稳定的结构单元。 DTs&#xff08;Dislocation Tangles&#xff0c;位错…...

Java集合(三)

目录 Java集合&#xff08;三&#xff09; Java双列集合体系介绍 HashMap类 HashMap类介绍 HashMap类常用方法 HashMap类元素遍历 LinkedHashMap类 LinkedHashMap类介绍 LinkedHashMap类常用方法 LinkedHashMap类元素遍历 Map接口自定义类型去重的方式 Set接口和Ma…...

python:给1个整数,你怎么判断是否等于2的幂次方?

最近在csdn上刷到一个比较简单的题目&#xff0c;题目要求不使用循环和递归来实现检查1个整数是否等于2的幂次方&#xff0c;题目如下&#xff1a; 题目的答案如下&#xff1a; def isPowerofTwo(n):z bin(n)[2:]print(bin(n))if z[0] ! 1:return Falsefor i in z[1:]:if i !…...

Centos7安装gitlab-ce(rpm安装方式)

本章教程&#xff0c;主要介绍如何在Centos7安装gitlab-ce。 一、安装基础环境 安装gitlab-ce之前&#xff0c;我们需要安装一下jdk版本。 sudo yum install java-11-openjdk-devel二、下载安装包 这里我们下载的是rpm包。更多gitlab-ce版本可以在这里查看&#xff1a;https://…...

Flutter 获取手机连接的Wifi信息

测试版本 Flutter&#xff1a;3.7.6Dart:2.19.3 network_info_plus: 4.0.1 前言 我在做设备配网的时候&#xff0c;需要选择配网的WiFi。用下拉选择框展示WiFi列表。现在有个需求&#xff1a;默认展示的设备为手机连接的wifi。要实现这个需求只要能够获取到手机连接的wifi信息…...

誉龙视音频 Third/TimeSyn 远程命令执行复现

0x01 漏洞描述&#xff1a; 誉龙公司定位为系统级的移动视音频记录解决方案提供商&#xff0c;凭借其深厚的行业经验&#xff0c;坚持自主研发&#xff0c;匠心打造记录仪领域行业生态&#xff0c;提供开放式的记录仪APK、GB28181 SDK、国网B协议、管理平台软件OEM。誉龙视音频…...

ATMEGA328P芯片引脚介绍

1.AVCC AVCC是ATmega328P芯片的模拟电源引脚。 AVCC引脚的定义 模拟电源引脚&#xff1a;AVCC&#xff08;Analog Voltage Common&#xff09;是ATmega328P微控制器中的模拟电源引脚&#xff0c;用于为模拟电路部分提供稳定的电源。功能描述&#xff1a;AVCC通常连接到一个干…...

现代前端构建工具对比:Vue CLI、Webpack 和 Vite

一、引言&#x1f31f; 在现代前端开发中&#xff0c;选择合适的构建工具对于提高项目的效率和可维护性至关重要。&#x1f6e0;️ Vue CLI、&#x1f4e6; Webpack 和 &#x1f680; Vite 是目前最流行的三个构建工具&#xff0c;它们各自具有独特的优势和适用场景。本文将深…...

代码随想录算法训练营第三九天| 198.打家劫舍 213.打家劫舍II 337.打家劫舍 III

今日任务 198.打家劫舍 213.打家劫舍II 337.打家劫舍 III 198.打家劫舍 题目链接&#xff1a; . - 力扣&#xff08;LeetCode&#xff09; class Solution {public int rob(int[] nums) {int[] dp new int[nums.length];if (nums.length 1) return nums[0];if (nums.lengt…...

阿里云AI基础设施全面升级,模型算力利用率提升超20%

来源首席数智官 9月20日&#xff0c;2024云栖大会现场&#xff0c;阿里云全面展示了全新升级后的AI Infra系列产品及能力。通过全栈优化&#xff0c;阿里云打造出一套稳定和高效的AI基础设施&#xff0c;连续训练有效时长大于99%&#xff0c;模型算力利用率提升20%以上。 “AI…...

Debezium日常分享系列之:将容器镜像移至 quay.io

Debezium日常分享系列之&#xff1a;将容器镜像移至 quay.io 在Debezium 3.0.0.Final发布之后&#xff0c;我们将不再向docker.io发布容器镜像更新。旧版本的Debezium 2.x和1.x镜像将继续保留在docker.io上&#xff1b;然而&#xff0c;所有未来的Debezium 2.7.x和3.x或更高版本…...

基于TCP实现聊天

TCP客户端代码 import java.io.*; import java.net.InetAddress; import java.net.Socket;public class TcpClientDemo01 {public static void main(String[] args) {Socket socket null;OutputStream os null;InputStream is null;BufferedReader reader null;try {// 1.…...

基于JavaSwing实现的酒店管理系统

一、项目介绍 > 欢迎使用酒店管理系统&#xff01; > 这是一个基于Java Swing开发&#xff0c;用于管理酒店预订、房间、订单和用户信息的系统。 > 适用于JAVA初学者作为入门学习项目。 二、项目演示 三、基础依赖 技术/框架版本描述Java8编程语言MySQL8.0数据…...

网络基础,协议,OSI分层,TCP/IP模型

网络的产生是数据交流的必然趋势&#xff0c;计算机之间的独立的个体&#xff0c;想要进行数据交互&#xff0c;一开始是使用磁盘进行数据拷贝&#xff0c;可是这样的数据拷贝效率很低&#xff0c;于是网络交互便出现了&#xff1b; 1.网络是什么 网络&#xff0c;顾名思义是…...

CefSharp_Vue交互(Element UI)_WinFormWeb应用---设置应用透明度(含示例代码)

一、界面预览 1.1 设置透明(整个页面透明80%示例) 限制输入值:10-100(数字太小会不好看见) 1.2 vue标题栏 //注册类与js调用 (async function(...

【OSS安全最佳实践】降低因账号密码泄露带来的未授权访问风险

如果因个人或者企业账号密码泄露引发了未经授权的访问&#xff0c;可能会出现非法用户对OSS资源进行违法操作&#xff0c;或者合法用户以未授权的方式对OSS资源进行各类操作&#xff0c;这将给数据安全带来极大的威胁。为此&#xff0c;OSS提供了在实施数据安全保护时需要考虑的…...

深入排查k8s集群6443端口连接拒绝:从kubectl故障到系统级修复

1. 当kubectl突然罢工&#xff1a;6443端口连接拒绝的紧急处理 那天早上我像往常一样打开终端&#xff0c;准备用kubectl get pods查看集群状态&#xff0c;结果终端冷冰冰地抛出一行错误&#xff1a;"Unable to connect to the server: dial tcp 192.168.1.1:6443: conne…...

从硬件到空域:拆解一个真实的无人机Remote ID广播包,聊聊合规与隐私

从硬件到空域&#xff1a;拆解无人机Remote ID广播包的技术与合规全景 当一架多旋翼无人机在低空掠过城市天际线时&#xff0c;它的存在不仅通过旋翼的嗡鸣声宣告&#xff0c;更通过无线电波向方圆数公里广播着自己的"数字身份证"。这种被称为Remote ID的技术&#x…...

不用npm!3分钟搞定微信小程序引入Animate.css的另类方法

微信小程序免npm引入Animate.css的极简方案 最近在开发微信小程序时&#xff0c;发现很多开发者都在寻找一种更简单的方法来引入Animate.css动画库&#xff0c;而不必依赖npm。对于不熟悉node环境的开发者来说&#xff0c;npm安装过程可能会遇到各种问题。今天我就分享一个完全…...

终极QMC音频解密方案:qmc-decoder如何3分钟转换100首加密音乐

终极QMC音频解密方案&#xff1a;qmc-decoder如何3分钟转换100首加密音乐 【免费下载链接】qmc-decoder Fastest & best convert qmc 2 mp3 | flac tools 项目地址: https://gitcode.com/gh_mirrors/qm/qmc-decoder 在数字音乐版权保护的浪潮中&#xff0c;QQ音乐QM…...

新手必看|SRC平台漏洞挖掘全攻略(2026干货版):平台详解+规则必记+实操步骤

新手必看&#xff5c;SRC平台漏洞挖掘全攻略&#xff08;2026 干货版&#xff09;&#xff1a;平台详解规则必记实操步骤 对于网络安全新手、计算机相关专业学生&#xff0c;以及想转型安全领域的从业者而言&#xff0c;SRC平台是合法练手、积累实战经验、衔接职场的核心载体。…...

从16QAM到256QAM:用Simulink星座图揭秘高阶调制的抗噪性能

高阶QAM调制的星座图分析与Simulink实战指南 在5G和Wi-Fi 6时代&#xff0c;256QAM已成为提升频谱效率的关键技术。但当我们从实验室的理想环境走向真实无线场景时&#xff0c;工程师们常面临一个核心矛盾&#xff1a;如何在频谱效率与系统稳定性之间找到最佳平衡点&#xff1…...

论文AI率从80%降到10%以下的完整攻略:实测3款降AI率工具真实效果

论文AI率从80%降到10%以下的完整攻略&#xff1a;实测3款降AI率工具真实效果 上个月我同学发来一张知网检测报告&#xff0c;AI率87%&#xff0c;整个人都懵了。她用DeepSeek写了大部分初稿&#xff0c;没想到检测会这么高。当时距离论文提交截止不到两周&#xff0c;她问我有没…...

FPGA开发实战——常见错误排查与优化技巧(持续更新)

1. Vivado仿真与PR Flow冲突问题实战解析 第一次用Vivado做PR&#xff08;Partial Reconfiguration&#xff09;项目时&#xff0c;我兴冲冲地点开仿真按钮&#xff0c;结果弹出一个让人崩溃的报错&#xff1a;"ERROR [Common 17-69] Command failed. Simulation for PR F…...

手把手教你用Vivado 2021配置Zynq UltraScale+ GTH回环测试(附工程源码)

Zynq UltraScale GTH回环测试实战指南&#xff1a;从原理到源码解析 在FPGA开发领域&#xff0c;高速串行接口的验证一直是工程师面临的关键挑战。Xilinx UltraScale架构中的GTH收发器以其高达16.3Gbps的线速率&#xff0c;成为医疗成像、雷达信号处理等高性能应用的理想选择。…...

RPCS3游戏汉化实战指南:从零构建多语言游戏体验

RPCS3游戏汉化实战指南&#xff1a;从零构建多语言游戏体验 【免费下载链接】rpcs3 PS3 emulator/debugger 项目地址: https://gitcode.com/GitHub_Trending/rp/rpcs3 还在为PS3经典游戏的日文界面而困扰吗&#xff1f;通过RPCS3模拟器的强大补丁系统&#xff0c;您可以…...