aboutsummaryrefslogtreecommitdiffstats
path: root/src/font.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/font.cpp')
-rw-r--r--src/font.cpp131
1 files changed, 131 insertions, 0 deletions
diff --git a/src/font.cpp b/src/font.cpp
new file mode 100644
index 0000000..adffa9c
--- /dev/null
+++ b/src/font.cpp
@@ -0,0 +1,131 @@
+#include <font.hpp>
+
+FT_Library FontSystem::ftLibrary;
+FT_Face FontSystem::ftFace;
+
+std::string FontSystem::fontFamily;
+std::map<int, std::vector<FT_Info>> FontSystem::fontData;
+
+int FontSystem::currentSize = 0;
+Color FontSystem::currentColor;
+float FontSystem::currentZ = -8.0f;
+
+void FontSystem::init(const std::string& ttf)
+{
+ FT_Init_FreeType(&ftLibrary);
+ FT_New_Face(ftLibrary, ttf.c_str(), 0, &ftFace);
+}
+
+void FontSystem::setFontSize(int size)
+{
+ auto result = fontData.try_emplace(size, 93);
+ if (result.second) {
+ FT_Set_Pixel_Sizes(ftFace, 0, size);
+
+ char c = 33;
+ for (auto& d : fontData.at(size)) {
+ glDeleteTextures(1, &d.tex);
+ glGenTextures(1, &d.tex); // Generate new texture name/locations?
+
+ // load the character from the font family file
+ FT_Load_Char(ftFace, c++, FT_LOAD_RENDER);
+
+ // transfer the character's bitmap (?) to a texture for rendering
+ glBindTexture(GL_TEXTURE_2D, d.tex);
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S , GL_CLAMP_TO_EDGE);
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T , GL_CLAMP_TO_EDGE);
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER , GL_LINEAR);
+ glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER , GL_LINEAR);
+ glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
+
+ /**
+ * The just-created texture will render red-on-black if we don't do anything to it, so
+ * here we create a buffer 4 times the size and transform the texture into an RGBA array,
+ * making it white-on-black.
+ */
+ auto& g = ftFace->glyph;
+ std::vector<uint32_t> buf (g->bitmap.width * g->bitmap.rows, 0xFFFFFFFF);
+ for (auto j = buf.size(); j--;)
+ buf[j] ^= !g->bitmap.buffer[j] ? buf[j] : 0;
+
+ d.wh.x = g->bitmap.width;
+ d.wh.y = g->bitmap.rows;
+ d.bl.x = g->bitmap_left;
+ d.bl.y = g->bitmap_top;
+ d.ad.x = g->advance.x >> 6;
+ d.ad.y = g->advance.y >> 6;
+ glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, g->bitmap.width, g->bitmap.rows,
+ 0, GL_RGBA, GL_UNSIGNED_BYTE, buf.data());
+ }
+ }
+
+ currentSize = size;
+}
+
+void FontSystem::setFontColor(float r, float g, float b)
+{
+ currentColor.red = r;
+ currentColor.green = g;
+ currentColor.blue = b;
+}
+
+void FontSystem::setFontZ(float z)
+{
+ currentZ = z;
+}
+
+vec2 FontSystem::putChar(float xx, float yy, char c)
+{
+ const auto& ch = fontData.at(currentSize)[c - 33];
+ int x = xx, y = yy;
+
+ // get dimensions of the rendered character
+ vec2 c1 = {
+ static_cast<float>(floor(x) + ch.bl.x),
+ static_cast<float>(floor(y) + ch.bl.y)
+ };
+ const auto& c2 = ch.wh;
+
+ Render::textShader.use();
+ Render::textShader.enable();
+
+ // draw the character
+ glActiveTexture(GL_TEXTURE0);
+ glBindTexture(GL_TEXTURE_2D, ch.tex);
+
+ glUniform4f(Render::textShader.uniform[WU_tex_color], 1.0f, 1.0f, 1.0f, 1.0f);
+
+ GLfloat tex_coord[] = {
+ 0.0, 1.0, // bottom left
+ 1.0, 1.0, // bottom right
+ 1.0, 0.0, // top right
+ 1.0, 0.0, // top right
+ 0.0, 0.0, // top left
+ 0.0, 1.0, // bottom left
+ };
+
+ GLfloat text_vert[] = {
+ c1.x, c1.y - c2.y, currentZ, // bottom left
+ c1.x + c2.x, c1.y - c2.y, currentZ, // bottom right
+ c1.x + c2.x, c1.y + c2.y - c2.y, currentZ, // top right
+ c1.x + c2.x, c1.y + c2.y - c2.y, currentZ, // top right
+ c1.x, c1.y + c2.y - c2.y, currentZ, // top left
+ c1.x, c1.y - c2.y, currentZ // bottom left
+ };
+
+ glUniform4f(Render::textShader.uniform[WU_tex_color],
+ currentColor.red, currentColor.green, currentColor.blue, currentColor.alpha);
+
+ glVertexAttribPointer(Render::textShader.coord, 3, GL_FLOAT, GL_FALSE, 0, text_vert);
+ glVertexAttribPointer(Render::textShader.tex, 2, GL_FLOAT, GL_FALSE, 0, tex_coord);
+ glDrawArrays(GL_TRIANGLES, 0, 6);
+
+ glUniform4f(Render::textShader.uniform[WU_tex_color], 1.0, 1.0, 1.0, 1.0);
+
+ Render::textShader.disable();
+ Render::textShader.unuse();
+
+ // return the width.
+ return ch.ad;
+}
+