// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2022 // // 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/MessageReaction.h" #include "td/telegram/ContactsManager.h" #include "td/telegram/MessagesManager.h" #include "td/telegram/Td.h" #include "td/utils/algorithm.h" #include "td/utils/logging.h" #include namespace td { td_api::object_ptr MessageReaction::get_message_reaction_object(Td *td) const { CHECK(!is_empty()); vector> recent_choosers; for (auto dialog_id : recent_chooser_dialog_ids_) { auto dialog_type = dialog_id.get_type(); if (dialog_type == DialogType::User) { auto user_id = dialog_id.get_user_id(); if (td->contacts_manager_->have_min_user(user_id)) { recent_choosers.push_back(td_api::make_object( td->contacts_manager_->get_user_id_object(user_id, "get_message_reaction_object"))); } else { LOG(ERROR) << "Skip unknown reacted " << user_id; } } else { if (!td->messages_manager_->have_dialog(dialog_id) && (td->messages_manager_->have_dialog_info(dialog_id) || (dialog_type == DialogType::Channel && td->contacts_manager_->have_min_channel(dialog_id.get_channel_id())))) { LOG(INFO) << "Force creation of " << dialog_id; td->messages_manager_->force_create_dialog(dialog_id, "get_message_reaction_object", true); } if (td->messages_manager_->have_dialog(dialog_id)) { recent_choosers.push_back(td_api::make_object(dialog_id.get())); } else { LOG(ERROR) << "Skip unknown reacted " << dialog_id; } } } return td_api::make_object(reaction_, choose_count_, is_chosen_, std::move(recent_choosers)); } bool operator==(const MessageReaction &lhs, const MessageReaction &rhs) { return lhs.reaction_ == rhs.reaction_ && lhs.choose_count_ == rhs.choose_count_ && lhs.is_chosen_ == rhs.is_chosen_ && lhs.recent_chooser_dialog_ids_ == rhs.recent_chooser_dialog_ids_; } StringBuilder &operator<<(StringBuilder &string_builder, const MessageReaction &reaction) { string_builder << '[' << reaction.reaction_ << (reaction.is_chosen_ ? " X " : " x ") << reaction.choose_count_; if (!reaction.recent_chooser_dialog_ids_.empty()) { string_builder << " by " << reaction.recent_chooser_dialog_ids_; } return string_builder << ']'; } unique_ptr MessageReactions::get_message_reactions( Td *td, tl_object_ptr &&reactions, bool is_bot) { if (reactions == nullptr || is_bot) { return nullptr; } auto result = make_unique(); result->can_see_all_choosers_ = reactions->can_see_list_; result->is_min_ = reactions->min_; std::unordered_set reaction_strings; std::unordered_set recent_choosers; for (auto &reaction_count : reactions->results_) { if (reaction_count->count_ <= 0 || reaction_count->count_ >= MessageReaction::MAX_CHOOSE_COUNT) { LOG(ERROR) << "Receive reaction " << reaction_count->reaction_ << " with invalid count " << reaction_count->count_; continue; } if (!reaction_strings.insert(reaction_count->reaction_).second) { LOG(ERROR) << "Receive duplicate reaction " << reaction_count->reaction_; continue; } vector recent_chooser_dialog_ids; vector> recent_chooser_min_channels; for (auto &user_reaction : reactions->recent_reactons_) { if (user_reaction->reaction_ == reaction_count->reaction_) { DialogId dialog_id(UserId(user_reaction->user_id_)); if (!dialog_id.is_valid()) { LOG(ERROR) << "Receive invalid " << dialog_id << " as a recent chooser"; continue; } if (!recent_choosers.insert(dialog_id).second) { LOG(ERROR) << "Receive duplicate " << dialog_id << " as a recent chooser"; continue; } if (!td->messages_manager_->have_dialog_info(dialog_id)) { auto dialog_type = dialog_id.get_type(); if (dialog_type == DialogType::User) { auto user_id = dialog_id.get_user_id(); if (!td->contacts_manager_->have_min_user(user_id)) { LOG(ERROR) << "Have no info about " << user_id; continue; } } else if (dialog_type == DialogType::Channel) { auto channel_id = dialog_id.get_channel_id(); auto min_channel = td->contacts_manager_->get_min_channel(channel_id); if (min_channel == nullptr) { LOG(ERROR) << "Have no info about reacted " << channel_id; continue; } recent_chooser_min_channels.emplace_back(channel_id, *min_channel); } else { LOG(ERROR) << "Have no info about reacted " << dialog_id; continue; } } recent_chooser_dialog_ids.push_back(dialog_id); if (recent_chooser_dialog_ids.size() == MessageReaction::MAX_RECENT_CHOOSERS) { break; } } } result->reactions_.emplace_back(std::move(reaction_count->reaction_), reaction_count->count_, reaction_count->chosen_, std::move(recent_chooser_dialog_ids), std::move(recent_chooser_min_channels)); } return result; } void MessageReactions::update_from(const MessageReactions &old_reactions) { if (old_reactions.has_pending_reaction_) { // we will ignore all updates, received while there is a pending reaction, so there are no reasons to update return; } CHECK(!has_pending_reaction_); if (is_min_ && !old_reactions.is_min_) { // chosen reaction was known, keep it is_min_ = false; for (const auto &old_reaction : old_reactions.reactions_) { if (old_reaction.is_chosen()) { for (auto &reaction : reactions_) { if (reaction.get_reaction() == old_reaction.get_reaction()) { reaction.set_is_chosen(true); } } } } } } bool MessageReactions::need_update_message_reactions(const MessageReactions *old_reactions, const MessageReactions *new_reactions) { if (old_reactions == nullptr) { // add reactions return new_reactions != nullptr; } if (old_reactions->has_pending_reaction_) { // ignore all updates, received while there is a pending reaction return false; } if (new_reactions == nullptr) { // remove reactions when they are disabled return true; } // has_pending_reaction_ and old_chosen_reaction_ don't affect visible state // compare all other fields return old_reactions->reactions_ != new_reactions->reactions_ || old_reactions->is_min_ != new_reactions->is_min_ || old_reactions->can_see_all_choosers_ != new_reactions->can_see_all_choosers_ || old_reactions->need_polling_ != new_reactions->need_polling_; } } // namespace td