diff --git a/drape/glsl_types.hpp b/drape/glsl_types.hpp index bb92a48ba1..844ba21116 100644 --- a/drape/glsl_types.hpp +++ b/drape/glsl_types.hpp @@ -54,4 +54,29 @@ inline vec2 ToVec2(m2::PointD const & pt) return glsl::vec2(pt.x, pt.y); } +template +uint8_t GetComponentCount() +{ + ASSERT(false, ()); + return 0; +} + +template <> +uint8_t GetComponentCount() +{ + return 2; +} + +template <> +uint8_t GetComponentCount() +{ + return 3; +} + +template <> +uint8_t GetComponentCount() +{ + return 4; +} + } // namespace glsl diff --git a/drape/shaders/line_fragment_shader.fsh b/drape/shaders/line_fragment_shader.fsh index 326874bc3f..8c17753893 100644 --- a/drape/shaders/line_fragment_shader.fsh +++ b/drape/shaders/line_fragment_shader.fsh @@ -1,34 +1,19 @@ -varying float v_dx; -varying vec4 v_radius; -varying vec2 v_type; - -varying vec2 v_color; -varying vec2 v_mask; +varying vec2 v_colorTexCoord; +varying vec2 v_maskTexCoord; +varying vec2 v_dxdy; uniform sampler2D u_colorTex; uniform sampler2D u_maskTex; void main(void) { - float squareDist = v_radius.x * v_radius.x; - float squareRadius = v_radius.y * v_radius.y; + float mask = texture2D(u_maskTex, v_maskTexCoord).a; + vec4 color = texture2D(u_colorTex, v_colorTexCoord); - float capY = (v_dx + 1.0 * sign(v_dx)) * v_radius.z / 2.0; - float joinY = (v_dx + 1.0) * v_radius.y / 2.0; + float domen = v_dxdy.x * v_dxdy.x + v_dxdy.y * v_dxdy.y; - if (v_type.x < -0.5) - { - float y = joinY; - if (v_type.y < 0.5) - y = v_radius.y - joinY; + if (mask < 0.1 || domen > 1.0) + discard; - if (squareDist + y * y > squareRadius) - discard; - } - else if (v_type.y > 0.1 && abs(v_dx) >= 1.0 && (squareDist + capY > squareRadius)) - discard; - - vec4 color = texture2D(u_colorTex, v_color); - color.a = texture2D(u_maskTex, v_mask).a; gl_FragColor = color; } diff --git a/drape/shaders/line_vertex_shader.vsh b/drape/shaders/line_vertex_shader.vsh index 537f4a4a87..5df46717b2 100644 --- a/drape/shaders/line_vertex_shader.vsh +++ b/drape/shaders/line_vertex_shader.vsh @@ -1,45 +1,26 @@ -attribute vec4 a_position; -attribute vec4 a_deltas; -attribute vec4 a_width_type; -attribute vec4 a_centres; -attribute vec2 a_color; -attribute vec2 a_mask; - -varying float v_dx; -varying vec4 v_radius; -varying vec2 v_type; - -varying vec2 v_color; -varying vec2 v_mask; +attribute vec3 a_position; +attribute vec2 a_normal; +attribute vec2 a_colorTexCoord; +attribute vec2 a_maskTexCoord; +attribute vec2 a_dxdy; uniform mat4 modelView; uniform mat4 projection; +varying vec2 v_colorTexCoord; +varying vec2 v_maskTexCoord; +varying vec2 v_dxdy; + void main(void) { - float r = abs(a_width_type.x); - vec2 dir = a_position.zw - a_position.xy; - float len = length(dir); - vec4 pos2 = vec4(a_position.xy, a_deltas.z, 1) * modelView; - vec4 direc = vec4(a_position.zw, a_deltas.z, 1) * modelView; - dir = direc.xy - pos2.xy; - float l2 = length(dir); - dir = normalize(dir); - dir *= len * r; - pos2 += vec4(dir, 0, 0); + float halfWidth = length(a_normal); + vec2 transformedAxisPos = (vec4(a_position.xy, 0.0, 1.0) * modelView).xy; + vec4 glbShiftPos = vec4(a_position.xy + a_normal, 0.0, 1.0); + vec2 shiftPos = (glbShiftPos * modelView).xy; + vec2 pxNormal = normalize(shiftPos - transformedAxisPos); - gl_Position = pos2 * projection; - - v_dx = (a_deltas.y + a_deltas.x * r / l2 * len); - v_radius.x = a_width_type.x; - v_radius.y = r; - v_radius.w = a_width_type.w; - vec2 centr1 = (vec4(a_centres.xy, 0, 1) * modelView).xy; - vec2 centr2 = (vec4(a_centres.zw, 0, 1) * modelView).xy; - float len2 = length(centr1 - centr2); - v_radius.z = len2; - v_type = a_width_type.yz; - - v_color = a_color; - v_mask = a_mask; + v_colorTexCoord = a_colorTexCoord; + v_maskTexCoord = a_maskTexCoord; + v_dxdy = a_dxdy; + gl_Position = vec4(transformedAxisPos + pxNormal * halfWidth, a_position.z, 1.0) * projection; } diff --git a/drape/utils/vertex_decl.cpp b/drape/utils/vertex_decl.cpp index 42276acb9d..f1fbdf44bd 100644 --- a/drape/utils/vertex_decl.cpp +++ b/drape/utils/vertex_decl.cpp @@ -3,6 +3,178 @@ namespace gpu { +namespace +{ + +enum VertexType +{ + SolidTexturing, + TextStatic, + TextDynamic, + Line, + TypeCount +}; + +struct BindingNode +{ + dp::BindingInfo m_info; + bool m_inited = false; +}; + +typedef dp::BindingInfo (*TInitFunction)(); + +dp::BindingInfo SolidTexturingBindingInit() +{ + STATIC_ASSERT(sizeof(SolidTexturingVertex) == (sizeof(SolidTexturingVertex::TPosition) + + sizeof(SolidTexturingVertex::TNormal) + + sizeof(SolidTexturingVertex::TTexCoord))); + + dp::BindingInfo info(3); + + dp::BindingDecl & posDecl = info.GetBindingDecl(0); + posDecl.m_attributeName = "a_position"; + posDecl.m_componentCount = glsl::GetComponentCount(); + posDecl.m_componentType = gl_const::GLFloatType; + posDecl.m_offset = 0; + posDecl.m_stride = sizeof(SolidTexturingVertex); + + dp::BindingDecl & normalDecl = info.GetBindingDecl(1); + normalDecl.m_attributeName = "a_normal"; + normalDecl.m_componentCount = glsl::GetComponentCount(); + normalDecl.m_componentType = gl_const::GLFloatType; + normalDecl.m_offset = sizeof(SolidTexturingVertex::TPosition); + normalDecl.m_stride = posDecl.m_stride; + + dp::BindingDecl & colorTexCoordDecl = info.GetBindingDecl(2); + colorTexCoordDecl.m_attributeName = "a_colorTexCoords"; + colorTexCoordDecl.m_componentCount = glsl::GetComponentCount(); + colorTexCoordDecl.m_componentType = gl_const::GLFloatType; + colorTexCoordDecl.m_offset = normalDecl.m_offset + sizeof(SolidTexturingVertex::TNormal); + colorTexCoordDecl.m_stride = posDecl.m_stride; + + return info; +} + +dp::BindingInfo TextStaticBindingInit() +{ + STATIC_ASSERT(sizeof(TextStaticVertex) == (sizeof(TextStaticVertex::TPosition) + + 3 * sizeof(TextStaticVertex::TTexCoord))); + dp::BindingInfo info(4); + + dp::BindingDecl & posDecl = info.GetBindingDecl(0); + posDecl.m_attributeName = "a_position"; + posDecl.m_componentCount = glsl::GetComponentCount(); + posDecl.m_componentType = gl_const::GLFloatType; + posDecl.m_offset = 0; + posDecl.m_stride = sizeof(TextStaticVertex); + + dp::BindingDecl & colorDecl = info.GetBindingDecl(1); + colorDecl.m_attributeName = "a_colorTexCoord"; + colorDecl.m_componentCount = glsl::GetComponentCount(); + colorDecl.m_componentType = gl_const::GLFloatType; + colorDecl.m_offset = sizeof(TextStaticVertex::TPosition); + colorDecl.m_stride = posDecl.m_stride; + + dp::BindingDecl & outlineDecl = info.GetBindingDecl(2); + outlineDecl.m_attributeName = "a_outlineColorTexCoord"; + outlineDecl.m_componentCount = glsl::GetComponentCount(); + outlineDecl.m_componentType = gl_const::GLFloatType; + outlineDecl.m_offset = colorDecl.m_offset + sizeof(TextStaticVertex::TTexCoord); + outlineDecl.m_stride = posDecl.m_stride; + + dp::BindingDecl & maskDecl = info.GetBindingDecl(3); + maskDecl.m_attributeName = "a_maskTexCoord"; + maskDecl.m_componentCount = glsl::GetComponentCount(); + maskDecl.m_componentType = gl_const::GLFloatType; + maskDecl.m_offset = outlineDecl.m_offset + sizeof(TextStaticVertex::TTexCoord); + maskDecl.m_stride = posDecl.m_stride; + + return info; +} + +dp::BindingInfo TextDynamicBindingInit() +{ + STATIC_ASSERT(sizeof(TextDynamicVertex) == sizeof(TextDynamicVertex::TNormal)); + dp::BindingInfo info(1, TextDynamicVertex::GetDynamicStreamID()); + + dp::BindingDecl & decl = info.GetBindingDecl(0); + decl.m_attributeName = "a_normal"; + decl.m_componentCount = glsl::GetComponentCount(); + decl.m_componentType = gl_const::GLFloatType; + decl.m_offset = 0; + decl.m_stride = sizeof(TextDynamicVertex); + + return info; +} + +dp::BindingInfo LineBindingInit() +{ + STATIC_ASSERT(sizeof(LineVertex) == sizeof(LineVertex::TPosition) + + 2 * sizeof(LineVertex::TNormal) + + 2 * sizeof(LineVertex::TTexCoord)); + dp::BindingInfo info(5); + + dp::BindingDecl & posDecl = info.GetBindingDecl(0); + posDecl.m_attributeName = "a_position"; + posDecl.m_componentCount = glsl::GetComponentCount(); + posDecl.m_componentType = gl_const::GLFloatType; + posDecl.m_offset = 0; + posDecl.m_stride = sizeof(LineVertex); + + dp::BindingDecl & normalDecl = info.GetBindingDecl(1); + normalDecl.m_attributeName = "a_normal"; + normalDecl.m_componentCount = glsl::GetComponentCount(); + normalDecl.m_componentType = gl_const::GLFloatType; + normalDecl.m_offset = posDecl.m_offset + sizeof(LineVertex::TPosition); + normalDecl.m_stride = posDecl.m_stride; + + dp::BindingDecl & colorDecl = info.GetBindingDecl(2); + colorDecl.m_attributeName = "a_colorTexCoord"; + colorDecl.m_componentCount = glsl::GetComponentCount(); + colorDecl.m_componentType = gl_const::GLFloatType; + colorDecl.m_offset = normalDecl.m_offset + sizeof(LineVertex::TNormal); + colorDecl.m_stride = posDecl.m_stride; + + dp::BindingDecl & maskDecl = info.GetBindingDecl(3); + maskDecl.m_attributeName = "a_maskTexCoord"; + maskDecl.m_componentCount = glsl::GetComponentCount(); + maskDecl.m_componentType = gl_const::GLFloatType; + maskDecl.m_offset = colorDecl.m_offset + sizeof(LineVertex::TTexCoord); + maskDecl.m_stride = posDecl.m_stride; + + dp::BindingDecl & dxdyDecl = info.GetBindingDecl(4); + dxdyDecl.m_attributeName = "a_dxdy"; + dxdyDecl.m_componentCount = glsl::GetComponentCount(); + dxdyDecl.m_componentType = gl_const::GLFloatType; + dxdyDecl.m_offset = maskDecl.m_offset + sizeof(LineVertex::TNormal); + dxdyDecl.m_stride = posDecl.m_stride; + + return info; +} + +BindingNode g_bindingNodes[TypeCount]; +TInitFunction g_initFunctions[TypeCount] = +{ + &SolidTexturingBindingInit, + &TextStaticBindingInit, + &TextDynamicBindingInit, + &LineBindingInit +}; + +dp::BindingInfo const & GetBinding(VertexType type) +{ + BindingNode & node = g_bindingNodes[type]; + if (!node.m_inited) + { + node.m_info = g_initFunctions[type](); + node.m_inited = true; + } + + return node.m_info; +} + +} // namespace + SolidTexturingVertex::SolidTexturingVertex() : m_position(0.0, 0.0, 0.0) , m_normal(0.0, 0.0) @@ -10,7 +182,8 @@ SolidTexturingVertex::SolidTexturingVertex() { } -SolidTexturingVertex::SolidTexturingVertex(glm::vec3 const & position, glm::vec2 const & normal, glm::vec2 const & colorTexCoord) +SolidTexturingVertex::SolidTexturingVertex(TPosition const & position, TNormal const & normal, + TTexCoord const & colorTexCoord) : m_position(position) , m_normal(normal) , m_colorTexCoord(colorTexCoord) @@ -19,98 +192,44 @@ SolidTexturingVertex::SolidTexturingVertex(glm::vec3 const & position, glm::vec2 dp::BindingInfo const & SolidTexturingVertex::GetBindingInfo() { - STATIC_ASSERT(sizeof(SolidTexturingVertex) == (sizeof(glsl::vec3) + 2 * sizeof(glsl::vec2))); + return GetBinding(SolidTexturing); +} - static dp::BindingInfo s_info(3); - static bool s_inited = false; +TextStaticVertex::TextStaticVertex() + : m_position(0.0, 0.0, 0.0) + , m_colorTexCoord(0.0, 0.0) + , m_outlineTexCoord(0.0, 0.0) + , m_maskTexCoord(0.0, 0.0) +{ +} - if (!s_inited) - { - s_inited = true; - dp::BindingDecl & posDecl = s_info.GetBindingDecl(0); - posDecl.m_attributeName = "a_position"; - posDecl.m_componentCount = 3; - posDecl.m_componentType = gl_const::GLFloatType; - posDecl.m_offset = 0; - posDecl.m_stride = sizeof(SolidTexturingVertex); - - dp::BindingDecl & normalDecl = s_info.GetBindingDecl(1); - normalDecl.m_attributeName = "a_normal"; - normalDecl.m_componentCount = 2; - normalDecl.m_componentType = gl_const::GLFloatType; - normalDecl.m_offset = sizeof(glsl::vec3); - normalDecl.m_stride = posDecl.m_stride; - - dp::BindingDecl & colorTexCoordDecl = s_info.GetBindingDecl(2); - colorTexCoordDecl.m_attributeName = "a_colorTexCoords"; - colorTexCoordDecl.m_componentCount = 2; - colorTexCoordDecl.m_componentType = gl_const::GLFloatType; - colorTexCoordDecl.m_offset = sizeof(glsl::vec3) + sizeof(glsl::vec2); - colorTexCoordDecl.m_stride = posDecl.m_stride; - } - - return s_info; +TextStaticVertex::TextStaticVertex(TPosition const & position, TTexCoord const & colorTexCoord, + TTexCoord const & outlineTexCoord, TTexCoord const & maskTexCoord) + : m_position(position) + , m_colorTexCoord(colorTexCoord) + , m_outlineTexCoord(outlineTexCoord) + , m_maskTexCoord(maskTexCoord) +{ } dp::BindingInfo const & TextStaticVertex::GetBindingInfo() { - STATIC_ASSERT(sizeof(TextStaticVertex) == (sizeof(glsl::vec3) + 3 * sizeof(glsl::vec2))); - static dp::BindingInfo s_info(4); - static bool s_inited = false; + return GetBinding(TextStatic); +} - if (!s_inited) - { - s_inited = true; +TextDynamicVertex::TextDynamicVertex() + : m_normal(0.0, 0.0) +{ +} - dp::BindingDecl & posDecl = s_info.GetBindingDecl(0); - posDecl.m_attributeName = "a_position"; - posDecl.m_componentCount = 3; - posDecl.m_componentType = gl_const::GLFloatType; - posDecl.m_offset = 0; - posDecl.m_stride = sizeof(TextStaticVertex); - - dp::BindingDecl & colorDecl = s_info.GetBindingDecl(1); - colorDecl.m_attributeName = "a_colorTexCoord"; - colorDecl.m_componentCount = 2; - colorDecl.m_componentType = gl_const::GLFloatType; - colorDecl.m_offset = sizeof(glsl::vec3); - colorDecl.m_stride = posDecl.m_stride; - - dp::BindingDecl & outlineDecl = s_info.GetBindingDecl(2); - outlineDecl.m_attributeName = "a_outlineColorTexCoord"; - outlineDecl.m_componentCount = 2; - outlineDecl.m_componentType = gl_const::GLFloatType; - outlineDecl.m_offset = colorDecl.m_offset + sizeof(glsl::vec2); - outlineDecl.m_stride = posDecl.m_stride; - - dp::BindingDecl & maskDecl = s_info.GetBindingDecl(3); - maskDecl.m_attributeName = "a_maskTexCoord"; - maskDecl.m_componentCount = 2; - maskDecl.m_componentType = gl_const::GLFloatType; - maskDecl.m_offset = outlineDecl.m_offset + sizeof(glsl::vec2); - maskDecl.m_stride = posDecl.m_stride; - } - - return s_info; +TextDynamicVertex::TextDynamicVertex(TNormal const & normal) + : m_normal(normal) +{ } dp::BindingInfo const & TextDynamicVertex::GetBindingInfo() { - STATIC_ASSERT(sizeof(TextDynamicVertex) == sizeof(glsl::vec2)); - static dp::BindingInfo s_info(1, GetDynamicStreamID()); - static bool s_inited = false; - - if (!s_inited) - { - dp::BindingDecl & decl = s_info.GetBindingDecl(0); - decl.m_attributeName = "a_normal"; - decl.m_componentCount = 2; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = sizeof(TextDynamicVertex); - } - - return s_info; + return GetBinding(TextDynamic); } uint32_t TextDynamicVertex::GetDynamicStreamID() @@ -118,5 +237,30 @@ uint32_t TextDynamicVertex::GetDynamicStreamID() return 0x7F; } +LineVertex::LineVertex() + : m_position(0.0, 0.0, 0.0) + , m_normal(0.0, 0.0) + , m_colorTexCoord(0.0, 0.0) + , m_maskTexCoord(0.0, 0.0) + , m_dxdy(0.0, 0.0) +{ +} + +LineVertex::LineVertex(TPosition const & position, TNormal const & normal, + TTexCoord const & color, TTexCoord const & mask, + TNormal const & dxdy) + : m_position(position) + , m_normal(normal) + , m_colorTexCoord(color) + , m_maskTexCoord(mask) + , m_dxdy(dxdy) +{ +} + +dp::BindingInfo const & LineVertex::GetBindingInfo() +{ + return GetBinding(Line); +} + } //namespace gpu diff --git a/drape/utils/vertex_decl.hpp b/drape/utils/vertex_decl.hpp index f4e9a86542..3ae565d61d 100644 --- a/drape/utils/vertex_decl.hpp +++ b/drape/utils/vertex_decl.hpp @@ -8,48 +8,50 @@ namespace gpu { -struct SolidTexturingVertex +struct BaseVertex +{ + typedef glsl::vec3 TPosition; + typedef glsl::vec2 TNormal; + typedef glsl::vec2 TTexCoord; +}; + +struct SolidTexturingVertex : BaseVertex { SolidTexturingVertex(); - SolidTexturingVertex(glsl::vec3 const & position, glsl::vec2 const & normal, glsl::vec2 const & colorTexCoord); + SolidTexturingVertex(TPosition const & position, TNormal const & normal, TTexCoord const & colorTexCoord); - glsl::vec3 m_position; - glsl::vec2 m_normal; - glsl::vec2 m_colorTexCoord; + TPosition m_position; + TNormal m_normal; + TTexCoord m_colorTexCoord; static dp::BindingInfo const & GetBindingInfo(); }; typedef buffer_vector TSolidTexVertexBuffer; -struct TextStaticVertex +struct TextStaticVertex : BaseVertex { public: - TextStaticVertex() = default; - TextStaticVertex(glsl::vec3 position, glsl::vec2 colorTexCoord, glsl::vec2 outlineTexCoord, glsl::vec2 maskTexCoord) - : m_position(position) - , m_colorTexCoord(colorTexCoord) - , m_outlineTexCoord(outlineTexCoord) - , m_maskTexCoord(maskTexCoord) - { - } + TextStaticVertex(); + TextStaticVertex(TPosition const & position, TTexCoord const & colorTexCoord, + TTexCoord const & outlineTexCoord, TTexCoord const & maskTexCoord); - glsl::vec3 m_position; - glsl::vec2 m_colorTexCoord; - glsl::vec2 m_outlineTexCoord; - glsl::vec2 m_maskTexCoord; + TPosition m_position; + TTexCoord m_colorTexCoord; + TTexCoord m_outlineTexCoord; + TTexCoord m_maskTexCoord; static dp::BindingInfo const & GetBindingInfo(); }; typedef buffer_vector TTextStaticVertexBuffer; -struct TextDynamicVertex +struct TextDynamicVertex : BaseVertex { - TextDynamicVertex() = default; - TextDynamicVertex(glsl::vec2 normal) : m_normal(normal) {} + TextDynamicVertex(); + TextDynamicVertex(TNormal const & normal); - glsl::vec2 m_normal; + TNormal m_normal; static dp::BindingInfo const & GetBindingInfo(); static uint32_t GetDynamicStreamID(); @@ -57,4 +59,20 @@ struct TextDynamicVertex typedef buffer_vector TTextDynamicVertexBuffer; +struct LineVertex : BaseVertex +{ + LineVertex(); + LineVertex(TPosition const & position, TNormal const & normal, + TTexCoord const & color, TTexCoord const & mask, + TNormal const & dxdy); + + TPosition m_position; + TNormal m_normal; + TTexCoord m_colorTexCoord; + TTexCoord m_maskTexCoord; + TNormal m_dxdy; + + static dp::BindingInfo const & GetBindingInfo(); +}; + } // namespace gpu diff --git a/drape_frontend/line_shape.cpp b/drape_frontend/line_shape.cpp index 048fe852f0..0b6c8f7868 100644 --- a/drape_frontend/line_shape.cpp +++ b/drape_frontend/line_shape.cpp @@ -1,5 +1,6 @@ #include "line_shape.hpp" +#include "../drape/utils/vertex_decl.hpp" #include "../drape/glsl_types.hpp" #include "../drape/glsl_func.hpp" #include "../drape/shader_def.hpp" @@ -8,264 +9,16 @@ #include "../drape/batcher.hpp" #include "../drape/texture_manager.hpp" -#include "../base/logging.hpp" - -#include "../std/algorithm.hpp" - namespace df { namespace { - -class GeomHelper -{ -public: - GeomHelper(glsl::vec2 const & prevPt, - glsl::vec2 const & curPt, - glsl::vec2 const & nextPt) - : m_prevPt(prevPt) - , m_currentPt(curPt) - , m_nextPt(nextPt) - { - Init(m_prevPt, m_currentPt, m_nextPt); - } - - void MoveNext(glsl::vec2 const & nextPt) - { - m_prevPt = m_currentPt; - m_currentPt = m_nextPt; - m_nextPt = nextPt; - Init(m_prevPt, m_currentPt, m_nextPt); - } - - glsl::vec2 m_prevPt, m_currentPt, m_nextPt; - glsl::vec2 m_leftBisector, m_rightBisector; - float m_dx; - float m_aspect; - -private: - /// Split angle v1-v2-v3 by bisector. - void Init(glsl::vec2 const & v1, glsl::vec2 const & v2, glsl::vec2 const & v3) - { - glsl::vec2 const dif21 = v2 - v1; - glsl::vec2 const dif32 = v3 - v2; - - float const l1 = glsl::length(dif21); - float const l2 = glsl::length(dif32); - float const l3 = glsl::length(v1 - v3); - m_aspect = l1 / l2; - - /// normal1 is normal from segment v2 - v1 - /// normal2 is normal from segment v3 - v2 - glsl::vec2 const normal1(-dif21.y / l1, dif21.x / l1); - glsl::vec2 const normal2(-dif32.y / l2, dif32.x / l2); - - m_leftBisector = normal1 + normal2; - - /// find cos(2a), where angle a is a half of v1-v2-v3 angle - float const cos2A = (l1 * l1 + l2 * l2 - l3 * l3) / (2.0f * l1 * l2); - float const sinA = sqrt((1.0f - cos2A) / 2.0f); - - m_leftBisector = m_leftBisector / (sinA * glsl::length(m_leftBisector)); - m_rightBisector = -m_leftBisector; - - float offsetLength = glsl::length(m_leftBisector - normal1); - - float const sign = (glsl::dot(dif21, m_leftBisector) > 0.0f) ? 1.0f : -1.0f; - m_dx = sign * 2.0f * (offsetLength / l1); - - m_leftBisector += v2; - m_rightBisector += v2; - } -}; - -class LineEnumerator -{ -public: - LineEnumerator(vector const & points, float GtoPScale, bool generateEndpoints) - : m_points(points) - , m_GtoPScale(GtoPScale) - , m_generateEndpoints(generateEndpoints) - { - ASSERT_GREATER(points.size(), 1, ()); - if (generateEndpoints) - m_iterIndex = -1; - } - - bool IsSolid() const - { - return m_isSolid; - } - - void SetColorParams(dp::TextureManager::ColorRegion const & color) - { - m_colorRegion = color; - } - - glsl::vec2 GetColorCoord() const - { - ASSERT(m_colorRegion.IsValid(), ()); - return glsl::ToVec2(m_colorRegion.GetTexRect().Center()); - } - - dp::RefPointer GetColorTexture() const - { - ASSERT(m_colorRegion.IsValid(), ()); - return m_colorRegion.GetTexture(); - } - - dp::RefPointer GetMaskTexture() const - { - ASSERT(m_stippleRegion.IsValid(), ()); - return m_stippleRegion.GetTexture(); - } - - void SetMaskParams(dp::TextureManager::StippleRegion const & stippleRegion, bool isSolid) - { - m_stippleRegion = stippleRegion; - m_isSolid = isSolid; - } - - dp::TextureManager::StippleRegion const & GetStippleMaskItem() const - { - ASSERT(!IsSolid(), ()); - ASSERT(m_stippleRegion.IsValid(), ()); - return m_stippleRegion; - } - - glsl::vec2 GetSolidMaskCoord() const - { - ASSERT(IsSolid(), ()); - ASSERT(m_stippleRegion.IsValid(), ()); - return glsl::ToVec2(m_stippleRegion.GetTexRect().Center()); - } - - glsl::vec2 GetNextPoint() - { - ASSERT(HasPoint(), ()); - glsl::vec2 result; - if (IsSolid()) - GetNextSolid(result); - else - GetNextStipple(result); - - return result; - } - - bool HasPoint() const - { - int additionalSize = m_generateEndpoints ? 1 : 0; - return m_iterIndex < static_cast(m_points.size() + additionalSize); - } - -private: - void GetNextSolid(glsl::vec2 & result) - { - if (GetGeneratedStart(result) || GetGeneratedEnd(result)) - return; - - result = glsl::ToVec2(m_points[m_iterIndex++]); - } - - void GetNextStipple(glsl::vec2 & result) - { - if (GetGeneratedStart(result) || GetGeneratedEnd(result)) - return; - - if (m_iterIndex == 0) - { - result = glsl::ToVec2(m_points[m_iterIndex++]); - return; - } - - m2::PointD const & currentPt = m_points[m_iterIndex]; - m2::PointD const & prevPt = m_points[m_iterIndex - 1]; - m2::PointD const segmentVector = currentPt - prevPt; - - float const pixelLength = segmentVector.Length() * m_GtoPScale; - int const maskPixelLength = m_stippleRegion.GetMaskPixelLength(); - int const patternPixelLength = m_stippleRegion.GetPatternPixelLength(); - - bool partialSegment = false; - int maxLength = maskPixelLength; - int pixelDiff = maskPixelLength - patternPixelLength; - int resetType = 0; - if (pixelLength > pixelDiff) - { - partialSegment = true; - maxLength = pixelDiff; - resetType = 1; - } - else if (pixelDiff == 0 && pixelLength > maskPixelLength) - partialSegment = true; - - if (partialSegment) - { - int const numParts = static_cast(ceilf(pixelLength / maxLength)); - m2::PointD const singlePart = segmentVector / (double)numParts; - - result = glsl::ToVec2(prevPt + singlePart * m_partGenerated); - - m_partGenerated++; - if (resetType == 0 && m_partGenerated == numParts) - { - m_partGenerated = 0; - m_iterIndex++; - } - else if (resetType == 1 && m_partGenerated > numParts) - { - m_partGenerated = 1; - m_iterIndex++; - } - } - else - result = glsl::ToVec2(m_points[m_iterIndex++]); - } - - bool GetGeneratedStart(glsl::vec2 & result) - { - if (m_iterIndex == -1) - { - ASSERT(m_generateEndpoints, ()); - glsl::vec2 const tmp = glsl::ToVec2(m_points[0]); - result = tmp + glsl::normalize(tmp - glsl::ToVec2(m_points[1])); - m_iterIndex = 0; - return true; - } - - return false; - } - - bool GetGeneratedEnd(glsl::vec2 & result) - { - if (m_iterIndex == static_cast(m_points.size())) - { - ASSERT(m_generateEndpoints, ()); - size_t const size = m_points.size(); - glsl::vec2 const tmp = glsl::ToVec2(m_points[size - 1]); - result = tmp + glsl::normalize(tmp - glsl::ToVec2(m_points[size - 2])); - m_iterIndex++; - return true; - } - - return false; - } - -private: - vector const & m_points; - int m_iterIndex = 0; - int m_partGenerated = 1; - - dp::TextureManager::ColorRegion m_colorRegion; - dp::TextureManager::StippleRegion m_stippleRegion; - bool m_isSolid = true; - - float m_GtoPScale; - bool m_generateEndpoints; -}; - -} // namespace + float const SEGMENT = 0.0f; + float const CAP = 1.0f; + float const LEFT_WIDTH = 1.0f; + float const RIGHT_WIDTH = -1.0f; +} LineShape::LineShape(m2::SharedSpline const & spline, LineViewParams const & params) @@ -277,247 +30,156 @@ LineShape::LineShape(m2::SharedSpline const & spline, void LineShape::Draw(dp::RefPointer batcher, dp::RefPointer textures) const { - LineEnumerator enumerator(m_spline->GetPath(), m_params.m_baseGtoPScale, m_params.m_cap != dp::ButtCap); + buffer_vector geometry; + vector const & path = m_spline->GetPath(); dp::TextureManager::ColorRegion colorRegion; textures->GetColorRegion(m_params.m_color, colorRegion); - enumerator.SetColorParams(colorRegion); + glsl::vec2 colorCoord(glsl::ToVec2(colorRegion.GetTexRect().Center())); dp::TextureManager::StippleRegion maskRegion; - if (!m_params.m_pattern.empty()) - textures->GetStippleRegion(m_params.m_pattern, maskRegion); - else - textures->GetStippleRegion(dp::TextureManager::TStipplePattern{1}, maskRegion); + textures->GetStippleRegion(dp::TextureManager::TStipplePattern{1}, maskRegion); + glsl::vec2 maskCoord(glsl::ToVec2(maskRegion.GetTexRect().Center())); - enumerator.SetMaskParams(maskRegion, m_params.m_pattern.empty()); - - float const joinType = m_params.m_join == dp::RoundJoin ? 1 : 0; - float const capType = m_params.m_cap == dp::RoundCap ? -1 : 1; float const halfWidth = m_params.m_width / 2.0f; - float const insetHalfWidth= 1.0f * halfWidth; + bool generateCap = m_params.m_cap != dp::ButtCap; - size_t size = m_spline->GetPath().size(); - if (m_params.m_cap != dp::ButtCap) - size += 2; - - // We generate quad for each line segment - // For line with 3 point A -- B -- C, we will generate - // A1 --- AB1 BC1 --- C1 - // | | | | - // A2 --- AB2 BC2 --- C2 - // By this segment count == points.size() - 1 - size_t const vertexCount = (size - 1) * 4; - vector vertexArray; - vector dxValsArray; - vector centersArray; - vector widthTypeArray; - vector colorArray; - vector maskArray; - - vertexArray.reserve(vertexCount); - dxValsArray.reserve(vertexCount); - centersArray.reserve(vertexCount); - widthTypeArray.reserve(vertexCount); - - glsl::vec2 v2 = enumerator.GetNextPoint(); - glsl::vec2 v3 = enumerator.GetNextPoint(); - glsl::vec2 v1 = 2.0f * v2 - v3; - - GeomHelper helper(v1, v2, v3); - - // Fill first two vertex of segment. A1 and A2 - if (m_params.m_cap != dp::ButtCap) + auto const calcTangentAndNormals = [&halfWidth](glsl::vec2 const & pt0, glsl::vec2 const & pt1, + glsl::vec2 & tangent, glsl::vec2 & leftNormal, + glsl::vec2 & rightNormal) { - vertexArray.push_back(glsl::vec4(helper.m_nextPt, helper.m_leftBisector)); - vertexArray.push_back(glsl::vec4(helper.m_nextPt, helper.m_rightBisector)); - widthTypeArray.push_back(glsl::vec4(halfWidth, capType, -1, insetHalfWidth)); - widthTypeArray.push_back(glsl::vec4(-halfWidth, capType, -1, insetHalfWidth)); - } - else + tangent = pt1 - pt0; + leftNormal = halfWidth * glsl::normalize(glsl::vec2(tangent.y, -tangent.x)); + rightNormal = -leftNormal; + }; + + float capType = m_params.m_cap == dp::RoundCap ? CAP : SEGMENT; + + glsl::vec2 leftSegment(SEGMENT, LEFT_WIDTH); + glsl::vec2 rightSegment(SEGMENT, RIGHT_WIDTH); + + if (generateCap) { - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_leftBisector)); - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_rightBisector)); - widthTypeArray.push_back(glsl::vec4(halfWidth, 0, joinType, insetHalfWidth)); - widthTypeArray.push_back(glsl::vec4(-halfWidth, 0, joinType, insetHalfWidth)); + glsl::vec2 startPoint = glsl::ToVec2(path[0]); + glsl::vec2 endPoint = glsl::ToVec2(path[1]); + glsl::vec2 tangent, leftNormal, rightNormal; + calcTangentAndNormals(startPoint, endPoint, tangent, leftNormal, rightNormal); + tangent = -halfWidth * glsl::normalize(tangent); + + glsl::vec3 pivot = glsl::vec3(startPoint, m_params.m_depth); + glsl::vec2 leftCap(capType, LEFT_WIDTH); + glsl::vec2 rightCap(capType, RIGHT_WIDTH); + + geometry.push_back(gpu::LineVertex(pivot, leftNormal + tangent, colorCoord, maskCoord, leftCap)); + geometry.push_back(gpu::LineVertex(pivot, rightNormal + tangent, colorCoord, maskCoord, rightCap)); + geometry.push_back(gpu::LineVertex(pivot, leftNormal, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(pivot, rightNormal, colorCoord, maskCoord, rightSegment)); } - dxValsArray.push_back(glsl::vec3(0.0f, -1.0f, m_params.m_depth)); - dxValsArray.push_back(glsl::vec3(0.0f, -1.0f, m_params.m_depth)); - centersArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_nextPt)); - centersArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_nextPt)); + glsl::vec2 prevPoint; + glsl::vec2 prevLeftNormal; + glsl::vec2 prevRightNormal; - //points in the middle - while (enumerator.HasPoint()) + for (size_t i = 1; i < path.size(); ++i) { - helper.MoveNext(enumerator.GetNextPoint()); - // In line like A -- AB BC -- C we fill AB points in this block - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_leftBisector)); - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_rightBisector)); - dxValsArray.push_back(glsl::vec3(helper.m_dx, 1.0f, m_params.m_depth)); - dxValsArray.push_back(glsl::vec3(-helper.m_dx, 1.0f, m_params.m_depth)); - centersArray.push_back(glsl::vec4(helper.m_prevPt, helper.m_currentPt)); - centersArray.push_back(glsl::vec4(helper.m_prevPt, helper.m_currentPt)); - widthTypeArray.push_back(glsl::vec4(halfWidth, 0, joinType, insetHalfWidth)); - widthTypeArray.push_back(glsl::vec4(-halfWidth, 0, joinType, insetHalfWidth)); + glsl::vec2 startPoint = glsl::ToVec2(path[i - 1]); + glsl::vec2 endPoint = glsl::ToVec2(path[i]); + glsl::vec2 tangent, leftNormal, rightNormal; + calcTangentAndNormals(startPoint, endPoint, tangent, leftNormal, rightNormal); - // As AB and BC point calculation on the same source point B - // we also fill BC points - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_leftBisector)); - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_rightBisector)); - dxValsArray.push_back(glsl::vec3(-helper.m_dx * helper.m_aspect, -1.0f, m_params.m_depth)); - dxValsArray.push_back(glsl::vec3(helper.m_dx * helper.m_aspect, -1.0f, m_params.m_depth)); - centersArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_nextPt)); - centersArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_nextPt)); - widthTypeArray.push_back(glsl::vec4(halfWidth, 0, joinType, insetHalfWidth)); - widthTypeArray.push_back(glsl::vec4(-halfWidth, 0, joinType, insetHalfWidth)); - } + glsl::vec3 startPivot = glsl::vec3(startPoint, m_params.m_depth); + glsl::vec3 endPivot = glsl::vec3(endPoint, m_params.m_depth); - // Here we fill last vertexes of last segment - v1 = helper.m_currentPt; - v2 = helper.m_nextPt; - v3 = 2.0f * v2 - v1; - helper = GeomHelper(v1, v2, v3); - - if (m_params.m_cap != dp::ButtCap) - { - vertexArray.push_back(glsl::vec4(helper.m_prevPt, helper.m_leftBisector)); - vertexArray.push_back(glsl::vec4(helper.m_prevPt, helper.m_rightBisector)); - widthTypeArray.push_back(glsl::vec4(halfWidth, capType, 1, insetHalfWidth)); - widthTypeArray.push_back(glsl::vec4(-halfWidth, capType, 1, insetHalfWidth)); - } - else - { - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_leftBisector)); - vertexArray.push_back(glsl::vec4(helper.m_currentPt, helper.m_rightBisector)); - widthTypeArray.push_back(glsl::vec4(halfWidth, 0, joinType, insetHalfWidth)); - widthTypeArray.push_back(glsl::vec4(-halfWidth, 0, joinType, insetHalfWidth)); - } - - dxValsArray.push_back(glsl::vec3(-helper.m_dx * helper.m_aspect, 1.0f, m_params.m_depth)); - dxValsArray.push_back(glsl::vec3(helper.m_dx * helper.m_aspect, 1.0f, m_params.m_depth)); - centersArray.push_back(glsl::vec4(helper.m_prevPt, helper.m_currentPt)); - centersArray.push_back(glsl::vec4(helper.m_prevPt, helper.m_currentPt)); - - ASSERT_EQUAL(vertexArray.size(), dxValsArray.size(), ()); - ASSERT_EQUAL(vertexArray.size(), centersArray.size(), ()); - ASSERT_EQUAL(vertexArray.size(), widthTypeArray.size(), ()); - ASSERT_EQUAL(vertexArray.size() % 4, 0, ()); - - colorArray.resize(vertexArray.size(), enumerator.GetColorCoord()); - - if (!enumerator.IsSolid()) - { - dp::TextureManager::StippleRegion const & region = enumerator.GetStippleMaskItem(); - m2::RectF const & stippleRect = region.GetTexRect(); - - float const maskTexLength = region.GetMaskPixelLength() / stippleRect.SizeX(); - float const patternTexLength = region.GetPatternPixelLength() / maskTexLength; - float const dxFactor = halfWidth / (2.0f * m_params.m_baseGtoPScale); - float patternStart = 0.0f; - size_t const quadCount = vertexArray.size() / 4; - for(size_t i = 0; i < quadCount; ++i) + // Create join beetween current segment and previous + if (i > 1) { - size_t const baseIndex = i * 4; - float dx1 = dxValsArray[baseIndex + 0].x * dxFactor; - float dx2 = dxValsArray[baseIndex + 1].x * dxFactor; - float dx3 = dxValsArray[baseIndex + 2].x * dxFactor; - float dx4 = dxValsArray[baseIndex + 3].x * dxFactor; - float const lengthPx = glsl::length(vertexArray[baseIndex].xy() - vertexArray[baseIndex + 2].xy()); - float const lengthTex = lengthPx * m_params.m_baseGtoPScale / (maskTexLength * (fabs(dx1) + fabs(dx3) + 1.0)); - float const f1 = fabs(dx1) * lengthTex; - float const length2 = (fabs(dx1) + 1.0) * lengthTex; + glsl::vec2 zeroNormal(0.0, 0.0); + glsl::vec2 prevForming, nextForming; + if (glsl::dot(prevLeftNormal, tangent) < 0) + { + prevForming = prevLeftNormal; + nextForming = leftNormal; + } + else + { + prevForming = prevRightNormal; + nextForming = rightNormal; + } - dx1 *= lengthTex; - dx2 *= lengthTex; - dx3 *= lengthTex; - dx4 *= lengthTex; + if (m_params.m_join == dp::BevelJoin) + { + geometry.push_back(gpu::LineVertex(startPivot, prevForming, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(startPivot, zeroNormal, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(startPivot, nextForming, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(startPivot, nextForming, colorCoord, maskCoord, leftSegment)); + } + else + { + glsl::vec2 middleForming = glsl::normalize(prevForming + nextForming); + glsl::vec2 zeroDxDy(0.0, 0.0); - maskArray.push_back(glsl::vec2(stippleRect.minX() + patternStart + dx1 + f1, stippleRect.minY())); - maskArray.push_back(glsl::vec2(stippleRect.minX() + patternStart + dx2 + f1, stippleRect.maxY())); - maskArray.push_back(glsl::vec2(stippleRect.minX() + patternStart + length2 + dx3, stippleRect.minY())); - maskArray.push_back(glsl::vec2(stippleRect.minX() + patternStart + length2 + dx4, stippleRect.maxY())); + if (m_params.m_join == dp::MiterJoin) + { + float const b = glsl::length(prevForming - nextForming) / 2.0; + float const a = glsl::length(prevForming); + middleForming *= static_cast(sqrt(a * a + b * b)); - patternStart += length2; - while (patternStart >= patternTexLength) - patternStart -= patternTexLength; + geometry.push_back(gpu::LineVertex(startPivot, prevForming, colorCoord, maskCoord, zeroDxDy)); + geometry.push_back(gpu::LineVertex(startPivot, zeroNormal, colorCoord, maskCoord, zeroDxDy)); + geometry.push_back(gpu::LineVertex(startPivot, middleForming, colorCoord, maskCoord, zeroDxDy)); + geometry.push_back(gpu::LineVertex(startPivot, nextForming, colorCoord, maskCoord, zeroDxDy)); + } + else + { + middleForming *= glsl::length(prevForming); + + glsl::vec2 dxdyLeft(0.0, -1.0); + glsl::vec2 dxdyRight(0.0, -1.0); + glsl::vec2 dxdyMiddle(1.0, 1.0); + geometry.push_back(gpu::LineVertex(startPivot, zeroNormal, colorCoord, maskCoord, zeroDxDy)); + geometry.push_back(gpu::LineVertex(startPivot, prevForming, colorCoord, maskCoord, dxdyLeft)); + geometry.push_back(gpu::LineVertex(startPivot, nextForming, colorCoord, maskCoord, dxdyRight)); + geometry.push_back(gpu::LineVertex(startPivot, middleForming, colorCoord, maskCoord, dxdyMiddle)); + } + } } + + geometry.push_back(gpu::LineVertex(startPivot, leftNormal, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(startPivot, rightNormal, colorCoord, maskCoord, rightSegment)); + geometry.push_back(gpu::LineVertex(endPivot, leftNormal, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(endPivot, rightNormal, colorCoord, maskCoord, rightSegment)); + + prevPoint = startPoint; + prevLeftNormal = leftNormal; + prevRightNormal = rightNormal; + } + + if (generateCap) + { + size_t lastPointIndex = path.size() - 1; + glsl::vec2 startPoint = glsl::ToVec2(path[lastPointIndex - 1]); + glsl::vec2 endPoint = glsl::ToVec2(path[lastPointIndex]); + glsl::vec2 tangent, leftNormal, rightNormal; + calcTangentAndNormals(startPoint, endPoint, tangent, leftNormal, rightNormal); + tangent = halfWidth * glsl::normalize(tangent); + + glsl::vec3 pivot = glsl::vec3(endPoint, m_params.m_depth); + glsl::vec2 leftCap(capType, LEFT_WIDTH); + glsl::vec2 rightCap(capType, RIGHT_WIDTH); + + geometry.push_back(gpu::LineVertex(pivot, leftNormal, colorCoord, maskCoord, leftSegment)); + geometry.push_back(gpu::LineVertex(pivot, rightNormal, colorCoord, maskCoord, rightSegment)); + geometry.push_back(gpu::LineVertex(pivot, leftNormal + tangent, colorCoord, maskCoord, leftCap)); + geometry.push_back(gpu::LineVertex(pivot, rightNormal + tangent, colorCoord, maskCoord, rightCap)); } - else - maskArray.resize(vertexArray.size(), enumerator.GetSolidMaskCoord()); dp::GLState state(gpu::LINE_PROGRAM, dp::GLState::GeometryLayer); - state.SetBlending(dp::Blending(true)); - state.SetColorTexture(enumerator.GetColorTexture()); - state.SetMaskTexture(enumerator.GetMaskTexture()); + state.SetBlending(true); + state.SetColorTexture(colorRegion.GetTexture()); + state.SetMaskTexture(maskRegion.GetTexture()); - dp::AttributeProvider provider(6, vertexArray.size()); - - { - dp::BindingInfo pos_dir(1); - dp::BindingDecl & decl = pos_dir.GetBindingDecl(0); - decl.m_attributeName = "a_position"; - decl.m_componentCount = 4; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = 0; - provider.InitStream(0, pos_dir, dp::MakeStackRefPointer(vertexArray.data())); - } - { - dp::BindingInfo deltas(1); - dp::BindingDecl & decl = deltas.GetBindingDecl(0); - decl.m_attributeName = "a_deltas"; - decl.m_componentCount = 3; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = 0; - - provider.InitStream(1, deltas, dp::MakeStackRefPointer(dxValsArray.data())); - } - { - dp::BindingInfo width_type(1); - dp::BindingDecl & decl = width_type.GetBindingDecl(0); - decl.m_attributeName = "a_width_type"; - decl.m_componentCount = 4; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = 0; - - provider.InitStream(2, width_type, dp::MakeStackRefPointer(widthTypeArray.data())); - } - { - dp::BindingInfo centres(1); - dp::BindingDecl & decl = centres.GetBindingDecl(0); - decl.m_attributeName = "a_centres"; - decl.m_componentCount = 4; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = 0; - - provider.InitStream(3, centres, dp::MakeStackRefPointer(centersArray.data())); - } - { - dp::BindingInfo colorBinding(1); - dp::BindingDecl & decl = colorBinding.GetBindingDecl(0); - decl.m_attributeName = "a_color"; - decl.m_componentCount = 2; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = 0; - - provider.InitStream(4, colorBinding, dp::MakeStackRefPointer(colorArray.data())); - } - - { - dp::BindingInfo ind(1); - dp::BindingDecl & decl = ind.GetBindingDecl(0); - decl.m_attributeName = "a_mask"; - decl.m_componentCount = 2; - decl.m_componentType = gl_const::GLFloatType; - decl.m_offset = 0; - decl.m_stride = 0; - - provider.InitStream(5, ind, dp::MakeStackRefPointer(maskArray.data())); - } + dp::AttributeProvider provider(1, geometry.size()); + provider.InitStream(0, gpu::LineVertex::GetBindingInfo(), dp::MakeStackRefPointer(geometry.data())); batcher->InsertListOfStrip(state, dp::MakeStackRefPointer(&provider), 4); } diff --git a/drape_head/testing_engine.cpp b/drape_head/testing_engine.cpp index 031b39440a..0451c921ba 100644 --- a/drape_head/testing_engine.cpp +++ b/drape_head/testing_engine.cpp @@ -381,7 +381,7 @@ void TestingEngine::DrawImpl() vector path; path.push_back(m2::PointD(92.277071f, 50.9271164f)); path.push_back(m2::PointD(98.277071f, 50.9271164f)); - path.push_back(m2::PointD(106.277071f, 47.9271164f)); + path.push_back(m2::PointD(106.277071f, 45.9271164f)); m2::SharedSpline spline(path); PathTextViewParams ptvp; @@ -394,11 +394,27 @@ void TestingEngine::DrawImpl() LineViewParams lvp; lvp.m_baseGtoPScale = ptvp.m_baseGtoPScale; lvp.m_depth = 90.0f; - lvp.m_cap = dp::RoundCap; + lvp.m_cap = dp::SquareCap; lvp.m_color = dp::Color::Red(); - lvp.m_width = 3.0f; + lvp.m_width = 16.0f; lvp.m_join = dp::BevelJoin; LineShape(spline, lvp).Draw(m_batcher.GetRefPointer(), m_textures.GetRefPointer()); + + { + vector path1; + path1.push_back(m2::PointD(92.277071f, 45.9271164f)); + path1.push_back(m2::PointD(98.277071f, 45.9271164f)); + path1.push_back(m2::PointD(98.277071f, 40.9271164f)); + path1.push_back(m2::PointD(100.277071f, 38.9271164f)); + path1.push_back(m2::PointD(101.277071f, 45.9271164f)); + path1.push_back(m2::PointD(102.277071f, 40.9271164f)); + m2::SharedSpline spl1(path1); + + lvp.m_join = dp::BevelJoin; + lvp.m_cap = dp::RoundCap; + lvp.m_color = dp::Color::Black(); + LineShape(spl1, lvp).Draw(m_batcher.GetRefPointer(), m_textures.GetRefPointer()); + } } void TestingEngine::DrawRects()