2018-02-27 05:10:08 +00:00
|
|
|
#include "controller.h"
|
|
|
|
|
2018-08-18 08:02:47 +00:00
|
|
|
#include "matriqueroom.h"
|
2018-09-10 01:51:02 +00:00
|
|
|
#include "matriqueuser.h"
|
2018-09-09 02:12:45 +00:00
|
|
|
#include "settings.h"
|
2018-08-18 08:02:47 +00:00
|
|
|
|
2018-07-15 08:02:26 +00:00
|
|
|
#include "events/eventcontent.h"
|
|
|
|
#include "events/roommessageevent.h"
|
2018-02-27 05:10:08 +00:00
|
|
|
|
2018-08-01 12:26:29 +00:00
|
|
|
#include "csapi/joining.h"
|
|
|
|
|
2018-08-04 20:35:31 +00:00
|
|
|
#include <QClipboard>
|
2018-09-09 02:12:45 +00:00
|
|
|
#include <QFile>
|
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
#include <QtCore/QDir>
|
|
|
|
#include <QtCore/QElapsedTimer>
|
|
|
|
#include <QtCore/QFileInfo>
|
|
|
|
#include <QtCore/QStandardPaths>
|
|
|
|
#include <QtCore/QStringBuilder>
|
|
|
|
#include <QtCore/QTimer>
|
|
|
|
#include <QtGui/QCloseEvent>
|
|
|
|
#include <QtGui/QDesktopServices>
|
|
|
|
#include <QtGui/QMovie>
|
|
|
|
#include <QtGui/QPixmap>
|
|
|
|
#include <QtNetwork/QAuthenticator>
|
|
|
|
#include <QtNetwork/QNetworkReply>
|
2018-07-17 08:18:50 +00:00
|
|
|
|
2018-07-15 08:02:26 +00:00
|
|
|
Controller::Controller(QObject* parent) : QObject(parent) {
|
2018-08-19 06:32:18 +00:00
|
|
|
tray->setIcon(QIcon(":/asset/img/icon.png"));
|
|
|
|
tray->setToolTip("Matrique");
|
2018-09-06 11:34:14 +00:00
|
|
|
connect(tray, &QSystemTrayIcon::activated,
|
|
|
|
[this](QSystemTrayIcon::ActivationReason r) {
|
2018-09-14 04:16:25 +00:00
|
|
|
if (r != QSystemTrayIcon::Context) emit showWindow();
|
2018-09-06 11:34:14 +00:00
|
|
|
});
|
2018-09-14 04:16:25 +00:00
|
|
|
connect(tray, &QSystemTrayIcon::messageClicked, [=] { emit showWindow(); });
|
|
|
|
trayMenu->addAction("Hide Window", [=] { emit hideWindow(); });
|
2018-08-19 06:32:18 +00:00
|
|
|
trayMenu->addAction("Quit", [=] { QApplication::quit(); });
|
|
|
|
tray->setContextMenu(trayMenu);
|
|
|
|
tray->show();
|
|
|
|
|
2018-08-18 08:02:47 +00:00
|
|
|
Connection::setRoomType<MatriqueRoom>();
|
2018-09-10 01:51:02 +00:00
|
|
|
Connection::setUserType<MatriqueUser>();
|
2018-08-18 08:02:47 +00:00
|
|
|
|
2018-09-09 13:13:43 +00:00
|
|
|
QTimer::singleShot(0, this, SLOT(invokeLogin()));
|
2018-02-27 11:07:50 +00:00
|
|
|
}
|
|
|
|
|
2018-09-09 13:13:43 +00:00
|
|
|
Controller::~Controller() {}
|
|
|
|
|
|
|
|
inline QString accessTokenFileName(const AccountSettings& account) {
|
|
|
|
QString fileName = account.userId();
|
|
|
|
fileName.replace(':', '_');
|
|
|
|
return QStandardPaths::writableLocation(
|
|
|
|
QStandardPaths::AppLocalDataLocation) +
|
|
|
|
'/' + fileName;
|
2018-07-17 08:18:50 +00:00
|
|
|
}
|
2018-02-27 11:07:50 +00:00
|
|
|
|
2018-09-09 02:12:45 +00:00
|
|
|
void Controller::loginWithCredentials(QString serverAddr, QString user,
|
|
|
|
QString pass) {
|
|
|
|
if (!user.isEmpty() && !pass.isEmpty()) {
|
|
|
|
Connection* m_connection = new Connection(this);
|
|
|
|
m_connection->setHomeserver(QUrl(serverAddr));
|
|
|
|
m_connection->connectToServer(user, pass, "");
|
|
|
|
connect(m_connection, &Connection::connected, [=] {
|
|
|
|
AccountSettings account(m_connection->userId());
|
|
|
|
account.setKeepLoggedIn(true);
|
|
|
|
account.clearAccessToken(); // Drop the legacy - just in case
|
|
|
|
account.setHomeserver(m_connection->homeserver());
|
|
|
|
account.setDeviceId(m_connection->deviceId());
|
|
|
|
account.setDeviceName("Matrique");
|
|
|
|
if (!saveAccessToken(account, m_connection->accessToken()))
|
|
|
|
qWarning() << "Couldn't save access token";
|
|
|
|
account.sync();
|
|
|
|
addConnection(m_connection);
|
|
|
|
});
|
2018-07-09 02:45:26 +00:00
|
|
|
}
|
2018-07-07 09:38:20 +00:00
|
|
|
}
|
|
|
|
|
2018-09-09 13:13:43 +00:00
|
|
|
void Controller::logout(Connection* conn) {
|
|
|
|
if (!conn) {
|
|
|
|
qCritical() << "Attempt to logout null connection";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SettingsGroup("Accounts").remove(conn->userId());
|
|
|
|
QFile(accessTokenFileName(AccountSettings(conn->userId()))).remove();
|
|
|
|
|
|
|
|
conn->logout();
|
|
|
|
}
|
|
|
|
|
2018-09-09 02:12:45 +00:00
|
|
|
void Controller::addConnection(Connection* c) {
|
|
|
|
Q_ASSERT_X(c, __FUNCTION__, "Attempt to add a null connection");
|
|
|
|
|
|
|
|
m_connections.push_back(c);
|
|
|
|
|
|
|
|
connect(c, &Connection::syncDone, this, [=] {
|
2018-09-10 00:06:32 +00:00
|
|
|
c->saveState();
|
|
|
|
c->sync(30000);
|
2018-09-09 02:12:45 +00:00
|
|
|
});
|
|
|
|
connect(c, &Connection::loggedOut, this, [=] { dropConnection(c); });
|
|
|
|
|
|
|
|
using namespace QMatrixClient;
|
|
|
|
|
|
|
|
c->sync(30000);
|
|
|
|
|
|
|
|
emit connectionAdded(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Controller::dropConnection(Connection* c) {
|
|
|
|
Q_ASSERT_X(c, __FUNCTION__, "Attempt to drop a null connection");
|
|
|
|
m_connections.removeOne(c);
|
|
|
|
|
2018-09-09 13:13:43 +00:00
|
|
|
emit connectionDropped(c);
|
2018-09-09 02:12:45 +00:00
|
|
|
c->deleteLater();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Controller::invokeLogin() {
|
|
|
|
using namespace QMatrixClient;
|
|
|
|
const auto accounts = SettingsGroup("Accounts").childGroups();
|
|
|
|
for (const auto& accountId : accounts) {
|
|
|
|
AccountSettings account{accountId};
|
|
|
|
if (!account.homeserver().isEmpty()) {
|
|
|
|
auto accessToken = loadAccessToken(account);
|
|
|
|
if (accessToken.isEmpty()) {
|
|
|
|
// Try to look in the legacy location (QSettings) and if found,
|
|
|
|
// migrate it from there to a file.
|
|
|
|
accessToken = account.accessToken().toLatin1();
|
|
|
|
if (accessToken.isEmpty())
|
|
|
|
continue; // No access token anywhere, no autologin
|
|
|
|
|
|
|
|
saveAccessToken(account, accessToken);
|
|
|
|
account.clearAccessToken(); // Clean the old place
|
|
|
|
}
|
|
|
|
|
|
|
|
auto c = new Connection(account.homeserver(), this);
|
|
|
|
auto deviceName = account.deviceName();
|
|
|
|
connect(c, &Connection::connected, this, [=] {
|
|
|
|
c->loadState();
|
|
|
|
addConnection(c);
|
|
|
|
});
|
|
|
|
c->connectWithToken(account.userId(), accessToken, account.deviceId());
|
2018-02-27 11:07:50 +00:00
|
|
|
}
|
2018-07-09 02:45:26 +00:00
|
|
|
}
|
2018-09-13 00:22:41 +00:00
|
|
|
emit initiated();
|
2018-02-27 11:07:50 +00:00
|
|
|
}
|
|
|
|
|
2018-09-09 02:12:45 +00:00
|
|
|
QByteArray Controller::loadAccessToken(const AccountSettings& account) {
|
|
|
|
QFile accountTokenFile{accessTokenFileName(account)};
|
|
|
|
if (accountTokenFile.open(QFile::ReadOnly)) {
|
|
|
|
if (accountTokenFile.size() < 1024) return accountTokenFile.readAll();
|
|
|
|
|
|
|
|
qWarning() << "File" << accountTokenFile.fileName() << "is"
|
|
|
|
<< accountTokenFile.size()
|
|
|
|
<< "bytes long - too long for a token, ignoring it.";
|
|
|
|
}
|
|
|
|
qWarning() << "Could not open access token file"
|
|
|
|
<< accountTokenFile.fileName();
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Controller::saveAccessToken(const AccountSettings& account,
|
|
|
|
const QByteArray& accessToken) {
|
|
|
|
// (Re-)Make a dedicated file for access_token.
|
|
|
|
QFile accountTokenFile{accessTokenFileName(account)};
|
|
|
|
accountTokenFile.remove(); // Just in case
|
|
|
|
|
|
|
|
auto fileDir = QFileInfo(accountTokenFile).dir();
|
|
|
|
if (!((fileDir.exists() || fileDir.mkpath(".")) &&
|
|
|
|
accountTokenFile.open(QFile::WriteOnly))) {
|
2018-09-14 04:16:25 +00:00
|
|
|
emit errorOccured("Cannot save access token.");
|
2018-09-09 02:12:45 +00:00
|
|
|
} else {
|
|
|
|
// Try to restrict access rights to the file. The below is useless
|
|
|
|
// on Windows: FAT doesn't control access at all and NTFS is
|
|
|
|
// incompatible with the UNIX perms model used by Qt. If the attempt
|
|
|
|
// didn't have the effect, at least ask the user if it's fine to save
|
|
|
|
// the token to a file readable by others.
|
|
|
|
// TODO: use system-specific API to ensure proper access.
|
|
|
|
if ((accountTokenFile.setPermissions(QFile::ReadOwner |
|
|
|
|
QFile::WriteOwner) &&
|
|
|
|
!(accountTokenFile.permissions() &
|
|
|
|
(QFile::ReadGroup | QFile::ReadOther)))) {
|
|
|
|
accountTokenFile.write(accessToken);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2018-02-27 11:37:53 +00:00
|
|
|
}
|
|
|
|
|
2018-09-09 13:13:43 +00:00
|
|
|
void Controller::joinRoom(Connection* c, const QString& alias) {
|
|
|
|
JoinRoomJob* joinRoomJob = c->joinRoom(alias);
|
|
|
|
setBusy(true);
|
|
|
|
joinRoomJob->connect(joinRoomJob, &JoinRoomJob::finished,
|
|
|
|
[=] { setBusy(false); });
|
2018-08-01 12:26:29 +00:00
|
|
|
}
|
|
|
|
|
2018-09-09 13:13:43 +00:00
|
|
|
void Controller::createRoom(Connection* c, const QString& name,
|
|
|
|
const QString& topic) {
|
|
|
|
CreateRoomJob* createRoomJob =
|
|
|
|
c->createRoom(Connection::PublishRoom, "", name, topic, QStringList());
|
|
|
|
setBusy(true);
|
|
|
|
createRoomJob->connect(createRoomJob, &CreateRoomJob::finished,
|
|
|
|
[=] { setBusy(false); });
|
2018-08-01 12:26:29 +00:00
|
|
|
}
|
2018-08-04 20:35:31 +00:00
|
|
|
|
|
|
|
void Controller::copyToClipboard(const QString& text) {
|
|
|
|
m_clipboard->setText(text);
|
|
|
|
}
|
2018-08-14 06:05:41 +00:00
|
|
|
|
2018-08-18 08:02:47 +00:00
|
|
|
void Controller::playAudio(QUrl localFile) {
|
|
|
|
QMediaPlayer* player = new QMediaPlayer;
|
|
|
|
player->setMedia(localFile);
|
|
|
|
player->play();
|
|
|
|
connect(player, &QMediaPlayer::stateChanged, [=] { player->deleteLater(); });
|
2018-08-17 04:55:57 +00:00
|
|
|
}
|
2018-08-19 06:32:18 +00:00
|
|
|
|
|
|
|
void Controller::showMessage(const QString& title, const QString& msg,
|
|
|
|
const QIcon& icon) {
|
|
|
|
tray->showMessage(title, msg, icon);
|
|
|
|
}
|
2018-09-10 01:51:02 +00:00
|
|
|
|
|
|
|
QImage Controller::safeImage(QImage image) {
|
|
|
|
if (image.isNull()) return QImage();
|
|
|
|
return image;
|
|
|
|
}
|
2018-09-10 07:01:01 +00:00
|
|
|
|
|
|
|
QColor Controller::color(QString userId) {
|
|
|
|
return QColor(SettingsGroup("UI/Color").value(userId, "#498882").toString());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Controller::setColor(QString userId, QColor newColor) {
|
|
|
|
SettingsGroup("UI/Color").setValue(userId, newColor.name());
|
|
|
|
}
|