From 0cd90b502238a889535400b9d8344aea9a95488a Mon Sep 17 00:00:00 2001 From: Sergey Magidovich Date: Wed, 23 Dec 2015 17:10:09 +0300 Subject: [PATCH] Implement type of a FeatureType serialization. --- base/base.pro | 2 +- base/base_tests/base_tests.pro | 2 +- base/base_tests/collection_cast_test.cpp | 11 ++ base/base_tests/enumerate_test.cpp | 39 ----- base/collection_cast.hpp | 29 ++++ base/enumerate.hpp | 100 ------------ base/range_iterator.hpp | 2 + editor/ui2oh.cpp | 13 +- editor/xml_feature.hpp | 3 +- indexer/feature.cpp | 22 +-- indexer/feature.hpp | 2 +- indexer/osm_editor.cpp | 184 ++++++++++++++++++++++- indexer/osm_editor.hpp | 12 +- qt/editor_dialog.cpp | 12 +- 14 files changed, 263 insertions(+), 170 deletions(-) create mode 100644 base/base_tests/collection_cast_test.cpp delete mode 100644 base/base_tests/enumerate_test.cpp create mode 100644 base/collection_cast.hpp delete mode 100644 base/enumerate.hpp diff --git a/base/base.pro b/base/base.pro index e9dc2f3a1c..642c3f56a8 100644 --- a/base/base.pro +++ b/base/base.pro @@ -38,9 +38,9 @@ HEADERS += \ buffer_vector.hpp \ cache.hpp \ cancellable.hpp \ + collection_cast.hpp \ condition.hpp \ const_helper.hpp \ - enumerate.hpp \ exception.hpp \ internal/message.hpp \ limited_priority_queue.hpp \ diff --git a/base/base_tests/base_tests.pro b/base/base_tests/base_tests.pro index 257cca190f..28744d497c 100644 --- a/base/base_tests/base_tests.pro +++ b/base/base_tests/base_tests.pro @@ -16,10 +16,10 @@ SOURCES += \ bits_test.cpp \ buffer_vector_test.cpp \ cache_test.cpp \ + collection_cast_test.cpp \ condition_test.cpp \ const_helper.cpp \ containers_test.cpp \ - enumerate_test.cpp \ logging_test.cpp \ math_test.cpp \ matrix_test.cpp \ diff --git a/base/base_tests/collection_cast_test.cpp b/base/base_tests/collection_cast_test.cpp new file mode 100644 index 0000000000..c5ead707b5 --- /dev/null +++ b/base/base_tests/collection_cast_test.cpp @@ -0,0 +1,11 @@ +#include "testing/testing.hpp" + +#include "base/collection_cast.hpp" + +#include "std/list.hpp" +#include "std/vector.hpp" + +UNIT_TEST(collection_cast) +{ + TEST_EQUAL((list{1, 2, 3, 4, }), my::collection_cast(vector {1, 2, 3, 4}), ()); +} diff --git a/base/base_tests/enumerate_test.cpp b/base/base_tests/enumerate_test.cpp deleted file mode 100644 index 5bd0d09825..0000000000 --- a/base/base_tests/enumerate_test.cpp +++ /dev/null @@ -1,39 +0,0 @@ -#include "testing/testing.hpp" - -#include "base/enumerate.hpp" - -#include "std/map.hpp" -#include "std/vector.hpp" - -UNIT_TEST(enumerate) -{ - { - map result; - for (auto const p : my::Enumerate(std::vector{1, 2, 3})) - result.insert(p); - TEST_EQUAL(result, (map{{0, 1}, {1, 2}, {2, 3}}), ()); - } - - { - map result; - for (auto const p : my::Enumerate(std::vector{1, 2, 3}, 10)) - result.insert(p); - TEST_EQUAL(result, (map{{10, 1}, {11, 2}, {12, 3}}), ()); - } - - { - std::vector const vec{1, 2, 3}; - map result; - for (auto const p : my::Enumerate(vec)) - result.insert(p); - TEST_EQUAL(result, (map{{0, 1}, {1, 2}, {2, 3}}), ()); - } - - { - std::vector vec{1, 2, 3, 4, 5, 6}; - for (auto const p : my::Enumerate(vec, -6)) - p.item *= p.index; - - TEST_EQUAL(vec, (std::vector{-6, -10, -12, -12, -10, -6}), ()); - } -} diff --git a/base/collection_cast.hpp b/base/collection_cast.hpp new file mode 100644 index 0000000000..a05780a2eb --- /dev/null +++ b/base/collection_cast.hpp @@ -0,0 +1,29 @@ +#pragma once + +#include + +namespace my +{ +namespace details +{ +template +struct ValueType +{ + using type = typename std::remove_reference::type::value_type; +}; + +template +using ValueTypeT = typename ValueType::type; +} // namespace details + +// Use this function to cast one collection to annother. +// I.E. list const myList = collection_cast(vector{1, 2, 4, 5}); +// More examples: +// auto const mySet = collection_cast("aaabcccd"); +// auto const myMap = collection_cast(vector>{{1, 2}, {3, 4}}); +template class To, typename From> +auto collection_cast(From && from) -> To> +{ + return To>(begin(from), end(from)); +} +} // namespace my diff --git a/base/enumerate.hpp b/base/enumerate.hpp deleted file mode 100644 index 60461beb3b..0000000000 --- a/base/enumerate.hpp +++ /dev/null @@ -1,100 +0,0 @@ -#pragma once - -#include "std/iterator.hpp" -#include "std/utility.hpp" - -namespace my -{ -namespace details -{ -template -struct ConstRespectiveIterator -{ - using type = typename TCollection::iterator; -}; - -template -struct ConstRespectiveIterator -{ - using type = typename TCollection::const_iterator; -}; - -template -using ConstRespectiveIteratorT = - typename ConstRespectiveIterator::type>::type; -} // namespace details - -template -struct IndexedElement : std::pair -{ - using TBase = std::pair; - using TBase::TBase; - - TIndex const index{TBase::first}; - TElement & item{TBase::second}; -}; - -template -IndexedElement MakeIndexedElement(TElement && item, TIndex counter = {}) -{ - return IndexedElement(counter, item); -} - -template -struct EnumeratingIterator -{ - using original_iterator = TIterator; - using original_reference = typename std::iterator_traits::reference; - - using difference_type = typename std::iterator_traits::difference_type; - using value_type = IndexedElement; - using pointer = value_type *; - using reference = value_type &; - using iterator_category = std::forward_iterator_tag; - - EnumeratingIterator(original_iterator const it, TIndex const counter) - : m_iterator(it), m_counter(counter) - { - } - - EnumeratingIterator & operator++() - { - ++m_iterator; - ++m_counter; - return *this; - } - - value_type operator*() { return MakeIndexedElement(*m_iterator, m_counter); } - bool operator==(EnumeratingIterator const & it) const { return m_iterator == it.m_iterator; } - bool operator!=(EnumeratingIterator const & it) const { return !(*this == it); } - original_iterator m_iterator; - TIndex m_counter; -}; - -template -struct EnumeratorWrapper -{ - using original_iterator = TIterator; - using iterator = EnumeratingIterator; - using value_type = typename std::iterator_traits::value_type; - - EnumeratorWrapper(original_iterator const begin, original_iterator const end, - TIndex const countFrom) - : m_begin(begin), m_end(end), m_countFrom(countFrom) - { - } - - iterator begin() { return {m_begin, m_countFrom}; } - iterator end() { return {m_end, {}}; } - original_iterator const m_begin; - original_iterator const m_end; - TIndex const m_countFrom; -}; - -template -auto Enumerate(TCollection && collection, TIndex const counter = {}) - -> EnumeratorWrapper, TIndex> -{ - return {begin(collection), end(collection), counter}; -} -} // namespace my diff --git a/base/range_iterator.hpp b/base/range_iterator.hpp index 438fb88516..cfc7ff11be 100644 --- a/base/range_iterator.hpp +++ b/base/range_iterator.hpp @@ -1,3 +1,5 @@ +#pragma once + #include "std/iterator.hpp" #include "std/type_traits.hpp" diff --git a/editor/ui2oh.cpp b/editor/ui2oh.cpp index de27826ab1..a77bfce3fc 100644 --- a/editor/ui2oh.cpp +++ b/editor/ui2oh.cpp @@ -1,7 +1,5 @@ #include "editor/ui2oh.hpp" -#include "base/enumerate.hpp" - #include "std/algorithm.hpp" #include "std/array.hpp" #include "std/string.hpp" @@ -74,9 +72,9 @@ bool ConvertOpeningHours(osmoh::OpeningHours const & oh, ui::TimeTableSet & tts) if (oh.IsTwentyFourHours()) return true; - for (auto const & p : my::Enumerate(oh.GetRule())) + bool first = true; + for (auto const & rulePart : oh.GetRule()) { - auto const & rulePart = p.item; ui::TimeTable tt; if (rulePart.HasWeekdays()) @@ -87,8 +85,13 @@ bool ConvertOpeningHours(osmoh::OpeningHours const & oh, ui::TimeTableSet & tts) tt.SetTwentyFourHours(false); SetUpTimeTable(rulePart.GetTimes(), tt); } + else + { + tt.SetTwentyFourHours(true); + } - bool const appended = p.index == 0 ? tts.Replace(tt, 0) : tts.Append(tt); + bool const appended = first ? tts.Replace(tt, 0) : tts.Append(tt); + first = false; if (!appended) return false; } diff --git a/editor/xml_feature.hpp b/editor/xml_feature.hpp index 0dfb7eb65e..07d06484f9 100644 --- a/editor/xml_feature.hpp +++ b/editor/xml_feature.hpp @@ -91,7 +91,8 @@ public: template void ForEachTag(TFunc && func) const { - // TODO(mgsergio): implement me. + for (auto const & tag : GetRootNode().select_nodes("tag")) + func(tag.node().attribute("k").value(), tag.node().attribute("v").value()); } string GetTagValue(string const & key) const; diff --git a/indexer/feature.cpp b/indexer/feature.cpp index 718f21a246..b340312568 100644 --- a/indexer/feature.cpp +++ b/indexer/feature.cpp @@ -1,9 +1,11 @@ +#include "indexer/classificator.hpp" #include "indexer/feature.hpp" #include "indexer/classificator.hpp" #include "indexer/feature_algo.hpp" #include "indexer/feature_loader_base.hpp" #include "indexer/feature_visibility.hpp" +#include "indexer/osm_editor.hpp" #include "geometry/distance.hpp" #include "geometry/robust_orientation.hpp" @@ -12,6 +14,8 @@ #include "base/range_iterator.hpp" +#include "std/algorithm.hpp" + using namespace feature; /////////////////////////////////////////////////////////////////////////////////////////////////// @@ -58,12 +62,8 @@ FeatureType FeatureType::FromXML(editor::XMLFeature const & xml) // EGeomType - // for (auto const i : my::Range(feature.GetTypesCount())) - // m_types[i] = - // Does the order matter? If so what order should be? - // TODO(mgsergio): Only features with single type are currently supported. - // TODO(mgsergio): Replace hardcode with real values. - feature.m_types[0] = classif().GetTypeByPath({"amenity", "atm"}); + auto const & types = osm::Editor::Instance().GetTypesOfFeature(xml); + copy(begin(types), end(types), begin(feature.m_types)); feature.m_bTypesParsed = true; for (auto const i : my::Range(1u, static_cast(feature::Metadata::FMD_COUNT))) @@ -105,12 +105,12 @@ editor::XMLFeature FeatureType::ToXML() const // feature.m_params.layer = // feature.m_params.rank = - // for (auto const i : my::Range(feature.GetTypesCount())) - // m_types[i] = - // Does the order matter? If so what order should be? - // TODO(mgsergio): Only features with single type are currently supported. ParseTypes(); - feature.SetTagValue("amenity", "atm"); // TODO(mgsergio): Replace hardcode with real values. + 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.hpp b/indexer/feature.hpp index 693e705cfd..9100d48b1a 100644 --- a/indexer/feature.hpp +++ b/indexer/feature.hpp @@ -105,7 +105,7 @@ public: } template - void ForEachType(ToDo f) const + void ForEachType(ToDo && f) const { ParseTypes(); diff --git a/indexer/osm_editor.cpp b/indexer/osm_editor.cpp index 07d36abbc8..acead90ba2 100644 --- a/indexer/osm_editor.cpp +++ b/indexer/osm_editor.cpp @@ -8,9 +8,13 @@ #include "editor/xml_feature.hpp" #include "base/logging.hpp" +#include "base/string_utils.hpp" #include "std/map.hpp" #include "std/set.hpp" +#include "std/unordered_set.hpp" + +#include #include "3party/pugixml/src/pugixml.hpp" @@ -34,7 +38,150 @@ namespace osm namespace { +// TODO(mgsergio): Replace hard-coded value with reading from file. string GetEditorFilePath() { return GetPlatform().WritablePathForFile(kEditorXMLFileName); } +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"}}; + +TStringPair MapsmeTypeToPair(string const & mapseType) +{ + 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); +} } // namespace Editor & Editor::Instance() @@ -124,7 +271,7 @@ void Editor::Save(string const & fullFilePath) const { xml_node mwmNode = root.append_child(kXmlMwmNode); mwmNode.append_attribute("name") = mwm.first.GetInfo()->GetCountryName().c_str(); - mwmNode.append_attribute("version") = mwm.first.GetInfo()->GetVersion(); + mwmNode.append_attribute("version") = static_cast(mwm.first.GetInfo()->GetVersion()); xml_node deleted = mwmNode.append_child(kDeleteSection); xml_node modified = mwmNode.append_child(kModifySection); xml_node created = mwmNode.append_child(kCreateSection); @@ -269,11 +416,42 @@ bool Editor::GetEditedFeature(MwmSet::MwmId const & mwmId, uint32_t offset, Feat return true; } -vector Editor::EditableMetadataForType(uint32_t type) const +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 || + feature.GetFeatureType() != feature::HEADER_GEOM_POINT || + feature.GetFeatureType() != feature::HEADER_GEOM_POINT_EX) + return {}; + // Enable opening hours for the first release. - return {Metadata::FMD_OPEN_HOURS}; + auto const & tags = GetTagsForType(types[0]); + if (tags.size() == 1 && IsTypeTag(tags[0])) + 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 bcb0b02c7c..20dbc97ca9 100644 --- a/indexer/osm_editor.hpp +++ b/indexer/osm_editor.hpp @@ -5,6 +5,8 @@ #include "indexer/feature_meta.hpp" #include "indexer/mwm_set.hpp" +#include "editor/xml_feature.hpp" + #include "std/ctime.hpp" #include "std/function.hpp" #include "std/map.hpp" @@ -17,6 +19,7 @@ class Index; namespace osm { +using TStringPair = pair; class Editor final { @@ -34,6 +37,9 @@ public: ECreated }; + using TTypes = vector; + using TTags = vector; + static Editor & Instance(); void SetMwmIdByNameAndVersionFn(TMwmIdByMapNameFn const & fn) { m_mwmIdByMapNameFn = fn; } @@ -65,7 +71,11 @@ public: /// Original feature with same FeatureID as newFeature is replaced by newFeature. void EditFeature(FeatureType & editedFeature); - vector EditableMetadataForType(uint32_t type) const; + 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. diff --git a/qt/editor_dialog.cpp b/qt/editor_dialog.cpp index 07241787a5..0c57625de2 100644 --- a/qt/editor_dialog.cpp +++ b/qt/editor_dialog.cpp @@ -6,6 +6,8 @@ #include "indexer/feature.hpp" #include "indexer/osm_editor.hpp" +#include "base/collection_cast.hpp" + #include "std/set.hpp" #include "std/vector.hpp" @@ -65,13 +67,9 @@ EditorDialog::EditorDialog(QWidget * parent, FeatureType const & feature) : QDia // All metadata rows. QVBoxLayout * metaRows = new QVBoxLayout(); // TODO(mgsergio): Load editable fields from metadata. Features can have several types, so we merge all editable fields here. - set editableMetadataFields; - // Merge editable fields for all feature's types. - feature.ForEachType([&editableMetadataFields](uint32_t type) - { - auto const editableFields = osm::Editor::Instance().EditableMetadataForType(type); - editableMetadataFields.insert(editableFields.begin(), editableFields.end()); - }); + set const editableMetadataFields = + my::collection_cast(osm::Editor::Instance().EditableMetadataForType(feature)); + for (Metadata::EType const field : editableMetadataFields) { QString const fieldName = QString::fromStdString(DebugPrint(field));