/* ============================================================ * * This file is a part of the rekonq project * * Copyright (C) 2008-2011 by Andrea Diamantini <adjam7 at gmail dot com> * Copyright (C) 2009 by Paweł Prażak <pawelprazak at gmail dot com> * Copyright (C) 2009-2011 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 "application.h" #include "application.moc" // Auto Includes #include "rekonq.h" // Local Includes #include "adblockmanager.h" #include "bookmarkprovider.h" #include "filterurljob.h" #include "historymanager.h" #include "iconmanager.h" #include "mainview.h" #include "mainwindow.h" #include "messagebar.h" #include "opensearchmanager.h" #include "sessionmanager.h" #include "stackedurlbar.h" #include "tabbar.h" #include "urlbar.h" #include "webtab.h" // KDE Includes #include <KCmdLineArgs> #include <KIcon> #include <KMessageBox> #include <KStandardDirs> #include <KStartupInfo> #include <ThreadWeaver/Weaver> #include <KAction> #include <KWindowSystem> #include <KWindowInfo> #include <KGlobal> #include <KCharsets> // Qt Includes #include <QVBoxLayout> using namespace ThreadWeaver; Application::Application() : KUniqueApplication() , _privateBrowsingAction(0) { connect(Weaver::instance(), SIGNAL(jobDone(ThreadWeaver::Job*)), this, SLOT(loadResolvedUrl(ThreadWeaver::Job*))); _privateBrowsingAction = new KAction(KIcon("view-media-artist"), i18n("Private &Browsing"), this); _privateBrowsingAction->setCheckable(true); connect(_privateBrowsingAction, SIGNAL(triggered(bool)), this, SLOT(setPrivateBrowsingMode(bool))); } Application::~Application() { // ok, we are closing well. // Don't recover on next load.. ReKonfig::setRecoverOnCrash(0); saveConfiguration(); Q_FOREACH(QWeakPointer<MainWindow> window, m_mainWindows) { delete window.data(); window.clear(); } if (!m_historyManager.isNull()) { delete m_historyManager.data(); m_historyManager.clear(); } if (!m_bookmarkProvider.isNull()) { delete m_bookmarkProvider.data(); m_bookmarkProvider.clear(); } if (!m_sessionManager.isNull()) { delete m_sessionManager.data(); m_sessionManager.clear(); } if (!m_opensearchManager.isNull()) { delete m_opensearchManager.data(); m_opensearchManager.clear(); } if (!m_iconManager.isNull()) { delete m_iconManager.data(); m_iconManager.clear(); } if (!m_adblockManager.isNull()) { delete m_adblockManager.data(); m_adblockManager.clear(); } } int Application::newInstance() { KCmdLineArgs::setCwd(QDir::currentPath().toUtf8()); KCmdLineArgs* args = KCmdLineArgs::parsedArgs(); // not that easy, indeed // We have to consider 3 variables here: // 1) Is first load? // 2) Are there arguments? // 3) Is rekonq recovering from crash? // so, we have 8 possible cases... bool isFirstLoad = m_mainWindows.isEmpty(); bool areThereArguments = (args->count() > 0); bool isRekonqCrashed = ReKonfig::recoverOnCrash(); // note that isRekonqCrashed is always true if it is not the first load // !isFirstLoad -> isRekonqCrashed kDebug() << "is first load? " << isFirstLoad; kDebug() << "are there arguments? " << areThereArguments; kDebug() << "is rekonq crashed? " << isRekonqCrashed; int exitValue = 1 * isFirstLoad + 2 * areThereArguments + 4 * isRekonqCrashed; if (isRekonqCrashed && isFirstLoad) { loadUrl(KUrl("about:closedTabs"), Rekonq::NewWindow); MessageBar *msgBar = new MessageBar(i18n("It seems rekonq was not closed properly. Do you want " "to restore the last saved session?") , mainWindow()->currentTab() , QMessageBox::Warning , MessageBar::Yes | MessageBar::No); connect(msgBar, SIGNAL(accepted()), sessionManager(), SLOT(restoreSession())); mainWindow()->currentTab()->insertBar(msgBar); } if (areThereArguments) { KUrl::List urlList; for (int i = 0; i < args->count(); ++i) { const KUrl u = args->url(i); if (u.isLocalFile() && QFile::exists(u.toLocalFile())) // "rekonq somefile.html" case urlList += u; else urlList += KUrl(args->arg(i)); // "rekonq kde.org" || "rekonq kde:kdialog" case } if (isFirstLoad && !isRekonqCrashed) { // No windows in the current desktop? No windows at all? // Create a new one and load there sites... loadUrl(urlList.at(0), Rekonq::NewWindow); } else { if (ReKonfig::openTabNoWindow()) { loadUrl(urlList.at(0), Rekonq::NewTab); if (!mainWindow()->isActiveWindow()) KWindowSystem::demandAttention(mainWindow()->winId(), true); } else loadUrl(urlList.at(0), Rekonq::NewWindow); } for (int i = 1; i < urlList.count(); ++i) loadUrl(urlList.at(i), Rekonq::NewTab); KStartupInfo::appStarted(); } else { if (isFirstLoad && !isRekonqCrashed) // we are starting rekonq, for the first time with no args: use startup behaviour { switch (ReKonfig::startupBehaviour()) { case 0: // open home page newMainWindow()->homePage(); break; case 1: // open new tab page loadUrl(KUrl("about:home"), Rekonq::NewWindow); break; case 2: // restore session sessionManager()->restoreSession(); kDebug() << "session restored following settings"; break; default: newMainWindow()->homePage(); break; } } else if (!isFirstLoad) // rekonq has just been started. Just open a new window { switch (ReKonfig::newTabsBehaviour()) { case 0: // new tab page loadUrl(KUrl("about:home") , Rekonq::NewWindow); break; case 1: // blank page loadUrl(KUrl("about:blank") , Rekonq::NewWindow); break; case 2: // homepage loadUrl(KUrl(ReKonfig::homePage()) , Rekonq::NewWindow); break; default: loadUrl(KUrl("about:blank") , Rekonq::NewWindow); break; } } } if (isFirstLoad) { // give me some time to do the other things.. QTimer::singleShot(100, this, SLOT(postLaunch())); } return exitValue; } Application *Application::instance() { return (qobject_cast<Application *>(QCoreApplication::instance())); } void Application::postLaunch() { // updating rekonq configuration updateConfiguration(); setWindowIcon(KIcon("rekonq")); historyManager(); sessionManager()->setSessionManagementEnabled(true); // bookmarks loading connect(bookmarkProvider(), SIGNAL(openUrl(const KUrl&, const Rekonq::OpenType&)), instance(), SLOT(loadUrl(const KUrl&, const Rekonq::OpenType&))); // crash recovering ReKonfig::setRecoverOnCrash(ReKonfig::recoverOnCrash() + 1); saveConfiguration(); } void Application::saveConfiguration() const { ReKonfig::self()->writeConfig(); } MainWindow *Application::mainWindow() { MainWindow *active = qobject_cast<MainWindow*>(QApplication::activeWindow()); if (!active) { if(m_mainWindows.isEmpty()) return 0; Q_FOREACH (const QWeakPointer<MainWindow> &pointer, m_mainWindows) { if (KWindowInfo(pointer.data()->effectiveWinId(),NET::WMDesktop,0).isOnCurrentDesktop()) return pointer.data(); } return m_mainWindows.at(0).data(); } return active; } HistoryManager *Application::historyManager() { if (m_historyManager.isNull()) { m_historyManager = new HistoryManager; } return m_historyManager.data(); } BookmarkProvider *Application::bookmarkProvider() { if (m_bookmarkProvider.isNull()) { m_bookmarkProvider = new BookmarkProvider; } return m_bookmarkProvider.data(); } SessionManager *Application::sessionManager() { if (m_sessionManager.isNull()) { m_sessionManager = new SessionManager; } return m_sessionManager.data(); } OpenSearchManager *Application::opensearchManager() { if (m_opensearchManager.isNull()) { m_opensearchManager = new OpenSearchManager; } return m_opensearchManager.data(); } IconManager *Application::iconManager() { if (m_iconManager.isNull()) { m_iconManager = new IconManager; } return m_iconManager.data(); } void Application::loadUrl(const KUrl& url, const Rekonq::OpenType& type) { if (url.isEmpty()) return; if (!url.isValid()) { KMessageBox::error(0, i18n("Malformed URL:\n%1", url.url(KUrl::RemoveTrailingSlash))); return; } // first, create the webview(s) to not let hangs UI.. WebTab *tab = 0; MainWindow *w = 0; w = (type == Rekonq::NewWindow) ? newMainWindow() : mainWindow(); switch (type) { case Rekonq::NewTab: if (ReKonfig::openTabNoWindow()) tab = w->mainView()->newWebTab(!ReKonfig::openTabsBack()); else { w = newMainWindow(); tab = w->mainView()->currentWebTab(); } break; case Rekonq::NewFocusedTab: tab = w->mainView()->newWebTab(true); break; case Rekonq::NewBackTab: tab = w->mainView()->newWebTab(false); break; case Rekonq::NewWindow: case Rekonq::CurrentTab: tab = w->mainView()->currentWebTab(); break; }; // rapidly show first loading url.. int tabIndex = w->mainView()->indexOf(tab); Q_ASSERT(tabIndex != -1); UrlBar *barForTab = qobject_cast<UrlBar *>(w->mainView()->widgetBar()->widget(tabIndex)); barForTab->activateSuggestions(false); barForTab->setQUrl(url); WebView *view = tab->view(); if (view) { FilterUrlJob *job = new FilterUrlJob(view, url.pathOrUrl(), this); Weaver::instance()->enqueue(job); } } MainWindow *Application::newMainWindow(bool withTab) { MainWindow *w = new MainWindow(); // This is used to track which window was activated most recently w->installEventFilter(this); if (withTab) w->mainView()->newWebTab(); // remember using newWebTab and NOT newTab here!! m_mainWindows.prepend(w); w->show(); return w; } void Application::removeMainWindow(MainWindow *window) { m_mainWindows.removeOne(window); } MainWindowList Application::mainWindowList() { return m_mainWindows; } AdBlockManager *Application::adblockManager() { if (m_adblockManager.isNull()) { m_adblockManager = new AdBlockManager; } return m_adblockManager.data(); } void Application::loadResolvedUrl(ThreadWeaver::Job *job) { FilterUrlJob *threadedJob = static_cast<FilterUrlJob *>(job); KUrl url = threadedJob->url(); WebView *view = threadedJob->view(); if (view) { view->load(url); } // Bye and thanks :) delete threadedJob; } void Application::newWindow() { loadUrl(KUrl("about:home"), Rekonq::NewWindow); mainWindow()->mainView()->currentUrlBar()->setFocus(); } bool Application::eventFilter( QObject* watched, QEvent* event ) { // Track which window was activated most recently to prefer it on window choosing // (e.g. when another application opens a link) if (event->type() == QEvent::WindowActivate) { MainWindow *window = qobject_cast<MainWindow*>(watched); if (window) { if (m_mainWindows.at(0).data() != window) { int index = m_mainWindows.indexOf(QWeakPointer<MainWindow>(window)); Q_ASSERT(index != -1); m_mainWindows.prepend(m_mainWindows.takeAt(index)); } } } return QObject::eventFilter( watched, event ); } void Application::updateConfiguration() { // ============== Tabs ================== bool b = ReKonfig::closeTabSelectPrevious(); Q_FOREACH(const QWeakPointer<MainWindow> &w, m_mainWindows) { MainView *mv = w.data()->mainView(); mv->updateTabBar(); mv->tabBar()->setAnimatedTabHighlighting( ReKonfig::animatedTabHighlighting() ); if (b) mv->tabBar()->setSelectionBehaviorOnRemove(QTabBar::SelectPreviousTab); else mv->tabBar()->setSelectionBehaviorOnRemove(QTabBar::SelectRightTab); } QWebSettings *defaultSettings = QWebSettings::globalSettings(); // =========== Fonts ============== defaultSettings->setFontFamily(QWebSettings::StandardFont, ReKonfig::standardFontFamily()); defaultSettings->setFontFamily(QWebSettings::FixedFont, ReKonfig::fixedFontFamily()); defaultSettings->setFontFamily(QWebSettings::SerifFont, ReKonfig::serifFontFamily()); defaultSettings->setFontFamily(QWebSettings::SansSerifFont, ReKonfig::sansSerifFontFamily()); defaultSettings->setFontFamily(QWebSettings::CursiveFont, ReKonfig::cursiveFontFamily()); defaultSettings->setFontFamily(QWebSettings::FantasyFont, ReKonfig::fantasyFontFamily()); // compute font size // (I have to admit I know nothing about these DPI questions..: copied from kwebkitpart, as someone suggested) // font size in pixels = font size in inches × screen dpi int defaultFontSize = ReKonfig::defaultFontSize(); int minimumFontSize = ReKonfig::minFontSize(); int logDpiY = mainWindow()->currentTab()->view()->logicalDpiY(); kDebug() << "Logical Dot per Inch Y: " << logDpiY; float toPix = (logDpiY < 96.0) ? 96.0 / 72.0 : logDpiY / 72.0 ; defaultSettings->setFontSize(QWebSettings::DefaultFontSize, qRound(defaultFontSize * toPix)); defaultSettings->setFontSize(QWebSettings::MinimumFontSize, qRound(minimumFontSize * toPix)); // encodings QString enc = ReKonfig::defaultEncoding(); defaultSettings->setDefaultTextEncoding(enc); // ================ WebKit ============================ defaultSettings->setAttribute(QWebSettings::AutoLoadImages, ReKonfig::autoLoadImages()); defaultSettings->setAttribute(QWebSettings::DnsPrefetchEnabled, ReKonfig::dnsPrefetch()); defaultSettings->setAttribute(QWebSettings::JavascriptEnabled, ReKonfig::javascriptEnabled()); defaultSettings->setAttribute(QWebSettings::JavaEnabled, ReKonfig::javaEnabled()); defaultSettings->setAttribute(QWebSettings::JavascriptCanOpenWindows, ReKonfig::javascriptCanOpenWindows()); defaultSettings->setAttribute(QWebSettings::JavascriptCanAccessClipboard, ReKonfig::javascriptCanAccessClipboard()); defaultSettings->setAttribute(QWebSettings::LinksIncludedInFocusChain, ReKonfig::linksIncludedInFocusChain()); defaultSettings->setAttribute(QWebSettings::ZoomTextOnly, ReKonfig::zoomTextOnly()); defaultSettings->setAttribute(QWebSettings::PrintElementBackgrounds, ReKonfig::printElementBackgrounds()); if (ReKonfig::pluginsEnabled() == 2) defaultSettings->setAttribute(QWebSettings::PluginsEnabled, false); else defaultSettings->setAttribute(QWebSettings::PluginsEnabled, true); // Enabling WebKit "Page Cache" feature: http://webkit.org/blog/427/webkit-page-cache-i-the-basics/ defaultSettings->setMaximumPagesInCache(3); // ===== HTML 5 features WebKit support ====== defaultSettings->setAttribute(QWebSettings::OfflineStorageDatabaseEnabled, ReKonfig::offlineStorageDatabaseEnabled()); defaultSettings->setAttribute(QWebSettings::OfflineWebApplicationCacheEnabled, ReKonfig::offlineWebApplicationCacheEnabled()); defaultSettings->setAttribute(QWebSettings::LocalStorageEnabled, ReKonfig::localStorageEnabled()); if (ReKonfig::localStorageEnabled()) { QString path = KStandardDirs::locateLocal("cache", QString("WebkitLocalStorage/rekonq"), true); path.remove("rekonq"); QWebSettings::setOfflineStoragePath(path); QWebSettings::setOfflineStorageDefaultQuota(50000); } // Applies user/system defined CSS to all open webpages. ReKonfig::userCSS().isEmpty() ? defaultSettings->setUserStyleSheetUrl(KUrl(KStandardDirs::locate("appdata" , "default_rekonq.css"))) : defaultSettings->setUserStyleSheetUrl(ReKonfig::userCSS()) ; // ====== load Settings on main classes historyManager()->loadSettings(); adblockManager()->loadSettings(); if (!ReKonfig::useFavicon()) mainWindow()->setWindowIcon(KIcon("rekonq")); else mainWindow()->changeWindowIcon(mainWindow()->mainView()->currentIndex()); // hovering unfocused tabs options switch (ReKonfig::hoveringTabOption()) { case 0: // tab previews case 3: // nothing for (int i = 0; i < mainWindow()->mainView()->tabBar()->count(); i++) { mainWindow()->mainView()->tabBar()->setTabToolTip(i, ""); } break; case 1: // title previews for (int i = 0; i < mainWindow()->mainView()->tabBar()->count(); i++) { mainWindow()->mainView()->tabBar()->setTabToolTip(i, mainWindow()->mainView()->tabText(i).remove('&')); } break; case 2: // url previews for (int i = 0; i < mainWindow()->mainView()->tabBar()->count(); i++) { mainWindow()->mainView()->tabBar()->setTabToolTip(i, mainWindow()->mainView()->webTab(i)->url().toMimeDataString()); } break; default: // non extant case break; } defaultSettings = 0; } void Application::addDownload(const QString &srcUrl, const QString &destUrl) { QWebSettings *globalSettings = QWebSettings::globalSettings(); if (globalSettings->testAttribute(QWebSettings::PrivateBrowsingEnabled)) return; QString downloadFilePath = KStandardDirs::locateLocal("appdata" , "downloads"); QFile downloadFile(downloadFilePath); if (!downloadFile.open(QFile::WriteOnly | QFile::Append)) { kDebug() << "Unable to open download file (WRITE mode).."; return; } QDataStream out(&downloadFile); out << srcUrl; out << destUrl; out << QDateTime::currentDateTime(); downloadFile.close(); } DownloadList Application::downloads() { DownloadList list; QString downloadFilePath = KStandardDirs::locateLocal("appdata" , "downloads"); QFile downloadFile(downloadFilePath); if (!downloadFile.open(QFile::ReadOnly)) { kDebug() << "Unable to open download file (READ mode).."; return list; } QDataStream in(&downloadFile); while (!in.atEnd()) { QString srcUrl; in >> srcUrl; QString destUrl; in >> destUrl; QDateTime dt; in >> dt; DownloadItem item(srcUrl, destUrl, dt); list << item; } return list; } bool Application::clearDownloadsHistory() { QString downloadFilePath = KStandardDirs::locateLocal("appdata" , "downloads"); QFile downloadFile(downloadFilePath); return downloadFile.remove(); } void Application::setPrivateBrowsingMode(bool b) { // NOTE // to let work nicely Private Browsing, we need the following: // - enable WebKit Private Browsing mode :) // - treat all cookies as session cookies // (so that they do not get saved to a persistent storage). Available from KDE SC 4.5.72, see BUG: 250122 // - favicons (fixed in rekonq 0.5.87) // - save actual session (to restore it when Private Mode is closed) and stop storing it // - disable history saving QWebSettings *settings = QWebSettings::globalSettings(); bool isJustEnabled = settings->testAttribute(QWebSettings::PrivateBrowsingEnabled); if (isJustEnabled == b) return; // uhm... something goes wrong... if (b) { QString caption = i18n("Are you sure you want to turn on private browsing?"); QString text = i18n("<b>%1</b>" "<p>rekonq will save your current tabs for when you'll stop private browsing the net.</p>", caption); int button = KMessageBox::warningContinueCancel(mainWindow(), text, caption, KStandardGuiItem::cont(), KStandardGuiItem::cancel(), i18n("don't ask again")); if (button != KMessageBox::Continue) return; settings->setAttribute(QWebSettings::PrivateBrowsingEnabled, true); _privateBrowsingAction->setChecked(true); Q_FOREACH(const QWeakPointer<MainWindow> &w, m_mainWindows) { w.data()->close(); } loadUrl(KUrl("about:home"), Rekonq::NewWindow); } else { Q_FOREACH(const QWeakPointer<MainWindow> &w, m_mainWindows) { w.data()->close(); } settings->setAttribute(QWebSettings::PrivateBrowsingEnabled, false); _privateBrowsingAction->setChecked(false); loadUrl(KUrl("about:blank"), Rekonq::NewWindow); if (!sessionManager()->restoreSession()) loadUrl(KUrl("about:home"), Rekonq::NewWindow); } }