/* * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ #include "TextExampleApp.hpp" #include "Scene/Scene.hpp" #include "Scene/Shader/Shader.hpp" #include "Scene/Geometry.hpp" #include "Scene/TextDrawable.hpp" #include "Scene/GeometryFactory.hpp" #include "Scene/Material.hpp" #include "Scene/Vertex.hpp" #include "Scene/SimpleDrawable.hpp" #include "Scene/UI/PerformanceInfo.hpp" #include "Scene/UniformBuffer.hpp" #include "Input/InputManager.hpp" #include "Host/GraphicsAppManager.hpp" #include "Host/GLFW/WindowGLFW.hpp" #include "Host/ResourceLoader.hpp" #include "Math/Math.hpp" #include "Base/EngineConfiguration.hpp" #include "Controller/FreeCamCameraController.hpp" #include "Image/ImageLoaderPng.hpp" #include "Scene/SdfFontAtlasGenerator.hpp" #include "Scene/IFontAtlasGenerator.hpp" #include "Scene/BitmapFontAtlasGenerator.hpp" #include #ifdef _WIN32 #undef TRANSPARENT #endif namespace OpenVulkano { using namespace Scene; using namespace Input; using namespace Math; namespace fs = std::filesystem; constexpr int CREATE_BITMAP_ATLAS = 0; constexpr int CREATE_NEW_ATLAS = 0; class TextExampleAppImpl final : public TextExampleApp { public: void Init() override { auto engineConfig = OpenVulkano::EngineConfiguration::GetEngineConfiguration(); engineConfig->SetPreferFramebufferFormatSRGB(false); std::srand(1); // Fix seed for random numbers m_scene.Init(); m_cam.Init(70, 16, 9, 0.1f, 100); m_scene.SetCamera(&m_cam); static std::vector> texts; texts.push_back(std::make_pair("ABab?.^{}_cdFGETG123)(", TextConfig())); texts.push_back(std::make_pair("Hello, World!", TextConfig())); texts.push_back(std::make_pair("\u0410\u0411\u0412\u041F", TextConfig())); texts.push_back(std::make_pair("Unsupported glyphs \u1E30\u1E31 are coming", TextConfig())); texts.push_back(std::make_pair("This is first line\nSecond gg line\nThird G line", TextConfig())); texts[0].second.applyBorder = true; texts[1].second.backgroundColor.a = 1; const int N = texts.size(); auto& resourceLoader = ResourceLoader::GetInstance(); const std::string fontPath = resourceLoader.GetResourcePath("Roboto-Regular.ttf"); m_nodesPool.resize(N * 3); m_drawablesPool.resize(N * 3); if constexpr (CREATE_BITMAP_ATLAS) { std::set s = BitmapFontAtlasGenerator::LoadAllGlyphs(fontPath); BitmapFontAtlasGenerator generator; generator.GenerateAtlas(fontPath, s); generator.SaveAtlasMetadataInfo("bitmap_atlas"); } #if defined(MSDFGEN_AVAILABLE) && CREATE_NEW_ATLAS std::set s = SdfFontAtlasGenerator::LoadAllGlyphs(fontPath); msdf_atlas::Charset charset; for (uint32_t c : s) { charset.add(c); } m_atlasGenerator.GenerateAtlas(fontPath, charset); m_msdfAtlasGenerator.GenerateAtlas(fontPath, charset); m_atlasGenerator.SaveAtlasMetadataInfo("sdf_atlas.png"); m_msdfAtlasGenerator.SaveAtlasMetadataInfo("msdf_atlas"); #else auto sdfMetadataInfo = resourceLoader.GetResource("sdf_atlas_packed.png"); auto msdfMetadataInfo = resourceLoader.GetResource("msdf_atlas_packed.png"); auto bitmapMetadataInfo = resourceLoader.GetResource("bitmap_atlas_packed.png"); #endif for (int i = 0; i < texts.size() * 3; i++) { int textIdx = i % texts.size(); TextDrawable* t = nullptr; #if defined(MSDFGEN_AVAILABLE) && CREATE_NEW_ATLAS if (i < texts.size()) { t = new TextDrawable(m_atlasGenerator.GetAtlasData(), texts[textIdx].second); t->SetShader(&TextDrawable::GetSdfDefaultShader()); } else { t = new TextDrawable(m_msdfAtlasGenerator.GetAtlasData(), texts[textIdx].second); t->SetShader(&TextDrawable::GetMsdfDefaultShader()); } #else int xOffset = 0; if (i < N) { t = new TextDrawable(sdfMetadataInfo, texts[textIdx].second); t->SetShader(&TextDrawable::GetSdfDefaultShader()); xOffset = -5; } else if (i >= N && i < N * 2) { t = new TextDrawable(msdfMetadataInfo, texts[textIdx].second); t->SetShader(&TextDrawable::GetMsdfDefaultShader()); xOffset = 15; } else { t = new TextDrawable(bitmapMetadataInfo, texts[textIdx].second); t->SetShader(&TextDrawable::GetBitmapDefaultShader()); xOffset = 35; } // OR use separate texture + metadata file //auto metadataInfo = resourceLoader.GetResource("atlas_metadata"); //auto data = resourceLoader.GetResource("roboto-regular-atlas.png"); //Image::ImageLoaderPng loader; //static auto image = loader.loadData(reinterpret_cast(data.Data()), data.Size()); //static Texture tex; //tex.resolution = image->resolution; //tex.textureBuffer = image->data.Data(); //tex.format = image->dataFormat; //tex.size = image->data.Size(); // 1 channel //TextDrawable* t = new TextDrawable(metadataInfo, &tex, texts[i].second); #endif // MSDFGEN_AVAILABLE t->GenerateText(texts[textIdx].first); m_drawablesPool[i] = t; m_nodesPool[i].Init(); m_nodesPool[i].SetMatrix(Math::Utils::translate(glm::mat4x4(1.f), Vector3f(xOffset, 2 - textIdx * 2, 0))); m_nodesPool[i].AddDrawable(m_drawablesPool[i]); m_scene.GetRoot()->AddChild(&m_nodesPool[i]); } GetGraphicsAppManager()->GetRenderer()->SetScene(&m_scene); m_camController.Init(&m_cam); m_camController.SetDefaultKeybindings(); m_camController.SetPosition({ 10, 0, 15 }); m_camController.SetBoostFactor(5); std::shared_ptr m_perfInfo = std::make_shared(); m_ui.AddElement(m_perfInfo); GetGraphicsAppManager()->GetRenderer()->SetActiveUi(&m_ui); } void Tick() override { m_camController.Tick(); } void Close() override { for (SimpleDrawable* d: m_drawablesPool) { d->Close(); delete d; } } private: OpenVulkano::Scene::Scene m_scene; PerspectiveCamera m_cam; OpenVulkano::FreeCamCameraController m_camController; #ifdef MSDFGEN_AVAILABLE SdfFontAtlasGeneratorT m_atlasGenerator; MsdfFontAtlasGenerator m_msdfAtlasGenerator; #endif std::vector m_drawablesPool; std::vector m_nodesPool; Vector3f_SIMD m_position = { 0, 0, -10 }; OpenVulkano::Scene::UI::SimpleUi m_ui; std::shared_ptr m_perfInfo; }; IGraphicsApp* TextExampleApp::Create() { return new TextExampleAppImpl(); } std::unique_ptr TextExampleApp::CreateUnique() { return std::make_unique(); } } #pragma clang diagnostic pop #pragma clang diagnostic pop