[generator:regions] Add place label nodes

This commit is contained in:
Anatoly Serdtcev 2019-06-20 18:39:59 +03:00 committed by Maksim Andrianov
parent 33e5054401
commit 74d68e730b
12 changed files with 154 additions and 52 deletions

View file

@ -144,25 +144,20 @@ void BuildTestData(std::vector<OsmElementData> const & testData,
}
}
class ToLabelingStringPolicy : public ToStringPolicyInterface
std::string ToLabelingString(vector<Node::Ptr> const & path)
{
public:
// ToStringPolicyInterface overrides:
std::string ToString(vector<Node::Ptr> const & path) const override
{
CHECK(!path.empty(), ());
std::stringstream stream;
auto i = path.begin();
auto const & countryName = (*i)->GetData().GetName();
CHECK(!countryName.empty(), ());
stream << (*i)->GetData().GetName();
++i;
for (; i != path.end(); ++i)
stream << ", " << GetLabel((*i)->GetData().GetLevel()) << ": " << (*i)->GetData().GetName();
CHECK(!path.empty(), ());
std::stringstream stream;
auto i = path.begin();
auto const & countryName = (*i)->GetData().GetName();
CHECK(!countryName.empty(), ());
stream << (*i)->GetData().GetName();
++i;
for (; i != path.end(); ++i)
stream << ", " << GetLabel((*i)->GetData().GetLevel()) << ": " << (*i)->GetData().GetName();
return stream.str();
}
};
return stream.str();
}
std::vector<std::string> GenerateTestRegions(std::vector<OsmElementData> const & testData)
{
@ -176,13 +171,13 @@ std::vector<std::string> GenerateTestRegions(std::vector<OsmElementData> const &
RegionInfo collector(filename);
BuildTestData(testData, regions, placePointsMap, collector);
RegionsBuilder builder(std::move(regions));
RegionsBuilder builder(std::move(regions), {});
std::vector<std::string> kvRegions;
builder.ForEachCountry([&](std::string const & name, Node::PtrList const & outers) {
for (auto const & tree : outers)
{
ForEachLevelPath(tree, [&] (std::vector<Node::Ptr> const & path) {
kvRegions.push_back(ToLabelingStringPolicy{}.ToString(path));
kvRegions.push_back(ToLabelingString(path));
});
}
});
@ -344,7 +339,7 @@ UNIT_TEST(RegionsBuilderTest_GetCountryNames)
{
auto const filename = MakeCollectorData();
RegionInfo collector(filename);
RegionsBuilder builder(MakeTestDataSet1(collector));
RegionsBuilder builder(MakeTestDataSet1(collector), {} /* placePointsMap */);
auto const & countryNames = builder.GetCountryNames();
TEST_EQUAL(countryNames.size(), 2, ());
TEST(std::count(std::begin(countryNames), std::end(countryNames), "Country_1"), ());
@ -355,7 +350,7 @@ UNIT_TEST(RegionsBuilderTest_GetCountries)
{
auto const filename = MakeCollectorData();
RegionInfo collector(filename);
RegionsBuilder builder(MakeTestDataSet1(collector));
RegionsBuilder builder(MakeTestDataSet1(collector), {} /* placePointsMap */);
auto const & countries = builder.GetCountriesOuters();
TEST_EQUAL(countries.size(), 3, ());
TEST_EQUAL(std::count_if(std::begin(countries), std::end(countries),
@ -371,7 +366,7 @@ UNIT_TEST(RegionsBuilderTest_GetCountryTrees)
auto const filename = MakeCollectorData();
RegionInfo collector(filename);
std::vector<std::string> bankOfNames;
RegionsBuilder builder(MakeTestDataSet1(collector));
RegionsBuilder builder(MakeTestDataSet1(collector), {} /* placePointsMap */);
builder.ForEachCountry([&](std::string const & name, Node::PtrList const & outers) {
for (auto const & tree : outers)
{

View file

@ -97,22 +97,29 @@ void CollectorRegionInfo::FillRegionData(base::GeoObjectId const & osmId, OsmEle
{
rd.m_osmId = osmId;
rd.m_place = EncodePlaceType(el.GetTag("place"));
auto const al = el.GetTag("admin_level");
if (al.empty())
return;
try
auto const al = el.GetTag("admin_level");
if (!al.empty())
{
auto const adminLevel = std::stoi(al);
// Administrative level is in the range [1 ... 12].
// https://wiki.openstreetmap.org/wiki/Tag:boundary=administrative
rd.m_adminLevel = (adminLevel >= 1 && adminLevel <= 12) ?
static_cast<AdminLevel>(adminLevel) : AdminLevel::Unknown;
try
{
auto const adminLevel = std::stoi(al);
// Administrative level is in the range [1 ... 12].
// https://wiki.openstreetmap.org/wiki/Tag:boundary=administrative
rd.m_adminLevel = (adminLevel >= 1 && adminLevel <= 12) ?
static_cast<AdminLevel>(adminLevel) : AdminLevel::Unknown;
}
catch (std::exception const & e) // std::invalid_argument, std::out_of_range
{
LOG(LWARNING, (e.what()));
rd.m_adminLevel = AdminLevel::Unknown;
}
}
catch (std::exception const & e) // std::invalid_argument, std::out_of_range
for (auto const & member : el.Members())
{
LOG(::base::LWARNING, (e.what()));
rd.m_adminLevel = AdminLevel::Unknown;
if (member.m_role == "label" && member.m_type == OsmElement::EntityType::Node)
rd.m_labelOsmId = base::MakeOsmNode(member.m_ref);
}
}

View file

@ -103,6 +103,7 @@ struct RegionData
base::GeoObjectId m_osmId;
AdminLevel m_adminLevel = AdminLevel::Unknown;
PlaceType m_place = PlaceType::Unknown;
base::GeoObjectId m_labelOsmId;
};
using MapRegionData = std::unordered_map<base::GeoObjectId, RegionData>;

View file

@ -50,12 +50,63 @@ Region::Region(PlacePoint const & place)
, RegionWithData(place.GetRegionData())
, m_polygon(std::make_shared<BoostPolygon>())
{
CHECK_NOT_EQUAL(place.GetPlaceType(), PlaceType::Unknown, ());
auto const radius = GetRadiusByPlaceType(place.GetPlaceType());
*m_polygon = MakePolygonWithRadius(place.GetPosition(), radius);
boost::geometry::envelope(*m_polygon, m_rect);
m_area = boost::geometry::area(*m_polygon);
}
std::string Region::GetEnglishOrTransliteratedName() const
{
if (m_placeLabel)
return m_placeLabel->GetEnglishOrTransliteratedName();
return RegionWithName::GetEnglishOrTransliteratedName();
}
std::string Region::GetName(int8_t lang) const
{
if (m_placeLabel)
return m_placeLabel->GetName();
return RegionWithName::GetName();
}
base::GeoObjectId Region::GetId() const
{
if (m_placeLabel)
return m_placeLabel->GetId();
return RegionWithData::GetId();
}
PlaceType Region::GetPlaceType() const
{
if (m_placeLabel)
return m_placeLabel->GetPlaceType();
return RegionWithData::GetPlaceType();
}
boost::optional<std::string> Region::GetIsoCode() const
{
if (m_placeLabel)
{
if (auto && isoCode = m_placeLabel->GetIsoCode())
return isoCode;
}
return RegionWithData::GetIsoCode();
}
void Region::SetLabel(PlacePoint const & place)
{
CHECK(!m_placeLabel, ());
m_placeLabel = place;
}
// static
double Region::GetRadiusByPlaceType(PlaceType place)
{
@ -127,6 +178,9 @@ bool Region::ContainsRect(Region const & smaller) const
BoostPoint Region::GetCenter() const
{
if (m_placeLabel)
return m_placeLabel->GetPosition();
BoostPoint p;
boost::geometry::centroid(m_rect, p);
return p;

View file

@ -2,6 +2,7 @@
#include "generator/feature_builder.hpp"
#include "generator/regions/region_base.hpp"
#include "generator/regions/place_point.hpp"
#include <memory>
@ -16,17 +17,27 @@ class RegionDataProxy;
namespace regions
{
class PlacePoint;
// This is a helper class that is needed to represent the region.
// With this view, further processing is simplified.
class Region : public RegionWithName, public RegionWithData
class Region : protected RegionWithName, protected RegionWithData
{
public:
explicit Region(feature::FeatureBuilder const & fb, RegionDataProxy const & rd);
// Build a region and its boundary based on the heuristic.
explicit Region(PlacePoint const & place);
// See RegionWithName::GetEnglishOrTransliteratedName().
std::string GetEnglishOrTransliteratedName() const;
std::string GetName(int8_t lang = StringUtf8Multilang::kDefaultCode) const;
base::GeoObjectId GetId() const;
using RegionWithData::GetAdminLevel;
PlaceType GetPlaceType() const;
boost::optional<std::string> GetIsoCode() const;
using RegionWithData::GetLabelOsmId;
void SetLabel(PlacePoint const & place);
bool Contains(Region const & smaller) const;
bool ContainsRect(Region const & smaller) const;
bool Contains(PlacePoint const & place) const;
@ -44,6 +55,7 @@ public:
private:
void FillPolygon(feature::FeatureBuilder const & fb);
boost::optional<PlacePoint> m_placeLabel;
std::shared_ptr<BoostPolygon> m_polygon;
BoostRect m_rect;
double m_area;

View file

@ -63,5 +63,10 @@ boost::optional<std::string> RegionWithData::GetIsoCode() const
{
return m_regionData.GetIsoCodeAlpha2();
}
boost::optional<base::GeoObjectId> RegionWithData::GetLabelOsmId() const
{
return m_regionData.GetLabelOsmId();
}
} // namespace regions
} // namespace generator

View file

@ -50,6 +50,8 @@ public:
base::GeoObjectId GetId() const;
boost::optional<std::string> GetIsoCode() const;
boost::optional<base::GeoObjectId> GetLabelOsmId() const;
AdminLevel GetAdminLevel() const { return m_regionData.GetAdminLevel(); }
PlaceType GetPlaceType() const { return m_regionData.GetPlaceType(); }

View file

@ -114,6 +114,16 @@ PlaceType BaseRegionDataProxy<T>::GetPlaceType() const
return regionData->second.m_place;
}
template <typename T>
boost::optional<base::GeoObjectId> BaseRegionDataProxy<T>::GetLabelOsmId() const
{
auto const & labelId = GetMapRegionData().at(m_osmId).m_labelOsmId;
if (!labelId.GetEncodedId())
return {};
return labelId;
}
template <typename T>
boost::optional<std::string> BaseRegionDataProxy<T>::GetIsoCodeAlpha2() const
{

View file

@ -69,6 +69,7 @@ public:
base::GeoObjectId const & GetOsmId() const;
AdminLevel GetAdminLevel() const;
PlaceType GetPlaceType() const;
boost::optional<base::GeoObjectId> GetLabelOsmId() const;
boost::optional<std::string> GetIsoCodeAlpha2() const;
boost::optional<std::string> GetIsoCodeAlpha3() const;

View file

@ -62,10 +62,12 @@ public:
auto timer = base::Timer{};
Transliteration::Instance().Init(GetPlatform().ResourcesDir());
RegionsBuilder::Regions regions = ReadAndFixData();
RegionsBuilder builder{std::move(regions), threadsCount};
GenerateRegions(builder);
RegionsBuilder::Regions regions;
PlacePointsMap placePointsMap;
std::tie(regions, placePointsMap) = ReadDatasetFromTmpMwm(m_pathInRegionsTmpMwm, m_regionsInfoCollector);
RegionsBuilder builder{std::move(regions), std::move(placePointsMap), threadsCount};
GenerateRegions(builder);
LOG(LINFO, ("Finish generating regions.", timer.ElapsedSeconds(), "seconds."));
}
@ -222,16 +224,6 @@ private:
return std::make_tuple(std::move(regions), std::move(placePointsMap));
}
RegionsBuilder::Regions ReadAndFixData()
{
RegionsBuilder::Regions regions;
PlacePointsMap placePointsMap;
std::tie(regions, placePointsMap) =
ReadDatasetFromTmpMwm(m_pathInRegionsTmpMwm, m_regionsInfoCollector);
FixRegionsWithPlacePointApproximation(placePointsMap, regions);
return regions;
}
void RepackTmpMwm()
{
feature::FeaturesCollector featuresCollector{m_pathOutRepackedRegionsTmpMwm};

View file

@ -1,5 +1,6 @@
#include "generator/regions/regions_builder.hpp"
#include "generator/regions/regions_fixer.hpp"
#include "generator/regions/specs/rus.hpp"
#include "base/assert.hpp"
@ -20,15 +21,35 @@ namespace generator
{
namespace regions
{
RegionsBuilder::RegionsBuilder(Regions && regions, size_t threadsCount)
RegionsBuilder::RegionsBuilder(Regions && regions, PlacePointsMap && placePointsMap,
size_t threadsCount)
: m_threadsCount(threadsCount)
{
ASSERT(m_threadsCount != 0, ());
MoveLabelPlacePoints(placePointsMap, regions);
FixRegionsWithPlacePointApproximation(placePointsMap, regions);
m_regionsInAreaOrder = FormRegionsInAreaOrder(std::move(regions));
m_countriesOuters = ExtractCountriesOuters(m_regionsInAreaOrder);
}
void RegionsBuilder::MoveLabelPlacePoints(PlacePointsMap & placePointsMap, Regions & regions)
{
for (auto & region : regions)
{
if (auto labelOsmId = region.GetLabelOsmId())
{
auto label = placePointsMap.find(*labelOsmId);
if (label != placePointsMap.end())
{
region.SetLabel(label->second);
placePointsMap.erase(label);
}
}
}
}
RegionsBuilder::Regions RegionsBuilder::FormRegionsInAreaOrder(Regions && regions)
{
auto const cmp = [](Region const & l, Region const & r) { return l.GetArea() > r.GetArea(); };

View file

@ -23,7 +23,8 @@ public:
using StringsList = std::vector<std::string>;
using CountryFn = std::function<void(std::string const &, Node::PtrList const &)>;
explicit RegionsBuilder(Regions && regions, size_t threadsCount = 1);
explicit RegionsBuilder(Regions && regions, PlacePointsMap && placePointsMap,
size_t threadsCount = 1);
Regions const & GetCountriesOuters() const;
StringsList GetCountryNames() const;
@ -32,6 +33,7 @@ public:
private:
static constexpr double kAreaRelativeErrorPercent = 0.1;
void MoveLabelPlacePoints(PlacePointsMap & placePointsMap, Regions & regions);
Regions FormRegionsInAreaOrder(Regions && regions);
Regions ExtractCountriesOuters(Regions & regions);
Node::PtrList BuildCountryRegionTrees(Regions const & outers,