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丰富易用的功能及多年积累下的生态,我们可以开发出更多有用的功能。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

动手学深度学习(李沐)PyTorch 第 1 章 引言

在线电子书 深度学习介绍 安装 使用conda环境 conda create -n d2l-zh python3.8 pip安装需要的包 pip install jupyter d2l torch torchvision下载代码并执行 wget https://zh-v2.d2l.ai/d2l-zh.zip unzip d2l-zh.zip jupyter notebookpip install rise如果不想使用jupyt…

Ubuntu20-xrdp与Windows-mstsc远程桌面连接

前期准备 两者在同一网段&#xff0c;网线连接。ubuntu端 sudo adduser yu //输入密码和确认密码&#xff0c;后面一路回车&#xff0c;新建用户yu&#xff0c;确保用户没有被登录 sudo apt install xrdp //安装xrdp sudo systemctl status xrdp //查看xrdp服务状态 sudo…

《飞机大战游戏》实训项目(Java GUI实现)(设计模式)(简易)

目录 一、最终实现后&#xff0c;效果如下。 &#xff08;1&#xff09;简单介绍本游戏项目&#xff08;待完善&#xff09; &#xff08;2&#xff09;运行效果图&#xff08;具体大家自己可以试&#xff09; 初始运行情况。 手动更换背景图。 通过子弹攻击敌机&#xff0c;累…

828华为云征文|Flexus云服务器X实例部署宝塔运维面板

本次华为云Flexus云服务器X实例部署宝塔运维面板教学&#xff0c;这次是推陈出新啊 之前的云耀云服务器L实例已经很不错了&#xff0c;大力赞叹华为云的 同时感谢华为云提供优惠卷&#xff0c;只能说白嫖真是太棒了 华为云近期正在筹办华为云828企业节活动&#xff0c;90款免…

HarmonyOS Next开发----使用XComponent自定义绘制

XComponent组件作为一种绘制组件&#xff0c;通常用于满足用户复杂的自定义绘制需求&#xff0c;其主要有两种类型"surface和component。对于surface类型可以将相关数据传入XComponent单独拥有的NativeWindow来渲染画面。 由于上层UI是采用arkTS开发&#xff0c;那么想要…

【RabbitMQ】消息分发、事务

消息分发 概念 RabbitMQ队列拥有多个消费者时&#xff0c;队列会把收到的消息分派给不同的消费者。每条消息只会发送给订阅该队列订阅列表里的一个消费者。这种方式非常适合扩展&#xff0c;如果现在负载加重&#xff0c;那么只需要创建更多的消费者来消费处理消息即可。 默…

深度学习01-概述

深度学习是机器学习的一个子集。机器学习是实现人工智能的一种途径&#xff0c;而深度学习则是通过多层神经网络模拟人类大脑的方式进行学习和知识提取。 深度学习的关键特点&#xff1a; 1. 自动提取特征&#xff1a;与传统的机器学习方法不同&#xff0c;深度学习不需要手动…

【数据库】常用数据库简介

目录 &#x1f354; 常用的关系型数据库 &#x1f354; Mysql简介 &#x1f354; SQL 简介 SQL语句的分类 SQL 写法 SQL 常用的数据类型 &#x1f354; DDL语句 对数据库的操作 对数据表的操作 &#x1f354; DML语句 插入数据 insert into 修改数据 update 删除数…

python实现多个pdf文件合并

打印发票时&#xff0c;需要将pdf合并成一个&#xff0c;单页两张打印。网上一些pdf合并逐渐收费&#xff0c;这玩意儿都能收费&#xff1f;自己写一个脚本使用。 实现代码&#xff1a; 输入pdf文件夹路径data_dir&#xff0c;统计目录下的“合并后的PDF”文件夹下&#xff0c;…

linux重要文件

/etc/sysconfig/network-scripts/ifcfg-eth1 网卡重启 /etc/init.d/network restart ifup ethname & ifdown ethname /etc/resolv.conf 设置Linux本地的客户端DNS的配置文件 linux客户端DNS可以在网卡配置文件(/etc/sysconfig/network/ifcfg-eth0 DNS2)里配置 也可以在/et…

