forked from organicmaps/organicmaps
Added construction of arrows by route's data
This commit is contained in:
parent
e04123350d
commit
4b3d1e2f35
10 changed files with 156 additions and 132 deletions
|
@ -186,7 +186,7 @@ void BackendRenderer::AcceptMessage(ref_ptr<Message> message)
|
|||
case Message::AddRoute:
|
||||
{
|
||||
ref_ptr<AddRouteMessage> msg = message;
|
||||
m_routeBuilder->Build(msg->GetRoutePolyline(), msg->GetColor(), m_texMng);
|
||||
m_routeBuilder->Build(msg->GetRoutePolyline(), msg->GetTurns(), msg->GetColor(), m_texMng);
|
||||
break;
|
||||
}
|
||||
case Message::RemoveRoute:
|
||||
|
|
|
@ -314,10 +314,10 @@ bool DrapeEngine::GetMyPosition(m2::PointD & myPosition)
|
|||
return hasPosition;
|
||||
}
|
||||
|
||||
void DrapeEngine::AddRoute(m2::PolylineD const & routePolyline, dp::Color const & color)
|
||||
void DrapeEngine::AddRoute(m2::PolylineD const & routePolyline, vector<double> const & turns, dp::Color const & color)
|
||||
{
|
||||
m_threadCommutator->PostMessage(ThreadsCommutator::ResourceUploadThread,
|
||||
make_unique_dp<AddRouteMessage>(routePolyline, color),
|
||||
make_unique_dp<AddRouteMessage>(routePolyline, turns, color),
|
||||
MessagePriority::Normal);
|
||||
}
|
||||
|
||||
|
|
|
@ -98,7 +98,7 @@ public:
|
|||
void DeselectObject();
|
||||
bool GetMyPosition(m2::PointD & myPosition);
|
||||
|
||||
void AddRoute(m2::PolylineD const & routePolyline, dp::Color const & color);
|
||||
void AddRoute(m2::PolylineD const & routePolyline, vector<double> const & turns, dp::Color const & color);
|
||||
void RemoveRoute(bool deactivateFollowing);
|
||||
|
||||
void SetWidgetLayout(gui::TWidgetsLayoutInfo && info);
|
||||
|
|
|
@ -500,7 +500,6 @@ void FrontendRenderer::RenderScene(ScreenBase const & modelView)
|
|||
m_selectionShape->Render(modelView, make_ref(m_gpuProgramManager), m_generalUniforms);
|
||||
}
|
||||
|
||||
m_routeRenderer->Render(modelView, make_ref(m_gpuProgramManager), m_generalUniforms);
|
||||
m_myPositionController->Render(modelView, make_ref(m_gpuProgramManager), m_generalUniforms);
|
||||
|
||||
for (; currentRenderGroup < m_renderGroups.size(); ++currentRenderGroup)
|
||||
|
@ -521,6 +520,9 @@ void FrontendRenderer::RenderScene(ScreenBase const & modelView)
|
|||
RenderSingleGroup(modelView, make_ref(group));
|
||||
}
|
||||
|
||||
GLFunctions::glClearDepth();
|
||||
m_routeRenderer->Render(modelView, make_ref(m_gpuProgramManager), m_generalUniforms);
|
||||
|
||||
GLFunctions::glClearDepth();
|
||||
|
||||
if (m_guiRenderer != nullptr)
|
||||
|
|
|
@ -497,8 +497,9 @@ private:
|
|||
class AddRouteMessage : public Message
|
||||
{
|
||||
public:
|
||||
AddRouteMessage(m2::PolylineD const & routePolyline, dp::Color const & color)
|
||||
AddRouteMessage(m2::PolylineD const & routePolyline, vector<double> const & turns, dp::Color const & color)
|
||||
: m_routePolyline(routePolyline)
|
||||
, m_turns(turns)
|
||||
, m_color(color)
|
||||
{}
|
||||
|
||||
|
@ -506,10 +507,12 @@ public:
|
|||
|
||||
m2::PolylineD const & GetRoutePolyline() { return m_routePolyline; }
|
||||
dp::Color const & GetColor() const { return m_color; }
|
||||
vector<double> const & GetTurns() const { return m_turns; }
|
||||
|
||||
private:
|
||||
m2::PolylineD m_routePolyline;
|
||||
dp::Color m_color;
|
||||
vector<double> m_turns;
|
||||
};
|
||||
|
||||
class RemoveRouteMessage : public Message
|
||||
|
|
|
@ -12,7 +12,8 @@ RouteBuilder::RouteBuilder(RouteBuilder::TFlushRouteFn const & flushRouteFn)
|
|||
, m_batcher(make_unique_dp<dp::Batcher>(ESTIMATE_BUFFER_SIZE, ESTIMATE_BUFFER_SIZE))
|
||||
{}
|
||||
|
||||
void RouteBuilder::Build(m2::PolylineD const & routePolyline, dp::Color const & color, ref_ptr<dp::TextureManager> textures)
|
||||
void RouteBuilder::Build(m2::PolylineD const & routePolyline, vector<double> const & turns,
|
||||
dp::Color const & color, ref_ptr<dp::TextureManager> textures)
|
||||
{
|
||||
CommonViewParams params;
|
||||
params.m_depth = 0.0f;
|
||||
|
@ -26,6 +27,7 @@ void RouteBuilder::Build(m2::PolylineD const & routePolyline, dp::Color const &
|
|||
routeData.m_arrowTextureRect = textureRect;
|
||||
routeData.m_joinsBounds = shape.GetJoinsBounds();
|
||||
routeData.m_length = shape.GetLength();
|
||||
routeData.m_turns = turns;
|
||||
|
||||
auto flushRoute = [this, &routeData](dp::GLState const & state, drape_ptr<dp::RenderBucket> && bucket)
|
||||
{
|
||||
|
|
|
@ -22,6 +22,7 @@ struct RouteData
|
|||
m2::RectF m_arrowTextureRect;
|
||||
vector<RouteJoinBounds> m_joinsBounds;
|
||||
double m_length;
|
||||
vector<double> m_turns;
|
||||
};
|
||||
|
||||
class RouteBuilder
|
||||
|
@ -31,7 +32,8 @@ public:
|
|||
|
||||
RouteBuilder(TFlushRouteFn const & flushRouteFn);
|
||||
|
||||
void Build(m2::PolylineD const & routePolyline, dp::Color const & color, ref_ptr<dp::TextureManager> textures);
|
||||
void Build(m2::PolylineD const & routePolyline, vector<double> const & turns,
|
||||
dp::Color const & color, ref_ptr<dp::TextureManager> textures);
|
||||
|
||||
private:
|
||||
TFlushRouteFn m_flushRouteFn;
|
||||
|
|
|
@ -22,11 +22,13 @@ float const halfWidthInPixel[] =
|
|||
2.0f, 2.5f, 3.5f, 5.0f, 7.5f, 10.0f, 14.0f, 18.0f, 36.0f,
|
||||
};
|
||||
|
||||
int const arrowAppearingZoomLevel = 14;
|
||||
|
||||
int const arrowPartsCount = 3;
|
||||
double const arrowHeightFactor = 96.0 / 36.0;
|
||||
double const arrowAspect = 400.0 / 192.0;
|
||||
double const arrowTailSize = 20.0 / 400.0;
|
||||
double const arrowHeadSize = 124.0 / 400.0;
|
||||
double const arrowHeadSize = 120.0 / 400.0;
|
||||
|
||||
struct RouteSegment
|
||||
{
|
||||
|
@ -41,42 +43,82 @@ struct RouteSegment
|
|||
{}
|
||||
};
|
||||
|
||||
int FindNearestAvailableSegment(bool isTail, double start, double end, vector<RouteSegment> const & segments)
|
||||
int CheckForIntersection(double start, double end, vector<RouteSegment> const & segments)
|
||||
{
|
||||
// check if distance is inside unavailable segment
|
||||
int startIndex = -1;
|
||||
int endIndex = -1;
|
||||
for (size_t i = 0; i < segments.size(); i++)
|
||||
{
|
||||
if (!segments[i].m_isAvailable && start >= segments[i].m_start && start <= segments[i].m_end)
|
||||
startIndex = i;
|
||||
if (segments[i].m_isAvailable)
|
||||
continue;
|
||||
|
||||
if (!segments[i].m_isAvailable && end >= segments[i].m_start && end <= segments[i].m_end)
|
||||
endIndex = i;
|
||||
if ((start >= segments[i].m_start && start <= segments[i].m_end) ||
|
||||
(end >= segments[i].m_start && end <= segments[i].m_end) ||
|
||||
(start < segments[i].m_start && end > segments[i].m_end))
|
||||
return i;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
int FindNearestAvailableSegment(double start, double end, vector<RouteSegment> const & segments)
|
||||
{
|
||||
double const threshold = 0.8;
|
||||
|
||||
// check if distance intersects unavailable segment
|
||||
int index = CheckForIntersection(start, end, segments);
|
||||
|
||||
// find nearest available segment if necessary
|
||||
int index = max(startIndex, endIndex);
|
||||
if (index != -1)
|
||||
{
|
||||
double const len = end - start;
|
||||
if (isTail)
|
||||
for (int i = index; i < (int)segments.size(); i++)
|
||||
{
|
||||
for (int i = index; i >= 0; i--)
|
||||
if (segments[i].m_isAvailable && len <= (segments[i].m_end - segments[i].m_start))
|
||||
return (int)i;
|
||||
}
|
||||
else
|
||||
{
|
||||
for (int i = index; i < (int)segments.size(); i++)
|
||||
if (segments[i].m_isAvailable && len <= (segments[i].m_end - segments[i].m_start))
|
||||
return (int)i;
|
||||
double const factor = (segments[i].m_end - segments[i].m_start) / len;
|
||||
if (segments[i].m_isAvailable && factor > threshold)
|
||||
return (int)i;
|
||||
}
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
void MergeAndClipBorders(vector<ArrowBorders> & borders, double scale, double arrowTextureWidth)
|
||||
{
|
||||
if (borders.empty())
|
||||
return;
|
||||
|
||||
// mark groups
|
||||
for (size_t i = 0; i < borders.size() - 1; i++)
|
||||
{
|
||||
if (borders[i].m_endDistance >= borders[i + 1].m_startDistance)
|
||||
borders[i + 1].m_groupIndex = borders[i].m_groupIndex;
|
||||
}
|
||||
|
||||
// merge groups
|
||||
int lastGroup = 0;
|
||||
size_t lastGroupIndex = 0;
|
||||
for (size_t i = 1; i < borders.size(); i++)
|
||||
{
|
||||
if (borders[i].m_groupIndex != lastGroup)
|
||||
{
|
||||
borders[lastGroupIndex].m_endDistance = borders[i - 1].m_endDistance;
|
||||
lastGroupIndex = i;
|
||||
lastGroup = borders[i].m_groupIndex;
|
||||
}
|
||||
else
|
||||
{
|
||||
borders[i].m_groupIndex = -1;
|
||||
}
|
||||
}
|
||||
borders[lastGroupIndex].m_endDistance = borders.back().m_endDistance;
|
||||
|
||||
// clip groups
|
||||
auto const iter = remove_if(borders.begin(), borders.end(),
|
||||
[&scale, &arrowTextureWidth](ArrowBorders const & borders)
|
||||
{
|
||||
return borders.m_groupIndex == -1;
|
||||
});
|
||||
borders.erase(iter, borders.end());
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
RouteGraphics::RouteGraphics(dp::GLState const & state,
|
||||
|
@ -123,51 +165,63 @@ void RouteRenderer::Render(ScreenBase const & screen, ref_ptr<dp::GpuProgramMana
|
|||
graphics.m_buffer->Render();
|
||||
|
||||
// arrows rendering
|
||||
double arrowHalfWidth = halfWidth * arrowHeightFactor;
|
||||
uniformStorage.SetFloatValue("u_halfWidth", arrowHalfWidth, arrowHalfWidth * screen.GetScale());
|
||||
uniformStorage.SetFloatValue("u_textureRect", m_routeData.m_arrowTextureRect.minX(),
|
||||
m_routeData.m_arrowTextureRect.minY(),
|
||||
m_routeData.m_arrowTextureRect.maxX(),
|
||||
m_routeData.m_arrowTextureRect.maxY());
|
||||
if (truncedZoom >= arrowAppearingZoomLevel)
|
||||
RenderArrow(graphics, halfWidth, screen, mng, commonUniforms);
|
||||
}
|
||||
}
|
||||
|
||||
m_turnPoints = { 0.0091, 0.0109 };
|
||||
double const textureWidth = 2.0 * arrowHalfWidth * arrowAspect;
|
||||
vector<ArrowBorders> arrowBorders;
|
||||
CalculateArrowBorders(0.001, screen.GetScale(), textureWidth, arrowHalfWidth * screen.GetScale(), arrowBorders);
|
||||
void RouteRenderer::RenderArrow(RouteGraphics const & graphics, float halfWidth, ScreenBase const & screen,
|
||||
ref_ptr<dp::GpuProgramManager> mng, dp::UniformValuesStorage const & commonUniforms)
|
||||
{
|
||||
double const arrowHalfWidth = halfWidth * arrowHeightFactor;
|
||||
double const arrowSize = 0.001;
|
||||
double const textureWidth = 2.0 * arrowHalfWidth * arrowAspect;
|
||||
|
||||
ref_ptr<dp::GpuProgram> prgArrow = mng->GetProgram(gpu::ROUTE_ARROW_PROGRAM);
|
||||
prgArrow->Bind();
|
||||
dp::ApplyState(graphics.m_state, prgArrow);
|
||||
dp::ApplyUniforms(commonUniforms, prgArrow);
|
||||
dp::UniformValuesStorage uniformStorage;
|
||||
uniformStorage.SetFloatValue("u_halfWidth", arrowHalfWidth, arrowHalfWidth * screen.GetScale());
|
||||
uniformStorage.SetFloatValue("u_textureRect", m_routeData.m_arrowTextureRect.minX(),
|
||||
m_routeData.m_arrowTextureRect.minY(),
|
||||
m_routeData.m_arrowTextureRect.maxX(),
|
||||
m_routeData.m_arrowTextureRect.maxY());
|
||||
|
||||
size_t const elementsCount = 16;
|
||||
vector<float> borders(elementsCount, 0.0);
|
||||
size_t index = 0;
|
||||
for (size_t i = 0; i < arrowBorders.size(); i++)
|
||||
// calculate arrows
|
||||
vector<ArrowBorders> arrowBorders;
|
||||
CalculateArrowBorders(arrowSize, screen.GetScale(), textureWidth, arrowHalfWidth * screen.GetScale(), arrowBorders);
|
||||
|
||||
// bind shaders
|
||||
ref_ptr<dp::GpuProgram> prgArrow = mng->GetProgram(gpu::ROUTE_ARROW_PROGRAM);
|
||||
prgArrow->Bind();
|
||||
dp::ApplyState(graphics.m_state, prgArrow);
|
||||
dp::ApplyUniforms(commonUniforms, prgArrow);
|
||||
|
||||
// split arrow's data by 16-elements buckets
|
||||
size_t const elementsCount = 16;
|
||||
vector<float> borders(elementsCount, 0.0);
|
||||
size_t index = 0;
|
||||
for (size_t i = 0; i < arrowBorders.size(); i++)
|
||||
{
|
||||
borders[index++] = arrowBorders[i].m_startDistance;
|
||||
borders[index++] = arrowBorders[i].m_startTexCoord;
|
||||
borders[index++] = arrowBorders[i].m_endDistance;
|
||||
borders[index++] = arrowBorders[i].m_endTexCoord;
|
||||
|
||||
// fill rests by zeros
|
||||
if (i == arrowBorders.size() - 1)
|
||||
{
|
||||
borders[index++] = arrowBorders[i].m_startDistance;
|
||||
borders[index++] = arrowBorders[i].m_startTexCoord;
|
||||
borders[index++] = arrowBorders[i].m_endDistance;
|
||||
borders[index++] = arrowBorders[i].m_endTexCoord;
|
||||
for (size_t j = index; j < elementsCount; j++)
|
||||
borders[j] = 0.0;
|
||||
|
||||
// fill rests by zeros
|
||||
if (i == arrowBorders.size() - 1)
|
||||
{
|
||||
for (size_t j = index; j < elementsCount; j++)
|
||||
borders[j] = 0.0;
|
||||
index = elementsCount;
|
||||
}
|
||||
|
||||
index = elementsCount;
|
||||
}
|
||||
// render arrow's parts
|
||||
if (index == elementsCount)
|
||||
{
|
||||
index = 0;
|
||||
uniformStorage.SetMatrix4x4Value("u_arrowBorders", borders.data());
|
||||
|
||||
// render arrow's parts
|
||||
if (index == elementsCount)
|
||||
{
|
||||
index = 0;
|
||||
uniformStorage.SetMatrix4x4Value("u_arrowBorders", borders.data());
|
||||
|
||||
dp::ApplyUniforms(uniformStorage, prgArrow);
|
||||
graphics.m_buffer->Render();
|
||||
}
|
||||
dp::ApplyUniforms(uniformStorage, prgArrow);
|
||||
graphics.m_buffer->Render();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -196,49 +250,6 @@ void RouteRenderer::UpdateDistanceFromBegin(double distanceFromBegin)
|
|||
m_distanceFromBegin = distanceFromBegin;
|
||||
}
|
||||
|
||||
void RouteRenderer::MergeAndClipBorders(vector<ArrowBorders> & borders, double scale, double arrowTextureWidth)
|
||||
{
|
||||
if (borders.empty())
|
||||
return;
|
||||
|
||||
// mark groups
|
||||
for (size_t i = 0; i < borders.size() - 1; i++)
|
||||
{
|
||||
if (borders[i].m_endDistance >= borders[i + 1].m_startDistance)
|
||||
borders[i + 1].m_groupIndex = borders[i].m_groupIndex;
|
||||
}
|
||||
|
||||
// merge groups
|
||||
int lastGroup = 0;
|
||||
size_t lastGroupIndex = 0;
|
||||
for (size_t i = 1; i < borders.size(); i++)
|
||||
{
|
||||
if (borders[i].m_groupIndex != lastGroup)
|
||||
{
|
||||
borders[lastGroupIndex].m_endDistance = borders[i - 1].m_endDistance;
|
||||
lastGroupIndex = i;
|
||||
lastGroup = borders[i].m_groupIndex;
|
||||
}
|
||||
else
|
||||
{
|
||||
borders[i].m_groupIndex = -1;
|
||||
}
|
||||
}
|
||||
borders[lastGroupIndex].m_endDistance = borders.back().m_endDistance;
|
||||
|
||||
// clip groups
|
||||
auto const iter = remove_if(borders.begin(), borders.end(),
|
||||
[&scale, &arrowTextureWidth](ArrowBorders const & borders)
|
||||
{
|
||||
if (borders.m_groupIndex == -1)
|
||||
return true;
|
||||
|
||||
double const distanceInPixels = (borders.m_endDistance - borders.m_startDistance) * 0.9 / scale;
|
||||
return distanceInPixels < (arrowHeadSize + arrowTailSize) * arrowTextureWidth;
|
||||
});
|
||||
borders.erase(iter, borders.end());
|
||||
}
|
||||
|
||||
void RouteRenderer::ApplyJoinsBounds(double arrowTextureWidth, double joinsBoundsScalar, double glbTailLength,
|
||||
double glbHeadLength, double scale, vector<ArrowBorders> & borders)
|
||||
{
|
||||
|
@ -261,23 +272,15 @@ void RouteRenderer::ApplyJoinsBounds(double arrowTextureWidth, double joinsBound
|
|||
}
|
||||
segments.back().m_end = m_routeData.m_length;
|
||||
|
||||
// shift tail and head of arrow if necessary
|
||||
// shift head of arrow if necessary
|
||||
bool needMerge = false;
|
||||
for (size_t i = 0; i < borders.size(); i++)
|
||||
{
|
||||
int tailIndex = FindNearestAvailableSegment(true /* isTail */, borders[i].m_startDistance,
|
||||
borders[i].m_startDistance + glbTailLength, segments);
|
||||
if (tailIndex != -1)
|
||||
{
|
||||
borders[i].m_startDistance = segments[tailIndex].m_end - glbTailLength;
|
||||
needMerge = true;
|
||||
}
|
||||
|
||||
int headIndex = FindNearestAvailableSegment(false /* isTail */, borders[i].m_endDistance - glbHeadLength,
|
||||
int headIndex = FindNearestAvailableSegment(borders[i].m_endDistance - glbHeadLength,
|
||||
borders[i].m_endDistance, segments);
|
||||
if (headIndex != -1)
|
||||
{
|
||||
borders[i].m_endDistance = segments[headIndex].m_start + glbHeadLength;
|
||||
borders[i].m_endDistance = min(m_routeData.m_length, segments[headIndex].m_start + glbHeadLength);
|
||||
needMerge = true;
|
||||
}
|
||||
}
|
||||
|
@ -290,30 +293,34 @@ void RouteRenderer::ApplyJoinsBounds(double arrowTextureWidth, double joinsBound
|
|||
void RouteRenderer::CalculateArrowBorders(double arrowLength, double scale, double arrowTextureWidth,
|
||||
double joinsBoundsScalar, vector<ArrowBorders> & borders)
|
||||
{
|
||||
if (m_turnPoints.empty())
|
||||
if (m_routeData.m_turns.empty())
|
||||
return;
|
||||
|
||||
double const halfLen = 0.5 * arrowLength;
|
||||
double halfLen = 0.5 * arrowLength;
|
||||
double const glbTextureWidth = arrowTextureWidth * scale;
|
||||
double const glbTailLength = arrowTailSize * glbTextureWidth;
|
||||
double const glbHeadLength = arrowHeadSize * glbTextureWidth;
|
||||
|
||||
borders.reserve(m_turnPoints.size() * arrowPartsCount);
|
||||
borders.reserve(m_routeData.m_turns.size() * arrowPartsCount);
|
||||
|
||||
double const halfTextureWidth = 0.5 * glbTextureWidth;
|
||||
if (halfLen < halfTextureWidth)
|
||||
halfLen = halfTextureWidth;
|
||||
|
||||
// initial filling
|
||||
for (size_t i = 0; i < m_turnPoints.size(); i++)
|
||||
for (size_t i = 0; i < m_routeData.m_turns.size(); i++)
|
||||
{
|
||||
ArrowBorders arrowBorders;
|
||||
arrowBorders.m_groupIndex = (int)i;
|
||||
arrowBorders.m_startDistance = m_turnPoints[i] - halfLen;
|
||||
arrowBorders.m_endDistance = m_turnPoints[i] + halfLen;
|
||||
arrowBorders.m_startDistance = max(0.0, m_routeData.m_turns[i] - halfLen * 0.8);
|
||||
arrowBorders.m_endDistance = min(m_routeData.m_length, m_routeData.m_turns[i] + halfLen * 1.2);
|
||||
borders.push_back(arrowBorders);
|
||||
}
|
||||
|
||||
// merge intersected borders and clip them
|
||||
MergeAndClipBorders(borders, scale, arrowTextureWidth);
|
||||
|
||||
// apply joins bounds to prevent draw arrow's head and tail on a join
|
||||
// apply joins bounds to prevent draw arrow's head on a join
|
||||
ApplyJoinsBounds(arrowTextureWidth, joinsBoundsScalar, glbTailLength,
|
||||
glbHeadLength, scale, borders);
|
||||
|
||||
|
|
|
@ -52,15 +52,16 @@ public:
|
|||
private:
|
||||
void CalculateArrowBorders(double arrowLength, double scale, double arrowTextureWidth,
|
||||
double joinsBoundsScalar, vector<ArrowBorders> & borders);
|
||||
void MergeAndClipBorders(vector<ArrowBorders> & borders, double scale, double arrowTextureWidth);
|
||||
|
||||
void ApplyJoinsBounds(double arrowTextureWidth, double joinsBoundsScalar, double glbTailLength,
|
||||
double glbHeadLength, double scale, vector<ArrowBorders> & borders);
|
||||
|
||||
void RenderArrow(RouteGraphics const & graphics, float halfWidth, ScreenBase const & screen,
|
||||
ref_ptr<dp::GpuProgramManager> mng, dp::UniformValuesStorage const & commonUniforms);
|
||||
|
||||
vector<RouteGraphics> m_routeGraphics;
|
||||
double m_distanceFromBegin;
|
||||
RouteData m_routeData;
|
||||
vector<double> m_turnPoints;
|
||||
};
|
||||
|
||||
} // namespace df
|
||||
|
|
|
@ -1960,6 +1960,7 @@ void Framework::CloseRouting()
|
|||
void Framework::InsertRoute(Route const & route)
|
||||
{
|
||||
ASSERT_THREAD_CHECKER(m_threadChecker, ("InsertRoute"));
|
||||
ASSERT(m_drapeEngine != nullptr, ());
|
||||
|
||||
if (route.GetPoly().GetSize() < 2)
|
||||
{
|
||||
|
@ -1967,9 +1968,15 @@ void Framework::InsertRoute(Route const & route)
|
|||
return;
|
||||
}
|
||||
|
||||
ASSERT(m_drapeEngine != nullptr, ());
|
||||
m_drapeEngine->AddRoute(route.GetPoly(), dp::Color(110, 180, 240, 200));
|
||||
|
||||
vector<double> turns;
|
||||
turns::TurnsGeomT const & turnsGeom = route.GetTurnsGeometry();
|
||||
if (!turnsGeom.empty())
|
||||
{
|
||||
turns.reserve(turnsGeom.size());
|
||||
for (size_t i = 0; i < turnsGeom.size(); i++)
|
||||
turns.push_back(turnsGeom[i].m_mercatorDistance);
|
||||
}
|
||||
m_drapeEngine->AddRoute(route.GetPoly(), turns, dp::Color(110, 180, 240, 160));
|
||||
|
||||
// TODO(@kuznetsov): Maybe we need some of this stuff
|
||||
//track.SetName(route.GetName());
|
||||
|
|
Loading…
Add table
Reference in a new issue