使用Qt WebEngineView在特定页面打开PDF



我想在Qt应用程序中创建一个简单的PDF查看器。一切都很好,接受通过URL在特定页面打开PDF。例如:

url = "file:///D://Repo//PdfViewer//PdfViewer//test.pdf";

有效,但

url = "file:///D://Repo//PdfViewer//PdfViewer//test.pdf#page=9";

剂量。我在某个地方读到,chrome不再正式支持#page=x,但我找不到更多的信息来解决这个问题。我在Qt的Nano浏览器示例中也遇到了同样的问题。

PdfViewer::PdfViewer(const QString &pdf_path, QWidget *parent)
: QWidget(parent), m_View(new QWebEngineView(this)), m_ExitButton(new QPushButton())
{
QUrl url = QUrl::fromLocalFile(pdf_path);
m_View->settings()->setAttribute(QWebEngineSettings::PluginsEnabled, true);
m_View->settings()->setAttribute(QWebEngineSettings::FullScreenSupportEnabled, true);
m_View->settings()->setAttribute(QWebEngineSettings::PdfViewerEnabled, true);
m_View->load(url);
m_ExitButton->setIcon(QIcon("Ok.png"));
QVBoxLayout *layout = new QVBoxLayout();
layout->addWidget(m_View);
layout->addWidget(m_ExitButton);
this->setLayout(layout);
connect(m_ExitButton, &QPushButton::clicked, this, &PdfViewer::close);
}

我使用的是问题5.13。

Qt WebEngine使用chromium pdf查看器,因此通过分析源代码,我发现了实现页面移动的函数:window.viewer.viewport_.goToPage(page),可以使用QWebEnginePage的runJavaScript()方法执行。

考虑到以上情况,解决方案是:

#include <QtWebEngineWidgets>
class PdfViewer: public QWidget{
Q_OBJECT
public:
PdfViewer(const QString &pdf_path, QWidget *parent=nullptr)
: QWidget(parent), m_View(new QWebEngineView(this)), m_ExitButton(new QPushButton())
{
QUrl url = QUrl::fromLocalFile(pdf_path);
m_View->settings()->setAttribute(QWebEngineSettings::PluginsEnabled, true);
m_View->settings()->setAttribute(QWebEngineSettings::FullScreenSupportEnabled, true);
m_View->settings()->setAttribute(QWebEngineSettings::PdfViewerEnabled, true);
m_View->load(url);
m_ExitButton->setIcon(QIcon("Ok.png"));
QVBoxLayout *layout = new QVBoxLayout();
layout->addWidget(m_View);
layout->addWidget(m_ExitButton);
this->setLayout(layout);
connect(m_ExitButton, &QPushButton::clicked, this, &PdfViewer::close);
connect(m_View, &QWebEngineView::loadFinished, this, &PdfViewer::on_finished);
}
private Q_SLOTS:
void on_finished(bool ok){
if(ok){
QTimer::singleShot(100, this, [this](){ goToPage(9); });
}
}
private:
void goToPage(int page){
m_View->page()->runJavaScript(QString("window.viewer.viewport_.goToPage(%1)").arg(page));
}
QWebEngineView *m_View;
QPushButton *m_ExitButton;
};
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QString fileName = QFileDialog::getOpenFileName(nullptr,
QObject::tr("Open Image"),
QDir::homePath(),
QObject::tr("PDF Files (*.pdf)"));
if(fileName.isEmpty())
return 0;
PdfViewer w(fileName);
w.resize(640, 480);
w.show();
return a.exec();
}
#include "main.moc"

这是QtWebEngine中缺少的功能,但现在最新的Qt 5.15版本(或Qt 6.2(支持它。https://bugreports.qt.io/browse/QTBUG-86152

最新更新