 dd81f5d59f
			
		
	
	
		dd81f5d59f
		
	
	
	
	
		
			
			base::lambda -> Fn (type alias for std::function). base::lambda_once -> FnMut (type alias for base::unique_function). base::lambda_guarded -> crl::guard. base::lambda_call_type_t -> crl::deduced_call_type.
		
			
				
	
	
		
			153 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			153 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #include "dialogs/dialogs_search_from_controllers.h"
 | |
| 
 | |
| #include "lang/lang_keys.h"
 | |
| #include "data/data_peer_values.h"
 | |
| #include "observer_peer.h"
 | |
| #include "auth_session.h"
 | |
| #include "apiwrap.h"
 | |
| 
 | |
| namespace Dialogs {
 | |
| 
 | |
| void ShowSearchFromBox(
 | |
| 		not_null<Window::Navigation*> navigation,
 | |
| 		not_null<PeerData*> peer,
 | |
| 		Fn<void(not_null<UserData*>)> callback,
 | |
| 		Fn<void()> closedCallback) {
 | |
| 	auto createController = [
 | |
| 		navigation,
 | |
| 		peer,
 | |
| 		callback = std::move(callback)
 | |
| 	]() -> std::unique_ptr<PeerListController> {
 | |
| 		if (peer) {
 | |
| 			if (auto chat = peer->asChat()) {
 | |
| 				return std::make_unique<Dialogs::ChatSearchFromController>(
 | |
| 					navigation,
 | |
| 					chat,
 | |
| 					std::move(callback));
 | |
| 			} else if (auto group = peer->asMegagroup()) {
 | |
| 				return std::make_unique<Dialogs::ChannelSearchFromController>(
 | |
| 					navigation,
 | |
| 					group,
 | |
| 					std::move(callback));
 | |
| 			}
 | |
| 		}
 | |
| 		return nullptr;
 | |
| 	};
 | |
| 	if (auto controller = createController()) {
 | |
| 		auto subscription = std::make_shared<rpl::lifetime>();
 | |
| 		auto box = Ui::show(Box<PeerListBox>(std::move(controller), [subscription](not_null<PeerListBox*> box) {
 | |
| 			box->addButton(langFactory(lng_cancel), [box, subscription] { box->closeBox(); });
 | |
| 		}), LayerOption::KeepOther);
 | |
| 		box->boxClosing() | rpl::start_with_next(
 | |
| 			std::move(closedCallback),
 | |
| 			*subscription);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ChatSearchFromController::ChatSearchFromController(
 | |
| 	not_null<Window::Navigation*> navigation,
 | |
| 	not_null<ChatData*> chat,
 | |
| 	Fn<void(not_null<UserData*>)> callback)
 | |
| : PeerListController()
 | |
| , _chat(chat)
 | |
| , _callback(std::move(callback)) {
 | |
| }
 | |
| 
 | |
| void ChatSearchFromController::prepare() {
 | |
| 	setSearchNoResultsText(lang(lng_blocked_list_not_found));
 | |
| 	delegate()->peerListSetSearchMode(PeerListSearchMode::Enabled);
 | |
| 	delegate()->peerListSetTitle(langFactory(lng_search_messages_from));
 | |
| 
 | |
| 	rebuildRows();
 | |
| 
 | |
| 	subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(Notify::PeerUpdate::Flag::MembersChanged, [this](const Notify::PeerUpdate &update) {
 | |
| 		if (update.peer == _chat) {
 | |
| 			rebuildRows();
 | |
| 		}
 | |
| 	}));
 | |
| }
 | |
| 
 | |
| void ChatSearchFromController::rowClicked(not_null<PeerListRow*> row) {
 | |
| 	Expects(row->peer()->isUser());
 | |
| 	_callback(row->peer()->asUser());
 | |
| }
 | |
| 
 | |
| void ChatSearchFromController::rebuildRows() {
 | |
| 	auto ms = getms();
 | |
| 	auto wasEmpty = !delegate()->peerListFullRowsCount();
 | |
| 
 | |
| 	auto now = unixtime();
 | |
| 	const auto byOnline = [&](not_null<UserData*> user) {
 | |
| 		return Data::SortByOnlineValue(user, now);
 | |
| 	};
 | |
| 	auto ordered = QMultiMap<TimeId, not_null<UserData*>>();
 | |
| 	if (_chat->noParticipantInfo()) {
 | |
| 		Auth().api().requestFullPeer(_chat);
 | |
| 	} else if (!_chat->participants.empty()) {
 | |
| 		for (const auto [user, version] : _chat->participants) {
 | |
| 			ordered.insertMulti(byOnline(user), user);
 | |
| 		}
 | |
| 	}
 | |
| 	for_const (auto user, _chat->lastAuthors) {
 | |
| 		if (user->isInaccessible()) continue;
 | |
| 		appendRow(user);
 | |
| 		if (!ordered.isEmpty()) {
 | |
| 			ordered.remove(byOnline(user), user);
 | |
| 		}
 | |
| 	}
 | |
| 	if (!ordered.isEmpty()) {
 | |
| 		for (auto i = ordered.cend(), b = ordered.cbegin(); i != b;) {
 | |
| 			appendRow(*(--i));
 | |
| 		}
 | |
| 	}
 | |
| 	checkForEmptyRows();
 | |
| 	delegate()->peerListRefreshRows();
 | |
| }
 | |
| 
 | |
| void ChatSearchFromController::checkForEmptyRows() {
 | |
| 	if (delegate()->peerListFullRowsCount()) {
 | |
| 		setDescriptionText(QString());
 | |
| 	} else {
 | |
| 		setDescriptionText(lang(lng_contacts_loading));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ChatSearchFromController::appendRow(not_null<UserData*> user) {
 | |
| 	if (!delegate()->peerListFindRow(user->id)) {
 | |
| 		delegate()->peerListAppendRow(std::make_unique<PeerListRow>(user));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ChannelSearchFromController::ChannelSearchFromController(
 | |
| 	not_null<Window::Navigation*> navigation,
 | |
| 	not_null<ChannelData*> channel,
 | |
| 	Fn<void(not_null<UserData*>)> callback)
 | |
| : ParticipantsBoxController(
 | |
| 	navigation,
 | |
| 	channel,
 | |
| 	ParticipantsBoxController::Role::Members)
 | |
| , _callback(std::move(callback)) {
 | |
| }
 | |
| 
 | |
| void ChannelSearchFromController::prepare() {
 | |
| 	ParticipantsBoxController::prepare();
 | |
| 	delegate()->peerListSetTitle(langFactory(lng_search_messages_from));
 | |
| }
 | |
| 
 | |
| void ChannelSearchFromController::rowClicked(not_null<PeerListRow*> row) {
 | |
| 	Expects(row->peer()->isUser());
 | |
| 	_callback(row->peer()->asUser());
 | |
| }
 | |
| 
 | |
| std::unique_ptr<PeerListRow> ChannelSearchFromController::createRow(not_null<UserData*> user) const {
 | |
| 	return std::make_unique<PeerListRow>(user);
 | |
| }
 | |
| 
 | |
| } // namespace Dialogs
 |