Qt WORD/PDF(二)使用 QtPdfium库实现 PDF操作、打印等

server/2024/12/18 21:17:51/

关于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)


一、简介

QtPdfium 是基于Pdfium库的一个Qt绑定。Pdfium是一个由Google开发并开源的PDF渲染引擎,它被广泛应用于Chrome浏览器和其他一些Google产品中。QtPdfium则是一个专门将Pdfium引入Qt项目的封装,使得Qt应用程序可以使用Pdfium的PDF渲染功能。

它的主要目标是高效的PDF文档渲染,特别是在Web浏览器等高性能需求的应用场景中。相较于Poppler,Pdfium侧重于简洁、高效的渲染和性能优化。

二、演示

在这里插入图片描述

三、代码部分

完整代码

mainwindow.cpp:

#include "mainwindow.h"// #include "ui_mainwindow.h"
#include <QVBoxLayout>
#include <QFileDialog>
#include <QPrinter>
#include <QPrintDialog>
#include <QPainter>
#include <QPrinterInfo>MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)// , ui(new Ui::MainWindow)
{// 设置窗口标题setWindowTitle("PDF Viewer & Printer");this->initWidget();// 主控件和布局QWidget *centralWidget = new QWidget(this);QVBoxLayout *layout = new QVBoxLayout(centralWidget);// PDF 预览区域m_pdfPreview = new QLabel("PDF 预览区域", this);m_pdfPreview->setFixedSize(700, 880);m_pdfPreview->setStyleSheet("border: 1px solid gray; background-color: #f0f0f0;");m_pdfPreview->setAlignment(Qt::AlignCenter);layout->addWidget(m_pdfPreview);//增加一行页码相关,页码控制区域//上一页 当前页/总页码 下一页QHBoxLayout *pageControlLayout = new QHBoxLayout();_prevPageButton = new QPushButton("上一页", this);_pageInfoLabel = new QLabel("0 / 0", this);_nextPageButton = new QPushButton("下一页", this);_pageInfoLabel->setAlignment(Qt::AlignCenter);// 将控件添加到水平布局中pageControlLayout->addWidget(_prevPageButton);pageControlLayout->addWidget(_pageInfoLabel);pageControlLayout->addWidget(_nextPageButton);layout->addLayout(pageControlLayout);// 打印设置区域QHBoxLayout *printerLayout = new QHBoxLayout();m_printerList = new QComboBox(this);m_currentPage = new QRadioButton("当前页", this);m_allPages = new QRadioButton("所有页", this);m_allPages->setChecked(true); // 默认打印所有页m_rangeGroup = new QButtonGroup(this);m_rangeGroup->addButton(m_currentPage);m_rangeGroup->addButton(m_allPages);printerLayout->addWidget(new QLabel("打印机:", this));printerLayout->addWidget(m_printerList);printerLayout->addWidget(m_currentPage);printerLayout->addWidget(m_allPages);layout->addLayout(printerLayout);// 按钮区域QHBoxLayout *buttonLayout = new QHBoxLayout();m_loadButton = new QPushButton("加载PDF", this);m_printButton = new QPushButton("打印PDF", this);buttonLayout->addWidget(m_loadButton);buttonLayout->addWidget(m_printButton);layout->addLayout(buttonLayout);setCentralWidget(centralWidget);// 填充打印机列表populatePrinters();// 信号与槽连接connect(_prevPageButton, &QPushButton::clicked, this, &MainWindow::goToPreviousPage);connect(_nextPageButton, &QPushButton::clicked, this, &MainWindow::goToNextPage);connect(m_loadButton, &QPushButton::clicked, this, &MainWindow::loadPdf);connect(m_printButton, &QPushButton::clicked, this, &MainWindow::printPdf);}MainWindow::~MainWindow()
{m_pdfium = nullptr;delete m_pdfium;
}//初始化界面
void MainWindow::initWidget()
{//初始化变量_currentPage = 0;               // 当前页码_totalPages = 0;                // 总页数
}void MainWindow::populatePrinters()
{// 获取系统打印机列表QList<QPrinterInfo> printers = QPrinterInfo::availablePrinters();for (const QPrinterInfo &printer : printers) {m_printerList->addItem(printer.printerName());}
}//鼠标滚轮事件,放大缩小
void MainWindow::wheelEvent(QWheelEvent *event)
{if(!isPdfValid) {return;}//根据鼠标滚轮的方向来缩放图像const QPoint numDegrees = event->angleDelta() / 8;const int numSteps = numDegrees.y() / 15;if(numSteps != 0){//更新缩放比例if(numSteps > 0) {//放大zoomScale *= 1.1;}else {//缩小,但要确保缩放比例不会小于某个最小值(比如0.1)zoomScale /= 1.1;zoomScale = qMax(zoomScale, 0.3);}//更新缩放后的图像并显示updateScaledPixmap();}
}//更新缩放图像
void MainWindow::updateScaledPixmap()
{//计算缩放后的尺寸QSize scaledSize = _showPixmap.size() * zoomScale;//缩放图像QPixmap scaledPixmap = _showPixmap.scaled(scaledSize,Qt::IgnoreAspectRatio,Qt::SmoothTransformation);//更新QLabel的Pixmapm_pdfPreview->setPixmap(scaledPixmap);
}void MainWindow::loadPdf()
{// 打开文件对话框加载 PDFQString fileName = QFileDialog::getOpenFileName(this, "选择PDF文件", "D:/23_Outproject/32_ZNPicture/Src/QtPdfiumTest2", "PDF Files (*.pdf)");if (fileName.isEmpty()) {isPdfValid = false;return;}m_pdfium = new QPdfium();// 加载 PDF 文件m_pdfium->loadFile(fileName);isPdfValid = m_pdfium->isValid();if (!isPdfValid) {m_pdfPreview->setText("加载PDF失败!");return;}_totalPages = m_pdfium->pageCount(); // 更新总页数_currentPage = 0;                    // 当前页重置为第一页_pageInfoLabel->setText(QString("%1 / %2").arg(_currentPage+1).arg(_totalPages));//预览第一页QPdfiumPage page = m_pdfium->page(_currentPage);if (page.isValid()) {QImage image = page.image(1.0); // 渲染为 QImage_showPixmap = QPixmap::fromImage(image).scaled (m_pdfPreview->size(),Qt::KeepAspectRatio,Qt::SmoothTransformation);m_pdfPreview->setPixmap(_showPixmap);}
}//上一页
void MainWindow::goToPreviousPage()
{if (_currentPage > 0) {_currentPage--;_pageInfoLabel->setText(QString("%1 / %2").arg(_currentPage+1).arg(_totalPages));QPdfiumPage page = m_pdfium->page(_currentPage);if (page.isValid()) {QImage image = page.image(1.0);_showPixmap = QPixmap::fromImage(image).scaled (m_pdfPreview->size(),Qt::KeepAspectRatio,Qt::SmoothTransformation);m_pdfPreview->setPixmap(_showPixmap);}}
}//下一页
void MainWindow::goToNextPage()
{if (_currentPage < _totalPages-1) {_currentPage++;_pageInfoLabel->setText(QString("%1 / %2").arg(_currentPage+1).arg(_totalPages));QPdfiumPage page = m_pdfium->page(_currentPage);if (page.isValid()) {QImage image = page.image(1.0);_showPixmap = QPixmap::fromImage(image).scaled (m_pdfPreview->size(),Qt::KeepAspectRatio,Qt::SmoothTransformation);m_pdfPreview->setPixmap(_showPixmap);}}
}void MainWindow::printPdf()
{// 调用打印功能if (!m_pdfium || !m_pdfium->isValid() || m_pdfium->pageCount() == 0) {m_pdfPreview->setText("未加载有效PDF,无法打印!");return;}// 创建打印机对象QPrinter printer(QPrinter::HighResolution);printer.setPrinterName(m_printerList->currentText());//是用来设置打印页面的方向,指定打印内容的布局是 纵向 (Portrait) 还是 横向 (Landscape)。printer.setOrientation(QPrinter::Portrait);printer.setPageSize(QPrinter::A4);// 设置打印范围int fromPage = 1;int toPage = m_pdfium->pageCount();if (m_currentPage->isChecked()) {fromPage = toPage = 1; // 当前页为第一页}// 打印 PDFQPainter painter(&printer);for (int i = fromPage; i <= toPage; ++i) {QPdfiumPage page = m_pdfium->page(i);if (!page.isValid()) continue;QImage image = page.image(1.0);QRect targetRect = printer.pageRect();QPixmap pixmap = QPixmap::fromImage(image).scaled (targetRect.size(),Qt::KeepAspectRatio,Qt::SmoothTransformation);painter.drawPixmap(0, 0, pixmap);if (i < toPage)printer.newPage();}painter.end();
}

mainwindow.h:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>#include <QFileDialog>
#include <QDesktopServices>
#include <QTime>
#include <QImage>
#include <QPixmap>
#include <QWheelEvent>
#include <QDebug>#include "qpdfium.h"#include <QPrinter>
#include <QPainter>
#include <QPrintDialog>
#include <QFileDialog>//Item
#include <QWheelEvent>
#include <QLabel>
#include <QPushButton>
#include <QComboBox>
#include <QRadioButton>
#include <QPrinter>
#include <QButtonGroup>QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();void initWidget();  //         初始化参数void showPdfData(int curPage);void updateScaledPixmap();protected:void wheelEvent(QWheelEvent *event) override;private slots:void goToPreviousPage();       // 上一页槽函数void goToNextPage();           // 下一页槽函数void loadPdf();void printPdf();private:Ui::MainWindow *ui;int m_curPage;                  //当前页int m_totalPage;                //总页数double  zoomScale;              //缩放比例bool    isPdfValid = false;     //打开pdf是否有效QString m_saveImagePath;        //图像保存路径QPixmap _showPixmap;           //界面显示图像//Item1QPushButton *_prevPageButton; // 上一页按钮QPushButton *_nextPageButton; // 下一页按钮QLabel      *_pageInfoLabel;  // 当前页/总页码显示//Item2QPdfium *m_pdfium = nullptr;    //QPdfium对象QLabel *m_pdfPreview;           // 用于显示PDF预览QPushButton *m_loadButton;      // 加载PDF按钮QPushButton *m_printButton;     // 打印按钮//基本参数int _currentPage;               // 当前页码int _totalPages;                // 总页数QComboBox *m_printerList;       // 打印设备列表QRadioButton *m_currentPage;    // 打印范围:当前页QRadioButton *m_allPages;       // 打印范围:所有页QButtonGroup *m_rangeGroup;     // 打印范围按钮组void populatePrinters();        // 获取系统打印机
};
#endif // MAINWINDOW_H

