AyuGramDesktop/Telegram/SourceFiles/main/session/send_as_peers.cpp
2025-02-12 17:24:24 +04:00

192 lines
5.3 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 } })
, _onlyMePaid({ session->user() }) {
_session->changes().peerUpdates(
Data::PeerUpdate::Flag::Rights
) | rpl::map([=](const Data::PeerUpdate &update) {
const auto peer = update.peer;
const auto channel = peer->asChannel();
const auto bits = 0
| (peer->amAnonymous() ? (1 << 0) : 0)
| ((channel && channel->isPublic()) ? (1 << 1) : 0)
| ((channel && channel->addsSignature()) ? (1 << 2) : 0)
| ((channel && channel->signatureProfiles()) ? (1 << 3) : 0);
return std::tuple(peer, bits);
}) | rpl::distinct_until_changed(
) | rpl::filter([=](not_null<PeerData*> peer, int) {
return _lists.contains(peer) || _lastRequestTime.contains(peer);
}) | rpl::start_with_next([=](not_null<PeerData*> peer, int) {
refresh(peer, true);
}, _lifetime);
}
bool SendAsPeers::shouldChoose(not_null<PeerData*> peer) {
refresh(peer);
const auto channel = peer->asBroadcast();
return Data::CanSendAnything(peer, false)
&& (list(peer).size() > 1)
&& (!channel
|| channel->addsSignature()
|| channel->signatureProfiles());
}
void SendAsPeers::refresh(not_null<PeerData*> peer, bool force) {
if (!peer->isChannel()) {
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);
request(peer, true);
}
const std::vector<SendAsPeer> &SendAsPeers::list(
not_null<PeerData*> peer) const {
const auto i = _lists.find(peer);
return (i != end(_lists)) ? i->second : _onlyMe;
}
const std::vector<not_null<PeerData*>> &SendAsPeers::paidReactionList(
not_null<PeerData*> peer) const {
const auto i = _paidReactionLists.find(peer);
return (i != end(_paidReactionLists)) ? i->second : _onlyMePaid;
}
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 fallback = peer->amAnonymous()
? peer
: peer->session().user();
if (!chosen) {
chosen = fallback->id;
}
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
: fallback;
}
void SendAsPeers::request(not_null<PeerData*> peer, bool forPaidReactions) {
using Flag = MTPchannels_GetSendAs::Flag;
peer->session().api().request(MTPchannels_GetSendAs(
MTP_flags(forPaidReactions ? Flag::f_for_paid_reactions : Flag()),
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 (forPaidReactions) {
auto peers = parsed | ranges::views::transform(
&SendAsPeer::peer
) | ranges::to_vector;
if (!peers.empty()) {
_paidReactionLists[peer] = std::move(peers);
} else {
_paidReactionLists.remove(peer);
}
} else {
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