// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2019 // // 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/Location.h" #include "td/telegram/Global.h" #include "td/telegram/misc.h" #include "td/telegram/secret_api.h" #include "td/telegram/td_api.h" #include "td/telegram/telegram_api.h" #include "td/utils/common.h" #include namespace td { void Location::init(double latitude, double longitude, int64 access_hash) { if (std::isfinite(latitude) && std::isfinite(longitude) && std::abs(latitude) <= 90 && std::abs(longitude) <= 180) { is_empty_ = false; latitude_ = latitude; longitude_ = longitude; access_hash_ = access_hash; G()->add_location_access_hash(latitude_, longitude_, access_hash_); } } Location::Location(double latitude, double longitude, int64 access_hash) { init(latitude, longitude, access_hash); } Location::Location(const tl_object_ptr &geo_point) : Location(geo_point->lat_, geo_point->long_, 0) { } Location::Location(const tl_object_ptr &geo_point_ptr) { if (geo_point_ptr == nullptr) { return; } switch (geo_point_ptr->get_id()) { case telegram_api::geoPointEmpty::ID: break; case telegram_api::geoPoint::ID: { auto geo_point = static_cast(geo_point_ptr.get()); init(geo_point->lat_, geo_point->long_, geo_point->access_hash_); break; } default: UNREACHABLE(); break; } } Location::Location(const tl_object_ptr &location) { if (location == nullptr) { return; } init(location->latitude_, location->longitude_, 0); } bool Location::empty() const { return is_empty_; } bool Location::is_valid_map_point() const { const double MAX_VALID_MAP_LATITUDE = 85.05112877; return !empty() && std::abs(latitude_) <= MAX_VALID_MAP_LATITUDE; } tl_object_ptr Location::get_location_object() const { if (empty()) { return nullptr; } return make_tl_object(latitude_, longitude_); } tl_object_ptr Location::get_input_geo_point() const { if (empty()) { return make_tl_object(); } return make_tl_object(latitude_, longitude_); } tl_object_ptr Location::get_input_media_geo_point() const { return make_tl_object(get_input_geo_point()); } SecretInputMedia Location::get_secret_input_media_geo_point() const { return SecretInputMedia{nullptr, make_tl_object(latitude_, longitude_)}; } bool operator==(const Location &lhs, const Location &rhs) { if (lhs.is_empty_) { return rhs.is_empty_; } return !rhs.is_empty_ && std::abs(lhs.latitude_ - rhs.latitude_) < 1e-6 && std::abs(lhs.longitude_ - rhs.longitude_) < 1e-6; } bool operator!=(const Location &lhs, const Location &rhs) { return !(lhs == rhs); } StringBuilder &operator<<(StringBuilder &string_builder, const Location &location) { if (location.empty()) { return string_builder << "Location[empty]"; } return string_builder << "Location[latitude = " << location.latitude_ << ", longitude = " << location.longitude_ << "]"; } Venue::Venue(const tl_object_ptr &geo_point_ptr, string title, string address, string provider, string id, string type) : location_(geo_point_ptr) , title_(std::move(title)) , address_(std::move(address)) , provider_(std::move(provider)) , id_(std::move(id)) , type_(std::move(type)) { } Venue::Venue(Location location, string title, string address, string provider, string id, string type) : location_(location) , title_(std::move(title)) , address_(std::move(address)) , provider_(std::move(provider)) , id_(std::move(id)) , type_(std::move(type)) { } Venue::Venue(const tl_object_ptr &venue) : location_(venue->location_) , title_(venue->title_) , address_(venue->address_) , provider_(venue->provider_) , id_(venue->id_) , type_(venue->type_) { } bool Venue::empty() const { return location_.empty(); } Location &Venue::location() { return location_; } const Location &Venue::location() const { return location_; } tl_object_ptr Venue::get_venue_object() const { return make_tl_object(location_.get_location_object(), title_, address_, provider_, id_, type_); } tl_object_ptr Venue::get_input_media_venue() const { return make_tl_object(location_.get_input_geo_point(), title_, address_, provider_, id_, type_); } SecretInputMedia Venue::get_secret_input_media_venue() const { return SecretInputMedia{nullptr, make_tl_object( location_.get_latitude(), location_.get_longitude(), title_, address_, provider_, id_)}; } tl_object_ptr Venue::get_input_bot_inline_message_media_venue( int32 flags, tl_object_ptr &&reply_markup) const { return make_tl_object( flags, location_.get_input_geo_point(), title_, address_, provider_, id_, type_, std::move(reply_markup)); } bool operator==(const Venue &lhs, const Venue &rhs) { return lhs.location_ == rhs.location_ && lhs.title_ == rhs.title_ && lhs.address_ == rhs.address_ && lhs.provider_ == rhs.provider_ && lhs.id_ == rhs.id_ && lhs.type_ == rhs.type_; } bool operator!=(const Venue &lhs, const Venue &rhs) { return !(lhs == rhs); } StringBuilder &operator<<(StringBuilder &string_builder, const Venue &venue) { return string_builder << "Venue[location = " << venue.location_ << ", title = " << venue.title_ << ", address = " << venue.address_ << ", provider = " << venue.provider_ << ", id = " << venue.id_ << ", type = " << venue.type_ << "]"; } Result> process_input_message_location( tl_object_ptr &&input_message_content) { CHECK(input_message_content != nullptr); CHECK(input_message_content->get_id() == td_api::inputMessageLocation::ID); auto input_location = static_cast(input_message_content.get()); Location location(input_location->location_); if (location.empty()) { return Status::Error(400, "Wrong location specified"); } constexpr int32 MIN_LIVE_LOCATION_PERIOD = 60; // seconds, server side limit constexpr int32 MAX_LIVE_LOCATION_PERIOD = 86400; // seconds, server side limit auto period = input_location->live_period_; if (period != 0 && (period < MIN_LIVE_LOCATION_PERIOD || period > MAX_LIVE_LOCATION_PERIOD)) { return Status::Error(400, "Wrong live location period specified"); } return std::make_pair(std::move(location), period); } Result process_input_message_venue(tl_object_ptr &&input_message_content) { CHECK(input_message_content != nullptr); CHECK(input_message_content->get_id() == td_api::inputMessageVenue::ID); auto venue = std::move(static_cast(input_message_content.get())->venue_); if (venue == nullptr) { return Status::Error(400, "Venue can't be empty"); } if (!clean_input_string(venue->title_)) { return Status::Error(400, "Venue title must be encoded in UTF-8"); } if (!clean_input_string(venue->address_)) { return Status::Error(400, "Venue address must be encoded in UTF-8"); } if (!clean_input_string(venue->provider_)) { return Status::Error(400, "Venue provider must be encoded in UTF-8"); } if (!clean_input_string(venue->id_)) { return Status::Error(400, "Venue identifier must be encoded in UTF-8"); } if (!clean_input_string(venue->type_)) { return Status::Error(400, "Venue type must be encoded in UTF-8"); } Venue result(venue); if (result.empty()) { return Status::Error(400, "Wrong venue location specified"); } return result; } } // namespace td