mirror of
				https://github.com/telegramdesktop/tdesktop
				synced 2025-10-25 14:58:42 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			660 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			660 lines
		
	
	
		
			19 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 "api/api_sending.h"
 | |
| 
 | |
| #include "api/api_text_entities.h"
 | |
| #include "base/random.h"
 | |
| #include "base/unixtime.h"
 | |
| #include "data/business/data_shortcut_messages.h"
 | |
| #include "data/data_document.h"
 | |
| #include "data/data_photo.h"
 | |
| #include "data/data_channel.h" // ChannelData::addsSignature.
 | |
| #include "data/data_user.h" // UserData::name
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_file_origin.h"
 | |
| #include "data/data_histories.h"
 | |
| #include "data/data_changes.h"
 | |
| #include "data/stickers/data_stickers.h"
 | |
| #include "history/history.h"
 | |
| #include "history/history_item.h"
 | |
| #include "history/history_item_helpers.h" // NewMessageFlags.
 | |
| #include "chat_helpers/message_field.h" // ConvertTextTagsToEntities.
 | |
| #include "chat_helpers/stickers_dice_pack.h" // DicePacks::kDiceString.
 | |
| #include "ui/text/text_entity.h" // TextWithEntities.
 | |
| #include "ui/item_text_options.h" // Ui::ItemTextOptions.
 | |
| #include "main/main_session.h"
 | |
| #include "main/main_app_config.h"
 | |
| #include "storage/localimageloader.h"
 | |
| #include "storage/file_upload.h"
 | |
| #include "mainwidget.h"
 | |
| #include "apiwrap.h"
 | |
| 
 | |
