// SPDX-FileCopyrightText: Nheko Contributors // // SPDX-License-Identifier: GPL-3.0-or-later #include #include #include #include #include "AliasEditModel.h" #include "BlurhashProvider.h" #include "Cache.h" #include "Cache_p.h" #include "ChatPage.h" #include "Clipboard.h" #include "ColorImageProvider.h" #include "CombinedImagePackModel.h" #include "CompletionProxyModel.h" #include "Config.h" #include "EventAccessors.h" #include "GridImagePackModel.h" #include "ImagePackListModel.h" #include "InviteesModel.h" #include "JdenticonProvider.h" #include "Logging.h" #include "LoginPage.h" #include "MainWindow.h" #include "MatrixClient.h" #include "MemberList.h" #include "MxcImageProvider.h" #include "PowerlevelsEditModels.h" #include "ReadReceiptsModel.h" #include "RegisterPage.h" #include "RoomDirectoryModel.h" #include "RoomsModel.h" #include "SingleImagePackModel.h" #include "TrayIcon.h" #include "UserDirectoryModel.h" #include "UserSettingsPage.h" #include "UsersModel.h" #include "Utils.h" #include "dock/Dock.h" #include "emoji/Provider.h" #include "encryption/DeviceVerificationFlow.h" #include "encryption/SelfVerificationStatus.h" #include "timeline/DelegateChooser.h" #include "timeline/TimelineFilter.h" #include "timeline/TimelineViewManager.h" #include "ui/HiddenEvents.h" #include "ui/MxcAnimatedImage.h" #include "ui/MxcMediaProxy.h" #include "ui/NhekoCursorShape.h" #include "ui/NhekoDropArea.h" #include "ui/NhekoEventObserver.h" #include "ui/NhekoGlobalObject.h" #include "ui/RoomSummary.h" #include "ui/UIA.h" #include "voip/CallManager.h" #include "voip/WebRTCSession.h" #ifdef NHEKO_DBUS_SYS #include "dbus/NhekoDBusApi.h" #endif Q_DECLARE_METATYPE(mtx::events::collections::TimelineEvents) Q_DECLARE_METATYPE(std::vector) Q_DECLARE_METATYPE(std::vector) Q_DECLARE_METATYPE(mtx::responses::PublicRoom) Q_DECLARE_METATYPE(mtx::responses::Profile) Q_DECLARE_METATYPE(mtx::responses::User) MainWindow *MainWindow::instance_ = nullptr; MainWindow::MainWindow(QWindow *parent) : QQuickView(parent) , userSettings_{UserSettings::instance()} { instance_ = this; MainWindow::setWindowTitle(0); setObjectName(QStringLiteral("MainWindow")); setResizeMode(QQuickView::SizeRootObjectToView); setMinimumHeight(conf::window::minHeight); setMinimumWidth(conf::window::minWidth); restoreWindowSize(); chat_page_ = new ChatPage(userSettings_, this); registerQmlTypes(); setColor(Theme::paletteFromTheme(userSettings_->theme()).window().color()); setSource(QUrl(QStringLiteral("qrc:///qml/Root.qml"))); trayIcon_ = new TrayIcon(QStringLiteral(":/logos/nheko.svg"), this); connect(chat_page_, &ChatPage::closing, this, [this] { switchToLoginPage(""); }); connect(chat_page_, &ChatPage::unreadMessages, this, &MainWindow::setWindowTitle); connect(chat_page_, SIGNAL(unreadMessages(int)), trayIcon_, SLOT(setUnreadCount(int))); connect(chat_page_, &ChatPage::showLoginPage, this, &MainWindow::switchToLoginPage); connect(chat_page_, &ChatPage::showNotification, this, &MainWindow::showNotification); connect(userSettings_.get(), &UserSettings::trayChanged, trayIcon_, &TrayIcon::setVisible); connect(trayIcon_, SIGNAL(activated(QSystemTrayIcon::ActivationReason)), this, SLOT(iconActivated(QSystemTrayIcon::ActivationReason))); trayIcon_->setVisible(userSettings_->tray()); dock_ = new Dock(this); connect(chat_page_, SIGNAL(unreadMessages(int)), dock_, SLOT(setUnreadCount(int))); // load cache on event loop QTimer::singleShot(0, this, [this] { if (hasActiveUser()) { QString token = userSettings_->accessToken(); QString home_server = userSettings_->homeserver(); QString user_id = userSettings_->userId(); QString device_id = userSettings_->deviceId(); http::client()->set_access_token(token.toStdString()); http::client()->set_server(home_server.toStdString()); http::client()->set_device_id(device_id.toStdString()); try { using namespace mtx::identifiers; http::client()->set_user(parse(user_id.toStdString())); } catch (const std::invalid_argument &) { nhlog::ui()->critical("bootstrapped with invalid user_id: {}", user_id.toStdString()); } nhlog::ui()->info("User already signed in, showing chat page"); showChatPage(); } }); } void MainWindow::registerQmlTypes() { qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType(); qRegisterMetaType>(); qRegisterMetaType>(); qRegisterMetaType>(); qRegisterMetaType(); qmlRegisterUncreatableMetaObject(qml_mtx_events::staticMetaObject, "im.nheko", 1, 0, "MtxEvent", QStringLiteral("Can't instantiate enum!")); qmlRegisterUncreatableMetaObject( olm::staticMetaObject, "im.nheko", 1, 0, "Olm", QStringLiteral("Can't instantiate enum!")); qmlRegisterUncreatableMetaObject(crypto::staticMetaObject, "im.nheko", 1, 0, "Crypto", QStringLiteral("Can't instantiate enum!")); qmlRegisterUncreatableMetaObject(verification::staticMetaObject, "im.nheko", 1, 0, "VerificationStatus", QStringLiteral("Can't instantiate enum!")); qmlRegisterType("im.nheko", 1, 0, "DelegateChoice"); qmlRegisterType("im.nheko", 1, 0, "DelegateChooser"); qmlRegisterType("im.nheko", 1, 0, "NhekoDropArea"); qmlRegisterType("im.nheko", 1, 0, "CursorShape"); qmlRegisterType("im.nheko", 1, 0, "EventObserver"); qmlRegisterType("im.nheko", 1, 0, "MxcAnimatedImage"); qmlRegisterType("im.nheko", 1, 0, "MxcMedia"); qmlRegisterType("im.nheko", 1, 0, "RoomDirectoryModel"); qmlRegisterType("im.nheko", 1, 0, "UserDirectoryModel"); qmlRegisterType("im.nheko", 1, 0, "Login"); qmlRegisterType("im.nheko", 1, 0, "Registration"); qmlRegisterType("im.nheko", 1, 0, "HiddenEvents"); qmlRegisterType("im.nheko", 1, 0, "TimelineFilter"); qmlRegisterUncreatableType( "im.nheko", 1, 0, "RoomSummary", QStringLiteral("Please use joinRoom to create a room summary.")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "AliasEditingModel", QStringLiteral("Please use editAliases to create the models")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "PowerlevelEditingModels", QStringLiteral("Please use editPowerlevels to create the models")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "DeviceVerificationFlow", QStringLiteral("Can't create verification flow from QML!")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "UserProfileModel", QStringLiteral("UserProfile needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "MemberList", QStringLiteral("MemberList needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "RoomSettingsModel", QStringLiteral("Room Settings needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "Room", QStringLiteral("Room needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "ImagePackListModel", QStringLiteral("ImagePackListModel needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "SingleImagePackModel", QStringLiteral("SingleImagePackModel needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "InviteesModel", QStringLiteral("InviteesModel needs to be instantiated on the C++ side")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "ReadReceiptsProxy", QStringLiteral("ReadReceiptsProxy needs to be instantiated on the C++ side")); qmlRegisterSingletonType( "im.nheko", 1, 0, "Clipboard", [](QQmlEngine *, QJSEngine *) -> QObject * { return new Clipboard(); }); qmlRegisterSingletonType( "im.nheko", 1, 0, "Nheko", [](QQmlEngine *, QJSEngine *) -> QObject * { return new Nheko(); }); qmlRegisterSingletonType( "im.nheko", 1, 0, "UserSettingsModel", [](QQmlEngine *, QJSEngine *) -> QObject * { return new UserSettingsModel(); }); qmlRegisterSingletonInstance("im.nheko", 1, 0, "Settings", userSettings_.data()); qRegisterMetaType(); qRegisterMetaType>(); qmlRegisterUncreatableType( "im.nheko", 1, 0, "FilteredCommunitiesModel", QStringLiteral("Use Communities.filtered() to create a FilteredCommunitiesModel")); qmlRegisterUncreatableType( "im.nheko", 1, 0, "MediaUpload", QStringLiteral("MediaUploads can not be created in Qml")); qmlRegisterUncreatableMetaObject(emoji::staticMetaObject, "im.nheko.EmojiModel", 1, 0, "EmojiCategory", QStringLiteral("Error: Only enums")); qmlRegisterType("im.nheko", 1, 0, "RoomDirectoryModel"); qmlRegisterSingletonType( "im.nheko", 1, 0, "SelfVerificationStatus", [](QQmlEngine *, QJSEngine *) -> QObject * { auto ptr = new SelfVerificationStatus(); QObject::connect(ChatPage::instance(), &ChatPage::initializeEmptyViews, ptr, &SelfVerificationStatus::invalidate); return ptr; }); qmlRegisterSingletonInstance("im.nheko", 1, 0, "MainWindow", this); qmlRegisterSingletonInstance("im.nheko", 1, 0, "UIA", UIA::instance()); qmlRegisterSingletonInstance( "im.nheko", 1, 0, "CallManager", ChatPage::instance()->callManager()); imgProvider = new MxcImageProvider(); engine()->addImageProvider(QStringLiteral("MxcImage"), imgProvider); engine()->addImageProvider(QStringLiteral("colorimage"), new ColorImageProvider()); engine()->addImageProvider(QStringLiteral("blurhash"), new BlurhashProvider()); if (JdenticonProvider::isAvailable()) engine()->addImageProvider(QStringLiteral("jdenticon"), new JdenticonProvider()); QObject::connect(engine(), &QQmlEngine::quit, &QGuiApplication::quit); #ifdef NHEKO_DBUS_SYS if (UserSettings::instance()->exposeDBusApi()) { if (QDBusConnection::sessionBus().isConnected() && QDBusConnection::sessionBus().registerService(NHEKO_DBUS_SERVICE_NAME)) { nheko::dbus::init(); nhlog::ui()->info("Initialized D-Bus"); dbusAvailable_ = true; } else nhlog::ui()->warn("Could not connect to D-Bus!"); } #endif } void MainWindow::setWindowTitle(int notificationCount) { QString name = QStringLiteral("nheko"); if (!userSettings_.data()->profile().isEmpty()) name += " | " + userSettings_.data()->profile(); if (notificationCount > 0) { name.append(QString{QStringLiteral(" (%1)")}.arg(notificationCount)); } QQuickView::setTitle(name); } bool MainWindow::event(QEvent *event) { auto type = event->type(); if (type == QEvent::Close) { closeEvent(static_cast(event)); } return QQuickView::event(event); } // HACK: https://bugreports.qt.io/browse/QTBUG-83972, qtwayland cannot auto hide menu void MainWindow::mousePressEvent(QMouseEvent *event) { #if defined(Q_OS_LINUX) if (QGuiApplication::platformName() == "wayland") { emit hideMenu(); } #endif return QQuickView::mousePressEvent(event); } void MainWindow::restoreWindowSize() { int savedWidth = userSettings_->qsettings()->value(QStringLiteral("window/width")).toInt(); int savedheight = userSettings_->qsettings()->value(QStringLiteral("window/height")).toInt(); nhlog::ui()->info("Restoring window size {}x{}", savedWidth, savedheight); if (savedWidth == 0 || savedheight == 0) resize(conf::window::width, conf::window::height); else resize(savedWidth, savedheight); } void MainWindow::saveCurrentWindowSize() { auto settings = userSettings_->qsettings(); QSize current = size(); settings->setValue(QStringLiteral("window/width"), current.width()); settings->setValue(QStringLiteral("window/height"), current.height()); } void MainWindow::showChatPage() { auto userid = QString::fromStdString(http::client()->user_id().to_string()); auto device_id = QString::fromStdString(http::client()->device_id()); auto homeserver = QString::fromStdString(http::client()->server_url()); auto token = QString::fromStdString(http::client()->access_token()); userSettings_.data()->setUserId(userid); userSettings_.data()->setAccessToken(token); userSettings_.data()->setDeviceId(device_id); userSettings_.data()->setHomeserver(homeserver); chat_page_->bootstrap(userid, homeserver, token); connect(cache::client(), &Cache::databaseReady, this, &MainWindow::secretsChanged); connect(cache::client(), &Cache::secretChanged, this, &MainWindow::secretsChanged); emit reload(); nhlog::ui()->info("Switching to chat page"); emit switchToChatPage(); } void MainWindow::closeEvent(QCloseEvent *event) { if (WebRTCSession::instance().state() != webrtc::State::DISCONNECTED) { if (QMessageBox::question( nullptr, QStringLiteral("nheko"), QStringLiteral("A call is in progress. Quit?")) != QMessageBox::Yes) { event->ignore(); return; } } if (!qApp->isSavingSession() && isVisible() && pageSupportsTray() && userSettings_->tray()) { event->ignore(); hide(); } } void MainWindow::iconActivated(QSystemTrayIcon::ActivationReason reason) { switch (reason) { case QSystemTrayIcon::Trigger: if (!isVisible()) { show(); } else { hide(); } break; default: break; } } bool MainWindow::hasActiveUser() { auto settings = userSettings_->qsettings(); QString prefix; if (userSettings_->profile() != QLatin1String("")) prefix = "profile/" + userSettings_->profile() + "/"; return settings->contains(prefix + "auth/access_token") && settings->contains(prefix + "auth/home_server") && settings->contains(prefix + "auth/user_id"); } bool MainWindow::pageSupportsTray() const { return !http::client()->access_token().empty(); } inline void MainWindow::showDialog(QWidget *dialog) { dialog->setWindowFlags(Qt::WindowType::Dialog | Qt::WindowType::WindowCloseButtonHint | Qt::WindowType::WindowTitleHint); dialog->raise(); dialog->show(); utils::centerWidget(dialog, this); dialog->window()->windowHandle()->setTransientParent(this); } void MainWindow::addPerRoomWindow(const QString &room, QWindow *window) { roomWindows_.insert(room, window); } void MainWindow::removePerRoomWindow(const QString &room, QWindow *window) { roomWindows_.remove(room, window); } QWindow * MainWindow::windowForRoom(const QString &room) { auto currMainWindowRoom = ChatPage::instance()->timelineManager()->rooms()->currentRoom(); if ((currMainWindowRoom && currMainWindowRoom->roomId() == room) || ChatPage::instance()->timelineManager()->rooms()->currentRoomPreview().roomid_ == room) return this; else if (auto res = roomWindows_.find(room); res != roomWindows_.end()) return res.value(); return nullptr; } QString MainWindow::focusedRoom() const { auto focus = QGuiApplication::focusWindow(); if (!focus) return {}; if (focus == this) { auto currMainWindowRoom = ChatPage::instance()->timelineManager()->rooms()->currentRoom(); if (currMainWindowRoom) return currMainWindowRoom->roomId(); else return ChatPage::instance()->timelineManager()->rooms()->currentRoomPreview().roomid_; } auto i = roomWindows_.constBegin(); while (i != roomWindows_.constEnd()) { if (i.value() == focus) return i.key(); ++i; } return nullptr; }