summaryrefslogtreecommitdiffhomepage
path: root/draw/floor.cpp
blob: 861336c72285238ec71bd155e4ebb983565246bc (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
#include "floor.hpp"
#include "shaders/tile.hpp"
#include "tile.hpp"
#include "chunk.hpp"
#include "tile-atlas.hpp"
#include <Magnum/GL/MeshView.h>

namespace floormat {

constexpr auto quad_index_count = 6;

floor_mesh::floor_mesh()
{
    _mesh.setCount((int)(quad_index_count * TILE_COUNT))
         .addVertexBuffer(_positions_buffer, 0, tile_shader::Position{})
         .addVertexBuffer(_vertex_buffer, 0, tile_shader::TextureCoordinates{})
         .setIndexBuffer(_index_buffer, 0, GL::MeshIndexType::UnsignedShort);
}

void floor_mesh::set_tile(quad_data& data, tile_ref& x)
{
    if (auto ground = x.ground(); ground)
    {
        auto texcoords = ground.atlas->texcoords_for_id(ground.variant);
        for (size_t i = 0; i < 4; i++)
            data[i] = { texcoords[i] };
    }
    else
        for (size_t i = 0; i < 4; i++)
            data[i] = {};
}

void floor_mesh::draw(tile_shader& shader, chunk& c)
{
    //_vertex_buffer.setData({nullptr, sizeof(quad_data) * TILE_COUNT}, Magnum::GL::BufferUsage::DynamicDraw); // orphan the buffer
    std::array<quad_data, TILE_COUNT> data;
    for (auto [x, idx, pt] : c) {
        set_tile(data[idx], x);
    }
    _vertex_buffer.setSubData(0, data);
    Magnum::GL::MeshView mesh{_mesh};

    tile_atlas* last_atlas = nullptr;
    std::size_t last_pos = 0;

    const auto do_draw = [&](std::size_t i, tile_atlas* atlas) {
        if (atlas == last_atlas)
            return;
        if (auto len = i - last_pos; last_atlas != nullptr && len > 0)
        {
            last_atlas->texture().bind(0);
            mesh.setCount((int)(quad_index_count * len));
            mesh.setIndexRange((int)(last_pos*quad_index_count), 0, quad_index_count*TILE_COUNT - 1);
            shader.draw(mesh);
        }
        last_atlas = atlas;
        last_pos = i;
    };

    for (auto [x, i, pt] : c)
        do_draw(i, x.ground_atlas().get());
    do_draw(TILE_COUNT, nullptr);
}

std::array<std::array<UnsignedShort, 6>, TILE_COUNT> floor_mesh::make_index_array()
{
    // NOLINTNEXTLINE(cppcoreguidelines-pro-type-member-init)
    std::array<std::array<UnsignedShort, quad_index_count>, TILE_COUNT> array;
    for (std::size_t i = 0; i < std::size(array); i++)
        array[i] = tile_atlas::indices(i);
    return array;
}

std::array<std::array<Vector3, 4>, TILE_COUNT> floor_mesh::make_position_array()
{
    std::array<std::array<Vector3, 4>, TILE_COUNT> array;
    for (std::uint8_t j = 0, k = 0; j < TILE_MAX_DIM; j++)
        for (std::uint8_t i = 0; i < TILE_MAX_DIM; i++, k++)
            array[k] = { tile_atlas::floor_quad(Vector3(i, j, 0) * TILE_SIZE, TILE_SIZE2) };
    return array;
}

} // namespace floormat