init
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
This commit is contained in:
410
Telegram/SourceFiles/intro/intro_password_check.cpp
Normal file
410
Telegram/SourceFiles/intro/intro_password_check.cpp
Normal file
@@ -0,0 +1,410 @@
|
||||
/*
|
||||
This file is part of Telegram Desktop,
|
||||
the official desktop application for the Telegram messaging service.
|
||||
|
||||
For license and copyright information please follow this link:
|
||||
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||
*/
|
||||
#include "intro/intro_password_check.h"
|
||||
|
||||
#include "intro/intro_widget.h"
|
||||
#include "core/core_cloud_password.h"
|
||||
#include "ui/boxes/confirm_box.h"
|
||||
#include "boxes/abstract_box.h"
|
||||
#include "boxes/passcode_box.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "intro/intro_signup.h"
|
||||
#include "ui/text/text_utilities.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/widgets/fields/input_field.h"
|
||||
#include "ui/widgets/fields/password_input.h"
|
||||
#include "main/main_account.h"
|
||||
#include "base/random.h"
|
||||
#include "styles/style_intro.h"
|
||||
#include "styles/style_boxes.h"
|
||||
|
||||
namespace Intro {
|
||||
namespace details {
|
||||
|
||||
PasswordCheckWidget::PasswordCheckWidget(
|
||||
QWidget *parent,
|
||||
not_null<Main::Account*> account,
|
||||
not_null<Data*> data)
|
||||
: Step(parent, account, data)
|
||||
, _passwordState(getData()->pwdState)
|
||||
, _pwdField(this, st::introPassword, tr::lng_signin_password())
|
||||
, _pwdHint(this, st::introPasswordHint)
|
||||
, _codeField(this, st::introPassword, tr::lng_signin_code())
|
||||
, _toRecover(this, tr::lng_signin_recover(tr::now))
|
||||
, _toPassword(this, tr::lng_signin_try_password(tr::now)) {
|
||||
Expects(_passwordState.hasPassword);
|
||||
|
||||
Lang::Updated(
|
||||
) | rpl::on_next([=] {
|
||||
refreshLang();
|
||||
}, lifetime());
|
||||
|
||||
_toRecover->addClickHandler([=] { toRecover(); });
|
||||
_toPassword->addClickHandler([=] { toPassword(); });
|
||||
connect(_pwdField, &Ui::PasswordInput::changed, [=] { hideError(); });
|
||||
_codeField->changes(
|
||||
) | rpl::on_next([=] {
|
||||
hideError();
|
||||
}, _codeField->lifetime());
|
||||
|
||||
setTitleText(tr::lng_signin_title());
|
||||
updateDescriptionText();
|
||||
|
||||
if (_passwordState.hint.isEmpty()) {
|
||||
_pwdHint->hide();
|
||||
} else {
|
||||
_pwdHint->setText(tr::lng_signin_hint(
|
||||
tr::now,
|
||||
lt_password_hint,
|
||||
_passwordState.hint));
|
||||
}
|
||||
_codeField->hide();
|
||||
_toPassword->hide();
|
||||
|
||||
setMouseTracking(true);
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::refreshLang() {
|
||||
if (_toRecover) {
|
||||
_toRecover->setText(tr::lng_signin_recover(tr::now));
|
||||
}
|
||||
if (_toPassword) {
|
||||
_toPassword->setText(
|
||||
tr::lng_signin_try_password(tr::now));
|
||||
}
|
||||
if (!_passwordState.hint.isEmpty()) {
|
||||
_pwdHint->setText(tr::lng_signin_hint(
|
||||
tr::now,
|
||||
lt_password_hint,
|
||||
_passwordState.hint));
|
||||
}
|
||||
updateControlsGeometry();
|
||||
}
|
||||
|
||||
int PasswordCheckWidget::errorTop() const {
|
||||
return contentTop() + st::introErrorBelowLinkTop;
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::resizeEvent(QResizeEvent *e) {
|
||||
Step::resizeEvent(e);
|
||||
updateControlsGeometry();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::updateControlsGeometry() {
|
||||
_pwdField->moveToLeft(contentLeft(), contentTop() + st::introPasswordTop);
|
||||
_pwdHint->moveToLeft(contentLeft() + st::buttonRadius, contentTop() + st::introPasswordHintTop);
|
||||
_codeField->moveToLeft(contentLeft(), contentTop() + st::introStepFieldTop);
|
||||
auto linkTop = _codeField->y() + _codeField->height() + st::introLinkTop;
|
||||
_toRecover->moveToLeft(contentLeft() + st::buttonRadius, linkTop);
|
||||
_toPassword->moveToLeft(contentLeft() + st::buttonRadius, linkTop);
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::setInnerFocus() {
|
||||
if (_pwdField->isHidden()) {
|
||||
_codeField->setFocusFast();
|
||||
} else {
|
||||
_pwdField->setFocusFast();
|
||||
}
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::activate() {
|
||||
if (_pwdField->isHidden() && _codeField->isHidden()) {
|
||||
Step::activate();
|
||||
_pwdField->show();
|
||||
_pwdHint->show();
|
||||
_toRecover->show();
|
||||
}
|
||||
setInnerFocus();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::cancelled() {
|
||||
api().request(base::take(_sentRequest)).cancel();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::pwdSubmitDone(
|
||||
bool recover,
|
||||
const MTPauth_Authorization &result) {
|
||||
_sentRequest = 0;
|
||||
if (recover) {
|
||||
cSetPasswordRecovered(true);
|
||||
}
|
||||
finish(result);
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::pwdSubmitFail(const MTP::Error &error) {
|
||||
if (MTP::IsFloodError(error)) {
|
||||
_sentRequest = 0;
|
||||
showError(tr::lng_flood_error());
|
||||
_pwdField->showError();
|
||||
return;
|
||||
}
|
||||
|
||||
_sentRequest = 0;
|
||||
const auto &type = error.type();
|
||||
if (type == u"PASSWORD_HASH_INVALID"_q
|
||||
|| type == u"SRP_PASSWORD_CHANGED"_q) {
|
||||
showError(tr::lng_signin_bad_password());
|
||||
_pwdField->selectAll();
|
||||
_pwdField->showError();
|
||||
} else if (type == u"PASSWORD_EMPTY"_q
|
||||
|| type == u"AUTH_KEY_UNREGISTERED"_q) {
|
||||
goBack();
|
||||
} else if (type == u"SRP_ID_INVALID"_q) {
|
||||
handleSrpIdInvalid();
|
||||
} else {
|
||||
if (Logs::DebugEnabled()) { // internal server error
|
||||
showError(rpl::single(type + ": " + error.description()));
|
||||
} else {
|
||||
showError(rpl::single(Lang::Hard::ServerError()));
|
||||
}
|
||||
_pwdField->setFocus();
|
||||
}
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::handleSrpIdInvalid() {
|
||||
const auto now = crl::now();
|
||||
if (_lastSrpIdInvalidTime > 0
|
||||
&& now - _lastSrpIdInvalidTime < Core::kHandleSrpIdInvalidTimeout) {
|
||||
_passwordState.mtp.request.id = 0;
|
||||
showError(rpl::single(Lang::Hard::ServerError()));
|
||||
} else {
|
||||
_lastSrpIdInvalidTime = now;
|
||||
requestPasswordData();
|
||||
}
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::checkPasswordHash() {
|
||||
if (_passwordState.mtp.request.id) {
|
||||
passwordChecked();
|
||||
} else {
|
||||
requestPasswordData();
|
||||
}
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::requestPasswordData() {
|
||||
api().request(base::take(_sentRequest)).cancel();
|
||||
_sentRequest = api().request(
|
||||
MTPaccount_GetPassword()
|
||||
).done([=](const MTPaccount_Password &result) {
|
||||
_sentRequest = 0;
|
||||
result.match([&](const MTPDaccount_password &data) {
|
||||
base::RandomAddSeed(bytes::make_span(data.vsecure_random().v));
|
||||
_passwordState = Core::ParseCloudPasswordState(data);
|
||||
passwordChecked();
|
||||
});
|
||||
}).send();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::passwordChecked() {
|
||||
const auto check = Core::ComputeCloudPasswordCheck(
|
||||
_passwordState.mtp.request,
|
||||
_passwordHash);
|
||||
if (!check) {
|
||||
return serverError();
|
||||
}
|
||||
_passwordState.mtp.request.id = 0;
|
||||
_sentRequest = api().request(
|
||||
MTPauth_CheckPassword(check.result)
|
||||
).done([=](const MTPauth_Authorization &result) {
|
||||
pwdSubmitDone(false, result);
|
||||
}).fail([=](const MTP::Error &error) {
|
||||
pwdSubmitFail(error);
|
||||
}).handleFloodErrors().send();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::serverError() {
|
||||
showError(rpl::single(Lang::Hard::ServerError()));
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::codeSubmitDone(
|
||||
const QString &code,
|
||||
const MTPBool &result) {
|
||||
auto fields = PasscodeBox::CloudFields::From(_passwordState);
|
||||
fields.fromRecoveryCode = code;
|
||||
fields.hasRecovery = false;
|
||||
fields.mtp.curRequest = {};
|
||||
fields.hasPassword = false;
|
||||
auto box = Box<PasscodeBox>(&api().instance(), nullptr, fields);
|
||||
const auto boxShared = std::make_shared<base::weak_qptr<PasscodeBox>>();
|
||||
|
||||
box->newAuthorization(
|
||||
) | rpl::on_next([=](const MTPauth_Authorization &result) {
|
||||
if (boxShared) {
|
||||
(*boxShared)->closeBox();
|
||||
}
|
||||
pwdSubmitDone(true, result);
|
||||
}, lifetime());
|
||||
|
||||
*boxShared = Ui::show(std::move(box));
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::codeSubmitFail(const MTP::Error &error) {
|
||||
if (MTP::IsFloodError(error)) {
|
||||
showError(tr::lng_flood_error());
|
||||
_codeField->showError();
|
||||
return;
|
||||
}
|
||||
|
||||
_sentRequest = 0;
|
||||
const auto &type = error.type();
|
||||
if (type == u"PASSWORD_EMPTY"_q
|
||||
|| type == u"AUTH_KEY_UNREGISTERED"_q) {
|
||||
goBack();
|
||||
} else if (type == u"PASSWORD_RECOVERY_NA"_q) {
|
||||
recoverStartFail(error);
|
||||
} else if (type == u"PASSWORD_RECOVERY_EXPIRED"_q) {
|
||||
_emailPattern = QString();
|
||||
toPassword();
|
||||
} else if (type == u"CODE_INVALID"_q) {
|
||||
showError(tr::lng_signin_wrong_code());
|
||||
_codeField->selectAll();
|
||||
_codeField->showError();
|
||||
} else {
|
||||
if (Logs::DebugEnabled()) { // internal server error
|
||||
showError(rpl::single(type + ": " + error.description()));
|
||||
} else {
|
||||
showError(rpl::single(Lang::Hard::ServerError()));
|
||||
}
|
||||
_codeField->setFocus();
|
||||
}
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::recoverStarted(const MTPauth_PasswordRecovery &result) {
|
||||
_emailPattern = qs(result.c_auth_passwordRecovery().vemail_pattern());
|
||||
updateDescriptionText();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::recoverStartFail(const MTP::Error &error) {
|
||||
_pwdField->show();
|
||||
_pwdHint->show();
|
||||
_codeField->hide();
|
||||
_pwdField->setFocus();
|
||||
updateDescriptionText();
|
||||
update();
|
||||
hideError();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::toRecover() {
|
||||
if (_passwordState.hasRecovery) {
|
||||
if (_sentRequest) {
|
||||
api().request(base::take(_sentRequest)).cancel();
|
||||
}
|
||||
hideError();
|
||||
_toRecover->hide();
|
||||
_toPassword->show();
|
||||
_pwdField->hide();
|
||||
_pwdHint->hide();
|
||||
_pwdField->setText(QString());
|
||||
_codeField->show();
|
||||
_codeField->setFocus();
|
||||
updateDescriptionText();
|
||||
if (_emailPattern.isEmpty()) {
|
||||
api().request(
|
||||
MTPauth_RequestPasswordRecovery()
|
||||
).done([=](const MTPauth_PasswordRecovery &result) {
|
||||
recoverStarted(result);
|
||||
}).fail([=](const MTP::Error &error) {
|
||||
recoverStartFail(error);
|
||||
}).send();
|
||||
}
|
||||
} else {
|
||||
const auto box = Ui::show(
|
||||
Ui::MakeInformBox(tr::lng_signin_no_email_forgot()));
|
||||
box->boxClosing(
|
||||
) | rpl::on_next([=] {
|
||||
showReset();
|
||||
}, box->lifetime());
|
||||
}
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::toPassword() {
|
||||
const auto box = Ui::show(
|
||||
Ui::MakeInformBox(tr::lng_signin_cant_email_forgot()));
|
||||
box->boxClosing(
|
||||
) | rpl::on_next([=] {
|
||||
showReset();
|
||||
}, box->lifetime());
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::showReset() {
|
||||
if (_sentRequest) {
|
||||
api().request(base::take(_sentRequest)).cancel();
|
||||
}
|
||||
_toRecover->show();
|
||||
_toPassword->hide();
|
||||
_pwdField->show();
|
||||
_pwdHint->show();
|
||||
_codeField->hide();
|
||||
_codeField->setText(QString());
|
||||
_pwdField->setFocus();
|
||||
showResetButton();
|
||||
updateDescriptionText();
|
||||
update();
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::updateDescriptionText() {
|
||||
auto pwdHidden = _pwdField->isHidden();
|
||||
auto emailPattern = _emailPattern;
|
||||
setDescriptionText(pwdHidden
|
||||
? tr::lng_signin_recover_desc(
|
||||
lt_email,
|
||||
rpl::single(Ui::Text::WrapEmailPattern(emailPattern)),
|
||||
tr::marked)
|
||||
: tr::lng_signin_desc(tr::marked));
|
||||
}
|
||||
|
||||
void PasswordCheckWidget::submit() {
|
||||
if (_sentRequest) {
|
||||
return;
|
||||
}
|
||||
if (_pwdField->isHidden()) {
|
||||
auto code = _codeField->getLastText().trimmed();
|
||||
if (code.isEmpty()) {
|
||||
_codeField->showError();
|
||||
return;
|
||||
}
|
||||
const auto send = crl::guard(this, [=] {
|
||||
_sentRequest = api().request(MTPauth_CheckRecoveryPassword(
|
||||
MTP_string(code)
|
||||
)).done([=](const MTPBool &result) {
|
||||
codeSubmitDone(code, result);
|
||||
}).fail([=](const MTP::Error &error) {
|
||||
codeSubmitFail(error);
|
||||
}).handleFloodErrors().send();
|
||||
});
|
||||
|
||||
if (_passwordState.notEmptyPassport) {
|
||||
const auto confirmed = [=](Fn<void()> &&close) {
|
||||
send();
|
||||
close();
|
||||
};
|
||||
Ui::show(Ui::MakeConfirmBox({
|
||||
.text = tr::lng_cloud_password_passport_losing(),
|
||||
.confirmed = confirmed,
|
||||
.confirmText = tr::lng_continue(),
|
||||
}));
|
||||
} else {
|
||||
send();
|
||||
}
|
||||
} else {
|
||||
hideError();
|
||||
|
||||
const auto password = _pwdField->getLastText().toUtf8();
|
||||
_passwordHash = Core::ComputeCloudPasswordHash(
|
||||
_passwordState.mtp.request.algo,
|
||||
bytes::make_span(password));
|
||||
checkPasswordHash();
|
||||
}
|
||||
}
|
||||
|
||||
rpl::producer<QString> PasswordCheckWidget::nextButtonText() const {
|
||||
return tr::lng_intro_submit();
|
||||
}
|
||||
|
||||
} // namespace details
|
||||
} // namespace Intro
|
||||
Reference in New Issue
Block a user