Features priority determenation.

This commit is contained in:
Lev Dragunov 2015-12-04 15:21:23 +03:00 committed by Sergey Yershov
parent 1c69f0b270
commit 3d0e4b76df
2 changed files with 138 additions and 7 deletions

View file

@ -1,9 +1,26 @@
#pragma once
#include "indexer/cell_id.hpp"
#include "indexer/drawing_rules.hpp"
#include "indexer/feature_data.hpp"
#include "indexer/feature_visibility.hpp"
#include "indexer/scales.hpp"
#include "geometry/any_rect2d.hpp"
#include "geometry/point2d.hpp"
#include "geometry/screenbase.hpp"
#include "geometry/tree4d.hpp"
#include "std/map.hpp"
#include "std/queue.hpp"
#include "std/vector.hpp"
namespace
{
double constexpr kPOIperTileSizeCoint = 6;
double constexpr kPointLookupDeltaDegrees = 0.000001;
} // namespace
namespace covering
{
class CellFeaturePair
@ -67,26 +84,139 @@ class DisplacementManager
{
public:
DisplacementManager(TSorter & sorter) : m_sorter(sorter) {}
template <class TFeature>
void Add(vector<int64_t> const & cells, uint32_t bucket, TFeature const & ft, uint32_t index)
{
if (!IsDisplacable(ft))
if (bucket != scales::GetUpperScale() && IsDisplacable(ft))
{
for (auto cell : cells)
m_sorter.Add(CellFeatureBucketTuple(CellFeaturePair(cell, index), bucket));
m_storage[bucket].emplace_back(cells, ft, index, bucket);
return;
}
// TODO(ldragunov) Put feature to displacement store.
for (auto const & cell : cells)
m_sorter.Add(CellFeatureBucketTuple(CellFeaturePair(cell, index), bucket));
}
void Displace()
{
m4::Tree<DisplacableNode> acceptedNodes;
list<DisplacableNode> deferredNodes;
for(uint32_t zoom = 0; zoom < scales::GetUpperScale(); ++zoom)
{
// Initialize queue.
priority_queue<DisplacableNode> displacableQueue;
for (auto const & node : m_storage[zoom])
displacableQueue.push(node);
for (auto const & node : deferredNodes)
displacableQueue.push(node);
deferredNodes.clear();
float const delta = CalculateDeltaForZoom(zoom);
size_t accepted = 0;
// Process queue.
while (!displacableQueue.empty())
{
ASSERT(!displacableQueue.empty(), ());
DisplacableNode const maxNode = displacableQueue.top();
displacableQueue.pop();
// Check if this node was displaced by higher features.
m2::RectD const displacementRect(maxNode.center.x - delta, maxNode.center.y - delta,
maxNode.center.x + delta, maxNode.center.y + delta);
bool isDisplacing = false;
acceptedNodes.ForEachInRect(displacementRect,
[&isDisplacing, &maxNode, &delta, &zoom](DisplacableNode const & node){
if ((maxNode.center - node.center).Length() < delta && node.maxZoomLevel > zoom)
isDisplacing = true;
});
if (isDisplacing)
{
deferredNodes.push_back(maxNode);
continue;
}
// Add feature to index otherwise.
for (auto const & cell : maxNode.cells)
m_sorter.Add(CellFeatureBucketTuple(CellFeaturePair(cell, maxNode.index), zoom));
acceptedNodes.Add(maxNode);
accepted++;
}
LOG(LINFO, ("Displacement for zoom", zoom,
"Features accepted:", accepted,
"Features discarted:", deferredNodes.size()));
}
}
private:
// TODO write displacement determination.
template <class TFeature>
bool IsDisplacable(TFeature const & /*ft*/) const noexcept
struct DisplacableNode
{
uint32_t index;
m2::PointD center;
vector<int64_t> cells;
int maxZoomLevel;
uint32_t priority;
DisplacableNode() : index(0), maxZoomLevel(0), priority(0) {}
template<class TFeature>
DisplacableNode(vector<int64_t> const & cells, TFeature const & ft, uint32_t index, int zoomLevel): index(index), center(ft.GetCenter()), cells(cells)
{
feature::TypesHolder const types(ft);
auto scaleRange = feature::GetDrawableScaleRange(types);
maxZoomLevel = scaleRange.second;
// Calculate depth field
drule::KeysT keys;
feature::GetDrawRule(ft, zoomLevel, keys);
drule::MakeUnique(keys);
float depth = 0;
size_t count = keys.size();
for (size_t i = 0; i < count; ++i)
{
if (depth < keys[i].m_priority)
depth = keys[i].m_priority;
}
float const kMinDepth = -100000.0f;
float const kMaxDepth = 100000.0f;
float const d = my::clamp(depth, kMinDepth, kMaxDepth) + kMaxDepth;
uint64_t rank = ft.GetRank();
priority = (static_cast<uint32_t>(d) << 8) || rank;
}
bool operator < (DisplacableNode const & node) const
{
return priority < node.priority;
}
bool operator == (DisplacableNode const & node) const
{
return index == node.index;
}
m2::RectD const GetLimitRect() const { return m2::RectD(center, center); }
};
template <class TFeature>
bool IsDisplacable(TFeature const & ft) const noexcept
{
feature::TypesHolder const types(ft);
if (types.GetGeoType() == feature::GEOM_POINT)
return true;
return false;
}
float CalculateDeltaForZoom(int32_t zoom)
{
double const worldSizeDevisor = 1 << zoom;
// Mercator SizeX and SizeY is equal
double const rectSize = (MercatorBounds::maxX - MercatorBounds::minX) / worldSizeDevisor;
return rectSize / kPOIperTileSizeCoint;
}
TSorter & m_sorter;
map<uint32_t, vector<DisplacableNode>> m_storage;
};
} // namespace indexer

View file

@ -159,6 +159,7 @@ void IndexScales(feature::DataHeader const & header, TFeaturesVector const & fea
vector<uint32_t> featuresInBucket(bucketsCount);
vector<uint32_t> cellsInBucket(bucketsCount);
features.ForEach(FeatureCoverer<TDisplacementManager>(header, manager, featuresInBucket, cellsInBucket));
manager.Displace();
sorter.SortAndFinish();
for (uint32_t bucket = 0; bucket < bucketsCount; ++bucket)