diff --git a/map/countries.cpp b/map/countries.cpp new file mode 100644 index 0000000000..0f14909047 --- /dev/null +++ b/map/countries.cpp @@ -0,0 +1,56 @@ +#include "countries.hpp" +#include "simple_tree.hpp" + +#include "../base/string_utils.hpp" + +#include "../geometry/cellid.hpp" + +#include "../std/iostream.hpp" + +typedef m2::CellId<9> CountryCellId; + +namespace mapinfo +{ + void CountryTreeNode::AddCellId(string const & cellId) + { + CountryCellId id = CountryCellId::FromString(cellId); + m_ids.push_back(static_cast(id.ToBitsAndLevel().first)); + ASSERT_EQUAL(CountryCellId::FromBitsAndLevel(m_ids.back(), 8).ToString(), cellId, ()); + } + + bool LoadCountries(SimpleTree & tree, std::istream & stream) + { + std::string line; + CountryTreeNode * value = &tree.Value(); + while (stream.good()) + { + std::getline(stream, line); + if (line.empty()) + continue; + + // calculate spaces - depth inside the tree + int spaces = 0; + for (size_t i = 0; i < line.size(); ++i) + { + if (line[i] == ' ') + ++spaces; + else + break; + } + switch (spaces) + { + case 0: // this is value for current node + value->AddCellId(line);; + break; + case 1: // country group + case 2: // country name + case 3: // region + value = &tree.AddAtDepth(spaces - 1, CountryTreeNode(line.substr(spaces))); + break; + default: + return false; + } + } + return true; + } +} diff --git a/map/countries.hpp b/map/countries.hpp new file mode 100644 index 0000000000..18dcd02e88 --- /dev/null +++ b/map/countries.hpp @@ -0,0 +1,33 @@ +#pragma once + +#include "../std/string.hpp" +#include "../std/vector.hpp" + +template class SimpleTree; + +namespace mapinfo +{ + /// used in SimpleTree when loading countries.txt + class CountryTreeNode + { + /// group, country or region + string m_name; + /// cell ids for the country + vector m_ids; + + public: + CountryTreeNode(string const & name = string()) : m_name(name) {} + + bool operator<(CountryTreeNode const & other) const + { + return m_name < other.m_name; + } + + string const & Name() const { return m_name; } + + void AddCellId(string const & cellId); + }; + + /// loads tree from given stream + bool LoadCountries(SimpleTree & tree, std::istream & stream); +} diff --git a/map/map.pro b/map/map.pro index c443c5b8c6..2ca7621717 100644 --- a/map/map.pro +++ b/map/map.pro @@ -21,6 +21,8 @@ HEADERS += \ storage.hpp \ render_queue.hpp \ render_queue_routine.hpp \ + simple_tree.hpp \ + countries.hpp \ SOURCES += \ feature_vec_model.cpp \ @@ -30,7 +32,8 @@ SOURCES += \ draw_processor.cpp \ storage.cpp \ render_queue.cpp \ - render_queue_routine.cpp + render_queue_routine.cpp \ + countries.cpp \ !iphonesimulator-g++42 { diff --git a/map/map_tests/map_tests.pro b/map/map_tests/map_tests.pro index d3772e7a03..9ac9bba602 100644 --- a/map/map_tests/map_tests.pro +++ b/map/map_tests/map_tests.pro @@ -20,3 +20,4 @@ SOURCES += \ ../../testing/testingmain.cpp \ navigator_test.cpp \ map_foreach_test.cpp \ + simple_tree_test.cpp \ diff --git a/map/map_tests/simple_tree_test.cpp b/map/map_tests/simple_tree_test.cpp new file mode 100644 index 0000000000..1b37c6c60b --- /dev/null +++ b/map/map_tests/simple_tree_test.cpp @@ -0,0 +1,50 @@ +#include "../../testing/testing.hpp" + +#include "../../platform/platform.hpp" + +#include "../../std/fstream.hpp" + +#include "../simple_tree.hpp" +#include "../countries.hpp" + +using namespace mapinfo; + +template +struct Calculator +{ + size_t count; + Calculator() : count(0) {} + void operator()(TNode const &) + { + ++count; + } +}; + +UNIT_TEST(SimpleTree) +{ + typedef SimpleTree mytree; + mytree tree; + std::ifstream file(GetPlatform().WritablePathForFile("countries.txt").c_str()); + TEST( LoadCountries(tree, file), ("countries.txt is absent or corrupted")); + + tree.Sort(); + + Calculator c1; + tree.ForEachSibling(c1); + TEST_GREATER_OR_EQUAL(c1.count, 6, ("At least 6 continents should be present")); + + // find europe + size_t i = 0; + for (; i < tree.SiblingsCount(); ++i) + if (tree[i].Value().Name() == "Europe") + break; + TEST_LESS(i, tree.SiblingsCount(), ("Europe is not present?")); + + Calculator c2; + tree[i].ForEachSibling(c2); + TEST_GREATER_OR_EQUAL(c2.count, 20, ("At least 20 countries should be present in Europe")); + + Calculator c3; + tree[i].ForEachChildren(c3); + TEST_GREATER(c3.count, c2.count, ("Europe should contain some regions")); +} diff --git a/map/simple_tree.hpp b/map/simple_tree.hpp new file mode 100644 index 0000000000..20bcb158bb --- /dev/null +++ b/map/simple_tree.hpp @@ -0,0 +1,103 @@ +#pragma once + +#include "../std/vector.hpp" + +template +class SimpleTree +{ + typedef std::vector > internal_container_type; + + T m_value; + internal_container_type m_siblings; + +public: + SimpleTree(T const & value = T()) : m_value(value) + { + } + + /// @return reference is valid only up to the next tree structure modification + T const & Value() const + { + return m_value; + } + + /// @return reference is valid only up to the next tree structure modification + T & Value() + { + return m_value; + } + + /// @return reference is valid only up to the next tree structure modification + T & AddAtDepth(int level, T const & value) + { + SimpleTree * node = this; + while (level-- > 0 && !node->m_siblings.empty()) + node = &node->m_siblings.back(); + return node->Add(value); + } + + /// @return reference is valid only up to the next tree structure modification + T & Add(T const & value) + { + m_siblings.push_back(SimpleTree(value)); + return m_siblings.back().Value(); + } + + bool operator<(SimpleTree const & other) const + { + return Value() < other.Value(); + } + + /// sorts siblings independently on each level by default + void Sort(bool onlySiblings = false) + { + std::sort(m_siblings.begin(), m_siblings.end()); + if (!onlySiblings) + for (typename internal_container_type::iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) + it->Sort(false); + } + + SimpleTree const & operator[](size_t index) const + { + return m_siblings.at(index); + } + + size_t SiblingsCount() const + { + return m_siblings.size(); + } + + template + void ForEachSibling(TFunctor & f) + { + for (typename internal_container_type::iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) + f(*it); + } + + template + void ForEachSibling(TFunctor & f) const + { + for (typename internal_container_type::const_iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) + f(*it); + } + + template + void ForEachChildren(TFunctor & f) + { + for (typename internal_container_type::iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) + { + it->ForEachChildren(f); + f(*it); + } + } + + template + void ForEachChildren(TFunctor & f) const + { + for (typename internal_container_type::const_iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) + { + it->ForEachChildren(f); + f(*it); + } + } +};