| namespace Api {
 | |
| namespace {
 | |
| 
 | |
| void InnerFillMessagePostFlags(
 | |
| 		const SendOptions &options,
 | |
| 		not_null<PeerData*> peer,
 | |
| 		MessageFlags &flags) {
 | |
| 	const auto anonymousPost = peer->amAnonymous();
 | |
| 	if (ShouldSendSilent(peer, options)) {
 | |
| 		flags |= MessageFlag::Silent;
 | |
| 	}
 | |
| 	if (!anonymousPost || options.sendAs) {
 | |
| 		flags |= MessageFlag::HasFromId;
 | |
| 		return;
 | |
| 	} else if (peer->asMegagroup()) {
 | |
| 		return;
 | |
| 	}
 | |
| 	flags |= MessageFlag::Post;
 | |
| 	// Don't display views and author of a new post when it's scheduled.
 | |
| 	if (options.scheduled) {
 | |
| 		return;
 | |
| 	}
 | |
| 	flags |= MessageFlag::HasViews;
 | |
| 	if (peer->asChannel()->addsSignature()) {
 | |
| 		flags |= MessageFlag::HasPostAuthor;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void SendSimpleMedia(SendAction action, MTPInputMedia inputMedia) {
 | |
| 	const auto history = action.history;
 | |
| 	const auto peer = history->peer;
 | |
| 	const auto session = &history->session();
 | |
| 	const auto api = &session->api();
 | |
| 
 | |
| 	action.clearDraft = false;
 | |
| 	action.generateLocal = false;
 | |
| 	api->sendAction(action);
 | |
| 
 | |
| 	const auto randomId = base::RandomValue<uint64>();
 | |
| 
 | |
| 	auto flags = NewMessageFlags(peer);
 | |
| 	auto sendFlags = MTPmessages_SendMedia::Flags(0);
 | |
| 	if (action.replyTo) {
 | |
| 		flags |= MessageFlag::HasReplyInfo;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_reply_to;
 | |
| 	}
 | |
| 	const auto anonymousPost = peer->amAnonymous();
 | |
| 	const auto silentPost = ShouldSendSilent(peer, action.options);
 | |
| 	InnerFillMessagePostFlags(action.options, peer, flags);
 | |
| 	if (silentPost) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_silent;
 | |
| 	}
 | |
| 	const auto sendAs = action.options.sendAs;
 | |
| 	const auto messageFromId = sendAs
 | |
| 		? sendAs->id
 | |
| 		: anonymousPost
 | |
| 		? 0
 | |
| 		: session->userPeerId();
 | |
| 	if (sendAs) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_send_as;
 | |
| 	}
 | |
| 	const auto messagePostAuthor = peer->isBroadcast()
 | |
| 		? session->user()->name()
 | |
| 		: QString();
 | |
| 
 | |
| 	if (action.options.scheduled) {
 | |
| 		flags |= MessageFlag::IsOrWasScheduled;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_schedule_date;
 | |
| 	}
 | |
| 	if (action.options.shortcutId) {
 | |
| 		flags |= MessageFlag::ShortcutMessage;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_quick_reply_shortcut;
 | |
| 	}
 | |
| 	if (action.options.effectId) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_effect;
 | |
| 	}
 | |
| 	if (action.options.invertCaption) {
 | |
| 		flags |= MessageFlag::InvertMedia;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_invert_media;
 | |
| 	}
 | |
| 
 | |
| 	auto &histories = history->owner().histories();
 | |
| 	histories.sendPreparedMessage(
 | |
| 		history,
 | |
| 		action.replyTo,
 | |
| 		randomId,
 | |
| 		Data::Histories::PrepareMessage<MTPmessages_SendMedia>(
 | |
| 			MTP_flags(sendFlags),
 | |
| 			peer->input,
 | |
| 			Data::Histories::ReplyToPlaceholder(),
 | |
| 			std::move(inputMedia),
 | |
| 			MTPstring(),
 | |
| 			MTP_long(randomId),
 | |
| 			MTPReplyMarkup(),
 | |
| 			MTPvector<MTPMessageEntity>(),
 | |
| 			MTP_int(action.options.scheduled),
 | |
| 			(sendAs ? sendAs->input : MTP_inputPeerEmpty()),
 | |
| 			Data::ShortcutIdToMTP(session, action.options.shortcutId),
 | |
| 			MTP_long(action.options.effectId)
 | |
| 		), [=](const MTPUpdates &result, const MTP::Response &response) {
 | |
| 	}, [=](const MTP::Error &error, const MTP::Response &response) {
 | |
| 		api->sendMessageFail(error, peer, randomId);
 | |
| 	});
 | |
| 
 | |
| 	api->finishForwarding(action);
 | |
| }
 | |
| 
 | |
| template <typename MediaData>
 | |
| void SendExistingMedia(
 | |
| 		MessageToSend &&message,
 | |
| 		not_null<MediaData*> media,
 | |
| 		Fn<MTPInputMedia()> inputMedia,
 | |
| 		Data::FileOrigin origin,
 | |
| 		std::optional<MsgId> localMessageId) {
 | |
| 	const auto history = message.action.history;
 | |
| 	const auto peer = history->peer;
 | |
| 	const auto session = &history->session();
 | |
| 	const auto api = &session->api();
 | |
| 
 | |
| 	message.action.clearDraft = false;
 | |
| 	message.action.generateLocal = true;
 | |
| 	api->sendAction(message.action);
 | |
| 
 | |
| 	const auto newId = FullMsgId(
 | |
| 		peer->id,
 | |
| 		localMessageId
 | |
| 			? (*localMessageId)
 | |
| 			: session->data().nextLocalMessageId());
 | |
| 	const auto randomId = base::RandomValue<uint64>();
 | |
| 	const auto &action = message.action;
 | |
| 
 | |
| 	auto flags = NewMessageFlags(peer);
 | |
| 	auto sendFlags = MTPmessages_SendMedia::Flags(0);
 | |
| 	if (action.replyTo) {
 | |
| 		flags |= MessageFlag::HasReplyInfo;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_reply_to;
 | |
| 	}
 | |
| 	const auto anonymousPost = peer->amAnonymous();
 | |
| 	const auto silentPost = ShouldSendSilent(peer, action.options);
 | |
| 	InnerFillMessagePostFlags(action.options, peer, flags);
 | |
| 	if (silentPost) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_silent;
 | |
| 	}
 | |
| 	const auto sendAs = action.options.sendAs;
 | |
| 	const auto messageFromId = sendAs
 | |
| 		? sendAs->id
 | |
| 		: anonymousPost
 | |
| 		? 0
 | |
| 		: session->userPeerId();
 | |
| 	if (sendAs) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_send_as;
 | |
| 	}
 | |
| 	const auto messagePostAuthor = peer->isBroadcast()
 | |
| 		? session->user()->name()
 | |
| 		: QString();
 | |
| 
 | |
