mirror of
https://github.com/Nheko-Reborn/nheko.git
synced 2024-11-22 19:08:58 +03:00
Initial qml root window
This commit is contained in:
parent
9a2e07cbce
commit
b106eafb0e
21 changed files with 426 additions and 397 deletions
|
@ -5,6 +5,7 @@
|
||||||
|
|
||||||
import QtGraphicalEffects 1.0
|
import QtGraphicalEffects 1.0
|
||||||
import QtQuick 2.12
|
import QtQuick 2.12
|
||||||
|
import QtQuick.Window 2.2
|
||||||
import im.nheko 1.0
|
import im.nheko 1.0
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
|
@ -15,7 +16,7 @@ Item {
|
||||||
|
|
||||||
Connections {
|
Connections {
|
||||||
function onFocusChanged() {
|
function onFocusChanged() {
|
||||||
if (TimelineManager.isWindowFocused) {
|
if (MainWindow.active) {
|
||||||
screenSaverTimer.stop();
|
screenSaverTimer.stop();
|
||||||
screenSaver.state = "Invisible";
|
screenSaver.state = "Invisible";
|
||||||
} else {
|
} else {
|
||||||
|
@ -25,14 +26,14 @@ Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
target: TimelineManager
|
target: MainWindow
|
||||||
}
|
}
|
||||||
|
|
||||||
Timer {
|
Timer {
|
||||||
id: screenSaverTimer
|
id: screenSaverTimer
|
||||||
|
|
||||||
interval: screenTimeout * 1000
|
interval: screenTimeout * 1000
|
||||||
running: true
|
running: !MainWindow.active
|
||||||
onTriggered: {
|
onTriggered: {
|
||||||
screenSaver.state = "Visible";
|
screenSaver.state = "Visible";
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,10 +17,12 @@ import QtQuick.Window 2.15
|
||||||
import im.nheko 1.0
|
import im.nheko 1.0
|
||||||
import im.nheko.EmojiModel 1.0
|
import im.nheko.EmojiModel 1.0
|
||||||
|
|
||||||
Page {
|
Pane {
|
||||||
id: timelineRoot
|
id: timelineRoot
|
||||||
|
|
||||||
palette: Nheko.colors
|
palette: Nheko.colors
|
||||||
|
background: null
|
||||||
|
padding: 0
|
||||||
|
|
||||||
FontMetrics {
|
FontMetrics {
|
||||||
id: fontMetrics
|
id: fontMetrics
|
||||||
|
@ -157,7 +159,6 @@ Page {
|
||||||
sequence: "Ctrl+K"
|
sequence: "Ctrl+K"
|
||||||
onActivated: {
|
onActivated: {
|
||||||
var quickSwitch = quickSwitcherComponent.createObject(timelineRoot);
|
var quickSwitch = quickSwitcherComponent.createObject(timelineRoot);
|
||||||
TimelineManager.focusTimeline();
|
|
||||||
quickSwitch.open();
|
quickSwitch.open();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -165,7 +166,6 @@ Page {
|
||||||
Shortcut {
|
Shortcut {
|
||||||
// Add alternative shortcut, because sometimes Alt+A is stolen by the TextEdit
|
// Add alternative shortcut, because sometimes Alt+A is stolen by the TextEdit
|
||||||
sequences: ["Alt+A", "Ctrl+Shift+A"]
|
sequences: ["Alt+A", "Ctrl+Shift+A"]
|
||||||
context: Qt.ApplicationShortcut
|
|
||||||
onActivated: Rooms.nextRoomWithActivity()
|
onActivated: Rooms.nextRoomWithActivity()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -366,9 +366,24 @@ Page {
|
||||||
id: mainWindow
|
id: mainWindow
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
initialItem: ChatPage {
|
initialItem: WelcomePage {
|
||||||
//anchors.fill: parent
|
//anchors.fill: parent
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: chatPage
|
||||||
|
|
||||||
|
ChatPage {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
function onSwitchToChatPage() {
|
||||||
|
console.log("AAAA");
|
||||||
|
mainWindow.replace(chatPage);
|
||||||
|
}
|
||||||
|
target: MainWindow
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ import im.nheko 1.0
|
||||||
Button {
|
Button {
|
||||||
id: control
|
id: control
|
||||||
|
|
||||||
implicitHeight: Math.ceil(control.contentItem.implicitHeight * 1.5)
|
implicitHeight: Math.ceil(control.contentItem.implicitHeight * 1.70)
|
||||||
implicitWidth: Math.ceil(control.contentItem.implicitWidth + control.contentItem.implicitHeight)
|
implicitWidth: Math.ceil(control.contentItem.implicitWidth + control.contentItem.implicitHeight)
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
|
|
||||||
|
@ -42,7 +42,7 @@ Button {
|
||||||
background: Rectangle {
|
background: Rectangle {
|
||||||
//height: control.contentItem.implicitHeight * 2
|
//height: control.contentItem.implicitHeight * 2
|
||||||
//width: control.contentItem.implicitWidth * 2
|
//width: control.contentItem.implicitWidth * 2
|
||||||
radius: height / 6
|
radius: height / 8
|
||||||
color: Qt.lighter(Nheko.colors.dark, control.down ? 1.4 : (control.hovered ? 1.2 : 1))
|
color: Qt.lighter(Nheko.colors.dark, control.down ? 1.4 : (control.hovered ? 1.2 : 1))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
76
resources/qml/pages/WelcomePage.qml
Normal file
76
resources/qml/pages/WelcomePage.qml
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQuick.Controls 2.15
|
||||||
|
import QtQuick.Layouts 1.2
|
||||||
|
import QtQuick.Window 2.15
|
||||||
|
import im.nheko 1.0
|
||||||
|
import "../components/"
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
FontMetrics {
|
||||||
|
id: fontMetrics
|
||||||
|
}
|
||||||
|
|
||||||
|
Shortcut {
|
||||||
|
sequence: StandardKey.Quit
|
||||||
|
onActivated: Qt.quit()
|
||||||
|
}
|
||||||
|
|
||||||
|
Item {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
}
|
||||||
|
|
||||||
|
Image {
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
source: "qrc:/logos/splash.png"
|
||||||
|
height: 256
|
||||||
|
width: 256
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
Layout.margins: Nheko.paddingLarge
|
||||||
|
Layout.bottomMargin: 0
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: qsTr("Welcome to nheko! The desktop client for the Matrix protocol.")
|
||||||
|
color: Nheko.colors.text
|
||||||
|
font.pointSize: fontMetrics.font.pointSize*2
|
||||||
|
wrapMode: Text.Wrap
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
Layout.margins: Nheko.paddingLarge
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: qsTr("Enjoy your stay!")
|
||||||
|
color: Nheko.colors.text
|
||||||
|
font.pointSize: fontMetrics.font.pointSize*1.5
|
||||||
|
wrapMode: Text.Wrap
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
Item {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
}
|
||||||
|
FlatButton {
|
||||||
|
Layout.margins: Nheko.paddingLarge
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
text: qsTr("REGISTER")
|
||||||
|
onClicked: {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
FlatButton {
|
||||||
|
Layout.margins: Nheko.paddingLarge
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
text: qsTr("LOGIN")
|
||||||
|
onClicked: {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Item {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Item {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
}
|
||||||
|
}
|
|
@ -110,6 +110,7 @@
|
||||||
<file>qml/TypingIndicator.qml</file>
|
<file>qml/TypingIndicator.qml</file>
|
||||||
<file>qml/NotificationWarning.qml</file>
|
<file>qml/NotificationWarning.qml</file>
|
||||||
<file>qml/pages/UserSettingsPage.qml</file>
|
<file>qml/pages/UserSettingsPage.qml</file>
|
||||||
|
<file>qml/pages/WelcomePage.qml</file>
|
||||||
<file>qml/components/AdaptiveLayout.qml</file>
|
<file>qml/components/AdaptiveLayout.qml</file>
|
||||||
<file>qml/components/AdaptiveLayoutElement.qml</file>
|
<file>qml/components/AdaptiveLayoutElement.qml</file>
|
||||||
<file>qml/components/AvatarListTile.qml</file>
|
<file>qml/components/AvatarListTile.qml</file>
|
||||||
|
|
|
@ -325,7 +325,7 @@ static void
|
||||||
fatalSecretError()
|
fatalSecretError()
|
||||||
{
|
{
|
||||||
QMessageBox::critical(
|
QMessageBox::critical(
|
||||||
ChatPage::instance(),
|
nullptr,
|
||||||
QCoreApplication::translate("SecretStorage", "Failed to connect to secret storage"),
|
QCoreApplication::translate("SecretStorage", "Failed to connect to secret storage"),
|
||||||
QCoreApplication::translate(
|
QCoreApplication::translate(
|
||||||
"SecretStorage",
|
"SecretStorage",
|
||||||
|
|
|
@ -44,8 +44,8 @@ Q_DECLARE_METATYPE(mtx::presence::PresenceState)
|
||||||
Q_DECLARE_METATYPE(mtx::secret_storage::AesHmacSha2KeyDescription)
|
Q_DECLARE_METATYPE(mtx::secret_storage::AesHmacSha2KeyDescription)
|
||||||
Q_DECLARE_METATYPE(SecretsToDecrypt)
|
Q_DECLARE_METATYPE(SecretsToDecrypt)
|
||||||
|
|
||||||
ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QObject *parent)
|
||||||
: QWidget(parent)
|
: QObject(parent)
|
||||||
, isConnected_(true)
|
, isConnected_(true)
|
||||||
, userSettings_{userSettings}
|
, userSettings_{userSettings}
|
||||||
, notificationsManager(this)
|
, notificationsManager(this)
|
||||||
|
@ -61,14 +61,8 @@ ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
||||||
qRegisterMetaType<mtx::secret_storage::AesHmacSha2KeyDescription>();
|
qRegisterMetaType<mtx::secret_storage::AesHmacSha2KeyDescription>();
|
||||||
qRegisterMetaType<SecretsToDecrypt>();
|
qRegisterMetaType<SecretsToDecrypt>();
|
||||||
|
|
||||||
topLayout_ = new QHBoxLayout(this);
|
|
||||||
topLayout_->setSpacing(0);
|
|
||||||
topLayout_->setContentsMargins(0, 0, 0, 0);
|
|
||||||
|
|
||||||
view_manager_ = new TimelineViewManager(callManager_, this);
|
view_manager_ = new TimelineViewManager(callManager_, this);
|
||||||
|
|
||||||
topLayout_->addWidget(view_manager_->getWidget());
|
|
||||||
|
|
||||||
connect(this,
|
connect(this,
|
||||||
&ChatPage::downloadedSecrets,
|
&ChatPage::downloadedSecrets,
|
||||||
this,
|
this,
|
||||||
|
@ -154,7 +148,7 @@ ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
||||||
[this](const QString &roomid, const QString &eventid) {
|
[this](const QString &roomid, const QString &eventid) {
|
||||||
Q_UNUSED(eventid)
|
Q_UNUSED(eventid)
|
||||||
view_manager_->rooms()->setCurrentRoom(roomid);
|
view_manager_->rooms()->setCurrentRoom(roomid);
|
||||||
activateWindow();
|
MainWindow::instance()->requestActivate();
|
||||||
});
|
});
|
||||||
connect(¬ificationsManager,
|
connect(¬ificationsManager,
|
||||||
&NotificationsManager::sendNotificationReply,
|
&NotificationsManager::sendNotificationReply,
|
||||||
|
@ -162,15 +156,13 @@ ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
||||||
[this](const QString &roomid, const QString &eventid, const QString &body) {
|
[this](const QString &roomid, const QString &eventid, const QString &body) {
|
||||||
view_manager_->rooms()->setCurrentRoom(roomid);
|
view_manager_->rooms()->setCurrentRoom(roomid);
|
||||||
view_manager_->queueReply(roomid, eventid, body);
|
view_manager_->queueReply(roomid, eventid, body);
|
||||||
activateWindow();
|
MainWindow::instance()->requestActivate();
|
||||||
});
|
});
|
||||||
|
|
||||||
connect(QCoreApplication::instance(), &QCoreApplication::aboutToQuit, this, [this]() {
|
connect(QCoreApplication::instance(), &QCoreApplication::aboutToQuit, this, [this]() {
|
||||||
// ensure the qml context is shutdown before we destroy all other singletons
|
// ensure the qml context is shutdown before we destroy all other singletons
|
||||||
// Otherwise Qml will try to access the room list or settings, after they have been
|
// Otherwise Qml will try to access the room list or settings, after they have been
|
||||||
// destroyed
|
// destroyed
|
||||||
topLayout_->removeWidget(view_manager_->getWidget());
|
|
||||||
delete view_manager_->getWidget();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
connect(
|
connect(
|
||||||
|
@ -201,7 +193,7 @@ ChatPage::ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
||||||
// TODO: Replace this once we have proper pushrules support. This is a horrible hack
|
// TODO: Replace this once we have proper pushrules support. This is a horrible hack
|
||||||
if (prevNotificationCount < notificationCount) {
|
if (prevNotificationCount < notificationCount) {
|
||||||
if (userSettings_->hasAlertOnNotification())
|
if (userSettings_->hasAlertOnNotification())
|
||||||
QApplication::alert(this);
|
MainWindow::instance()->alert(0);
|
||||||
}
|
}
|
||||||
prevNotificationCount = notificationCount;
|
prevNotificationCount = notificationCount;
|
||||||
|
|
||||||
|
@ -331,7 +323,7 @@ ChatPage::bootstrap(QString userid, QString homeserver, QString token)
|
||||||
} else if (cacheVersion == cache::CacheVersion::Older) {
|
} else if (cacheVersion == cache::CacheVersion::Older) {
|
||||||
if (!cache::runMigrations()) {
|
if (!cache::runMigrations()) {
|
||||||
QMessageBox::critical(
|
QMessageBox::critical(
|
||||||
this,
|
nullptr,
|
||||||
tr("Cache migration failed!"),
|
tr("Cache migration failed!"),
|
||||||
tr("Migrating the cache to the current version failed. "
|
tr("Migrating the cache to the current version failed. "
|
||||||
"This can have different reasons. Please open an "
|
"This can have different reasons. Please open an "
|
||||||
|
@ -344,7 +336,7 @@ ChatPage::bootstrap(QString userid, QString homeserver, QString token)
|
||||||
return;
|
return;
|
||||||
} else if (cacheVersion == cache::CacheVersion::Newer) {
|
} else if (cacheVersion == cache::CacheVersion::Newer) {
|
||||||
QMessageBox::critical(
|
QMessageBox::critical(
|
||||||
this,
|
nullptr,
|
||||||
tr("Incompatible cache version"),
|
tr("Incompatible cache version"),
|
||||||
tr("The cache on your disk is newer than this version of Nheko "
|
tr("The cache on your disk is newer than this version of Nheko "
|
||||||
"supports. Please update Nheko or clear your cache."));
|
"supports. Please update Nheko or clear your cache."));
|
||||||
|
@ -690,7 +682,7 @@ ChatPage::joinRoomVia(const std::string &room_id,
|
||||||
if (promptForConfirmation &&
|
if (promptForConfirmation &&
|
||||||
QMessageBox::Yes !=
|
QMessageBox::Yes !=
|
||||||
QMessageBox::question(
|
QMessageBox::question(
|
||||||
this,
|
nullptr,
|
||||||
tr("Confirm join"),
|
tr("Confirm join"),
|
||||||
tr("Do you really want to join %1?").arg(QString::fromStdString(room_id))))
|
tr("Do you really want to join %1?").arg(QString::fromStdString(room_id))))
|
||||||
return;
|
return;
|
||||||
|
@ -776,7 +768,7 @@ ChatPage::inviteUser(QString userid, QString reason)
|
||||||
{
|
{
|
||||||
auto room = currentRoom();
|
auto room = currentRoom();
|
||||||
|
|
||||||
if (QMessageBox::question(this,
|
if (QMessageBox::question(nullptr,
|
||||||
tr("Confirm invite"),
|
tr("Confirm invite"),
|
||||||
tr("Do you really want to invite %1 (%2)?")
|
tr("Do you really want to invite %1 (%2)?")
|
||||||
.arg(cache::displayName(room, userid), userid)) != QMessageBox::Yes)
|
.arg(cache::displayName(room, userid), userid)) != QMessageBox::Yes)
|
||||||
|
@ -800,7 +792,7 @@ ChatPage::kickUser(QString userid, QString reason)
|
||||||
{
|
{
|
||||||
auto room = currentRoom();
|
auto room = currentRoom();
|
||||||
|
|
||||||
if (QMessageBox::question(this,
|
if (QMessageBox::question(nullptr,
|
||||||
tr("Confirm kick"),
|
tr("Confirm kick"),
|
||||||
tr("Do you really want to kick %1 (%2)?")
|
tr("Do you really want to kick %1 (%2)?")
|
||||||
.arg(cache::displayName(room, userid), userid)) != QMessageBox::Yes)
|
.arg(cache::displayName(room, userid), userid)) != QMessageBox::Yes)
|
||||||
|
@ -825,7 +817,7 @@ ChatPage::banUser(QString userid, QString reason)
|
||||||
auto room = currentRoom();
|
auto room = currentRoom();
|
||||||
|
|
||||||
if (QMessageBox::question(
|
if (QMessageBox::question(
|
||||||
this,
|
nullptr,
|
||||||
tr("Confirm ban"),
|
tr("Confirm ban"),
|
||||||
tr("Do you really want to ban %1 (%2)?").arg(cache::displayName(room, userid), userid)) !=
|
tr("Do you really want to ban %1 (%2)?").arg(cache::displayName(room, userid), userid)) !=
|
||||||
QMessageBox::Yes)
|
QMessageBox::Yes)
|
||||||
|
@ -849,7 +841,7 @@ ChatPage::unbanUser(QString userid, QString reason)
|
||||||
{
|
{
|
||||||
auto room = currentRoom();
|
auto room = currentRoom();
|
||||||
|
|
||||||
if (QMessageBox::question(this,
|
if (QMessageBox::question(nullptr,
|
||||||
tr("Confirm unban"),
|
tr("Confirm unban"),
|
||||||
tr("Do you really want to unban %1 (%2)?")
|
tr("Do you really want to unban %1 (%2)?")
|
||||||
.arg(cache::displayName(room, userid), userid)) != QMessageBox::Yes)
|
.arg(cache::displayName(room, userid), userid)) != QMessageBox::Yes)
|
||||||
|
@ -1083,7 +1075,7 @@ ChatPage::decryptDownloadedSecrets(mtx::secret_storage::AesHmacSha2KeyDescriptio
|
||||||
const SecretsToDecrypt &secrets)
|
const SecretsToDecrypt &secrets)
|
||||||
{
|
{
|
||||||
QString text = QInputDialog::getText(
|
QString text = QInputDialog::getText(
|
||||||
ChatPage::instance(),
|
nullptr,
|
||||||
QCoreApplication::translate("CrossSigningSecrets", "Decrypt secrets"),
|
QCoreApplication::translate("CrossSigningSecrets", "Decrypt secrets"),
|
||||||
keyDesc.name.empty()
|
keyDesc.name.empty()
|
||||||
? QCoreApplication::translate(
|
? QCoreApplication::translate(
|
||||||
|
@ -1115,7 +1107,7 @@ ChatPage::decryptDownloadedSecrets(mtx::secret_storage::AesHmacSha2KeyDescriptio
|
||||||
|
|
||||||
if (!decryptionKey) {
|
if (!decryptionKey) {
|
||||||
QMessageBox::information(
|
QMessageBox::information(
|
||||||
ChatPage::instance(),
|
nullptr,
|
||||||
QCoreApplication::translate("CrossSigningSecrets", "Decryption failed"),
|
QCoreApplication::translate("CrossSigningSecrets", "Decryption failed"),
|
||||||
QCoreApplication::translate("CrossSigningSecrets",
|
QCoreApplication::translate("CrossSigningSecrets",
|
||||||
"Failed to decrypt secrets with the "
|
"Failed to decrypt secrets with the "
|
||||||
|
@ -1209,7 +1201,7 @@ ChatPage::startChat(QString userid)
|
||||||
|
|
||||||
if (QMessageBox::Yes !=
|
if (QMessageBox::Yes !=
|
||||||
QMessageBox::question(
|
QMessageBox::question(
|
||||||
this,
|
nullptr,
|
||||||
tr("Confirm invite"),
|
tr("Confirm invite"),
|
||||||
tr("Do you really want to start a private chat with %1?").arg(userid)))
|
tr("Do you really want to start a private chat with %1?").arg(userid)))
|
||||||
return;
|
return;
|
||||||
|
@ -1395,7 +1387,7 @@ ChatPage::handleMatrixUri(const QUrl &uri)
|
||||||
bool
|
bool
|
||||||
ChatPage::isRoomActive(const QString &room_id)
|
ChatPage::isRoomActive(const QString &room_id)
|
||||||
{
|
{
|
||||||
return isActiveWindow() && currentRoom() == room_id;
|
return MainWindow::instance()->isActive() && currentRoom() == room_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString
|
QString
|
||||||
|
|
|
@ -51,12 +51,12 @@ struct Rooms;
|
||||||
|
|
||||||
using SecretsToDecrypt = std::map<std::string, mtx::secret_storage::AesHmacSha2EncryptedData>;
|
using SecretsToDecrypt = std::map<std::string, mtx::secret_storage::AesHmacSha2EncryptedData>;
|
||||||
|
|
||||||
class ChatPage : public QWidget
|
class ChatPage : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent = nullptr);
|
ChatPage(QSharedPointer<UserSettings> userSettings, QObject *parent = nullptr);
|
||||||
|
|
||||||
// Initialize all the components of the UI.
|
// Initialize all the components of the UI.
|
||||||
void bootstrap(QString userid, QString homeserver, QString token);
|
void bootstrap(QString userid, QString homeserver, QString token);
|
||||||
|
|
|
@ -13,77 +13,108 @@
|
||||||
#include <mtx/requests.hpp>
|
#include <mtx/requests.hpp>
|
||||||
#include <mtx/responses/login.hpp>
|
#include <mtx/responses/login.hpp>
|
||||||
|
|
||||||
|
#include "BlurhashProvider.h"
|
||||||
#include "Cache.h"
|
#include "Cache.h"
|
||||||
#include "Cache_p.h"
|
#include "Cache_p.h"
|
||||||
#include "ChatPage.h"
|
#include "ChatPage.h"
|
||||||
|
#include "Clipboard.h"
|
||||||
|
#include "ColorImageProvider.h"
|
||||||
|
#include "CombinedImagePackModel.h"
|
||||||
|
#include "CompletionProxyModel.h"
|
||||||
#include "Config.h"
|
#include "Config.h"
|
||||||
|
#include "EventAccessors.h"
|
||||||
|
#include "ImagePackListModel.h"
|
||||||
|
#include "InviteesModel.h"
|
||||||
#include "JdenticonProvider.h"
|
#include "JdenticonProvider.h"
|
||||||
#include "Logging.h"
|
#include "Logging.h"
|
||||||
#include "LoginPage.h"
|
#include "LoginPage.h"
|
||||||
#include "MainWindow.h"
|
#include "MainWindow.h"
|
||||||
#include "MatrixClient.h"
|
#include "MatrixClient.h"
|
||||||
#include "MemberList.h"
|
#include "MemberList.h"
|
||||||
|
#include "MxcImageProvider.h"
|
||||||
|
#include "ReadReceiptsModel.h"
|
||||||
#include "RegisterPage.h"
|
#include "RegisterPage.h"
|
||||||
|
#include "RoomDirectoryModel.h"
|
||||||
|
#include "RoomsModel.h"
|
||||||
|
#include "SingleImagePackModel.h"
|
||||||
#include "TrayIcon.h"
|
#include "TrayIcon.h"
|
||||||
#include "UserSettingsPage.h"
|
#include "UserSettingsPage.h"
|
||||||
|
#include "UsersModel.h"
|
||||||
#include "Utils.h"
|
#include "Utils.h"
|
||||||
#include "WelcomePage.h"
|
#include "WelcomePage.h"
|
||||||
|
#include "emoji/EmojiModel.h"
|
||||||
|
#include "emoji/Provider.h"
|
||||||
|
#include "encryption/DeviceVerificationFlow.h"
|
||||||
|
#include "encryption/SelfVerificationStatus.h"
|
||||||
|
#include "timeline/DelegateChooser.h"
|
||||||
|
#include "timeline/TimelineViewManager.h"
|
||||||
#include "ui/LoadingIndicator.h"
|
#include "ui/LoadingIndicator.h"
|
||||||
|
#include "ui/MxcAnimatedImage.h"
|
||||||
|
#include "ui/MxcMediaProxy.h"
|
||||||
|
#include "ui/NhekoCursorShape.h"
|
||||||
|
#include "ui/NhekoDropArea.h"
|
||||||
|
#include "ui/NhekoGlobalObject.h"
|
||||||
#include "ui/OverlayModal.h"
|
#include "ui/OverlayModal.h"
|
||||||
#include "ui/SnackBar.h"
|
#include "ui/SnackBar.h"
|
||||||
|
#include "ui/UIA.h"
|
||||||
#include "voip/WebRTCSession.h"
|
#include "voip/WebRTCSession.h"
|
||||||
|
|
||||||
#include "dialogs/CreateRoom.h"
|
#include "dialogs/CreateRoom.h"
|
||||||
|
|
||||||
|
Q_DECLARE_METATYPE(mtx::events::collections::TimelineEvents)
|
||||||
|
Q_DECLARE_METATYPE(std::vector<DeviceInfo>)
|
||||||
|
Q_DECLARE_METATYPE(std::vector<mtx::responses::PublicRoomsChunk>)
|
||||||
|
|
||||||
MainWindow *MainWindow::instance_ = nullptr;
|
MainWindow *MainWindow::instance_ = nullptr;
|
||||||
|
|
||||||
MainWindow::MainWindow(QWidget *parent)
|
MainWindow::MainWindow(QWindow *parent)
|
||||||
: QMainWindow(parent)
|
: QQuickView(parent)
|
||||||
, userSettings_{UserSettings::instance()}
|
, userSettings_{UserSettings::instance()}
|
||||||
{
|
{
|
||||||
instance_ = this;
|
instance_ = this;
|
||||||
|
|
||||||
QMainWindow::setWindowTitle(0);
|
MainWindow::setWindowTitle(0);
|
||||||
setObjectName(QStringLiteral("MainWindow"));
|
setObjectName(QStringLiteral("MainWindow"));
|
||||||
|
setResizeMode(QQuickView::SizeRootObjectToView);
|
||||||
modal_ = new OverlayModal(this);
|
setMinimumHeight(400);
|
||||||
|
setMinimumWidth(400);
|
||||||
restoreWindowSize();
|
restoreWindowSize();
|
||||||
|
|
||||||
QFont font;
|
chat_page_ = new ChatPage(userSettings_, this);
|
||||||
font.setStyleStrategy(QFont::PreferAntialias);
|
registerQmlTypes();
|
||||||
setFont(font);
|
|
||||||
|
setColor(Theme::paletteFromTheme(userSettings_->theme()).window().color());
|
||||||
|
setSource(QUrl(QStringLiteral("qrc:///qml/Root.qml")));
|
||||||
|
// modal_ = new OverlayModal(this);
|
||||||
|
|
||||||
|
|
||||||
|
// QFont font;
|
||||||
|
// font.setStyleStrategy(QFont::PreferAntialias);
|
||||||
|
// setFont(font);
|
||||||
|
|
||||||
trayIcon_ = new TrayIcon(QStringLiteral(":/logos/nheko.svg"), this);
|
trayIcon_ = new TrayIcon(QStringLiteral(":/logos/nheko.svg"), this);
|
||||||
|
|
||||||
welcome_page_ = new WelcomePage(this);
|
// welcome_page_ = new WelcomePage(this);
|
||||||
login_page_ = new LoginPage(this);
|
// login_page_ = new LoginPage(this);
|
||||||
register_page_ = new RegisterPage(this);
|
// register_page_ = new RegisterPage(this);
|
||||||
chat_page_ = new ChatPage(userSettings_, this);
|
|
||||||
|
|
||||||
// Initialize sliding widget manager.
|
//// Initialize sliding widget manager.
|
||||||
pageStack_ = new QStackedWidget(this);
|
|
||||||
pageStack_->addWidget(welcome_page_);
|
|
||||||
pageStack_->addWidget(login_page_);
|
|
||||||
pageStack_->addWidget(register_page_);
|
|
||||||
pageStack_->addWidget(chat_page_);
|
|
||||||
|
|
||||||
setCentralWidget(pageStack_);
|
// connect(welcome_page_, SIGNAL(userLogin()), this, SLOT(showLoginPage()));
|
||||||
|
// connect(welcome_page_, SIGNAL(userRegister()), this, SLOT(showRegisterPage()));
|
||||||
|
|
||||||
connect(welcome_page_, SIGNAL(userLogin()), this, SLOT(showLoginPage()));
|
// connect(login_page_, SIGNAL(backButtonClicked()), this, SLOT(showWelcomePage()));
|
||||||
connect(welcome_page_, SIGNAL(userRegister()), this, SLOT(showRegisterPage()));
|
// connect(login_page_, &LoginPage::loggingIn, this, &MainWindow::showOverlayProgressBar);
|
||||||
|
// connect(register_page_, &RegisterPage::registering, this,
|
||||||
connect(login_page_, SIGNAL(backButtonClicked()), this, SLOT(showWelcomePage()));
|
// &MainWindow::showOverlayProgressBar); connect(login_page_, &LoginPage::errorOccurred, this,
|
||||||
connect(login_page_, &LoginPage::loggingIn, this, &MainWindow::showOverlayProgressBar);
|
// [this]() { removeOverlayProgressBar(); }); connect(
|
||||||
connect(register_page_, &RegisterPage::registering, this, &MainWindow::showOverlayProgressBar);
|
// register_page_, &RegisterPage::errorOccurred, this, [this]() { removeOverlayProgressBar();
|
||||||
connect(login_page_, &LoginPage::errorOccurred, this, [this]() { removeOverlayProgressBar(); });
|
// });
|
||||||
connect(
|
// connect(register_page_, SIGNAL(backButtonClicked()), this, SLOT(showWelcomePage()));
|
||||||
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::closing, this, &MainWindow::showWelcomePage);
|
||||||
connect(
|
// connect(
|
||||||
chat_page_, &ChatPage::showOverlayProgressBar, this, &MainWindow::showOverlayProgressBar);
|
// chat_page_, &ChatPage::showOverlayProgressBar, this, &MainWindow::showOverlayProgressBar);
|
||||||
connect(chat_page_, &ChatPage::unreadMessages, this, &MainWindow::setWindowTitle);
|
connect(chat_page_, &ChatPage::unreadMessages, this, &MainWindow::setWindowTitle);
|
||||||
connect(chat_page_, SIGNAL(unreadMessages(int)), trayIcon_, SLOT(setUnreadCount(int)));
|
connect(chat_page_, SIGNAL(unreadMessages(int)), trayIcon_, SLOT(setUnreadCount(int)));
|
||||||
connect(chat_page_, &ChatPage::showLoginPage, this, [this](const QString &msg) {
|
connect(chat_page_, &ChatPage::showLoginPage, this, [this](const QString &msg) {
|
||||||
|
@ -101,15 +132,12 @@ MainWindow::MainWindow(QWidget *parent)
|
||||||
|
|
||||||
connect(this, &MainWindow::focusChanged, chat_page_, &ChatPage::chatFocusChanged);
|
connect(this, &MainWindow::focusChanged, chat_page_, &ChatPage::chatFocusChanged);
|
||||||
|
|
||||||
connect(login_page_, &LoginPage::loginOk, this, [this](const mtx::responses::Login &res) {
|
// connect(login_page_, &LoginPage::loginOk, this, [this](const mtx::responses::Login &res) {
|
||||||
http::client()->set_user(res.user_id);
|
// http::client()->set_user(res.user_id);
|
||||||
showChatPage();
|
// showChatPage();
|
||||||
});
|
// });
|
||||||
|
|
||||||
connect(register_page_, &RegisterPage::registerOk, this, &MainWindow::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());
|
trayIcon_->setVisible(userSettings_->tray());
|
||||||
|
|
||||||
|
@ -133,11 +161,167 @@ MainWindow::MainWindow(QWidget *parent)
|
||||||
user_id.toStdString());
|
user_id.toStdString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nhlog::ui()->info("User already signed in, showing chat page");
|
||||||
showChatPage();
|
showChatPage();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
MainWindow::registerQmlTypes()
|
||||||
|
{
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationAccept>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationCancel>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationDone>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationKey>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationMac>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationReady>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationRequest>();
|
||||||
|
qRegisterMetaType<mtx::events::msg::KeyVerificationStart>();
|
||||||
|
qRegisterMetaType<CombinedImagePackModel *>();
|
||||||
|
qRegisterMetaType<mtx::events::collections::TimelineEvents>();
|
||||||
|
qRegisterMetaType<std::vector<DeviceInfo>>();
|
||||||
|
|
||||||
|
qRegisterMetaType<std::vector<mtx::responses::PublicRoomsChunk>>();
|
||||||
|
|
||||||
|
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<DelegateChoice>("im.nheko", 1, 0, "DelegateChoice");
|
||||||
|
qmlRegisterType<DelegateChooser>("im.nheko", 1, 0, "DelegateChooser");
|
||||||
|
qmlRegisterType<NhekoDropArea>("im.nheko", 1, 0, "NhekoDropArea");
|
||||||
|
qmlRegisterType<NhekoCursorShape>("im.nheko", 1, 0, "CursorShape");
|
||||||
|
qmlRegisterType<MxcAnimatedImage>("im.nheko", 1, 0, "MxcAnimatedImage");
|
||||||
|
qmlRegisterType<MxcMediaProxy>("im.nheko", 1, 0, "MxcMedia");
|
||||||
|
qmlRegisterType<RoomDirectoryModel>("im.nheko", 1, 0, "RoomDirectoryModel");
|
||||||
|
qmlRegisterUncreatableType<DeviceVerificationFlow>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"DeviceVerificationFlow",
|
||||||
|
QStringLiteral("Can't create verification flow from QML!"));
|
||||||
|
qmlRegisterUncreatableType<UserProfile>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"UserProfileModel",
|
||||||
|
QStringLiteral("UserProfile needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<MemberList>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"MemberList",
|
||||||
|
QStringLiteral("MemberList needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<RoomSettings>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"RoomSettingsModel",
|
||||||
|
QStringLiteral("Room Settings needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<TimelineModel>(
|
||||||
|
"im.nheko", 1, 0, "Room", QStringLiteral("Room needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<ImagePackListModel>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"ImagePackListModel",
|
||||||
|
QStringLiteral("ImagePackListModel needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<SingleImagePackModel>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"SingleImagePackModel",
|
||||||
|
QStringLiteral("SingleImagePackModel needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<InviteesModel>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"InviteesModel",
|
||||||
|
QStringLiteral("InviteesModel needs to be instantiated on the C++ side"));
|
||||||
|
qmlRegisterUncreatableType<ReadReceiptsProxy>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"ReadReceiptsProxy",
|
||||||
|
QStringLiteral("ReadReceiptsProxy needs to be instantiated on the C++ side"));
|
||||||
|
|
||||||
|
qmlRegisterSingletonType<Clipboard>(
|
||||||
|
"im.nheko", 1, 0, "Clipboard", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
||||||
|
return new Clipboard();
|
||||||
|
});
|
||||||
|
qmlRegisterSingletonType<Nheko>(
|
||||||
|
"im.nheko", 1, 0, "Nheko", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
||||||
|
return new Nheko();
|
||||||
|
});
|
||||||
|
qmlRegisterSingletonType<UserSettingsModel>(
|
||||||
|
"im.nheko", 1, 0, "UserSettingsModel", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
||||||
|
return new UserSettingsModel();
|
||||||
|
});
|
||||||
|
|
||||||
|
qmlRegisterSingletonInstance("im.nheko", 1, 0, "Settings", userSettings_.data());
|
||||||
|
|
||||||
|
qRegisterMetaType<mtx::events::collections::TimelineEvents>();
|
||||||
|
qRegisterMetaType<std::vector<DeviceInfo>>();
|
||||||
|
|
||||||
|
qmlRegisterUncreatableType<FilteredCommunitiesModel>(
|
||||||
|
"im.nheko",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"FilteredCommunitiesModel",
|
||||||
|
QStringLiteral("Use Communities.filtered() to create a FilteredCommunitiesModel"));
|
||||||
|
|
||||||
|
qmlRegisterType<emoji::EmojiModel>("im.nheko.EmojiModel", 1, 0, "EmojiModel");
|
||||||
|
qmlRegisterUncreatableType<emoji::Emoji>(
|
||||||
|
"im.nheko.EmojiModel", 1, 0, "Emoji", QStringLiteral("Used by emoji models"));
|
||||||
|
qmlRegisterUncreatableMetaObject(emoji::staticMetaObject,
|
||||||
|
"im.nheko.EmojiModel",
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
"EmojiCategory",
|
||||||
|
QStringLiteral("Error: Only enums"));
|
||||||
|
|
||||||
|
qmlRegisterType<RoomDirectoryModel>("im.nheko", 1, 0, "RoomDirectoryModel");
|
||||||
|
|
||||||
|
qmlRegisterSingletonType<SelfVerificationStatus>(
|
||||||
|
"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());
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::setWindowTitle(int notificationCount)
|
MainWindow::setWindowTitle(int notificationCount)
|
||||||
{
|
{
|
||||||
|
@ -148,20 +332,23 @@ MainWindow::setWindowTitle(int notificationCount)
|
||||||
if (notificationCount > 0) {
|
if (notificationCount > 0) {
|
||||||
name.append(QString{QStringLiteral(" (%1)")}.arg(notificationCount));
|
name.append(QString{QStringLiteral(" (%1)")}.arg(notificationCount));
|
||||||
}
|
}
|
||||||
QMainWindow::setWindowTitle(name);
|
QQuickView::setTitle(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
MainWindow::event(QEvent *event)
|
MainWindow::event(QEvent *event)
|
||||||
{
|
{
|
||||||
auto type = event->type();
|
auto type = event->type();
|
||||||
if (type == QEvent::WindowActivate) {
|
|
||||||
|
if (type == QEvent::Close) {
|
||||||
|
closeEvent(static_cast<QCloseEvent *>(event));
|
||||||
|
} else if (type == QEvent::WindowActivate) {
|
||||||
emit focusChanged(true);
|
emit focusChanged(true);
|
||||||
} else if (type == QEvent::WindowDeactivate) {
|
} else if (type == QEvent::WindowDeactivate) {
|
||||||
emit focusChanged(false);
|
emit focusChanged(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
return QMainWindow::event(event);
|
return QQuickView::event(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -196,19 +383,13 @@ MainWindow::removeOverlayProgressBar()
|
||||||
|
|
||||||
connect(timer, &QTimer::timeout, this, [this, timer]() {
|
connect(timer, &QTimer::timeout, this, [this, timer]() {
|
||||||
timer->deleteLater();
|
timer->deleteLater();
|
||||||
|
|
||||||
if (modal_)
|
|
||||||
modal_->hide();
|
|
||||||
|
|
||||||
if (spinner_)
|
|
||||||
spinner_->stop();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// FIXME: Snackbar doesn't work if it's initialized in the constructor.
|
// FIXME: Snackbar doesn't work if it's initialized in the constructor.
|
||||||
QTimer::singleShot(0, this, [this]() {
|
// QTimer::singleShot(0, this, [this]() {
|
||||||
snackBar_ = new SnackBar(this);
|
// snackBar_ = new SnackBar(this);
|
||||||
connect(chat_page_, &ChatPage::showNotification, snackBar_, &SnackBar::showMessage);
|
// connect(chat_page_, &ChatPage::showNotification, snackBar_, &SnackBar::showMessage);
|
||||||
});
|
//});
|
||||||
|
|
||||||
timer->start(50);
|
timer->start(50);
|
||||||
}
|
}
|
||||||
|
@ -229,17 +410,14 @@ MainWindow::showChatPage()
|
||||||
|
|
||||||
showOverlayProgressBar();
|
showOverlayProgressBar();
|
||||||
|
|
||||||
pageStack_->setCurrentWidget(chat_page_);
|
// login_page_->reset();
|
||||||
|
|
||||||
pageStack_->removeWidget(welcome_page_);
|
|
||||||
pageStack_->removeWidget(login_page_);
|
|
||||||
pageStack_->removeWidget(register_page_);
|
|
||||||
|
|
||||||
login_page_->reset();
|
|
||||||
chat_page_->bootstrap(userid, homeserver, token);
|
chat_page_->bootstrap(userid, homeserver, token);
|
||||||
connect(cache::client(), &Cache::databaseReady, this, &MainWindow::secretsChanged);
|
connect(cache::client(), &Cache::databaseReady, this, &MainWindow::secretsChanged);
|
||||||
connect(cache::client(), &Cache::secretChanged, this, &MainWindow::secretsChanged);
|
connect(cache::client(), &Cache::secretChanged, this, &MainWindow::secretsChanged);
|
||||||
|
|
||||||
emit reload();
|
emit reload();
|
||||||
|
nhlog::ui()->info("Switching to chat page");
|
||||||
|
emit switchToChatPage();
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -247,7 +425,7 @@ MainWindow::closeEvent(QCloseEvent *event)
|
||||||
{
|
{
|
||||||
if (WebRTCSession::instance().state() != webrtc::State::DISCONNECTED) {
|
if (WebRTCSession::instance().state() != webrtc::State::DISCONNECTED) {
|
||||||
if (QMessageBox::question(
|
if (QMessageBox::question(
|
||||||
this, QStringLiteral("nheko"), QStringLiteral("A call is in progress. Quit?")) !=
|
nullptr, QStringLiteral("nheko"), QStringLiteral("A call is in progress. Quit?")) !=
|
||||||
QMessageBox::Yes) {
|
QMessageBox::Yes) {
|
||||||
event->ignore();
|
event->ignore();
|
||||||
return;
|
return;
|
||||||
|
@ -292,20 +470,13 @@ MainWindow::hasActiveUser()
|
||||||
void
|
void
|
||||||
MainWindow::showOverlayProgressBar()
|
MainWindow::showOverlayProgressBar()
|
||||||
{
|
{
|
||||||
spinner_ = new LoadingIndicator(this);
|
|
||||||
spinner_->setFixedHeight(100);
|
|
||||||
spinner_->setFixedWidth(100);
|
|
||||||
spinner_->setObjectName(QStringLiteral("ChatPageLoadSpinner"));
|
|
||||||
spinner_->start();
|
|
||||||
|
|
||||||
showSolidOverlayModal(spinner_);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::openCreateRoomDialog(
|
MainWindow::openCreateRoomDialog(
|
||||||
std::function<void(const mtx::requests::CreateRoom &request)> callback)
|
std::function<void(const mtx::requests::CreateRoom &request)> callback)
|
||||||
{
|
{
|
||||||
auto dialog = new dialogs::CreateRoom(this);
|
auto dialog = new dialogs::CreateRoom(nullptr);
|
||||||
connect(dialog,
|
connect(dialog,
|
||||||
&dialogs::CreateRoom::createRoom,
|
&dialogs::CreateRoom::createRoom,
|
||||||
this,
|
this,
|
||||||
|
@ -315,50 +486,36 @@ MainWindow::openCreateRoomDialog(
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::showTransparentOverlayModal(QWidget *content, QFlags<Qt::AlignmentFlag> flags)
|
MainWindow::showTransparentOverlayModal(QWidget *, QFlags<Qt::AlignmentFlag>)
|
||||||
{
|
{}
|
||||||
modal_->setWidget(content);
|
|
||||||
modal_->setColor(QColor(30, 30, 30, 150));
|
|
||||||
modal_->setDismissible(true);
|
|
||||||
modal_->setContentAlignment(flags);
|
|
||||||
modal_->raise();
|
|
||||||
modal_->show();
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::showSolidOverlayModal(QWidget *content, QFlags<Qt::AlignmentFlag> flags)
|
MainWindow::showSolidOverlayModal(QWidget *, QFlags<Qt::AlignmentFlag>)
|
||||||
{
|
{
|
||||||
modal_->setWidget(content);
|
|
||||||
modal_->setColor(QColor(30, 30, 30));
|
|
||||||
modal_->setDismissible(false);
|
|
||||||
modal_->setContentAlignment(flags);
|
|
||||||
modal_->raise();
|
|
||||||
modal_->show();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
MainWindow::hasActiveDialogs() const
|
MainWindow::hasActiveDialogs() const
|
||||||
{
|
{
|
||||||
return modal_ && modal_->isVisible();
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
MainWindow::pageSupportsTray() const
|
MainWindow::pageSupportsTray() const
|
||||||
{
|
{
|
||||||
return !welcome_page_->isVisible() && !login_page_->isVisible() && !register_page_->isVisible();
|
return false; //! welcome_page_->isVisible() && !login_page_->isVisible() &&
|
||||||
|
//! !register_page_->isVisible();
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::hideOverlay()
|
MainWindow::hideOverlay()
|
||||||
{
|
{
|
||||||
if (modal_)
|
|
||||||
modal_->hide();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void
|
inline void
|
||||||
MainWindow::showDialog(QWidget *dialog)
|
MainWindow::showDialog(QWidget *dialog)
|
||||||
{
|
{
|
||||||
utils::centerWidget(dialog, this);
|
// utils::centerWidget(dialog, this);
|
||||||
dialog->raise();
|
dialog->raise();
|
||||||
dialog->show();
|
dialog->show();
|
||||||
}
|
}
|
||||||
|
@ -367,23 +524,13 @@ void
|
||||||
MainWindow::showWelcomePage()
|
MainWindow::showWelcomePage()
|
||||||
{
|
{
|
||||||
removeOverlayProgressBar();
|
removeOverlayProgressBar();
|
||||||
pageStack_->addWidget(welcome_page_);
|
|
||||||
pageStack_->setCurrentWidget(welcome_page_);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::showLoginPage()
|
MainWindow::showLoginPage()
|
||||||
{
|
{
|
||||||
if (modal_)
|
|
||||||
modal_->hide();
|
|
||||||
|
|
||||||
pageStack_->addWidget(login_page_);
|
|
||||||
pageStack_->setCurrentWidget(login_page_);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
MainWindow::showRegisterPage()
|
MainWindow::showRegisterPage()
|
||||||
{
|
{}
|
||||||
pageStack_->addWidget(register_page_);
|
|
||||||
pageStack_->setCurrentWidget(register_page_);
|
|
||||||
}
|
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
|
||||||
#include <QMainWindow>
|
#include <QQuickView>
|
||||||
#include <QSharedPointer>
|
#include <QSharedPointer>
|
||||||
#include <QStackedWidget>
|
#include <QStackedWidget>
|
||||||
#include <QSystemTrayIcon>
|
#include <QSystemTrayIcon>
|
||||||
|
@ -28,6 +28,7 @@ class OverlayModal;
|
||||||
class SnackBar;
|
class SnackBar;
|
||||||
class TrayIcon;
|
class TrayIcon;
|
||||||
class UserSettings;
|
class UserSettings;
|
||||||
|
class MxcImageProvider;
|
||||||
|
|
||||||
namespace mtx {
|
namespace mtx {
|
||||||
namespace requests {
|
namespace requests {
|
||||||
|
@ -42,17 +43,12 @@ class MemberList;
|
||||||
class ReCaptcha;
|
class ReCaptcha;
|
||||||
}
|
}
|
||||||
|
|
||||||
class MainWindow : public QMainWindow
|
class MainWindow : public QQuickView
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
Q_PROPERTY(int x READ x CONSTANT)
|
|
||||||
Q_PROPERTY(int y READ y CONSTANT)
|
|
||||||
Q_PROPERTY(int width READ width CONSTANT)
|
|
||||||
Q_PROPERTY(int height READ height CONSTANT)
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit MainWindow(QWidget *parent = nullptr);
|
explicit MainWindow(QWindow *parent = nullptr);
|
||||||
|
|
||||||
static MainWindow *instance() { return instance_; }
|
static MainWindow *instance() { return instance_; }
|
||||||
void saveCurrentWindowSize();
|
void saveCurrentWindowSize();
|
||||||
|
@ -67,8 +63,10 @@ public:
|
||||||
showTransparentOverlayModal(QWidget *content,
|
showTransparentOverlayModal(QWidget *content,
|
||||||
QFlags<Qt::AlignmentFlag> flags = Qt::AlignTop | Qt::AlignHCenter);
|
QFlags<Qt::AlignmentFlag> flags = Qt::AlignTop | Qt::AlignHCenter);
|
||||||
|
|
||||||
|
MxcImageProvider *imageProvider() { return imgProvider; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void closeEvent(QCloseEvent *event) override;
|
void closeEvent(QCloseEvent *event);
|
||||||
bool event(QEvent *event) override;
|
bool event(QEvent *event) override;
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
|
@ -97,6 +95,9 @@ signals:
|
||||||
void reload();
|
void reload();
|
||||||
void secretsChanged();
|
void secretsChanged();
|
||||||
|
|
||||||
|
void switchToChatPage();
|
||||||
|
void switchToWelcomePage();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void showDialog(QWidget *dialog);
|
void showDialog(QWidget *dialog);
|
||||||
bool hasActiveUser();
|
bool hasActiveUser();
|
||||||
|
@ -106,6 +107,8 @@ private:
|
||||||
//! Check if the current page supports the "minimize to tray" functionality.
|
//! Check if the current page supports the "minimize to tray" functionality.
|
||||||
bool pageSupportsTray() const;
|
bool pageSupportsTray() const;
|
||||||
|
|
||||||
|
void registerQmlTypes();
|
||||||
|
|
||||||
static MainWindow *instance_;
|
static MainWindow *instance_;
|
||||||
|
|
||||||
//! The initial welcome screen.
|
//! The initial welcome screen.
|
||||||
|
@ -114,16 +117,11 @@ private:
|
||||||
LoginPage *login_page_;
|
LoginPage *login_page_;
|
||||||
//! The register page.
|
//! The register page.
|
||||||
RegisterPage *register_page_;
|
RegisterPage *register_page_;
|
||||||
//! A stacked widget that handles the transitions between widgets.
|
|
||||||
QStackedWidget *pageStack_;
|
|
||||||
//! The main chat area.
|
//! The main chat area.
|
||||||
ChatPage *chat_page_;
|
ChatPage *chat_page_;
|
||||||
QSharedPointer<UserSettings> userSettings_;
|
QSharedPointer<UserSettings> userSettings_;
|
||||||
//! Tray icon that shows the unread message count.
|
//! Tray icon that shows the unread message count.
|
||||||
TrayIcon *trayIcon_;
|
TrayIcon *trayIcon_;
|
||||||
//! Notifications display.
|
|
||||||
SnackBar *snackBar_ = nullptr;
|
MxcImageProvider *imgProvider = nullptr;
|
||||||
//! Overlay modal used to project other widgets.
|
|
||||||
OverlayModal *modal_ = nullptr;
|
|
||||||
LoadingIndicator *spinner_ = nullptr;
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
#include <QPainter>
|
#include <QPainter>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
|
#include <QWindow>
|
||||||
|
|
||||||
#include "TrayIcon.h"
|
#include "TrayIcon.h"
|
||||||
|
|
||||||
|
@ -100,7 +101,7 @@ MsgCountComposedIcon::pixmap(const QSize &size, QIcon::Mode mode, QIcon::State s
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
TrayIcon::TrayIcon(const QString &filename, QWidget *parent)
|
TrayIcon::TrayIcon(const QString &filename, QWindow *parent)
|
||||||
: QSystemTrayIcon(parent)
|
: QSystemTrayIcon(parent)
|
||||||
{
|
{
|
||||||
#if defined(Q_OS_MAC) || defined(Q_OS_WIN)
|
#if defined(Q_OS_MAC) || defined(Q_OS_WIN)
|
||||||
|
@ -110,13 +111,13 @@ TrayIcon::TrayIcon(const QString &filename, QWidget *parent)
|
||||||
setIcon(QIcon(icon_));
|
setIcon(QIcon(icon_));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
QMenu *menu = new QMenu(parent);
|
QMenu *menu = new QMenu();
|
||||||
setContextMenu(menu);
|
setContextMenu(menu);
|
||||||
|
|
||||||
viewAction_ = new QAction(tr("Show"), this);
|
viewAction_ = new QAction(tr("Show"), this);
|
||||||
quitAction_ = new QAction(tr("Quit"), this);
|
quitAction_ = new QAction(tr("Quit"), this);
|
||||||
|
|
||||||
connect(viewAction_, SIGNAL(triggered()), parent, SLOT(show()));
|
connect(viewAction_, &QAction::triggered, parent, &QWindow::show);
|
||||||
connect(quitAction_, &QAction::triggered, this, QApplication::quit);
|
connect(quitAction_, &QAction::triggered, this, QApplication::quit);
|
||||||
|
|
||||||
menu->addAction(viewAction_);
|
menu->addAction(viewAction_);
|
||||||
|
|
|
@ -40,7 +40,7 @@ class TrayIcon : public QSystemTrayIcon
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
TrayIcon(const QString &filename, QWidget *parent);
|
TrayIcon(const QString &filename, QWindow *parent);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void setUnreadCount(int count);
|
void setUnreadCount(int count);
|
||||||
|
|
|
@ -1518,7 +1518,7 @@ UserSettingsModel::setData(const QModelIndex &index, const QVariant &value, int
|
||||||
QString homeFolder =
|
QString homeFolder =
|
||||||
QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
||||||
auto filepath = QFileDialog::getOpenFileName(
|
auto filepath = QFileDialog::getOpenFileName(
|
||||||
MainWindow::instance(), tr("Select a file"), homeFolder, tr("All Files (*)"));
|
nullptr, tr("Select a file"), homeFolder, tr("All Files (*)"));
|
||||||
if (!filepath.isEmpty()) {
|
if (!filepath.isEmpty()) {
|
||||||
i->setRingtone(filepath);
|
i->setRingtone(filepath);
|
||||||
i->setRingtone(filepath);
|
i->setRingtone(filepath);
|
||||||
|
@ -1600,11 +1600,11 @@ UserSettingsModel::importSessionKeys()
|
||||||
{
|
{
|
||||||
const QString homeFolder = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
const QString homeFolder = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
||||||
const QString fileName = QFileDialog::getOpenFileName(
|
const QString fileName = QFileDialog::getOpenFileName(
|
||||||
MainWindow::instance(), tr("Open Sessions File"), homeFolder, QLatin1String(""));
|
nullptr, tr("Open Sessions File"), homeFolder, QLatin1String(""));
|
||||||
|
|
||||||
QFile file(fileName);
|
QFile file(fileName);
|
||||||
if (!file.open(QIODevice::ReadOnly)) {
|
if (!file.open(QIODevice::ReadOnly)) {
|
||||||
QMessageBox::warning(MainWindow::instance(), tr("Error"), file.errorString());
|
QMessageBox::warning(nullptr, tr("Error"), file.errorString());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1612,7 +1612,7 @@ UserSettingsModel::importSessionKeys()
|
||||||
auto payload = std::string(bin.data(), bin.size());
|
auto payload = std::string(bin.data(), bin.size());
|
||||||
|
|
||||||
bool ok;
|
bool ok;
|
||||||
auto password = QInputDialog::getText(MainWindow::instance(),
|
auto password = QInputDialog::getText(nullptr,
|
||||||
tr("File Password"),
|
tr("File Password"),
|
||||||
tr("Enter the passphrase to decrypt the file:"),
|
tr("Enter the passphrase to decrypt the file:"),
|
||||||
QLineEdit::Password,
|
QLineEdit::Password,
|
||||||
|
@ -1622,8 +1622,7 @@ UserSettingsModel::importSessionKeys()
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (password.isEmpty()) {
|
if (password.isEmpty()) {
|
||||||
QMessageBox::warning(
|
QMessageBox::warning(nullptr, tr("Error"), tr("The password cannot be empty"));
|
||||||
MainWindow::instance(), tr("Error"), tr("The password cannot be empty"));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1631,7 +1630,7 @@ UserSettingsModel::importSessionKeys()
|
||||||
auto sessions = mtx::crypto::decrypt_exported_sessions(payload, password.toStdString());
|
auto sessions = mtx::crypto::decrypt_exported_sessions(payload, password.toStdString());
|
||||||
cache::importSessionKeys(std::move(sessions));
|
cache::importSessionKeys(std::move(sessions));
|
||||||
} catch (const std::exception &e) {
|
} catch (const std::exception &e) {
|
||||||
QMessageBox::warning(MainWindow::instance(), tr("Error"), e.what());
|
QMessageBox::warning(nullptr, tr("Error"), e.what());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void
|
void
|
||||||
|
@ -1639,7 +1638,7 @@ UserSettingsModel::exportSessionKeys()
|
||||||
{
|
{
|
||||||
// Open password dialog.
|
// Open password dialog.
|
||||||
bool ok;
|
bool ok;
|
||||||
auto password = QInputDialog::getText(MainWindow::instance(),
|
auto password = QInputDialog::getText(nullptr,
|
||||||
tr("File Password"),
|
tr("File Password"),
|
||||||
tr("Enter passphrase to encrypt your session keys:"),
|
tr("Enter passphrase to encrypt your session keys:"),
|
||||||
QLineEdit::Password,
|
QLineEdit::Password,
|
||||||
|
@ -1649,19 +1648,18 @@ UserSettingsModel::exportSessionKeys()
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (password.isEmpty()) {
|
if (password.isEmpty()) {
|
||||||
QMessageBox::warning(
|
QMessageBox::warning(nullptr, tr("Error"), tr("The password cannot be empty"));
|
||||||
MainWindow::instance(), tr("Error"), tr("The password cannot be empty"));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Open file dialog to save the file.
|
// Open file dialog to save the file.
|
||||||
const QString homeFolder = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
const QString homeFolder = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
||||||
const QString fileName = QFileDialog::getSaveFileName(
|
const QString fileName = QFileDialog::getSaveFileName(
|
||||||
MainWindow::instance(), tr("File to save the exported session keys"), homeFolder);
|
nullptr, tr("File to save the exported session keys"), homeFolder);
|
||||||
|
|
||||||
QFile file(fileName);
|
QFile file(fileName);
|
||||||
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
||||||
QMessageBox::warning(MainWindow::instance(), tr("Error"), file.errorString());
|
QMessageBox::warning(nullptr, tr("Error"), file.errorString());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1679,7 +1677,7 @@ UserSettingsModel::exportSessionKeys()
|
||||||
out << prefix << newline << b64 << newline << suffix << newline;
|
out << prefix << newline << b64 << newline << suffix << newline;
|
||||||
file.close();
|
file.close();
|
||||||
} catch (const std::exception &e) {
|
} catch (const std::exception &e) {
|
||||||
QMessageBox::warning(MainWindow::instance(), tr("Error"), e.what());
|
QMessageBox::warning(nullptr, tr("Error"), e.what());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void
|
void
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
#include <QLibraryInfo>
|
#include <QLibraryInfo>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QPoint>
|
#include <QPoint>
|
||||||
|
#include <QQuickView>
|
||||||
#include <QScreen>
|
#include <QScreen>
|
||||||
#include <QStandardPaths>
|
#include <QStandardPaths>
|
||||||
#include <QTranslator>
|
#include <QTranslator>
|
||||||
|
@ -279,6 +280,7 @@ main(int argc, char *argv[])
|
||||||
font.setPointSizeF(settings.lock()->fontSize());
|
font.setPointSizeF(settings.lock()->fontSize());
|
||||||
|
|
||||||
app.setFont(font);
|
app.setFont(font);
|
||||||
|
settings.lock()->applyTheme();
|
||||||
|
|
||||||
if (QLocale().language() == QLocale::C)
|
if (QLocale().language() == QLocale::C)
|
||||||
QLocale::setDefault(QLocale(QLocale::English, QLocale::UnitedKingdom));
|
QLocale::setDefault(QLocale(QLocale::English, QLocale::UnitedKingdom));
|
||||||
|
@ -296,15 +298,16 @@ main(int argc, char *argv[])
|
||||||
app.installTranslator(&appTranslator);
|
app.installTranslator(&appTranslator);
|
||||||
|
|
||||||
MainWindow w;
|
MainWindow w;
|
||||||
|
// QQuickView w;
|
||||||
|
|
||||||
// Move the MainWindow to the center
|
// Move the MainWindow to the center
|
||||||
w.move(screenCenter(w.width(), w.height()));
|
// w.move(screenCenter(w.width(), w.height()));
|
||||||
|
|
||||||
if (!(settings.lock()->startInTray() && settings.lock()->tray()))
|
if (!(settings.lock()->startInTray() && settings.lock()->tray()))
|
||||||
w.show();
|
w.show();
|
||||||
|
|
||||||
QObject::connect(&app, &QApplication::aboutToQuit, &w, [&w]() {
|
QObject::connect(&app, &QApplication::aboutToQuit, &w, [&w]() {
|
||||||
w.saveCurrentWindowSize();
|
// w.saveCurrentWindowSize();
|
||||||
if (http::client() != nullptr) {
|
if (http::client() != nullptr) {
|
||||||
nhlog::net()->debug("shutting down all I/O threads & open connections");
|
nhlog::net()->debug("shutting down all I/O threads & open connections");
|
||||||
http::client()->close(true);
|
http::client()->close(true);
|
||||||
|
@ -314,7 +317,7 @@ main(int argc, char *argv[])
|
||||||
QObject::connect(&app, &SingleApplication::instanceStarted, &w, [&w]() {
|
QObject::connect(&app, &SingleApplication::instanceStarted, &w, [&w]() {
|
||||||
w.show();
|
w.show();
|
||||||
w.raise();
|
w.raise();
|
||||||
w.activateWindow();
|
w.requestActivate();
|
||||||
});
|
});
|
||||||
|
|
||||||
// It seems like handling the message in a blocking manner is a no-go. I have no idea how to
|
// It seems like handling the message in a blocking manner is a no-go. I have no idea how to
|
||||||
|
|
|
@ -266,8 +266,8 @@ void
|
||||||
InputBar::openFileSelection()
|
InputBar::openFileSelection()
|
||||||
{
|
{
|
||||||
const QString homeFolder = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
const QString homeFolder = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
|
||||||
const auto fileName = QFileDialog::getOpenFileName(
|
const auto fileName =
|
||||||
ChatPage::instance(), tr("Select a file"), homeFolder, tr("All Files (*)"));
|
QFileDialog::getOpenFileName(nullptr, tr("Select a file"), homeFolder, tr("All Files (*)"));
|
||||||
|
|
||||||
if (fileName.isEmpty())
|
if (fileName.isEmpty())
|
||||||
return;
|
return;
|
||||||
|
@ -659,7 +659,7 @@ InputBar::command(const QString &command, QString args)
|
||||||
void
|
void
|
||||||
InputBar::showPreview(const QMimeData &source, const QString &path, const QStringList &formats)
|
InputBar::showPreview(const QMimeData &source, const QString &path, const QStringList &formats)
|
||||||
{
|
{
|
||||||
auto *previewDialog_ = new dialogs::PreviewUploadOverlay(ChatPage::instance());
|
auto *previewDialog_ = new dialogs::PreviewUploadOverlay(nullptr);
|
||||||
previewDialog_->setAttribute(Qt::WA_DeleteOnClose);
|
previewDialog_->setAttribute(Qt::WA_DeleteOnClose);
|
||||||
|
|
||||||
// Force SVG to _not_ be handled as an image, but as raw data
|
// Force SVG to _not_ be handled as an image, but as raw data
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "Cache_p.h"
|
#include "Cache_p.h"
|
||||||
#include "ChatPage.h"
|
#include "ChatPage.h"
|
||||||
#include "Logging.h"
|
#include "Logging.h"
|
||||||
|
#include "MainWindow.h"
|
||||||
#include "MatrixClient.h"
|
#include "MatrixClient.h"
|
||||||
#include "MxcImageProvider.h"
|
#include "MxcImageProvider.h"
|
||||||
#include "TimelineModel.h"
|
#include "TimelineModel.h"
|
||||||
|
@ -275,7 +276,7 @@ RoomlistModel::addRoom(const QString &room_id, bool suppressInsertNotification)
|
||||||
|
|
||||||
connect(newRoom.data(),
|
connect(newRoom.data(),
|
||||||
&TimelineModel::newEncryptedImage,
|
&TimelineModel::newEncryptedImage,
|
||||||
manager->imageProvider(),
|
MainWindow::instance()->imageProvider(),
|
||||||
&MxcImageProvider::addEncryptionInfo);
|
&MxcImageProvider::addEncryptionInfo);
|
||||||
connect(newRoom.data(),
|
connect(newRoom.data(),
|
||||||
&TimelineModel::forwardToRoom,
|
&TimelineModel::forwardToRoom,
|
||||||
|
@ -509,7 +510,7 @@ RoomlistModel::sync(const mtx::responses::Sync &sync_)
|
||||||
// room_model->addEvents(room.timeline);
|
// room_model->addEvents(room.timeline);
|
||||||
connect(room_model.data(),
|
connect(room_model.data(),
|
||||||
&TimelineModel::newCallEvent,
|
&TimelineModel::newCallEvent,
|
||||||
manager->callManager(),
|
ChatPage::instance()->callManager(),
|
||||||
&CallManager::syncEvent,
|
&CallManager::syncEvent,
|
||||||
Qt::UniqueConnection);
|
Qt::UniqueConnection);
|
||||||
|
|
||||||
|
|
|
@ -1031,7 +1031,7 @@ TimelineModel::setCurrentIndex(int index)
|
||||||
if (index != oldIndex)
|
if (index != oldIndex)
|
||||||
emit currentIndexChanged(index);
|
emit currentIndexChanged(index);
|
||||||
|
|
||||||
if (!ChatPage::instance()->isActiveWindow())
|
if (!MainWindow::instance()->isActive())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (!currentId.startsWith('m')) {
|
if (!currentId.startsWith('m')) {
|
||||||
|
@ -1495,7 +1495,7 @@ TimelineModel::saveMedia(const QString &eventId) const
|
||||||
const QString openLocation = downloadsFolder + "/" + originalFilename;
|
const QString openLocation = downloadsFolder + "/" + originalFilename;
|
||||||
|
|
||||||
const QString filename =
|
const QString filename =
|
||||||
QFileDialog::getSaveFileName(manager_->getWidget(), dialogTitle, openLocation, filterString);
|
QFileDialog::getSaveFileName(nullptr, dialogTitle, openLocation, filterString);
|
||||||
|
|
||||||
if (filename.isEmpty())
|
if (filename.isEmpty())
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -8,7 +8,6 @@
|
||||||
#include <QDropEvent>
|
#include <QDropEvent>
|
||||||
#include <QFileDialog>
|
#include <QFileDialog>
|
||||||
#include <QMetaType>
|
#include <QMetaType>
|
||||||
#include <QPalette>
|
|
||||||
#include <QQmlContext>
|
#include <QQmlContext>
|
||||||
#include <QQmlEngine>
|
#include <QQmlEngine>
|
||||||
#include <QStandardPaths>
|
#include <QStandardPaths>
|
||||||
|
@ -45,10 +44,6 @@
|
||||||
#include "ui/NhekoGlobalObject.h"
|
#include "ui/NhekoGlobalObject.h"
|
||||||
#include "ui/UIA.h"
|
#include "ui/UIA.h"
|
||||||
|
|
||||||
Q_DECLARE_METATYPE(mtx::events::collections::TimelineEvents)
|
|
||||||
Q_DECLARE_METATYPE(std::vector<DeviceInfo>)
|
|
||||||
Q_DECLARE_METATYPE(std::vector<mtx::responses::PublicRoomsChunk>)
|
|
||||||
|
|
||||||
namespace msgs = mtx::events::msg;
|
namespace msgs = mtx::events::msg;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
@ -102,19 +97,6 @@ void
|
||||||
TimelineViewManager::updateColorPalette()
|
TimelineViewManager::updateColorPalette()
|
||||||
{
|
{
|
||||||
userColors.clear();
|
userColors.clear();
|
||||||
|
|
||||||
if (ChatPage::instance()->userSettings()->theme() == QLatin1String("light")) {
|
|
||||||
view->rootContext()->setContextProperty(QStringLiteral("currentActivePalette"), QPalette());
|
|
||||||
view->rootContext()->setContextProperty(QStringLiteral("currentInactivePalette"),
|
|
||||||
QPalette());
|
|
||||||
} else if (ChatPage::instance()->userSettings()->theme() == QLatin1String("dark")) {
|
|
||||||
view->rootContext()->setContextProperty(QStringLiteral("currentActivePalette"), QPalette());
|
|
||||||
view->rootContext()->setContextProperty(QStringLiteral("currentInactivePalette"),
|
|
||||||
QPalette());
|
|
||||||
} else {
|
|
||||||
view->rootContext()->setContextProperty(QStringLiteral("currentActivePalette"), QPalette());
|
|
||||||
view->rootContext()->setContextProperty(QStringLiteral("currentInactivePalette"), nullptr);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QColor
|
QColor
|
||||||
|
@ -126,112 +108,15 @@ TimelineViewManager::userColor(QString id, QColor background)
|
||||||
return userColors.value(idx);
|
return userColors.value(idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
TimelineViewManager::TimelineViewManager(CallManager *callManager, ChatPage *parent)
|
TimelineViewManager::TimelineViewManager(CallManager *, ChatPage *parent)
|
||||||
: QObject(parent)
|
: QObject(parent)
|
||||||
, imgProvider(new MxcImageProvider())
|
|
||||||
, colorImgProvider(new ColorImageProvider())
|
|
||||||
, blurhashProvider(new BlurhashProvider())
|
|
||||||
, jdenticonProvider(new JdenticonProvider())
|
|
||||||
, rooms_(new RoomlistModel(this))
|
, rooms_(new RoomlistModel(this))
|
||||||
, communities_(new CommunitiesModel(this))
|
, communities_(new CommunitiesModel(this))
|
||||||
, callManager_(callManager)
|
|
||||||
, verificationManager_(new VerificationManager(this))
|
, verificationManager_(new VerificationManager(this))
|
||||||
, presenceEmitter(new PresenceEmitter(this))
|
, presenceEmitter(new PresenceEmitter(this))
|
||||||
{
|
{
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationAccept>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationCancel>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationDone>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationKey>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationMac>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationReady>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationRequest>();
|
|
||||||
qRegisterMetaType<mtx::events::msg::KeyVerificationStart>();
|
|
||||||
qRegisterMetaType<CombinedImagePackModel *>();
|
|
||||||
|
|
||||||
qRegisterMetaType<std::vector<mtx::responses::PublicRoomsChunk>>();
|
|
||||||
|
|
||||||
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<DelegateChoice>("im.nheko", 1, 0, "DelegateChoice");
|
|
||||||
qmlRegisterType<DelegateChooser>("im.nheko", 1, 0, "DelegateChooser");
|
|
||||||
qmlRegisterType<NhekoDropArea>("im.nheko", 1, 0, "NhekoDropArea");
|
|
||||||
qmlRegisterType<NhekoCursorShape>("im.nheko", 1, 0, "CursorShape");
|
|
||||||
qmlRegisterType<MxcAnimatedImage>("im.nheko", 1, 0, "MxcAnimatedImage");
|
|
||||||
qmlRegisterType<MxcMediaProxy>("im.nheko", 1, 0, "MxcMedia");
|
|
||||||
qmlRegisterUncreatableType<DeviceVerificationFlow>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"DeviceVerificationFlow",
|
|
||||||
QStringLiteral("Can't create verification flow from QML!"));
|
|
||||||
qmlRegisterUncreatableType<UserProfile>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"UserProfileModel",
|
|
||||||
QStringLiteral("UserProfile needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<MemberList>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"MemberList",
|
|
||||||
QStringLiteral("MemberList needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<RoomSettings>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"RoomSettingsModel",
|
|
||||||
QStringLiteral("Room Settings needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<TimelineModel>(
|
|
||||||
"im.nheko", 1, 0, "Room", QStringLiteral("Room needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<ImagePackListModel>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"ImagePackListModel",
|
|
||||||
QStringLiteral("ImagePackListModel needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<SingleImagePackModel>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"SingleImagePackModel",
|
|
||||||
QStringLiteral("SingleImagePackModel needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<InviteesModel>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"InviteesModel",
|
|
||||||
QStringLiteral("InviteesModel needs to be instantiated on the C++ side"));
|
|
||||||
qmlRegisterUncreatableType<ReadReceiptsProxy>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"ReadReceiptsProxy",
|
|
||||||
QStringLiteral("ReadReceiptsProxy needs to be instantiated on the C++ side"));
|
|
||||||
|
|
||||||
static auto self = this;
|
static auto self = this;
|
||||||
qmlRegisterSingletonInstance("im.nheko", 1, 0, "MainWindow", MainWindow::instance());
|
|
||||||
qmlRegisterSingletonInstance("im.nheko", 1, 0, "TimelineManager", self);
|
qmlRegisterSingletonInstance("im.nheko", 1, 0, "TimelineManager", self);
|
||||||
qmlRegisterSingletonInstance("im.nheko", 1, 0, "UIA", UIA::instance());
|
|
||||||
qmlRegisterSingletonType<RoomlistModel>(
|
qmlRegisterSingletonType<RoomlistModel>(
|
||||||
"im.nheko", 1, 0, "Rooms", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
"im.nheko", 1, 0, "Rooms", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
||||||
auto ptr = new FilteredRoomlistModel(self->rooms_);
|
auto ptr = new FilteredRoomlistModel(self->rooms_);
|
||||||
|
@ -247,79 +132,15 @@ TimelineViewManager::TimelineViewManager(CallManager *callManager, ChatPage *par
|
||||||
return ptr;
|
return ptr;
|
||||||
});
|
});
|
||||||
qmlRegisterSingletonInstance("im.nheko", 1, 0, "Communities", self->communities_);
|
qmlRegisterSingletonInstance("im.nheko", 1, 0, "Communities", self->communities_);
|
||||||
qmlRegisterSingletonInstance(
|
|
||||||
"im.nheko", 1, 0, "Settings", ChatPage::instance()->userSettings().data());
|
|
||||||
qmlRegisterSingletonInstance(
|
|
||||||
"im.nheko", 1, 0, "CallManager", ChatPage::instance()->callManager());
|
|
||||||
qmlRegisterSingletonType<Clipboard>(
|
|
||||||
"im.nheko", 1, 0, "Clipboard", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
|
||||||
return new Clipboard();
|
|
||||||
});
|
|
||||||
qmlRegisterSingletonType<Nheko>(
|
|
||||||
"im.nheko", 1, 0, "Nheko", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
|
||||||
return new Nheko();
|
|
||||||
});
|
|
||||||
qmlRegisterSingletonType<UserSettingsModel>(
|
|
||||||
"im.nheko", 1, 0, "UserSettingsModel", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
|
||||||
return new UserSettingsModel();
|
|
||||||
});
|
|
||||||
qmlRegisterSingletonInstance("im.nheko", 1, 0, "VerificationManager", verificationManager_);
|
qmlRegisterSingletonInstance("im.nheko", 1, 0, "VerificationManager", verificationManager_);
|
||||||
qmlRegisterSingletonInstance("im.nheko", 1, 0, "Presence", presenceEmitter);
|
qmlRegisterSingletonInstance("im.nheko", 1, 0, "Presence", presenceEmitter);
|
||||||
qmlRegisterSingletonType<SelfVerificationStatus>(
|
|
||||||
"im.nheko", 1, 0, "SelfVerificationStatus", [](QQmlEngine *, QJSEngine *) -> QObject * {
|
|
||||||
auto ptr = new SelfVerificationStatus();
|
|
||||||
QObject::connect(ChatPage::instance(),
|
|
||||||
&ChatPage::initializeEmptyViews,
|
|
||||||
ptr,
|
|
||||||
&SelfVerificationStatus::invalidate);
|
|
||||||
return ptr;
|
|
||||||
});
|
|
||||||
|
|
||||||
qRegisterMetaType<mtx::events::collections::TimelineEvents>();
|
|
||||||
qRegisterMetaType<std::vector<DeviceInfo>>();
|
|
||||||
|
|
||||||
qmlRegisterUncreatableType<FilteredCommunitiesModel>(
|
|
||||||
"im.nheko",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"FilteredCommunitiesModel",
|
|
||||||
QStringLiteral("Use Communities.filtered() to create a FilteredCommunitiesModel"));
|
|
||||||
|
|
||||||
qmlRegisterType<emoji::EmojiModel>("im.nheko.EmojiModel", 1, 0, "EmojiModel");
|
|
||||||
qmlRegisterUncreatableType<emoji::Emoji>(
|
|
||||||
"im.nheko.EmojiModel", 1, 0, "Emoji", QStringLiteral("Used by emoji models"));
|
|
||||||
qmlRegisterUncreatableMetaObject(emoji::staticMetaObject,
|
|
||||||
"im.nheko.EmojiModel",
|
|
||||||
1,
|
|
||||||
0,
|
|
||||||
"EmojiCategory",
|
|
||||||
QStringLiteral("Error: Only enums"));
|
|
||||||
|
|
||||||
qmlRegisterType<RoomDirectoryModel>("im.nheko", 1, 0, "RoomDirectoryModel");
|
|
||||||
|
|
||||||
#ifdef USE_QUICK_VIEW
|
|
||||||
view = new QQuickView(parent);
|
|
||||||
container = QWidget::createWindowContainer(view, parent);
|
|
||||||
#else
|
|
||||||
view = new QQuickWidget(parent);
|
|
||||||
container = view;
|
|
||||||
view->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
||||||
container->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
|
|
||||||
|
|
||||||
connect(view, &QQuickWidget::statusChanged, this, [](QQuickWidget::Status status) {
|
|
||||||
nhlog::ui()->debug("Status changed to {}", status);
|
|
||||||
});
|
|
||||||
#endif
|
|
||||||
container->setMinimumSize(200, 200);
|
|
||||||
updateColorPalette();
|
updateColorPalette();
|
||||||
view->engine()->addImageProvider(QStringLiteral("MxcImage"), imgProvider);
|
|
||||||
view->engine()->addImageProvider(QStringLiteral("colorimage"), colorImgProvider);
|
|
||||||
view->engine()->addImageProvider(QStringLiteral("blurhash"), blurhashProvider);
|
|
||||||
if (JdenticonProvider::isAvailable())
|
|
||||||
view->engine()->addImageProvider(QStringLiteral("jdenticon"), jdenticonProvider);
|
|
||||||
view->setSource(QUrl(QStringLiteral("qrc:///qml/Root.qml")));
|
|
||||||
|
|
||||||
connect(parent, &ChatPage::themeChanged, this, &TimelineViewManager::updateColorPalette);
|
connect(UserSettings::instance().get(),
|
||||||
|
&UserSettings::themeChanged,
|
||||||
|
this,
|
||||||
|
&TimelineViewManager::updateColorPalette);
|
||||||
connect(parent,
|
connect(parent,
|
||||||
&ChatPage::receivedRoomDeviceVerificationRequest,
|
&ChatPage::receivedRoomDeviceVerificationRequest,
|
||||||
verificationManager_,
|
verificationManager_,
|
||||||
|
@ -379,7 +200,8 @@ void
|
||||||
TimelineViewManager::setVideoCallItem()
|
TimelineViewManager::setVideoCallItem()
|
||||||
{
|
{
|
||||||
WebRTCSession::instance().setVideoItem(
|
WebRTCSession::instance().setVideoItem(
|
||||||
view->rootObject()->findChild<QQuickItem *>(QStringLiteral("videoCallItem")));
|
MainWindow::instance()->rootObject()->findChild<QQuickItem *>(
|
||||||
|
QStringLiteral("videoCallItem")));
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -401,7 +223,7 @@ TimelineViewManager::showEvent(const QString &room_id, const QString &event_id)
|
||||||
if (auto room = rooms_->getRoomById(room_id)) {
|
if (auto room = rooms_->getRoomById(room_id)) {
|
||||||
if (rooms_->currentRoom() != room) {
|
if (rooms_->currentRoom() != room) {
|
||||||
rooms_->setCurrentRoom(room_id);
|
rooms_->setCurrentRoom(room_id);
|
||||||
container->setFocus();
|
MainWindow::instance()->requestActivate();
|
||||||
nhlog::ui()->info("Activated room {}", room_id.toStdString());
|
nhlog::ui()->info("Activated room {}", room_id.toStdString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -439,7 +261,7 @@ TimelineViewManager::saveMedia(QString mxcUrl)
|
||||||
QStandardPaths::writableLocation(QStandardPaths::DownloadLocation);
|
QStandardPaths::writableLocation(QStandardPaths::DownloadLocation);
|
||||||
const QString openLocation = downloadsFolder + "/" + mxcUrl.splitRef(u'/').constLast();
|
const QString openLocation = downloadsFolder + "/" + mxcUrl.splitRef(u'/').constLast();
|
||||||
|
|
||||||
const QString filename = QFileDialog::getSaveFileName(getWidget(), {}, openLocation);
|
const QString filename = QFileDialog::getSaveFileName(nullptr, {}, openLocation);
|
||||||
|
|
||||||
if (filename.isEmpty())
|
if (filename.isEmpty())
|
||||||
return;
|
return;
|
||||||
|
@ -590,12 +412,6 @@ TimelineViewManager::completerFor(QString completerName, QString roomId)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
TimelineViewManager::focusTimeline()
|
|
||||||
{
|
|
||||||
getWidget()->setFocus();
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
TimelineViewManager::forwardMessageToRoom(mtx::events::collections::TimelineEvents *e,
|
TimelineViewManager::forwardMessageToRoom(mtx::events::collections::TimelineEvents *e,
|
||||||
QString roomId)
|
QString roomId)
|
||||||
|
|
|
@ -8,8 +8,6 @@
|
||||||
#include <QHash>
|
#include <QHash>
|
||||||
#include <QQuickItem>
|
#include <QQuickItem>
|
||||||
#include <QQuickTextDocument>
|
#include <QQuickTextDocument>
|
||||||
#include <QQuickView>
|
|
||||||
#include <QQuickWidget>
|
|
||||||
#include <QWidget>
|
#include <QWidget>
|
||||||
|
|
||||||
#include <mtx/common.hpp>
|
#include <mtx/common.hpp>
|
||||||
|
@ -48,12 +46,9 @@ class TimelineViewManager : public QObject
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TimelineViewManager(CallManager *callManager, ChatPage *parent = nullptr);
|
TimelineViewManager(CallManager *callManager, ChatPage *parent = nullptr);
|
||||||
QWidget *getWidget() const { return container; }
|
|
||||||
|
|
||||||
void sync(const mtx::responses::Sync &sync_);
|
void sync(const mtx::responses::Sync &sync_);
|
||||||
|
|
||||||
MxcImageProvider *imageProvider() { return imgProvider; }
|
|
||||||
CallManager *callManager() { return callManager_; }
|
|
||||||
VerificationManager *verificationManager() { return verificationManager_; }
|
VerificationManager *verificationManager() { return verificationManager_; }
|
||||||
|
|
||||||
void clearAll() { rooms_->clear(); }
|
void clearAll() { rooms_->clear(); }
|
||||||
|
@ -105,7 +100,6 @@ public slots:
|
||||||
}
|
}
|
||||||
|
|
||||||
void showEvent(const QString &room_id, const QString &event_id);
|
void showEvent(const QString &room_id, const QString &event_id);
|
||||||
void focusTimeline();
|
|
||||||
|
|
||||||
void updateColorPalette();
|
void updateColorPalette();
|
||||||
void queueReply(const QString &roomid, const QString &repliedToEvent, const QString &replyBody);
|
void queueReply(const QString &roomid, const QString &repliedToEvent, const QString &replyBody);
|
||||||
|
@ -122,18 +116,6 @@ public slots:
|
||||||
RoomlistModel *rooms() { return rooms_; }
|
RoomlistModel *rooms() { return rooms_; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
#ifdef USE_QUICK_VIEW
|
|
||||||
QQuickView *view;
|
|
||||||
#else
|
|
||||||
QQuickWidget *view;
|
|
||||||
#endif
|
|
||||||
QWidget *container;
|
|
||||||
|
|
||||||
MxcImageProvider *imgProvider;
|
|
||||||
ColorImageProvider *colorImgProvider;
|
|
||||||
BlurhashProvider *blurhashProvider;
|
|
||||||
JdenticonProvider *jdenticonProvider;
|
|
||||||
|
|
||||||
bool isInitialSync_ = true;
|
bool isInitialSync_ = true;
|
||||||
bool isWindowFocused_ = false;
|
bool isWindowFocused_ = false;
|
||||||
|
|
||||||
|
@ -141,7 +123,6 @@ private:
|
||||||
CommunitiesModel *communities_ = nullptr;
|
CommunitiesModel *communities_ = nullptr;
|
||||||
|
|
||||||
// don't move this above the rooms_
|
// don't move this above the rooms_
|
||||||
CallManager *callManager_ = nullptr;
|
|
||||||
VerificationManager *verificationManager_ = nullptr;
|
VerificationManager *verificationManager_ = nullptr;
|
||||||
PresenceEmitter *presenceEmitter = nullptr;
|
PresenceEmitter *presenceEmitter = nullptr;
|
||||||
|
|
||||||
|
|
|
@ -139,5 +139,5 @@ Nheko::openCreateRoomDialog() const
|
||||||
void
|
void
|
||||||
Nheko::reparent(QWindow *win) const
|
Nheko::reparent(QWindow *win) const
|
||||||
{
|
{
|
||||||
win->setTransientParent(MainWindow::instance()->windowHandle());
|
win->setTransientParent(MainWindow::instance());
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,7 +13,6 @@
|
||||||
#include <mtx/responses/common.hpp>
|
#include <mtx/responses/common.hpp>
|
||||||
|
|
||||||
#include "Logging.h"
|
#include "Logging.h"
|
||||||
#include "MainWindow.h"
|
|
||||||
#include "dialogs/FallbackAuth.h"
|
#include "dialogs/FallbackAuth.h"
|
||||||
#include "dialogs/ReCaptcha.h"
|
#include "dialogs/ReCaptcha.h"
|
||||||
|
|
||||||
|
@ -71,7 +70,7 @@ UIA::genericHandler(QString context)
|
||||||
emit phoneNumber();
|
emit phoneNumber();
|
||||||
} else if (current_stage == mtx::user_interactive::auth_types::recaptcha) {
|
} else if (current_stage == mtx::user_interactive::auth_types::recaptcha) {
|
||||||
auto captchaDialog =
|
auto captchaDialog =
|
||||||
new dialogs::ReCaptcha(QString::fromStdString(u.session), MainWindow::instance());
|
new dialogs::ReCaptcha(QString::fromStdString(u.session), nullptr);
|
||||||
captchaDialog->setWindowTitle(context);
|
captchaDialog->setWindowTitle(context);
|
||||||
|
|
||||||
connect(
|
connect(
|
||||||
|
@ -95,7 +94,7 @@ UIA::genericHandler(QString context)
|
||||||
} else if (current_stage == mtx::user_interactive::auth_types::registration_token) {
|
} else if (current_stage == mtx::user_interactive::auth_types::registration_token) {
|
||||||
bool ok;
|
bool ok;
|
||||||
QString token =
|
QString token =
|
||||||
QInputDialog::getText(MainWindow::instance(),
|
QInputDialog::getText(nullptr,
|
||||||
context,
|
context,
|
||||||
tr("Please enter a valid registration token."),
|
tr("Please enter a valid registration token."),
|
||||||
QLineEdit::Normal,
|
QLineEdit::Normal,
|
||||||
|
@ -113,7 +112,7 @@ UIA::genericHandler(QString context)
|
||||||
// use fallback
|
// use fallback
|
||||||
auto dialog = new dialogs::FallbackAuth(QString::fromStdString(current_stage),
|
auto dialog = new dialogs::FallbackAuth(QString::fromStdString(current_stage),
|
||||||
QString::fromStdString(u.session),
|
QString::fromStdString(u.session),
|
||||||
MainWindow::instance());
|
nullptr);
|
||||||
dialog->setWindowTitle(context);
|
dialog->setWindowTitle(context);
|
||||||
|
|
||||||
connect(dialog, &dialogs::FallbackAuth::confirmation, this, [h, u, dialog]() {
|
connect(dialog, &dialogs::FallbackAuth::confirmation, this, [h, u, dialog]() {
|
||||||
|
|
Loading…
Reference in a new issue