aboutsummaryrefslogtreecommitdiffstats
path: root/src/texture.cpp
blob: b013bef5c6d85af7768860f5efd363527e2e3f9d (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
/**
 * @file texture.cpp
 * Handles all texture loading 
 *
 * Copyright (C) 2019  Belle-Isle, Andrew <drumsetmonkey@gmail.com>
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */


#include "texture.hpp"

#include <soil/SOIL.h>

#include <unordered_map>
#include <iostream>

struct TextureData
{
    GLuint tex = 0;
    int width = 0;
    int height= 0;
};

// Stores a list of all textures we've already loaded. This makes sure we don't
// waste our precious CPU cycles reloading a texture.
std::unordered_map<std::string, TextureData> textureCache;

TextureData loadTexture(std::string filename)
{
    TextureData tex;

    // Search to see if this texture has already been loading
    auto cacheSearch = textureCache.find(filename);

    if (cacheSearch == textureCache.end()) {
    // If this texture hasn't been loading

        unsigned char* image = SOIL_load_image(filename.c_str(),
                                               &(tex.width), &(tex.height), 0,
                                               SOIL_LOAD_RGBA);

        glGenTextures(1, &(tex.tex));
        glBindTexture(GL_TEXTURE_2D, tex.tex);
        glPixelStoref(GL_UNPACK_ALIGNMENT, 1);
        glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
        glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
        glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
        glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
        glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, tex.width, tex.height, 0,
                     GL_RGBA, GL_UNSIGNED_BYTE, image);

        SOIL_free_image_data(image);

        // Add new texture to the texture cache
        textureCache.emplace(filename, tex);

    } else {
    // If this texture has been loaded, just return the loaded texture
        tex = cacheSearch->second;
    }

    return tex;
}

void Texture::loadFromString(std::string filename)
{
    TextureData d = loadTexture(filename);

    tex = d.tex;
    width = d.width;
    height = d.height;
}

Texture::Texture(std::string filename)
{
    loadFromString(filename);
}

Texture::Texture(sol::object param)
{
    if (param.get_type() == sol::type::string) {
        loadFromString(param.as<std::string>());
    } else if (param.get_type() == sol::type::table) {
        sol::table tab = param;

        // If there is a filename given, load that file to get image data
        if (tab["file"] == sol::type::string)
            loadFromString(tab.get<std::string>("file"));
        else
            return; // If we don't have image data just return a null image

        if (tab["offset"] == sol::type::table) {
            sol::table off = tab["offset"];
            if (off["x"] == sol::type::number)
                offset.x = off.get<float>("x")/width;
            if (off["y"] == sol::type::number)
                offset.y = off.get<float>("y")/height;
        }

        if (tab["size"] == sol::type::table) {
            sol::table siz = tab["size"];
            if (siz["x"] == sol::type::number)
                size.x = siz.get<float>("x")/width;
            if (siz["y"] == sol::type::number)
                size.y = siz.get<float>("y")/height;
        }
    }
}