/* ============================================================ * * 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 "bookmarkprovider.h" #include "searchengine.h" #include "websnap.h" // KDE Includes #include <KService> #include <KUriFilterData> #include <KStandardShortcut> #include <KMenu> #include <KActionMenu> #include <ktoolinvocation.h> #include <KStandardDirs> // 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 <QtGui/QWindowsStyle> #include <QtDBus/QDBusConnectionInterface> #include <QtDBus/QDBusInterface> #include <QtDBus/QDBusReply> WebView::WebView(QWidget* parent) : KWebView(parent, false) , _mousePos(QPoint(0, 0)) , _autoScrollTimer(new QTimer(this)) , _VScrollSpeed(0) , _HScrollSpeed(0) , _canEnableAutoScroll(true) , _isAutoScrollEnabled(false) , _smoothScrollTimer(new QTimer(this)) , _smoothScrolling(false) , _dy(0) { WebPage *page = new WebPage(this); setPage(page); // // NOTE This is a lot hackish. We copied it from Arora, but using a "Windows Style" // // seems really a pity to me. The problem is that using a KStyle everything seems "broken" // // (at least with dark colors). So I think the code should be somthing like: // KStyle s; // setPalette( s.standardPalette() ); // // But it doesn't work :( // // We'll see in next KDE releases... QPalette p; if (p.color(QPalette::ButtonText).lightness() > 50) { //if it is a dark theme QWindowsStyle s; p = s.standardPalette(); setPalette(p); } // 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 &))); // Auto scroll timer connect(_autoScrollTimer, SIGNAL(timeout()), this, SLOT(scrollFrameChanged())); _autoScrollTimer->setInterval(100); // Smooth scroll timer connect(_smoothScrollTimer, SIGNAL(timeout()), this, SLOT(scrollTick())); _smoothScrollTimer->setInterval(16); } WebView::~WebView() { delete _autoScrollTimer; if (_smoothScrolling) stopScrolling(); delete _smoothScrollTimer; disconnect(); WebPage* p = page(); QPixmap preview = WebSnap::renderClosingPagePreview(*p); QString path = WebSnap::imagePathFromUrl(p->mainFrame()->url().toString()); QFile::remove(path); preview.save(path); } 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; KAction *inspectAction = new KAction(KIcon("layer-visible-on"), i18n("Inspect Element"), this); connect(inspectAction, SIGNAL(triggered(bool)), this, SLOT(inspect())); // 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(); menu.addAction(inspectAction); // 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(); menu.addAction(inspectAction); } // Open url text in new tab/window if (result.linkUrl().isEmpty()) { QString text = selectedText(); text = text.trimmed(); 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 (!KStandardDirs::findExe("kget").isNull() && 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")); menu.addAction(inspectAction); 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; _autoScrollTimer->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 (!_autoScrollTimer->isActive()) _autoScrollTimer->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::NewFocusedTab); } 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::NewTab); } 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::NewTab); } 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 (!_autoScrollTimer->isActive()) _autoScrollTimer->start(); return; } if (event->key() == Qt::Key_Down) { _VScrollSpeed++; if (!_autoScrollTimer->isActive()) _autoScrollTimer->start(); return; } if (event->key() == Qt::Key_Right) { _HScrollSpeed++; if (!_autoScrollTimer->isActive()) _autoScrollTimer->start(); return; } if (event->key() == Qt::Key_Left) { _HScrollSpeed--; if (!_autoScrollTimer->isActive()) _autoScrollTimer->start(); return; } if (_autoScrollTimer->isActive()) { _autoScrollTimer->stop(); } else { if (_VScrollSpeed || _HScrollSpeed) _autoScrollTimer->start(); } } KWebView::keyPressEvent(event); } void WebView::wheelEvent(QWheelEvent *event) { // To let some websites (eg: google maps) to handle wheel events int ypos = page()->currentFrame()->scrollPosition().y(); KWebView::wheelEvent(event); // Sync with the zoom slider if (event->modifiers() == Qt::ControlModifier) { // Limits of the slider if (zoomFactor() > 1.9) setZoomFactor(1.9); else if (zoomFactor() < 0.1) setZoomFactor(0.1); // Round the factor (Fix slider's end value) int newFactor = zoomFactor() * 10; if ((zoomFactor() * 10 - newFactor) > 0.5) newFactor++; emit zoomChanged(newFactor); } else if (ReKonfig::smoothScrolling() && ypos != page()->currentFrame()->scrollPosition().y()) { page()->currentFrame()->setScrollPosition(QPoint(page()->currentFrame()->scrollPosition().x(), ypos)); int numDegrees = event->delta() / 8; int numSteps = numDegrees / 15; if ((numSteps > 0) != !_scrollBottom) stopScrolling(); if (numSteps > 0) _scrollBottom = false; else _scrollBottom = true; setupSmoothScrolling(QApplication::wheelScrollLines() * abs(numSteps) * 20); } } 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::NewTab); } 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; } void WebView::setupSmoothScrolling(int posY) { int ddy = qMax(_smoothScrollSteps ? abs(_dy)/_smoothScrollSteps : 0,3); _dy += posY; if (_dy <= 0) { stopScrolling(); return; } _smoothScrollSteps = 8; if (_dy / _smoothScrollSteps < ddy) { _smoothScrollSteps = (abs(_dy)+ddy-1)/ddy; if (_smoothScrollSteps < 1) _smoothScrollSteps = 1; } _smoothScrollTime.start(); if (!_smoothScrolling) { _smoothScrolling = true; _smoothScrollTimer->start(); scrollTick(); } } void WebView::scrollTick() { if (_dy == 0) { stopScrolling(); return; } if (_smoothScrollSteps < 1) _smoothScrollSteps = 1; int takesteps = _smoothScrollTime.restart() / 16; int scroll_y = 0; if (takesteps < 1) takesteps = 1; if (takesteps > _smoothScrollSteps) takesteps = _smoothScrollSteps; for(int i = 0; i < takesteps; i++) { int ddy = (_dy / (_smoothScrollSteps+1)) * 2; // limit step to requested scrolling distance if (abs(ddy) > abs(_dy)) ddy = _dy; // update remaining scroll _dy -= ddy; scroll_y += ddy; _smoothScrollSteps--; } if (_scrollBottom) page()->currentFrame()->scroll(0, scroll_y); else page()->currentFrame()->scroll(0, -scroll_y); } void WebView::stopScrolling() { _smoothScrollTimer->stop(); _dy = 0; _smoothScrolling = false; }