/* ************************************************************************** */ /* */ /* ::: :::::::: */ /* text_pipeline.cpp :+: :+: :+: */ /* +:+ +:+ +:+ */ /* By: maldavid +#+ +:+ +#+ */ /* +#+#+#+#+#+ +#+ */ /* Created: 2023/04/06 16:41:13 by maldavid #+# #+# */ /* Updated: 2023/12/14 17:49:37 by maldavid ### ########.fr */ /* */ /* ************************************************************************** */ #include #include #include #include #include #define STB_RECT_PACK_IMPLEMENTATION #include #include #define STB_TRUETYPE_IMPLEMENTATION #define STB_malloc(x, u) ((void)(u), MemManager::malloc(x)) #define STB_free(x, u) ((void)(u), MemManager::free(x)) #include constexpr const int RANGE = 1024; namespace mlx { TextDrawData::TextDrawData(std::string _text, int _color, int _x, int _y) : x(_x), y(_y), color(_color), text(std::move(_text)) {} void TextDrawData::init(TextLibrary& library, Font* const font) noexcept { std::vector vertexData; std::vector indexData; float stb_x = 0.0f; float stb_y = 0.0f; for(char c : text) { if(c < 32) continue; stbtt_aligned_quad q; stbtt_GetPackedQuad(font->getCharData().data(), RANGE, RANGE, c - 32, &stb_x, &stb_y, &q, 1); std::size_t index = vertexData.size(); vertexData.emplace_back(glm::vec2{q.x0, q.y0}, glm::vec4{color & 0x00FF0000, color & 0x0000FF00, color & 0x000000FF, 0xFFFFFFFF}, glm::vec2{q.s0, q.t0}); vertexData.emplace_back(glm::vec2{q.x1, q.y0}, glm::vec4{color & 0x00FF0000, color & 0x0000FF00, color & 0x000000FF, 0xFFFFFFFF}, glm::vec2{q.s1, q.t0}); vertexData.emplace_back(glm::vec2{q.x1, q.y1}, glm::vec4{color & 0x00FF0000, color & 0x0000FF00, color & 0x000000FF, 0xFFFFFFFF}, glm::vec2{q.s1, q.t1}); vertexData.emplace_back(glm::vec2{q.x0, q.y1}, glm::vec4{color & 0x00FF0000, color & 0x0000FF00, color & 0x000000FF, 0xFFFFFFFF}, glm::vec2{q.s0, q.t1}); indexData.emplace_back(index + 0); indexData.emplace_back(index + 1); indexData.emplace_back(index + 2); indexData.emplace_back(index + 2); indexData.emplace_back(index + 3); indexData.emplace_back(index + 0); } std::shared_ptr text_data = std::make_shared(); text_data->init(text, font, std::move(vertexData), std::move(indexData)); id = library.addTextToLibrary(text_data); #ifdef DEBUG core::error::report(e_kind::message, "Text put : registered new text to render"); #endif } void TextPutPipeline::init(Renderer* renderer) noexcept { _renderer = renderer; _font_in_use = &const_cast(*_font_set.emplace(*_renderer, "default", dogica_ttf, 6.0f).first); } void TextPutPipeline::loadFont(const std::filesystem::path& filepath, float scale) { if(filepath.string() == "default") // we're sure it is already loaded _font_in_use = &const_cast(*_font_set.emplace(*_renderer, "default", dogica_ttf, scale).first); else _font_in_use = &const_cast(*_font_set.emplace(*_renderer, filepath, scale).first); } void TextPutPipeline::put(int x, int y, int color, std::string str) { auto res = _drawlist.emplace(std::move(str), color, x, y); if(res.second) const_cast(*res.first).init(_library, _font_in_use); else { auto text_ptr = _library.getTextData(res.first->id); if(*_font_in_use != text_ptr->getFontInUse()) { _library.removeTextFromLibrary(res.first->id); const_cast(*res.first).init(_library, _font_in_use); } } } void TextPutPipeline::render(std::array& sets) { for(auto& draw : _drawlist) { std::shared_ptr draw_data = _library.getTextData(draw.id); const TextureAtlas& atlas = draw_data->getFontInUse().getAtlas(); draw_data->bind(*_renderer); atlas.updateSet(0); sets[1] = const_cast(atlas).getSet(); vkCmdBindDescriptorSets(_renderer->getActiveCmdBuffer().get(), VK_PIPELINE_BIND_POINT_GRAPHICS, _renderer->getPipeline().getPipelineLayout(), 0, sets.size(), sets.data(), 0, nullptr); atlas.render(*_renderer, draw.x, draw.y, draw_data->getIBOsize()); } } void TextPutPipeline::destroy() noexcept { _library.clearLibrary(); _drawlist.clear(); _font_set.clear(); } }