1
mirror of https://invent.kde.org/network/falkon.git synced 2024-09-23 02:32:10 +02:00
falkonOfficial/src/webview/tabwidget.cpp

505 lines
16 KiB
C++

/* ============================================================
* QupZilla - WebKit based browser
* Copyright (C) 2010-2011 nowrep
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
* ============================================================ */
#include "webview.h"
#include "webpage.h"
#include "qupzilla.h"
#include "tabwidget.h"
#include "tabbar.h"
#include "locationbar.h"
#include "mainapplication.h"
#include "webtab.h"
#include "clickablelabel.h"
class NewTabButton : public QToolButton
{
public:
explicit NewTabButton(QWidget* parent ) : QToolButton(parent)
{
}
QSize sizeHint() const
{
QSize siz = QToolButton::sizeHint();
#ifdef Q_WS_X11
siz.setWidth(25);
#else
siz.setWidth(26);
#endif
return siz;
}
private:
void paintEvent(QPaintEvent*)
{
QPainter p(this);
QStyleOptionTabV3 opt;
opt.init(this);
style()->drawControl(QStyle::CE_TabBarTab, &opt, &p, this);
QPixmap pix(":/icons/other/list-add.png");
QRect r = this->rect();
r.setHeight(r.height()+3);
#ifdef Q_WS_X11
r.setWidth(r.width()-1);
#else
r.setWidth(r.width()+3);
#endif
style()->drawItemPixmap(&p, r, Qt::AlignCenter, pix);
}
};
class TabListButton : public QToolButton
{
public:
explicit TabListButton(QWidget* parent ) : QToolButton(parent)
{
}
QSize sizeHint() const
{
QSize siz = QToolButton::sizeHint();
siz.setWidth(20);
return siz;
}
private:
void paintEvent(QPaintEvent*)
{
QPainter p(this);
QStyleOptionToolButton opt;
opt.init(this);
if (isDown())
opt.state |= QStyle::State_On;
if (opt.state & QStyle::State_MouseOver)
opt.activeSubControls = QStyle::SC_ToolButton;
if (!isChecked() && !isDown())
opt.state |= QStyle::State_Raised;
opt.state |= QStyle::State_AutoRaise;
style()->drawComplexControl(QStyle::CC_ToolButton, &opt, &p, this);
}
};
TabWidget::TabWidget(QupZilla* mainClass, QWidget* parent) :
QTabWidget(parent)
,p_QupZilla(mainClass)
,m_canRestoreTab(false)
,m_lastTabIndex(0)
,m_lastTabUrl(0)
,m_lastTabHistory(0)
{
m_tabBar = new TabBar(p_QupZilla);
setTabBar(m_tabBar);
setObjectName("tabWidget");
setStyleSheet("QTabBar::tab{ max-width:250px; }");
loadSettings();
connect(this, SIGNAL(currentChanged(int)), this, SLOT(tabChanged(int)));
connect(this, SIGNAL(currentChanged(int)), p_QupZilla, SLOT(refreshHistory()));
connect(this, SIGNAL(currentChanged(int)), p_QupZilla->locationBar(), SLOT(siteIconChanged()));
connect(this, SIGNAL(tabCloseRequested(int)), this, SLOT(closeTab(int)));
connect(m_tabBar, SIGNAL(backTab(int)), this, SLOT(backTab(int)));
connect(m_tabBar, SIGNAL(forwardTab(int)), this, SLOT(forwardTab(int)));
connect(m_tabBar, SIGNAL(reloadTab(int)), this, SLOT(reloadTab(int)));
connect(m_tabBar, SIGNAL(stopTab(int)), this, SLOT(stopTab(int)));
connect(m_tabBar, SIGNAL(closeTab(int)), this, SLOT(closeTab(int)));
connect(m_tabBar, SIGNAL(closeAllButCurrent(int)), this, SLOT(closeAllButCurrent(int)));
connect(m_tabBar, SIGNAL(duplicateTab(int)), this, SLOT(duplicateTab(int)));
m_buttonListTabs = new TabListButton(this);
m_menuTabs = new QMenu();
m_buttonListTabs->setMenu(m_menuTabs);
m_buttonListTabs->setPopupMode(QToolButton::InstantPopup);
m_buttonListTabs->setToolTip(tr("Show list of opened tabs"));
connect(m_menuTabs, SIGNAL(aboutToShow()), this, SLOT(aboutToShowTabsMenu()));
setCornerWidget(m_buttonListTabs);
m_buttonAddTab = new NewTabButton(this);
m_buttonAddTab->setToolTip(tr("Add Tab"));
connect(m_buttonAddTab, SIGNAL(clicked()), p_QupZilla, SLOT(addTab()));
setCornerWidget(m_buttonAddTab, Qt::TopLeftCorner);
}
void TabWidget::loadSettings()
{
QSettings settings(mApp->getActiveProfil()+"settings.ini", QSettings::IniFormat);
settings.beginGroup("Browser-Tabs-Settings");
m_hideCloseButtonWithOneTab = settings.value("hideCloseButtonWithOneTab",false).toBool();
m_hideTabBarWithOneTab = settings.value("hideTabsWithOneTab",false).toBool();
settings.endGroup();
settings.beginGroup("Web-URL-Settings");
m_urlOnNewTab = settings.value("newTabUrl","").toUrl();
settings.endGroup();
m_tabBar->loadSettings();
}
void TabWidget::aboutToShowTabsMenu()
{
m_menuTabs->clear();
WebView* actView = weView();
if (!actView)
return;
for (int i = 0; i<count(); i++) {
WebView* view = weView(i);
if (!view)
continue;
QAction* action = new QAction(this);
if (view == actView)
action->setIcon(QIcon(":/icons/menu/circle.png"));
else
action->setIcon(LocationBar::icon(view->url()));
if (view->title().isEmpty()) {
if (view->isLoading()) {
action->setText(tr("Loading..."));
action->setIcon(QIcon(":/icons/other/progress.gif"));
}else
action->setText(tr("No Named Page"));
}
else{
QString title = view->title();
if (title.length()>40) {
title.truncate(40);
title+="..";
}
action->setText(title);
}
action->setData(i);
connect(action, SIGNAL(triggered()), this, SLOT(actionChangeIndex()));
m_menuTabs->addAction(action);
}
m_menuTabs->addSeparator();
m_menuTabs->addAction(tr("Actually You have %1 opened tabs").arg(count()))->setEnabled(false);
}
void TabWidget::actionChangeIndex()
{
if (QAction* action = qobject_cast<QAction*>(sender())) {
setCurrentIndex(action->data().toInt());
}
}
int TabWidget::addView(QUrl url, QString title, OpenUrlIn openIn, bool selectLine)
{
if (url.isEmpty())
url = m_urlOnNewTab;
int index = addTab(new WebTab(p_QupZilla),"");
setTabText(index, title);
weView(index)->animationLoading(index, true)->movie()->stop();
weView(index)->animationLoading(index, false)->setPixmap(LocationBar::icon(url).pixmap(16,16));
if (openIn == TabWidget::NewSelectedTab) {
setCurrentIndex(index);
p_QupZilla->locationBar()->setText(url.toEncoded());
p_QupZilla->locationBar()->setCursorPosition(0);
}
if (count() == 1 && m_hideTabBarWithOneTab)
tabBar()->setVisible(false);
else tabBar()->setVisible(true);
if (count() == 1 && m_hideCloseButtonWithOneTab)
tabBar()->setTabsClosable(false);
else tabBar()->setTabsClosable(true);
connect(weView(index), SIGNAL(siteIconChanged()), p_QupZilla->locationBar(), SLOT(siteIconChanged()));
connect(weView(index), SIGNAL(showUrl(QUrl)), p_QupZilla->locationBar(), SLOT(showUrl(QUrl)));
connect(weView(index), SIGNAL(wantsCloseTab(int)), this, SLOT(closeTab(int)));
connect(weView(index), SIGNAL(changed()), mApp, SLOT(setChanged()));
connect(weView(index), SIGNAL(ipChanged(QString)), p_QupZilla->ipLabel(), SLOT(setText(QString)));
if (url.isValid())
weView(index)->load(url);
if (selectLine)
p_QupZilla->locationBar()->setFocus();
return index;
}
void TabWidget::setTabText(int index, const QString& text)
{
QString newtext = text + " ";
if (WebTab* webTab = qobject_cast<WebTab*>(p_QupZilla->tabWidget()->widget(index)) ) {
if (webTab->isPinned())
newtext = "";
}
QTabWidget::setTabText(index, newtext);
}
void TabWidget::closeTab(int index)
{
if (count() == 1)
return;
if (index == -1)
index = currentIndex();
else m_lastTabIndex-=1;
if (weView(index)) {
disconnect(weView(index), SIGNAL(siteIconChanged()), p_QupZilla->locationBar(), SLOT(siteIconChanged()));
disconnect(weView(index), SIGNAL(showUrl(QUrl)), p_QupZilla->locationBar(), SLOT(showUrl(QUrl)));
disconnect(weView(index), SIGNAL(wantsCloseTab(int)), this, SLOT(closeTab(int)));
disconnect(weView(index), SIGNAL(changed()), mApp, SLOT(setChanged()));
disconnect(weView(index), SIGNAL(ipChanged(QString)), p_QupZilla->ipLabel(), SLOT(setText(QString)));
//Save last tab url and history
if (!weView(index)->url().isEmpty()) {
m_lastTabUrl = weView(index)->url().toString();
QDataStream tabHistoryStream(&m_lastTabHistory, QIODevice::WriteOnly);
tabHistoryStream << *weView(index)->history();
m_canRestoreTab = true;
}
//weView(index)->page()->~QWebPage();
//weView(index)->~QWebView();
delete weView(index);
removeTab(index);
if (count() == 1 && m_hideCloseButtonWithOneTab)
tabBar()->setTabsClosable(false);
if (count() == 1 && m_hideTabBarWithOneTab)
tabBar()->setVisible(false);
}
// if (count() < 1)
// p_QupZilla->close();
}
void TabWidget::tabChanged(int index)
{
if (index<0)
return;
QString title = p_QupZilla->weView()->title();
if (title.isEmpty())
title = tr("No Named Page");
p_QupZilla->setWindowTitle(title + " - QupZilla");
p_QupZilla->locationBar()->showUrl(weView()->url(),false);
p_QupZilla->ipLabel()->setText(weView()->getIp());
if (p_QupZilla->inspectorDock() && p_QupZilla->inspectorDock()->isVisible())
p_QupZilla->showInspector();
weView()->setFocus();
m_lastTabIndex = index;
m_tabBar->updateCloseButton(index);
}
void TabWidget::reloadAllTabs()
{
for (int i = 0;i<count();i++) {
reloadTab(i);
}
}
void TabWidget::closeAllButCurrent(int index)
{
WebTab* akt = qobject_cast<WebTab*>(widget(index));
foreach (WebTab* tab, allTabs(false)) {
if (akt == widget(tab->view()->tabIndex()))
continue;
closeTab(tab->view()->tabIndex());
}
}
void TabWidget::duplicateTab(int index)
{
QUrl url = weView(index)->url();
QByteArray history;
QDataStream tabHistoryStream(&history, QIODevice::WriteOnly);
tabHistoryStream << *weView(index)->history();
int id = addView(url, tr("New tab"), TabWidget::NewSelectedTab);
QDataStream historyStream(history);
historyStream >> *weView(id)->history();
}
void TabWidget::restoreClosedTab()
{
if (m_lastTabUrl.isEmpty())
return;
int index = addView(QUrl());
QDataStream historyStream(m_lastTabHistory);
historyStream >> *weView(index)->history();
weView(index)->load(m_lastTabUrl);
m_canRestoreTab = false;
}
QList<WebTab*> TabWidget::allTabs(bool withPinned)
{
QList<WebTab*> allTabs;
for (int i = 0; i < count(); i++) {
WebTab* tab = qobject_cast<WebTab*>(widget(i));
if (!tab || (!withPinned && tab->isPinned()) )
continue;
allTabs.append(tab);
}
return allTabs;
}
void TabWidget::savePinnedTabs()
{
QByteArray data;
QDataStream stream(&data, QIODevice::WriteOnly);
QStringList tabs;
QList<QByteArray> tabsHistory;
for (int i = 0; i < count(); ++i) {
if (WebView* tab = weView(i)) {
WebTab* webTab = qobject_cast<WebTab*>(widget(i));
if (!webTab || !webTab->isPinned())
continue;
tabs.append(QString::fromUtf8(tab->url().toEncoded()));
if (tab->history()->count() != 0) {
QByteArray tabHistory;
QDataStream tabHistoryStream(&tabHistory, QIODevice::WriteOnly);
tabHistoryStream << *tab->history();
tabsHistory.append(tabHistory);
} else {
tabsHistory << QByteArray();
}
} else {
tabs.append(QString::null);
tabsHistory.append(QByteArray());
}
}
stream << tabs;
stream << tabsHistory;
QFile file(mApp->getActiveProfil()+"pinnedTabs.dat");
file.open(QIODevice::WriteOnly);
file.write(data);
file.close();
}
void TabWidget::restorePinnedTabs()
{
QFile file(mApp->getActiveProfil()+"pinnedTabs.dat");
file.open(QIODevice::ReadOnly);
QByteArray sd = file.readAll();
file.close();
QDataStream stream(&sd, QIODevice::ReadOnly);
if (stream.atEnd())
return;
QStringList pinnedTabs;
stream >> pinnedTabs;
QList<QByteArray> tabHistory;
stream >> tabHistory;
for (int i = 0; i < pinnedTabs.count(); ++i) {
QUrl url = QUrl::fromEncoded(pinnedTabs.at(i).toUtf8());
QByteArray historyState = tabHistory.value(i);
int addedIndex;
if (!historyState.isEmpty()) {
addedIndex= addView(QUrl());
QDataStream historyStream(historyState);
historyStream >> *weView(addedIndex)->history();
weView(addedIndex)->load(url);
} else {
addedIndex = addView(url);
}
WebTab* webTab = (WebTab*)widget(addedIndex);
if (webTab)
webTab->setPinned(true);
m_tabBar->moveTab(addedIndex, i);
m_tabBar->updateCloseButton(i);
}
}
QByteArray TabWidget::saveState()
{
QByteArray data;
QDataStream stream(&data, QIODevice::WriteOnly);
QStringList tabs;
QList<QByteArray> tabsHistory;
for (int i = 0; i < count(); ++i) {
if (WebView* tab = weView(i)) {
WebTab* webTab = qobject_cast<WebTab*>(widget(i));
if (webTab && webTab->isPinned())
continue;
tabs.append(QString::fromUtf8(tab->url().toEncoded()));
if (tab->history()->count() != 0) {
QByteArray tabHistory;
QDataStream tabHistoryStream(&tabHistory, QIODevice::WriteOnly);
tabHistoryStream << *tab->history();
tabsHistory.append(tabHistory);
} else {
tabsHistory << QByteArray();
}
} else {
tabs.append(QString::null);
tabsHistory.append(QByteArray());
}
}
stream << tabs;
stream << currentIndex();
stream << tabsHistory;
return data;
}
bool TabWidget::restoreState(const QByteArray &state)
{
QByteArray sd = state;
QDataStream stream(&sd, QIODevice::ReadOnly);
if (stream.atEnd())
return false;
QStringList openTabs;
int currentTab;
QList<QByteArray> tabHistory;
stream >> openTabs;
stream >> currentTab;
stream >> tabHistory;
for (int i = 0; i < openTabs.count(); ++i) {
QUrl url = QUrl::fromEncoded(openTabs.at(i).toUtf8());
QByteArray historyState = tabHistory.value(i);
if (!historyState.isEmpty()) {
int index = addView(QUrl());
QDataStream historyStream(historyState);
historyStream >> *weView(index)->history();
weView(index)->load(url);
} else {
addView(url);
}
}
setCurrentIndex(currentTab);
return true;
}
TabWidget::~TabWidget()
{
int index = currentIndex();
closeAllButCurrent(index);
closeTab(index);
delete m_menuTabs;
delete m_buttonAddTab;
delete m_buttonListTabs;
}