browserwindow.cpp Example File
webenginewidgets/simplebrowser/browserwindow.cpp
/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the examples of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:BSD$
** 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.
**
** BSD License Usage
** Alternatively, you may use this file under the terms of the BSD license
** as follows:
**
** "Redistribution and use in source and binary forms, with or without
** modification, are permitted provided that the following conditions are
** met:
** * Redistributions of source code must retain the above copyright
** notice, this list of conditions and the following disclaimer.
** * Redistributions in binary form must reproduce the above copyright
** notice, this list of conditions and the following disclaimer in
** the documentation and/or other materials provided with the
** distribution.
** * Neither the name of The Qt Company Ltd nor the names of its
** contributors may be used to endorse or promote products derived
** from this software without specific prior written permission.
**
**
** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "browser.h"
#include "browserwindow.h"
#include "downloadmanagerwidget.h"
#include "tabwidget.h"
#include "webview.h"
#include <QApplication>
#include <QCloseEvent>
#include <QDesktopWidget>
#include <QEvent>
#include <QFileDialog>
#include <QInputDialog>
#include <QMenuBar>
#include <QMessageBox>
#include <QProgressBar>
#include <QStatusBar>
#include <QToolBar>
#include <QVBoxLayout>
#include <QWebEngineProfile>
BrowserWindow:: BrowserWindow(Browser * browser, QWebEngineProfile * profile, bool forDevTools)
: m_browser(browser)
, m_profile(profile)
, m_tabWidget(new TabWidget(profile, this ))
, m_progressBar(nullptr)
, m_historyBackAction(nullptr)
, m_historyForwardAction(nullptr)
, m_stopAction(nullptr)
, m_reloadAction(nullptr)
, m_stopReloadAction(nullptr)
, m_urlLineEdit(nullptr)
, m_favAction(nullptr)
{
setAttribute(Qt :: WA_DeleteOnClose, true );
setFocusPolicy(Qt :: ClickFocus);
if (! forDevTools) {
m_progressBar = new QProgressBar (this );
QToolBar * toolbar = createToolBar();
addToolBar(toolbar);
menuBar()- > addMenu(createFileMenu(m_tabWidget));
menuBar()- > addMenu(createEditMenu());
menuBar()- > addMenu(createViewMenu(toolbar));
menuBar()- > addMenu(createWindowMenu(m_tabWidget));
menuBar()- > addMenu(createHelpMenu());
}
QWidget * centralWidget = new QWidget (this );
QVBoxLayout * layout = new QVBoxLayout ;
layout- > setSpacing(0 );
layout- > setMargin(0 );
if (! forDevTools) {
addToolBarBreak();
m_progressBar- > setMaximumHeight(1 );
m_progressBar- > setTextVisible(false );
m_progressBar- > setStyleSheet(QStringLiteral ("QProgressBar {border: 0px} QProgressBar::chunk {background-color: #da4453}" ));
layout- > addWidget(m_progressBar);
}
layout- > addWidget(m_tabWidget);
centralWidget- > setLayout(layout);
setCentralWidget(centralWidget);
connect(m_tabWidget, & TabWidget:: titleChanged, this , & BrowserWindow:: handleWebViewTitleChanged);
if (! forDevTools) {
connect(m_tabWidget, & TabWidget:: linkHovered, [ this ] (const QString & url) {
statusBar()- > showMessage(url);
});
connect(m_tabWidget, & TabWidget:: loadProgress, this , & BrowserWindow:: handleWebViewLoadProgress);
connect(m_tabWidget, & TabWidget:: webActionEnabledChanged, this , & BrowserWindow:: handleWebActionEnabledChanged);
connect(m_tabWidget, & TabWidget:: urlChanged, [ this ] (const QUrl & url) {
m_urlLineEdit- > setText(url. toDisplayString());
});
connect(m_tabWidget, & TabWidget:: favIconChanged, m_favAction, & QAction :: setIcon);
connect(m_tabWidget, & TabWidget:: devToolsRequested, this , & BrowserWindow:: handleDevToolsRequested);
connect(m_urlLineEdit, & QLineEdit :: returnPressed, [ this ] () {
m_tabWidget- > setUrl(QUrl :: fromUserInput(m_urlLineEdit- > text()));
});
QAction * focusUrlLineEditAction = new QAction (this );
addAction(focusUrlLineEditAction);
focusUrlLineEditAction- > setShortcut(QKeySequence (Qt :: CTRL | Qt :: Key_L));
connect(focusUrlLineEditAction, & QAction :: triggered, this , [ this ] () {
m_urlLineEdit- > setFocus(Qt :: ShortcutFocusReason);
});
}
handleWebViewTitleChanged(QString ());
m_tabWidget- > createTab();
}
QSize BrowserWindow:: sizeHint() const
{
QRect desktopRect = QApplication :: desktop()- > screenGeometry();
QSize size = desktopRect. size() * qreal (0.9 );
return size;
}
QMenu * BrowserWindow:: createFileMenu(TabWidget * tabWidget)
{
QMenu * fileMenu = new QMenu (tr("&File" ));
fileMenu- > addAction(tr("&New Window" ), this , & BrowserWindow:: handleNewWindowTriggered, QKeySequence :: New);
fileMenu- > addAction(tr("New &Incognito Window" ), this , & BrowserWindow:: handleNewIncognitoWindowTriggered);
QAction * newTabAction = new QAction (tr("New &Tab" ), this );
newTabAction- > setShortcuts(QKeySequence :: AddTab);
connect(newTabAction, & QAction :: triggered, this , [ this ] () {
m_tabWidget- > createTab();
m_urlLineEdit- > setFocus();
});
fileMenu- > addAction(newTabAction);
fileMenu- > addAction(tr("&Open File..." ), this , & BrowserWindow:: handleFileOpenTriggered, QKeySequence :: Open);
fileMenu- > addSeparator();
QAction * closeTabAction = new QAction (tr("&Close Tab" ), this );
closeTabAction- > setShortcuts(QKeySequence :: Close);
connect(closeTabAction, & QAction :: triggered, [ tabWidget] () {
tabWidget- > closeTab(tabWidget- > currentIndex());
});
fileMenu- > addAction(closeTabAction);
QAction * closeAction = new QAction (tr("&Quit" ), this );
closeAction- > setShortcut(QKeySequence (Qt :: CTRL | Qt :: Key_Q));
connect(closeAction, & QAction :: triggered, this , & QWidget :: close);
fileMenu- > addAction(closeAction);
connect(fileMenu, & QMenu :: aboutToShow, [ this , closeAction] () {
if (m_browser- > windows(). count() = = 1 )
closeAction- > setText(tr("&Quit" ));
else
closeAction- > setText(tr("&Close Window" ));
});
return fileMenu;
}
QMenu * BrowserWindow:: createEditMenu()
{
QMenu * editMenu = new QMenu (tr("&Edit" ));
QAction * findAction = editMenu- > addAction(tr("&Find" ));
findAction- > setShortcuts(QKeySequence :: Find);
connect(findAction, & QAction :: triggered, this , & BrowserWindow:: handleFindActionTriggered);
QAction * findNextAction = editMenu- > addAction(tr("Find &Next" ));
findNextAction- > setShortcut(QKeySequence :: FindNext);
connect(findNextAction, & QAction :: triggered, [ this ] () {
if (! currentTab() | | m_lastSearch. isEmpty())
return ;
currentTab()- > findText(m_lastSearch);
});
QAction * findPreviousAction = editMenu- > addAction(tr("Find &Previous" ));
findPreviousAction- > setShortcut(QKeySequence :: FindPrevious);
connect(findPreviousAction, & QAction :: triggered, [ this ] () {
if (! currentTab() | | m_lastSearch. isEmpty())
return ;
currentTab()- > findText(m_lastSearch, QWebEnginePage :: FindBackward);
});
return editMenu;
}
QMenu * BrowserWindow:: createViewMenu(QToolBar * toolbar)
{
QMenu * viewMenu = new QMenu (tr("&View" ));
m_stopAction = viewMenu- > addAction(tr("&Stop" ));
QList < QKeySequence > shortcuts;
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_Period));
shortcuts. append(Qt :: Key_Escape);
m_stopAction- > setShortcuts(shortcuts);
connect(m_stopAction, & QAction :: triggered, [ this ] () {
m_tabWidget- > triggerWebPageAction(QWebEnginePage :: Stop);
});
m_reloadAction = viewMenu- > addAction(tr("Reload Page" ));
m_reloadAction- > setShortcuts(QKeySequence :: Refresh);
connect(m_reloadAction, & QAction :: triggered, [ this ] () {
m_tabWidget- > triggerWebPageAction(QWebEnginePage :: Reload);
});
QAction * zoomIn = viewMenu- > addAction(tr("Zoom &In" ));
zoomIn- > setShortcut(QKeySequence (Qt :: CTRL | Qt :: Key_Plus));
connect(zoomIn, & QAction :: triggered, [ this ] () {
if (currentTab())
currentTab()- > setZoomFactor(currentTab()- > zoomFactor() + 0.1 );
});
QAction * zoomOut = viewMenu- > addAction(tr("Zoom &Out" ));
zoomOut- > setShortcut(QKeySequence (Qt :: CTRL | Qt :: Key_Minus));
connect(zoomOut, & QAction :: triggered, [ this ] () {
if (currentTab())
currentTab()- > setZoomFactor(currentTab()- > zoomFactor() - 0.1 );
});
QAction * resetZoom = viewMenu- > addAction(tr("Reset &Zoom" ));
resetZoom- > setShortcut(QKeySequence (Qt :: CTRL | Qt :: Key_0));
connect(resetZoom, & QAction :: triggered, [ this ] () {
if (currentTab())
currentTab()- > setZoomFactor(1.0 );
});
viewMenu- > addSeparator();
QAction * viewToolbarAction = new QAction (tr("Hide Toolbar" ), this );
viewToolbarAction- > setShortcut(tr("Ctrl+|" ));
connect(viewToolbarAction, & QAction :: triggered, [ toolbar, viewToolbarAction] () {
if (toolbar- > isVisible()) {
viewToolbarAction- > setText(tr("Show Toolbar" ));
toolbar- > close();
} else {
viewToolbarAction- > setText(tr("Hide Toolbar" ));
toolbar- > show();
}
});
viewMenu- > addAction(viewToolbarAction);
QAction * viewStatusbarAction = new QAction (tr("Hide Status Bar" ), this );
viewStatusbarAction- > setShortcut(tr("Ctrl+/" ));
connect(viewStatusbarAction, & QAction :: triggered, [ this , viewStatusbarAction] () {
if (statusBar()- > isVisible()) {
viewStatusbarAction- > setText(tr("Show Status Bar" ));
statusBar()- > close();
} else {
viewStatusbarAction- > setText(tr("Hide Status Bar" ));
statusBar()- > show();
}
});
viewMenu- > addAction(viewStatusbarAction);
return viewMenu;
}
QMenu * BrowserWindow:: createWindowMenu(TabWidget * tabWidget)
{
QMenu * menu = new QMenu (tr("&Window" ));
QAction * nextTabAction = new QAction (tr("Show Next Tab" ), this );
QList < QKeySequence > shortcuts;
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_BraceRight));
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_PageDown));
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_BracketRight));
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_Less));
nextTabAction- > setShortcuts(shortcuts);
connect(nextTabAction, & QAction :: triggered, tabWidget, & TabWidget:: nextTab);
QAction * previousTabAction = new QAction (tr("Show Previous Tab" ), this );
shortcuts. clear();
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_BraceLeft));
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_PageUp));
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_BracketLeft));
shortcuts. append(QKeySequence (Qt :: CTRL | Qt :: Key_Greater));
previousTabAction- > setShortcuts(shortcuts);
connect(previousTabAction, & QAction :: triggered, tabWidget, & TabWidget:: previousTab);
connect(menu, & QMenu :: aboutToShow, [ this , menu, nextTabAction, previousTabAction] () {
menu- > clear();
menu- > addAction(nextTabAction);
menu- > addAction(previousTabAction);
menu- > addSeparator();
QVector < BrowserWindow* > windows = m_browser- > windows();
int index(- 1 );
for (auto window : windows) {
QAction * action = menu- > addAction(window- > windowTitle(), this , & BrowserWindow:: handleShowWindowTriggered);
action- > setData(+ + index);
action- > setCheckable(true );
if (window = = this )
action- > setChecked(true );
}
});
return menu;
}
QMenu * BrowserWindow:: createHelpMenu()
{
QMenu * helpMenu = new QMenu (tr("&Help" ));
helpMenu- > addAction(tr("About &Qt" ), qApp , QApplication :: aboutQt);
return helpMenu;
}
QToolBar * BrowserWindow:: createToolBar()
{
QToolBar * navigationBar = new QToolBar (tr("Navigation" ));
navigationBar- > setMovable(false );
navigationBar- > toggleViewAction()- > setEnabled(false );
m_historyBackAction = new QAction (this );
QList < QKeySequence > backShortcuts = QKeySequence :: keyBindings(QKeySequence :: Back);
for (auto it = backShortcuts. begin(); it ! = backShortcuts. end();) {
// Chromium already handles navigate on backspace when appropriate.
if ((* it)[ 0 ] = = Qt :: Key_Backspace)
it = backShortcuts. erase(it);
else
+ + it;
}
// For some reason Qt doesn't bind the dedicated Back key to Back.
backShortcuts. append(QKeySequence (Qt :: Key_Back));
m_historyBackAction- > setShortcuts(backShortcuts);
m_historyBackAction- > setIconVisibleInMenu(false );
m_historyBackAction- > setIcon(QIcon (QStringLiteral (":go-previous.png" )));
m_historyBackAction- > setToolTip(tr("Go back in history" ));
connect(m_historyBackAction, & QAction :: triggered, [ this ] () {
m_tabWidget- > triggerWebPageAction(QWebEnginePage :: Back);
});
navigationBar- > addAction(m_historyBackAction);
m_historyForwardAction = new QAction (this );
QList < QKeySequence > fwdShortcuts = QKeySequence :: keyBindings(QKeySequence :: Forward);
for (auto it = fwdShortcuts. begin(); it ! = fwdShortcuts. end();) {
if (((* it)[ 0 ] & Qt :: Key_unknown) = = Qt :: Key_Backspace)
it = fwdShortcuts. erase(it);
else
+ + it;
}
fwdShortcuts. append(QKeySequence (Qt :: Key_Forward));
m_historyForwardAction- > setShortcuts(fwdShortcuts);
m_historyForwardAction- > setIconVisibleInMenu(false );
m_historyForwardAction- > setIcon(QIcon (QStringLiteral (":go-next.png" )));
m_historyForwardAction- > setToolTip(tr("Go forward in history" ));
connect(m_historyForwardAction, & QAction :: triggered, [ this ] () {
m_tabWidget- > triggerWebPageAction(QWebEnginePage :: Forward);
});
navigationBar- > addAction(m_historyForwardAction);
m_stopReloadAction = new QAction (this );
connect(m_stopReloadAction, & QAction :: triggered, [ this ] () {
m_tabWidget- > triggerWebPageAction(QWebEnginePage :: WebAction(m_stopReloadAction- > data(). toInt()));
});
navigationBar- > addAction(m_stopReloadAction);
m_urlLineEdit = new QLineEdit (this );
m_favAction = new QAction (this );
m_urlLineEdit- > addAction(m_favAction, QLineEdit :: LeadingPosition);
m_urlLineEdit- > setClearButtonEnabled(true );
navigationBar- > addWidget(m_urlLineEdit);
auto downloadsAction = new QAction (this );
downloadsAction- > setIcon(QIcon (QStringLiteral (":go-bottom.png" )));
downloadsAction- > setToolTip(tr("Show downloads" ));
navigationBar- > addAction(downloadsAction);
connect(downloadsAction, & QAction :: triggered, [ this ] () {
m_browser- > downloadManagerWidget(). show();
});
return navigationBar;
}
void BrowserWindow:: handleWebActionEnabledChanged(QWebEnginePage :: WebAction action, bool enabled)
{
switch (action) {
case QWebEnginePage :: Back:
m_historyBackAction- > setEnabled(enabled);
break ;
case QWebEnginePage :: Forward:
m_historyForwardAction- > setEnabled(enabled);
break ;
case QWebEnginePage :: Reload:
m_reloadAction- > setEnabled(enabled);
break ;
case QWebEnginePage :: Stop:
m_stopAction- > setEnabled(enabled);
break ;
default :
qWarning ("Unhandled webActionChanged signal" );
}
}
void BrowserWindow:: handleWebViewTitleChanged(const QString & title)
{
QString suffix = m_profile- > isOffTheRecord()
? tr("Qt Simple Browser (Incognito)" )
: tr("Qt Simple Browser" );
if (title. isEmpty())
setWindowTitle(suffix);
else
setWindowTitle(title + " - " + suffix);
}
void BrowserWindow:: handleNewWindowTriggered()
{
BrowserWindow * window = m_browser- > createWindow();
window- > m_urlLineEdit- > setFocus();
}
void BrowserWindow:: handleNewIncognitoWindowTriggered()
{
BrowserWindow * window = m_browser- > createWindow(/* offTheRecord: */ true );
window- > m_urlLineEdit- > setFocus();
}
void BrowserWindow:: handleFileOpenTriggered()
{
QUrl url = QFileDialog :: getOpenFileUrl(this , tr("Open Web Resource" ), QString (),
tr("Web Resources (*.html *.htm *.svg *.png *.gif *.svgz);;All files (*.*)" ));
if (url. isEmpty())
return ;
currentTab()- > setUrl(url);
}
void BrowserWindow:: handleFindActionTriggered()
{
if (! currentTab())
return ;
bool ok = false ;
QString search = QInputDialog :: getText(this , tr("Find" ),
tr("Find:" ), QLineEdit :: Normal,
m_lastSearch, & ok);
if (ok & & ! search. isEmpty()) {
m_lastSearch = search;
currentTab()- > findText(m_lastSearch, 0 , [ this ] (bool found) {
if (! found)
statusBar()- > showMessage(tr("\"%1\" not found." ). arg(m_lastSearch));
});
}
}
void BrowserWindow:: closeEvent(QCloseEvent * event)
{
if (m_tabWidget- > count() > 1 ) {
int ret = QMessageBox :: warning(this , tr("Confirm close" ),
tr("Are you sure you want to close the window ?\n"
"There are %1 tabs open." ). arg(m_tabWidget- > count()),
QMessageBox :: Yes | QMessageBox :: No, QMessageBox :: No);
if (ret = = QMessageBox :: No) {
event- > ignore();
return ;
}
}
event- > accept();
deleteLater();
}
TabWidget * BrowserWindow:: tabWidget() const
{
return m_tabWidget;
}
WebView * BrowserWindow:: currentTab() const
{
return m_tabWidget- > currentWebView();
}
void BrowserWindow:: handleWebViewLoadProgress(int progress)
{
static QIcon stopIcon(QStringLiteral (":process-stop.png" ));
static QIcon reloadIcon(QStringLiteral (":view-refresh.png" ));
if (0 < progress & & progress < 100 ) {
m_stopReloadAction- > setData(QWebEnginePage :: Stop);
m_stopReloadAction- > setIcon(stopIcon);
m_stopReloadAction- > setToolTip(tr("Stop loading the current page" ));
m_progressBar- > setValue(progress);
} else {
m_stopReloadAction- > setData(QWebEnginePage :: Reload);
m_stopReloadAction- > setIcon(reloadIcon);
m_stopReloadAction- > setToolTip(tr("Reload the current page" ));
m_progressBar- > setValue(0 );
}
}
void BrowserWindow:: handleShowWindowTriggered()
{
if (QAction * action = qobject_cast< QAction * > (sender())) {
int offset = action- > data(). toInt();
QVector < BrowserWindow* > windows = m_browser- > windows();
windows. at(offset)- > activateWindow();
windows. at(offset)- > currentTab()- > setFocus();
}
}
void BrowserWindow:: handleDevToolsRequested(QWebEnginePage * source)
{
source- > setDevToolsPage(m_browser- > createDevToolsWindow()- > currentTab()- > page());
source- > triggerAction(QWebEnginePage :: InspectElement);
}