pro 中需要增加对打印的支持

QT       += core gui printsupport

简要分析

1. 初始化界面 (MainWindow::MainWindow)

  • 创建并配置了主界面控件,包括 PDF 预览区域、页码控制、打印设置、按钮区域。
  • 使用 QVBoxLayoutQHBoxLayout 布局管理器来组织控件。
  • QLabel 用于显示 PDF 预览,QPushButton 控件用于加载 PDF 文件和打印 PDF。
  • QComboBoxQRadioButton 用于选择打印机和打印范围(当前页或所有页)。
  • 设置了信号与槽连接:控制按钮的点击事件(加载 PDF、打印 PDF、翻页等)。

2. 控件的布局和显示

  • PDF 预览区域使用 QLabel 来显示 PDF 页面图像,用户可以通过上下页按钮查看不同页。
  • 页码信息通过 QLabel 显示(例如 “1 / 10” 表示当前第 1 页,总共有 10 页)。
  • 打印区域提供了选择打印机和打印选项(当前页或所有页)。
  • 通过 QPushButton 控件加载和打印 PDF 文件。

3. 加载 PDF 文件 (MainWindow::loadPdf)

  • 使用 QFileDialog::getOpenFileName 打开文件对话框,让用户选择 PDF 文件。
  • 通过 QPdfium 类(可能是一个自定义类)加载 PDF 文件。若加载成功,更新总页数并显示第一页。
  • QPdfiumPage 用于获取指定页的图像,将其转换为 QPixmap 后显示在 QLabel 中。

