mirror of
https://github.com/Nheko-Reborn/nheko.git
synced 2024-11-25 20:48:52 +03:00
6c57fa6c5b
This also adds a property `roomId` to TimelineModel.
502 lines
15 KiB
C++
502 lines
15 KiB
C++
// SPDX-FileCopyrightText: 2017 Konstantinos Sideris <siderisk@auth.gr>
|
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
//
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
#include <QApplication>
|
|
#include <QLayout>
|
|
#include <QMessageBox>
|
|
#include <QPluginLoader>
|
|
#include <QSettings>
|
|
#include <QShortcut>
|
|
|
|
#include <mtx/requests.hpp>
|
|
#include <mtx/responses/login.hpp>
|
|
|
|
#include "Cache.h"
|
|
#include "Cache_p.h"
|
|
#include "ChatPage.h"
|
|
#include "Config.h"
|
|
#include "Logging.h"
|
|
#include "LoginPage.h"
|
|
#include "MainWindow.h"
|
|
#include "MatrixClient.h"
|
|
#include "MemberList.h"
|
|
#include "RegisterPage.h"
|
|
#include "TrayIcon.h"
|
|
#include "UserSettingsPage.h"
|
|
#include "Utils.h"
|
|
#include "WebRTCSession.h"
|
|
#include "WelcomePage.h"
|
|
#include "ui/LoadingIndicator.h"
|
|
#include "ui/OverlayModal.h"
|
|
#include "ui/SnackBar.h"
|
|
|
|
#include "dialogs/CreateRoom.h"
|
|
#include "dialogs/JoinRoom.h"
|
|
#include "dialogs/LeaveRoom.h"
|
|
#include "dialogs/Logout.h"
|
|
#include "dialogs/ReadReceipts.h"
|
|
|
|
MainWindow *MainWindow::instance_ = nullptr;
|
|
|
|
MainWindow::MainWindow(QWidget *parent)
|
|
: QMainWindow(parent)
|
|
, userSettings_{UserSettings::instance()}
|
|
{
|
|
instance_ = this;
|
|
|
|
setWindowTitle(0);
|
|
setObjectName("MainWindow");
|
|
|
|
modal_ = new OverlayModal(this);
|
|
|
|
restoreWindowSize();
|
|
|
|
QFont font;
|
|
font.setStyleStrategy(QFont::PreferAntialias);
|
|
setFont(font);
|
|
|
|
trayIcon_ = new TrayIcon(":/logos/nheko.svg", this);
|
|
|
|
welcome_page_ = new WelcomePage(this);
|
|
login_page_ = new LoginPage(this);
|
|
register_page_ = new RegisterPage(this);
|
|
chat_page_ = new ChatPage(userSettings_, this);
|
|
userSettingsPage_ = new UserSettingsPage(userSettings_, this);
|
|
|
|
// Initialize sliding widget manager.
|
|
pageStack_ = new QStackedWidget(this);
|
|
pageStack_->addWidget(welcome_page_);
|
|
pageStack_->addWidget(login_page_);
|
|
pageStack_->addWidget(register_page_);
|
|
pageStack_->addWidget(chat_page_);
|
|
pageStack_->addWidget(userSettingsPage_);
|
|
|
|
setCentralWidget(pageStack_);
|
|
|
|
connect(welcome_page_, SIGNAL(userLogin()), this, SLOT(showLoginPage()));
|
|
connect(welcome_page_, SIGNAL(userRegister()), this, SLOT(showRegisterPage()));
|
|
|
|
connect(login_page_, SIGNAL(backButtonClicked()), this, SLOT(showWelcomePage()));
|
|
connect(login_page_, &LoginPage::loggingIn, this, &MainWindow::showOverlayProgressBar);
|
|
connect(
|
|
register_page_, &RegisterPage::registering, this, &MainWindow::showOverlayProgressBar);
|
|
connect(
|
|
login_page_, &LoginPage::errorOccurred, this, [this]() { removeOverlayProgressBar(); });
|
|
connect(register_page_, &RegisterPage::errorOccurred, this, [this]() {
|
|
removeOverlayProgressBar();
|
|
});
|
|
connect(register_page_, SIGNAL(backButtonClicked()), this, SLOT(showWelcomePage()));
|
|
|
|
connect(chat_page_, &ChatPage::closing, this, &MainWindow::showWelcomePage);
|
|
connect(
|
|
chat_page_, &ChatPage::showOverlayProgressBar, this, &MainWindow::showOverlayProgressBar);
|
|
connect(chat_page_, &ChatPage::unreadMessages, this, &MainWindow::setWindowTitle);
|
|
connect(chat_page_, SIGNAL(unreadMessages(int)), trayIcon_, SLOT(setUnreadCount(int)));
|
|
connect(chat_page_, &ChatPage::showLoginPage, this, [this](const QString &msg) {
|
|
login_page_->showError(msg);
|
|
showLoginPage();
|
|
});
|
|
|
|
connect(userSettingsPage_, &UserSettingsPage::moveBack, this, [this]() {
|
|
pageStack_->setCurrentWidget(chat_page_);
|
|
});
|
|
|
|
connect(
|
|
userSettingsPage_, SIGNAL(trayOptionChanged(bool)), trayIcon_, SLOT(setVisible(bool)));
|
|
connect(
|
|
userSettingsPage_, &UserSettingsPage::themeChanged, chat_page_, &ChatPage::themeChanged);
|
|
connect(trayIcon_,
|
|
SIGNAL(activated(QSystemTrayIcon::ActivationReason)),
|
|
this,
|
|
SLOT(iconActivated(QSystemTrayIcon::ActivationReason)));
|
|
|
|
connect(chat_page_, SIGNAL(contentLoaded()), this, SLOT(removeOverlayProgressBar()));
|
|
|
|
connect(this, &MainWindow::focusChanged, chat_page_, &ChatPage::chatFocusChanged);
|
|
|
|
connect(
|
|
chat_page_, &ChatPage::showUserSettingsPage, this, &MainWindow::showUserSettingsPage);
|
|
|
|
connect(login_page_, &LoginPage::loginOk, this, [this](const mtx::responses::Login &res) {
|
|
http::client()->set_user(res.user_id);
|
|
showChatPage();
|
|
});
|
|
|
|
connect(register_page_, &RegisterPage::registerOk, this, &MainWindow::showChatPage);
|
|
|
|
QShortcut *quitShortcut = new QShortcut(QKeySequence::Quit, this);
|
|
connect(quitShortcut, &QShortcut::activated, this, QApplication::quit);
|
|
|
|
trayIcon_->setVisible(userSettings_->tray());
|
|
|
|
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>(user_id.toStdString()));
|
|
} catch (const std::invalid_argument &) {
|
|
nhlog::ui()->critical("bootstrapped with invalid user_id: {}",
|
|
user_id.toStdString());
|
|
}
|
|
|
|
showChatPage();
|
|
}
|
|
|
|
if (loadJdenticonPlugin()) {
|
|
nhlog::ui()->info("loaded jdenticon.");
|
|
}
|
|
}
|
|
|
|
void
|
|
MainWindow::setWindowTitle(int notificationCount)
|
|
{
|
|
QString name = "nheko";
|
|
|
|
if (!userSettings_.data()->profile().isEmpty())
|
|
name += " | " + userSettings_.data()->profile();
|
|
if (notificationCount > 0) {
|
|
name.append(QString{" (%1)"}.arg(notificationCount));
|
|
}
|
|
QMainWindow::setWindowTitle(name);
|
|
}
|
|
|
|
bool
|
|
MainWindow::event(QEvent *event)
|
|
{
|
|
auto type = event->type();
|
|
if (type == QEvent::WindowActivate) {
|
|
emit focusChanged(true);
|
|
} else if (type == QEvent::WindowDeactivate) {
|
|
emit focusChanged(false);
|
|
}
|
|
|
|
return QMainWindow::event(event);
|
|
}
|
|
|
|
void
|
|
MainWindow::restoreWindowSize()
|
|
{
|
|
QSettings settings;
|
|
int savedWidth = settings.value("window/width").toInt();
|
|
int savedheight = settings.value("window/height").toInt();
|
|
|
|
if (savedWidth == 0 || savedheight == 0)
|
|
resize(conf::window::width, conf::window::height);
|
|
else
|
|
resize(savedWidth, savedheight);
|
|
}
|
|
|
|
void
|
|
MainWindow::saveCurrentWindowSize()
|
|
{
|
|
QSettings settings;
|
|
QSize current = size();
|
|
|
|
settings.setValue("window/width", current.width());
|
|
settings.setValue("window/height", current.height());
|
|
}
|
|
|
|
void
|
|
MainWindow::removeOverlayProgressBar()
|
|
{
|
|
QTimer *timer = new QTimer(this);
|
|
timer->setSingleShot(true);
|
|
|
|
connect(timer, &QTimer::timeout, [this, timer]() {
|
|
timer->deleteLater();
|
|
|
|
if (modal_)
|
|
modal_->hide();
|
|
|
|
if (spinner_)
|
|
spinner_->stop();
|
|
});
|
|
|
|
// FIXME: Snackbar doesn't work if it's initialized in the constructor.
|
|
QTimer::singleShot(0, this, [this]() {
|
|
snackBar_ = new SnackBar(this);
|
|
connect(chat_page_, &ChatPage::showNotification, snackBar_, &SnackBar::showMessage);
|
|
});
|
|
|
|
timer->start(50);
|
|
}
|
|
|
|
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() + ":" +
|
|
std::to_string(http::client()->port()));
|
|
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);
|
|
|
|
showOverlayProgressBar();
|
|
|
|
pageStack_->setCurrentWidget(chat_page_);
|
|
|
|
pageStack_->removeWidget(welcome_page_);
|
|
pageStack_->removeWidget(login_page_);
|
|
pageStack_->removeWidget(register_page_);
|
|
|
|
login_page_->reset();
|
|
chat_page_->bootstrap(userid, homeserver, token);
|
|
connect(cache::client(),
|
|
&Cache::secretChanged,
|
|
userSettingsPage_,
|
|
&UserSettingsPage::updateSecretStatus);
|
|
emit reload();
|
|
}
|
|
|
|
void
|
|
MainWindow::closeEvent(QCloseEvent *event)
|
|
{
|
|
if (WebRTCSession::instance().state() != webrtc::State::DISCONNECTED) {
|
|
if (QMessageBox::question(this, "nheko", "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()
|
|
{
|
|
QSettings settings;
|
|
QString prefix;
|
|
if (userSettings_->profile() != "")
|
|
prefix = "profile/" + userSettings_->profile() + "/";
|
|
|
|
return settings.contains(prefix + "auth/access_token") &&
|
|
settings.contains(prefix + "auth/home_server") &&
|
|
settings.contains(prefix + "auth/user_id");
|
|
}
|
|
|
|
void
|
|
MainWindow::openLeaveRoomDialog(const QString &room_id)
|
|
{
|
|
auto dialog = new dialogs::LeaveRoom(this);
|
|
connect(dialog, &dialogs::LeaveRoom::leaving, this, [this, room_id]() {
|
|
chat_page_->leaveRoom(room_id);
|
|
});
|
|
|
|
showDialog(dialog);
|
|
}
|
|
|
|
void
|
|
MainWindow::showOverlayProgressBar()
|
|
{
|
|
spinner_ = new LoadingIndicator(this);
|
|
spinner_->setFixedHeight(100);
|
|
spinner_->setFixedWidth(100);
|
|
spinner_->setObjectName("ChatPageLoadSpinner");
|
|
spinner_->start();
|
|
|
|
showSolidOverlayModal(spinner_);
|
|
}
|
|
|
|
void
|
|
MainWindow::openJoinRoomDialog(std::function<void(const QString &room_id)> callback)
|
|
{
|
|
auto dialog = new dialogs::JoinRoom(this);
|
|
connect(dialog, &dialogs::JoinRoom::joinRoom, this, [callback](const QString &room) {
|
|
if (!room.isEmpty())
|
|
callback(room);
|
|
});
|
|
|
|
showDialog(dialog);
|
|
}
|
|
|
|
void
|
|
MainWindow::openCreateRoomDialog(
|
|
std::function<void(const mtx::requests::CreateRoom &request)> callback)
|
|
{
|
|
auto dialog = new dialogs::CreateRoom(this);
|
|
connect(dialog,
|
|
&dialogs::CreateRoom::createRoom,
|
|
this,
|
|
[callback](const mtx::requests::CreateRoom &request) { callback(request); });
|
|
|
|
showDialog(dialog);
|
|
}
|
|
|
|
void
|
|
MainWindow::showTransparentOverlayModal(QWidget *content, QFlags<Qt::AlignmentFlag> flags)
|
|
{
|
|
modal_->setWidget(content);
|
|
modal_->setColor(QColor(30, 30, 30, 150));
|
|
modal_->setDismissible(true);
|
|
modal_->setContentAlignment(flags);
|
|
modal_->raise();
|
|
modal_->show();
|
|
}
|
|
|
|
void
|
|
MainWindow::showSolidOverlayModal(QWidget *content, QFlags<Qt::AlignmentFlag> flags)
|
|
{
|
|
modal_->setWidget(content);
|
|
modal_->setColor(QColor(30, 30, 30));
|
|
modal_->setDismissible(false);
|
|
modal_->setContentAlignment(flags);
|
|
modal_->raise();
|
|
modal_->show();
|
|
}
|
|
|
|
void
|
|
MainWindow::openLogoutDialog()
|
|
{
|
|
auto dialog = new dialogs::Logout(this);
|
|
connect(dialog, &dialogs::Logout::loggingOut, this, [this]() {
|
|
if (WebRTCSession::instance().state() != webrtc::State::DISCONNECTED) {
|
|
if (QMessageBox::question(
|
|
this, "nheko", "A call is in progress. Log out?") !=
|
|
QMessageBox::Yes) {
|
|
return;
|
|
}
|
|
WebRTCSession::instance().end();
|
|
}
|
|
chat_page_->initiateLogout();
|
|
});
|
|
|
|
showDialog(dialog);
|
|
}
|
|
|
|
void
|
|
MainWindow::openReadReceiptsDialog(const QString &event_id)
|
|
{
|
|
auto dialog = new dialogs::ReadReceipts(this);
|
|
|
|
const auto room_id = chat_page_->currentRoom();
|
|
|
|
try {
|
|
dialog->addUsers(cache::readReceipts(event_id, room_id));
|
|
} catch (const lmdb::error &) {
|
|
nhlog::db()->warn("failed to retrieve read receipts for {} {}",
|
|
event_id.toStdString(),
|
|
chat_page_->currentRoom().toStdString());
|
|
dialog->deleteLater();
|
|
|
|
return;
|
|
}
|
|
|
|
showDialog(dialog);
|
|
}
|
|
|
|
bool
|
|
MainWindow::hasActiveDialogs() const
|
|
{
|
|
return !modal_ && modal_->isVisible();
|
|
}
|
|
|
|
bool
|
|
MainWindow::pageSupportsTray() const
|
|
{
|
|
return !welcome_page_->isVisible() && !login_page_->isVisible() &&
|
|
!register_page_->isVisible();
|
|
}
|
|
|
|
void
|
|
MainWindow::hideOverlay()
|
|
{
|
|
if (modal_)
|
|
modal_->hide();
|
|
}
|
|
|
|
inline void
|
|
MainWindow::showDialog(QWidget *dialog)
|
|
{
|
|
utils::centerWidget(dialog, this);
|
|
dialog->raise();
|
|
dialog->show();
|
|
}
|
|
|
|
bool
|
|
MainWindow::loadJdenticonPlugin()
|
|
{
|
|
QDir pluginsDir(qApp->applicationDirPath());
|
|
|
|
bool plugins = pluginsDir.cd("plugins");
|
|
if (plugins) {
|
|
foreach (QString fileName, pluginsDir.entryList(QDir::Files)) {
|
|
QPluginLoader pluginLoader(pluginsDir.absoluteFilePath(fileName));
|
|
QObject *plugin = pluginLoader.instance();
|
|
if (plugin) {
|
|
jdenticonInteface_ = qobject_cast<JdenticonInterface *>(plugin);
|
|
if (jdenticonInteface_) {
|
|
nhlog::ui()->info("Found jdenticon plugin.");
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
nhlog::ui()->info("jdenticon plugin not found.");
|
|
return false;
|
|
}
|
|
void
|
|
MainWindow::showWelcomePage()
|
|
{
|
|
removeOverlayProgressBar();
|
|
pageStack_->addWidget(welcome_page_);
|
|
pageStack_->setCurrentWidget(welcome_page_);
|
|
}
|
|
|
|
void
|
|
MainWindow::showLoginPage()
|
|
{
|
|
if (modal_)
|
|
modal_->hide();
|
|
|
|
pageStack_->addWidget(login_page_);
|
|
pageStack_->setCurrentWidget(login_page_);
|
|
}
|
|
|
|
void
|
|
MainWindow::showRegisterPage()
|
|
{
|
|
pageStack_->addWidget(register_page_);
|
|
pageStack_->setCurrentWidget(register_page_);
|
|
}
|
|
|
|
void
|
|
MainWindow::showUserSettingsPage()
|
|
{
|
|
pageStack_->setCurrentWidget(userSettingsPage_);
|
|
}
|