Files
qt-creator/src/plugins/help/webenginehelpviewer.cpp

381 lines
13 KiB
C++
Raw Normal View History

/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of Qt Creator.
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3 as published by the Free Software
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
** included in the packaging of this file. Please review the following
** information to ensure the GNU General Public License requirements will
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
**
****************************************************************************/
#include "webenginehelpviewer.h"
#include "helpconstants.h"
#include "localhelpmanager.h"
#include "openpagesmanager.h"
#include <utils/qtcassert.h>
#include <QBuffer>
#include <QContextMenuEvent>
#include <QCoreApplication>
#include <QDesktopServices>
#include <QTimer>
#include <QVBoxLayout>
#if QT_VERSION >= QT_VERSION_CHECK(6, 2, 0)
#include <QWebEngineContextMenuRequest>
#else
#include <QWebEngineContextMenuData>
#endif
#include <QWebEngineHistory>
#include <QWebEngineProfile>
#include <QWebEngineSettings>
#include <QWebEngineUrlRequestJob>
using namespace Help;
using namespace Help::Internal;
HelpUrlSchemeHandler::HelpUrlSchemeHandler(QObject *parent) :
QWebEngineUrlSchemeHandler(parent)
{
}
void HelpUrlSchemeHandler::requestStarted(QWebEngineUrlRequestJob *job)
{
const QUrl url = job->requestUrl();
if (!HelpViewer::isLocalUrl(url)) {
job->fail(QWebEngineUrlRequestJob::RequestDenied);
return;
}
LocalHelpManager::HelpData data = LocalHelpManager::helpData(url);
auto buffer = new QBuffer(job);
buffer->setData(data.data);
job->reply(data.mimeType.toUtf8(), buffer);
}
static HelpUrlSchemeHandler *helpUrlSchemeHandler()
{
static HelpUrlSchemeHandler *schemeHandler = nullptr;
if (!schemeHandler)
schemeHandler = new HelpUrlSchemeHandler(LocalHelpManager::instance());
return schemeHandler;
}
HelpUrlRequestInterceptor::HelpUrlRequestInterceptor(QObject *parent)
: QWebEngineUrlRequestInterceptor(parent)
{}
void HelpUrlRequestInterceptor::interceptRequest(QWebEngineUrlRequestInfo &info)
{
if (!HelpViewer::isLocalUrl(info.requestUrl())
&& info.navigationType() != QWebEngineUrlRequestInfo::NavigationTypeLink) {
info.block(true);
}
}
static HelpUrlRequestInterceptor *helpurlRequestInterceptor()
{
static HelpUrlRequestInterceptor *interceptor = nullptr;
if (!interceptor)
interceptor = new HelpUrlRequestInterceptor(LocalHelpManager::instance());
return interceptor;
}
WebEngineHelpViewer::WebEngineHelpViewer(QWidget *parent) :
HelpViewer(parent),
m_widget(new WebView(this))
{
// some of these should already be that way by default, but better be sure
QWebEngineSettings *settings = m_widget->settings();
settings->setAttribute(QWebEngineSettings::JavascriptCanOpenWindows, false);
settings->setAttribute(QWebEngineSettings::LocalContentCanAccessRemoteUrls, false);
settings->setAttribute(QWebEngineSettings::XSSAuditingEnabled, true);
settings->setAttribute(QWebEngineSettings::PluginsEnabled, false);
settings->setAttribute(QWebEngineSettings::AllowRunningInsecureContent, false);
settings->setAttribute(QWebEngineSettings::AllowGeolocationOnInsecureOrigins, false);
settings->setAttribute(QWebEngineSettings::AllowWindowActivationFromJavaScript, false);
m_widget->setPage(new WebEngineHelpPage(this));
auto layout = new QVBoxLayout;
setLayout(layout);
layout->setContentsMargins(0, 0, 0, 0);
layout->addWidget(m_widget, 10);
QPalette p = palette();
p.setColor(QPalette::Base, Qt::white);
p.setColor(QPalette::Text, Qt::black);
setPalette(p);
connect(m_widget, &QWebEngineView::urlChanged, this, &WebEngineHelpViewer::sourceChanged);
connect(m_widget, &QWebEngineView::loadStarted, this, [this] {
slotLoadStarted();
// Work around QTBUG-65223: if only anchor changed, we never get a loadFinished signal
// If a link is clicked in a page, it can happen that the new URL has not yet been set,
// so we need to delay a bit...
QTimer::singleShot(/*magic timeout=*/150, this, [this] {
QUrl urlWithoutFragment = source();
urlWithoutFragment.setFragment(QString());
if (urlWithoutFragment == m_previousUrlWithoutFragment)
slotLoadFinished();
m_previousUrlWithoutFragment = urlWithoutFragment;
});
});
connect(m_widget, &QWebEngineView::loadFinished, this, &WebEngineHelpViewer::slotLoadFinished);
connect(m_widget, &QWebEngineView::titleChanged, this, &WebEngineHelpViewer::titleChanged);
connect(m_widget->page(), &QWebEnginePage::linkHovered, this, &WebEngineHelpViewer::setToolTip);
connect(m_widget->pageAction(QWebEnginePage::Back), &QAction::changed, this, [this]() {
emit backwardAvailable(isBackwardAvailable());
});
connect(m_widget->pageAction(QWebEnginePage::Forward), &QAction::changed, this, [this]() {
emit forwardAvailable(isForwardAvailable());
});
QAction* action = m_widget->pageAction(QWebEnginePage::OpenLinkInNewTab);
action->setText(QCoreApplication::translate("HelpViewer",
Constants::TR_OPEN_LINK_AS_NEW_PAGE));
QWebEnginePage *viewPage = m_widget->page();
QTC_ASSERT(viewPage, return);
QWebEngineProfile *viewProfile = viewPage->profile();
QTC_ASSERT(viewProfile, return);
if (!viewProfile->urlSchemeHandler("qthelp"))
viewProfile->installUrlSchemeHandler("qthelp", helpUrlSchemeHandler());
viewProfile->setUrlRequestInterceptor(helpurlRequestInterceptor());
}
void WebEngineHelpViewer::setViewerFont(const QFont &font)
{
QWebEngineSettings *webSettings = m_widget->settings();
webSettings->setFontFamily(QWebEngineSettings::StandardFont, font.family());
webSettings->setFontSize(QWebEngineSettings::DefaultFontSize, font.pointSize());
}
void WebEngineHelpViewer::setScale(qreal scale)
{
m_widget->setZoomFactor(scale);
}
QString WebEngineHelpViewer::title() const
{
return m_widget->title();
}
QUrl WebEngineHelpViewer::source() const
{
return m_widget->url();
}
void WebEngineHelpViewer::setSource(const QUrl &url)
{
m_widget->setUrl(url);
}
void WebEngineHelpViewer::setHtml(const QString &html)
{
m_widget->setHtml(html);
}
QString WebEngineHelpViewer::selectedText() const
{
return m_widget->selectedText();
}
bool WebEngineHelpViewer::isForwardAvailable() const
{
// m_view->history()->canGoForward()
return m_widget->pageAction(QWebEnginePage::Forward)->isEnabled();
}
bool WebEngineHelpViewer::isBackwardAvailable() const
{
return m_widget->pageAction(QWebEnginePage::Back)->isEnabled();
}
void WebEngineHelpViewer::addBackHistoryItems(QMenu *backMenu)
{
if (QWebEngineHistory *history = m_widget->history()) {
QList<QWebEngineHistoryItem> items = history->backItems(history->count());
for (int i = items.count() - 1; i >= 0; --i) {
QWebEngineHistoryItem item = items.at(i);
auto action = new QAction(backMenu);
action->setText(item.title());
connect(action, &QAction::triggered, this, [this,item]() {
if (QWebEngineHistory *history = m_widget->history())
history->goToItem(item);
});
backMenu->addAction(action);
}
}
}
void WebEngineHelpViewer::addForwardHistoryItems(QMenu *forwardMenu)
{
if (QWebEngineHistory *history = m_widget->history()) {
QList<QWebEngineHistoryItem> items = history->forwardItems(history->count());
for (int i = 0; i < items.count(); ++i) {
QWebEngineHistoryItem item = items.at(i);
auto action = new QAction(forwardMenu);
action->setText(item.title());
connect(action, &QAction::triggered, this, [this,item]() {
if (QWebEngineHistory *history = m_widget->history())
history->goToItem(item);
});
forwardMenu->addAction(action);
}
}
}
bool WebEngineHelpViewer::findText(const QString &text, Core::FindFlags flags, bool incremental,
bool fromSearch, bool *wrapped)
{
Q_UNUSED(incremental)
Q_UNUSED(fromSearch)
if (wrapped)
*wrapped = false; // missing feature in QWebEngine
QWebEnginePage::FindFlags webEngineFlags;
if (flags & Core::FindBackward)
webEngineFlags |= QWebEnginePage::FindBackward;
if (flags & Core::FindCaseSensitively)
webEngineFlags |= QWebEnginePage::FindCaseSensitively;
// QWebEngineView's findText is asynchronous, and the variant taking a callback runs the
// callback on the main thread, so blocking here becomes ugly too
// So we just claim that the search succeeded
m_widget->findText(text, webEngineFlags);
return true;
}
WebEngineHelpPage *WebEngineHelpViewer::page() const
{
return static_cast<WebEngineHelpPage *>(m_widget->page());
}
void WebEngineHelpViewer::copy()
{
m_widget->triggerPageAction(QWebEnginePage::Copy);
}
void WebEngineHelpViewer::stop()
{
m_widget->triggerPageAction(QWebEnginePage::Stop);
}
void WebEngineHelpViewer::forward()
{
m_widget->triggerPageAction(QWebEnginePage::Forward);
}
void WebEngineHelpViewer::backward()
{
m_widget->triggerPageAction(QWebEnginePage::Back);
}
void WebEngineHelpViewer::print(QPrinter *printer)
{
Q_UNUSED(printer)
}
WebEngineHelpPage::WebEngineHelpPage(QObject *parent)
: QWebEnginePage(parent)
{
}
bool WebEngineHelpPage::acceptNavigationRequest(const QUrl &url,
QWebEnginePage::NavigationType type,
bool isMainFrame)
{
Q_UNUSED(type)
Q_UNUSED(isMainFrame)
if (HelpViewer::isLocalUrl(url))
return true;
QDesktopServices::openUrl(url);
return false;
}
WebView::WebView(WebEngineHelpViewer *viewer)
: QWebEngineView(viewer)
, m_viewer(viewer)
{}
bool WebView::event(QEvent *ev)
{
// work around QTBUG-43602
if (ev->type() == QEvent::ChildAdded) {
auto ce = static_cast<QChildEvent *>(ev);
ce->child()->installEventFilter(this);
} else if (ev->type() == QEvent::ChildRemoved) {
auto ce = static_cast<QChildEvent *>(ev);
ce->child()->removeEventFilter(this);
}
return QWebEngineView::event(ev);
}
bool WebView::eventFilter(QObject *src, QEvent *e)
{
Q_UNUSED(src)
// work around QTBUG-43602
if (m_viewer->isScrollWheelZoomingEnabled() && e->type() == QEvent::Wheel) {
auto we = static_cast<QWheelEvent *>(e);
if (we->modifiers() == Qt::ControlModifier)
return true;
}
return false;
}
void WebView::contextMenuEvent(QContextMenuEvent *event)
{
#if QT_VERSION >= QT_VERSION_CHECK(6, 2, 0)
QMenu *menu = createStandardContextMenu();
#else
QMenu *menu = page()->createStandardContextMenu();
#endif
// insert Open as New Page etc if OpenLinkInThisWindow is also there
const QList<QAction*> actions = menu->actions();
auto it = std::find(actions.cbegin(), actions.cend(),
page()->action(QWebEnginePage::OpenLinkInThisWindow));
if (it != actions.cend()) {
// insert after
++it;
QAction *before = (it == actions.cend() ? 0 : *it);
#if QT_VERSION >= QT_VERSION_CHECK(6, 2, 0)
QUrl url = lastContextMenuRequest()->linkUrl();
#else
QUrl url = page()->contextMenuData().linkUrl();
#endif
if (m_viewer->isActionVisible(HelpViewer::Action::NewPage)) {
auto openLink = new QAction(QCoreApplication::translate("HelpViewer",
Constants::TR_OPEN_LINK_AS_NEW_PAGE), menu);
connect(openLink, &QAction::triggered, m_viewer, [this, url] {
m_viewer->newPageRequested(url);
});
menu->insertAction(before, openLink);
}
if (m_viewer->isActionVisible(HelpViewer::Action::ExternalWindow)) {
auto openLink = new QAction(QCoreApplication::translate("HelpViewer",
Constants::TR_OPEN_LINK_IN_WINDOW), menu);
connect(openLink, &QAction::triggered, m_viewer, [this, url] {
m_viewer->externalPageRequested(url);
});
menu->insertAction(before, openLink);
}
}
connect(menu, &QMenu::aboutToHide, menu, &QObject::deleteLater);
menu->popup(event->globalPos());
}