5237 lines
		
	
	
	
		
			145 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			5237 lines
		
	
	
	
		
			145 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 "history/history_item.h"
 | |
| 
 | |
| #include "lang/lang_keys.h"
 | |
| #include "mainwidget.h"
 | |
| #include "calls/calls_instance.h" // Core::App().calls().joinGroupCall.
 | |
| #include "history/view/history_view_item_preview.h"
 | |
| #include "history/view/history_view_message.h"
 | |
| #include "history/view/history_view_service_message.h"
 | |
| #include "history/view/media/history_view_media_grouped.h"
 | |
| #include "history/history_item_components.h"
 | |
| #include "history/history_item_helpers.h"
 | |
| #include "history/history_unread_things.h"
 | |
| #include "history/history.h"
 | |
| #include "mtproto/mtproto_config.h"
 | |
| #include "ui/text/format_values.h"
 | |
| #include "ui/text/text_isolated_emoji.h"
 | |
| #include "ui/text/text_utilities.h"
 | |
| #include "storage/file_upload.h"
 | |
| #include "storage/storage_shared_media.h"
 | |
| #include "main/main_account.h"
 | |
| #include "main/main_domain.h"
 | |
| #include "main/main_session.h"
 | |
| #include "main/main_session_settings.h"
 | |
| #include "menu/menu_ttl_validator.h"
 | |
| #include "apiwrap.h"
 | |
| #include "media/audio/media_audio.h"
 | |
| #include "core/application.h"
 | |
| #include "window/window_controller.h"
 | |
| #include "window/window_session_controller.h"
 | |
| #include "core/click_handler_types.h"
 | |
| #include "base/unixtime.h"
 | |
| #include "base/timer_rpl.h"
 | |
| #include "api/api_text_entities.h"
 | |
| #include "api/api_updates.h"
 | |
| #include "data/notify/data_notify_settings.h"
 | |
| #include "data/data_bot_app.h"
 | |
| #include "data/data_scheduled_messages.h"
 | |
| #include "data/data_changes.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_message_reactions.h"
 | |
| #include "data/data_folder.h"
 | |
| #include "data/data_forum.h"
 | |
| #include "data/data_forum_topic.h"
 | |
| #include "data/data_channel.h"
 | |
| #include "data/data_chat.h"
 | |
| #include "data/data_game.h"
 | |
| #include "data/data_user.h"
 | |
| #include "data/data_group_call.h" // Data::GroupCall::id().
 | |
| #include "data/data_poll.h" // PollData::publicVotes.
 | |
| #include "data/data_sponsored_messages.h"
 | |
| #include "data/data_stories.h"
 | |
| #include "data/data_web_page.h"
 | |
| #include "chat_helpers/stickers_gift_box_pack.h"
 | |
| #include "payments/payments_checkout_process.h" // CheckoutProcess::Start.
 | |
| #include "platform/platform_notifications_manager.h"
 | |
| #include "spellcheck/spellcheck_highlight_syntax.h"
 | |
| #include "styles/style_dialogs.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kNotificationTextLimit = 255;
 | |
| constexpr auto kPinnedMessageTextLimit = 16;
 | |
| 
 | |
| using ItemPreview = HistoryView::ItemPreview;
 | |
| 
 | |
| template <typename T>
 | |
| [[nodiscard]] PreparedServiceText PrepareEmptyText(const T &) {
 | |
| 	return PreparedServiceText();
 | |
| };
 | |
| 
 | |
| template <typename T>
 | |
| [[nodiscard]] PreparedServiceText PrepareErrorText(const T &data) {
 | |
| 	if constexpr (!std::is_same_v<T, MTPDmessageActionEmpty>) {
 | |
| 		const auto name = QString::fromUtf8(typeid(data).name());
 | |
| 		LOG(("API Error: %1 received.").arg(name));
 | |
| 	}
 | |
| 	return PreparedServiceText{ { tr::lng_message_empty(tr::now) } };
 | |
| }
 | |
| 
 | |
| [[nodiscard]] TextWithEntities SpoilerLoginCode(TextWithEntities text) {
 | |
| 	const auto r = QRegularExpression(u"([\\d\\-]{5,7})"_q);
 | |
| 	const auto m = r.match(text.text);
 | |
| 	if (!m.hasMatch()) {
 | |
| 		return text;
 | |
| 	}
 | |
| 	const auto codeStart = int(m.capturedStart(1));
 | |
| 	const auto codeLength = int(m.capturedLength(1));
 | |
| 	auto i = text.entities.begin();
 | |
| 	const auto e = text.entities.end();
 | |
| 	while (i != e && i->offset() < codeStart) {
 | |
| 		if (i->offset() + i->length() > codeStart) {
 | |
| 			return text; // Entities should not intersect code.
 | |
| 		}
 | |
| 		++i;
 | |
| 	}
 | |
| 	text.entities.insert(i, { EntityType::Spoiler, codeStart, codeLength });
 | |
| 	return text;
 | |
| }
 | |
| 
 | |
