Replaced custom GroupMembersWidget with PeerList.

This commit is contained in:
23rd 2025-08-11 15:50:32 +03:00
parent 04479ad660
commit a2847246e6
4 changed files with 116 additions and 510 deletions

View file

@ -5806,11 +5806,7 @@ void HistoryWidget::showMembersDropdown() {
if (!_membersDropdown) {
_membersDropdown.create(this, st::membersInnerDropdown);
_membersDropdown->setOwnedWidget(
object_ptr<HistoryView::GroupMembersWidget>(
this,
controller(),
_peer,
st::membersInnerItem));
object_ptr<HistoryView::GroupMembersWidget>(this, controller(), _peer));
_membersDropdown->resizeToWidth(st::membersInnerWidth);
_membersDropdown->setMaxHeight(countMembersDropdownHeightMax());

View file

@ -7,478 +7,107 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "history/view/history_view_group_members_widget.h"
#include "api/api_chat_participants.h"
#include "styles/style_profile.h"
#include "ui/widgets/labels.h"
#include "ui/boxes/confirm_box.h"
#include "boxes/peers/edit_participant_box.h"
#include "boxes/peers/edit_participants_box.h"
#include "base/unixtime.h"
#include "ui/widgets/popup_menu.h"
#include "mtproto/mtproto_config.h"
#include "data/data_peer_values.h"
#include "data/data_channel.h"
#include "data/data_chat.h"
#include "data/data_user.h"
#include "data/data_changes.h"
#include "mainwidget.h"
#include "apiwrap.h"
#include "main/main_session.h"
#include "lang/lang_keys.h"
#include "window/window_session_controller.h"
#include "styles/style_boxes.h"
#include "styles/style_chat.h"
namespace HistoryView {
namespace {
using UpdateFlag = Data::PeerUpdate::Flag;
class GroupMembersWidgetController : public ParticipantsBoxController {
public:
using ParticipantsBoxController::ParticipantsBoxController;
protected:
base::unique_qptr<Ui::PopupMenu> rowContextMenu(
QWidget *parent,
not_null<PeerListRow*> row) override {
return nullptr;
}
};
} // namespace
GroupMembersWidget::Member::Member(not_null<UserData*> user) : Item(user) {
}
not_null<UserData*> GroupMembersWidget::Member::user() const {
return static_cast<UserData*>(peer.get());
}
GroupMembersWidget::GroupMembersWidget(
QWidget *parent,
not_null<Window::SessionController*> controller,
not_null<PeerData*> peer,
const style::PeerListItem &st)
: PeerListWidget(parent, peer, QString(), st, tr::lng_profile_kick(tr::now))
, _controller(controller)
, _updateOnlineTimer([=] { updateOnlineDisplay(); }) {
peer->session().changes().peerUpdates(
UpdateFlag::Admins
| UpdateFlag::Members
| UpdateFlag::OnlineStatus
) | rpl::start_with_next([=](const Data::PeerUpdate &update) {
notifyPeerUpdated(update);
}, lifetime());
setRemovedCallback([=](PeerData *selectedPeer) {
removePeer(selectedPeer);
});
setSelectedCallback([=](PeerData *selectedPeer) {
controller->showPeerInfo(selectedPeer);
});
setUpdateItemCallback([=](Item *item) {
updateItemStatusText(item);
});
setPreloadMoreCallback([=] {
preloadMore();
});
refreshMembers();
not_null<Ui::RpWidget*> parent,
not_null<Window::SessionNavigation*> navigation,
not_null<PeerData*> peer)
: Ui::RpWidget(parent)
, _show(navigation->uiShow())
, _listController(std::make_unique<GroupMembersWidgetController>(
navigation,
peer,
ParticipantsBoxController::Role::Profile)) {
_listController->setStoriesShown(true);
setupList();
setContent(_list.data());
_listController->setDelegate(static_cast<PeerListDelegate*>(this));
}
void GroupMembersWidget::removePeer(PeerData *selectedPeer) {
const auto user = selectedPeer->asUser();
Assert(user != nullptr);
const auto text = tr::lng_profile_sure_kick(
tr::now,
lt_user, user->firstName);
const auto currentRestrictedRights = [&]() -> ChatRestrictionsInfo {
if (auto channel = peer()->asMegagroup()) {
auto it = channel->mgInfo->lastRestricted.find(user);
if (it != channel->mgInfo->lastRestricted.cend()) {
return it->second.rights;
}
void GroupMembersWidget::setupList() {
const auto topSkip = 0;
_listController->setStyleOverrides(&st::groupMembersWidgetList);
_listController->setStoriesShown(true);
_list = object_ptr<PeerListContent>(this, _listController.get());
widthValue() | rpl::start_with_next([this](int newWidth) {
if (newWidth > 0) {
_list->resizeToWidth(newWidth);
}
return ChatRestrictionsInfo();
}();
const auto peer = this->peer();
const auto callback = [=, controller = _controller] {
controller->hideLayer();
if (const auto chat = peer->asChat()) {
chat->session().api().chatParticipants().kick(chat, user);
controller->showPeerHistory(
chat->id,
Window::SectionShow::Way::ClearStack,
ShowAtTheEndMsgId);
} else if (const auto channel = peer->asChannel()) {
channel->session().api().chatParticipants().kick(
channel,
user,
currentRestrictedRights);
}, _list->lifetime());
_list->heightValue() | rpl::start_with_next([=](int listHeight) {
if (const auto newHeight = topSkip + listHeight; newHeight > 0) {
resize(width(), newHeight);
}
};
_controller->show(Ui::MakeConfirmBox({
.text = text,
.confirmed = crl::guard(&peer->session(), callback),
.confirmText = tr::lng_box_remove(),
}));
}, _list->lifetime());
_list->moveToLeft(0, topSkip);
}
void GroupMembersWidget::notifyPeerUpdated(const Data::PeerUpdate &update) {
if (update.peer != peer()) {
if (update.flags & UpdateFlag::OnlineStatus) {
if (auto user = update.peer->asUser()) {
refreshUserOnline(user);
}
}
return;
}
if (update.flags & UpdateFlag::Members) {
refreshMembers();
contentSizeUpdated();
}
if (update.flags & UpdateFlag::Admins) {
if (const auto chat = peer()->asChat()) {
for (const auto item : items()) {
setItemFlags(getMember(item), chat);
}
} else if (const auto megagroup = peer()->asMegagroup()) {
for (const auto item : items()) {
setItemFlags(getMember(item), megagroup);
}
}
}
this->update();
void GroupMembersWidget::peerListSetTitle(rpl::producer<QString> title) {
}
void GroupMembersWidget::refreshUserOnline(UserData *user) {
auto it = _membersByUser.find(user);
if (it == _membersByUser.cend()) return;
_now = base::unixtime::now();
auto member = getMember(it->second);
member->lastseen = user->lastseen();
member->statusHasOnlineColor = Data::OnlineTextActive(user, _now);
member->onlineForSort = user->isSelf()
? std::numeric_limits<TimeId>::max()
: Data::SortByOnlineValue(user, _now);
member->statusText = QString();
sortMembers();
update();
void GroupMembersWidget::peerListSetAdditionalTitle(
rpl::producer<QString> title) {
}
void GroupMembersWidget::preloadMore() {
//
// This can cause a ddos, because lastParticipants may never reach members count.
//
//if (auto megagroup = peer()->asMegagroup()) {
// auto &megagroupInfo = megagroup->mgInfo;
// if (!megagroupInfo->lastParticipants.isEmpty() && megagroupInfo->lastParticipants.size() < megagroup->membersCount()) {
// peer()->session().api().requestLast(megagroup, false);
// }
//}
bool GroupMembersWidget::peerListIsRowChecked(not_null<PeerListRow*> row) {
return false;
}
void GroupMembersWidget::updateItemStatusText(Item *item) {
auto member = getMember(item);
auto user = member->user();
if (member->statusText.isEmpty() || (member->onlineTextTill <= _now)) {
if (user->isBot()) {
const auto seesAllMessages = user->botInfo->readsAllHistory
|| member->rank.has_value();
member->statusText = seesAllMessages
? tr::lng_status_bot_reads_all(tr::now)
: tr::lng_status_bot_not_reads_all(tr::now);
member->onlineTextTill = _now + 86400;
} else {
member->statusHasOnlineColor = member->lastseen.isOnline(_now);
member->statusText = Data::OnlineText(member->lastseen, _now);
const auto changeInMs = Data::OnlineChangeTimeout(
member->lastseen,
_now);
member->onlineTextTill = _now + TimeId(changeInMs / 1000);
}
}
if (_updateOnlineAt <= _now || _updateOnlineAt > member->onlineTextTill) {
_updateOnlineAt = member->onlineTextTill;
_updateOnlineTimer.callOnce((_updateOnlineAt - _now + 1) * 1000);
}
int GroupMembersWidget::peerListSelectedRowsCount() {
return 0;
}
void GroupMembersWidget::refreshMembers() {
_now = base::unixtime::now();
if (const auto chat = peer()->asChat()) {
checkSelfAdmin(chat);
if (chat->noParticipantInfo()) {
chat->session().api().requestFullPeer(chat);
}
fillChatMembers(chat);
} else if (const auto megagroup = peer()->asMegagroup()) {
if (megagroup->lastParticipantsRequestNeeded()) {
megagroup->session().api().chatParticipants().requestLast(
megagroup);
}
fillMegagroupMembers(megagroup);
}
sortMembers();
refreshVisibility();
void GroupMembersWidget::peerListScrollToTop() {
}
void GroupMembersWidget::checkSelfAdmin(not_null<ChatData*> chat) {
if (chat->participants.empty()) {
return;
}
//const auto self = chat->session().user();
//if (chat->amAdmin() && !chat->admins.contains(self)) {
// chat->admins.insert(self);
//} else if (!chat->amAdmin() && chat->admins.contains(self)) {
// chat->admins.remove(self);
//}
void GroupMembersWidget::peerListAddSelectedPeerInBunch(
not_null<PeerData*> peer) {
Unexpected("Item selection in Info::Profile::Members.");
}
void GroupMembersWidget::sortMembers() {
if (!_sortByOnline || !itemsCount()) return;
sortItems([this](Item *a, Item *b) {
return getMember(a)->onlineForSort > getMember(b)->onlineForSort;
});
updateOnlineCount();
void GroupMembersWidget::peerListAddSelectedRowInBunch(
not_null<PeerListRow*> row) {
Unexpected("Item selection in Info::Profile::Members.");
}
void GroupMembersWidget::updateOnlineCount() {
bool onlyMe = true;
int newOnlineCount = 0;
for (const auto item : items()) {
auto member = getMember(item);
auto user = member->user();
auto isOnline = !user->isBot()
&& member->lastseen.isOnline(_now);
if (member->statusHasOnlineColor != isOnline) {
member->statusHasOnlineColor = isOnline;
member->statusText = QString();
}
if (member->statusHasOnlineColor) {
++newOnlineCount;
if (!user->isSelf()) {
onlyMe = false;
}
}
}
if (newOnlineCount == 1 && onlyMe) {
newOnlineCount = 0;
}
if (_onlineCount != newOnlineCount) {
_onlineCount = newOnlineCount;
}
void GroupMembersWidget::peerListFinishSelectedRowsBunch() {
}
auto GroupMembersWidget::addUser(
not_null<ChatData*> chat,
not_null<UserData*> user)
-> not_null<Member*> {
const auto member = computeMember(user);
setItemFlags(member, chat);
addItem(member);
return member;
std::shared_ptr<Main::SessionShow> GroupMembersWidget::peerListUiShow() {
return _show;
}
void GroupMembersWidget::fillChatMembers(not_null<ChatData*> chat) {
if (chat->participants.empty()) {
return;
}
clearItems();
if (!chat->amIn()) {
return;
}
_sortByOnline = true;
reserveItemsForSize(chat->participants.size());
addUser(chat, chat->session().user())->onlineForSort
= std::numeric_limits<TimeId>::max();
for (const auto &user : chat->participants) {
if (!user->isSelf()) {
addUser(chat, user);
}
}
void GroupMembersWidget::peerListSetDescription(
object_ptr<Ui::FlatLabel> description) {
description.destroy();
}
void GroupMembersWidget::setItemFlags(
not_null<Item*> item,
not_null<ChatData*> chat) {
const auto user = getMember(item)->user();
const auto isCreator = (peerFromUser(chat->creator) == item->peer->id);
const auto isAdmin = (item->peer->isSelf() && chat->hasAdminRights())
|| chat->admins.contains(user);
const auto rank = isCreator
? tr::lng_owner_badge(tr::now)
: isAdmin
? tr::lng_admin_badge(tr::now)
: std::optional<QString>();
item->rank = rank;
item->rankWidth = rank ? st::normalFont->width(*rank) : 0;
if (item->peer->id == chat->session().userPeerId()) {
item->hasRemoveLink = false;
} else if (chat->amCreator()
|| ((chat->adminRights() & ChatAdminRight::BanUsers)
&& !rank.has_value())) {
item->hasRemoveLink = true;
} else if (chat->invitedByMe.contains(user) && !rank.has_value()) {
item->hasRemoveLink = true;
} else {
item->hasRemoveLink = false;
}
}
auto GroupMembersWidget::addUser(
not_null<ChannelData*> megagroup,
not_null<UserData*> user)
-> not_null<Member*> {
const auto member = computeMember(user);
setItemFlags(member, megagroup);
addItem(member);
return member;
}
void GroupMembersWidget::fillMegagroupMembers(
not_null<ChannelData*> megagroup) {
Expects(megagroup->mgInfo != nullptr);
if (megagroup->mgInfo->lastParticipants.empty()) {
return;
} else if (!megagroup->canViewMembers()) {
clearItems();
return;
}
_sortByOnline = (megagroup->membersCount() > 0)
&& (megagroup->membersCount()
<= megagroup->session().serverConfig().chatSizeMax);
auto &membersList = megagroup->mgInfo->lastParticipants;
if (_sortByOnline) {
clearItems();
reserveItemsForSize(membersList.size());
if (megagroup->amIn()) {
addUser(megagroup, megagroup->session().user())->onlineForSort
= std::numeric_limits<TimeId>::max();
}
} else if (membersList.size() >= itemsCount()) {
if (addUsersToEnd(megagroup)) {
return;
}
}
if (!_sortByOnline) {
clearItems();
reserveItemsForSize(membersList.size());
}
for (const auto user : membersList) {
if (!_sortByOnline || !user->isSelf()) {
addUser(megagroup, user);
}
}
}
bool GroupMembersWidget::addUsersToEnd(not_null<ChannelData*> megagroup) {
auto &membersList = megagroup->mgInfo->lastParticipants;
auto &itemsList = items();
for (int i = 0, count = itemsList.size(); i < count; ++i) {
if (itemsList[i]->peer != membersList.at(i)) {
return false;
}
}
reserveItemsForSize(membersList.size());
for (int i = itemsCount(), count = membersList.size(); i < count; ++i) {
addUser(megagroup, membersList.at(i));
}
return true;
}
void GroupMembersWidget::setItemFlags(
not_null<Item*> item,
not_null<ChannelData*> megagroup) {
const auto amCreator = item->peer->isSelf() && megagroup->amCreator();
const auto amAdmin = item->peer->isSelf() && megagroup->hasAdminRights();
const auto user = getMember(item)->user();
const auto adminIt = megagroup->mgInfo->lastAdmins.find(user);
const auto isAdmin = (adminIt != megagroup->mgInfo->lastAdmins.cend());
const auto isCreator = (megagroup->mgInfo->creator == item->peer);
const auto rankIt = megagroup->mgInfo->admins.find(peerToUser(user->id));
const auto adminCanEdit = isAdmin && adminIt->second.canEdit;
const auto rank = (amCreator || isCreator)
? (!megagroup->mgInfo->creatorRank.isEmpty()
? megagroup->mgInfo->creatorRank
: tr::lng_owner_badge(tr::now))
: (amAdmin || isAdmin)
? ((rankIt != megagroup->mgInfo->admins.end()
&& !rankIt->second.isEmpty())
? rankIt->second
: tr::lng_admin_badge(tr::now))
: std::optional<QString>();
if (item->rank != rank) {
item->rank = rank;
item->rankWidth = rank ? st::normalFont->width(*rank) : 0;
auto user = item->peer->asUser();
Assert(user != nullptr);
if (user->isBot()) {
// Update "has access to messages" status.
item->statusText = QString();
updateItemStatusText(item);
}
}
if (item->peer->isSelf()) {
item->hasRemoveLink = false;
} else if (megagroup->amCreator()
|| (megagroup->canBanMembers()
&& (!rank.has_value() || adminCanEdit))) {
item->hasRemoveLink = true;
} else {
item->hasRemoveLink = false;
}
}
auto GroupMembersWidget::computeMember(not_null<UserData*> user)
-> not_null<Member*> {
auto it = _membersByUser.find(user);
if (it == _membersByUser.cend()) {
auto member = new Member(user);
it = _membersByUser.emplace(user, member).first;
member->lastseen = user->lastseen();
member->statusHasOnlineColor = !user->isBot()
&& member->lastseen.isOnline(_now);
member->onlineForSort = Data::SortByOnlineValue(user, _now);
}
return it->second;
}
void GroupMembersWidget::updateOnlineDisplay() {
if (_sortByOnline) {
_now = base::unixtime::now();
bool changed = false;
for (const auto item : items()) {
if (!item->statusHasOnlineColor) {
if (!item->peer->isSelf()) {
continue;
} else {
break;
}
}
auto member = getMember(item);
bool isOnline = !member->user()->isBot()
&& member->lastseen.isOnline(_now);
if (!isOnline) {
changed = true;
}
}
if (changed) {
updateOnlineCount();
}
}
update();
}
GroupMembersWidget::~GroupMembersWidget() {
auto members = base::take(_membersByUser);
for (const auto &[_, member] : members) {
delete member;
}
void GroupMembersWidget::peerListShowRowMenu(
not_null<PeerListRow*> row,
bool highlightRow,
Fn<void(not_null<Ui::PopupMenu*>)> destroyed) {
}
} // namespace HistoryView

View file

@ -7,87 +7,51 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "base/timer.h"
#include "data/data_lastseen_status.h"
#include "profile/profile_block_peer_list.h"
namespace Ui {
class FlatLabel;
} // namespace Ui
namespace Data {
struct PeerUpdate;
} // namespace Data
#include "boxes/peer_list_box.h"
namespace Window {
class SessionController;
class SessionNavigation;
} // namespace Window
class ParticipantsBoxController;
namespace HistoryView {
class GroupMembersWidget : public Profile::PeerListWidget {
class GroupMembersWidget
: public Ui::RpWidget
, public PeerListContentDelegate {
public:
GroupMembersWidget(
QWidget *parent,
not_null<Window::SessionController*> controller,
not_null<PeerData*> peer,
const style::PeerListItem &st);
int onlineCount() const {
return _onlineCount;
}
~GroupMembersWidget();
not_null<Ui::RpWidget*> parent,
not_null<Window::SessionNavigation*> navigation,
not_null<PeerData*> peer);
private:
void updateOnlineDisplay();
void setupList();
// Observed notifications.
void notifyPeerUpdated(const Data::PeerUpdate &update);
// PeerListContentDelegate interface.
void peerListSetTitle(rpl::producer<QString> title) override;
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
bool peerListIsRowChecked(not_null<PeerListRow*> row) override;
int peerListSelectedRowsCount() override;
void peerListScrollToTop() override;
void peerListAddSelectedPeerInBunch(
not_null<PeerData*> peer) override;
void peerListAddSelectedRowInBunch(
not_null<PeerListRow*> row) override;
void peerListFinishSelectedRowsBunch() override;
void peerListSetDescription(
object_ptr<Ui::FlatLabel> description) override;
std::shared_ptr<Main::SessionShow> peerListUiShow() override;
void peerListShowRowMenu(
not_null<PeerListRow*> row,
bool highlightRow,
Fn<void(not_null<Ui::PopupMenu*>)> destroyed = nullptr) override;
// PeerListContentDelegate interface.
void removePeer(PeerData *selectedPeer);
void refreshMembers();
void fillChatMembers(not_null<ChatData*> chat);
void fillMegagroupMembers(not_null<ChannelData*> megagroup);
void sortMembers();
void updateOnlineCount();
void checkSelfAdmin(not_null<ChatData*> chat);
void preloadMore();
void refreshUserOnline(UserData *user);
struct Member : public Item {
explicit Member(not_null<UserData*> user);
not_null<UserData*> user() const;
TimeId onlineTextTill = 0;
Data::LastseenStatus lastseen;
TimeId onlineForSort = 0;
};
Member *getMember(Item *item) {
return static_cast<Member*>(item);
}
void updateItemStatusText(Item *item);
not_null<Member*> computeMember(not_null<UserData*> user);
not_null<Member*> addUser(not_null<ChatData*> chat, not_null<UserData*> user);
not_null<Member*> addUser(not_null<ChannelData*> megagroup, not_null<UserData*> user);
void setItemFlags(not_null<Item*> item, not_null<ChatData*> chat);
void setItemFlags(
not_null<Item*> item,
not_null<ChannelData*> megagroup);
bool addUsersToEnd(not_null<ChannelData*> megagroup);
const not_null<Window::SessionController*> _controller;
base::flat_map<UserData*, Member*> _membersByUser;
bool _sortByOnline = false;
TimeId _now = 0;
int _onlineCount = 0;
TimeId _updateOnlineAt = 0;
base::Timer _updateOnlineTimer;
std::shared_ptr<Main::SessionShow> _show;
object_ptr<PeerListContent> _list = { nullptr };
std::unique_ptr<ParticipantsBoxController> _listController;
};

View file

@ -1428,3 +1428,20 @@ lowTonText: FlatLabel(defaultFlatLabel) {
minWidth: 100px;
align: align(top);
}
groupMembersWidgetList: PeerList(defaultPeerList) {
item: PeerListItem(defaultPeerListItem) {
photoPosition: point(12px, 6px);
namePosition: point(68px, 11px);
statusPosition: point(68px, 31px);
checkbox: RoundImageCheckbox(defaultPeerListCheckbox) {
selectExtendTwice: 1px;
imageRadius: 21px;
imageSmallRadius: 19px;
check: RoundCheckbox(defaultPeerListCheck) {
size: 0px;
}
}
nameFgChecked: contactsNameFg;
}
}