Make FeatureBuilder1 geometry more generic. Now we store the set of polygons for GEOM_AREA.

Holes are defined by WindingOdd rule in tesselator.cpp.
This commit is contained in:
vng 2011-09-02 16:06:14 +03:00 committed by Alex Zolotarev
parent 6fb7c31b7b
commit acca93ee07
8 changed files with 99 additions and 97 deletions

View file

@ -16,9 +16,15 @@ using namespace feature;
// FeatureBuilder1 implementation
///////////////////////////////////////////////////////////////////////////////////////////////////
FeatureBuilder1::FeatureBuilder1()
{
m_Polygons.push_back(points_t());
}
bool FeatureBuilder1::IsGeometryClosed() const
{
return (m_Geometry.size() > 2 && m_Geometry.front() == m_Geometry.back());
points_t const & poly = GetGeometry();
return (poly.size() > 2 && poly.front() == poly.back());
}
void FeatureBuilder1::SetCenter(m2::PointD const & p)
@ -30,25 +36,26 @@ void FeatureBuilder1::SetCenter(m2::PointD const & p)
void FeatureBuilder1::AddPoint(m2::PointD const & p)
{
m_Geometry.push_back(p);
m_Polygons.front().push_back(p);
m_LimitRect.Add(p);
}
void FeatureBuilder1::SetAreaAddHoles(list<points_t> const & holes)
{
m_Params.SetGeomType(GEOM_AREA);
m_Holes.clear();
m_Polygons.resize(1);
if (holes.empty()) return;
m2::Region<m2::PointD> rgn(m_Geometry.begin(), m_Geometry.end());
points_t const & poly = GetGeometry();
m2::Region<m2::PointD> rgn(poly.begin(), poly.end());
for (list<points_t>::const_iterator i = holes.begin(); i != holes.end(); ++i)
{
ASSERT ( !i->empty(), () );
if (rgn.Contains(i->front()))
m_Holes.push_back(*i);
m_Polygons.push_back(*i);
}
}
@ -155,15 +162,12 @@ bool FeatureBuilder1::operator == (FeatureBuilder1 const & fb) const
if (!is_equal(m_LimitRect, fb.m_LimitRect))
return false;
if (!is_equal(m_Geometry, fb.m_Geometry))
if (m_Polygons.size() != fb.m_Polygons.size())
return false;
if (m_Holes.size() != fb.m_Holes.size())
return false;
list<points_t>::const_iterator i = m_Holes.begin();
list<points_t>::const_iterator j = fb.m_Holes.begin();
for (; i != m_Holes.end(); ++i, ++j)
list<points_t>::const_iterator i = m_Polygons.begin();
list<points_t>::const_iterator j = fb.m_Polygons.begin();
for (; i != m_Polygons.end(); ++i, ++j)
if (!is_equal(*i, *j))
return false;
@ -175,15 +179,19 @@ bool FeatureBuilder1::CheckValid() const
CHECK(m_Params.CheckValid(), ());
EGeomType const type = m_Params.GetGeomType();
CHECK ( type != GEOM_UNDEFINED, () );
CHECK(type != GEOM_LINE || m_Geometry.size() >= 2, ());
points_t const & poly = GetGeometry();
CHECK(type != GEOM_AREA || m_Geometry.size() >= 3, ());
if (type == GEOM_LINE)
CHECK(poly.size() >= 2, (*this));
CHECK(m_Holes.empty() || type == GEOM_AREA, ());
for (list<points_t>::const_iterator i = m_Holes.begin(); i != m_Holes.end(); ++i)
CHECK(i->size() >= 3, ());
if (type == GEOM_AREA)
{
// Use '4' because the first point should be equal with the last one
for (list<points_t>::const_iterator i = m_Polygons.begin(); i != m_Polygons.end(); ++i)
CHECK(i->size() >= 4, ());
}
return true;
}
@ -211,16 +219,11 @@ void FeatureBuilder1::Serialize(buffer_t & data) const
PushBackByteSink<buffer_t> sink(data);
EGeomType const type = m_Params.GetGeomType();
if (type != GEOM_POINT)
serial::SaveOuterPath(m_Geometry, cp, sink);
if (type == GEOM_AREA)
if (m_Params.GetGeomType() != GEOM_POINT)
{
WriteVarUint(sink, uint32_t(m_Holes.size()));
WriteVarUint(sink, uint32_t(m_Polygons.size()));
for (list<points_t>::const_iterator i = m_Holes.begin(); i != m_Holes.end(); ++i)
for (list<points_t>::const_iterator i = m_Polygons.begin(); i != m_Polygons.end(); ++i)
serial::SaveOuterPath(*i, cp, sink);
}
@ -252,17 +255,15 @@ void FeatureBuilder1::Deserialize(buffer_t & data)
return;
}
serial::LoadOuterPath(source, cp, m_Geometry);
CalcRect(m_Geometry, m_LimitRect);
m_Polygons.clear();
uint32_t const count = ReadVarUint<uint32_t>(source);
ASSERT_GREATER ( count, 0, () );
if (type == GEOM_AREA)
for (uint32_t i = 0; i < count; ++i)
{
uint32_t const count = ReadVarUint<uint32_t>(source);
for (uint32_t i = 0; i < count; ++i)
{
m_Holes.push_back(points_t());
serial::LoadOuterPath(source, cp, m_Holes.back());
}
m_Polygons.push_back(points_t());
serial::LoadOuterPath(source, cp, m_Polygons.back());
CalcRect(m_Polygons.back(), m_LimitRect);
}
CHECK ( CheckValid(), () );
@ -286,6 +287,7 @@ string debug_print(FeatureBuilder1 const & f)
ostringstream out;
for (size_t i = 0; i < f.m_osmIds.size(); ++i)
out << f.m_osmIds[i].Type() << " id=" << f.m_osmIds[i].Id() << " ";
switch (f.GetGeomType())
{
case feature::GEOM_POINT: out << "(" << f.m_Center << ")"; break;
@ -303,7 +305,7 @@ string debug_print(FeatureBuilder1 const & f)
bool FeatureBuilder2::IsDrawableInRange(int lowS, int highS) const
{
if (!m_Geometry.empty())
if (!GetGeometry().empty())
{
FeatureBase const fb = GetFeatureBase();

View file

@ -18,6 +18,8 @@ class FeatureBuilder1
friend string debug_print(FeatureBuilder1 const & f);
public:
FeatureBuilder1();
/// @name Geometry manipulating functions.
//@{
/// Set center (origin) point of feature and set that feature is point.
@ -57,23 +59,18 @@ public:
bool IsGeometryClosed() const;
inline size_t GetPointsCount() const { return m_Geometry.size(); }
template <class ToDo>
void ForEachPointRef(ToDo & toDo) const
{
for_each(m_Geometry.begin(), m_Geometry.end(), bind<void>(ref(toDo), _1));
}
inline size_t GetPointsCount() const { return GetGeometry().size(); }
// stops processing when functor returns false
template <class ToDo>
void ForEachTruePointRef(ToDo & toDo) const
void ForEachGeometryPoint(ToDo & toDo) const
{
if (m_Params.GetGeomType() == feature::GEOM_POINT)
toDo(m_Center);
else
{
for (points_t::const_iterator it = m_Geometry.begin(); it != m_Geometry.end(); ++it)
points_t const & poly = GetGeometry();
for (points_t::const_iterator it = poly.begin(); it != poly.end(); ++it)
if (!toDo(*it))
return;
}
@ -112,13 +109,10 @@ protected:
typedef vector<m2::PointD> points_t;
/// Can be one of the following:
/// - geometry in line-feature
/// - boundary in area-feature
points_t m_Geometry; // Check HEADER_IS_LINE
inline points_t const & GetGeometry() const { return m_Polygons.front(); }
/// List of holes in area-feature.
list<points_t> m_Holes; // Check HEADER_IS_AREA
/// List of geometry polygons.
list<points_t> m_Polygons; // Check HEADER_IS_AREA
};
/// Used for serialization of features during final pass.
@ -154,8 +148,8 @@ public:
bool IsArea() const { return (m_Params.GetTypeMask() & feature::HEADER_GEOM_AREA) != 0; }
bool IsDrawableInRange(int lowS, int highS) const;
points_t const & GetGeometry() const { return m_Geometry; }
list<points_t> const & GetHoles() const { return m_Holes; }
points_t const & GetOuterPoly() const { return GetGeometry(); }
list<points_t> const & GetPolygons() const { return m_Polygons; }
/// @name Overwrite from base_type.
//@{

View file

@ -13,7 +13,8 @@ MergedFeatureBuilder1::MergedFeatureBuilder1(FeatureBuilder1 const & fb)
void MergedFeatureBuilder1::SetRound()
{
m_isRound = true;
m_roundBounds[0] = m_roundBounds[1] = m_Geometry;
m_roundBounds[0] = m_roundBounds[1] = GetGeometry();
}
void MergedFeatureBuilder1::AppendFeature(MergedFeatureBuilder1 const & fb, bool fromBegin, bool toBack)
@ -21,12 +22,15 @@ void MergedFeatureBuilder1::AppendFeature(MergedFeatureBuilder1 const & fb, bool
// Also merge Osm IDs for debugging
m_osmIds.insert(m_osmIds.end(), fb.m_osmIds.begin(), fb.m_osmIds.end());
points_t & thisG = m_Polygons.front();
points_t const & fbG = fb.GetGeometry();
if (fb.m_isRound)
{
if (toBack)
m_roundBounds[1] = fb.m_Geometry;
m_roundBounds[1] = fbG;
else
m_roundBounds[0] = fb.m_Geometry;
m_roundBounds[0] = fbG;
return;
}
@ -37,28 +41,28 @@ void MergedFeatureBuilder1::AppendFeature(MergedFeatureBuilder1 const & fb, bool
m_isRound = false;
for (size_t i = 0; i < fb.m_Geometry.size(); ++i)
m_LimitRect.Add(fb.m_Geometry[i]);
for (size_t i = 0; i < fbG.size(); ++i)
m_LimitRect.Add(fbG[i]);
if (fromBegin)
{
if (toBack)
m_Geometry.insert(m_Geometry.end(), fb.m_Geometry.begin() + 1, fb.m_Geometry.end());
thisG.insert(thisG.end(), fbG.begin() + 1, fbG.end());
else
m_Geometry.insert(m_Geometry.begin(), fb.m_Geometry.begin(), fb.m_Geometry.end() - 1);
thisG.insert(thisG.begin(), fbG.begin(), fbG.end() - 1);
}
else
{
if (toBack)
m_Geometry.insert(m_Geometry.end(), fb.m_Geometry.rbegin() + 1, fb.m_Geometry.rend());
thisG.insert(thisG.end(), fbG.rbegin() + 1, fbG.rend());
else
m_Geometry.insert(m_Geometry.begin(), fb.m_Geometry.rbegin(), fb.m_Geometry.rend() - 1);
thisG.insert(thisG.begin(), fbG.rbegin(), fbG.rend() - 1);
}
}
bool MergedFeatureBuilder1::EqualGeometry(MergedFeatureBuilder1 const & fb) const
{
return (m_Geometry == fb.m_Geometry);
return (GetGeometry() == fb.GetGeometry());
}
pair<m2::PointD, bool> MergedFeatureBuilder1::GetKeyPoint(size_t i) const
@ -89,9 +93,11 @@ size_t MergedFeatureBuilder1::GetKeyPointsCount() const
double MergedFeatureBuilder1::GetPriority() const
{
points_t const & poly = GetGeometry();
double pr = 0.0;
for (size_t i = 1; i < m_Geometry.size(); ++i)
pr += m_Geometry[i-1].SquareLength(m_Geometry[i]);
for (size_t i = 1; i < poly.size(); ++i)
pr += poly[i-1].SquareLength(poly[i]);
return pr;
}

View file

@ -26,10 +26,10 @@ public:
bool EqualGeometry(MergedFeatureBuilder1 const & fb) const;
inline bool NotEmpty() const { return !m_Geometry.empty(); }
inline bool NotEmpty() const { return !GetGeometry().empty(); }
inline m2::PointD FirstPoint() const { return m_Geometry.front(); }
inline m2::PointD LastPoint() const { return m_Geometry.back(); }
inline m2::PointD FirstPoint() const { return GetGeometry().front(); }
inline m2::PointD LastPoint() const { return GetGeometry().back(); }
inline void SetType(uint32_t type) { m_Params.SetType(type); }
inline bool PopAnyType(uint32_t & type) { return m_Params.PopAnyType(type); }
@ -43,8 +43,9 @@ public:
template <class ToDo> void ForEachMiddlePoints(ToDo toDo) const
{
for (size_t i = 1; i < m_Geometry.size()-1; ++i)
toDo(m_Geometry[i]);
points_t const & poly = GetGeometry();
for (size_t i = 1; i < poly.size()-1; ++i)
toDo(poly[i]);
}
pair<m2::PointD, bool> GetKeyPoint(size_t i) const;

View file

@ -48,7 +48,7 @@ namespace
m_midLoc = m2::PointD(0, 0);
m_locCount = 0;
ft.ForEachPointRef(*this);
ft.ForEachGeometryPoint(*this);
m_midLoc = m_midLoc / m_locCount;
uint64_t const pointAsInt64 = PointToInt64(m_midLoc.x, m_midLoc.y, m_coordBits);
@ -59,12 +59,13 @@ namespace
m_vec.push_back(make_pair(order, pos));
}
void operator() (m2::PointD const & p)
bool operator() (m2::PointD const & p)
{
m_midLoc += p;
m_midAll += p;
++m_locCount;
++m_allCount;
return true;
}
m2::PointD GetCenter() const { return m_midAll / m_allCount; }
@ -147,7 +148,7 @@ namespace feature
private:
typedef vector<m2::PointD> points_t;
typedef list<points_t> holes_t;
typedef list<points_t> polygons_t;
class GeometryHolder
{
@ -169,14 +170,14 @@ namespace feature
serial::SaveOuterPath(points, m_codingParams, *m_rMain.m_geoFile[i]);
}
void WriteOuterTriangles(points_t const & bound, holes_t const & holes, int i)
void WriteOuterTriangles(polygons_t const & polys, int i)
{
m_buffer.m_trgMask |= (1 << i);
m_buffer.m_trgOffset.push_back(m_rMain.GetFileSize(*m_rMain.m_trgFile[i]));
// tesselation
tesselator::TrianglesInfo info;
tesselator::TesselateInterior(bound, holes, info);
tesselator::TesselateInterior(polys, info);
serial::TrianglesChainSaver saver(m_codingParams);
@ -255,7 +256,7 @@ namespace feature
points_t const & GetSourcePoints()
{
return (!m_current.empty() ? m_current : m_rFB.GetGeometry());
return (!m_current.empty() ? m_current : m_rFB.GetOuterPoly());
}
void AddPoints(points_t const & points, int scaleIndex)
@ -327,12 +328,12 @@ namespace feature
return true;
}
void AddTriangles(points_t const & bound, holes_t const & holes, int scaleIndex)
void AddTriangles(polygons_t const & polys, int scaleIndex)
{
CHECK ( m_buffer.m_innerTrg.empty(), () );
m_trgInner = false;
WriteOuterTriangles(bound, holes, scaleIndex);
WriteOuterTriangles(polys, scaleIndex);
}
};
@ -361,23 +362,25 @@ namespace feature
{
// simplify and serialize triangles
holes_t const & holes = fb.GetHoles();
if (holes.empty() && holder.TryToMakeStrip(points))
polygons_t const & polys = fb.GetPolygons();
if (polys.size() == 1 && holder.TryToMakeStrip(points))
continue;
holes_t simpleHoles;
for (holes_t::const_iterator iH = holes.begin(); iH != holes.end(); ++iH)
polygons_t simplified;
simplified.push_back(points);
polygons_t::const_iterator iH = polys.begin();
for (++iH; iH != polys.end(); ++iH)
{
simpleHoles.push_back(points_t());
simplified.push_back(points_t());
SimplifyPoints(*iH, simpleHoles.back(), m_header.GetScale(i));
SimplifyPoints(*iH, simplified.back(), m_header.GetScale(i));
if (simpleHoles.back().size() < 3)
simpleHoles.pop_back();
if (simplified.back().size() < 3)
simplified.pop_back();
}
holder.AddTriangles(points, simpleHoles, i);
holder.AddTriangles(simplified, i);
}
}
}