| [[nodiscard]] bool HasNotEmojiAndSpaces(const QString &text) {
 | |
| 	if (text.isEmpty()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	auto emoji = 0;
 | |
| 	auto start = text.data();
 | |
| 	const auto end = start + text.size();
 | |
| 	while (start < end) {
 | |
| 		if (start->isSpace()) {
 | |
| 			++start;
 | |
| 		} else if (Ui::Emoji::Find(start, end, &emoji)) {
 | |
| 			start += emoji;
 | |
| 		} else {
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| void HistoryItem::HistoryItem::Destroyer::operator()(HistoryItem *value) {
 | |
| 	if (value) {
 | |
| 		value->destroy();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| struct HistoryItem::CreateConfig {
 | |
| 	ReplyFields reply;
 | |
| 
 | |
| 	UserId viaBotId = 0;
 | |
| 	int viewsCount = -1;
 | |
| 	int forwardsCount = -1;
 | |
| 	QString postAuthor;
 | |
| 
 | |
| 	MsgId originalId = 0;
 | |
| 	TimeId originalDate = 0;
 | |
| 	PeerId originalSenderId = 0;
 | |
| 	QString originalSenderName;
 | |
| 	QString originalPostAuthor;
 | |
| 
 | |
| 	QString forwardPsaType;
 | |
| 	PeerId savedFromPeer = 0;
 | |
| 	MsgId savedFromMsgId = 0;
 | |
| 
 | |
| 	TimeId editDate = 0;
 | |
| 	HistoryMessageMarkupData markup;
 | |
| 	HistoryMessageRepliesData replies;
 | |
| 	bool imported = false;
 | |
| 
 | |
| 	// For messages created from existing messages (forwarded).
 | |
| 	const HistoryMessageReplyMarkup *inlineMarkup = nullptr;
 | |
| };
 | |
| 
 | |
| void HistoryItem::FillForwardedInfo(
 | |
| 		CreateConfig &config,
 | |
| 		const MTPDmessageFwdHeader &data) {
 | |
| 	config.originalId = data.vchannel_post().value_or_empty();
 | |
| 	config.originalDate = data.vdate().v;
 | |
| 	if (const auto fromId = data.vfrom_id()) {
 | |
| 		config.originalSenderId = peerFromMTP(*fromId);
 | |
| 	}
 | |
| 	config.originalSenderName = qs(data.vfrom_name().value_or_empty());
 | |
| 	config.originalPostAuthor = qs(data.vpost_author().value_or_empty());
 | |
| 	config.forwardPsaType = qs(data.vpsa_type().value_or_empty());
 | |
| 	const auto savedFromPeer = data.vsaved_from_peer();
 | |
| 	const auto savedFromMsgId = data.vsaved_from_msg_id();
 | |
| 	if (savedFromPeer && savedFromMsgId) {
 | |
| 		config.savedFromPeer = peerFromMTP(*savedFromPeer);
 | |
| 		config.savedFromMsgId = savedFromMsgId->v;
 | |
| 	}
 | |
| 	config.imported = data.is_imported();
 | |
| }
 | |
| 
 | |
| std::unique_ptr<Data::Media> HistoryItem::CreateMedia(
 | |
| 		not_null<HistoryItem*> item,
 | |
| 		const MTPMessageMedia &media) {
 | |
| 	using Result = std::unique_ptr<Data::Media>;
 | |
| 	return media.match([&](const MTPDmessageMediaContact &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaContact>(
 | |
| 			item,
 | |
| 			media.vuser_id().v,
 | |
| 			qs(media.vfirst_name()),
 | |
| 			qs(media.vlast_name()),
 | |
| 			qs(media.vphone_number()));
 | |
| 	}, [&](const MTPDmessageMediaGeo &media) -> Result {
 | |
| 		return media.vgeo().match([&](const MTPDgeoPoint &point) -> Result {
 | |
| 			return std::make_unique<Data::MediaLocation>(
 | |
| 				item,
 | |
| 				Data::LocationPoint(point));
 | |
| 		}, [](const MTPDgeoPointEmpty &) -> Result {
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaGeoLive &media) -> Result {
 | |
| 		return media.vgeo().match([&](const MTPDgeoPoint &point) -> Result {
 | |
| 			return std::make_unique<Data::MediaLocation>(
 | |
| 				item,
 | |
| 				Data::LocationPoint(point));
 | |
| 		}, [](const MTPDgeoPointEmpty &) -> Result {
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaVenue &media) -> Result {
 | |
| 		return media.vgeo().match([&](const MTPDgeoPoint &point) -> Result {
 | |
| 			return std::make_unique<Data::MediaLocation>(
 | |
| 				item,
 | |
| 				Data::LocationPoint(point),
 | |
| 				qs(media.vtitle()),
 | |
| 				qs(media.vaddress()));
 | |
| 		}, [](const MTPDgeoPointEmpty &data) -> Result {
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaPhoto &media) -> Result {
 | |
| 		const auto photo = media.vphoto();
 | |
| 		if (media.vttl_seconds()) {
 | |
| 			LOG(("App Error: "
 | |
| 				"Unexpected MTPMessageMediaPhoto "
 | |
| 				"with ttl_seconds in CreateMedia."));
 | |
| 			return nullptr;
 | |
| 		} else if (!photo) {
 | |
| 			LOG(("API Error: "
 | |
| 				"Got MTPMessageMediaPhoto "
 | |
| 				"without photo and without ttl_seconds."));
 | |
| 			return nullptr;
 | |
| 		}
 | |
| 		return photo->match([&](const MTPDphoto &photo) -> Result {
 | |
| 			return std::make_unique<Data::MediaPhoto>(
 | |
| 				item,
 | |
| 				item->history()->owner().processPhoto(photo),
 | |
| 				media.is_spoiler());
 | |
| 		}, [](const MTPDphotoEmpty &) -> Result {
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaDocument &media) -> Result {
 | |
| 		const auto document = media.vdocument();
 | |
| 		if (media.vttl_seconds()) {
 | |
| 			LOG(("App Error: "
 | |
| 				"Unexpected MTPMessageMediaDocument "
 | |
| 				"with ttl_seconds in CreateMedia."));
 | |
| 			return nullptr;
 | |
| 		} else if (!document) {
 | |
| 			LOG(("API Error: "
 | |
| 				"Got MTPMessageMediaDocument "
 | |
| 				"without document and without ttl_seconds."));
 | |
| 			return nullptr;
 | |
| 		}
 | |
| 		return document->match([&](const MTPDdocument &document) -> Result {
 | |
| 			return std::make_unique<Data::MediaFile>(
 | |
| 				item,
 | |
| 				item->history()->owner().processDocument(document),
 | |
| 				media.is_nopremium(),
 | |
| 				media.is_spoiler());
 | |
| 		}, [](const MTPDdocumentEmpty &) -> Result {
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaWebPage &media) {
 | |
| 		using Flag = MediaWebPageFlag;
 | |
| 		const auto flags = Flag()
 | |
| 			| (media.is_force_large_media()
 | |
| 				? Flag::ForceLargeMedia
 | |
| 				: Flag())
 | |
| 			| (media.is_force_small_media()
 | |
| 				? Flag::ForceSmallMedia
 | |
| 				: Flag())
 | |
| 			| (media.is_manual() ? Flag::Manual : Flag())
 | |
| 			| (media.is_safe() ? Flag::Safe : Flag());
 | |
| 		return media.vwebpage().match([](const MTPDwebPageEmpty &) -> Result {
 | |
| 			return nullptr;
 | |
| 		}, [&](const MTPDwebPagePending &webpage) -> Result {
 | |
| 			return std::make_unique<Data::MediaWebPage>(
 | |
| 				item,
 | |
| 				item->history()->owner().processWebpage(webpage),
 | |
| 				flags);
 | |
| 		}, [&](const MTPDwebPage &webpage) -> Result {
 | |
| 			return std::make_unique<Data::MediaWebPage>(
 | |
| 				item,
 | |
| 				item->history()->owner().processWebpage(webpage),
 | |
| 				flags);
 | |
| 		}, [](const MTPDwebPageNotModified &) -> Result {
 | |
| 			LOG(("API Error: "
 | |
| 				"webPageNotModified is unexpected in message media."));
 | |
| 			return nullptr;
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaGame &media) -> Result {
 | |
| 		return media.vgame().match([&](const MTPDgame &game) {
 | |
| 			return std::make_unique<Data::MediaGame>(
 | |
| 				item,
 | |
| 				item->history()->owner().processGame(game));
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageMediaInvoice &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaInvoice>(
 | |
| 			item,
 | |
| 			Data::ComputeInvoiceData(item, media));
 | |
| 	}, [&](const MTPDmessageMediaPoll &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaPoll>(
 | |
| 			item,
 | |
| 			item->history()->owner().processPoll(media));
 | |
| 	}, [&](const MTPDmessageMediaDice &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaDice>(
 | |
| 			item,
 | |
| 			qs(media.vemoticon()),
 | |
| 			media.vvalue().v);
 | |
| 	}, [&](const MTPDmessageMediaStory &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaStory>(item, FullStoryId{
 | |
| 			peerFromMTP(media.vpeer()),
 | |
| 			media.vid().v,
 | |
| 		}, media.is_via_mention());
 | |
| 	}, [&](const MTPDmessageMediaGiveaway &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaGiveawayStart>(
 | |
| 			item,
 | |
| 			Data::ComputeGiveawayStartData(item, media));
 | |
| 	}, [&](const MTPDmessageMediaGiveawayResults &media) -> Result {
 | |
| 		return std::make_unique<Data::MediaGiveawayResults>(
 | |
| 			item,
 | |
| 			Data::ComputeGiveawayResultsData(item, media));
 | |
| 	}, [](const MTPDmessageMediaEmpty &) -> Result {
 | |
| 		return nullptr;
 | |
| 	}, [](const MTPDmessageMediaUnsupported &) -> Result {
 | |
| 		return nullptr;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	const MTPDmessage &data,
 | |
| 	MessageFlags localFlags)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		FlagsFromMTP(id, data.vflags().v, localFlags),
 | |
| 		data.vdate().v,
 | |
| 		data.vfrom_id() ? peerFromMTP(*data.vfrom_id()) : PeerId(0)) {
 | |
| 	const auto media = data.vmedia();
 | |
| 	const auto checked = media
 | |
| 		? CheckMessageMedia(*media)
 | |
| 		: MediaCheckResult::Good;
 | |
| 	if (checked == MediaCheckResult::Unsupported) {
 | |
| 		_flags &= ~MessageFlag::HasPostAuthor;
 | |
| 		_flags |= MessageFlag::Legacy;
 | |
| 		createComponents(data);
 | |
| 		setText(UnsupportedMessageText());
 | |
| 	} else if (checked == MediaCheckResult::Empty) {
 | |
| 		AddComponents(HistoryServiceData::Bit());
 | |
| 		setServiceText({
 | |
| 			tr::lng_message_empty(tr::now, Ui::Text::WithEntities)
 | |
| 		});
 | |
| 	} else if (checked == MediaCheckResult::HasTimeToLive) {
 | |
| 		createServiceFromMtp(data);
 | |
| 		applyTTL(data);
 | |
| 	} else if (checked == MediaCheckResult::HasStoryMention) {
 | |
| 		setMedia(*data.vmedia());
 | |
| 		createServiceFromMtp(data);
 | |
| 		applyTTL(data);
 | |
| 	} else {
 | |
| 		createComponents(data);
 | |
| 		if (const auto media = data.vmedia()) {
 | |
| 			setMedia(*media);
 | |
| 		}
 | |
| 		auto textWithEntities = TextWithEntities{
 | |
| 			qs(data.vmessage()),
 | |
| 			Api::EntitiesFromMTP(
 | |
| 				&history->session(),
 | |
| 				data.ventities().value_or_empty())
 | |
| 		};
 | |
| 		setText(_media ? textWithEntities : EnsureNonEmpty(textWithEntities));
 | |
| 		if (const auto groupedId = data.vgrouped_id()) {
 | |
| 			setGroupId(
 | |
| 				MessageGroupId::FromRaw(
 | |
| 					history->peer->id,
 | |
| 					groupedId->v,
 | |
| 					_flags & MessageFlag::IsOrWasScheduled));
 | |
| 		}
 | |
| 		setReactions(data.vreactions());
 | |
| 		applyTTL(data);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	const MTPDmessageService &data,
 | |
| 	MessageFlags localFlags)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		FlagsFromMTP(id, data.vflags().v, localFlags),
 | |
| 		data.vdate().v,
 | |
| 		data.vfrom_id() ? peerFromMTP(*data.vfrom_id()) : PeerId(0)) {
 | |
| 	if (data.vaction().type() != mtpc_messageActionPhoneCall) {
 | |
| 		createServiceFromMtp(data);
 | |
| 	} else {
 | |
| 		createComponents(CreateConfig());
 | |
| 		_media = std::make_unique<Data::MediaCall>(
 | |
| 			this,
 | |
| 			Data::ComputeCallData(data.vaction().c_messageActionPhoneCall()));
 | |
| 		setTextValue({});
 | |
| 	}
 | |
| 	applyTTL(data);
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	const MTPDmessageEmpty &data,
 | |
| 	MessageFlags localFlags)
 | |
| : HistoryItem(
 | |
| 	history,
 | |
| 	id,
 | |
| 	localFlags,
 | |
| 	TimeId(0),
 | |
| 	PreparedServiceText{ tr::lng_message_empty(
 | |
| 		tr::now,
 | |
| 		Ui::Text::WithEntities) }) {
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	TimeId date,
 | |
| 	PreparedServiceText &&message,
 | |
| 	PeerId from,
 | |
| 	PhotoData *photo)
 | |
| : HistoryItem(history, id, flags, date, from) {
 | |
| 	setServiceText(std::move(message));
 | |
| 	if (photo) {
 | |
| 		_media = std::make_unique<Data::MediaPhoto>(
 | |
| 			this,
 | |
| 			history->peer,
 | |
| 			photo);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	TimeId date,
 | |
| 	PeerId from,
 | |
| 	const QString &postAuthor,
 | |
| 	not_null<HistoryItem*> original,
 | |
| 	MsgId topicRootId)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		(NewForwardedFlags(history->peer, from, original) | flags),
 | |
| 		date,
 | |
| 		from) {
 | |
| 	const auto peer = history->peer;
 | |
| 
 | |
| 	auto config = CreateConfig();
 | |
| 
 | |
| 	const auto originalMedia = original->media();
 | |
| 	const auto dropForwardInfo = original->computeDropForwardedInfo();
 | |
| 	config.reply.messageId = config.reply.topMessageId = topicRootId;
 | |
| 	config.reply.topicPost = (topicRootId != 0) ? 1 : 0;
 | |
| 	if (const auto originalReply = original->Get<HistoryMessageReply>()) {
 | |
| 		if (originalReply->external()) {
 | |
| 			config.reply = originalReply->fields().clone(this);
 | |
| 			if (!config.reply.externalPeerId) {
 | |
| 				config.reply.messageId = 0;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	if (!dropForwardInfo) {
 | |
| 		config.originalDate = original->originalDate();
 | |
| 		if (const auto info = original->hiddenSenderInfo()) {
 | |
| 			config.originalSenderName = info->name;
 | |
| 		} else if (const auto originalSender = original->originalSender()) {
 | |
| 			config.originalSenderId = originalSender->id;
 | |
| 			if (originalSender->isChannel()) {
 | |
| 				config.originalId = original->originalId();
 | |
| 			}
 | |
| 		} else {
 | |
| 			Unexpected("Corrupt forwarded information in message.");
 | |
| 		}
 | |
| 		config.originalPostAuthor = original->originalPostAuthor();
 | |
| 	}
 | |
| 	if (peer->isSelf()) {
 | |
| 		//
 | |
| 		// iOS app sends you to the original post if we forward a forward from channel.
 | |
| 		// But server returns not the original post but the forward in saved_from_...
 | |
| 		//
 | |
| 		//if (config.originalId) {
 | |
| 		//	config.savedFromPeer = config.senderOriginal;
 | |
| 		//	config.savedFromMsgId = config.originalId;
 | |
| 		//} else {
 | |
| 			config.savedFromPeer = original->history()->peer->id;
 | |
| 			config.savedFromMsgId = original->id;
 | |
| 		//}
 | |
| 	}
 | |
| 	if (flags & MessageFlag::HasPostAuthor) {
 | |
| 		config.postAuthor = postAuthor;
 | |
| 	}
 | |
| 	if (const auto fwdViaBot = original->viaBot()) {
 | |
| 		config.viaBotId = peerToUser(fwdViaBot->id);
 | |
| 	} else if (originalMedia && originalMedia->game()) {
 | |
| 		if (const auto sender = original->originalSender()) {
 | |
| 			if (const auto user = sender->asUser()) {
 | |
| 				if (user->isBot()) {
 | |
| 					config.viaBotId = peerToUser(user->id);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	const auto fwdViewsCount = original->viewsCount();
 | |
| 	if (fwdViewsCount > 0) {
 | |
| 		config.viewsCount = fwdViewsCount;
 | |
| 	} else if ((isPost() && !isScheduled())
 | |
| 		|| (original->originalSender()
 | |
| 			&& original->originalSender()->isChannel())) {
 | |
| 		config.viewsCount = 1;
 | |
| 	}
 | |
| 
 | |
| 	const auto mediaOriginal = original->media();
 | |
| 	if (CopyMarkupToForward(original)) {
 | |
| 		config.inlineMarkup = original->inlineReplyMarkup();
 | |
| 	}
 | |
| 	createComponents(std::move(config));
 | |
| 
 | |
| 	const auto ignoreMedia = [&] {
 | |
| 		if (mediaOriginal && mediaOriginal->webpage()) {
 | |
| 			if (peer->amRestricted(ChatRestriction::EmbedLinks)) {
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 		return false;
 | |
| 	};
 | |
| 	if (mediaOriginal && !ignoreMedia()) {
 | |
| 		_media = mediaOriginal->clone(this);
 | |
| 		if (original->invertMedia()) {
 | |
| 			_flags |= MessageFlag::InvertMedia;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	const auto dropCustomEmoji = dropForwardInfo
 | |
| 		&& !history->session().premium()
 | |
| 		&& !history->peer->isSelf();
 | |
| 	setText(dropCustomEmoji
 | |
| 		? DropCustomEmoji(original->originalText())
 | |
| 		: original->originalText());
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	FullReplyTo replyTo,
 | |
| 	UserId viaBotId,
 | |
| 	TimeId date,
 | |
| 	PeerId from,
 | |
| 	const QString &postAuthor,
 | |
| 	const TextWithEntities &textWithEntities,
 | |
| 	const MTPMessageMedia &media,
 | |
| 	HistoryMessageMarkupData &&markup,
 | |
| 	uint64 groupedId)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		flags,
 | |
| 		date,
 | |
| 		(flags & MessageFlag::HasFromId) ? from : 0) {
 | |
| 	createComponentsHelper(
 | |
| 		flags,
 | |
| 		replyTo,
 | |
| 		viaBotId,
 | |
| 		postAuthor,
 | |
| 		std::move(markup));
 | |
| 	setMedia(media);
 | |
| 	setText(textWithEntities);
 | |
| 	if (groupedId) {
 | |
| 		setGroupId(MessageGroupId::FromRaw(
 | |
| 			history->peer->id,
 | |
| 			groupedId,
 | |
| 			flags & MessageFlag::IsOrWasScheduled));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	FullReplyTo replyTo,
 | |
| 	UserId viaBotId,
 | |
| 	TimeId date,
 | |
| 	PeerId from,
 | |
| 	const QString &postAuthor,
 | |
| 	not_null<DocumentData*> document,
 | |
| 	const TextWithEntities &caption,
 | |
| 	HistoryMessageMarkupData &&markup)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		flags,
 | |
| 		date,
 | |
| 		(flags & MessageFlag::HasFromId) ? from : 0) {
 | |
| 	createComponentsHelper(
 | |
| 		flags,
 | |
| 		replyTo,
 | |
| 		viaBotId,
 | |
| 		postAuthor,
 | |
| 		std::move(markup));
 | |
| 
 | |
| 	const auto skipPremiumEffect = !history->session().premium();
 | |
| 	const auto spoiler = false;
 | |
| 	_media = std::make_unique<Data::MediaFile>(
 | |
| 		this,
 | |
| 		document,
 | |
| 		skipPremiumEffect,
 | |
| 		spoiler);
 | |
| 	setText(caption);
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	FullReplyTo replyTo,
 | |
| 	UserId viaBotId,
 | |
| 	TimeId date,
 | |
| 	PeerId from,
 | |
| 	const QString &postAuthor,
 | |
| 	not_null<PhotoData*> photo,
 | |
| 	const TextWithEntities &caption,
 | |
| 	HistoryMessageMarkupData &&markup)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		flags,
 | |
| 		date,
 | |
| 		(flags & MessageFlag::HasFromId) ? from : 0) {
 | |
| 	createComponentsHelper(
 | |
| 		flags,
 | |
| 		replyTo,
 | |
| 		viaBotId,
 | |
| 		postAuthor,
 | |
| 		std::move(markup));
 | |
| 
 | |
| 	const auto spoiler = false;
 | |
| 	_media = std::make_unique<Data::MediaPhoto>(this, photo, spoiler);
 | |
| 	setText(caption);
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	FullReplyTo replyTo,
 | |
| 	UserId viaBotId,
 | |
| 	TimeId date,
 | |
| 	PeerId from,
 | |
| 	const QString &postAuthor,
 | |
| 	not_null<GameData*> game,
 | |
| 	HistoryMessageMarkupData &&markup)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		flags,
 | |
| 		date,
 | |
| 		(flags & MessageFlag::HasFromId) ? from : 0) {
 | |
| 	createComponentsHelper(
 | |
| 		flags,
 | |
| 		replyTo,
 | |
| 		viaBotId,
 | |
| 		postAuthor,
 | |
| 		std::move(markup));
 | |
| 
 | |
| 	_media = std::make_unique<Data::MediaGame>(this, game);
 | |
| 	setTextValue({});
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	Data::SponsoredFrom from,
 | |
| 	const TextWithEntities &textWithEntities,
 | |
| 	HistoryItem *injectedAfter)
 | |
| : HistoryItem(
 | |
| 		history,
 | |
| 		id,
 | |
| 		((history->peer->isChannel() ? MessageFlag::Post : MessageFlag(0))
 | |
| 			//| (from.peer ? MessageFlag::HasFromId : MessageFlag(0))
 | |
| 			| MessageFlag::Local),
 | |
| 		HistoryItem::NewMessageDate(injectedAfter
 | |
| 			? injectedAfter->date()
 | |
| 			: 0),
 | |
| 		/*from.peer ? from.peer->id : */PeerId(0)) {
 | |
| 	_flags |= MessageFlag::Sponsored;
 | |
| 
 | |
| 	const auto webPageType = !from.externalLink.isEmpty()
 | |
| 		? WebPageType::None
 | |
| 		: from.isExactPost
 | |
| 		? WebPageType::Message
 | |
| 		: (from.botLinkInfo && !from.botLinkInfo->botAppName.isEmpty())
 | |
| 		? WebPageType::BotApp
 | |
| 		: from.botLinkInfo
 | |
| 		? WebPageType::Bot
 | |
| 		: from.isBroadcast
 | |
| 		? WebPageType::Channel
 | |
| 		: (from.peer && from.peer->isUser())
 | |
| 		? WebPageType::User
 | |
| 		: WebPageType::Group;
 | |
| 
 | |
| 	const auto webpage = history->peer->owner().webpage(
 | |
| 		history->peer->owner().nextLocalMessageId().bare,
 | |
| 		webPageType,
 | |
| 		from.externalLink,
 | |
| 		from.externalLink,
 | |
| 		from.isRecommended
 | |
| 			? tr::lng_recommended_message_title(tr::now)
 | |
| 			: tr::lng_sponsored_message_title(tr::now),
 | |
| 		from.title,
 | |
| 		textWithEntities,
 | |
| 		(from.webpageOrBotPhotoId
 | |
| 			? history->owner().photo(from.webpageOrBotPhotoId).get()
 | |
| 			: nullptr),
 | |
| 		nullptr,
 | |
| 		WebPageCollage(),
 | |
| 		0,
 | |
| 		QString(),
 | |
| 		false,
 | |
| 		0);
 | |
| 	auto webpageMedia = std::make_unique<Data::MediaWebPage>(
 | |
| 		this,
 | |
| 		webpage,
 | |
| 		MediaWebPageFlag::Sponsored);
 | |
| 	_media = std::move(webpageMedia);
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MessageFlags flags,
 | |
| 	TimeId date,
 | |
| 	PeerId from)
 | |
| : id(id)
 | |
| , _history(history)
 | |
| , _from(from ? history->owner().peer(from) : history->peer)
 | |
| , _flags(FinalizeMessageFlags(flags))
 | |
| , _date(date) {
 | |
| 	if (isHistoryEntry() && IsClientMsgId(id)) {
 | |
| 		_history->registerClientSideMessage(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	not_null<Data::Story*> story)
 | |
| : id(StoryIdToMsgId(story->id()))
 | |
| , _history(history)
 | |
| , _from(history->peer)
 | |
| , _flags(MessageFlag::Local
 | |
| 	| MessageFlag::Outgoing
 | |
| 	| MessageFlag::FakeHistoryItem
 | |
| 	| MessageFlag::StoryItem)
 | |
| , _date(story->date()) {
 | |
| 	setStoryFields(story);
 | |
| }
 | |
| 
 | |
| HistoryItem::~HistoryItem() {
 | |
| 	_media = nullptr;
 | |
| 	clearSavedMedia();
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		reply->clearData(this);
 | |
| 	}
 | |
| 	clearDependencyMessage();
 | |
| 	applyTTL(0);
 | |
| }
 | |
| 
 | |
| TimeId HistoryItem::date() const {
 | |
| 	return _date;
 | |
| }
 | |
| 
 | |
| TimeId HistoryItem::NewMessageDate(TimeId scheduled) {
 | |
| 	return scheduled ? scheduled : base::unixtime::now();
 | |
| }
 | |
| 
 | |
| HistoryServiceDependentData *HistoryItem::GetServiceDependentData() {
 | |
| 	if (const auto pinned = Get<HistoryServicePinned>()) {
 | |
| 		return pinned;
 | |
| 	} else if (const auto gamescore = Get<HistoryServiceGameScore>()) {
 | |
| 		return gamescore;
 | |
| 	} else if (const auto payment = Get<HistoryServicePayment>()) {
 | |
| 		return payment;
 | |
| 	} else if (const auto info = Get<HistoryServiceTopicInfo>()) {
 | |
| 		return info;
 | |
| 	} else if (const auto same = Get<HistoryServiceSameBackground>()) {
 | |
| 		return same;
 | |
| 	} else if (const auto results = Get<HistoryServiceGiveawayResults>()) {
 | |
| 		return results;
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| auto HistoryItem::GetServiceDependentData() const
 | |
| -> const HistoryServiceDependentData * {
 | |
| 	return const_cast<HistoryItem*>(this)->GetServiceDependentData();
 | |
| }
 | |
| 
 | |
| void HistoryItem::dependencyItemRemoved(not_null<HistoryItem*> dependency) {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto documentId = reply->replyToDocumentId;
 | |
| 		reply->itemRemoved(this, dependency);
 | |
| 		if (documentId != reply->replyToDocumentId
 | |
| 			&& generateLocalEntitiesByReply()) {
 | |
| 			_history->owner().requestItemTextRefresh(this);
 | |
| 		}
 | |
| 	} else {
 | |
| 		clearDependencyMessage();
 | |
| 		updateDependentServiceText();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::dependencyStoryRemoved(
 | |
| 		not_null<Data::Story*> dependency) {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto documentId = reply->replyToDocumentId;
 | |
| 		reply->storyRemoved(this, dependency);
 | |
| 		if (documentId != reply->replyToDocumentId
 | |
| 			&& generateLocalEntitiesByReply()) {
 | |
| 			_history->owner().requestItemTextRefresh(this);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateDependencyItem() {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto documentId = reply->replyToDocumentId;
 | |
| 		const auto webpageId = reply->replyToWebPageId;
 | |
| 		reply->updateData(this, true);
 | |
| 		const auto mediaIdChanged = (documentId != reply->replyToDocumentId)
 | |
| 			|| (webpageId != reply->replyToWebPageId);
 | |
| 		if (mediaIdChanged && generateLocalEntitiesByReply()) {
 | |
| 			_history->owner().requestItemTextRefresh(this);
 | |
| 		}
 | |
| 	} else if (GetServiceDependentData()) {
 | |
| 		updateServiceDependent(true);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateDependentServiceText() {
 | |
| 	if (Has<HistoryServicePinned>()) {
 | |
| 		updateServiceText(preparePinnedText());
 | |
| 	} else if (Has<HistoryServiceGameScore>()) {
 | |
| 		updateServiceText(prepareGameScoreText());
 | |
| 	} else if (Has<HistoryServicePayment>()) {
 | |
| 		updateServiceText(preparePaymentSentText());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::updateServiceDependent(bool force) {
 | |
| 	auto dependent = GetServiceDependentData();
 | |
| 	Assert(dependent != nullptr);
 | |
| 
 | |
| 	if (!force) {
 | |
| 		if (!dependent->msgId || dependent->msg) {
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!dependent->lnk) {
 | |
| 		dependent->lnk = JumpToMessageClickHandler(
 | |
| 			(dependent->peerId
 | |
| 				? _history->owner().peer(dependent->peerId)
 | |
| 				: _history->peer),
 | |
| 			dependent->msgId,
 | |
| 			fullId());
 | |
| 	}
 | |
| 	auto gotDependencyItem = false;
 | |
| 	if (!dependent->msg) {
 | |
| 		dependent->msg = _history->owner().message(
 | |
| 			(dependent->peerId
 | |
| 				? dependent->peerId
 | |
| 				: _history->peer->id),
 | |
| 			dependent->msgId);
 | |
| 		if (dependent->msg) {
 | |
| 			if (dependent->msg->isEmpty()) {
 | |
| 				// Really it is deleted.
 | |
| 				dependent->msg = nullptr;
 | |
| 				force = true;
 | |
| 			} else {
 | |
| 				_history->owner().registerDependentMessage(
 | |
| 					this,
 | |
| 					dependent->msg);
 | |
| 				gotDependencyItem = true;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	if (dependent->msg) {
 | |
| 		updateDependentServiceText();
 | |
| 	} else if (force) {
 | |
| 		if (dependent->msgId > 0) {
 | |
| 			dependent->msgId = 0;
 | |
| 			gotDependencyItem = true;
 | |
| 		}
 | |
| 		updateDependentServiceText();
 | |
| 	}
 | |
| 	if (force && gotDependencyItem) {
 | |
| 		Core::App().notifications().checkDelayed();
 | |
| 	}
 | |
| 	return (dependent->msg || !dependent->msgId);
 | |
| }
 | |
| 
 | |
| MsgId HistoryItem::dependencyMsgId() const {
 | |
| 	if (auto dependent = GetServiceDependentData()) {
 | |
| 		return dependent->msgId;
 | |
| 	}
 | |
| 	return replyToId();
 | |
| }
 | |
| 
 | |
| void HistoryItem::checkBuyButton() {
 | |
| 	if (const auto invoice = _media ? _media->invoice() : nullptr) {
 | |
| 		if (invoice->receiptMsgId) {
 | |
| 			replaceBuyWithReceiptInMarkup();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::notificationReady() const {
 | |
| 	if (const auto dependent = GetServiceDependentData()) {
 | |
| 		return (dependent->msg || !dependent->msgId);
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void HistoryItem::finishEdition(int oldKeyboardTop) {
 | |
| 	if (const auto group = _history->owner().groups().find(this)) {
 | |
| 		for (const auto &item : group->items) {
 | |
| 			_history->owner().requestItemViewRefresh(item);
 | |
| 			item->invalidateChatListEntry();
 | |
| 		}
 | |
| 	} else {
 | |
| 		_history->owner().requestItemViewRefresh(this);
 | |
| 		invalidateChatListEntry();
 | |
| 	}
 | |
| 
 | |
| 	// Should be completely redesigned as the oldTop no longer exists.
 | |
| 	//if (oldKeyboardTop >= 0) { // #TODO edit bot message
 | |
| 	//	if (auto keyboard = Get<HistoryMessageReplyMarkup>()) {
 | |
| 	//		keyboard->oldTop = oldKeyboardTop;
 | |
| 	//	}
 | |
| 	//}
 | |
| 
 | |
| 	_history->owner().updateDependentMessages(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setGroupId(MessageGroupId groupId) {
 | |
| 	Expects(!_groupId);
 | |
| 
 | |
| 	_groupId = groupId;
 | |
| 	_history->owner().groups().registerMessage(this);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::checkCommentsLinkedChat(ChannelId id) const {
 | |
| 	if (!id) {
 | |
| 		return true;
 | |
| 	} else if (const auto channel = _history->peer->asChannel()) {
 | |
| 		if (channel->linkedChatKnown()
 | |
| 			|| !(channel->flags() & ChannelDataFlag::HasLink)) {
 | |
| 			const auto linked = channel->linkedChat();
 | |
| 			if (!linked || peerToChannel(linked->id) != id) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void HistoryItem::setReplyMarkup(HistoryMessageMarkupData &&markup) {
 | |
| 	const auto requestUpdate = [&] {
 | |
| 		history()->owner().requestItemResize(this);
 | |
| 		history()->session().changes().messageUpdated(
 | |
| 			this,
 | |
| 			Data::MessageUpdate::Flag::ReplyMarkup);
 | |
| 	};
 | |
| 	if (markup.isNull()) {
 | |
| 		if (_flags & MessageFlag::HasReplyMarkup) {
 | |
| 			_flags &= ~MessageFlag::HasReplyMarkup;
 | |
| 			if (Has<HistoryMessageReplyMarkup>()) {
 | |
| 				RemoveComponents(HistoryMessageReplyMarkup::Bit());
 | |
| 			}
 | |
| 			requestUpdate();
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	// optimization: don't create markup component for the case
 | |
| 	// MTPDreplyKeyboardHide with flags = 0, assume it has f_zero flag
 | |
| 	if (markup.isTrivial()) {
 | |
| 		bool changed = false;
 | |
| 		if (Has<HistoryMessageReplyMarkup>()) {
 | |
| 			RemoveComponents(HistoryMessageReplyMarkup::Bit());
 | |
| 			changed = true;
 | |
| 		}
 | |
| 		if (!(_flags & MessageFlag::HasReplyMarkup)) {
 | |
| 			_flags |= MessageFlag::HasReplyMarkup;
 | |
| 			changed = true;
 | |
| 		}
 | |
| 		if (changed) {
 | |
| 			requestUpdate();
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (!(_flags & MessageFlag::HasReplyMarkup)) {
 | |
| 			_flags |= MessageFlag::HasReplyMarkup;
 | |
| 		}
 | |
| 		if (!Has<HistoryMessageReplyMarkup>()) {
 | |
| 			AddComponents(HistoryMessageReplyMarkup::Bit());
 | |
| 		}
 | |
| 		Get<HistoryMessageReplyMarkup>()->updateData(std::move(markup));
 | |
| 		requestUpdate();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setCommentsInboxReadTill(MsgId readTillId) {
 | |
| 	const auto views = Get<HistoryMessageViews>();
 | |
| 	if (!views) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto newReadTillId = std::max(readTillId.bare, int64(1));
 | |
| 	const auto ignore = (newReadTillId < views->commentsInboxReadTillId);
 | |
| 	if (ignore) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto changed = (newReadTillId > views->commentsInboxReadTillId);
 | |
| 	if (!changed) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto wasUnread = areCommentsUnread();
 | |
| 	views->commentsInboxReadTillId = newReadTillId;
 | |
| 	if (wasUnread && !areCommentsUnread()) {
 | |
| 		_history->owner().requestItemRepaint(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setCommentsMaxId(MsgId maxId) {
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		if (views->commentsMaxId != maxId) {
 | |
| 			const auto wasUnread = areCommentsUnread();
 | |
| 			views->commentsMaxId = maxId;
 | |
| 			if (wasUnread != areCommentsUnread()) {
 | |
| 				_history->owner().requestItemRepaint(this);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setCommentsPossibleMaxId(MsgId possibleMaxId) {
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		if (views->commentsMaxId < possibleMaxId) {
 | |
| 			const auto wasUnread = areCommentsUnread();
 | |
| 			views->commentsMaxId = possibleMaxId;
 | |
| 			if (!wasUnread && areCommentsUnread()) {
 | |
| 				_history->owner().requestItemRepaint(this);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::areCommentsUnread() const {
 | |
| 	const auto views = Get<HistoryMessageViews>();
 | |
| 	if (!views
 | |
| 		|| !views->commentsMegagroupId
 | |
| 		|| !checkCommentsLinkedChat(views->commentsMegagroupId)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	const auto till = views->commentsInboxReadTillId;
 | |
| 	if (views->commentsInboxReadTillId < 2 || views->commentsMaxId <= till) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	const auto group = views->commentsMegagroupId
 | |
| 		? _history->owner().historyLoaded(
 | |
| 			peerFromChannel(views->commentsMegagroupId))
 | |
| 		: _history.get();
 | |
| 	return !group || (views->commentsMaxId > group->inboxReadTillId());
 | |
| }
 | |
| 
 | |
| FullMsgId HistoryItem::commentsItemId() const {
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		return FullMsgId(
 | |
| 			PeerId(views->commentsMegagroupId),
 | |
| 			views->commentsRootId);
 | |
| 	}
 | |
| 	return FullMsgId();
 | |
| }
 | |
| 
 | |
| void HistoryItem::setCommentsItemId(FullMsgId id) {
 | |
| 	if (id.peer == _history->peer->id) {
 | |
| 		if (id.msg != this->id) {
 | |
| 			if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 				reply->setTopMessageId(id.msg);
 | |
| 			}
 | |
| 		}
 | |
| 	} else if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		if (const auto channelId = peerToChannel(id.peer)) {
 | |
| 			if (views->commentsMegagroupId != channelId) {
 | |
| 				views->commentsMegagroupId = channelId;
 | |
| 				_history->owner().requestItemResize(this);
 | |
| 			}
 | |
| 			views->commentsRootId = id.msg;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setServiceText(PreparedServiceText &&prepared) {
 | |
| 	AddComponents(HistoryServiceData::Bit());
 | |
| 	_flags &= ~MessageFlag::HasTextLinks;
 | |
| 	const auto data = Get<HistoryServiceData>();
 | |
| 	const auto had = !_text.empty();
 | |
| 	_text = std::move(prepared.text);
 | |
| 	data->textLinks = std::move(prepared.links);
 | |
| 	if (had) {
 | |
| 		_history->owner().requestItemTextRefresh(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateServiceText(PreparedServiceText &&text) {
 | |
| 	setServiceText(std::move(text));
 | |
| 	_history->owner().requestItemResize(this);
 | |
| 	invalidateChatListEntry();
 | |
| 	_history->owner().updateDependentMessages(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateStoryMentionText() {
 | |
| 	setServiceText(prepareStoryMentionText());
 | |
| }
 | |
| 
 | |
| HistoryMessageReplyMarkup *HistoryItem::inlineReplyMarkup() {
 | |
| 	if (const auto markup = Get<HistoryMessageReplyMarkup>()) {
 | |
| 		if (markup->data.flags & ReplyMarkupFlag::Inline) {
 | |
| 			return markup;
 | |
| 		}
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| ReplyKeyboard *HistoryItem::inlineReplyKeyboard() {
 | |
| 	if (const auto markup = inlineReplyMarkup()) {
 | |
| 		return markup->inlineKeyboard.get();
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| ChannelData *HistoryItem::discussionPostOriginalSender() const {
 | |
| 	if (!_history->peer->isMegagroup()) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		const auto from = forwarded->savedFromPeer;
 | |
| 		if (const auto result = from ? from->asChannel() : nullptr) {
 | |
| 			return result;
 | |
| 		}
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isDiscussionPost() const {
 | |
| 	return (discussionPostOriginalSender() != nullptr);
 | |
| }
 | |
| 
 | |
| HistoryItem *HistoryItem::lookupDiscussionPostOriginal() const {
 | |
| 	if (!_history->peer->isMegagroup()) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 	const auto forwarded = Get<HistoryMessageForwarded>();
 | |
| 	if (!forwarded
 | |
| 		|| !forwarded->savedFromPeer
 | |
| 		|| !forwarded->savedFromMsgId) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 	return _history->owner().message(
 | |
| 		forwarded->savedFromPeer->id,
 | |
| 		forwarded->savedFromMsgId);
 | |
| }
 | |
| 
 | |
| PeerData *HistoryItem::computeDisplayFrom() const {
 | |
| 	if (const auto sender = discussionPostOriginalSender()) {
 | |
| 		return sender;
 | |
| 	} else if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		if (_history->peer->isSelf()
 | |
| 			|| _history->peer->isRepliesChat()
 | |
| 			|| forwarded->imported) {
 | |
| 			return forwarded->originalSender;
 | |
| 		}
 | |
| 	}
 | |
| 	return author().get();
 | |
| }
 | |
| 
 | |
| PeerData *HistoryItem::displayFrom() const {
 | |
| 	if (!(_flags & MessageFlag::DisplayFromChecked)) {
 | |
| 		_flags |= MessageFlag::DisplayFromChecked;
 | |
| 		_displayFrom = computeDisplayFrom();
 | |
| 	}
 | |
| 	return _displayFrom;
 | |
| }
 | |
| 
 | |
| uint8 HistoryItem::colorIndex() const {
 | |
| 	if (const auto from = displayFrom()) {
 | |
| 		return from->colorIndex();
 | |
| 	} else if (const auto info = hiddenSenderInfo()) {
 | |
| 		return info->colorIndex;
 | |
| 	}
 | |
| 	Unexpected("No displayFrom and no hiddenSenderInfo.");
 | |
| }
 | |
| 
 | |
| std::unique_ptr<HistoryView::Element> HistoryItem::createView(
 | |
| 		not_null<HistoryView::ElementDelegate*> delegate,
 | |
| 		HistoryView::Element *replacing) {
 | |
| 	if (isService()) {
 | |
| 		return std::make_unique<HistoryView::Service>(
 | |
| 			delegate,
 | |
| 			this,
 | |
| 			replacing);
 | |
| 	}
 | |
| 	return std::make_unique<HistoryView::Message>(delegate, this, replacing);
 | |
| }
 | |
| 
 | |
| void HistoryItem::invalidateChatListEntry() {
 | |
| 	_history->session().changes().messageUpdated(
 | |
| 		this,
 | |
| 		Data::MessageUpdate::Flag::DialogRowRefresh);
 | |
| 	_history->lastItemDialogsView().itemInvalidated(this);
 | |
| 	if (const auto topic = this->topic()) {
 | |
| 		topic->lastItemDialogsView().itemInvalidated(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::customEmojiRepaint() {
 | |
| 	if (!(_flags & MessageFlag::CustomEmojiRepainting)) {
 | |
| 		_flags |= MessageFlag::CustomEmojiRepainting;
 | |
| 		_history->owner().requestItemRepaint(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::finishEditionToEmpty() {
 | |
| 	finishEdition(-1);
 | |
| 	_history->itemVanished(this);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::hasUnreadMediaFlag() const {
 | |
| 	if (_history->peer->isChannel()) {
 | |
| 		const auto passed = base::unixtime::now() - date();
 | |
| 		const auto &config = _history->session().serverConfig();
 | |
| 		if (passed >= config.channelsReadMediaPeriod) {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	return _flags & MessageFlag::MediaIsUnread;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isUnreadMention() const {
 | |
| 	return !out() && mentionsMe() && (_flags & MessageFlag::MediaIsUnread);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::hasUnreadReaction() const {
 | |
| 	return (_flags & MessageFlag::HasUnreadReaction);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::mentionsMe() const {
 | |
| 	if (Has<HistoryServicePinned>()
 | |
| 		&& !Core::App().settings().notifyAboutPinned()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return _flags & MessageFlag::MentionsMe;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isUnreadMedia() const {
 | |
| 	if (!hasUnreadMediaFlag()) {
 | |
| 		return false;
 | |
| 	} else if (const auto media = this->media()) {
 | |
| 		if (const auto document = media->document()) {
 | |
| 			if (document->isVoiceMessage() || document->isVideoMessage()) {
 | |
| 				return (media->webpage() == nullptr);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isIncomingUnreadMedia() const {
 | |
| 	return !out() && isUnreadMedia();
 | |
| }
 | |
| 
 | |
| void HistoryItem::markMediaAndMentionRead() {
 | |
| 	_flags &= ~MessageFlag::MediaIsUnread;
 | |
| 
 | |
| 	if (mentionsMe()) {
 | |
| 		_history->updateChatListEntry();
 | |
| 		_history->unreadMentions().erase(id);
 | |
| 		if (const auto topic = this->topic()) {
 | |
| 			topic->updateChatListEntry();
 | |
| 			topic->unreadMentions().erase(id);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (const auto selfdestruct = Get<HistoryServiceSelfDestruct>()) {
 | |
| 		if (selfdestruct->destructAt == crl::time()) {
 | |
| 			const auto ttl = selfdestruct->timeToLive;
 | |
| 			if (const auto maybeTime = std::get_if<crl::time>(&ttl)) {
 | |
| 				const auto time = *maybeTime;
 | |
| 				selfdestruct->destructAt = crl::now() + time;
 | |
| 				_history->owner().selfDestructIn(this, time);
 | |
| 			} else {
 | |
| 				selfdestruct->destructAt = TimeToLiveSingleView();
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::markReactionsRead() {
 | |
| 	if (_reactions) {
 | |
| 		_reactions->markRead();
 | |
| 	}
 | |
| 	_flags &= ~MessageFlag::HasUnreadReaction;
 | |
| 	_history->updateChatListEntry();
 | |
| 	_history->unreadReactions().erase(id);
 | |
| 	if (const auto topic = this->topic()) {
 | |
| 		topic->updateChatListEntry();
 | |
| 		topic->unreadReactions().erase(id);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::markContentsRead(bool fromThisClient) {
 | |
| 	if (hasUnreadReaction()) {
 | |
| 		if (fromThisClient) {
 | |
| 			_history->owner().requestUnreadReactionsAnimation(this);
 | |
| 		}
 | |
| 		markReactionsRead();
 | |
| 		return true;
 | |
| 	} else if (isUnreadMention() || isIncomingUnreadMedia()) {
 | |
| 		markMediaAndMentionRead();
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void HistoryItem::setIsPinned(bool pinned) {
 | |
| 	const auto changed = (isPinned() != pinned);
 | |
| 	if (pinned) {
 | |
| 		_flags |= MessageFlag::Pinned;
 | |
| 		auto &storage = _history->session().storage();
 | |
| 		storage.add(Storage::SharedMediaAddExisting(
 | |
| 			_history->peer->id,
 | |
| 			MsgId(0), // topicRootId
 | |
| 			Storage::SharedMediaType::Pinned,
 | |
| 			id,
 | |
| 			{ id, id }));
 | |
| 		_history->setHasPinnedMessages(true);
 | |
| 		if (const auto topic = this->topic()) {
 | |
| 			storage.add(Storage::SharedMediaAddExisting(
 | |
| 				_history->peer->id,
 | |
| 				topic->rootId(),
 | |
| 				Storage::SharedMediaType::Pinned,
 | |
| 				id,
 | |
| 				{ id, id }));
 | |
| 			topic->setHasPinnedMessages(true);
 | |
| 		}
 | |
| 	} else {
 | |
| 		_flags &= ~MessageFlag::Pinned;
 | |
| 		_history->session().storage().remove(Storage::SharedMediaRemoveOne(
 | |
| 			_history->peer->id,
 | |
| 			Storage::SharedMediaType::Pinned,
 | |
| 			id));
 | |
| 	}
 | |
| 	if (changed) {
 | |
| 		_history->owner().notifyItemDataChange(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::returnSavedMedia() {
 | |
| 	if (!isEditingMedia()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto wasGrouped = history()->owner().groups().isGrouped(this);
 | |
| 	_media = std::move(_savedLocalEditMediaData->media);
 | |
| 	setText(_savedLocalEditMediaData->text);
 | |
| 	clearSavedMedia();
 | |
| 	if (wasGrouped) {
 | |
| 		history()->owner().groups().refreshMessage(this, true);
 | |
| 	} else {
 | |
| 		history()->owner().requestItemViewRefresh(this);
 | |
| 		history()->owner().updateDependentMessages(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::savePreviousMedia() {
 | |
| 	Expects(_media != nullptr);
 | |
| 
 | |
| 	using Data = SavedMediaData;
 | |
| 	_savedLocalEditMediaData = std::make_unique<Data>(Data{
 | |
| 		.text = originalText(),
 | |
| 		.media = _media->clone(this),
 | |
| 	});
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isEditingMedia() const {
 | |
| 	return _savedLocalEditMediaData != nullptr;
 | |
| }
 | |
| 
 | |
| void HistoryItem::clearSavedMedia() {
 | |
| 	_savedLocalEditMediaData = nullptr;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::definesReplyKeyboard() const {
 | |
| 	if (const auto markup = Get<HistoryMessageReplyMarkup>()) {
 | |
| 		if (markup->data.flags & ReplyMarkupFlag::Inline) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	// optimization: don't create markup component for the case
 | |
| 	// MTPDreplyKeyboardHide with flags = 0, assume it has f_zero flag
 | |
| 	return (_flags & MessageFlag::HasReplyMarkup);
 | |
| }
 | |
| 
 | |
| ReplyMarkupFlags HistoryItem::replyKeyboardFlags() const {
 | |
| 	Expects(definesReplyKeyboard());
 | |
| 
 | |
| 	if (const auto markup = Get<HistoryMessageReplyMarkup>()) {
 | |
| 		return markup->data.flags;
 | |
| 	}
 | |
| 
 | |
| 	// optimization: don't create markup component for the case
 | |
| 	// MTPDreplyKeyboardHide with flags = 0, assume it has f_zero flag
 | |
| 	return ReplyMarkupFlag::None;
 | |
| }
 | |
| 
 | |
| void HistoryItem::addLogEntryOriginal(
 | |
| 		WebPageId localId,
 | |
| 		const QString &label,
 | |
| 		const TextWithEntities &content) {
 | |
| 	Expects(isAdminLogEntry());
 | |
| 
 | |
| 	AddComponents(HistoryMessageLogEntryOriginal::Bit());
 | |
| 	Get<HistoryMessageLogEntryOriginal>()->page = _history->owner().webpage(
 | |
| 		localId,
 | |
| 		label,
 | |
| 		content);
 | |
| }
 | |
| 
 | |
| PeerData *HistoryItem::specialNotificationPeer() const {
 | |
| 	return (mentionsMe() && !_history->peer->isUser())
 | |
| 		? from().get()
 | |
| 		: nullptr;
 | |
| }
 | |
| 
 | |
| UserData *HistoryItem::viaBot() const {
 | |
| 	if (const auto via = Get<HistoryMessageVia>()) {
 | |
| 		return via->bot;
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| UserData *HistoryItem::getMessageBot() const {
 | |
| 	if (const auto bot = viaBot()) {
 | |
| 		return bot;
 | |
| 	}
 | |
| 	auto bot = from()->asUser();
 | |
| 	if (!bot) {
 | |
| 		bot = _history->peer->asUser();
 | |
| 	}
 | |
| 	return (bot && bot->isBot()) ? bot : nullptr;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isHistoryEntry() const {
 | |
| 	return (_flags & MessageFlag::HistoryEntry);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isAdminLogEntry() const {
 | |
| 	return (_flags & MessageFlag::AdminLogEntry);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isFromScheduled() const {
 | |
| 	return isHistoryEntry()
 | |
| 		&& (_flags & MessageFlag::IsOrWasScheduled);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isScheduled() const {
 | |
| 	return !isHistoryEntry()
 | |
| 		&& !isAdminLogEntry()
 | |
| 		&& (_flags & MessageFlag::IsOrWasScheduled);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isSponsored() const {
 | |
| 	return _flags & MessageFlag::Sponsored;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::skipNotification() const {
 | |
| 	if (isSilent() && (_flags & MessageFlag::IsContactSignUp)) {
 | |
| 		return true;
 | |
| 	} else if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		if (forwarded->imported) {
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isUserpicSuggestion() const {
 | |
| 	return (_flags & MessageFlag::IsUserpicSuggestion);
 | |
| }
 | |
| 
 | |
| void HistoryItem::destroy() {
 | |
| 	_history->destroyMessage(this);
 | |
| }
 | |
| 
 | |
| not_null<Data::Thread*> HistoryItem::notificationThread() const {
 | |
| 	if (const auto rootId = topicRootId()) {
 | |
| 		if (const auto forum = _history->asForum()) {
 | |
| 			return forum->enforceTopicFor(rootId);
 | |
| 		}
 | |
| 	}
 | |
| 	return _history;
 | |
| }
 | |
| 
 | |
| Data::ForumTopic *HistoryItem::topic() const {
 | |
| 	if (const auto rootId = topicRootId()) {
 | |
| 		if (const auto forum = _history->asForum()) {
 | |
| 			return forum->topicFor(rootId);
 | |
| 		}
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| void HistoryItem::refreshMainView() {
 | |
| 	if (const auto view = mainView()) {
 | |
| 		_history->owner().notifyHistoryChangeDelayed(_history);
 | |
| 		view->refreshInBlock();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::removeMainView() {
 | |
| 	if (const auto view = mainView()) {
 | |
| 		_history->owner().notifyHistoryChangeDelayed(_history);
 | |
| 		view->removeFromBlock();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::clearMainView() {
 | |
| 	_mainView = nullptr;
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyEdition(HistoryMessageEdition &&edition) {
 | |
| 	int keyboardTop = -1;
 | |
| 	//if (!pendingResize()) {// #TODO edit bot message
 | |
| 	//	if (auto keyboard = inlineReplyKeyboard()) {
 | |
| 	//		int h = st::msgBotKbButton.margin + keyboard->naturalHeight();
 | |
| 	//		keyboardTop = _height - h + st::msgBotKbButton.margin - marginBottom();
 | |
| 	//	}
 | |
| 	//}
 | |
| 
 | |
| 	const auto updatingSavedLocalEdit = !edition.savePreviousMedia
 | |
| 		&& (_savedLocalEditMediaData != nullptr);
 | |
| 	if (!_savedLocalEditMediaData && edition.savePreviousMedia) {
 | |
| 		savePreviousMedia();
 | |
| 	}
 | |
| 	Assert(!updatingSavedLocalEdit || !isLocalUpdateMedia());
 | |
| 
 | |
| 	if (edition.isEditHide) {
 | |
| 		_flags |= MessageFlag::HideEdited;
 | |
| 	} else {
 | |
| 		_flags &= ~MessageFlag::HideEdited;
 | |
| 	}
 | |
| 	if (edition.invertMedia) {
 | |
| 		_flags |= MessageFlag::InvertMedia;
 | |
| 	} else {
 | |
| 		_flags &= ~MessageFlag::InvertMedia;
 | |
| 	}
 | |
| 
 | |
| 	if (edition.editDate != -1) {
 | |
| 		//_flags |= MTPDmessage::Flag::f_edit_date;
 | |
| 		if (!Has<HistoryMessageEdited>()) {
 | |
| 			AddComponents(HistoryMessageEdited::Bit());
 | |
| 		}
 | |
| 		auto edited = Get<HistoryMessageEdited>();
 | |
| 		edited->date = edition.editDate;
 | |
| 	}
 | |
| 
 | |
| 	if (!edition.useSameMarkup) {
 | |
| 		setReplyMarkup(base::take(edition.replyMarkup));
 | |
| 	}
 | |
| 	if (updatingSavedLocalEdit) {
 | |
| 		_savedLocalEditMediaData->media = edition.mtpMedia
 | |
| 			? CreateMedia(this, *edition.mtpMedia)
 | |
| 			: nullptr;
 | |
| 	} else {
 | |
| 		removeFromSharedMediaIndex();
 | |
| 		refreshMedia(edition.mtpMedia);
 | |
| 	}
 | |
| 	if (!edition.useSameReactions) {
 | |
| 		updateReactions(edition.mtpReactions);
 | |
| 	}
 | |
| 	if (!edition.useSameViews) {
 | |
| 		changeViewsCount(edition.views);
 | |
| 	}
 | |
| 	if (!edition.useSameForwards) {
 | |
| 		setForwardsCount(edition.forwards);
 | |
| 	}
 | |
| 	const auto &checkedMedia = updatingSavedLocalEdit
 | |
| 		? _savedLocalEditMediaData->media
 | |
| 		: _media;
 | |
| 	auto updatedText = checkedMedia
 | |
| 		? edition.textWithEntities
 | |
| 		: EnsureNonEmpty(edition.textWithEntities);
 | |
| 	if (updatingSavedLocalEdit) {
 | |
| 		_savedLocalEditMediaData->text = std::move(updatedText);
 | |
| 	} else {
 | |
| 		setText(std::move(updatedText));
 | |
| 		addToSharedMediaIndex();
 | |
| 	}
 | |
| 	if (!edition.useSameReplies) {
 | |
| 		if (!edition.replies.isNull) {
 | |
| 			if (checkRepliesPts(edition.replies)) {
 | |
| 				setReplies(base::take(edition.replies));
 | |
| 			}
 | |
| 		} else {
 | |
| 			clearReplies();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	applyTTL(edition.ttl);
 | |
| 
 | |
| 	finishEdition(keyboardTop);
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyChanges(not_null<Data::Story*> story) {
 | |
| 	Expects(_flags & MessageFlag::StoryItem);
 | |
| 	Expects(StoryIdFromMsgId(id) == story->id());
 | |
| 
 | |
| 	_media = nullptr;
 | |
| 	setStoryFields(story);
 | |
| 
 | |
| 	finishEdition(-1);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setStoryFields(not_null<Data::Story*> story) {
 | |
| 	const auto spoiler = false;
 | |
| 	if (const auto photo = story->photo()) {
 | |
| 		_media = std::make_unique<Data::MediaPhoto>(this, photo, spoiler);
 | |
| 	} else if (const auto document = story->document()) {
 | |
| 		_media = std::make_unique<Data::MediaFile>(
 | |
| 			this,
 | |
| 			document,
 | |
| 			/*skipPremiumEffect=*/false,
 | |
| 			spoiler);
 | |
| 	}
 | |
| 	setText(story->caption());
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyEdition(const MTPDmessageService &message) {
 | |
| 	if (message.vaction().type() == mtpc_messageActionHistoryClear) {
 | |
| 		const auto wasGrouped = history()->owner().groups().isGrouped(this);
 | |
| 		setReplyMarkup({});
 | |
| 		removeFromSharedMediaIndex();
 | |
| 		refreshMedia(nullptr);
 | |
| 		setTextValue({});
 | |
| 		changeViewsCount(-1);
 | |
| 		setForwardsCount(-1);
 | |
| 		if (wasGrouped) {
 | |
| 			history()->owner().groups().unregisterMessage(this);
 | |
| 		}
 | |
| 		if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 			reply->clearData(this);
 | |
| 		}
 | |
| 		clearDependencyMessage();
 | |
| 		UpdateComponents(0);
 | |
| 		createServiceFromMtp(message);
 | |
| 		applyServiceDateEdition(message);
 | |
| 		finishEditionToEmpty();
 | |
| 		_flags &= ~MessageFlag::DisplayFromChecked;
 | |
| 	} else if (isService()) {
 | |
| 		if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 			reply->clearData(this);
 | |
| 		}
 | |
| 		clearDependencyMessage();
 | |
| 		UpdateComponents(0);
 | |
| 		createServiceFromMtp(message);
 | |
| 		applyServiceDateEdition(message);
 | |
| 		finishEdition(-1);
 | |
| 		_flags &= ~MessageFlag::DisplayFromChecked;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyEdition(const MTPMessageExtendedMedia &media) {
 | |
| 	if (const auto existing = this->media()) {
 | |
| 		if (existing->updateExtendedMedia(this, media)) {
 | |
| 			checkBuyButton();
 | |
| 			finishEdition(-1);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::applySentMessage(const MTPDmessage &data) {
 | |
| 	updateSentContent({
 | |
| 		qs(data.vmessage()),
 | |
| 		Api::EntitiesFromMTP(
 | |
| 			&_history->session(),
 | |
| 			data.ventities().value_or_empty())
 | |
| 	}, data.vmedia());
 | |
| 	updateReplyMarkup(HistoryMessageMarkupData(data.vreply_markup()));
 | |
| 	updateForwardedInfo(data.vfwd_from());
 | |
| 	changeViewsCount(data.vviews().value_or(-1));
 | |
| 	if (const auto replies = data.vreplies()) {
 | |
| 		setReplies(HistoryMessageRepliesData(replies));
 | |
| 	} else {
 | |
| 		clearReplies();
 | |
| 	}
 | |
| 	setForwardsCount(data.vforwards().value_or(-1));
 | |
| 	if (const auto reply = data.vreply_to()) {
 | |
| 		reply->match([&](const MTPDmessageReplyHeader &data) {
 | |
| 			const auto replyToPeer = data.vreply_to_peer_id()
 | |
| 				? peerFromMTP(*data.vreply_to_peer_id())
 | |
| 				: PeerId();
 | |
| 			if (!replyToPeer || replyToPeer == history()->peer->id) {
 | |
| 				if (const auto replyToId = data.vreply_to_msg_id()) {
 | |
| 					setReplyFields(
 | |
| 						replyToId->v,
 | |
| 						data.vreply_to_top_id().value_or(replyToId->v),
 | |
| 						data.is_forum_topic());
 | |
| 				}
 | |
| 			}
 | |
| 		}, [](const MTPDmessageReplyStoryHeader &data) {
 | |
| 		});
 | |
| 	}
 | |
| 	setPostAuthor(data.vpost_author().value_or_empty());
 | |
| 	setIsPinned(data.is_pinned());
 | |
| 	contributeToSlowmode(data.vdate().v);
 | |
| 	addToSharedMediaIndex();
 | |
| 	invalidateChatListEntry();
 | |
| 	if (const auto period = data.vttl_period(); period && period->v > 0) {
 | |
| 		applyTTL(data.vdate().v + period->v);
 | |
| 	} else {
 | |
| 		applyTTL(0);
 | |
| 	}
 | |
| 	_history->owner().notifyItemDataChange(this);
 | |
| 	_history->owner().requestItemTextRefresh(this);
 | |
| 	_history->owner().updateDependentMessages(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::applySentMessage(
 | |
| 		const QString &text,
 | |
| 		const MTPDupdateShortSentMessage &data,
 | |
| 		bool wasAlready) {
 | |
| 	updateSentContent({
 | |
| 		text,
 | |
| 		Api::EntitiesFromMTP(
 | |
| 			&_history->session(),
 | |
| 			data.ventities().value_or_empty())
 | |
| 		}, data.vmedia());
 | |
| 	contributeToSlowmode(data.vdate().v);
 | |
| 	if (!wasAlready) {
 | |
| 		addToSharedMediaIndex();
 | |
| 	}
 | |
| 	invalidateChatListEntry();
 | |
| 	if (const auto period = data.vttl_period(); period && period->v > 0) {
 | |
| 		applyTTL(data.vdate().v + period->v);
 | |
| 	} else {
 | |
| 		applyTTL(0);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateSentContent(
 | |
| 		const TextWithEntities &textWithEntities,
 | |
| 		const MTPMessageMedia *media) {
 | |
| 	if (_savedLocalEditMediaData) {
 | |
| 		return;
 | |
| 	}
 | |
| 	setText(textWithEntities);
 | |
| 	if (_flags & MessageFlag::FromInlineBot) {
 | |
| 		if (!media || !_media || !_media->updateInlineResultMedia(*media)) {
 | |
| 			refreshSentMedia(media);
 | |
| 		}
 | |
| 		_flags &= ~MessageFlag::FromInlineBot;
 | |
| 	} else if (media || _media) {
 | |
| 		if (!media || !_media || !_media->updateSentMedia(*media)) {
 | |
| 			refreshSentMedia(media);
 | |
| 		}
 | |
| 	}
 | |
| 	history()->owner().requestItemResize(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateForwardedInfo(const MTPMessageFwdHeader *fwd) {
 | |
| 	const auto forwarded = Get<HistoryMessageForwarded>();
 | |
| 	if (!fwd) {
 | |
| 		if (forwarded) {
 | |
| 			LOG(("API Error: Server removed forwarded information."));
 | |
| 		}
 | |
| 		return;
 | |
| 	} else if (!forwarded) {
 | |
| 		LOG(("API Error: Server added forwarded information."));
 | |
| 		return;
 | |
| 	}
 | |
| 	fwd->match([&](const MTPDmessageFwdHeader &data) {
 | |
| 		auto config = CreateConfig();
 | |
| 		FillForwardedInfo(config, data);
 | |
| 		setupForwardedComponent(config);
 | |
| 		history()->owner().requestItemResize(this);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyEditionToHistoryCleared() {
 | |
| 	applyEdition(
 | |
| 		MTP_messageService(
 | |
| 			MTP_flags(0),
 | |
| 			MTP_int(id),
 | |
| 			peerToMTP(PeerId(0)), // from_id
 | |
| 			peerToMTP(_history->peer->id),
 | |
| 			MTPMessageReplyHeader(),
 | |
| 			MTP_int(date()),
 | |
| 			MTP_messageActionHistoryClear(),
 | |
| 			MTPint() // ttl_period
 | |
| 		).c_messageService());
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateReplyMarkup(HistoryMessageMarkupData &&markup) {
 | |
| 	setReplyMarkup(std::move(markup));
 | |
| }
 | |
| 
 | |
| void HistoryItem::contributeToSlowmode(TimeId realDate) {
 | |
| 	if (const auto channel = history()->peer->asChannel()) {
 | |
| 		if (out() && isRegular() && !isService()) {
 | |
| 			channel->growSlowmodeLastMessage(realDate ? realDate : date());
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::addToUnreadThings(HistoryUnreadThings::AddType type) {
 | |
| 	if (!isRegular()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto mention = isUnreadMention();
 | |
| 	const auto reaction = hasUnreadReaction();
 | |
| 	if (!mention && !reaction) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto topic = this->topic();
 | |
| 	const auto history = this->history();
 | |
| 	const auto changes = &history->session().changes();
 | |
| 	if (mention) {
 | |
| 		if (history->unreadMentions().add(id, type)) {
 | |
| 			changes->historyUpdated(
 | |
| 				history,
 | |
| 				Data::HistoryUpdate::Flag::UnreadMentions);
 | |
| 		}
 | |
| 		if (topic && topic->unreadMentions().add(id, type)) {
 | |
| 			changes->topicUpdated(
 | |
| 				topic,
 | |
| 				Data::TopicUpdate::Flag::UnreadMentions);
 | |
| 		}
 | |
| 	}
 | |
| 	if (reaction) {
 | |
| 		const auto toHistory = history->unreadReactions().add(id, type);
 | |
| 		const auto toTopic = topic && topic->unreadReactions().add(id, type);
 | |
| 		if (toHistory || toTopic) {
 | |
| 			if (type == HistoryUnreadThings::AddType::New) {
 | |
| 				changes->messageUpdated(
 | |
| 					this,
 | |
| 					Data::MessageUpdate::Flag::NewUnreadReaction);
 | |
| 			}
 | |
| 			if (hasUnreadReaction()) {
 | |
| 				if (toHistory) {
 | |
| 					changes->historyUpdated(
 | |
| 						history,
 | |
| 						Data::HistoryUpdate::Flag::UnreadReactions);
 | |
| 				}
 | |
| 				if (toTopic) {
 | |
| 					changes->topicUpdated(
 | |
| 						topic,
 | |
| 						Data::TopicUpdate::Flag::UnreadReactions);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::destroyHistoryEntry() {
 | |
| 	if (isUnreadMention()) {
 | |
| 		history()->unreadMentions().erase(id);
 | |
| 		if (const auto topic = this->topic()) {
 | |
| 			topic->unreadMentions().erase(id);
 | |
| 		}
 | |
| 	}
 | |
| 	if (hasUnreadReaction()) {
 | |
| 		history()->unreadReactions().erase(id);
 | |
| 		if (const auto topic = this->topic()) {
 | |
| 			topic->unreadReactions().erase(id);
 | |
| 		}
 | |
| 	}
 | |
| 	if (isRegular() && _history->peer->isMegagroup()) {
 | |
| 		if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 			changeReplyToTopCounter(reply, -1);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| Storage::SharedMediaTypesMask HistoryItem::sharedMediaTypes() const {
 | |
| 	auto result = Storage::SharedMediaTypesMask {};
 | |
| 	const auto media = _savedLocalEditMediaData
 | |
| 		? _savedLocalEditMediaData->media.get()
 | |
| 		: _media.get();
 | |
| 	if (media) {
 | |
| 		result.set(media->sharedMediaTypes());
 | |
| 	}
 | |
| 	if (hasTextLinks()) {
 | |
| 		result.set(Storage::SharedMediaType::Link);
 | |
| 	}
 | |
| 	if (isPinned()) {
 | |
| 		result.set(Storage::SharedMediaType::Pinned);
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void HistoryItem::indexAsNewItem() {
 | |
| 	if (isRegular()) {
 | |
| 		addToUnreadThings(HistoryUnreadThings::AddType::New);
 | |
| 	}
 | |
| 	addToSharedMediaIndex();
 | |
| }
 | |
| 
 | |
| void HistoryItem::addToSharedMediaIndex() {
 | |
| 	if (isRegular()) {
 | |
| 		if (const auto types = sharedMediaTypes()) {
 | |
| 			_history->session().storage().add(Storage::SharedMediaAddNew(
 | |
| 				_history->peer->id,
 | |
| 				topicRootId(),
 | |
| 				types,
 | |
| 				id));
 | |
| 			if (types.test(Storage::SharedMediaType::Pinned)) {
 | |
| 				_history->setHasPinnedMessages(true);
 | |
| 				if (const auto topic = this->topic()) {
 | |
| 					topic->setHasPinnedMessages(true);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::removeFromSharedMediaIndex() {
 | |
| 	if (isRegular()) {
 | |
| 		if (const auto types = sharedMediaTypes()) {
 | |
| 			_history->session().storage().remove(
 | |
| 				Storage::SharedMediaRemoveOne(
 | |
| 					_history->peer->id,
 | |
| 					types,
 | |
| 					id));
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::incrementReplyToTopCounter() {
 | |
| 	if (isRegular() && _history->peer->isMegagroup()) {
 | |
| 		_history->session().changes().messageUpdated(
 | |
| 			this,
 | |
| 			Data::MessageUpdate::Flag::ReplyToTopAdded);
 | |
| 		if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 			changeReplyToTopCounter(reply, 1);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::changeReplyToTopCounter(
 | |
| 		not_null<HistoryMessageReply*> reply,
 | |
| 		int delta) {
 | |
| 	const auto topId = reply->topMessageId();
 | |
| 	if (!topId) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto top = _history->owner().message(_history->peer->id, topId);
 | |
| 	if (!top) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto from = displayFrom();
 | |
| 	const auto replier = from ? from->id : PeerId();
 | |
| 	top->changeRepliesCount(delta, replier);
 | |
| 	if (const auto original = top->lookupDiscussionPostOriginal()) {
 | |
| 		original->changeRepliesCount(delta, replier);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QString HistoryItem::notificationHeader() const {
 | |
| 	if (isService()) {
 | |
| 		return QString();
 | |
| 	} else if (out() && isFromScheduled() && !_history->peer->isSelf()) {
 | |
| 		return tr::lng_from_you(tr::now);
 | |
| 	} else if (!_history->peer->isUser() && !isPost()) {
 | |
| 		return from()->name();
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| void HistoryItem::setRealId(MsgId newId) {
 | |
| 	Expects(_flags & MessageFlag::BeingSent);
 | |
| 	Expects(IsClientMsgId(id));
 | |
| 
 | |
| 	const auto oldId = std::exchange(id, newId);
 | |
| 	_flags &= ~(MessageFlag::BeingSent | MessageFlag::Local);
 | |
| 	if (isRegular()) {
 | |
| 		_history->unregisterClientSideMessage(this);
 | |
| 	}
 | |
| 	_history->owner().notifyItemIdChange({ fullId(), oldId });
 | |
| 
 | |
| 	// We don't fire MessageUpdate::Flag::ReplyMarkup and update keyboard
 | |
| 	// in history widget, because it can't exist for an outgoing message.
 | |
| 	// Only inline keyboards can be in outgoing messages.
 | |
| 	if (const auto markup = inlineReplyMarkup()) {
 | |
| 		if (markup->inlineKeyboard) {
 | |
| 			markup->inlineKeyboard->updateMessageId();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	_history->owner().notifyItemDataChange(this);
 | |
| 	_history->owner().groups().refreshMessage(this);
 | |
| 	_history->owner().requestItemResize(this);
 | |
| 
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		incrementReplyToTopCounter();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canPin() const {
 | |
| 	if (!isRegular() || isService()) {
 | |
| 		return false;
 | |
| 	} else if (const auto m = media(); m && m->call()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return _history->peer->canPinMessages();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::allowsSendNow() const {
 | |
| 	return !isService()
 | |
| 		&& isScheduled()
 | |
| 		&& !isSending()
 | |
| 		&& !hasFailed()
 | |
| 		&& !isEditingMedia();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::allowsForward() const {
 | |
| 	return !isService()
 | |
| 		&& isRegular()
 | |
| 		&& !forbidsForward()
 | |
| 		&& history()->peer->allowsForwarding()
 | |
| 		&& (!_media || _media->allowsForward());
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isTooOldForEdit(TimeId now) const {
 | |
| 	return !_history->peer->canEditMessagesIndefinitely()
 | |
| 		&& !isScheduled()
 | |
| 		&& (now - date() >= _history->session().serverConfig().editTimeLimit);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::allowsEdit(TimeId now) const {
 | |
| 	return !isService()
 | |
| 		&& canBeEdited()
 | |
| 		&& !isTooOldForEdit(now)
 | |
| 		&& (!_media || _media->allowsEdit())
 | |
| 		&& !isLegacyMessage()
 | |
| 		&& !isEditingMedia();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canBeEdited() const {
 | |
| 	if ((!isRegular() && !isScheduled())
 | |
| 		|| Has<HistoryMessageVia>()
 | |
| 		|| Has<HistoryMessageForwarded>()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	const auto peer = _history->peer;
 | |
| 	if (peer->isSelf()) {
 | |
| 		return true;
 | |
| 	} else if (const auto channel = peer->asChannel()) {
 | |
| 		if (isPost() && channel->canEditMessages()) {
 | |
| 			return true;
 | |
| 		} else if (out()) {
 | |
| 			if (isPost()) {
 | |
| 				return channel->canPostMessages();
 | |
| 			} else if (const auto topic = this->topic()) {
 | |
| 				return Data::CanSendAnything(topic);
 | |
| 			} else {
 | |
| 				return Data::CanSendAnything(channel);
 | |
| 			}
 | |
| 		} else {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	return out();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canStopPoll() const {
 | |
| 	return canBeEdited() && isRegular();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::forbidsForward() const {
 | |
| 	return (_flags & MessageFlag::NoForwards);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::forbidsSaving() const {
 | |
| 	if (forbidsForward()) {
 | |
| 		return true;
 | |
| 	} else if (const auto invoice = _media ? _media->invoice() : nullptr) {
 | |
| 		return (invoice->extendedMedia != nullptr);
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canDelete() const {
 | |
| 	if (isSponsored()) {
 | |
| 		return false;
 | |
| 	} else if (IsStoryMsgId(id)) {
 | |
| 		return false;
 | |
| 	} else if (isService() && !isRegular()) {
 | |
| 		return false;
 | |
| 	} else if (topicRootId() == id) {
 | |
| 		return false;
 | |
| 	} else if (!isHistoryEntry() && !isScheduled()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	auto channel = _history->peer->asChannel();
 | |
| 	if (!channel) {
 | |
| 		return !isGroupMigrate();
 | |
| 	}
 | |
| 
 | |
| 	if (id == 1) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (channel->canDeleteMessages()) {
 | |
| 		return true;
 | |
| 	} else if (out() && !isService()) {
 | |
| 		return isPost() ? channel->canPostMessages() : true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canDeleteForEveryone(TimeId now) const {
 | |
| 	const auto peer = _history->peer;
 | |
| 	const auto &config = _history->session().serverConfig();
 | |
| 	const auto messageToMyself = peer->isSelf();
 | |
| 	const auto messageTooOld = messageToMyself
 | |
| 		? false
 | |
| 		: peer->isUser()
 | |
| 		? (now - date() >= config.revokePrivateTimeLimit)
 | |
| 		: (now - date() >= config.revokeTimeLimit);
 | |
| 	if (!isRegular() || messageToMyself || messageTooOld || isPost()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (peer->isChannel()) {
 | |
| 		return false;
 | |
| 	} else if (const auto user = peer->asUser()) {
 | |
| 		// Bots receive all messages and there is no sense in revoking them.
 | |
| 		// See https://github.com/telegramdesktop/tdesktop/issues/3818
 | |
| 		if (user->isBot() && !user->isSupport()) {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	if (const auto media = this->media()) {
 | |
| 		if (!media->allowsRevoke(now)) {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!out()) {
 | |
| 		if (const auto chat = peer->asChat()) {
 | |
| 			if (!chat->canDeleteMessages()) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		} else if (peer->isUser()) {
 | |
| 			return config.revokePrivateInbox;
 | |
| 		} else {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::suggestReport() const {
 | |
| 	if (out() || isService() || !isRegular()) {
 | |
| 		return false;
 | |
| 	} else if (const auto channel = _history->peer->asChannel()) {
 | |
| 		return true;
 | |
| 	} else if (const auto user = _history->peer->asUser()) {
 | |
| 		return user->isBot();
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::suggestBanReport() const {
 | |
| 	const auto channel = _history->peer->asChannel();
 | |
| 	if (!channel || !channel->canRestrictParticipant(from())) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return !isPost() && !out();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::suggestDeleteAllReport() const {
 | |
| 	auto channel = _history->peer->asChannel();
 | |
| 	if (!channel || !channel->canDeleteMessages()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return !isPost() && !out();
 | |
| }
 | |
| 
 | |
| ChatRestriction HistoryItem::requiredSendRight() const {
 | |
| 	const auto media = this->media();
 | |
| 	if (media && media->game()) {
 | |
| 		return ChatRestriction::SendGames;
 | |
| 	}
 | |
| 	const auto photo = (media && !media->webpage())
 | |
| 		? media->photo()
 | |
| 		: nullptr;
 | |
| 	const auto document = (media && !media->webpage())
 | |
| 		? media->document()
 | |
| 		: nullptr;
 | |
| 	if (photo) {
 | |
| 		return ChatRestriction::SendPhotos;
 | |
| 	} else if (document) {
 | |
| 		return document->requiredSendRight();
 | |
| 	} else if (media && media->poll()) {
 | |
| 		return ChatRestriction::SendPolls;
 | |
| 	}
 | |
| 	return ChatRestriction::SendOther;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::requiresSendInlineRight() const {
 | |
| 	return Has<HistoryMessageVia>();
 | |
| }
 | |
| 
 | |
| std::optional<QString> HistoryItem::errorTextForForward(
 | |
| 		not_null<Data::Thread*> to) const {
 | |
| 	const auto requiredRight = requiredSendRight();
 | |
| 	const auto requiresInline = requiresSendInlineRight();
 | |
| 	const auto peer = to->peer();
 | |
| 	constexpr auto kInline = ChatRestriction::SendInline;
 | |
| 	if (const auto error = Data::RestrictionError(peer, requiredRight)) {
 | |
| 		return *error;
 | |
| 	} else if (requiresInline && !Data::CanSend(to, kInline)) {
 | |
| 		return Data::RestrictionError(peer, kInline).value_or(
 | |
| 			tr::lng_forward_cant(tr::now));
 | |
| 	} else if (_media
 | |
| 		&& _media->poll()
 | |
| 		&& _media->poll()->publicVotes()
 | |
| 		&& peer->isBroadcast()) {
 | |
| 		return tr::lng_restricted_send_public_polls(tr::now);
 | |
| 	} else if (!Data::CanSend(to, requiredRight, false)) {
 | |
| 		return tr::lng_forward_cant(tr::now);
 | |
| 	}
 | |
| 	return {};
 | |
| }
 | |
| 
 | |
| const HistoryMessageTranslation *HistoryItem::translation() const {
 | |
| 	return Get<HistoryMessageTranslation>();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::translationShowRequiresCheck(LanguageId to) const {
 | |
| 	// Check if a call to translationShowRequiresRequest(to) is not a no-op.
 | |
| 	if (!to) {
 | |
| 		if (const auto translation = Get<HistoryMessageTranslation>()) {
 | |
| 			return (!translation->failed && translation->text.empty())
 | |
| 				|| translation->used;
 | |
| 		}
 | |
| 		return false;
 | |
| 	} else if (const auto translation = Get<HistoryMessageTranslation>()) {
 | |
| 		if (translation->to == to) {
 | |
| 			return !translation->used && !translation->text.empty();
 | |
| 		}
 | |
| 		return true;
 | |
| 	} else {
 | |
| 		return true;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::translationShowRequiresRequest(LanguageId to) {
 | |
| 	// When changing be sure to reflect in translationShowRequiresCheck(to).
 | |
| 	if (!to) {
 | |
| 		if (const auto translation = Get<HistoryMessageTranslation>()) {
 | |
| 			if (!translation->failed && translation->text.empty()) {
 | |
| 				Assert(!translation->used);
 | |
| 				RemoveComponents(HistoryMessageTranslation::Bit());
 | |
| 			} else {
 | |
| 				translationToggle(translation, false);
 | |
| 			}
 | |
| 		}
 | |
| 		return false;
 | |
| 	} else if (const auto translation = Get<HistoryMessageTranslation>()) {
 | |
| 		if (translation->to == to) {
 | |
| 			translationToggle(translation, true);
 | |
| 			return false;
 | |
| 		}
 | |
| 		translationToggle(translation, false);
 | |
| 		translation->to = to;
 | |
| 		translation->requested = true;
 | |
| 		translation->failed = false;
 | |
| 		translation->text = {};
 | |
| 		return true;
 | |
| 	} else {
 | |
| 		AddComponents(HistoryMessageTranslation::Bit());
 | |
| 		const auto added = Get<HistoryMessageTranslation>();
 | |
| 		added->to = to;
 | |
| 		added->requested = true;
 | |
| 		return true;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::translationToggle(
 | |
| 		not_null<HistoryMessageTranslation*> translation,
 | |
| 		bool used) {
 | |
| 	if (translation->used != used && !translation->text.empty()) {
 | |
| 		translation->used = used;
 | |
| 		_history->owner().requestItemTextRefresh(this);
 | |
| 		_history->owner().updateDependentMessages(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::translationDone(LanguageId to, TextWithEntities result) {
 | |
| 	const auto set = [&](not_null<HistoryMessageTranslation*> translation) {
 | |
| 		if (result.empty()) {
 | |
| 			translation->failed = true;
 | |
| 		} else {
 | |
| 			translation->text = std::move(result);
 | |
| 			if (_history->translatedTo() == to) {
 | |
| 				translationToggle(translation, true);
 | |
| 			}
 | |
| 		}
 | |
| 	};
 | |
| 	if (const auto translation = Get<HistoryMessageTranslation>()) {
 | |
| 		if (translation->to == to && translation->text.empty()) {
 | |
| 			translation->requested = false;
 | |
| 			set(translation);
 | |
| 		}
 | |
| 	} else {
 | |
| 		AddComponents(HistoryMessageTranslation::Bit());
 | |
| 		const auto added = Get<HistoryMessageTranslation>();
 | |
| 		added->to = to;
 | |
| 		set(added);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canReact() const {
 | |
| 	if (!isRegular() || isService()) {
 | |
| 		return false;
 | |
| 	} else if (const auto media = this->media()) {
 | |
| 		if (media->call()) {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void HistoryItem::toggleReaction(
 | |
| 		const Data::ReactionId &reaction,
 | |
| 		ReactionSource source) {
 | |
| 	if (!_reactions) {
 | |
| 		_reactions = std::make_unique<Data::MessageReactions>(this);
 | |
| 		const auto canViewReactions = !isDiscussionPost()
 | |
| 			&& (_history->peer->isChat() || _history->peer->isMegagroup());
 | |
| 		if (canViewReactions) {
 | |
| 			_flags |= MessageFlag::CanViewReactions;
 | |
| 		}
 | |
| 		_reactions->add(reaction, (source == ReactionSource::Selector));
 | |
| 	} else if (ranges::contains(_reactions->chosen(), reaction)) {
 | |
| 		_reactions->remove(reaction);
 | |
| 		if (_reactions->empty()) {
 | |
| 			_reactions = nullptr;
 | |
| 			_flags &= ~MessageFlag::CanViewReactions;
 | |
| 			_history->owner().notifyItemDataChange(this);
 | |
| 		}
 | |
| 	} else {
 | |
| 		_reactions->add(reaction, (source == ReactionSource::Selector));
 | |
| 	}
 | |
| 	_history->owner().notifyItemDataChange(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateReactionsUnknown() {
 | |
| 	_reactionsLastRefreshed = 1;
 | |
| }
 | |
| 
 | |
| const std::vector<Data::MessageReaction> &HistoryItem::reactions() const {
 | |
| 	static const auto kEmpty = std::vector<Data::MessageReaction>();
 | |
| 	return _reactions ? _reactions->list() : kEmpty;
 | |
| }
 | |
| 
 | |
| auto HistoryItem::recentReactions() const
 | |
| -> const base::flat_map<
 | |
| 		Data::ReactionId,
 | |
| 		std::vector<Data::RecentReaction>> & {
 | |
| 	static const auto kEmpty = base::flat_map<
 | |
| 		Data::ReactionId,
 | |
| 		std::vector<Data::RecentReaction>>();
 | |
| 	return _reactions ? _reactions->recent() : kEmpty;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canViewReactions() const {
 | |
| 	return (_flags & MessageFlag::CanViewReactions)
 | |
| 		&& _reactions
 | |
| 		&& !_reactions->list().empty();
 | |
| }
 | |
| 
 | |
| std::vector<Data::ReactionId> HistoryItem::chosenReactions() const {
 | |
| 	return _reactions
 | |
| 		? _reactions->chosen()
 | |
| 		: std::vector<Data::ReactionId>();
 | |
| }
 | |
| 
 | |
| Data::ReactionId HistoryItem::lookupUnreadReaction(
 | |
| 		not_null<UserData*> from) const {
 | |
| 	if (!_reactions) {
 | |
| 		return {};
 | |
| 	}
 | |
| 	const auto recent = _reactions->recent();
 | |
| 	for (const auto &[id, list] : _reactions->recent()) {
 | |
| 		const auto i = ranges::find(
 | |
| 			list,
 | |
| 			from,
 | |
| 			&Data::RecentReaction::peer);
 | |
| 		if (i != end(list) && i->unread) {
 | |
| 			return id;
 | |
| 		}
 | |
| 	}
 | |
| 	return {};
 | |
| }
 | |
| 
 | |
| crl::time HistoryItem::lastReactionsRefreshTime() const {
 | |
| 	return _reactionsLastRefreshed;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::hasDirectLink() const {
 | |
| 	return isRegular() && _history->peer->isChannel();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::changesWallPaper() const {
 | |
| 	if (const auto media = _media.get()) {
 | |
| 		return media->paper() != nullptr;
 | |
| 	}
 | |
| 	return Has<HistoryServiceSameBackground>();
 | |
| }
 | |
| 
 | |
| FullMsgId HistoryItem::fullId() const {
 | |
| 	return FullMsgId(_history->peer->id, id);
 | |
| }
 | |
| 
 | |
| GlobalMsgId HistoryItem::globalId() const {
 | |
| 	return { fullId(), _history->session().uniqueId() };
 | |
| }
 | |
| 
 | |
| Data::MessagePosition HistoryItem::position() const {
 | |
| 	return { .fullId = fullId(), .date = date() };
 | |
| }
 | |
| 
 | |
| bool HistoryItem::computeDropForwardedInfo() const {
 | |
| 	const auto media = this->media();
 | |
| 	return (media && media->dropForwardedInfo())
 | |
| 		|| (_history->peer->isSelf()
 | |
| 			&& !Has<HistoryMessageForwarded>()
 | |
| 			&& (!media || !media->forceForwardedInfo()));
 | |
| }
 | |
| 
 | |
| bool HistoryItem::inThread(MsgId rootId) const {
 | |
| 	return (replyToTop() == rootId)
 | |
| 		|| (topicRootId() == rootId);
 | |
| }
 | |
| 
 | |
| not_null<PeerData*> HistoryItem::author() const {
 | |
| 	return (isPost() && !isSponsored()) ? _history->peer : from();
 | |
| }
 | |
| 
 | |
| TimeId HistoryItem::originalDate() const {
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		return forwarded->originalDate;
 | |
| 	}
 | |
| 	return date();
 | |
| }
 | |
| 
 | |
| PeerData *HistoryItem::originalSender() const {
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		return forwarded->originalSender;
 | |
| 	}
 | |
| 	const auto peer = _history->peer;
 | |
| 	return (peer->isChannel() && !peer->isMegagroup()) ? peer : from();
 | |
| }
 | |
| 
 | |
| const HiddenSenderInfo *HistoryItem::hiddenSenderInfo() const {
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		return forwarded->hiddenSenderInfo.get();
 | |
| 	}
 | |
| 	return nullptr;
 | |
| }
 | |
| 
 | |
| not_null<PeerData*> HistoryItem::fromOriginal() const {
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		if (forwarded->originalSender) {
 | |
| 			if (const auto user = forwarded->originalSender->asUser()) {
 | |
| 				return user;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return from();
 | |
| }
 | |
| 
 | |
| QString HistoryItem::originalPostAuthor() const {
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		return forwarded->originalPostAuthor;
 | |
| 	} else if (const auto msgsigned = Get<HistoryMessageSigned>()) {
 | |
| 		if (!msgsigned->isAnonymousRank) {
 | |
| 			return msgsigned->postAuthor;
 | |
| 		}
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| MsgId HistoryItem::originalId() const {
 | |
| 	if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		return forwarded->originalId;
 | |
| 	}
 | |
| 	return id;
 | |
| }
 | |
| 
 | |
| const TextWithEntities &HistoryItem::originalText() const {
 | |
| 	static const auto kEmpty = TextWithEntities();
 | |
| 	return isService() ? kEmpty : _text;
 | |
| }
 | |
| 
 | |
| const TextWithEntities &HistoryItem::translatedText() const {
 | |
| 	if (isService()) {
 | |
| 		static const auto kEmpty = TextWithEntities();
 | |
| 		return kEmpty;
 | |
| 	} else if (const auto translation = this->translation()
 | |
| 		; translation
 | |
| 		&& translation->used
 | |
| 		&& (translation->to == history()->translatedTo())) {
 | |
| 		return translation->text;
 | |
| 	} else {
 | |
| 		return originalText();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| TextWithEntities HistoryItem::translatedTextWithLocalEntities() const {
 | |
| 	if (isService()) {
 | |
| 		return {};
 | |
| 	} else {
 | |
| 		return withLocalEntities(translatedText());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| TextForMimeData HistoryItem::clipboardText() const {
 | |
| 	return isService()
 | |
| 		? TextForMimeData()
 | |
| 		: TextForMimeData::WithExpandedLinks(_text);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::changeViewsCount(int count) {
 | |
| 	const auto views = Get<HistoryMessageViews>();
 | |
| 	if (!views
 | |
| 		|| views->views.count == count
 | |
| 		|| (count >= 0 && views->views.count > count)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	views->views.count = count;
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void HistoryItem::setForwardsCount(int count) {
 | |
| 	const auto views = Get<HistoryMessageViews>();
 | |
| 	if (!views
 | |
| 		|| views->forwardsCount == count
 | |
| 		|| (count >= 0 && views->forwardsCount > count)) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	views->forwardsCount = count;
 | |
| 	history()->owner().notifyItemDataChange(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setPostAuthor(const QString &postAuthor) {
 | |
| 	auto msgsigned = Get<HistoryMessageSigned>();
 | |
| 	if (postAuthor.isEmpty()) {
 | |
| 		if (!msgsigned) {
 | |
| 			return;
 | |
| 		}
 | |
| 		RemoveComponents(HistoryMessageSigned::Bit());
 | |
| 		history()->owner().requestItemResize(this);
 | |
| 		return;
 | |
| 	}
 | |
| 	if (!msgsigned) {
 | |
| 		AddComponents(HistoryMessageSigned::Bit());
 | |
| 		msgsigned = Get<HistoryMessageSigned>();
 | |
| 	} else if (msgsigned->postAuthor == postAuthor) {
 | |
| 		return;
 | |
| 	}
 | |
| 	msgsigned->postAuthor = postAuthor;
 | |
| 	msgsigned->isAnonymousRank = !isDiscussionPost()
 | |
| 		&& this->author()->isMegagroup();
 | |
| 	history()->owner().requestItemResize(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setReplies(HistoryMessageRepliesData &&data) {
 | |
| 	if (data.isNull) {
 | |
| 		return;
 | |
| 	}
 | |
| 	auto views = Get<HistoryMessageViews>();
 | |
| 	if (!views) {
 | |
| 		AddComponents(HistoryMessageViews::Bit());
 | |
| 		views = Get<HistoryMessageViews>();
 | |
| 	}
 | |
| 	const auto &repliers = data.recentRepliers;
 | |
| 	const auto count = data.repliesCount;
 | |
| 	const auto channelId = data.channelId;
 | |
| 	const auto readTillId = data.readMaxId
 | |
| 		? std::max({
 | |
| 			views->commentsInboxReadTillId.bare,
 | |
| 			data.readMaxId.bare,
 | |
| 			int64(1),
 | |
| 		})
 | |
| 		: views->commentsInboxReadTillId;
 | |
| 	const auto maxId = data.maxId ? data.maxId : views->commentsMaxId;
 | |
| 	const auto countsChanged = (views->replies.count != count)
 | |
| 		|| (views->commentsInboxReadTillId != readTillId)
 | |
| 		|| (views->commentsMaxId != maxId);
 | |
| 	const auto megagroupChanged = (views->commentsMegagroupId != channelId);
 | |
| 	const auto recentChanged = (views->recentRepliers != repliers);
 | |
| 	if (!countsChanged && !megagroupChanged && !recentChanged) {
 | |
| 		return;
 | |
| 	}
 | |
| 	views->replies.count = count;
 | |
| 	if (recentChanged) {
 | |
| 		views->recentRepliers = repliers;
 | |
| 	}
 | |
| 	const auto wasUnread = areCommentsUnread();
 | |
| 	views->commentsMegagroupId = channelId;
 | |
| 	views->commentsInboxReadTillId = readTillId;
 | |
| 	views->commentsMaxId = maxId;
 | |
| 	if (wasUnread != areCommentsUnread()) {
 | |
| 		history()->owner().requestItemRepaint(this);
 | |
| 	}
 | |
| 	refreshRepliesText(views, megagroupChanged);
 | |
| }
 | |
| 
 | |
| void HistoryItem::clearReplies() {
 | |
| 	auto views = Get<HistoryMessageViews>();
 | |
| 	if (!views) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto viewsPart = views->views;
 | |
| 	if (viewsPart.count < 0) {
 | |
| 		RemoveComponents(HistoryMessageViews::Bit());
 | |
| 	} else {
 | |
| 		*views = HistoryMessageViews();
 | |
| 		views->views = viewsPart;
 | |
| 	}
 | |
| 	history()->owner().requestItemResize(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::refreshRepliesText(
 | |
| 		not_null<HistoryMessageViews*> views,
 | |
| 		bool forceResize) {
 | |
| 	if (views->commentsMegagroupId) {
 | |
| 		views->replies.text = (views->replies.count > 0)
 | |
| 			? tr::lng_comments_open_count(
 | |
| 				tr::now,
 | |
| 				lt_count_short,
 | |
| 				views->replies.count)
 | |
| 			: tr::lng_comments_open_none(tr::now);
 | |
| 		views->replies.textWidth = st::semiboldFont->width(
 | |
| 			views->replies.text);
 | |
| 		views->repliesSmall.text = (views->replies.count > 0)
 | |
| 			? Lang::FormatCountToShort(views->replies.count).string
 | |
| 			: QString();
 | |
| 		const auto hadText = (views->repliesSmall.textWidth > 0);
 | |
| 		views->repliesSmall.textWidth = (views->replies.count > 0)
 | |
| 			? st::semiboldFont->width(views->repliesSmall.text)
 | |
| 			: 0;
 | |
| 		const auto hasText = (views->repliesSmall.textWidth > 0);
 | |
| 		if (hasText != hadText) {
 | |
| 			forceResize = true;
 | |
| 		}
 | |
| 	}
 | |
| 	if (forceResize) {
 | |
| 		history()->owner().requestItemResize(this);
 | |
| 	} else {
 | |
| 		history()->owner().requestItemRepaint(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::changeRepliesCount(int delta, PeerId replier) {
 | |
| 	const auto views = Get<HistoryMessageViews>();
 | |
| 	const auto limit = HistoryMessageViews::kMaxRecentRepliers;
 | |
| 	if (!views) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	// Update full count.
 | |
| 	if (views->replies.count < 0) {
 | |
| 		return;
 | |
| 	}
 | |
| 	views->replies.count = std::max(views->replies.count + delta, 0);
 | |
| 	if (replier && views->commentsMegagroupId) {
 | |
| 		if (delta < 0) {
 | |
| 			views->recentRepliers.erase(
 | |
| 				ranges::remove(views->recentRepliers, replier),
 | |
| 				end(views->recentRepliers));
 | |
| 		} else if (!ranges::contains(views->recentRepliers, replier)) {
 | |
| 			views->recentRepliers.insert(views->recentRepliers.begin(), replier);
 | |
| 			while (views->recentRepliers.size() > limit) {
 | |
| 				views->recentRepliers.pop_back();
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	refreshRepliesText(views);
 | |
| 	history()->owner().notifyItemDataChange(this);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setReplyFields(
 | |
| 		MsgId replyTo,
 | |
| 		MsgId replyToTop,
 | |
| 		bool isForumPost) {
 | |
| 	if (isScheduled()) {
 | |
| 		return;
 | |
| 	} else if (const auto data = GetServiceDependentData()) {
 | |
| 		if ((data->topId != replyToTop) && !IsServerMsgId(data->topId)) {
 | |
| 			data->topId = replyToTop;
 | |
| 			if (isForumPost) {
 | |
| 				data->topicPost = true;
 | |
| 			}
 | |
| 		}
 | |
| 	} else if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto increment = (reply->topMessageId() != replyToTop)
 | |
| 			&& !IsServerMsgId(reply->topMessageId());
 | |
| 		reply->updateFields(this, replyTo, replyToTop, isForumPost);
 | |
| 		if (increment) {
 | |
| 			incrementReplyToTopCounter();
 | |
| 		}
 | |
| 	}
 | |
| 	if (const auto topic = this->topic()) {
 | |
| 		topic->maybeSetLastMessage(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateDate(TimeId newDate) {
 | |
| 	if (canUpdateDate() && _date != newDate) {
 | |
| 		_date = newDate;
 | |
| 		_history->owner().requestItemViewRefresh(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canUpdateDate() const {
 | |
| 	return isScheduled();
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyTTL(TimeId destroyAt) {
 | |
| 	const auto previousDestroyAt = std::exchange(_ttlDestroyAt, destroyAt);
 | |
| 	if (previousDestroyAt) {
 | |
| 		_history->owner().unregisterMessageTTL(previousDestroyAt, this);
 | |
| 	}
 | |
| 	if (!_ttlDestroyAt) {
 | |
| 		return;
 | |
| 	} else if (base::unixtime::now() >= _ttlDestroyAt) {
 | |
| 		const auto session = &_history->session();
 | |
| 		crl::on_main(session, [session, id = fullId()]{
 | |
| 			if (const auto item = session->data().message(id)) {
 | |
| 				item->destroy();
 | |
| 			}
 | |
| 		});
 | |
| 	} else {
 | |
| 		_history->owner().registerMessageTTL(_ttlDestroyAt, this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::replaceBuyWithReceiptInMarkup() {
 | |
| 	if (const auto markup = inlineReplyMarkup()) {
 | |
| 		for (auto &row : markup->data.rows) {
 | |
| 			for (auto &button : row) {
 | |
| 				if (button.type == HistoryMessageMarkupButton::Type::Buy) {
 | |
| 					const auto receipt = tr::lng_payments_receipt_button(tr::now);
 | |
| 					if (button.text != receipt) {
 | |
| 						button.text = receipt;
 | |
| 						if (markup->inlineKeyboard) {
 | |
| 							markup->inlineKeyboard = nullptr;
 | |
| 							_history->owner().requestItemResize(this);
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isUploading() const {
 | |
| 	return _media && _media->uploading();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isRegular() const {
 | |
| 	return isHistoryEntry() && !isLocal();
 | |
| }
 | |
| 
 | |
| int HistoryItem::viewsCount() const {
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		return std::max(views->views.count, 0);
 | |
| 	}
 | |
| 	return hasViews() ? 1 : -1;
 | |
| }
 | |
| 
 | |
| int HistoryItem::repliesCount() const {
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		if (!checkCommentsLinkedChat(views->commentsMegagroupId)) {
 | |
| 			return 0;
 | |
| 		}
 | |
| 		return std::max(views->replies.count, 0);
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::repliesAreComments() const {
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		return (views->commentsMegagroupId != 0)
 | |
| 			&& checkCommentsLinkedChat(views->commentsMegagroupId);
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::externalReply() const {
 | |
| 	if (!_history->peer->isRepliesChat()) {
 | |
| 		return false;
 | |
| 	} else if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		return forwarded->savedFromPeer && forwarded->savedFromMsgId;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::hasExtendedMediaPreview() const {
 | |
| 	if (const auto media = _media.get()) {
 | |
| 		if (const auto invoice = media->invoice()) {
 | |
| 			return (invoice->extendedPreview && !invoice->extendedMedia);
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void HistoryItem::sendFailed() {
 | |
| 	Expects(_flags & MessageFlag::BeingSent);
 | |
| 	Expects(!(_flags & MessageFlag::SendingFailed));
 | |
| 
 | |
| 	_flags = (_flags | MessageFlag::SendingFailed) & ~MessageFlag::BeingSent;
 | |
| 	_history->owner().notifyItemDataChange(this);
 | |
| 	_history->session().changes().historyUpdated(
 | |
| 		_history,
 | |
| 		Data::HistoryUpdate::Flag::ClientSideMessages);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::needCheck() const {
 | |
| 	return (out() && !isEmpty())
 | |
| 		|| (!isRegular() && _history->peer->isSelf());
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isService() const {
 | |
| 	return Has<HistoryServiceData>();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::unread(not_null<Data::Thread*> thread) const {
 | |
| 	// Messages from myself are always read, unless scheduled.
 | |
| 	if (_history->peer->isSelf() && !isFromScheduled()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	// All messages in converted chats are always read.
 | |
| 	if (_history->peer->migrateTo()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (isRegular()) {
 | |
| 		if (!thread->isServerSideUnread(this)) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		if (out()) {
 | |
| 			if (const auto user = _history->peer->asUser()) {
 | |
| 				if (user->isBot() && !user->isSupport()) {
 | |
| 					return false;
 | |
| 				}
 | |
| 			} else if (const auto channel = _history->peer->asChannel()) {
 | |
| 				if (!channel->isMegagroup()) {
 | |
| 					return false;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	return out() || (_flags & MessageFlag::ClientSideUnread);
 | |
| }
 | |
| 
 | |
| MsgId HistoryItem::replyToId() const {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		return reply->messageId();
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| FullMsgId HistoryItem::replyToFullId() const {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto peer = reply->externalPeerId();
 | |
| 		return { peer ? peer : history()->peer->id, reply->messageId() };
 | |
| 	}
 | |
| 	return {};
 | |
| }
 | |
| 
 | |
| MsgId HistoryItem::replyToTop() const {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		return reply->topMessageId();
 | |
| 	} else if (const auto data = GetServiceDependentData()) {
 | |
| 		return data->topId;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| MsgId HistoryItem::topicRootId() const {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()
 | |
| 		; reply && reply->topicPost()) {
 | |
| 		return reply->topMessageId();
 | |
| 	} else if (const auto data = GetServiceDependentData()
 | |
| 		; data && data->topicPost && data->topId) {
 | |
| 		return data->topId;
 | |
| 	} else if (const auto info = Get<HistoryServiceTopicInfo>()) {
 | |
| 		if (info->created()) {
 | |
| 			return id;
 | |
| 		}
 | |
| 	}
 | |
| 	return Data::ForumTopic::kGeneralId;
 | |
| }
 | |
| 
 | |
| FullStoryId HistoryItem::replyToStory() const {
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		if (reply->storyId()) {
 | |
| 			const auto peerId = reply->externalPeerId()
 | |
| 				? reply->externalPeerId()
 | |
| 				: _history->peer->id;
 | |
| 			return { .peer = peerId, .story = reply->storyId() };
 | |
| 		}
 | |
| 	}
 | |
| 	return {};
 | |
| }
 | |
| 
 | |
| FullReplyTo HistoryItem::replyTo() const {
 | |
| 	auto result = FullReplyTo{
 | |
| 		.topicRootId = topicRootId(),
 | |
| 	};
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto &fields = reply->fields();
 | |
| 		const auto peer = fields.externalPeerId;
 | |
| 		const auto replyToPeer = peer ? peer : _history->peer->id;
 | |
| 		if (const auto id = fields.messageId) {
 | |
| 			result.messageId = { replyToPeer, id };
 | |
| 			result.quote = fields.quote;
 | |
| 			result.quoteOffset = fields.quoteOffset;
 | |
| 		}
 | |
| 		if (const auto id = fields.storyId) {
 | |
| 			result.storyId = { replyToPeer, id };
 | |
| 		}
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void HistoryItem::setText(const TextWithEntities &textWithEntities) {
 | |
| 	for (const auto &entity : textWithEntities.entities) {
 | |
| 		auto type = entity.type();
 | |
| 		if (type == EntityType::Url
 | |
| 			|| type == EntityType::CustomUrl
 | |
| 			|| type == EntityType::Email) {
 | |
| 			_flags |= MessageFlag::HasTextLinks;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	setTextValue((_media && _media->consumeMessageText(textWithEntities))
 | |
| 		? TextWithEntities()
 | |
| 		: std::move(textWithEntities));
 | |
| }
 | |
| 
 | |
| void HistoryItem::setTextValue(TextWithEntities text, bool force) {
 | |
| 	if (const auto processId = Spellchecker::TryHighlightSyntax(text)) {
 | |
| 		_flags |= MessageFlag::InHighlightProcess;
 | |
| 		history()->owner().registerHighlightProcess(processId, this);
 | |
| 	}
 | |
| 	const auto had = !_text.empty();
 | |
| 	_text = std::move(text);
 | |
| 	RemoveComponents(HistoryMessageTranslation::Bit());
 | |
| 	if (had || force) {
 | |
| 		history()->owner().requestItemTextRefresh(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::inHighlightProcess() const {
 | |
| 	return _flags & MessageFlag::InHighlightProcess;
 | |
| }
 | |
| 
 | |
| void HistoryItem::highlightProcessDone() {
 | |
| 	Expects(inHighlightProcess());
 | |
| 
 | |
| 	_flags &= ~MessageFlag::InHighlightProcess;
 | |
| 	if (!_text.empty()) {
 | |
| 		setTextValue(base::take(_text), true);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::showNotification() const {
 | |
| 	const auto channel = _history->peer->asChannel();
 | |
| 	if (channel && !channel->amIn()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return (out() || _history->peer->isSelf())
 | |
| 		? isFromScheduled()
 | |
| 		: unread(notificationThread());
 | |
| }
 | |
| 
 | |
| void HistoryItem::markClientSideAsRead() {
 | |
| 	_flags &= ~MessageFlag::ClientSideUnread;
 | |
| }
 | |
| 
 | |
| MessageGroupId HistoryItem::groupId() const {
 | |
| 	return _groupId;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isEmpty() const {
 | |
| 	return _text.empty()
 | |
| 		&& !_media
 | |
| 		&& !Has<HistoryMessageLogEntryOriginal>();
 | |
| }
 | |
| 
 | |
| TextWithEntities HistoryItem::notificationText(
 | |
| 		NotificationTextOptions options) const {
 | |
| 	auto result = [&] {
 | |
| 		if (_media && !isService()) {
 | |
| 			return _media->notificationText();
 | |
| 		} else if (!emptyText()) {
 | |
| 			return _text;
 | |
| 		}
 | |
| 		return TextWithEntities();
 | |
| 	}();
 | |
| 	if (options.spoilerLoginCode
 | |
| 		&& !out()
 | |
| 		&& history()->peer->isNotificationsUser()) {
 | |
| 		result = SpoilerLoginCode(std::move(result));
 | |
| 	}
 | |
| 	if (result.text.size() <= kNotificationTextLimit) {
 | |
| 		return result;
 | |
| 	}
 | |
| 	return Ui::Text::Mid(result, 0, kNotificationTextLimit).append(
 | |
| 		Ui::kQEllipsis);
 | |
| }
 | |
| 
 | |
| ItemPreview HistoryItem::toPreview(ToPreviewOptions options) const {
 | |
| 	if (isService()) {
 | |
| 		// Don't show small media for service messages (chat photo changed).
 | |
| 		// Because larger version is shown exactly to the left of the small.
 | |
| 		//auto media = _media ? _media->toPreview(options) : ItemPreview();
 | |
| 		return {
 | |
| 			.text = Ui::Text::Colorized(notificationText()),
 | |
| 			//.images = std::move(media.images),
 | |
| 			//.loadingContext = std::move(media.loadingContext),
 | |
| 		};
 | |
| 	}
 | |
| 
 | |
| 	auto result = [&]() -> ItemPreview {
 | |
| 		if (_media) {
 | |
| 			return _media->toPreview(options);
 | |
| 		} else if (!emptyText()) {
 | |
| 			return { .text = options.translated ? translatedText() : _text };
 | |
| 		}
 | |
| 		return {};
 | |
| 	}();
 | |
| 	if (options.spoilerLoginCode
 | |
| 		&& !out()
 | |
| 		&& history()->peer->isNotificationsUser()) {
 | |
| 		result.text = SpoilerLoginCode(std::move(result.text));
 | |
| 	}
 | |
| 	const auto fromSender = [](not_null<PeerData*> sender) {
 | |
| 		return sender->isSelf()
 | |
| 			? tr::lng_from_you(tr::now)
 | |
| 			: sender->shortName();
 | |
| 	};
 | |
| 	result.icon = (Get<HistoryMessageForwarded>() != nullptr)
 | |
| 		? ItemPreview::Icon::ForwardedMessage
 | |
| 		: replyToStory().valid()
 | |
| 		? ItemPreview::Icon::ReplyToStory
 | |
| 		: ItemPreview::Icon::None;
 | |
| 	const auto fromForwarded = [&]() -> std::optional<QString> {
 | |
| 		if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 			return forwarded->originalSender
 | |
| 				? fromSender(forwarded->originalSender)
 | |
| 				: forwarded->hiddenSenderInfo->name;
 | |
| 		}
 | |
| 		return {};
 | |
| 	};
 | |
| 	const auto sender = [&]() -> std::optional<QString> {
 | |
| 		if (options.hideSender || isPost() || isEmpty()) {
 | |
| 			return {};
 | |
| 		} else if (!_history->peer->isUser()) {
 | |
| 			if (const auto from = displayFrom()) {
 | |
| 				return fromSender(from);
 | |
| 			}
 | |
| 			return fromForwarded();
 | |
| 		} else if (_history->peer->isSelf()) {
 | |
| 			return fromForwarded();
 | |
| 		}
 | |
| 		return {};
 | |
| 	}();
 | |
| 	if (!sender) {
 | |
| 		return result;
 | |
| 	}
 | |
| 	const auto topic = options.ignoreTopic ? nullptr : this->topic();
 | |
| 	return Dialogs::Ui::PreviewWithSender(
 | |
| 		std::move(result),
 | |
| 		*sender,
 | |
| 		topic ? topic->titleWithIcon() : TextWithEntities());
 | |
| }
 | |
| 
 | |
| TextWithEntities HistoryItem::inReplyText() const {
 | |
| 	if (!isService()) {
 | |
| 		return toPreview({
 | |
| 			.hideSender = true,
 | |
| 			.generateImages = false,
 | |
| 			.translated = true,
 | |
| 		}).text;
 | |
| 	}
 | |
| 	auto result = notificationText();
 | |
| 	const auto &name = author()->name();
 | |
| 	TextUtilities::Trim(result);
 | |
| 	if (result.text.startsWith(name)) {
 | |
| 		result = Ui::Text::Mid(result, name.size());
 | |
| 		TextUtilities::Trim(result);
 | |
| 	}
 | |
| 	return Ui::Text::Colorized(result);
 | |
| }
 | |
| 
 | |
| const std::vector<ClickHandlerPtr> &HistoryItem::customTextLinks() const {
 | |
| 	static const auto kEmpty = std::vector<ClickHandlerPtr>();
 | |
| 	const auto service = Get<HistoryServiceData>();
 | |
| 	return service ? service->textLinks : kEmpty;
 | |
| }
 | |
| 
 | |
| void HistoryItem::createComponents(CreateConfig &&config) {
 | |
| 	uint64 mask = 0;
 | |
| 	if (config.reply.messageId
 | |
| 		|| config.reply.externalSenderId
 | |
| 		|| !config.reply.externalSenderName.isEmpty()
 | |
| 		|| config.reply.storyId) {
 | |
| 		mask |= HistoryMessageReply::Bit();
 | |
| 	}
 | |
| 	if (config.viaBotId) {
 | |
| 		mask |= HistoryMessageVia::Bit();
 | |
| 	}
 | |
| 	if (config.viewsCount >= 0 || !config.replies.isNull) {
 | |
| 		mask |= HistoryMessageViews::Bit();
 | |
| 	}
 | |
| 	if (!config.postAuthor.isEmpty()) {
 | |
| 		mask |= HistoryMessageSigned::Bit();
 | |
| 	} else if (_history->peer->isMegagroup() // Discussion posts signatures.
 | |
| 		&& config.savedFromPeer
 | |
| 		&& !config.originalPostAuthor.isEmpty()) {
 | |
| 		const auto savedFrom = _history->owner().peerLoaded(
 | |
| 			config.savedFromPeer);
 | |
| 		if (savedFrom && savedFrom->isChannel()) {
 | |
| 			mask |= HistoryMessageSigned::Bit();
 | |
| 		}
 | |
| 	} else if ((_history->peer->isSelf() || _history->peer->isRepliesChat())
 | |
| 		&& !config.originalPostAuthor.isEmpty()) {
 | |
| 		mask |= HistoryMessageSigned::Bit();
 | |
| 	}
 | |
| 	if (config.editDate != TimeId(0)) {
 | |
| 		mask |= HistoryMessageEdited::Bit();
 | |
| 	}
 | |
| 	if (config.originalDate != 0) {
 | |
| 		mask |= HistoryMessageForwarded::Bit();
 | |
| 	}
 | |
| 	if (!config.markup.isTrivial()) {
 | |
| 		mask |= HistoryMessageReplyMarkup::Bit();
 | |
| 	} else if (config.inlineMarkup) {
 | |
| 		mask |= HistoryMessageReplyMarkup::Bit();
 | |
| 	}
 | |
| 
 | |
| 	UpdateComponents(mask);
 | |
| 
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		reply->set(std::move(config.reply));
 | |
| 		if (!reply->updateData(this)) {
 | |
| 			if (const auto messageId = reply->messageId()) {
 | |
| 				RequestDependentMessageItem(
 | |
| 					this,
 | |
| 					reply->externalPeerId(),
 | |
| 					reply->messageId());
 | |
| 			} else if (reply->storyId()) {
 | |
| 				RequestDependentMessageStory(
 | |
| 					this,
 | |
| 					reply->externalPeerId(),
 | |
| 					reply->storyId());
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	if (const auto via = Get<HistoryMessageVia>()) {
 | |
| 		via->create(&_history->owner(), config.viaBotId);
 | |
| 	}
 | |
| 	if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 		changeViewsCount(config.viewsCount);
 | |
| 		if (config.replies.isNull
 | |
| 			&& isSending()
 | |
| 			&& config.markup.isNull()) {
 | |
| 			if (const auto broadcast = _history->peer->asBroadcast()) {
 | |
| 				if (const auto linked = broadcast->linkedChat()) {
 | |
| 					config.replies.isNull = false;
 | |
| 					config.replies.channelId = peerToChannel(linked->id);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		setForwardsCount(config.forwardsCount);
 | |
| 		setReplies(std::move(config.replies));
 | |
| 	}
 | |
| 	if (const auto edited = Get<HistoryMessageEdited>()) {
 | |
| 		edited->date = config.editDate;
 | |
| 	}
 | |
| 	if (const auto msgsigned = Get<HistoryMessageSigned>()) {
 | |
| 		msgsigned->postAuthor = config.postAuthor.isEmpty()
 | |
| 			? config.originalPostAuthor
 | |
| 			: config.postAuthor;
 | |
| 		msgsigned->isAnonymousRank = !isDiscussionPost()
 | |
| 			&& author()->isMegagroup();
 | |
| 	}
 | |
| 	setupForwardedComponent(config);
 | |
| 	if (const auto markup = Get<HistoryMessageReplyMarkup>()) {
 | |
| 		if (!config.markup.isTrivial()) {
 | |
| 			markup->updateData(std::move(config.markup));
 | |
| 		} else if (config.inlineMarkup) {
 | |
| 			markup->createForwarded(*config.inlineMarkup);
 | |
| 		}
 | |
| 		if (markup->data.flags & ReplyMarkupFlag::HasSwitchInlineButton) {
 | |
| 			_flags |= MessageFlag::HasSwitchInlineButton;
 | |
| 		}
 | |
| 	} else if (!config.markup.isNull()) {
 | |
| 		_flags |= MessageFlag::HasReplyMarkup;
 | |
| 	} else {
 | |
| 		_flags &= ~MessageFlag::HasReplyMarkup;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::checkRepliesPts(
 | |
| 		const HistoryMessageRepliesData &data) const {
 | |
| 	const auto channel = _history->peer->asChannel();
 | |
| 	const auto pts = channel
 | |
| 		? channel->pts()
 | |
| 		: _history->session().updates().pts();
 | |
| 	return (data.pts >= pts);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setupForwardedComponent(const CreateConfig &config) {
 | |
| 	const auto forwarded = Get<HistoryMessageForwarded>();
 | |
| 	if (!forwarded) {
 | |
| 		return;
 | |
| 	}
 | |
| 	forwarded->originalDate = config.originalDate;
 | |
| 	const auto originalSender = config.originalSenderId
 | |
| 		? config.originalSenderId
 | |
| 		: !config.originalSenderName.isEmpty()
 | |
| 		? PeerId()
 | |
| 		: from()->id;
 | |
| 	forwarded->originalSender = originalSender
 | |
| 		? _history->owner().peer(originalSender).get()
 | |
| 		: nullptr;
 | |
| 	if (!forwarded->originalSender) {
 | |
| 		forwarded->hiddenSenderInfo = std::make_unique<HiddenSenderInfo>(
 | |
| 			config.originalSenderName,
 | |
| 			config.imported);
 | |
| 	}
 | |
| 	forwarded->originalId = config.originalId;
 | |
| 	forwarded->originalPostAuthor = config.originalPostAuthor;
 | |
| 	forwarded->psaType = config.forwardPsaType;
 | |
| 	forwarded->savedFromPeer = _history->owner().peerLoaded(
 | |
| 		config.savedFromPeer);
 | |
| 	forwarded->savedFromMsgId = config.savedFromMsgId;
 | |
| 	forwarded->imported = config.imported;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::generateLocalEntitiesByReply() const {
 | |
| 	using namespace HistoryView;
 | |
| 	if (!_media) {
 | |
| 		return true;
 | |
| 	} else if (const auto document = _media->document()) {
 | |
| 		return !DurationForTimestampLinks(document);
 | |
| 	} else if (const auto webpage = _media->webpage()) {
 | |
| 		return (webpage->type != WebPageType::Video)
 | |
| 			&& !DurationForTimestampLinks(webpage);
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| TextWithEntities HistoryItem::withLocalEntities(
 | |
| 		const TextWithEntities &textWithEntities) const {
 | |
| 	using namespace HistoryView;
 | |
| 	if (!generateLocalEntitiesByReply()) {
 | |
| 		if (!_media) {
 | |
| 		} else if (const auto document = _media->document()) {
 | |
| 			if (const auto duration = DurationForTimestampLinks(document)) {
 | |
| 				return AddTimestampLinks(
 | |
| 					textWithEntities,
 | |
| 					duration,
 | |
| 					TimestampLinkBase(document, fullId()));
 | |
| 			}
 | |
| 		} else if (const auto webpage = _media->webpage()) {
 | |
| 			if (const auto duration = DurationForTimestampLinks(webpage)) {
 | |
| 				return AddTimestampLinks(
 | |
| 					textWithEntities,
 | |
| 					duration,
 | |
| 					TimestampLinkBase(webpage, fullId()));
 | |
| 			}
 | |
| 		}
 | |
| 		return textWithEntities;
 | |
| 	}
 | |
| 	if (const auto reply = Get<HistoryMessageReply>()) {
 | |
| 		const auto document = reply->replyToDocumentId
 | |
| 			? _history->owner().document(reply->replyToDocumentId).get()
 | |
| 			: nullptr;
 | |
| 		const auto webpage = reply->replyToWebPageId
 | |
| 			? _history->owner().webpage(reply->replyToWebPageId).get()
 | |
| 			: nullptr;
 | |
| 		if (document) {
 | |
| 			if (const auto duration = DurationForTimestampLinks(document)) {
 | |
| 				const auto context = reply->resolvedMessage->fullId();
 | |
| 				return AddTimestampLinks(
 | |
| 					textWithEntities,
 | |
| 					duration,
 | |
| 					TimestampLinkBase(document, context));
 | |
| 			}
 | |
| 		} else if (webpage) {
 | |
| 			if (const auto duration = DurationForTimestampLinks(webpage)) {
 | |
| 				const auto context = reply->resolvedMessage->fullId();
 | |
| 				return AddTimestampLinks(
 | |
| 					textWithEntities,
 | |
| 					duration,
 | |
| 					TimestampLinkBase(webpage, context));
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return textWithEntities;
 | |
| }
 | |
| 
 | |
| void HistoryItem::createComponentsHelper(
 | |
| 		MessageFlags flags,
 | |
| 		FullReplyTo replyTo,
 | |
| 		UserId viaBotId,
 | |
| 		const QString &postAuthor,
 | |
| 		HistoryMessageMarkupData &&markup) {
 | |
| 	auto config = CreateConfig();
 | |
| 	config.viaBotId = viaBotId;
 | |
| 	if (flags & MessageFlag::HasReplyInfo) {
 | |
| 		config.reply.messageId = replyTo.messageId.msg;
 | |
| 		config.reply.storyId = replyTo.storyId.story;
 | |
| 		config.reply.externalPeerId = replyTo.storyId
 | |
| 			? replyTo.storyId.peer
 | |
| 			: (replyTo.messageId && replyTo.messageId.peer
 | |
| 				!= history()->peer->id)
 | |
| 			? replyTo.messageId.peer
 | |
| 			: PeerId();
 | |
| 		const auto to = LookupReplyTo(_history, replyTo.messageId);
 | |
| 		const auto replyToTop = replyTo.topicRootId
 | |
| 			? replyTo.topicRootId
 | |
| 			: LookupReplyToTop(_history, to);
 | |
| 		config.reply.topMessageId = replyToTop
 | |
| 			? replyToTop
 | |
| 			: (replyTo.messageId.peer == history()->peer->id)
 | |
| 			? replyTo.messageId.msg
 | |
| 			: MsgId();
 | |
| 		const auto forum = _history->asForum();
 | |
| 		const auto topic = forum
 | |
| 			? forum->topicFor(replyTo.topicRootId)
 | |
| 			: nullptr;
 | |
| 		if (!config.reply.externalPeerId
 | |
| 			&& topic
 | |
| 			&& to
 | |
| 			&& topic->rootId() != to->topicRootId()) {
 | |
| 			config.reply.externalPeerId = replyTo.messageId.peer;
 | |
| 		}
 | |
| 		const auto topicPost = config.reply.externalPeerId
 | |
| 			? (replyTo.topicRootId
 | |
| 				&& (replyTo.topicRootId != Data::ForumTopic::kGeneralId))
 | |
| 			: (LookupReplyIsTopicPost(to)
 | |
| 				|| (to && to->Has<HistoryServiceTopicInfo>())
 | |
| 				|| (forum && forum->creating(config.reply.topMessageId)));
 | |
| 		config.reply.topicPost = topicPost ? 1 : 0;
 | |
| 		config.reply.manualQuote = replyTo.quote.empty() ? 0 : 1;
 | |
| 		config.reply.quoteOffset = replyTo.quoteOffset;
 | |
| 		config.reply.quote = std::move(replyTo.quote);
 | |
| 	}
 | |
| 	config.markup = std::move(markup);
 | |
| 	if (flags & MessageFlag::HasPostAuthor) config.postAuthor = postAuthor;
 | |
| 	if (flags & MessageFlag::HasViews) config.viewsCount = 1;
 | |
| 
 | |
| 	createComponents(std::move(config));
 | |
| }
 | |
| 
 | |
| void HistoryItem::setReactions(const MTPMessageReactions *reactions) {
 | |
| 	Expects(!_reactions);
 | |
| 
 | |
| 	if (changeReactions(reactions) && _reactions->hasUnread()) {
 | |
| 		_flags |= MessageFlag::HasUnreadReaction;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::updateReactions(const MTPMessageReactions *reactions) {
 | |
| 	const auto wasRecentUsers = LookupRecentUnreadReactedUsers(this);
 | |
| 	const auto hadUnread = hasUnreadReaction();
 | |
| 	const auto changed = changeReactions(reactions);
 | |
| 	if (!changed) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto hasUnread = _reactions && _reactions->hasUnread();
 | |
| 	if (hasUnread && !hadUnread) {
 | |
| 		_flags |= MessageFlag::HasUnreadReaction;
 | |
| 
 | |
| 		addToUnreadThings(HistoryUnreadThings::AddType::New);
 | |
| 	} else if (!hasUnread && hadUnread) {
 | |
| 		markReactionsRead();
 | |
| 	}
 | |
| 	CheckReactionNotificationSchedule(this, wasRecentUsers);
 | |
| 	_history->owner().notifyItemDataChange(this);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::changeReactions(const MTPMessageReactions *reactions) {
 | |
| 	if (reactions || _reactionsLastRefreshed) {
 | |
| 		_reactionsLastRefreshed = crl::now();
 | |
| 	}
 | |
| 	if (!reactions) {
 | |
| 		_flags &= ~MessageFlag::CanViewReactions;
 | |
| 		return (base::take(_reactions) != nullptr);
 | |
| 	}
 | |
| 	return reactions->match([&](const MTPDmessageReactions &data) {
 | |
| 		if (data.is_can_see_list()) {
 | |
| 			_flags |= MessageFlag::CanViewReactions;
 | |
| 		} else {
 | |
| 			_flags &= ~MessageFlag::CanViewReactions;
 | |
| 		}
 | |
| 		if (data.vresults().v.isEmpty()) {
 | |
| 			return (base::take(_reactions) != nullptr);
 | |
| 		} else if (!_reactions) {
 | |
| 			_reactions = std::make_unique<Data::MessageReactions>(this);
 | |
| 		}
 | |
| 		const auto min = data.is_min();
 | |
| 		const auto &list = data.vresults().v;
 | |
| 		const auto &recent = data.vrecent_reactions().value_or_empty();
 | |
| 		if (min && hasUnreadReaction()) {
 | |
| 			// We can't update reactions from min if we have unread.
 | |
| 			if (_reactions->checkIfChanged(list, recent, min)) {
 | |
| 				updateReactionsUnknown();
 | |
| 			}
 | |
| 			return false;
 | |
| 		}
 | |
| 		return _reactions->change(list, recent, min);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyTTL(const MTPDmessage &data) {
 | |
| 	if (const auto period = data.vttl_period()) {
 | |
| 		if (period->v > 0) {
 | |
| 			applyTTL(data.vdate().v + period->v);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyTTL(const MTPDmessageService &data) {
 | |
| 	if (const auto period = data.vttl_period()) {
 | |
| 		if (period->v > 0) {
 | |
| 			applyTTL(data.vdate().v + period->v);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::createComponents(const MTPDmessage &data) {
 | |
| 	auto config = CreateConfig();
 | |
| 	if (const auto forwarded = data.vfwd_from()) {
 | |
| 		forwarded->match([&](const MTPDmessageFwdHeader &data) {
 | |
| 			FillForwardedInfo(config, data);
 | |
| 		});
 | |
| 	}
 | |
| 	if (const auto reply = data.vreply_to()) {
 | |
| 		config.reply = ReplyFieldsFromMTP(this, *reply);
 | |
| 	}
 | |
| 	config.viaBotId = data.vvia_bot_id().value_or_empty();
 | |
| 	config.viewsCount = data.vviews().value_or(-1);
 | |
| 	config.forwardsCount = data.vforwards().value_or(-1);
 | |
| 	config.replies = isScheduled()
 | |
| 		? HistoryMessageRepliesData()
 | |
| 		: HistoryMessageRepliesData(data.vreplies());
 | |
| 	config.markup = HistoryMessageMarkupData(data.vreply_markup());
 | |
| 	config.editDate = data.vedit_date().value_or_empty();
 | |
| 	config.postAuthor = qs(data.vpost_author().value_or_empty());
 | |
| 	createComponents(std::move(config));
 | |
| }
 | |
| 
 | |
| void HistoryItem::refreshMedia(const MTPMessageMedia *media) {
 | |
| 	const auto was = (_media != nullptr);
 | |
| 	if (const auto invoice = was ? _media->invoice() : nullptr) {
 | |
| 		if (invoice->extendedMedia) {
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 	_media = nullptr;
 | |
| 	if (media) {
 | |
| 		setMedia(*media);
 | |
| 	}
 | |
| 	if (was || _media) {
 | |
| 		if (const auto views = Get<HistoryMessageViews>()) {
 | |
| 			refreshRepliesText(views);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::refreshSentMedia(const MTPMessageMedia *media) {
 | |
| 	const auto wasGrouped = history()->owner().groups().isGrouped(this);
 | |
| 	refreshMedia(media);
 | |
| 	if (wasGrouped) {
 | |
| 		history()->owner().groups().refreshMessage(this);
 | |
| 	} else {
 | |
| 		history()->owner().requestItemViewRefresh(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::createServiceFromMtp(const MTPDmessage &message) {
 | |
| 	AddComponents(HistoryServiceData::Bit());
 | |
| 
 | |
| 	const auto unread = message.is_media_unread();
 | |
| 	const auto media = message.vmedia();
 | |
| 	Assert(media != nullptr);
 | |
| 
 | |
| 	media->match([&](const MTPDmessageMediaPhoto &data) {
 | |
| 		if (unread) {
 | |
| 			const auto ttl = data.vttl_seconds();
 | |
| 			Assert(ttl != nullptr);
 | |
| 
 | |
| 			setSelfDestruct(HistoryServiceSelfDestruct::Type::Photo, *ttl);
 | |
| 			if (out()) {
 | |
| 				setServiceText({
 | |
| 					tr::lng_ttl_photo_sent(tr::now, Ui::Text::WithEntities)
 | |
| 				});
 | |
| 			} else {
 | |
| 				auto result = PreparedServiceText();
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.text = tr::lng_ttl_photo_received(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					Ui::Text::WithEntities);
 | |
| 				setServiceText(std::move(result));
 | |
| 			}
 | |
| 		} else {
 | |
| 			setServiceText({
 | |
| 				tr::lng_ttl_photo_expired(tr::now, Ui::Text::WithEntities)
 | |
| 			});
 | |
| 		}
 | |
| 	}, [&](const MTPDmessageMediaDocument &data) {
 | |
| 		if (unread) {
 | |
| 			const auto ttl = data.vttl_seconds();
 | |
| 			Assert(ttl != nullptr);
 | |
| 
 | |
| 			setSelfDestruct(HistoryServiceSelfDestruct::Type::Video, *ttl);
 | |
| 			if (out()) {
 | |
| 				setServiceText({
 | |
| 					tr::lng_ttl_video_sent(tr::now, Ui::Text::WithEntities)
 | |
| 				});
 | |
| 			} else {
 | |
| 				auto result = PreparedServiceText();
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.text = tr::lng_ttl_video_received(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					Ui::Text::WithEntities);
 | |
| 				setServiceText(std::move(result));
 | |
| 			}
 | |
| 		} else {
 | |
| 			setServiceText({
 | |
| 				tr::lng_ttl_video_expired(tr::now, Ui::Text::WithEntities)
 | |
| 			});
 | |
| 		}
 | |
| 	}, [&](const MTPDmessageMediaStory &data) {
 | |
| 		setServiceText(prepareStoryMentionText());
 | |
| 	}, [](const auto &) {
 | |
| 		Unexpected("Media type in HistoryItem::createServiceFromMtp()");
 | |
| 	});
 | |
| 
 | |
| 	if (const auto reactions = message.vreactions()) {
 | |
| 		updateReactions(reactions);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::createServiceFromMtp(const MTPDmessageService &message) {
 | |
| 	AddComponents(HistoryServiceData::Bit());
 | |
| 
 | |
| 	const auto &action = message.vaction();
 | |
| 	const auto type = action.type();
 | |
| 	if (type == mtpc_messageActionPinMessage) {
 | |
| 		UpdateComponents(HistoryServicePinned::Bit());
 | |
| 	} else if (type == mtpc_messageActionTopicCreate
 | |
| 		|| type == mtpc_messageActionTopicEdit) {
 | |
| 		UpdateComponents(HistoryServiceTopicInfo::Bit());
 | |
| 		const auto info = Get<HistoryServiceTopicInfo>();
 | |
| 		info->topicPost = true;
 | |
| 		if (type == mtpc_messageActionTopicEdit) {
 | |
| 			const auto &data = action.c_messageActionTopicEdit();
 | |
| 			if (const auto title = data.vtitle()) {
 | |
| 				info->title = qs(*title);
 | |
| 				info->renamed = true;
 | |
| 			}
 | |
| 			if (const auto icon = data.vicon_emoji_id()) {
 | |
| 				info->iconId = icon->v;
 | |
| 				info->reiconed = true;
 | |
| 			}
 | |
| 			if (const auto closed = data.vclosed()) {
 | |
| 				info->closed = mtpIsTrue(*closed);
 | |
| 				info->reopened = !info->closed;
 | |
| 			}
 | |
| 			if (const auto hidden = data.vhidden()) {
 | |
| 				info->hidden = mtpIsTrue(*hidden);
 | |
| 				info->unhidden = !info->hidden;
 | |
| 			}
 | |
| 		} else {
 | |
| 			const auto &data = action.c_messageActionTopicCreate();
 | |
| 			info->title = qs(data.vtitle());
 | |
| 			info->iconId = data.vicon_emoji_id().value_or_empty();
 | |
| 		}
 | |
| 	} else if (type == mtpc_messageActionSetChatTheme) {
 | |
| 		setupChatThemeChange();
 | |
| 	} else if (type == mtpc_messageActionSetMessagesTTL) {
 | |
| 		setupTTLChange();
 | |
| 	} else if (type == mtpc_messageActionGameScore) {
 | |
| 		const auto &data = action.c_messageActionGameScore();
 | |
| 		UpdateComponents(HistoryServiceGameScore::Bit());
 | |
| 		Get<HistoryServiceGameScore>()->score = data.vscore().v;
 | |
| 	} else if (type == mtpc_messageActionPaymentSent) {
 | |
| 		const auto &data = action.c_messageActionPaymentSent();
 | |
| 		UpdateComponents(HistoryServicePayment::Bit());
 | |
| 		const auto amount = data.vtotal_amount().v;
 | |
| 		const auto currency = qs(data.vcurrency());
 | |
| 		const auto payment = Get<HistoryServicePayment>();
 | |
| 		const auto id = fullId();
 | |
| 		const auto owner = &_history->owner();
 | |
| 		payment->slug = data.vinvoice_slug().value_or_empty();
 | |
| 		payment->recurringInit = data.is_recurring_init();
 | |
| 		payment->recurringUsed = data.is_recurring_used();
 | |
| 		payment->amount = Ui::FillAmountAndCurrency(amount, currency);
 | |
| 		payment->invoiceLink = std::make_shared<LambdaClickHandler>([=](
 | |
| 				ClickContext context) {
 | |
| 			using namespace Payments;
 | |
| 			const auto my = context.other.value<ClickHandlerContext>();
 | |
| 			const auto weak = my.sessionWindow;
 | |
| 			if (const auto item = owner->message(id)) {
 | |
| 				CheckoutProcess::Start(
 | |
| 					item,
 | |
| 					Mode::Receipt,
 | |
| 					crl::guard(weak, [=](auto) { weak->window().activate(); }));
 | |
| 			}
 | |
| 		});
 | |
| 	} else if (type == mtpc_messageActionGroupCall
 | |
| 		|| type == mtpc_messageActionGroupCallScheduled) {
 | |
| 		const auto started = (type == mtpc_messageActionGroupCall);
 | |
| 		const auto &callData = started
 | |
| 			? action.c_messageActionGroupCall().vcall()
 | |
| 			: action.c_messageActionGroupCallScheduled().vcall();
 | |
| 		const auto duration = started
 | |
| 			? action.c_messageActionGroupCall().vduration()
 | |
| 			: tl::conditional<MTPint>();
 | |
| 		if (duration) {
 | |
| 			RemoveComponents(HistoryServiceOngoingCall::Bit());
 | |
| 		} else {
 | |
| 			UpdateComponents(HistoryServiceOngoingCall::Bit());
 | |
| 			const auto call = Get<HistoryServiceOngoingCall>();
 | |
| 			call->id = CallIdFromInput(callData);
 | |
| 			call->link = GroupCallClickHandler(_history->peer, call->id);
 | |
| 		}
 | |
| 	} else if (type == mtpc_messageActionInviteToGroupCall) {
 | |
| 		const auto &data = action.c_messageActionInviteToGroupCall();
 | |
| 		const auto id = CallIdFromInput(data.vcall());
 | |
| 		const auto peer = _history->peer;
 | |
| 		const auto has = PeerHasThisCall(peer, id);
 | |
| 		auto hasLink = !has.has_value()
 | |
| 			? PeerHasThisCallValue(peer, id)
 | |
| 			: (*has)
 | |
| 			? PeerHasThisCallValue(
 | |
| 				peer,
 | |
| 				id) | rpl::skip(1) | rpl::type_erased()
 | |
| 			: rpl::producer<bool>();
 | |
| 		if (!hasLink) {
 | |
| 			RemoveComponents(HistoryServiceOngoingCall::Bit());
 | |
| 		} else {
 | |
| 			UpdateComponents(HistoryServiceOngoingCall::Bit());
 | |
| 			const auto call = Get<HistoryServiceOngoingCall>();
 | |
| 			call->id = id;
 | |
| 			call->lifetime.destroy();
 | |
| 
 | |
| 			const auto users = data.vusers().v;
 | |
| 			std::move(hasLink) | rpl::start_with_next([=](bool has) {
 | |
| 				updateServiceText(
 | |
| 					prepareInvitedToCallText(
 | |
| 						ParseInvitedToCallUsers(this, users),
 | |
| 						has ? id : 0));
 | |
| 				if (!has) {
 | |
| 					RemoveComponents(HistoryServiceOngoingCall::Bit());
 | |
| 				}
 | |
| 			}, call->lifetime);
 | |
| 		}
 | |
| 	} else if (type == mtpc_messageActionSetChatWallPaper) {
 | |
| 		if (action.c_messageActionSetChatWallPaper().is_same()) {
 | |
| 			UpdateComponents(HistoryServiceSameBackground::Bit());
 | |
| 		} else {
 | |
| 			RemoveComponents(HistoryServiceSameBackground::Bit());
 | |
| 		}
 | |
| 	} else if (type == mtpc_messageActionGiveawayResults) {
 | |
| 		UpdateComponents(HistoryServiceGiveawayResults::Bit());
 | |
| 	}
 | |
| 	if (const auto replyTo = message.vreply_to()) {
 | |
| 		replyTo->match([&](const MTPDmessageReplyHeader &data) {
 | |
| 			const auto peerId = data.vreply_to_peer_id()
 | |
| 				? peerFromMTP(*data.vreply_to_peer_id())
 | |
| 				: _history->peer->id;
 | |
| 			if (const auto dependent = GetServiceDependentData()) {
 | |
| 				const auto id = data.vreply_to_msg_id().value_or_empty();
 | |
| 				if (id) {
 | |
| 					dependent->peerId = (peerId != _history->peer->id)
 | |
| 						? peerId
 | |
| 						: 0;
 | |
| 					dependent->msgId = id;
 | |
| 					dependent->topId = data.vreply_to_top_id().value_or(id);
 | |
| 					dependent->topicPost = data.is_forum_topic()
 | |
| 						|| Has<HistoryServiceTopicInfo>();
 | |
| 					if (!updateServiceDependent()) {
 | |
| 						RequestDependentMessageItem(
 | |
| 							this,
 | |
| 							(dependent->peerId
 | |
| 								? dependent->peerId
 | |
| 								: _history->peer->id),
 | |
| 							dependent->msgId);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}, [](const MTPDmessageReplyStoryHeader &data) {
 | |
| 		});
 | |
| 	}
 | |
| 	setServiceMessageByAction(action);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setMedia(const MTPMessageMedia &media) {
 | |
| 	_media = CreateMedia(this, media);
 | |
| 	checkStoryForwardInfo();
 | |
| 	checkBuyButton();
 | |
| }
 | |
| 
 | |
| void HistoryItem::checkStoryForwardInfo() {
 | |
| 	if (const auto storyId = _media ? _media->storyId() : FullStoryId()) {
 | |
| 		const auto adding = !Has<HistoryMessageForwarded>();
 | |
| 		if (adding) {
 | |
| 			AddComponents(HistoryMessageForwarded::Bit());
 | |
| 		}
 | |
| 		const auto forwarded = Get<HistoryMessageForwarded>();
 | |
| 		if (forwarded->story || adding) {
 | |
| 			const auto peer = history()->owner().peer(storyId.peer);
 | |
| 			forwarded->story = true;
 | |
| 			forwarded->originalSender = peer;
 | |
| 		}
 | |
| 	} else if (const auto forwarded = Get<HistoryMessageForwarded>()) {
 | |
| 		if (forwarded->story) {
 | |
| 			RemoveComponents(HistoryMessageForwarded::Bit());
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyServiceDateEdition(const MTPDmessageService &data) {
 | |
| 	const auto date = data.vdate().v;
 | |
| 	if (_date == date) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_date = date;
 | |
| }
 | |
| 
 | |
| void HistoryItem::setServiceMessageByAction(const MTPmessageAction &action) {
 | |
| 	auto prepareChatAddUserText = [this](const MTPDmessageActionChatAddUser &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		auto &users = action.vusers().v;
 | |
| 		if (users.size() == 1) {
 | |
| 			auto u = _history->owner().user(users[0].v);
 | |
| 			if (u == _from) {
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.text = tr::lng_action_user_joined(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.links.push_back(u->createOpenLink());
 | |
| 				result.text = tr::lng_action_add_user(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					lt_user,
 | |
| 					Ui::Text::Link(u->name(), 2), // Link 2.
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		} else if (users.isEmpty()) {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_add_user(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_user,
 | |
| 				{ .text = u"somebody"_q },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			for (auto i = 0, l = int(users.size()); i != l; ++i) {
 | |
| 				auto user = _history->owner().user(users[i].v);
 | |
| 				result.links.push_back(user->createOpenLink());
 | |
| 
 | |
| 				auto linkText = Ui::Text::Link(user->name(), 2 + i);
 | |
| 				if (i == 0) {
 | |
| 					result.text = linkText;
 | |
| 				} else if (i + 1 == l) {
 | |
| 					result.text = tr::lng_action_add_users_and_last(
 | |
| 						tr::now,
 | |
| 						lt_accumulated,
 | |
| 						result.text,
 | |
| 						lt_user,
 | |
| 						linkText,
 | |
| 						Ui::Text::WithEntities);
 | |
| 				} else {
 | |
| 					result.text = tr::lng_action_add_users_and_one(
 | |
| 						tr::now,
 | |
| 						lt_accumulated,
 | |
| 						result.text,
 | |
| 						lt_user,
 | |
| 						linkText,
 | |
| 						Ui::Text::WithEntities);
 | |
| 				}
 | |
| 			}
 | |
| 			result.text = tr::lng_action_add_users_many(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_users,
 | |
| 				result.text,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatJoinedByLink = [this](const MTPDmessageActionChatJoinedByLink &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.text = tr::lng_action_user_joined_by_link(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatCreate = [this](const MTPDmessageActionChatCreate &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.text = tr::lng_action_created_chat(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			lt_title,
 | |
| 			{ .text = qs(action.vtitle()) },
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChannelCreate = [this](const MTPDmessageActionChannelCreate &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (isPost()) {
 | |
| 			result.text = tr::lng_action_created_channel(
 | |
| 				tr::now,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_created_chat(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_title,
 | |
| 				{ .text = qs(action.vtitle()) },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatDeletePhoto = [&](const MTPDmessageActionChatDeletePhoto &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (isPost()) {
 | |
| 			result.text = tr::lng_action_removed_photo_channel(
 | |
| 				tr::now,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_removed_photo(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatDeleteUser = [this](const MTPDmessageActionChatDeleteUser &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (peerFromUser(action.vuser_id()) == _from->id) {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_user_left(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			auto user = _history->owner().user(action.vuser_id().v);
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.links.push_back(user->createOpenLink());
 | |
| 			result.text = tr::lng_action_kick_user(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_user,
 | |
| 				Ui::Text::Link(user->name(), 2), // Link 2.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatEditPhoto = [this](const MTPDmessageActionChatEditPhoto &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (isPost()) {
 | |
| 			result.text = tr::lng_action_changed_photo_channel(
 | |
| 				tr::now,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_changed_photo(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatEditTitle = [this](const MTPDmessageActionChatEditTitle &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (isPost()) {
 | |
| 			result.text = tr::lng_action_changed_title_channel(
 | |
| 				tr::now,
 | |
| 				lt_title,
 | |
| 				{ .text = (qs(action.vtitle())) },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_changed_title(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_title,
 | |
| 				{ .text = qs(action.vtitle()) },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto preparePinMessage = [&](const MTPDmessageActionPinMessage &) {
 | |
| 		return preparePinnedText();
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGameScore = [&](const MTPDmessageActionGameScore &) {
 | |
| 		return prepareGameScoreText();
 | |
| 	};
 | |
| 
 | |
| 	auto preparePhoneCall = [&](const MTPDmessageActionPhoneCall &) -> PreparedServiceText {
 | |
| 		Unexpected("PhoneCall type in setServiceMessageFromMtp.");
 | |
| 	};
 | |
| 
 | |
| 	auto preparePaymentSent = [&](const MTPDmessageActionPaymentSent &) {
 | |
| 		return preparePaymentSentText();
 | |
| 	};
 | |
| 
 | |
| 	auto prepareScreenshotTaken = [this](const MTPDmessageActionScreenshotTaken &) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (out()) {
 | |
| 			result.text = tr::lng_action_you_took_screenshot(
 | |
| 				tr::now,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_took_screenshot(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareCustomAction = [&](const MTPDmessageActionCustomAction &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.text = { .text = qs(action.vmessage()) };
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareBotAllowed = [&](const MTPDmessageActionBotAllowed &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (action.is_attach_menu()) {
 | |
| 			result.text = {
 | |
| 				tr::lng_action_attach_menu_bot_allowed(tr::now)
 | |
| 			};
 | |
| 		} else if (action.is_from_request()) {
 | |
| 			result.text = {
 | |
| 				tr::lng_action_webapp_bot_allowed(tr::now)
 | |
| 			};
 | |
| 		} else if (const auto app = action.vapp()) {
 | |
| 			const auto bot = history()->peer->asUser();
 | |
| 			const auto botId = bot ? bot->id : PeerId();
 | |
| 			const auto info = history()->owner().processBotApp(botId, *app);
 | |
| 			const auto url = (bot && info)
 | |
| 				? history()->session().createInternalLinkFull(
 | |
| 					bot->username() + '/' + info->shortName)
 | |
| 				: QString();
 | |
| 			result.text = tr::lng_action_bot_allowed_from_app(
 | |
| 				tr::now,
 | |
| 				lt_app,
 | |
| 				(url.isEmpty()
 | |
| 					? TextWithEntities{ u"App"_q }
 | |
| 					: Ui::Text::Link(info->title, url)),
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			const auto domain = qs(action.vdomain().value_or_empty());
 | |
| 			result.text = tr::lng_action_bot_allowed_from_domain(
 | |
| 				tr::now,
 | |
| 				lt_domain,
 | |
| 				Ui::Text::Link(domain, u"http://"_q + domain),
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareSecureValuesSent = [&](const MTPDmessageActionSecureValuesSent &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		auto documents = QStringList();
 | |
| 		for (const auto &type : action.vtypes().v) {
 | |
| 			documents.push_back([&] {
 | |
| 				switch (type.type()) {
 | |
| 				case mtpc_secureValueTypePersonalDetails:
 | |
| 					return tr::lng_action_secure_personal_details(tr::now);
 | |
| 				case mtpc_secureValueTypePassport:
 | |
| 				case mtpc_secureValueTypeDriverLicense:
 | |
| 				case mtpc_secureValueTypeIdentityCard:
 | |
| 				case mtpc_secureValueTypeInternalPassport:
 | |
| 					return tr::lng_action_secure_proof_of_identity(tr::now);
 | |
| 				case mtpc_secureValueTypeAddress:
 | |
| 					return tr::lng_action_secure_address(tr::now);
 | |
| 				case mtpc_secureValueTypeUtilityBill:
 | |
| 				case mtpc_secureValueTypeBankStatement:
 | |
| 				case mtpc_secureValueTypeRentalAgreement:
 | |
| 				case mtpc_secureValueTypePassportRegistration:
 | |
| 				case mtpc_secureValueTypeTemporaryRegistration:
 | |
| 					return tr::lng_action_secure_proof_of_address(tr::now);
 | |
| 				case mtpc_secureValueTypePhone:
 | |
| 					return tr::lng_action_secure_phone(tr::now);
 | |
| 				case mtpc_secureValueTypeEmail:
 | |
| 					return tr::lng_action_secure_email(tr::now);
 | |
| 				}
 | |
| 				Unexpected("Type in prepareSecureValuesSent.");
 | |
| 			}());
 | |
| 		};
 | |
| 		result.links.push_back(_history->peer->createOpenLink());
 | |
| 		result.text = tr::lng_action_secure_values_sent(
 | |
| 			tr::now,
 | |
| 			lt_user,
 | |
| 			Ui::Text::Link(_history->peer->name(), QString()), // Link 1.
 | |
| 			lt_documents,
 | |
| 			{ .text = documents.join(", ") },
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareContactSignUp = [this](const MTPDmessageActionContactSignUp &data) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.text = tr::lng_action_user_registered(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareProximityReached = [this](const MTPDmessageActionGeoProximityReached &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		const auto fromId = peerFromMTP(action.vfrom_id());
 | |
| 		const auto fromPeer = _history->owner().peer(fromId);
 | |
| 		const auto toId = peerFromMTP(action.vto_id());
 | |
| 		const auto toPeer = _history->owner().peer(toId);
 | |
| 		const auto selfId = _from->session().userPeerId();
 | |
| 		const auto distanceMeters = action.vdistance().v;
 | |
| 		const auto distance = [&] {
 | |
| 			if (distanceMeters >= 1000) {
 | |
| 				const auto km = (10 * (distanceMeters / 10)) / 1000.;
 | |
| 				return tr::lng_action_proximity_distance_km(
 | |
| 					tr::now,
 | |
| 					lt_count,
 | |
| 					km);
 | |
| 			} else {
 | |
| 				return tr::lng_action_proximity_distance_m(
 | |
| 					tr::now,
 | |
| 					lt_count,
 | |
| 					distanceMeters);
 | |
| 			}
 | |
| 		}();
 | |
| 		result.text = [&] {
 | |
| 			if (fromId == selfId) {
 | |
| 				result.links.push_back(toPeer->createOpenLink());
 | |
| 				return tr::lng_action_you_proximity_reached(
 | |
| 					tr::now,
 | |
| 					lt_distance,
 | |
| 					{ .text = distance },
 | |
| 					lt_user,
 | |
| 					Ui::Text::Link(toPeer->name(), QString()), // Link 1.
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else if (toId == selfId) {
 | |
| 				result.links.push_back(fromPeer->createOpenLink());
 | |
| 				return tr::lng_action_proximity_reached_you(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					Ui::Text::Link(fromPeer->name(), QString()), // Link 1.
 | |
| 					lt_distance,
 | |
| 					{ .text = distance },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.links.push_back(fromPeer->createOpenLink());
 | |
| 				result.links.push_back(toPeer->createOpenLink());
 | |
| 				return tr::lng_action_proximity_reached(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					Ui::Text::Link(fromPeer->name(), 1), // Link 1.
 | |
| 					lt_distance,
 | |
| 					{ .text = distance },
 | |
| 					lt_user,
 | |
| 					Ui::Text::Link(toPeer->name(), 2), // Link 2.
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		}();
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGroupCall = [this](const MTPDmessageActionGroupCall &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (const auto duration = action.vduration()) {
 | |
| 			const auto seconds = duration->v;
 | |
| 			const auto days = seconds / 86400;
 | |
| 			const auto hours = seconds / 3600;
 | |
| 			const auto minutes = seconds / 60;
 | |
| 			auto text = (days > 1)
 | |
| 				? tr::lng_days(tr::now, lt_count, days)
 | |
| 				: (hours > 1)
 | |
| 				? tr::lng_hours(tr::now, lt_count, hours)
 | |
| 				: (minutes > 1)
 | |
| 				? tr::lng_minutes(tr::now, lt_count, minutes)
 | |
| 				: tr::lng_seconds(tr::now, lt_count, seconds);
 | |
| 			if (_history->peer->isBroadcast()) {
 | |
| 				result.text = tr::lng_action_group_call_finished(
 | |
| 					tr::now,
 | |
| 					lt_duration,
 | |
| 					{ .text = text },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.text = tr::lng_action_group_call_finished_group(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					lt_duration,
 | |
| 					{ .text = text },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 			return result;
 | |
| 		}
 | |
| 		if (_history->peer->isBroadcast()) {
 | |
| 			result.text = tr::lng_action_group_call_started_channel(
 | |
| 				tr::now,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_group_call_started_group(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareInviteToGroupCall = [this](const MTPDmessageActionInviteToGroupCall &action) {
 | |
| 		const auto callId = CallIdFromInput(action.vcall());
 | |
| 		const auto owner = &_history->owner();
 | |
| 		const auto peer = _history->peer;
 | |
| 		for (const auto &id : action.vusers().v) {
 | |
| 			const auto user = owner->user(id.v);
 | |
| 			if (callId) {
 | |
| 				owner->registerInvitedToCallUser(callId, peer, user);
 | |
| 			}
 | |
| 		};
 | |
| 		const auto linkCallId = PeerHasThisCall(peer, callId).value_or(false)
 | |
| 			? callId
 | |
| 			: 0;
 | |
| 		return prepareInvitedToCallText(
 | |
| 			ParseInvitedToCallUsers(this, action.vusers().v),
 | |
| 			linkCallId);
 | |
| 	};
 | |
| 
 | |
| 	auto prepareSetMessagesTTL = [this](const MTPDmessageActionSetMessagesTTL &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		const auto period = action.vperiod().v;
 | |
| 		const auto duration = (period == 5)
 | |
| 			? u"5 seconds"_q
 | |
| 			: Ui::FormatTTL(period);
 | |
| 		if (const auto from = action.vauto_setting_from(); from && period) {
 | |
| 			if (const auto peer = _from->owner().peer(peerFromUser(*from))) {
 | |
| 				result.text = (peer->id == peer->session().userPeerId())
 | |
| 					? tr::lng_action_ttl_global_me(
 | |
| 						tr::now,
 | |
| 						lt_duration,
 | |
| 						{ .text = duration },
 | |
| 						Ui::Text::WithEntities)
 | |
| 					: tr::lng_action_ttl_global(
 | |
| 						tr::now,
 | |
| 						lt_from,
 | |
| 						Ui::Text::Link(peer->name(), 1), // Link 1.
 | |
| 						lt_duration,
 | |
| 						{ .text = duration },
 | |
| 						Ui::Text::WithEntities);
 | |
| 				return result;
 | |
| 			}
 | |
| 		}
 | |
| 		if (isPost()) {
 | |
| 			if (!period) {
 | |
| 				result.text = tr::lng_action_ttl_removed_channel(
 | |
| 					tr::now,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.text = tr::lng_action_ttl_changed_channel(
 | |
| 					tr::now,
 | |
| 					lt_duration,
 | |
| 					{ .text = duration },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		} else if (_from->isSelf()) {
 | |
| 			if (!period) {
 | |
| 				result.text = tr::lng_action_ttl_removed_you(
 | |
| 					tr::now,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.text = tr::lng_action_ttl_changed_you(
 | |
| 					tr::now,
 | |
| 					lt_duration,
 | |
| 					{ .text = duration },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			if (!period) {
 | |
| 				result.text = tr::lng_action_ttl_removed(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.text = tr::lng_action_ttl_changed(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					lt_duration,
 | |
| 					{ .text = duration },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGroupCallScheduled = [&](const MTPDmessageActionGroupCallScheduled &data) {
 | |
| 		return prepareCallScheduledText(data.vschedule_date().v);
 | |
| 	};
 | |
| 
 | |
| 	auto prepareSetChatTheme = [this](const MTPDmessageActionSetChatTheme &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		const auto text = qs(action.vemoticon());
 | |
| 		if (!text.isEmpty()) {
 | |
| 			if (_from->isSelf()) {
 | |
| 				result.text = tr::lng_action_you_theme_changed(
 | |
| 					tr::now,
 | |
| 					lt_emoji,
 | |
| 					{ .text = text },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.text = tr::lng_action_theme_changed(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					lt_emoji,
 | |
| 					{ .text = text },
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		} else {
 | |
| 			if (_from->isSelf()) {
 | |
| 				result.text = tr::lng_action_you_theme_disabled(
 | |
| 					tr::now,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.links.push_back(fromLink());
 | |
| 				result.text = tr::lng_action_theme_disabled(
 | |
| 					tr::now,
 | |
| 					lt_from,
 | |
| 					fromLinkText(), // Link 1.
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareChatJoinedByRequest = [this](const MTPDmessageActionChatJoinedByRequest &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.text = tr::lng_action_user_joined_by_request(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareWebViewDataSent = [](const MTPDmessageActionWebViewDataSent &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.text = tr::lng_action_webview_data_done(
 | |
| 			tr::now,
 | |
| 			lt_text,
 | |
| 			{ .text = qs(action.vtext()) },
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGiftPremium = [&](
 | |
| 			const MTPDmessageActionGiftPremium &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		const auto isSelf = (_from->id == _from->session().userPeerId());
 | |
| 		const auto peer = isSelf ? _history->peer : _from;
 | |
| 		_history->session().giftBoxStickersPacks().load();
 | |
| 		const auto amount = action.vamount().v;
 | |
| 		const auto currency = qs(action.vcurrency());
 | |
| 		result.links.push_back(peer->createOpenLink());
 | |
| 		result.text = (isSelf
 | |
| 			? tr::lng_action_gift_received_me
 | |
| 			: tr::lng_action_gift_received)(
 | |
| 				tr::now,
 | |
| 				lt_user,
 | |
| 				Ui::Text::Link(peer->name(), 1), // Link 1.
 | |
| 				lt_cost,
 | |
| 				{ Ui::FillAmountAndCurrency(amount, currency) },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareTopicCreate = [&](const MTPDmessageActionTopicCreate &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		const auto topicUrl = u"internal:url:https://t.me/c/%1/%2"_q
 | |
| 			.arg(peerToChannel(_history->peer->id).bare)
 | |
| 			.arg(id.bare);
 | |
| 		result.text = tr::lng_action_topic_created(
 | |
| 			tr::now,
 | |
| 			lt_topic,
 | |
| 			Ui::Text::Link(
 | |
| 				Data::ForumTopicIconWithTitle(
 | |
| 					id,
 | |
| 					action.vicon_emoji_id().value_or_empty(),
 | |
| 					qs(action.vtitle())),
 | |
| 				topicUrl),
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareTopicEdit = [&](const MTPDmessageActionTopicEdit &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		if (const auto closed = action.vclosed()) {
 | |
| 			result.text = { mtpIsTrue(*closed)
 | |
| 				? tr::lng_action_topic_closed_inside(tr::now)
 | |
| 				: tr::lng_action_topic_reopened_inside(tr::now) };
 | |
| 		} else if (const auto hidden = action.vhidden()) {
 | |
| 			result.text = { mtpIsTrue(*hidden)
 | |
| 				? tr::lng_action_topic_hidden_inside(tr::now)
 | |
| 				: tr::lng_action_topic_unhidden_inside(tr::now) };
 | |
| 		} else if (!action.vtitle()) {
 | |
| 			if (const auto icon = action.vicon_emoji_id()) {
 | |
| 				if (const auto iconId = icon->v) {
 | |
| 					result.links.push_back(fromLink());
 | |
| 					result.text = tr::lng_action_topic_icon_changed(
 | |
| 						tr::now,
 | |
| 						lt_from,
 | |
| 						fromLinkText(), // Link 1.
 | |
| 						lt_link,
 | |
| 						{ tr::lng_action_topic_placeholder(tr::now) },
 | |
| 						lt_emoji,
 | |
| 						Data::SingleCustomEmoji(iconId),
 | |
| 						Ui::Text::WithEntities);
 | |
| 				} else {
 | |
| 					result.links.push_back(fromLink());
 | |
| 					result.text = tr::lng_action_topic_icon_removed(
 | |
| 						tr::now,
 | |
| 						lt_from,
 | |
| 						fromLinkText(), // Link 1.
 | |
| 						lt_link,
 | |
| 						{ tr::lng_action_topic_placeholder(tr::now) },
 | |
| 						Ui::Text::WithEntities);
 | |
| 				}
 | |
| 			}
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_topic_renamed(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_link,
 | |
| 				{ tr::lng_action_topic_placeholder(tr::now) },
 | |
| 				lt_title,
 | |
| 				Data::ForumTopicIconWithTitle(
 | |
| 					topicRootId(),
 | |
| 					action.vicon_emoji_id().value_or_empty(),
 | |
| 					qs(*action.vtitle())),
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		if (result.text.empty()) {
 | |
| 			result.text = { tr::lng_message_empty(tr::now) };
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareSuggestProfilePhoto = [this](const MTPDmessageActionSuggestProfilePhoto &action) {
 | |
| 		auto result = PreparedServiceText{};
 | |
| 		const auto isSelf = (_from->id == _from->session().userPeerId());
 | |
| 		const auto isVideo = action.vphoto().match([&](const MTPDphoto &data) {
 | |
| 			return data.vvideo_sizes().has_value()
 | |
| 				&& !data.vvideo_sizes()->v.isEmpty();
 | |
| 		}, [](const MTPDphotoEmpty &) {
 | |
| 			return false;
 | |
| 		});
 | |
| 		const auto peer = isSelf ? history()->peer : _from;
 | |
| 		const auto user = peer->asUser();
 | |
| 		const auto name = (user && !user->firstName.isEmpty())
 | |
| 			? user->firstName
 | |
| 			: peer->name();
 | |
| 		result.links.push_back(peer->createOpenLink());
 | |
| 		result.text = (isSelf
 | |
| 			? (isVideo
 | |
| 				? tr::lng_action_suggested_video_me
 | |
| 				: tr::lng_action_suggested_photo_me)
 | |
| 			: (isVideo
 | |
| 				? tr::lng_action_suggested_video
 | |
| 				: tr::lng_action_suggested_photo))(
 | |
| 				tr::now,
 | |
| 				lt_user,
 | |
| 				Ui::Text::Link(name, 1), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareRequestedPeer = [&](
 | |
| 			const MTPDmessageActionRequestedPeer &action) {
 | |
| 		auto result = PreparedServiceText{};
 | |
| 		result.links.push_back(fromLink());
 | |
| 
 | |
| 		const auto &list = action.vpeers().v;
 | |
| 		for (auto i = 0, count = int(list.size()); i != count; ++i) {
 | |
| 			const auto id = peerFromMTP(list[i]);
 | |
| 
 | |
| 			auto user = _history->owner().peer(id);
 | |
| 			result.links.push_back(user->createOpenLink());
 | |
| 
 | |
| 			auto linkText = Ui::Text::Link(user->name(), 2 + i);
 | |
| 			if (i == 0) {
 | |
| 				result.text = linkText;
 | |
| 			} else if (i + 1 == count) {
 | |
| 				result.text = tr::lng_action_add_users_and_last(
 | |
| 					tr::now,
 | |
| 					lt_accumulated,
 | |
| 					result.text,
 | |
| 					lt_user,
 | |
| 					linkText,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.text = tr::lng_action_add_users_and_one(
 | |
| 					tr::now,
 | |
| 					lt_accumulated,
 | |
| 					result.text,
 | |
| 					lt_user,
 | |
| 					linkText,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		result.text = tr::lng_action_shared_chat_with_bot(
 | |
| 			tr::now,
 | |
| 			lt_chat,
 | |
| 			result.text,
 | |
| 			lt_bot,
 | |
| 			Ui::Text::Link(history()->peer->name(), 2),
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareSetChatWallPaper = [&](
 | |
| 			const MTPDmessageActionSetChatWallPaper &action) {
 | |
| 		const auto isSelf = (_from->id == _from->session().userPeerId());
 | |
| 		const auto same = action.is_same();
 | |
| 		const auto both = action.is_for_both();
 | |
| 		const auto peer = isSelf ? history()->peer : _from;
 | |
| 		const auto user = peer->asUser();
 | |
| 		const auto name = (user && !user->firstName.isEmpty())
 | |
| 			? user->firstName
 | |
| 			: peer->name();
 | |
| 		auto result = PreparedServiceText{};
 | |
| 		if (!isSelf) {
 | |
| 			result.links.push_back(peer->createOpenLink());
 | |
| 		}
 | |
| 		result.text = isSelf
 | |
| 			? ((!same && both)
 | |
| 				? tr::lng_action_set_wallpaper_both_me(
 | |
| 					tr::now,
 | |
| 					lt_user,
 | |
| 					Ui::Text::Link(Ui::Text::Bold(name), 1),
 | |
| 					Ui::Text::WithEntities)
 | |
| 				: (same
 | |
| 					? tr::lng_action_set_same_wallpaper_me
 | |
| 					: tr::lng_action_set_wallpaper_me)(
 | |
| 						tr::now,
 | |
| 						Ui::Text::WithEntities))
 | |
| 			: (same
 | |
| 				? tr::lng_action_set_same_wallpaper
 | |
| 				: tr::lng_action_set_wallpaper)(
 | |
| 					tr::now,
 | |
| 					lt_user,
 | |
| 					Ui::Text::Link(Ui::Text::Bold(name), 1),
 | |
| 					Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGiftCode = [&](const MTPDmessageActionGiftCode &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		_history->session().giftBoxStickersPacks().load();
 | |
| 		if (const auto boosted = action.vboost_peer()) {
 | |
| 			result.text = {
 | |
| 				(action.is_unclaimed()
 | |
| 					? tr::lng_prize_unclaimed_about
 | |
| 					: action.is_via_giveaway()
 | |
| 					? tr::lng_prize_about
 | |
| 					: tr::lng_prize_gift_about)(
 | |
| 						tr::now,
 | |
| 						lt_channel,
 | |
| 						_from->owner().peer(
 | |
| 							peerFromMTP(*action.vboost_peer()))->name()),
 | |
| 			};
 | |
| 		} else {
 | |
| 			const auto isSelf = (_from->id == _from->session().userPeerId());
 | |
| 			const auto peer = isSelf ? _history->peer : _from;
 | |
| 			result.links.push_back(peer->createOpenLink());
 | |
| 			result.text = (isSelf
 | |
| 				? tr::lng_action_gift_received_me
 | |
| 				: tr::lng_action_gift_received)(
 | |
| 					tr::now,
 | |
| 					lt_user,
 | |
| 					Ui::Text::Link(peer->name(), 1), // Link 1.
 | |
| 					lt_cost,
 | |
| 					{ Ui::FillAmountAndCurrency(
 | |
| 						action.vamount().value_or_empty(),
 | |
| 						qs(action.vcurrency().value_or_empty())) },
 | |
| 					Ui::Text::WithEntities);
 | |
| 
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGiveawayLaunch = [&](const MTPDmessageActionGiveawayLaunch &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.text = tr::lng_action_giveaway_started(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			Ui::Text::WithEntities);
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	auto prepareGiveawayResults = [&](const MTPDmessageActionGiveawayResults &action) {
 | |
| 		auto result = PreparedServiceText();
 | |
| 		const auto winners = action.vwinners_count().v;
 | |
| 		const auto unclaimed = action.vunclaimed_count().v;
 | |
| 		result.text = {
 | |
| 			(!winners
 | |
| 				? tr::lng_action_giveaway_results_none(tr::now)
 | |
| 				: unclaimed
 | |
| 				? tr::lng_action_giveaway_results_some(tr::now)
 | |
| 				: tr::lng_action_giveaway_results(
 | |
| 					tr::now,
 | |
| 					lt_count,
 | |
| 					winners))
 | |
| 		};
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	setServiceText(action.match(
 | |
| 		prepareChatAddUserText,
 | |
| 		prepareChatJoinedByLink,
 | |
| 		prepareChatCreate,
 | |
| 		PrepareEmptyText<MTPDmessageActionChatMigrateTo>,
 | |
| 		PrepareEmptyText<MTPDmessageActionChannelMigrateFrom>,
 | |
| 		PrepareEmptyText<MTPDmessageActionHistoryClear>,
 | |
| 		prepareChannelCreate,
 | |
| 		prepareChatDeletePhoto,
 | |
| 		prepareChatDeleteUser,
 | |
| 		prepareChatEditPhoto,
 | |
| 		prepareChatEditTitle,
 | |
| 		preparePinMessage,
 | |
| 		prepareGameScore,
 | |
| 		preparePhoneCall,
 | |
| 		preparePaymentSent,
 | |
| 		prepareScreenshotTaken,
 | |
| 		prepareCustomAction,
 | |
| 		prepareBotAllowed,
 | |
| 		prepareSecureValuesSent,
 | |
| 		prepareContactSignUp,
 | |
| 		prepareProximityReached,
 | |
| 		PrepareErrorText<MTPDmessageActionPaymentSentMe>,
 | |
| 		PrepareErrorText<MTPDmessageActionSecureValuesSentMe>,
 | |
| 		prepareGroupCall,
 | |
| 		prepareInviteToGroupCall,
 | |
| 		prepareSetMessagesTTL,
 | |
| 		prepareGroupCallScheduled,
 | |
| 		prepareSetChatTheme,
 | |
| 		prepareChatJoinedByRequest,
 | |
| 		prepareWebViewDataSent,
 | |
| 		prepareGiftPremium,
 | |
| 		prepareTopicCreate,
 | |
| 		prepareTopicEdit,
 | |
| 		PrepareErrorText<MTPDmessageActionWebViewDataSentMe>,
 | |
| 		prepareSuggestProfilePhoto,
 | |
| 		prepareRequestedPeer,
 | |
| 		prepareSetChatWallPaper,
 | |
| 		prepareGiftCode,
 | |
| 		prepareGiveawayLaunch,
 | |
| 		prepareGiveawayResults,
 | |
| 		PrepareErrorText<MTPDmessageActionEmpty>));
 | |
| 
 | |
| 	// Additional information.
 | |
| 	applyAction(action);
 | |
| }
 | |
| 
 | |
| void HistoryItem::applyAction(const MTPMessageAction &action) {
 | |
| 	action.match([&](const MTPDmessageActionChatAddUser &data) {
 | |
| 		if (const auto channel = _history->peer->asMegagroup()) {
 | |
| 			const auto selfUserId = _history->session().userId();
 | |
| 			for (const auto &item : data.vusers().v) {
 | |
| 				if (peerFromUser(item) == selfUserId) {
 | |
| 					channel->mgInfo->joinedMessageFound = true;
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}, [&](const MTPDmessageActionChatJoinedByLink &data) {
 | |
| 		if (_from->isSelf()) {
 | |
| 			if (const auto channel = _history->peer->asMegagroup()) {
 | |
| 				channel->mgInfo->joinedMessageFound = true;
 | |
| 			}
 | |
| 		}
 | |
| 	}, [&](const MTPDmessageActionChatEditPhoto &data) {
 | |
| 		data.vphoto().match([&](const MTPDphoto &photo) {
 | |
| 			_media = std::make_unique<Data::MediaPhoto>(
 | |
| 				this,
 | |
| 				_history->peer,
 | |
| 				_history->owner().processPhoto(photo));
 | |
| 		}, [](const MTPDphotoEmpty &) {
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageActionChatCreate &) {
 | |
| 		_flags |= MessageFlag::IsGroupEssential;
 | |
| 	}, [&](const MTPDmessageActionChannelCreate &) {
 | |
| 		_flags |= MessageFlag::IsGroupEssential;
 | |
| 	}, [&](const MTPDmessageActionChatMigrateTo &) {
 | |
| 		_flags |= MessageFlag::IsGroupEssential;
 | |
| 	}, [&](const MTPDmessageActionChannelMigrateFrom &) {
 | |
| 		_flags |= MessageFlag::IsGroupEssential;
 | |
| 	}, [&](const MTPDmessageActionContactSignUp &) {
 | |
| 		_flags |= MessageFlag::IsContactSignUp;
 | |
| 	}, [&](const MTPDmessageActionChatJoinedByRequest &data) {
 | |
| 		if (_from->isSelf()) {
 | |
| 			if (const auto channel = _history->peer->asMegagroup()) {
 | |
| 				channel->mgInfo->joinedMessageFound = true;
 | |
| 			}
 | |
| 		}
 | |
| 	}, [&](const MTPDmessageActionGiftPremium &data) {
 | |
| 		_media = std::make_unique<Data::MediaGiftBox>(
 | |
| 			this,
 | |
| 			_from,
 | |
| 			data.vmonths().v);
 | |
| 	}, [&](const MTPDmessageActionSuggestProfilePhoto &data) {
 | |
| 		data.vphoto().match([&](const MTPDphoto &photo) {
 | |
| 			_flags |= MessageFlag::IsUserpicSuggestion;
 | |
| 			_media = std::make_unique<Data::MediaPhoto>(
 | |
| 				this,
 | |
| 				history()->peer,
 | |
| 				history()->owner().processPhoto(photo));
 | |
| 		}, [](const MTPDphotoEmpty &) {
 | |
| 		});
 | |
| 	}, [&](const MTPDmessageActionSetChatWallPaper &data) {
 | |
| 		if (!data.is_same()) {
 | |
| 			using namespace Data;
 | |
| 			const auto session = &history()->session();
 | |
| 			const auto &attached = data.vwallpaper();
 | |
| 			if (const auto paper = WallPaper::Create(session, attached)) {
 | |
| 				_media = std::make_unique<MediaWallPaper>(
 | |
| 					this,
 | |
| 					*paper,
 | |
| 					data.is_for_both());
 | |
| 			}
 | |
| 		}
 | |
| 	}, [&](const MTPDmessageActionGiftCode &data) {
 | |
| 		const auto boostedId = data.vboost_peer()
 | |
| 			? peerToChannel(peerFromMTP(*data.vboost_peer()))
 | |
| 			: ChannelId();
 | |
| 		_media = std::make_unique<Data::MediaGiftBox>(
 | |
| 			this,
 | |
| 			_from,
 | |
| 			Data::GiftCode{
 | |
| 				.slug = qs(data.vslug()),
 | |
| 				.channel = (boostedId
 | |
| 					? history()->owner().channel(boostedId).get()
 | |
| 					: nullptr),
 | |
| 				.months = data.vmonths().v,
 | |
| 				.viaGiveaway = data.is_via_giveaway(),
 | |
| 				.unclaimed = data.is_unclaimed(),
 | |
| 			});
 | |
| 	}, [](const auto &) {
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void HistoryItem::setSelfDestruct(
 | |
| 		HistoryServiceSelfDestruct::Type type,
 | |
| 		MTPint mtpTTLvalue) {
 | |
| 	UpdateComponents(HistoryServiceSelfDestruct::Bit());
 | |
| 	const auto selfdestruct = Get<HistoryServiceSelfDestruct>();
 | |
| 	if (mtpTTLvalue.v == TimeId(0x7FFFFFFF)) {
 | |
| 		selfdestruct->timeToLive = TimeToLiveSingleView();
 | |
| 	} else {
 | |
| 		selfdestruct->timeToLive = mtpTTLvalue.v * crl::time(1000);
 | |
| 	}
 | |
| 	selfdestruct->type = type;
 | |
| }
 | |
| 
 | |
| PreparedServiceText HistoryItem::prepareInvitedToCallText(
 | |
| 		const std::vector<not_null<UserData*>> &users,
 | |
| 		CallId linkCallId) {
 | |
| 	auto chatText = tr::lng_action_invite_user_chat(
 | |
| 		tr::now,
 | |
| 		Ui::Text::WithEntities);
 | |
| 	auto result = PreparedServiceText();
 | |
| 	result.links.push_back(fromLink());
 | |
| 	auto linkIndex = 1;
 | |
| 	if (linkCallId) {
 | |
| 		const auto peer = _history->peer;
 | |
| 		result.links.push_back(GroupCallClickHandler(peer, linkCallId));
 | |
| 		chatText = Ui::Text::Link(chatText.text, ++linkIndex);
 | |
| 	}
 | |
| 	if (users.size() == 1) {
 | |
| 		auto user = users[0];
 | |
| 		result.links.push_back(user->createOpenLink());
 | |
| 		result.text = tr::lng_action_invite_user(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			lt_user,
 | |
| 			Ui::Text::Link(user->name(), ++linkIndex), // Link N.
 | |
| 			lt_chat,
 | |
| 			chatText,
 | |
| 			Ui::Text::WithEntities);
 | |
| 	} else if (users.empty()) {
 | |
| 		result.text = tr::lng_action_invite_user(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			lt_user,
 | |
| 			{ .text = u"somebody"_q },
 | |
| 			lt_chat,
 | |
| 			chatText,
 | |
| 			Ui::Text::WithEntities);
 | |
| 	} else {
 | |
| 		for (auto i = 0, l = int(users.size()); i != l; ++i) {
 | |
| 			const auto user = users[i];
 | |
| 			result.links.push_back(user->createOpenLink());
 | |
| 
 | |
| 			auto linkText = Ui::Text::Link(user->name(), ++linkIndex);
 | |
| 			if (i == 0) {
 | |
| 				result.text = linkText;
 | |
| 			} else if (i + 1 == l) {
 | |
| 				result.text = tr::lng_action_invite_users_and_last(
 | |
| 					tr::now,
 | |
| 					lt_accumulated,
 | |
| 					result.text,
 | |
| 					lt_user,
 | |
| 					linkText,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			} else {
 | |
| 				result.text = tr::lng_action_invite_users_and_one(
 | |
| 					tr::now,
 | |
| 					lt_accumulated,
 | |
| 					result.text,
 | |
| 					lt_user,
 | |
| 					linkText,
 | |
| 					Ui::Text::WithEntities);
 | |
| 			}
 | |
| 		}
 | |
| 		result.text = tr::lng_action_invite_users_many(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			lt_users,
 | |
| 			result.text,
 | |
| 			lt_chat,
 | |
| 			chatText,
 | |
| 			Ui::Text::WithEntities);
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| PreparedServiceText HistoryItem::preparePinnedText() {
 | |
| 	auto result = PreparedServiceText();
 | |
| 	auto pinned = Get<HistoryServicePinned>();
 | |
| 	if (pinned && pinned->msg) {
 | |
| 		const auto mediaText = [&] {
 | |
| 			using TTL = HistoryServiceSelfDestruct;
 | |
| 			if (const auto media = pinned->msg->media()) {
 | |
| 				return media->pinnedTextSubstring();
 | |
| 			} else if (const auto selfdestruct = pinned->msg->Get<TTL>()) {
 | |
| 				if (selfdestruct->type == TTL::Type::Photo) {
 | |
| 					return tr::lng_action_pinned_media_photo(tr::now);
 | |
| 				} else if (selfdestruct->type == TTL::Type::Video) {
 | |
| 					return tr::lng_action_pinned_media_video(tr::now);
 | |
| 				}
 | |
| 			}
 | |
| 			return QString();
 | |
| 		}();
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.links.push_back(pinned->lnk);
 | |
| 		if (mediaText.isEmpty()) {
 | |
| 			auto original = pinned->msg->translatedText();
 | |
| 			auto cutAt = 0;
 | |
| 			auto limit = kPinnedMessageTextLimit;
 | |
| 			auto size = original.text.size();
 | |
| 			for (; limit != 0;) {
 | |
| 				--limit;
 | |
| 				if (cutAt >= size) break;
 | |
| 				if (original.text.at(cutAt).isLowSurrogate()
 | |
| 					&& (cutAt + 1 < size)
 | |
| 					&& original.text.at(cutAt + 1).isHighSurrogate()) {
 | |
| 					cutAt += 2;
 | |
| 				} else {
 | |
| 					++cutAt;
 | |
| 				}
 | |
| 			}
 | |
| 			if (!limit && cutAt + 5 < size) {
 | |
| 				original = Ui::Text::Mid(original, 0, cutAt).append(
 | |
| 					Ui::kQEllipsis);
 | |
| 			}
 | |
| 			original = Ui::Text::Link(
 | |
| 				Ui::Text::Filtered(
 | |
| 					std::move(original),
 | |
| 					{
 | |
| 						EntityType::Spoiler,
 | |
| 						EntityType::StrikeOut,
 | |
| 						EntityType::Italic,
 | |
| 						EntityType::CustomEmoji,
 | |
| 					}),
 | |
| 				2);
 | |
| 			result.text = tr::lng_action_pinned_message(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_text,
 | |
| 				std::move(original), // Link 2.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.text = tr::lng_action_pinned_media(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_media,
 | |
| 				Ui::Text::Link(mediaText, 2), // Link 2.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 	} else if (pinned && pinned->msgId) {
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.links.push_back(pinned->lnk);
 | |
| 		result.text = tr::lng_action_pinned_media(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			lt_media,
 | |
| 			Ui::Text::Link(tr::lng_contacts_loading(tr::now), 2), // Link 2.
 | |
| 			Ui::Text::WithEntities);
 | |
| 	} else {
 | |
| 		result.links.push_back(fromLink());
 | |
| 		result.text = tr::lng_action_pinned_media(
 | |
| 			tr::now,
 | |
| 			lt_from,
 | |
| 			fromLinkText(), // Link 1.
 | |
| 			lt_media,
 | |
| 			{ .text = tr::lng_deleted_message(tr::now) },
 | |
| 			Ui::Text::WithEntities);
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| PreparedServiceText HistoryItem::prepareGameScoreText() {
 | |
| 	auto result = PreparedServiceText();
 | |
| 	auto gamescore = Get<HistoryServiceGameScore>();
 | |
| 
 | |
| 	auto computeGameTitle = [&]() -> TextWithEntities {
 | |
| 		if (gamescore && gamescore->msg) {
 | |
| 			if (const auto media = gamescore->msg->media()) {
 | |
| 				if (const auto game = media->game()) {
 | |
| 					const auto row = 0;
 | |
| 					const auto column = 0;
 | |
| 					result.links.push_back(
 | |
| 						std::make_shared<ReplyMarkupClickHandler>(
 | |
| 							&_history->owner(),
 | |
| 							row,
 | |
| 							column,
 | |
| 							gamescore->msg->fullId()));
 | |
| 					auto titleText = game->title;
 | |
| 					return Ui::Text::Link(titleText, QString());
 | |
| 				}
 | |
| 			}
 | |
| 			return tr::lng_deleted_message(tr::now, Ui::Text::WithEntities);
 | |
| 		} else if (gamescore && gamescore->msgId) {
 | |
| 			return tr::lng_contacts_loading(tr::now, Ui::Text::WithEntities);
 | |
| 		}
 | |
| 		return {};
 | |
| 	};
 | |
| 
 | |
| 	const auto scoreNumber = gamescore ? gamescore->score : 0;
 | |
| 	if (_from->isSelf()) {
 | |
| 		auto gameTitle = computeGameTitle();
 | |
| 		if (gameTitle.text.isEmpty()) {
 | |
| 			result.text = tr::lng_action_game_you_scored_no_game(
 | |
| 				tr::now,
 | |
| 				lt_count,
 | |
| 				scoreNumber,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.text = tr::lng_action_game_you_scored(
 | |
| 				tr::now,
 | |
| 				lt_count,
 | |
| 				scoreNumber,
 | |
| 				lt_game,
 | |
| 				gameTitle,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 	} else {
 | |
| 		result.links.push_back(fromLink());
 | |
| 		auto gameTitle = computeGameTitle();
 | |
| 		if (gameTitle.text.isEmpty()) {
 | |
| 			result.text = tr::lng_action_game_score_no_game(
 | |
| 				tr::now,
 | |
| 				lt_count,
 | |
| 				scoreNumber,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.text = tr::lng_action_game_score(
 | |
| 				tr::now,
 | |
| 				lt_count,
 | |
| 				scoreNumber,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_game,
 | |
| 				gameTitle,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| PreparedServiceText HistoryItem::preparePaymentSentText() {
 | |
| 	auto result = PreparedServiceText();
 | |
| 	const auto payment = Get<HistoryServicePayment>();
 | |
| 	Assert(payment != nullptr);
 | |
| 
 | |
| 	auto invoiceTitle = [&] {
 | |
| 		if (payment->msg) {
 | |
| 			if (const auto media = payment->msg->media()) {
 | |
| 				if (const auto invoice = media->invoice()) {
 | |
| 					return Ui::Text::Link(invoice->title, QString());
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return TextWithEntities();
 | |
| 	}();
 | |
| 
 | |
| 	if (invoiceTitle.text.isEmpty()) {
 | |
| 		if (payment->recurringUsed) {
 | |
| 			result.text = tr::lng_action_payment_used_recurring(
 | |
| 				tr::now,
 | |
| 				lt_amount,
 | |
| 				{ .text = payment->amount },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.text = (payment->recurringInit
 | |
| 				? tr::lng_action_payment_init_recurring
 | |
| 				: tr::lng_action_payment_done)(
 | |
| 					tr::now,
 | |
| 					lt_amount,
 | |
| 					{ .text = payment->amount },
 | |
| 					lt_user,
 | |
| 					{ .text = _history->peer->name() },
 | |
| 					Ui::Text::WithEntities);
 | |
| 		}
 | |
| 	} else {
 | |
| 		result.text = (payment->recurringInit
 | |
| 			? tr::lng_action_payment_init_recurring_for
 | |
| 			: tr::lng_action_payment_done_for)(
 | |
| 				tr::now,
 | |
| 				lt_amount,
 | |
| 				{ .text = payment->amount },
 | |
| 				lt_user,
 | |
| 				{ .text = _history->peer->name() },
 | |
| 				lt_invoice,
 | |
| 				invoiceTitle,
 | |
| 				Ui::Text::WithEntities);
 | |
| 		if (payment->msg) {
 | |
| 			result.links.push_back(payment->lnk);
 | |
| 		}
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| PreparedServiceText HistoryItem::prepareStoryMentionText() {
 | |
| 	auto result = PreparedServiceText();
 | |
| 	const auto peer = history()->peer;
 | |
| 	result.links.push_back(peer->createOpenLink());
 | |
| 	const auto phrase = (this->media() && this->media()->storyExpired(true))
 | |
| 		? (out()
 | |
| 			? tr::lng_action_story_mention_me_unavailable
 | |
| 			: tr::lng_action_story_mention_unavailable)
 | |
| 		: (out()
 | |
| 			? tr::lng_action_story_mention_me
 | |
| 			: tr::lng_action_story_mention);
 | |
| 	result.text = phrase(
 | |
| 		tr::now,
 | |
| 		lt_user,
 | |
| 		Ui::Text::Wrapped(
 | |
| 			Ui::Text::Bold(peer->shortName()),
 | |
| 			EntityType::CustomUrl,
 | |
| 			u"internal:index"_q + QChar(1)),
 | |
| 		Ui::Text::WithEntities);
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| PreparedServiceText HistoryItem::prepareCallScheduledText(
 | |
| 		TimeId scheduleDate) {
 | |
| 	const auto call = Get<HistoryServiceOngoingCall>();
 | |
| 	Assert(call != nullptr);
 | |
| 
 | |
| 	const auto scheduled = base::unixtime::parse(scheduleDate);
 | |
| 	const auto date = scheduled.date();
 | |
| 	const auto now = QDateTime::currentDateTime();
 | |
| 	const auto secsToDateAddDays = [&](int days) {
 | |
| 		return now.secsTo(QDateTime(date.addDays(days), QTime(0, 0)));
 | |
| 	};
 | |
| 	auto result = PreparedServiceText();
 | |
| 	const auto prepareWithDate = [&](const QString &date) {
 | |
| 		if (_history->peer->isBroadcast()) {
 | |
| 			result.text = tr::lng_action_group_call_scheduled_channel(
 | |
| 				tr::now,
 | |
| 				lt_date,
 | |
| 				{ .text = date },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		} else {
 | |
| 			result.links.push_back(fromLink());
 | |
| 			result.text = tr::lng_action_group_call_scheduled_group(
 | |
| 				tr::now,
 | |
| 				lt_from,
 | |
| 				fromLinkText(), // Link 1.
 | |
| 				lt_date,
 | |
| 				{ .text = date },
 | |
| 				Ui::Text::WithEntities);
 | |
| 		}
 | |
| 	};
 | |
| 	const auto time = QLocale().toString(
 | |
| 		scheduled.time(),
 | |
| 		QLocale::ShortFormat);
 | |
| 	const auto prepareGeneric = [&] {
 | |
| 		prepareWithDate(tr::lng_group_call_starts_date(
 | |
| 			tr::now,
 | |
| 			lt_date,
 | |
| 			langDayOfMonthFull(date),
 | |
| 			lt_time,
 | |
| 			time));
 | |
| 	};
 | |
| 	auto nextIn = TimeId(0);
 | |
| 	if (now.date().addDays(1) < scheduled.date()) {
 | |
| 		nextIn = secsToDateAddDays(-1);
 | |
| 		prepareGeneric();
 | |
| 	} else if (now.date().addDays(1) == scheduled.date()) {
 | |
| 		nextIn = secsToDateAddDays(0);
 | |
| 		prepareWithDate(
 | |
| 			tr::lng_group_call_starts_tomorrow(tr::now, lt_time, time));
 | |
| 	} else if (now.date() == scheduled.date()) {
 | |
| 		nextIn = secsToDateAddDays(1);
 | |
| 		prepareWithDate(
 | |
| 			tr::lng_group_call_starts_today(tr::now, lt_time, time));
 | |
| 	} else {
 | |
| 		prepareGeneric();
 | |
| 	}
 | |
| 	if (nextIn) {
 | |
| 		call->lifetime = base::timer_once(
 | |
| 			(nextIn + 2) * crl::time(1000)
 | |
| 		) | rpl::start_with_next([=] {
 | |
| 			updateServiceText(prepareCallScheduledText(scheduleDate));
 | |
| 		});
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| TextWithEntities HistoryItem::fromLinkText() const {
 | |
| 	return Ui::Text::Link(_from->name(), 1);
 | |
| }
 | |
| 
 | |
| ClickHandlerPtr HistoryItem::fromLink() const {
 | |
| 	return _from->createOpenLink();
 | |
| }
 | |
| 
 | |
| crl::time HistoryItem::getSelfDestructIn(crl::time now) {
 | |
| 	if (const auto selfdestruct = Get<HistoryServiceSelfDestruct>()) {
 | |
| 		const auto at = std::get_if<crl::time>(&selfdestruct->destructAt);
 | |
| 		if (at && (*at) > 0) {
 | |
| 			const auto destruct = *at;
 | |
| 			if (destruct <= now) {
 | |
| 				auto text = [&] {
 | |
| 					switch (selfdestruct->type) {
 | |
| 					case HistoryServiceSelfDestruct::Type::Photo:
 | |
| 						return tr::lng_ttl_photo_expired(tr::now);
 | |
| 					case HistoryServiceSelfDestruct::Type::Video:
 | |
| 						return tr::lng_ttl_video_expired(tr::now);
 | |
| 					}
 | |
| 					Unexpected("Type in HistoryServiceSelfDestruct::Type");
 | |
| 				};
 | |
| 				setServiceText({ TextWithEntities{ .text = text() } });
 | |
| 				return 0;
 | |
| 			}
 | |
| 			return destruct - now;
 | |
| 		}
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void HistoryItem::cacheOnlyEmojiAndSpaces(bool only) {
 | |
| 	_flags |= MessageFlag::OnlyEmojiAndSpacesSet;
 | |
| 	if (only) {
 | |
| 		_flags |= MessageFlag::OnlyEmojiAndSpaces;
 | |
| 	} else {
 | |
| 		_flags &= ~MessageFlag::OnlyEmojiAndSpaces;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::isOnlyEmojiAndSpaces() const {
 | |
| 	if (!(_flags & MessageFlag::OnlyEmojiAndSpacesSet)) {
 | |
| 		const_cast<HistoryItem*>(this)->cacheOnlyEmojiAndSpaces(
 | |
| 			!HasNotEmojiAndSpaces(_text.text));
 | |
| 	}
 | |
| 	return (_flags & MessageFlag::OnlyEmojiAndSpaces);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setupChatThemeChange() {
 | |
| 	if (const auto user = history()->peer->asUser()) {
 | |
| 		auto link = std::make_shared<LambdaClickHandler>([=](
 | |
| 				ClickContext context) {
 | |
| 			const auto my = context.other.value<ClickHandlerContext>();
 | |
| 			if (const auto controller = my.sessionWindow.get()) {
 | |
| 				controller->toggleChooseChatTheme(user);
 | |
| 			}
 | |
| 		});
 | |
| 
 | |
| 		UpdateComponents(HistoryServiceChatThemeChange::Bit());
 | |
| 		Get<HistoryServiceChatThemeChange>()->link = std::move(link);
 | |
| 	} else {
 | |
| 		RemoveComponents(HistoryServiceChatThemeChange::Bit());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setupTTLChange() {
 | |
| 	const auto peer = history()->peer;
 | |
| 	auto link = std::make_shared<LambdaClickHandler>([=](
 | |
| 			ClickContext context) {
 | |
| 		const auto my = context.other.value<ClickHandlerContext>();
 | |
| 		if (const auto controller = my.sessionWindow.get()) {
 | |
| 			const auto validator = TTLMenu::TTLValidator(
 | |
| 				controller->uiShow(),
 | |
| 				peer);
 | |
| 			if (validator.can()) {
 | |
| 				validator.showBox();
 | |
| 			}
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	UpdateComponents(HistoryServiceTTLChange::Bit());
 | |
| 	Get<HistoryServiceTTLChange>()->link = std::move(link);
 | |
| }
 | |
| 
 | |
| void HistoryItem::clearDependencyMessage() {
 | |
| 	if (const auto dependent = GetServiceDependentData()) {
 | |
| 		if (dependent->msg) {
 | |
| 			_history->owner().unregisterDependentMessage(
 | |
| 				this,
 | |
| 				dependent->msg);
 | |
| 			dependent->msg = nullptr;
 | |
| 			dependent->msgId = 0;
 | |
| 		}
 | |
| 	}
 | |
| }
 | 
