BLT-With-Graphics-Template/src/blt/gfx/renderer/font_renderer.cpp

448 lines
17 KiB
C++

/*
* <Short Description>
* Copyright (C) 2024 Brett Terpstra
*
* 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 <https://www.gnu.org/licenses/>.
*/
#include <blt/gfx/renderer/font_renderer.h>
#include <blt/gfx/renderer/shaders/2d_font.vert>
#include <blt/gfx/renderer/shaders/2d_font.frag>
#include "blt/gfx/window.h"
#include <blt/gfx/shader.h>
#include <blt/gfx/imgui/ImGuiUtils.h>
#include <cctype>
// TODO: signed distance fonts
//#include <msdfgen.h>
#include <utility>
namespace blt::gfx
{
font_texture_atlas::font_texture_atlas(blt::i32 dimensions, blt::u32 padding): texture_gl2D(dimensions, dimensions, GL_R8), padding(padding)
{
bind();
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAX_LEVEL, 0);
}
font_texture_atlas::added_font_results_t font_texture_atlas::add_font(const font::font_file_t& file, float size)
{
return add_font(file, file.character_min_index, size);
}
font_texture_atlas::added_font_results_t font_texture_atlas::add_font(const font::font_file_t& file, blt::u64 start, float size)
{
blt::u32 max_height = 0;
blt::u64 min_index_used = file.character_max_index;
blt::u64 max_index_used = start;
file.font.set_pixel_sizes(0, static_cast<blt::u32>(size));
for (auto i = start; i < file.character_max_index; i++)
{
// returns an empty optional if there is no error code
if (file.font.load_char(i))
continue;
auto& face = *file.font.get();
auto width = face->glyph->bitmap.width;
auto height = face->glyph->bitmap.rows;
max_height = std::max(max_height, height);
// if adding this width overflows, we need to move to the next row.
if (used_width + width > static_cast<blt::u32>(getWidth()))
{
used_height += max_height + padding;
max_height = 0;
used_width = 0;
}
// if we can't fit the height of this character we should move to the next texture atlas.
if (used_height + height > static_cast<blt::u32>(getHeight()))
return {i, min_index_used, max_index_used};
min_index_used = std::min(min_index_used, i);
max_index_used = std::max(max_index_used, i);
auto begin_width = used_width;
auto begin_height = used_height;
auto end_width = used_width + width;
auto end_height = used_height + height;
used_width += width + padding;
// BLT_TRACE("%d %d %d %d", begin_width, begin_height, width, height);
// upload the texture
upload(face->glyph->bitmap.buffer, GL_RED, 0, static_cast<blt::i32>(begin_width), static_cast<blt::i32>(begin_height),
static_cast<blt::i32>(width), static_cast<blt::i32>(height), GL_UNSIGNED_BYTE);
glyphs[file.font.get_name()].insert(std::pair{i, bounded_t{
font::glyph_t{
{width, height},
{face->glyph->bitmap_left, face->glyph->bitmap_top},
face->glyph->advance.x},
{begin_width, begin_height}, {end_width, end_height}}
});
}
bind();
generateMipmaps();
return {-1ul, min_index_used, max_index_used};
}
font_generator_t::font_generator_t(blt::i32 dimensions, blt::u32 padding, float gen_size):
dimensions(dimensions), padding(padding), gen_size(gen_size)
{}
void font_generator_t::add_font(const font::font_file_t& file)
{
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
auto& vec = atlases;
if (atlases.empty())
atlases.emplace_back(std::make_unique<font_texture_atlas>(dimensions, padding));
auto& font_vec = font_associations[file.font.get_name()];
blt::u64 start_index = file.character_min_index;
while (start_index != -1ul)
{
// add font and associate results with a font file
auto results = atlases.back()->add_font(file, start_index, gen_size);
font_vec.emplace_back(atlases.back().get(), results.min_index_used, results.max_index_used);
// didn't fit, need to push new atlas
if (results.last_inserted_index != -1ul)
vec.emplace_back(std::make_unique<font_texture_atlas>(dimensions, padding));
start_index = results.last_inserted_index;
}
glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
}
font_renderer_t::font_renderer_t() = default;
void font_renderer_t::create(float gen_size, blt::i32 dimensions)
{
font_shader = shader_t::make_unique(shader_2d_font_vert, shader_2d_font_frag);
font_shader->bind();
font_shader->bindAttribute(0, "vertex");
font::create();
if (dimensions == 0)
{
int size;
glGetIntegerv(GL_MAX_TEXTURE_SIZE, &size);
dimensions = std::min(size, 2048);
BLT_INFO("Max texture size %d, font renderer will use size %d", size, dimensions);
}
generator = std::make_unique<font_generator_t>(dimensions, 2, gen_size);
}
void font_renderer_t::create_default(float gen_size, blt::i32 dimensions)
{
create(gen_size, dimensions);
// add_default_font(reinterpret_cast<const blt::u8*>(font::default_font_compressed_data), font::default_font_compressed_size, true);
add_default_font(reinterpret_cast<const blt::u8*>(ImGui::Spectrum::SourceSansProRegular_compressed_data),
ImGui::Spectrum::SourceSansProRegular_compressed_size, true);
}
void font_renderer_t::cleanup()
{
for (auto& n : added_texts)
n = nullptr;
for (auto& c : cached_deallocated_text_objects)
c = nullptr;
added_texts.clear();
cached_deallocated_text_objects.clear();
text_ptr_to_index.clear();
font_shader = nullptr;
generator = nullptr;
blt::gfx::font::cleanup();
}
font_renderer_t::compiled_text_t* font_renderer_t::create_text(std::string_view str, float size, std::string_view font)
{
std::unique_ptr<font_renderer_t::compiled_text_t> ptr = allocate_text();
ptr->setText(str, size, font);
auto index = added_texts.size();
added_texts.push_back(std::move(ptr));
text_ptr_to_index[added_texts.back().get()] = index;
return added_texts.back().get();
}
void font_renderer_t::destroy_text(font_renderer_t::compiled_text_t* text)
{
auto index = text_ptr_to_index[text];
text_ptr_to_index.erase(text);
cached_deallocated_text_objects.push_back(std::move(added_texts[index]));
added_texts[index] = nullptr;
added_texts.erase(added_texts.begin() + static_cast<blt::ptrdiff_t>(index));
}
void font_renderer_t::render()
{
#ifndef __EMSCRIPTEN__
glDisable(GL_MULTISAMPLE);
#endif
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
font_shader->bind();
glActiveTexture(GL_TEXTURE0);
for (auto [text, index] : text_ptr_to_index)
{
// stupid hack
if (text_ptr_to_text_index.find(text) != text_ptr_to_text_index.end())
continue;
text->bind();
blt::mat4x4 transform;
transform.translate(text->position);
transform.scale(text->scale);
transform.rotateZ(blt::toRadians(text->rotation));
font_shader->setMatrix("transform", transform);
font_shader->setFloat("z_index", text->z_index);
font_shader->setVec4("color", text->color);
text->draw();
}
for (auto& [text, list] : render_list)
{
text->bind();
render_context_t before{text->current_size};
text->save_to(before);
for (const auto& context : list)
{
text->restore_from(context);
blt::mat4x4 transform;
transform.translate(text->position);
transform.scale(text->scale);
transform.rotateZ(blt::toRadians(text->rotation));
font_shader->setMatrix("transform", transform);
font_shader->setFloat("z_index", text->z_index);
font_shader->setVec4("color", text->color);
text->draw();
}
text->restore_from(before);
if (last_rendered_strings.find(text) != last_rendered_strings.end())
last_rendered_strings.erase(text);
}
render_list.clear();
for (auto& text : last_rendered_strings)
{
rendered_strings.erase(text_ptr_to_text_index[text]);
text_ptr_to_text_index.erase(text);
cached_deallocated_text_objects.emplace_back(std::move(added_texts[text_ptr_to_index[text]]));
text_ptr_to_index.erase(text);
}
last_rendered_strings.clear();
for (auto& text : currently_rendered_strings)
last_rendered_strings.insert(text);
currently_rendered_strings.clear();
#ifndef __EMSCRIPTEN__
glEnable(GL_MULTISAMPLE);
#endif
}
void font_renderer_t::add_default_font(std::string_view path, std::string_view name)
{
font::font_face_t default_font_face{path, name};
font::font_file_t default_font{default_font_face, 0, 256};
add_font(default_font);
}
void font_renderer_t::add_default_font(const blt::u8* data, blt::size_t size, bool compressed, std::string_view name)
{
font::font_face_t default_font_face{name, data, size, compressed};
font::font_file_t default_font{default_font_face, 0, 256};
add_font(default_font);
}
std::unique_ptr<font_renderer_t::compiled_text_t> font_renderer_t::allocate_text()
{
std::unique_ptr<font_renderer_t::compiled_text_t> ptr;
if (!cached_deallocated_text_objects.empty())
{
ptr = std::move(*cached_deallocated_text_objects.begin());
cached_deallocated_text_objects.erase(cached_deallocated_text_objects.begin());
} else
{
ptr = std::unique_ptr<compiled_text_t>(new compiled_text_t{*generator});
}
return ptr;
}
font_renderer_t::render_context_t& font_renderer_t::render_text(std::string_view str, float size, std::string_view font)
{
auto str_iter = rendered_strings.find(text_index_t{std::string(str), std::string(font)});
compiled_text_t* ptr;
if (str_iter != rendered_strings.end())
{
ptr = str_iter->second;
currently_rendered_strings.insert(ptr);
ptr->setSize(size);
} else
{
auto index = added_texts.size();
added_texts.emplace_back(allocate_text());
ptr = added_texts.back().get();
ptr->setText(str, size);
text_ptr_to_text_index[ptr] = text_index_t{std::string(str), std::string(font)};
rendered_strings[text_index_t{std::string(str), std::string(font)}] = ptr;
text_ptr_to_index[ptr] = index;
currently_rendered_strings.insert(ptr);
}
auto& vec = render_list[ptr];
vec.emplace_back(size);
return vec.back();
}
font_renderer_t::compiled_text_t& font_renderer_t::compiled_text_t::setText(std::string_view str, float size, std::string_view f)
{
if (str == contents)
return *this;
contents = str;
font = f;
setSize(size);
recompile();
return *this;
}
font_renderer_t::compiled_text_t::compiled_text_t(font_generator_t& generator): generator(generator)
{
auto vbo = vertex_array_t::make_vbo({});
vbo->vbo.create();
vbo->vbo.bind();
vbo->vbo.allocate(sizeof(float) * 6 * 4, GL_DYNAMIC_DRAW);
vao.bind();
vao.bindVBO(vbo, 0, 4, GL_FLOAT, 4 * sizeof(float), 0);
}
void font_renderer_t::compiled_text_t::draw()
{
font_texture_atlas* last_texture = nullptr;
for (const auto& render : renders)
{
if (last_texture != render.texture)
{
last_texture = render.texture;
render.texture->bind();
}
glDrawArrays(GL_TRIANGLES, static_cast<int>(render.render_start), static_cast<int>(render.render_count));
}
}
font_renderer_t::compiled_text_t& font_renderer_t::compiled_text_t::recompile()
{
static std::vector<float> vertices;
vertices.clear();
renders.clear();
font_texture_atlas* last_texture = nullptr;
blt::size_t draw_start = 0;
blt::size_t draw_count = 0;
const float sc = generator.get_generated_size() / current_size;
float global_x = 0;
float global_y = 0;
// TODO: parker UTF8
for (const auto& c : contents)
{
if (bounds.y() > 0 && std::abs(global_y) >= bounds.y() * sc)
continue;
if (c == '\n')
{
global_x = 0;
global_y -= generator.get_generated_size();
continue;
}
auto& texture = generator.get_texture(std::string(font), c);
auto& ch = texture.get_glyph(c, font);
if (std::isblank(c))
{
if (bounds.x() > 0 && global_x > bounds.x() * sc)
{
global_x = 0;
global_y -= generator.get_generated_size();
continue;
}
}
if (last_texture == nullptr)
last_texture = &texture;
else if (last_texture != &texture)
{
renders.emplace_back(last_texture, draw_start, draw_count);
draw_start += draw_count;
draw_count = 0;
}
auto dims = static_cast<float>(generator.get_dimensions());
float x_pos = global_x + static_cast<float>(ch.glyph.bearing.x());
float y_pos = global_y - static_cast<float>(ch.glyph.size.y() - ch.glyph.bearing.y());
auto w = static_cast<float>(ch.glyph.size.x());
auto h = static_cast<float>(ch.glyph.size.y());
auto texture_min_x = static_cast<float>(ch.min.x()) / dims;
auto texture_min_y = static_cast<float>(ch.min.y()) / dims;
auto texture_max_x = static_cast<float>(ch.max.x()) / dims;
auto texture_max_y = static_cast<float>(ch.max.y()) / dims;
// BLT_TRACE("%c: %f | xy[%f %f] wh[%f %f] | min[%f %f] max[%f %f] |", c, x, x_pos, y_pos, w, h, texture_min_x, texture_min_y, texture_max_x, texture_max_y);
std::array<float, 6 * 4> local_vert = {
x_pos, y_pos + h, texture_min_x, texture_min_y,
x_pos, y_pos, texture_min_x, texture_max_y,
x_pos + w, y_pos, texture_max_x, texture_max_y,
x_pos, y_pos + h, texture_min_x, texture_min_y,
x_pos + w, y_pos, texture_max_x, texture_max_y,
x_pos + w, y_pos + h, texture_max_x, texture_min_y
};
vertices.insert(vertices.end(), local_vert.begin(), local_vert.end());
draw_count += 6;
global_x += static_cast<float>(ch.glyph.advance >> 6);
}
// BLT_TRACE("size: %ld %ld %ld", draw_count, vertices.size(), vertices.size() / 4);
renders.emplace_back(last_texture, draw_start, draw_count);
vao.getBuffer(0).update(static_cast<long>(vertices.size() * sizeof(float)), vertices.data());
return *this;
}
bool font_renderer_t::text_index_t::operator==(const font_renderer_t::text_index_t& rhs) const
{
return text == rhs.text && font == rhs.font;
}
bool font_renderer_t::text_index_t::operator!=(const font_renderer_t::text_index_t& rhs) const
{
return !(rhs == *this);
}
}