tabwidget.cpp
上传用户:huahtool
上传日期:2015-12-10
资源大小:1089k
文件大小:27k
源码类别:

浏览器

开发平台:

Visual C++

  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2008 Nokia Corporation and/or its subsidiary(-ies).
  4. ** Contact: Qt Software Information (qt-info@nokia.com)
  5. **
  6. ** This file is part of the demonstration applications of the Qt Toolkit.
  7. **
  8. ** Commercial Usage
  9. ** Licensees holding valid Qt Commercial licenses may use this file in
  10. ** accordance with the Qt Commercial License Agreement provided with the
  11. ** Software or, alternatively, in accordance with the terms contained in
  12. ** a written agreement between you and Nokia.
  13. **
  14. **
  15. ** GNU General Public License Usage
  16. ** Alternatively, this file may be used under the terms of the GNU
  17. ** General Public License versions 2.0 or 3.0 as published by the Free
  18. ** Software Foundation and appearing in the file LICENSE.GPL included in
  19. ** the packaging of this file.  Please review the following information
  20. ** to ensure GNU General Public Licensing requirements will be met:
  21. ** http://www.fsf.org/licensing/licenses/info/GPLv2.html and
  22. ** http://www.gnu.org/copyleft/gpl.html.  In addition, as a special
  23. ** exception, Nokia gives you certain additional rights. These rights
  24. ** are described in the Nokia Qt GPL Exception version 1.3, included in
  25. ** the file GPL_EXCEPTION.txt in this package.
  26. **
  27. ** Qt for Windows(R) Licensees
  28. ** As a special exception, Nokia, as the sole copyright holder for Qt
  29. ** Designer, grants users of the Qt/Eclipse Integration plug-in the
  30. ** right for the Qt/Eclipse Integration to link to functionality
  31. ** provided by Qt Designer and its related libraries.
  32. **
  33. ** If you are unsure which license is appropriate for your use, please
  34. ** contact the sales department at qt-sales@nokia.com.
  35. **
  36. ****************************************************************************/
  37. #include "tabwidget.h"
  38. #include "browserapplication.h"
  39. #include "browsermainwindow.h"
  40. #include "history.h"
  41. #include "urllineedit.h"
  42. #include "webview.h"
  43. #include <QtGui/QClipboard>
  44. #include <QtGui/QCompleter>
  45. #include <QtGui/QListView>
  46. #include <QtGui/QMenu>
  47. #include <QtGui/QMessageBox>
  48. #include <QtGui/QMouseEvent>
  49. #include <QtGui/QStackedWidget>
  50. #include <QtGui/QStyle>
  51. #include <QtGui/QToolButton>
  52. #include <QtCore/QDebug>
  53. TabBar::TabBar(QWidget *parent)
  54.     : QTabBar(parent)
  55. {
  56.     setContextMenuPolicy(Qt::CustomContextMenu);
  57.     setAcceptDrops(true);
  58.     connect(this, SIGNAL(customContextMenuRequested(const QPoint &)),
  59.             this, SLOT(contextMenuRequested(const QPoint &)));
  60.     QString alt = QLatin1String("Alt+%1");
  61.     for (int i = 1; i <= 10; ++i) {
  62.         int key = i;
  63.         if (key == 10)
  64.             key = 0;
  65.         QShortcut *shortCut = new QShortcut(alt.arg(key), this);
  66.         m_tabShortcuts.append(shortCut);
  67.         connect(shortCut, SIGNAL(activated()), this, SLOT(selectTabAction()));
  68.     }
  69. }
  70. void TabBar::selectTabAction()
  71. {
  72.     if (QShortcut *shortCut = qobject_cast<QShortcut*>(sender())) {
  73.         int index = m_tabShortcuts.indexOf(shortCut);
  74.         if (index == 0)
  75.             index = 10;
  76.         setCurrentIndex(index);
  77.     }
  78. }
  79. void TabBar::contextMenuRequested(const QPoint &position)
  80. {
  81.     QMenu menu;
  82.     menu.addAction(tr("New &Tab"), this, SIGNAL(newTab()), QKeySequence::AddTab);
  83.     int index = tabAt(position);
  84.     if (-1 != index) {
  85.         QAction *action = menu.addAction(tr("Clone Tab"),
  86.                 this, SLOT(cloneTab()));
  87.         action->setData(index);
  88.         menu.addSeparator();
  89.         action = menu.addAction(tr("&Close Tab"),
  90.                 this, SLOT(closeTab()), QKeySequence::Close);
  91.         action->setData(index);
  92.         action = menu.addAction(tr("Close &Other Tabs"),
  93.                 this, SLOT(closeOtherTabs()));
  94.         action->setData(index);
  95.         menu.addSeparator();
  96.         action = menu.addAction(tr("Reload Tab"),
  97.                 this, SLOT(reloadTab()), QKeySequence::Refresh);
  98.         action->setData(index);
  99.     } else {
  100.         menu.addSeparator();
  101.     }
  102.     menu.addAction(tr("Reload All Tabs"), this, SIGNAL(reloadAllTabs()));
  103.     menu.exec(QCursor::pos());
  104. }
  105. void TabBar::cloneTab()
  106. {
  107.     if (QAction *action = qobject_cast<QAction*>(sender())) {
  108.         int index = action->data().toInt();
  109.         emit cloneTab(index);
  110.     }
  111. }
  112. void TabBar::closeTab()
  113. {
  114.     if (QAction *action = qobject_cast<QAction*>(sender())) {
  115.         int index = action->data().toInt();
  116.         emit closeTab(index);
  117.     }
  118. }
  119. void TabBar::closeOtherTabs()
  120. {
  121.     if (QAction *action = qobject_cast<QAction*>(sender())) {
  122.         int index = action->data().toInt();
  123.         emit closeOtherTabs(index);
  124.     }
  125. }
  126. void TabBar::mousePressEvent(QMouseEvent *event)
  127. {
  128.     if (event->button() == Qt::LeftButton)
  129.         m_dragStartPos = event->pos();
  130.     QTabBar::mousePressEvent(event);
  131. }
  132. void TabBar::mouseMoveEvent(QMouseEvent *event)
  133. {
  134.     if (event->buttons() == Qt::LeftButton
  135.         && (event->pos() - m_dragStartPos).manhattanLength() > QApplication::startDragDistance()) {
  136.         QDrag *drag = new QDrag(this);
  137.         QMimeData *mimeData = new QMimeData;
  138.         QList<QUrl> urls;
  139.         int index = tabAt(event->pos());
  140.         QUrl url = tabData(index).toUrl();
  141.         urls.append(url);
  142.         mimeData->setUrls(urls);
  143.         mimeData->setText(tabText(index));
  144.         mimeData->setData(QLatin1String("action"), "tab-reordering");
  145.         drag->setMimeData(mimeData);
  146.         drag->exec();
  147.     }
  148.     QTabBar::mouseMoveEvent(event);
  149. }
  150. void TabBar::dragEnterEvent(QDragEnterEvent *event)
  151. {
  152.     const QMimeData *mimeData = event->mimeData();
  153.     QStringList formats = mimeData->formats();
  154.     if (formats.contains(QLatin1String("action"))
  155.         && (mimeData->data(QLatin1String("action")) == "tab-reordering")) {
  156.         event->acceptProposedAction();
  157.     }
  158.     QTabBar::dragEnterEvent(event);
  159. }
  160. void TabBar::dropEvent(QDropEvent *event)
  161. {
  162.     int fromIndex = tabAt(m_dragStartPos);
  163.     int toIndex = tabAt(event->pos());
  164.     if (fromIndex != toIndex) {
  165.         emit tabMoveRequested(fromIndex, toIndex);
  166.         event->acceptProposedAction();
  167.     }
  168.     QTabBar::dropEvent(event);
  169. }
  170. // When index is -1 index chooses the current tab
  171. void TabWidget::reloadTab(int index)
  172. {
  173.     if (index < 0)
  174.         index = currentIndex();
  175.     if (index < 0 || index >= count())
  176.         return;
  177.     QWidget *widget = this->widget(index);
  178.     if (WebView *tab = qobject_cast<WebView*>(widget))
  179.         tab->reload();
  180. }
  181. void TabBar::reloadTab()
  182. {
  183.     if (QAction *action = qobject_cast<QAction*>(sender())) {
  184.         int index = action->data().toInt();
  185.         emit reloadTab(index);
  186.     }
  187. }
  188. TabWidget::TabWidget(QWidget *parent)
  189.     : QTabWidget(parent)
  190.     , m_recentlyClosedTabsAction(0)
  191.     , m_newTabAction(0)
  192.     , m_closeTabAction(0)
  193.     , m_nextTabAction(0)
  194.     , m_previousTabAction(0)
  195.     , m_recentlyClosedTabsMenu(0)
  196.     , m_lineEditCompleter(0)
  197.     , m_lineEdits(0)
  198.     , m_tabBar(new TabBar(this))
  199. {
  200.     setElideMode(Qt::ElideRight);
  201.     connect(m_tabBar, SIGNAL(newTab()), this, SLOT(newTab()));
  202.     connect(m_tabBar, SIGNAL(closeTab(int)), this, SLOT(closeTab(int)));
  203.     connect(m_tabBar, SIGNAL(cloneTab(int)), this, SLOT(cloneTab(int)));
  204.     connect(m_tabBar, SIGNAL(closeOtherTabs(int)), this, SLOT(closeOtherTabs(int)));
  205.     connect(m_tabBar, SIGNAL(reloadTab(int)), this, SLOT(reloadTab(int)));
  206.     connect(m_tabBar, SIGNAL(reloadAllTabs()), this, SLOT(reloadAllTabs()));
  207.     connect(m_tabBar, SIGNAL(tabMoveRequested(int, int)), this, SLOT(moveTab(int, int)));
  208.     setTabBar(m_tabBar);
  209.     // Actions
  210.     m_newTabAction = new QAction(QIcon(QLatin1String(":addtab.png")), tr("New &Tab"), this);
  211.     m_newTabAction->setShortcuts(QKeySequence::AddTab);
  212.     m_newTabAction->setIconVisibleInMenu(false);
  213.     connect(m_newTabAction, SIGNAL(triggered()), this, SLOT(newTab()));
  214.     m_closeTabAction = new QAction(QIcon(QLatin1String(":closetab.png")), tr("&Close Tab"), this);
  215.     m_closeTabAction->setShortcuts(QKeySequence::Close);
  216.     m_closeTabAction->setIconVisibleInMenu(false);
  217.     connect(m_closeTabAction, SIGNAL(triggered()), this, SLOT(closeTab()));
  218.     m_nextTabAction = new QAction(tr("Show Next Tab"), this);
  219.     QList<QKeySequence> shortcuts;
  220.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_BraceRight));
  221.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_PageDown));
  222.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_BracketRight));
  223.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_Less));
  224.     m_nextTabAction->setShortcuts(shortcuts);
  225.     connect(m_nextTabAction, SIGNAL(triggered()), this, SLOT(nextTab()));
  226.     m_previousTabAction = new QAction(tr("Show Previous Tab"), this);
  227.     shortcuts.clear();
  228.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_BraceLeft));
  229.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_PageUp));
  230.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_BracketLeft));
  231.     shortcuts.append(QKeySequence(Qt::CTRL | Qt::Key_Greater));
  232.     m_previousTabAction->setShortcuts(shortcuts);
  233.     connect(m_previousTabAction, SIGNAL(triggered()), this, SLOT(previousTab()));
  234.     m_recentlyClosedTabsMenu = new QMenu(this);
  235.     connect(m_recentlyClosedTabsMenu, SIGNAL(aboutToShow()),
  236.             this, SLOT(aboutToShowRecentTabsMenu()));
  237.     connect(m_recentlyClosedTabsMenu, SIGNAL(triggered(QAction *)),
  238.             this, SLOT(aboutToShowRecentTriggeredAction(QAction *)));
  239.     m_recentlyClosedTabsAction = new QAction(tr("Recently Closed Tabs"), this);
  240.     m_recentlyClosedTabsAction->setMenu(m_recentlyClosedTabsMenu);
  241.     m_recentlyClosedTabsAction->setEnabled(false);
  242.     // corner buttons
  243.     QToolButton *addTabButton = new QToolButton(this);
  244.     addTabButton->setDefaultAction(m_newTabAction);
  245.     addTabButton->setAutoRaise(true);
  246.     addTabButton->setToolButtonStyle(Qt::ToolButtonIconOnly);
  247.     setCornerWidget(addTabButton, Qt::TopLeftCorner);
  248.     QToolButton *closeTabButton = new QToolButton(this);
  249.     closeTabButton->setDefaultAction(m_closeTabAction);
  250.     closeTabButton->setAutoRaise(true);
  251.     closeTabButton->setToolButtonStyle(Qt::ToolButtonIconOnly);
  252.     setCornerWidget(closeTabButton, Qt::TopRightCorner);
  253.     connect(this, SIGNAL(currentChanged(int)),
  254.             this, SLOT(currentChanged(int)));
  255.     m_lineEdits = new QStackedWidget(this);
  256. }
  257. void TabWidget::clear()
  258. {
  259.     // clear the recently closed tabs
  260.     m_recentlyClosedTabs.clear();
  261.     // clear the line edit history
  262.     for (int i = 0; i < m_lineEdits->count(); ++i) {
  263.         QLineEdit *qLineEdit = lineEdit(i);
  264.         qLineEdit->setText(qLineEdit->text());
  265.     }
  266. }
  267. void TabWidget::moveTab(int fromIndex, int toIndex)
  268. {
  269.     disconnect(this, SIGNAL(currentChanged(int)),
  270.         this, SLOT(currentChanged(int)));
  271.     QWidget *tabWidget = widget(fromIndex);
  272.     QIcon icon = tabIcon(fromIndex);
  273.     QString text = tabText(fromIndex);
  274.     QVariant data = m_tabBar->tabData(fromIndex);
  275.     removeTab(fromIndex);
  276.     insertTab(toIndex, tabWidget, icon, text);
  277.     m_tabBar->setTabData(toIndex, data);
  278.     connect(this, SIGNAL(currentChanged(int)),
  279.         this, SLOT(currentChanged(int)));
  280.     setCurrentIndex(toIndex);
  281. }
  282. void TabWidget::addWebAction(QAction *action, QWebPage::WebAction webAction)
  283. {
  284.     if (!action)
  285.         return;
  286.     m_actions.append(new WebActionMapper(action, webAction, this));
  287. }
  288. void TabWidget::currentChanged(int index)
  289. {
  290.     WebView *webView = this->webView(index);
  291.     if (!webView)
  292.         return;
  293.     Q_ASSERT(m_lineEdits->count() == count());
  294.     WebView *oldWebView = this->webView(m_lineEdits->currentIndex());
  295.     if (oldWebView) {
  296.         disconnect(oldWebView, SIGNAL(statusBarMessage(const QString&)),
  297.                 this, SIGNAL(showStatusBarMessage(const QString&)));
  298.         disconnect(oldWebView->page(), SIGNAL(linkHovered(const QString&, const QString&, const QString&)),
  299.                 this, SIGNAL(linkHovered(const QString&)));
  300.         disconnect(oldWebView, SIGNAL(loadProgress(int)),
  301.                 this, SIGNAL(loadProgress(int)));
  302.     }
  303.     connect(webView, SIGNAL(statusBarMessage(const QString&)),
  304.             this, SIGNAL(showStatusBarMessage(const QString&)));
  305.     connect(webView->page(), SIGNAL(linkHovered(const QString&, const QString&, const QString&)),
  306.             this, SIGNAL(linkHovered(const QString&)));
  307.     connect(webView, SIGNAL(loadProgress(int)),
  308.             this, SIGNAL(loadProgress(int)));
  309.     for (int i = 0; i < m_actions.count(); ++i) {
  310.         WebActionMapper *mapper = m_actions[i];
  311.         mapper->updateCurrent(webView->page());
  312.     }
  313.     emit setCurrentTitle(webView->title());
  314.     m_lineEdits->setCurrentIndex(index);
  315.     emit loadProgress(webView->progress());
  316.     emit showStatusBarMessage(webView->lastStatusBarText());
  317.     if (webView->url().isEmpty())
  318.         m_lineEdits->currentWidget()->setFocus();
  319.     else
  320.         webView->setFocus();
  321. }
  322. QAction *TabWidget::newTabAction() const
  323. {
  324.     return m_newTabAction;
  325. }
  326. QAction *TabWidget::closeTabAction() const
  327. {
  328.     return m_closeTabAction;
  329. }
  330. QAction *TabWidget::recentlyClosedTabsAction() const
  331. {
  332.     return m_recentlyClosedTabsAction;
  333. }
  334. QAction *TabWidget::nextTabAction() const
  335. {
  336.     return m_nextTabAction;
  337. }
  338. QAction *TabWidget::previousTabAction() const
  339. {
  340.     return m_previousTabAction;
  341. }
  342. QWidget *TabWidget::lineEditStack() const
  343. {
  344.     return m_lineEdits;
  345. }
  346. QLineEdit *TabWidget::currentLineEdit() const
  347. {
  348.     return lineEdit(m_lineEdits->currentIndex());
  349. }
  350. WebView *TabWidget::currentWebView() const
  351. {
  352.     return webView(currentIndex());
  353. }
  354. QLineEdit *TabWidget::lineEdit(int index) const
  355. {
  356.     UrlLineEdit *urlLineEdit = qobject_cast<UrlLineEdit*>(m_lineEdits->widget(index));
  357.     if (urlLineEdit)
  358.         return urlLineEdit->lineEdit();
  359.     return 0;
  360. }
  361. WebView *TabWidget::webView(int index) const
  362. {
  363.     QWidget *widget = this->widget(index);
  364.     if (WebView *webView = qobject_cast<WebView*>(widget)) {
  365.         return webView;
  366.     } else {
  367.         // optimization to delay creating the first webview
  368.         if (count() == 1) {
  369.             TabWidget *that = const_cast<TabWidget*>(this);
  370.             that->setUpdatesEnabled(false);
  371.             that->newTab();
  372.             that->closeTab(0);
  373.             that->setUpdatesEnabled(true);
  374.             return currentWebView();
  375.         }
  376.     }
  377.     return 0;
  378. }
  379. int TabWidget::webViewIndex(WebView *webView) const
  380. {
  381.     int index = indexOf(webView);
  382.     return index;
  383. }
  384. WebView *TabWidget::newTab(bool makeCurrent)
  385. {
  386.     // line edit
  387.     UrlLineEdit *urlLineEdit = new UrlLineEdit;
  388.     QLineEdit *lineEdit = urlLineEdit->lineEdit();
  389.     if (!m_lineEditCompleter && count() > 0) {
  390.         HistoryCompletionModel *completionModel = new HistoryCompletionModel(this);
  391.         completionModel->setSourceModel(BrowserApplication::historyManager()->historyFilterModel());
  392.         m_lineEditCompleter = new QCompleter(completionModel, this);
  393.         // Should this be in Qt by default?
  394.         QAbstractItemView *popup = m_lineEditCompleter->popup();
  395.         QListView *listView = qobject_cast<QListView*>(popup);
  396.         if (listView)
  397.             listView->setUniformItemSizes(true);
  398.     }
  399.     lineEdit->setCompleter(m_lineEditCompleter);
  400.     connect(lineEdit, SIGNAL(returnPressed()), this, SLOT(lineEditReturnPressed()));
  401.     m_lineEdits->addWidget(urlLineEdit);
  402.     m_lineEdits->setSizePolicy(lineEdit->sizePolicy());
  403.     // optimization to delay creating the more expensive WebView, history, etc
  404.     if (count() == 0) {
  405.         QWidget *emptyWidget = new QWidget;
  406.         QPalette p = emptyWidget->palette();
  407.         p.setColor(QPalette::Window, palette().color(QPalette::Base));
  408.         emptyWidget->setPalette(p);
  409.         emptyWidget->setAutoFillBackground(true);
  410.         disconnect(this, SIGNAL(currentChanged(int)),
  411.             this, SLOT(currentChanged(int)));
  412.         addTab(emptyWidget, tr("(Untitled)"));
  413.         connect(this, SIGNAL(currentChanged(int)),
  414.             this, SLOT(currentChanged(int)));
  415.         return 0;
  416.     }
  417.     // webview
  418.     WebView *webView = new WebView;
  419.     urlLineEdit->setWebView(webView);
  420.     connect(webView, SIGNAL(loadStarted()),
  421.             this, SLOT(webViewLoadStarted()));
  422.     connect(webView, SIGNAL(loadFinished(bool)),
  423.             this, SLOT(webViewIconChanged()));
  424.     connect(webView, SIGNAL(iconChanged()),
  425.             this, SLOT(webViewIconChanged()));
  426.     connect(webView, SIGNAL(titleChanged(const QString &)),
  427.             this, SLOT(webViewTitleChanged(const QString &)));
  428.     connect(webView, SIGNAL(urlChanged(const QUrl &)),
  429.             this, SLOT(webViewUrlChanged(const QUrl &)));
  430.     connect(webView->page(), SIGNAL(windowCloseRequested()),
  431.             this, SLOT(windowCloseRequested()));
  432.     connect(webView->page(), SIGNAL(geometryChangeRequested(const QRect &)),
  433.             this, SIGNAL(geometryChangeRequested(const QRect &)));
  434.     connect(webView->page(), SIGNAL(printRequested(QWebFrame *)),
  435.             this, SIGNAL(printRequested(QWebFrame *)));
  436.     connect(webView->page(), SIGNAL(menuBarVisibilityChangeRequested(bool)),
  437.             this, SIGNAL(menuBarVisibilityChangeRequested(bool)));
  438.     connect(webView->page(), SIGNAL(statusBarVisibilityChangeRequested(bool)),
  439.             this, SIGNAL(statusBarVisibilityChangeRequested(bool)));
  440.     connect(webView->page(), SIGNAL(toolBarVisibilityChangeRequested(bool)),
  441.             this, SIGNAL(toolBarVisibilityChangeRequested(bool)));
  442.     addTab(webView, tr("(Untitled)"));
  443.     if (makeCurrent)
  444.         setCurrentWidget(webView);
  445.     // webview actions
  446.     for (int i = 0; i < m_actions.count(); ++i) {
  447.         WebActionMapper *mapper = m_actions[i];
  448.         mapper->addChild(webView->page()->action(mapper->webAction()));
  449.     }
  450.     if (count() == 1)
  451.         currentChanged(currentIndex());
  452.     emit tabsChanged();
  453.     return webView;
  454. }
  455. void TabWidget::reloadAllTabs()
  456. {
  457.     for (int i = 0; i < count(); ++i) {
  458.         QWidget *tabWidget = widget(i);
  459.         if (WebView *tab = qobject_cast<WebView*>(tabWidget)) {
  460.             tab->reload();
  461.         }
  462.     }
  463. }
  464. void TabWidget::lineEditReturnPressed()
  465. {
  466.     if (QLineEdit *lineEdit = qobject_cast<QLineEdit*>(sender())) {
  467.         emit loadPage(lineEdit->text());
  468.         if (m_lineEdits->currentWidget() == lineEdit)
  469.             currentWebView()->setFocus();
  470.     }
  471. }
  472. void TabWidget::windowCloseRequested()
  473. {
  474.     WebPage *webPage = qobject_cast<WebPage*>(sender());
  475.     WebView *webView = qobject_cast<WebView*>(webPage->view());
  476.     int index = webViewIndex(webView);
  477.     if (index >= 0) {
  478.         if (count() == 1)
  479.             webView->webPage()->mainWindow()->close();
  480.         else
  481.             closeTab(index);
  482.     }
  483. }
  484. void TabWidget::closeOtherTabs(int index)
  485. {
  486.     if (-1 == index)
  487.         return;
  488.     for (int i = count() - 1; i > index; --i)
  489.         closeTab(i);
  490.     for (int i = index - 1; i >= 0; --i)
  491.         closeTab(i);
  492. }
  493. // When index is -1 index chooses the current tab
  494. void TabWidget::cloneTab(int index)
  495. {
  496.     if (index < 0)
  497.         index = currentIndex();
  498.     if (index < 0 || index >= count())
  499.         return;
  500.     WebView *tab = newTab(false);
  501.     tab->setUrl(webView(index)->url());
  502. }
  503. // When index is -1 index chooses the current tab
  504. void TabWidget::closeTab(int index)
  505. {
  506.     if (index < 0)
  507.         index = currentIndex();
  508.     if (index < 0 || index >= count())
  509.         return;
  510.     bool hasFocus = false;
  511.     if (WebView *tab = webView(index)) {
  512.         if (tab->isModified()) {
  513.             QMessageBox closeConfirmation(tab);
  514.             closeConfirmation.setWindowFlags(Qt::Sheet);
  515.             closeConfirmation.setWindowTitle(tr("Do you really want to close this page?"));
  516.             closeConfirmation.setInformativeText(tr("You have modified this page and when closing it you would lose the modification.n"
  517.                                                      "Do you really want to close this page?n"));
  518.             closeConfirmation.setIcon(QMessageBox::Question);
  519.             closeConfirmation.addButton(QMessageBox::Yes);
  520.             closeConfirmation.addButton(QMessageBox::No);
  521.             closeConfirmation.setEscapeButton(QMessageBox::No);
  522.             if (closeConfirmation.exec() == QMessageBox::No)
  523.                 return;
  524.         }
  525.         hasFocus = tab->hasFocus();
  526.         m_recentlyClosedTabsAction->setEnabled(true);
  527.         m_recentlyClosedTabs.prepend(tab->url());
  528.         if (m_recentlyClosedTabs.size() >= TabWidget::m_recentlyClosedTabsSize)
  529.             m_recentlyClosedTabs.removeLast();
  530.     }
  531.     QWidget *lineEdit = m_lineEdits->widget(index);
  532.     m_lineEdits->removeWidget(lineEdit);
  533.     lineEdit->deleteLater();
  534.     QWidget *webView = widget(index);
  535.     removeTab(index);
  536.     webView->deleteLater();
  537.     emit tabsChanged();
  538.     if (hasFocus && count() > 0)
  539.         currentWebView()->setFocus();
  540.     if (count() == 0)
  541.         emit lastTabClosed();
  542. }
  543. void TabWidget::webViewLoadStarted()
  544. {
  545.     WebView *webView = qobject_cast<WebView*>(sender());
  546.     int index = webViewIndex(webView);
  547.     if (-1 != index) {
  548.         QIcon icon(QLatin1String(":loading.gif"));
  549.         setTabIcon(index, icon);
  550.     }
  551. }
  552. void TabWidget::webViewIconChanged()
  553. {
  554.     WebView *webView = qobject_cast<WebView*>(sender());
  555.     int index = webViewIndex(webView);
  556.     if (-1 != index) {
  557.         QIcon icon = BrowserApplication::instance()->icon(webView->url());
  558.         setTabIcon(index, icon);
  559.     }
  560. }
  561. void TabWidget::webViewTitleChanged(const QString &title)
  562. {
  563.     WebView *webView = qobject_cast<WebView*>(sender());
  564.     int index = webViewIndex(webView);
  565.     if (-1 != index) {
  566.         setTabText(index, title);
  567.     }
  568.     if (currentIndex() == index)
  569.         emit setCurrentTitle(title);
  570.     BrowserApplication::historyManager()->updateHistoryItem(webView->url(), title);
  571. }
  572. void TabWidget::webViewUrlChanged(const QUrl &url)
  573. {
  574.     WebView *webView = qobject_cast<WebView*>(sender());
  575.     int index = webViewIndex(webView);
  576.     if (-1 != index) {
  577.         m_tabBar->setTabData(index, url);
  578.     }
  579.     emit tabsChanged();
  580. }
  581. void TabWidget::aboutToShowRecentTabsMenu()
  582. {
  583.     m_recentlyClosedTabsMenu->clear();
  584.     for (int i = 0; i < m_recentlyClosedTabs.count(); ++i) {
  585.         QAction *action = new QAction(m_recentlyClosedTabsMenu);
  586.         action->setData(m_recentlyClosedTabs.at(i));
  587.         QIcon icon = BrowserApplication::instance()->icon(m_recentlyClosedTabs.at(i));
  588.         action->setIcon(icon);
  589.         action->setText(m_recentlyClosedTabs.at(i).toString());
  590.         m_recentlyClosedTabsMenu->addAction(action);
  591.     }
  592. }
  593. void TabWidget::aboutToShowRecentTriggeredAction(QAction *action)
  594. {
  595.     QUrl url = action->data().toUrl();
  596.     loadUrlInCurrentTab(url);
  597. }
  598. void TabWidget::mouseDoubleClickEvent(QMouseEvent *event)
  599. {
  600.     if (!childAt(event->pos())
  601.             // Remove the line below when QTabWidget does not have a one pixel frame
  602.             && event->pos().y() < (tabBar()->y() + tabBar()->height())) {
  603.         newTab();
  604.         return;
  605.     }
  606.     QTabWidget::mouseDoubleClickEvent(event);
  607. }
  608. void TabWidget::contextMenuEvent(QContextMenuEvent *event)
  609. {
  610.     if (!childAt(event->pos())) {
  611.         m_tabBar->contextMenuRequested(event->pos());
  612.         return;
  613.     }
  614.     QTabWidget::contextMenuEvent(event);
  615. }
  616. void TabWidget::mouseReleaseEvent(QMouseEvent *event)
  617. {
  618.     if (event->button() == Qt::MidButton && !childAt(event->pos())
  619.             // Remove the line below when QTabWidget does not have a one pixel frame
  620.             && event->pos().y() < (tabBar()->y() + tabBar()->height())) {
  621.         QUrl url(QApplication::clipboard()->text(QClipboard::Selection));
  622.         if (!url.isEmpty() && url.isValid() && !url.scheme().isEmpty()) {
  623.             WebView *webView = newTab();
  624.             webView->setUrl(url);
  625.         }
  626.     }
  627. }
  628. void TabWidget::loadUrlInCurrentTab(const QUrl &url)
  629. {
  630.     WebView *webView = currentWebView();
  631.     if (webView) {
  632.         webView->loadUrl(url);
  633.         webView->setFocus();
  634.     }
  635. }
  636. void TabWidget::nextTab()
  637. {
  638.     int next = currentIndex() + 1;
  639.     if (next == count())
  640.         next = 0;
  641.     setCurrentIndex(next);
  642. }
  643. void TabWidget::previousTab()
  644. {
  645.     int next = currentIndex() - 1;
  646.     if (next < 0)
  647.         next = count() - 1;
  648.     setCurrentIndex(next);
  649. }
  650. static const qint32 TabWidgetMagic = 0xaa;
  651. QByteArray TabWidget::saveState() const
  652. {
  653.     int version = 1;
  654.     QByteArray data;
  655.     QDataStream stream(&data, QIODevice::WriteOnly);
  656.     stream << qint32(TabWidgetMagic);
  657.     stream << qint32(version);
  658.     QStringList tabs;
  659.     for (int i = 0; i < count(); ++i) {
  660.         if (WebView *tab = qobject_cast<WebView*>(widget(i))) {
  661.             tabs.append(tab->url().toString());
  662.         } else {
  663.             tabs.append(QString::null);
  664.         }
  665.     }
  666.     stream << tabs;
  667.     stream << currentIndex();
  668.     return data;
  669. }
  670. bool TabWidget::restoreState(const QByteArray &state)
  671. {
  672.     int version = 1;
  673.     QByteArray sd = state;
  674.     QDataStream stream(&sd, QIODevice::ReadOnly);
  675.     if (stream.atEnd())
  676.         return false;
  677.     qint32 marker;
  678.     qint32 v;
  679.     stream >> marker;
  680.     stream >> v;
  681.     if (marker != TabWidgetMagic || v != version)
  682.         return false;
  683.     QStringList openTabs;
  684.     stream >> openTabs;
  685.     for (int i = 0; i < openTabs.count(); ++i) {
  686.         if (i != 0)
  687.             newTab();
  688.         loadPage(openTabs.at(i));
  689.     }
  690.     int currentTab;
  691.     stream >> currentTab;
  692.     setCurrentIndex(currentTab);
  693.     return true;
  694. }
  695. WebActionMapper::WebActionMapper(QAction *root, QWebPage::WebAction webAction, QObject *parent)
  696.     : QObject(parent)
  697.     , m_currentParent(0)
  698.     , m_root(root)
  699.     , m_webAction(webAction)
  700. {
  701.     if (!m_root)
  702.         return;
  703.     connect(m_root, SIGNAL(triggered()), this, SLOT(rootTriggered()));
  704.     connect(root, SIGNAL(destroyed(QObject *)), this, SLOT(rootDestroyed()));
  705.     root->setEnabled(false);
  706. }
  707. void WebActionMapper::rootDestroyed()
  708. {
  709.     m_root = 0;
  710. }
  711. void WebActionMapper::currentDestroyed()
  712. {
  713.     updateCurrent(0);
  714. }
  715. void WebActionMapper::addChild(QAction *action)
  716. {
  717.     if (!action)
  718.         return;
  719.     connect(action, SIGNAL(changed()), this, SLOT(childChanged()));
  720. }
  721. QWebPage::WebAction WebActionMapper::webAction() const
  722. {
  723.     return m_webAction;
  724. }
  725. void WebActionMapper::rootTriggered()
  726. {
  727.     if (m_currentParent) {
  728.         QAction *gotoAction = m_currentParent->action(m_webAction);
  729.         gotoAction->trigger();
  730.     }
  731. }
  732. void WebActionMapper::childChanged()
  733. {
  734.     if (QAction *source = qobject_cast<QAction*>(sender())) {
  735.         if (m_root
  736.             && m_currentParent
  737.             && source->parent() == m_currentParent) {
  738.             m_root->setChecked(source->isChecked());
  739.             m_root->setEnabled(source->isEnabled());
  740.         }
  741.     }
  742. }
  743. void WebActionMapper::updateCurrent(QWebPage *currentParent)
  744. {
  745.     if (m_currentParent)
  746.         disconnect(m_currentParent, SIGNAL(destroyed(QObject *)),
  747.                    this, SLOT(currentDestroyed()));
  748.     m_currentParent = currentParent;
  749.     if (!m_root)
  750.         return;
  751.     if (!m_currentParent) {
  752.         m_root->setEnabled(false);
  753.         m_root->setChecked(false);
  754.         return;
  755.     }
  756.     QAction *source = m_currentParent->action(m_webAction);
  757.     m_root->setChecked(source->isChecked());
  758.     m_root->setEnabled(source->isEnabled());
  759.     connect(m_currentParent, SIGNAL(destroyed(QObject *)),
  760.             this, SLOT(currentDestroyed()));
  761. }