/* ============================================================ * 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 "webpage.h" #include "tabbedwebview.h" #include "tabwidget.h" #include "qupzilla.h" #include "downloadmanager.h" #include "webpluginfactory.h" #include "mainapplication.h" #ifdef NONBLOCK_JS_DIALOGS #include "ui_jsconfirm.h" #include "ui_jsalert.h" #include "ui_jsprompt.h" #endif #include "ui_closedialog.h" #include "widget.h" #include "globalfunctions.h" #include "pluginproxy.h" #include "speeddial.h" #include "popupwebpage.h" #include "popupwebview.h" #include "networkmanagerproxy.h" #include "adblockicon.h" QString WebPage::m_lastUploadLocation = QDir::homePath(); QString WebPage::m_userAgent = QString(); WebPage::WebPage(QupZilla* mainClass) : QWebPage() , p_QupZilla(mainClass) , m_view(0) , m_speedDial(mApp->plugins()->speedDial()) , m_fileWatcher(0) , m_runningLoop(0) , m_blockAlerts(false) , m_secureStatus(false) , m_isClosing(false) { m_networkProxy = new NetworkManagerProxy(this); m_networkProxy->setPrimaryNetworkAccessManager(mApp->networkManager()); m_networkProxy->setPage(this); setNetworkAccessManager(m_networkProxy); setForwardUnsupportedContent(true); setPluginFactory(new WebPluginFactory(this)); history()->setMaximumItemCount(20); connect(this, SIGNAL(unsupportedContent(QNetworkReply*)), this, SLOT(handleUnsupportedContent(QNetworkReply*))); connect(this, SIGNAL(loadProgress(int)), this, SLOT(progress(int))); connect(this, SIGNAL(loadFinished(bool)), this, SLOT(finished())); connect(this, SIGNAL(printRequested(QWebFrame*)), this, SLOT(printFrame(QWebFrame*))); connect(this, SIGNAL(downloadRequested(QNetworkRequest)), this, SLOT(downloadRequested(QNetworkRequest))); connect(mainFrame(), SIGNAL(javaScriptWindowObjectCleared()), this, SLOT(addJavaScriptObject())); } QUrl WebPage::url() const { return mainFrame()->url(); } void WebPage::setWebView(TabbedWebView* view) { if (m_view == view) { return; } if (m_view) { delete m_view; m_view = 0; } m_view = view; m_view->setWebPage(this); connect(m_view, SIGNAL(urlChanged(QUrl)), this, SLOT(urlChanged(QUrl))); } void WebPage::scheduleAdjustPage() { if (m_view && m_view->isLoading()) { m_adjustingScheduled = true; } else { mainFrame()->setZoomFactor(mainFrame()->zoomFactor() + 1); mainFrame()->setZoomFactor(mainFrame()->zoomFactor() - 1); } } bool WebPage::isRunningLoop() { return m_runningLoop; } void WebPage::setUserAgent(const QString &agent) { if (!agent.isEmpty()) { m_userAgent = QString("%1 (QupZilla %2)").arg(agent, QupZilla::VERSION); } else { m_userAgent = agent; } } void WebPage::urlChanged(const QUrl &url) { Q_UNUSED(url) m_adBlockedEntries.clear(); m_blockAlerts = false; } void WebPage::progress(int prog) { Q_UNUSED(prog) bool secStatus = sslCertificate().isValid(); if (secStatus != m_secureStatus) { m_secureStatus = secStatus; emit privacyChanged(sslCertificate().isValid()); } } void WebPage::finished() { progress(100); if (m_adjustingScheduled) { m_adjustingScheduled = false; mainFrame()->setZoomFactor(mainFrame()->zoomFactor() + 1); mainFrame()->setZoomFactor(mainFrame()->zoomFactor() - 1); } if (url().scheme() == "file") { if (!m_fileWatcher) { m_fileWatcher = new QFileSystemWatcher(this); connect(m_fileWatcher, SIGNAL(fileChanged(QString)), this, SLOT(watchedFileChanged(QString))); } const QString &filePath = url().toLocalFile(); if (QFile::exists(filePath) && !m_fileWatcher->files().contains(filePath)) { m_fileWatcher->addPath(filePath); } } else if (m_fileWatcher && !m_fileWatcher->files().isEmpty()) { m_fileWatcher->removePaths(m_fileWatcher->files()); } QTimer::singleShot(100, this, SLOT(cleanBlockedObjects())); } void WebPage::watchedFileChanged(const QString &file) { if (url().toLocalFile() == file) { triggerAction(QWebPage::Reload); } } void WebPage::printFrame(QWebFrame* frame) { WebView* webView = qobject_cast(view()); if (!webView) { return; } webView->printPage(frame); } void WebPage::addJavaScriptObject() { if (url().toString() != "qupzilla:speeddial") { return; } mainFrame()->addToJavaScriptWindowObject("speeddial", m_speedDial); m_speedDial->addWebFrame(mainFrame()); } void WebPage::handleUnsupportedContent(QNetworkReply* reply) { if (!reply) { return; } const QUrl &url = reply->url(); switch (reply->error()) { case QNetworkReply::NoError: if (reply->header(QNetworkRequest::ContentTypeHeader).isValid()) { QString requestUrl = reply->request().url().toString(QUrl::RemoveFragment | QUrl::RemoveQuery); if (requestUrl.endsWith(".swf")) { QWebElement docElement = mainFrame()->documentElement(); QWebElement object = docElement.findFirst(QString("object[src=\"%1\"]").arg(requestUrl)); QWebElement embed = docElement.findFirst(QString("embed[src=\"%1\"]").arg(requestUrl)); if (!object.isNull() || !embed.isNull()) { qDebug() << "WebPage::UnsupportedContent" << url << "Attempt to download flash object on site!"; reply->deleteLater(); return; } } DownloadManager* dManager = mApp->downManager(); dManager->handleUnsupportedContent(reply, this); return; } case QNetworkReply::ProtocolUnknownError: qDebug() << "WebPage::UnsupportedContent" << url << "ProtocolUnknowError"; // We are not going to end in endless new tab opening loop in case // user has QupZilla as default provider for http / https urls if (!url.scheme().startsWith("http")) { QDesktopServices::openUrl(url); } reply->deleteLater(); return; default: break; } qDebug() << "WebPage::UnsupportedContent error" << url << reply->errorString(); reply->deleteLater(); } void WebPage::downloadRequested(const QNetworkRequest &request) { DownloadManager* dManager = mApp->downManager(); dManager->download(request, this); } void WebPage::setSSLCertificate(const QSslCertificate &cert) { // if (cert != m_SslCert) m_SslCert = cert; } QSslCertificate WebPage::sslCertificate() { if (url().scheme() == "https" && m_SslCert.subjectInfo(QSslCertificate::CommonName).remove("*").contains(QRegExp(url().host()))) { return m_SslCert; } else { return QSslCertificate(); } } bool WebPage::acceptNavigationRequest(QWebFrame* frame, const QNetworkRequest &request, NavigationType type) { m_lastRequest = request; m_lastRequestType = type; const QString &scheme = request.url().scheme(); if (scheme == "mailto" || scheme == "ftp") { QDesktopServices::openUrl(request.url()); return false; } if (type == QWebPage::NavigationTypeFormResubmitted) { QString message = tr("To show this page, QupZilla must resend request which do it again \n" "(like searching on making an shoping, which has been already done.)"); bool result = (QMessageBox::question(view(), tr("Confirm form resubmission"), message, QMessageBox::Yes | QMessageBox::No, QMessageBox::No) == QMessageBox::Yes); if (!result) { return false; } } bool accept = QWebPage::acceptNavigationRequest(frame, request, type); return accept; } void WebPage::populateNetworkRequest(QNetworkRequest &request) { WebPage* pagePointer = this; QVariant variant = qVariantFromValue((void*) pagePointer); request.setAttribute((QNetworkRequest::Attribute)(QNetworkRequest::User + 100), variant); request.setAttribute((QNetworkRequest::Attribute)(QNetworkRequest::User + 101), m_lastRequestType); } QWebPage* WebPage::createWindow(QWebPage::WebWindowType type) { return new PopupWebPage(type, p_QupZilla); } void WebPage::addAdBlockRule(const QString &filter, const QUrl &url) { AdBlockedEntry entry; entry.rule = filter; entry.url = url; if (!m_adBlockedEntries.contains(entry)) { m_adBlockedEntries.append(entry); } } void WebPage::cleanBlockedObjects() { QStringList findingStrings; foreach(const AdBlockedEntry & entry, m_adBlockedEntries) { if (entry.url.toString().endsWith(".js")) { continue; } findingStrings.append(entry.url.toString()); const QUrl &mainFrameUrl = url(); if (entry.url.scheme() == mainFrameUrl.scheme() && entry.url.host() == mainFrameUrl.host()) { //May be relative url QString relativeUrl = qz_makeRelativeUrl(mainFrameUrl, entry.url).toString(); findingStrings.append(relativeUrl); if (relativeUrl.startsWith("/")) { findingStrings.append(relativeUrl.right(relativeUrl.size() - 1)); } } } const QWebElement &docElement = mainFrame()->documentElement(); QWebElementCollection elements; foreach(const QString & s, findingStrings) { elements.append(docElement.findAll("*[src=\"" + s + "\"]")); } foreach(QWebElement element, elements) { element.setStyleProperty("visibility", "hidden"); } } QString WebPage::userAgentForUrl(const QUrl &url) const { if (m_userAgent.isEmpty()) { m_userAgent = QWebPage::userAgentForUrl(url); #ifdef Q_WS_MAC #ifdef __i386__ || __x86_64__ m_userAgent.replace("PPC Mac OS X", "Intel Mac OS X"); #endif #endif } return m_userAgent; } bool WebPage::supportsExtension(Extension extension) const { Q_UNUSED(extension) return true; } bool WebPage::extension(Extension extension, const ExtensionOption* option, ExtensionReturn* output) { if (extension == ChooseMultipleFilesExtension) { const QWebPage::ChooseMultipleFilesExtensionOption* exOption = static_cast(option); QWebPage::ChooseMultipleFilesExtensionReturn* exReturn = static_cast(output); if (!exOption || !exReturn) { return QWebPage::extension(extension, option, output); } QString suggestedFileName; if (!exOption->suggestedFileNames.isEmpty()) { suggestedFileName = exOption->suggestedFileNames.first(); } exReturn->fileNames = QFileDialog::getOpenFileNames(0, tr("Select files to upload..."), suggestedFileName); return true; } const ErrorPageExtensionOption* exOption = static_cast(option); ErrorPageExtensionReturn* exReturn = static_cast(output); if (!exOption || !exReturn) { return QWebPage::extension(extension, option, output); } WebPage* erPage = qobject_cast(exOption->frame->page()); if (!erPage) { return QWebPage::extension(extension, option, output); } QString errorString; if (exOption->domain == QWebPage::QtNetwork) { switch (exOption->error) { case QNetworkReply::ConnectionRefusedError: errorString = tr("Server refused the connection"); break; case QNetworkReply::RemoteHostClosedError: errorString = tr("Server closed the connection"); break; case QNetworkReply::HostNotFoundError: errorString = tr("Server not found"); break; case QNetworkReply::TimeoutError: errorString = tr("Connection timed out"); break; case QNetworkReply::SslHandshakeFailedError: errorString = tr("Untrusted connection"); break; case QNetworkReply::TemporaryNetworkFailureError: errorString = tr("Temporary network failure"); break; case QNetworkReply::ProxyConnectionRefusedError: errorString = tr("Proxy connection refused"); break; case QNetworkReply::ProxyNotFoundError: errorString = tr("Proxy server not found"); break; case QNetworkReply::ProxyTimeoutError: errorString = tr("Proxy connection timed out"); break; case QNetworkReply::ProxyAuthenticationRequiredError: errorString = tr("Proxy authentication required"); break; case QNetworkReply::ContentNotFoundError: errorString = tr("Content not found"); break; case QNetworkReply::ContentAccessDenied: if (exOption->errorString.startsWith("AdBlockRule")) { if (exOption->frame != erPage->mainFrame()) { //Content in