2021-03-05 00:35:15 +01:00
|
|
|
// SPDX-FileCopyrightText: 2017 Konstantinos Sideris <siderisk@auth.gr>
|
|
|
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2019-12-23 05:22:03 +01:00
|
|
|
#include <iostream>
|
|
|
|
|
2017-04-06 01:06:42 +02:00
|
|
|
#include <QApplication>
|
2018-07-15 13:53:18 +02:00
|
|
|
#include <QCommandLineParser>
|
2021-01-10 18:36:06 +01:00
|
|
|
#include <QDesktopServices>
|
2017-04-15 16:23:35 +02:00
|
|
|
#include <QDesktopWidget>
|
2018-06-12 19:36:16 +02:00
|
|
|
#include <QDir>
|
2017-11-16 15:33:52 +01:00
|
|
|
#include <QFile>
|
2017-04-06 01:06:42 +02:00
|
|
|
#include <QFontDatabase>
|
2019-07-05 03:20:19 +02:00
|
|
|
#include <QGuiApplication>
|
2017-12-14 20:55:00 +01:00
|
|
|
#include <QLabel>
|
2017-05-29 18:09:12 +02:00
|
|
|
#include <QLibraryInfo>
|
2018-07-22 19:26:50 +02:00
|
|
|
#include <QMessageBox>
|
2017-12-14 20:55:00 +01:00
|
|
|
#include <QPoint>
|
2019-07-05 03:20:19 +02:00
|
|
|
#include <QScreen>
|
2017-10-28 14:46:39 +02:00
|
|
|
#include <QSettings>
|
2018-06-09 15:03:14 +02:00
|
|
|
#include <QStandardPaths>
|
2017-05-29 18:09:12 +02:00
|
|
|
#include <QTranslator>
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2021-01-10 18:36:06 +01:00
|
|
|
#include "ChatPage.h"
|
2017-12-14 20:55:00 +01:00
|
|
|
#include "Config.h"
|
2018-07-17 15:37:25 +02:00
|
|
|
#include "Logging.h"
|
2017-04-06 01:06:42 +02:00
|
|
|
#include "MainWindow.h"
|
2018-06-09 15:03:14 +02:00
|
|
|
#include "MatrixClient.h"
|
2018-07-22 18:48:58 +02:00
|
|
|
#include "Utils.h"
|
2018-09-30 13:33:54 +02:00
|
|
|
#include "config/nheko.h"
|
2020-02-23 13:40:04 +01:00
|
|
|
#include "singleapplication.h"
|
2017-12-14 20:55:00 +01:00
|
|
|
|
2018-09-19 22:38:36 +02:00
|
|
|
#if defined(Q_OS_MAC)
|
|
|
|
#include "emoji/MacHelper.h"
|
|
|
|
#endif
|
|
|
|
|
2019-12-14 23:48:02 +01:00
|
|
|
#ifdef QML_DEBUGGING
|
|
|
|
#include <QQmlDebuggingEnabler>
|
|
|
|
QQmlDebuggingEnabler enabler;
|
|
|
|
#endif
|
|
|
|
|
2019-12-23 05:22:03 +01:00
|
|
|
#if defined(Q_OS_LINUX)
|
|
|
|
#include <boost/stacktrace.hpp>
|
|
|
|
#include <csignal>
|
|
|
|
|
2018-07-01 18:40:53 +02:00
|
|
|
void
|
|
|
|
stacktraceHandler(int signum)
|
|
|
|
{
|
|
|
|
std::signal(signum, SIG_DFL);
|
2018-07-01 21:57:14 +02:00
|
|
|
boost::stacktrace::safe_dump_to("./nheko-backtrace.dump");
|
2018-07-01 18:40:53 +02:00
|
|
|
std::raise(SIGABRT);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
registerSignalHandlers()
|
|
|
|
{
|
|
|
|
std::signal(SIGSEGV, &stacktraceHandler);
|
|
|
|
std::signal(SIGABRT, &stacktraceHandler);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
// No implementation for systems with no stacktrace support.
|
|
|
|
void
|
|
|
|
registerSignalHandlers()
|
|
|
|
{}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2017-12-14 20:55:00 +01:00
|
|
|
QPoint
|
|
|
|
screenCenter(int width, int height)
|
|
|
|
{
|
2019-07-05 03:20:19 +02:00
|
|
|
// Deprecated in 5.13: QRect screenGeometry = QApplication::desktop()->screenGeometry();
|
|
|
|
QRect screenGeometry = QGuiApplication::primaryScreen()->geometry();
|
2017-12-14 20:55:00 +01:00
|
|
|
|
|
|
|
int x = (screenGeometry.width() - width) / 2;
|
|
|
|
int y = (screenGeometry.height() - height) / 2;
|
|
|
|
|
|
|
|
return QPoint(x, y);
|
|
|
|
}
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2018-06-12 19:36:16 +02:00
|
|
|
void
|
2021-01-19 19:44:22 +01:00
|
|
|
createStandardDirectory(QStandardPaths::StandardLocation path)
|
2018-06-12 19:36:16 +02:00
|
|
|
{
|
2021-01-19 19:44:22 +01:00
|
|
|
auto dir = QStandardPaths::writableLocation(path);
|
2018-06-12 19:36:16 +02:00
|
|
|
|
|
|
|
if (!QDir().mkpath(dir)) {
|
|
|
|
throw std::runtime_error(
|
|
|
|
("Unable to create state directory:" + dir).toStdString().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-20 12:47:22 +02:00
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
2017-04-06 01:06:42 +02:00
|
|
|
{
|
2020-11-10 03:28:41 +01:00
|
|
|
QCoreApplication::setApplicationName("nheko");
|
2020-06-11 23:08:15 +02:00
|
|
|
QCoreApplication::setApplicationVersion(nheko::version);
|
|
|
|
QCoreApplication::setOrganizationName("nheko");
|
|
|
|
QCoreApplication::setAttribute(Qt::AA_DontCreateNativeWidgetSiblings);
|
|
|
|
QCoreApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
|
|
|
|
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
|
|
|
|
|
|
|
|
// this needs to be after setting the application name. Or how would we find our settings
|
|
|
|
// file then?
|
2018-08-12 08:33:36 +02:00
|
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_WIN) || defined(Q_OS_FREEBSD)
|
2018-07-22 18:48:58 +02:00
|
|
|
if (qgetenv("QT_SCALE_FACTOR").size() == 0) {
|
|
|
|
float factor = utils::scaleFactor();
|
|
|
|
|
|
|
|
if (factor != -1)
|
|
|
|
qputenv("QT_SCALE_FACTOR", QString::number(factor).toUtf8());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-11-10 03:28:41 +01:00
|
|
|
// This is some hacky programming, but it's necessary (AFAIK?) to get the unique config name
|
|
|
|
// parsed before the SingleApplication userdata is set.
|
|
|
|
QString userdata{""};
|
2021-01-10 18:36:06 +01:00
|
|
|
QString matrixUri;
|
2021-02-04 01:02:38 +01:00
|
|
|
for (int i = 1; i < argc; ++i) {
|
2021-01-10 18:36:06 +01:00
|
|
|
QString arg{argv[i]};
|
|
|
|
if (arg.startsWith("--profile=")) {
|
|
|
|
arg.remove("--profile=");
|
|
|
|
userdata = arg;
|
|
|
|
} else if (arg.startsWith("--p=")) {
|
|
|
|
arg.remove("-p=");
|
|
|
|
userdata = arg;
|
|
|
|
} else if (arg == "--profile" || arg == "-p") {
|
2020-11-10 03:28:41 +01:00
|
|
|
if (i < argc - 1) // if i is less than argc - 1, we still have a parameter
|
|
|
|
// left to process as the name
|
|
|
|
{
|
|
|
|
++i; // the next arg is the name, so increment
|
|
|
|
userdata = QString{argv[i]};
|
|
|
|
}
|
2021-01-10 18:36:06 +01:00
|
|
|
} else if (arg.startsWith("matrix:")) {
|
|
|
|
matrixUri = arg;
|
2020-11-10 03:28:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-23 13:40:04 +01:00
|
|
|
SingleApplication app(argc,
|
|
|
|
argv,
|
2021-01-10 18:36:06 +01:00
|
|
|
true,
|
2020-02-23 13:40:04 +01:00
|
|
|
SingleApplication::Mode::User |
|
|
|
|
SingleApplication::Mode::ExcludeAppPath |
|
2021-01-10 18:36:06 +01:00
|
|
|
SingleApplication::Mode::ExcludeAppVersion |
|
|
|
|
SingleApplication::Mode::SecondaryNotification,
|
2020-11-10 03:28:41 +01:00
|
|
|
100,
|
|
|
|
userdata);
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2021-01-10 18:36:06 +01:00
|
|
|
if (app.isSecondary()) {
|
|
|
|
// open uri in main instance
|
|
|
|
app.sendMessage(matrixUri.toUtf8());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-07-15 13:53:18 +02:00
|
|
|
QCommandLineParser parser;
|
|
|
|
parser.addHelpOption();
|
|
|
|
parser.addVersionOption();
|
2020-04-15 23:28:04 +02:00
|
|
|
QCommandLineOption debugOption("debug", "Enable debug output");
|
|
|
|
parser.addOption(debugOption);
|
2020-10-22 01:20:02 +02:00
|
|
|
|
|
|
|
// This option is not actually parsed via Qt due to the need to parse it before the app
|
|
|
|
// name is set. It only exists to keep Qt from complaining about the --profile/-p
|
|
|
|
// option and thereby crashing the app.
|
2020-10-27 17:45:28 +01:00
|
|
|
QCommandLineOption configName(
|
|
|
|
QStringList() << "p"
|
|
|
|
<< "profile",
|
|
|
|
QCoreApplication::tr("Create a unique profile, which allows you to log into several "
|
|
|
|
"accounts at the same time and start multiple instances of nheko."),
|
|
|
|
QCoreApplication::tr("profile"),
|
|
|
|
QCoreApplication::tr("profile name"));
|
2020-10-22 01:20:02 +02:00
|
|
|
parser.addOption(configName);
|
2020-10-27 17:45:28 +01:00
|
|
|
|
2018-07-15 13:53:18 +02:00
|
|
|
parser.process(app);
|
2017-05-03 22:32:51 +02:00
|
|
|
|
2020-12-03 02:23:28 +01:00
|
|
|
app.setWindowIcon(QIcon::fromTheme("nheko", QIcon{":/logos/nheko.png"}));
|
2018-06-09 15:03:14 +02:00
|
|
|
|
|
|
|
http::init();
|
|
|
|
|
2021-01-19 19:44:22 +01:00
|
|
|
createStandardDirectory(QStandardPaths::CacheLocation);
|
|
|
|
createStandardDirectory(QStandardPaths::AppDataLocation);
|
2018-06-12 19:36:16 +02:00
|
|
|
|
2018-07-01 18:40:53 +02:00
|
|
|
registerSignalHandlers();
|
|
|
|
|
2020-04-15 23:28:04 +02:00
|
|
|
if (parser.isSet(debugOption))
|
|
|
|
nhlog::enable_debug_log_from_commandline = true;
|
|
|
|
|
2018-06-09 15:03:14 +02:00
|
|
|
try {
|
2018-06-14 01:28:35 +02:00
|
|
|
nhlog::init(QString("%1/nheko.log")
|
|
|
|
.arg(QStandardPaths::writableLocation(QStandardPaths::CacheLocation))
|
|
|
|
.toStdString());
|
2018-06-09 15:03:14 +02:00
|
|
|
} catch (const spdlog::spdlog_ex &ex) {
|
|
|
|
std::cout << "Log initialization failed: " << ex.what() << std::endl;
|
|
|
|
std::exit(1);
|
|
|
|
}
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2020-11-10 03:28:41 +01:00
|
|
|
if (parser.isSet(configName))
|
2020-12-27 22:56:43 +01:00
|
|
|
UserSettings::initialize(parser.value(configName));
|
|
|
|
else
|
|
|
|
UserSettings::initialize(std::nullopt);
|
|
|
|
|
|
|
|
auto settings = UserSettings::instance().toWeakRef();
|
2018-10-01 16:56:46 +02:00
|
|
|
|
2018-10-07 12:27:20 +02:00
|
|
|
QFont font;
|
2020-12-27 22:56:43 +01:00
|
|
|
QString userFontFamily = settings.lock()->font();
|
2021-02-09 16:26:18 +01:00
|
|
|
if (!userFontFamily.isEmpty() && userFontFamily != "default") {
|
2019-01-19 17:20:41 +01:00
|
|
|
font.setFamily(userFontFamily);
|
|
|
|
}
|
2020-12-27 22:56:43 +01:00
|
|
|
font.setPointSizeF(settings.lock()->fontSize());
|
2018-10-01 16:56:46 +02:00
|
|
|
|
|
|
|
app.setFont(font);
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2017-09-10 11:59:21 +02:00
|
|
|
QString lang = QLocale::system().name();
|
2017-05-29 18:09:12 +02:00
|
|
|
|
2017-09-10 11:59:21 +02:00
|
|
|
QTranslator qtTranslator;
|
2020-07-10 03:17:23 +02:00
|
|
|
qtTranslator.load(
|
|
|
|
QLocale(), "qt", "_", QLibraryInfo::location(QLibraryInfo::TranslationsPath));
|
2017-09-10 11:59:21 +02:00
|
|
|
app.installTranslator(&qtTranslator);
|
2017-05-29 18:09:12 +02:00
|
|
|
|
2017-09-10 11:59:21 +02:00
|
|
|
QTranslator appTranslator;
|
2020-07-10 03:17:23 +02:00
|
|
|
appTranslator.load(QLocale(), "nheko", "_", ":/translations");
|
2017-09-10 11:59:21 +02:00
|
|
|
app.installTranslator(&appTranslator);
|
2017-05-29 18:09:12 +02:00
|
|
|
|
2020-11-10 03:28:41 +01:00
|
|
|
MainWindow w;
|
2017-04-15 16:23:35 +02:00
|
|
|
|
2017-09-10 11:59:21 +02:00
|
|
|
// Move the MainWindow to the center
|
2017-12-14 20:55:00 +01:00
|
|
|
w.move(screenCenter(w.width(), w.height()));
|
2018-05-08 22:53:40 +02:00
|
|
|
|
2021-01-14 13:56:22 +01:00
|
|
|
if (!(settings.lock()->startInTray() && settings.lock()->tray()))
|
2018-05-08 22:53:40 +02:00
|
|
|
w.show();
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2018-06-10 19:03:45 +02:00
|
|
|
QObject::connect(&app, &QApplication::aboutToQuit, &w, [&w]() {
|
|
|
|
w.saveCurrentWindowSize();
|
2018-07-15 18:09:08 +02:00
|
|
|
if (http::client() != nullptr) {
|
2018-06-22 18:47:55 +02:00
|
|
|
nhlog::net()->debug("shutting down all I/O threads & open connections");
|
2018-07-15 18:09:08 +02:00
|
|
|
http::client()->close(true);
|
2018-06-22 18:47:55 +02:00
|
|
|
nhlog::net()->debug("bye");
|
2018-06-10 19:03:45 +02:00
|
|
|
}
|
|
|
|
});
|
2020-02-23 13:40:04 +01:00
|
|
|
QObject::connect(&app, &SingleApplication::instanceStarted, &w, [&w]() {
|
|
|
|
w.show();
|
|
|
|
w.raise();
|
|
|
|
w.activateWindow();
|
|
|
|
});
|
2017-07-30 12:59:28 +02:00
|
|
|
|
2021-01-10 18:36:06 +01:00
|
|
|
QObject::connect(
|
|
|
|
&app,
|
|
|
|
&SingleApplication::receivedMessage,
|
|
|
|
ChatPage::instance(),
|
|
|
|
[&](quint32, QByteArray message) { ChatPage::instance()->handleMatrixUri(message); });
|
|
|
|
|
|
|
|
QMetaObject::Connection uriConnection;
|
|
|
|
if (app.isPrimary() && !matrixUri.isEmpty()) {
|
|
|
|
uriConnection = QObject::connect(ChatPage::instance(),
|
|
|
|
&ChatPage::contentLoaded,
|
|
|
|
ChatPage::instance(),
|
|
|
|
[&uriConnection, matrixUri]() {
|
|
|
|
ChatPage::instance()->handleMatrixUri(
|
|
|
|
matrixUri.toUtf8());
|
|
|
|
QObject::disconnect(uriConnection);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
QDesktopServices::setUrlHandler("matrix", ChatPage::instance(), "handleMatrixUri");
|
|
|
|
|
2018-09-19 22:38:36 +02:00
|
|
|
#if defined(Q_OS_MAC)
|
|
|
|
// Temporary solution for the emoji picker until
|
|
|
|
// nheko has a proper menu bar with more functionality.
|
|
|
|
MacHelper::initializeMenus();
|
|
|
|
#endif
|
|
|
|
|
2018-06-14 01:28:35 +02:00
|
|
|
nhlog::ui()->info("starting nheko {}", nheko::version);
|
2018-06-09 15:03:14 +02:00
|
|
|
|
2017-09-10 11:59:21 +02:00
|
|
|
return app.exec();
|
2017-04-06 01:06:42 +02:00
|
|
|
}
|