forked from organicmaps/organicmaps-tmp
Path text rendering without perspective deformation.
Conflicts: drape_frontend/path_text_shape.cpp
This commit is contained in:
parent
82683cbfaa
commit
c2e1b3132d
6 changed files with 174 additions and 88 deletions
|
@ -1,7 +1,6 @@
|
|||
#include "drape_frontend/path_text_shape.hpp"
|
||||
#include "drape_frontend/text_handle.hpp"
|
||||
#include "drape_frontend/text_layout.hpp"
|
||||
#include "drape_frontend/visual_params.hpp"
|
||||
#include "drape_frontend/intrusive_vector.hpp"
|
||||
|
||||
#include "drape/attribute_provider.hpp"
|
||||
|
@ -32,16 +31,24 @@ class PathTextHandle : public df::TextHandle
|
|||
public:
|
||||
PathTextHandle(m2::SharedSpline const & spl,
|
||||
df::SharedTextLayout const & layout,
|
||||
float const mercatorOffset, float const depth,
|
||||
float const mercatorOffset,
|
||||
float depth,
|
||||
uint32_t textIndex,
|
||||
uint64_t priority,
|
||||
ref_ptr<dp::TextureManager> textureManager)
|
||||
: TextHandle(FeatureID(), layout->GetText(), dp::Center, priority, textureManager)
|
||||
ref_ptr<dp::TextureManager> textureManager,
|
||||
bool isBillboard)
|
||||
: TextHandle(FeatureID(), layout->GetText(), dp::Center, priority, textureManager, isBillboard)
|
||||
, m_spline(spl)
|
||||
, m_layout(layout)
|
||||
, m_textIndex(textIndex)
|
||||
, m_globalOffset(mercatorOffset)
|
||||
, m_depth(depth)
|
||||
{
|
||||
m_centerPointIter = m_spline.CreateIterator();
|
||||
m_centerPointIter.Advance(mercatorOffset);
|
||||
|
||||
m2::Spline::iterator centerPointIter = m_spline.CreateIterator();
|
||||
centerPointIter.Advance(m_globalOffset);
|
||||
m_globalPivot = centerPointIter.m_pos;
|
||||
m_buffer.resize(4 * m_layout->GetGlyphCount());
|
||||
}
|
||||
|
||||
double GetMinScaleInPerspective() const override { return 0.5; }
|
||||
|
@ -51,20 +58,70 @@ public:
|
|||
if (!df::TextHandle::Update(screen))
|
||||
return false;
|
||||
|
||||
if (m_normals.empty())
|
||||
m_normals.resize(4 * m_layout->GetGlyphCount());
|
||||
if (m_buffer.empty())
|
||||
m_buffer.resize(4 * m_layout->GetGlyphCount());
|
||||
|
||||
m2::Spline pixelSpline(m_spline->GetSize());
|
||||
|
||||
return m_layout->CacheDynamicGeometry(m_centerPointIter, m_depth, screen, m_normals);
|
||||
vector<m2::PointD> const & globalPoints = m_spline->GetPath();
|
||||
for (auto pos : globalPoints)
|
||||
{
|
||||
pos = screen.GtoP(pos);
|
||||
|
||||
if (screen.isPerspective())
|
||||
{
|
||||
if (!screen.PixelRect().IsPointInside(pos))
|
||||
continue;
|
||||
|
||||
pos = screen.PtoP3d(pos);
|
||||
}
|
||||
|
||||
pixelSpline.AddPoint(pos);
|
||||
}
|
||||
|
||||
m2::Spline::iterator centerPointIter;
|
||||
centerPointIter.Attach(pixelSpline);
|
||||
|
||||
if (screen.isPerspective())
|
||||
{
|
||||
vector<float> offsets;
|
||||
df::PathTextLayout::CalculatePositions(offsets, pixelSpline.GetLength(), 1.0,
|
||||
m_layout->GetPixelLength());
|
||||
|
||||
if (offsets.size() <= m_textIndex)
|
||||
return false;
|
||||
|
||||
centerPointIter.Advance(offsets[m_textIndex]);
|
||||
m_globalPivot = screen.PtoG(screen.P3dtoP(centerPointIter.m_pos));
|
||||
}
|
||||
else
|
||||
{
|
||||
centerPointIter.Advance(m_globalOffset / screen.GetScale());
|
||||
m_globalPivot = screen.PtoG(centerPointIter.m_pos);
|
||||
}
|
||||
return m_layout->CacheDynamicGeometry(centerPointIter, m_depth, m_globalPivot, m_buffer);
|
||||
}
|
||||
|
||||
m2::RectD GetPixelRect(ScreenBase const & screen, bool perspective) const override
|
||||
{
|
||||
if (perspective)
|
||||
return GetPixelRectPerspective(screen);
|
||||
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::PointD const pixelPivot(screen.GtoP(m_centerPointIter.m_pos));
|
||||
m2::RectD result;
|
||||
for (gpu::TextDynamicVertex const & v : m_normals)
|
||||
for (gpu::TextDynamicVertex const & v : m_buffer)
|
||||
result.Add(pixelPivot + glsl::ToPoint(v.m_normal));
|
||||
|
||||
return result;
|
||||
|
@ -72,19 +129,29 @@ public:
|
|||
|
||||
void GetPixelShape(ScreenBase const & screen, Rects & rects, bool perspective) const override
|
||||
{
|
||||
m2::PointD const pixelPivot(screen.GtoP(m_centerPointIter.m_pos));
|
||||
for (size_t quadIndex = 0; quadIndex < m_normals.size(); quadIndex += 4)
|
||||
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_normals[quadIndex].m_normal));
|
||||
r.Add(pixelPivot + glsl::ToPoint(m_normals[quadIndex + 1].m_normal));
|
||||
r.Add(pixelPivot + glsl::ToPoint(m_normals[quadIndex + 2].m_normal));
|
||||
r.Add(pixelPivot + glsl::ToPoint(m_normals[quadIndex + 3].m_normal));
|
||||
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));
|
||||
}
|
||||
|
||||
m2::RectD const screenRect = perspective ? screen.PixelRectIn3d() : screen.PixelRect();
|
||||
if (perspective)
|
||||
r = m2::RectF(GetPerspectiveRect(m2::RectD(r), screen));
|
||||
|
||||
if (screenRect.IsIntersect(m2::RectD(r)))
|
||||
rects.emplace_back(move(r));
|
||||
}
|
||||
|
@ -105,9 +172,11 @@ public:
|
|||
|
||||
private:
|
||||
m2::SharedSpline m_spline;
|
||||
m2::Spline::iterator m_centerPointIter;
|
||||
float const m_depth;
|
||||
df::SharedTextLayout m_layout;
|
||||
uint32_t const m_textIndex;
|
||||
m2::PointD m_globalPivot;
|
||||
float const m_globalOffset;
|
||||
float const m_depth;
|
||||
};
|
||||
|
||||
}
|
||||
|
@ -137,12 +206,13 @@ uint64_t PathTextShape::GetOverlayPriority() const
|
|||
void PathTextShape::DrawPathTextPlain(ref_ptr<dp::TextureManager> textures,
|
||||
ref_ptr<dp::Batcher> batcher,
|
||||
unique_ptr<PathTextLayout> && layout,
|
||||
buffer_vector<float, 32> const & offsets) const
|
||||
vector<float> const & offsets) const
|
||||
{
|
||||
dp::TextureManager::ColorRegion color;
|
||||
textures->GetColorRegion(m_params.m_textFont.m_color, color);
|
||||
|
||||
dp::GLState state(gpu::TEXT_PROGRAM, dp::GLState::OverlayLayer);
|
||||
state.SetProgram3dIndex(gpu::TEXT_BILLBOARD_PROGRAM);
|
||||
state.SetColorTexture(color.GetTexture());
|
||||
state.SetMaskTexture(layout->GetMaskTexture());
|
||||
|
||||
|
@ -150,15 +220,13 @@ void PathTextShape::DrawPathTextPlain(ref_ptr<dp::TextureManager> textures,
|
|||
gpu::TTextStaticVertexBuffer staticBuffer;
|
||||
gpu::TTextDynamicVertexBuffer dynBuffer;
|
||||
SharedTextLayout layoutPtr(layout.release());
|
||||
for (float offset : offsets)
|
||||
for (size_t textIndex = 0; textIndex < offsets.size(); ++textIndex)
|
||||
{
|
||||
float offset = offsets[textIndex];
|
||||
staticBuffer.clear();
|
||||
dynBuffer.clear();
|
||||
|
||||
Spline::iterator iter = m_spline.CreateIterator();
|
||||
iter.Advance(offset);
|
||||
layoutPtr->CacheStaticGeometry(color, staticBuffer);
|
||||
|
||||
dynBuffer.resize(staticBuffer.size());
|
||||
|
||||
dp::AttributeProvider provider(2, staticBuffer.size());
|
||||
|
@ -167,8 +235,10 @@ void PathTextShape::DrawPathTextPlain(ref_ptr<dp::TextureManager> textures,
|
|||
|
||||
drape_ptr<dp::OverlayHandle> handle = make_unique_dp<PathTextHandle>(m_spline, layoutPtr, offset,
|
||||
m_params.m_depth,
|
||||
textIndex,
|
||||
GetOverlayPriority(),
|
||||
textures);
|
||||
textures,
|
||||
true);
|
||||
batcher->InsertListOfStrip(state, make_ref(&provider), move(handle), 4);
|
||||
}
|
||||
}
|
||||
|
@ -176,7 +246,7 @@ void PathTextShape::DrawPathTextPlain(ref_ptr<dp::TextureManager> textures,
|
|||
void PathTextShape::DrawPathTextOutlined(ref_ptr<dp::TextureManager> textures,
|
||||
ref_ptr<dp::Batcher> batcher,
|
||||
unique_ptr<PathTextLayout> && layout,
|
||||
buffer_vector<float, 32> const & offsets) const
|
||||
vector<float> const & offsets) const
|
||||
{
|
||||
dp::TextureManager::ColorRegion color;
|
||||
dp::TextureManager::ColorRegion outline;
|
||||
|
@ -184,6 +254,7 @@ void PathTextShape::DrawPathTextOutlined(ref_ptr<dp::TextureManager> textures,
|
|||
textures->GetColorRegion(m_params.m_textFont.m_outlineColor, outline);
|
||||
|
||||
dp::GLState state(gpu::TEXT_OUTLINED_PROGRAM, dp::GLState::OverlayLayer);
|
||||
state.SetProgram3dIndex(gpu::TEXT_OUTLINED_BILLBOARD_PROGRAM);
|
||||
state.SetColorTexture(color.GetTexture());
|
||||
state.SetMaskTexture(layout->GetMaskTexture());
|
||||
|
||||
|
@ -191,15 +262,13 @@ void PathTextShape::DrawPathTextOutlined(ref_ptr<dp::TextureManager> textures,
|
|||
gpu::TTextOutlinedStaticVertexBuffer staticBuffer;
|
||||
gpu::TTextDynamicVertexBuffer dynBuffer;
|
||||
SharedTextLayout layoutPtr(layout.release());
|
||||
for (float offset : offsets)
|
||||
for (size_t textIndex = 0; textIndex < offsets.size(); ++textIndex)
|
||||
{
|
||||
float offset = offsets[textIndex];
|
||||
staticBuffer.clear();
|
||||
dynBuffer.clear();
|
||||
|
||||
Spline::iterator iter = m_spline.CreateIterator();
|
||||
iter.Advance(offset);
|
||||
layoutPtr->CacheStaticGeometry(color, outline, staticBuffer);
|
||||
|
||||
dynBuffer.resize(staticBuffer.size());
|
||||
|
||||
dp::AttributeProvider provider(2, staticBuffer.size());
|
||||
|
@ -208,8 +277,10 @@ void PathTextShape::DrawPathTextOutlined(ref_ptr<dp::TextureManager> textures,
|
|||
|
||||
drape_ptr<dp::OverlayHandle> handle = make_unique_dp<PathTextHandle>(m_spline, layoutPtr, offset,
|
||||
m_params.m_depth,
|
||||
textIndex,
|
||||
GetOverlayPriority(),
|
||||
textures);
|
||||
textures,
|
||||
true);
|
||||
batcher->InsertListOfStrip(state, make_ref(&provider), move(handle), 4);
|
||||
}
|
||||
}
|
||||
|
@ -223,38 +294,12 @@ void PathTextShape::Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManage
|
|||
if (glyphCount == 0)
|
||||
return;
|
||||
|
||||
//we leave a little space on either side of the text that would
|
||||
//remove the comparison for equality of spline portions
|
||||
float const kTextBorder = 4.0f;
|
||||
float const textLength = kTextBorder + layout->GetPixelLength();
|
||||
float const pathGlbLength = m_spline->GetLength();
|
||||
|
||||
// on next readable scale m_scaleGtoP will be twice
|
||||
if (textLength > pathGlbLength * 2.0 * m_params.m_baseGtoPScale)
|
||||
vector<float> offsets;
|
||||
PathTextLayout::CalculatePositions(offsets, m_spline->GetLength(), m_params.m_baseGtoPScale,
|
||||
layout->GetPixelLength());
|
||||
if (offsets.empty())
|
||||
return;
|
||||
|
||||
float const kPathLengthScalar = 0.75;
|
||||
float const pathLength = kPathLengthScalar * m_params.m_baseGtoPScale * pathGlbLength;
|
||||
|
||||
float const etalonEmpty = max(300 * df::VisualParams::Instance().GetVisualScale(), (double)textLength);
|
||||
float const minPeriodSize = etalonEmpty + textLength;
|
||||
float const twoTextAndEmpty = minPeriodSize + textLength;
|
||||
|
||||
buffer_vector<float, 32> offsets;
|
||||
if (pathLength < twoTextAndEmpty)
|
||||
{
|
||||
// if we can't place 2 text and empty part on path
|
||||
// we place only one text on center of path
|
||||
offsets.push_back(pathGlbLength / 2.0f);
|
||||
}
|
||||
else
|
||||
{
|
||||
double const textCount = max(floor(pathLength / minPeriodSize), 1.0);
|
||||
double const glbTextLen = pathGlbLength / textCount;
|
||||
for (double offset = 0.5 * glbTextLen; offset < pathGlbLength; offset += glbTextLen)
|
||||
offsets.push_back(offset);
|
||||
}
|
||||
|
||||
if (m_params.m_textFont.m_outlineColor == dp::Color::Transparent())
|
||||
DrawPathTextPlain(textures, batcher, move(layout), offsets);
|
||||
else
|
||||
|
|
|
@ -22,11 +22,11 @@ private:
|
|||
void DrawPathTextPlain(ref_ptr<dp::TextureManager> textures,
|
||||
ref_ptr<dp::Batcher> batcher,
|
||||
unique_ptr<PathTextLayout> && layout,
|
||||
buffer_vector<float, 32> const & offests) const;
|
||||
vector<float> const & offests) const;
|
||||
void DrawPathTextOutlined(ref_ptr<dp::TextureManager> textures,
|
||||
ref_ptr<dp::Batcher> batcher,
|
||||
unique_ptr<PathTextLayout> && layout,
|
||||
buffer_vector<float, 32> const & offsets) const;
|
||||
vector<float> const & offsets) const;
|
||||
|
||||
m2::SharedSpline m_spline;
|
||||
PathTextViewParams m_params;
|
||||
|
|
|
@ -7,8 +7,9 @@ namespace df
|
|||
|
||||
TextHandle::TextHandle(FeatureID const & id, strings::UniString const & text,
|
||||
dp::Anchor anchor, uint64_t priority,
|
||||
ref_ptr<dp::TextureManager> textureManager)
|
||||
: OverlayHandle(id, anchor, priority, false)
|
||||
ref_ptr<dp::TextureManager> textureManager,
|
||||
bool isBillboard)
|
||||
: OverlayHandle(id, anchor, priority, isBillboard)
|
||||
, m_forceUpdateNormals(false)
|
||||
, m_isLastVisible(false)
|
||||
, m_text(text)
|
||||
|
@ -22,7 +23,7 @@ TextHandle::TextHandle(FeatureID const & id, strings::UniString const & text,
|
|||
gpu::TTextDynamicVertexBuffer && normals,
|
||||
bool isBillboard)
|
||||
: OverlayHandle(id, anchor, priority, isBillboard)
|
||||
, m_normals(move(normals))
|
||||
, m_buffer(move(normals))
|
||||
, m_forceUpdateNormals(false)
|
||||
, m_isLastVisible(false)
|
||||
, m_text(text)
|
||||
|
@ -41,12 +42,12 @@ void TextHandle::GetAttributeMutation(ref_ptr<dp::AttributeBufferMutator> mutato
|
|||
|
||||
TOffsetNode const & node = GetOffsetNode(gpu::TextDynamicVertex::GetDynamicStreamID());
|
||||
ASSERT(node.first.GetElementSize() == sizeof(gpu::TextDynamicVertex), ());
|
||||
ASSERT(node.second.m_count == m_normals.size(), ());
|
||||
ASSERT(node.second.m_count == m_buffer.size(), ());
|
||||
|
||||
uint32_t byteCount = m_normals.size() * sizeof(gpu::TextDynamicVertex);
|
||||
uint32_t byteCount = m_buffer.size() * sizeof(gpu::TextDynamicVertex);
|
||||
void * buffer = mutator->AllocateMutationBuffer(byteCount);
|
||||
if (isVisible)
|
||||
memcpy(buffer, m_normals.data(), byteCount);
|
||||
memcpy(buffer, m_buffer.data(), byteCount);
|
||||
else
|
||||
memset(buffer, 0, byteCount);
|
||||
|
||||
|
|
|
@ -19,7 +19,8 @@ class TextHandle : public dp::OverlayHandle
|
|||
public:
|
||||
TextHandle(FeatureID const & id, strings::UniString const & text,
|
||||
dp::Anchor anchor, uint64_t priority,
|
||||
ref_ptr<dp::TextureManager> textureManager);
|
||||
ref_ptr<dp::TextureManager> textureManager,
|
||||
bool isBillboard = false);
|
||||
|
||||
TextHandle(FeatureID const & id, strings::UniString const & text,
|
||||
dp::Anchor anchor, uint64_t priority,
|
||||
|
@ -37,7 +38,7 @@ public:
|
|||
void SetForceUpdateNormals(bool forceUpdate) const;
|
||||
|
||||
protected:
|
||||
gpu::TTextDynamicVertexBuffer m_normals;
|
||||
gpu::TTextDynamicVertexBuffer m_buffer;
|
||||
mutable bool m_forceUpdateNormals;
|
||||
|
||||
private:
|
||||
|
|
|
@ -1,4 +1,5 @@
|
|||
#include "drape_frontend/text_layout.hpp"
|
||||
#include "drape_frontend/visual_params.hpp"
|
||||
|
||||
#include "drape_frontend/visual_params.hpp"
|
||||
|
||||
|
@ -457,37 +458,36 @@ void PathTextLayout::CacheStaticGeometry(dp::TextureManager::ColorRegion const &
|
|||
}
|
||||
|
||||
bool PathTextLayout::CacheDynamicGeometry(m2::Spline::iterator const & iter, const float depth,
|
||||
ScreenBase const & screen,
|
||||
m2::PointD const & globalPivot,
|
||||
gpu::TTextDynamicVertexBuffer & buffer) const
|
||||
{
|
||||
float const scalePtoG = screen.GetScale();
|
||||
float const glbHalfLength = 0.5 * GetPixelLength() * scalePtoG;
|
||||
float const halfLength = 0.5 * GetPixelLength();
|
||||
|
||||
m2::Spline::iterator beginIter = iter;
|
||||
beginIter.Advance(-glbHalfLength);
|
||||
beginIter.Advance(-halfLength);
|
||||
m2::Spline::iterator endIter = iter;
|
||||
endIter.Advance(glbHalfLength);
|
||||
endIter.Advance(halfLength);
|
||||
if (beginIter.BeginAgain() || endIter.BeginAgain())
|
||||
return false;
|
||||
|
||||
float const halfFontSize = 0.5 * GetPixelHeight();
|
||||
float advanceSign = 1.0f;
|
||||
m2::Spline::iterator penIter = beginIter;
|
||||
if (screen.GtoP(beginIter.m_pos).x > screen.GtoP(endIter.m_pos).x)
|
||||
if (beginIter.m_pos.x > endIter.m_pos.x)
|
||||
{
|
||||
advanceSign = -advanceSign;
|
||||
penIter = endIter;
|
||||
}
|
||||
|
||||
glsl::vec2 pxPivot = glsl::ToVec2(screen.GtoP(iter.m_pos));
|
||||
glsl::vec2 pxPivot = glsl::ToVec2(iter.m_pos);
|
||||
buffer.resize(4 * m_metrics.size());
|
||||
for (size_t i = 0; i < m_metrics.size(); ++i)
|
||||
{
|
||||
GlyphRegion const & g = m_metrics[i];
|
||||
m2::PointF pxSize = m2::PointF(g.GetPixelSize()) * m_textSizeRatio;
|
||||
|
||||
m2::PointD const pxBase = screen.GtoP(penIter.m_pos);
|
||||
m2::PointD const pxShiftBase = screen.GtoP(penIter.m_pos + penIter.m_dir);
|
||||
m2::PointD const pxBase = penIter.m_pos;
|
||||
m2::PointD const pxShiftBase = penIter.m_pos + penIter.m_dir;
|
||||
|
||||
glsl::vec2 tangent = advanceSign * glsl::normalize(glsl::ToVec2(pxShiftBase - pxBase));
|
||||
glsl::vec2 normal = glsl::normalize(glsl::vec2(-tangent.y, tangent.x));
|
||||
|
@ -501,7 +501,7 @@ bool PathTextLayout::CacheDynamicGeometry(m2::Spline::iterator const & iter, con
|
|||
|
||||
size_t baseIndex = 4 * i;
|
||||
|
||||
glsl::vec3 pivot(glsl::ToVec2(iter.m_pos), depth);
|
||||
glsl::vec3 pivot(glsl::ToVec2(globalPivot), depth);
|
||||
buffer[baseIndex + 0] = gpu::TextDynamicVertex(pivot, formingVector + normal * bottomVector + tangent * xOffset);
|
||||
buffer[baseIndex + 1] = gpu::TextDynamicVertex(pivot, formingVector + normal * upVector + tangent * xOffset);
|
||||
buffer[baseIndex + 2] = gpu::TextDynamicVertex(pivot, formingVector + normal * bottomVector + tangent * (pxSize.x + xOffset));
|
||||
|
@ -509,7 +509,7 @@ bool PathTextLayout::CacheDynamicGeometry(m2::Spline::iterator const & iter, con
|
|||
|
||||
float const xAdvance = g.GetAdvanceX() * m_textSizeRatio;
|
||||
glsl::vec2 currentTangent = glsl::ToVec2(penIter.m_dir);
|
||||
penIter.Advance(advanceSign * xAdvance * scalePtoG);
|
||||
penIter.Advance(advanceSign * xAdvance);
|
||||
float const dotProduct = glsl::dot(currentTangent, glsl::ToVec2(penIter.m_dir));
|
||||
if (dotProduct < kValidSplineTurn)
|
||||
return false;
|
||||
|
@ -518,6 +518,42 @@ bool PathTextLayout::CacheDynamicGeometry(m2::Spline::iterator const & iter, con
|
|||
return true;
|
||||
}
|
||||
|
||||
// static
|
||||
void PathTextLayout::CalculatePositions(vector<float> & offsets, float splineLength,
|
||||
float splineScaleToPixel, float textPixelLength)
|
||||
{
|
||||
//we leave a little space on either side of the text that would
|
||||
//remove the comparison for equality of spline portions
|
||||
float const kTextBorder = 4.0f;
|
||||
float const textLength = kTextBorder + textPixelLength;
|
||||
float const pathGlbLength = splineLength;
|
||||
|
||||
// on next readable scale m_scaleGtoP will be twice
|
||||
if (textLength > pathGlbLength * 2.0 * splineScaleToPixel)
|
||||
return;
|
||||
|
||||
float const kPathLengthScalar = 0.75;
|
||||
float const pathLength = kPathLengthScalar * splineScaleToPixel * pathGlbLength;
|
||||
|
||||
float const etalonEmpty = max(300 * df::VisualParams::Instance().GetVisualScale(), (double)textLength);
|
||||
float const minPeriodSize = etalonEmpty + textLength;
|
||||
float const twoTextAndEmpty = minPeriodSize + textLength;
|
||||
|
||||
if (pathLength < twoTextAndEmpty)
|
||||
{
|
||||
// if we can't place 2 text and empty part on path
|
||||
// we place only one text on center of path
|
||||
offsets.push_back(pathGlbLength / 2.0f);
|
||||
}
|
||||
else
|
||||
{
|
||||
double const textCount = max(floor(pathLength / minPeriodSize), 1.0);
|
||||
double const glbTextLen = pathGlbLength / textCount;
|
||||
for (double offset = 0.5 * glbTextLen; offset < pathGlbLength; offset += glbTextLen)
|
||||
offsets.push_back(offset);
|
||||
}
|
||||
}
|
||||
|
||||
///////////////////////////////////////////////////////////////
|
||||
SharedTextLayout::SharedTextLayout(PathTextLayout * layout)
|
||||
: m_layout(layout)
|
||||
|
|
|
@ -87,6 +87,9 @@ public:
|
|||
PathTextLayout(strings::UniString const & text,
|
||||
float fontSize, ref_ptr<dp::TextureManager> textures);
|
||||
|
||||
static void CalculatePositions(vector<float> & offsets, float splineLength,
|
||||
float splineScaleToPixel, float textPixelLength);
|
||||
|
||||
void CacheStaticGeometry(dp::TextureManager::ColorRegion const & colorRegion,
|
||||
dp::TextureManager::ColorRegion const & outlineRegion,
|
||||
gpu::TTextOutlinedStaticVertexBuffer & staticBuffer) const;
|
||||
|
@ -96,7 +99,7 @@ public:
|
|||
|
||||
bool CacheDynamicGeometry(m2::Spline::iterator const & iter,
|
||||
float const depth,
|
||||
ScreenBase const & screen,
|
||||
m2::PointD const & globalPivot,
|
||||
gpu::TTextDynamicVertexBuffer & buffer) const;
|
||||
};
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue