Qt WORD/PDF(四)使用 QAxObject 对 Word 替换(QWidget)
关于QT Widget 其它文章请点击这里: QT Widget
国际站点 GitHub: https://github.com/chenchuhan
国内站点 Gitee : https://gitee.com/chuck_chee
姊妹篇:
Qt WORD/PDF(一)使用 QtPdfium库实现 PDF 操作
Qt WORD/PDF(二)使用 QtPdfium库实现 PDF 预览、打印等
Qt WORD/PDF(三)使用 QAxObject 对 Word 替换(QML)
Qt WORD/PDF(四)使用 QAxObject 对 Word 替换(QWidget)
一、QAxObject 简介
QAxObject 是 Qt 提供的一个类,它用于与 COM(Component Object Model)对象进行交互。COM 是一种微软的技术,广泛用于各种应用程序之间的通信,尤其在 Windows 平台上,很多软件和系统组件都是基于 COM 构建的。QAxObject 类提供了一个 Qt 风格的接口,简化了与这些 COM 对象的交互。
本文主要使用 QAxObject 操作 word 文档,使用键值对,对模板文件进行替换操作,导出相应的文档,特别适合输出报告。
本文采用 Qt Widget 纯代码的方式
环境:
QT5.15.2 + MSVC2019 + Widget
二、演示
实现功能:
- 用户可以选择一个模板文件,并进行占位符的批量替换。
- 用户可以设置替换后文档的保存路径。
- 支持通过界面交互实现选择文件、显示信息以及执行替换操作。
- 使用 QAxObject 实现了与 Word 的 COM 接口的交互,允许直接操作 Word 文档中的内容。
三、代码
完整代码
mainwindow.cpp:
#include "mainwindow.h"
// #include "ui_mainwindow.h"#include <QFileDialog>
#include <QDebug>
#include <QMessageBox>MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)// , ui(new Ui::MainWindow)
{// ui->setupUi(this);// 初始化 UIauto *centralWidget = new QWidget(this);auto *mainLayout = new QVBoxLayout;mainLayout->setContentsMargins(10, 10, 10, 10); // 设置布局边距mainLayout->setSpacing(10); // 设置控件之间的间距// setMinimumSize(600, 520); // 设置窗口最小宽度为600,高度为400// 模板文件选择auto *templateLayout = new QHBoxLayout;auto *templateLabel = new QLabel("打开模板:", this);templatePathEdit = new QLineEdit(this);auto *browseTemplateButton = new QPushButton("浏览", this);templateLayout->addWidget(templateLabel);templateLayout->addWidget(templatePathEdit);templateLayout->addWidget(browseTemplateButton);connect(browseTemplateButton, &QPushButton::clicked, this, &MainWindow::browseTemplateFile);// 输出文件选择auto *outputLayout = new QHBoxLayout;auto *outputLabel = new QLabel("输出路径:", this);outputPathEdit = new QLineEdit(this);auto *browseOutputButton = new QPushButton("浏览", this);outputLayout->addWidget(outputLabel);outputLayout->addWidget(outputPathEdit);outputLayout->addWidget(browseOutputButton);connect(browseOutputButton, &QPushButton::clicked, this, &MainWindow::browseOutputFile);// 键值对表格auto *placeholdersLabel = new QLabel("键值对替换:", this);placeholdersTable = new QTableWidget(this);placeholdersTable->setColumnCount(2);placeholdersTable->setHorizontalHeaderLabels({"占位符", "替换值"});placeholdersTable->setRowCount(5); // 默认三行// 设置默认值placeholdersTable->setItem(0, 0, new QTableWidgetItem("[A]"));placeholdersTable->setItem(0, 1, new QTableWidgetItem("柯布"));placeholdersTable->setItem(1, 0, new QTableWidgetItem("[B]"));placeholdersTable->setItem(1, 1, new QTableWidgetItem("阿瑟"));placeholdersTable->setItem(2, 0, new QTableWidgetItem("[C]"));placeholdersTable->setItem(2, 1, new QTableWidgetItem("杜拉"));placeholdersTable->setItem(3, 0, new QTableWidgetItem("[D]"));placeholdersTable->setItem(3, 1, new QTableWidgetItem("伊姆斯"));// 替换按钮replaceButton = new QPushButton("执行替换", this);connect(replaceButton, &QPushButton::clicked, this, &MainWindow::replaceInWord);// 布局整合mainLayout->addLayout(templateLayout);mainLayout->addLayout(outputLayout);mainLayout->addWidget(placeholdersLabel);mainLayout->addWidget(placeholdersTable);mainLayout->addWidget(replaceButton);centralWidget->setLayout(mainLayout);setCentralWidget(centralWidget);setWindowTitle("Word 替换工具");resize(1000, 600); // 初始窗口大小
}MainWindow::~MainWindow()
{// delete ui;
}void MainWindow::browseTemplateFile() {QString filePath = QFileDialog::getOpenFileName(this, "选择模板文件", QString(), "Word 文件 (*.docx *.doc)");if (!filePath.isEmpty()) {templatePathEdit->setText(filePath);wordApp = new QAxObject("Word.Application");if (wordApp->isNull()) {qDebug() << "Failed to initialize Word.Application.";delete wordApp;return ;}// 隐藏 Word 窗口wordApp->setProperty("Visible", true);//打开指定文档QAxObject *documents = wordApp->querySubObject("Documents");QAxObject *document = documents->querySubObject("Open(const QString&)", filePath);if (document == nullptr) {QMessageBox::critical(this, "错误", "无法打开 Word 文件!");return;}}
}void MainWindow::browseOutputFile() {QString filePath = QFileDialog::getSaveFileName(this, "选择输出文件", QString(), "Word 文件 (*.docx *.doc)");if (!filePath.isEmpty()) {outputPathEdit->setText(filePath);}
}void MainWindow::replaceInWord() {QString templatePath = templatePathEdit->text();QString outputPath = outputPathEdit->text();if (templatePath.isEmpty() || outputPath.isEmpty()) {QMessageBox::warning(this, "错误", "请填写模板路径和输出路径!");return;}QMap<QString, QString> placeholders;for (int row = 0; row < placeholdersTable->rowCount(); ++row) {QString key = placeholdersTable->item(row, 0) ? placeholdersTable->item(row, 0)->text() : QString();QString value = placeholdersTable->item(row, 1) ? placeholdersTable->item(row, 1)->text() : QString();if (!key.isEmpty()) {placeholders.insert(key, value);}}if (placeholders.isEmpty()) {QMessageBox::warning(this, "错误", "请填写至少一个占位符和替换值!");return;}if (replaceMultiple(templatePath, outputPath, placeholders)) {QMessageBox::information(this, "成功", "替换完成!");} else {QMessageBox::critical(this, "失败", "替换失败!");}
}bool MainWindow::replaceMultiple(const QString &templatePath, const QString &outputPath, const QMap<QString, QString> &placeholders) {qDebug() << "Received data:" << placeholders;qDebug() << "Template Path:" << templatePath;qDebug() << "Output Path:" << outputPath;if (!QFile::exists(templatePath)) {qDebug() << "Template file does not exist:" << templatePath;return false;}qDebug() << "QFile::exists ok" ;// 打开模板文件QAxObject *documents = wordApp->querySubObject("Documents");QAxObject *document = documents->querySubObject("Open(const QString&)", templatePath);// 查找占位符并替换//使用 Find.Execute 查找占位符,使用 TypeText 方法替换为新内容QAxObject *selection = wordApp->querySubObject("Selection");// 获取 Find 对象QAxObject *find = selection->querySubObject("Find");qDebug() << "start placeholde";// 遍历占位符键值对, 替换未成功,则有问题for (auto it = placeholders.begin(); it != placeholders.end(); ++it) {QString placeholder = it.key();QString newContent = it.value();bool isFound = true;//可替换多个,且重复的while (isFound) {// 查找目标文本并替换// isFound = find->dynamicCall("Execute(const QString&)", placeholder).toBool();isFound = find->dynamicCall("Execute(QString, bool, bool, bool, bool, bool, bool, int)",placeholder, // 要查找的字符串false, // 区分大小写false, // 完整单词false, // 使用通配符false, // 忽略标点符号false, // 忽略空格true, // 向前查找1).toBool(); // 查找范围:整个文档if (isFound) {// 替换文本selection->dynamicCall("TypeText(const QString&)", newContent);}}}qDebug() << "All Find operation succeed!";document->dynamicCall("SaveAs(const QString&)", outputPath);// 关闭文档document->dynamicCall("Close()");wordApp->dynamicCall("Quit()");delete wordApp;return true;
}
mainwindow.h:
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QTableWidget>
#include <QLineEdit>
#include <QPushButton>
#include <QVBoxLayout>
#include <QHBoxLayout>
#include <QLabel>
#include <QMap>
#include <QAxObject>
#include <QAxWidget>QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();private slots:void browseTemplateFile();void browseOutputFile();void replaceInWord();private:QLineEdit *templatePathEdit;QLineEdit *outputPathEdit;QTableWidget *placeholdersTable;QPushButton *replaceButton;bool replaceMultiple(const QString &templatePath, const QString &outputPath, const QMap<QString, QString> &placeholders);QAxObject *wordApp = nullptr;QAxWidget *wordPreview = nullptr;private:Ui::MainWindow *ui;
};
#endif // MAINWINDOW_H
pro 中需要增加对 QAxObject 的支持
QT += core gui axcontainer
四、分析:
这段Qt C++代码实现了一个简单的“Word 文件替换工具”,允许用户通过一个图形界面选择Word模板文件、设置输出路径,并在Word文档中进行占位符的替换操作。
1. 类的构造函数 (MainWindow::MainWindow)
- UI设置:
- 使用
QWidget
创建中央窗口,QVBoxLayout
为主布局,内部包含多个控件(如标签、输入框、按钮等)。 - 通过
QHBoxLayout
设置了模板文件选择区域(输入框和浏览按钮)、输出路径选择区域(输入框和浏览按钮)、以及键值对表格用于占位符替换。 - 还创建了一个
QTableWidget
来管理占位符和替换值的键值对。初始化了5行默认数据。
- 使用
- 控件连接:
- 点击“浏览”按钮会触发文件选择对话框,并通过信号槽机制连接相应的函数(
browseTemplateFile
和browseOutputFile
)。 - 替换按钮 (
replaceButton
) 连接到replaceInWord
函数。
- 点击“浏览”按钮会触发文件选择对话框,并通过信号槽机制连接相应的函数(
- Word预览:
wordPreview
是一个QAxWidget
控件,允许通过 ActiveX 技术与 Word 应用进行交互。它在browseTemplateFile
中初始化并用于打开 Word 文件。
2. 浏览模板文件 (browseTemplateFile
)
- 打开文件对话框 (
QFileDialog::getOpenFileName
) 选择模板文件,文件路径显示在templatePathEdit
中。 - 使用
QAxWidget
来查询 Word 的应用对象wordApp
,并打开用户选择的文件。 - 通过
Word.Application
创建 ActiveX 对象,加载模板文件并将其显示在打印预览模式。 - 如果没有成功加载Word文件,会弹出错误提示。
3. 浏览输出文件 (browseOutputFile
)
- 打开保存文件对话框 (
QFileDialog::getSaveFileName
) 选择输出文件路径,并将路径显示在outputPathEdit
中。
4. 替换操作 (replaceInWord
)
- 从 UI 中获取模板文件路径和输出文件路径,如果路径为空,弹出警告。
- 获取表格中的占位符及其对应替换值,并构建一个
QMap
来存储这些键值对。 - 调用
replaceMultiple
函数进行批量替换操作。
5. 替换逻辑 (replaceMultiple
)
- 文件存在性检查: 使用
QFile::exists
检查模板文件是否存在。 - 打开Word文档: 使用
QAxObject
打开模板文件,获取Selection
对象和Find
对象来查找占位符。 - 查找和替换: 遍历占位符的键值对,使用
Find.Execute
查找占位符,并通过TypeText
替换为新内容。替换过程中使用了while
循环,确保文档中所有的占位符都能被替换(即使它们重复出现)。 - 保存文件: 替换完成后,使用
SaveAs
保存文件到指定的输出路径。 - 关闭和退出: 完成替换后,关闭文档并退出Word应用。
总结
该程序使用 Qt 的 QAxObject
来与 Microsoft Word 进行交互,实现了以下功能:
- 用户可以选择一个模板文件,并进行占位符的批量替换。
- 用户可以设置替换后文档的保存路径。
- 支持通过界面交互实现选择文件、显示信息以及执行替换操作。
- 使用
QAxObject
实现了与 Word 的 COM 接口的交互,允许直接操作 Word 文档中的内容。
关于QGC地面站其它文章请点击这里: QT Widget
相关文章:

Qt WORD/PDF(四)使用 QAxObject 对 Word 替换(QWidget)
关于QT Widget 其它文章请点击这里: QT Widget 国际站点 GitHub: https://github.com/chenchuhan 国内站点 Gitee : https://gitee.com/chuck_chee 姊妹篇: Qt WORD/PDF(一)使用 QtPdfium库实现 PDF 操作 Qt WORD/PDF(二…...

音视频学习(二十四):hls协议
基本原理 HLS协议通过将视频文件切分成多个小的媒体段(通常是10秒左右的.ts文件),并通过HTTP传输给客户端。视频播放过程中,客户端按顺序请求这些小段文件来逐步播放整个视频流。HLS还支持多种码率,以便适应不同网络条…...

UniDepth 学习笔记
摘要 准确的单目度量深度估计(MMDE)是解决三维感知和建模中下游任务的关键。然而,最近的MMDE方法的显著准确性仅限于其训练领域。这些方法存在适度的域间隙,也不能推广到看不见的域,这阻碍了它们的实际适用性。本文提出…...

PVE——OpenWRT 硬盘 size单位的调整
问题:初始状态为120MB 还需要进行计算,如果通过图形界面添加磁盘会出现单位不变的情况。 进入命令行前记得给你的虚拟机拍照,防止误操作 通过ssh 进入PVE命令行 按需添加容量即可 不到1G 会显示M 超过1G 不是G整数均为M单位。 …...

Android-ImagesPickers 拍照崩溃优化
Android-ImagesPickers 作为老牌图片选择器,帮助了很多牛马宝宝,刚好最近用到了多相册选择以及拍照,可能是高版本机型问题,导致拍照后就闪退 原作者文章以及git Android实用视图动画及工具系列之九:漂亮的图片选择器…...

Linux dd 命令详解:工作原理与实用指南(C/C++代码实现)
这段代码是一个模仿 Linux dd 命令的工具,它用于在不同文件之间复制数据。dd 是一个非常强大的命令行工具,可以用于数据备份、转换和复制。下面我将详细解释这段代码的原理、实现方式以及如何运行和测试。 Linux dd 命令的工作原理 dd 命令是 Unix 和 …...

Golang学习历程【第一篇 入门】
Golang学习历程【第一篇 入门Hello World】 1. 学习文档2. Window 本地安装Go2.1 安装2.2 验证 3. 开发环境——VsCode3.1 VsCode 安装3.2 安装插件3.2.1 language 语言汉化插件安装3.2.2 Go插件安装 4. Hello World 入门4.1 建工程4.2 创建项目文件4.3 编写Hello World程序4.4…...

青少年编程与数学 02-004 Go语言Web编程 01课题、Web应用程序
青少年编程与数学 02-004 Go语言Web编程 01课题、Web应用程序 课题摘要:一、Web应用程序二、Web服务器(一)什么是Web服务器(二)Web服务器配置1. 选择服务器软件2. 安装服务器软件3. 配置服务器4. 安全设置5. 部署网站内容6. 测试服…...

【mysql】如何解决主从架构从库延迟问题
目录 1. 说明2.优化主库的写入性能3. 优化网络性能4. 增强从库的硬件性能5. 调整从库的配置6. 主从架构优化7. 监控和调优8.使用 GTID 和 Group Replication 1. 说明 1.在 MySQL 数据库中,从库延迟(replication lag)是指主库和从库之间的数据…...

前端学习-获取DOM对象(二十一)
目录 前言 目标 提问 学习路径 根据CSS选择器来获取DOM元素 其他获取DOM元素的方法 根据CSS选择器来获取DOM元素 选择匹配的第一个元素 语法 示例 参数 返回值 选择匹配的多个元素语法 参数 字符串返回值 示例 补充 其它获取DOM元素方法 根据id获取一个元素 …...

PCL点云库入门——PCL库中Eigen数学工具库的基本使用(持续更新)
0、前言 PCL点云库中的算法都基于Eigen数学工具库来实现的,因此,掌握Eigen库对于深入理解和应用PCL点云库至关重要。Eigen库不仅提供了高效的矩阵和向量运算,还支持复杂的线性代数、几何变换等操作,为PCL点云处理提供了强大的数学…...

CLion Inlay Hints - 取消 CLion 灰色的参数和类型提示
CLion Inlay Hints - 取消 CLion 灰色的参数和类型提示 1. Parameter hints for C/C1.1. Toggle parameter hints globally 2. Type hints for C/C2.1. Toggle type hints globally 3. Toggle inlay hints globallyReferences https://www.jetbrains.com/help/clio…...

2025山东科技大学考研专业课复习资料一览
[冲刺]2025年山东科技大学020200应用经济学《814经济学之西方经济学[宏观部分]》考研学霸狂刷870题[简答论述计算题]1小时前[强化]2025年山东科技大学085600材料与化工《817物理化学》考研强化检测5套卷22小时前[冲刺]2025年山东科技大学030100法学《704综合一[法理学、国际法学…...

vue3 v-model实例之二,tab标签页的实现
<template><div><Tab v-model"activeTab" :list"tabs" /><div><p>当前激活的 Tab 索引: {{ activeTab }}</p></div></div> </template><script setup> import { ref } from vue; import Tab …...

东方通TongWeb7.0.4.9M4部署SuperMap iServer 11.2.1
一、软件版本 操作系统: CentOS Linux release 7.5.1804 (Core)JDK:11.0.18东方通:TongWeb7.0.4.9M4SuperMap iServer:11.2.1 JDK和TongWeb软件分享: 链接: https://pan.baidu.com/s/1HGDTPnPID0PEOMbg3FjTVQ?pwdbh8v 提取码: bh8v 东方通软…...

QT绘制同心扇形
void ChartForm::paintEvent(QPaintEvent *) {QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);// 设置抗锯齿painter.save();// 设置无边框(不需要设置QPen,因为默认是不绘制边框的)QPen pen(Qt::NoPen);// QPen pen…...

2012年西部数学奥林匹克试题(几何)
2012/G1 △ A B C \triangle ABC △ABC 内有一点 P P P, P P P 在 A B AB AB, A C AC AC 上的投影分别为 E E E, F F F, 射线 B P BP BP, C P CP CP 分别交 △ A B C \triangle ABC △ABC 的外接圆于点 M M M, N N N. r r r 为 △ A B C \triangle ABC △ABC 的内…...

8位移位寄存器的verilog语言
module shift_register (output reg [7:0] Q, // 8位移位寄存器输出input D, // 输入数据input rst, // 复位信号input clk // 时钟信号 );always (posedge clk) beginif (!rst)Q < 8b00000000; // 复位时将Q清零elseQ < {Q[6:0], D}; // 否则…...

【苍穹外卖】学习心得体会-随笔
前言 写了很久,终于可以完整运行项目了,记录下这几天的心得体会回顾一下知识点 第一天、Git 分布式版本控制工具 一、Git概述 定义:是分布式版本控制工具,用于管理软件开发中的源代码文件,像Java类、xml文件、html…...

MySQL学习之表的增删改
MySQL学习之表的增删改 语法总结: INSERT INTO 表名 (字段名1, 字段名2, ...) VALUES (值1, 值2, ...); //指定字段添加数据 INSERT INTO 表名 VALUES (值1, 值2, ...); //给全部字段添加数据 INSERT INTO 表名 VALUES (值1, 值2, ...), (值1, 值2, ...), (值1, …...

电脑excel词典(xllex.dll)文件丢失是或损坏是什么原因?“xllex.dll文件缺失“要怎么解决?
Excel词典(xllex.dll)文件丢失或损坏?别担心,这里有解决之道! 在日常的电脑使用和办公软件操作中,我们偶尔会碰到一些让人头疼的问题,比如Excel突然提示“Excel词典(xllex.dll&…...

【CSS in Depth 2 精译_084】第 14 章:CSS 蒙版、形状与剪切概述 + 14.1:CSS 滤镜
当前内容所在位置(可进入专栏查看其他译好的章节内容) 第四部分 视觉增强技术 ✔️【第 14 章 蒙版、形状与剪切】 ✔️ 14.1 滤镜 ✔️ 14.1.1 滤镜的类型 ✔️14.1.2 背景滤镜 ✔️ 14.2 蒙版 文章目录 第 14 章 蒙版、形状与剪切 Masks, shapes, and…...

gorm源码解析(四):事务,预编译
文章目录 前言事务自己控制事务用 Transaction方法包装事务 预编译事务结合预编译总结 前言 前几篇文章介绍gorm的整体设计,增删改查的具体实现流程。本文将聚焦与事务和预编译部分 事务 自己控制事务 用gorm框架,可以自己控制事务的Begin࿰…...

前端优雅(装逼)写法(updating····)
1.>>右位移运算符取整数 它将一个数字的二进制位向右移动指定的位数,并在左侧填充符号位(即负数用1填充,正数用0填充)。 比如 2.99934 >> 0:取整结果是2,此处取整并非四舍五入 2.99934 会先…...

黑马Java面试教程_P7_常见集合_P4_HashMap
系列博客目录 文章目录 系列博客目录4. HashMap相关面试题4.4 面试题-HashMap的put方法的具体流程 频54.4.1 hashMap常见属性4.4.2 源码分析 HashMap的构造函数面试文稿: 4.5 讲一讲HashMap的扩容机制 难3频4面试文稿: 4.6 面试题-hashMap的寻址算法 难4…...

使用 CFD 加强水资源管理:全面概述
探索 CFD(计算流体动力学)在增强保护人类健康的土木和水利工程实践方面的重大贡献。 挑战 水资源管理是指规划、开发、分配和管理水资源最佳利用的做法。它包括广泛的活动,旨在确保水得到有效和可持续的利用,以满足各种需求&…...

XXE练习
pikachu-XXE靶场 1.POC:攻击测试 <?xml version"1.0"?> <!DOCTYPE foo [ <!ENTITY xxe "a">]> <foo>&xxe;</foo> 2.EXP:查看文件 <?xml version"1.0"?> <!DOCTYPE foo [ <!ENTITY xxe SY…...

R语言读取hallmarks的gmt文档的不同姿势整理
不同格式各有所用 1.读取数据框格式的 hallmarks <- clusterProfiler::read.gmt("~/genelist/h.all.v7.4.symbols.gmt") #返回的是表格 hallmarks$term<- gsub(HALLMARK_,"",hallmarks$term)适配Y叔的clusterProfiler的后续分析,比如整理后geneli…...

【Nginx-4】Nginx负载均衡策略详解
在现代Web应用中,随着用户访问量的增加,单台服务器往往难以承受巨大的流量压力。为了解决这一问题,负载均衡技术应运而生。Nginx作为一款高性能的Web服务器和反向代理服务器,提供了多种负载均衡策略,能够有效地将请求分…...

Python 的 Decimal的错误计算
摘要 阐述在使用 Python的 Decimal类时,可能产生的错误计算。 在 详述 BigDecimal 的错误计算 中,笔者较为详细地说明了 Java的 BigDecimal可能出错的原因。类似地,Python的 decimal模块中有个 Decimal类,也可用于高精度的十进制…...