263 lines
10 KiB
C++
263 lines
10 KiB
C++
#include "./singleinstance.h"
|
|
|
|
#include "../gui/trayicon.h"
|
|
#include "../gui/traywidget.h"
|
|
|
|
#include <syncthingwidgets/misc/syncthinglauncher.h>
|
|
#include <syncthingwidgets/settings/settings.h>
|
|
|
|
#include <syncthingconnector/syncthingprocess.h>
|
|
#ifdef LIB_SYNCTHING_CONNECTOR_SUPPORT_SYSTEMD
|
|
#include <syncthingconnector/syncthingservice.h>
|
|
#endif
|
|
|
|
#ifdef SYNCTHINGTRAY_USE_LIBSYNCTHING
|
|
#include <syncthing/interface.h>
|
|
#endif
|
|
|
|
#include "resources/config.h"
|
|
#include "resources/qtconfig.h"
|
|
|
|
#include <c++utilities/application/argumentparser.h>
|
|
#include <c++utilities/application/commandlineutils.h>
|
|
#include <c++utilities/misc/parseerror.h>
|
|
|
|
#include <qtutilities/resources/importplugin.h>
|
|
#include <qtutilities/resources/qtconfigarguments.h>
|
|
#include <qtutilities/resources/resources.h>
|
|
#include <qtutilities/settingsdialog/qtsettings.h>
|
|
|
|
#include <QApplication>
|
|
#include <QMessageBox>
|
|
#include <QNetworkAccessManager>
|
|
#include <QSettings>
|
|
#include <QStringBuilder>
|
|
|
|
#include <iostream>
|
|
|
|
using namespace std;
|
|
using namespace CppUtilities;
|
|
using namespace QtGui;
|
|
using namespace Data;
|
|
|
|
// import static icon engine plugin
|
|
#ifdef QT_FORK_AWESOME_ICON_ENGINE_STATIC
|
|
#include <QtPlugin>
|
|
Q_IMPORT_PLUGIN(ForkAwesomeIconEnginePlugin)
|
|
#endif
|
|
|
|
ENABLE_QT_RESOURCES_OF_STATIC_DEPENDENCIES
|
|
|
|
#ifdef LIB_SYNCTHING_CONNECTOR_SUPPORT_SYSTEMD
|
|
void handleSystemdServiceError(const QString &context, const QString &name, const QString &message)
|
|
{
|
|
auto *const msgBox = new QMessageBox;
|
|
msgBox->setAttribute(Qt::WA_DeleteOnClose);
|
|
msgBox->setIcon(QMessageBox::Critical);
|
|
msgBox->setText(QCoreApplication::translate("main", "Unable to ") + context);
|
|
msgBox->setInformativeText(name % QStringLiteral(":\n") % message);
|
|
msgBox->show();
|
|
}
|
|
#endif
|
|
|
|
int initSyncthingTray(bool windowed, bool waitForTray, const Argument &connectionConfigArg)
|
|
{
|
|
// get settings
|
|
auto &settings = Settings::values();
|
|
static const auto defaultConnection = std::vector<const char *>({ "" });
|
|
const auto &connectionConfigurations
|
|
= connectionConfigArg.isPresent() && !connectionConfigArg.values().empty() ? connectionConfigArg.values() : defaultConnection;
|
|
|
|
// handle "windowed" case
|
|
if (windowed) {
|
|
// launch Syncthing if configured
|
|
settings.launcher.autostart();
|
|
|
|
// show a window for each connection
|
|
for (const auto *const connectionConfig : connectionConfigurations) {
|
|
auto *const trayWidget = new TrayWidget();
|
|
trayWidget->setAttribute(Qt::WA_DeleteOnClose);
|
|
trayWidget->show();
|
|
trayWidget->applySettings(QString::fromLocal8Bit(connectionConfig));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#ifndef QT_NO_SYSTEMTRAYICON
|
|
// check whether system tray is available
|
|
if (!QSystemTrayIcon::isSystemTrayAvailable() && !waitForTray) {
|
|
QMessageBox::critical(nullptr, QApplication::applicationName(),
|
|
QApplication::translate("main",
|
|
"The system tray is (currently) not available. You could open the tray menu as a regular window using the --windowed flag, though."
|
|
"It is also possible to start Syncthing Tray with --wait to wait until the system tray becomes available instead of showing this "
|
|
"message."));
|
|
return -1;
|
|
}
|
|
|
|
// launch Syncthing if configured
|
|
settings.launcher.autostart();
|
|
|
|
// show a tray icon for each connection
|
|
TrayWidget *widget;
|
|
for (const auto *const connectionConfig : connectionConfigurations) {
|
|
auto *const trayIcon = new TrayIcon(QString::fromLocal8Bit(connectionConfig), QApplication::instance());
|
|
trayIcon->show();
|
|
widget = &trayIcon->trayMenu().widget();
|
|
}
|
|
|
|
// show "first launch" message box
|
|
if (!settings.firstLaunch) {
|
|
return 0;
|
|
}
|
|
QMessageBox msgBox;
|
|
msgBox.setIcon(QMessageBox::Information);
|
|
msgBox.setText(QCoreApplication::translate("main", "You must configure how to connect to Syncthing when using Syncthing Tray the first time."));
|
|
msgBox.setInformativeText(QCoreApplication::translate(
|
|
"main", "Note that the settings dialog allows importing URL, credentials and API-key from the local Syncthing configuration."));
|
|
msgBox.exec();
|
|
widget->showSettingsDialog();
|
|
return 0;
|
|
|
|
#else
|
|
// show error if system tray is not supported by Qt
|
|
QMessageBox::critical(nullptr, QApplication::applicationName(),
|
|
QApplication::translate("main",
|
|
"The Qt libraries have not been built with tray icon support. You could open the tray menu as a regular "
|
|
"window using the -w flag, though."));
|
|
return -2;
|
|
#endif
|
|
}
|
|
|
|
void trigger(bool tray, bool webUi)
|
|
{
|
|
if (TrayWidget::instances().empty() || !(tray || webUi)) {
|
|
return;
|
|
}
|
|
auto *const trayWidget = TrayWidget::instances().front();
|
|
if (webUi) {
|
|
trayWidget->showWebUi();
|
|
}
|
|
if (tray) {
|
|
trayWidget->showUsingPositioningSettings();
|
|
}
|
|
}
|
|
|
|
void shutdownSyncthingTray()
|
|
{
|
|
Settings::save();
|
|
Settings::Launcher::terminate();
|
|
}
|
|
|
|
int runApplication(int argc, const char *const *argv)
|
|
{
|
|
// setup argument parser
|
|
SET_APPLICATION_INFO;
|
|
CMD_UTILS_CONVERT_ARGS_TO_UTF8;
|
|
CMD_UTILS_HANDLE_VIRTUAL_TERMINAL_PROCESSING;
|
|
auto parser = ArgumentParser();
|
|
auto qtConfigArgs = QT_CONFIG_ARGUMENTS();
|
|
auto windowedArg = ConfigValueArgument("windowed", 'w', "opens the tray menu as a regular window");
|
|
auto showWebUiArg = ConfigValueArgument("webui", '\0', "instantly shows the web UI - meant for creating shortcut to web UI");
|
|
auto triggerArg = ConfigValueArgument("trigger", '\0', "instantly shows the left-click tray menu - meant for creating a shortcut");
|
|
auto waitForTrayArg = ConfigValueArgument("wait", '\0',
|
|
"wait until the system tray becomes available instead of showing an error message if the system tray is not available on start-up");
|
|
auto connectionArg = ConfigValueArgument("connection", '\0', "specifies one or more connection configurations to be used", { "config name" });
|
|
connectionArg.setRequiredValueCount(Argument::varValueCount);
|
|
auto configPathArg = ConfigValueArgument("config-dir-path", '\0', "specifies the path to the configuration directory", { "path" });
|
|
configPathArg.setEnvironmentVariable(PROJECT_VARNAME_UPPER "_CONFIG_DIR");
|
|
auto singleInstance = Argument("single-instance", '\0', "does nothing if a tray icon is already shown");
|
|
auto newInstanceArg = Argument("new-instance", '\0', "disable the usual single-process behavior");
|
|
auto &widgetsGuiArg = qtConfigArgs.qtWidgetsGuiArg();
|
|
widgetsGuiArg.addSubArguments(
|
|
{ &windowedArg, &showWebUiArg, &triggerArg, &waitForTrayArg, &connectionArg, &configPathArg, &singleInstance, &newInstanceArg });
|
|
#ifdef SYNCTHINGTRAY_USE_LIBSYNCTHING
|
|
auto cliArg = OperationArgument("cli", 'c', "run Syncthing's CLI");
|
|
auto cliHelp = ConfigValueArgument("help", 'h', "show help for Syncthing's CLI");
|
|
cliArg.setRequiredValueCount(Argument::varValueCount);
|
|
cliArg.setFlags(Argument::Flags::Greedy, true);
|
|
cliArg.setSubArguments({ &cliHelp });
|
|
#endif
|
|
|
|
parser.setMainArguments({ &qtConfigArgs.qtWidgetsGuiArg(),
|
|
#ifdef SYNCTHINGTRAY_USE_LIBSYNCTHING
|
|
&cliArg,
|
|
#endif
|
|
&parser.noColorArg(), &parser.helpArg() });
|
|
parser.parseArgs(argc, argv);
|
|
#ifdef SYNCTHINGTRAY_USE_LIBSYNCTHING
|
|
if (cliArg.isPresent()) {
|
|
CMD_UTILS_START_CONSOLE;
|
|
return static_cast<int>(LibSyncthing::runCli(cliArg.values()));
|
|
}
|
|
#endif
|
|
if (!qtConfigArgs.qtWidgetsGuiArg().isPresent()) {
|
|
return 0;
|
|
}
|
|
|
|
// handle override for config dir
|
|
if (const char *const configPathDir = configPathArg.firstValue()) {
|
|
QSettings::setPath(QSettings::IniFormat, QSettings::UserScope, QString::fromLocal8Bit(configPathDir));
|
|
}
|
|
|
|
// check whether runApplication() has been called for the first time
|
|
static auto firstRun = true;
|
|
if (firstRun) {
|
|
firstRun = false;
|
|
|
|
// do first-time initializations
|
|
SET_QT_APPLICATION_INFO;
|
|
QApplication application(argc, const_cast<char **>(argv));
|
|
QGuiApplication::setQuitOnLastWindowClosed(false);
|
|
SingleInstance singleInstance(argc, argv, newInstanceArg.isPresent());
|
|
networkAccessManager().setParent(&singleInstance);
|
|
QObject::connect(&singleInstance, &SingleInstance::newInstance, &runApplication);
|
|
Settings::restore();
|
|
Settings::values().qt.apply();
|
|
qtConfigArgs.applySettings(true);
|
|
LOAD_QT_TRANSLATIONS;
|
|
SyncthingLauncher launcher;
|
|
SyncthingLauncher::setMainInstance(&launcher);
|
|
#ifdef LIB_SYNCTHING_CONNECTOR_SUPPORT_SYSTEMD
|
|
SyncthingService service;
|
|
SyncthingService::setMainInstance(&service);
|
|
Settings::values().systemd.setupService(service);
|
|
QObject::connect(&service, &SyncthingService::errorOccurred, &handleSystemdServiceError);
|
|
#endif
|
|
|
|
// init Syncthing Tray and immediately shutdown on failure
|
|
if (const auto res = initSyncthingTray(windowedArg.isPresent(), waitForTrayArg.isPresent(), connectionArg)) {
|
|
shutdownSyncthingTray();
|
|
return res;
|
|
}
|
|
|
|
// trigger UI and enter event loop
|
|
QObject::connect(&application, &QCoreApplication::aboutToQuit, &shutdownSyncthingTray);
|
|
trigger(triggerArg.isPresent(), showWebUiArg.isPresent());
|
|
return application.exec();
|
|
}
|
|
|
|
// trigger actions if --webui or --trigger is present but don't create a new tray icon
|
|
const auto firstInstance = TrayWidget::instances().empty();
|
|
if (!firstInstance && (showWebUiArg.isPresent() || triggerArg.isPresent())) {
|
|
trigger(triggerArg.isPresent(), showWebUiArg.isPresent());
|
|
return 0;
|
|
}
|
|
|
|
// don't create a new instance if --single-instance has been specified
|
|
if (!firstInstance && singleInstance.isPresent()) {
|
|
return 0;
|
|
}
|
|
|
|
// create new/additional tray icon
|
|
const auto res = initSyncthingTray(windowedArg.isPresent(), waitForTrayArg.isPresent(), connectionArg);
|
|
if (!res) {
|
|
trigger(triggerArg.isPresent(), showWebUiArg.isPresent());
|
|
}
|
|
return res;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
return runApplication(argc, argv);
|
|
}
|