/* ============================================================ * QupZilla - WebKit based browser * Copyright (C) 2010-2012 David Rosca * * 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 "tabwidget.h" #include "tabbar.h" #include "tabbedwebview.h" #include "webpage.h" #include "qupzilla.h" #include "iconprovider.h" #include "mainapplication.h" #include "webtab.h" #include "clickablelabel.h" #include "closedtabsmanager.h" #include "progressbar.h" #include "navigationbar.h" #include "locationbar.h" #include "websearchbar.h" #include "settings.h" #include #include #include #include #include #include AddTabButton::AddTabButton(TabWidget* tabWidget, TabBar* tabBar) : ToolButton(tabWidget) , m_tabBar(tabBar) , m_tabWidget(tabWidget) { setObjectName("tabwidget-button-addtab"); setAutoRaise(true); setFocusPolicy(Qt::NoFocus); setAcceptDrops(true); setToolTip(TabWidget::tr("New Tab")); } void AddTabButton::dragEnterEvent(QDragEnterEvent* event) { const QMimeData* mime = event->mimeData(); if (mime->hasUrls()) { event->acceptProposedAction(); return; } ToolButton::dragEnterEvent(event); } void AddTabButton::dropEvent(QDropEvent* event) { const QMimeData* mime = event->mimeData(); if (!mime->hasUrls()) { ToolButton::dropEvent(event); return; } foreach(const QUrl & url, mime->urls()) { m_tabWidget->addView(url, Qz::NT_SelectedTabAtTheEnd); } } TabWidget::TabWidget(QupZilla* mainClass, QWidget* parent) : QTabWidget(parent) , p_QupZilla(mainClass) , m_lastTabIndex(0) , m_lastBackgroundTabIndex(-1) , m_isClosingToLastTabIndex(false) , m_closedTabsManager(new ClosedTabsManager) , m_locationBars(new QStackedWidget) { setObjectName("tabwidget"); m_tabBar = new TabBar(p_QupZilla, this); setTabBar(m_tabBar); connect(this, SIGNAL(currentChanged(int)), this, SLOT(currentTabChanged(int))); connect(this, SIGNAL(currentChanged(int)), p_QupZilla, SLOT(refreshHistory())); 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))); connect(m_tabBar, SIGNAL(moveAddTabButton(int)), this, SLOT(moveAddTabButton(int))); connect(m_tabBar, SIGNAL(showButtons()), this, SLOT(showButtons())); connect(m_tabBar, SIGNAL(hideButtons()), this, SLOT(hideButtons())); m_buttonListTabs = new ToolButton(this); m_buttonListTabs->setObjectName("tabwidget-button-opentabs"); m_menuTabs = new QMenu(this); m_buttonListTabs->setMenu(m_menuTabs); m_buttonListTabs->setPopupMode(QToolButton::InstantPopup); m_buttonListTabs->setToolTip(tr("List of tabs")); m_buttonListTabs->setAutoRaise(true); m_buttonListTabs->setFocusPolicy(Qt::NoFocus); m_buttonAddTab = new AddTabButton(this, m_tabBar); connect(m_buttonAddTab, SIGNAL(clicked()), p_QupZilla, SLOT(addTab())); connect(m_menuTabs, SIGNAL(aboutToShow()), this, SLOT(aboutToShowClosedTabsMenu())); loadSettings(); } void TabWidget::loadSettings() { Settings settings; settings.beginGroup("Browser-Tabs-Settings"); m_hideTabBarWithOneTab = settings.value("hideTabsWithOneTab", false).toBool(); m_dontQuitWithOneTab = settings.value("dontQuitWithOneTab", false).toBool(); m_closedInsteadOpened = settings.value("closedInsteadOpenedTabs", false).toBool(); m_newTabAfterActive = settings.value("newTabAfterActive", true).toBool(); settings.endGroup(); settings.beginGroup("Web-URL-Settings"); m_urlOnNewTab = settings.value("newTabUrl", "qupzilla:speeddial").toUrl(); settings.endGroup(); m_tabBar->loadSettings(); } void TabWidget::resizeEvent(QResizeEvent* e) { QPoint posit; posit.setY(0); posit.setX(width() - m_buttonListTabs->width()); m_buttonListTabs->move(posit); QTabWidget::resizeEvent(e); } WebTab* TabWidget::weTab() { return weTab(currentIndex()); } WebTab* TabWidget::weTab(int index) { return qobject_cast(widget(index)); } void TabWidget::createKeyPressEvent(QKeyEvent* event) { QTabWidget::keyPressEvent(event); } void TabWidget::showButtons() { m_buttonListTabs->show(); m_buttonAddTab->show(); } void TabWidget::hideButtons() { m_buttonListTabs->hide(); m_buttonAddTab->hide(); } void TabWidget::moveAddTabButton(int posX) { int posY = (m_tabBar->height() - m_buttonAddTab->height()) / 2; m_buttonAddTab->move(posX, posY); } void TabWidget::aboutToShowTabsMenu() { m_menuTabs->clear(); WebTab* actTab = weTab(); if (!actTab) { return; } for (int i = 0; i < count(); i++) { WebTab* tab = weTab(i); if (!tab) { continue; } QAction* action = new QAction(this); if (tab == actTab) { action->setIcon(QIcon(":/icons/menu/dot.png")); } else { action->setIcon(tab->icon()); } if (tab->title().isEmpty()) { if (tab->isLoading()) { action->setText(tr("Loading...")); action->setIcon(QIcon(":/icons/other/progress.gif")); } else { action->setText(tr("No Named Page")); } } else { QString title = tab->title(); title.replace("&", "&&"); 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("Currently 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(const QUrl &url, const Qz::NewTabPositionFlags &openFlags, bool selectLine) { return addView(QNetworkRequest(url), openFlags, selectLine); } int TabWidget::addView(const QNetworkRequest &req, const Qz::NewTabPositionFlags &openFlags, bool selectLine) { return addView(req, tr("New tab"), openFlags, selectLine); } int TabWidget::addView(const QUrl &url, const QString &title, const Qz::NewTabPositionFlags &openFlags, bool selectLine, int position) { return addView(QNetworkRequest(url), title, openFlags, selectLine, position); } int TabWidget::addView(QNetworkRequest req, const QString &title, const Qz::NewTabPositionFlags &openFlags, bool selectLine, int position) { QUrl url = req.url(); m_lastTabIndex = currentIndex(); if (url.isEmpty() && !(openFlags & Qz::NT_CleanTab)) { url = m_urlOnNewTab; } if (position == -1 && m_newTabAfterActive && !(openFlags & Qz::NT_TabAtTheEnd)) { // If we are opening newBgTab from pinned tab, make sure it won't be // opened between other pinned tabs if (openFlags &Qz::NT_NotSelectedTab && m_lastBackgroundTabIndex != -1) { position = m_lastBackgroundTabIndex + 1; } else { position = qMax(currentIndex() + 1, m_tabBar->pinnedTabsCount()); } } LocationBar* locBar = new LocationBar(p_QupZilla); m_locationBars->addWidget(locBar); int index; if (position == -1) { index = addTab(new WebTab(p_QupZilla, locBar), ""); } else { index = insertTab(position, new WebTab(p_QupZilla, locBar), ""); } TabbedWebView* webView = weTab(index)->view(); locBar->setWebView(webView); setTabText(index, title); webView->animationLoading(index, true)->movie()->stop(); webView->animationLoading(index, false)->setPixmap(_iconForUrl(url).pixmap(16, 16)); if (openFlags & Qz::NT_SelectedTab) { setCurrentIndex(index); } else { m_lastBackgroundTabIndex = index; } if (count() == 1 && m_hideTabBarWithOneTab) { tabBar()->setVisible(false); } else { tabBar()->setVisible(true); } connect(webView, SIGNAL(wantsCloseTab(int)), this, SLOT(closeTab(int))); connect(webView, SIGNAL(changed()), mApp, SLOT(setStateChanged())); connect(webView, SIGNAL(ipChanged(QString)), p_QupZilla->ipLabel(), SLOT(setText(QString))); if (url.isValid()) { req.setUrl(url); webView->load(req); } if (selectLine) { p_QupZilla->locationBar()->setFocus(); } if (openFlags & Qz::NT_SelectedTab) { m_isClosingToLastTabIndex = true; } return index; } void TabWidget::closeTab(int index) { if (index == -1) { index = currentIndex(); } WebTab* webTab = weTab(index); if (!webTab) { return; } TabbedWebView* webView = webTab->view(); WebPage* webPage = webView->webPage(); if (count() == 1) { if (m_dontQuitWithOneTab) { webView->load(m_urlOnNewTab); return; } else { p_QupZilla->close(); return; } } if (webTab->isPinned()) { emit pinnedTabClosed(); } m_locationBars->removeWidget(webView->webTab()->locationBar()); disconnect(webView, SIGNAL(wantsCloseTab(int)), this, SLOT(closeTab(int))); disconnect(webView, SIGNAL(changed()), mApp, SLOT(setStateChanged())); disconnect(webView, SIGNAL(ipChanged(QString)), p_QupZilla->ipLabel(), SLOT(setText(QString))); //Save last tab url and history m_closedTabsManager->saveView(webView, index); if (m_isClosingToLastTabIndex && m_lastTabIndex < count() && index == currentIndex()) { setCurrentIndex(m_lastTabIndex); } if (count() == 2 && m_hideTabBarWithOneTab) { tabBar()->setVisible(false); } m_lastBackgroundTabIndex = -1; webPage->disconnectObjects(); webView->disconnectObjects(); webTab->disconnectObjects(); webTab->deleteLater(); } void TabWidget::currentTabChanged(int index) { if (index < 0) { return; } m_isClosingToLastTabIndex = false; m_lastBackgroundTabIndex = -1; WebTab* webTab = weTab(index); LocationBar* locBar = webTab->locationBar(); if (m_locationBars->indexOf(locBar) != -1) { m_locationBars->setCurrentWidget(locBar); } webTab->setCurrentTab(); p_QupZilla->currentTabChanged(); m_tabBar->updateCloseButton(index); } void TabWidget::tabMoved(int before, int after) { Q_UNUSED(before) Q_UNUSED(after) m_isClosingToLastTabIndex = false; m_lastBackgroundTabIndex = -1; } void TabWidget::setTabText(int index, const QString &text) { QString newtext = text; newtext.replace("&", "&&"); // Avoid Alt+letter shortcuts if (WebTab* webTab = weTab(index)) { if (webTab->isPinned()) { newtext = ""; } } QTabWidget::setTabText(index, newtext); } void TabWidget::reloadTab(int index) { weTab(index)->reload(); } void TabWidget::showTabBar() { if (count() == 1 && m_hideTabBarWithOneTab) { tabBar()->hide(); } else { tabBar()->show(); } } void TabWidget::reloadAllTabs() { for (int i = 0; i < count(); i++) { reloadTab(i); } } void TabWidget::stopTab(int index) { weTab(index)->stop(); } void TabWidget::closeAllButCurrent(int index) { WebTab* akt = weTab(index); foreach(WebTab * tab, allTabs(false)) { int tabIndex = tab->tabIndex(); if (akt == widget(tabIndex)) { continue; } closeTab(tabIndex); } } int TabWidget::duplicateTab(int index) { WebView* view = weTab(index)->view(); const QUrl &url = view->url(); QByteArray history; QDataStream tabHistoryStream(&history, QIODevice::WriteOnly); tabHistoryStream << *view->history(); QNetworkRequest req(url); req.setRawHeader("Referer", url.toEncoded()); int id = addView(req, tabText(index), Qz::NT_CleanNotSelectedTab); QDataStream historyStream(history); historyStream >> *view->history(); return id; } 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, Qz::NT_CleanSelectedTab, false, tab.position); WebTab* webTab = weTab(index); webTab->p_restoreTab(tab.url, tab.history); } void TabWidget::restoreAllClosedTabs() { if (!m_closedTabsManager->isClosedTabAvailable()) { return; } const QList &closedTabs = m_closedTabsManager->allClosedTabs(); foreach(const ClosedTabsManager::Tab & tab, closedTabs) { int index = addView(QUrl(), tab.title, Qz::NT_CleanSelectedTab); WebTab* webTab = weTab(index); webTab->p_restoreTab(tab.url, tab.history); } m_closedTabsManager->clearList(); } void TabWidget::clearClosedTabsList() { m_closedTabsManager->clearList(); } bool TabWidget::canRestoreTab() { return m_closedTabsManager->isClosedTabAvailable(); } void TabWidget::aboutToShowClosedTabsMenu() { if (!m_closedInsteadOpened) { aboutToShowTabsMenu(); } else { m_menuTabs->clear(); int i = 0; foreach(const ClosedTabsManager::Tab & tab, this->closedTabsManager()->allClosedTabs()) { QString title = tab.title; if (title.length() > 40) { title.truncate(40); title += ".."; } m_menuTabs->addAction(_iconForUrl(tab.url), title, this, SLOT(restoreClosedTab()))->setData(i); i++; } m_menuTabs->addSeparator(); if (i == 0) { m_menuTabs->addAction(tr("Empty"))->setEnabled(false); } else { m_menuTabs->addAction(tr("Restore All Closed Tabs"), this, SLOT(restoreAllClosedTabs())); m_menuTabs->addAction(tr("Clear list"), this, SLOT(clearClosedTabsList())); } } } QList TabWidget::allTabs(bool withPinned) { QList allTabs; for (int i = 0; i < count(); i++) { WebTab* tab = weTab(i); if (!tab || (!withPinned && tab->isPinned())) { continue; } allTabs.append(tab); } return allTabs; } static const qint32 TabWidgetFileVersion = 0xaaf; void TabWidget::savePinnedTabs() { QByteArray data; QDataStream stream(&data, QIODevice::WriteOnly); QStringList tabs; QList tabsHistory; for (int i = 0; i < count(); ++i) { if (WebTab* tab = weTab(i)) { if (!tab->isPinned()) { continue; } tabs.append(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->getActiveProfilPath() + "pinnedtabs.dat"); file.open(QIODevice::WriteOnly); file.write(data); file.close(); } void TabWidget::restorePinnedTabs() { QFile file(mApp->getActiveProfilPath() + "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(), Qz::NT_CleanSelectedTab); weTab(i)->p_restoreTab(url, historyState); } else { addedIndex = addView(url); } WebTab* webTab = weTab(addedIndex); if (webTab) { webTab->setPinned(true); emit pinnedTabAdded(); } m_tabBar->updateCloseButton(addedIndex); m_tabBar->moveTab(addedIndex, i); } } QByteArray TabWidget::saveState() { QList tabList; for (int i = 0; i < count(); ++i) { WebTab* webTab = weTab(i); if (!webTab || webTab->isPinned()) { continue; } WebTab::SavedTab tab(webTab); tabList.append(tab); } QByteArray data; QDataStream stream(&data, QIODevice::WriteOnly); stream << TabWidgetFileVersion; stream << tabList.count(); foreach(const WebTab::SavedTab & tab, tabList) { stream << tab; } stream << currentIndex(); return data; } bool TabWidget::restoreState(QByteArray &state) { QDataStream stream(&state, QIODevice::ReadOnly); if (stream.atEnd()) { return false; } qint32 fileVersion = 0; int tabListCount = 0; int currentTab = 0; stream >> fileVersion; stream >> tabListCount; if (fileVersion != TabWidgetFileVersion) { return false; } for (int i = 0; i < tabListCount; ++i) { WebTab::SavedTab tab; stream >> tab; int index = addView(QUrl(), Qz::NT_CleanSelectedTab); weTab(index)->restoreTab(tab); } stream >> currentTab; setCurrentIndex(currentTab); currentTabChanged(currentTab); return true; } void TabWidget::disconnectObjects() { disconnect(this); disconnect(mApp); disconnect(p_QupZilla); disconnect(p_QupZilla->ipLabel()); } TabWidget::~TabWidget() { delete m_closedTabsManager; }