| 	auto caption = TextWithEntities{
 | |
| 		message.textWithTags.text,
 | |
| 		TextUtilities::ConvertTextTagsToEntities(message.textWithTags.tags)
 | |
| 	};
 | |
| 	TextUtilities::Trim(caption);
 | |
| 	auto sentEntities = EntitiesToMTP(
 | |
| 		session,
 | |
| 		caption.entities,
 | |
| 		ConvertOption::SkipLocal);
 | |
| 	if (!sentEntities.v.isEmpty()) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_entities;
 | |
| 	}
 | |
| 	const auto captionText = caption.text;
 | |
| 
 | |
| 	if (action.options.scheduled) {
 | |
| 		flags |= MessageFlag::IsOrWasScheduled;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_schedule_date;
 | |
| 	}
 | |
| 	if (action.options.shortcutId) {
 | |
| 		flags |= MessageFlag::ShortcutMessage;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_quick_reply_shortcut;
 | |
| 	}
 | |
| 	if (action.options.effectId) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_effect;
 | |
| 	}
 | |
| 	if (action.options.invertCaption) {
 | |
| 		flags |= MessageFlag::InvertMedia;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_invert_media;
 | |
| 	}
 | |
| 
 | |
| 	session->data().registerMessageRandomId(randomId, newId);
 | |
| 
 | |
| 	history->addNewLocalMessage({
 | |
| 		.id = newId.msg,
 | |
| 		.flags = flags,
 | |
| 		.from = messageFromId,
 | |
| 		.replyTo = action.replyTo,
 | |
| 		.date = HistoryItem::NewMessageDate(action.options),
 | |
| 		.shortcutId = action.options.shortcutId,
 | |
| 		.postAuthor = messagePostAuthor,
 | |
| 		.effectId = action.options.effectId,
 | |
| 	}, media, caption);
 | |
| 
 | |
| 	const auto performRequest = [=](const auto &repeatRequest) -> void {
 | |
| 		auto &histories = history->owner().histories();
 | |
| 		const auto session = &history->session();
 | |
| 		const auto usedFileReference = media->fileReference();
 | |
| 		histories.sendPreparedMessage(
 | |
| 			history,
 | |
| 			action.replyTo,
 | |
| 			randomId,
 | |
| 			Data::Histories::PrepareMessage<MTPmessages_SendMedia>(
 | |
| 				MTP_flags(sendFlags),
 | |
| 				peer->input,
 | |
| 				Data::Histories::ReplyToPlaceholder(),
 | |
| 				inputMedia(),
 | |
| 				MTP_string(captionText),
 | |
| 				MTP_long(randomId),
 | |
| 				MTPReplyMarkup(),
 | |
| 				sentEntities,
 | |
| 				MTP_int(action.options.scheduled),
 | |
| 				(sendAs ? sendAs->input : MTP_inputPeerEmpty()),
 | |
| 				Data::ShortcutIdToMTP(session, action.options.shortcutId),
 | |
| 				MTP_long(action.options.effectId)
 | |
| 			), [=](const MTPUpdates &result, const MTP::Response &response) {
 | |
| 		}, [=](const MTP::Error &error, const MTP::Response &response) {
 | |
| 			if (error.code() == 400
 | |
| 				&& error.type().startsWith(u"FILE_REFERENCE_"_q)) {
 | |
| 				api->refreshFileReference(origin, [=](const auto &result) {
 | |
| 					if (media->fileReference() != usedFileReference) {
 | |
| 						repeatRequest(repeatRequest);
 | |
| 					} else {
 | |
| 						api->sendMessageFail(error, peer, randomId, newId);
 | |
| 					}
 | |
| 				});
 | |
| 			} else {
 | |
| 				api->sendMessageFail(error, peer, randomId, newId);
 | |
| 			}
 | |
| 		});
 | |
| 	};
 | |
| 	performRequest(performRequest);
 | |
| 
 | |
