diff --git a/drape/overlay_handle.hpp b/drape/overlay_handle.hpp index 43ab1c2012..1ed5027bc6 100644 --- a/drape/overlay_handle.hpp +++ b/drape/overlay_handle.hpp @@ -103,6 +103,7 @@ public: virtual m2::PointD GetPivot(ScreenBase const & screen, bool perspective) const; + virtual void BeforeUpdate() {} virtual bool Update(ScreenBase const & /*screen*/) { return true; } virtual m2::RectD GetPixelRect(ScreenBase const & screen, bool perspective) const = 0; diff --git a/drape/overlay_tree.cpp b/drape/overlay_tree.cpp index e22616765a..58d4588102 100644 --- a/drape/overlay_tree.cpp +++ b/drape/overlay_tree.cpp @@ -158,7 +158,7 @@ void OverlayTree::Add(ref_ptr handle) } // Clip handles which are out of screen if these handles were not displacers - // last time. Also clip all handles in reserve projection. + // last time. Also clip all handles in reverse projection. m2::RectD const pixelRect = handle->GetExtendedPixelRect(modelView); if (modelView.IsReverseProjection3d(pixelRect.Center()) || (m_displacers.find(handle) == m_displacers.end() && diff --git a/drape/render_bucket.cpp b/drape/render_bucket.cpp index fb6a70f7d0..e6b8e2f981 100644 --- a/drape/render_bucket.cpp +++ b/drape/render_bucket.cpp @@ -56,8 +56,15 @@ void RenderBucket::AddOverlayHandle(drape_ptr && handle) m_overlay.push_back(move(handle)); } +void RenderBucket::BeforeUpdate() +{ + for (drape_ptr & overlayHandle : m_overlay) + overlayHandle->BeforeUpdate(); +} + void RenderBucket::Update(ScreenBase const & modelView) { + BeforeUpdate(); for (drape_ptr & overlayHandle : m_overlay) { if (overlayHandle->IsVisible()) @@ -67,6 +74,7 @@ void RenderBucket::Update(ScreenBase const & modelView) void RenderBucket::CollectOverlayHandles(ref_ptr tree) { + BeforeUpdate(); for (drape_ptr const & overlayHandle : m_overlay) tree->Add(make_ref(overlayHandle)); } diff --git a/drape/render_bucket.hpp b/drape/render_bucket.hpp index 05efe6ef3f..95137e4dc6 100644 --- a/drape/render_bucket.hpp +++ b/drape/render_bucket.hpp @@ -55,6 +55,9 @@ public: void SetFeatureMinZoom(int minZoom); int GetMinZoom() const { return m_featuresMinZoom; } +private: + void BeforeUpdate(); + private: int m_featuresMinZoom = numeric_limits::max(); diff --git a/drape_frontend/CMakeLists.txt b/drape_frontend/CMakeLists.txt index 83b3587fc5..dd783d77ef 100644 --- a/drape_frontend/CMakeLists.txt +++ b/drape_frontend/CMakeLists.txt @@ -142,6 +142,8 @@ set( overlays_tracker.hpp path_symbol_shape.cpp path_symbol_shape.hpp + path_text_handle.cpp + path_text_handle.hpp path_text_shape.cpp path_text_shape.hpp poi_symbol_shape.cpp diff --git a/drape_frontend/apply_feature_functors.cpp b/drape_frontend/apply_feature_functors.cpp index 3139c311ba..70a4c9ee11 100644 --- a/drape_frontend/apply_feature_functors.cpp +++ b/drape_frontend/apply_feature_functors.cpp @@ -393,7 +393,7 @@ m2::PointF GetShieldOffset(dp::Anchor anchor, double borderWidth, double borderH return offset; } -void CalculateRoadShieldPositions(std::vector const & offsets, +void CalculateRoadShieldPositions(std::vector const & offsets, m2::SharedSpline const & spline, std::vector & shieldPositions) { @@ -1150,10 +1150,10 @@ void ApplyLineFeatureAdditional::Finish(ref_ptr texMng, { for (auto const & spline : m_clippedSplines) { - float const pixelLength = 300.0f * vs; - std::vector offsets; - PathTextLayout::CalculatePositions(static_cast(spline->GetLength()), - m_currentScaleGtoP, pixelLength, offsets); + double const pixelLength = 300.0 * vs; + std::vector offsets; + PathTextLayout::CalculatePositions(spline->GetLength(), m_currentScaleGtoP, + pixelLength, offsets); if (!offsets.empty()) CalculateRoadShieldPositions(offsets, spline, shieldPositions); } diff --git a/drape_frontend/drape_frontend.pro b/drape_frontend/drape_frontend.pro index 7702640e43..f463bb3a8a 100755 --- a/drape_frontend/drape_frontend.pro +++ b/drape_frontend/drape_frontend.pro @@ -70,6 +70,7 @@ SOURCES += \ overlay_batcher.cpp \ overlays_tracker.cpp \ path_symbol_shape.cpp \ + path_text_handle.cpp \ path_text_shape.cpp \ poi_symbol_shape.cpp \ postprocess_renderer.cpp \ @@ -181,6 +182,7 @@ HEADERS += \ overlay_batcher.hpp \ overlays_tracker.hpp \ path_symbol_shape.hpp \ + path_text_handle.hpp \ path_text_shape.hpp \ poi_symbol_shape.hpp \ postprocess_renderer.hpp \ diff --git a/drape_frontend/path_text_handle.cpp b/drape_frontend/path_text_handle.cpp new file mode 100644 index 0000000000..f1d1033200 --- /dev/null +++ b/drape_frontend/path_text_handle.cpp @@ -0,0 +1,272 @@ +#include "drape_frontend/path_text_handle.hpp" + +namespace df +{ + +PathTextContext::PathTextContext(m2::SharedSpline const & spline) + : m_globalSpline(spline) +{} + +void PathTextContext::SetLayout(drape_ptr && layout, double baseGtoPScale) +{ + m_layout = std::move(layout); + m_globalOffsets.clear(); + m_globalPivots.clear(); + PathTextLayout::CalculatePositions(m_globalSpline->GetLength(), baseGtoPScale, + m_layout->GetPixelLength(), m_globalOffsets); + m_globalPivots.reserve(m_globalOffsets.size()); + for (auto offset : m_globalOffsets) + m_globalPivots.push_back(m_globalSpline->GetPoint(offset).m_pos); +} + +ref_ptr const PathTextContext::GetLayout() const +{ + return make_ref(m_layout); +} + +void PathTextContext::BeforeUpdate() +{ + m_updated = false; +} + +std::vector const & PathTextContext::GetOffsets() const +{ + return m_globalOffsets; +} + +bool PathTextContext::GetPivot(size_t textIndex, m2::PointD & pivot, + m2::Spline::iterator & centerPointIter) const +{ + if (textIndex >= m_centerGlobalPivots.size()) + return false; + + pivot = m_centerGlobalPivots[textIndex]; + centerPointIter = m_centerPointIters[textIndex]; + return true; +} + +void PathTextContext::Update(ScreenBase const & screen) +{ + if (m_updated) + return; + m_updated = true; + + m_pixel3dSplines.clear(); + m_centerPointIters.clear(); + m_centerGlobalPivots.clear(); + + m2::Spline pixelSpline(m_globalSpline->GetSize()); + for (auto pos : m_globalSpline->GetPath()) + { + pos = screen.GtoP(pos); + if (screen.IsReverseProjection3d(pos)) + { + if (pixelSpline.GetSize() > 1) + { + m_pixel3dSplines.push_back(pixelSpline); + pixelSpline.Clear(); + } + continue; + } + pixelSpline.AddPoint(screen.PtoP3d(pos)); + } + + if (pixelSpline.GetSize() > 1) + m_pixel3dSplines.emplace_back(std::move(pixelSpline)); + + if (m_pixel3dSplines.empty()) + return; + + for (size_t i = 0, sz = m_globalOffsets.size(); i < sz; ++i) + { + if (screen.isPerspective()) + { + m2::PointD const pt2d = screen.GtoP(m_globalPivots[i]); + if (!screen.IsReverseProjection3d(pt2d)) + { + m_centerPointIters.push_back(GetProjectedPoint(m_pixel3dSplines, screen.PtoP3d(pt2d))); + m_centerGlobalPivots.push_back(m_globalPivots[i]); + } + } + else + { + m_centerPointIters.push_back(m_pixel3dSplines.front().GetPoint(m_globalOffsets[i] / screen.GetScale())); + m_centerGlobalPivots.push_back(m_globalPivots[i]); + } + } +} + +m2::Spline::iterator PathTextContext::GetProjectedPoint(std::vector const & splines, + m2::PointD const & pt) const +{ + m2::Spline::iterator iter; + double minDist = std::numeric_limits::max(); + + for (auto const & spline : splines) + { + auto const & path = spline.GetPath(); + if (path.size() < 2) + continue; + + double step = 0; + + for (size_t i = 0, sz = path.size(); i < sz - 1; ++i) + { + double const segLength = spline.GetLengths()[i]; + m2::PointD const segDir = spline.GetDirections()[i]; + + m2::PointD v = pt - path[i]; + double const t = m2::DotProduct(segDir, v); + + m2::PointD nearestPt; + double nearestStep; + if (t <= 0) + { + nearestPt = path[i]; + nearestStep = step; + } + else if (t >= segLength) + { + nearestPt = path[i + 1]; + nearestStep = step + segLength; + } + else + { + nearestPt = path[i] + segDir * t; + nearestStep = step + t; + } + + double const dist = pt.SquareLength(nearestPt); + if (dist < minDist) + { + minDist = dist; + iter = spline.GetPoint(nearestStep); + + double const kEps = 1e-5; + if (minDist < kEps) + return iter; + } + + step += segLength; + } + } + + return iter; +} + + +PathTextHandle::PathTextHandle(dp::OverlayID const & id, + std::shared_ptr const & context, + float depth, uint32_t textIndex, + uint64_t priority, int fixedHeight, + ref_ptr textureManager, + bool isBillboard) + : TextHandle(id, context->GetLayout()->GetText(), dp::Center, priority, + fixedHeight, textureManager, isBillboard) + , m_context(context) + , m_textIndex(textIndex) + , m_depth(depth) +{ + m_buffer.resize(4 * m_context->GetLayout()->GetGlyphCount()); +} + +bool PathTextHandle::Update(ScreenBase const & screen) +{ + if (!df::TextHandle::Update(screen)) + return false; + + m_context->Update(screen); + + m2::Spline::iterator centerPointIter; + if (!m_context->GetPivot(m_textIndex, m_globalPivot, centerPointIter)) + return false; + + return m_context->GetLayout()->CacheDynamicGeometry(centerPointIter, m_depth, m_globalPivot, m_buffer); +} + +void PathTextHandle::BeforeUpdate() +{ + m_context->BeforeUpdate(); +} + +m2::RectD PathTextHandle::GetPixelRect(ScreenBase const & screen, bool perspective) const +{ + m2::PointD const pixelPivot(screen.GtoP(m_globalPivot)); + + if (perspective) + { + if (IsBillboard()) + { + m2::RectD r = GetPixelRect(screen, false); + m2::PointD pixelPivotPerspective = screen.PtoP3d(pixelPivot); + r.Offset(-pixelPivot); + r.Offset(pixelPivotPerspective); + + return r; + } + return GetPixelRectPerspective(screen); + } + + m2::RectD result; + for (gpu::TextDynamicVertex const & v : m_buffer) + result.Add(pixelPivot + glsl::ToPoint(v.m_normal)); + + return result; +} + +void PathTextHandle::GetPixelShape(ScreenBase const & screen, bool perspective, Rects & rects) const +{ + m2::PointD const pixelPivot(screen.GtoP(m_globalPivot)); + for (size_t quadIndex = 0; quadIndex < m_buffer.size(); quadIndex += 4) + { + m2::RectF r; + r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex].m_normal)); + r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex + 1].m_normal)); + r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex + 2].m_normal)); + r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex + 3].m_normal)); + + if (perspective) + { + if (IsBillboard()) + { + m2::PointD const pxPivotPerspective = screen.PtoP3d(pixelPivot); + + r.Offset(-pixelPivot); + r.Offset(pxPivotPerspective); + } + else + { + r = m2::RectF(GetPerspectiveRect(m2::RectD(r), screen)); + } + } + + bool const needAddRect = perspective ? !screen.IsReverseProjection3d(r.Center()) : true; + if (needAddRect) + rects.emplace_back(move(r)); + } +} + +void PathTextHandle::GetAttributeMutation(ref_ptr mutator) const +{ + // for visible text paths we always update normals + SetForceUpdateNormals(IsVisible()); + TextHandle::GetAttributeMutation(mutator); +} + +uint64_t PathTextHandle::GetPriorityMask() const +{ + return dp::kPriorityMaskManual | dp::kPriorityMaskRank; +} + +bool PathTextHandle::Enable3dExtention() const +{ + // Do not extend overlays for path texts. + return false; +} + +bool PathTextHandle::HasLinearFeatureShape() const +{ + return true; +} + +} // namespace df diff --git a/drape_frontend/path_text_handle.hpp b/drape_frontend/path_text_handle.hpp new file mode 100644 index 0000000000..d7a644ddaf --- /dev/null +++ b/drape_frontend/path_text_handle.hpp @@ -0,0 +1,74 @@ +#pragma once + +#include "drape_frontend/text_handle.hpp" +#include "drape_frontend/text_layout.hpp" + +#include "geometry/spline.hpp" + +#include +#include +#include + +namespace df +{ +class PathTextContext +{ +public: + PathTextContext(m2::SharedSpline const & spline); + + void SetLayout(drape_ptr && layout, double baseGtoPScale); + ref_ptr const GetLayout() const; + + bool GetPivot(size_t textIndex, m2::PointD & pivot, + m2::Spline::iterator & centerPointIter) const; + + void BeforeUpdate(); + void Update(ScreenBase const & screen); + + std::vector const & GetOffsets() const; + +private: + m2::Spline::iterator GetProjectedPoint(std::vector const & splines, + m2::PointD const & pt) const; + +private: + std::vector m_globalPivots; + std::vector m_globalOffsets; + m2::SharedSpline m_globalSpline; + + std::vector m_pixel3dSplines; + std::vector m_centerPointIters; + std::vector m_centerGlobalPivots; + + drape_ptr m_layout; + bool m_updated = false; +}; + +class PathTextHandle : public df::TextHandle +{ +public: + PathTextHandle(dp::OverlayID const & id, + std::shared_ptr const & context, + float depth, uint32_t textIndex, + uint64_t priority, int fixedHeight, + ref_ptr textureManager, + bool isBillboard); + + void BeforeUpdate() override; + bool Update(ScreenBase const & screen) override; + + m2::RectD GetPixelRect(ScreenBase const & screen, bool perspective) const override; + void GetPixelShape(ScreenBase const & screen, bool perspective, Rects & rects) const override; + void GetAttributeMutation(ref_ptr mutator) const override; + uint64_t GetPriorityMask() const override; + bool Enable3dExtention() const override; + bool HasLinearFeatureShape() const override; + +private: + m2::SharedSpline m_spline; + std::shared_ptr m_context; + uint32_t const m_textIndex; + m2::PointD m_globalPivot; + float const m_depth; +}; +} // namespace df diff --git a/drape_frontend/path_text_shape.cpp b/drape_frontend/path_text_shape.cpp index f426dd36a6..4c6c676c3a 100644 --- a/drape_frontend/path_text_shape.cpp +++ b/drape_frontend/path_text_shape.cpp @@ -1,6 +1,6 @@ #include "drape_frontend/path_text_shape.hpp" +#include "drape_frontend/path_text_handle.hpp" #include "drape_frontend/shader_def.hpp" -#include "drape_frontend/text_handle.hpp" #include "drape/attribute_provider.hpp" #include "drape/batcher.hpp" @@ -18,194 +18,6 @@ using m2::Spline; -namespace -{ -class PathTextHandle : public df::TextHandle -{ -public: - PathTextHandle(dp::OverlayID const & id, m2::SharedSpline const & spl, - df::SharedTextLayout const & layout, - float mercatorOffset, float depth, uint32_t textIndex, - uint64_t priority, int fixedHeight, - ref_ptr textureManager, - bool isBillboard) - : TextHandle(id, layout->GetText(), dp::Center, priority, fixedHeight, - textureManager, isBillboard) - , m_spline(spl) - , m_layout(layout) - , m_textIndex(textIndex) - , m_globalOffset(mercatorOffset) - , m_depth(depth) - { - - m2::Spline::iterator centerPointIter = m_spline.CreateIterator(); - centerPointIter.Advance(m_globalOffset); - m_globalPivot = centerPointIter.m_pos; - m_buffer.resize(4 * m_layout->GetGlyphCount()); - } - - bool Update(ScreenBase const & screen) override - { - if (!df::TextHandle::Update(screen)) - return false; - - auto const & globalPoints = m_spline->GetPath(); - m2::Spline pixelSpline(m_spline->GetSize()); - m2::Spline::iterator centerPointIter; - - if (screen.isPerspective()) - { - // In perspective mode we draw the first label only. - if (m_textIndex != 0) - return false; - - float pixelOffset = 0.0f; - bool foundOffset = false; - for (auto pos : globalPoints) - { - pos = screen.GtoP(pos); - if (!screen.PixelRect().IsPointInside(pos)) - { - if ((foundOffset = CalculatePerspectiveOffsets(pixelSpline, m_textIndex, pixelOffset))) - break; - - pixelSpline = m2::Spline(m_spline->GetSize()); - continue; - } - pixelSpline.AddPoint(screen.PtoP3d(pos)); - } - - // We aren't able to place the only label anywhere. - if (!foundOffset && !CalculatePerspectiveOffsets(pixelSpline, m_textIndex, pixelOffset)) - return false; - - centerPointIter.Attach(pixelSpline); - centerPointIter.Advance(pixelOffset); - m_globalPivot = screen.PtoG(screen.P3dtoP(centerPointIter.m_pos)); - } - else - { - for (auto const & pos : globalPoints) - pixelSpline.AddPoint(screen.GtoP(pos)); - - centerPointIter.Attach(pixelSpline); - centerPointIter.Advance(m_globalOffset / screen.GetScale()); - m_globalPivot = screen.PtoG(centerPointIter.m_pos); - } - - if (m_buffer.empty()) - m_buffer.resize(4 * m_layout->GetGlyphCount()); - return m_layout->CacheDynamicGeometry(centerPointIter, m_depth, m_globalPivot, m_buffer); - } - - m2::RectD GetPixelRect(ScreenBase const & screen, bool perspective) const override - { - m2::PointD const pixelPivot(screen.GtoP(m_globalPivot)); - - if (perspective) - { - if (IsBillboard()) - { - m2::RectD r = GetPixelRect(screen, false); - m2::PointD pixelPivotPerspective = screen.PtoP3d(pixelPivot); - r.Offset(-pixelPivot); - r.Offset(pixelPivotPerspective); - - return r; - } - return GetPixelRectPerspective(screen); - } - - m2::RectD result; - for (gpu::TextDynamicVertex const & v : m_buffer) - result.Add(pixelPivot + glsl::ToPoint(v.m_normal)); - - return result; - } - - void GetPixelShape(ScreenBase const & screen, bool perspective, Rects & rects) const override - { - m2::PointD const pixelPivot(screen.GtoP(m_globalPivot)); - for (size_t quadIndex = 0; quadIndex < m_buffer.size(); quadIndex += 4) - { - m2::RectF r; - r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex].m_normal)); - r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex + 1].m_normal)); - r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex + 2].m_normal)); - r.Add(pixelPivot + glsl::ToPoint(m_buffer[quadIndex + 3].m_normal)); - - if (perspective) - { - if (IsBillboard()) - { - m2::PointD const pxPivotPerspective = screen.PtoP3d(pixelPivot); - - r.Offset(-pixelPivot); - r.Offset(pxPivotPerspective); - } - else - { - r = m2::RectF(GetPerspectiveRect(m2::RectD(r), screen)); - } - } - - bool const needAddRect = perspective ? !screen.IsReverseProjection3d(r.Center()) : true; - if (needAddRect) - rects.emplace_back(move(r)); - } - } - - void GetAttributeMutation(ref_ptr mutator) const override - { - // for visible text paths we always update normals - SetForceUpdateNormals(IsVisible()); - TextHandle::GetAttributeMutation(mutator); - } - - uint64_t GetPriorityMask() const override - { - return dp::kPriorityMaskManual | dp::kPriorityMaskRank; - } - - bool Enable3dExtention() const override - { - // Do not extend overlays for path texts. - return false; - } - - bool HasLinearFeatureShape() const override - { - return true; - } - -private: - bool CalculatePerspectiveOffsets(m2::Spline const & pixelSpline, uint32_t textIndex, - float & pixelOffset) const - { - if (pixelSpline.GetSize() < 2) - return false; - - float offset = 0.0f; - if (!df::PathTextLayout::CalculatePerspectivePosition(static_cast(pixelSpline.GetLength()), - m_layout->GetPixelLength(), textIndex, offset)) - { - return false; - } - - pixelOffset = offset; - return true; - } - -private: - m2::SharedSpline m_spline; - df::SharedTextLayout m_layout; - uint32_t const m_textIndex; - m2::PointD m_globalPivot; - float const m_globalOffset; - float const m_depth; -}; -} // namespace - namespace df { PathTextShape::PathTextShape(m2::SharedSpline const & spline, @@ -215,7 +27,9 @@ PathTextShape::PathTextShape(m2::SharedSpline const & spline, , m_params(params) , m_tileCoords(tileKey.GetTileCoords()) , m_baseTextIndex(baseTextIndex) -{} +{ + m_context.reset(new PathTextContext(m_spline)); +} bool PathTextShape::CalculateLayout(ref_ptr textures) { @@ -223,19 +37,18 @@ bool PathTextShape::CalculateLayout(ref_ptr textures) if (!m_params.m_auxText.empty()) text += " " + m_params.m_auxText; - m_layout = SharedTextLayout(new PathTextLayout(m_params.m_tileCenter, - strings::MakeUniString(text), - m_params.m_textFont.m_size, - m_params.m_textFont.m_isSdf, - textures)); - uint32_t const glyphCount = m_layout->GetGlyphCount(); + auto layout = make_unique_dp(m_params.m_tileCenter, + strings::MakeUniString(text), + m_params.m_textFont.m_size, + m_params.m_textFont.m_isSdf, + textures); + uint32_t const glyphCount = layout->GetGlyphCount(); if (glyphCount == 0) return false; - PathTextLayout::CalculatePositions(static_cast(m_spline->GetLength()), - m_params.m_baseGtoPScale, m_layout->GetPixelLength(), - m_offsets); - return !m_offsets.empty(); + m_context->SetLayout(std::move(layout), m_params.m_baseGtoPScale); + + return !m_context->GetOffsets().empty(); } uint64_t PathTextShape::GetOverlayPriority(uint32_t textIndex, size_t textLength) const @@ -261,29 +74,30 @@ uint64_t PathTextShape::GetOverlayPriority(uint32_t textIndex, size_t textLength void PathTextShape::DrawPathTextPlain(ref_ptr textures, ref_ptr batcher) const { - ASSERT(!m_layout.IsNull(), ()); - ASSERT(!m_offsets.empty(), ()); + auto const layout = m_context->GetLayout(); + ASSERT(layout != nullptr, ()); + ASSERT(!m_context->GetOffsets().empty(), ()); dp::TextureManager::ColorRegion color; textures->GetColorRegion(m_params.m_textFont.m_color, color); - dp::GLState state(m_layout->GetFixedHeight() > 0 ? gpu::TEXT_FIXED_PROGRAM : gpu::TEXT_PROGRAM, + + dp::GLState state(layout->GetFixedHeight() > 0 ? gpu::TEXT_FIXED_PROGRAM : gpu::TEXT_PROGRAM, dp::GLState::OverlayLayer); - state.SetProgram3dIndex(m_layout->GetFixedHeight() > 0 ? gpu::TEXT_FIXED_BILLBOARD_PROGRAM : - gpu::TEXT_BILLBOARD_PROGRAM); + state.SetProgram3dIndex(layout->GetFixedHeight() > 0 ? gpu::TEXT_FIXED_BILLBOARD_PROGRAM : + gpu::TEXT_BILLBOARD_PROGRAM); state.SetColorTexture(color.GetTexture()); - state.SetMaskTexture(m_layout->GetMaskTexture()); + state.SetMaskTexture(layout->GetMaskTexture()); gpu::TTextStaticVertexBuffer staticBuffer; gpu::TTextDynamicVertexBuffer dynBuffer; - for (uint32_t textIndex = 0; textIndex < static_cast(m_offsets.size()); ++textIndex) + for (uint32_t textIndex = 0; textIndex < m_context->GetOffsets().size(); ++textIndex) { - float const offset = m_offsets[textIndex]; staticBuffer.clear(); dynBuffer.clear(); - m_layout->CacheStaticGeometry(color, staticBuffer); + layout->CacheStaticGeometry(color, staticBuffer); dynBuffer.resize(staticBuffer.size()); dp::AttributeProvider provider(2, static_cast(staticBuffer.size())); @@ -292,15 +106,16 @@ void PathTextShape::DrawPathTextPlain(ref_ptr textures, provider.InitStream(1, gpu::TextDynamicVertex::GetBindingInfo(), make_ref(dynBuffer.data())); batcher->InsertListOfStrip(state, make_ref(&provider), - CreateOverlayHandle(m_layout, textIndex, offset, textures), 4); + CreateOverlayHandle(textIndex, textures), 4); } } void PathTextShape::DrawPathTextOutlined(ref_ptr textures, ref_ptr batcher) const { - ASSERT(!m_layout.IsNull(), ()); - ASSERT(!m_offsets.empty(), ()); + auto const layout = m_context->GetLayout(); + ASSERT(layout != nullptr, ()); + ASSERT(!m_context->GetOffsets().empty(), ()); dp::TextureManager::ColorRegion color; dp::TextureManager::ColorRegion outline; @@ -310,17 +125,16 @@ void PathTextShape::DrawPathTextOutlined(ref_ptr textures, dp::GLState state(gpu::TEXT_OUTLINED_PROGRAM, dp::GLState::OverlayLayer); state.SetProgram3dIndex(gpu::TEXT_OUTLINED_BILLBOARD_PROGRAM); state.SetColorTexture(color.GetTexture()); - state.SetMaskTexture(m_layout->GetMaskTexture()); + state.SetMaskTexture(layout->GetMaskTexture()); gpu::TTextOutlinedStaticVertexBuffer staticBuffer; gpu::TTextDynamicVertexBuffer dynBuffer; - for (uint32_t textIndex = 0; textIndex < static_cast(m_offsets.size()); ++textIndex) + for (uint32_t textIndex = 0; textIndex < m_context->GetOffsets().size(); ++textIndex) { - float const offset = m_offsets[textIndex]; staticBuffer.clear(); dynBuffer.clear(); - m_layout->CacheStaticGeometry(color, outline, staticBuffer); + layout->CacheStaticGeometry(color, outline, staticBuffer); dynBuffer.resize(staticBuffer.size()); dp::AttributeProvider provider(2, static_cast(staticBuffer.size())); @@ -329,25 +143,25 @@ void PathTextShape::DrawPathTextOutlined(ref_ptr textures, provider.InitStream(1, gpu::TextDynamicVertex::GetBindingInfo(), make_ref(dynBuffer.data())); batcher->InsertListOfStrip(state, make_ref(&provider), - CreateOverlayHandle(m_layout, textIndex, offset, textures), 4); + CreateOverlayHandle(textIndex, textures), 4); } } -drape_ptr PathTextShape::CreateOverlayHandle(SharedTextLayout const & layoutPtr, - uint32_t textIndex, float offset, +drape_ptr PathTextShape::CreateOverlayHandle(uint32_t textIndex, ref_ptr textures) const { dp::OverlayID const overlayId = dp::OverlayID(m_params.m_featureID, m_tileCoords, m_baseTextIndex + textIndex); - auto const priority = GetOverlayPriority(textIndex, layoutPtr->GetText().size()); - return make_unique_dp(overlayId, m_spline, layoutPtr, offset, m_params.m_depth, - textIndex, priority, layoutPtr->GetFixedHeight(), + auto const layout = m_context->GetLayout(); + auto const priority = GetOverlayPriority(textIndex, layout->GetText().size()); + return make_unique_dp(overlayId, m_context, m_params.m_depth, + textIndex, priority, layout->GetFixedHeight(), textures, true /* isBillboard */); } void PathTextShape::Draw(ref_ptr batcher, ref_ptr textures) const { - if (m_layout.IsNull() || m_offsets.empty()) + if (m_context->GetLayout() == nullptr || m_context->GetOffsets().empty()) return; if (m_params.m_textFont.m_outlineColor == dp::Color::Transparent()) diff --git a/drape_frontend/path_text_shape.hpp b/drape_frontend/path_text_shape.hpp index 93e588f7f7..6dc6a6f5dd 100644 --- a/drape_frontend/path_text_shape.hpp +++ b/drape_frontend/path_text_shape.hpp @@ -1,6 +1,7 @@ #pragma once #include "drape_frontend/map_shape.hpp" +#include "drape_frontend/path_text_handle.hpp" #include "drape_frontend/shape_view_params.hpp" #include "drape_frontend/text_layout.hpp" @@ -23,7 +24,7 @@ public: TileKey const & tileKey, uint32_t baseTextIndex); bool CalculateLayout(ref_ptr textures); - std::vector GetOffsets() const { return m_offsets; } + std::vector GetOffsets() const { return m_context->GetOffsets(); } void Draw(ref_ptr batcher, ref_ptr textures) const override; MapShapeType GetType() const override { return MapShapeType::OverlayType; } @@ -35,15 +36,13 @@ private: ref_ptr batcher) const; void DrawPathTextOutlined(ref_ptr textures, ref_ptr batcher) const; - drape_ptr CreateOverlayHandle(SharedTextLayout const & layoutPtr, - uint32_t textIndex, float offset, + drape_ptr CreateOverlayHandle(uint32_t textIndex, ref_ptr textures) const; m2::SharedSpline m_spline; PathTextViewParams m_params; m2::PointI const m_tileCoords; uint32_t const m_baseTextIndex; - SharedTextLayout m_layout; - std::vector m_offsets; + std::shared_ptr m_context; }; } // namespace df diff --git a/drape_frontend/text_layout.cpp b/drape_frontend/text_layout.cpp index eb4d22ac36..11df700bec 100644 --- a/drape_frontend/text_layout.cpp +++ b/drape_frontend/text_layout.cpp @@ -279,10 +279,10 @@ void CalculateOffsets(dp::Anchor anchor, float textRatio, pixelSize = m2::PointF(maxLength, summaryHeight); } -float GetTextMinPeriod(float pixelTextLength) +double GetTextMinPeriod(double pixelTextLength) { - float const vs = static_cast(df::VisualParams::Instance().GetVisualScale()); - float const etalonEmpty = std::max(300.0f * vs, pixelTextLength); + double const vs = df::VisualParams::Instance().GetVisualScale(); + double const etalonEmpty = std::max(300.0 * vs, pixelTextLength); return etalonEmpty + pixelTextLength; } } // namespace @@ -479,37 +479,26 @@ bool PathTextLayout::CacheDynamicGeometry(m2::Spline::iterator const & iter, flo return true; } -float PathTextLayout::CalculateTextLength(float textPixelLength) +double PathTextLayout::CalculateTextLength(double textPixelLength) { // We leave a little space on each side of the text. - float const kTextBorder = 4.0f; + double const kTextBorder = 4.0; return kTextBorder + textPixelLength; } -bool PathTextLayout::CalculatePerspectivePosition(float pixelSplineLength, float textPixelLength, - uint32_t textIndex, float & offset) +void PathTextLayout::CalculatePositions(double splineLength, double splineScaleToPixel, + double textPixelLength, std::vector & offsets) { - UNUSED_VALUE(textIndex); - float const textLength = CalculateTextLength(textPixelLength); - if (textLength > pixelSplineLength * 2.0f) - return false; - offset = pixelSplineLength * 0.5f; - return true; -} - -void PathTextLayout::CalculatePositions(float splineLength, float splineScaleToPixel, - float textPixelLength, std::vector & offsets) -{ - float const textLength = CalculateTextLength(textPixelLength); + double const textLength = CalculateTextLength(textPixelLength); // On the next scale m_scaleGtoP will be twice. if (textLength > splineLength * 2.0f * splineScaleToPixel) return; - float const kPathLengthScalar = 0.75; - float const pathLength = kPathLengthScalar * splineScaleToPixel * splineLength; - float const minPeriodSize = GetTextMinPeriod(textLength); - float const twoTextsAndEmpty = minPeriodSize + textLength; + double const kPathLengthScalar = 0.75; + double const pathLength = kPathLengthScalar * splineScaleToPixel * splineLength; + double const minPeriodSize = GetTextMinPeriod(textLength); + double const twoTextsAndEmpty = minPeriodSize + textLength; if (pathLength < twoTextsAndEmpty) { @@ -519,39 +508,11 @@ void PathTextLayout::CalculatePositions(float splineLength, float splineScaleToP } else { - double const textCount = std::max(floor(static_cast(pathLength / minPeriodSize)), 1.0); + double const textCount = std::max(floor(pathLength / minPeriodSize), 1.0); double const glbTextLen = splineLength / textCount; for (double offset = 0.5 * glbTextLen; offset < splineLength; offset += glbTextLen) - offsets.push_back(static_cast(offset)); + offsets.push_back(offset); } } -SharedTextLayout::SharedTextLayout(PathTextLayout * layout) - : m_layout(layout) -{} - -bool SharedTextLayout::IsNull() const -{ - return m_layout == nullptr; -} - -void SharedTextLayout::Reset(PathTextLayout * layout) -{ - m_layout.reset(layout); -} - -PathTextLayout * SharedTextLayout::GetRaw() -{ - return m_layout.get(); -} - -PathTextLayout * SharedTextLayout::operator->() -{ - return m_layout.get(); -} - -PathTextLayout const * SharedTextLayout::operator->() const -{ - return m_layout.get(); -} } // namespace df diff --git a/drape_frontend/text_layout.hpp b/drape_frontend/text_layout.hpp index e22fb15b59..5c46dd6837 100644 --- a/drape_frontend/text_layout.hpp +++ b/drape_frontend/text_layout.hpp @@ -94,31 +94,12 @@ public: m2::PointD const & globalPivot, gpu::TTextDynamicVertexBuffer & buffer) const; - static bool CalculatePerspectivePosition(float pixelSplineLength, float textPixelLength, - uint32_t textIndex, float & offset); - - static void CalculatePositions(float splineLength, float splineScaleToPixel, - float textPixelLength, std::vector & offsets); + static void CalculatePositions(double splineLength, double splineScaleToPixel, + double textPixelLength, std::vector & offsets); private: - static float CalculateTextLength(float textPixelLength); + static double CalculateTextLength(double textPixelLength); m2::PointD m_tileCenter; }; -class SharedTextLayout -{ -public: - SharedTextLayout() = default; - SharedTextLayout(PathTextLayout * layout); - - bool IsNull() const; - void Reset(PathTextLayout * layout); - PathTextLayout * GetRaw(); - - PathTextLayout * operator->(); - PathTextLayout const * operator->() const; - -private: - std::shared_ptr m_layout; -}; } // namespace df diff --git a/geometry/spline.cpp b/geometry/spline.cpp index c74df1013a..3ea90bc752 100644 --- a/geometry/spline.cpp +++ b/geometry/spline.cpp @@ -116,6 +116,13 @@ double Spline::GetLength() const return accumulate(m_length.begin(), m_length.end(), 0.0); } +void Spline::Clear() +{ + m_position.clear(); + m_direction.clear(); + m_length.clear(); +} + Spline::iterator::iterator() : m_checker(false) , m_spl(NULL) diff --git a/geometry/spline.hpp b/geometry/spline.hpp index 2ad552d4e0..07132fe173 100644 --- a/geometry/spline.hpp +++ b/geometry/spline.hpp @@ -56,6 +56,8 @@ public: size_t GetSize() const; vector const & GetPath() const { return m_position; } vector const & GetLengths() const { return m_length; } + vector const & GetDirections() const { return m_direction; } + void Clear(); iterator GetPoint(double step) const; diff --git a/xcode/drape_frontend/drape_frontend.xcodeproj/project.pbxproj b/xcode/drape_frontend/drape_frontend.xcodeproj/project.pbxproj index 0a328cba48..a2447b69c6 100644 --- a/xcode/drape_frontend/drape_frontend.xcodeproj/project.pbxproj +++ b/xcode/drape_frontend/drape_frontend.xcodeproj/project.pbxproj @@ -49,7 +49,6 @@ 45580ABB1E28DB2600CD535D /* scenario_manager.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 45580AB91E28DB2600CD535D /* scenario_manager.hpp */; }; 456B3F991ED464FE009B3D1F /* postprocess_renderer.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 456B3F971ED464FE009B3D1F /* postprocess_renderer.cpp */; }; 456B3F9A1ED464FE009B3D1F /* postprocess_renderer.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 456B3F981ED464FE009B3D1F /* postprocess_renderer.hpp */; }; - 457D89251E7AE89500049500 /* custom_symbol.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 457D89241E7AE89500049500 /* custom_symbol.hpp */; }; 45B4B8CB1CF5C16B00A54761 /* screen_animations.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 45B4B8C71CF5C16B00A54761 /* screen_animations.cpp */; }; 45B4B8CC1CF5C16B00A54761 /* screen_animations.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 45B4B8C81CF5C16B00A54761 /* screen_animations.hpp */; }; 45B4B8CD1CF5C16B00A54761 /* screen_operations.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 45B4B8C91CF5C16B00A54761 /* screen_operations.cpp */; }; @@ -60,7 +59,6 @@ 56BF56DB1C7608C0006DD7CB /* choose_position_mark.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56BF56D91C7608C0006DD7CB /* choose_position_mark.hpp */; }; 56D545661C74A44900E3719C /* overlay_batcher.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56D545641C74A44900E3719C /* overlay_batcher.cpp */; }; 56D545671C74A44900E3719C /* overlay_batcher.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56D545651C74A44900E3719C /* overlay_batcher.hpp */; }; - 670947911BDF9BE1005014C0 /* intrusive_vector.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 670947421BDF9BE0005014C0 /* intrusive_vector.hpp */; }; 670947921BDF9BE1005014C0 /* kinetic_scroller.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670947431BDF9BE0005014C0 /* kinetic_scroller.cpp */; }; 670947931BDF9BE1005014C0 /* kinetic_scroller.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 670947441BDF9BE0005014C0 /* kinetic_scroller.hpp */; }; 670947941BDF9BE1005014C0 /* line_shape_helper.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 670947451BDF9BE0005014C0 /* line_shape_helper.cpp */; }; @@ -215,6 +213,9 @@ 67E91C7E1BDFC85E005CEE88 /* visual_params.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 6709478F1BDF9BE1005014C0 /* visual_params.cpp */; }; BB035F6F1E3A2AAE00519962 /* drape_measurer.cpp in Sources */ = {isa = PBXBuildFile; fileRef = BB035F6D1E3A2AAE00519962 /* drape_measurer.cpp */; }; BB035F701E3A2AAE00519962 /* drape_measurer.hpp in Headers */ = {isa = PBXBuildFile; fileRef = BB035F6E1E3A2AAE00519962 /* drape_measurer.hpp */; }; + BB7D67D21F34A62C002FD122 /* custom_features_context.hpp in Headers */ = {isa = PBXBuildFile; fileRef = BB7D67CF1F34A62C002FD122 /* custom_features_context.hpp */; }; + BB7D67D31F34A62C002FD122 /* path_text_handle.cpp in Sources */ = {isa = PBXBuildFile; fileRef = BB7D67D01F34A62C002FD122 /* path_text_handle.cpp */; }; + BB7D67D41F34A62C002FD122 /* path_text_handle.hpp in Headers */ = {isa = PBXBuildFile; fileRef = BB7D67D11F34A62C002FD122 /* path_text_handle.hpp */; }; BBD8F8791E96A51A00BAEB72 /* drape_hints.hpp in Headers */ = {isa = PBXBuildFile; fileRef = BBD8F8781E96A51A00BAEB72 /* drape_hints.hpp */; }; BBD9E2D31F009D9E00DF189A /* user_mark_generator.cpp in Sources */ = {isa = PBXBuildFile; fileRef = BBD9E2D11F009D9E00DF189A /* user_mark_generator.cpp */; }; BBD9E2D41F009D9E00DF189A /* user_mark_generator.hpp in Headers */ = {isa = PBXBuildFile; fileRef = BBD9E2D21F009D9E00DF189A /* user_mark_generator.hpp */; }; @@ -275,7 +276,6 @@ 4560692B1EB9F9D2009AB7B7 /* shaders_lib.glsl */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; name = shaders_lib.glsl; path = shaders/shaders_lib.glsl; sourceTree = ""; }; 456B3F971ED464FE009B3D1F /* postprocess_renderer.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = postprocess_renderer.cpp; sourceTree = ""; }; 456B3F981ED464FE009B3D1F /* postprocess_renderer.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = postprocess_renderer.hpp; sourceTree = ""; }; - 457D89241E7AE89500049500 /* custom_symbol.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = custom_symbol.hpp; sourceTree = ""; }; 45B4B8C71CF5C16B00A54761 /* screen_animations.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = screen_animations.cpp; sourceTree = ""; }; 45B4B8C81CF5C16B00A54761 /* screen_animations.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = screen_animations.hpp; sourceTree = ""; }; 45B4B8C91CF5C16B00A54761 /* screen_operations.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = screen_operations.cpp; sourceTree = ""; }; @@ -341,7 +341,6 @@ 56D545641C74A44900E3719C /* overlay_batcher.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = overlay_batcher.cpp; sourceTree = ""; }; 56D545651C74A44900E3719C /* overlay_batcher.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = overlay_batcher.hpp; sourceTree = ""; }; 6709473A1BDF9B82005014C0 /* libdrape_frontend.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = libdrape_frontend.a; sourceTree = BUILT_PRODUCTS_DIR; }; - 670947421BDF9BE0005014C0 /* intrusive_vector.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = intrusive_vector.hpp; sourceTree = ""; }; 670947431BDF9BE0005014C0 /* kinetic_scroller.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = kinetic_scroller.cpp; sourceTree = ""; }; 670947441BDF9BE0005014C0 /* kinetic_scroller.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = kinetic_scroller.hpp; sourceTree = ""; }; 670947451BDF9BE0005014C0 /* line_shape_helper.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = line_shape_helper.cpp; sourceTree = ""; }; @@ -497,6 +496,9 @@ 677A2DE41C0DD55D00635A00 /* requested_tiles.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = requested_tiles.hpp; sourceTree = ""; }; BB035F6D1E3A2AAE00519962 /* drape_measurer.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = drape_measurer.cpp; sourceTree = ""; }; BB035F6E1E3A2AAE00519962 /* drape_measurer.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = drape_measurer.hpp; sourceTree = ""; }; + BB7D67CF1F34A62C002FD122 /* custom_features_context.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = custom_features_context.hpp; sourceTree = ""; }; + BB7D67D01F34A62C002FD122 /* path_text_handle.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = path_text_handle.cpp; sourceTree = ""; }; + BB7D67D11F34A62C002FD122 /* path_text_handle.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = path_text_handle.hpp; sourceTree = ""; }; BBD8F8781E96A51A00BAEB72 /* drape_hints.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = drape_hints.hpp; sourceTree = ""; }; BBD9E2D11F009D9E00DF189A /* user_mark_generator.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = user_mark_generator.cpp; sourceTree = ""; }; BBD9E2D21F009D9E00DF189A /* user_mark_generator.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = user_mark_generator.hpp; sourceTree = ""; }; @@ -613,6 +615,7 @@ EB2B77FE1EEDD418002697B6 /* metaline_manager.hpp */, 451A2A801EE8464E003E05A4 /* circles_pack_shape.cpp */, 451A2A811EE8464E003E05A4 /* circles_pack_shape.hpp */, + BB7D67CF1F34A62C002FD122 /* custom_features_context.hpp */, 456B3F971ED464FE009B3D1F /* postprocess_renderer.cpp */, 456B3F981ED464FE009B3D1F /* postprocess_renderer.hpp */, 45BB025B1EB8BE5200FE5C0C /* shader_def.cpp */, @@ -620,7 +623,6 @@ 45BB02221EB8BE1400FE5C0C /* shaders */, 672D24981E892768004BB7B1 /* overlays_tracker.cpp */, 672D24991E892768004BB7B1 /* overlays_tracker.hpp */, - 457D89241E7AE89500049500 /* custom_symbol.hpp */, 453EEA6B1E3A28F400505E09 /* colored_symbol_shape.cpp */, 453EEA6C1E3A28F400505E09 /* colored_symbol_shape.hpp */, 347F52071DC2334A0064B273 /* drape_api_builder.cpp */, @@ -680,7 +682,6 @@ 670947E11BDF9BEC005014C0 /* engine_context.hpp */, 670947E21BDF9BEC005014C0 /* frontend_renderer.cpp */, 670947E31BDF9BEC005014C0 /* frontend_renderer.hpp */, - 670947421BDF9BE0005014C0 /* intrusive_vector.hpp */, 670947431BDF9BE0005014C0 /* kinetic_scroller.cpp */, 670947441BDF9BE0005014C0 /* kinetic_scroller.hpp */, 670947451BDF9BE0005014C0 /* line_shape_helper.cpp */, @@ -704,6 +705,8 @@ 670947591BDF9BE0005014C0 /* navigator.hpp */, 6709475A1BDF9BE0005014C0 /* path_symbol_shape.cpp */, 6709475B1BDF9BE0005014C0 /* path_symbol_shape.hpp */, + BB7D67D01F34A62C002FD122 /* path_text_handle.cpp */, + BB7D67D11F34A62C002FD122 /* path_text_handle.hpp */, 6709475C1BDF9BE0005014C0 /* path_text_shape.cpp */, 6709475D1BDF9BE0005014C0 /* path_text_shape.hpp */, 6709475E1BDF9BE0005014C0 /* poi_symbol_shape.cpp */, @@ -864,6 +867,7 @@ files = ( 670947971BDF9BE1005014C0 /* line_shape.hpp in Headers */, 56BF56DB1C7608C0006DD7CB /* choose_position_mark.hpp in Headers */, + BB7D67D21F34A62C002FD122 /* custom_features_context.hpp in Headers */, EB2B78001EEDD418002697B6 /* metaline_manager.hpp in Headers */, 45580ABB1E28DB2600CD535D /* scenario_manager.hpp in Headers */, 670948031BDF9BF5005014C0 /* drape_engine.hpp in Headers */, @@ -903,7 +907,6 @@ 670947AA1BDF9BE1005014C0 /* path_symbol_shape.hpp in Headers */, 6709483E1BDF9C48005014C0 /* copyright_label.hpp in Headers */, 675D21921BFB871D00717E4F /* text_engine.h in Headers */, - 457D89251E7AE89500049500 /* custom_symbol.hpp in Headers */, 675D218D1BFB871D00717E4F /* proto_to_styles.hpp in Headers */, 670947951BDF9BE1005014C0 /* line_shape_helper.hpp in Headers */, EB2B78041EEDD439002697B6 /* read_metaline_task.hpp in Headers */, @@ -941,7 +944,6 @@ 670E393B1C46C59000E9C0A6 /* batch_merge_helper.hpp in Headers */, 670947931BDF9BE1005014C0 /* kinetic_scroller.hpp in Headers */, 670947E71BDF9BEC005014C0 /* frontend_renderer.hpp in Headers */, - 670947911BDF9BE1005014C0 /* intrusive_vector.hpp in Headers */, 670947F71BDF9BF5005014C0 /* apply_feature_functors.hpp in Headers */, 670947CD1BDF9BE1005014C0 /* tile_info.hpp in Headers */, 6709486B1BDF9C7F005014C0 /* circle_info.hpp in Headers */, @@ -956,6 +958,7 @@ 670948201BDF9C39005014C0 /* value_mapping.hpp in Headers */, 670948441BDF9C48005014C0 /* drape_gui.hpp in Headers */, 670947B41BDF9BE1005014C0 /* render_group.hpp in Headers */, + BB7D67D41F34A62C002FD122 /* path_text_handle.hpp in Headers */, 452C9EE51CEDCF3200A55E57 /* sequence_animation.hpp in Headers */, 670948771BDF9C7F005014C0 /* glyph_cache_impl.hpp in Headers */, 670947A11BDF9BE1005014C0 /* message_subclasses.hpp in Headers */, @@ -1127,6 +1130,7 @@ 670947AD1BDF9BE1005014C0 /* poi_symbol_shape.cpp in Sources */, 6709479F1BDF9BE1005014C0 /* message_queue.cpp in Sources */, 452C9EE41CEDCF3200A55E57 /* sequence_animation.cpp in Sources */, + BB7D67D31F34A62C002FD122 /* path_text_handle.cpp in Sources */, 452C9ED41CEDCF3200A55E57 /* animation.cpp in Sources */, 670947A71BDF9BE1005014C0 /* navigator.cpp in Sources */, 6709484F1BDF9C48005014C0 /* skin.cpp in Sources */,