如何使用qt开发一个xml发票浏览器,实现按发票样式显示

news2025/4/21 18:26:20

使用Qt开发一个按发票样式显示的XML发票浏览器,如下图所示样式:

一、需求:

1、按税务发票样式显示。

2、拖入即可显示。

3、正确解析xml文件。

二、实现

可以按照以下步骤进行:

1. 创建Qt项目

  • 打开Qt Creator,创建一个新的Qt Widgets Application项目。

  • 设置项目名称和路径,选择合适的Qt版本。

2. 导入必要的模块

在项目的.pro文件中,确保导入了xml模块,以便使用Qt的XML解析功能:

pro复制

QT += core gui widgets xml

3. 设计UI界面

  • 使用Qt Designer设计发票浏览器的UI界面。可以添加一个QTextBrowserQTextEdit控件用于显示发票内容,还可以添加一些按钮用于加载和解析XML文件。

  • 保存UI文件并生成对应的.h.cpp文件。

4. 加载和解析XML文件

Qt提供了多种方式来解析XML文件,常用的有QDomDocumentQXmlStreamReader。也可以使用python来解析,参考文章。《如何用Python编程实现自动整理XML发票文件_python解析xml发票-CSDN博客》

5. 显示发票内容

将解析出的发票内容格式化后显示在QTextBrowserQTextEdit控件中。可以使用HTML格式来实现发票的样式显示。或者使用QWidget和QTableWidget、QFormLayout、QVBoxLayout等进行界面布局。

cpp复制

#include <QTextBrowser>
#include <QTextStream>

void displayInvoice(QTextBrowser *browser, const QDomDocument &doc) {
    QDomElement root = doc.documentElement();
    QDomNodeList items = root.elementsByTagName("item");

    QString html;
    QTextStream stream(&html);
    stream << "<html><body>";
    stream << "<h1>发票内容</h1>";
    stream << "<table border='1'>";
    stream << "<tr><th>名称</th><th>金额</th></tr>";

    for (int i = 0; i < items.size(); ++i) {
        QDomElement item = items.at(i).toElement();
        QString name = item.firstChildElement("name").text();
        QString amount = item.firstChildElement("amount").text();
        stream << "<tr><td>" << name << "</td><td>" << amount << "</td></tr>";
    }

    stream << "</table>";
    stream << "</body></html>";

    browser->setHtml(html);
}

6. 整合功能

在主窗口类中,将加载、解析和显示功能整合起来。例如,可以为按钮添加槽函数来触发这些操作。

cpp复制

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    connect(ui->loadButton, &QPushButton::clicked, this, &MainWindow::onLoadClicked);
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::onLoadClicked() {
    QString filePath = QFileDialog::getOpenFileName(this, "选择XML文件", "", "XML文件 (*.xml)");
    if (!filePath.isEmpty()) {
        QDomDocument doc;
        QFile file(filePath);
        if (file.open(QIODevice::ReadOnly)) {
            if (doc.setContent(&file)) {
                displayInvoice(ui->textBrowser, doc);
            } else {
                QMessageBox::critical(this, "错误", "无法解析XML文件");
            }
            file.close();
        } else {
            QMessageBox::critical(this, "错误", "无法打开文件");
        }
    }
}

7. 支持拖入

要支持拖入文件的功能,可以通过为窗口启用拖放功能来实现。在Qt中,可以通过设置窗口的dragEnterEventdragMoveEventdropEvent等事件来处理拖放操作。

以下是实现拖入文件功能的详细步骤:

7.1. 启用窗口的拖放功能

在窗口类的构造函数中,启用拖放功能并设置接受拖放的文件类型。

cpp复制

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    // 启用拖放功能
    setAcceptDrops(true);

    connect(ui->loadButton, &QPushButton::clicked, this, &MainWindow::onLoadClicked);
}

7.2. 重写拖放事件处理函数

重写dragEnterEventdragMoveEventdropEvent函数来处理拖放操作。

dragEnterEvent

当文件被拖入窗口时,此事件被触发。可以在这里检查拖入的文件是否符合要求(例如是否是XML文件)。

cpp复制

void MainWindow::dragEnterEvent(QDragEnterEvent *event) {
    if (event->mimeData()->hasUrls()) {
        event->acceptProposedAction(); // 接受拖放操作
    } else {
        event->ignore(); // 忽略其他类型的拖放操作
    }
}
dragMoveEvent

当文件在窗口内移动时,此事件被触发。可以在这里提供视觉反馈,例如高亮显示可放置区域。

cpp复制

void MainWindow::dragMoveEvent(QDragMoveEvent *event) {
    if (event->mimeData()->hasUrls()) {
        event->acceptProposedAction();
    } else {
        event->ignore();
    }
}
dropEvent

当文件被释放(放下)时,此事件被触发。可以在这里获取文件路径并处理文件。

