2
0
mirror of https://github.com/telegramdesktop/tdesktop synced 2025-08-26 12:18:01 +00:00
tdesktop/Telegram/SourceFiles/main/main_accounts.cpp

323 lines
7.6 KiB
C++
Raw Normal View History

2020-06-15 20:25:02 +04:00
/*
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 "main/main_accounts.h"
#include "core/application.h"
#include "core/shortcuts.h"
#include "main/main_account.h"
#include "main/main_session.h"
#include "data/data_session.h"
#include "mtproto/mtproto_config.h"
#include "mtproto/mtproto_dc_options.h"
2020-06-15 20:25:02 +04:00
#include "storage/storage_accounts.h"
#include "storage/localstorage.h"
2020-06-16 18:19:23 +04:00
#include "facades.h"
2020-06-15 20:25:02 +04:00
namespace Main {
Accounts::Accounts(const QString &dataName)
: _dataName(dataName)
, _local(std::make_unique<Storage::Accounts>(this, dataName)) {
}
Accounts::~Accounts() = default;
bool Accounts::started() const {
return !_accounts.empty();
}
Storage::StartResult Accounts::start(const QByteArray &passcode) {
Expects(!started());
2020-06-16 13:40:43 +04:00
const auto result = _local->start(passcode);
2020-06-15 20:25:02 +04:00
if (result == Storage::StartResult::Success) {
activateAfterStarting();
2020-06-15 20:25:02 +04:00
if (Local::oldSettingsVersion() < AppVersion) {
Local::writeSettings();
}
} else {
Assert(!started());
}
return result;
}
void Accounts::finish() {
_activeIndex = -1;
_active = nullptr;
base::take(_accounts);
}
2020-06-16 13:40:43 +04:00
void Accounts::accountAddedInStorage(
int index,
std::unique_ptr<Account> account) {
Expects(account != nullptr);
Expects(!_accounts.contains(index));
if (_accounts.empty()) {
_activeIndex = index;
}
_accounts.emplace(index, std::move(account));
};
void Accounts::resetWithForgottenPasscode() {
if (_accounts.empty()) {
_local->startFromScratch();
activateAfterStarting();
} else {
for (const auto &[index, account] : _accounts) {
account->logOut();
}
}
}
void Accounts::activateAfterStarting() {
Expects(started());
for (const auto &[index, account] : _accounts) {
watchSession(account.get());
}
activate(_activeIndex);
2020-06-16 18:19:23 +04:00
removePasscodeIfEmpty();
2020-06-16 13:40:43 +04:00
}
2020-06-15 20:25:02 +04:00
const base::flat_map<int, std::unique_ptr<Account>> &Accounts::list() const {
return _accounts;
}
rpl::producer<Account*> Accounts::activeValue() const {
return _active.value();
}
2020-06-16 10:42:47 +04:00
int Accounts::activeIndex() const {
Expects(_accounts.contains(_activeIndex));
return _activeIndex;
}
2020-06-15 20:25:02 +04:00
Account &Accounts::active() const {
Expects(!_accounts.empty());
Ensures(_active.current() != nullptr);
return *_active.current();
}
rpl::producer<not_null<Account*>> Accounts::activeChanges() const {
return _active.changes() | rpl::map([](Account *value) {
return not_null{ value };
});
}
rpl::producer<Session*> Accounts::activeSessionChanges() const {
return _activeSessions.events();
}
rpl::producer<Session*> Accounts::activeSessionValue() const {
const auto current = (_accounts.empty() || !active().sessionExists())
? nullptr
: &active().session();
return rpl::single(current) | rpl::then(_activeSessions.events());
}
int Accounts::unreadBadge() const {
return _unreadBadge;
}
bool Accounts::unreadBadgeMuted() const {
return _unreadBadgeMuted;
}
rpl::producer<> Accounts::unreadBadgeChanges() const {
return _unreadBadgeChanges.events();
}
void Accounts::notifyUnreadBadgeChanged() {
for (const auto &[index, account] : _accounts) {
if (account->sessionExists()) {
account->session().data().notifyUnreadBadgeChanged();
}
}
}
void Accounts::updateUnreadBadge() {
_unreadBadge = 0;
_unreadBadgeMuted = true;
for (const auto &[index, account] : _accounts) {
if (account->sessionExists()) {
const auto data = &account->session().data();
_unreadBadge += data->unreadBadge();
if (!data->unreadBadgeMuted()) {
_unreadBadgeMuted = false;
}
}
}
_unreadBadgeChanges.fire({});
}
void Accounts::scheduleUpdateUnreadBadge() {
if (_unreadBadgeUpdateScheduled) {
return;
}
_unreadBadgeUpdateScheduled = true;
Core::App().postponeCall(crl::guard(&Core::App(), [=] {
_unreadBadgeUpdateScheduled = false;
updateUnreadBadge();
}));
}
int Accounts::add(MTP::Environment environment) {
Expects(_active.current() != nullptr);
static const auto cloneConfig = [](const MTP::Config &config) {
return std::make_unique<MTP::Config>(config);
};
static const auto accountConfig = [](not_null<Account*> account) {
return cloneConfig(account->mtp().config());
};
auto config = [&] {
if (_active.current()->mtp().environment() == environment) {
return accountConfig(_active.current());
}
for (const auto &[index, account] : list()) {
if (account->mtp().environment() == environment) {
return accountConfig(account.get());
}
}
return (environment == MTP::Environment::Production)
? cloneConfig(Core::App().fallbackProductionConfig())
: std::make_unique<MTP::Config>(environment);
}();
2020-06-15 20:25:02 +04:00
auto index = 0;
while (_accounts.contains(index)) {
++index;
}
2020-06-16 10:42:47 +04:00
const auto account = _accounts.emplace(
index,
std::make_unique<Account>(_dataName, index)
).first->second.get();
_local->startAdded(account, std::move(config));
2020-06-16 13:40:43 +04:00
watchSession(account);
2020-06-15 20:25:02 +04:00
return index;
}
2020-06-16 13:40:43 +04:00
void Accounts::watchSession(not_null<Account*> account) {
account->sessionValue(
) | rpl::filter([=](Session *session) {
return session != nullptr;
}) | rpl::start_with_next([=](Session *session) {
session->data().unreadBadgeChanges(
) | rpl::start_with_next([=] {
scheduleUpdateUnreadBadge();
}, session->lifetime());
}, account->lifetime());
2020-06-16 13:40:43 +04:00
account->sessionChanges(
) | rpl::filter([=](Session *session) {
return !session;
}) | rpl::start_with_next([=] {
scheduleUpdateUnreadBadge();
2020-06-16 13:40:43 +04:00
if (account == _active.current()) {
activateAuthedAccount();
}
crl::on_main(&Core::App(), [=] {
removeRedundantAccounts();
});
}, account->lifetime());
}
void Accounts::activateAuthedAccount() {
Expects(started());
if (_active.current()->sessionExists()) {
return;
}
for (auto i = _accounts.begin(); i != _accounts.end(); ++i) {
if (i->second->sessionExists()) {
activate(i->first);
return;
}
}
}
2020-06-16 18:19:23 +04:00
bool Accounts::removePasscodeIfEmpty() {
if (_accounts.size() != 1 || _active.current()->sessionExists()) {
return false;
}
Local::reset();
if (!Global::LocalPasscode()) {
return false;
}
// We completely logged out, remove the passcode if it was there.
Core::App().unlockPasscode();
_local->setPasscode(QByteArray());
return true;
}
2020-06-16 13:40:43 +04:00
void Accounts::removeRedundantAccounts() {
Expects(started());
const auto was = _accounts.size();
activateAuthedAccount();
for (auto i = _accounts.begin(); i != _accounts.end();) {
if (i->second.get() == _active.current()
|| i->second->sessionExists()) {
++i;
continue;
}
checkForLastProductionConfig(i->second.get());
2020-06-16 13:40:43 +04:00
i = _accounts.erase(i);
}
2020-06-16 18:19:23 +04:00
if (!removePasscodeIfEmpty() && _accounts.size() != was) {
2020-06-16 13:40:43 +04:00
scheduleWriteAccounts();
}
}
void Accounts::checkForLastProductionConfig(
not_null<Main::Account*> account) {
const auto mtp = &account->mtp();
if (mtp->environment() != MTP::Environment::Production) {
return;
}
for (const auto &[index, other] : _accounts) {
if (other.get() != account
&& other->mtp().environment() == MTP::Environment::Production) {
return;
}
}
Core::App().refreshFallbackProductionConfig(mtp->config());
}
2020-06-15 20:25:02 +04:00
void Accounts::activate(int index) {
Expects(_accounts.contains(index));
const auto changed = (_activeIndex != index);
2020-06-15 20:25:02 +04:00
_activeLifetime.destroy();
_activeIndex = index;
_active = _accounts.find(index)->second.get();
_active.current()->sessionValue(
) | rpl::start_to_stream(_activeSessions, _activeLifetime);
2020-06-16 10:42:47 +04:00
if (changed) {
scheduleWriteAccounts();
}
2020-06-16 13:40:43 +04:00
}
void Accounts::scheduleWriteAccounts() {
if (_writeAccountsScheduled) {
return;
}
_writeAccountsScheduled = true;
2020-06-16 10:42:47 +04:00
crl::on_main(&Core::App(), [=] {
2020-06-16 13:40:43 +04:00
_writeAccountsScheduled = false;
2020-06-16 10:42:47 +04:00
_local->writeAccounts();
});
2020-06-15 20:25:02 +04:00
}
} // namespace Main