summaryrefslogtreecommitdiff
path: root/src/mbgl/renderer/symbol_bucket.cpp
blob: 3ef0686e4d0f699dff233bdaecbcf55c4e2d2318 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#include <mbgl/renderer/symbol_bucket.hpp>
#include <mbgl/renderer/painter.hpp>
#include <mbgl/style/layers/symbol_layer.hpp>
#include <mbgl/shader/sdf_shader.hpp>
#include <mbgl/shader/icon_shader.hpp>
#include <mbgl/shader/collision_box_shader.hpp>
#include <mbgl/gl/gl.hpp>

namespace mbgl {

using namespace style;

SymbolBucket::SymbolBucket(const MapMode mode_,
                           style::SymbolLayoutProperties layout_,
                           bool sdfIcons_,
                           bool iconsNeedLinear_)
    : mode(mode_),
      layout(std::move(layout_)),
      sdfIcons(sdfIcons_),
      iconsNeedLinear(iconsNeedLinear_) {
}

void SymbolBucket::upload(gl::Context& context) {
    if (hasTextData()) {
        text.vertices.upload(context);
        text.triangles.upload(context);
    }
    if (hasIconData()) {
        icon.vertices.upload(context);
        icon.triangles.upload(context);
    }

    uploaded = true;
}

void SymbolBucket::render(Painter& painter,
                          PaintParameters& parameters,
                          const Layer& layer,
                          const RenderTile& tile) {
    painter.renderSymbol(parameters, *this, *layer.as<SymbolLayer>(), tile);
}

bool SymbolBucket::hasData() const {
    assert(false); // Should be calling SymbolLayout::hasSymbolInstances() instead.
    return false;
}

bool SymbolBucket::hasTextData() const {
    return !text.groups.empty();
}

bool SymbolBucket::hasIconData() const {
    return !icon.groups.empty();
}

bool SymbolBucket::hasCollisionBoxData() const {
    return !collisionBox.groups.empty();
}

bool SymbolBucket::needsClipping() const {
    return mode == MapMode::Still;
}

void SymbolBucket::drawGlyphs(SDFShader& shader,
                              gl::Context& context,
                              PaintMode paintMode) {
    GLbyte* vertex_index = BUFFER_OFFSET_0;
    GLbyte* elements_index = BUFFER_OFFSET_0;
    for (auto& group : text.groups) {
        assert(group);
        group->array[paintMode == PaintMode::Overdraw ? 1 : 0].bind(
            shader, text.vertices, text.triangles, vertex_index, context);
        MBGL_CHECK_ERROR(glDrawElements(GL_TRIANGLES, group->elements_length * 3, GL_UNSIGNED_SHORT,
                                        elements_index));
        vertex_index += group->vertex_length * text.vertices.itemSize;
        elements_index += group->elements_length * text.triangles.itemSize;
    }
}

void SymbolBucket::drawIcons(SDFShader& shader,
                             gl::Context& context,
                             PaintMode paintMode) {
    GLbyte* vertex_index = BUFFER_OFFSET_0;
    GLbyte* elements_index = BUFFER_OFFSET_0;
    for (auto& group : icon.groups) {
        assert(group);
        group->array[paintMode == PaintMode::Overdraw ? 1 : 0].bind(
            shader, icon.vertices, icon.triangles, vertex_index, context);
        MBGL_CHECK_ERROR(glDrawElements(GL_TRIANGLES, group->elements_length * 3, GL_UNSIGNED_SHORT,
                                        elements_index));
        vertex_index += group->vertex_length * icon.vertices.itemSize;
        elements_index += group->elements_length * icon.triangles.itemSize;
    }
}

void SymbolBucket::drawIcons(IconShader& shader,
                             gl::Context& context,
                             PaintMode paintMode) {
    GLbyte* vertex_index = BUFFER_OFFSET_0;
    GLbyte* elements_index = BUFFER_OFFSET_0;
    for (auto& group : icon.groups) {
        assert(group);
        group->array[paintMode == PaintMode::Overdraw ? 3 : 2].bind(
            shader, icon.vertices, icon.triangles, vertex_index, context);
        MBGL_CHECK_ERROR(glDrawElements(GL_TRIANGLES, group->elements_length * 3, GL_UNSIGNED_SHORT,
                                        elements_index));
        vertex_index += group->vertex_length * icon.vertices.itemSize;
        elements_index += group->elements_length * icon.triangles.itemSize;
    }
}

void SymbolBucket::drawCollisionBoxes(CollisionBoxShader& shader,
                                      gl::Context& context) {
    GLbyte* vertex_index = BUFFER_OFFSET_0;
    for (auto& group : collisionBox.groups) {
        group->array[0].bind(shader, collisionBox.vertices, vertex_index, context);
        MBGL_CHECK_ERROR(glDrawArrays(GL_LINES, 0, group->vertex_length));
    }
}

} // namespace mbgl