View file

@ -195,7 +195,7 @@ namespace feature
for (size_t i = 0; i < m_Countries.size(); ++i)
{
PointChecker doCheck(m_Countries[i]->m_regions);
m_FB.ForEachTruePointRef(doCheck);
m_FB.ForEachGeometryPoint(doCheck);
if (doCheck.m_belongs)
m_pPolygonizer->EmitFeature(m_Countries[i], m_FB);

View file

@ -26,7 +26,7 @@ namespace tesselator
}
};
void TesselateInterior(PointsT const & bound, HolesT const & holes, TrianglesInfo & info)
void TesselateInterior(PolygonsT const & polys, TrianglesInfo & info)
{
tess::VectorDispatcher disp;
tess::Tesselator tess;
@ -35,11 +35,7 @@ namespace tesselator
tess.beginPolygon();
tess.beginContour();
for_each(bound.begin(), bound.end(), AddTessPointF(tess));
tess.endContour();
for (HolesT::const_iterator it = holes.begin(); it != holes.end(); ++it)
for (PolygonsT::const_iterator it = polys.begin(); it != polys.end(); ++it)
{
tess.beginContour();
for_each(it->begin(), it->end(), AddTessPointF(tess));

View file

@ -13,7 +13,7 @@
namespace tesselator
{
typedef vector<m2::PointD> PointsT;
typedef list<PointsT> HolesT;
typedef list<PointsT> PolygonsT;
struct Triangle
{
@ -162,5 +162,5 @@ namespace tesselator
};
/// Main tesselate function.
void TesselateInterior(PointsT const & bound, HolesT const & holes, TrianglesInfo & info);
void TesselateInterior(PolygonsT const & polys, TrianglesInfo & info);
}