diff --git a/indexer/displacement_manager.hpp b/indexer/displacement_manager.hpp index eb9d4fdd19..e723ec1a68 100644 --- a/indexer/displacement_manager.hpp +++ b/indexer/displacement_manager.hpp @@ -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 void Add(vector 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 acceptedNodes; + list deferredNodes; + for(uint32_t zoom = 0; zoom < scales::GetUpperScale(); ++zoom) + { + // Initialize queue. + priority_queue 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 - bool IsDisplacable(TFeature const & /*ft*/) const noexcept + struct DisplacableNode { + uint32_t index; + m2::PointD center; + vector cells; + + int maxZoomLevel; + uint32_t priority; + + DisplacableNode() : index(0), maxZoomLevel(0), priority(0) {} + + template + DisplacableNode(vector 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(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 + 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> m_storage; }; } // namespace indexer diff --git a/indexer/scale_index_builder.hpp b/indexer/scale_index_builder.hpp index 1dbfb758df..31715970fa 100644 --- a/indexer/scale_index_builder.hpp +++ b/indexer/scale_index_builder.hpp @@ -159,6 +159,7 @@ void IndexScales(feature::DataHeader const & header, TFeaturesVector const & fea vector featuresInBucket(bucketsCount); vector cellsInBucket(bucketsCount); features.ForEach(FeatureCoverer(header, manager, featuresInBucket, cellsInBucket)); + manager.Displace(); sorter.SortAndFinish(); for (uint32_t bucket = 0; bucket < bucketsCount; ++bucket)