| 	api->finishForwarding(action);
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| void SendExistingDocument(
 | |
| 		MessageToSend &&message,
 | |
| 		not_null<DocumentData*> document,
 | |
| 		std::optional<MsgId> localMessageId) {
 | |
| 	const auto inputMedia = [=] {
 | |
| 		return MTP_inputMediaDocument(
 | |
| 			MTP_flags(0),
 | |
| 			document->mtpInput(),
 | |
| 			MTPint(), // ttl_seconds
 | |
| 			MTPstring()); // query
 | |
| 	};
 | |
| 	SendExistingMedia(
 | |
| 		std::move(message),
 | |
| 		document,
 | |
| 		inputMedia,
 | |
| 		document->stickerOrGifOrigin(),
 | |
| 		std::move(localMessageId));
 | |
| 
 | |
| 	if (document->sticker()) {
 | |
| 		document->owner().stickers().incrementSticker(document);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void SendExistingPhoto(
 | |
| 		MessageToSend &&message,
 | |
| 		not_null<PhotoData*> photo,
 | |
| 		std::optional<MsgId> localMessageId) {
 | |
| 	const auto inputMedia = [=] {
 | |
| 		return MTP_inputMediaPhoto(
 | |
| 			MTP_flags(0),
 | |
| 			photo->mtpInput(),
 | |
| 			MTPint());
 | |
| 	};
 | |
| 	SendExistingMedia(
 | |
| 		std::move(message),
 | |
| 		photo,
 | |
| 		inputMedia,
 | |
| 		Data::FileOrigin(),
 | |
| 		std::move(localMessageId));
 | |
| }
 | |
| 
 | |
| bool SendDice(MessageToSend &message) {
 | |
| 	const auto full = QStringView(message.textWithTags.text).trimmed();
 | |
| 	auto length = 0;
 | |
| 	if (!Ui::Emoji::Find(full.data(), full.data() + full.size(), &length)
 | |
| 		|| length != full.size()
 | |
| 		|| !message.textWithTags.tags.isEmpty()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	auto &config = message.action.history->session().appConfig();
 | |
| 	static const auto hardcoded = std::vector<QString>{
 | |
| 		Stickers::DicePacks::kDiceString,
 | |
| 		Stickers::DicePacks::kDartString,
 | |
| 		Stickers::DicePacks::kSlotString,
 | |
| 		Stickers::DicePacks::kFballString,
 | |
| 		Stickers::DicePacks::kFballString + QChar(0xFE0F),
 | |
| 		Stickers::DicePacks::kBballString,
 | |
| 	};
 | |
| 	const auto list = config.get<std::vector<QString>>(
 | |
| 		"emojies_send_dice",
 | |
| 		hardcoded);
 | |
| 	const auto emoji = full.toString();
 | |
| 	if (!ranges::contains(list, emoji)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	const auto history = message.action.history;
 | |
| 	const auto peer = history->peer;
 | |
| 	const auto session = &history->session();
 | |
| 	const auto api = &session->api();
 | |
| 
 | |
| 	message.textWithTags = TextWithTags();
 | |
| 	message.action.clearDraft = false;
 | |
| 	message.action.generateLocal = true;
 | |
| 
 | |
| 
 | |
| 	const auto &action = message.action;
 | |
| 	api->sendAction(action);
 | |
| 
 | |
| 	const auto newId = FullMsgId(
 | |
| 		peer->id,
 | |
| 		session->data().nextLocalMessageId());
 | |
| 	const auto randomId = base::RandomValue<uint64>();
 | |
| 
 | |
| 	auto &histories = history->owner().histories();
 | |
| 	auto flags = NewMessageFlags(peer);
 | |
| 	auto sendFlags = MTPmessages_SendMedia::Flags(0);
 | |
| 	if (action.replyTo) {
 | |
| 		flags |= MessageFlag::HasReplyInfo;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_reply_to;
 | |
| 	}
 | |
| 	const auto anonymousPost = peer->amAnonymous();
 | |
| 	const auto silentPost = ShouldSendSilent(peer, action.options);
 | |
| 	InnerFillMessagePostFlags(action.options, peer, flags);
 | |
| 	if (silentPost) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_silent;
 | |
| 	}
 | |
| 	const auto sendAs = action.options.sendAs;
 | |
| 	const auto messageFromId = sendAs
 | |
| 		? sendAs->id
 | |
| 		: anonymousPost
 | |
| 		? 0
 | |
| 		: session->userPeerId();
 | |
| 	if (sendAs) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_send_as;
 | |
| 	}
 | |
| 	const auto messagePostAuthor = peer->isBroadcast()
 | |
| 		? session->user()->name()
 | |
| 		: QString();
 | |
| 
 | |
| 	if (action.options.scheduled) {
 | |
| 		flags |= MessageFlag::IsOrWasScheduled;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_schedule_date;
 | |
| 	}
 | |
| 	if (action.options.shortcutId) {
 | |
| 		flags |= MessageFlag::ShortcutMessage;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_quick_reply_shortcut;
 | |
| 	}
 | |
| 	if (action.options.effectId) {
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_effect;
 | |
| 	}
 | |
| 	if (action.options.invertCaption) {
 | |
| 		flags |= MessageFlag::InvertMedia;
 | |
| 		sendFlags |= MTPmessages_SendMedia::Flag::f_invert_media;
 | |
| 	}
 | |
| 
 | |
| 	session->data().registerMessageRandomId(randomId, newId);
 | |
| 
 | |
| 	history->addNewLocalMessage({
 | |
| 		.id = newId.msg,
 | |
| 		.flags = flags,
 | |
| 		.from = messageFromId,
 | |
| 		.replyTo = action.replyTo,
 | |
| 		.date = HistoryItem::NewMessageDate(action.options),
 | |
| 		.shortcutId = action.options.shortcutId,
 | |
| 		.postAuthor = messagePostAuthor,
 | |
| 		.effectId = action.options.effectId,
 | |
| 	}, TextWithEntities(), MTP_messageMediaDice(
 | |
| 		MTP_int(0),
 | |
| 		MTP_string(emoji)));
 | |
| 	histories.sendPreparedMessage(
 | |
| 		history,
 | |
| 		action.replyTo,
 | |
| 		randomId,
 | |
| 		Data::Histories::PrepareMessage<MTPmessages_SendMedia>(
 | |
| 			MTP_flags(sendFlags),
 | |
| 			peer->input,
 | |
| 			Data::Histories::ReplyToPlaceholder(),
 | |
| 			MTP_inputMediaDice(MTP_string(emoji)),
 | |
| 			MTP_string(),
 | |
| 			MTP_long(randomId),
 | |
| 			MTPReplyMarkup(),
 | |
| 			MTP_vector<MTPMessageEntity>(),
 | |
| 			MTP_int(action.options.scheduled),
 | |
| 			(sendAs ? sendAs->input : MTP_inputPeerEmpty()),
 | |
| 			Data::ShortcutIdToMTP(session, action.options.shortcutId),
 | |
| 			MTP_long(action.options.effectId)
 | |
| 		), [=](const MTPUpdates &result, const MTP::Response &response) {
 | |
| 	}, [=](const MTP::Error &error, const MTP::Response &response) {
 | |
| 		api->sendMessageFail(error, peer, randomId, newId);
 | |
| 	});
 | |
| 	api->finishForwarding(action);
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void SendLocation(SendAction action, float64 lat, float64 lon) {
 | |
| 	SendSimpleMedia(
 | |
| 		action,
 | |
| 		MTP_inputMediaGeoPoint(
 | |
| 			MTP_inputGeoPoint(
 | |
| 				MTP_flags(0),
 | |
| 				MTP_double(lat),
 | |
| 				MTP_double(lon),
 | |
| 				MTPint()))); // accuracy_radius
 | |
| }
 | |
| 
 | |
| void SendVenue(SendAction action, Data::InputVenue venue) {
 | |
| 	SendSimpleMedia(
 | |
| 		action,
 | |
| 		MTP_inputMediaVenue(
 | |
| 			MTP_inputGeoPoint(
 | |
| 				MTP_flags(0),
 | |
| 				MTP_double(venue.lat),
 | |
| 				MTP_double(venue.lon),
 | |
| 				MTPint()), // accuracy_radius
 | |
| 			MTP_string(venue.title),
 | |
| 			MTP_string(venue.address),
 | |
| 			MTP_string(venue.provider),
 | |
| 			MTP_string(venue.id),
 | |
| 			MTP_string(venue.venueType)));
 | |
| }
 | |
| 
 | |
| void FillMessagePostFlags(
 | |
| 		const SendAction &action,
 | |
| 		not_null<PeerData*> peer,
 | |
| 		MessageFlags &flags) {
 | |
| 	InnerFillMessagePostFlags(action.options, peer, flags);
 | |
| }
 | |
| 
 | |
| void SendConfirmedFile(
 | |
| 		not_null<Main::Session*> session,
 | |
| 		const std::shared_ptr<FilePrepareResult> &file) {
 | |
| 	const auto isEditing = (file->type != SendMediaType::Audio)
 | |
| 		&& (file->to.replaceMediaOf != 0);
 | |
| 	const auto newId = FullMsgId(
 | |
| 		file->to.peer,
 | |
| 		(isEditing
 | |
| 			? file->to.replaceMediaOf
 | |
| 			: session->data().nextLocalMessageId()));
 | |
| 	const auto groupId = file->album ? file->album->groupId : uint64(0);
 | |
| 	if (file->album) {
 | |
| 		const auto proj = [](const SendingAlbum::Item &item) {
 | |
| 			return item.taskId;
 | |
| 		};
 | |
| 		const auto it = ranges::find(file->album->items, file->taskId, proj);
 | |
| 		Assert(it != file->album->items.end());
 | |
| 
 | |
| 		it->msgId = newId;
 | |
| 	}
 | |
| 
 | |
| 	const auto itemToEdit = isEditing
 | |
| 		? session->data().message(newId)
 | |
| 		: nullptr;
 | |
| 	const auto history = session->data().history(file->to.peer);
 | |
| 	const auto peer = history->peer;
 | |
| 
 | |
| 	if (!isEditing) {
 | |
| 		const auto histories = &session->data().histories();
 | |
| 		file->to.replyTo.messageId = histories->convertTopicReplyToId(
 | |
| 			history,
 | |
| 			file->to.replyTo.messageId);
 | |
| 		file->to.replyTo.topicRootId = histories->convertTopicReplyToId(
 | |
| 			history,
 | |
| 			file->to.replyTo.topicRootId);
 | |
| 	}
 | |
| 
 | |
| 	session->uploader().upload(newId, file);
 | |
| 
 | |
| 	auto action = SendAction(history, file->to.options);
 | |
| 	action.clearDraft = false;
 | |
| 	action.replyTo = file->to.replyTo;
 | |
| 	action.generateLocal = true;
 | |
| 	action.replaceMediaOf = file->to.replaceMediaOf;
 | |
| 	session->api().sendAction(action);
 | |
| 
 | |
| 	auto caption = TextWithEntities{
 | |
| 		file->caption.text,
 | |
| 		TextUtilities::ConvertTextTagsToEntities(file->caption.tags)
 | |
| 	};
 | |
| 	const auto prepareFlags = Ui::ItemTextOptions(
 | |
| 		history,
 | |
| 		session->user()).flags;
 | |
| 	TextUtilities::PrepareForSending(caption, prepareFlags);
 | |
| 	TextUtilities::Trim(caption);
 | |
| 
 | |
| 	auto flags = isEditing ? MessageFlags() : NewMessageFlags(peer);
 | |
| 	if (file->to.replyTo) {
 | |
| 		flags |= MessageFlag::HasReplyInfo;
 | |
| 	}
 | |
| 	const auto anonymousPost = peer->amAnonymous();
 | |
| 	FillMessagePostFlags(action, peer, flags);
 | |
| 	if (file->to.options.scheduled) {
 | |
| 		flags |= MessageFlag::IsOrWasScheduled;
 | |
| 
 | |
| 		// Scheduled messages have no 'edited' badge.
 | |
| 		flags |= MessageFlag::HideEdited;
 | |
| 	}
 | |
| 	if (file->to.options.shortcutId) {
 | |
| 		flags |= MessageFlag::ShortcutMessage;
 | |
| 
 | |
| 		// Shortcut messages have no 'edited' badge.
 | |
| 		flags |= MessageFlag::HideEdited;
 | |
| 	}
 | |
| 	if (file->type == SendMediaType::Audio) {
 | |
| 		if (!peer->isChannel() || peer->isMegagroup()) {
 | |
| 			flags |= MessageFlag::MediaIsUnread;
 | |
| 		}
 | |
| 	}
 | |
| 	if (file->to.options.invertCaption) {
 | |
| 		flags |= MessageFlag::InvertMedia;
 | |
| 	}
 | |
| 
 | |
| 	const auto messageFromId = file->to.options.sendAs
 | |
| 		? file->to.options.sendAs->id
 | |
| 		: anonymousPost
 | |
| 		? PeerId()
 | |
| 		: session->userPeerId();
 | |
| 	const auto messagePostAuthor = peer->isBroadcast()
 | |
| 		? session->user()->name()
 | |
| 		: QString();
 | |
| 
 | |
| 	const auto media = MTPMessageMedia([&] {
 | |
| 		if (file->type == SendMediaType::Photo) {
 | |
| 			using Flag = MTPDmessageMediaPhoto::Flag;
 | |
| 			return MTP_messageMediaPhoto(
 | |
| 				MTP_flags(Flag::f_photo
 | |
| 					| (file->spoiler ? Flag::f_spoiler : Flag())),
 | |
| 				file->photo,
 | |
| 				MTPint());
 | |
| 		} else if (file->type == SendMediaType::File) {
 | |
| 			using Flag = MTPDmessageMediaDocument::Flag;
 | |
| 			return MTP_messageMediaDocument(
 | |
| 				MTP_flags(Flag::f_document
 | |
| 					| (file->spoiler ? Flag::f_spoiler : Flag())),
 | |
| 				file->document,
 | |
| 				MTPDocument(), // alt_document
 | |
| 				MTPint());
 | |
| 		} else if (file->type == SendMediaType::Audio) {
 | |
| 			const auto ttlSeconds = file->to.options.ttlSeconds;
 | |
| 			const auto isVoice = [&] {
 | |
| 				return file->document.match([](const MTPDdocumentEmpty &d) {
 | |
| 					return false;
 | |
| 				}, [](const MTPDdocument &d) {
 | |
| 					return ranges::any_of(d.vattributes().v, [&](
 | |
| 							const MTPDocumentAttribute &attribute) {
 | |
| 						using Att = MTPDdocumentAttributeAudio;
 | |
| 						return attribute.match([](const Att &data) -> bool {
 | |
| 							return data.vflags().v & Att::Flag::f_voice;
 | |
| 						}, [](const auto &) {
 | |
| 							return false;
 | |
| 						});
 | |
| 					});
 | |
| 				});
 | |
| 			}();
 | |
| 			using Flag = MTPDmessageMediaDocument::Flag;
 | |
| 			return MTP_messageMediaDocument(
 | |
| 				MTP_flags(Flag::f_document
 | |
| 					| (isVoice ? Flag::f_voice : Flag())
 | |
| 					| (ttlSeconds ? Flag::f_ttl_seconds : Flag())),
 | |
| 				file->document,
 | |
| 				MTPDocument(), // alt_document
 | |
| 				MTP_int(ttlSeconds));
 | |
| 		} else {
 | |
| 			Unexpected("Type in sendFilesConfirmed.");
 | |
| 		}
 | |
| 	}());
 | |
| 
 | |
| 	if (itemToEdit) {
 | |
| 		auto edition = HistoryMessageEdition();
 | |
| 		edition.isEditHide = (flags & MessageFlag::HideEdited);
 | |
| 		edition.editDate = 0;
 | |
| 		edition.ttl = 0;
 | |
| 		edition.mtpMedia = &media;
 | |
| 		edition.textWithEntities = caption;
 | |
| 		edition.invertMedia = file->to.options.invertCaption;
 | |
| 		edition.useSameViews = true;
 | |
| 		edition.useSameForwards = true;
 | |
| 		edition.useSameMarkup = true;
 | |
| 		edition.useSameReplies = true;
 | |
| 		edition.useSameReactions = true;
 | |
| 		edition.savePreviousMedia = true;
 | |
| 		itemToEdit->applyEdition(std::move(edition));
 | |
| 	} else {
 | |
| 		history->addNewLocalMessage({
 | |
| 			.id = newId.msg,
 | |
| 			.flags = flags,
 | |
| 			.from = messageFromId,
 | |
| 			.replyTo = file->to.replyTo,
 | |
| 			.date = HistoryItem::NewMessageDate(file->to.options),
 | |
| 			.shortcutId = file->to.options.shortcutId,
 | |
| 			.postAuthor = messagePostAuthor,
 | |
| 			.groupedId = groupId,
 | |
| 			.effectId = file->to.options.effectId,
 | |
| 		}, caption, media);
 | |
| 	}
 | |
| 
 | |
| 	if (isEditing) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	session->data().sendHistoryChangeNotifications();
 | |
| 	if (!itemToEdit) {
 | |
| 		session->changes().historyUpdated(
 | |
| 			history,
 | |
| 			(action.options.scheduled
 | |
| 				? Data::HistoryUpdate::Flag::ScheduledSent
 | |
| 				: Data::HistoryUpdate::Flag::MessageSent));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| } // namespace Api
 |