/* * Copyright (C) by Duncan Mac-Vicar P. * Copyright (C) by Klaas Freitag * Copyright (C) by Daniel Molkentin * * 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) 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. */ #include #include "mirall/application.h" #include "mirall/systray.h" #include "mirall/folder.h" #include "mirall/folderman.h" #include "mirall/folderwatcher.h" #include "mirall/folderwizard.h" #include "mirall/networklocation.h" #include "mirall/folder.h" #include "mirall/owncloudsetupwizard.h" #include "mirall/owncloudinfo.h" #include "mirall/sslerrordialog.h" #include "mirall/theme.h" #include "mirall/mirallconfigfile.h" #include "mirall/updatedetector.h" #include "mirall/logger.h" #include "mirall/settingsdialog.h" #include "mirall/itemprogressdialog.h" #include "mirall/utility.h" #include "mirall/inotify.h" #include "mirall/connectionvalidator.h" #include "creds/abstractcredentials.h" #if defined(Q_OS_WIN) #include #endif #include #include #include #include #include #include #include #include #include namespace Mirall { // application logging handler. void mirallLogCatcher(QtMsgType type, const char *msg) { Q_UNUSED(type) // qDebug() exports to local8Bit, which is not always UTF-8 Logger::instance()->mirallLog( QString::fromLocal8Bit(msg) ); } namespace { static const char optionsC[] = "Options:\n" " -h --help : show this help screen.\n" " --logwindow : open a window to show log output.\n" " --logfile : write log output to file .\n" " --logdir : write each sync log output in a new file\n" " in directory .\n" " --logexpire : removes logs older than hours.\n" " (to be used with --logdir)\n" " --logflush : flush the log file after every write.\n" " --confdir : Use the given configuration directory.\n" ; QString applicationTrPath() { #ifdef Q_OS_LINUX // FIXME - proper path! return QString::fromLatin1("/usr/share/%1/i18n/").arg(Theme::instance()->appName()); #endif #ifdef Q_OS_MAC return QApplication::applicationDirPath()+QLatin1String("/../Resources/Translations"); // path defaults to app dir. #endif #ifdef Q_OS_WIN32 return QApplication::applicationDirPath(); #endif } } // ---------------------------------------------------------------------------------- Application::Application(int &argc, char **argv) : SharedTools::QtSingleApplication(argc, argv), _tray(0), _networkMgr(new QNetworkConfigurationManager(this)), _sslErrorDialog(0), _contextMenu(0), _recentActionsMenu(0), _theme(Theme::instance()), _logBrowser(0), _logExpire(0), _showLogWindow(false), _logFlush(false), _helpOnly(false) { setApplicationName( _theme->appNameGUI() ); setWindowIcon( _theme->applicationIcon() ); parseOptions(arguments()); setupTranslations(); setupLogBrowser(); //no need to waste time; if ( _helpOnly ) return; connect( this, SIGNAL(messageReceived(QString)), SLOT(slotParseOptions(QString))); connect( Logger::instance(), SIGNAL(guiLog(QString,QString)), this, SLOT(slotShowTrayMessage(QString,QString))); connect( Logger::instance(), SIGNAL(optionalGuiLog(QString,QString)), this, SLOT(slotShowOptionalTrayMessage(QString,QString))); connect( Logger::instance(), SIGNAL(guiMessage(QString,QString)), this, SLOT(slotShowGuiMessage(QString,QString))); ProgressDispatcher *pd = ProgressDispatcher::instance(); connect( pd, SIGNAL(progressInfo(QString,Progress::Info)), this, SLOT(slotUpdateProgress(QString,Progress::Info)) ); connect( pd, SIGNAL(progressSyncProblem(QString,Progress::SyncProblem)), SLOT(slotProgressSyncProblem(QString,Progress::SyncProblem))); // create folder manager for sync folder management FolderMan *folderMan = FolderMan::instance(); connect( folderMan, SIGNAL(folderSyncStateChange(QString)), this,SLOT(slotSyncStateChange(QString))); folderMan->setSyncEnabled(false); /* use a signal mapper to map the open requests to the alias names */ _folderOpenActionMapper = new QSignalMapper(this); connect(_folderOpenActionMapper, SIGNAL(mapped(const QString &)), this, SLOT(slotFolderOpenAction(const QString &))); setQuitOnLastWindowClosed(false); qRegisterMetaType("Progress::Kind"); qRegisterMetaType("Progress::Info"); #if 0 qDebug() << "* Network is" << (_networkMgr->isOnline() ? "online" : "offline"); foreach (const QNetworkConfiguration& netCfg, _networkMgr->allConfigurations(QNetworkConfiguration::Active)) { //qDebug() << "Network:" << netCfg.identifier(); } #endif MirallConfigFile cfg; _theme->setSystrayUseMonoIcons(cfg.monoIcons()); connect (_theme, SIGNAL(systrayUseMonoIconsChanged(bool)), SLOT(slotUseMonoIconsChanged(bool))); setupActions(); setupSystemTray(); slotSetupProxy(); folderMan->setupFolders(); // startup procedure. QTimer::singleShot( 0, this, SLOT( slotCheckConnection() )); if( !cfg.ownCloudSkipUpdateCheck() ) { QTimer::singleShot( 3000, this, SLOT( slotStartUpdateDetector() )); } connect( ownCloudInfo::instance(), SIGNAL(sslFailed(QNetworkReply*, QList)), this,SLOT(slotSSLFailed(QNetworkReply*, QList))); connect( ownCloudInfo::instance(), SIGNAL(quotaUpdated(qint64,qint64)), SLOT(slotRefreshQuotaDisplay(qint64, qint64))); qDebug() << "Network Location: " << NetworkLocation::currentLocation().encoded(); } Application::~Application() { delete _tray; // needed, see ctor qDebug() << "* Mirall shutdown"; } void Application::slotStartUpdateDetector() { UpdateDetector *updateDetector = new UpdateDetector(this); updateDetector->versionCheck(_theme); } void Application::slotCheckConnection() { if( checkConfigExists(false) ) { MirallConfigFile cfg; AbstractCredentials* credentials(cfg.getCredentials()); if (! credentials->ready()) { connect( credentials, SIGNAL(fetched()), this, SLOT(slotCredentialsFetched())); credentials->fetch(); } else { runValidator(); } } else { // the call to checkConfigExists opens the setup wizard // if the config does not exist. Nothing to do here. } } void Application::slotCredentialsFetched() { MirallConfigFile cfg; AbstractCredentials* credentials(cfg.getCredentials()); disconnect(credentials, SIGNAL(fetched()), this, SLOT(slotCredentialsFetched())); runValidator(); } void Application::runValidator() { _conValidator = new ConnectionValidator(); connect( _conValidator, SIGNAL(connectionResult(ConnectionValidator::Status)), this, SLOT(slotConnectionValidatorResult(ConnectionValidator::Status)) ); _conValidator->checkConnection(); } void Application::slotConnectionValidatorResult(ConnectionValidator::Status status) { qDebug() << "Connection Validator Result: " << _conValidator->statusString(status); if( status == ConnectionValidator::Connected ) { FolderMan *folderMan = FolderMan::instance(); qDebug() << "######## Connection and Credentials are ok!"; folderMan->setSyncEnabled(true); _tray->setIcon( _theme->syncStateIcon( SyncResult::NotYetStarted, true ) ); _tray->show(); int cnt = folderMan->map().size(); slotShowTrayMessage(tr("%1 Sync Started").arg(_theme->appNameGUI()), tr("Sync started for %n configured sync folder(s).","", cnt)); // queue up the sync for all folders. folderMan->slotScheduleAllFolders(); computeOverallSyncStatus(); setupContextMenu(); } else if( status == ConnectionValidator::NotConfigured ) { // this can not happen, it should be caught in first step of startup. } else { // What else? } _conValidator->deleteLater(); } void Application::slotSSLFailed( QNetworkReply *reply, QList errors ) { qDebug() << "SSL-Warnings happened for url " << reply->url().toString(); if( ownCloudInfo::instance()->certsUntrusted() ) { // User decided once to untrust. Honor this decision. qDebug() << "Untrusted by user decision, returning."; return; } QString configHandle = ownCloudInfo::instance()->configHandle(reply); // make the ssl dialog aware of the custom config. It loads known certs. if( ! _sslErrorDialog ) { _sslErrorDialog = new SslErrorDialog; } _sslErrorDialog->setCustomConfigHandle( configHandle ); if( _sslErrorDialog->setErrorList( errors ) ) { // all ssl certs are known and accepted. We can ignore the problems right away. qDebug() << "Certs are already known and trusted, Warnings are not valid."; reply->ignoreSslErrors(); } else { if( _sslErrorDialog->exec() == QDialog::Accepted ) { if( _sslErrorDialog->trustConnection() ) { reply->ignoreSslErrors(); } else { // User does not want to trust. ownCloudInfo::instance()->setCertsUntrusted(true); } } else { ownCloudInfo::instance()->setCertsUntrusted(true); } } } void Application::slotownCloudWizardDone( int res ) { FolderMan *folderMan = FolderMan::instance(); if( res == QDialog::Accepted ) { int cnt = folderMan->setupFolders(); qDebug() << "Set up " << cnt << " folders."; // We have some sort of configuration. Enable autostart Utility::setLaunchOnStartup(_theme->appName(), _theme->appNameGUI(), true); // FIXME! // _statusDialog->setFolderList( folderMan->map() ); } folderMan->setSyncEnabled( true ); if( res == QDialog::Accepted ) { slotCheckConnection(); } } void Application::setupActions() { _actionOpenoC = new QAction(tr("Open %1 in browser").arg(_theme->appNameGUI()), this); QObject::connect(_actionOpenoC, SIGNAL(triggered(bool)), SLOT(slotOpenOwnCloud())); _actionQuota = new QAction(tr("Calculating quota..."), this); _actionQuota->setEnabled( false ); _actionStatus = new QAction(tr("Unknown status"), this); _actionStatus->setEnabled( false ); _actionSettings = new QAction(tr("Settings..."), this); _actionRecent = new QAction(tr("Details..."), this); _actionRecent->setEnabled( true ); QObject::connect(_actionRecent, SIGNAL(triggered(bool)), SLOT(slotItemProgressDialog())); QObject::connect(_actionSettings, SIGNAL(triggered(bool)), SLOT(slotSettings())); _actionHelp = new QAction(tr("Help"), this); QObject::connect(_actionHelp, SIGNAL(triggered(bool)), SLOT(slotHelp())); _actionQuit = new QAction(tr("Quit"), this); QObject::connect(_actionQuit, SIGNAL(triggered(bool)), SLOT(quit())); } void Application::setupSystemTray() { // Setting a parent heres will crash on X11 since by the time qapp runs // its childrens dtors, the X11->screen variable queried for is gone -> crash _tray = new Systray(); _tray->setIcon( _theme->syncStateIcon( SyncResult::NotYetStarted, true ) ); connect(_tray,SIGNAL(activated(QSystemTrayIcon::ActivationReason)), SLOT(slotTrayClicked(QSystemTrayIcon::ActivationReason))); setupContextMenu(); _tray->show(); } void Application::setupContextMenu() { bool isConfigured = ownCloudInfo::instance()->isConfigured(); FolderMan *folderMan = FolderMan::instance(); _actionOpenoC->setEnabled(isConfigured); if( _contextMenu ) { _contextMenu->clear(); _recentActionsMenu->clear(); _recentActionsMenu->addAction(tr("None.")); _recentActionsMenu->addAction(_actionRecent); } else { _contextMenu = new QMenu(); _recentActionsMenu = _contextMenu->addMenu(tr("Recent Changes")); // this must be called only once after creating the context menu, or // it will trigger a bug in Ubuntu's SNI bridge patch (11.10, 12.04). _tray->setContextMenu(_contextMenu); } _contextMenu->setTitle(_theme->appNameGUI() ); _contextMenu->addAction(_actionOpenoC); int folderCnt = folderMan->map().size(); // add open actions for all sync folders to the tray menu if( _theme->singleSyncFolder() ) { // there should be exactly one folder. No sync-folder add action will be shown. QStringList li = folderMan->map().keys(); if( li.size() == 1 ) { Folder *folder = folderMan->map().value(li.first()); if( folder ) { // if there is singleFolder mode, a generic open action is displayed. QAction *action = new QAction( tr("Open local folder '%1'").arg(_theme->appNameGUI()), this); connect( action, SIGNAL(triggered()),_folderOpenActionMapper,SLOT(map())); _folderOpenActionMapper->setMapping( action, folder->alias() ); _contextMenu->addAction(action); } } } else { // show a grouping with more than one folder. if ( folderCnt > 1) { _contextMenu->addAction(tr("Managed Folders:"))->setDisabled(true); } foreach (Folder *folder, folderMan->map() ) { QAction *action = new QAction( tr("Open folder '%1'").arg(folder->alias()), this ); connect( action, SIGNAL(triggered()),_folderOpenActionMapper,SLOT(map())); _folderOpenActionMapper->setMapping( action, folder->alias() ); _contextMenu->addAction(action); } } _contextMenu->addSeparator(); _contextMenu->addAction(_actionQuota); _contextMenu->addSeparator(); _contextMenu->addAction(_actionStatus); _contextMenu->addMenu(_recentActionsMenu); _contextMenu->addSeparator(); _contextMenu->addAction(_actionSettings); _contextMenu->addAction(_actionHelp); _contextMenu->addSeparator(); _contextMenu->addAction(_actionQuit); } void Application::setupLogBrowser() { // might be called from second instance if (!_logBrowser) { // init the log browser. _logBrowser = new LogBrowser; qInstallMsgHandler( mirallLogCatcher ); // ## TODO: allow new log name maybe? if (!_logDirectory.isEmpty()) { enterNextLogFile(); } else if (!_logFile.isEmpty()) { qDebug() << "Logging into logfile: " << _logFile << " with flush " << _logFlush; _logBrowser->setLogFile( _logFile, _logFlush ); } } if (_showLogWindow) slotOpenLogBrowser(); qDebug() << QString::fromLatin1( "################## %1 %2 (%3) %4").arg(_theme->appName()) .arg( QLocale::system().name() ) .arg(property("ui_lang").toString()) .arg(_theme->version()); } void Application::enterNextLogFile() { if (_logBrowser && !_logDirectory.isEmpty()) { QDir dir(_logDirectory); if (!dir.exists()) { dir.mkpath("."); } // Find out what is the file with the highest nymber if any QStringList files = dir.entryList(QStringList("owncloud.log.*"), QDir::Files); QRegExp rx("owncloud.log.(\\d+)"); uint maxNumber = 0; QDateTime now = QDateTime::currentDateTime(); foreach(const QString &s, files) { if (rx.exactMatch(s)) { maxNumber = qMax(maxNumber, rx.cap(1).toUInt()); if (_logExpire > 0) { QFileInfo fileInfo = dir.absoluteFilePath(s); if (fileInfo.lastModified().addSecs(60*60 * _logExpire) < now) { dir.remove(s); } } } } QString filename = _logDirectory + "/owncloud.log." + QString::number(maxNumber+1); _logBrowser->setLogFile(filename , _logFlush); } } QNetworkProxy proxyFromConfig(const MirallConfigFile& cfg) { QNetworkProxy proxy; if (cfg.proxyHostName().isEmpty()) return QNetworkProxy(); proxy.setHostName(cfg.proxyHostName()); proxy.setPort(cfg.proxyPort()); if (cfg.proxyNeedsAuth()) { proxy.setUser(cfg.proxyUser()); proxy.setPassword(cfg.proxyPassword()); } return proxy; } void Application::slotSetupProxy() { Mirall::MirallConfigFile cfg; int proxyType = cfg.proxyType(); QNetworkProxy proxy = proxyFromConfig(cfg); switch(proxyType) { case QNetworkProxy::NoProxy: QNetworkProxy::setApplicationProxy(QNetworkProxy::NoProxy); break; case QNetworkProxy::DefaultProxy: QNetworkProxyFactory::setUseSystemConfiguration(true); break; case QNetworkProxy::Socks5Proxy: proxy.setType(QNetworkProxy::Socks5Proxy); QNetworkProxy::setApplicationProxy(proxy); break; case QNetworkProxy::HttpProxy: proxy.setType(QNetworkProxy::HttpProxy); QNetworkProxy::setApplicationProxy(proxy); break; default: break; } } void Application::slotRefreshQuotaDisplay( qint64 total, qint64 used ) { if (total == 0) { _actionQuota->setText(tr("Quota n/a")); return; } double percent = used/(double)total*100; QString percentFormatted = Utility::compactFormatDouble(percent, 1); QString totalFormatted = Utility::octetsToString(total); _actionQuota->setText(tr("%1% of %2 used").arg(percentFormatted).arg(totalFormatted)); } void Application::slotUseMonoIconsChanged(bool) { computeOverallSyncStatus(); } void Application::slotProgressSyncProblem(const QString& folder, const Progress::SyncProblem& problem) { Q_UNUSED(folder); Q_UNUSED(problem); // display a warn icon if warnings happend. QIcon warnIcon(":/mirall/resources/warning-16"); _actionRecent->setIcon(warnIcon); rebuildRecentMenus(); } void Application::rebuildRecentMenus() { _recentActionsMenu->clear(); const QList& progressInfoList = ProgressDispatcher::instance()->recentChangedItems(5); if( progressInfoList.size() == 0 ) { _recentActionsMenu->addAction(tr("No items synced recently")); } else { QListIterator i(progressInfoList); i.toBack(); while(i.hasPrevious()) { Progress::Info info = i.previous(); QString kindStr = Progress::asResultString(info.kind); QString timeStr = info.timestamp.toString("hh:mm"); QString actionText = tr("%1 (%2, %3)").arg(info.current_file).arg(kindStr).arg(timeStr); _recentActionsMenu->addAction( actionText ); } } // add a more... entry. _recentActionsMenu->addAction(_actionRecent); } void Application::slotUpdateProgress(const QString &folder, const Progress::Info& progress) { Q_UNUSED(folder); // shows an entry in the context menu. QString curAmount = Utility::octetsToString(progress.overall_current_bytes); QString totalAmount = Utility::octetsToString(progress.overall_transmission_size); _actionStatus->setText(tr("Syncing %1 of %2 (%3 of %4) ").arg(progress.current_file_no) .arg(progress.overall_file_count).arg(curAmount, totalAmount)); // wipe the problem list at start of sync. if( progress.kind == Progress::StartSync ) { _actionRecent->setIcon( QIcon() ); // Fixme: Set a "in-progress"-item eventually. } // If there was a change in the file list, redo the progress menu. if( progress.kind == Progress::EndDownload || progress.kind == Progress::EndUpload || progress.kind == Progress::EndDelete ) { rebuildRecentMenus(); } if (progress.kind == Progress::EndSync) { rebuildRecentMenus(); // show errors. QTimer::singleShot(2000, this, SLOT(slotDisplayIdle())); } } void Application::slotDisplayIdle() { _actionStatus->setText(tr("In Sync")); } void Application::slotHelp() { QDesktopServices::openUrl(QUrl(_theme->helpUrl())); } /* * open the folder with the given Alais */ void Application::slotFolderOpenAction( const QString& alias ) { Folder *f = FolderMan::instance()->folder(alias); qDebug() << "opening local url " << f->path(); if( f ) { QUrl url(f->path(), QUrl::TolerantMode); url.setScheme( QLatin1String("file") ); #ifdef Q_OS_WIN32 // work around a bug in QDesktopServices on Win32, see i-net QString filePath = f->path(); if (filePath.startsWith(QLatin1String("\\\\")) || filePath.startsWith(QLatin1String("//"))) url.setUrl(QDir::toNativeSeparators(filePath)); else url = QUrl::fromLocalFile(filePath); #endif QDesktopServices::openUrl(url); } } void Application::slotOpenOwnCloud() { MirallConfigFile cfgFile; QString url = cfgFile.ownCloudUrl(); QDesktopServices::openUrl( url ); } void Application::slotTrayClicked( QSystemTrayIcon::ActivationReason reason ) { // A click on the tray icon should only open the status window on Win and // Linux, not on Mac. They want a menu entry. #if defined Q_WS_WIN || defined Q_WS_X11 if( reason == QSystemTrayIcon::Trigger ) { checkConfigExists(true); // start settings if config is existing. } #endif } bool Application::checkConfigExists(bool openSettings) { // if no config file is there, start the configuration wizard. MirallConfigFile cfgFile; if( cfgFile.exists() ) { if( openSettings ) { slotSettings(); } return true; } else { qDebug() << "No configured folders yet, starting setup wizard"; OwncloudSetupWizard::runWizard(this, SLOT(slotownCloudWizardDone(int))); return false; } } void Application::slotOpenLogBrowser() { _logBrowser->show(); _logBrowser->raise(); } // slot hit when a folder gets changed in the settings dialog. void Application::slotFoldersChanged() { computeOverallSyncStatus(); setupContextMenu(); } void Application::slotSettings() { if (_settingsDialog.isNull()) { _settingsDialog = new SettingsDialog(this); _settingsDialog->setAttribute( Qt::WA_DeleteOnClose, true ); _settingsDialog->show(); } Utility::raiseDialog(_settingsDialog); } void Application::slotItemProgressDialog() { if (_progressDialog.isNull()) { _progressDialog = new ItemProgressDialog(this); _progressDialog->setAttribute( Qt::WA_DeleteOnClose, true ); _progressDialog->setupList(); _progressDialog->show(); } Utility::raiseDialog(_progressDialog); } void Application::slotParseOptions(const QString &opts) { QStringList options = opts.split(QLatin1Char('|')); parseOptions(options); setupLogBrowser(); } void Application::slotShowTrayMessage(const QString &title, const QString &msg) { if( _tray ) _tray->showMessage(title, msg); else qDebug() << "Tray not ready: " << msg; } void Application::slotShowOptionalTrayMessage(const QString &title, const QString &msg) { MirallConfigFile cfg; if (cfg.optionalDesktopNotifications()) slotShowTrayMessage(title, msg); } void Application::slotShowGuiMessage(const QString &title, const QString &message) { QMessageBox *msgBox = new QMessageBox; msgBox->setAttribute(Qt::WA_DeleteOnClose); msgBox->setText(message); msgBox->setWindowTitle(title); msgBox->setIcon(QMessageBox::Information); msgBox->open(); } void Application::slotSyncStateChange( const QString& alias ) { FolderMan *folderMan = FolderMan::instance(); const SyncResult& result = folderMan->syncResult( alias ); emit folderStateChanged( folderMan->folder(alias) ); computeOverallSyncStatus(); qDebug() << "Sync state changed for folder " << alias << ": " << result.statusString(); if (result.status() == SyncResult::Success || result.status() == SyncResult::Error) { enterNextLogFile(); } if( _progressDialog ) { _progressDialog->setSyncResult(result); } } void Application::parseOptions(const QStringList &options) { QStringListIterator it(options); // skip file name; if (it.hasNext()) it.next(); //parse options; if help or bad option exit while (it.hasNext()) { QString option = it.next(); if (option == QLatin1String("--help") || option == QLatin1String("-h")) { setHelp(); break; } else if (option == QLatin1String("--logwindow") || option == QLatin1String("-l")) { _showLogWindow = true; } else if (option == QLatin1String("--logfile")) { if (it.hasNext() && !it.peekNext().startsWith(QLatin1String("--"))) { _logFile = it.next(); } else { setHelp(); } } else if (option == QLatin1String("--logdir")) { if (it.hasNext() && !it.peekNext().startsWith(QLatin1String("--"))) { _logDirectory = it.next(); } else { setHelp(); } } else if (option == QLatin1String("--logexpire")) { if (it.hasNext() && !it.peekNext().startsWith(QLatin1String("--"))) { _logExpire = it.next().toInt(); } else { setHelp(); } } else if (option == QLatin1String("--logflush")) { _logFlush = true; } else if (option == QLatin1String("--confdir")) { if (it.hasNext() && !it.peekNext().startsWith(QLatin1String("--"))) { QString confDir = it.next(); MirallConfigFile::setConfDir( confDir ); } else { showHelp(); } } else { setHelp(); break; } } } void Application::computeOverallSyncStatus() { // display the info of the least successful sync (eg. not just display the result of the latest sync QString trayMessage; FolderMan *folderMan = FolderMan::instance(); Folder::Map map = folderMan->map(); SyncResult overallResult = FolderMan::accountStatus(map.values()); // create the tray blob message, check if we have an defined state if( overallResult.status() != SyncResult::Undefined ) { QStringList allStatusStrings; foreach(Folder* folder, map.values()) { qDebug() << "Folder in overallStatus Message: " << folder << " with name " << folder->alias(); QString folderMessage = folderMan->statusToString(folder->syncResult().status(), folder->syncEnabled()); allStatusStrings += tr("Folder %1: %2").arg(folder->alias(), folderMessage); } if( ! allStatusStrings.isEmpty() ) trayMessage = allStatusStrings.join(QLatin1String("\n")); else trayMessage = tr("No sync folders configured."); QIcon statusIcon = _theme->syncStateIcon( overallResult.status(), true); _tray->setIcon( statusIcon ); _tray->setToolTip(trayMessage); } } // Helpers for displaying messages. Note that there is no console on Windows. #ifdef Q_OS_WIN // Format as
 HTML