4. 翻页功能

  • goToPreviousPagegoToNextPage 用于上一页和下一页功能,页面切换时更新页码并渲染该页图像。

5. 打印功能 (MainWindow::printPdf)

  • 使用 QPrinter 设置打印机、页面大小、打印方向等。
  • QPainter 用于将 PDF 页面的图像绘制到打印机上。
  • 如果选择打印当前页,则仅打印当前页;如果选择打印所有页,则打印整个 PDF 文件的所有页面。
  • 在每一页打印完后,调用 printer.newPage() 使打印机跳转到下一页。

6. 鼠标滚轮放大缩小

  • wheelEvent 捕获鼠标滚轮事件,根据滚动方向来缩放 PDF 页面的显示。缩放比例由 zoomScale 控制,放大时乘以 1.1,缩小时除以 1.1。 调整后更新显示;

7. 打印机列表填充

  • populatePrinters 方法列出系统中的打印机,将其添加到 QComboBox 控件中

注意:

  1. 使用 QtPdfium 偶尔无法读中文路径的PDF,不知道是不是库本身的问题;
  2. PDF 渲染效率:每次翻页时都重新加载页面并进行图像渲染,若 PDF 页数较多或者页面较复杂时可能会影响性能。可以考虑缓存已经渲染过的页面,减少重复渲染

