tdesktop/Telegram/SourceFiles/main/session/send_as_peers.cpp

161 lines
4.2 KiB
C++

/*
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/session/send_as_peers.h"
#include "data/data_user.h"
#include "data/data_channel.h"
#include "data/data_session.h"
#include "data/data_changes.h"
#include "main/main_session.h"
#include "apiwrap.h"
namespace Main {
namespace {
constexpr auto kRequestEach = 30 * crl::time(1000);
} // namespace
SendAsPeers::SendAsPeers(not_null<Session*> session)
: _session(session)
, _onlyMe({ { .peer = session->user(), .premiumRequired = false } }) {
_session->changes().peerUpdates(
Data::PeerUpdate::Flag::Rights
) | rpl::map([=](const Data::PeerUpdate &update) {
const auto peer = update.peer;
const auto channel = peer->asChannel();
return std::tuple(
peer,
peer->amAnonymous(),
channel ? channel->isPublic() : false);
}) | rpl::distinct_until_changed(
) | rpl::filter([=](not_null<PeerData*> peer, bool, bool) {
return _lists.contains(peer);
}) | rpl::start_with_next([=](not_null<PeerData*> peer, bool, bool) {
refresh(peer, true);
}, _lifetime);
}
bool SendAsPeers::shouldChoose(not_null<PeerData*> peer) {
refresh(peer);
return peer->canWrite(false) && (list(peer).size() > 1);
}
void SendAsPeers::refresh(not_null<PeerData*> peer, bool force) {
if (!peer->isMegagroup()) {
return;
}
const auto now = crl::now();
const auto i = _lastRequestTime.find(peer);
const auto when = (i == end(_lastRequestTime)) ? -1 : i->second;
if (!force && (when >= 0 && now < when + kRequestEach)) {
return;
}
_lastRequestTime[peer] = now;
request(peer);
}
const std::vector<SendAsPeer> &SendAsPeers::list(
not_null<PeerData*> peer) const {
const auto i = _lists.find(peer);
return (i != end(_lists)) ? i->second : _onlyMe;
}
rpl::producer<not_null<PeerData*>> SendAsPeers::updated() const {
return _updates.events();
}
void SendAsPeers::saveChosen(
not_null<PeerData*> peer,
not_null<PeerData*> chosen) {
peer->session().api().request(MTPmessages_SaveDefaultSendAs(
peer->input,
chosen->input
)).send();
setChosen(peer, chosen->id);
}
void SendAsPeers::setChosen(not_null<PeerData*> peer, PeerId chosenId) {
if (chosen(peer) == chosenId) {
return;
}
const auto fallback = peer->amAnonymous()
? peer
: peer->session().user();
if (fallback->id == chosenId) {
_chosen.remove(peer);
} else {
_chosen[peer] = chosenId;
}
_updates.fire_copy(peer);
}
PeerId SendAsPeers::chosen(not_null<PeerData*> peer) const {
const auto i = _chosen.find(peer);
return (i != end(_chosen)) ? i->second : PeerId();
}
not_null<PeerData*> SendAsPeers::resolveChosen(
not_null<PeerData*> peer) const {
return ResolveChosen(peer, list(peer), chosen(peer));
}
not_null<PeerData*> SendAsPeers::ResolveChosen(
not_null<PeerData*> peer,
const std::vector<SendAsPeer> &list,
PeerId chosen) {
const auto i = ranges::find(list, chosen, [](const SendAsPeer &as) {
return as.peer->id;
});
return (i != end(list))
? i->peer
: !list.empty()
? list.front().peer
: (peer->isMegagroup() && peer->amAnonymous())
? peer
: peer->session().user();
}
void SendAsPeers::request(not_null<PeerData*> peer) {
peer->session().api().request(MTPchannels_GetSendAs(
peer->input
)).done([=](const MTPchannels_SendAsPeers &result) {
auto parsed = std::vector<SendAsPeer>();
auto &owner = peer->owner();
result.match([&](const MTPDchannels_sendAsPeers &data) {
owner.processUsers(data.vusers());
owner.processChats(data.vchats());
const auto &list = data.vpeers().v;
parsed.reserve(list.size());
for (const auto &as : list) {
const auto &data = as.data();
const auto peerId = peerFromMTP(data.vpeer());
if (const auto peer = owner.peerLoaded(peerId)) {
parsed.push_back({
.peer = peer,
.premiumRequired = data.is_premium_required(),
});
}
}
});
if (parsed.size() > 1) {
auto &now = _lists[peer];
if (now != parsed) {
now = std::move(parsed);
_updates.fire_copy(peer);
}
} else if (const auto i = _lists.find(peer); i != end(_lists)) {
_lists.erase(i);
_updates.fire_copy(peer);
}
}).send();
}
} // namespace Main