forked from organicmaps/organicmaps
Change LocalityIndex depth
This commit is contained in:
parent
e01c0f206d
commit
4bf5605859
11 changed files with 139 additions and 129 deletions
|
@ -7,8 +7,6 @@
|
|||
|
||||
#define POINT_COORD_BITS 30
|
||||
|
||||
typedef m2::CellId<19> RectId;
|
||||
|
||||
m2::PointU PointD2PointU(double x, double y, uint32_t coordBits);
|
||||
inline m2::PointU PointD2PointU(m2::PointD const & pt, uint32_t coordBits)
|
||||
{
|
||||
|
|
|
@ -5,6 +5,10 @@
|
|||
|
||||
#include "base/assert.hpp"
|
||||
|
||||
using RectId = m2::CellId<19>;
|
||||
|
||||
// 24 is enough to have cell size < 2.5m * 2.5m for world.
|
||||
using LocalityCellId = m2::CellId<24>;
|
||||
|
||||
template <int MinX, int MinY, int MaxX, int MaxY>
|
||||
struct Bounds
|
||||
|
|
|
@ -1,17 +1,16 @@
|
|||
#include "indexer/feature_covering.hpp"
|
||||
|
||||
#include "indexer/cell_coverer.hpp"
|
||||
#include "indexer/cell_id.hpp"
|
||||
#include "indexer/feature.hpp"
|
||||
#include "indexer/locality_object.hpp"
|
||||
#include "indexer/scales.hpp"
|
||||
|
||||
#include "geometry/covering_utils.hpp"
|
||||
|
||||
using namespace std;
|
||||
|
||||
namespace
|
||||
{
|
||||
|
||||
// This class should only be used with covering::CoverObject()!
|
||||
template <int DEPTH_LEVELS>
|
||||
class FeatureIntersector
|
||||
{
|
||||
public:
|
||||
|
@ -30,7 +29,7 @@ public:
|
|||
// 1. Here we don't need to differentiate between CELL_OBJECT_INTERSECT and OBJECT_INSIDE_CELL.
|
||||
// 2. We can return CELL_OBJECT_INTERSECT instead of CELL_INSIDE_OBJECT - it's just
|
||||
// a performance penalty.
|
||||
covering::CellObjectIntersection operator() (RectId const & cell) const
|
||||
covering::CellObjectIntersection operator()(m2::CellId<DEPTH_LEVELS> const & cell) const
|
||||
{
|
||||
using namespace covering;
|
||||
|
||||
|
@ -91,12 +90,11 @@ public:
|
|||
return CELL_OBJECT_NO_INTERSECTION;
|
||||
}
|
||||
|
||||
typedef CellIdConverter<MercatorBounds, RectId> CellIdConverterType;
|
||||
using CellIdConverter = CellIdConverter<MercatorBounds, m2::CellId<DEPTH_LEVELS>>;
|
||||
|
||||
m2::PointD ConvertPoint(m2::PointD const & p)
|
||||
{
|
||||
m2::PointD const pt(CellIdConverterType::XToCellIdX(p.x),
|
||||
CellIdConverterType::YToCellIdY(p.y));
|
||||
m2::PointD const pt(CellIdConverter::XToCellIdX(p.x), CellIdConverter::YToCellIdY(p.y));
|
||||
m_rect.Add(pt);
|
||||
return pt;
|
||||
}
|
||||
|
@ -112,7 +110,8 @@ public:
|
|||
}
|
||||
};
|
||||
|
||||
void GetIntersection(FeatureType const & f, FeatureIntersector & fIsect)
|
||||
template <int DEPTH_LEVELS>
|
||||
void GetIntersection(FeatureType const & f, FeatureIntersector<DEPTH_LEVELS> & fIsect)
|
||||
{
|
||||
// We need to cover feature for the best geometry, because it's indexed once for the
|
||||
// first top level scale. Do reset current cached geometry first.
|
||||
|
@ -126,19 +125,22 @@ void GetIntersection(FeatureType const & f, FeatureIntersector & fIsect)
|
|||
f.GetLimitRect(scale).IsValid(), (f.DebugString(scale)));
|
||||
}
|
||||
|
||||
vector<int64_t> CoverIntersection(FeatureIntersector const & fIsect, int cellDepth,
|
||||
template <int DEPTH_LEVELS>
|
||||
vector<int64_t> CoverIntersection(FeatureIntersector<DEPTH_LEVELS> const & fIsect, int cellDepth,
|
||||
uint64_t cellPenaltyArea)
|
||||
{
|
||||
if (fIsect.m_trg.empty() && fIsect.m_polyline.size() == 1)
|
||||
{
|
||||
m2::PointD const pt = fIsect.m_polyline[0];
|
||||
return vector<int64_t>(
|
||||
1, RectId::FromXY(static_cast<uint32_t>(pt.x), static_cast<uint32_t>(pt.y),
|
||||
RectId::DEPTH_LEVELS - 1).ToInt64(cellDepth));
|
||||
1, m2::CellId<DEPTH_LEVELS>::FromXY(static_cast<uint32_t>(pt.x),
|
||||
static_cast<uint32_t>(pt.y), DEPTH_LEVELS - 1)
|
||||
.ToInt64(cellDepth));
|
||||
}
|
||||
|
||||
vector<RectId> cells;
|
||||
covering::CoverObject(fIsect, cellPenaltyArea, cells, cellDepth, RectId::Root());
|
||||
vector<m2::CellId<DEPTH_LEVELS>> cells;
|
||||
covering::CoverObject(fIsect, cellPenaltyArea, cells, cellDepth,
|
||||
m2::CellId<DEPTH_LEVELS>::Root());
|
||||
|
||||
vector<int64_t> res(cells.size());
|
||||
for (size_t i = 0; i < cells.size(); ++i)
|
||||
|
@ -146,13 +148,13 @@ vector<int64_t> CoverIntersection(FeatureIntersector const & fIsect, int cellDep
|
|||
|
||||
return res;
|
||||
}
|
||||
}
|
||||
} // namespace
|
||||
|
||||
namespace covering
|
||||
{
|
||||
vector<int64_t> CoverFeature(FeatureType const & f, int cellDepth, uint64_t cellPenaltyArea)
|
||||
{
|
||||
FeatureIntersector fIsect;
|
||||
FeatureIntersector<RectId::DEPTH_LEVELS> fIsect;
|
||||
GetIntersection(f, fIsect);
|
||||
return CoverIntersection(fIsect, cellDepth, cellPenaltyArea);
|
||||
}
|
||||
|
@ -160,7 +162,7 @@ vector<int64_t> CoverFeature(FeatureType const & f, int cellDepth, uint64_t cell
|
|||
vector<int64_t> CoverLocality(indexer::LocalityObject const & o, int cellDepth,
|
||||
uint64_t cellPenaltyArea)
|
||||
{
|
||||
FeatureIntersector fIsect;
|
||||
FeatureIntersector<LocalityCellId::DEPTH_LEVELS> fIsect;
|
||||
o.ForEachPoint(fIsect);
|
||||
o.ForEachTriangle(fIsect);
|
||||
return CoverIntersection(fIsect, cellDepth, cellPenaltyArea);
|
||||
|
@ -192,91 +194,4 @@ Intervals SortAndMergeIntervals(Intervals const & v)
|
|||
SortAndMergeIntervals(v, res);
|
||||
return res;
|
||||
}
|
||||
|
||||
void AppendLowerLevels(RectId id, int cellDepth, Intervals & intervals)
|
||||
{
|
||||
int64_t idInt64 = id.ToInt64(cellDepth);
|
||||
intervals.push_back(make_pair(idInt64, idInt64 + id.SubTreeSize(cellDepth)));
|
||||
while (id.Level() > 0)
|
||||
{
|
||||
id = id.Parent();
|
||||
idInt64 = id.ToInt64(cellDepth);
|
||||
intervals.push_back(make_pair(idInt64, idInt64 + 1));
|
||||
}
|
||||
}
|
||||
|
||||
void CoverViewportAndAppendLowerLevels(m2::RectD const & r, int cellDepth, Intervals & res)
|
||||
{
|
||||
vector<RectId> ids;
|
||||
ids.reserve(SPLIT_RECT_CELLS_COUNT);
|
||||
CoverRect<MercatorBounds, RectId>(r, SPLIT_RECT_CELLS_COUNT, cellDepth, ids);
|
||||
|
||||
Intervals intervals;
|
||||
for (size_t i = 0; i < ids.size(); ++i)
|
||||
AppendLowerLevels(ids[i], cellDepth, intervals);
|
||||
|
||||
SortAndMergeIntervals(intervals, res);
|
||||
}
|
||||
|
||||
RectId GetRectIdAsIs(m2::RectD const & r)
|
||||
{
|
||||
double const eps = MercatorBounds::GetCellID2PointAbsEpsilon();
|
||||
using TConverter = CellIdConverter<MercatorBounds, RectId>;
|
||||
|
||||
return TConverter::Cover2PointsWithCell(
|
||||
MercatorBounds::ClampX(r.minX() + eps),
|
||||
MercatorBounds::ClampY(r.minY() + eps),
|
||||
MercatorBounds::ClampX(r.maxX() - eps),
|
||||
MercatorBounds::ClampY(r.maxY() - eps));
|
||||
}
|
||||
|
||||
int GetCodingDepth(int scale)
|
||||
{
|
||||
int const delta = scales::GetUpperScale() - scale;
|
||||
ASSERT_GREATER_OR_EQUAL ( delta, 0, () );
|
||||
|
||||
return (RectId::DEPTH_LEVELS - delta);
|
||||
}
|
||||
|
||||
Intervals const & CoveringGetter::Get(int scale)
|
||||
{
|
||||
int const cellDepth = GetCodingDepth(scale);
|
||||
int const ind = (cellDepth == RectId::DEPTH_LEVELS ? 0 : 1);
|
||||
|
||||
if (m_res[ind].empty())
|
||||
{
|
||||
switch (m_mode)
|
||||
{
|
||||
case ViewportWithLowLevels:
|
||||
CoverViewportAndAppendLowerLevels(m_rect, cellDepth, m_res[ind]);
|
||||
break;
|
||||
|
||||
case LowLevelsOnly:
|
||||
{
|
||||
RectId id = GetRectIdAsIs(m_rect);
|
||||
while (id.Level() >= cellDepth)
|
||||
id = id.Parent();
|
||||
AppendLowerLevels(id, cellDepth, m_res[ind]);
|
||||
|
||||
// Check for optimal result intervals.
|
||||
#if 0
|
||||
size_t oldSize = m_res[ind].size();
|
||||
Intervals res;
|
||||
SortAndMergeIntervals(m_res[ind], res);
|
||||
if (res.size() != oldSize)
|
||||
LOG(LINFO, ("Old =", oldSize, "; New =", res.size()));
|
||||
res.swap(m_res[ind]);
|
||||
#endif
|
||||
break;
|
||||
}
|
||||
|
||||
case FullCover:
|
||||
m_res[ind].push_back(Intervals::value_type(0, static_cast<int64_t>((1ULL << 63) - 1)));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return m_res[ind];
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -1,9 +1,18 @@
|
|||
#pragma once
|
||||
|
||||
#include "indexer/cell_coverer.hpp"
|
||||
#include "indexer/cell_id.hpp"
|
||||
#include "indexer/scales.hpp"
|
||||
|
||||
#include "geometry/mercator.hpp"
|
||||
#include "geometry/rect2d.hpp"
|
||||
|
||||
#include "coding/point_to_integer.hpp"
|
||||
|
||||
#include "base/logging.hpp"
|
||||
|
||||
#include <cstdint>
|
||||
#include <utility>
|
||||
#include <vector>
|
||||
|
||||
class FeatureType;
|
||||
|
@ -25,19 +34,56 @@ std::vector<int64_t> CoverFeature(FeatureType const & feature, int cellDepth,
|
|||
std::vector<int64_t> CoverLocality(indexer::LocalityObject const & o, int cellDepth,
|
||||
uint64_t cellPenaltyArea);
|
||||
|
||||
void AppendLowerLevels(RectId id, int cellDepth, Intervals & intervals);
|
||||
|
||||
// Cover viewport with RectIds and append their RectIds as well.
|
||||
void CoverViewportAndAppendLowerLevels(m2::RectD const & rect, int cellDepth,
|
||||
Intervals & intervals);
|
||||
|
||||
// Given a vector of intervals [a, b), sort them and merge overlapping intervals.
|
||||
Intervals SortAndMergeIntervals(Intervals const & intervals);
|
||||
void SortAndMergeIntervals(Intervals v, Intervals & res);
|
||||
|
||||
RectId GetRectIdAsIs(m2::RectD const & r);
|
||||
template <int DEPTH_LEVELS>
|
||||
m2::CellId<DEPTH_LEVELS> GetRectIdAsIs(m2::RectD const & r)
|
||||
{
|
||||
double const eps = MercatorBounds::GetCellID2PointAbsEpsilon();
|
||||
using Converter = CellIdConverter<MercatorBounds, m2::CellId<DEPTH_LEVELS>>;
|
||||
|
||||
return Converter::Cover2PointsWithCell(
|
||||
MercatorBounds::ClampX(r.minX() + eps), MercatorBounds::ClampY(r.minY() + eps),
|
||||
MercatorBounds::ClampX(r.maxX() - eps), MercatorBounds::ClampY(r.maxY() - eps));
|
||||
}
|
||||
|
||||
// Calculate cell coding depth according to max visual scale for mwm.
|
||||
int GetCodingDepth(int scale);
|
||||
template <int DEPTH_LEVELS>
|
||||
int GetCodingDepth(int scale)
|
||||
{
|
||||
int const delta = scales::GetUpperScale() - scale;
|
||||
ASSERT_GREATER_OR_EQUAL(delta, 0, ());
|
||||
return DEPTH_LEVELS - delta;
|
||||
}
|
||||
|
||||
template <int DEPTH_LEVELS>
|
||||
void AppendLowerLevels(m2::CellId<DEPTH_LEVELS> id, int cellDepth, Intervals & intervals)
|
||||
{
|
||||
int64_t idInt64 = id.ToInt64(cellDepth);
|
||||
intervals.push_back(std::make_pair(idInt64, idInt64 + id.SubTreeSize(cellDepth)));
|
||||
while (id.Level() > 0)
|
||||
{
|
||||
id = id.Parent();
|
||||
idInt64 = id.ToInt64(cellDepth);
|
||||
intervals.push_back(make_pair(idInt64, idInt64 + 1));
|
||||
}
|
||||
}
|
||||
|
||||
template <int DEPTH_LEVELS>
|
||||
void CoverViewportAndAppendLowerLevels(m2::RectD const & r, int cellDepth, Intervals & res)
|
||||
{
|
||||
std::vector<m2::CellId<DEPTH_LEVELS>> ids;
|
||||
ids.reserve(SPLIT_RECT_CELLS_COUNT);
|
||||
CoverRect<MercatorBounds, m2::CellId<DEPTH_LEVELS>>(r, SPLIT_RECT_CELLS_COUNT, cellDepth, ids);
|
||||
|
||||
Intervals intervals;
|
||||
for (auto const & id : ids)
|
||||
AppendLowerLevels<DEPTH_LEVELS>(id, cellDepth, intervals);
|
||||
|
||||
SortAndMergeIntervals(intervals, res);
|
||||
}
|
||||
|
||||
enum CoveringMode
|
||||
{
|
||||
|
@ -56,6 +102,47 @@ class CoveringGetter
|
|||
public:
|
||||
CoveringGetter(m2::RectD const & r, CoveringMode mode) : m_rect(r), m_mode(mode) {}
|
||||
|
||||
Intervals const & Get(int scale);
|
||||
};
|
||||
template <int DEPTH_LEVELS>
|
||||
Intervals const & Get(int scale)
|
||||
{
|
||||
int const cellDepth = GetCodingDepth<DEPTH_LEVELS>(scale);
|
||||
int const ind = (cellDepth == DEPTH_LEVELS ? 0 : 1);
|
||||
|
||||
if (m_res[ind].empty())
|
||||
{
|
||||
switch (m_mode)
|
||||
{
|
||||
case ViewportWithLowLevels:
|
||||
CoverViewportAndAppendLowerLevels<DEPTH_LEVELS>(m_rect, cellDepth, m_res[ind]);
|
||||
break;
|
||||
|
||||
case LowLevelsOnly:
|
||||
{
|
||||
m2::CellId<DEPTH_LEVELS> id = GetRectIdAsIs<DEPTH_LEVELS>(m_rect);
|
||||
while (id.Level() >= cellDepth)
|
||||
id = id.Parent();
|
||||
AppendLowerLevels<DEPTH_LEVELS>(id, cellDepth, m_res[ind]);
|
||||
|
||||
// Check for optimal result intervals.
|
||||
#if 0
|
||||
size_t oldSize = m_res[ind].size();
|
||||
Intervals res;
|
||||
SortAndMergeIntervals(m_res[ind], res);
|
||||
if (res.size() != oldSize)
|
||||
LOG(LINFO, ("Old =", oldSize, "; New =", res.size()));
|
||||
res.swap(m_res[ind]);
|
||||
#endif
|
||||
break;
|
||||
}
|
||||
|
||||
case FullCover:
|
||||
m_res[ind].push_back(
|
||||
Intervals::value_type(0, static_cast<int64_t>((uint64_t{1} << 63) - 1)));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return m_res[ind];
|
||||
}
|
||||
};
|
||||
}
|
||||
|
|
|
@ -114,7 +114,7 @@ private:
|
|||
scale = lastScale;
|
||||
|
||||
// Use last coding scale for covering (see index_builder.cpp).
|
||||
covering::Intervals const & interval = cov.Get(lastScale);
|
||||
covering::Intervals const & intervals = cov.Get<RectId::DEPTH_LEVELS>(lastScale);
|
||||
|
||||
// Prepare features reading.
|
||||
FeaturesVector const fv(pValue->m_cont, header, pValue->m_table.get());
|
||||
|
@ -125,7 +125,7 @@ private:
|
|||
CheckUniqueIndexes checkUnique(header.GetFormat() >= version::Format::v5);
|
||||
MwmId const & mwmID = handle.GetId();
|
||||
|
||||
for (auto const & i : interval)
|
||||
for (auto const & i : intervals)
|
||||
{
|
||||
index.ForEachInIntervalAndScale(
|
||||
[&](uint32_t index)
|
||||
|
@ -186,14 +186,14 @@ private:
|
|||
scale = lastScale;
|
||||
|
||||
// Use last coding scale for covering (see index_builder.cpp).
|
||||
covering::Intervals const & interval = cov.Get(lastScale);
|
||||
covering::Intervals const & intervals = cov.Get<RectId::DEPTH_LEVELS>(lastScale);
|
||||
ScaleIndex<ModelReaderPtr> const index(pValue->m_cont.GetReader(INDEX_FILE_TAG), pValue->m_factory);
|
||||
|
||||
// Iterate through intervals.
|
||||
CheckUniqueIndexes checkUnique(header.GetFormat() >= version::Format::v5);
|
||||
MwmId const & mwmID = handle.GetId();
|
||||
|
||||
for (auto const & i : interval)
|
||||
for (auto const & i : intervals)
|
||||
{
|
||||
index.ForEachInIntervalAndScale(
|
||||
[&](uint32_t index)
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
|
||||
#include "indexer/indexer_tests_support/test_with_custom_mwms.hpp"
|
||||
|
||||
#include "indexer/cell_id.hpp"
|
||||
#include "indexer/data_factory.hpp"
|
||||
#include "indexer/data_header.hpp"
|
||||
#include "indexer/feature.hpp"
|
||||
|
@ -49,7 +50,7 @@ public:
|
|||
covering::CoveringGetter covering(rect, covering::ViewportWithLowLevels);
|
||||
|
||||
vector<uint32_t> indices;
|
||||
for (auto const & interval : covering.Get(scale))
|
||||
for (auto const & interval : covering.Get<RectId::DEPTH_LEVELS>(scale))
|
||||
{
|
||||
index.ForEachInIntervalAndScale([&](uint32_t index) { indices.push_back(index); },
|
||||
interval.first, interval.second, scale);
|
||||
|
|
|
@ -32,7 +32,8 @@ public:
|
|||
void ForEachInRect(ProcessObject const & processObject, m2::RectD const & rect) const
|
||||
{
|
||||
covering::CoveringGetter cov(rect, covering::CoveringMode::ViewportWithLowLevels);
|
||||
covering::Intervals const & intervals = cov.Get(scales::GetUpperScale());
|
||||
covering::Intervals const & intervals =
|
||||
cov.Get<LocalityCellId::DEPTH_LEVELS>(scales::GetUpperScale());
|
||||
|
||||
for (auto const & i : intervals)
|
||||
{
|
||||
|
|
|
@ -16,6 +16,7 @@
|
|||
|
||||
#include "defines.hpp"
|
||||
|
||||
#include <cstdint>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
|
@ -35,8 +36,9 @@ void BuildLocalityIndex(TObjectsVector const & objects, TWriter & writer,
|
|||
1024 * 1024 /* bufferBytes */, tmpFilePrefix + CELL2LOCALITY_TMP_EXT, out);
|
||||
objects.ForEach([&sorter](indexer::LocalityObject const & o) {
|
||||
// @todo(t.yan): adjust cellPenaltyArea for whole world locality index.
|
||||
std::vector<int64_t> const cells =
|
||||
covering::CoverLocality(o, GetCodingDepth(scales::GetUpperScale()), 250 /* cellPenaltyArea */);
|
||||
std::vector<int64_t> const cells = covering::CoverLocality(
|
||||
o, GetCodingDepth<LocalityCellId::DEPTH_LEVELS>(scales::GetUpperScale()),
|
||||
250 /* cellPenaltyArea */);
|
||||
for (auto const & cell : cells)
|
||||
sorter.Add(CellValuePair<uint64_t>(cell, o.GetStoredId()));
|
||||
});
|
||||
|
@ -48,7 +50,7 @@ void BuildLocalityIndex(TObjectsVector const & objects, TWriter & writer,
|
|||
|
||||
{
|
||||
BuildIntervalIndex(cellsToValue.begin(), cellsToValue.end(), writer,
|
||||
RectId::DEPTH_LEVELS * 2 + 1);
|
||||
LocalityCellId::DEPTH_LEVELS * 2 + 1);
|
||||
}
|
||||
}
|
||||
} // namespace covering
|
||||
|
|
|
@ -37,7 +37,7 @@ public:
|
|||
, m_scalesIdx(0)
|
||||
, m_bucketsCount(header.GetLastScale() + 1)
|
||||
, m_displacement(manager)
|
||||
, m_codingDepth(covering::GetCodingDepth(header.GetLastScale()))
|
||||
, m_codingDepth(covering::GetCodingDepth<RectId::DEPTH_LEVELS>(header.GetLastScale()))
|
||||
, m_featuresInBucket(featuresInBucket)
|
||||
, m_cellsInBucket(cellsInBucket)
|
||||
{
|
||||
|
|
|
@ -1,11 +1,13 @@
|
|||
#include "search/mwm_context.hpp"
|
||||
|
||||
#include "indexer/cell_id.hpp"
|
||||
|
||||
namespace search
|
||||
{
|
||||
void CoverRect(m2::RectD const & rect, int scale, covering::Intervals & result)
|
||||
{
|
||||
covering::CoveringGetter covering(rect, covering::ViewportWithLowLevels);
|
||||
auto const & intervals = covering.Get(scale);
|
||||
auto const & intervals = covering.Get<RectId::DEPTH_LEVELS>(scale);
|
||||
result.insert(result.end(), intervals.begin(), intervals.end());
|
||||
}
|
||||
|
||||
|
|
|
@ -1,11 +1,11 @@
|
|||
#include "search/street_vicinity_loader.hpp"
|
||||
|
||||
#include "indexer/cell_id.hpp"
|
||||
#include "indexer/feature_covering.hpp"
|
||||
#include "indexer/feature_decl.hpp"
|
||||
#include "indexer/index.hpp"
|
||||
|
||||
#include "geometry/mercator.hpp"
|
||||
|
||||
#include "geometry/point2d.hpp"
|
||||
|
||||
#include "base/math.hpp"
|
||||
|
@ -58,7 +58,7 @@ void StreetVicinityLoader::LoadStreet(uint32_t featureId, Street & street)
|
|||
street.m_rect.Add(MercatorBounds::RectByCenterXYAndSizeInMeters(point, m_offsetMeters));
|
||||
|
||||
covering::CoveringGetter coveringGetter(street.m_rect, covering::ViewportWithLowLevels);
|
||||
auto const & intervals = coveringGetter.Get(m_scale);
|
||||
auto const & intervals = coveringGetter.Get<RectId::DEPTH_LEVELS>(m_scale);
|
||||
m_context->ForEachIndex(intervals, m_scale, MakeBackInsertFunctor(street.m_features));
|
||||
|
||||
street.m_calculator = make_unique<ProjectionOnStreetCalculator>(points);
|
||||
|
|
Loading…
Add table
Reference in a new issue