diff --git a/drape_frontend/apply_feature_functors.cpp b/drape_frontend/apply_feature_functors.cpp index db537572f9..365368c0ee 100644 --- a/drape_frontend/apply_feature_functors.cpp +++ b/drape_frontend/apply_feature_functors.cpp @@ -18,8 +18,11 @@ #include "drape/stipple_pen_resource.hpp" #include "drape/utils/projection.hpp" +#include "base/logging.hpp" + #include "std/algorithm.hpp" #include "std/utility.hpp" +#include "std/mutex.hpp" namespace df { @@ -32,6 +35,71 @@ dp::Color ToDrapeColor(uint32_t src) return dp::Extract(src, 255 - (src >> 24)); } +#ifdef CALC_FILTERED_POINTS +class LinesStat +{ +public: + ~LinesStat() + { + map zoomValues; + for (pair const & f : m_features) + { + TValue & v = zoomValues[f.first.second]; + v.m_neededPoints += f.second.m_neededPoints; + v.m_readedPoints += f.second.m_readedPoints; + } + + LOG(LINFO, ("========================")); + for (pair const & v : zoomValues) + LOG(LINFO, ("Zoom = ", v.first, " Percent = ", 1 - v.second.m_neededPoints / (double)v.second.m_readedPoints)); + } + + static LinesStat & Get() + { + static LinesStat s_stat; + return s_stat; + } + + void InsertLine(FeatureID const & id, double scale, int vertexCount, int renderVertexCount) + { + int s = 0; + double factor = 5.688; + while (factor < scale) + { + s++; + factor = factor * 2.0; + } + + InsertLine(id, s, vertexCount, renderVertexCount); + } + + void InsertLine(FeatureID const & id, int scale, int vertexCount, int renderVertexCount) + { + TKey key(id, scale); + lock_guard g(m_mutex); + if (m_features.find(key) != m_features.end()) + return; + + TValue & v = m_features[key]; + v.m_readedPoints = vertexCount; + v.m_neededPoints = renderVertexCount; + } + +private: + LinesStat() = default; + + using TKey = pair; + struct TValue + { + int m_readedPoints = 0; + int m_neededPoints = 0; + }; + + map m_features; + mutex m_mutex; +}; +#endif + void Extract(::LineDefProto const * lineRule, df::LineViewParams & params) { @@ -259,15 +327,41 @@ ApplyLineFeature::ApplyLineFeature(ref_ptr context, FeatureID con double currentScaleGtoP) : TBase(context, id, captions) , m_currentScaleGtoP(currentScaleGtoP) + , m_sqrScale(math::sqr(m_currentScaleGtoP)) +#ifdef CALC_FILTERED_POINTS + , m_readedCount(0) +#endif { } void ApplyLineFeature::operator() (m2::PointD const & point) { +#ifdef CALC_FILTERED_POINTS + ++m_readedCount; +#endif + if (m_spline.IsNull()) m_spline.Reset(new m2::Spline()); - m_spline->AddPoint(point); + if (m_spline->IsEmpty()) + { + m_spline->AddPoint(point); + m_lastAddedPoint = point; + } + else + { + static float minSegmentLength = math::sqr(4.0 * df::VisualParams::Instance().GetVisualScale()); + if ((m_spline->GetSize() > 1 && point.SquareLength(m_lastAddedPoint) * m_sqrScale < minSegmentLength) || + m_spline->IsPrelonging(point)) + { + m_spline->ReplacePoint(point); + } + else + { + m_spline->AddPoint(point); + m_lastAddedPoint = point; + } + } } bool ApplyLineFeature::HasGeometry() const @@ -331,6 +425,10 @@ void ApplyLineFeature::ProcessRule(Stylist::TRuleWrapper const & rule) void ApplyLineFeature::Finish() { +#ifdef CALC_FILTERED_POINTS + LinesStat::Get().InsertLine(m_id, m_currentScaleGtoP, m_readedCount, m_spline->GetSize()); +#endif + string const & roadNumber = m_captions.GetRoadNumber(); if (roadNumber.empty()) return; diff --git a/drape_frontend/apply_feature_functors.hpp b/drape_frontend/apply_feature_functors.hpp index a83fc9be23..3a3bc04722 100644 --- a/drape_frontend/apply_feature_functors.hpp +++ b/drape_frontend/apply_feature_functors.hpp @@ -14,6 +14,8 @@ class CircleRuleProto; class SymbolRuleProto; class CaptionDefProto; +//#define CALC_FILTERED_POINTS + namespace df { @@ -91,6 +93,12 @@ public: private: m2::SharedSpline m_spline; double m_currentScaleGtoP; + double m_sqrScale; + m2::PointD m_lastAddedPoint; + +#ifdef CALC_FILTERED_POINTS + int m_readedCount; +#endif }; } // namespace df diff --git a/geometry/spline.cpp b/geometry/spline.cpp index 418bea301e..ed4617b381 100644 --- a/geometry/spline.cpp +++ b/geometry/spline.cpp @@ -44,6 +44,38 @@ void Spline::AddPoint(PointD const & pt) } } +void Spline::ReplacePoint(PointD const & pt) +{ + ASSERT(m_position.size() > 1, ()); + ASSERT(!m_length.empty(), ()); + ASSERT(!m_direction.empty(), ()); + m_position.pop_back(); + m_length.pop_back(); + m_direction.pop_back(); + AddPoint(pt); +} + +bool Spline::IsPrelonging(PointD const & pt) +{ + if (m_position.size() < 2) + return false; + + PointD dir = pt - m_position.back(); + if (dir.IsAlmostZero()) + return true; + + dir = dir.Normalize(); + PointD prevDir = m_direction.back().Normalize(); + + double const MAX_ANGLE_THRESHOLD = 0.98; + return fabs(DotProduct(prevDir, dir)) > MAX_ANGLE_THRESHOLD; +} + +size_t Spline::GetSize() const +{ + return m_position.size(); +} + bool Spline::IsEmpty() const { return m_position.empty(); diff --git a/geometry/spline.hpp b/geometry/spline.hpp index e2800e4669..9af7097e9d 100644 --- a/geometry/spline.hpp +++ b/geometry/spline.hpp @@ -49,6 +49,9 @@ public: Spline const & operator = (Spline const & spl); void AddPoint(PointD const & pt); + void ReplacePoint(PointD const & pt); + bool IsPrelonging(PointD const & pt); + size_t GetSize() const; vector const & GetPath() const { return m_position; } template