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;
}
}
}
|