| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | /*
 | 
					
						
							|  |  |  | 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
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | #include "api/api_who_reacted.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "history/history_item.h"
 | 
					
						
							|  |  |  | #include "history/history.h"
 | 
					
						
							|  |  |  | #include "data/data_peer.h"
 | 
					
						
							|  |  |  | #include "data/data_chat.h"
 | 
					
						
							|  |  |  | #include "data/data_channel.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | #include "data/data_document.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | #include "data/data_user.h"
 | 
					
						
							|  |  |  | #include "data/data_changes.h"
 | 
					
						
							|  |  |  | #include "data/data_session.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | #include "data/data_media_types.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | #include "main/main_app_config.h"
 | 
					
						
							|  |  |  | #include "main/main_session.h"
 | 
					
						
							|  |  |  | #include "main/main_account.h"
 | 
					
						
							|  |  |  | #include "base/unixtime.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | #include "base/weak_ptr.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | #include "ui/controls/who_reacted_context_action.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | #include "apiwrap.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | #include "styles/style_chat.h"
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Api { | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | constexpr auto kContextReactionsLimit = 50; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | struct Peers { | 
					
						
							|  |  |  | 	std::vector<PeerId> list; | 
					
						
							|  |  |  | 	bool unknown = false; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | inline bool operator==(const Peers &a, const Peers &b) noexcept { | 
					
						
							|  |  |  | 	return (a.list == b.list) && (a.unknown == b.unknown); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | struct PeerWithReaction { | 
					
						
							|  |  |  | 	PeerId peer = 0; | 
					
						
							|  |  |  | 	QString reaction; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | inline bool operator==( | 
					
						
							|  |  |  | 		const PeerWithReaction &a, | 
					
						
							|  |  |  | 		const PeerWithReaction &b) noexcept { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	return (a.peer == b.peer) && (a.reaction == b.reaction); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | struct PeersWithReactions { | 
					
						
							|  |  |  | 	std::vector<PeerWithReaction> list; | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 	std::vector<PeerId> read; | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 	int fullReactionsCount = 0; | 
					
						
							|  |  |  | 	bool unknown = false; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | inline bool operator==( | 
					
						
							|  |  |  | 		const PeersWithReactions &a, | 
					
						
							|  |  |  | 		const PeersWithReactions &b) noexcept { | 
					
						
							|  |  |  | 	return (a.fullReactionsCount == b.fullReactionsCount) | 
					
						
							|  |  |  | 		&& (a.list == b.list) | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 		&& (a.read == b.read) | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		&& (a.unknown == b.unknown); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | struct CachedRead { | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 	CachedRead() | 
					
						
							|  |  |  | 	: data(Peers{ .unknown = true }) { | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 	rpl::variable<Peers> data; | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 	mtpRequestId requestId = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | struct CachedReacted { | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 	CachedReacted() | 
					
						
							|  |  |  | 	: data(PeersWithReactions{ .unknown = true }) { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 	rpl::variable<PeersWithReactions> data; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	mtpRequestId requestId = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | struct Context { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	base::flat_map<not_null<HistoryItem*>, CachedRead> cachedRead; | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 	base::flat_map< | 
					
						
							|  |  |  | 		not_null<HistoryItem*>, | 
					
						
							|  |  |  | 		base::flat_map<QString, CachedReacted>> cachedReacted; | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 	base::flat_map<not_null<Main::Session*>, rpl::lifetime> subscriptions; | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	[[nodiscard]] CachedRead &cacheRead(not_null<HistoryItem*> item) { | 
					
						
							|  |  |  | 		const auto i = cachedRead.find(item); | 
					
						
							|  |  |  | 		if (i != end(cachedRead)) { | 
					
						
							|  |  |  | 			return i->second; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		return cachedRead.emplace(item, CachedRead()).first->second; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 	[[nodiscard]] CachedReacted &cacheReacted( | 
					
						
							|  |  |  | 			not_null<HistoryItem*> item, | 
					
						
							|  |  |  | 			const QString &reaction) { | 
					
						
							|  |  |  | 		auto &map = cachedReacted[item]; | 
					
						
							|  |  |  | 		const auto i = map.find(reaction); | 
					
						
							|  |  |  | 		if (i != end(map)) { | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 			return i->second; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 		return map.emplace(reaction, CachedReacted()).first->second; | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | struct Userpic { | 
					
						
							|  |  |  | 	not_null<PeerData*> peer; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	QString reaction; | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 	mutable std::shared_ptr<Data::CloudImageView> view; | 
					
						
							|  |  |  | 	mutable InMemoryKey uniqueKey; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct State { | 
					
						
							|  |  |  | 	std::vector<Userpic> userpics; | 
					
						
							|  |  |  | 	Ui::WhoReadContent current; | 
					
						
							|  |  |  | 	base::has_weak_ptr guard; | 
					
						
							|  |  |  | 	bool someUserpicsNotLoaded = false; | 
					
						
							|  |  |  | 	bool scheduled = false; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | [[nodiscard]] auto Contexts() | 
					
						
							|  |  |  | -> base::flat_map<not_null<QWidget*>, std::unique_ptr<Context>> & { | 
					
						
							|  |  |  | 	static auto result = base::flat_map< | 
					
						
							|  |  |  | 		not_null<QWidget*>, | 
					
						
							|  |  |  | 		std::unique_ptr<Context>>(); | 
					
						
							|  |  |  | 	return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [[nodiscard]] not_null<Context*> ContextAt(not_null<QWidget*> key) { | 
					
						
							|  |  |  | 	auto &contexts = Contexts(); | 
					
						
							|  |  |  | 	const auto i = contexts.find(key); | 
					
						
							|  |  |  | 	if (i != end(contexts)) { | 
					
						
							|  |  |  | 		return i->second.get(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto result = contexts.emplace( | 
					
						
							|  |  |  | 		key, | 
					
						
							|  |  |  | 		std::make_unique<Context>()).first->second.get(); | 
					
						
							|  |  |  | 	QObject::connect(key.get(), &QObject::destroyed, [=] { | 
					
						
							|  |  |  | 		auto &contexts = Contexts(); | 
					
						
							|  |  |  | 		const auto i = contexts.find(key); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		for (auto &[item, entry] : i->second->cachedRead) { | 
					
						
							|  |  |  | 			if (const auto requestId = entry.requestId) { | 
					
						
							|  |  |  | 				item->history()->session().api().request(requestId).cancel(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 		for (auto &[item, map] : i->second->cachedReacted) { | 
					
						
							|  |  |  | 			for (auto &[reaction, entry] : map) { | 
					
						
							|  |  |  | 				if (const auto requestId = entry.requestId) { | 
					
						
							|  |  |  | 					item->history()->session().api().request(requestId).cancel(); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		contexts.erase(i); | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 	return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | [[nodiscard]] not_null<Context*> PreparedContextAt( | 
					
						
							|  |  |  | 		not_null<QWidget*> key, | 
					
						
							|  |  |  | 		not_null<Main::Session*> session) { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	const auto context = ContextAt(key); | 
					
						
							|  |  |  | 	if (context->subscriptions.contains(session)) { | 
					
						
							|  |  |  | 		return context; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	session->changes().messageUpdates( | 
					
						
							|  |  |  | 		Data::MessageUpdate::Flag::Destroyed | 
					
						
							|  |  |  | 	) | rpl::start_with_next([=](const Data::MessageUpdate &update) { | 
					
						
							|  |  |  | 		const auto i = context->cachedRead.find(update.item); | 
					
						
							|  |  |  | 		if (i != end(context->cachedRead)) { | 
					
						
							|  |  |  | 			session->api().request(i->second.requestId).cancel(); | 
					
						
							|  |  |  | 			context->cachedRead.erase(i); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		const auto j = context->cachedReacted.find(update.item); | 
					
						
							|  |  |  | 		if (j != end(context->cachedReacted)) { | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 			for (auto &[reaction, entry] : j->second) { | 
					
						
							|  |  |  | 				session->api().request(entry.requestId).cancel(); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 			context->cachedReacted.erase(j); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}, context->subscriptions[session]); | 
					
						
							|  |  |  | 	return context; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 11:47:02 +03:00
										 |  |  | [[nodiscard]] QImage GenerateUserpic(Userpic &userpic, int size) { | 
					
						
							|  |  |  | 	size *= style::DevicePixelRatio(); | 
					
						
							|  |  |  | 	auto result = userpic.peer->generateUserpicImage(userpic.view, size); | 
					
						
							|  |  |  | 	result.setDevicePixelRatio(style::DevicePixelRatio()); | 
					
						
							|  |  |  | 	return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-29 17:54:01 +03:00
										 |  |  | [[nodiscard]] Ui::WhoReadType DetectSeenType(not_null<HistoryItem*> item) { | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 	if (const auto media = item->media()) { | 
					
						
							|  |  |  | 		if (!media->webpage()) { | 
					
						
							|  |  |  | 			if (const auto document = media->document()) { | 
					
						
							|  |  |  | 				if (document->isVoiceMessage()) { | 
					
						
							|  |  |  | 					return Ui::WhoReadType::Listened; | 
					
						
							|  |  |  | 				} else if (document->isVideoMessage()) { | 
					
						
							|  |  |  | 					return Ui::WhoReadType::Watched; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return Ui::WhoReadType::Seen; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | [[nodiscard]] rpl::producer<Peers> WhoReadIds( | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 		not_null<HistoryItem*> item, | 
					
						
							|  |  |  | 		not_null<QWidget*> context) { | 
					
						
							|  |  |  | 	auto weak = QPointer<QWidget>(context.get()); | 
					
						
							|  |  |  | 	const auto session = &item->history()->session(); | 
					
						
							|  |  |  | 	return [=](auto consumer) { | 
					
						
							|  |  |  | 		if (!weak) { | 
					
						
							|  |  |  | 			return rpl::lifetime(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		const auto context = PreparedContextAt(weak.data(), session); | 
					
						
							|  |  |  | 		auto &entry = context->cacheRead(item); | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 		if (!entry.requestId) { | 
					
						
							|  |  |  | 			entry.requestId = session->api().request( | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 				MTPmessages_GetMessageReadParticipants( | 
					
						
							|  |  |  | 					item->history()->peer->input, | 
					
						
							|  |  |  | 					MTP_int(item->id) | 
					
						
							|  |  |  | 				) | 
					
						
							|  |  |  | 			).done([=](const MTPVector<MTPlong> &result) { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 				auto &entry = context->cacheRead(item); | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 				entry.requestId = 0; | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 				auto parsed = Peers(); | 
					
						
							|  |  |  | 				parsed.list.reserve(result.v.size()); | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 				for (const auto &id : result.v) { | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 					parsed.list.push_back(UserId(id)); | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 				entry.data = std::move(parsed); | 
					
						
							| 
									
										
										
										
											2021-11-26 23:46:53 +03:00
										 |  |  | 			}).fail([=] { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 				auto &entry = context->cacheRead(item); | 
					
						
							| 
									
										
										
										
											2021-09-08 13:50:48 +03:00
										 |  |  | 				entry.requestId = 0; | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 				if (entry.data.current().unknown) { | 
					
						
							|  |  |  | 					entry.data = Peers(); | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			}).send(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		return entry.data.value().start_existing(consumer); | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 	}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | [[nodiscard]] PeersWithReactions WithEmptyReactions( | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 		Peers &&peers) { | 
					
						
							|  |  |  | 	auto result = PeersWithReactions{ | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		.list = peers.list | ranges::views::transform([](PeerId peer) { | 
					
						
							|  |  |  | 			return PeerWithReaction{.peer = peer }; | 
					
						
							|  |  |  | 		}) | ranges::to_vector, | 
					
						
							|  |  |  | 		.unknown = peers.unknown, | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 	result.read = std::move(peers.list); | 
					
						
							|  |  |  | 	return result; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | [[nodiscard]] rpl::producer<PeersWithReactions> WhoReactedIds( | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		not_null<HistoryItem*> item, | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 		const QString &reaction, | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		not_null<QWidget*> context) { | 
					
						
							|  |  |  | 	auto weak = QPointer<QWidget>(context.get()); | 
					
						
							|  |  |  | 	const auto session = &item->history()->session(); | 
					
						
							|  |  |  | 	return [=](auto consumer) { | 
					
						
							|  |  |  | 		if (!weak) { | 
					
						
							|  |  |  | 			return rpl::lifetime(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		const auto context = PreparedContextAt(weak.data(), session); | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 		auto &entry = context->cacheReacted(item, reaction); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		if (!entry.requestId) { | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 			using Flag = MTPmessages_GetMessageReactionsList::Flag; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 			entry.requestId = session->api().request( | 
					
						
							|  |  |  | 				MTPmessages_GetMessageReactionsList( | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 					MTP_flags(reaction.isEmpty() | 
					
						
							|  |  |  | 						? Flag(0) | 
					
						
							|  |  |  | 						: Flag::f_reaction), | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 					item->history()->peer->input, | 
					
						
							|  |  |  | 					MTP_int(item->id), | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 					MTP_string(reaction), | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 					MTPstring(), // offset
 | 
					
						
							|  |  |  | 					MTP_int(kContextReactionsLimit) | 
					
						
							|  |  |  | 				) | 
					
						
							|  |  |  | 			).done([=](const MTPmessages_MessageReactionsList &result) { | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 				auto &entry = context->cacheReacted(item, reaction); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 				entry.requestId = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				result.match([&]( | 
					
						
							|  |  |  | 						const MTPDmessages_messageReactionsList &data) { | 
					
						
							|  |  |  | 					session->data().processUsers(data.vusers()); | 
					
						
							| 
									
										
										
										
											2022-01-21 15:48:14 +03:00
										 |  |  | 					session->data().processChats(data.vchats()); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 					auto parsed = PeersWithReactions{ | 
					
						
							|  |  |  | 						.fullReactionsCount = data.vcount().v, | 
					
						
							|  |  |  | 					}; | 
					
						
							|  |  |  | 					parsed.list.reserve(data.vreactions().v.size()); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 					for (const auto &vote : data.vreactions().v) { | 
					
						
							|  |  |  | 						vote.match([&](const auto &data) { | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 							parsed.list.push_back(PeerWithReaction{ | 
					
						
							| 
									
										
										
										
											2022-01-21 15:48:14 +03:00
										 |  |  | 								.peer = peerFromMTP(data.vpeer_id()), | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 								.reaction = qs(data.vreaction()), | 
					
						
							|  |  |  | 							}); | 
					
						
							|  |  |  | 						}); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 					entry.data = std::move(parsed); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 				}); | 
					
						
							|  |  |  | 			}).fail([=] { | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 				auto &entry = context->cacheReacted(item, reaction); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 				entry.requestId = 0; | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 				if (entry.data.current().unknown) { | 
					
						
							|  |  |  | 					entry.data = PeersWithReactions(); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			}).send(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		return entry.data.value().start_existing(consumer); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [[nodiscard]] auto WhoReadOrReactedIds( | 
					
						
							|  |  |  | 	not_null<HistoryItem*> item, | 
					
						
							|  |  |  | 	not_null<QWidget*> context) | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | -> rpl::producer<PeersWithReactions> { | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	return rpl::combine( | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 		WhoReactedIds(item, QString(), context), | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		WhoReadIds(item, context) | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 	) | rpl::map([=](PeersWithReactions &&reacted, Peers &&read) { | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		if (reacted.unknown || read.unknown) { | 
					
						
							|  |  |  | 			return PeersWithReactions{ .unknown = true }; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 		auto &list = reacted.list; | 
					
						
							|  |  |  | 		for (const auto &peer : read.list) { | 
					
						
							|  |  |  | 			if (!ranges::contains(list, peer, &PeerWithReaction::peer)) { | 
					
						
							|  |  |  | 				list.push_back({ .peer = peer }); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 		reacted.read = std::move(read.list); | 
					
						
							| 
									
										
										
										
											2022-01-18 21:52:17 +03:00
										 |  |  | 		return std::move(reacted); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | bool UpdateUserpics( | 
					
						
							|  |  |  | 		not_null<State*> state, | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 		not_null<HistoryItem*> item, | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		const std::vector<PeerWithReaction> &ids) { | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 	auto &owner = item->history()->owner(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	struct ResolvedPeer { | 
					
						
							|  |  |  | 		PeerData *peer = nullptr; | 
					
						
							|  |  |  | 		QString reaction; | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 	const auto peers = ranges::views::all( | 
					
						
							|  |  |  | 		ids | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	) | ranges::views::transform([&](PeerWithReaction id) { | 
					
						
							|  |  |  | 		return ResolvedPeer{ | 
					
						
							|  |  |  | 			.peer = owner.peerLoaded(id.peer), | 
					
						
							|  |  |  | 			.reaction = id.reaction, | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 	}) | ranges::views::filter([](ResolvedPeer resolved) { | 
					
						
							|  |  |  | 		return resolved.peer != nullptr; | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 	}) | ranges::to_vector; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto same = ranges::equal( | 
					
						
							|  |  |  | 		state->userpics, | 
					
						
							|  |  |  | 		peers, | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		ranges::equal_to(), | 
					
						
							|  |  |  | 		&Userpic::peer, | 
					
						
							|  |  |  | 		[](const ResolvedPeer &r) { return not_null{ r.peer }; }); | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 	if (same) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	auto &was = state->userpics; | 
					
						
							|  |  |  | 	auto now = std::vector<Userpic>(); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 	for (const auto &resolved : peers) { | 
					
						
							|  |  |  | 		const auto peer = not_null{ resolved.peer }; | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		if (ranges::contains(now, peer, &Userpic::peer)) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		const auto i = ranges::find(was, peer, &Userpic::peer); | 
					
						
							|  |  |  | 		if (i != end(was)) { | 
					
						
							|  |  |  | 			now.push_back(std::move(*i)); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		now.push_back(Userpic{ | 
					
						
							|  |  |  | 			.peer = peer, | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 			.reaction = resolved.reaction, | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		}); | 
					
						
							|  |  |  | 		auto &userpic = now.back(); | 
					
						
							|  |  |  | 		userpic.uniqueKey = peer->userpicUniqueKey(userpic.view); | 
					
						
							|  |  |  | 		peer->loadUserpic(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	was = std::move(now); | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RegenerateUserpics(not_null<State*> state, int small, int large) { | 
					
						
							|  |  |  | 	Expects(state->userpics.size() == state->current.participants.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	state->someUserpicsNotLoaded = false; | 
					
						
							|  |  |  | 	const auto count = int(state->userpics.size()); | 
					
						
							|  |  |  | 	for (auto i = 0; i != count; ++i) { | 
					
						
							|  |  |  | 		auto &userpic = state->userpics[i]; | 
					
						
							|  |  |  | 		auto &participant = state->current.participants[i]; | 
					
						
							|  |  |  | 		const auto peer = userpic.peer; | 
					
						
							|  |  |  | 		const auto key = peer->userpicUniqueKey(userpic.view); | 
					
						
							|  |  |  | 		if (peer->hasUserpic() && peer->useEmptyUserpic(userpic.view)) { | 
					
						
							|  |  |  | 			state->someUserpicsNotLoaded = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (userpic.uniqueKey == key) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		participant.userpicKey = userpic.uniqueKey = key; | 
					
						
							| 
									
										
										
										
											2021-09-09 11:47:02 +03:00
										 |  |  | 		participant.userpicLarge = GenerateUserpic(userpic, large); | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		if (i < Ui::WhoReadParticipant::kMaxSmallUserpics) { | 
					
						
							| 
									
										
										
										
											2021-09-09 11:47:02 +03:00
										 |  |  | 			participant.userpicSmall = GenerateUserpic(userpic, small); | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RegenerateParticipants(not_null<State*> state, int small, int large) { | 
					
						
							|  |  |  | 	auto old = base::take(state->current.participants); | 
					
						
							|  |  |  | 	auto &now = state->current.participants; | 
					
						
							|  |  |  | 	now.reserve(state->userpics.size()); | 
					
						
							|  |  |  | 	for (auto &userpic : state->userpics) { | 
					
						
							|  |  |  | 		const auto peer = userpic.peer; | 
					
						
							|  |  |  | 		const auto id = peer->id.value; | 
					
						
							|  |  |  | 		const auto was = ranges::find(old, id, &Ui::WhoReadParticipant::id); | 
					
						
							|  |  |  | 		if (was != end(old)) { | 
					
						
							|  |  |  | 			was->name = peer->name; | 
					
						
							|  |  |  | 			now.push_back(std::move(*was)); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		now.push_back({ | 
					
						
							|  |  |  | 			.name = peer->name, | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 			.reaction = userpic.reaction, | 
					
						
							| 
									
										
										
										
											2021-09-09 11:47:02 +03:00
										 |  |  | 			.userpicLarge = GenerateUserpic(userpic, large), | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 			.userpicKey = userpic.uniqueKey, | 
					
						
							|  |  |  | 			.id = id, | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 		if (now.size() <= Ui::WhoReadParticipant::kMaxSmallUserpics) { | 
					
						
							| 
									
										
										
										
											2021-09-09 11:47:02 +03:00
										 |  |  | 			now.back().userpicSmall = GenerateUserpic(userpic, small); | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	RegenerateUserpics(state, small, large); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | rpl::producer<Ui::WhoReadContent> WhoReacted( | 
					
						
							|  |  |  | 		not_null<HistoryItem*> item, | 
					
						
							|  |  |  | 		const QString &reaction, | 
					
						
							|  |  |  | 		not_null<QWidget*> context, | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 		const style::WhoRead &st, | 
					
						
							|  |  |  | 		std::shared_ptr<WhoReadList> whoReadIds) { | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 	const auto small = st.userpics.size; | 
					
						
							|  |  |  | 	const auto large = st.photoSize; | 
					
						
							|  |  |  | 	return [=](auto consumer) { | 
					
						
							|  |  |  | 		auto lifetime = rpl::lifetime(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 		const auto resolveWhoRead = reaction.isEmpty() | 
					
						
							|  |  |  | 			&& WhoReadExists(item); | 
					
						
							| 
									
										
										
										
											2021-12-29 17:54:01 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 		const auto state = lifetime.make_state<State>(); | 
					
						
							|  |  |  | 		const auto pushNext = [=] { | 
					
						
							|  |  |  | 			consumer.put_next_copy(state->current); | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 		const auto resolveWhoReacted = !reaction.isEmpty() | 
					
						
							|  |  |  | 			|| item->canViewReactions(); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		auto idsWithReactions = (resolveWhoRead && resolveWhoReacted) | 
					
						
							|  |  |  | 			? WhoReadOrReactedIds(item, context) | 
					
						
							|  |  |  | 			: resolveWhoRead | 
					
						
							|  |  |  | 			? (WhoReadIds(item, context) | rpl::map(WithEmptyReactions)) | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 			: WhoReactedIds(item, reaction, context); | 
					
						
							| 
									
										
										
										
											2021-12-29 17:54:01 +03:00
										 |  |  | 		state->current.type = resolveWhoRead | 
					
						
							|  |  |  | 			? DetectSeenType(item) | 
					
						
							|  |  |  | 			: Ui::WhoReadType::Reacted; | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		if (resolveWhoReacted) { | 
					
						
							| 
									
										
										
										
											2021-12-29 17:54:01 +03:00
										 |  |  | 			const auto &list = item->reactions(); | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 			state->current.fullReactionsCount = reaction.isEmpty() | 
					
						
							|  |  |  | 				? ranges::accumulate( | 
					
						
							|  |  |  | 					list, | 
					
						
							|  |  |  | 					0, | 
					
						
							|  |  |  | 					ranges::plus{}, | 
					
						
							|  |  |  | 					[](const auto &pair) { return pair.second; }) | 
					
						
							|  |  |  | 				: list.contains(reaction) | 
					
						
							|  |  |  | 				? list.find(reaction)->second | 
					
						
							|  |  |  | 				: 0; | 
					
						
							| 
									
										
										
										
											2021-12-29 17:54:01 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 			// #TODO reactions
 | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 			state->current.singleReaction = !reaction.isEmpty() | 
					
						
							|  |  |  | 				? reaction | 
					
						
							|  |  |  | 				: (list.size() == 1) | 
					
						
							| 
									
										
										
										
											2021-12-30 16:07:52 +03:00
										 |  |  | 				? list.front().first | 
					
						
							|  |  |  | 				: QString(); | 
					
						
							| 
									
										
										
										
											2021-12-24 13:51:53 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		std::move( | 
					
						
							|  |  |  | 			idsWithReactions | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 		) | rpl::start_with_next([=](PeersWithReactions &&peers) { | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 			if (peers.unknown) { | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 				state->userpics.clear(); | 
					
						
							| 
									
										
										
										
											2021-09-09 16:16:19 +03:00
										 |  |  | 				consumer.put_next(Ui::WhoReadContent{ | 
					
						
							|  |  |  | 					.type = state->current.type, | 
					
						
							| 
									
										
										
										
											2022-01-14 18:42:24 +03:00
										 |  |  | 					.fullReactionsCount = state->current.fullReactionsCount, | 
					
						
							| 
									
										
										
										
											2022-01-18 19:55:24 +03:00
										 |  |  | 					.fullReadCount = state->current.fullReadCount, | 
					
						
							| 
									
										
										
										
											2021-09-09 16:16:19 +03:00
										 |  |  | 					.unknown = true, | 
					
						
							|  |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 				return; | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 			state->current.fullReadCount = int(peers.read.size()); | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 			state->current.fullReactionsCount = peers.fullReactionsCount; | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 			if (whoReadIds) { | 
					
						
							| 
									
										
										
										
											2022-02-02 15:40:39 +03:00
										 |  |  | 				const auto reacted = peers.list.size() - ranges::count( | 
					
						
							|  |  |  | 					peers.list, | 
					
						
							|  |  |  | 					QString(), | 
					
						
							|  |  |  | 					&PeerWithReaction::reaction); | 
					
						
							|  |  |  | 				whoReadIds->list = (peers.read.size() > reacted) | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | 					? std::move(peers.read) | 
					
						
							|  |  |  | 					: std::vector<PeerId>(); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 			if (UpdateUserpics(state, item, peers.list)) { | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 				RegenerateParticipants(state, small, large); | 
					
						
							|  |  |  | 				pushNext(); | 
					
						
							| 
									
										
										
										
											2021-12-31 00:59:29 +03:00
										 |  |  | 			} else if (peers.list.empty()) { | 
					
						
							| 
									
										
										
										
											2021-09-27 18:51:50 +04:00
										 |  |  | 				pushNext(); | 
					
						
							| 
									
										
										
										
											2021-09-09 00:10:49 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}, lifetime); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		item->history()->session().downloaderTaskFinished( | 
					
						
							|  |  |  | 		) | rpl::filter([=] { | 
					
						
							|  |  |  | 			return state->someUserpicsNotLoaded && !state->scheduled; | 
					
						
							|  |  |  | 		}) | rpl::start_with_next([=] { | 
					
						
							|  |  |  | 			for (const auto &userpic : state->userpics) { | 
					
						
							|  |  |  | 				if (userpic.peer->userpicUniqueKey(userpic.view) | 
					
						
							|  |  |  | 					!= userpic.uniqueKey) { | 
					
						
							|  |  |  | 					state->scheduled = true; | 
					
						
							|  |  |  | 					crl::on_main(&state->guard, [=] { | 
					
						
							|  |  |  | 						state->scheduled = false; | 
					
						
							|  |  |  | 						RegenerateUserpics(state, small, large); | 
					
						
							|  |  |  | 						pushNext(); | 
					
						
							|  |  |  | 					}); | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}, lifetime); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return lifetime; | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 20:46:10 +03:00
										 |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WhoReadExists(not_null<HistoryItem*> item) { | 
					
						
							|  |  |  | 	if (!item->out()) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto type = DetectSeenType(item); | 
					
						
							|  |  |  | 	const auto unseen = (type == Ui::WhoReadType::Seen) | 
					
						
							|  |  |  | 		? item->unread() | 
					
						
							|  |  |  | 		: item->isUnreadMedia(); | 
					
						
							|  |  |  | 	if (unseen) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto history = item->history(); | 
					
						
							|  |  |  | 	const auto peer = history->peer; | 
					
						
							|  |  |  | 	const auto chat = peer->asChat(); | 
					
						
							|  |  |  | 	const auto megagroup = peer->asMegagroup(); | 
					
						
							|  |  |  | 	if (!chat && !megagroup) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto &appConfig = peer->session().account().appConfig(); | 
					
						
							|  |  |  | 	const auto expirePeriod = TimeId(appConfig.get<double>( | 
					
						
							|  |  |  | 		"chat_read_mark_expire_period", | 
					
						
							|  |  |  | 		7 * 86400.)); | 
					
						
							|  |  |  | 	if (item->date() + expirePeriod <= base::unixtime::now()) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto maxCount = int(appConfig.get<double>( | 
					
						
							|  |  |  | 		"chat_read_mark_size_threshold", | 
					
						
							|  |  |  | 		50)); | 
					
						
							|  |  |  | 	const auto count = megagroup ? megagroup->membersCount() : chat->count; | 
					
						
							|  |  |  | 	if (count <= 0 || count > maxCount) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WhoReactedExists(not_null<HistoryItem*> item) { | 
					
						
							|  |  |  | 	return item->canViewReactions() || WhoReadExists(item); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rpl::producer<Ui::WhoReadContent> WhoReacted( | 
					
						
							|  |  |  | 		not_null<HistoryItem*> item, | 
					
						
							|  |  |  | 		not_null<QWidget*> context, | 
					
						
							|  |  |  | 		const style::WhoRead &st, | 
					
						
							|  |  |  | 		std::shared_ptr<WhoReadList> whoReadIds) { | 
					
						
							|  |  |  | 	return WhoReacted(item, QString(), context, st, std::move(whoReadIds)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rpl::producer<Ui::WhoReadContent> WhoReacted( | 
					
						
							|  |  |  | 		not_null<HistoryItem*> item, | 
					
						
							|  |  |  | 		const QString &reaction, | 
					
						
							|  |  |  | 		not_null<QWidget*> context, | 
					
						
							|  |  |  | 		const style::WhoRead &st) { | 
					
						
							|  |  |  | 	return WhoReacted(item, reaction, context, st, nullptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 18:44:01 +03:00
										 |  |  | } // namespace Api
 |