static inline void toHtml(QString &t)
{
    t.replace(QLatin1Char('&'), QLatin1String("&"));
    t.replace(QLatin1Char('<'), QLatin1String("<"));
    t.replace(QLatin1Char('>'), QLatin1String(">"));
    t.insert(0, QLatin1String("
"));
    t.append(QLatin1String("
")); } static void displayHelpText(QString t) // No console on Windows. { toHtml(t); QMessageBox::information(0, Theme::instance()->appNameGUI(), t); } #else static void displayHelpText(const QString &t) { std::cout << qPrintable(t); } #endif void Application::showHelp() { setHelp(); QString helpText; QTextStream stream(&helpText); stream << _theme->appName().toLatin1().constData() << QLatin1String(" version ") << _theme->version().toLatin1().constData() << endl; stream << QLatin1String("File synchronisation desktop utility.") << endl << endl << QLatin1String(optionsC); if (_theme->appName() == QLatin1String("ownCloud")) stream << endl << "For more information, see http://www.owncloud.org" << endl; displayHelpText(helpText); } void Application::setHelp() { _helpOnly = true; } #if defined(Q_OS_WIN) bool Application::winEventFilter(MSG *pMsg, long *result) { if (pMsg->message == WM_POWERBROADCAST) { switch(pMsg->wParam) { case PBT_APMPOWERSTATUSCHANGE: qDebug() << "WM_POWERBROADCAST: Power state changed"; break; case PBT_APMSUSPEND: qDebug() << "WM_POWERBROADCAST: Entering low power state"; break; case PBT_APMRESUMEAUTOMATIC: qDebug() << "WM_POWERBROADCAST: Resuming from low power state"; break; default: break; } return true; } return SharedTools::QtSingleApplication::winEventFilter(pMsg, result); } #endif QString substLang(const QString &lang) { // Map the more apropriate script codes // to country codes as used by Qt and // transifex translation conventions. // Simplified Chinese if (lang == QLatin1String("zh_Hans")) return QLatin1String("zh_CN"); // Traditional Chinese if (lang == QLatin1String("zh_Hant")) return QLatin1String("zh_TW"); return lang; } void Application::setupTranslations() { QStringList uiLanguages; // uiLanguages crashes on Windows with 4.8.0 release builds #if (QT_VERSION >= 0x040801) || (QT_VERSION >= 0x040800 && !defined(Q_OS_WIN)) uiLanguages = QLocale::system().uiLanguages(); #else // older versions need to fall back to the systems locale uiLanguages << QLocale::system().name(); #endif QString enforcedLocale = Theme::instance()->enforcedLocale(); if (!enforcedLocale.isEmpty()) uiLanguages.prepend(enforcedLocale); QTranslator *translator = new QTranslator(this); QTranslator *qtTranslator = new QTranslator(this); QTranslator *qtkeychainTranslator = new QTranslator(this); foreach(QString lang, uiLanguages) { lang.replace(QLatin1Char('-'), QLatin1Char('_')); // work around QTBUG-25973 lang = substLang(lang); const QString trPath = applicationTrPath(); const QString trFile = QLatin1String("mirall_") + lang; if (translator->load(trFile, trPath) || lang.startsWith(QLatin1String("en"))) { // Permissive approach: Qt and keychain translations // may be missing, but Qt translations must be there in order // for us to accept the language. Otherwise, we try with the next. // "en" is an exeption as it is the default language and may not // have a translation file provided. qDebug() << Q_FUNC_INFO << "Using" << lang << "translation"; setProperty("ui_lang", lang); const QString qtTrPath = QLibraryInfo::location(QLibraryInfo::TranslationsPath); const QString qtTrFile = QLatin1String("qt_") + lang; if (qtTranslator->load(qtTrFile, qtTrPath)) { qtTranslator->load(qtTrFile, trPath); } const QString qtkeychainFile = QLatin1String("qt_") + lang; if (!qtkeychainTranslator->load(qtkeychainFile, qtTrPath)) { qtkeychainTranslator->load(qtkeychainFile, trPath); } if (!translator->isEmpty()) installTranslator(translator); if (!qtTranslator->isEmpty()) installTranslator(qtTranslator); if (!qtkeychainTranslator->isEmpty()) installTranslator(qtkeychainTranslator); break; } if (property("ui_lang").isNull()) setProperty("ui_lang", "C"); } } bool Application::giveHelp() { return _helpOnly; } } // namespace Mirall