// // 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/DraftMessage.h" #include "td/telegram/Global.h" #include "td/telegram/MessageEntity.h" #include "td/telegram/MessagesManager.h" #include "td/telegram/misc.h" #include "td/telegram/ServerMessageId.h" #include "td/telegram/Td.h" #include "td/telegram/UpdatesManager.h" #include "td/utils/buffer.h" #include "td/utils/logging.h" namespace td { class SaveDraftMessageQuery final : public Td::ResultHandler { Promise promise_; DialogId dialog_id_; public: explicit SaveDraftMessageQuery(Promise &&promise) : promise_(std::move(promise)) { } void send(DialogId dialog_id, const unique_ptr &draft_message) { dialog_id_ = dialog_id; auto input_peer = td_->messages_manager_->get_input_peer(dialog_id, AccessRights::Write); if (input_peer == nullptr) { LOG(INFO) << "Can't update draft message because have no write access to " << dialog_id; return on_error(Status::Error(400, "Can't save draft message")); } int32 flags = 0; ServerMessageId reply_to_message_id; vector> input_message_entities; if (draft_message != nullptr) { if (draft_message->reply_to_message_id.is_valid() && draft_message->reply_to_message_id.is_server()) { reply_to_message_id = draft_message->reply_to_message_id.get_server_message_id(); flags |= telegram_api::messages_saveDraft::REPLY_TO_MSG_ID_MASK; } if (draft_message->input_message_text.disable_web_page_preview) { flags |= telegram_api::messages_saveDraft::NO_WEBPAGE_MASK; } input_message_entities = get_input_message_entities( td_->contacts_manager_.get(), draft_message->input_message_text.text.entities, "SaveDraftMessageQuery"); if (!input_message_entities.empty()) { flags |= telegram_api::messages_saveDraft::ENTITIES_MASK; } } send_query(G()->net_query_creator().create( telegram_api::messages_saveDraft( flags, false /*ignored*/, reply_to_message_id.get(), 0, std::move(input_peer), draft_message == nullptr ? string() : draft_message->input_message_text.text.text, std::move(input_message_entities)), {{dialog_id}})); } void on_result(BufferSlice packet) final { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(result_ptr.move_as_error()); } bool result = result_ptr.ok(); if (!result) { return on_error(Status::Error(400, "Save draft failed")); } promise_.set_value(Unit()); } void on_error(Status status) final { if (!td_->messages_manager_->on_get_dialog_error(dialog_id_, status, "SaveDraftMessageQuery")) { LOG(ERROR) << "Receive error for SaveDraftMessageQuery: " << status; } promise_.set_error(std::move(status)); } }; class GetAllDraftsQuery final : public Td::ResultHandler { public: void send() { send_query(G()->net_query_creator().create(telegram_api::messages_getAllDrafts())); } void on_result(BufferSlice packet) final { auto result_ptr = fetch_result(packet); if (result_ptr.is_error()) { return on_error(result_ptr.move_as_error()); } auto ptr = result_ptr.move_as_ok(); LOG(INFO) << "Receive result for GetAllDraftsQuery: " << to_string(ptr); td_->updates_manager_->on_get_updates(std::move(ptr), Promise()); } void on_error(Status status) final { if (!G()->is_expected_error(status)) { LOG(ERROR) << "Receive error for GetAllDraftsQuery: " << status; } status.ignore(); } }; bool need_update_draft_message(const unique_ptr &old_draft_message, const unique_ptr &new_draft_message, bool from_update) { if (new_draft_message == nullptr) { return old_draft_message != nullptr; } else { if (old_draft_message == nullptr) { return true; } if (old_draft_message->reply_to_message_id == new_draft_message->reply_to_message_id && old_draft_message->input_message_text == new_draft_message->input_message_text) { return old_draft_message->date < new_draft_message->date; } else { return !from_update || old_draft_message->date <= new_draft_message->date; } } } td_api::object_ptr get_draft_message_object(const unique_ptr &draft_message) { if (draft_message == nullptr) { return nullptr; } return td_api::make_object(draft_message->reply_to_message_id.get(), draft_message->date, get_input_message_text_object(draft_message->input_message_text)); } unique_ptr get_draft_message(ContactsManager *contacts_manager, telegram_api::object_ptr &&draft_message_ptr) { if (draft_message_ptr == nullptr) { return nullptr; } auto constructor_id = draft_message_ptr->get_id(); switch (constructor_id) { case telegram_api::draftMessageEmpty::ID: return nullptr; case telegram_api::draftMessage::ID: { auto draft = move_tl_object_as(draft_message_ptr); auto flags = draft->flags_; auto result = make_unique(); result->date = draft->date_; if ((flags & telegram_api::draftMessage::REPLY_TO_MSG_ID_MASK) != 0) { result->reply_to_message_id = MessageId(ServerMessageId(draft->reply_to_msg_id_)); if (!result->reply_to_message_id.is_valid()) { LOG(ERROR) << "Receive " << result->reply_to_message_id << " as reply_to_message_id in the draft"; result->reply_to_message_id = MessageId(); } } auto entities = get_message_entities(contacts_manager, std::move(draft->entities_), "draftMessage"); auto status = fix_formatted_text(draft->message_, entities, true, true, true, true, true); if (status.is_error()) { LOG(ERROR) << "Receive error " << status << " while parsing draft " << draft->message_; if (!clean_input_string(draft->message_)) { draft->message_.clear(); } entities = find_entities(draft->message_, false, true); } result->input_message_text.text = FormattedText{std::move(draft->message_), std::move(entities)}; result->input_message_text.disable_web_page_preview = draft->no_webpage_; result->input_message_text.clear_draft = false; return result; } default: UNREACHABLE(); return nullptr; } } Result> get_draft_message(Td *td, DialogId dialog_id, td_api::object_ptr &&draft_message) { if (draft_message == nullptr) { return nullptr; } auto result = make_unique(); result->date = G()->unix_time(); result->reply_to_message_id = MessageId(draft_message->reply_to_message_id_); if (result->reply_to_message_id != MessageId() && !result->reply_to_message_id.is_valid()) { return Status::Error(400, "Invalid reply_to_message_id specified"); } auto input_message_content = std::move(draft_message->input_message_text_); if (input_message_content != nullptr) { if (input_message_content->get_id() != td_api::inputMessageText::ID) { return Status::Error(400, "Input message content type must be InputMessageText"); } TRY_RESULT(message_content, process_input_message_text(td, dialog_id, std::move(input_message_content), false, true)); result->input_message_text = std::move(message_content); } return std::move(result); } void save_draft_message(Td *td, DialogId dialog_id, const unique_ptr &draft_message, Promise &&promise) { td->create_handler(std::move(promise))->send(dialog_id, draft_message); } void load_all_draft_messages(Td *td) { td->create_handler()->send(); } } // namespace td