diff --git a/package.json b/package.json index c7d82f1e15f..62c1a7940a8 100644 --- a/package.json +++ b/package.json @@ -18,7 +18,7 @@ }, "devDependencies": { "aws-sdk": "^2.3.5", - "mapbox-gl-test-suite": "mapbox/mapbox-gl-test-suite#1e0d988e187a6c9a7a74fa8d7feb9c66c2258123", + "mapbox-gl-test-suite": "mapbox/mapbox-gl-test-suite#d974ec6b3748a258f8ddd7528e049493390177b4", "node-gyp": "^3.3.1", "request": "^2.72.0", "tape": "^4.5.1" diff --git a/src/mbgl/renderer/fill_bucket.cpp b/src/mbgl/renderer/fill_bucket.cpp index 1388c6dfb7d..48d9ae88aac 100644 --- a/src/mbgl/renderer/fill_bucket.cpp +++ b/src/mbgl/renderer/fill_bucket.cpp @@ -6,6 +6,7 @@ #include #include #include +#include #include #include @@ -245,3 +246,15 @@ void FillBucket::drawVertices(OutlineShader& shader, gl::GLObjectStore& glObject elements_index += group->elements_length * lineElementsBuffer.itemSize; } } + +void FillBucket::drawVertices(OutlinePatternShader& shader, gl::GLObjectStore& glObjectStore) { + GLbyte* vertex_index = BUFFER_OFFSET(0); + GLbyte* elements_index = BUFFER_OFFSET(0); + for (auto& group : lineGroups) { + assert(group); + group->array[1].bind(shader, vertexBuffer, lineElementsBuffer, vertex_index, glObjectStore); + MBGL_CHECK_ERROR(glDrawElements(GL_LINES, group->elements_length * 2, GL_UNSIGNED_SHORT, elements_index)); + vertex_index += group->vertex_length * vertexBuffer.itemSize; + elements_index += group->elements_length * lineElementsBuffer.itemSize; + } +} diff --git a/src/mbgl/renderer/fill_bucket.hpp b/src/mbgl/renderer/fill_bucket.hpp index 61a3b986ba8..bd6a9a9cf47 100644 --- a/src/mbgl/renderer/fill_bucket.hpp +++ b/src/mbgl/renderer/fill_bucket.hpp @@ -16,6 +16,7 @@ namespace mbgl { class FillVertexBuffer; class OutlineShader; +class OutlinePatternShader; class PlainShader; class PatternShader; @@ -26,7 +27,7 @@ class FillBucket : public Bucket { static void free(void *userData, void *ptr); typedef ElementGroup<2> TriangleGroup; - typedef ElementGroup<1> LineGroup; + typedef ElementGroup<2> LineGroup; public: FillBucket(); @@ -43,6 +44,7 @@ class FillBucket : public Bucket { void drawElements(PlainShader&, gl::GLObjectStore&); void drawElements(PatternShader&, gl::GLObjectStore&); void drawVertices(OutlineShader&, gl::GLObjectStore&); + void drawVertices(OutlinePatternShader&, gl::GLObjectStore&); private: TESSalloc *allocator; diff --git a/src/mbgl/renderer/painter.cpp b/src/mbgl/renderer/painter.cpp index 552859343b6..1fbce82c038 100644 --- a/src/mbgl/renderer/painter.cpp +++ b/src/mbgl/renderer/painter.cpp @@ -20,6 +20,7 @@ #include #include #include +#include #include #include #include @@ -49,6 +50,7 @@ Painter::Painter(const TransformState& state_, gl::GLObjectStore& glObjectStore_ plainShader = std::make_unique(glObjectStore); outlineShader = std::make_unique(glObjectStore); + outlinePatternShader = std::make_unique(glObjectStore); lineShader = std::make_unique(glObjectStore); linesdfShader = std::make_unique(glObjectStore); linepatternShader = std::make_unique(glObjectStore); diff --git a/src/mbgl/renderer/painter.hpp b/src/mbgl/renderer/painter.hpp index 5dc87f9c9f9..2fa5c026045 100644 --- a/src/mbgl/renderer/painter.hpp +++ b/src/mbgl/renderer/painter.hpp @@ -53,6 +53,7 @@ class BackgroundLayer; class SDFShader; class PlainShader; class OutlineShader; +class OutlinePatternShader; class LineShader; class LinejoinShader; class LineSDFShader; @@ -190,6 +191,7 @@ class Painter : private util::noncopyable { std::unique_ptr plainShader; std::unique_ptr outlineShader; + std::unique_ptr outlinePatternShader; std::unique_ptr lineShader; std::unique_ptr linesdfShader; std::unique_ptr linepatternShader; diff --git a/src/mbgl/renderer/painter_fill.cpp b/src/mbgl/renderer/painter_fill.cpp index 0452b532ac2..0808cd2ac11 100644 --- a/src/mbgl/renderer/painter_fill.cpp +++ b/src/mbgl/renderer/painter_fill.cpp @@ -4,6 +4,7 @@ #include #include #include +#include #include #include @@ -108,6 +109,44 @@ void Painter::renderFill(FillBucket& bucket, const FillLayer& layer, const TileI // Draw the actual triangles into the color & stencil buffer. setDepthSublayer(0); bucket.drawElements(*patternShader, glObjectStore); + + if (properties.fillAntialias && stroke_color == fill_color) { + config.program = outlinePatternShader->getID(); + outlinePatternShader->u_matrix = vtxMatrix; + config.lineWidth = 2.0f; + + // Draw the entire line + outlinePatternShader->u_world = {{ + static_cast(frame.framebufferSize[0]), + static_cast(frame.framebufferSize[1]) + }}; + + outlinePatternShader->u_pattern_tl_a = (*posA).tl; + outlinePatternShader->u_pattern_br_a = (*posA).br; + outlinePatternShader->u_pattern_tl_b = (*posB).tl; + outlinePatternShader->u_pattern_br_b = (*posB).br; + outlinePatternShader->u_opacity = properties.fillOpacity; + outlinePatternShader->u_image = 0; + outlinePatternShader->u_mix = properties.fillPattern.value.t; + + outlinePatternShader->u_patternscale_a = {{ + 1.0f / id.pixelsToTileUnits(imageSizeScaledA[0], state.getIntegerZoom()), + 1.0f / id.pixelsToTileUnits(imageSizeScaledB[1], state.getIntegerZoom()) + }}; + outlinePatternShader->u_patternscale_b = {{ + 1.0f / id.pixelsToTileUnits(imageSizeScaledB[0], state.getIntegerZoom()), + 1.0f / id.pixelsToTileUnits(imageSizeScaledB[1], state.getIntegerZoom()) + }}; + + outlinePatternShader->u_offset_a = std::array{{offsetAx, offsetAy}}; + outlinePatternShader->u_offset_b = std::array{{offsetBx, offsetBy}}; + + config.activeTexture = GL_TEXTURE0; + spriteAtlas->bind(true, glObjectStore); + + setDepthSublayer(2); + bucket.drawVertices(*outlinePatternShader, glObjectStore); + } } } else { @@ -135,7 +174,7 @@ void Painter::renderFill(FillBucket& bucket, const FillLayer& layer, const TileI config.lineWidth = 2.0f; // This is always fixed and does not depend on the pixelRatio! outlineShader->u_color = fill_color; - + // Draw the entire line outlineShader->u_world = {{ static_cast(frame.framebufferSize[0]), diff --git a/src/mbgl/shader/outlinepattern.fragment.glsl b/src/mbgl/shader/outlinepattern.fragment.glsl new file mode 100644 index 00000000000..dd1efd63e67 --- /dev/null +++ b/src/mbgl/shader/outlinepattern.fragment.glsl @@ -0,0 +1,30 @@ +uniform float u_opacity; +uniform vec2 u_pattern_tl_a; +uniform vec2 u_pattern_br_a; +uniform vec2 u_pattern_tl_b; +uniform vec2 u_pattern_br_b; +uniform float u_mix; + +uniform sampler2D u_image; + +varying vec2 v_pos_a; +varying vec2 v_pos_b; +varying vec2 v_pos; + +void main() { + vec2 imagecoord = mod(v_pos_a, 1.0); + vec2 pos = mix(u_pattern_tl_a, u_pattern_br_a, imagecoord); + vec4 color1 = texture2D(u_image, pos); + + vec2 imagecoord_b = mod(v_pos_b, 1.0); + vec2 pos2 = mix(u_pattern_tl_b, u_pattern_br_b, imagecoord_b); + vec4 color2 = texture2D(u_image, pos2); + + // find distance to outline for alpha interpolation + + float dist = length(v_pos - gl_FragCoord.xy); + float alpha = smoothstep(1.0, 0.0, dist); + + + gl_FragColor = mix(color1, color2, u_mix) * alpha * u_opacity; +} \ No newline at end of file diff --git a/src/mbgl/shader/outlinepattern.vertex.glsl b/src/mbgl/shader/outlinepattern.vertex.glsl new file mode 100644 index 00000000000..e31b77b560f --- /dev/null +++ b/src/mbgl/shader/outlinepattern.vertex.glsl @@ -0,0 +1,21 @@ +uniform vec2 u_patternscale_a; +uniform vec2 u_patternscale_b; +uniform vec2 u_offset_a; +uniform vec2 u_offset_b; + +attribute vec2 a_pos; + +uniform mat4 u_matrix; +uniform vec2 u_world; + +varying vec2 v_pos_a; +varying vec2 v_pos_b; +varying vec2 v_pos; + + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); + v_pos_a = u_patternscale_a * a_pos + u_offset_a; + v_pos_b = u_patternscale_b * a_pos + u_offset_b; + v_pos = (gl_Position.xy/gl_Position.w + 1.0) / 2.0 * u_world; +} diff --git a/src/mbgl/shader/outlinepattern_shader.cpp b/src/mbgl/shader/outlinepattern_shader.cpp new file mode 100644 index 00000000000..44b76b74c0b --- /dev/null +++ b/src/mbgl/shader/outlinepattern_shader.cpp @@ -0,0 +1,21 @@ +#include +#include +#include +#include + +#include + +using namespace mbgl; + +OutlinePatternShader::OutlinePatternShader(gl::GLObjectStore& glObjectStore) + : Shader( + "outlinepattern", + shaders::outlinepattern::vertex, shaders::outlinepattern::fragment, + glObjectStore + ) { +} + +void OutlinePatternShader::bind(GLbyte *offset) { + MBGL_CHECK_ERROR(glEnableVertexAttribArray(a_pos)); + MBGL_CHECK_ERROR(glVertexAttribPointer(a_pos, 2, GL_SHORT, false, 0, offset)); +} diff --git a/src/mbgl/shader/outlinepattern_shader.hpp b/src/mbgl/shader/outlinepattern_shader.hpp new file mode 100644 index 00000000000..eed3839fd37 --- /dev/null +++ b/src/mbgl/shader/outlinepattern_shader.hpp @@ -0,0 +1,32 @@ +#ifndef MBGL_SHADER_SHADER_OUTLINEPATTERN +#define MBGL_SHADER_SHADER_OUTLINEPATTERN + +#include +#include + +namespace mbgl { + +class OutlinePatternShader : public Shader { +public: + OutlinePatternShader(gl::GLObjectStore&); + + void bind(GLbyte *offset) final; + + UniformMatrix<4> u_matrix = {"u_matrix", *this}; + Uniform> u_pattern_tl_a = {"u_pattern_tl_a", *this}; + Uniform> u_pattern_br_a = {"u_pattern_br_a", *this}; + Uniform> u_pattern_tl_b = {"u_pattern_tl_b", *this}; + Uniform> u_pattern_br_b = {"u_pattern_br_b", *this}; + Uniform u_opacity = {"u_opacity", *this}; + Uniform u_mix = {"u_mix", *this}; + Uniform u_image = {"u_image", *this}; + Uniform> u_patternscale_a = {"u_patternscale_a", *this}; + Uniform> u_patternscale_b = {"u_patternscale_b", *this}; + Uniform> u_offset_a = {"u_offset_a", *this}; + Uniform> u_offset_b = {"u_offset_b", *this}; + Uniform> u_world = {"u_world", *this}; +}; + +} // namespace mbgl + +#endif \ No newline at end of file diff --git a/test/fixtures/map/offline/expected.png b/test/fixtures/map/offline/expected.png index 1b1c2be4c6c..c7bfb79fbfb 100644 Binary files a/test/fixtures/map/offline/expected.png and b/test/fixtures/map/offline/expected.png differ diff --git a/test/fixtures/map/offline/style.json b/test/fixtures/map/offline/style.json index 978df3aae36..e92dc2fb0ad 100644 --- a/test/fixtures/map/offline/style.json +++ b/test/fixtures/map/offline/style.json @@ -21,7 +21,8 @@ "source": "mapbox", "source-layer": "water", "paint": { - "fill-pattern": "noise" + "fill-pattern": "noise", + "fill-antialias": false } }, { "id": "admin",