Java_Day04学习

类继承实例 package com.dx.test03; public class extendsTest {public static void main(String args[]) {// 实例化一个Cat对象&#xff0c;设置属性name和age&#xff0c;调用voice()和eat()方法&#xff0c;再打印出名字和年龄信息/********* begin *********/Cat cat ne…

Pandas -----------------------基础知识(一)

目录 Series对象 属性和方法 布尔值列表获取Series对象中部分数据 运算 DateFrame对象 常用属性 常见方法 运算 总结 Series对象 是DataFrame的列对象或者行对象 生成Series对象生成索引使用元组创建Series对象使用字典创建Series对象 通过Pandas创建对象 自定义索引 …

面试官问:你最自豪的成就是什么?

当面试官问你最自豪的成就是什么&#xff0c;我们首先分析面试官为什么这么问&#xff0c;他想通过这问题得到什么信息&#xff1f; 你最自豪的成就是什么&#xff1f; 其实反应了一个人的职业驱动力&#xff0c;比如我们常说的&#xff1a;上进心&#xff0c;主动积极性&…

【机器学习-监督学习】朴素贝叶斯

【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈Python机器学习 ⌋ ⌋ ⌋ 机器学习是一门人工智能的分支学科&#xff0c;通过算法和模型让计算机从数据中学习&#xff0c;进行模型训练和优化&#xff0c;做出预测、分类和决策支持。Python成为机器学习的首选语言&#xff0c;…

【小沐学GIS】基于Openstreetmap创建Sionna RT场景(Python)

文章目录 1、简介1.1 blender 2、下载和安装2.1 Python2.2 jupyter 3、运行结语 1、简介 1.1 blender https://www.blender.org/ Blender 是一款免费开源的3D创作套件。 使用 Blender&#xff0c;您可以创建3D可视化效果&#xff0c;例如静态图像、3D动画、VFX&#xff08;…

【UE5】将2D切片图渲染为体积纹理,最终实现使用RT实时绘制体积纹理【第一篇-原理】

如果想直接制作&#xff0c;请看【第二篇】内容 这次做一个这样的东西&#xff0c;通过在2DRT上实时绘制&#xff0c;生成动态的体积纹理&#xff0c;也就是可以runtime的VDB 设想的文章流程: 对原理进行学习制作体积渲染制作实时绘制 第一篇&#xff08;本篇&#xff09;是对“…

【Rust练习】16.模式

文章题目来自&#xff1a;https://practice-zh.course.rs/pattern-match/patterns.html 1 &#x1f31f;&#x1f31f; 使用 | 可以匹配多个值, 而使用 … 可以匹配一个闭区间的数值序列 fn main() {} fn match_number(n: i32) {match n {// 匹配一个单独的值1 > println!(…

【赵渝强老师】K8s中的Deployment控制器

K8s的Deployment将Pod部署成无状态的应用程序&#xff0c;它只关心Pod的数量、Pod更新方式、使用的镜像和资源限制等。由于是无状态的管理方式&#xff0c;因此Deployment中没有角色和顺序的概念&#xff0c;换句话说&#xff1a;Deployment中没有状态。   通过使用Deploymen…

【远程调用PythonAPI-flask】

文章目录 前言一、Pycharm创建flask项目1.创建虚拟环境2.创建flask项目 二、远程调用PythonAPI——SpringBoot项目集成1.修改PyCharm的host配置2.防火墙设置3.SpringBoot远程调用PythonAPI 前言 解决Pycharm运行Flask指定ip、端口更改无效的问题 首先先创建一个新的flask项目&…

C语言 | Leetcode C语言题解之第415题字符串相加

题目&#xff1a; 题解&#xff1a; char* addStrings(char* num1, char* num2) {int i strlen(num1) - 1, j strlen(num2) - 1, add 0;char* ans (char*)malloc(sizeof(char) * (fmax(i, j) 3));int len 0;while (i > 0 || j > 0 || add ! 0) {int x i > 0 ?…