forked from organicmaps/organicmaps
[generator] Added ClustersFinder.
This commit is contained in:
parent
175ff54ffb
commit
3b26b2e7cb
3 changed files with 105 additions and 32 deletions
|
@ -235,7 +235,7 @@ private:
|
|||
{
|
||||
for (auto const & id : fbWithIds.second)
|
||||
{
|
||||
if (cities.count(id) != 0)
|
||||
if (cities.count(id) == 0)
|
||||
continue;
|
||||
|
||||
auto static const kPromoType = classif().GetTypeByPath({"sponsored", "promo_catalog"});
|
||||
|
|
|
@ -81,19 +81,15 @@ bool IsTheSamePlace(T const & left, T const & right)
|
|||
return dist <= GetThresholdM(localityL);
|
||||
}
|
||||
|
||||
template <typename Iterator>
|
||||
Iterator FindGroupOfTheSamePlaces(Iterator start, Iterator end)
|
||||
std::vector<std::vector<FeaturePlace>> FindClusters(std::vector<FeaturePlace> && places)
|
||||
{
|
||||
CHECK(start != end, ());
|
||||
auto next = start;
|
||||
while (++next != end)
|
||||
{
|
||||
if (!IsTheSamePlace(*next, *start))
|
||||
return next;
|
||||
|
||||
start = next;
|
||||
}
|
||||
return end;
|
||||
auto func = [](FeaturePlace const & fp) {
|
||||
auto const & localityChecker = ftypes::IsLocalityChecker::Instance();
|
||||
auto const locality = localityChecker.GetType(GetPlaceType(fp.GetFb()));
|
||||
return GetThresholdM(locality);
|
||||
};
|
||||
ClustersFinder<FeaturePlace, std::vector> f(std::move(places), func, IsTheSamePlace<FeaturePlace>);
|
||||
return f.Find();
|
||||
}
|
||||
} // namespace
|
||||
|
||||
|
@ -131,6 +127,11 @@ m2::RectD const & FeaturePlace::GetLimitRect() const
|
|||
return m_limitRect;
|
||||
}
|
||||
|
||||
base::GeoObjectId FeaturePlace::GetMostGenericOsmId() const
|
||||
{
|
||||
return GetFb().GetMostGenericOsmId();
|
||||
}
|
||||
|
||||
uint8_t FeaturePlace::GetRank() const
|
||||
{
|
||||
return GetFb().GetRank();
|
||||
|
@ -194,19 +195,11 @@ std::vector<PlaceProcessor::PlaceWithIds> PlaceProcessor::ProcessPlaces()
|
|||
places.reserve(nameToGeoObjectIdToFeaturePlaces.second.size());
|
||||
for (auto const & geoObjectIdToFeaturePlaces : nameToGeoObjectIdToFeaturePlaces.second)
|
||||
places.emplace_back(geoObjectIdToFeaturePlaces.second);
|
||||
// Firstly, objects are geometrically sorted.
|
||||
std::sort(std::begin(places), std::end(places), [](auto const & l, auto const & r) {
|
||||
auto const & rectL = l.GetLimitRect();
|
||||
auto const & rectR = r.GetLimitRect();
|
||||
return rectL.Center() < rectR.Center();
|
||||
});
|
||||
// Secondly, a group of similar places is searched for, then a better place is searched in
|
||||
// this group.
|
||||
auto start = std::begin(places);
|
||||
while (start != std::cend(places))
|
||||
|
||||
auto const clusters = FindClusters(std::move(places));
|
||||
for (auto const & cluster : clusters)
|
||||
{
|
||||
auto end = FindGroupOfTheSamePlaces(start, std::end(places));
|
||||
auto best = std::max_element(start, end, IsWorsePlace<FeaturePlace>);
|
||||
auto best = std::max_element(std::cbegin(cluster), std::cend(cluster), IsWorsePlace<FeaturePlace>);
|
||||
auto bestFb = best->GetFb();
|
||||
auto const & localityChecker = ftypes::IsLocalityChecker::Instance();
|
||||
if (bestFb.IsArea() && localityChecker.GetType(GetPlaceType(bestFb)) != ftypes::NONE)
|
||||
|
@ -215,17 +208,15 @@ std::vector<PlaceProcessor::PlaceWithIds> PlaceProcessor::ProcessPlaces()
|
|||
TransformAreaToPoint(bestFb);
|
||||
}
|
||||
std::vector<base::GeoObjectId> ids;
|
||||
ids.reserve(static_cast<size_t>(std::distance(start, end)));
|
||||
std::transform(start, end, std::back_inserter(ids), [](auto const & place) {
|
||||
return place.GetMostGenericOsmId();
|
||||
});
|
||||
ids.reserve(cluster.size());
|
||||
std::transform(std::cbegin(cluster), std::cend(cluster), std::back_inserter(ids),
|
||||
[](auto const & place) { return place.GetMostGenericOsmId(); });
|
||||
finalPlaces.emplace_back(std::move(bestFb), std::move(ids));
|
||||
if (m_boundariesTable)
|
||||
FillTable(start, end, best);
|
||||
|
||||
start = end;
|
||||
FillTable(std::cbegin(cluster), std::cend(cluster), best);
|
||||
}
|
||||
}
|
||||
|
||||
return finalPlaces;
|
||||
}
|
||||
|
||||
|
|
|
@ -3,15 +3,96 @@
|
|||
#include "generator/cities_boundaries_builder.hpp"
|
||||
#include "generator/feature_builder.hpp"
|
||||
|
||||
#include "geometry/mercator.hpp"
|
||||
#include "geometry/tree4d.hpp"
|
||||
|
||||
#include "base/geo_object_id.hpp"
|
||||
|
||||
#include <functional>
|
||||
#include <memory>
|
||||
#include <queue>
|
||||
#include <set>
|
||||
#include <string>
|
||||
#include <unordered_map>
|
||||
#include <vector>
|
||||
|
||||
namespace generator
|
||||
{
|
||||
template <typename T, template<typename, typename> class Container, typename Alloc = std::allocator<T>>
|
||||
class ClustersFinder
|
||||
{
|
||||
public:
|
||||
using DistanceFunc = std::function<double(T const &)>;
|
||||
using IsSameFunc = std::function<bool(T const &, T const &)>;
|
||||
using ConstIterator = T const *;
|
||||
|
||||
ClustersFinder(Container<T, Alloc> && container, DistanceFunc distanceFunc, IsSameFunc isSameFunc)
|
||||
: m_container(std::move(container)), m_distanseFunc(distanceFunc), m_isSameFunc(isSameFunc)
|
||||
{
|
||||
for (auto const & e : m_container)
|
||||
m_tree.Add(&e);
|
||||
}
|
||||
|
||||
std::vector<std::vector<T>> Find()
|
||||
{
|
||||
std::vector<std::vector<T>> clusters;
|
||||
std::set<ConstIterator> unviewed;
|
||||
for (auto const & e : m_container)
|
||||
unviewed.insert(&e);
|
||||
|
||||
while (!unviewed.empty())
|
||||
clusters.emplace_back(FindOneCluster(*std::cbegin(unviewed), unviewed));
|
||||
|
||||
return clusters;
|
||||
}
|
||||
|
||||
private:
|
||||
struct TraitsDef
|
||||
{
|
||||
m2::RectD const LimitRect(ConstIterator const & it) const { return it->GetLimitRect(); }
|
||||
};
|
||||
|
||||
std::vector<T> FindOneCluster(ConstIterator const & it, std::set<ConstIterator> & unviewed)
|
||||
{
|
||||
std::vector<T> cluster{*it};
|
||||
std::queue<ConstIterator> queue;
|
||||
queue.emplace(it);
|
||||
unviewed.erase(it);
|
||||
while (!queue.empty())
|
||||
{
|
||||
auto const current = queue.front();
|
||||
queue.pop();
|
||||
auto const queryBbox = GetBboxFor(current);
|
||||
m_tree.ForEachInRect(queryBbox, [&](auto const & conditate) {
|
||||
if (current == conditate || unviewed.count(conditate) == 0 || !m_isSameFunc(*current, *conditate))
|
||||
return;
|
||||
|
||||
unviewed.erase(conditate);
|
||||
queue.emplace(conditate);
|
||||
cluster.emplace_back(*conditate);
|
||||
});
|
||||
}
|
||||
|
||||
return cluster;
|
||||
}
|
||||
|
||||
m2::RectD GetBboxFor(ConstIterator const & it)
|
||||
{
|
||||
m2::RectD bbox;
|
||||
it->GetLimitRect().ForEachCorner([&](auto const & p) {
|
||||
auto const dist = m_distanseFunc(*it);
|
||||
bbox.Add(MercatorBounds::RectByCenterXYAndSizeInMeters(p, dist));
|
||||
});
|
||||
|
||||
return bbox;
|
||||
}
|
||||
|
||||
Container<T, Alloc> m_container;
|
||||
DistanceFunc m_distanseFunc;
|
||||
IsSameFunc m_isSameFunc;
|
||||
m4::Tree<ConstIterator, TraitsDef> m_tree;
|
||||
};
|
||||
|
||||
bool NeedProcessPlace(feature::FeatureBuilder const & fb);
|
||||
|
||||
// This structure encapsulates work with elements of different types.
|
||||
|
@ -25,6 +106,7 @@ public:
|
|||
feature::FeatureBuilder const & GetFb() const;
|
||||
FeaturesBuilders const & GetFbs() const;
|
||||
m2::RectD const & GetLimitRect() const;
|
||||
base::GeoObjectId GetMostGenericOsmId() const;
|
||||
uint8_t GetRank() const;
|
||||
std::string GetName() const;
|
||||
m2::PointD GetKeyPoint() const;
|
||||
|
|
Loading…
Add table
Reference in a new issue