// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023 // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #include "td/telegram/DialogAction.h" #include "td/telegram/misc.h" #include "td/telegram/ServerMessageId.h" #include "td/telegram/telegram_api.h" #include "td/utils/emoji.h" #include "td/utils/misc.h" #include "td/utils/Slice.h" #include "td/utils/SliceBuilder.h" #include "td/utils/utf8.h" namespace td { bool DialogAction::is_valid_emoji(string &emoji) { if (!clean_input_string(emoji)) { return false; } return is_emoji(emoji); } void DialogAction::init(Type type) { type_ = type; progress_ = 0; emoji_.clear(); } void DialogAction::init(Type type, int32 progress) { type_ = type; progress_ = clamp(progress, 0, 100); emoji_.clear(); } void DialogAction::init(Type type, string emoji) { if (is_valid_emoji(emoji)) { type_ = type; progress_ = 0; emoji_ = std::move(emoji); } else { init(Type::Cancel); } } void DialogAction::init(Type type, int32 message_id, string emoji, const string &data) { if (ServerMessageId(message_id).is_valid() && is_valid_emoji(emoji) && check_utf8(data)) { type_ = type; progress_ = message_id; emoji_ = PSTRING() << emoji << '\xFF' << data; } else { init(Type::Cancel); } } DialogAction::DialogAction(Type type, int32 progress) { init(type, progress); } DialogAction::DialogAction(td_api::object_ptr &&action) { if (action == nullptr) { return; } switch (action->get_id()) { case td_api::chatActionCancel::ID: init(Type::Cancel); break; case td_api::chatActionTyping::ID: init(Type::Typing); break; case td_api::chatActionRecordingVideo::ID: init(Type::RecordingVideo); break; case td_api::chatActionUploadingVideo::ID: { auto uploading_action = move_tl_object_as(action); init(Type::UploadingVideo, uploading_action->progress_); break; } case td_api::chatActionRecordingVoiceNote::ID: init(Type::RecordingVoiceNote); break; case td_api::chatActionUploadingVoiceNote::ID: { auto uploading_action = move_tl_object_as(action); init(Type::UploadingVoiceNote, uploading_action->progress_); break; } case td_api::chatActionUploadingPhoto::ID: { auto uploading_action = move_tl_object_as(action); init(Type::UploadingPhoto, uploading_action->progress_); break; } case td_api::chatActionUploadingDocument::ID: { auto uploading_action = move_tl_object_as(action); init(Type::UploadingDocument, uploading_action->progress_); break; } case td_api::chatActionChoosingLocation::ID: init(Type::ChoosingLocation); break; case td_api::chatActionChoosingContact::ID: init(Type::ChoosingContact); break; case td_api::chatActionStartPlayingGame::ID: init(Type::StartPlayingGame); break; case td_api::chatActionRecordingVideoNote::ID: init(Type::RecordingVideoNote); break; case td_api::chatActionUploadingVideoNote::ID: { auto uploading_action = move_tl_object_as(action); init(Type::UploadingVideoNote, uploading_action->progress_); break; } case td_api::chatActionChoosingSticker::ID: init(Type::ChoosingSticker); break; case td_api::chatActionWatchingAnimations::ID: { auto watching_animations_action = move_tl_object_as(action); init(Type::WatchingAnimations, std::move(watching_animations_action->emoji_)); break; } default: UNREACHABLE(); break; } } DialogAction::DialogAction(telegram_api::object_ptr &&action) { switch (action->get_id()) { case telegram_api::sendMessageCancelAction::ID: init(Type::Cancel); break; case telegram_api::sendMessageTypingAction::ID: init(Type::Typing); break; case telegram_api::sendMessageRecordVideoAction::ID: init(Type::RecordingVideo); break; case telegram_api::sendMessageUploadVideoAction::ID: { auto upload_video_action = move_tl_object_as(action); init(Type::UploadingVideo, upload_video_action->progress_); break; } case telegram_api::sendMessageRecordAudioAction::ID: init(Type::RecordingVoiceNote); break; case telegram_api::sendMessageUploadAudioAction::ID: { auto upload_audio_action = move_tl_object_as(action); init(Type::UploadingVoiceNote, upload_audio_action->progress_); break; } case telegram_api::sendMessageUploadPhotoAction::ID: { auto upload_photo_action = move_tl_object_as(action); init(Type::UploadingPhoto, upload_photo_action->progress_); break; } case telegram_api::sendMessageUploadDocumentAction::ID: { auto upload_document_action = move_tl_object_as(action); init(Type::UploadingDocument, upload_document_action->progress_); break; } case telegram_api::sendMessageGeoLocationAction::ID: init(Type::ChoosingLocation); break; case telegram_api::sendMessageChooseContactAction::ID: init(Type::ChoosingContact); break; case telegram_api::sendMessageGamePlayAction::ID: init(Type::StartPlayingGame); break; case telegram_api::sendMessageRecordRoundAction::ID: init(Type::RecordingVideoNote); break; case telegram_api::sendMessageUploadRoundAction::ID: { auto upload_round_action = move_tl_object_as(action); init(Type::UploadingVideoNote, upload_round_action->progress_); break; } case telegram_api::speakingInGroupCallAction::ID: init(Type::SpeakingInVoiceChat); break; case telegram_api::sendMessageHistoryImportAction::ID: { auto history_import_action = move_tl_object_as(action); init(Type::ImportingMessages, history_import_action->progress_); break; } case telegram_api::sendMessageChooseStickerAction::ID: init(Type::ChoosingSticker); break; case telegram_api::sendMessageEmojiInteractionSeen::ID: { auto emoji_interaction_seen_action = move_tl_object_as(action); init(Type::WatchingAnimations, std::move(emoji_interaction_seen_action->emoticon_)); break; } case telegram_api::sendMessageEmojiInteraction::ID: { auto emoji_interaction_action = move_tl_object_as(action); init(Type::ClickingAnimatedEmoji, emoji_interaction_action->msg_id_, std::move(emoji_interaction_action->emoticon_), emoji_interaction_action->interaction_->data_); break; } default: UNREACHABLE(); break; } } tl_object_ptr DialogAction::get_input_send_message_action() const { switch (type_) { case Type::Cancel: return make_tl_object(); case Type::Typing: return make_tl_object(); case Type::RecordingVideo: return make_tl_object(); case Type::UploadingVideo: return make_tl_object(progress_); case Type::RecordingVoiceNote: return make_tl_object(); case Type::UploadingVoiceNote: return make_tl_object(progress_); case Type::UploadingPhoto: return make_tl_object(progress_); case Type::UploadingDocument: return make_tl_object(progress_); case Type::ChoosingLocation: return make_tl_object(); case Type::ChoosingContact: return make_tl_object(); case Type::StartPlayingGame: return make_tl_object(); case Type::RecordingVideoNote: return make_tl_object(); case Type::UploadingVideoNote: return make_tl_object(progress_); case Type::SpeakingInVoiceChat: return make_tl_object(); case Type::ImportingMessages: return make_tl_object(progress_); case Type::ChoosingSticker: return make_tl_object(); case Type::WatchingAnimations: return make_tl_object(emoji_); case Type::ClickingAnimatedEmoji: default: UNREACHABLE(); return nullptr; } } tl_object_ptr DialogAction::get_secret_input_send_message_action() const { switch (type_) { case Type::Cancel: return make_tl_object(); case Type::Typing: return make_tl_object(); case Type::RecordingVideo: return make_tl_object(); case Type::UploadingVideo: return make_tl_object(); case Type::RecordingVoiceNote: return make_tl_object(); case Type::UploadingVoiceNote: return make_tl_object(); case Type::UploadingPhoto: return make_tl_object(); case Type::UploadingDocument: return make_tl_object(); case Type::ChoosingLocation: return make_tl_object(); case Type::ChoosingContact: return make_tl_object(); case Type::StartPlayingGame: return make_tl_object(); case Type::RecordingVideoNote: return make_tl_object(); case Type::UploadingVideoNote: return make_tl_object(); case Type::SpeakingInVoiceChat: return make_tl_object(); case Type::ImportingMessages: return make_tl_object(); case Type::ChoosingSticker: return make_tl_object(); case Type::WatchingAnimations: return make_tl_object(); case Type::ClickingAnimatedEmoji: default: UNREACHABLE(); return nullptr; } } tl_object_ptr DialogAction::get_chat_action_object() const { switch (type_) { case Type::Cancel: return td_api::make_object(); case Type::Typing: return td_api::make_object(); case Type::RecordingVideo: return td_api::make_object(); case Type::UploadingVideo: return td_api::make_object(progress_); case Type::RecordingVoiceNote: return td_api::make_object(); case Type::UploadingVoiceNote: return td_api::make_object(progress_); case Type::UploadingPhoto: return td_api::make_object(progress_); case Type::UploadingDocument: return td_api::make_object(progress_); case Type::ChoosingLocation: return td_api::make_object(); case Type::ChoosingContact: return td_api::make_object(); case Type::StartPlayingGame: return td_api::make_object(); case Type::RecordingVideoNote: return td_api::make_object(); case Type::UploadingVideoNote: return td_api::make_object(progress_); case Type::ChoosingSticker: return td_api::make_object(); case Type::WatchingAnimations: return td_api::make_object(emoji_); case Type::ImportingMessages: case Type::SpeakingInVoiceChat: case Type::ClickingAnimatedEmoji: default: UNREACHABLE(); return td_api::make_object(); } } bool DialogAction::is_canceled_by_message_of_type(MessageContentType message_content_type) const { if (message_content_type == MessageContentType::None) { return true; } if (type_ == Type::Typing) { return message_content_type == MessageContentType::Text || message_content_type == MessageContentType::Game || can_have_message_content_caption(message_content_type); } switch (message_content_type) { case MessageContentType::Animation: case MessageContentType::Audio: case MessageContentType::Document: return type_ == Type::UploadingDocument; case MessageContentType::ExpiredPhoto: case MessageContentType::Photo: return type_ == Type::UploadingPhoto; case MessageContentType::ExpiredVideo: case MessageContentType::Video: return type_ == Type::RecordingVideo || type_ == Type::UploadingVideo; case MessageContentType::VideoNote: return type_ == Type::RecordingVideoNote || type_ == Type::UploadingVideoNote; case MessageContentType::VoiceNote: return type_ == Type::RecordingVoiceNote || type_ == Type::UploadingVoiceNote; case MessageContentType::Contact: return type_ == Type::ChoosingContact; case MessageContentType::LiveLocation: case MessageContentType::Location: case MessageContentType::Venue: return type_ == Type::ChoosingLocation; case MessageContentType::Sticker: return type_ == Type::ChoosingSticker; case MessageContentType::Game: case MessageContentType::Invoice: case MessageContentType::Text: case MessageContentType::Unsupported: case MessageContentType::ChatCreate: case MessageContentType::ChatChangeTitle: case MessageContentType::ChatChangePhoto: case MessageContentType::ChatDeletePhoto: case MessageContentType::ChatDeleteHistory: case MessageContentType::ChatAddUsers: case MessageContentType::ChatJoinedByLink: case MessageContentType::ChatDeleteUser: case MessageContentType::ChatMigrateTo: case MessageContentType::ChannelCreate: case MessageContentType::ChannelMigrateFrom: case MessageContentType::PinMessage: case MessageContentType::GameScore: case MessageContentType::ScreenshotTaken: case MessageContentType::ChatSetTtl: case MessageContentType::Call: case MessageContentType::PaymentSuccessful: case MessageContentType::ContactRegistered: case MessageContentType::CustomServiceAction: case MessageContentType::WebsiteConnected: case MessageContentType::PassportDataSent: case MessageContentType::PassportDataReceived: case MessageContentType::Poll: case MessageContentType::Dice: case MessageContentType::ProximityAlertTriggered: case MessageContentType::GroupCall: case MessageContentType::InviteToGroupCall: case MessageContentType::ChatSetTheme: case MessageContentType::WebViewDataSent: case MessageContentType::WebViewDataReceived: case MessageContentType::GiftPremium: case MessageContentType::TopicCreate: case MessageContentType::TopicEdit: case MessageContentType::SuggestProfilePhoto: case MessageContentType::WriteAccessAllowed: case MessageContentType::RequestedDialog: case MessageContentType::WebViewWriteAccessAllowed: case MessageContentType::SetBackground: case MessageContentType::Story: case MessageContentType::WriteAccessAllowedByRequest: case MessageContentType::GiftCode: case MessageContentType::Giveaway: return false; default: UNREACHABLE(); return false; } } DialogAction DialogAction::get_uploading_action(MessageContentType message_content_type, int32 progress) { switch (message_content_type) { case MessageContentType::Animation: case MessageContentType::Audio: case MessageContentType::Document: return DialogAction(Type::UploadingDocument, progress); case MessageContentType::Photo: return DialogAction(Type::UploadingPhoto, progress); case MessageContentType::Video: return DialogAction(Type::UploadingVideo, progress); case MessageContentType::VideoNote: return DialogAction(Type::UploadingVideoNote, progress); case MessageContentType::VoiceNote: return DialogAction(Type::UploadingVoiceNote, progress); default: return DialogAction(); } } DialogAction DialogAction::get_typing_action() { return DialogAction(Type::Typing, 0); } DialogAction DialogAction::get_speaking_action() { return DialogAction(Type::SpeakingInVoiceChat, 0); } int32 DialogAction::get_importing_messages_action_progress() const { if (type_ != Type::ImportingMessages) { return -1; } return progress_; } string DialogAction::get_watching_animations_emoji() const { if (type_ == Type::WatchingAnimations) { return emoji_; } return string(); } DialogAction::ClickingAnimateEmojiInfo DialogAction::get_clicking_animated_emoji_action_info() const { ClickingAnimateEmojiInfo result; if (type_ == Type::ClickingAnimatedEmoji) { auto pos = emoji_.find('\xFF'); CHECK(pos < emoji_.size()); result.message_id = progress_; result.emoji = emoji_.substr(0, pos); result.data = emoji_.substr(pos + 1); } return result; } StringBuilder &operator<<(StringBuilder &string_builder, const DialogAction &action) { string_builder << "ChatAction"; const char *type = [action_type = action.type_] { switch (action_type) { case DialogAction::Type::Cancel: return "Cancel"; case DialogAction::Type::Typing: return "Typing"; case DialogAction::Type::RecordingVideo: return "RecordingVideo"; case DialogAction::Type::UploadingVideo: return "UploadingVideo"; case DialogAction::Type::RecordingVoiceNote: return "RecordingVoiceNote"; case DialogAction::Type::UploadingVoiceNote: return "UploadingVoiceNote"; case DialogAction::Type::UploadingPhoto: return "UploadingPhoto"; case DialogAction::Type::UploadingDocument: return "UploadingDocument"; case DialogAction::Type::ChoosingLocation: return "ChoosingLocation"; case DialogAction::Type::ChoosingContact: return "ChoosingContact"; case DialogAction::Type::StartPlayingGame: return "StartPlayingGame"; case DialogAction::Type::RecordingVideoNote: return "RecordingVideoNote"; case DialogAction::Type::UploadingVideoNote: return "UploadingVideoNote"; case DialogAction::Type::SpeakingInVoiceChat: return "SpeakingInVoiceChat"; case DialogAction::Type::ImportingMessages: return "ImportingMessages"; case DialogAction::Type::ChoosingSticker: return "ChoosingSticker"; case DialogAction::Type::WatchingAnimations: return "WatchingAnimations"; case DialogAction::Type::ClickingAnimatedEmoji: return "ClickingAnimatedEmoji"; default: UNREACHABLE(); return "Cancel"; } }(); string_builder << type << "Action"; if (action.type_ == DialogAction::Type::ClickingAnimatedEmoji) { auto pos = action.emoji_.find('\xFF'); CHECK(pos < action.emoji_.size()); string_builder << '(' << action.progress_ << ")(" << Slice(action.emoji_).substr(0, pos) << ")(" << Slice(action.emoji_).substr(pos + 1) << ')'; } else { if (action.progress_ != 0) { string_builder << '(' << action.progress_ << "%)"; } if (!action.emoji_.empty()) { string_builder << '(' << action.emoji_ << ')'; } } return string_builder; } } // namespace td