2023-02-22 01:48:49 +03:00
|
|
|
// SPDX-FileCopyrightText: Nheko Contributors
|
2021-03-05 02:35:15 +03:00
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2019-12-23 07:22:03 +03:00
|
|
|
#include <iostream>
|
|
|
|
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QApplication>
|
2018-07-15 14:53:18 +03:00
|
|
|
#include <QCommandLineParser>
|
2021-01-10 20:36:06 +03:00
|
|
|
#include <QDesktopServices>
|
2018-06-12 20:36:16 +03:00
|
|
|
#include <QDir>
|
2017-11-16 17:33:52 +03:00
|
|
|
#include <QFile>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QFontDatabase>
|
2017-12-14 22:55:00 +03:00
|
|
|
#include <QLabel>
|
2017-05-29 19:09:12 +03:00
|
|
|
#include <QLibraryInfo>
|
2018-07-22 20:26:50 +03:00
|
|
|
#include <QMessageBox>
|
2017-12-14 22:55:00 +03:00
|
|
|
#include <QPoint>
|
2022-01-12 21:09:46 +03:00
|
|
|
#include <QQuickView>
|
2019-07-05 04:20:19 +03:00
|
|
|
#include <QScreen>
|
2018-06-09 16:03:14 +03:00
|
|
|
#include <QStandardPaths>
|
2017-05-29 19:09:12 +03:00
|
|
|
#include <QTranslator>
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2024-01-09 05:05:00 +03:00
|
|
|
// in theory we can enable this everywhere, but the header is missing on some of our CI systems and
|
|
|
|
// it is too much effort to install.
|
2024-06-20 20:22:20 +03:00
|
|
|
#if __has_include(<QtGui/qpa/qplatformwindow_p.h>)
|
2023-11-20 05:20:26 +03:00
|
|
|
#include <QtGui/qpa/qplatformwindow_p.h>
|
|
|
|
#endif
|
|
|
|
|
2023-11-20 04:38:58 +03:00
|
|
|
#include <kdsingleapplication.h>
|
|
|
|
|
2023-10-21 20:45:00 +03:00
|
|
|
#include "Cache.h"
|
2024-01-15 00:25:58 +03:00
|
|
|
#include "CallManager.h"
|
2021-01-10 20:36:06 +03:00
|
|
|
#include "ChatPage.h"
|
2018-07-17 16:37:25 +03:00
|
|
|
#include "Logging.h"
|
2017-04-06 02:06:42 +03:00
|
|
|
#include "MainWindow.h"
|
2018-06-09 16:03:14 +03:00
|
|
|
#include "MatrixClient.h"
|
2018-07-22 19:48:58 +03:00
|
|
|
#include "Utils.h"
|
2018-09-30 14:33:54 +03:00
|
|
|
#include "config/nheko.h"
|
2017-12-14 22:55:00 +03:00
|
|
|
|
2024-01-09 05:17:28 +03:00
|
|
|
#if defined(Q_OS_MACOS)
|
2018-09-19 23:38:36 +03:00
|
|
|
#include "emoji/MacHelper.h"
|
2022-11-04 19:42:09 +03:00
|
|
|
#include "notifications/Manager.h"
|
2018-09-19 23:38:36 +03:00
|
|
|
#endif
|
|
|
|
|
2023-01-22 07:01:50 +03:00
|
|
|
#ifdef GSTREAMER_AVAILABLE
|
2022-01-03 00:15:10 +03:00
|
|
|
#include <QAbstractEventDispatcher>
|
|
|
|
#include <gst/gst.h>
|
2023-01-22 07:01:50 +03:00
|
|
|
|
|
|
|
#include "voip/CallDevices.h"
|
2022-01-03 00:15:10 +03:00
|
|
|
#endif
|
|
|
|
|
2019-12-15 01:48:02 +03:00
|
|
|
#ifdef QML_DEBUGGING
|
|
|
|
#include <QQmlDebuggingEnabler>
|
2024-01-08 02:21:42 +03:00
|
|
|
QQmlTriviallyDestructibleDebuggingEnabler enabler;
|
2019-12-15 01:48:02 +03:00
|
|
|
#endif
|
|
|
|
|
2021-06-30 16:01:41 +03:00
|
|
|
#if HAVE_BACKTRACE_SYMBOLS_FD
|
2019-12-23 07:22:03 +03:00
|
|
|
#include <csignal>
|
2021-06-30 16:01:41 +03:00
|
|
|
#include <execinfo.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
2019-12-23 07:22:03 +03:00
|
|
|
|
2018-07-01 19:40:53 +03:00
|
|
|
void
|
|
|
|
stacktraceHandler(int signum)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
std::signal(signum, SIG_DFL);
|
2021-06-30 16:01:41 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
// boost::stacktrace::safe_dump_to("./nheko-backtrace.dump");
|
2021-06-30 16:01:41 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
// see
|
|
|
|
// https://stackoverflow.com/questions/77005/how-to-automatically-generate-a-stacktrace-when-my-program-crashes/77336#77336
|
|
|
|
void *array[50];
|
2022-10-26 02:10:35 +03:00
|
|
|
int size;
|
2021-06-30 16:01:41 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
// get void*'s for all entries on the stack
|
|
|
|
size = backtrace(array, 50);
|
2021-06-30 16:01:41 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
// print out all the frames to stderr
|
|
|
|
fprintf(stderr, "Error: signal %d:\n", signum);
|
|
|
|
backtrace_symbols_fd(array, size, STDERR_FILENO);
|
2021-06-30 16:01:41 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
int file = ::open("/tmp/nheko-crash.dump",
|
|
|
|
O_CREAT | O_WRONLY | O_TRUNC
|
2021-06-30 16:01:41 +03:00
|
|
|
#if defined(S_IWUSR) && defined(S_IRUSR)
|
2021-09-18 01:22:33 +03:00
|
|
|
,
|
|
|
|
S_IWUSR | S_IRUSR
|
2021-06-30 16:01:41 +03:00
|
|
|
#elif defined(S_IWRITE) && defined(S_IREAD)
|
2021-09-18 01:22:33 +03:00
|
|
|
,
|
|
|
|
S_IWRITE | S_IREAD
|
2021-06-30 16:01:41 +03:00
|
|
|
#endif
|
2021-09-18 01:22:33 +03:00
|
|
|
);
|
|
|
|
if (file != -1) {
|
|
|
|
constexpr char header[] = "Error: signal\n";
|
|
|
|
[[maybe_unused]] auto ret = write(file, header, std::size(header) - 1);
|
|
|
|
backtrace_symbols_fd(array, size, file);
|
|
|
|
close(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::raise(SIGABRT);
|
2018-07-01 19:40:53 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
registerSignalHandlers()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
std::signal(SIGSEGV, &stacktraceHandler);
|
|
|
|
std::signal(SIGABRT, &stacktraceHandler);
|
2018-07-01 19:40:53 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
// No implementation for systems with no stacktrace support.
|
|
|
|
void
|
|
|
|
registerSignalHandlers()
|
2022-09-25 21:05:08 +03:00
|
|
|
{
|
|
|
|
}
|
2018-07-01 19:40:53 +03:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2024-01-09 05:17:28 +03:00
|
|
|
#if defined(GSTREAMER_AVAILABLE) && (defined(Q_OS_MACOS) || defined(Q_OS_WINDOWS))
|
2022-01-03 00:15:10 +03:00
|
|
|
GMainLoop *gloop = 0;
|
|
|
|
GThread *gthread = 0;
|
|
|
|
|
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
static gpointer glibMainLoopThreadFunc(gpointer)
|
|
|
|
{
|
|
|
|
gloop = g_main_loop_new(0, false);
|
|
|
|
g_main_loop_run(gloop);
|
|
|
|
g_main_loop_unref(gloop);
|
|
|
|
gloop = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} // extern "C"
|
|
|
|
#endif
|
|
|
|
|
2017-12-14 22:55:00 +03:00
|
|
|
QPoint
|
|
|
|
screenCenter(int width, int height)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
// Deprecated in 5.13: QRect screenGeometry = QApplication::desktop()->screenGeometry();
|
|
|
|
QRect screenGeometry = QGuiApplication::primaryScreen()->geometry();
|
2017-12-14 22:55:00 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
int x = (screenGeometry.width() - width) / 2;
|
|
|
|
int y = (screenGeometry.height() - height) / 2;
|
2017-12-14 22:55:00 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
return QPoint(x, y);
|
2017-12-14 22:55:00 +03:00
|
|
|
}
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2018-06-12 20:36:16 +03:00
|
|
|
void
|
2021-01-19 21:44:22 +03:00
|
|
|
createStandardDirectory(QStandardPaths::StandardLocation path)
|
2018-06-12 20:36:16 +03:00
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
auto dir = QStandardPaths::writableLocation(path);
|
2018-06-12 20:36:16 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
if (!QDir().mkpath(dir)) {
|
|
|
|
throw std::runtime_error(("Unable to create state directory:" + dir).toStdString().c_str());
|
|
|
|
}
|
2018-06-12 20:36:16 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2021-12-29 06:28:08 +03:00
|
|
|
QCoreApplication::setApplicationName(QStringLiteral("nheko"));
|
2021-09-18 01:22:33 +03:00
|
|
|
QCoreApplication::setApplicationVersion(nheko::version);
|
2021-12-29 06:28:08 +03:00
|
|
|
QCoreApplication::setOrganizationName(QStringLiteral("nheko"));
|
2021-09-18 01:22:33 +03:00
|
|
|
|
2023-02-25 03:46:18 +03:00
|
|
|
// Disable the qml disk cache by default to prevent crashes on updates. See
|
|
|
|
// https://github.com/Nheko-Reborn/nheko/issues/1383
|
|
|
|
if (qgetenv("NHEKO_ALLOW_QML_DISK_CACHE").size() == 0) {
|
|
|
|
qputenv("QML_DISABLE_DISK_CACHE", "1");
|
|
|
|
}
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
// this needs to be after setting the application name. Or how would we find our settings
|
|
|
|
// file then?
|
2024-06-20 22:40:53 +03:00
|
|
|
#if !defined(Q_OS_MACOS)
|
2021-09-18 01:22:33 +03:00
|
|
|
if (qgetenv("QT_SCALE_FACTOR").size() == 0) {
|
|
|
|
float factor = utils::scaleFactor();
|
2018-07-22 19:48:58 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
if (factor != -1)
|
|
|
|
qputenv("QT_SCALE_FACTOR", QString::number(factor).toUtf8());
|
|
|
|
}
|
2018-07-22 19:48:58 +03:00
|
|
|
#endif
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
QString matrixUri;
|
|
|
|
for (int i = 1; i < argc; ++i) {
|
|
|
|
QString arg{argv[i]};
|
2024-05-27 22:15:44 +03:00
|
|
|
if (arg.startsWith(QLatin1String("matrix:"))) {
|
2021-09-18 01:22:33 +03:00
|
|
|
matrixUri = arg;
|
2019-01-19 19:20:41 +03:00
|
|
|
}
|
2021-09-18 01:22:33 +03:00
|
|
|
}
|
|
|
|
|
2023-11-20 04:38:58 +03:00
|
|
|
QApplication app(argc, argv);
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
QCommandLineParser parser;
|
|
|
|
parser.addHelpOption();
|
|
|
|
parser.addVersionOption();
|
2022-08-23 22:47:52 +03:00
|
|
|
QCommandLineOption debugOption(QStringLiteral("debug"),
|
|
|
|
QObject::tr("Alias for '--log-level trace'."));
|
2021-09-18 01:22:33 +03:00
|
|
|
parser.addOption(debugOption);
|
2022-08-23 22:47:52 +03:00
|
|
|
QCommandLineOption logLevel(
|
|
|
|
QStringList() << QStringLiteral("l") << QStringLiteral("log-level"),
|
|
|
|
QObject::tr("Set the global log level, or a comma-separated list of <component>=<level> "
|
|
|
|
"pairs, or both. For example, to set the default log level to 'warn' but "
|
|
|
|
"disable logging for the 'ui' component, pass 'warn,ui=off'. "
|
|
|
|
"levels:{trace,debug,info,warning,error,critical,off} "
|
|
|
|
"components:{crypto,db,mtx,net,qml,ui}"),
|
|
|
|
QObject::tr("level"));
|
|
|
|
parser.addOption(logLevel);
|
|
|
|
QCommandLineOption logType(
|
|
|
|
QStringList() << QStringLiteral("L") << QStringLiteral("log-type"),
|
|
|
|
QObject::tr("Set the log output type. A comma-separated list is allowed. "
|
|
|
|
"The default is 'file,stderr'. types:{file,stderr,none}"),
|
|
|
|
QObject::tr("type"));
|
|
|
|
parser.addOption(logType);
|
2023-10-21 20:45:00 +03:00
|
|
|
QCommandLineOption compactDb(
|
|
|
|
QStringList() << QStringLiteral("C") << QStringLiteral("compact"),
|
|
|
|
QObject::tr("Recompacts the database which might improve performance."));
|
|
|
|
parser.addOption(compactDb);
|
2021-09-18 01:22:33 +03: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.
|
|
|
|
QCommandLineOption configName(
|
2021-12-29 08:01:38 +03:00
|
|
|
QStringList() << QStringLiteral("p") << QStringLiteral("profile"),
|
2022-09-30 07:23:39 +03:00
|
|
|
QCoreApplication::tr("Create a unique profile which allows you to log into several "
|
2021-09-18 01:22:33 +03:00
|
|
|
"accounts at the same time and start multiple instances of nheko."),
|
|
|
|
QCoreApplication::tr("profile"),
|
|
|
|
QCoreApplication::tr("profile name"));
|
|
|
|
parser.addOption(configName);
|
|
|
|
|
|
|
|
parser.process(app);
|
|
|
|
|
2023-10-21 20:45:00 +03:00
|
|
|
if (parser.isSet(compactDb))
|
|
|
|
cache::setNeedsCompactFlag();
|
|
|
|
|
2024-05-27 22:15:44 +03:00
|
|
|
if (parser.isSet(configName))
|
|
|
|
UserSettings::initialize(parser.value(configName));
|
|
|
|
else
|
|
|
|
UserSettings::initialize(std::nullopt);
|
|
|
|
|
|
|
|
auto settings = UserSettings::instance().toWeakRef();
|
|
|
|
|
|
|
|
auto profileName = settings.lock()->profile();
|
|
|
|
|
|
|
|
KDSingleApplication singleapp(
|
|
|
|
QStringLiteral("im.nheko.nheko-%1")
|
|
|
|
.arg(profileName == QLatin1String("default") ? QLatin1String("") : profileName));
|
|
|
|
|
2021-09-28 00:20:25 +03:00
|
|
|
// This check needs to happen _after_ process(), so that we actually print help for --help when
|
|
|
|
// Nheko is already running.
|
2023-11-20 04:38:58 +03:00
|
|
|
if (!singleapp.isPrimaryInstance()) {
|
2023-11-20 05:20:26 +03:00
|
|
|
auto token = qgetenv("XDG_ACTIVATION_TOKEN");
|
|
|
|
|
2024-06-20 20:22:20 +03:00
|
|
|
#if __has_include(<QtGui/qpa/qplatformwindow_p.h>) && \
|
|
|
|
((QT_VERSION >= QT_VERSION_CHECK(6, 7, 0) && QT_CONFIG(wayland)) || \
|
|
|
|
(QT_VERSION < QT_VERSION_CHECK(6, 7, 0) && defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)))
|
2023-11-20 05:20:26 +03:00
|
|
|
// getting a valid activation token on wayland is a bit of a pain, it works most reliably
|
|
|
|
// when you have an actual window, that has the focus...
|
|
|
|
auto waylandApp = app.nativeInterface<QNativeInterface::QWaylandApplication>();
|
2024-05-16 13:32:50 +03:00
|
|
|
// When the token is set in the env, use it by default as that's what we're supposed to do
|
|
|
|
// But leave a env knob so users can workaround terminal emulators that leak tokens
|
|
|
|
if (waylandApp &&
|
|
|
|
(!qEnvironmentVariableIsEmpty("NHEKO_FORCE_ACTIVATION_SPLASH") || token.isEmpty())) {
|
2023-11-20 05:20:26 +03:00
|
|
|
QQuickView window;
|
|
|
|
window.setTitle("Activate main instance");
|
|
|
|
window.setMaximumSize(QSize(100, 50));
|
|
|
|
window.setMinimumSize(QSize(100, 50));
|
|
|
|
window.setResizeMode(QQuickView::ResizeMode::SizeRootObjectToView);
|
|
|
|
window.setSource(QUrl(QStringLiteral("qrc:///resources/qml/ui/Spinner.qml")));
|
|
|
|
window.show();
|
|
|
|
auto waylandWindow =
|
|
|
|
window.nativeInterface<QNativeInterface::Private::QWaylandWindow>();
|
|
|
|
if (waylandWindow) {
|
|
|
|
std::cout << "Launching temp window to activate main instance!\n";
|
|
|
|
QObject::connect(
|
|
|
|
waylandWindow,
|
|
|
|
&QNativeInterface::Private::QWaylandWindow::xdgActivationTokenCreated,
|
|
|
|
waylandWindow,
|
|
|
|
[&token, &app](QString newToken) { // clazy:exclude=lambda-in-connect
|
|
|
|
token = newToken.toUtf8();
|
|
|
|
app.exit();
|
|
|
|
},
|
|
|
|
Qt::SingleShotConnection);
|
|
|
|
QTimer::singleShot(100, waylandWindow, [waylandWindow, waylandApp] {
|
|
|
|
waylandWindow->requestXdgActivationToken(waylandApp->lastInputSerial());
|
|
|
|
});
|
|
|
|
app.exec();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
std::cout << "Activating main app (instead of opening it a second time)."
|
|
|
|
<< token.toStdString() << std::endl;
|
|
|
|
|
2021-11-05 01:10:03 +03:00
|
|
|
// open uri in main instance
|
2023-11-20 04:38:58 +03:00
|
|
|
// TODO(Nico): Send also an activation token.
|
2023-11-20 05:20:26 +03:00
|
|
|
singleapp.sendMessage("activate" + token);
|
2023-11-20 04:38:58 +03:00
|
|
|
|
|
|
|
if (!matrixUri.isEmpty()) {
|
|
|
|
std::cout << "Sending Matrix URL to main application: " << matrixUri.toStdString()
|
|
|
|
<< std::endl;
|
|
|
|
// open uri in main instance
|
|
|
|
singleapp.sendMessage(matrixUri.toUtf8());
|
|
|
|
}
|
|
|
|
|
2021-09-28 00:20:25 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-09 05:17:28 +03:00
|
|
|
#if !defined(Q_OS_MACOS)
|
2021-12-29 06:28:08 +03:00
|
|
|
app.setWindowIcon(QIcon::fromTheme(QStringLiteral("nheko"), QIcon{":/logos/nheko.png"}));
|
2021-12-15 01:05:48 +03:00
|
|
|
#endif
|
2022-02-10 01:46:49 +03:00
|
|
|
#ifdef NHEKO_FLATPAK
|
2023-06-03 01:50:55 +03:00
|
|
|
app.setDesktopFileName(QStringLiteral("im.nheko.Nheko"));
|
2022-02-10 01:46:49 +03:00
|
|
|
#else
|
|
|
|
app.setDesktopFileName(QStringLiteral("nheko"));
|
|
|
|
#endif
|
2021-12-20 00:54:50 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
http::init();
|
|
|
|
|
|
|
|
createStandardDirectory(QStandardPaths::CacheLocation);
|
|
|
|
createStandardDirectory(QStandardPaths::AppDataLocation);
|
|
|
|
|
|
|
|
registerSignalHandlers();
|
|
|
|
|
2024-01-09 05:17:28 +03:00
|
|
|
#if defined(GSTREAMER_AVAILABLE) && (defined(Q_OS_MACOS) || defined(Q_OS_WINDOWS))
|
2022-01-03 00:15:10 +03:00
|
|
|
// If the version of Qt we're running in does not use GLib, we need to
|
|
|
|
// start a GMainLoop so that gstreamer can dispatch events.
|
|
|
|
const QMetaObject *mo = QAbstractEventDispatcher::instance(qApp->thread())->metaObject();
|
|
|
|
if (gloop == 0 && strcmp(mo->className(), "QEventDispatcherGlib") != 0 &&
|
|
|
|
strcmp(mo->superClass()->className(), "QEventDispatcherGlib") != 0) {
|
|
|
|
gthread = g_thread_new(0, glibMainLoopThreadFunc, 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
try {
|
2022-08-23 22:47:52 +03:00
|
|
|
QString level;
|
|
|
|
if (parser.isSet(logLevel)) {
|
|
|
|
level = parser.value(logLevel);
|
|
|
|
} else if (parser.isSet(debugOption)) {
|
|
|
|
level = "trace";
|
|
|
|
} else {
|
|
|
|
level = qEnvironmentVariable("NHEKO_LOG_LEVEL");
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList targets =
|
|
|
|
(parser.isSet(logType) ? parser.value(logType)
|
|
|
|
: qEnvironmentVariable("NHEKO_LOG_TYPE", "file,stderr"))
|
|
|
|
.split(',', Qt::SkipEmptyParts);
|
|
|
|
targets.removeAll("none");
|
|
|
|
bool to_stderr = bool(targets.removeAll("stderr"));
|
|
|
|
QString path = targets.removeAll("file")
|
|
|
|
? QDir(QStandardPaths::writableLocation(QStandardPaths::CacheLocation))
|
|
|
|
.filePath("nheko.log")
|
|
|
|
: QLatin1String("");
|
|
|
|
if (!targets.isEmpty()) {
|
|
|
|
std::cerr << "Invalid log type '" << targets.first().toStdString().c_str() << "'"
|
|
|
|
<< std::endl;
|
|
|
|
std::exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
nhlog::init(level, path, to_stderr);
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
} catch (const spdlog::spdlog_ex &ex) {
|
2022-08-23 22:47:52 +03:00
|
|
|
std::cerr << "Log initialization failed: " << ex.what() << std::endl;
|
2021-09-18 01:22:33 +03:00
|
|
|
std::exit(1);
|
|
|
|
}
|
|
|
|
|
2023-06-05 01:18:17 +03:00
|
|
|
auto filter = new NhekoFixupPaletteEventFilter(&app);
|
|
|
|
app.installEventFilter(filter);
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
QFont font;
|
|
|
|
QString userFontFamily = settings.lock()->font();
|
2021-12-29 06:28:08 +03:00
|
|
|
if (!userFontFamily.isEmpty() && userFontFamily != QLatin1String("default")) {
|
2021-09-18 01:22:33 +03:00
|
|
|
font.setFamily(userFontFamily);
|
|
|
|
}
|
|
|
|
font.setPointSizeF(settings.lock()->fontSize());
|
|
|
|
|
|
|
|
app.setFont(font);
|
2022-01-12 21:09:46 +03:00
|
|
|
settings.lock()->applyTheme();
|
2021-09-18 01:22:33 +03:00
|
|
|
|
2021-11-27 05:05:32 +03:00
|
|
|
if (QLocale().language() == QLocale::C)
|
|
|
|
QLocale::setDefault(QLocale(QLocale::English, QLocale::UnitedKingdom));
|
2021-09-18 01:22:33 +03:00
|
|
|
|
|
|
|
QTranslator qtTranslator;
|
2023-06-02 02:37:53 +03:00
|
|
|
if (qtTranslator.load(QLocale(),
|
2023-07-13 02:10:09 +03:00
|
|
|
QStringLiteral("qtbase"),
|
2023-06-02 02:37:53 +03:00
|
|
|
QStringLiteral("_"),
|
2023-07-13 02:10:09 +03:00
|
|
|
QLibraryInfo::path(QLibraryInfo::TranslationsPath))) {
|
2023-06-02 01:24:26 +03:00
|
|
|
app.installTranslator(&qtTranslator);
|
2023-07-13 02:10:09 +03:00
|
|
|
} else
|
|
|
|
qDebug() << "Failed to load qtbase translations: "
|
|
|
|
<< QLibraryInfo::path(QLibraryInfo::TranslationsPath);
|
|
|
|
QTranslator qmlTranslator;
|
|
|
|
if (qmlTranslator.load(QLocale(),
|
|
|
|
QStringLiteral("qtdeclarative"),
|
|
|
|
QStringLiteral("_"),
|
|
|
|
QLibraryInfo::path(QLibraryInfo::TranslationsPath))) {
|
|
|
|
app.installTranslator(&qmlTranslator);
|
|
|
|
} else
|
|
|
|
qDebug() << "Failed to load qtdeclarative translations";
|
2021-09-18 01:22:33 +03:00
|
|
|
|
|
|
|
QTranslator appTranslator;
|
2023-06-02 02:37:53 +03:00
|
|
|
if (appTranslator.load(QLocale(),
|
|
|
|
QStringLiteral("nheko"),
|
|
|
|
QStringLiteral("_"),
|
|
|
|
QStringLiteral(":/translations")))
|
2023-06-02 01:24:26 +03:00
|
|
|
app.installTranslator(&appTranslator);
|
2023-07-13 02:10:09 +03:00
|
|
|
else
|
|
|
|
qDebug() << "Failed to load nheko translations";
|
2021-09-18 01:22:33 +03:00
|
|
|
|
2023-06-19 02:38:40 +03:00
|
|
|
MainWindow w(nullptr);
|
2022-01-12 21:09:46 +03:00
|
|
|
// QQuickView w;
|
2021-09-18 01:22:33 +03:00
|
|
|
|
|
|
|
// Move the MainWindow to the center
|
2022-01-12 21:09:46 +03:00
|
|
|
// w.move(screenCenter(w.width(), w.height()));
|
2021-09-18 01:22:33 +03:00
|
|
|
|
|
|
|
if (!(settings.lock()->startInTray() && settings.lock()->tray()))
|
|
|
|
w.show();
|
|
|
|
|
|
|
|
QObject::connect(&app, &QApplication::aboutToQuit, &w, [&w]() {
|
2023-02-01 04:54:01 +03:00
|
|
|
ChatPage::instance()->removeAllNotifications();
|
2022-01-29 20:10:42 +03:00
|
|
|
w.saveCurrentWindowSize();
|
2021-09-18 01:22:33 +03:00
|
|
|
if (http::client() != nullptr) {
|
|
|
|
nhlog::net()->debug("shutting down all I/O threads & open connections");
|
|
|
|
http::client()->close(true);
|
|
|
|
nhlog::net()->debug("bye");
|
2021-01-10 20:36:06 +03:00
|
|
|
}
|
2024-01-15 00:25:58 +03:00
|
|
|
// This is required in order to destroy CallManager's QMediaPlayer, in turn allowing it
|
|
|
|
// to destroy its GstPipeline so that gst_deinit() can return.
|
|
|
|
ChatPage::instance()->callManager()->deleteLater();
|
2021-09-18 01:22:33 +03:00
|
|
|
});
|
|
|
|
|
2021-12-17 07:53:34 +03:00
|
|
|
// It seems like handling the message in a blocking manner is a no-go. I have no idea how to
|
|
|
|
// fix that, so just use a queued connection for now... (ASAN does not cooperate and just
|
|
|
|
// hides the crash D:)
|
2021-09-18 01:22:33 +03:00
|
|
|
QObject::connect(
|
2023-11-20 04:38:58 +03:00
|
|
|
&singleapp,
|
|
|
|
&KDSingleApplication::messageReceived,
|
2021-09-18 01:22:33 +03:00
|
|
|
ChatPage::instance(),
|
2023-11-20 04:38:58 +03:00
|
|
|
[&](QByteArray message) {
|
|
|
|
if (message.isEmpty() || message.startsWith("activate")) {
|
2023-11-20 05:20:26 +03:00
|
|
|
auto token = message.remove(0, sizeof("activate") - 1);
|
|
|
|
if (!token.isEmpty()) {
|
|
|
|
nhlog::ui()->debug("Setting activation token to: {}", token.toStdString());
|
|
|
|
qputenv("XDG_ACTIVATION_TOKEN", token);
|
|
|
|
}
|
2023-11-20 04:38:58 +03:00
|
|
|
w.show();
|
|
|
|
w.raise();
|
|
|
|
w.requestActivate();
|
|
|
|
} else {
|
|
|
|
QString m = QString::fromUtf8(message);
|
|
|
|
ChatPage::instance()->handleMatrixUri(m);
|
|
|
|
}
|
2021-12-17 07:53:34 +03:00
|
|
|
},
|
|
|
|
Qt::QueuedConnection);
|
2021-09-18 01:22:33 +03:00
|
|
|
|
|
|
|
QMetaObject::Connection uriConnection;
|
2023-11-20 04:38:58 +03:00
|
|
|
if (singleapp.isPrimaryInstance() && !matrixUri.isEmpty()) {
|
2021-12-17 07:53:34 +03:00
|
|
|
uriConnection = QObject::connect(ChatPage::instance(),
|
|
|
|
&ChatPage::contentLoaded,
|
|
|
|
ChatPage::instance(),
|
|
|
|
[&uriConnection, matrixUri]() {
|
|
|
|
ChatPage::instance()->handleMatrixUri(matrixUri);
|
|
|
|
QObject::disconnect(uriConnection);
|
|
|
|
});
|
2021-09-18 01:22:33 +03:00
|
|
|
}
|
2021-12-29 08:01:38 +03:00
|
|
|
QDesktopServices::setUrlHandler(
|
|
|
|
QStringLiteral("matrix"), ChatPage::instance(), "handleMatrixUri");
|
2021-01-10 20:36:06 +03:00
|
|
|
|
2024-01-09 05:17:28 +03:00
|
|
|
#if defined(Q_OS_MACOS)
|
2021-09-18 01:22:33 +03:00
|
|
|
// Temporary solution for the emoji picker until
|
|
|
|
// nheko has a proper menu bar with more functionality.
|
|
|
|
MacHelper::initializeMenus();
|
2022-11-04 19:42:09 +03:00
|
|
|
|
|
|
|
// Need to set up notification delegate so users can respond to messages from within the
|
|
|
|
// notification itself.
|
|
|
|
NotificationsManager::attachToMacNotifCenter();
|
2018-09-19 23:38:36 +03:00
|
|
|
#endif
|
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
nhlog::ui()->info("starting nheko {}", nheko::version);
|
2018-06-09 16:03:14 +03:00
|
|
|
|
2023-01-22 07:01:50 +03:00
|
|
|
auto returnvalue = app.exec();
|
|
|
|
|
|
|
|
#ifdef GSTREAMER_AVAILABLE
|
|
|
|
CallDevices::instance().deinit();
|
|
|
|
|
|
|
|
gst_deinit();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return returnvalue;
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|