Review fixes

This commit is contained in:
Maksim Andrianov 2020-04-13 13:14:18 +03:00 committed by mpimenov
parent 2e6b81f0d0
commit b73da24e1f
5 changed files with 107 additions and 145 deletions

View file

@ -20,50 +20,53 @@ namespace
{
using namespace feature;
template<typename T>
template <typename T>
struct RemoveCvref
{
typedef std::remove_cv_t<std::remove_reference_t<T>> type;
};
template<typename T>
template <typename T>
using RemoveCvrefT = typename RemoveCvref<T>::type;
template<typename T>
template <typename T>
m2::RectD GetLimitRect(T && t)
{
using Type = RemoveCvrefT<T>;
if constexpr(std::is_same_v<Type, FeatureBuilder>)
return t.GetLimitRect();
else if constexpr(std::is_same_v<Type, m2::PointD>)
if constexpr(std::is_same_v<Type, m2::PointD>)
return m2::RectD(t, t);
else
UNREACHABLE();
UNREACHABLE();
}
template<typename T, typename F>
bool ForAnyGeometryPoint(T && t, F && f)
template <typename T, typename F>
bool ForAnyPoint(T && t, F && f)
{
using Type = RemoveCvrefT<T>;
if constexpr(std::is_same_v<Type, FeatureBuilder>)
return t.ForAnyGeometryPoint(f);
return t.ForAnyPoint(std::forward<F>(f));
if constexpr(std::is_same_v<Type, m2::PointD>)
return f(std::forward<T>(t));
UNREACHABLE();
}
template <typename T, typename F>
void ForEachPoint(T && t, F && f)
{
using Type = RemoveCvrefT<T>;
if constexpr(std::is_same_v<Type, FeatureBuilder>)
t.ForEachPoint(std::forward<F>(f));
else if constexpr(std::is_same_v<Type, m2::PointD>)
return f(t);
f(std::forward<T>(t));
else
UNREACHABLE();
}
template<typename T, typename F>
void ForEachGeometryPoint(T && t, F && f)
{
ForAnyGeometryPoint(std::forward<T>(t), [&](auto const & p) {
f(p);
return false;
});
}
// An implementation for CountriesFilesAffiliation class.
template<typename T>
template <typename T>
std::vector<std::string> GetAffiliations(T && t,
borders::CountryPolygonsCollection const & countryPolygonsTree,
bool haveBordersForWholeWorld)
@ -85,7 +88,7 @@ std::vector<std::string> GetAffiliations(T && t,
for (borders::CountryPolygons const & countryPolygons : countriesContainer)
{
auto const need = ForAnyGeometryPoint(t, [&](auto const & point) {
auto const need = ForAnyPoint(t, [&](auto const & point) {
return countryPolygons.Contains(point);
});
@ -125,12 +128,12 @@ std::optional<std::string> IsOneCountryForLimitRect(m2::RectD const & limitRect,
return country ? country->GetName() : std::optional<std::string>{};
}
template<typename T>
template <typename T>
std::vector<std::string> GetHonestAffiliations(T && t, IndexSharedPtr const & index)
{
std::vector<std::string> affiliations;
std::unordered_set<borders::CountryPolygons const *> countires;
ForEachGeometryPoint(t, [&](auto const & point) {
ForEachPoint(t, [&](auto const & point) {
std::vector<CountriesFilesIndexAffiliation::Value> values;
boost::geometry::index::query(*index, boost::geometry::index::covers(point),
std::back_inserter(values));
@ -156,7 +159,7 @@ std::vector<std::string> GetHonestAffiliations(T && t, IndexSharedPtr const & in
return affiliations;
}
template<typename T>
template <typename T>
std::vector<std::string> GetAffiliations(T && t, IndexSharedPtr const & index)
{
auto const oneCountry = IsOneCountryForLimitRect(GetLimitRect(t), index);
@ -195,21 +198,21 @@ CountriesFilesIndexAffiliation::CountriesFilesIndexAffiliation(std::string const
static std::mutex cacheMutex;
static std::unordered_map<std::string, std::shared_ptr<Tree>> cache;
auto const key = borderPath + std::to_string(haveBordersForWholeWorld);
std::lock_guard<std::mutex> lock(cacheMutex);
auto const it = cache.find(key);
if (it != std::cend(cache))
{
std::lock_guard<std::mutex> lock(cacheMutex);
auto const it = cache.find(key);
if (it != std::cend(cache))
{
m_index = it->second;
return;
}
m_index = it->second;
return;
}
auto const net = generator::cells_merger::MakeNet(0.2 /* step */,
mercator::Bounds::kMinX, mercator::Bounds::kMinY,
mercator::Bounds::kMaxX, mercator::Bounds::kMaxY);
auto const index = BuildIndex(net);
m_index = index;
std::lock_guard<std::mutex> lock(cacheMutex);
cache.emplace(key, index);
}

View file

@ -28,63 +28,34 @@ CoastlineFeaturesGenerator::CoastlineFeaturesGenerator()
namespace
{
m2::RectD GetLimitRect(RegionT const & rgn)
{
RectT r = rgn.GetRect();
return m2::RectD(r.minX(), r.minY(), r.maxX(), r.maxY());
}
m2::RectD GetLimitRect(RegionT const & rgn)
{
RectT r = rgn.GetRect();
return m2::RectD(r.minX(), r.minY(), r.maxX(), r.maxY());
}
inline PointT D2I(m2::PointD const & p)
{
m2::PointU const pu = PointDToPointU(p, kPointCoordBits);
return PointT(static_cast<int32_t>(pu.x), static_cast<int32_t>(pu.y));
}
template <class Tree> class DoCreateRegion
{
Tree & m_tree;
RegionT m_rgn;
m2::PointD m_pt;
bool m_exist;
public:
template <typename T>
DoCreateRegion(T && tree) : m_tree(std::forward<T>(tree)), m_exist(false) {}
bool operator()(m2::PointD const & p)
{
// This logic is to skip last polygon point (that is equal to first).
if (m_exist)
{
// add previous point to region
m_rgn.AddPoint(D2I(m_pt));
}
else
m_exist = true;
// save current point
m_pt = p;
return true;
}
void EndRegion()
{
m_tree.Add(m_rgn, GetLimitRect(m_rgn));
m_rgn = RegionT();
m_exist = false;
}
};
inline PointT D2I(m2::PointD const & p)
{
m2::PointU const pu = PointDToPointU(p, kPointCoordBits);
return PointT(static_cast<int32_t>(pu.x), static_cast<int32_t>(pu.y));
}
} // namespace
void CoastlineFeaturesGenerator::AddRegionToTree(FeatureBuilder const & fb)
{
ASSERT ( fb.IsGeometryClosed(), () );
ASSERT(fb.IsGeometryClosed(), ());
DoCreateRegion<TTree> createRgn(m_tree);
fb.ForEachGeometryPointEx(createRgn);
fb.ForEachPolygon([&](auto const & polygon) {
if (polygon.empty())
return;
RegionT rgn;
for (auto it = std::next(std::cbegin(polygon)); it != std::cend(polygon); ++it)
rgn.AddPoint(D2I(*it));
auto const limitRect = GetLimitRect(rgn);
m_tree.Add(std::move(rgn), limitRect);
});
}
void CoastlineFeaturesGenerator::Process(FeatureBuilder const & fb)

View file

@ -70,7 +70,7 @@ public:
size_t GetTypesCount() const { return m_params.m_types.size(); }
template <class ToDo>
void ForEachGeometryPointEx(ToDo && toDo) const
void ForEachPoint(ToDo && toDo) const
{
if (IsPoint())
{
@ -78,48 +78,34 @@ public:
}
else
{
for (PointSeq const & points : m_polygons)
for (auto const & points : m_polygons)
{
for (auto const & pt : points)
{
if (!toDo(pt))
return;
}
toDo.EndRegion();
toDo(pt);
}
}
}
template <class ToDo>
void ForEachGeometryPoint(ToDo && toDo) const
{
ToDoWrapper<ToDo> wrapper(std::forward<ToDo>(toDo));
ForEachGeometryPointEx(std::move(wrapper));
}
template <class ToDo>
bool ForAnyGeometryPointEx(ToDo && toDo) const
bool ForAnyPoint(ToDo && toDo) const
{
if (IsPoint())
return toDo(m_center);
for (PointSeq const & points : m_polygons)
for (auto const & points : m_polygons)
{
for (auto const & pt : points)
{
if (toDo(pt))
return true;
}
toDo.EndRegion();
if (base::AnyOf(points, std::forward<ToDo>(toDo)))
return true;
}
return false;
}
template <class ToDo>
bool ForAnyGeometryPoint(ToDo && toDo) const
void ForEachPolygon(ToDo && toDo) const
{
ToDoWrapper<ToDo> wrapper(std::forward<ToDo>(toDo));
return ForAnyGeometryPointEx(std::move(wrapper));
for (auto const & points : m_polygons)
toDo(points);
}
// To work with geometry type.
@ -210,18 +196,6 @@ public:
bool IsCoastCell() const { return (m_coastCell != -1); }
protected:
template <class ToDo>
class ToDoWrapper
{
public:
ToDoWrapper(ToDo && toDo) : m_toDo(std::forward<ToDo>(toDo)) {}
bool operator()(m2::PointD const & p) { return m_toDo(p); }
void EndRegion() {}
private:
ToDo && m_toDo;
};
// Can be one of the following:
// - point in point-feature
// - origin point of text [future] in line-feature

View file

@ -26,7 +26,7 @@ void CalculateMidPoints::operator()(FeatureBuilder const & ft, uint64_t pos)
m_midLoc = m2::PointD::Zero();;
m_locCount = 0;
ft.ForEachGeometryPoint(*this);
ft.ForEachPoint(*this);
ASSERT_NOT_EQUAL(m_locCount, 0, ());
m_midLoc = m_midLoc / m_locCount;

View file

@ -25,14 +25,16 @@ public:
static const std::string kOne;
static const std::string kTwo;
static constexpr m2::PointD kPointOne1{3.0, 3.0};
static constexpr m2::PointD kPointOne2{4.0, 4.0};
static constexpr m2::PointD kPointTwo1{10.0, 3.0};
static constexpr m2::PointD kPointTwo2{11.0, 4.0};
static constexpr m2::PointD kPointOneTwo1{7.0, 4.0};
static constexpr m2::PointD kPointOneTwo2{9.0, 5.0};
static constexpr m2::PointD kPointOneBb{1.0, 9.0};
static constexpr m2::PointD kPointTwoBb{14.0, 9.0};
static constexpr m2::PointD kPointInsideOne1{3.0, 3.0};
static constexpr m2::PointD kPointInsideOne2{4.0, 4.0};
static constexpr m2::PointD kPointInsideTwo1{10.0, 3.0};
static constexpr m2::PointD kPointInsideTwo2{11.0, 4.0};
static constexpr m2::PointD kPointInsideOneAndTwo1{7.0, 4.0};
static constexpr m2::PointD kPointInsideOneAndTwo2{9.0, 5.0};
static constexpr m2::PointD kPointOnBorderOne{1.0, 6.0};
static constexpr m2::PointD kPointOnBorderTwo{14.0, 6.0};
static constexpr m2::PointD kPointInsideOneBoundingBox{1.0, 9.0};
static constexpr m2::PointD kPointInsideTwoBoundingBox{14.0, 9.0};
AffiliationTests()
{
@ -102,31 +104,41 @@ bool Test(std::vector<std::string> && res, std::set<std::string> const & answ)
void TestCountriesAffiliationInsideBorders(feature::AffiliationInterface const & affiliation)
{
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOne1), {AffiliationTests::kOne}),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOne1),
{AffiliationTests::kOne}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOne2), {AffiliationTests::kOne}),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOne2),
{AffiliationTests::kOne}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointTwo1), {AffiliationTests::kTwo}),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOnBorderOne),
{AffiliationTests::kOne}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointTwo2), {AffiliationTests::kTwo}),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideTwo1),
{AffiliationTests::kTwo}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOneTwo1),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideTwo2),
{AffiliationTests::kTwo}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOneAndTwo1),
{AffiliationTests::kOne, AffiliationTests::kTwo}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOneTwo2),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOneAndTwo2),
{AffiliationTests::kOne, AffiliationTests::kTwo}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOnBorderTwo),
{AffiliationTests::kTwo}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::MakeLineFb(
{AffiliationTests::kPointOne1, AffiliationTests::kPointOne2})),
{AffiliationTests::kPointInsideOne1, AffiliationTests::kPointInsideOne2})),
{AffiliationTests::kOne}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::MakeLineFb(
{AffiliationTests::kPointTwo1, AffiliationTests::kPointTwo2})),
{AffiliationTests::kPointInsideTwo1, AffiliationTests::kPointInsideTwo2})),
{AffiliationTests::kTwo}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::MakeLineFb(
{AffiliationTests::kPointOne1, AffiliationTests::kPointTwo1})),
{AffiliationTests::kPointInsideOne1, AffiliationTests::kPointInsideTwo1})),
{AffiliationTests::kOne, AffiliationTests::kTwo}),
());
}
@ -139,17 +151,19 @@ void TestCountriesFilesAffiliation(std::string const & borderPath)
TestCountriesAffiliationInsideBorders(affiliation);
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOneBb), {}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointTwoBb), {}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOneBoundingBox), {}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideTwoBoundingBox), {}), ());
}
{
T affiliation(borderPath, true /* haveBordersForWholeWorld */);
TestCountriesAffiliationInsideBorders(affiliation);
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOneBb), {AffiliationTests::kOne}),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOneBoundingBox),
{AffiliationTests::kOne}),
());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointTwoBb), {AffiliationTests::kTwo}),
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideTwoBoundingBox),
{AffiliationTests::kTwo}),
());
}
}
@ -159,12 +173,12 @@ UNIT_CLASS_TEST(AffiliationTests, SingleAffiliationTests)
std::string const kName = "Name";
feature::SingleAffiliation affiliation(kName);
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOne1), {kName}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOne1), {kName}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointOneTwo1), {kName}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::kPointInsideOneAndTwo1), {kName}), ());
TEST(Test(affiliation.GetAffiliations(AffiliationTests::MakeLineFb(
{AffiliationTests::kPointOne1, AffiliationTests::kPointTwo1})),
{AffiliationTests::kPointInsideOne1, AffiliationTests::kPointInsideTwo1})),
{kName}),
());