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

152 lines
3.9 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({ 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();
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() && (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<not_null<PeerData*>> &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<not_null<PeerData*>> &list,
PeerId chosen) {
const auto i = ranges::find(list, chosen, &PeerData::id);
return (i != end(list))
? (*i)
: !list.empty()
? list.front()
: (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 list = std::vector<not_null<PeerData*>>();
auto &owner = peer->owner();
result.match([&](const MTPDchannels_sendAsPeers &data) {
owner.processUsers(data.vusers());
owner.processChats(data.vchats());
for (const auto &id : data.vpeers().v) {
if (const auto peer = owner.peerLoaded(peerFromMTP(id))) {
list.push_back(peer);
}
}
});
if (list.size() > 1) {
auto &now = _lists[peer];
if (now != list) {
now = std::move(list);
_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