505 lines
16 KiB
C++
Executable file
505 lines
16 KiB
C++
Executable file
// For license of this file, see <project-root-folder>/LICENSE.md.
|
|
|
|
#include "miscellaneous/application.h"
|
|
|
|
#include "dynamic-shortcuts/dynamicshortcuts.h"
|
|
#include "exceptions/applicationexception.h"
|
|
#include "gui/dialogs/formabout.h"
|
|
#include "gui/dialogs/formmain.h"
|
|
#include "gui/feedmessageviewer.h"
|
|
#include "gui/feedsview.h"
|
|
#include "gui/messagebox.h"
|
|
#include "gui/statusbar.h"
|
|
#include "miscellaneous/debugging.h"
|
|
#include "miscellaneous/feedreader.h"
|
|
#include "miscellaneous/iconfactory.h"
|
|
#include "miscellaneous/iofactory.h"
|
|
#include "miscellaneous/mutex.h"
|
|
#include "network-web/webfactory.h"
|
|
#include "services/abstract/serviceroot.h"
|
|
#include "services/owncloud/owncloudserviceentrypoint.h"
|
|
#include "services/standard/standardserviceentrypoint.h"
|
|
#include "services/standard/standardserviceroot.h"
|
|
#include "services/tt-rss/ttrssserviceentrypoint.h"
|
|
|
|
#include <QProcess>
|
|
#include <QSessionManager>
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
#include "network-web/adblock/adblockicon.h"
|
|
#include "network-web/adblock/adblockmanager.h"
|
|
#include "network-web/networkurlinterceptor.h"
|
|
#include "network-web/rssguardschemehandler.h"
|
|
#include "network-web/urlinterceptor.h"
|
|
|
|
#include <QWebEngineDownloadItem>
|
|
#include <QWebEngineProfile>
|
|
#include <QWebEngineScript>
|
|
#include <QWebEngineScriptCollection>
|
|
#endif
|
|
|
|
Application::Application(const QString& id, int& argc, char** argv)
|
|
: QtSingleApplication(id, argc, argv),
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
m_urlInterceptor(new NetworkUrlInterceptor(this)),
|
|
#endif
|
|
|
|
m_feedReader(nullptr),
|
|
m_updateFeedsLock(new Mutex()), m_mainForm(nullptr),
|
|
m_trayIcon(nullptr), m_settings(Settings::setupSettings(this)), m_webFactory(new WebFactory(this)),
|
|
m_system(new SystemFactory(this)), m_skins(new SkinFactory(this)),
|
|
m_localization(new Localization(this)), m_icons(new IconFactory(this)),
|
|
m_database(new DatabaseFactory(this)), m_downloadManager(nullptr), m_shouldRestart(false) {
|
|
|
|
// Setup debug output system.
|
|
qInstallMessageHandler(Debugging::debugHandler);
|
|
|
|
//: Abbreviation of language, e.g. en.
|
|
//: Use ISO 639-1 code here combined with ISO 3166-1 (alpha-2) code.
|
|
//: Examples: "cs", "en", "it", "cs_CZ", "en_GB", "en_US".
|
|
QObject::tr("LANG_ABBREV");
|
|
|
|
//: Name of translator - optional.
|
|
QObject::tr("LANG_AUTHOR");
|
|
|
|
connect(this, &Application::aboutToQuit, this, &Application::onAboutToQuit);
|
|
connect(this, &Application::commitDataRequest, this, &Application::onCommitData);
|
|
connect(this, &Application::saveStateRequest, this, &Application::onSaveState);
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
connect(QWebEngineProfile::defaultProfile(), &QWebEngineProfile::downloadRequested, this, &Application::downloadRequested);
|
|
|
|
QWebEngineProfile::defaultProfile()->setRequestInterceptor(m_urlInterceptor);
|
|
m_urlInterceptor->loadSettings();
|
|
QWebEngineProfile::defaultProfile()->installUrlSchemeHandler(QByteArray(APP_LOW_NAME),
|
|
new RssGuardSchemeHandler(QWebEngineProfile::defaultProfile()));
|
|
#endif
|
|
|
|
if (arguments().contains(QL1S("-log"))) {
|
|
Debugging::instance()->setTargetFile(IOFactory::getSystemFolder(QStandardPaths::TempLocation) +
|
|
QDir::separator() + QL1S("rssguard.log"));
|
|
}
|
|
|
|
m_webFactory->updateProxy();
|
|
}
|
|
|
|
Application::~Application() {
|
|
qDebug("Destroying Application instance.");
|
|
}
|
|
|
|
void Application::reactOnForeignNotifications() {
|
|
connect(this, &Application::messageReceived, this, &Application::processExecutionMessage);
|
|
}
|
|
|
|
void Application::hideOrShowMainForm() {
|
|
// Display main window.
|
|
if (qApp->settings()->value(GROUP(GUI), SETTING(GUI::MainWindowStartsHidden)).toBool() && SystemTrayIcon::isSystemTrayActivated()) {
|
|
qDebug("Hiding the main window when the application is starting.");
|
|
mainForm()->switchVisibility(true);
|
|
}
|
|
else {
|
|
qDebug("Showing the main window when the application is starting.");
|
|
mainForm()->show();
|
|
}
|
|
}
|
|
|
|
void Application::loadDynamicShortcuts() {
|
|
DynamicShortcuts::load(userActions());
|
|
}
|
|
|
|
void Application::showPolls() const {
|
|
if (isFirstRun(APP_VERSION)) {
|
|
//web()->openUrlInExternalBrowser(QSL("https://goo.gl/forms/7bJNr33Ii22Q1c3k2"));
|
|
}
|
|
}
|
|
|
|
void Application::offerChanges() const {
|
|
if (isFirstRun() || isFirstRun(APP_VERSION)) {
|
|
qApp->showGuiMessage(QSL(APP_NAME), QObject::tr("Welcome to %1.\n\nPlease, check NEW stuff included in this\n"
|
|
"version by clicking this popup notification.").arg(APP_LONG_NAME),
|
|
QSystemTrayIcon::NoIcon, nullptr, false, [] {
|
|
FormAbout(qApp->mainForm()).exec();
|
|
});
|
|
}
|
|
}
|
|
|
|
bool Application::isAlreadyRunning() {
|
|
return sendMessage((QStringList() << APP_IS_RUNNING << Application::arguments().mid(1)).join(ARGUMENTS_LIST_SEPARATOR));
|
|
}
|
|
|
|
FeedReader* Application::feedReader() {
|
|
return m_feedReader;
|
|
}
|
|
|
|
QList<QAction*> Application::userActions() {
|
|
if (m_mainForm != nullptr && m_userActions.isEmpty()) {
|
|
m_userActions = m_mainForm->allActions();
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
m_userActions.append(AdBlockManager::instance()->adBlockIcon());
|
|
#endif
|
|
}
|
|
|
|
return m_userActions;
|
|
}
|
|
|
|
bool Application::isFirstRun() const {
|
|
return settings()->value(GROUP(General), SETTING(General::FirstRun)).toBool();
|
|
}
|
|
|
|
bool Application::isFirstRun(const QString& version) const {
|
|
if (version == APP_VERSION) {
|
|
// Check this only if checked version is equal to actual version.
|
|
return settings()->value(GROUP(General), QString(General::FirstRun) + QL1C('_') + version, true).toBool();
|
|
}
|
|
else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
WebFactory* Application::web() const {
|
|
return m_webFactory;
|
|
}
|
|
|
|
SystemFactory* Application::system() {
|
|
return m_system;
|
|
}
|
|
|
|
SkinFactory* Application::skins() {
|
|
return m_skins;
|
|
}
|
|
|
|
Localization* Application::localization() {
|
|
return m_localization;
|
|
}
|
|
|
|
DatabaseFactory* Application::database() {
|
|
return m_database;
|
|
}
|
|
|
|
void Application::eliminateFirstRun() {
|
|
settings()->setValue(GROUP(General), General::FirstRun, false);
|
|
}
|
|
|
|
void Application::eliminateFirstRun(const QString& version) {
|
|
settings()->setValue(GROUP(General), QString(General::FirstRun) + QL1C('_') + version, false);
|
|
}
|
|
|
|
void Application::setFeedReader(FeedReader* feed_reader) {
|
|
m_feedReader = feed_reader;
|
|
connect(m_feedReader, &FeedReader::feedUpdatesStarted, this, &Application::onFeedUpdatesStarted);
|
|
connect(m_feedReader, &FeedReader::feedUpdatesProgress, this, &Application::onFeedUpdatesProgress);
|
|
connect(m_feedReader, &FeedReader::feedUpdatesFinished, this, &Application::onFeedUpdatesFinished);
|
|
}
|
|
|
|
IconFactory* Application::icons() {
|
|
return m_icons;
|
|
}
|
|
|
|
DownloadManager* Application::downloadManager() {
|
|
if (m_downloadManager == nullptr) {
|
|
m_downloadManager = new DownloadManager();
|
|
connect(m_downloadManager, &DownloadManager::downloadFinished, mainForm()->statusBar(), &StatusBar::clearProgressDownload);
|
|
connect(m_downloadManager, &DownloadManager::downloadProgressed, mainForm()->statusBar(), &StatusBar::showProgressDownload);
|
|
}
|
|
|
|
return m_downloadManager;
|
|
}
|
|
|
|
Settings* Application::settings() const {
|
|
return m_settings;
|
|
}
|
|
|
|
Mutex* Application::feedUpdateLock() {
|
|
return m_updateFeedsLock.data();
|
|
}
|
|
|
|
FormMain* Application::mainForm() {
|
|
return m_mainForm;
|
|
}
|
|
|
|
QWidget* Application::mainFormWidget() {
|
|
return m_mainForm;
|
|
}
|
|
|
|
void Application::setMainForm(FormMain* main_form) {
|
|
m_mainForm = main_form;
|
|
}
|
|
|
|
QString Application::configFolder() {
|
|
return IOFactory::getSystemFolder(QStandardPaths::GenericConfigLocation);
|
|
}
|
|
|
|
QString Application::userDataAppFolder() {
|
|
// In "app" folder, we would like to separate all user data into own subfolder,
|
|
// therefore stick to "data" folder in this mode.
|
|
return applicationDirPath() + QDir::separator() + QSL("data");
|
|
}
|
|
|
|
QString Application::userDataFolder() {
|
|
if (settings()->type() == SettingsProperties::Portable) {
|
|
return userDataAppFolder();
|
|
}
|
|
else {
|
|
return userDataHomeFolder();
|
|
}
|
|
}
|
|
|
|
QString Application::userDataHomeFolder() {
|
|
// Fallback folder.
|
|
const QString home_folder = homeFolder() + QDir::separator() + QSL(APP_LOW_H_NAME) + QDir::separator() + QSL("data");
|
|
|
|
if (QDir().exists(home_folder)) {
|
|
return home_folder;
|
|
}
|
|
else {
|
|
#if defined (Q_OS_ANDROID)
|
|
return IOFactory::getSystemFolder(QStandardPaths::GenericDataLocation) + QDir::separator() + QSL(APP_NAME);
|
|
#else
|
|
return configFolder() + QDir::separator() + QSL(APP_NAME);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
QString Application::tempFolder() {
|
|
return IOFactory::getSystemFolder(QStandardPaths::TempLocation);
|
|
}
|
|
|
|
QString Application::documentsFolder() {
|
|
return IOFactory::getSystemFolder(QStandardPaths::DocumentsLocation);
|
|
}
|
|
|
|
QString Application::homeFolder() {
|
|
#if defined (Q_OS_ANDROID)
|
|
return IOFactory::getSystemFolder(QStandardPaths::GenericDataLocation);
|
|
#else
|
|
return IOFactory::getSystemFolder(QStandardPaths::HomeLocation);
|
|
#endif
|
|
}
|
|
|
|
void Application::backupDatabaseSettings(bool backup_database, bool backup_settings,
|
|
const QString& target_path, const QString& backup_name) {
|
|
if (!QFileInfo(target_path).isWritable()) {
|
|
throw ApplicationException(tr("Output directory is not writable."));
|
|
}
|
|
|
|
if (backup_settings) {
|
|
settings()->sync();
|
|
|
|
if (!IOFactory::copyFile(settings()->fileName(), target_path + QDir::separator() + backup_name + BACKUP_SUFFIX_SETTINGS)) {
|
|
throw ApplicationException(tr("Settings file not copied to output directory successfully."));
|
|
}
|
|
}
|
|
|
|
if (backup_database &&
|
|
(database()->activeDatabaseDriver() == DatabaseFactory::UsedDriver::SQLITE ||
|
|
database()->activeDatabaseDriver() == DatabaseFactory::UsedDriver::SQLITE_MEMORY)) {
|
|
// We need to save the database first.
|
|
database()->saveDatabase();
|
|
|
|
if (!IOFactory::copyFile(database()->sqliteDatabaseFilePath(),
|
|
target_path + QDir::separator() + backup_name + BACKUP_SUFFIX_DATABASE)) {
|
|
throw ApplicationException(tr("Database file not copied to output directory successfully."));
|
|
}
|
|
}
|
|
}
|
|
|
|
void Application::restoreDatabaseSettings(bool restore_database, bool restore_settings,
|
|
const QString& source_database_file_path, const QString& source_settings_file_path) {
|
|
if (restore_database) {
|
|
if (!qApp->database()->initiateRestoration(source_database_file_path)) {
|
|
throw ApplicationException(tr("Database restoration was not initiated. Make sure that output directory is writable."));
|
|
}
|
|
}
|
|
|
|
if (restore_settings) {
|
|
if (!qApp->settings()->initiateRestoration(source_settings_file_path)) {
|
|
throw ApplicationException(tr("Settings restoration was not initiated. Make sure that output directory is writable."));
|
|
}
|
|
}
|
|
}
|
|
|
|
void Application::processExecutionMessage(const QString& message) {
|
|
qDebug("Received '%s' execution message from another application instance.", qPrintable(message));
|
|
const QStringList messages = message.split(ARGUMENTS_LIST_SEPARATOR);
|
|
|
|
if (messages.contains(APP_QUIT_INSTANCE)) {
|
|
quit();
|
|
}
|
|
else {
|
|
foreach (const QString& msg, messages) {
|
|
if (msg == APP_IS_RUNNING) {
|
|
showGuiMessage(APP_NAME, tr("Application is already running."), QSystemTrayIcon::Information);
|
|
mainForm()->display();
|
|
}
|
|
else if (msg.startsWith(QL1S(URI_SCHEME_FEED_SHORT))) {
|
|
// Application was running, and someone wants to add new feed.
|
|
StandardServiceRoot* root = qApp->feedReader()->feedsModel()->standardServiceRoot();
|
|
|
|
if (root != nullptr) {
|
|
root->checkArgumentForFeedAdding(msg);
|
|
}
|
|
else {
|
|
showGuiMessage(tr("Cannot add feed"),
|
|
tr("Feed cannot be added because standard RSS/ATOM account is not enabled."),
|
|
QSystemTrayIcon::Warning, qApp->mainForm(),
|
|
true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
SystemTrayIcon* Application::trayIcon() {
|
|
if (m_trayIcon == nullptr) {
|
|
if (qApp->settings()->value(GROUP(GUI), SETTING(GUI::MonochromeTrayIcon)).toBool()) {
|
|
m_trayIcon = new SystemTrayIcon(APP_ICON_MONO_PATH, APP_ICON_MONO_PLAIN_PATH, m_mainForm);
|
|
}
|
|
else {
|
|
m_trayIcon = new SystemTrayIcon(APP_ICON_PATH, APP_ICON_PLAIN_PATH, m_mainForm);
|
|
}
|
|
|
|
connect(m_trayIcon, &SystemTrayIcon::shown, m_feedReader->feedsModel(), &FeedsModel::notifyWithCounts);
|
|
connect(m_feedReader->feedsModel(), &FeedsModel::messageCountsChanged, m_trayIcon, &SystemTrayIcon::setNumber);
|
|
}
|
|
|
|
return m_trayIcon;
|
|
}
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
NetworkUrlInterceptor* Application::urlIinterceptor() {
|
|
return m_urlInterceptor;
|
|
}
|
|
|
|
#endif
|
|
|
|
void Application::showTrayIcon() {
|
|
// Display tray icon if it is enabled and available.
|
|
if (SystemTrayIcon::isSystemTrayActivated()) {
|
|
qDebug("Showing tray icon.");
|
|
trayIcon()->show();
|
|
}
|
|
}
|
|
|
|
void Application::deleteTrayIcon() {
|
|
if (m_trayIcon != nullptr) {
|
|
qDebug("Disabling tray icon, deleting it and raising main application window.");
|
|
m_mainForm->display();
|
|
delete m_trayIcon;
|
|
m_trayIcon = nullptr;
|
|
|
|
// Make sure that application quits when last window is closed.
|
|
setQuitOnLastWindowClosed(true);
|
|
}
|
|
}
|
|
|
|
void Application::showGuiMessage(const QString& title, const QString& message,
|
|
QSystemTrayIcon::MessageIcon message_type, QWidget* parent,
|
|
bool show_at_least_msgbox, std::function<void()> functor) {
|
|
if (SystemTrayIcon::areNotificationsEnabled() && SystemTrayIcon::isSystemTrayActivated()) {
|
|
trayIcon()->showMessage(title, message, message_type, TRAY_ICON_BUBBLE_TIMEOUT, std::move(functor));
|
|
}
|
|
else if (show_at_least_msgbox) {
|
|
// Tray icon or OSD is not available, display simple text box.
|
|
MessageBox::show(parent, QMessageBox::Icon(message_type), title, message);
|
|
}
|
|
else {
|
|
qDebug("Silencing GUI message: '%s'.", qPrintable(message));
|
|
}
|
|
}
|
|
|
|
void Application::onCommitData(QSessionManager& manager) {
|
|
qDebug("OS asked application to commit its data.");
|
|
manager.setRestartHint(QSessionManager::RestartNever);
|
|
manager.release();
|
|
}
|
|
|
|
void Application::onSaveState(QSessionManager& manager) {
|
|
qDebug("OS asked application to save its state.");
|
|
manager.setRestartHint(QSessionManager::RestartNever);
|
|
manager.release();
|
|
}
|
|
|
|
void Application::onAboutToQuit() {
|
|
eliminateFirstRun();
|
|
eliminateFirstRun(APP_VERSION);
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
AdBlockManager::instance()->save();
|
|
#endif
|
|
|
|
// Make sure that we obtain close lock BEFORE even trying to quit the application.
|
|
const bool locked_safely = feedUpdateLock()->tryLock(4 * CLOSE_LOCK_TIMEOUT);
|
|
|
|
processEvents();
|
|
qDebug("Cleaning up resources and saving application state.");
|
|
|
|
#if defined(Q_OS_WIN)
|
|
system()->removeTrolltechJunkRegistryKeys();
|
|
#endif
|
|
|
|
qApp->feedReader()->quit();
|
|
database()->saveDatabase();
|
|
|
|
if (mainForm() != nullptr) {
|
|
mainForm()->saveSize();
|
|
}
|
|
|
|
if (locked_safely) {
|
|
// Application obtained permission to close in a safe way.
|
|
qDebug("Close lock was obtained safely.");
|
|
|
|
// We locked the lock to exit peacefully, unlock it to avoid warnings.
|
|
feedUpdateLock()->unlock();
|
|
}
|
|
else {
|
|
// Request for write lock timed-out. This means
|
|
// that some critical action can be processed right now.
|
|
qDebug("Close lock timed-out.");
|
|
}
|
|
|
|
// Now, we can check if application should just quit or restart itself.
|
|
if (m_shouldRestart) {
|
|
finish();
|
|
qDebug("Killing local peer connection to allow another instance to start.");
|
|
|
|
// TODO: Start RSS Guard with sleep before it cross-platform way if possible.
|
|
// sleep 5 && "<rssguard-start>".
|
|
if (QProcess::startDetached(QString("\"") + QDir::toNativeSeparators(applicationFilePath()) + QString("\""))) {
|
|
qDebug("New application instance was started.");
|
|
}
|
|
else {
|
|
qWarning("New application instance was not started successfully.");
|
|
}
|
|
}
|
|
}
|
|
|
|
void Application::restart() {
|
|
m_shouldRestart = true;
|
|
quit();
|
|
}
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
void Application::downloadRequested(QWebEngineDownloadItem* download_item) {
|
|
downloadManager()->download(download_item->url());
|
|
download_item->cancel();
|
|
download_item->deleteLater();
|
|
}
|
|
|
|
#endif
|
|
|
|
void Application::onFeedUpdatesStarted() {}
|
|
|
|
void Application::onFeedUpdatesProgress(const Feed* feed, int current, int total) {
|
|
Q_UNUSED(feed)
|
|
Q_UNUSED(current)
|
|
Q_UNUSED(total)
|
|
}
|
|
|
|
void Application::onFeedUpdatesFinished(const FeedDownloadResults& results) {
|
|
if (!results.updatedFeeds().isEmpty()) {
|
|
// Now, inform about results via GUI message/notification.
|
|
qApp->showGuiMessage(tr("New messages downloaded"), results.overview(10), QSystemTrayIcon::NoIcon, nullptr, false);
|
|
}
|
|
}
|