aboutsummaryrefslogtreecommitdiffstats
path: root/src/texture.cpp
blob: 44e48a0f1cae9bed9fcee4acf323266d7ffae1bd (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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
#include <texture.hpp>

#include <algorithm>
#include <string>
#include <stdexcept>

#include <SDL2/SDL_image.h>

#include <config.hpp>
#include <debug.hpp>
#include <error.hpp>

namespace Colors
{
	ColorTex white;
	ColorTex black;
	ColorTex red;
	ColorTex blue;

	GLfloat texCoord[12] = {
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
	};

	void init(void) {
		white = ColorTex(Color(255, 255, 255));
		black = ColorTex(Color(0,   0,   0  ));
		red   = ColorTex(Color(255, 0,   0  ));
		blue  = ColorTex(Color(0,   0,   255));
	}
}

void loadTexture(const std::string& file, Texture& texture);

Texture::Texture(const std::string& file, const GLuint& t, const vec2& v)
	: name(file), tex(t), dim(v)
{
	if (t == 0xFFFFF && !file.empty())
		loadTexture(file, *this);
}

Texture::Texture(const std::string& file, bool hline)
{
	loadTexture(file, *this);
	if (hline)
		dim /= game::HLINE;
}

const std::string& Texture::getName(void) const
{
	return name;
}

const vec2& Texture::getDim(void) const
{
	return dim;
}

ColorTex::ColorTex(void)
{
	Texture();
}

ColorTex::ColorTex(const Color& color)
{
	unsigned char data[4] = {
		static_cast<unsigned char>(color.red),
		static_cast<unsigned char>(color.green),
		static_cast<unsigned char>(color.blue),
		static_cast<unsigned char>(color.alpha),
	};

	GLuint object;
	glGenTextures(1, &object);				// Turns "object" into a texture
	glBindTexture(GL_TEXTURE_2D, object);	// Binds "object" to the top of the stack
	glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE, data);

	tex = object;
	dim = vec2(1, 1);
}

static std::vector<Texture> loadedTextures;

void loadTexture(const std::string& file, Texture& texture)
{
	auto preloaded =
		std::find_if(std::begin(loadedTextures), std::end(loadedTextures),
		[&file](const Texture& t) { return (t.getName() == file); });

	if (preloaded == std::end(loadedTextures)) {
		auto image = IMG_Load(file.c_str());
		UserAssert(image != nullptr, "File not found: " + file);

#ifdef DEBUG
		DEBUG_printf("Loaded image file: %s\n", file.c_str());
#endif // DEBUG

		// load texture through OpenGL
		GLuint object;
		glGenTextures(1, &object);				// Turns "object" into a texture
		glBindTexture(GL_TEXTURE_2D, object);	// Binds "object" to the top of the stack
		glPixelStoref(GL_UNPACK_ALIGNMENT, 1);
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);	// Sets the "min" filter
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);	// The the "max" filter of the stack
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); // Wrap the texture to the matrix
		glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); //
		glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, image->w, image->h, 0,
					 GL_RGBA, GL_UNSIGNED_BYTE, image->pixels);

		// add texture to loadedTextures
		loadedTextures.emplace_back(file, object, vec2(image->w, image->h));

		texture = loadedTextures.back();

		// free the SDL_Surface
		SDL_FreeSurface(image);
	} else {
		texture = *preloaded;
	}
}


TextureIterator::TextureIterator(const std::vector<std::string> &l)
{
	for (const auto& s : l)
		textures.emplace_back(s);
	position = std::begin(textures);
}

void TextureIterator::operator++(int) noexcept
{
	if (++position >= std::end(textures))
		position = std::end(textures) - 1;
	position->use();
}

void TextureIterator::operator--(int) noexcept
{
	if (--position < std::begin(textures))
		position = std::begin(textures);
	position->use();
}

void TextureIterator::operator()(const int &index)
{
	if (index < 0 || index > static_cast<int>(textures.size()))
		throw std::invalid_argument("texture index out of range");

	position = std::begin(textures) + index;
	position->use();
}


void unloadTextures(void)
{
	while (!loadedTextures.empty()) {
		loadedTextures.back().destroy();
		loadedTextures.pop_back();
	}
}