参考:

Qt下使用QtPdfium处理PDF文档


关于QT Widget 其它文章请点击这里:     QT Widget


http://www.ppmy.cn/server/151271.html

相关文章

牛客周赛71(字符串,状压dp)

目录 B. 宝石手串 D. 气球谜题 B. 宝石手串 &#xff08;1&#xff09;两种扩容方式&#xff1a; // 法一&#xff1a;直接加&#xff08;通常用于拼接字符串&#xff09;s s// 法二&#xff1a;一个一个字符加&#xff08;用于加单个字符&#xff09;for (…

[Unity]Unity跨平台开发之针对Android开发

用户手册的这一部分包含Android平台关于输入&#xff08;input&#xff09;、资产管理&#xff08;asset management&#xff09;和调试&#xff08;debugging&#xff09;等相关主题的开发信息。 Android移动脚本编写 注意&#xff1a;安卓可以在C#中使用UNITY_ANDROID来进行…

阿里数据仓库-数据模型建设方法总结

一、大数据领域建模综述 1.1 为什么需要数据建模 有结构地分类组织和存储是我们面临的一个挑战。 数据模型强调从业务、数据存取和使用角度合理存储数据。 数据模型方法,以便在性能、成本、效率之间取得最佳平衡 成本:良好的数据模型能极大地减少不必要的数据冗余,也能实现…

k8s中设置annotation的方法总结

k8s中设置annotation的方法总结 annotation是什么 在 Kubernetes 中&#xff0c;Annotations 是一种用于向 Kubernetes 对象附加非标识性元数据的机制。 annotation有什么用 annotation与 Labels 类似&#xff0c;但有一些关键区别和特定用途。 常用于存储与对象相关的配置…

Spring Boot 条件注解:@ConditionalOnProperty 完全解析

在 Spring Boot 项目中&#xff0c;有时候我们希望根据配置文件中的某个属性值来决定是否启用某个功能或加载某个组件。此时&#xff0c;ConditionalOnProperty 注解就可以发挥作用。它通过配置文件的属性值控制 Bean 或配置类的加载&#xff0c;使得我们的程序更具灵活性。 本…

Flink SQL保留关键字

官方参考资料&#xff1a; SQL | Apache Flink 在使用flink时&#xff0c;如果不小心用到了flink保留关键字&#xff0c;会产生关键字错误。关于flink关键字&#xff0c;官方原话是“Although not every SQL feature is implemented yet, some string combinations are alread…

限制redis内存

要限制Redis的内存使用&#xff0c;可以在Redis的配置文件中设置maxmemory参数。以下是如何在Docker环境中限制Redis内存的步骤&#xff1a; 编辑Redis配置文件&#xff1a; 已经创建了Redis的配置文件/mydata/redis/conf/redis.conf&#xff0c;现在需要在这个文件中添加或修…

每天40分玩转Django:Django视图和URL

Django视图和URL 一、课程概述 学习项目具体内容预计用时视图基础函数视图、类视图、视图装饰器90分钟URL配置URL模式、路由系统、命名URL60分钟请求处理请求对象、响应对象、中间件90分钟 二、视图基础 2.1 函数视图 # blog/views.py from django.shortcuts import render…