mirror of https://github.com/procxx/kepka.git
				
				
				
			
		
			
				
	
	
		
			528 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			528 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop version of Telegram messaging app, see https://telegram.org
 | |
| 
 | |
| Telegram Desktop is free software: you can redistribute it and/or modify
 | |
| it under the terms of the GNU General Public License as published by
 | |
| the Free Software Foundation, either version 3 of the License, or
 | |
| (at your option) any later version.
 | |
| 
 | |
| It is distributed in the hope that it will be useful,
 | |
| but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | |
| GNU General Public License for more details.
 | |
| 
 | |
| In addition, as a special exception, the copyright holders give permission
 | |
| to link the code of portions of this program with the OpenSSL library.
 | |
| 
 | |
| Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | |
| Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
 | |
| */
 | |
| #include "history/history_admin_log_item.h"
 | |
| 
 | |
| #include "app.h"
 | |
| #include "boxes/sticker_set_box.h"
 | |
| #include "history/history_admin_log_inner.h"
 | |
| #include "history/history_message.h"
 | |
| #include "history/history_service.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "messenger.h"
 | |
| 
 | |
| namespace AdminLog {
 | |
| namespace {
 | |
| 
 | |
| TextWithEntities PrepareText(const QString &value, const QString &emptyValue) {
 | |
| 	auto result = TextWithEntities{TextUtilities::Clean(value)};
 | |
| 	if (result.text.isEmpty()) {
 | |
| 		result.text = emptyValue;
 | |
| 		if (!emptyValue.isEmpty()) {
 | |
| 			result.entities.push_back(EntityInText(EntityInTextItalic, 0, emptyValue.size()));
 | |
| 		}
 | |
| 	} else {
 | |
| 		TextUtilities::ParseEntities(result,
 | |
| 		                             TextParseLinks | TextParseMentions | TextParseHashtags | TextParseBotCommands);
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| MTPMessage PrepareLogMessage(const MTPMessage &message, MsgId newId, qint32 newDate) {
 | |
| 	switch (message.type()) {
 | |
| 	case mtpc_messageEmpty: return MTP_messageEmpty(MTP_int(newId));
 | |
| 	case mtpc_messageService: {
 | |
| 		auto &data = message.c_messageService();
 | |
| 		auto flags =
 | |
| 		    data.vflags.v & ~(MTPDmessageService::Flag::f_out |
 | |
| 		                      MTPDmessageService::Flag::f_post /* | MTPDmessageService::Flag::f_reply_to_msg_id*/);
 | |
| 		return MTP_messageService(MTP_flags(flags), MTP_int(newId), data.vfrom_id, data.vto_id, data.vreply_to_msg_id,
 | |
| 		                          MTP_int(newDate), data.vaction);
 | |
| 	} break;
 | |
| 	case mtpc_message: {
 | |
| 		auto &data = message.c_message();
 | |
| 		auto flags = data.vflags.v & ~(MTPDmessage::Flag::f_out | MTPDmessage::Flag::f_post |
 | |
| 		                               MTPDmessage::Flag::f_reply_to_msg_id | MTPDmessage::Flag::f_edit_date);
 | |
| 		return MTP_message(MTP_flags(flags), MTP_int(newId), data.vfrom_id, data.vto_id, data.vfwd_from,
 | |
| 		                   data.vvia_bot_id, data.vreply_to_msg_id, MTP_int(newDate), data.vmessage, data.vmedia,
 | |
| 		                   data.vreply_markup, data.ventities, data.vviews, data.vedit_date, MTP_string(""));
 | |
| 	} break;
 | |
| 	}
 | |
| 	Unexpected("Type in PrepareLogMessage()");
 | |
| }
 | |
| 
 | |
| bool MediaCanHaveCaption(const MTPMessage &message) {
 | |
| 	if (message.type() != mtpc_message) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	auto &data = message.c_message();
 | |
| 	auto mediaType = data.has_media() ? data.vmedia.type() : mtpc_messageMediaEmpty;
 | |
| 	return (mediaType == mtpc_messageMediaDocument || mediaType == mtpc_messageMediaPhoto);
 | |
| }
 | |
| 
 | |
| TextWithEntities ExtractEditedText(const MTPMessage &message) {
 | |
| 	if (message.type() != mtpc_message) {
 | |
| 		return TextWithEntities();
 | |
| 	}
 | |
| 	auto &data = message.c_message();
 | |
| 	auto mediaType = data.has_media() ? data.vmedia.type() : mtpc_messageMediaEmpty;
 | |
| 	if (mediaType == mtpc_messageMediaDocument) {
 | |
| 		auto &document = data.vmedia.c_messageMediaDocument();
 | |
| 		return PrepareText(document.has_caption() ? qs(document.vcaption) : QString(), QString());
 | |
| 	} else if (mediaType == mtpc_messageMediaPhoto) {
 | |
| 		auto &photo = data.vmedia.c_messageMediaPhoto();
 | |
| 		return PrepareText(photo.has_caption() ? qs(photo.vcaption) : QString(), QString());
 | |
| 	}
 | |
| 	auto text = TextUtilities::Clean(qs(data.vmessage));
 | |
| 	auto entities = data.has_entities() ? TextUtilities::EntitiesFromMTP(data.ventities.v) : EntitiesInText();
 | |
| 	return {text, entities};
 | |
| }
 | |
| 
 | |
| PhotoData *GenerateChatPhoto(ChannelId channelId, quint64 logEntryId, MTPint date, const MTPDchatPhoto &photo) {
 | |
| 	// We try to make a unique photoId that will stay the same for each pair (channelId, logEntryId).
 | |
| 	static const auto RandomIdPart = rand_value<quint64>();
 | |
| 	auto mixinIdPart = (static_cast<quint64>(static_cast<quint32>(channelId)) << 32) ^ logEntryId;
 | |
| 	auto photoId = RandomIdPart ^ mixinIdPart;
 | |
| 
 | |
| 	auto photoSizes = QVector<MTPPhotoSize>();
 | |
| 	photoSizes.reserve(2);
 | |
| 	photoSizes.push_back(MTP_photoSize(MTP_string("a"), photo.vphoto_small, MTP_int(160), MTP_int(160), MTP_int(0)));
 | |
| 	photoSizes.push_back(MTP_photoSize(MTP_string("c"), photo.vphoto_big, MTP_int(640), MTP_int(640), MTP_int(0)));
 | |
| 	return App::feedPhoto(
 | |
| 	    MTP_photo(MTP_flags(0), MTP_long(photoId), MTP_long(0), date, MTP_vector<MTPPhotoSize>(photoSizes)));
 | |
| }
 | |
| 
 | |
| const auto CollectChanges = [](auto &phraseMap, auto plusFlags, auto minusFlags) {
 | |
| 	auto withPrefix = [&phraseMap](auto flags, QChar prefix) {
 | |
| 		auto result = QString();
 | |
| 		for (auto &phrase : phraseMap) {
 | |
| 			if (flags & phrase.first) {
 | |
| 				result.append('\n' + (prefix + lang(phrase.second)));
 | |
| 			}
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 	const auto kMinus = QChar(0x2212);
 | |
| 	return withPrefix(plusFlags & ~minusFlags, '+') + withPrefix(minusFlags & ~plusFlags, kMinus);
 | |
| };
 | |
| 
 | |
| auto GenerateAdminChangeText(not_null<ChannelData *> channel, const TextWithEntities &user,
 | |
|                              const MTPChannelAdminRights *newRights, const MTPChannelAdminRights *prevRights) {
 | |
| 	using Flag = MTPDchannelAdminRights::Flag;
 | |
| 	using Flags = MTPDchannelAdminRights::Flags;
 | |
| 
 | |
| 	Expects(!newRights || newRights->type() == mtpc_channelAdminRights);
 | |
| 	Expects(!prevRights || prevRights->type() == mtpc_channelAdminRights);
 | |
| 	auto newFlags = newRights ? newRights->c_channelAdminRights().vflags.v : MTPDchannelAdminRights::Flags(0);
 | |
| 	auto prevFlags = prevRights ? prevRights->c_channelAdminRights().vflags.v : MTPDchannelAdminRights::Flags(0);
 | |
| 	auto result = lng_admin_log_promoted__generic(lt_user, user);
 | |
| 
 | |
| 	auto inviteKey = Flag::f_invite_users | Flag::f_invite_link;
 | |
| 	auto useInviteLinkPhrase = channel->isMegagroup() && channel->anyoneCanAddMembers();
 | |
| 	auto invitePhrase = (useInviteLinkPhrase ? lng_admin_log_admin_invite_link : lng_admin_log_admin_invite_users);
 | |
| 	static auto phraseMap = std::map<Flags, LangKey>{
 | |
| 	    {Flag::f_change_info, lng_admin_log_admin_change_info},
 | |
| 	    {Flag::f_post_messages, lng_admin_log_admin_post_messages},
 | |
| 	    {Flag::f_edit_messages, lng_admin_log_admin_edit_messages},
 | |
| 	    {Flag::f_delete_messages, lng_admin_log_admin_delete_messages},
 | |
| 	    {Flag::f_ban_users, lng_admin_log_admin_ban_users},
 | |
| 	    {inviteKey, invitePhrase},
 | |
| 	    {Flag::f_pin_messages, lng_admin_log_admin_pin_messages},
 | |
| 	    {Flag::f_add_admins, lng_admin_log_admin_add_admins},
 | |
| 	};
 | |
| 	phraseMap[inviteKey] = invitePhrase;
 | |
| 
 | |
| 	if (!channel->isMegagroup()) {
 | |
| 		// Don't display "Ban users" changes in channels.
 | |
| 		newFlags &= ~Flag::f_ban_users;
 | |
| 		prevFlags &= ~Flag::f_ban_users;
 | |
| 	}
 | |
| 
 | |
| 	auto changes = CollectChanges(phraseMap, newFlags, prevFlags);
 | |
| 	if (!changes.isEmpty()) {
 | |
| 		result.text.append('\n' + changes);
 | |
| 	}
 | |
| 
 | |
| 	return result;
 | |
| };
 | |
| 
 | |
| auto GenerateBannedChangeText(const TextWithEntities &user, const MTPChannelBannedRights *newRights,
 | |
|                               const MTPChannelBannedRights *prevRights) {
 | |
| 	using Flag = MTPDchannelBannedRights::Flag;
 | |
| 	using Flags = MTPDchannelBannedRights::Flags;
 | |
| 
 | |
| 	Expects(!newRights || newRights->type() == mtpc_channelBannedRights);
 | |
| 	Expects(!prevRights || prevRights->type() == mtpc_channelBannedRights);
 | |
| 	auto newFlags = newRights ? newRights->c_channelBannedRights().vflags.v : MTPDchannelBannedRights::Flags(0);
 | |
| 	auto prevFlags = prevRights ? prevRights->c_channelBannedRights().vflags.v : MTPDchannelBannedRights::Flags(0);
 | |
| 	auto newUntil = newRights ? newRights->c_channelBannedRights().vuntil_date : MTP_int(0);
 | |
| 	auto indefinitely = ChannelData::IsRestrictedForever(newUntil.v);
 | |
| 	if (newFlags & Flag::f_view_messages) {
 | |
| 		return lng_admin_log_banned__generic(lt_user, user);
 | |
| 	}
 | |
| 	auto untilText = indefinitely ? lang(lng_admin_log_restricted_forever) :
 | |
| 	                                lng_admin_log_restricted_until(lt_date, langDateTime(::date(newUntil)));
 | |
| 	auto result = lng_admin_log_restricted__generic(lt_user, user, lt_until, TextWithEntities{untilText});
 | |
| 
 | |
| 	static auto phraseMap = std::map<Flags, LangKey>{
 | |
| 	    {Flag::f_view_messages, lng_admin_log_banned_view_messages},
 | |
| 	    {Flag::f_send_messages, lng_admin_log_banned_send_messages},
 | |
| 	    {Flag::f_send_media, lng_admin_log_banned_send_media},
 | |
| 	    {Flag::f_send_stickers | Flag::f_send_gifs | Flag::f_send_inline | Flag::f_send_games,
 | |
| 	     lng_admin_log_banned_send_stickers},
 | |
| 	    {Flag::f_embed_links, lng_admin_log_banned_embed_links},
 | |
| 	};
 | |
| 	auto changes = CollectChanges(phraseMap, prevFlags, newFlags);
 | |
| 	if (!changes.isEmpty()) {
 | |
| 		result.text.append('\n' + changes);
 | |
| 	}
 | |
| 
 | |
| 	return result;
 | |
| };
 | |
| 
 | |
| auto GenerateUserString(MTPint userId) {
 | |
| 	// User name in "User name (@username)" format with entities.
 | |
| 	auto user = App::user(userId.v);
 | |
| 	auto name = TextWithEntities{App::peerName(user)};
 | |
| 	name.entities.push_back(EntityInText(EntityInTextMentionName, 0, name.text.size(),
 | |
| 	                                     QString::number(user->id) + '.' + QString::number(user->access)));
 | |
| 	auto username = user->userName();
 | |
| 	if (username.isEmpty()) {
 | |
| 		return name;
 | |
| 	}
 | |
| 	auto mention = TextWithEntities{'@' + username};
 | |
| 	mention.entities.push_back(EntityInText(EntityInTextMention, 0, mention.text.size()));
 | |
| 	return lng_admin_log_user_with_username__generic(lt_name, name, lt_mention, mention);
 | |
| }
 | |
| 
 | |
| auto GenerateParticipantChangeTextInner(not_null<ChannelData *> channel, const MTPChannelParticipant &participant,
 | |
|                                         const MTPChannelParticipant *oldParticipant) {
 | |
| 	auto oldType = oldParticipant ? oldParticipant->type() : 0;
 | |
| 
 | |
| 	auto resultForParticipant = [channel, oldParticipant, oldType](auto &&data) {
 | |
| 		auto user = GenerateUserString(data.vuser_id);
 | |
| 		if (oldType == mtpc_channelParticipantAdmin) {
 | |
| 			return GenerateAdminChangeText(channel, user, nullptr,
 | |
| 			                               &oldParticipant->c_channelParticipantAdmin().vadmin_rights);
 | |
| 		} else if (oldType == mtpc_channelParticipantBanned) {
 | |
| 			return GenerateBannedChangeText(user, nullptr,
 | |
| 			                                &oldParticipant->c_channelParticipantBanned().vbanned_rights);
 | |
| 		}
 | |
| 		return lng_admin_log_invited__generic(lt_user, user);
 | |
| 	};
 | |
| 
 | |
| 	switch (participant.type()) {
 | |
| 	case mtpc_channelParticipantCreator: {
 | |
| 		// No valid string here :(
 | |
| 		auto &data = participant.c_channelParticipantCreator();
 | |
| 		return lng_admin_log_invited__generic(lt_user, GenerateUserString(data.vuser_id));
 | |
| 	} break;
 | |
| 
 | |
| 	case mtpc_channelParticipant: return resultForParticipant(participant.c_channelParticipant());
 | |
| 	case mtpc_channelParticipantSelf: return resultForParticipant(participant.c_channelParticipantSelf());
 | |
| 
 | |
| 	case mtpc_channelParticipantAdmin: {
 | |
| 		auto &data = participant.c_channelParticipantAdmin();
 | |
| 		auto user = GenerateUserString(data.vuser_id);
 | |
| 		return GenerateAdminChangeText(channel, user, &data.vadmin_rights,
 | |
| 		                               (oldType == mtpc_channelParticipantAdmin) ?
 | |
| 		                                   &oldParticipant->c_channelParticipantAdmin().vadmin_rights :
 | |
| 		                                   nullptr);
 | |
| 	} break;
 | |
| 
 | |
| 	case mtpc_channelParticipantBanned: {
 | |
| 		auto &data = participant.c_channelParticipantBanned();
 | |
| 		auto user = GenerateUserString(data.vuser_id);
 | |
| 		return GenerateBannedChangeText(user, &data.vbanned_rights,
 | |
| 		                                (oldType == mtpc_channelParticipantBanned) ?
 | |
| 		                                    &oldParticipant->c_channelParticipantBanned().vbanned_rights :
 | |
| 		                                    nullptr);
 | |
| 	} break;
 | |
| 	}
 | |
| 
 | |
| 	Unexpected("Participant type in GenerateParticipantChangeTextInner()");
 | |
| }
 | |
| 
 | |
| TextWithEntities GenerateParticipantChangeText(not_null<ChannelData *> channel,
 | |
|                                                const MTPChannelParticipant &participant,
 | |
|                                                const MTPChannelParticipant *oldParticipant = nullptr) {
 | |
| 	auto result = GenerateParticipantChangeTextInner(channel, participant, oldParticipant);
 | |
| 	result.entities.push_front(EntityInText(EntityInTextItalic, 0, result.text.size()));
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| void GenerateItems(not_null<History *> history, LocalIdManager &idManager, const MTPDchannelAdminLogEvent &event,
 | |
|                    Fn<void(HistoryItemOwned item)> callback) {
 | |
| 	Expects(history->peer->isChannel());
 | |
| 
 | |
| 	auto id = event.vid.v;
 | |
| 	auto from = App::user(event.vuser_id.v);
 | |
| 	auto channel = history->peer->asChannel();
 | |
| 	auto &action = event.vaction;
 | |
| 	auto date = event.vdate;
 | |
| 	auto addPart = [&callback](not_null<HistoryItem *> item) { return callback(HistoryItemOwned(item)); };
 | |
| 
 | |
| 	using Flag = MTPDmessage::Flag;
 | |
| 	auto fromName = App::peerName(from);
 | |
| 	auto fromLink = from->createOpenLink();
 | |
| 	auto fromLinkText = textcmdLink(1, fromName);
 | |
| 
 | |
| 	auto addSimpleServiceMessage = [&](const QString &text, PhotoData *photo = nullptr) {
 | |
| 		auto message = HistoryService::PreparedText{text};
 | |
| 		message.links.push_back(fromLink);
 | |
| 		addPart(
 | |
| 		    HistoryService::create(history, idManager.next(), ::date(date), message, 0, peerToUser(from->id), photo));
 | |
| 	};
 | |
| 
 | |
| 	auto createChangeTitle = [&](const MTPDchannelAdminLogEventActionChangeTitle &action) {
 | |
| 		auto text = (channel->isMegagroup() ? lng_action_changed_title : lng_admin_log_changed_title_channel)(
 | |
| 		    lt_from, fromLinkText, lt_title, qs(action.vnew_value));
 | |
| 		addSimpleServiceMessage(text);
 | |
| 	};
 | |
| 
 | |
| 	auto createChangeAbout = [&](const MTPDchannelAdminLogEventActionChangeAbout &action) {
 | |
| 		auto newValue = qs(action.vnew_value);
 | |
| 		auto oldValue = qs(action.vprev_value);
 | |
| 		auto text = (channel->isMegagroup() ?
 | |
| 		                 (newValue.isEmpty() ? lng_admin_log_removed_description_group :
 | |
| 		                                       lng_admin_log_changed_description_group) :
 | |
| 		                 (newValue.isEmpty() ? lng_admin_log_removed_description_channel :
 | |
| 		                                       lng_admin_log_changed_description_channel))(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 
 | |
| 		auto bodyFlags = Flag::f_entities | Flag::f_from_id;
 | |
| 		auto bodyReplyTo = 0;
 | |
| 		auto bodyViaBotId = 0;
 | |
| 		auto newDescription = PrepareText(newValue, QString());
 | |
| 		auto body = HistoryMessage::create(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId,
 | |
| 		                                   ::date(date), peerToUser(from->id), QString(), newDescription);
 | |
| 		if (!oldValue.isEmpty()) {
 | |
| 			auto oldDescription = PrepareText(oldValue, QString());
 | |
| 			body->addLogEntryOriginal(id, lang(lng_admin_log_previous_description), oldDescription);
 | |
| 		}
 | |
| 		addPart(body);
 | |
| 	};
 | |
| 
 | |
| 	auto createChangeUsername = [&](const MTPDchannelAdminLogEventActionChangeUsername &action) {
 | |
| 		auto newValue = qs(action.vnew_value);
 | |
| 		auto oldValue = qs(action.vprev_value);
 | |
| 		auto text = (channel->isMegagroup() ?
 | |
| 		                 (newValue.isEmpty() ? lng_admin_log_removed_link_group : lng_admin_log_changed_link_group) :
 | |
| 		                 (newValue.isEmpty() ? lng_admin_log_removed_link_channel :
 | |
| 		                                       lng_admin_log_changed_link_channel))(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 
 | |
| 		auto bodyFlags = Flag::f_entities | Flag::f_from_id;
 | |
| 		auto bodyReplyTo = 0;
 | |
| 		auto bodyViaBotId = 0;
 | |
| 		auto newLink = newValue.isEmpty() ?
 | |
| 		                   TextWithEntities() :
 | |
| 		                   PrepareText(Messenger::Instance().createInternalLinkFull(newValue), QString());
 | |
| 		auto body = HistoryMessage::create(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId,
 | |
| 		                                   ::date(date), peerToUser(from->id), QString(), newLink);
 | |
| 		if (!oldValue.isEmpty()) {
 | |
| 			auto oldLink = PrepareText(Messenger::Instance().createInternalLinkFull(oldValue), QString());
 | |
| 			body->addLogEntryOriginal(id, lang(lng_admin_log_previous_link), oldLink);
 | |
| 		}
 | |
| 		addPart(body);
 | |
| 	};
 | |
| 
 | |
| 	auto createChangePhoto = [&](const MTPDchannelAdminLogEventActionChangePhoto &action) {
 | |
| 		switch (action.vnew_photo.type()) {
 | |
| 		case mtpc_chatPhoto: {
 | |
| 			auto photo = GenerateChatPhoto(channel->bareId(), id, date, action.vnew_photo.c_chatPhoto());
 | |
| 			auto text = (channel->isMegagroup() ? lng_admin_log_changed_photo_group :
 | |
| 			                                      lng_admin_log_changed_photo_channel)(lt_from, fromLinkText);
 | |
| 			addSimpleServiceMessage(text, photo);
 | |
| 		} break;
 | |
| 		case mtpc_chatPhotoEmpty: {
 | |
| 			auto text = (channel->isMegagroup() ? lng_admin_log_removed_photo_group :
 | |
| 			                                      lng_admin_log_removed_photo_channel)(lt_from, fromLinkText);
 | |
| 			addSimpleServiceMessage(text);
 | |
| 		} break;
 | |
| 		default: Unexpected("ChatPhoto type in createChangePhoto()");
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	auto createToggleInvites = [&](const MTPDchannelAdminLogEventActionToggleInvites &action) {
 | |
| 		auto enabled = (action.vnew_value.type() == mtpc_boolTrue);
 | |
| 		auto text = (enabled ? lng_admin_log_invites_enabled : lng_admin_log_invites_disabled)(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 	};
 | |
| 
 | |
| 	auto createToggleSignatures = [&](const MTPDchannelAdminLogEventActionToggleSignatures &action) {
 | |
| 		auto enabled = (action.vnew_value.type() == mtpc_boolTrue);
 | |
| 		auto text =
 | |
| 		    (enabled ? lng_admin_log_signatures_enabled : lng_admin_log_signatures_disabled)(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 	};
 | |
| 
 | |
| 	auto createUpdatePinned = [&](const MTPDchannelAdminLogEventActionUpdatePinned &action) {
 | |
| 		if (action.vmessage.type() == mtpc_messageEmpty) {
 | |
| 			auto text = lng_admin_log_unpinned_message(lt_from, fromLinkText);
 | |
| 			addSimpleServiceMessage(text);
 | |
| 		} else {
 | |
| 			auto text = lng_admin_log_pinned_message(lt_from, fromLinkText);
 | |
| 			addSimpleServiceMessage(text);
 | |
| 
 | |
| 			auto applyServiceAction = false;
 | |
| 			auto detachExistingItem = false;
 | |
| 			addPart(history->createItem(PrepareLogMessage(action.vmessage, idManager.next(), date.v),
 | |
| 			                            applyServiceAction, detachExistingItem));
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	auto createEditMessage = [&](const MTPDchannelAdminLogEventActionEditMessage &action) {
 | |
| 		auto newValue = ExtractEditedText(action.vnew_message);
 | |
| 		auto canHaveCaption = MediaCanHaveCaption(action.vnew_message);
 | |
| 		auto text =
 | |
| 		    (canHaveCaption ? (newValue.text.isEmpty() ? lng_admin_log_removed_caption : lng_admin_log_edited_caption) :
 | |
| 		                      lng_admin_log_edited_message)(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 
 | |
| 		auto oldValue = ExtractEditedText(action.vprev_message);
 | |
| 		auto applyServiceAction = false;
 | |
| 		auto detachExistingItem = false;
 | |
| 		auto body = history->createItem(PrepareLogMessage(action.vnew_message, idManager.next(), date.v),
 | |
| 		                                applyServiceAction, detachExistingItem);
 | |
| 		if (oldValue.text.isEmpty()) {
 | |
| 			oldValue = PrepareText(QString(), lang(lng_admin_log_empty_text));
 | |
| 		}
 | |
| 		body->addLogEntryOriginal(
 | |
| 		    id, lang(canHaveCaption ? lng_admin_log_previous_caption : lng_admin_log_previous_message), oldValue);
 | |
| 		addPart(body);
 | |
| 	};
 | |
| 
 | |
| 	auto createDeleteMessage = [&](const MTPDchannelAdminLogEventActionDeleteMessage &action) {
 | |
| 		auto text = lng_admin_log_deleted_message(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 
 | |
| 		auto applyServiceAction = false;
 | |
| 		auto detachExistingItem = false;
 | |
| 		addPart(history->createItem(PrepareLogMessage(action.vmessage, idManager.next(), date.v), applyServiceAction,
 | |
| 		                            detachExistingItem));
 | |
| 	};
 | |
| 
 | |
| 	auto createParticipantJoin = [&]() {
 | |
| 		auto text = (channel->isMegagroup() ? lng_admin_log_participant_joined :
 | |
| 		                                      lng_admin_log_participant_joined_channel)(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 	};
 | |
| 
 | |
| 	auto createParticipantLeave = [&]() {
 | |
| 		auto text = (channel->isMegagroup() ? lng_admin_log_participant_left :
 | |
| 		                                      lng_admin_log_participant_left_channel)(lt_from, fromLinkText);
 | |
| 		addSimpleServiceMessage(text);
 | |
| 	};
 | |
| 
 | |
| 	auto createParticipantInvite = [&](const MTPDchannelAdminLogEventActionParticipantInvite &action) {
 | |
| 		auto bodyFlags = Flag::f_entities | Flag::f_from_id;
 | |
| 		auto bodyReplyTo = 0;
 | |
| 		auto bodyViaBotId = 0;
 | |
| 		auto bodyText = GenerateParticipantChangeText(channel, action.vparticipant);
 | |
| 		addPart(HistoryMessage::create(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date),
 | |
| 		                               peerToUser(from->id), QString(), bodyText));
 | |
| 	};
 | |
| 
 | |
| 	auto createParticipantToggleBan = [&](const MTPDchannelAdminLogEventActionParticipantToggleBan &action) {
 | |
| 		auto bodyFlags = Flag::f_entities | Flag::f_from_id;
 | |
| 		auto bodyReplyTo = 0;
 | |
| 		auto bodyViaBotId = 0;
 | |
| 		auto bodyText = GenerateParticipantChangeText(channel, action.vnew_participant, &action.vprev_participant);
 | |
| 		addPart(HistoryMessage::create(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date),
 | |
| 		                               peerToUser(from->id), QString(), bodyText));
 | |
| 	};
 | |
| 
 | |
| 	auto createParticipantToggleAdmin = [&](const MTPDchannelAdminLogEventActionParticipantToggleAdmin &action) {
 | |
| 		auto bodyFlags = Flag::f_entities | Flag::f_from_id;
 | |
| 		auto bodyReplyTo = 0;
 | |
| 		auto bodyViaBotId = 0;
 | |
| 		auto bodyText = GenerateParticipantChangeText(channel, action.vnew_participant, &action.vprev_participant);
 | |
| 		addPart(HistoryMessage::create(history, idManager.next(), bodyFlags, bodyReplyTo, bodyViaBotId, ::date(date),
 | |
| 		                               peerToUser(from->id), QString(), bodyText));
 | |
| 	};
 | |
| 
 | |
| 	auto createChangeStickerSet = [&](const MTPDchannelAdminLogEventActionChangeStickerSet &action) {
 | |
| 		auto set = action.vnew_stickerset;
 | |
| 		auto removed = (set.type() == mtpc_inputStickerSetEmpty);
 | |
| 		if (removed) {
 | |
| 			auto text = lng_admin_log_removed_stickers_group(lt_from, fromLinkText);
 | |
| 			addSimpleServiceMessage(text);
 | |
| 		} else {
 | |
| 			auto text = lng_admin_log_changed_stickers_group(lt_from, fromLinkText, lt_sticker_set,
 | |
| 			                                                 textcmdLink(2, lang(lng_admin_log_changed_stickers_set)));
 | |
| 			auto setLink = MakeShared<LambdaClickHandler>([set] { Ui::show(Box<StickerSetBox>(set)); });
 | |
| 			auto message = HistoryService::PreparedText{text};
 | |
| 			message.links.push_back(fromLink);
 | |
| 			message.links.push_back(setLink);
 | |
| 			addPart(HistoryService::create(history, idManager.next(), ::date(date), message, 0, peerToUser(from->id)));
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	switch (action.type()) {
 | |
| 	case mtpc_channelAdminLogEventActionChangeTitle:
 | |
| 		createChangeTitle(action.c_channelAdminLogEventActionChangeTitle());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionChangeAbout:
 | |
| 		createChangeAbout(action.c_channelAdminLogEventActionChangeAbout());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionChangeUsername:
 | |
| 		createChangeUsername(action.c_channelAdminLogEventActionChangeUsername());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionChangePhoto:
 | |
| 		createChangePhoto(action.c_channelAdminLogEventActionChangePhoto());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionToggleInvites:
 | |
| 		createToggleInvites(action.c_channelAdminLogEventActionToggleInvites());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionToggleSignatures:
 | |
| 		createToggleSignatures(action.c_channelAdminLogEventActionToggleSignatures());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionUpdatePinned:
 | |
| 		createUpdatePinned(action.c_channelAdminLogEventActionUpdatePinned());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionEditMessage:
 | |
| 		createEditMessage(action.c_channelAdminLogEventActionEditMessage());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionDeleteMessage:
 | |
| 		createDeleteMessage(action.c_channelAdminLogEventActionDeleteMessage());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionParticipantJoin: createParticipantJoin(); break;
 | |
| 	case mtpc_channelAdminLogEventActionParticipantLeave: createParticipantLeave(); break;
 | |
| 	case mtpc_channelAdminLogEventActionParticipantInvite:
 | |
| 		createParticipantInvite(action.c_channelAdminLogEventActionParticipantInvite());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionParticipantToggleBan:
 | |
| 		createParticipantToggleBan(action.c_channelAdminLogEventActionParticipantToggleBan());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionParticipantToggleAdmin:
 | |
| 		createParticipantToggleAdmin(action.c_channelAdminLogEventActionParticipantToggleAdmin());
 | |
| 		break;
 | |
| 	case mtpc_channelAdminLogEventActionChangeStickerSet:
 | |
| 		createChangeStickerSet(action.c_channelAdminLogEventActionChangeStickerSet());
 | |
| 		break;
 | |
| 	default: Unexpected("channelAdminLogEventAction type in AdminLog::Item::Item()");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| } // namespace AdminLog
 |