matrixion/src/LoginPage.cpp

493 lines
18 KiB
C++
Raw Normal View History

2017-04-06 02:06:42 +03:00
/*
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
2020-05-10 00:31:00 +03:00
#include <QDesktopServices>
2020-05-25 14:03:49 +03:00
#include <QLabel>
2020-01-31 08:12:02 +03:00
#include <QPainter>
#include <QStyleOption>
2018-02-28 22:14:41 +03:00
#include <mtx/identifiers.hpp>
2020-05-10 00:31:00 +03:00
#include <mtx/requests.hpp>
2020-01-31 18:08:30 +03:00
#include <mtx/responses/login.hpp>
2018-02-28 22:14:41 +03:00
#include "Config.h"
#include "Logging.h"
2017-11-22 22:13:22 +03:00
#include "LoginPage.h"
2017-10-28 15:46:39 +03:00
#include "MatrixClient.h"
2020-05-10 00:31:00 +03:00
#include "SSOHandler.h"
2018-07-17 16:37:25 +03:00
#include "ui/FlatButton.h"
#include "ui/LoadingIndicator.h"
#include "ui/OverlayModal.h"
#include "ui/RaisedButton.h"
#include "ui/TextField.h"
2017-04-06 02:06:42 +03:00
2020-05-10 00:31:00 +03:00
Q_DECLARE_METATYPE(LoginPage::LoginMethod)
2018-02-28 22:14:41 +03:00
using namespace mtx::identifiers;
LoginPage::LoginPage(QWidget *parent)
2017-08-20 13:47:22 +03:00
: QWidget(parent)
, inferredServerAddress_()
2017-04-06 02:06:42 +03:00
{
2020-05-10 00:31:00 +03:00
qRegisterMetaType<LoginPage::LoginMethod>("LoginPage::LoginMethod");
2017-09-10 12:58:00 +03:00
top_layout_ = new QVBoxLayout();
top_bar_layout_ = new QHBoxLayout();
top_bar_layout_->setSpacing(0);
top_bar_layout_->setMargin(0);
back_button_ = new FlatButton(this);
back_button_->setMinimumSize(QSize(30, 30));
top_bar_layout_->addWidget(back_button_, 0, Qt::AlignLeft | Qt::AlignVCenter);
top_bar_layout_->addStretch(1);
QIcon icon;
2017-10-15 22:08:51 +03:00
icon.addFile(":/icons/icons/ui/angle-pointing-to-left.png");
2017-09-10 12:58:00 +03:00
back_button_->setIcon(icon);
2017-10-15 22:08:51 +03:00
back_button_->setIconSize(QSize(32, 32));
2017-09-10 12:58:00 +03:00
2017-10-15 22:08:51 +03:00
QIcon logo;
logo.addFile(":/logos/login.png");
2017-09-10 12:58:00 +03:00
logo_ = new QLabel(this);
2017-10-15 22:08:51 +03:00
logo_->setPixmap(logo.pixmap(128));
2017-09-10 12:58:00 +03:00
logo_layout_ = new QHBoxLayout();
logo_layout_->setContentsMargins(0, 0, 0, 20);
logo_layout_->addWidget(logo_, 0, Qt::AlignHCenter);
form_wrapper_ = new QHBoxLayout();
form_widget_ = new QWidget();
form_widget_->setMinimumSize(QSize(350, 200));
form_layout_ = new QVBoxLayout();
form_layout_->setSpacing(20);
form_layout_->setContentsMargins(0, 0, 0, 30);
form_widget_->setLayout(form_layout_);
form_wrapper_->addStretch(1);
form_wrapper_->addWidget(form_widget_);
form_wrapper_->addStretch(1);
matrixid_input_ = new TextField(this);
matrixid_input_->setLabel(tr("Matrix ID"));
matrixid_input_->setPlaceholderText(tr("e.g @joe:matrix.org"));
matrixid_input_->setToolTip(
tr("Your login name. A mxid should start with @ followed by the user id. After the user "
"id you need to include your server name after a :.\nYou can also put your homeserver "
"address there, if your server doesn't support .well-known lookup.\nExample: "
"@user:server.my\nIf Nheko fails to discover your homeserver, it will show you a "
"field to enter the server manually."));
2017-09-10 12:58:00 +03:00
spinner_ = new LoadingIndicator(this);
spinner_->setFixedHeight(40);
spinner_->setFixedWidth(40);
spinner_->hide();
errorIcon_ = new QLabel(this);
errorIcon_->setPixmap(QPixmap(":/icons/icons/error.png"));
errorIcon_->hide();
matrixidLayout_ = new QHBoxLayout();
matrixidLayout_->addWidget(matrixid_input_, 0, Qt::AlignVCenter);
QFont font;
error_matrixid_label_ = new QLabel(this);
error_matrixid_label_->setFont(font);
matrixid_error_layout_ = new QVBoxLayout();
matrixid_error_layout_->addWidget(error_matrixid_label_, 0, Qt::AlignHCenter);
2017-09-10 12:58:00 +03:00
password_input_ = new TextField(this);
password_input_->setLabel(tr("Password"));
password_input_->setEchoMode(QLineEdit::Password);
password_input_->setToolTip("Your password.");
2017-09-10 12:58:00 +03:00
deviceName_ = new TextField(this);
deviceName_->setLabel(tr("Device name"));
deviceName_->setToolTip(
tr("A name for this device, which will be shown to others, when verifying your devices. "
2020-05-25 14:03:49 +03:00
"If none is provided a default is used."));
2017-09-10 12:58:00 +03:00
serverInput_ = new TextField(this);
serverInput_->setLabel("Homeserver address");
serverInput_->setPlaceholderText("matrix.org");
serverInput_->setToolTip(tr("The address that can be used to contact you homeservers "
"client API.\nExample: https://server.my:8787"));
2017-09-10 12:58:00 +03:00
serverInput_->hide();
serverLayout_ = new QHBoxLayout();
serverLayout_->addWidget(serverInput_, 0, Qt::AlignVCenter);
form_layout_->addLayout(matrixidLayout_);
form_layout_->addLayout(matrixid_error_layout_);
form_layout_->addWidget(password_input_);
2020-06-06 00:34:00 +03:00
form_layout_->addWidget(deviceName_, Qt::AlignHCenter);
2017-09-10 12:58:00 +03:00
form_layout_->addLayout(serverLayout_);
2020-11-24 00:10:43 +03:00
error_matrixid_label_->hide();
2017-09-10 12:58:00 +03:00
button_layout_ = new QHBoxLayout();
button_layout_->setSpacing(0);
button_layout_->setContentsMargins(0, 0, 0, 30);
login_button_ = new RaisedButton(tr("LOGIN"), this);
login_button_->setMinimumSize(350, 65);
login_button_->setFontSize(20);
login_button_->setCornerRadius(3);
button_layout_->addStretch(1);
button_layout_->addWidget(login_button_);
button_layout_->addStretch(1);
error_label_ = new QLabel(this);
error_label_->setFont(font);
error_label_->setWordWrap(true);
2017-09-10 12:58:00 +03:00
top_layout_->addLayout(top_bar_layout_);
top_layout_->addStretch(1);
top_layout_->addLayout(logo_layout_);
top_layout_->addLayout(form_wrapper_);
top_layout_->addStretch(1);
top_layout_->addLayout(button_layout_);
top_layout_->addWidget(error_label_, 0, Qt::AlignHCenter);
top_layout_->addStretch(1);
setLayout(top_layout_);
connect(this, &LoginPage::versionOkCb, this, &LoginPage::versionOk);
connect(this, &LoginPage::versionErrorCb, this, &LoginPage::versionError);
connect(this, &LoginPage::loginErrorCb, this, &LoginPage::loginError);
2017-09-10 12:58:00 +03:00
connect(back_button_, SIGNAL(clicked()), this, SLOT(onBackButtonClicked()));
connect(login_button_, SIGNAL(clicked()), this, SLOT(onLoginButtonClicked()));
connect(matrixid_input_, SIGNAL(returnPressed()), login_button_, SLOT(click()));
connect(password_input_, SIGNAL(returnPressed()), login_button_, SLOT(click()));
connect(deviceName_, SIGNAL(returnPressed()), login_button_, SLOT(click()));
2017-09-10 12:58:00 +03:00
connect(serverInput_, SIGNAL(returnPressed()), login_button_, SLOT(click()));
connect(matrixid_input_, SIGNAL(editingFinished()), this, SLOT(onMatrixIdEntered()));
connect(serverInput_, SIGNAL(editingFinished()), this, SLOT(onServerAddressEntered()));
2017-04-06 02:06:42 +03:00
}
2020-05-25 14:03:49 +03:00
void
LoginPage::loginError(const QString &msg)
{
error_label_->setText(msg);
}
void
LoginPage::matrixIdError(const QString &msg)
{
error_matrixid_label_->show();
error_matrixid_label_->setText(msg);
matrixid_input_->setValid(false);
}
bool
LoginPage::isMatrixIdValid()
{
QRegularExpressionValidator v(QRegularExpression("@.+?:.{3,}"), this);
QString s = matrixid_input_->text();
2020-11-23 03:20:37 +03:00
int pos = 0;
return v.validate(s, pos);
}
2017-08-20 13:47:22 +03:00
void
LoginPage::onMatrixIdEntered()
2017-04-06 02:06:42 +03:00
{
error_label_->setText("");
2017-04-06 02:06:42 +03:00
User user;
if (!isMatrixIdValid()) {
matrixIdError("You have entered an invalid Matrix ID e.g @joe:matrix.org");
2020-11-23 09:44:30 +03:00
return;
} else {
error_matrixid_label_->setText("");
error_matrixid_label_->hide();
matrixid_input_->setValid(true);
}
2018-02-28 22:14:41 +03:00
try {
user = parse<User>(matrixid_input_->text().toStdString());
2018-02-28 22:14:41 +03:00
} catch (const std::exception &e) {
return loginError("You have entered an invalid Matrix ID e.g @joe:matrix.org");
2018-02-28 22:14:41 +03:00
}
QString homeServer = QString::fromStdString(user.hostname());
2017-09-10 12:58:00 +03:00
if (homeServer != inferredServerAddress_) {
serverInput_->hide();
serverLayout_->removeWidget(errorIcon_);
errorIcon_->hide();
if (serverInput_->isVisible()) {
matrixidLayout_->removeWidget(spinner_);
serverLayout_->addWidget(spinner_, 0, Qt::AlignVCenter | Qt::AlignRight);
spinner_->start();
} else {
serverLayout_->removeWidget(spinner_);
matrixidLayout_->addWidget(spinner_, 0, Qt::AlignVCenter | Qt::AlignRight);
spinner_->start();
}
inferredServerAddress_ = homeServer;
serverInput_->setText(homeServer);
http::client()->set_server(user.hostname());
http::client()->well_known([this](const mtx::responses::WellKnown &res,
mtx::http::RequestErr err) {
if (err) {
using namespace boost::beast::http;
if (err->status_code == status::not_found) {
nhlog::net()->info("Autodiscovery: No .well-known.");
checkHomeserverVersion();
return;
}
if (!err->parse_error.empty()) {
emit versionErrorCb(
tr("Autodiscovery failed. Received malformed response."));
nhlog::net()->error(
"Autodiscovery failed. Received malformed response.");
return;
}
emit versionErrorCb(tr("Autodiscovery failed. Unknown error when "
"requesting .well-known."));
nhlog::net()->error("Autodiscovery failed. Unknown error when "
2020-05-10 00:31:00 +03:00
"requesting .well-known. {}",
err->error_code.message());
return;
}
nhlog::net()->info("Autodiscovery: Discovered '" + res.homeserver.base_url +
"'");
http::client()->set_server(res.homeserver.base_url);
checkHomeserverVersion();
});
2017-09-10 12:58:00 +03:00
}
2017-04-06 02:06:42 +03:00
}
void
LoginPage::checkHomeserverVersion()
{
http::client()->versions(
[this](const mtx::responses::Versions &, mtx::http::RequestErr err) {
if (err) {
using namespace boost::beast::http;
if (err->status_code == status::not_found) {
emit versionErrorCb(tr("The required endpoints were not found. "
"Possibly not a Matrix server."));
return;
}
if (!err->parse_error.empty()) {
emit versionErrorCb(tr("Received malformed response. Make sure "
"the homeserver domain is valid."));
return;
}
emit versionErrorCb(tr(
"An unknown error occured. Make sure the homeserver domain is valid."));
return;
}
2020-05-10 00:31:00 +03:00
http::client()->get_login(
[this](mtx::responses::LoginFlows flows, mtx::http::RequestErr err) {
if (err || flows.flows.empty())
emit versionOkCb(LoginMethod::Password);
if (flows.flows[0].type == mtx::user_interactive::auth_types::sso)
emit versionOkCb(LoginMethod::SSO);
else
emit versionOkCb(LoginMethod::Password);
});
});
}
2017-08-20 13:47:22 +03:00
void
LoginPage::onServerAddressEntered()
{
2017-09-10 12:58:00 +03:00
error_label_->setText("");
http::client()->set_server(serverInput_->text().toStdString());
checkHomeserverVersion();
2017-09-10 12:58:00 +03:00
serverLayout_->removeWidget(errorIcon_);
errorIcon_->hide();
serverLayout_->addWidget(spinner_, 0, Qt::AlignVCenter | Qt::AlignRight);
spinner_->start();
}
2017-08-20 13:47:22 +03:00
void
LoginPage::versionError(const QString &error)
{
2017-09-10 12:58:00 +03:00
error_label_->setText(error);
serverInput_->show();
spinner_->stop();
serverLayout_->removeWidget(spinner_);
serverLayout_->addWidget(errorIcon_, 0, Qt::AlignVCenter | Qt::AlignRight);
errorIcon_->show();
matrixidLayout_->removeWidget(spinner_);
}
2017-08-20 13:47:22 +03:00
void
2020-05-10 02:00:20 +03:00
LoginPage::versionOk(LoginMethod loginMethod_)
{
2020-05-10 02:00:20 +03:00
this->loginMethod = loginMethod_;
2020-05-10 00:31:00 +03:00
2017-09-10 12:58:00 +03:00
serverLayout_->removeWidget(spinner_);
matrixidLayout_->removeWidget(spinner_);
spinner_->stop();
2020-05-10 00:31:00 +03:00
if (loginMethod == LoginMethod::SSO) {
password_input_->hide();
login_button_->setText(tr("SSO LOGIN"));
} else {
password_input_->show();
login_button_->setText(tr("LOGIN"));
}
2017-09-10 12:58:00 +03:00
if (serverInput_->isVisible())
serverInput_->hide();
}
2017-08-20 13:47:22 +03:00
void
LoginPage::onLoginButtonClicked()
{
2017-09-10 12:58:00 +03:00
error_label_->setText("");
User user;
if (!isMatrixIdValid()) {
matrixIdError("You have entered an invalid Matrix ID e.g @joe:matrix.org");
2020-11-23 09:44:30 +03:00
return;
} else {
error_matrixid_label_->setText("");
error_matrixid_label_->hide();
matrixid_input_->setValid(true);
}
try {
user = parse<User>(matrixid_input_->text().toStdString());
} catch (const std::exception &e) {
return loginError("You have entered an invalid Matrix ID e.g @joe:matrix.org");
2017-09-10 12:58:00 +03:00
}
2020-05-10 00:31:00 +03:00
if (loginMethod == LoginMethod::Password) {
if (password_input_->text().isEmpty())
return loginError(tr("Empty password"));
http::client()->login(
user.localpart(),
password_input_->text().toStdString(),
deviceName_->text().trimmed().isEmpty() ? initialDeviceName()
: deviceName_->text().toStdString(),
[this](const mtx::responses::Login &res, mtx::http::RequestErr err) {
if (err) {
emit loginError(QString::fromStdString(err->matrix_error.error));
emit errorOccurred();
return;
}
2020-05-10 00:31:00 +03:00
if (res.well_known) {
http::client()->set_server(res.well_known->homeserver.base_url);
nhlog::net()->info("Login requested to user server: " +
res.well_known->homeserver.base_url);
}
2020-05-10 00:31:00 +03:00
emit loginOk(res);
});
} else {
auto sso = new SSOHandler();
connect(sso, &SSOHandler::ssoSuccess, this, [this, sso](std::string token) {
mtx::requests::Login req{};
2020-05-10 02:00:20 +03:00
req.token = token;
req.type = mtx::user_interactive::auth_types::token;
2020-05-10 00:31:00 +03:00
req.device_id = deviceName_->text().trimmed().isEmpty()
? initialDeviceName()
: deviceName_->text().toStdString();
http::client()->login(
req, [this](const mtx::responses::Login &res, mtx::http::RequestErr err) {
if (err) {
emit loginError(
QString::fromStdString(err->matrix_error.error));
emit errorOccurred();
return;
}
if (res.well_known) {
http::client()->set_server(
res.well_known->homeserver.base_url);
nhlog::net()->info("Login requested to user server: " +
res.well_known->homeserver.base_url);
}
emit loginOk(res);
});
sso->deleteLater();
});
connect(sso, &SSOHandler::ssoFailed, this, [this, sso]() {
emit loginError(tr("SSO login failed"));
emit errorOccurred();
sso->deleteLater();
});
2020-05-10 00:31:00 +03:00
QDesktopServices::openUrl(
QString::fromStdString(http::client()->login_sso_redirect(sso->url())));
}
emit loggingIn();
}
2017-08-20 13:47:22 +03:00
void
LoginPage::reset()
2017-04-09 02:32:48 +03:00
{
2017-09-10 12:58:00 +03:00
matrixid_input_->clear();
password_input_->clear();
2020-05-10 00:31:00 +03:00
password_input_->show();
2017-09-10 12:58:00 +03:00
serverInput_->clear();
2017-09-10 12:58:00 +03:00
spinner_->stop();
errorIcon_->hide();
serverLayout_->removeWidget(spinner_);
serverLayout_->removeWidget(errorIcon_);
matrixidLayout_->removeWidget(spinner_);
2017-09-10 12:58:00 +03:00
inferredServerAddress_.clear();
2017-04-09 02:32:48 +03:00
}
2017-08-20 13:47:22 +03:00
void
LoginPage::onBackButtonClicked()
2017-04-06 02:06:42 +03:00
{
2017-09-10 12:58:00 +03:00
emit backButtonClicked();
2017-04-06 02:06:42 +03:00
}
void
LoginPage::paintEvent(QPaintEvent *)
{
QStyleOption opt;
opt.init(this);
QPainter p(this);
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
}