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
 |