From 9e8e49b8d9ef3bb66d7d3413ee8f698b7cd0cd92 Mon Sep 17 00:00:00 2001 From: John Preston Date: Fri, 16 Jun 2017 14:05:13 +0300 Subject: [PATCH] Add other search to admin/ban/restrict in channel. Search in chats / contacts / global in AddParticipantBoxController. Also move all channel participants box controllers to another module. --- .../profile/profile_block_settings.cpp | 1000 +-------------- .../profile/profile_channel_controllers.cpp | 1095 +++++++++++++++++ .../profile/profile_channel_controllers.h | 220 ++++ Telegram/gyp/telegram_sources.txt | 2 + 4 files changed, 1319 insertions(+), 998 deletions(-) create mode 100644 Telegram/SourceFiles/profile/profile_channel_controllers.cpp create mode 100644 Telegram/SourceFiles/profile/profile_channel_controllers.h diff --git a/Telegram/SourceFiles/profile/profile_block_settings.cpp b/Telegram/SourceFiles/profile/profile_block_settings.cpp index 62cbe8d0c..07bbfa22d 100644 --- a/Telegram/SourceFiles/profile/profile_block_settings.cpp +++ b/Telegram/SourceFiles/profile/profile_block_settings.cpp @@ -20,13 +20,12 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #include "profile/profile_block_settings.h" +#include "profile/profile_channel_controllers.h" #include "styles/style_profile.h" #include "ui/widgets/buttons.h" #include "ui/widgets/checkbox.h" -#include "boxes/confirm_box.h" #include "boxes/contacts_box.h" -#include "boxes/peer_list_box.h" -#include "boxes/edit_participant_box.h" +#include "boxes/confirm_box.h" #include "observer_peer.h" #include "auth_session.h" #include "mainwidget.h" @@ -34,1001 +33,6 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "lang/lang_keys.h" namespace Profile { -namespace { - -constexpr auto kBannedPerPage = 40; -constexpr auto kAdminsPerPage = 200; -constexpr auto kParticipantsPerPage = 200; - -// Viewing admins, banned or restricted users list with search. -class ParticipantsBoxController : public PeerListController, private base::Subscriber, private MTP::Sender, public base::enable_weak_from_this { -public: - enum class Role { - Admins, - Restricted, - Kicked, - }; - static void Start(gsl::not_null channel, Role role); - - struct Additional { - std::map, MTPChannelAdminRights> adminRights; - std::set> adminCanEdit; - std::map, gsl::not_null> adminPromotedBy; - std::map, MTPChannelBannedRights> restrictedRights; - std::set> kicked; - std::set> external; - std::set> infoNotLoaded; - UserData *creator = nullptr; - }; - - ParticipantsBoxController(gsl::not_null channel, Role role); - - void addNewItem(); - - void prepare() override; - void rowClicked(gsl::not_null row) override; - void rowActionClicked(gsl::not_null row) override; - void loadMoreRows() override; - - void peerListSearchAddRow(gsl::not_null peer) override; - - // Callback(gsl::not_null) - template - static void HandleParticipant(const MTPChannelParticipant &participant, Role role, gsl::not_null additional, Callback callback); - -private: - void editAdmin(gsl::not_null user); - void editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights); - void editRestricted(gsl::not_null user); - void editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights); - void removeKicked(gsl::not_null row, gsl::not_null user); - bool appendRow(gsl::not_null user); - bool prependRow(gsl::not_null user); - std::unique_ptr createRow(gsl::not_null user) const; - - gsl::not_null _channel; - Role _role = Role::Admins; - int _offset = 0; - mtpRequestId _loadRequestId = 0; - bool _allLoaded = false; - Additional _additional; - QPointer _editBox; - QPointer _addBox; - -}; - -// Banned and restricted users server side search. -class BannedBoxSearchController : public PeerListSearchController, private MTP::Sender { -public: - using Role = ParticipantsBoxController::Role; - using Additional = ParticipantsBoxController::Additional; - - BannedBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional); - - void searchQuery(const QString &query) override; - bool isLoading() override; - bool loadMoreRows() override; - -private: - bool searchInCache(); - void searchOnServer(); - void searchDone(const MTPchannels_ChannelParticipants &result, mtpRequestId requestId); - - gsl::not_null _channel; - Role _role = Role::Restricted; - gsl::not_null _additional; - - base::Timer _timer; - QString _query; - mtpRequestId _requestId = 0; - int _offset = 0; - bool _allLoaded = false; - std::map _cache; - std::map> _queries; // query, offset - -}; - -// Adding an admin, banned or restricted user from channel members with search + contacts search + global search. -class AddParticipantBoxController : public PeerListController, private base::Subscriber, private MTP::Sender, public base::enable_weak_from_this { -public: - using Role = ParticipantsBoxController::Role; - using Additional = ParticipantsBoxController::Additional; - - AddParticipantBoxController(gsl::not_null channel, Role role, base::lambda doneCallback); - - void prepare() override; - void rowClicked(gsl::not_null row) override; - void loadMoreRows() override; - - void peerListSearchAddRow(gsl::not_null peer) override; - - // Callback(gsl::not_null) - template - static void HandleParticipant(const MTPChannelParticipant &participant, gsl::not_null additional, Callback callback); - -private: - template - bool checkInfoLoaded(gsl::not_null user, Callback callback); - - void editAdmin(gsl::not_null user, bool sure = false); - void editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights); - void editRestricted(gsl::not_null user, bool sure = false); - void editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights); - void kickUser(gsl::not_null user, bool sure = false); - void restrictUserSure(gsl::not_null user, const MTPChannelBannedRights &rights); - bool appendRow(gsl::not_null user); - bool prependRow(gsl::not_null user); - std::unique_ptr createRow(gsl::not_null user) const; - - gsl::not_null _channel; - Role _role = Role::Admins; - int _offset = 0; - mtpRequestId _loadRequestId = 0; - bool _allLoaded = false; - Additional _additional; - QPointer _editBox; - base::lambda _doneCallback; - -}; - -// Finds channel members, then contacts, then global search results. -class AddParticipantBoxSearchController : public PeerListSearchController, private MTP::Sender { -public: - using Role = ParticipantsBoxController::Role; - using Additional = ParticipantsBoxController::Additional; - - AddParticipantBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional); - - void searchQuery(const QString &query) override; - bool isLoading() override; - bool loadMoreRows() override; - -private: - bool searchInCache(); - void searchOnServer(); - void searchDone(const MTPchannels_ChannelParticipants &result, mtpRequestId requestId); - - gsl::not_null _channel; - Role _role = Role::Admins; - gsl::not_null _additional; - - base::Timer _timer; - QString _query; - mtpRequestId _requestId = 0; - int _offset = 0; - bool _allLoaded = false; - std::map _cache; - std::map> _queries; // query, offset - -}; - -ParticipantsBoxController::ParticipantsBoxController(gsl::not_null channel, Role role) : PeerListController((role == Role::Admins) ? nullptr : std::make_unique(channel, role, &_additional)) -, _channel(channel) -, _role(role) { - if (_channel->mgInfo) { - _additional.creator = _channel->mgInfo->creator; - } -} - -void ParticipantsBoxController::Start(gsl::not_null channel, Role role) { - auto controller = std::make_unique(channel, role); - auto initBox = [role, channel, controller = controller.get()](PeerListBox *box) { - box->addButton(langFactory(lng_close), [box] { box->closeBox(); }); - auto buttonText = [role] { - switch (role) { - case Role::Admins: return langFactory(lng_channel_add_admin); - case Role::Restricted: return langFactory(lng_channel_add_restricted); - case Role::Kicked: return langFactory(lng_channel_add_banned); - } - Unexpected("Role value in ParticipantsBoxController::Start()"); - }; - if ((role == Role::Admins && channel->canAddAdmins()) || ((role == Role::Restricted || role == Role::Kicked) && channel->canBanMembers())) { - box->addLeftButton(buttonText(), [controller] { controller->addNewItem(); }); - } - }; - Ui::show(Box(std::move(controller), std::move(initBox)), KeepOtherLayers); -} - -void ParticipantsBoxController::addNewItem() { - auto weak = base::weak_unique_ptr(this); - _addBox = Ui::show(Box(std::make_unique(_channel, _role, [weak] { - if (weak && weak->_addBox) { - weak->_addBox->closeBox(); - } - }), [](PeerListBox *box) { - box->addButton(langFactory(lng_cancel), [box] { box->closeBox(); }); - }), KeepOtherLayers); -} - -void ParticipantsBoxController::peerListSearchAddRow(gsl::not_null peer) { - Expects(_role != Role::Admins); - PeerListController::peerListSearchAddRow(peer); - if (_role == Role::Restricted && delegate()->peerListFullRowsCount() > 0) { - setDescriptionText(QString()); - } -} - -template -void ParticipantsBoxController::HandleParticipant(const MTPChannelParticipant &participant, Role role, gsl::not_null additional, Callback callback) { - if (role == Role::Admins && participant.type() == mtpc_channelParticipantAdmin) { - auto &admin = participant.c_channelParticipantAdmin(); - if (auto user = App::userLoaded(admin.vuser_id.v)) { - additional->adminRights[user] = admin.vadmin_rights; - if (admin.is_can_edit()) { - additional->adminCanEdit.emplace(user); - } else { - additional->adminCanEdit.erase(user); - } - if (auto promoted = App::userLoaded(admin.vpromoted_by.v)) { - auto it = additional->adminPromotedBy.find(user); - if (it == additional->adminPromotedBy.end()) { - additional->adminPromotedBy.emplace(user, promoted); - } else { - it->second = promoted; - } - } else { - LOG(("API Error: No user %1 for admin promoted by.").arg(admin.vpromoted_by.v)); - } - callback(user); - } - } else if (role == Role::Admins && participant.type() == mtpc_channelParticipantCreator) { - auto &creator = participant.c_channelParticipantCreator(); - if (auto user = App::userLoaded(creator.vuser_id.v)) { - additional->creator = user; - callback(user); - } - } else if ((role == Role::Restricted || role == Role::Kicked) && participant.type() == mtpc_channelParticipantBanned) { - auto &banned = participant.c_channelParticipantBanned(); - if (auto user = App::userLoaded(banned.vuser_id.v)) { - additional->restrictedRights[user] = banned.vbanned_rights; - callback(user); - } - } else { - LOG(("API Error: Bad participant type got while requesting for participants: %1").arg(participant.type())); - } -} - -void ParticipantsBoxController::prepare() { - if (_role == Role::Admins) { - delegate()->peerListSetSearchMode(PeerListSearchMode::Local); - delegate()->peerListSetTitle(langFactory(lng_channel_admins)); - } else { - delegate()->peerListSetSearchMode(PeerListSearchMode::Complex); - delegate()->peerListSetTitle(langFactory((_role == Role::Restricted) ? lng_restricted_list_title : lng_banned_list_title)); - } - setDescriptionText(lang(lng_contacts_loading)); - setSearchNoResultsText(lang(lng_blocked_list_not_found)); - delegate()->peerListRefreshRows(); - - loadMoreRows(); -} - -void ParticipantsBoxController::loadMoreRows() { - if (searchController() && searchController()->loadMoreRows()) { - return; - } - if (_loadRequestId || _allLoaded) { - return; - } - - auto filter = [this] { - if (_role == Role::Admins) { - return MTP_channelParticipantsAdmins(); - } else if (_role == Role::Restricted) { - return MTP_channelParticipantsBanned(MTP_string(QString())); - } - return MTP_channelParticipantsKicked(MTP_string(QString())); - }; - auto perPage = (_role == Role::Admins) ? kAdminsPerPage : kBannedPerPage; - _loadRequestId = request(MTPchannels_GetParticipants(_channel->inputChannel, filter(), MTP_int(_offset), MTP_int(perPage))).done([this](const MTPchannels_ChannelParticipants &result) { - Expects(result.type() == mtpc_channels_channelParticipants); - - _loadRequestId = 0; - - if (!_offset) { - setDescriptionText((_role == Role::Restricted) ? lang(lng_group_blocked_list_about) : QString()); - } - auto &participants = result.c_channels_channelParticipants(); - App::feedUsers(participants.vusers); - - auto &list = participants.vparticipants.v; - if (list.isEmpty()) { - _allLoaded = true; - } else { - for_const (auto &participant, list) { - ++_offset; - HandleParticipant(participant, _role, &_additional, [this](gsl::not_null user) { - appendRow(user); - }); - } - } - delegate()->peerListRefreshRows(); - }).fail([this](const RPCError &error) { - _loadRequestId = 0; - }).send(); -} - -void ParticipantsBoxController::rowClicked(gsl::not_null row) { - Ui::showPeerHistoryAsync(row->peer()->id, ShowAtUnreadMsgId); -} - -void ParticipantsBoxController::rowActionClicked(gsl::not_null row) { - auto user = row->peer()->asUser(); - Expects(user != nullptr); - - if (_role == Role::Admins) { - editAdmin(user); - } else if (_role == Role::Restricted) { - editRestricted(user); - } else { - removeKicked(row, user); - } -} - -void ParticipantsBoxController::editAdmin(gsl::not_null user) { - if (_additional.adminCanEdit.find(user) == _additional.adminCanEdit.end()) { - return; - } - - auto it = _additional.adminRights.find(user); - t_assert(it != _additional.adminRights.cend()); - auto weak = base::weak_unique_ptr(this); - _editBox = Ui::show(Box(_channel, user, it->second, [megagroup = _channel.get(), user, weak](const MTPChannelAdminRights &rights) { - MTP::send(MTPchannels_EditAdmin(megagroup->inputChannel, user->inputUser, rights), rpcDone([megagroup, user, weak, rights](const MTPUpdates &result) { - if (App::main()) App::main()->sentUpdatesReceived(result); - megagroup->applyEditAdmin(user, rights); - if (weak) { - weak->editAdminDone(user, rights); - } - })); - }), KeepOtherLayers); -} - -void ParticipantsBoxController::editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights) { - if (_editBox) _editBox->closeBox(); - if (rights.c_channelAdminRights().vflags.v == 0) { - if (auto row = delegate()->peerListFindRow(user->id)) { - delegate()->peerListRemoveRow(row); - if (!delegate()->peerListFullRowsCount()) { - setDescriptionText(lang(lng_blocked_list_not_found)); - } - delegate()->peerListRefreshRows(); - } - } else { - _additional.adminRights[user] = rights; - } -} - -void ParticipantsBoxController::editRestricted(gsl::not_null user) { - auto it = _additional.restrictedRights.find(user); - t_assert(it != _additional.restrictedRights.cend()); - auto weak = base::weak_unique_ptr(this); - _editBox = Ui::show(Box(_channel, user, it->second, [megagroup = _channel.get(), user, weak](const MTPChannelBannedRights &rights) { - MTP::send(MTPchannels_EditBanned(megagroup->inputChannel, user->inputUser, rights), rpcDone([megagroup, user, weak, rights](const MTPUpdates &result) { - if (App::main()) App::main()->sentUpdatesReceived(result); - megagroup->applyEditBanned(user, rights); - if (weak) { - weak->editRestrictedDone(user, rights); - } - })); - }), KeepOtherLayers); -} - -void ParticipantsBoxController::editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights) { - if (_editBox) _editBox->closeBox(); - if (rights.c_channelBannedRights().vflags.v == 0 || rights.c_channelBannedRights().is_view_messages()) { - if (auto row = delegate()->peerListFindRow(user->id)) { - delegate()->peerListRemoveRow(row); - if (!delegate()->peerListFullRowsCount()) { - setDescriptionText(lang(lng_blocked_list_not_found)); - } - delegate()->peerListRefreshRows(); - } - } else { - _additional.restrictedRights[user] = rights; - } -} - -void ParticipantsBoxController::removeKicked(gsl::not_null row, gsl::not_null user) { - delegate()->peerListRemoveRow(row); - delegate()->peerListRefreshRows(); - - AuthSession::Current().api().unblockParticipant(_channel, user); -} - -bool ParticipantsBoxController::appendRow(gsl::not_null user) { - if (delegate()->peerListFindRow(user->id)) { - return false; - } - delegate()->peerListAppendRow(createRow(user)); - if (_role != Role::Kicked) { - setDescriptionText(QString()); - } - return true; -} - -bool ParticipantsBoxController::prependRow(gsl::not_null user) { - if (delegate()->peerListFindRow(user->id)) { - return false; - } - delegate()->peerListPrependRow(createRow(user)); - if (_role != Role::Kicked) { - setDescriptionText(QString()); - } - return true; -} - -std::unique_ptr ParticipantsBoxController::createRow(gsl::not_null user) const { - auto row = std::make_unique(user); - if (_role == Role::Admins) { - auto promotedBy = _additional.adminPromotedBy.find(user); - if (promotedBy == _additional.adminPromotedBy.end()) { - row->setCustomStatus(lang(lng_channel_admin_status_creator)); - } else { - row->setCustomStatus(lng_channel_admin_status_promoted_by(lt_user, App::peerName(promotedBy->second))); - } - } - if (_role == Role::Restricted || (_role == Role::Admins && _additional.adminCanEdit.find(user) != _additional.adminCanEdit.end())) { - row->setActionLink(lang(lng_profile_edit_permissions)); - } else if (_role == Role::Kicked) { - row->setActionLink(lang(lng_blocked_list_unblock)); - } - return std::move(row); -} - -BannedBoxSearchController::BannedBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional) -: _channel(channel) -, _role(role) -, _additional(additional) { - Expects(role != Role::Admins); - _timer.setCallback([this] { searchOnServer(); }); -} - -void BannedBoxSearchController::searchQuery(const QString &query) { - if (_query != query) { - _query = query; - _offset = 0; - _requestId = 0; - _allLoaded = false; - if (!_query.isEmpty() && !searchInCache()) { - _timer.callOnce(AutoSearchTimeout); - } else { - _timer.cancel(); - } - } -} - -bool BannedBoxSearchController::searchInCache() { - auto it = _cache.find(_query); - if (it != _cache.cend()) { - _requestId = 0; - searchDone(it->second, _requestId); - return true; - } - return false; -} - -void BannedBoxSearchController::searchOnServer() { - Expects(!_query.isEmpty()); - loadMoreRows(); -} - -void BannedBoxSearchController::searchDone(const MTPchannels_ChannelParticipants &result, mtpRequestId requestId) { - Expects(result.type() == mtpc_channels_channelParticipants); - - auto &participants = result.c_channels_channelParticipants(); - auto query = _query; - if (requestId) { - App::feedUsers(participants.vusers); - auto it = _queries.find(requestId); - if (it != _queries.cend()) { - query = it->second.first; // query - if (it->second.second == 0) { // offset - _cache[query] = result; - } - _queries.erase(it); - } - } - - if (_requestId == requestId) { - _requestId = 0; - auto &list = participants.vparticipants.v; - if (list.isEmpty()) { - _allLoaded = true; - } else { - for_const (auto &participant, list) { - ++_offset; - ParticipantsBoxController::HandleParticipant(participant, _role, _additional, [this](gsl::not_null user) { - delegate()->peerListSearchAddRow(user); - }); - } - } - delegate()->peerListSearchRefreshRows(); - } -} - -bool BannedBoxSearchController::isLoading() { - return _timer.isActive() || _requestId; -} - -bool BannedBoxSearchController::loadMoreRows() { - if (_query.isEmpty()) { - return false; - } - if (!_allLoaded && !isLoading()) { - auto filter = (_role == Role::Restricted) ? MTP_channelParticipantsBanned(MTP_string(_query)) : MTP_channelParticipantsKicked(MTP_string(_query)); - _requestId = request(MTPchannels_GetParticipants(_channel->inputChannel, filter, MTP_int(_offset), MTP_int(kBannedPerPage))).done([this](const MTPchannels_ChannelParticipants &result, mtpRequestId requestId) { - searchDone(result, requestId); - }).fail([this](const RPCError &error, mtpRequestId requestId) { - if (_requestId == requestId) { - _requestId = 0; - _allLoaded = true; - delegate()->peerListSearchRefreshRows(); - } - }).send(); - _queries.emplace(_requestId, std::make_pair(_query, _offset)); - } - return true; -} - -AddParticipantBoxController::AddParticipantBoxController(gsl::not_null channel, Role role, base::lambda doneCallback) -: _channel(channel) -, _role(role) -, _doneCallback(std::move(doneCallback)) { - if (_channel->mgInfo) { - _additional.creator = _channel->mgInfo->creator; - } -} - -void AddParticipantBoxController::peerListSearchAddRow(gsl::not_null peer) { - if (peer->isSelf()) { - return; - } - PeerListController::peerListSearchAddRow(peer); -} - -void AddParticipantBoxController::prepare() { - delegate()->peerListSetSearchMode(PeerListSearchMode::Complex); - auto title = [this] { - switch (_role) { - case Role::Admins: return langFactory(lng_channel_add_admin); - case Role::Restricted: return langFactory(lng_channel_add_restricted); - case Role::Kicked: return langFactory(lng_channel_add_banned); - } - Unexpected("Role in AddParticipantBoxController::prepare()"); - }; - delegate()->peerListSetTitle(title()); - setDescriptionText(lang(lng_contacts_loading)); - setSearchNoResultsText(lang(lng_blocked_list_not_found)); - delegate()->peerListRefreshRows(); - - loadMoreRows(); -} - -void AddParticipantBoxController::loadMoreRows() { - if (searchController() && searchController()->loadMoreRows()) { - return; - } - if (_loadRequestId || _allLoaded) { - return; - } - - _loadRequestId = request(MTPchannels_GetParticipants(_channel->inputChannel, MTP_channelParticipantsRecent(), MTP_int(_offset), MTP_int(kParticipantsPerPage))).done([this](const MTPchannels_ChannelParticipants &result) { - Expects(result.type() == mtpc_channels_channelParticipants); - - _loadRequestId = 0; - - auto &participants = result.c_channels_channelParticipants(); - App::feedUsers(participants.vusers); - - auto &list = participants.vparticipants.v; - if (list.isEmpty()) { - _allLoaded = true; - } else { - for_const (auto &participant, list) { - ++_offset; - HandleParticipant(participant, &_additional, [this](gsl::not_null user) { - appendRow(user); - }); - } - } - if (delegate()->peerListFullRowsCount() > 0) { - setDescriptionText(QString()); - } - delegate()->peerListRefreshRows(); - }).fail([this](const RPCError &error) { - _loadRequestId = 0; - }).send(); -} - -void AddParticipantBoxController::rowClicked(gsl::not_null row) { - auto user = row->peer()->asUser(); - switch (_role) { - case Role::Admins: return editAdmin(user); - case Role::Restricted: return editRestricted(user); - case Role::Kicked: return kickUser(user); - } - Unexpected("Role in AddParticipantBoxController::rowClicked()"); -} - -template -bool AddParticipantBoxController::checkInfoLoaded(gsl::not_null user, Callback callback) { - if (_additional.infoNotLoaded.find(user) == _additional.infoNotLoaded.end()) { - return true; - } - - // We don't know what this user status is in the group. - request(MTPchannels_GetParticipant(_channel->inputChannel, user->inputUser)).done([this, user, callback](const MTPchannels_ChannelParticipant &result) { - Expects(result.type() == mtpc_channels_channelParticipant); - auto &participant = result.c_channels_channelParticipant(); - App::feedUsers(participant.vusers); - HandleParticipant(participant.vparticipant, &_additional, [](gsl::not_null) {}); - _additional.infoNotLoaded.erase(user); - callback(); - }).fail([this, user, callback](const RPCError &error) { - _additional.infoNotLoaded.erase(user); - _additional.external.emplace(user); - callback(); - }).send(); - return false; -} - -void AddParticipantBoxController::editAdmin(gsl::not_null user, bool sure) { - if (!checkInfoLoaded(user, [this, user] { editAdmin(user); })) { - return; - } - - if (sure && _editBox) { - // Close the confirmation box. - _editBox->closeBox(); - } - - // Check restrictions. - auto weak = base::weak_unique_ptr(this); - auto alreadyIt = _additional.adminRights.find(user); - auto currentRights = EditAdminBox::DefaultRights(_channel); - if (alreadyIt != _additional.adminRights.end() || _additional.creator == user) { - // The user is already an admin. - if (_additional.adminCanEdit.find(user) == _additional.adminCanEdit.end() || _additional.creator == user) { - Ui::show(Box(lang(lng_error_cant_edit_admin)), KeepOtherLayers); - return; - } - currentRights = alreadyIt->second; - } else if (_additional.restrictedRights.find(user) != _additional.restrictedRights.end() || _additional.kicked.find(user) != _additional.kicked.end()) { - // The user is banned or restricted. - if (_channel->canBanMembers()) { - if (!sure) { - _editBox = Ui::show(Box(lang(lng_sure_add_admin_unban), [weak, user] { weak->editAdmin(user, true); }), KeepOtherLayers); - return; - } - } else { - Ui::show(Box(lang(lng_error_cant_add_admin_unban)), KeepOtherLayers); - return; - } - } else if (_additional.external.find(user) != _additional.external.end()) { - // The user is not in the group yet. - if (_channel->canAddMembers()) { - if (!sure) { - _editBox = Ui::show(Box(lang(lng_sure_add_admin_invite), [weak, user] { weak->editAdmin(user, true); }), KeepOtherLayers); - return; - } - } else { - Ui::show(Box(lang(lng_error_cant_add_admin_invite)), KeepOtherLayers); - return; - } - } - - // Finally edit the admin. - _editBox = Ui::show(Box(_channel, user, currentRights, [megagroup = _channel.get(), user, weak](const MTPChannelAdminRights &rights) { - MTP::send(MTPchannels_EditAdmin(megagroup->inputChannel, user->inputUser, rights), rpcDone([megagroup, user, weak, rights](const MTPUpdates &result) { - if (App::main()) App::main()->sentUpdatesReceived(result); - megagroup->applyEditAdmin(user, rights); - if (weak) { - weak->editAdminDone(user, rights); - } - })); - }), KeepOtherLayers); -} - -void AddParticipantBoxController::editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights) { - if (_editBox) _editBox->closeBox(); - _additional.restrictedRights.erase(user); - _additional.kicked.erase(user); - _additional.external.erase(user); - if (rights.c_channelAdminRights().vflags.v == 0) { - _additional.adminRights.erase(user); - _additional.adminPromotedBy.erase(user); - _additional.adminCanEdit.erase(user); - } else { - _additional.adminRights[user] = rights; - _additional.adminCanEdit.emplace(user); - auto it = _additional.adminPromotedBy.find(user); - if (it == _additional.adminPromotedBy.end()) { - _additional.adminPromotedBy.emplace(user, App::self()); - } - } - _doneCallback(); -} - -void AddParticipantBoxController::editRestricted(gsl::not_null user, bool sure) { - if (!checkInfoLoaded(user, [this, user] { editRestricted(user); })) { - return; - } - - if (sure && _editBox) { - // Close the confirmation box. - _editBox->closeBox(); - } - - // Check restrictions. - auto weak = base::weak_unique_ptr(this); - auto alreadyIt = _additional.restrictedRights.find(user); - auto currentRights = EditRestrictedBox::DefaultRights(_channel); - if (alreadyIt != _additional.restrictedRights.end()) { - // The user is already banned or restricted. - currentRights = alreadyIt->second; - } else if (_additional.adminRights.find(user) != _additional.adminRights.end() || _additional.creator == user) { - // The user is an admin or creator. - if (_additional.adminCanEdit.find(user) != _additional.adminCanEdit.end()) { - if (!sure) { - _editBox = Ui::show(Box(lang(lng_sure_ban_admin), [weak, user] { weak->editRestricted(user, true); }), KeepOtherLayers); - return; - } - } else { - Ui::show(Box(lang(lng_error_cant_ban_admin)), KeepOtherLayers); - return; - } - } - - // Finally edit the restricted. - _editBox = Ui::show(Box(_channel, user, currentRights, [user, weak](const MTPChannelBannedRights &rights) { - if (weak) { - weak->restrictUserSure(user, rights); - } - }), KeepOtherLayers); -} - -void AddParticipantBoxController::restrictUserSure(gsl::not_null user, const MTPChannelBannedRights &rights) { - auto weak = base::weak_unique_ptr(this); - MTP::send(MTPchannels_EditBanned(_channel->inputChannel, user->inputUser, rights), rpcDone([megagroup = _channel.get(), user, weak, rights](const MTPUpdates &result) { - if (App::main()) App::main()->sentUpdatesReceived(result); - megagroup->applyEditBanned(user, rights); - if (weak) { - weak->editRestrictedDone(user, rights); - } - })); -} - -void AddParticipantBoxController::editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights) { - if (_editBox) _editBox->closeBox(); - _additional.adminRights.erase(user); - _additional.adminCanEdit.erase(user); - _additional.adminPromotedBy.erase(user); - if (rights.c_channelBannedRights().vflags.v == 0) { - _additional.restrictedRights.erase(user); - _additional.kicked.erase(user); - } else if (rights.c_channelBannedRights().vflags.v & MTPDchannelBannedRights::Flag::f_view_messages) { - _additional.restrictedRights.erase(user); - _additional.kicked.emplace(user); - } else { - _additional.restrictedRights[user] = rights; - _additional.kicked.erase(user); - } - _doneCallback(); -} - -void AddParticipantBoxController::kickUser(gsl::not_null user, bool sure) { - if (!checkInfoLoaded(user, [this, user] { kickUser(user); })) { - return; - } - - // Check restrictions. - auto weak = base::weak_unique_ptr(this); - if (_additional.adminRights.find(user) != _additional.adminRights.end() || _additional.creator == user) { - // The user is an admin or creator. - if (_additional.adminCanEdit.find(user) != _additional.adminCanEdit.end()) { - if (!sure) { - Ui::show(Box(lang(lng_sure_ban_admin), [weak, user] { weak->kickUser(user, true); }), KeepOtherLayers); - return; - } - } else { - Ui::show(Box(lang(lng_error_cant_ban_admin)), KeepOtherLayers); - return; - } - } - - // Finally kick him. - if (!sure) { - auto text = lng_sure_ban_user_group(lt_user, App::peerName(user)); - Ui::show(Box(text, [weak, user] { weak->kickUser(user, true); }), KeepOtherLayers); - return; - } - restrictUserSure(user, ChannelData::KickedRestrictedRights()); -} - -bool AddParticipantBoxController::appendRow(gsl::not_null user) { - if (delegate()->peerListFindRow(user->id) || user->isSelf()) { - return false; - } - delegate()->peerListAppendRow(createRow(user)); - return true; -} - -bool AddParticipantBoxController::prependRow(gsl::not_null user) { - if (delegate()->peerListFindRow(user->id)) { - return false; - } - delegate()->peerListPrependRow(createRow(user)); - return true; -} - -std::unique_ptr AddParticipantBoxController::createRow(gsl::not_null user) const { - auto row = std::make_unique(user); - return std::move(row); -} - -template -void AddParticipantBoxController::HandleParticipant(const MTPChannelParticipant &participant, gsl::not_null additional, Callback callback) { - switch (participant.type()) { - case mtpc_channelParticipantAdmin: { - auto &admin = participant.c_channelParticipantAdmin(); - if (auto user = App::userLoaded(admin.vuser_id.v)) { - additional->infoNotLoaded.erase(user); - additional->restrictedRights.erase(user); - additional->kicked.erase(user); - additional->adminRights[user] = admin.vadmin_rights; - if (admin.is_can_edit()) { - additional->adminCanEdit.emplace(user); - } else { - additional->adminCanEdit.erase(user); - } - if (auto promoted = App::userLoaded(admin.vpromoted_by.v)) { - auto it = additional->adminPromotedBy.find(user); - if (it == additional->adminPromotedBy.end()) { - additional->adminPromotedBy.emplace(user, promoted); - } else { - it->second = promoted; - } - } else { - LOG(("API Error: No user %1 for admin promoted by.").arg(admin.vpromoted_by.v)); - } - callback(user); - } - } break; - case mtpc_channelParticipantCreator: { - auto &creator = participant.c_channelParticipantCreator(); - if (auto user = App::userLoaded(creator.vuser_id.v)) { - additional->infoNotLoaded.erase(user); - additional->creator = user; - callback(user); - } - } break; - case mtpc_channelParticipantBanned: { - auto &banned = participant.c_channelParticipantBanned(); - if (auto user = App::userLoaded(banned.vuser_id.v)) { - additional->infoNotLoaded.erase(user); - additional->adminRights.erase(user); - additional->adminCanEdit.erase(user); - additional->adminPromotedBy.erase(user); - if (banned.is_left()) { - additional->kicked.emplace(user); - } else { - additional->kicked.erase(user); - } - additional->restrictedRights[user] = banned.vbanned_rights; - callback(user); - } - } break; - case mtpc_channelParticipant: { - auto &data = participant.c_channelParticipant(); - if (auto user = App::userLoaded(data.vuser_id.v)) { - additional->infoNotLoaded.erase(user); - additional->adminRights.erase(user); - additional->adminCanEdit.erase(user); - additional->adminPromotedBy.erase(user); - additional->restrictedRights.erase(user); - additional->kicked.erase(user); - callback(user); - } - } break; - default: Unexpected("Participant type in AddParticipantBoxController::HandleParticipant()"); - } -} - -AddParticipantBoxSearchController::AddParticipantBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional) -: _channel(channel) -, _role(role) -, _additional(additional) { - _timer.setCallback([this] { searchOnServer(); }); -} - -void AddParticipantBoxSearchController::searchQuery(const QString &query) { - if (_query != query) { - _query = query; - _offset = 0; - _requestId = 0; - _allLoaded = false; - if (!_query.isEmpty() && !searchInCache()) { - _timer.callOnce(AutoSearchTimeout); - } else { - _timer.cancel(); - } - } -} - -bool AddParticipantBoxSearchController::searchInCache() { - auto it = _cache.find(_query); - if (it != _cache.cend()) { - _requestId = 0; - searchDone(it->second, _requestId); - return true; - } - return false; -} - -void AddParticipantBoxSearchController::searchOnServer() { - Expects(!_query.isEmpty()); - loadMoreRows(); -} - -void AddParticipantBoxSearchController::searchDone(const MTPchannels_ChannelParticipants &result, mtpRequestId requestId) { - Expects(result.type() == mtpc_channels_channelParticipants); - - auto &participants = result.c_channels_channelParticipants(); - auto query = _query; - if (requestId) { - App::feedUsers(participants.vusers); - auto it = _queries.find(requestId); - if (it != _queries.cend()) { - query = it->second.first; // query - if (it->second.second == 0) { // offset - _cache[query] = result; - } - _queries.erase(it); - } - } - - if (_requestId == requestId) { - _requestId = 0; - auto &list = participants.vparticipants.v; - if (list.isEmpty()) { - _allLoaded = true; - } else { - for_const (auto &participant, list) { - ++_offset; - AddParticipantBoxController::HandleParticipant(participant, _additional, [this](gsl::not_null user) { - delegate()->peerListSearchAddRow(user); - }); - } - } - delegate()->peerListSearchRefreshRows(); - } -} - -bool AddParticipantBoxSearchController::isLoading() { - return _timer.isActive() || _requestId; -} - -bool AddParticipantBoxSearchController::loadMoreRows() { - if (_query.isEmpty()) { - return false; - } - if (!_allLoaded && !isLoading()) { - _requestId = request(MTPchannels_GetParticipants(_channel->inputChannel, MTP_channelParticipantsSearch(MTP_string(_query)), MTP_int(_offset), MTP_int(kBannedPerPage))).done([this](const MTPchannels_ChannelParticipants &result, mtpRequestId requestId) { - searchDone(result, requestId); - }).fail([this](const RPCError &error, mtpRequestId requestId) { - if (_requestId == requestId) { - _requestId = 0; - _allLoaded = true; - delegate()->peerListSearchRefreshRows(); - } - }).send(); - _queries.emplace(_requestId, std::make_pair(_query, _offset)); - } - return true; -} - -} // namespace using UpdateFlag = Notify::PeerUpdate::Flag; diff --git a/Telegram/SourceFiles/profile/profile_channel_controllers.cpp b/Telegram/SourceFiles/profile/profile_channel_controllers.cpp new file mode 100644 index 000000000..60c98fbd7 --- /dev/null +++ b/Telegram/SourceFiles/profile/profile_channel_controllers.cpp @@ -0,0 +1,1095 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#include "profile/profile_channel_controllers.h" + +#include "boxes/edit_participant_box.h" +#include "boxes/confirm_box.h" +#include "auth_session.h" +#include "apiwrap.h" +#include "lang/lang_keys.h" +#include "mainwidget.h" +#include "dialogs/dialogs_indexed_list.h" + +namespace Profile { +namespace { + +constexpr auto kParticipantsFirstPageCount = 16; +constexpr auto kParticipantsPerPage = 200; + +} // namespace + +ParticipantsBoxController::ParticipantsBoxController(gsl::not_null channel, Role role) : PeerListController((role == Role::Admins) ? nullptr : std::make_unique(channel, role, &_additional)) +, _channel(channel) +, _role(role) { + if (_channel->mgInfo) { + _additional.creator = _channel->mgInfo->creator; + } +} + +void ParticipantsBoxController::Start(gsl::not_null channel, Role role) { + auto controller = std::make_unique(channel, role); + auto initBox = [role, channel, controller = controller.get()](PeerListBox *box) { + box->addButton(langFactory(lng_close), [box] { box->closeBox(); }); + auto canAddNewItem = [role, channel] { + switch (role) { + case Role::Admins: return channel->canAddAdmins(); + case Role::Restricted: + case Role::Kicked: return channel->canBanMembers(); + } + Unexpected("Role value in ParticipantsBoxController::Start()"); + }; + auto addNewItemText = [role] { + switch (role) { + case Role::Admins: return langFactory(lng_channel_add_admin); + case Role::Restricted: return langFactory(lng_channel_add_restricted); + case Role::Kicked: return langFactory(lng_channel_add_banned); + } + Unexpected("Role value in ParticipantsBoxController::Start()"); + }; + if (canAddNewItem()) { + box->addLeftButton(addNewItemText(), [controller] { controller->addNewItem(); }); + } + }; + Ui::show(Box(std::move(controller), std::move(initBox)), KeepOtherLayers); +} + +void ParticipantsBoxController::addNewItem() { + auto weak = base::weak_unique_ptr(this); + _addBox = Ui::show(Box(std::make_unique(_channel, _role, [weak] { + if (weak && weak->_addBox) { + weak->_addBox->closeBox(); + } + }), [](PeerListBox *box) { + box->addButton(langFactory(lng_cancel), [box] { box->closeBox(); }); + }), KeepOtherLayers); +} + +void ParticipantsBoxController::peerListSearchAddRow(gsl::not_null peer) { + Expects(_role != Role::Admins); + PeerListController::peerListSearchAddRow(peer); + if (_role == Role::Restricted && delegate()->peerListFullRowsCount() > 0) { + setDescriptionText(QString()); + } +} + +std::unique_ptr ParticipantsBoxController::createSearchRow(gsl::not_null peer) { + if (auto user = peer->asUser()) { + return createRow(user); + } + return std::unique_ptr(); +} + +template +void ParticipantsBoxController::HandleParticipant(const MTPChannelParticipant &participant, Role role, gsl::not_null additional, Callback callback) { + if (role == Role::Admins && participant.type() == mtpc_channelParticipantAdmin) { + auto &admin = participant.c_channelParticipantAdmin(); + if (auto user = App::userLoaded(admin.vuser_id.v)) { + additional->adminRights[user] = admin.vadmin_rights; + if (admin.is_can_edit()) { + additional->adminCanEdit.emplace(user); + } else { + additional->adminCanEdit.erase(user); + } + if (auto promoted = App::userLoaded(admin.vpromoted_by.v)) { + auto it = additional->adminPromotedBy.find(user); + if (it == additional->adminPromotedBy.end()) { + additional->adminPromotedBy.emplace(user, promoted); + } else { + it->second = promoted; + } + } else { + LOG(("API Error: No user %1 for admin promoted by.").arg(admin.vpromoted_by.v)); + } + callback(user); + } + } else if (role == Role::Admins && participant.type() == mtpc_channelParticipantCreator) { + auto &creator = participant.c_channelParticipantCreator(); + if (auto user = App::userLoaded(creator.vuser_id.v)) { + additional->creator = user; + callback(user); + } + } else if ((role == Role::Restricted || role == Role::Kicked) && participant.type() == mtpc_channelParticipantBanned) { + auto &banned = participant.c_channelParticipantBanned(); + if (auto user = App::userLoaded(banned.vuser_id.v)) { + additional->restrictedRights[user] = banned.vbanned_rights; + callback(user); + } + } else { + LOG(("API Error: Bad participant type got while requesting for participants: %1").arg(participant.type())); + } +} + +void ParticipantsBoxController::prepare() { + if (_role == Role::Admins) { + delegate()->peerListSetSearchMode(PeerListSearchMode::Local); + delegate()->peerListSetTitle(langFactory(lng_channel_admins)); + } else { + delegate()->peerListSetSearchMode(PeerListSearchMode::Complex); + delegate()->peerListSetTitle(langFactory((_role == Role::Restricted) ? lng_restricted_list_title : lng_banned_list_title)); + } + setDescriptionText(lang(lng_contacts_loading)); + setSearchNoResultsText(lang(lng_blocked_list_not_found)); + delegate()->peerListRefreshRows(); + + loadMoreRows(); +} + +void ParticipantsBoxController::loadMoreRows() { + if (searchController() && searchController()->loadMoreRows()) { + return; + } + if (_loadRequestId || _allLoaded) { + return; + } + + auto filter = [this] { + if (_role == Role::Admins) { + return MTP_channelParticipantsAdmins(); + } else if (_role == Role::Restricted) { + return MTP_channelParticipantsBanned(MTP_string(QString())); + } + return MTP_channelParticipantsKicked(MTP_string(QString())); + }; + + // First query is small and fast, next loads a lot of rows. + auto perPage = (_offset > 0) ? kParticipantsPerPage : kParticipantsFirstPageCount; + _loadRequestId = request(MTPchannels_GetParticipants(_channel->inputChannel, filter(), MTP_int(_offset), MTP_int(perPage))).done([this](const MTPchannels_ChannelParticipants &result) { + Expects(result.type() == mtpc_channels_channelParticipants); + + _loadRequestId = 0; + + if (!_offset) { + setDescriptionText((_role == Role::Restricted) ? lang(lng_group_blocked_list_about) : QString()); + } + auto &participants = result.c_channels_channelParticipants(); + App::feedUsers(participants.vusers); + + auto &list = participants.vparticipants.v; + if (list.isEmpty()) { + // To be sure - wait for a whole empty result list. + _allLoaded = true; + } else { + for_const (auto &participant, list) { + ++_offset; + HandleParticipant(participant, _role, &_additional, [this](gsl::not_null user) { + appendRow(user); + }); + } + } + delegate()->peerListRefreshRows(); + }).fail([this](const RPCError &error) { + _loadRequestId = 0; + }).send(); +} + +void ParticipantsBoxController::rowClicked(gsl::not_null row) { + Ui::showPeerHistoryAsync(row->peer()->id, ShowAtUnreadMsgId); +} + +void ParticipantsBoxController::rowActionClicked(gsl::not_null row) { + auto user = row->peer()->asUser(); + Expects(user != nullptr); + + if (_role == Role::Admins) { + editAdmin(user); + } else if (_role == Role::Restricted) { + editRestricted(user); + } else { + removeKicked(row, user); + } +} + +void ParticipantsBoxController::editAdmin(gsl::not_null user) { + if (_additional.adminCanEdit.find(user) == _additional.adminCanEdit.end()) { + return; + } + + auto it = _additional.adminRights.find(user); + t_assert(it != _additional.adminRights.cend()); + auto weak = base::weak_unique_ptr(this); + _editBox = Ui::show(Box(_channel, user, it->second, [megagroup = _channel.get(), user, weak](const MTPChannelAdminRights &rights) { + MTP::send(MTPchannels_EditAdmin(megagroup->inputChannel, user->inputUser, rights), rpcDone([megagroup, user, weak, rights](const MTPUpdates &result) { + AuthSession::Current().api().applyUpdates(result); + megagroup->applyEditAdmin(user, rights); + if (weak) { + weak->editAdminDone(user, rights); + } + })); + }), KeepOtherLayers); +} + +void ParticipantsBoxController::editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights) { + if (_editBox) _editBox->closeBox(); + if (rights.c_channelAdminRights().vflags.v == 0) { + if (auto row = delegate()->peerListFindRow(user->id)) { + delegate()->peerListRemoveRow(row); + if (!delegate()->peerListFullRowsCount()) { + setDescriptionText(lang(lng_blocked_list_not_found)); + } + delegate()->peerListRefreshRows(); + } + } else { + _additional.adminRights[user] = rights; + } +} + +void ParticipantsBoxController::editRestricted(gsl::not_null user) { + auto it = _additional.restrictedRights.find(user); + t_assert(it != _additional.restrictedRights.cend()); + auto weak = base::weak_unique_ptr(this); + _editBox = Ui::show(Box(_channel, user, it->second, [megagroup = _channel.get(), user, weak](const MTPChannelBannedRights &rights) { + MTP::send(MTPchannels_EditBanned(megagroup->inputChannel, user->inputUser, rights), rpcDone([megagroup, user, weak, rights](const MTPUpdates &result) { + AuthSession::Current().api().applyUpdates(result); + megagroup->applyEditBanned(user, rights); + if (weak) { + weak->editRestrictedDone(user, rights); + } + })); + }), KeepOtherLayers); +} + +void ParticipantsBoxController::editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights) { + if (_editBox) _editBox->closeBox(); + if (rights.c_channelBannedRights().vflags.v == 0 || rights.c_channelBannedRights().is_view_messages()) { + if (auto row = delegate()->peerListFindRow(user->id)) { + delegate()->peerListRemoveRow(row); + if (!delegate()->peerListFullRowsCount()) { + setDescriptionText(lang(lng_blocked_list_not_found)); + } + delegate()->peerListRefreshRows(); + } + } else { + _additional.restrictedRights[user] = rights; + } +} + +void ParticipantsBoxController::removeKicked(gsl::not_null row, gsl::not_null user) { + delegate()->peerListRemoveRow(row); + delegate()->peerListRefreshRows(); + + AuthSession::Current().api().unblockParticipant(_channel, user); +} + +bool ParticipantsBoxController::appendRow(gsl::not_null user) { + if (delegate()->peerListFindRow(user->id)) { + return false; + } + delegate()->peerListAppendRow(createRow(user)); + if (_role != Role::Kicked) { + setDescriptionText(QString()); + } + return true; +} + +bool ParticipantsBoxController::prependRow(gsl::not_null user) { + if (delegate()->peerListFindRow(user->id)) { + return false; + } + delegate()->peerListPrependRow(createRow(user)); + if (_role != Role::Kicked) { + setDescriptionText(QString()); + } + return true; +} + +std::unique_ptr ParticipantsBoxController::createRow(gsl::not_null user) const { + auto row = std::make_unique(user); + if (_role == Role::Admins) { + auto promotedBy = _additional.adminPromotedBy.find(user); + if (promotedBy == _additional.adminPromotedBy.end()) { + row->setCustomStatus(lang(lng_channel_admin_status_creator)); + } else { + row->setCustomStatus(lng_channel_admin_status_promoted_by(lt_user, App::peerName(promotedBy->second))); + } + } + if (_role == Role::Restricted || (_role == Role::Admins && _additional.adminCanEdit.find(user) != _additional.adminCanEdit.end())) { + row->setActionLink(lang(lng_profile_edit_permissions)); + } else if (_role == Role::Kicked) { + row->setActionLink(lang(lng_blocked_list_unblock)); + } + return std::move(row); +} + +BannedBoxSearchController::BannedBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional) +: _channel(channel) +, _role(role) +, _additional(additional) { + Expects(role != Role::Admins); + _timer.setCallback([this] { searchOnServer(); }); +} + +void BannedBoxSearchController::searchQuery(const QString &query) { + if (_query != query) { + _query = query; + _offset = 0; + _requestId = 0; + _allLoaded = false; + if (!_query.isEmpty() && !searchInCache()) { + _timer.callOnce(AutoSearchTimeout); + } else { + _timer.cancel(); + } + } +} + +void BannedBoxSearchController::searchOnServer() { + Expects(!_query.isEmpty()); + loadMoreRows(); +} + +bool BannedBoxSearchController::isLoading() { + return _timer.isActive() || _requestId; +} + +bool BannedBoxSearchController::searchInCache() { + auto it = _cache.find(_query); + if (it != _cache.cend()) { + _requestId = 0; + searchDone(_requestId, it->second.result, it->second.requestedCount); + return true; + } + return false; +} + +bool BannedBoxSearchController::loadMoreRows() { + if (_query.isEmpty()) { + return false; + } + if (!_allLoaded && !isLoading()) { + auto filter = (_role == Role::Restricted) ? MTP_channelParticipantsBanned(MTP_string(_query)) : MTP_channelParticipantsKicked(MTP_string(_query)); + + // For search we request a lot of rows from the first query. + // (because we've waited for search request by timer already, + // so we don't expect it to be fast, but we want to fill cache). + auto perPage = kParticipantsPerPage; + _requestId = request(MTPchannels_GetParticipants(_channel->inputChannel, filter, MTP_int(_offset), MTP_int(perPage))).done([this, perPage](const MTPchannels_ChannelParticipants &result, mtpRequestId requestId) { + searchDone(requestId, result, perPage); + }).fail([this](const RPCError &error, mtpRequestId requestId) { + if (_requestId == requestId) { + _requestId = 0; + _allLoaded = true; + delegate()->peerListSearchRefreshRows(); + } + }).send(); + + auto entry = Query(); + entry.text = _query; + entry.offset = _offset; + _queries.emplace(_requestId, entry); + } + return true; +} + +void BannedBoxSearchController::searchDone(mtpRequestId requestId, const MTPchannels_ChannelParticipants &result, int requestedCount) { + Expects(result.type() == mtpc_channels_channelParticipants); + + auto &participants = result.c_channels_channelParticipants(); + auto query = _query; + if (requestId) { + App::feedUsers(participants.vusers); + auto it = _queries.find(requestId); + if (it != _queries.cend()) { + query = it->second.text; + if (it->second.offset == 0) { + auto &entry = _cache[query]; + entry.result = result; + entry.requestedCount = requestedCount; + } + _queries.erase(it); + } + } + + if (_requestId == requestId) { + _requestId = 0; + auto &list = participants.vparticipants.v; + if (list.size() < requestedCount) { + // We want cache to have full information about a query with small + // results count (if we don't need the second request). So we don't + // wait for an empty results list unlike the non-search peer list. + _allLoaded = true; + } + for_const (auto &participant, list) { + ParticipantsBoxController::HandleParticipant(participant, _role, _additional, [this](gsl::not_null user) { + delegate()->peerListSearchAddRow(user); + }); + } + _offset += list.size(); + delegate()->peerListSearchRefreshRows(); + } +} + +AddParticipantBoxController::AddParticipantBoxController(gsl::not_null channel, Role role, base::lambda doneCallback) : PeerListController(std::make_unique(channel, role, &_additional)) +, _channel(channel) +, _role(role) +, _doneCallback(std::move(doneCallback)) { + if (_channel->mgInfo) { + _additional.creator = _channel->mgInfo->creator; + } +} + +void AddParticipantBoxController::peerListSearchAddRow(gsl::not_null peer) { + if (peer->isSelf()) { + return; + } + PeerListController::peerListSearchAddRow(peer); +} + +std::unique_ptr AddParticipantBoxController::createSearchRow(gsl::not_null peer) { + if (auto user = peer->asUser()) { + return createRow(user); + } + return std::unique_ptr(); +} + +void AddParticipantBoxController::prepare() { + delegate()->peerListSetSearchMode(PeerListSearchMode::Complex); + auto title = [this] { + switch (_role) { + case Role::Admins: return langFactory(lng_channel_add_admin); + case Role::Restricted: return langFactory(lng_channel_add_restricted); + case Role::Kicked: return langFactory(lng_channel_add_banned); + } + Unexpected("Role in AddParticipantBoxController::prepare()"); + }; + delegate()->peerListSetTitle(title()); + setDescriptionText(lang(lng_contacts_loading)); + setSearchNoResultsText(lang(lng_blocked_list_not_found)); + delegate()->peerListRefreshRows(); + + loadMoreRows(); +} + +void AddParticipantBoxController::loadMoreRows() { + if (searchController() && searchController()->loadMoreRows()) { + return; + } + if (_loadRequestId || _allLoaded) { + return; + } + + // First query is small and fast, next loads a lot of rows. + auto perPage = (_offset > 0) ? kParticipantsPerPage : kParticipantsFirstPageCount; + _loadRequestId = request(MTPchannels_GetParticipants(_channel->inputChannel, MTP_channelParticipantsRecent(), MTP_int(_offset), MTP_int(perPage))).done([this](const MTPchannels_ChannelParticipants &result) { + Expects(result.type() == mtpc_channels_channelParticipants); + + _loadRequestId = 0; + + auto &participants = result.c_channels_channelParticipants(); + App::feedUsers(participants.vusers); + + auto &list = participants.vparticipants.v; + if (list.isEmpty()) { + // To be sure - wait for a whole empty result list. + _allLoaded = true; + } else { + for_const (auto &participant, list) { + ++_offset; + HandleParticipant(participant, &_additional, [this](gsl::not_null user) { + appendRow(user); + }); + } + } + if (delegate()->peerListFullRowsCount() > 0) { + setDescriptionText(QString()); + } + delegate()->peerListRefreshRows(); + }).fail([this](const RPCError &error) { + _loadRequestId = 0; + }).send(); +} + +void AddParticipantBoxController::rowClicked(gsl::not_null row) { + auto user = row->peer()->asUser(); + switch (_role) { + case Role::Admins: return editAdmin(user); + case Role::Restricted: return editRestricted(user); + case Role::Kicked: return kickUser(user); + } + Unexpected("Role in AddParticipantBoxController::rowClicked()"); +} + +template +bool AddParticipantBoxController::checkInfoLoaded(gsl::not_null user, Callback callback) { + if (_additional.infoNotLoaded.find(user) == _additional.infoNotLoaded.end()) { + return true; + } + + // We don't know what this user status is in the group. + request(MTPchannels_GetParticipant(_channel->inputChannel, user->inputUser)).done([this, user, callback](const MTPchannels_ChannelParticipant &result) { + Expects(result.type() == mtpc_channels_channelParticipant); + auto &participant = result.c_channels_channelParticipant(); + App::feedUsers(participant.vusers); + HandleParticipant(participant.vparticipant, &_additional, [](gsl::not_null) {}); + _additional.infoNotLoaded.erase(user); + callback(); + }).fail([this, user, callback](const RPCError &error) { + _additional.infoNotLoaded.erase(user); + _additional.external.emplace(user); + callback(); + }).send(); + return false; +} + +void AddParticipantBoxController::editAdmin(gsl::not_null user, bool sure) { + if (!checkInfoLoaded(user, [this, user] { editAdmin(user); })) { + return; + } + + if (sure && _editBox) { + // Close the confirmation box. + _editBox->closeBox(); + } + + // Check restrictions. + auto weak = base::weak_unique_ptr(this); + auto alreadyIt = _additional.adminRights.find(user); + auto currentRights = EditAdminBox::DefaultRights(_channel); + if (alreadyIt != _additional.adminRights.end() || _additional.creator == user) { + // The user is already an admin. + if (_additional.adminCanEdit.find(user) == _additional.adminCanEdit.end() || _additional.creator == user) { + Ui::show(Box(lang(lng_error_cant_edit_admin)), KeepOtherLayers); + return; + } + currentRights = alreadyIt->second; + } else if (_additional.kicked.find(user) != _additional.kicked.end()) { + // The user is banned. + if (_channel->canAddMembers()) { + if (_channel->canBanMembers()) { + if (!sure) { + _editBox = Ui::show(Box(lang(lng_sure_add_admin_unban), [weak, user] { weak->editAdmin(user, true); }), KeepOtherLayers); + return; + } + } else { + Ui::show(Box(lang(lng_error_cant_add_admin_unban)), KeepOtherLayers); + return; + } + } else { + Ui::show(Box(lang(lng_error_cant_add_admin_invite)), KeepOtherLayers); + return; + } + } else if (_additional.restrictedRights.find(user) != _additional.restrictedRights.end()) { + // The user is restricted. + if (_channel->canBanMembers()) { + if (!sure) { + _editBox = Ui::show(Box(lang(lng_sure_add_admin_unban), [weak, user] { weak->editAdmin(user, true); }), KeepOtherLayers); + return; + } + } else { + Ui::show(Box(lang(lng_error_cant_add_admin_unban)), KeepOtherLayers); + return; + } + } else if (_additional.external.find(user) != _additional.external.end()) { + // The user is not in the group yet. + if (_channel->canAddMembers()) { + if (!sure) { + _editBox = Ui::show(Box(lang(lng_sure_add_admin_invite), [weak, user] { weak->editAdmin(user, true); }), KeepOtherLayers); + return; + } + } else { + Ui::show(Box(lang(lng_error_cant_add_admin_invite)), KeepOtherLayers); + return; + } + } + + // Finally edit the admin. + _editBox = Ui::show(Box(_channel, user, currentRights, [megagroup = _channel.get(), user, weak](const MTPChannelAdminRights &rights) { + MTP::send(MTPchannels_EditAdmin(megagroup->inputChannel, user->inputUser, rights), rpcDone([megagroup, user, weak, rights](const MTPUpdates &result) { + AuthSession::Current().api().applyUpdates(result); + megagroup->applyEditAdmin(user, rights); + if (weak) { + weak->editAdminDone(user, rights); + } + })); + }), KeepOtherLayers); +} + +void AddParticipantBoxController::editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights) { + if (_editBox) _editBox->closeBox(); + _additional.restrictedRights.erase(user); + _additional.kicked.erase(user); + _additional.external.erase(user); + if (rights.c_channelAdminRights().vflags.v == 0) { + _additional.adminRights.erase(user); + _additional.adminPromotedBy.erase(user); + _additional.adminCanEdit.erase(user); + } else { + _additional.adminRights[user] = rights; + _additional.adminCanEdit.emplace(user); + auto it = _additional.adminPromotedBy.find(user); + if (it == _additional.adminPromotedBy.end()) { + _additional.adminPromotedBy.emplace(user, App::self()); + } + } + _doneCallback(); +} + +void AddParticipantBoxController::editRestricted(gsl::not_null user, bool sure) { + if (!checkInfoLoaded(user, [this, user] { editRestricted(user); })) { + return; + } + + if (sure && _editBox) { + // Close the confirmation box. + _editBox->closeBox(); + } + + // Check restrictions. + auto weak = base::weak_unique_ptr(this); + auto alreadyIt = _additional.restrictedRights.find(user); + auto currentRights = EditRestrictedBox::DefaultRights(_channel); + if (alreadyIt != _additional.restrictedRights.end()) { + // The user is already banned or restricted. + currentRights = alreadyIt->second; + } else if (_additional.adminRights.find(user) != _additional.adminRights.end() || _additional.creator == user) { + // The user is an admin or creator. + if (_additional.adminCanEdit.find(user) != _additional.adminCanEdit.end()) { + if (!sure) { + _editBox = Ui::show(Box(lang(lng_sure_ban_admin), [weak, user] { weak->editRestricted(user, true); }), KeepOtherLayers); + return; + } + } else { + Ui::show(Box(lang(lng_error_cant_ban_admin)), KeepOtherLayers); + return; + } + } + + // Finally edit the restricted. + _editBox = Ui::show(Box(_channel, user, currentRights, [user, weak](const MTPChannelBannedRights &rights) { + if (weak) { + weak->restrictUserSure(user, rights); + } + }), KeepOtherLayers); +} + +void AddParticipantBoxController::restrictUserSure(gsl::not_null user, const MTPChannelBannedRights &rights) { + if (_editBox) _editBox->closeBox(); + auto weak = base::weak_unique_ptr(this); + MTP::send(MTPchannels_EditBanned(_channel->inputChannel, user->inputUser, rights), rpcDone([megagroup = _channel.get(), user, weak, rights](const MTPUpdates &result) { + AuthSession::Current().api().applyUpdates(result); + megagroup->applyEditBanned(user, rights); + if (weak) { + weak->editRestrictedDone(user, rights); + } + })); +} + +void AddParticipantBoxController::editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights) { + if (_editBox) _editBox->closeBox(); + _additional.adminRights.erase(user); + _additional.adminCanEdit.erase(user); + _additional.adminPromotedBy.erase(user); + if (rights.c_channelBannedRights().vflags.v == 0) { + _additional.restrictedRights.erase(user); + _additional.kicked.erase(user); + } else { + _additional.restrictedRights[user] = rights; + if (rights.c_channelBannedRights().vflags.v & MTPDchannelBannedRights::Flag::f_view_messages) { + _additional.kicked.emplace(user); + } else { + _additional.kicked.erase(user); + } + } + _doneCallback(); +} + +void AddParticipantBoxController::kickUser(gsl::not_null user, bool sure) { + if (!checkInfoLoaded(user, [this, user] { kickUser(user); })) { + return; + } + + // Check restrictions. + auto weak = base::weak_unique_ptr(this); + if (_additional.adminRights.find(user) != _additional.adminRights.end() || _additional.creator == user) { + // The user is an admin or creator. + if (_additional.adminCanEdit.find(user) != _additional.adminCanEdit.end()) { + if (!sure) { + _editBox = Ui::show(Box(lang(lng_sure_ban_admin), [weak, user] { weak->kickUser(user, true); }), KeepOtherLayers); + return; + } + } else { + Ui::show(Box(lang(lng_error_cant_ban_admin)), KeepOtherLayers); + return; + } + } + + // Finally kick him. + if (!sure) { + auto text = lng_sure_ban_user_group(lt_user, App::peerName(user)); + _editBox = Ui::show(Box(text, [weak, user] { weak->kickUser(user, true); }), KeepOtherLayers); + return; + } + restrictUserSure(user, ChannelData::KickedRestrictedRights()); +} + +bool AddParticipantBoxController::appendRow(gsl::not_null user) { + if (delegate()->peerListFindRow(user->id) || user->isSelf()) { + return false; + } + delegate()->peerListAppendRow(createRow(user)); + return true; +} + +bool AddParticipantBoxController::prependRow(gsl::not_null user) { + if (delegate()->peerListFindRow(user->id)) { + return false; + } + delegate()->peerListPrependRow(createRow(user)); + return true; +} + +std::unique_ptr AddParticipantBoxController::createRow(gsl::not_null user) const { + auto row = std::make_unique(user); + return std::move(row); +} + +template +void AddParticipantBoxController::HandleParticipant(const MTPChannelParticipant &participant, gsl::not_null additional, Callback callback) { + switch (participant.type()) { + case mtpc_channelParticipantAdmin: { + auto &admin = participant.c_channelParticipantAdmin(); + if (auto user = App::userLoaded(admin.vuser_id.v)) { + additional->infoNotLoaded.erase(user); + additional->restrictedRights.erase(user); + additional->kicked.erase(user); + additional->adminRights[user] = admin.vadmin_rights; + if (admin.is_can_edit()) { + additional->adminCanEdit.emplace(user); + } else { + additional->adminCanEdit.erase(user); + } + if (auto promoted = App::userLoaded(admin.vpromoted_by.v)) { + auto it = additional->adminPromotedBy.find(user); + if (it == additional->adminPromotedBy.end()) { + additional->adminPromotedBy.emplace(user, promoted); + } else { + it->second = promoted; + } + } else { + LOG(("API Error: No user %1 for admin promoted by.").arg(admin.vpromoted_by.v)); + } + callback(user); + } + } break; + case mtpc_channelParticipantCreator: { + auto &creator = participant.c_channelParticipantCreator(); + if (auto user = App::userLoaded(creator.vuser_id.v)) { + additional->infoNotLoaded.erase(user); + additional->creator = user; + callback(user); + } + } break; + case mtpc_channelParticipantBanned: { + auto &banned = participant.c_channelParticipantBanned(); + if (auto user = App::userLoaded(banned.vuser_id.v)) { + additional->infoNotLoaded.erase(user); + additional->adminRights.erase(user); + additional->adminCanEdit.erase(user); + additional->adminPromotedBy.erase(user); + if (banned.is_left()) { + additional->kicked.emplace(user); + } else { + additional->kicked.erase(user); + } + additional->restrictedRights[user] = banned.vbanned_rights; + callback(user); + } + } break; + case mtpc_channelParticipant: { + auto &data = participant.c_channelParticipant(); + if (auto user = App::userLoaded(data.vuser_id.v)) { + additional->infoNotLoaded.erase(user); + additional->adminRights.erase(user); + additional->adminCanEdit.erase(user); + additional->adminPromotedBy.erase(user); + additional->restrictedRights.erase(user); + additional->kicked.erase(user); + callback(user); + } + } break; + default: Unexpected("Participant type in AddParticipantBoxController::HandleParticipant()"); + } +} + +AddParticipantBoxSearchController::AddParticipantBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional) +: _channel(channel) +, _role(role) +, _additional(additional) { + _timer.setCallback([this] { searchOnServer(); }); +} + +void AddParticipantBoxSearchController::searchQuery(const QString &query) { + if (_query != query) { + _query = query; + _offset = 0; + _requestId = 0; + _participantsLoaded = false; + _chatsContactsAdded = false; + _globalLoaded = false; + if (!_query.isEmpty() && !searchParticipantsInCache()) { + _timer.callOnce(AutoSearchTimeout); + } else { + _timer.cancel(); + } + } +} + +void AddParticipantBoxSearchController::searchOnServer() { + Expects(!_query.isEmpty()); + loadMoreRows(); +} + +bool AddParticipantBoxSearchController::isLoading() { + return _timer.isActive() || _requestId; +} + +bool AddParticipantBoxSearchController::searchParticipantsInCache() { + auto it = _participantsCache.find(_query); + if (it != _participantsCache.cend()) { + _requestId = 0; + searchParticipantsDone(_requestId, it->second.result, it->second.requestedCount); + return true; + } + return false; +} + +bool AddParticipantBoxSearchController::searchGlobalInCache() { + auto it = _globalCache.find(_query); + if (it != _globalCache.cend()) { + _requestId = 0; + searchGlobalDone(_requestId, it->second); + return true; + } + return false; +} + +bool AddParticipantBoxSearchController::loadMoreRows() { + if (_query.isEmpty()) { + return false; + } + if (_globalLoaded) { + return true; + } + if (_participantsLoaded) { + if (!_chatsContactsAdded) { + addChatsContacts(); + } + if (!isLoading() && !searchGlobalInCache()) { + requestGlobal(); + } + } else if (!isLoading()) { + requestParticipants(); + } + return true; +} + +void AddParticipantBoxSearchController::requestParticipants() { + // For search we request a lot of rows from the first query. + // (because we've waited for search request by timer already, + // so we don't expect it to be fast, but we want to fill cache). + auto perPage = kParticipantsPerPage; + _requestId = request(MTPchannels_GetParticipants(_channel->inputChannel, MTP_channelParticipantsSearch(MTP_string(_query)), MTP_int(_offset), MTP_int(perPage))).done([this, perPage](const MTPchannels_ChannelParticipants &result, mtpRequestId requestId) { + searchParticipantsDone(requestId, result, perPage); + }).fail([this](const RPCError &error, mtpRequestId requestId) { + if (_requestId == requestId) { + _requestId = 0; + _participantsLoaded = true; + loadMoreRows(); + delegate()->peerListSearchRefreshRows(); + } + }).send(); + auto entry = Query(); + entry.text = _query; + entry.offset = _offset; + _participantsQueries.emplace(_requestId, entry); +} + +void AddParticipantBoxSearchController::searchParticipantsDone(mtpRequestId requestId, const MTPchannels_ChannelParticipants &result, int requestedCount) { + Expects(result.type() == mtpc_channels_channelParticipants); + + auto &participants = result.c_channels_channelParticipants(); + auto query = _query; + if (requestId) { + App::feedUsers(participants.vusers); + auto it = _participantsQueries.find(requestId); + if (it != _participantsQueries.cend()) { + query = it->second.text; + if (it->second.offset == 0) { + auto &entry = _participantsCache[query]; + entry.result = result; + entry.requestedCount = requestedCount; + } + _participantsQueries.erase(it); + } + } + + if (_requestId == requestId) { + _requestId = 0; + auto &list = participants.vparticipants.v; + if (list.size() < requestedCount) { + // We want cache to have full information about a query with small + // results count (if we don't need the second request). So we don't + // wait for an empty results list unlike the non-search peer list. + _participantsLoaded = true; + if (list.empty() && _offset == 0) { + // No results, so we want to request global search immediately. + loadMoreRows(); + } + } + for_const (auto &participant, list) { + AddParticipantBoxController::HandleParticipant(participant, _additional, [this](gsl::not_null user) { + delegate()->peerListSearchAddRow(user); + }); + } + _offset += list.size(); + delegate()->peerListSearchRefreshRows(); + } +} + +void AddParticipantBoxSearchController::requestGlobal() { + if (_query.size() < MinUsernameLength) { + _globalLoaded = true; + return; + } + + auto perPage = SearchPeopleLimit; + _requestId = request(MTPcontacts_Search(MTP_string(_query), MTP_int(perPage))).done([this](const MTPcontacts_Found &result, mtpRequestId requestId) { + searchGlobalDone(requestId, result); + }).fail([this](const RPCError &error, mtpRequestId requestId) { + if (_requestId == requestId) { + _requestId = 0; + _globalLoaded = true; + delegate()->peerListSearchRefreshRows(); + } + }).send(); + _globalQueries.emplace(_requestId, _query); +} + +void AddParticipantBoxSearchController::searchGlobalDone(mtpRequestId requestId, const MTPcontacts_Found &result) { + Expects(result.type() == mtpc_contacts_found); + + auto &found = result.c_contacts_found(); + auto query = _query; + if (requestId) { + App::feedUsers(found.vusers); + App::feedChats(found.vchats); + auto it = _globalQueries.find(requestId); + if (it != _globalQueries.cend()) { + query = it->second; + _globalCache[query] = result; + _globalQueries.erase(it); + } + } + + if (_requestId == requestId) { + _requestId = 0; + _globalLoaded = true; + for_const (auto &mtpPeer, found.vresults.v) { + auto peerId = peerFromMTP(mtpPeer); + if (auto peer = App::peerLoaded(peerId)) { + if (auto user = peer->asUser()) { + if (_additional->adminRights.find(user) == _additional->adminRights.cend() + && _additional->restrictedRights.find(user) == _additional->restrictedRights.cend() + && _additional->external.find(user) == _additional->external.cend() + && _additional->kicked.find(user) == _additional->kicked.cend() + && _additional->creator != user) { + _additional->infoNotLoaded.emplace(user); + } + delegate()->peerListSearchAddRow(user); + } + } + } + delegate()->peerListSearchRefreshRows(); + } +} + +void AddParticipantBoxSearchController::addChatsContacts() { + _chatsContactsAdded = true; + + auto filterWordList = _query.split(cWordSplit(), QString::SkipEmptyParts); + auto wordsCount = filterWordList.size(); + auto wordList = QStringList(); + wordList.reserve(wordsCount); + for_const (auto &word, filterWordList) { + auto trimmed = word.trimmed(); + if (!trimmed.isEmpty()) { + wordList.push_back(trimmed); + } + } + if (wordList.empty()) { + return; + } + + auto getSmallestIndex = [&wordList](Dialogs::IndexedList *list) -> const Dialogs::List* { + if (list->isEmpty()) { + return nullptr; + } + + auto result = (const Dialogs::List*)nullptr; + for_const (auto &word, wordList) { + auto found = list->filtered(word[0]); + if (found->isEmpty()) { + return nullptr; + } + if (!result || result->size() > found->size()) { + result = found; + } + } + return result; + }; + auto dialogsIndex = getSmallestIndex(App::main()->dialogsList()); + auto contactsIndex = getSmallestIndex(App::main()->contactsNoDialogsList()); + + auto allWordsAreFound = [&wordList](const OrderedSet &names) { + auto hasNamePartStartingWith = [&names](const QString &word) { + for_const (auto &namePart, names) { + if (namePart.startsWith(word)) { + return true; + } + } + return false; + }; + + for_const (auto &word, wordList) { + if (!hasNamePartStartingWith(word)) { + return false; + } + } + return true; + }; + auto filterAndAppend = [this, allWordsAreFound](const Dialogs::List *list) { + if (!list) { + return; + } + + for_const (auto row, *list) { + if (auto user = row->history()->peer->asUser()) { + if (allWordsAreFound(user->names)) { + delegate()->peerListSearchAddRow(user); + } + } + } + }; + filterAndAppend(dialogsIndex); + filterAndAppend(contactsIndex); + delegate()->peerListSearchRefreshRows(); +} + +} // namespace Profile diff --git a/Telegram/SourceFiles/profile/profile_channel_controllers.h b/Telegram/SourceFiles/profile/profile_channel_controllers.h new file mode 100644 index 000000000..695f79fd3 --- /dev/null +++ b/Telegram/SourceFiles/profile/profile_channel_controllers.h @@ -0,0 +1,220 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#pragma once + +#include "boxes/peer_list_box.h" +#include "mtproto/sender.h" +#include "base/weak_unique_ptr.h" + +namespace Profile { + +// Viewing admins, banned or restricted users list with search. +class ParticipantsBoxController : public PeerListController, private base::Subscriber, private MTP::Sender, public base::enable_weak_from_this { +public: + enum class Role { + Admins, + Restricted, + Kicked, + }; + static void Start(gsl::not_null channel, Role role); + + struct Additional { + std::map, MTPChannelAdminRights> adminRights; + std::set> adminCanEdit; + std::map, gsl::not_null> adminPromotedBy; + std::map, MTPChannelBannedRights> restrictedRights; + std::set> kicked; + std::set> external; + std::set> infoNotLoaded; + UserData *creator = nullptr; + }; + + ParticipantsBoxController(gsl::not_null channel, Role role); + + void addNewItem(); + + void prepare() override; + void rowClicked(gsl::not_null row) override; + void rowActionClicked(gsl::not_null row) override; + void loadMoreRows() override; + + void peerListSearchAddRow(gsl::not_null peer) override; + std::unique_ptr createSearchRow(gsl::not_null peer) override; + + // Callback(gsl::not_null) + template + static void HandleParticipant(const MTPChannelParticipant &participant, Role role, gsl::not_null additional, Callback callback); + +private: + void editAdmin(gsl::not_null user); + void editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights); + void editRestricted(gsl::not_null user); + void editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights); + void removeKicked(gsl::not_null row, gsl::not_null user); + bool appendRow(gsl::not_null user); + bool prependRow(gsl::not_null user); + std::unique_ptr createRow(gsl::not_null user) const; + + gsl::not_null _channel; + Role _role = Role::Admins; + int _offset = 0; + mtpRequestId _loadRequestId = 0; + bool _allLoaded = false; + Additional _additional; + QPointer _editBox; + QPointer _addBox; + +}; + +// Banned and restricted users server side search. +class BannedBoxSearchController : public PeerListSearchController, private MTP::Sender { +public: + using Role = ParticipantsBoxController::Role; + using Additional = ParticipantsBoxController::Additional; + + BannedBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional); + + void searchQuery(const QString &query) override; + bool isLoading() override; + bool loadMoreRows() override; + +private: + struct CacheEntry { + MTPchannels_ChannelParticipants result; + int requestedCount = 0; + }; + struct Query { + QString text; + int offset = 0; + }; + + void searchOnServer(); + bool searchInCache(); + void searchDone(mtpRequestId requestId, const MTPchannels_ChannelParticipants &result, int requestedCount); + + gsl::not_null _channel; + Role _role = Role::Restricted; + gsl::not_null _additional; + + base::Timer _timer; + QString _query; + mtpRequestId _requestId = 0; + int _offset = 0; + bool _allLoaded = false; + std::map _cache; + std::map _queries; + +}; + +// Adding an admin, banned or restricted user from channel members with search + contacts search + global search. +class AddParticipantBoxController : public PeerListController, private base::Subscriber, private MTP::Sender, public base::enable_weak_from_this { +public: + using Role = ParticipantsBoxController::Role; + using Additional = ParticipantsBoxController::Additional; + + AddParticipantBoxController(gsl::not_null channel, Role role, base::lambda doneCallback); + + void prepare() override; + void rowClicked(gsl::not_null row) override; + void loadMoreRows() override; + + void peerListSearchAddRow(gsl::not_null peer) override; + std::unique_ptr createSearchRow(gsl::not_null peer) override; + + // Callback(gsl::not_null) + template + static void HandleParticipant(const MTPChannelParticipant &participant, gsl::not_null additional, Callback callback); + +private: + template + bool checkInfoLoaded(gsl::not_null user, Callback callback); + + void editAdmin(gsl::not_null user, bool sure = false); + void editAdminDone(gsl::not_null user, const MTPChannelAdminRights &rights); + void editRestricted(gsl::not_null user, bool sure = false); + void editRestrictedDone(gsl::not_null user, const MTPChannelBannedRights &rights); + void kickUser(gsl::not_null user, bool sure = false); + void restrictUserSure(gsl::not_null user, const MTPChannelBannedRights &rights); + bool appendRow(gsl::not_null user); + bool prependRow(gsl::not_null user); + std::unique_ptr createRow(gsl::not_null user) const; + + gsl::not_null _channel; + Role _role = Role::Admins; + int _offset = 0; + mtpRequestId _loadRequestId = 0; + bool _allLoaded = false; + Additional _additional; + QPointer _editBox; + base::lambda _doneCallback; + +}; + +// Finds channel members, then contacts, then global search results. +class AddParticipantBoxSearchController : public PeerListSearchController, private MTP::Sender { +public: + using Role = ParticipantsBoxController::Role; + using Additional = ParticipantsBoxController::Additional; + + AddParticipantBoxSearchController(gsl::not_null channel, Role role, gsl::not_null additional); + + void searchQuery(const QString &query) override; + bool isLoading() override; + bool loadMoreRows() override; + +private: + struct CacheEntry { + MTPchannels_ChannelParticipants result; + int requestedCount = 0; + }; + struct Query { + QString text; + int offset = 0; + }; + + void searchOnServer(); + bool searchParticipantsInCache(); + void searchParticipantsDone(mtpRequestId requestId, const MTPchannels_ChannelParticipants &result, int requestedCount); + bool searchGlobalInCache(); + void searchGlobalDone(mtpRequestId requestId, const MTPcontacts_Found &result); + void requestParticipants(); + void addChatsContacts(); + void requestGlobal(); + + gsl::not_null _channel; + Role _role = Role::Admins; + gsl::not_null _additional; + + base::Timer _timer; + QString _query; + mtpRequestId _requestId = 0; + int _offset = 0; + bool _participantsLoaded = false; + bool _chatsContactsAdded = false; + bool _globalLoaded = false; + std::map _participantsCache; + std::map _participantsQueries; + std::map _globalCache; + std::map _globalQueries; + +}; + +} // namespace Profile diff --git a/Telegram/gyp/telegram_sources.txt b/Telegram/gyp/telegram_sources.txt index 47f841b44..8bf675954 100644 --- a/Telegram/gyp/telegram_sources.txt +++ b/Telegram/gyp/telegram_sources.txt @@ -345,6 +345,8 @@ <(src_loc)/profile/profile_block_shared_media.h <(src_loc)/profile/profile_block_widget.cpp <(src_loc)/profile/profile_block_widget.h +<(src_loc)/profile/profile_channel_controllers.cpp +<(src_loc)/profile/profile_channel_controllers.h <(src_loc)/profile/profile_common_groups_section.cpp <(src_loc)/profile/profile_common_groups_section.h <(src_loc)/profile/profile_cover_drop_area.cpp