cpp复制

void MainWindow::dropEvent(QDropEvent *event) {
    if (event->mimeData()->hasUrls()) {
        QList<QUrl> urls = event->mimeData()->urls();
        if (!urls.isEmpty()) {
            QString filePath = urls.first().toLocalFile(); // 获取第一个文件的路径
            if (!filePath.isEmpty()) {
                loadInvoice(filePath); // 调用加载发票的函数
            }
        }
    }
    event->acceptProposedAction();
}

7.3. 加载和显示发票内容

loadInvoice函数中,解析XML文件并显示发票内容。这里可以复用前面提到的解析和显示代码。

cpp复制

void MainWindow::loadInvoice(const QString &filePath) {
    QDomDocument doc;
    QFile file(filePath);
    if (file.open(QIODevice::ReadOnly)) {
        if (doc.setContent(&file)) {
            displayInvoice(ui->textBrowser, doc); // 显示发票内容
        } else {
            QMessageBox::critical(this, "错误", "无法解析XML文件");
        }
        file.close();
    } else {
        QMessageBox::critical(this, "错误", "无法打开文件");
    }
}

三. 编译和运行

  • 编译项目并运行。

  • 使用“加载”按钮选择一个XML发票文件,程序将解析并按样式显示发票内容。

  • 将XML文件拖入窗口,程序将解析并按样式显示发票内容。

  • 通过以上步骤,

通过以上步骤,你可以开发一个简单的XML发票浏览器,按发票样式显示内容。你的发票浏览器不仅可以通过按钮加载文件,还可以支持拖入文件的功能。根据实际需求,还可以进一步优化UI界面和功能。

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

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

相关文章

解析 JavaScript 面试题:`index | 0` 确保数组索引为整数

文章目录 一、JavaScript 中的数字类型二、按位或运算符 | 的作用&#xff08;一&#xff09;对于整数&#xff08;二&#xff09;对于小数&#xff08;三&#xff09;对于非数字值 三、用于数组索引的意义 在 JavaScript 面试中&#xff0c;常常会涉及到一些看似简单却蕴含着深…

46 map与set

目录 一、序列式容器和关联式容器 二、set系列的使用 &#xff08;一&#xff09;set和mutilset参考文档链接 &#xff08;二&#xff09;set类模板介绍 1、set类声明 2、set的构造和迭代器 3、set的增删查 &#xff08;三&#xff09;multiset类模板 1、multiset和se…

RAGFlow和Dify对比

‌ RAGFlow和Dify都是基于大语言模型&#xff08;LLM&#xff09;的应用开发平台&#xff0c;具有相似的功能和应用场景&#xff0c;但它们在技术架构、部署要求和用户体验上存在一些差异。‌‌ RAGFlow和Dify对比 2025-02-13 22.08 RAGFlow‌ ‌技术栈‌&#xff1a;RAGFlow…

Dart 3.5语法 14-16

017自定代码段让变量有默认值 List下标访问和2种for循环遍历_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1RZ421p7BL?spm_id_from333.788.videopod.episodes&vd_source68aea1c1d33b45ca3285a52d4ef7365f&p42原作者链接&#xff0c;此为修订补充版本 014main…

yanshee机器人初次使用说明(备注)-PyCharm

准备 需要&#xff1a; 1&#xff0c;&#xff08;优必选&#xff09;yanshee机器人Yanshee 开发者说明 2&#xff0c;手机-联网简单操控 / HDMI线与显示器和键鼠标-图形化开发环境 / 笔记本&#xff08;VNC-内置图形化开发环境/PyCharm等平台&#xff09;。 3&#xff0c;P…

面试题:如何在10亿个数中判断某个数是否存在?

参考视频 参考视频&#xff1a; 如何用10只老鼠试出藏在99瓶清水中的那瓶毒药 参考视频

【设计模式】【行为型模式】观察者模式(Observer)

&#x1f44b;hi&#xff0c;我不是一名外包公司的员工&#xff0c;也不会偷吃茶水间的零食&#xff0c;我的梦想是能写高端CRUD &#x1f525; 2025本人正在沉淀中… 博客更新速度 &#x1f44d; 欢迎点赞、收藏、关注&#xff0c;跟上我的更新节奏 &#x1f3b5; 当你的天空突…

[创业之路-299]:图解金融体系结构

一、金融体系结构 1.1 概述 金融体系结构是一个国家以行政的、法律的形式和运用经济规律确定的金融系统结构&#xff0c;以及构成这个系统的各种类型的银行和非银行金融机构的职能作用和相互关系。以下是对金融体系结构的详细分析&#xff1a; 1、金融体系的构成要素 现代金…

STM32、GD32驱动TM1640原理图、源码分享

