forked from organicmaps/organicmaps-tmp
[drape] Move dashed lines to separate shader. Remove all discards from shaders.
This commit is contained in:
parent
6a94fc8496
commit
ab1d72dd6f
13 changed files with 398 additions and 254 deletions
|
@ -55,4 +55,27 @@ uint8_t FillDecl(size_t index, string const & attrName, dp::BindingInfo & info,
|
|||
return sizeof(TFieldType);
|
||||
}
|
||||
|
||||
template <typename TVertex>
|
||||
class BindingFiller
|
||||
{
|
||||
public:
|
||||
BindingFiller(uint8_t count, uint8_t id = 0)
|
||||
: m_info(count, id)
|
||||
{
|
||||
}
|
||||
|
||||
template<typename TFieldType>
|
||||
void FillDecl(string const & attrName)
|
||||
{
|
||||
m_offset += dp::FillDecl<TFieldType, TVertex>(m_index, attrName, m_info, m_offset);
|
||||
++m_index;
|
||||
}
|
||||
|
||||
dp::BindingInfo m_info;
|
||||
|
||||
private:
|
||||
size_t m_index = 0;
|
||||
uint8_t m_offset = 0;
|
||||
};
|
||||
|
||||
} // namespace dp
|
||||
|
|
|
@ -13,6 +13,8 @@ SOURCES += glfunctions.cpp
|
|||
|
||||
OTHER_FILES += \
|
||||
shaders/compass_vertex_shader.vsh \
|
||||
shaders/dashed_fragment_shader.fsh \
|
||||
shaders/dashed_vertex_shader.vsh \
|
||||
shaders/line_fragment_shader.fsh \
|
||||
shaders/line_vertex_shader.vsh \
|
||||
shaders/my_position_shader.vsh \
|
||||
|
|
14
drape/shaders/dashed_fragment_shader.fsh
Normal file
14
drape/shaders/dashed_fragment_shader.fsh
Normal file
|
@ -0,0 +1,14 @@
|
|||
varying vec2 v_colorTexCoord;
|
||||
varying vec2 v_maskTexCoord;
|
||||
|
||||
uniform sampler2D u_colorTex;
|
||||
uniform sampler2D u_maskTex;
|
||||
uniform float u_opacity;
|
||||
|
||||
void main(void)
|
||||
{
|
||||
vec4 color = texture2D(u_colorTex, v_colorTexCoord);
|
||||
vec4 mask = texture2D(u_maskTex, v_maskTexCoord);
|
||||
color.a = color.a * mask.a * u_opacity;
|
||||
gl_FragColor = color;
|
||||
}
|
26
drape/shaders/dashed_vertex_shader.vsh
Normal file
26
drape/shaders/dashed_vertex_shader.vsh
Normal file
|
@ -0,0 +1,26 @@
|
|||
attribute vec3 a_position;
|
||||
attribute vec2 a_normal;
|
||||
attribute vec2 a_colorTexCoord;
|
||||
attribute vec2 a_maskTexCoord;
|
||||
|
||||
uniform mat4 modelView;
|
||||
uniform mat4 projection;
|
||||
|
||||
varying vec2 v_colorTexCoord;
|
||||
varying vec2 v_maskTexCoord;
|
||||
|
||||
void main(void)
|
||||
{
|
||||
float halfWidth = length(a_normal);
|
||||
vec2 transformedAxisPos = (vec4(a_position.xy, 0.0, 1.0) * modelView).xy;
|
||||
if (halfWidth != 0.0)
|
||||
{
|
||||
vec4 glbShiftPos = vec4(a_position.xy + a_normal, 0.0, 1.0);
|
||||
vec2 shiftPos = (glbShiftPos * modelView).xy;
|
||||
transformedAxisPos = transformedAxisPos + normalize(shiftPos - transformedAxisPos) * halfWidth;
|
||||
}
|
||||
|
||||
v_colorTexCoord = a_colorTexCoord;
|
||||
v_maskTexCoord = a_maskTexCoord;
|
||||
gl_Position = vec4(transformedAxisPos, a_position.z, 1.0) * projection;
|
||||
}
|
|
@ -1,16 +1,10 @@
|
|||
varying vec2 v_colorTexCoord;
|
||||
varying vec2 v_maskTexCoord;
|
||||
|
||||
uniform sampler2D u_colorTex;
|
||||
uniform sampler2D u_maskTex;
|
||||
uniform float u_opacity;
|
||||
|
||||
void main(void)
|
||||
{
|
||||
float mask = texture2D(u_maskTex, v_maskTexCoord).a;
|
||||
if (mask < 0.1)
|
||||
discard;
|
||||
|
||||
vec4 color = texture2D(u_colorTex, v_colorTexCoord);
|
||||
color.a *= u_opacity;
|
||||
gl_FragColor = color;
|
||||
|
|
|
@ -1,7 +1,6 @@
|
|||
attribute vec3 a_position;
|
||||
attribute vec2 a_normal;
|
||||
attribute vec2 a_colorTexCoord;
|
||||
attribute vec2 a_maskTexCoord;
|
||||
|
||||
uniform mat4 modelView;
|
||||
uniform mat4 projection;
|
||||
|
@ -21,6 +20,5 @@ void main(void)
|
|||
}
|
||||
|
||||
v_colorTexCoord = a_colorTexCoord;
|
||||
v_maskTexCoord = a_maskTexCoord;
|
||||
gl_Position = vec4(transformedAxisPos, a_position.z, 1.0) * projection;
|
||||
}
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
LINE_PROGRAM line_vertex_shader.vsh line_fragment_shader.fsh
|
||||
TEXTURING_PROGRAM texturing_vertex_shader.vsh texturing_fragment_shader.fsh
|
||||
DASHED_LINE_PROGRAM dashed_vertex_shader.vsh dashed_fragment_shader.fsh
|
||||
TEXT_PROGRAM text_vertex_shader.vsh text_fragment_shader.fsh
|
||||
COMPASS_PROGRAM compass_vertex_shader.vsh texturing_fragment_shader.fsh
|
||||
RULER_PROGRAM ruler_vertex_shader.vsh texturing_fragment_shader.fsh
|
||||
|
|
|
@ -6,9 +6,6 @@ varying vec2 v_colorTexCoords;
|
|||
void main(void)
|
||||
{
|
||||
vec4 finalColor = texture2D(u_colorTex, v_colorTexCoords);
|
||||
if (finalColor.a < 0.1)
|
||||
discard;
|
||||
|
||||
finalColor.a *= u_opacity;
|
||||
gl_FragColor = finalColor;
|
||||
}
|
||||
|
|
|
@ -12,6 +12,7 @@ enum VertexType
|
|||
TextStatic,
|
||||
TextDynamic,
|
||||
Line,
|
||||
DashedLine,
|
||||
Route,
|
||||
TypeCount
|
||||
};
|
||||
|
@ -30,121 +31,64 @@ dp::BindingInfo SolidTexturingBindingInit()
|
|||
sizeof(SolidTexturingVertex::TNormal) +
|
||||
sizeof(SolidTexturingVertex::TTexCoord)), "");
|
||||
|
||||
dp::BindingInfo info(3);
|
||||
dp::BindingFiller<SolidTexturingVertex> filler(3);
|
||||
filler.FillDecl<SolidTexturingVertex::TPosition>("a_position");
|
||||
filler.FillDecl<SolidTexturingVertex::TNormal>("a_normal");
|
||||
filler.FillDecl<SolidTexturingVertex::TTexCoord>("a_colorTexCoords");
|
||||
|
||||
dp::BindingDecl & posDecl = info.GetBindingDecl(0);
|
||||
posDecl.m_attributeName = "a_position";
|
||||
posDecl.m_componentCount = glsl::GetComponentCount<SolidTexturingVertex::TPosition>();
|
||||
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<SolidTexturingVertex::TNormal>();
|
||||
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<SolidTexturingVertex::TTexCoord>();
|
||||
colorTexCoordDecl.m_componentType = gl_const::GLFloatType;
|
||||
colorTexCoordDecl.m_offset = normalDecl.m_offset + sizeof(SolidTexturingVertex::TNormal);
|
||||
colorTexCoordDecl.m_stride = posDecl.m_stride;
|
||||
|
||||
return info;
|
||||
return filler.m_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<TextStaticVertex::TPosition>();
|
||||
posDecl.m_componentType = gl_const::GLFloatType;
|
||||
posDecl.m_offset = 0;
|
||||
posDecl.m_stride = sizeof(TextStaticVertex);
|
||||
dp::BindingFiller<TextStaticVertex> filler(4);
|
||||
filler.FillDecl<TextStaticVertex::TPosition>("a_position");
|
||||
filler.FillDecl<TextStaticVertex::TTexCoord>("a_colorTexCoord");
|
||||
filler.FillDecl<TextStaticVertex::TTexCoord>("a_outlineColorTexCoord");
|
||||
filler.FillDecl<TextStaticVertex::TTexCoord>("a_maskTexCoord");
|
||||
|
||||
dp::BindingDecl & colorDecl = info.GetBindingDecl(1);
|
||||
colorDecl.m_attributeName = "a_colorTexCoord";
|
||||
colorDecl.m_componentCount = glsl::GetComponentCount<TextStaticVertex::TTexCoord>();
|
||||
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<TextStaticVertex::TTexCoord>();
|
||||
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<TextStaticVertex::TTexCoord>();
|
||||
maskDecl.m_componentType = gl_const::GLFloatType;
|
||||
maskDecl.m_offset = outlineDecl.m_offset + sizeof(TextStaticVertex::TTexCoord);
|
||||
maskDecl.m_stride = posDecl.m_stride;
|
||||
|
||||
return info;
|
||||
return filler.m_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<TextDynamicVertex::TNormal>();
|
||||
decl.m_componentType = gl_const::GLFloatType;
|
||||
decl.m_offset = 0;
|
||||
decl.m_stride = sizeof(TextDynamicVertex);
|
||||
dp::BindingFiller<TextDynamicVertex> filler(1, TextDynamicVertex::GetDynamicStreamID());
|
||||
filler.FillDecl<TextDynamicVertex::TNormal>("a_normal");
|
||||
|
||||
return info;
|
||||
return filler.m_info;
|
||||
}
|
||||
|
||||
dp::BindingInfo LineBindingInit()
|
||||
{
|
||||
static_assert(sizeof(LineVertex) == sizeof(LineVertex::TPosition) +
|
||||
sizeof(LineVertex::TNormal) +
|
||||
2 * sizeof(LineVertex::TTexCoord), "");
|
||||
dp::BindingInfo info(4);
|
||||
>
|
||||
sizeof(LineVertex::TTexCoord), "");
|
||||
dp::BindingFiller<LineVertex> filler(3);
|
||||
filler.FillDecl<LineVertex::TPosition>("a_position");
|
||||
filler.FillDecl<LineVertex::TNormal>("a_normal");
|
||||
filler.FillDecl<LineVertex::TTexCoord>("a_colorTexCoord");
|
||||
|
||||
dp::BindingDecl & posDecl = info.GetBindingDecl(0);
|
||||
posDecl.m_attributeName = "a_position";
|
||||
posDecl.m_componentCount = glsl::GetComponentCount<LineVertex::TPosition>();
|
||||
posDecl.m_componentType = gl_const::GLFloatType;
|
||||
posDecl.m_offset = 0;
|
||||
posDecl.m_stride = sizeof(LineVertex);
|
||||
return filler.m_info;
|
||||
}
|
||||
|
||||
dp::BindingDecl & normalDecl = info.GetBindingDecl(1);
|
||||
normalDecl.m_attributeName = "a_normal";
|
||||
normalDecl.m_componentCount = glsl::GetComponentCount<LineVertex::TNormal>();
|
||||
normalDecl.m_componentType = gl_const::GLFloatType;
|
||||
normalDecl.m_offset = posDecl.m_offset + sizeof(LineVertex::TPosition);
|
||||
normalDecl.m_stride = posDecl.m_stride;
|
||||
dp::BindingInfo DashedLineBindingInit()
|
||||
{
|
||||
static_assert(sizeof(DashedLineVertex) == sizeof(DashedLineVertex::TPosition) +
|
||||
sizeof(DashedLineVertex::TNormal) +
|
||||
2 * sizeof(DashedLineVertex::TTexCoord), "");
|
||||
|
||||
dp::BindingDecl & colorDecl = info.GetBindingDecl(2);
|
||||
colorDecl.m_attributeName = "a_colorTexCoord";
|
||||
colorDecl.m_componentCount = glsl::GetComponentCount<LineVertex::TTexCoord>();
|
||||
colorDecl.m_componentType = gl_const::GLFloatType;
|
||||
colorDecl.m_offset = normalDecl.m_offset + sizeof(LineVertex::TNormal);
|
||||
colorDecl.m_stride = posDecl.m_stride;
|
||||
dp::BindingFiller<DashedLineVertex> filler(4);
|
||||
filler.FillDecl<DashedLineVertex::TPosition>("a_position");
|
||||
filler.FillDecl<DashedLineVertex::TNormal>("a_normal");
|
||||
filler.FillDecl<DashedLineVertex::TTexCoord>("a_colorTexCoord");
|
||||
filler.FillDecl<DashedLineVertex::TTexCoord>("a_maskTexCoord");
|
||||
|
||||
dp::BindingDecl & maskDecl = info.GetBindingDecl(3);
|
||||
maskDecl.m_attributeName = "a_maskTexCoord";
|
||||
maskDecl.m_componentCount = glsl::GetComponentCount<LineVertex::TTexCoord>();
|
||||
maskDecl.m_componentType = gl_const::GLFloatType;
|
||||
maskDecl.m_offset = colorDecl.m_offset + sizeof(LineVertex::TTexCoord);
|
||||
maskDecl.m_stride = posDecl.m_stride;
|
||||
|
||||
return info;
|
||||
return filler.m_info;
|
||||
}
|
||||
|
||||
dp::BindingInfo RouteBindingInit()
|
||||
|
@ -152,30 +96,13 @@ dp::BindingInfo RouteBindingInit()
|
|||
STATIC_ASSERT(sizeof(RouteVertex) == sizeof(RouteVertex::TPosition) +
|
||||
sizeof(RouteVertex::TNormal) +
|
||||
sizeof(RouteVertex::TLength));
|
||||
dp::BindingInfo info(3);
|
||||
|
||||
dp::BindingDecl & posDecl = info.GetBindingDecl(0);
|
||||
posDecl.m_attributeName = "a_position";
|
||||
posDecl.m_componentCount = glsl::GetComponentCount<RouteVertex::TPosition>();
|
||||
posDecl.m_componentType = gl_const::GLFloatType;
|
||||
posDecl.m_offset = 0;
|
||||
posDecl.m_stride = sizeof(RouteVertex);
|
||||
dp::BindingFiller<RouteVertex> filler(3);
|
||||
filler.FillDecl<RouteVertex::TPosition>("a_position");
|
||||
filler.FillDecl<RouteVertex::TNormal>("a_normal");
|
||||
filler.FillDecl<RouteVertex::TLength>("a_length");
|
||||
|
||||
dp::BindingDecl & normalDecl = info.GetBindingDecl(1);
|
||||
normalDecl.m_attributeName = "a_normal";
|
||||
normalDecl.m_componentCount = glsl::GetComponentCount<RouteVertex::TNormal>();
|
||||
normalDecl.m_componentType = gl_const::GLFloatType;
|
||||
normalDecl.m_offset = posDecl.m_offset + sizeof(RouteVertex::TPosition);
|
||||
normalDecl.m_stride = posDecl.m_stride;
|
||||
|
||||
dp::BindingDecl & lengthDecl = info.GetBindingDecl(2);
|
||||
lengthDecl.m_attributeName = "a_length";
|
||||
lengthDecl.m_componentCount = glsl::GetComponentCount<RouteVertex::TLength>();
|
||||
lengthDecl.m_componentType = gl_const::GLFloatType;
|
||||
lengthDecl.m_offset = normalDecl.m_offset + sizeof(RouteVertex::TNormal);
|
||||
lengthDecl.m_stride = posDecl.m_stride;
|
||||
|
||||
return info;
|
||||
return filler.m_info;
|
||||
}
|
||||
|
||||
BindingNode g_bindingNodes[TypeCount];
|
||||
|
@ -185,6 +112,7 @@ TInitFunction g_initFunctions[TypeCount] =
|
|||
&TextStaticBindingInit,
|
||||
&TextDynamicBindingInit,
|
||||
&LineBindingInit,
|
||||
&DashedLineBindingInit,
|
||||
&RouteBindingInit
|
||||
};
|
||||
|
||||
|
@ -268,16 +196,13 @@ 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)
|
||||
{
|
||||
}
|
||||
|
||||
LineVertex::LineVertex(TPosition const & position, TNormal const & normal,
|
||||
TTexCoord const & color, TTexCoord const & mask)
|
||||
LineVertex::LineVertex(TPosition const & position, TNormal const & normal, TTexCoord const & color)
|
||||
: m_position(position)
|
||||
, m_normal(normal)
|
||||
, m_colorTexCoord(color)
|
||||
, m_maskTexCoord(mask)
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -286,6 +211,23 @@ dp::BindingInfo const & LineVertex::GetBindingInfo()
|
|||
return GetBinding(Line);
|
||||
}
|
||||
|
||||
DashedLineVertex::DashedLineVertex()
|
||||
: m_maskTexCoord(0.0, 0.0)
|
||||
{
|
||||
}
|
||||
|
||||
DashedLineVertex::DashedLineVertex(TPosition const & position, TNormal const & normal,
|
||||
TTexCoord const & color, TTexCoord const & mask)
|
||||
: LineVertex(position, normal, color)
|
||||
, m_maskTexCoord(mask)
|
||||
{
|
||||
}
|
||||
|
||||
dp::BindingInfo const & DashedLineVertex::GetBindingInfo()
|
||||
{
|
||||
return GetBinding(DashedLine);
|
||||
}
|
||||
|
||||
RouteVertex::RouteVertex()
|
||||
: m_position(0.0, 0.0, 0.0)
|
||||
, m_normal(0.0, 0.0)
|
||||
|
|
|
@ -62,12 +62,21 @@ typedef buffer_vector<TextDynamicVertex, 128> TTextDynamicVertexBuffer;
|
|||
struct LineVertex : BaseVertex
|
||||
{
|
||||
LineVertex();
|
||||
LineVertex(TPosition const & position, TNormal const & normal,
|
||||
TTexCoord const & color, TTexCoord const & mask);
|
||||
LineVertex(TPosition const & position, TNormal const & normal, TTexCoord const & color);
|
||||
|
||||
TPosition m_position;
|
||||
TNormal m_normal;
|
||||
TTexCoord m_colorTexCoord;
|
||||
|
||||
static dp::BindingInfo const & GetBindingInfo();
|
||||
};
|
||||
|
||||
struct DashedLineVertex : LineVertex
|
||||
{
|
||||
DashedLineVertex();
|
||||
DashedLineVertex(TPosition const & position, TNormal const & normal,
|
||||
TTexCoord const & color, TTexCoord const & mask);
|
||||
|
||||
TTexCoord m_maskTexCoord;
|
||||
|
||||
static dp::BindingInfo const & GetBindingInfo();
|
||||
|
|
|
@ -751,6 +751,7 @@ void FrontendRenderer::Routine::Do()
|
|||
context->setDefaultFramebuffer();
|
||||
bool const hasAsyncRoutines = m_renderer.m_texMng->UpdateDynamicTextures();
|
||||
m_renderer.RenderScene(modelView);
|
||||
GLFunctions::glFlush();
|
||||
bool const animActive = InterpolationHolder::Instance().Advance(frameTime);
|
||||
modelView = m_renderer.UpdateScene(viewChanged);
|
||||
|
||||
|
|
|
@ -18,55 +18,206 @@ namespace df
|
|||
|
||||
namespace
|
||||
{
|
||||
using LV = gpu::LineVertex;
|
||||
using TGeometryBuffer = buffer_vector<gpu::LineVertex, 256>;
|
||||
|
||||
class TextureCoordGenerator
|
||||
class TextureCoordGenerator
|
||||
{
|
||||
public:
|
||||
TextureCoordGenerator(dp::TextureManager::StippleRegion const & region, float const baseGtoPScale)
|
||||
: m_region(region)
|
||||
, m_baseGtoPScale(baseGtoPScale)
|
||||
, m_maskLength(static_cast<float>(m_region.GetMaskPixelLength()))
|
||||
{
|
||||
public:
|
||||
TextureCoordGenerator(float const baseGtoPScale)
|
||||
: m_baseGtoPScale(baseGtoPScale)
|
||||
}
|
||||
|
||||
float GetUvOffsetByDistance(float distance) const
|
||||
{
|
||||
return (distance * m_baseGtoPScale / m_maskLength);
|
||||
}
|
||||
|
||||
glsl::vec2 GetTexCoordsByDistance(float distance) const
|
||||
{
|
||||
float normalizedOffset = min(GetUvOffsetByDistance(distance), 1.0f);
|
||||
return GetTexCoords(normalizedOffset);
|
||||
}
|
||||
|
||||
glsl::vec2 GetTexCoords(float normalizedOffset) const
|
||||
{
|
||||
m2::RectF const & texRect = m_region.GetTexRect();
|
||||
return glsl::vec2(texRect.minX() + normalizedOffset * texRect.SizeX(), texRect.Center().y);
|
||||
}
|
||||
|
||||
float GetMaskLength() const { return m_maskLength; }
|
||||
|
||||
dp::TextureManager::StippleRegion const & GetRegion()
|
||||
{
|
||||
return m_region;
|
||||
}
|
||||
|
||||
private:
|
||||
dp::TextureManager::StippleRegion m_region;
|
||||
float const m_baseGtoPScale;
|
||||
float m_maskLength = 0.0f;
|
||||
};
|
||||
|
||||
template <typename TVertex>
|
||||
class BaseLineBuilder
|
||||
{
|
||||
public:
|
||||
BaseLineBuilder(dp::TextureManager::ColorRegion const & color, float pxHalfWidth)
|
||||
: m_color(color)
|
||||
, m_colorCoord(glsl::ToVec2(m_color.GetTexRect().Center()))
|
||||
, m_pxHalfWidth(pxHalfWidth)
|
||||
{
|
||||
}
|
||||
|
||||
void GetTexturingInfo(float const globalLength, int & steps, float & maskSize)
|
||||
{
|
||||
UNUSED_VALUE(globalLength);
|
||||
UNUSED_VALUE(steps);
|
||||
UNUSED_VALUE(maskSize);
|
||||
}
|
||||
|
||||
dp::BindingInfo const & GetBindingInfo()
|
||||
{
|
||||
return TVertex::GetBindingInfo();
|
||||
}
|
||||
|
||||
ref_ptr<void> GetLineData()
|
||||
{
|
||||
return make_ref(m_geometry.data());
|
||||
}
|
||||
|
||||
size_t GetLineSize()
|
||||
{
|
||||
return m_geometry.size();
|
||||
}
|
||||
|
||||
ref_ptr<void> GetJoinData()
|
||||
{
|
||||
return make_ref(m_joinGeom.data());
|
||||
}
|
||||
|
||||
size_t GetJoinSize()
|
||||
{
|
||||
return m_joinGeom.size();
|
||||
}
|
||||
|
||||
float GetHalfWidth()
|
||||
{
|
||||
return m_pxHalfWidth;
|
||||
}
|
||||
|
||||
protected:
|
||||
using V = TVertex;
|
||||
using TGeometryBuffer = buffer_vector<V, 256>;
|
||||
|
||||
TGeometryBuffer m_geometry;
|
||||
TGeometryBuffer m_joinGeom;
|
||||
|
||||
dp::TextureManager::ColorRegion m_color;
|
||||
glsl::vec2 const m_colorCoord;
|
||||
float const m_pxHalfWidth;
|
||||
};
|
||||
|
||||
class SolidLineBuilder : public BaseLineBuilder<gpu::LineVertex>
|
||||
{
|
||||
using TBase = BaseLineBuilder<gpu::LineVertex>;
|
||||
|
||||
public:
|
||||
SolidLineBuilder(dp::TextureManager::ColorRegion const & color, float const pxHalfWidth)
|
||||
: TBase(color, pxHalfWidth)
|
||||
{
|
||||
}
|
||||
|
||||
dp::GLState GetState()
|
||||
{
|
||||
dp::GLState state(gpu::LINE_PROGRAM, dp::GLState::GeometryLayer);
|
||||
state.SetColorTexture(m_color.GetTexture());
|
||||
return state;
|
||||
}
|
||||
|
||||
void SubmitVertex(LineSegment const & segment, glsl::vec3 const & pivot,
|
||||
glsl::vec2 const & normal, float offsetFromStart)
|
||||
{
|
||||
UNUSED_VALUE(segment);
|
||||
UNUSED_VALUE(offsetFromStart);
|
||||
|
||||
m_geometry.emplace_back(V(pivot, m_pxHalfWidth * normal, m_colorCoord));
|
||||
}
|
||||
|
||||
void SubmitJoin(glsl::vec3 const & pivot, vector<glsl::vec2> const & normals)
|
||||
{
|
||||
size_t const trianglesCount = normals.size() / 3;
|
||||
for (int j = 0; j < trianglesCount; j++)
|
||||
{
|
||||
size_t baseIndex = 3 * j;
|
||||
m_joinGeom.push_back(V(pivot, normals[baseIndex + 0], m_colorCoord));
|
||||
m_joinGeom.push_back(V(pivot, normals[baseIndex + 1], m_colorCoord));
|
||||
m_joinGeom.push_back(V(pivot, normals[baseIndex + 2], m_colorCoord));
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
void SetRegion(dp::TextureManager::StippleRegion const & region, bool isSolid)
|
||||
class DashedLineBuilder : public BaseLineBuilder<gpu::DashedLineVertex>
|
||||
{
|
||||
using TBase = BaseLineBuilder<gpu::DashedLineVertex>;
|
||||
|
||||
public:
|
||||
DashedLineBuilder(dp::TextureManager::ColorRegion const & color,
|
||||
dp::TextureManager::StippleRegion const & stipple,
|
||||
float glbHalfWidth, float pxHalfWidth, float baseGtoP)
|
||||
: TBase(color, pxHalfWidth)
|
||||
, m_texCoordGen(stipple, baseGtoP)
|
||||
, m_glbHalfWidth(glbHalfWidth)
|
||||
, m_baseGtoPScale(baseGtoP)
|
||||
{
|
||||
}
|
||||
|
||||
void GetTexturingInfo(float const globalLength, int & steps, float & maskSize)
|
||||
{
|
||||
float const pixelLen = globalLength * m_baseGtoPScale;
|
||||
steps = static_cast<int>((pixelLen + m_texCoordGen.GetMaskLength() - 1) / m_texCoordGen.GetMaskLength());
|
||||
maskSize = globalLength / steps;
|
||||
}
|
||||
|
||||
dp::GLState GetState()
|
||||
{
|
||||
dp::GLState state(gpu::DASHED_LINE_PROGRAM, dp::GLState::GeometryLayer);
|
||||
state.SetColorTexture(m_color.GetTexture());
|
||||
state.SetMaskTexture(m_texCoordGen.GetRegion().GetTexture());
|
||||
return state;
|
||||
}
|
||||
|
||||
void SubmitVertex(LineSegment const & segment, glsl::vec3 const & pivot,
|
||||
glsl::vec2 const & normal, float offsetFromStart)
|
||||
{
|
||||
float distance = GetProjectionLength(pivot.xy() + m_glbHalfWidth * normal,
|
||||
segment.m_points[StartPoint],
|
||||
segment.m_points[EndPoint]) - offsetFromStart;
|
||||
|
||||
m_geometry.emplace_back(V(pivot, m_pxHalfWidth * normal, m_colorCoord, m_texCoordGen.GetTexCoordsByDistance(distance)));
|
||||
}
|
||||
|
||||
void SubmitJoin(glsl::vec3 const & pivot, vector<glsl::vec2> const & normals)
|
||||
{
|
||||
size_t const trianglesCount = normals.size() / 3;
|
||||
for (int j = 0; j < trianglesCount; j++)
|
||||
{
|
||||
m_isSolid = isSolid;
|
||||
m_region = region;
|
||||
if (!m_isSolid)
|
||||
m_maskLength = static_cast<float>(m_region.GetMaskPixelLength());
|
||||
size_t baseIndex = 3 * j;
|
||||
glsl::vec2 texCoord = m_texCoordGen.GetTexCoords(0.0f);
|
||||
m_joinGeom.push_back(V(pivot, normals[baseIndex + 0], m_colorCoord, texCoord));
|
||||
m_joinGeom.push_back(V(pivot, normals[baseIndex + 1], m_colorCoord, texCoord));
|
||||
m_joinGeom.push_back(V(pivot, normals[baseIndex + 2], m_colorCoord, texCoord));
|
||||
}
|
||||
}
|
||||
|
||||
float GetUvOffsetByDistance(float distance) const
|
||||
{
|
||||
return m_isSolid ? 1.0 : (distance * m_baseGtoPScale / m_maskLength);
|
||||
}
|
||||
private:
|
||||
TextureCoordGenerator m_texCoordGen;
|
||||
float const m_glbHalfWidth;
|
||||
float const m_baseGtoPScale;
|
||||
};
|
||||
|
||||
glsl::vec2 GetTexCoordsByDistance(float distance) const
|
||||
{
|
||||
float normalizedOffset = min(GetUvOffsetByDistance(distance), 1.0f);
|
||||
return GetTexCoords(normalizedOffset);
|
||||
}
|
||||
|
||||
glsl::vec2 GetTexCoords(float normalizedOffset) const
|
||||
{
|
||||
m2::RectF const & texRect = m_region.GetTexRect();
|
||||
if (m_isSolid)
|
||||
return glsl::ToVec2(texRect.Center());
|
||||
|
||||
return glsl::vec2(texRect.minX() + normalizedOffset * texRect.SizeX(), texRect.Center().y);
|
||||
}
|
||||
|
||||
float GetMaskLength() const { return m_maskLength; }
|
||||
bool IsSolid() const { return m_isSolid; }
|
||||
|
||||
private:
|
||||
float const m_baseGtoPScale;
|
||||
dp::TextureManager::StippleRegion m_region;
|
||||
float m_maskLength = 0.0f;
|
||||
bool m_isSolid = true;
|
||||
};
|
||||
}
|
||||
} // namespace
|
||||
|
||||
LineShape::LineShape(m2::SharedSpline const & spline,
|
||||
LineViewParams const & params)
|
||||
|
@ -76,58 +227,19 @@ LineShape::LineShape(m2::SharedSpline const & spline,
|
|||
ASSERT_GREATER(m_spline->GetPath().size(), 1, ());
|
||||
}
|
||||
|
||||
void LineShape::Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManager> textures) const
|
||||
template <typename TBuilder>
|
||||
void LineShape::Draw(TBuilder & builder, ref_ptr<dp::Batcher> batcher) const
|
||||
{
|
||||
TGeometryBuffer geometry;
|
||||
TGeometryBuffer joinsGeometry;
|
||||
vector<m2::PointD> const & path = m_spline->GetPath();
|
||||
ASSERT(path.size() > 1, ());
|
||||
|
||||
// set up uv-generator
|
||||
dp::TextureManager::ColorRegion colorRegion;
|
||||
textures->GetColorRegion(m_params.m_color, colorRegion);
|
||||
glsl::vec2 colorCoord(glsl::ToVec2(colorRegion.GetTexRect().Center()));
|
||||
|
||||
TextureCoordGenerator texCoordGen(m_params.m_baseGtoPScale);
|
||||
dp::TextureManager::StippleRegion maskRegion;
|
||||
if (m_params.m_pattern.empty())
|
||||
textures->GetStippleRegion(dp::TextureManager::TStipplePattern{1}, maskRegion);
|
||||
else
|
||||
textures->GetStippleRegion(m_params.m_pattern, maskRegion);
|
||||
|
||||
texCoordGen.SetRegion(maskRegion, m_params.m_pattern.empty());
|
||||
|
||||
float const halfWidth = m_params.m_width / 2.0f;
|
||||
float const glbHalfWidth = halfWidth / m_params.m_baseGtoPScale;
|
||||
|
||||
// skip joins generation
|
||||
float const halfWidth = builder.GetHalfWidth();
|
||||
float const joinsGenerationThreshold = 2.5f;
|
||||
bool generateJoins = true;
|
||||
if (halfWidth <= joinsGenerationThreshold)
|
||||
generateJoins = false;
|
||||
|
||||
auto const getLineVertex = [&](LineSegment const & segment, glsl::vec3 const & pivot,
|
||||
glsl::vec2 const & normal, float offsetFromStart)
|
||||
{
|
||||
float distance = GetProjectionLength(pivot.xy() + glbHalfWidth * normal,
|
||||
segment.m_points[StartPoint],
|
||||
segment.m_points[EndPoint]) - offsetFromStart;
|
||||
|
||||
return LV(pivot, halfWidth * normal, colorCoord, texCoordGen.GetTexCoordsByDistance(distance));
|
||||
};
|
||||
|
||||
auto const generateTriangles = [&](glsl::vec3 const & pivot, vector<glsl::vec2> const & normals)
|
||||
{
|
||||
size_t const trianglesCount = normals.size() / 3;
|
||||
for (int j = 0; j < trianglesCount; j++)
|
||||
{
|
||||
glsl::vec2 const uv = texCoordGen.GetTexCoords(0.0f);
|
||||
joinsGeometry.push_back(LV(pivot, normals[3 * j], colorCoord, uv));
|
||||
joinsGeometry.push_back(LV(pivot, normals[3 * j + 1], colorCoord, uv));
|
||||
joinsGeometry.push_back(LV(pivot, normals[3 * j + 2], colorCoord, uv));
|
||||
}
|
||||
};
|
||||
|
||||
// constuct segments
|
||||
vector<LineSegment> segments;
|
||||
segments.reserve(path.size() - 1);
|
||||
|
@ -146,12 +258,7 @@ void LineShape::Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManager> t
|
|||
float const initialGlobalLength = glsl::length(segments[i].m_points[EndPoint] - segments[i].m_points[StartPoint]);
|
||||
int steps = 1;
|
||||
float maskSize = initialGlobalLength;
|
||||
if (!texCoordGen.IsSolid())
|
||||
{
|
||||
float const pixelLen = initialGlobalLength * m_params.m_baseGtoPScale;
|
||||
steps = static_cast<int>((pixelLen + texCoordGen.GetMaskLength() - 1) / texCoordGen.GetMaskLength());
|
||||
maskSize = initialGlobalLength / steps;
|
||||
}
|
||||
builder.GetTexturingInfo(initialGlobalLength, steps, maskSize);
|
||||
|
||||
// generate main geometry
|
||||
float currentSize = 0;
|
||||
|
@ -166,10 +273,10 @@ void LineShape::Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManager> t
|
|||
glsl::vec2 const leftNormal = GetNormal(segments[i], true /* isLeft */, BaseNormal);
|
||||
glsl::vec2 const rightNormal = GetNormal(segments[i], false /* isLeft */, BaseNormal);
|
||||
|
||||
geometry.push_back(getLineVertex(segments[i], currentStartPivot, rightNormal, offsetFromStart));
|
||||
geometry.push_back(getLineVertex(segments[i], currentStartPivot, leftNormal, offsetFromStart));
|
||||
geometry.push_back(getLineVertex(segments[i], newPivot, rightNormal, offsetFromStart));
|
||||
geometry.push_back(getLineVertex(segments[i], newPivot, leftNormal, offsetFromStart));
|
||||
builder.SubmitVertex(segments[i], currentStartPivot, rightNormal, offsetFromStart);
|
||||
builder.SubmitVertex(segments[i], currentStartPivot, leftNormal, offsetFromStart);
|
||||
builder.SubmitVertex(segments[i], newPivot, rightNormal, offsetFromStart);
|
||||
builder.SubmitVertex(segments[i], newPivot, leftNormal, offsetFromStart);
|
||||
|
||||
currentStartPivot = newPivot;
|
||||
}
|
||||
|
@ -190,48 +297,74 @@ void LineShape::Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManager> t
|
|||
GenerateJoinNormals(m_params.m_join, n1, n2, halfWidth, segments[i].m_hasLeftJoin[EndPoint],
|
||||
widthScalar, normals);
|
||||
|
||||
generateTriangles(glsl::vec3(segments[i].m_points[EndPoint], m_params.m_depth), normals);
|
||||
}
|
||||
|
||||
// generate caps
|
||||
if (i == 0)
|
||||
{
|
||||
vector<glsl::vec2> normals;
|
||||
normals.reserve(24);
|
||||
GenerateCapNormals(m_params.m_cap, segments[i].m_leftNormals[StartPoint],
|
||||
segments[i].m_rightNormals[StartPoint], -segments[i].m_tangent,
|
||||
halfWidth, true /* isStart */, normals);
|
||||
|
||||
generateTriangles(glsl::vec3(segments[i].m_points[StartPoint], m_params.m_depth), normals);
|
||||
}
|
||||
|
||||
if (i == segments.size() - 1)
|
||||
{
|
||||
vector<glsl::vec2> normals;
|
||||
normals.reserve(24);
|
||||
GenerateCapNormals(m_params.m_cap, segments[i].m_leftNormals[EndPoint],
|
||||
segments[i].m_rightNormals[EndPoint], segments[i].m_tangent,
|
||||
halfWidth, false /* isStart */, normals);
|
||||
|
||||
generateTriangles(glsl::vec3(segments[i].m_points[EndPoint], m_params.m_depth), normals);
|
||||
builder.SubmitJoin(glsl::vec3(segments[i].m_points[EndPoint], m_params.m_depth), normals);
|
||||
}
|
||||
}
|
||||
|
||||
dp::GLState state(gpu::LINE_PROGRAM, dp::GLState::GeometryLayer);
|
||||
state.SetColorTexture(colorRegion.GetTexture());
|
||||
state.SetMaskTexture(maskRegion.GetTexture());
|
||||
vector<glsl::vec2> normals;
|
||||
normals.reserve(24);
|
||||
|
||||
dp::AttributeProvider provider(1, geometry.size());
|
||||
provider.InitStream(0, gpu::LineVertex::GetBindingInfo(), make_ref(geometry.data()));
|
||||
batcher->InsertListOfStrip(state, make_ref(&provider), 4);
|
||||
|
||||
if (!joinsGeometry.empty())
|
||||
{
|
||||
dp::AttributeProvider joinsProvider(1, joinsGeometry.size());
|
||||
joinsProvider.InitStream(0, gpu::LineVertex::GetBindingInfo(), make_ref(joinsGeometry.data()));
|
||||
LineSegment const & segment = segments[0];
|
||||
GenerateCapNormals(m_params.m_cap,
|
||||
segment.m_leftNormals[StartPoint],
|
||||
segment.m_rightNormals[StartPoint],
|
||||
-segment.m_tangent,
|
||||
halfWidth, true /* isStart */, normals);
|
||||
|
||||
builder.SubmitJoin(glsl::vec3(segment.m_points[StartPoint], m_params.m_depth), normals);
|
||||
normals.clear();
|
||||
}
|
||||
|
||||
{
|
||||
size_t index = segments.size() - 1;
|
||||
LineSegment const & segment = segments[index];
|
||||
GenerateCapNormals(m_params.m_cap,
|
||||
segment.m_leftNormals[EndPoint],
|
||||
segment.m_rightNormals[EndPoint],
|
||||
segment.m_tangent,
|
||||
halfWidth, false /* isStart */, normals);
|
||||
|
||||
builder.SubmitJoin(glsl::vec3(segment.m_points[EndPoint], m_params.m_depth), normals);
|
||||
}
|
||||
|
||||
dp::GLState state = builder.GetState();
|
||||
|
||||
dp::AttributeProvider provider(1, builder.GetLineSize());
|
||||
provider.InitStream(0, builder.GetBindingInfo(), builder.GetLineData());
|
||||
batcher->InsertListOfStrip(state, make_ref(&provider), dp::Batcher::VertexPerQuad);
|
||||
|
||||
size_t joinSize = builder.GetJoinSize();
|
||||
if (joinSize > 0)
|
||||
{
|
||||
dp::AttributeProvider joinsProvider(1, joinSize);
|
||||
joinsProvider.InitStream(0, builder.GetBindingInfo(), builder.GetJoinData());
|
||||
batcher->InsertTriangleList(state, make_ref(&joinsProvider));
|
||||
}
|
||||
}
|
||||
|
||||
void LineShape::Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManager> textures) const
|
||||
{
|
||||
dp::TextureManager::ColorRegion colorRegion;
|
||||
textures->GetColorRegion(m_params.m_color, colorRegion);
|
||||
float const pxHalfWidth = m_params.m_width / 2.0f;
|
||||
|
||||
if (m_params.m_pattern.empty())
|
||||
{
|
||||
SolidLineBuilder builder(colorRegion, pxHalfWidth);
|
||||
Draw(builder, batcher);
|
||||
}
|
||||
else
|
||||
{
|
||||
dp::TextureManager::StippleRegion maskRegion;
|
||||
textures->GetStippleRegion(m_params.m_pattern, maskRegion);
|
||||
|
||||
float const glbHalfWidth = pxHalfWidth / m_params.m_baseGtoPScale;
|
||||
|
||||
DashedLineBuilder builder(colorRegion, maskRegion, glbHalfWidth, pxHalfWidth, m_params.m_baseGtoPScale);
|
||||
Draw(builder, batcher);
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace df
|
||||
|
||||
|
|
|
@ -16,6 +16,10 @@ public:
|
|||
|
||||
virtual void Draw(ref_ptr<dp::Batcher> batcher, ref_ptr<dp::TextureManager> textures) const;
|
||||
|
||||
private:
|
||||
template <typename TBuilder>
|
||||
void Draw(TBuilder & builder, ref_ptr<dp::Batcher> batcher) const;
|
||||
|
||||
private:
|
||||
LineViewParams m_params;
|
||||
m2::SharedSpline m_spline;
|
||||
|
|
Loading…
Add table
Reference in a new issue