/* ============================================================ * 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 . * ============================================================ */ #include "webview.h" #include "webpage.h" #include "qupzilla.h" #include "tabwidget.h" #include "tabbar.h" #include "iconprovider.h" #include "mainapplication.h" #include "webtab.h" #include "clickablelabel.h" #include "closedtabsmanager.h" class NewTabButton : public QToolButton { public: explicit NewTabButton(QWidget* parent ) : QToolButton(parent) { #ifndef Q_WS_WIN setIcon(QIcon::fromTheme("list-add")); setIconSize(QSize(16,16)); setAutoRaise(true); #endif } QSize sizeHint() const { QSize siz = QToolButton::sizeHint(); siz.setWidth(26); return siz; } #ifdef Q_WS_WIN 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); r.setWidth(r.width()+3); style()->drawItemPixmap(&p, r, Qt::AlignCenter, pix); } #endif }; 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_lastTabIndex(0) ,m_isClosingToLastTabIndex(false) ,m_closedTabsManager(new ClosedTabsManager(this)) { m_tabBar = new TabBar(p_QupZilla); setTabBar(m_tabBar); setObjectName("tabWidget"); setStyleSheet("QTabBar::tab{ max-width:250px; }"); 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))); connect(m_tabBar, SIGNAL(tabMoved(int,int)), this, SLOT(tabMoved(int,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); loadSettings(); } 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(); settings.beginGroup("Browser-View-Settings"); bool showAddTab = settings.value("showAddTabButton", true).toBool(); m_buttonAddTab->setVisible(showAddTab); if (showAddTab && !cornerWidget(Qt::TopLeftCorner)) setCornerWidget(m_buttonAddTab, Qt::TopLeftCorner); else if (!showAddTab && cornerWidget(Qt::TopLeftCorner)) setCornerWidget(0, Qt::TopLeftCorner); settings.endGroup(); m_tabBar->loadSettings(); } void TabWidget::aboutToShowTabsMenu() { m_menuTabs->clear(); WebView* actView = weView(); if (!actView) return; for (int i = 0; isetIcon(QIcon(":/icons/menu/circle.png")); else action->setIcon(_iconForUrl(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(sender())) { setCurrentIndex(action->data().toInt()); } } int TabWidget::addView(QUrl url, const QString &title, OpenUrlIn openIn, bool selectLine) { m_lastTabIndex = currentIndex(); 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(_iconForUrl(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(); if (openIn == NewSelectedTab) m_isClosingToLastTabIndex = true; return index; } void TabWidget::setTabText(int index, const QString& text) { QString newtext = text + " "; if (WebTab* webTab = qobject_cast(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(); 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 m_closedTabsManager->saveView(weView(index)); if (m_isClosingToLastTabIndex && m_lastTabIndex < count()) setCurrentIndex(m_lastTabIndex); 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::tabMoved(int before, int after) { Q_UNUSED(before) Q_UNUSED(after) m_isClosingToLastTabIndex = false; } void TabWidget::tabChanged(int index) { if (index < 0) return; m_isClosingToLastTabIndex = false; 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_tabBar->updateCloseButton(index); } void TabWidget::reloadAllTabs() { for (int i = 0; i(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_closedTabsManager->isClosedTabAvailable()) return; ClosedTabsManager::Tab tab; QAction* action = qobject_cast(sender()); if (action && action->data().toInt() != 0) tab = m_closedTabsManager->getTabAt(action->data().toInt()); else tab = m_closedTabsManager->getFirstClosedTab(); int index = addView(QUrl(), tab.title); QDataStream historyStream(tab.history); historyStream >> *weView(index)->history(); weView(index)->load(tab.url); } void TabWidget::restoreAllClosedTabs() { if (!m_closedTabsManager->isClosedTabAvailable()) return; QList closedTabs = m_closedTabsManager->allClosedTabs(); foreach (ClosedTabsManager::Tab tab, closedTabs) { int index = addView(QUrl(), tab.title); QDataStream historyStream(tab.history); historyStream >> *weView(index)->history(); weView(index)->load(tab.url); } m_closedTabsManager->clearList(); } bool TabWidget::canRestoreTab() { return m_closedTabsManager->isClosedTabAvailable(); } QList TabWidget::allTabs(bool withPinned) { QList allTabs; for (int i = 0; i < count(); i++) { WebTab* tab = qobject_cast(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 tabsHistory; for (int i = 0; i < count(); ++i) { if (WebView* tab = weView(i)) { WebTab* webTab = qobject_cast(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 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 tabsHistory; for (int i = 0; i < count(); ++i) { if (WebView* tab = weView(i)) { WebTab* webTab = qobject_cast(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 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; }