nheko/src/MainWindow.cpp

510 lines
16 KiB
C++
Raw Normal View History

2017-04-06 01:06:42 +02:00
/*
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
2017-09-24 16:39:06 +02:00
#include <QApplication>
2017-04-06 01:06:42 +02:00
#include <QLayout>
2019-01-26 19:11:30 +01:00
#include <QPluginLoader>
2017-04-06 01:06:42 +02:00
#include <QSettings>
#include <QShortcut>
2017-10-28 14:46:39 +02:00
#include <mtx/requests.hpp>
2017-10-28 14:46:39 +02:00
#include "ChatPage.h"
#include "Config.h"
2018-07-17 15:37:25 +02:00
#include "Logging.h"
2017-10-28 14:46:39 +02:00
#include "LoginPage.h"
#include "MainWindow.h"
#include "MatrixClient.h"
#include "RegisterPage.h"
#include "TrayIcon.h"
2017-11-01 23:41:13 +01:00
#include "UserSettingsPage.h"
2018-09-21 10:30:02 +02:00
#include "Utils.h"
2017-10-28 14:46:39 +02:00
#include "WelcomePage.h"
2018-07-17 15:37:25 +02:00
#include "ui/LoadingIndicator.h"
#include "ui/OverlayModal.h"
#include "ui/SnackBar.h"
#include "dialogs/CreateRoom.h"
#include "dialogs/InviteUsers.h"
#include "dialogs/JoinRoom.h"
#include "dialogs/LeaveRoom.h"
#include "dialogs/Logout.h"
2018-07-17 15:37:25 +02:00
#include "dialogs/MemberList.h"
2018-08-11 12:50:56 +02:00
#include "dialogs/ReadReceipts.h"
2018-07-17 15:37:25 +02:00
#include "dialogs/RoomSettings.h"
2017-04-06 01:06:42 +02:00
MainWindow *MainWindow::instance_ = nullptr;
2017-04-06 01:06:42 +02:00
MainWindow::MainWindow(QWidget *parent)
2017-08-20 12:47:22 +02:00
: QMainWindow(parent)
2017-04-06 01:06:42 +02:00
{
2017-09-10 11:58:00 +02:00
setWindowTitle("nheko");
setObjectName("MainWindow");
2017-05-14 20:10:03 +02:00
2018-08-11 12:50:56 +02:00
modal_ = new OverlayModal(this);
2017-09-10 11:58:00 +02:00
restoreWindowSize();
2017-05-14 20:10:03 +02:00
2018-09-30 12:24:36 +02:00
QFont font;
2017-09-10 11:58:00 +02:00
font.setStyleStrategy(QFont::PreferAntialias);
setFont(font);
2017-05-14 20:10:03 +02:00
2017-11-01 23:41:13 +01:00
userSettings_ = QSharedPointer<UserSettings>(new UserSettings);
trayIcon_ = new TrayIcon(":/logos/nheko-32.png", this);
2017-11-01 23:41:13 +01:00
welcome_page_ = new WelcomePage(this);
login_page_ = new LoginPage(this);
register_page_ = new RegisterPage(this);
chat_page_ = new ChatPage(userSettings_, this);
2017-11-01 23:41:13 +01:00
userSettingsPage_ = new UserSettingsPage(userSettings_, this);
2017-04-06 01:06:42 +02:00
2017-09-10 11:58:00 +02:00
// Initialize sliding widget manager.
2017-09-30 16:05:05 +02:00
pageStack_ = new QStackedWidget(this);
pageStack_->addWidget(welcome_page_);
pageStack_->addWidget(login_page_);
pageStack_->addWidget(register_page_);
pageStack_->addWidget(chat_page_);
2017-11-01 23:41:13 +01:00
pageStack_->addWidget(userSettingsPage_);
2017-04-06 01:06:42 +02:00
2017-09-30 16:05:05 +02:00
setCentralWidget(pageStack_);
2017-04-06 01:06:42 +02:00
2017-09-10 11:58:00 +02:00
connect(welcome_page_, SIGNAL(userLogin()), this, SLOT(showLoginPage()));
connect(welcome_page_, SIGNAL(userRegister()), this, SLOT(showRegisterPage()));
2017-04-06 01:06:42 +02:00
2017-09-10 11:58:00 +02:00
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();
});
2017-09-10 11:58:00 +02:00
connect(register_page_, SIGNAL(backButtonClicked()), this, SLOT(showWelcomePage()));
2017-04-06 01:06:42 +02:00
2018-04-24 22:57:49 +02:00
connect(chat_page_, &ChatPage::closing, this, &MainWindow::showWelcomePage);
connect(
chat_page_, &ChatPage::showOverlayProgressBar, this, &MainWindow::showOverlayProgressBar);
2017-09-10 11:58:00 +02:00
connect(
chat_page_, SIGNAL(changeWindowTitle(QString)), this, SLOT(setWindowTitle(QString)));
connect(chat_page_, SIGNAL(unreadMessages(int)), trayIcon_, SLOT(setUnreadCount(int)));
connect(chat_page_, &ChatPage::showLoginPage, this, [this](const QString &msg) {
2017-10-20 21:32:48 +02:00
login_page_->loginError(msg);
showLoginPage();
});
2017-05-21 15:36:06 +02:00
connect(userSettingsPage_, &UserSettingsPage::moveBack, this, [this]() {
2017-11-01 23:41:13 +01:00
pageStack_->setCurrentWidget(chat_page_);
});
2017-11-02 21:02:31 +01:00
connect(
userSettingsPage_, SIGNAL(trayOptionChanged(bool)), trayIcon_, SLOT(setVisible(bool)));
2019-01-24 05:54:35 +01:00
connect(userSettingsPage_, &UserSettingsPage::themeChanged, this, []() {
Cache::clearUserColors();
});
connect(
userSettingsPage_, &UserSettingsPage::themeChanged, chat_page_, &ChatPage::themeChanged);
2017-09-10 11:58:00 +02:00
connect(trayIcon_,
SIGNAL(activated(QSystemTrayIcon::ActivationReason)),
this,
SLOT(iconActivated(QSystemTrayIcon::ActivationReason)));
2017-09-10 11:58:00 +02:00
connect(chat_page_, SIGNAL(contentLoaded()), this, SLOT(removeOverlayProgressBar()));
2017-11-01 23:41:13 +01:00
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();
2018-05-02 14:30:08 +02:00
});
connect(register_page_, &RegisterPage::registerOk, this, &MainWindow::showChatPage);
QShortcut *quitShortcut = new QShortcut(QKeySequence::Quit, this);
connect(quitShortcut, &QShortcut::activated, this, QApplication::quit);
QShortcut *quickSwitchShortcut = new QShortcut(QKeySequence("Ctrl+K"), this);
connect(quickSwitchShortcut, &QShortcut::activated, this, [this]() {
if (chat_page_->isVisible() && !hasActiveDialogs())
chat_page_->showQuickSwitcher();
});
2017-09-10 11:58:00 +02:00
QSettings settings;
trayIcon_->setVisible(userSettings_->isTrayEnabled());
2017-09-10 11:58:00 +02:00
if (hasActiveUser()) {
QString token = settings.value("auth/access_token").toString();
QString home_server = settings.value("auth/home_server").toString();
QString user_id = settings.value("auth/user_id").toString();
QString device_id = settings.value("auth/device_id").toString();
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 &e) {
nhlog::ui()->critical("bootstrapped with invalid user_id: {}",
user_id.toStdString());
}
showChatPage();
2017-09-10 11:58:00 +02:00
}
if (loadJdenticonPlugin()) {
nhlog::ui()->info("loaded jdenticon.");
}
2017-04-06 01:06:42 +02:00
}
void
MainWindow::showEvent(QShowEvent *event)
{
adjustSideBars();
QMainWindow::showEvent(event);
}
void
MainWindow::resizeEvent(QResizeEvent *event)
{
adjustSideBars();
QMainWindow::resizeEvent(event);
}
void
MainWindow::adjustSideBars()
{
2018-10-07 13:09:47 +02:00
const auto sz = utils::calculateSidebarSizes(QFont{});
const uint64_t timelineWidth = chat_page_->timelineWidth();
const uint64_t minAvailableWidth = sz.collapsePoint + sz.groups;
if (timelineWidth < minAvailableWidth && !chat_page_->isSideBarExpanded()) {
chat_page_->hideSideBars();
} else {
chat_page_->showSideBars();
}
}
2017-08-20 12:47:22 +02:00
void
MainWindow::restoreWindowSize()
{
2017-09-10 11:58:00 +02:00
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);
}
2017-08-20 12:47:22 +02:00
void
MainWindow::saveCurrentWindowSize()
{
2017-09-10 11:58:00 +02:00
QSettings settings;
QSize current = size();
2017-09-10 11:58:00 +02:00
settings.setValue("window/width", current.width());
settings.setValue("window/height", current.height());
}
2017-08-20 12:47:22 +02:00
void
MainWindow::removeOverlayProgressBar()
{
2017-09-10 11:58:00 +02:00
QTimer *timer = new QTimer(this);
timer->setSingleShot(true);
connect(timer, &QTimer::timeout, [this, timer]() {
2017-09-10 11:58:00 +02:00
timer->deleteLater();
2018-08-11 12:50:56 +02:00
if (modal_)
modal_->hide();
2018-08-11 12:50:56 +02:00
if (spinner_)
2017-10-07 19:09:34 +02:00
spinner_->stop();
2017-09-10 11:58:00 +02:00
});
2017-10-08 21:38:38 +02:00
// FIXME: Snackbar doesn't work if it's initialized in the constructor.
QTimer::singleShot(0, this, [this]() {
2018-08-11 12:50:56 +02:00
snackBar_ = new SnackBar(this);
connect(chat_page_, &ChatPage::showNotification, snackBar_, &SnackBar::showMessage);
2017-10-08 21:38:38 +02:00
});
2018-08-11 12:50:56 +02:00
timer->start(50);
}
2017-08-20 12:47:22 +02:00
void
MainWindow::showChatPage()
2017-04-06 01:06:42 +02:00
{
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());
2017-09-10 11:58:00 +02:00
QSettings settings;
settings.setValue("auth/access_token", token);
settings.setValue("auth/home_server", homeserver);
settings.setValue("auth/user_id", userid);
settings.setValue("auth/device_id", device_id);
2017-09-10 11:58:00 +02:00
showOverlayProgressBar();
2017-09-30 16:05:05 +02:00
pageStack_->setCurrentWidget(chat_page_);
2017-09-10 11:58:00 +02:00
pageStack_->removeWidget(welcome_page_);
pageStack_->removeWidget(login_page_);
pageStack_->removeWidget(register_page_);
2017-09-10 11:58:00 +02:00
login_page_->reset();
chat_page_->bootstrap(userid, homeserver, token);
instance_ = this;
2017-04-06 01:06:42 +02:00
}
2017-08-20 12:47:22 +02:00
void
MainWindow::closeEvent(QCloseEvent *event)
2017-05-21 15:36:06 +02:00
{
if (!qApp->isSavingSession() && isVisible() && pageSupportsTray() &&
userSettings_->isTrayEnabled()) {
2017-09-10 11:58:00 +02:00
event->ignore();
hide();
}
2017-05-21 15:36:06 +02:00
}
2017-08-20 12:47:22 +02:00
void
MainWindow::iconActivated(QSystemTrayIcon::ActivationReason reason)
2017-05-21 15:36:06 +02:00
{
2017-09-10 11:58:00 +02:00
switch (reason) {
case QSystemTrayIcon::Trigger:
if (!isVisible()) {
show();
} else {
hide();
}
break;
default:
break;
}
2017-05-21 15:36:06 +02:00
}
2017-08-20 12:47:22 +02:00
bool
MainWindow::hasActiveUser()
{
2017-09-10 11:58:00 +02:00
QSettings settings;
2017-09-10 11:58:00 +02:00
return settings.contains("auth/access_token") && settings.contains("auth/home_server") &&
settings.contains("auth/user_id");
}
2018-07-20 11:02:35 +02:00
void
MainWindow::openUserProfile(const QString &user_id, const QString &room_id)
{
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::UserProfile(this);
dialog->init(user_id, room_id);
2018-07-20 11:02:35 +02:00
2018-09-21 10:30:02 +02:00
showDialog(dialog);
2018-07-20 11:02:35 +02:00
}
2018-04-30 20:41:47 +02:00
void
MainWindow::openRoomSettings(const QString &room_id)
{
const auto roomToSearch = room_id.isEmpty() ? chat_page_->currentRoom() : "";
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::RoomSettings(roomToSearch, this);
2018-04-30 20:41:47 +02:00
2018-09-21 10:30:02 +02:00
showDialog(dialog);
2018-04-30 20:41:47 +02:00
}
2018-05-01 18:35:28 +02:00
void
MainWindow::openMemberListDialog(const QString &room_id)
{
const auto roomToSearch = room_id.isEmpty() ? chat_page_->currentRoom() : "";
2018-09-19 21:42:26 +02:00
auto dialog = new dialogs::MemberList(roomToSearch, this);
2018-05-01 18:35:28 +02:00
2018-09-21 10:30:02 +02:00
showDialog(dialog);
2018-05-01 18:35:28 +02:00
}
void
MainWindow::openLeaveRoomDialog(const QString &room_id)
{
auto roomToLeave = room_id.isEmpty() ? chat_page_->currentRoom() : room_id;
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::LeaveRoom(this);
2018-09-19 21:42:26 +02:00
connect(dialog, &dialogs::LeaveRoom::leaving, this, [this, roomToLeave]() {
chat_page_->leaveRoom(roomToLeave);
2018-08-11 12:50:56 +02:00
});
2018-09-21 10:30:02 +02:00
showDialog(dialog);
}
void
MainWindow::showOverlayProgressBar()
{
2018-08-11 12:50:56 +02:00
spinner_ = new LoadingIndicator(this);
spinner_->setFixedHeight(100);
spinner_->setFixedWidth(100);
spinner_->setObjectName("ChatPageLoadSpinner");
spinner_->start();
2018-08-11 12:50:56 +02:00
showSolidOverlayModal(spinner_);
}
void
MainWindow::openInviteUsersDialog(std::function<void(const QStringList &invitees)> callback)
{
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::InviteUsers(this);
2018-09-19 21:42:26 +02:00
connect(dialog, &dialogs::InviteUsers::sendInvites, this, [callback](QStringList invitees) {
if (!invitees.isEmpty())
callback(invitees);
});
2018-09-21 10:30:02 +02:00
showDialog(dialog);
}
void
MainWindow::openJoinRoomDialog(std::function<void(const QString &room_id)> callback)
{
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::JoinRoom(this);
2018-09-19 21:42:26 +02:00
connect(dialog, &dialogs::JoinRoom::joinRoom, this, [callback](const QString &room) {
if (!room.isEmpty())
callback(room);
});
2018-09-21 10:30:02 +02:00
showDialog(dialog);
}
void
MainWindow::openCreateRoomDialog(
std::function<void(const mtx::requests::CreateRoom &request)> callback)
{
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::CreateRoom(this);
connect(dialog,
2018-09-19 21:42:26 +02:00
&dialogs::CreateRoom::createRoom,
2018-08-11 12:50:56 +02:00
this,
2018-09-19 21:42:26 +02:00
[callback](const mtx::requests::CreateRoom &request) { callback(request); });
2018-09-21 10:30:02 +02:00
showDialog(dialog);
2018-08-11 12:50:56 +02:00
}
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()
{
2018-08-11 12:50:56 +02:00
auto dialog = new dialogs::Logout(this);
2018-09-19 21:42:26 +02:00
connect(
dialog, &dialogs::Logout::loggingOut, this, [this]() { chat_page_->initiateLogout(); });
2018-08-11 12:50:56 +02:00
2018-09-21 10:30:02 +02:00
showDialog(dialog);
2018-08-11 12:50:56 +02:00
}
void
MainWindow::openReadReceiptsDialog(const QString &event_id)
{
auto dialog = new dialogs::ReadReceipts(this);
const auto room_id = chat_page_->currentRoom();
try {
dialog->addUsers(cache::client()->readReceipts(event_id, room_id));
} catch (const lmdb::error &e) {
nhlog::db()->warn("failed to retrieve read receipts for {} {}",
event_id.toStdString(),
chat_page_->currentRoom().toStdString());
dialog->deleteLater();
return;
}
2018-09-21 10:30:02 +02:00
showDialog(dialog);
}
bool
MainWindow::hasActiveDialogs() const
{
2018-08-11 12:50:56 +02:00
return !modal_ && modal_->isVisible();
}
bool
MainWindow::pageSupportsTray() const
{
return !welcome_page_->isVisible() && !login_page_->isVisible() &&
!register_page_->isVisible();
}
2018-08-11 12:50:56 +02:00
void
MainWindow::hideOverlay()
{
if (modal_)
modal_->hide();
}
2018-09-21 10:30:02 +02:00
inline void
MainWindow::showDialog(QWidget *dialog)
{
utils::centerWidget(dialog, this);
dialog->raise();
dialog->show();
}
bool
MainWindow::loadJdenticonPlugin()
{
2019-01-26 19:11:30 +01:00
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;
}
}
}
}
2019-01-26 19:11:30 +01:00
nhlog::ui()->info("jdenticon plugin not found.");
return false;
}