tdesktop/Telegram/SourceFiles/info/statistics/info_statistics_list_controllers.cpp

392 lines
9.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 "info/statistics/info_statistics_list_controllers.h"
#include "api/api_statistics.h"
#include "boxes/peer_list_controllers.h"
#include "data/data_channel.h"
#include "data/data_session.h"
#include "data/data_user.h"
#include "history/history_item.h"
#include "lang/lang_keys.h"
#include "main/main_session.h"
#include "settings/settings_common.h"
#include "ui/widgets/buttons.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "styles/style_settings.h"
namespace Info::Statistics {
namespace {
void AddSubsectionTitle(
not_null<Ui::VerticalLayout*> container,
rpl::producer<QString> title) {
const auto &subtitlePadding = st::settingsButton.padding;
::Settings::AddSubsectionTitle(
container,
std::move(title),
{ 0, -subtitlePadding.top(), 0, -subtitlePadding.bottom() });
}
[[nodiscard]] QString FormatText(
int value1, tr::phrase<lngtag_count> phrase1,
int value2, tr::phrase<lngtag_count> phrase2,
int value3, tr::phrase<lngtag_count> phrase3) {
const auto separator = u", "_q;
auto resultText = QString();
if (value1 > 0) {
resultText += phrase1(tr::now, lt_count, value1);
}
if (value2 > 0) {
if (!resultText.isEmpty()) {
resultText += separator;
}
resultText += phrase2(tr::now, lt_count, value2);
}
if (value3 > 0) {
if (!resultText.isEmpty()) {
resultText += separator;
}
resultText += phrase3(tr::now, lt_count, value3);
}
return resultText;
}
struct Descriptor final {
Api::PublicForwards::Slice firstSlice;
Fn<void(FullMsgId)> showPeerHistory;
not_null<PeerData*> peer;
FullMsgId contextId;
};
struct MembersDescriptor final {
not_null<Main::Session*> session;
Fn<void(not_null<PeerData*>)> showPeerInfo;
Data::SupergroupStatistics data;
};
class PeerListRowWithMsgId : public PeerListRow {
public:
using PeerListRow::PeerListRow;
void setMsgId(MsgId msgId);
[[nodiscard]] MsgId msgId() const;
private:
MsgId _msgId;
};
void PeerListRowWithMsgId::setMsgId(MsgId msgId) {
_msgId = msgId;
}
MsgId PeerListRowWithMsgId::msgId() const {
return _msgId;
}
class MembersController final : public PeerListController {
public:
MembersController(MembersDescriptor d);
Main::Session &session() const override;
void prepare() override;
void rowClicked(not_null<PeerListRow*> row) override;
void loadMoreRows() override;
void setLimit(int limit);
private:
void addRows(int from, int to);
const not_null<Main::Session*> _session;
Fn<void(not_null<PeerData*>)> _showPeerInfo;
Data::SupergroupStatistics _data;
int _limit = 0;
};
MembersController::MembersController(MembersDescriptor d)
: _session(std::move(d.session))
, _showPeerInfo(std::move(d.showPeerInfo))
, _data(std::move(d.data)) {
}
Main::Session &MembersController::session() const {
return *_session;
}
void MembersController::setLimit(int limit) {
addRows(_limit, limit);
_limit = limit;
}
void MembersController::addRows(int from, int to) {
const auto addRow = [&](UserId userId, QString text) {
const auto user = _session->data().user(userId);
auto row = std::make_unique<PeerListRow>(user);
row->setCustomStatus(std::move(text));
delegate()->peerListAppendRow(std::move(row));
};
if (!_data.topSenders.empty()) {
for (auto i = from; i < to; i++) {
const auto &member = _data.topSenders[i];
addRow(
member.userId,
FormatText(
member.sentMessageCount,
tr::lng_stats_member_messages,
member.averageCharacterCount,
tr::lng_stats_member_characters,
0,
{}));
}
} else if (!_data.topAdministrators.empty()) {
for (auto i = from; i < to; i++) {
const auto &admin = _data.topAdministrators[i];
addRow(
admin.userId,
FormatText(
admin.deletedMessageCount,
tr::lng_stats_member_deletions,
admin.bannedUserCount,
tr::lng_stats_member_bans,
admin.restrictedUserCount,
tr::lng_stats_member_restrictions));
}
} else if (!_data.topInviters.empty()) {
for (auto i = from; i < to; i++) {
const auto &inviter = _data.topInviters[i];
addRow(
inviter.userId,
FormatText(
inviter.addedMemberCount,
tr::lng_stats_member_invitations,
0,
{},
0,
{}));
}
}
}
void MembersController::prepare() {
}
void MembersController::loadMoreRows() {
}
void MembersController::rowClicked(not_null<PeerListRow*> row) {
crl::on_main([=, peer = row->peer()] {
_showPeerInfo(peer);
});
}
class PublicForwardsController final : public PeerListController {
public:
explicit PublicForwardsController(Descriptor d);
Main::Session &session() const override;
void prepare() override;
void rowClicked(not_null<PeerListRow*> row) override;
void loadMoreRows() override;
private:
bool appendRow(not_null<PeerData*> peer, MsgId msgId);
void applySlice(const Api::PublicForwards::Slice &slice);
const not_null<Main::Session*> _session;
Fn<void(FullMsgId)> _showPeerHistory;
Api::PublicForwards _api;
Api::PublicForwards::Slice _firstSlice;
Api::PublicForwards::OffsetToken _apiToken;
bool _allLoaded = false;
};
PublicForwardsController::PublicForwardsController(Descriptor d)
: _session(&d.peer->session())
, _showPeerHistory(std::move(d.showPeerHistory))
, _api(d.peer->asChannel(), d.contextId)
, _firstSlice(std::move(d.firstSlice)) {
}
Main::Session &PublicForwardsController::session() const {
return *_session;
}
void PublicForwardsController::prepare() {
applySlice(base::take(_firstSlice));
delegate()->peerListRefreshRows();
}
void PublicForwardsController::loadMoreRows() {
if (_allLoaded) {
return;
}
_api.request(_apiToken, [=](const Api::PublicForwards::Slice &slice) {
applySlice(slice);
});
}
void PublicForwardsController::applySlice(
const Api::PublicForwards::Slice &slice) {
_allLoaded = slice.allLoaded;
_apiToken = slice.token;
for (const auto &item : slice.list) {
if (const auto peer = session().data().peerLoaded(item.peer)) {
appendRow(peer, item.msg);
}
}
delegate()->peerListRefreshRows();
}
void PublicForwardsController::rowClicked(not_null<PeerListRow*> row) {
const auto rowWithMsgId = static_cast<PeerListRowWithMsgId*>(row.get());
crl::on_main([=, msgId = rowWithMsgId->msgId(), peer = row->peer()] {
_showPeerHistory({ peer->id, msgId });
});
}
bool PublicForwardsController::appendRow(
not_null<PeerData*> peer,
MsgId msgId) {
if (delegate()->peerListFindRow(peer->id.value)) {
return false;
}
auto row = std::make_unique<PeerListRowWithMsgId>(peer);
row->setMsgId(msgId);
const auto members = peer->asChannel()->membersCount();
const auto message = peer->owner().message({ peer->id, msgId });
const auto views = message ? message->viewsCount() : 0;
const auto membersText = !members
? QString()
: peer->isMegagroup()
? tr::lng_chat_status_members(tr::now, lt_count_decimal, members)
: tr::lng_chat_status_subscribers(tr::now, lt_count_decimal, members);
const auto viewsText = views
? tr::lng_stats_recent_messages_views({}, lt_count_decimal, views)
: QString();
const auto resultText = (membersText.isEmpty() || viewsText.isEmpty())
? membersText + viewsText
: QString("%1, %2").arg(membersText, viewsText);
row->setCustomStatus(resultText);
delegate()->peerListAppendRow(std::move(row));
return true;
}
} // namespace
void AddPublicForwards(
const Api::MessageStatistics &firstSliceHolder,
not_null<Ui::VerticalLayout*> container,
Fn<void(FullMsgId)> showPeerHistory,
not_null<PeerData*> peer,
FullMsgId contextId) {
if (!peer->isChannel()) {
return;
}
struct State final {
State(Descriptor d) : controller(std::move(d)) {
}
PeerListContentDelegateSimple delegate;
PublicForwardsController controller;
};
const auto state = container->lifetime().make_state<State>(Descriptor{
firstSliceHolder.firstSlice(),
std::move(showPeerHistory),
peer,
contextId,
});
if (const auto total = firstSliceHolder.firstSlice().total; total > 0) {
AddSubsectionTitle(
container,
tr::lng_stats_overview_message_public_share(
lt_count_decimal,
rpl::single<float64>(total)));
}
state->delegate.setContent(container->add(
object_ptr<PeerListContent>(container, &state->controller)));
state->controller.setDelegate(&state->delegate);
}
void AddMembersList(
Data::SupergroupStatistics data,
not_null<Ui::VerticalLayout*> container,
Fn<void(not_null<PeerData*>)> showPeerInfo,
not_null<PeerData*> peer,
rpl::producer<QString> title) {
if (!peer->isMegagroup()) {
return;
}
const auto max = !data.topSenders.empty()
? data.topSenders.size()
: !data.topAdministrators.empty()
? data.topAdministrators.size()
: !data.topInviters.empty()
? data.topInviters.size()
: 0;
if (!max) {
return;
}
constexpr auto kPerPage = 40;
struct State final {
State(MembersDescriptor d) : controller(std::move(d)) {
}
PeerListContentDelegateSimple delegate;
MembersController controller;
int limit = 0;
};
auto d = MembersDescriptor{
&peer->session(),
std::move(showPeerInfo),
std::move(data),
};
const auto state = container->lifetime().make_state<State>(std::move(d));
AddSubsectionTitle(container, std::move(title));
state->delegate.setContent(container->add(
object_ptr<PeerListContent>(container, &state->controller)));
state->controller.setDelegate(&state->delegate);
const auto wrap = container->add(
object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
container,
object_ptr<Ui::SettingsButton>(
container,
tr::lng_stories_show_more())),
{ 0, -st::settingsButton.padding.top(), 0, 0 });
const auto button = wrap->entity();
const auto showMore = [=] {
state->limit = std::min(int(max), state->limit + kPerPage);
state->controller.setLimit(state->limit);
if (state->limit == max) {
wrap->toggle(false, anim::type::instant);
}
container->resizeToWidth(container->width());
};
button->setClickedCallback(showMore);
showMore();
}
} // namespace Info::Statistics