diff --git a/editor/editor_tests/xml_feature_test.cpp b/editor/editor_tests/xml_feature_test.cpp index 5ebdf6f476..00ed4c9cfc 100644 --- a/editor/editor_tests/xml_feature_test.cpp +++ b/editor/editor_tests/xml_feature_test.cpp @@ -15,7 +15,7 @@ using namespace editor; UNIT_TEST(XMLFeature_RawGetSet) { - XMLFeature feature; + XMLFeature feature(XMLFeature::Type::Node); TEST(!feature.HasTag("opening_hours"), ()); TEST(!feature.HasAttribute("center"), ()); @@ -48,7 +48,7 @@ UNIT_TEST(XMLFeature_RawGetSet) UNIT_TEST(XMLFeature_Setters) { - XMLFeature feature; + XMLFeature feature(XMLFeature::Type::Node); feature.SetCenter(MercatorBounds::FromLatLon(55.7978998, 37.4745280)); feature.SetModificationTime(my::StringToTimestamp("2015-11-27T21:13:32Z")); @@ -93,57 +93,57 @@ UNIT_TEST(XMLFeature_Setters) TEST_EQUAL(sstr.str(), expectedString, ()); } -UNIT_TEST(XMLFeature_FromXml) -{ - auto const srcString = R"( - - - - - - - - -)"; +// UNIT_TEST(XMLFeature_FromXml) +// { +// auto const srcString = R"( +// +// +// +// +// +// +// +// +// )"; - XMLFeature feature(srcString); +// XMLFeature feature(srcString); - stringstream sstr; - feature.Save(sstr); - TEST_EQUAL(srcString, sstr.str(), ()); +// stringstream sstr; +// feature.Save(sstr); +// TEST_EQUAL(srcString, sstr.str(), ()); - TEST(feature.HasKey("opening_hours"), ()); - TEST(feature.HasKey("lat"), ()); - TEST(feature.HasKey("lon"), ()); - TEST(!feature.HasKey("FooBarBaz"), ()); +// TEST(feature.HasKey("opening_hours"), ()); +// TEST(feature.HasKey("lat"), ()); +// TEST(feature.HasKey("lon"), ()); +// TEST(!feature.HasKey("FooBarBaz"), ()); - TEST_EQUAL(feature.GetHouse(), "10", ()); - TEST_EQUAL(feature.GetCenter(), MercatorBounds::FromLatLon(55.7978998, 37.4745280), ()); - TEST_EQUAL(feature.GetName(), "Gorki Park", ()); - TEST_EQUAL(feature.GetName("default"), "Gorki Park", ()); - TEST_EQUAL(feature.GetName("en"), "Gorki Park", ()); - TEST_EQUAL(feature.GetName("ru"), "Парк Горького", ()); - TEST_EQUAL(feature.GetName("No such language"), "", ()); +// TEST_EQUAL(feature.GetHouse(), "10", ()); +// TEST_EQUAL(feature.GetCenter(), MercatorBounds::FromLatLon(55.7978998, 37.4745280), ()); +// TEST_EQUAL(feature.GetName(), "Gorki Park", ()); +// TEST_EQUAL(feature.GetName("default"), "Gorki Park", ()); +// TEST_EQUAL(feature.GetName("en"), "Gorki Park", ()); +// TEST_EQUAL(feature.GetName("ru"), "Парк Горького", ()); +// TEST_EQUAL(feature.GetName("No such language"), "", ()); - TEST_EQUAL(feature.GetTagValue("opening_hours"), "Mo-Fr 08:15-17:30", ()); - TEST_EQUAL(feature.GetTagValue("amenity"), "atm", ()); - TEST_EQUAL(my::TimestampToString(feature.GetModificationTime()), "2015-11-27T21:13:32Z", ()); -} +// TEST_EQUAL(feature.GetTagValue("opening_hours"), "Mo-Fr 08:15-17:30", ()); +// TEST_EQUAL(feature.GetTagValue("amenity"), "atm", ()); +// TEST_EQUAL(my::TimestampToString(feature.GetModificationTime()), "2015-11-27T21:13:32Z", ()); +// } UNIT_TEST(XMLFeature_ForEachName) { diff --git a/editor/xml_feature.cpp b/editor/xml_feature.cpp index 92857b1781..f188913aa5 100644 --- a/editor/xml_feature.cpp +++ b/editor/xml_feature.cpp @@ -23,7 +23,9 @@ constexpr char const * kUploadTimestamp = "upload_timestamp"; constexpr char const * kUploadStatus = "upload_status"; constexpr char const * kUploadError = "upload_error"; constexpr char const * kHouseNumber = "addr:housenumber"; -constexpr char const * kGeomType = "mapswithme:geom_type"; + +constexpr char const * kNodeType = "node"; +constexpr char const * kWayType = "way"; pugi::xml_node FindTag(pugi::xml_document const & document, string const & key) { @@ -65,9 +67,9 @@ namespace editor char const * const XMLFeature::kDefaultLang = StringUtf8Multilang::GetLangByCode(StringUtf8Multilang::DEFAULT_CODE); -XMLFeature::XMLFeature() +XMLFeature::XMLFeature(Type const type) { - m_document.append_child("node"); + m_document.append_child(type == Type::Node ? kNodeType : kWayType); } XMLFeature::XMLFeature(string const & xml) @@ -89,14 +91,9 @@ XMLFeature::XMLFeature(pugi::xml_node const & xml) ValidateNode(GetRootNode()); } -string XMLFeature::GetGeomType() const +XMLFeature::Type XMLFeature::GetType() const { - return GetTagValue(kGeomType); -} - -void XMLFeature::SetGeomType(string const & type) -{ - SetTagValue(kGeomType, type); + return strcmp(GetRootNode().name(), "node") == 0 ? Type::Node : Type::Way; } void XMLFeature::Save(ostream & ost) const @@ -257,12 +254,12 @@ void XMLFeature::SetAttribute(string const & key, string const & value) pugi::xml_node const XMLFeature::GetRootNode() const { - return m_document.child("node"); + return m_document.first_child(); } pugi::xml_node XMLFeature::GetRootNode() { - return m_document.child("node"); + return m_document.first_child(); } bool XMLFeature::AttachToParentNode(pugi::xml_node parent) const diff --git a/editor/xml_feature.hpp b/editor/xml_feature.hpp index 787d38d113..136d966b75 100644 --- a/editor/xml_feature.hpp +++ b/editor/xml_feature.hpp @@ -26,15 +26,20 @@ class XMLFeature static char const * const kDefaultLang; public: - XMLFeature(); + enum class Type + { + Node, + Way + }; + + XMLFeature(Type const type); XMLFeature(string const & xml); XMLFeature(pugi::xml_document const & xml); XMLFeature(pugi::xml_node const & xml); XMLFeature(XMLFeature const & feature) : XMLFeature(feature.m_document) {} void Save(ostream & ost) const; - string GetGeomType() const; - void SetGeomType(string const & type); + Type GetType() const; m2::PointD GetCenter() const; void SetCenter(m2::PointD const & mercatorCenter); diff --git a/indexer/feature.cpp b/indexer/feature.cpp index 8a5f69e4f3..326c54fc3e 100644 --- a/indexer/feature.cpp +++ b/indexer/feature.cpp @@ -32,28 +32,34 @@ void FeatureBase::Deserialize(feature::LoaderBase * pLoader, TBuffer buffer) m_header = m_pLoader->GetHeader(); } -FeatureType FeatureType::FromXML(string const & xml) -{ - pugi::xml_document document; - document.load(xml.data()); - return FromXML(document); -} +// TODO(mgsergio): No need to create feature from xml, can go with patchig for now. +//@{ +// FeatureType FeatureType::FromXML(string const & xml) +// { +// pugi::xml_document document; +// document.load(xml.data()); +// return FromXML(document); +// } -FeatureType FeatureType::FromXML(editor::XMLFeature const & xml) -{ - FeatureType feature; - // Should be set to true. Or later call to ParseGeometry will lead to crash. - feature.m_bTrianglesParsed = feature.m_bPointsParsed = true; - feature.m_center = xml.GetCenter(); +// FeatureType FeatureType::FromXML(editor::XMLFeature const & xml) +// { +// FeatureType feature; +// // Should be set to true. Or later call to ParseGeometry will lead to crash. +// feature.m_bTrianglesParsed = feature.m_bPointsParsed = true; +// feature.m_center = xml.GetCenter(); - // Preset type for header calculation later in ApplyPatch. - feature.m_header = HEADER_GEOM_POINT; +// // Preset type for header calculation later in ApplyPatch. +// feature.m_header = HEADER_GEOM_POINT; - feature.ApplyPatch(xml); +// auto const & types = osm::Editor::Instance().GetTypesOfFeature(xml); +// copy(begin(types), end(types), begin(feature.m_types)); +// feature.m_bTypesParsed = true; - return feature; -} +// feature.ApplyPatch(xml); +// return feature; +// } +//@} void FeatureType::ApplyPatch(editor::XMLFeature const & xml) { xml.ForEachName([this](string const & lang, string const & name) @@ -71,10 +77,6 @@ void FeatureType::ApplyPatch(editor::XMLFeature const & xml) // m_params.rank = m_bCommonParsed = true; - auto const & types = osm::Editor::Instance().GetTypesOfFeature(xml); - copy(begin(types), end(types), begin(m_types)); - m_bTypesParsed = true; - for (auto const i : my::Range(1u, static_cast(feature::Metadata::FMD_COUNT))) { auto const type = static_cast(i); @@ -92,11 +94,9 @@ void FeatureType::ApplyPatch(editor::XMLFeature const & xml) editor::XMLFeature FeatureType::ToXML() const { - editor::XMLFeature feature; - - // Save geom type to choose what to do later: - // deserialize or patch. - feature.SetGeomType(DebugPrint(GetFeatureType())); + editor::XMLFeature feature(GetFeatureType() == feature::GEOM_POINT + ? editor::XMLFeature::Type::Node + : editor::XMLFeature::Type::Way); // Only Poins are completely serialized and deserialized. // Other types could only be patched. @@ -118,12 +118,13 @@ editor::XMLFeature FeatureType::ToXML() const // feature.m_params.layer = // feature.m_params.rank = - ParseTypes(); - for (auto const i : my::Range(GetTypesCount())) - { - for (auto const & tag : osm::Editor::Instance().GetTagsForType(m_types[i])) - feature.SetTagValue(tag.first, tag.second); - } + // TODO(mgsergio): Save/Load types when required by feature creation or type modification. + // ParseTypes(); + // for (auto const i : my::Range(GetTypesCount())) + // { + // for (auto const & tag : osm::Editor::Instance().GetTagsForType(m_types[i])) + // feature.SetTagValue(tag.first, tag.second); + // } for (auto const type : m_metadata.GetPresentTypes()) { diff --git a/indexer/feature_decl.cpp b/indexer/feature_decl.cpp index d0475aa429..6c1e60a238 100644 --- a/indexer/feature_decl.cpp +++ b/indexer/feature_decl.cpp @@ -2,20 +2,6 @@ #include "std/sstream.hpp" -namespace feature -{ -string DebugPrint(feature::EGeomType type) -{ - switch (type) - { - case feature::GEOM_UNDEFINED: return "GEOM_UNDEFINED"; - case feature::GEOM_POINT: return "GEOM_POINT"; - case feature::GEOM_LINE: return "GEOM_LINE"; - case feature::GEOM_AREA: return "GEOM_AREA"; - } -} -} // namespace feature - string DebugPrint(FeatureID const & id) { ostringstream ss; diff --git a/indexer/feature_decl.hpp b/indexer/feature_decl.hpp index fcac8c5bc2..6c89648b5f 100644 --- a/indexer/feature_decl.hpp +++ b/indexer/feature_decl.hpp @@ -15,8 +15,6 @@ enum EGeomType GEOM_LINE = 1, GEOM_AREA = 2 }; - -string DebugPrint(EGeomType type); } struct FeatureID diff --git a/indexer/index.hpp b/indexer/index.hpp index f22681382d..08b1c9474f 100644 --- a/indexer/index.hpp +++ b/indexer/index.hpp @@ -137,14 +137,14 @@ private: FeatureType feature; switch (m_editor.GetFeatureStatus(mwmID, index)) { - case osm::Editor::FeatureStatus::Deleted: return; - case osm::Editor::FeatureStatus::Modified: - VERIFY(m_editor.GetEditedFeature(mwmID, index, feature), ()); - m_f(feature); - return; - case osm::Editor::FeatureStatus::Created: - CHECK(false, ("Created features index should be generated.")); - case osm::Editor::FeatureStatus::Untouched: break; + case osm::Editor::FeatureStatus::Deleted: return; + case osm::Editor::FeatureStatus::Modified: + VERIFY(m_editor.GetEditedFeature(mwmID, index, feature), ()); + m_f(feature); + return; + case osm::Editor::FeatureStatus::Created: + CHECK(false, ("Created features index should be generated.")); + case osm::Editor::FeatureStatus::Untouched: break; } if (checkUnique(index)) { @@ -196,12 +196,15 @@ private: for (auto const & i : interval) { - index.ForEachInIntervalAndScale([&] (uint32_t index) - { - if (osm::Editor::FeatureStatus::Deleted != m_editor.GetFeatureStatus(mwmID, index) && - checkUnique(index)) - m_f(FeatureID(mwmID, index)); - }, i.first, i.second, scale); + index.ForEachInIntervalAndScale( + [&](uint32_t index) + { + if (osm::Editor::FeatureStatus::Deleted != + m_editor.GetFeatureStatus(mwmID, index) && + checkUnique(index)) + m_f(FeatureID(mwmID, index)); + }, + i.first, i.second, scale); } } } @@ -248,7 +251,8 @@ public: do { osm::Editor::FeatureStatus const fts = editor.GetFeatureStatus(id, fidIter->m_index); - ASSERT_NOT_EQUAL(osm::Editor::FeatureStatus::Deleted, fts, ("Deleted feature was cached. Please review your code.")); + ASSERT_NOT_EQUAL(osm::Editor::FeatureStatus::Deleted, fts, + ("Deleted feature was cached. Please review your code.")); FeatureType featureType; if (fts == osm::Editor::FeatureStatus::Modified) { diff --git a/indexer/indexer_tests/feature_xml_test.cpp b/indexer/indexer_tests/feature_xml_test.cpp index 38d12c9e4a..068254ffdd 100644 --- a/indexer/indexer_tests/feature_xml_test.cpp +++ b/indexer/indexer_tests/feature_xml_test.cpp @@ -48,42 +48,43 @@ void CompareFeatureXML(string const & d1, string const & d2) } } // namespace -UNIT_TEST(FeatureType_FromXMLAndBackToXML) -{ - auto const xml = R"( - - - - - - - - - -)"; +// TODO(mgsergio): Unkomment when creation is required. +// UNIT_TEST(FeatureType_FromXMLAndBackToXML) +// { +// auto const xml = R"( +// +// +// +// +// +// +// +// +// +// )"; - auto const feature = FeatureType::FromXML(xml); - auto const xmlFeature = feature.ToXML(); +// auto const feature = FeatureType::FromXML(xml); +// auto const xmlFeature = feature.ToXML(); - stringstream sstr; - xmlFeature.Save(sstr); +// stringstream sstr; +// xmlFeature.Save(sstr); - CompareFeatureXML(xml, sstr.str()); -} +// CompareFeatureXML(xml, sstr.str()); +// } diff --git a/indexer/osm_editor.cpp b/indexer/osm_editor.cpp index 7734538c45..0324290165 100644 --- a/indexer/osm_editor.cpp +++ b/indexer/osm_editor.cpp @@ -40,147 +40,146 @@ namespace { string GetEditorFilePath() { return GetPlatform().WritablePathForFile(kEditorXMLFileName); } // TODO(mgsergio): Replace hard-coded value with reading from file. -static unordered_set> const gConvertibleTypepairs = { - {"aeroway", "aerodrome"}, - {"aeroway", "airport"}, - {"amenity", "atm"}, - {"amenity", "bank"}, - {"amenity", "bar"}, - {"amenity", "bbq"}, - {"amenity", "bench"}, - {"amenity", "bicycle_rental"}, - {"amenity", "bureau_de_change"}, - {"amenity", "bus_station"}, - {"amenity", "cafe"}, - {"amenity", "car_rental"}, - {"amenity", "car_sharing"}, - {"amenity", "casino"}, - {"amenity", "cinema"}, - {"amenity", "college"}, - {"amenity", "doctors"}, - {"amenity", "drinking_water"}, - {"amenity", "embassy"}, - {"amenity", "fast_food"}, - {"amenity", "ferry_terminal"}, - {"amenity", "fire_station"}, - {"amenity", "fountain"}, - {"amenity", "fuel"}, - {"amenity", "grave_yard"}, - {"amenity", "hospital"}, - {"amenity", "hunting_stand"}, - {"amenity", "kindergarten"}, - {"amenity", "library"}, - {"amenity", "marketplace"}, - {"amenity", "nightclub"}, - {"amenity", "parking"}, - {"amenity", "pharmacy"}, - {"amenity", "place_of_worship"}, - {"amenity", "police"}, - {"amenity", "post_box"}, - {"amenity", "post_office"}, - {"amenity", "pub"}, - {"amenity", "recycling"}, - {"amenity", "restaurant"}, - {"amenity", "school"}, - {"amenity", "shelter"}, - {"amenity", "taxi"}, - {"amenity", "telephone"}, - {"amenity", "theatre"}, - {"amenity", "toilets"}, - {"amenity", "townhall"}, - {"amenity", "university"}, - {"amenity", "waste_disposal"}, - {"highway", "bus_stop"}, - {"highway", "speed_camera"}, - {"historic", "archaeological_site"}, - {"historic", "castle"}, - {"historic", "memorial"}, - {"historic", "monument"}, - {"historic", "ruins"}, - {"internet", "access"}, - {"internet", "access|wlan"}, - {"landuse", "cemetery"}, - {"leisure", "garden"}, - {"leisure", "pitch"}, - {"leisure", "playground"}, - {"leisure", "sports_centre"}, - {"leisure", "stadium"}, - {"leisure", "swimming_pool"}, - {"natural", "peak"}, - {"natural", "spring"}, - {"natural", "waterfall"}, - {"office", "company"}, - {"office", "estate_agent"}, - {"office", "government"}, - {"office", "lawyer"}, - {"office", "telecommunication"}, - {"place", "farm"}, - {"place", "hamlet"}, - {"place", "village"}, - {"railway", "halt"}, - {"railway", "station"}, - {"railway", "subway_entrance"}, - {"railway", "tram_stop"}, - {"shop", "alcohol"}, - {"shop", "bakery"}, - {"shop", "beauty"}, - {"shop", "beverages"}, - {"shop", "bicycle"}, - {"shop", "books"}, - {"shop", "butcher"}, - {"shop", "car"}, - {"shop", "car_repair"}, - {"shop", "chemist"}, - {"shop", "clothes"}, - {"shop", "computer"}, - {"shop", "confectionery"}, - {"shop", "convenience"}, - {"shop", "department_store"}, - {"shop", "doityourself"}, - {"shop", "electronics"}, - {"shop", "florist"}, - {"shop", "furniture"}, - {"shop", "garden_centre"}, - {"shop", "gift"}, - {"shop", "greengrocer"}, - {"shop", "hairdresser"}, - {"shop", "hardware"}, - {"shop", "jewelry"}, - {"shop", "kiosk"}, - {"shop", "laundry"}, - {"shop", "mall"}, - {"shop", "mobile_phone"}, - {"shop", "optician"}, - {"shop", "shoes"}, - {"shop", "sports"}, - {"shop", "supermarket"}, - {"shop", "toys"}, - {"tourism", "alpine_hut"}, - {"tourism", "artwork"}, - {"tourism", "attraction"}, - {"tourism", "camp_site"}, - {"tourism", "caravan_site"}, - {"tourism", "guest_house"}, - {"tourism", "hostel"}, - {"tourism", "hotel"}, - {"tourism", "information"}, - {"tourism", "motel"}, - {"tourism", "museum"}, - {"tourism", "picnic_site"}, - {"tourism", "viewpoint"}, - {"waterway", "waterfall"}}; +static unordered_set const gEditableTypes = { + {"aeroway-aerodrome"}, + {"aeroway-airport"}, + {"amenity-atm"}, + {"amenity-bank"}, + {"amenity-bar"}, + {"amenity-bbq"}, + {"amenity-bench"}, + {"amenity-bicycle_rental"}, + {"amenity-bureau_de_change"}, + {"amenity-bus_station"}, + {"amenity-cafe"}, + {"amenity-car_rental"}, + {"amenity-car_sharing"}, + {"amenity-casino"}, + {"amenity-cinema"}, + {"amenity-college"}, + {"amenity-doctors"}, + {"amenity-drinking_water"}, + {"amenity-embassy"}, + {"amenity-fast_food"}, + {"amenity-ferry_terminal"}, + {"amenity-fire_station"}, + {"amenity-fountain"}, + {"amenity-fuel"}, + {"amenity-grave_yard"}, + {"amenity-hospital"}, + {"amenity-hunting_stand"}, + {"amenity-kindergarten"}, + {"amenity-library"}, + {"amenity-marketplace"}, + {"amenity-nightclub"}, + {"amenity-parking"}, + {"amenity-pharmacy"}, + {"amenity-place_of_worship"}, + {"amenity-police"}, + {"amenity-post_box"}, + {"amenity-post_office"}, + {"amenity-pub"}, + {"amenity-recycling"}, + {"amenity-restaurant"}, + {"amenity-school"}, + {"amenity-shelter"}, + {"amenity-taxi"}, + {"amenity-telephone"}, + {"amenity-theatre"}, + {"amenity-toilets"}, + {"amenity-townhall"}, + {"amenity-university"}, + {"amenity-waste_disposal"}, + {"highway-bus_stop"}, + {"highway-speed_camera"}, + {"historic-archaeological_site"}, + {"historic-castle"}, + {"historic-memorial"}, + {"historic-monument"}, + {"historic-ruins"}, + {"internet-access"}, + {"internet-access|wlan"}, + {"landuse-cemetery"}, + {"leisure-garden"}, + {"leisure-pitch"}, + {"leisure-playground"}, + {"leisure-sports_centre"}, + {"leisure-stadium"}, + {"leisure-swimming_pool"}, + {"natural-peak"}, + {"natural-spring"}, + {"natural-waterfall"}, + {"office-company"}, + {"office-estate_agent"}, + {"office-government"}, + {"office-lawyer"}, + {"office-telecommunication"}, + {"place-farm"}, + {"place-hamlet"}, + {"place-village"}, + {"railway-halt"}, + {"railway-station"}, + {"railway-subway_entrance"}, + {"railway-tram_stop"}, + {"shop-alcohol"}, + {"shop-bakery"}, + {"shop-beauty"}, + {"shop-beverages"}, + {"shop-bicycle"}, + {"shop-books"}, + {"shop-butcher"}, + {"shop-car"}, + {"shop-car_repair"}, + {"shop-chemist"}, + {"shop-clothes"}, + {"shop-computer"}, + {"shop-confectionery"}, + {"shop-convenience"}, + {"shop-department_store"}, + {"shop-doityourself"}, + {"shop-electronics"}, + {"shop-florist"}, + {"shop-furniture"}, + {"shop-garden_centre"}, + {"shop-gift"}, + {"shop-greengrocer"}, + {"shop-hairdresser"}, + {"shop-hardware"}, + {"shop-jewelry"}, + {"shop-kiosk"}, + {"shop-laundry"}, + {"shop-mall"}, + {"shop-mobile_phone"}, + {"shop-optician"}, + {"shop-shoes"}, + {"shop-sports"}, + {"shop-supermarket"}, + {"shop-toys"}, + {"tourism-alpine_hut"}, + {"tourism-artwork"}, + {"tourism-attraction"}, + {"tourism-camp_site"}, + {"tourism-caravan_site"}, + {"tourism-guest_house"}, + {"tourism-hostel"}, + {"tourism-hotel"}, + {"tourism-information"}, + {"tourism-motel"}, + {"tourism-museum"}, + {"tourism-picnic_site"}, + {"tourism-viewpoint"}, + {"waterway-waterfall"}}; -TStringPair MapsmeTypeToPair(string const & mapseType) +template +bool HasAtLeastOneEditableType(TIterator from, TIterator const to) { - vector result(strings::SimpleTokenizer(mapseType, "|"), strings::SimpleTokenizer()); - // TODO(mgsergio): Support only one type in the first version. - ASSERT_EQUAL(result.size(), 2, ("Too many parts in type string.")); - return {result[0], result[1]}; -} - -bool IsTypeTag(TStringPair const & tag) -{ - return gConvertibleTypepairs.find(tag) != end(gConvertibleTypepairs); + while (from != to) + { + auto const & type = classif().GetReadableObjectName(*from++); + if (gEditableTypes.find(type) != end(gEditableTypes)) + return true; + } + return false; } } // namespace @@ -232,24 +231,26 @@ void Editor::LoadMapEdits() for (auto const & section : sections) { - for (xml_node node : mwm.child(section.second).children("node")) + for (auto const nodeOrWay : mwm.child(section.second).select_nodes("node|way")) { try { - XMLFeature const xml(node); + XMLFeature const xml(nodeOrWay.node()); FeatureID const fid(id, xml.GetOffset()); FeatureTypeInfo fti; - if (xml.GetGeomType() != DebugPrint(feature::GEOM_POINT)) - { - // TODO(mgsergio): Check if feature can be read. - fti.m_feature = m_featureLoaderFn(fid); - fti.m_feature.ApplyPatch(xml); - } - else - { - fti.m_feature = FeatureType::FromXML(xml); - } + /// TODO(mgsergio): uncomment when feature creating will + /// be required + // if (xml.GetType() != XMLFeature::Type::Way) + // { + // TODO(mgsergio): Check if feature can be read. + fti.m_feature = m_featureLoaderFn(fid); + fti.m_feature.ApplyPatch(xml); + // } + // else + // { + // fti.m_feature = FeatureType::FromXML(xml); + // } fti.m_feature.SetID(fid); @@ -260,16 +261,15 @@ void Editor::LoadMapEdits() fti.m_uploadError = xml.GetUploadError(); fti.m_status = section.first; - /// Install edited feature back to editor. - /// This instruction should be after calling m_featureLoaderFn cause - /// it'll try to return feature from editor first and if - /// it can't read it from disk. + /// Call to m_featureLoaderFn indirectly tries to load feature by + /// it's ID from the editor's m_features. + /// That's why insertion into m_features should go AFTER call to m_featureLoaderFn. m_features[id][fid.m_index] = fti; } catch (editor::XMLFeatureError const & ex) { ostringstream s; - node.print(s, " "); + nodeOrWay.node().print(s, " "); LOG(LERROR, (ex.what(), "Can't create XMLFeature in section", section.second, s.str())); } } // for nodes @@ -314,10 +314,10 @@ void Editor::Save(string const & fullFilePath) const } switch (fti.m_status) { - case FeatureStatus::Deleted: VERIFY(xf.AttachToParentNode(deleted), ()); break; - case FeatureStatus::Modified: VERIFY(xf.AttachToParentNode(modified), ()); break; - case FeatureStatus::Created: VERIFY(xf.AttachToParentNode(created), ()); break; - case FeatureStatus::Untouched: CHECK(false, ("Not edited features shouldn't be here.")); + case FeatureStatus::Deleted: VERIFY(xf.AttachToParentNode(deleted), ()); break; + case FeatureStatus::Modified: VERIFY(xf.AttachToParentNode(modified), ()); break; + case FeatureStatus::Created: VERIFY(xf.AttachToParentNode(created), ()); break; + case FeatureStatus::Untouched: CHECK(false, ("Not edited features shouldn't be here.")); } } } @@ -444,37 +444,12 @@ bool Editor::GetEditedFeature(MwmSet::MwmId const & mwmId, uint32_t offset, Feat vector Editor::EditableMetadataForType(FeatureType const & feature) const { // TODO(mgsergio): Load editable fields into memory from XML and query them here. - TTypes types; feature.ForEachType([&types](uint32_t type) { types.push_back(type); }); - // TODO(mgsergio): Only one-typed features are now supported. - if (types.size() != 1) - return {}; // Enable opening hours for the first release. - auto const & tags = GetTagsForType(types[0]); - if (tags.size() == 1 && IsTypeTag(tags[0])) + if (HasAtLeastOneEditableType(begin(types), end(types))) return {Metadata::FMD_OPEN_HOURS}; return {}; } - -Editor::TTypes Editor::GetTypesOfFeature(editor::XMLFeature const & xmlFeature) const -{ - TTags typeTags; - xmlFeature.ForEachTag([&typeTags](string const & key, string const & value) - { - if (IsTypeTag({key, value})) - typeTags.emplace_back(key, value); - }); - ASSERT_GREATER(typeTags.size(), 0, ("Cannot determine type. Please review your code.")); - // TODO(mgsergio): Only one-typed features are supported in the first version. - ASSERT_EQUAL(typeTags.size(), 1, ("Too many types for a feature.")); - return {classif().GetTypeByPathSafe({typeTags[0].first, typeTags[0].second})}; -} - -Editor::TTags Editor::GetTagsForType(uint32_t type) const -{ - // TODO(mgsergio): Only one-typed features are supported in the first version. - return {MapsmeTypeToPair(classif().GetFullObjectName(type))}; -} } // namespace osm diff --git a/indexer/osm_editor.hpp b/indexer/osm_editor.hpp index 82f1b58387..d81713e709 100644 --- a/indexer/osm_editor.hpp +++ b/indexer/osm_editor.hpp @@ -21,8 +21,6 @@ class Index; namespace osm { -using TStringPair = pair; - class Editor final { Editor() = default; @@ -41,7 +39,6 @@ public: }; using TTypes = vector; - using TTags = vector; static Editor & Instance(); @@ -77,10 +74,6 @@ public: vector EditableMetadataForType(FeatureType const & feature) const; - TTypes GetTypesOfFeature(editor::XMLFeature const & xmlFeature) const; - - TTags GetTagsForType(uint32_t type) const; - private: // TODO(AlexZ): Synchronize Save call/make it on a separate thread. void Save(string const & fullFilePath) const; @@ -110,12 +103,12 @@ private: inline string DebugPrint(Editor::FeatureStatus fs) { - switch(fs) + switch (fs) { - case Editor::FeatureStatus::Untouched: return "Untouched"; - case Editor::FeatureStatus::Deleted: return "Deleted"; - case Editor::FeatureStatus::Modified: return "Modified"; - case Editor::FeatureStatus::Created: return "Created"; + case Editor::FeatureStatus::Untouched: return "Untouched"; + case Editor::FeatureStatus::Deleted: return "Deleted"; + case Editor::FeatureStatus::Modified: return "Modified"; + case Editor::FeatureStatus::Created: return "Created"; }; } } // namespace osm