一、原理图分享 二、源码分享 /************************************************* * copyright: * author:Xupeng * date:2024-07-18 * description: **************************************************/ #include "smg.h"#define DBG_TAG "smg&…

框架ThinkPHP(小迪网络安全笔记~

免责声明&#xff1a;本文章仅用于交流学习&#xff0c;因文章内容而产生的任何违法&未授权行为&#xff0c;与文章作者无关&#xff01;&#xff01;&#xff01; 附&#xff1a;完整笔记目录~ ps&#xff1a;本人小白&#xff0c;笔记均在个人理解基础上整理&#xff0c;…

Postman如何流畅使用DeepSeek

上次写了一篇文章是用chatBox调用api的方式使用DeepSeek&#xff0c;但是实际只能请求少数几次就不再能给回响应。这回我干脆用最原生的方法Postman调用接口请求好了。 1. 通过下载安装Postman软件 postman下载(https://pan.quark.cn/s/c8d1c7d526f3)&#xff0c;包含7.0和10…

土星云边缘计算微服务器 SE110S-WA32加持DeepSeek,本地部署企业私有推理大模型!

模型介绍 DeepSeek-R1-Distill-Qwen-7B是一款高性能的语言模型&#xff0c;基于DeepSeek-R1的推理能力&#xff0c;通过蒸馏技术将推理模式迁移到较小的Qwen模型上&#xff0c;在保持高性能的同时&#xff0c;显著降低了资源消耗&#xff0c;更适合在资源受限的环境中部署。 该…

Linux权限提升-内核溢出

一&#xff1a;Web到Linux-内核溢出Dcow 复现环境&#xff1a;https://www.vulnhub.com/entry/lampiao-1,249/ 1.信息收集&#xff1a;探测⽬标ip及开发端⼝ 2.Web漏洞利⽤&#xff1a; 查找drupal相关漏洞 search drupal # 进⾏漏洞利⽤ use exploit/unix/webapp/drupal_dr…

ThinkPHP8视图赋值与渲染

【图书介绍】《ThinkPHP 8高效构建Web应用》-CSDN博客 《2025新书 ThinkPHP 8高效构建Web应用 编程与应用开发丛书 夏磊 清华大学出版社教材书籍 9787302678236 ThinkPHP 8高效构建Web应用》【摘要 书评 试读】- 京东图书 在控制器操作中&#xff0c;使用view函数可以传入视图…

微信小程序网络请求封装

微信小程序的网络请求为什么要封装&#xff1f;封装使用有什么好处&#xff1f; 封装的目的是为了偷懒&#xff0c;试想一下每次都要wx.request&#xff0c;巴拉巴拉传一堆参数&#xff0c;是不是很麻烦&#xff0c;有些公共的参数例如header&#xff0c;baseUrl是不是可以封装…

瑞芯微烧写工具

文章目录 前言一、安装驱动二、安装烧写工具1.直接解压压缩包2. 如何使用 三、MASKROM 裸机必备四、LOADER 烧写&#xff0c;前提是搞过第三步没问题五、Update.img包的烧录六、linux下烧写总结 前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 项目需要…

抖音SEO短视频矩阵系统源码:短视频流量密码揭秘

在开发短视频SEO优化排名技术时&#xff0c;仅通过get和set这两个代理无法完全实现目标。实际上&#xff0c;还需要实现has、ownKeys以及getOwnPropertyDescriptor等代理&#xff0c;以更全面地控制私有属性的访问权限。这些代理对于限制对私有属性的访问至关重要。 该技术主要…

【工业安全】-CVE-2022-35561- Tenda W6路由器 栈溢出漏洞

文章目录 1.漏洞描述 2.环境搭建 3.漏洞复现 4.漏洞分析 4.1&#xff1a;代码分析 4.2&#xff1a;流量分析 5.poc代码&#xff1a; 1.漏洞描述 漏洞编号&#xff1a;CVE-2022-35561 漏洞名称&#xff1a;Tenda W6 栈溢出漏洞 威胁等级&#xff1a;高危 漏洞详情&#xff1…

【GRPO】GRPO原理原文翻译

论文&#xff1a;DeepSeekMath: Pushing the Limits of Mathematical Reasoning in Open Language Models 注&#xff01;这里我仅仅翻译GRPO部分供学习使用。其他部分请去看原文。 4. 强化学习&#xff08;Reinforcement Learning&#xff09; 4.1. 群组相对策略优化&#xf…

侯捷 C++ 课程学习笔记:C++ 新标准 11/14 的革新与实战应用

在侯捷老师的 C 系列课程中&#xff0c;《C 新标准 11/14》这门课程让我对现代 C 编程有了全新的认识。C11 和 C14 是 C 语言发展史上的重要里程碑&#xff0c;它们引入了大量新特性&#xff0c;极大地提升了语言的表达能力和开发效率。侯捷老师通过深入浅出的讲解和丰富的实战…