/* * This file Copyright (C) 2009-2015 Mnemosyne LLC * * It may be used under the GNU GPL versions 2 or 3 * or any future license endorsed by Mnemosyne LLC. * */ #include #include #include #include #include #include #include #include #ifdef QT_DBUS_LIB #include #include #include #endif #include #include #include #include #include "AddData.h" #include "Application.h" #include "Formatter.h" #include "InteropHelper.h" #include "MainWindow.h" #include "OptionsDialog.h" #include "Prefs.h" #include "Session.h" #include "TorrentModel.h" #include "WatchDir.h" namespace { QLatin1String const MY_CONFIG_NAME("transmission"); QLatin1String const MY_READABLE_NAME("transmission-qt"); tr_option const opts[] = { { 'g', "config-dir", "Where to look for configuration files", "g", true, "" }, { 'm', "minimized", "Start minimized in system tray", "m", false, nullptr }, { 'p', "port", "Port to use when connecting to an existing session", "p", true, "" }, { 'r', "remote", "Connect to an existing session at the specified hostname", "r", true, "" }, { 'u', "username", "Username to use when connecting to an existing session", "u", true, "" }, { 'v', "version", "Show version number and exit", "v", false, nullptr }, { 'w', "password", "Password to use when connecting to an existing session", "w", true, "" }, { 0, nullptr, nullptr, nullptr, false, nullptr } }; char const* getUsage() { return "Usage:\n" " transmission [OPTIONS...] [torrent files]"; } enum { STATS_REFRESH_INTERVAL_MSEC = 3000, SESSION_REFRESH_INTERVAL_MSEC = 3000, MODEL_REFRESH_INTERVAL_MSEC = 3000 }; bool loadTranslation(QTranslator& translator, QString const& name, QLocale const& locale, QStringList const& searchDirectories) { for (QString const& directory : searchDirectories) { if (translator.load(locale, name, QLatin1String("_"), directory)) { return true; } } return false; } } // namespace Application::Application(int& argc, char** argv) : QApplication(argc, argv), myPrefs(nullptr), mySession(nullptr), myModel(nullptr), myWindow(nullptr), myWatchDir(nullptr), myLastFullUpdateTime(0) { setApplicationName(MY_CONFIG_NAME); loadTranslations(); Formatter::initUnits(); #if defined(_WIN32) || defined(__APPLE__) if (QIcon::themeName().isEmpty()) { QIcon::setThemeName(QLatin1String("Faenza")); } #endif // set the default icon QIcon icon = QIcon::fromTheme(QLatin1String("transmission")); if (icon.isNull()) { QList sizes; sizes << 16 << 22 << 24 << 32 << 48 << 64 << 72 << 96 << 128 << 192 << 256; for (int const size : sizes) { icon.addPixmap(QPixmap(QString::fromLatin1(":/icons/transmission-%1.png").arg(size))); } } setWindowIcon(icon); #ifdef __APPLE__ setAttribute(Qt::AA_DontShowIconsInMenus); #endif // parse the command-line arguments int c; bool minimized = false; char const* optarg; QString host; QString port; QString username; QString password; QString configDir; QStringList filenames; while ((c = tr_getopt(getUsage(), argc, const_cast(argv), opts, &optarg)) != TR_OPT_DONE) { switch (c) { case 'g': configDir = QString::fromUtf8(optarg); break; case 'p': port = QString::fromUtf8(optarg); break; case 'r': host = QString::fromUtf8(optarg); break; case 'u': username = QString::fromUtf8(optarg); break; case 'w': password = QString::fromUtf8(optarg); break; case 'm': minimized = true; break; case 'v': std::cerr << MY_READABLE_NAME.latin1() << ' ' << LONG_VERSION_STRING << std::endl; quitLater(); return; case TR_OPT_ERR: std::cerr << qPrintable(QObject::tr("Invalid option")) << std::endl; tr_getopt_usage(MY_READABLE_NAME.latin1(), getUsage(), opts); quitLater(); return; default: filenames.append(QString::fromUtf8(optarg)); break; } } // try to delegate the work to an existing copy of Transmission // before starting ourselves... InteropHelper interopClient; if (interopClient.isConnected()) { bool delegated = false; for (QString const& filename : filenames) { QString metainfo; AddData a(filename); switch (a.type) { case AddData::URL: metainfo = a.url.toString(); break; case AddData::MAGNET: metainfo = a.magnet; break; case AddData::FILENAME: case AddData::METAINFO: metainfo = QString::fromLatin1(a.toBase64()); break; default: break; } if (!metainfo.isEmpty() && interopClient.addMetainfo(metainfo)) { delegated = true; } } if (delegated) { quitLater(); return; } } // set the fallback config dir if (configDir.isNull()) { configDir = QString::fromUtf8(tr_getDefaultConfigDir("transmission")); } // ensure our config directory exists QDir dir(configDir); if (!dir.exists()) { dir.mkpath(configDir); } // is this the first time we've run transmission? bool const firstTime = !dir.exists(QLatin1String("settings.json")); // initialize the prefs myPrefs = new Prefs(configDir); if (!host.isNull()) { myPrefs->set(Prefs::SESSION_REMOTE_HOST, host); } if (!port.isNull()) { myPrefs->set(Prefs::SESSION_REMOTE_PORT, port.toUInt()); } if (!username.isNull()) { myPrefs->set(Prefs::SESSION_REMOTE_USERNAME, username); } if (!password.isNull()) { myPrefs->set(Prefs::SESSION_REMOTE_PASSWORD, password); } if (!host.isNull() || !port.isNull() || !username.isNull() || !password.isNull()) { myPrefs->set(Prefs::SESSION_IS_REMOTE, true); } if (myPrefs->getBool(Prefs::START_MINIMIZED)) { minimized = true; } // start as minimized only if the system tray present if (!myPrefs->getBool(Prefs::SHOW_TRAY_ICON)) { minimized = false; } mySession = new Session(configDir, *myPrefs); myModel = new TorrentModel(*myPrefs); myWindow = new MainWindow(*mySession, *myPrefs, *myModel, minimized); myWatchDir = new WatchDir(*myModel); connect(myModel, &TorrentModel::torrentsAdded, this, &Application::onTorrentsAdded); connect(myModel, &TorrentModel::torrentsCompleted, this, &Application::onTorrentsCompleted); connect(myModel, &TorrentModel::torrentsNeedInfo, this, &Application::onTorrentsNeedInfo); connect(myPrefs, &Prefs::changed, this, &Application::refreshPref); connect(mySession, &Session::sourceChanged, this, &Application::onSessionSourceChanged); connect(mySession, &Session::torrentsRemoved, myModel, &TorrentModel::removeTorrents); connect(mySession, &Session::torrentsUpdated, myModel, &TorrentModel::updateTorrents); connect(myWatchDir, &WatchDir::torrentFileAdded, this, &Application::addTorrent); // init from preferences for (auto const key : { Prefs::DIR_WATCH }) { refreshPref(key); } QTimer* timer = &myModelTimer; connect(timer, &QTimer::timeout, this, &Application::refreshTorrents); timer->setSingleShot(false); timer->setInterval(MODEL_REFRESH_INTERVAL_MSEC); timer->start(); timer = &myStatsTimer; connect(timer, &QTimer::timeout, mySession, &Session::refreshSessionStats); timer->setSingleShot(false); timer->setInterval(STATS_REFRESH_INTERVAL_MSEC); timer->start(); timer = &mySessionTimer; connect(timer, &QTimer::timeout, mySession, &Session::refreshSessionInfo); timer->setSingleShot(false); timer->setInterval(SESSION_REFRESH_INTERVAL_MSEC); timer->start(); maybeUpdateBlocklist(); if (!firstTime) { mySession->restart(); } else { myWindow->openSession(); } if (!myPrefs->getBool(Prefs::USER_HAS_GIVEN_INFORMED_CONSENT)) { auto* dialog = new QMessageBox(QMessageBox::Information, QString(), tr("Transmission is a file sharing program."), QMessageBox::Ok | QMessageBox::Cancel, myWindow); dialog->setInformativeText(tr("When you run a torrent, its data will be made available to others by means of upload. " "Any content you share is your sole responsibility.")); dialog->button(QMessageBox::Ok)->setText(tr("I &Agree")); dialog->setDefaultButton(QMessageBox::Ok); dialog->setModal(true); connect(dialog, SIGNAL(finished(int)), this, SLOT(consentGiven(int))); dialog->setAttribute(Qt::WA_DeleteOnClose); dialog->show(); } for (QString const& filename : filenames) { addTorrent(filename); } InteropHelper::registerObject(this); } void Application::loadTranslations() { QStringList const qtQmDirs = QStringList() << QLibraryInfo::location(QLibraryInfo::TranslationsPath) << #ifdef TRANSLATIONS_DIR QString::fromUtf8(TRANSLATIONS_DIR) << #endif (applicationDirPath() + QLatin1String("/translations")); QStringList const appQmDirs = QStringList() << #ifdef TRANSLATIONS_DIR QString::fromUtf8(TRANSLATIONS_DIR) << #endif (applicationDirPath() + QLatin1String("/translations")); QString const qtFileName = QLatin1String("qtbase"); QLocale const locale; QLocale const englishLocale(QLocale::English, QLocale::UnitedStates); if (loadTranslation(myQtTranslator, qtFileName, locale, qtQmDirs) || loadTranslation(myQtTranslator, qtFileName, englishLocale, qtQmDirs)) { installTranslator(&myQtTranslator); } if (loadTranslation(myAppTranslator, MY_CONFIG_NAME, locale, appQmDirs) || loadTranslation(myAppTranslator, MY_CONFIG_NAME, englishLocale, appQmDirs)) { installTranslator(&myAppTranslator); } } void Application::quitLater() { QTimer::singleShot(0, this, SLOT(quit())); } void Application::onTorrentsEdited(torrent_ids_t const& ids) { // the backend's tr_info has changed, so reload those fields mySession->initTorrents(ids); } QStringList Application::getNames(torrent_ids_t const& ids) const { QStringList names; for (auto const& id : ids) { names.push_back(myModel->getTorrentFromId(id)->name()); } names.sort(); return names; } void Application::onTorrentsAdded(torrent_ids_t const& ids) { if (myPrefs->getBool(Prefs::SHOW_NOTIFICATION_ON_ADD)) { auto const title = tr("Torrent(s) Added", nullptr, ids.size()); auto const body = getNames(ids).join(QStringLiteral("\n")); notifyApp(title, body); } } void Application::onTorrentsCompleted(torrent_ids_t const& ids) { if (myPrefs->getBool(Prefs::SHOW_NOTIFICATION_ON_COMPLETE)) { auto const title = tr("Torrent Completed", nullptr, ids.size()); auto const body = getNames(ids).join(QStringLiteral("\n")); notifyApp(title, body); } if (myPrefs->getBool(Prefs::COMPLETE_SOUND_ENABLED)) { #if defined(Q_OS_WIN) || defined(Q_OS_MAC) beep(); #else QProcess::execute(myPrefs->getString(Prefs::COMPLETE_SOUND_COMMAND)); #endif } } void Application::onTorrentsNeedInfo(torrent_ids_t const& ids) { if (!ids.empty()) { mySession->initTorrents(ids); } } /*** **** ***/ void Application::consentGiven(int result) { if (result == QMessageBox::Ok) { myPrefs->set(Prefs::USER_HAS_GIVEN_INFORMED_CONSENT, true); } else { quit(); } } Application::~Application() { if (myPrefs != nullptr && myWindow != nullptr) { QRect const mainwinRect(myWindow->geometry()); myPrefs->set(Prefs::MAIN_WINDOW_HEIGHT, std::max(100, mainwinRect.height())); myPrefs->set(Prefs::MAIN_WINDOW_WIDTH, std::max(100, mainwinRect.width())); myPrefs->set(Prefs::MAIN_WINDOW_X, mainwinRect.x()); myPrefs->set(Prefs::MAIN_WINDOW_Y, mainwinRect.y()); } delete myWatchDir; delete myWindow; delete myModel; delete mySession; delete myPrefs; } /*** **** ***/ void Application::refreshPref(int key) { switch (key) { case Prefs::BLOCKLIST_UPDATES_ENABLED: maybeUpdateBlocklist(); break; case Prefs::DIR_WATCH: case Prefs::DIR_WATCH_ENABLED: { QString const path(myPrefs->getString(Prefs::DIR_WATCH)); bool const isEnabled(myPrefs->getBool(Prefs::DIR_WATCH_ENABLED)); myWatchDir->setPath(path, isEnabled); break; } default: break; } } void Application::maybeUpdateBlocklist() { if (!myPrefs->getBool(Prefs::BLOCKLIST_UPDATES_ENABLED)) { return; } QDateTime const lastUpdatedAt = myPrefs->getDateTime(Prefs::BLOCKLIST_DATE); QDateTime const nextUpdateAt = lastUpdatedAt.addDays(7); QDateTime const now = QDateTime::currentDateTime(); if (now < nextUpdateAt) { mySession->updateBlocklist(); myPrefs->set(Prefs::BLOCKLIST_DATE, now); } } void Application::onSessionSourceChanged() { mySession->initTorrents(); mySession->refreshSessionStats(); mySession->refreshSessionInfo(); } void Application::refreshTorrents() { // usually we just poll the torrents that have shown recent activity, // but we also periodically ask for updates on the others to ensure // nothing's falling through the cracks. time_t const now = time(nullptr); if (myLastFullUpdateTime + 60 >= now) { mySession->refreshActiveTorrents(); } else { myLastFullUpdateTime = now; mySession->refreshAllTorrents(); } } /*** **** ***/ void Application::addTorrent(AddData const& addme) { if (addme.type == addme.NONE) { return; } if (!myPrefs->getBool(Prefs::OPTIONS_PROMPT)) { mySession->addTorrent(addme); } else { auto* o = new OptionsDialog(*mySession, *myPrefs, addme, myWindow); o->show(); } raise(); } /*** **** ***/ void Application::raise() { alert(myWindow); } bool Application::notifyApp(QString const& title, QString const& body) const { #ifdef QT_DBUS_LIB QLatin1String const dbusServiceName("org.freedesktop.Notifications"); QLatin1String const dbusInterfaceName("org.freedesktop.Notifications"); QLatin1String const dbusPath("/org/freedesktop/Notifications"); QDBusConnection bus = QDBusConnection::sessionBus(); if (bus.isConnected()) { QDBusMessage m = QDBusMessage::createMethodCall(dbusServiceName, dbusPath, dbusInterfaceName, QLatin1String("Notify")); QVariantList args; args.append(QLatin1String("Transmission")); // app_name args.append(0U); // replaces_id args.append(QLatin1String("transmission")); // icon args.append(title); // summary args.append(body); // body args.append(QStringList()); // actions - unused for plain passive popups args.append(QVariantMap()); // hints - unused atm args.append(static_cast(-1)); // use the default timeout period m.setArguments(args); QDBusReply const replyMsg = bus.call(m); if (replyMsg.isValid() && replyMsg.value() > 0) { return true; } } #endif myWindow->trayIcon().showMessage(title, body); return true; } FaviconCache& Application::faviconCache() { return myFavicons; } /*** **** ***/ int tr_main(int argc, char* argv[]) { InteropHelper::initialize(); #if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0) Application::setAttribute(Qt::AA_EnableHighDpiScaling); #endif Application::setAttribute(Qt::AA_UseHighDpiPixmaps); Application app(argc, argv); return app.exec(); }