/* ============================================================ * * This file is a part of the rekonq project * * Copyright (C) 2008-2010 by Andrea Diamantini <adjam7 at gmail dot com> * Copyright (C) 2009-2010 by Lionel Chauvin <megabigbug@yahoo.fr> * * * 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 2 of * the License or (at your option) version 3 or any later version * accepted by the membership of KDE e.V. (or its successor approved * by the membership of KDE e.V.), which shall act as a proxy * defined in Section 14 of version 3 of the license. * * 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/>. * * ============================================================ */ // Self Includes #include "webview.h" #include "webview.moc" // Auto Includes #include "rekonq.h" // Local Includes #include "mainwindow.h" #include "mainview.h" #include "webpage.h" #include "bookmarksmanager.h" #include "searchengine.h" // KDE Includes #include <KService> #include <KUriFilterData> #include <KStandardShortcut> #include <KMenu> #include <KActionMenu> #include <ktoolinvocation.h> // Qt Includes #include <QtCore/QDir> #include <QtGui/QAction> #include <QtGui/QContextMenuEvent> #include <QtGui/QWheelEvent> #include <QtGui/QMouseEvent> #include <QtGui/QClipboard> #include <QtGui/QKeyEvent> #include <QtGui/QLayout> #include <QtDBus/QDBusConnectionInterface> #include <QtDBus/QDBusInterface> #include <QtDBus/QDBusReply> WebView::WebView(QWidget* parent) : KWebView(parent, false) , _mousePos(QPoint(0, 0)) , _scrollTimer(new QTimer(this)) , _VScrollSpeed(0) , _HScrollSpeed(0) , _canEnableAutoScroll(true) , _isAutoScrollEnabled(false) { WebPage *page = new WebPage(this); setPage(page); // download system connect(this, SIGNAL(linkShiftClicked(const KUrl &)), page, SLOT(downloadUrl(const KUrl &))); connect(page, SIGNAL(downloadRequested(const QNetworkRequest &)), page, SLOT(downloadRequest(const QNetworkRequest &))); // middle click || ctrl + click signal connect(this, SIGNAL(linkMiddleOrCtrlClicked(const KUrl &)), this, SLOT(loadUrlInNewTab(const KUrl &))); // loadUrl signal connect(this, SIGNAL(loadUrl(const KUrl &, const Rekonq::OpenType &)), Application::instance(), SLOT(loadUrl(const KUrl &, const Rekonq::OpenType &))); // scrolling timer connect(_scrollTimer, SIGNAL(timeout()), this, SLOT(scrollFrameChanged())); _scrollTimer->setInterval(100); } WebView::~WebView() { delete _scrollTimer; disconnect(); } WebPage *WebView::page() { WebPage *page = qobject_cast<WebPage *>(KWebView::page()); return page; } void WebView::contextMenuEvent(QContextMenuEvent *event) { QWebHitTestResult result = page()->mainFrame()->hitTestContent(event->pos()); MainWindow *mainwindow = Application::instance()->mainWindow(); KMenu menu(this); QAction *a; // is a link? if (!result.linkUrl().isEmpty()) { // link actions a = new KAction(KIcon("tab-new"), i18n("Open in New &Tab"), this); a->setData(result.linkUrl()); connect(a, SIGNAL(triggered(bool)), this, SLOT(openLinkInNewTab())); menu.addAction(a); a = new KAction(KIcon("window-new"), i18n("Open in New &Window"), this); a->setData(result.linkUrl()); connect(a, SIGNAL(triggered(bool)), this, SLOT(openLinkInNewWindow())); menu.addAction(a); menu.addAction(pageAction(KWebPage::DownloadLinkToDisk)); menu.addAction(pageAction(KWebPage::CopyLinkToClipboard)); menu.addSeparator(); } // is content editable && selected? Add CUT if (result.isContentEditable() && result.isContentSelected()) { // actions for text selected in field menu.addAction(pageAction(KWebPage::Cut)); } // is content selected) Add COPY if (result.isContentSelected()) { a = pageAction(KWebPage::Copy); if (!result.linkUrl().isEmpty()) a->setText(i18n("Copy Text")); //for link else a->setText(i18n("Copy")); menu.addAction(a); } // is content editable? Add PASTE if (result.isContentEditable()) { menu.addAction(pageAction(KWebPage::Paste)); } // is content selected? Add SEARCH actions if (result.isContentSelected()) { KActionMenu *searchMenu = new KActionMenu(KIcon("edit-find"), i18n("Search with"), this); foreach(KService::Ptr engine, SearchEngine::favorites()) { a = new KAction(engine->name(), this); a->setIcon(Application::icon(SearchEngine::buildQuery(engine, ""))); a->setData(engine->entryPath()); connect(a, SIGNAL(triggered(bool)), this, SLOT(search())); searchMenu->addAction(a); } if (!searchMenu->menu()->isEmpty()) { menu.addAction(searchMenu); } menu.addSeparator(); // TODO Add translate, show translation } // is an image? if (!result.pixmap().isNull()) { menu.addSeparator(); // TODO remove copy_this_image action a = new KAction(KIcon("view-media-visualization"), i18n("&View Image"), this); a->setData(result.imageUrl()); connect(a, SIGNAL(triggered(Qt::MouseButtons, Qt::KeyboardModifiers)), this, SLOT(viewImage(Qt::MouseButtons, Qt::KeyboardModifiers))); menu.addAction(a); menu.addAction(pageAction(KWebPage::DownloadImageToDisk)); menu.addAction(pageAction(KWebPage::CopyImageToClipboard)); menu.addSeparator(); } // Open url text in new tab/window if (result.linkUrl().isEmpty()) { QString text = selectedText(); if (text.startsWith(QL1S("http://")) || text.startsWith(QL1S("https://")) || text.startsWith(QL1S("www.")) ) { QString truncatedURL = text; if (text.length() > 18) { truncatedURL.truncate(15); truncatedURL += "..."; } //open selected text url in a new tab a = new KAction(KIcon("tab-new"), i18n("Open '%1' in New Tab", truncatedURL), this); a->setData(QUrl(text)); connect(a, SIGNAL(triggered(bool)), this, SLOT(openLinkInNewTab())); menu.addAction(a); //open selected text url in a new window a = new KAction(KIcon("window-new"), i18n("Open '%1' in New Window", truncatedURL), this); a->setData(QUrl(text)); connect(a, SIGNAL(triggered(bool)), this, SLOT(openLinkInNewWindow())); menu.addAction(a); menu.addSeparator(); } } // page actions if (!result.isContentSelected() && result.linkUrl().isEmpty()) { // navigation QWebHistory *history = page()->history(); if (history->canGoBack()) { menu.addAction(pageAction(KWebPage::Back)); } if (history->canGoForward()) { menu.addAction(pageAction(KWebPage::Forward)); } menu.addAction(mainwindow->actionByName("view_redisplay")); if (result.pixmap().isNull()) { menu.addSeparator(); menu.addAction(mainwindow->actionByName("new_tab")); menu.addAction(mainwindow->actionByName("new_window")); menu.addSeparator(); //Frame KActionMenu *frameMenu = new KActionMenu(i18n("Current Frame"), this); frameMenu->addAction(pageAction(KWebPage::OpenFrameInNewWindow)); a = new KAction(KIcon("document-print-frame"), i18n("Print Frame"), this); connect(a, SIGNAL(triggered()), this, SLOT(printFrame())); frameMenu->addAction(a); menu.addAction(frameMenu); menu.addSeparator(); // Page Actions menu.addAction(pageAction(KWebPage::SelectAll)); menu.addAction(mainwindow->actionByName(KStandardAction::name(KStandardAction::SaveAs))); if (ReKonfig::kgetList()) { a = new KAction(KIcon("kget"), i18n("List All Links"), this); connect(a, SIGNAL(triggered(bool)), page(), SLOT(downloadAllContentsWithKGet())); menu.addAction(a); } menu.addAction(mainwindow->actionByName("page_source")); a = new KAction(KIcon("layer-visible-on"), i18n("Inspect Element"), this); connect(a, SIGNAL(triggered(bool)), this, SLOT(inspect())); menu.addAction(a); a = Application::bookmarkProvider()->actionByName("rekonq_add_bookmark"); menu.addAction(a); } if (mainwindow->isFullScreen()) { menu.addSeparator(); menu.addAction(mainwindow->actionByName("fullscreen")); } } menu.exec(mapToGlobal(event->pos())); } void WebView::mousePressEvent(QMouseEvent *event) { if (_isAutoScrollEnabled) { setCursor(Qt::ArrowCursor); _VScrollSpeed = 0; _HScrollSpeed = 0; _scrollTimer->stop(); _isAutoScrollEnabled = false; return; } QWebHitTestResult result = page()->mainFrame()->hitTestContent(event->pos()); _canEnableAutoScroll = ReKonfig::autoScroll() && !result.isContentEditable() && result.linkUrl().isEmpty(); switch (event->button()) { case Qt::XButton1: triggerPageAction(KWebPage::Back); break; case Qt::XButton2: triggerPageAction(KWebPage::Forward); break; case Qt::MidButton: if (_canEnableAutoScroll && !_isAutoScrollEnabled) { setCursor(KIcon("transform-move").pixmap(32)); _clickPos = event->pos(); _isAutoScrollEnabled = true; } break; default: break; }; KWebView::mousePressEvent(event); } void WebView::mouseMoveEvent(QMouseEvent *event) { _mousePos = event->pos(); if (_isAutoScrollEnabled) { QPoint r = _mousePos - _clickPos; _HScrollSpeed = r.x() / 2; // you are too fast.. _VScrollSpeed = r.y() / 2; if (!_scrollTimer->isActive()) _scrollTimer->start(); return; } if (Application::instance()->mainWindow()->isFullScreen()) { if (event->pos().y() >= 0 && event->pos().y() <= 4) { Application::instance()->mainWindow()->setWidgetsVisible(true); } else { Application::instance()->mainWindow()->setWidgetsVisible(false); } } KWebView::mouseMoveEvent(event); } void WebView::search() { KAction *a = qobject_cast<KAction*>(sender()); KService::Ptr engine = KService::serviceByDesktopPath(a->data().toString()); KUrl urlSearch = KUrl(SearchEngine::buildQuery(engine, selectedText())); emit loadUrl(urlSearch, Rekonq::NewCurrentTab); } void WebView::printFrame() { Application::instance()->mainWindow()->printRequested(page()->currentFrame()); } void WebView::viewImage(Qt::MouseButtons buttons, Qt::KeyboardModifiers modifiers) { KAction *a = qobject_cast<KAction*>(sender()); KUrl url(a->data().toUrl()); if (modifiers & Qt::ControlModifier || buttons == Qt::MidButton) { emit loadUrl(url, Rekonq::SettingOpenTab); } else { emit loadUrl(url, Rekonq::CurrentTab); } } void WebView::openLinkInNewWindow() { KAction *a = qobject_cast<KAction*>(sender()); KUrl url(a->data().toUrl()); emit loadUrl(url, Rekonq::NewWindow); } void WebView::openLinkInNewTab() { KAction *a = qobject_cast<KAction*>(sender()); KUrl url(a->data().toUrl()); emit loadUrl(url, Rekonq::SettingOpenTab); } void WebView::keyPressEvent(QKeyEvent *event) { if (event->modifiers() == Qt::ControlModifier) { if (event->key() == Qt::Key_C) { triggerPageAction(KWebPage::Copy); return; } if (event->key() == Qt::Key_A) { triggerPageAction(KWebPage::SelectAll); return; } } if (!_canEnableAutoScroll) { KWebView::keyPressEvent(event); return; } // Auto Scrolling if (event->modifiers() == Qt::ShiftModifier) { if (event->key() == Qt::Key_Up) { _VScrollSpeed--; if (!_scrollTimer->isActive()) _scrollTimer->start(); return; } if (event->key() == Qt::Key_Down) { _VScrollSpeed++; if (!_scrollTimer->isActive()) _scrollTimer->start(); return; } if (event->key() == Qt::Key_Right) { _HScrollSpeed++; if (!_scrollTimer->isActive()) _scrollTimer->start(); return; } if (event->key() == Qt::Key_Left) { _HScrollSpeed--; if (!_scrollTimer->isActive()) _scrollTimer->start(); return; } if (_scrollTimer->isActive()) { _scrollTimer->stop(); } else { if (_VScrollSpeed || _HScrollSpeed) _scrollTimer->start(); } } KWebView::keyPressEvent(event); } void WebView::wheelEvent(QWheelEvent *event) { // Sync with the zoom slider if (event->modifiers() == Qt::ControlModifier) { emit zoomChanged(zoomFactor()); } KWebView::wheelEvent(event); // Limits of the slider if (zoomFactor() > 1.9) setZoomFactor(1.9); else if (zoomFactor() < 0.1) setZoomFactor(0.1); } void WebView::inspect() { QAction *a = Application::instance()->mainWindow()->actionByName("web_inspector"); if (a && !a->isChecked()) a->trigger(); pageAction(QWebPage::InspectElement)->trigger(); } void WebView::loadUrlInNewTab(const KUrl &url) { emit loadUrl(url, Rekonq::SettingOpenTab); } void WebView::scrollFrameChanged() { // do the scrolling page()->currentFrame()->scroll(_HScrollSpeed, _VScrollSpeed); // check if we reached the end int y = page()->currentFrame()->scrollPosition().y(); if (y == 0 || y == page()->currentFrame()->scrollBarMaximum(Qt::Vertical)) _VScrollSpeed = 0; int x = page()->currentFrame()->scrollPosition().x(); if (x == 0 || x == page()->currentFrame()->scrollBarMaximum(Qt::Horizontal)) _HScrollSpeed = 0; }