diff --git a/assets/dawnrpg/CMakeLists.txt b/assets/dawnrpg/CMakeLists.txt new file mode 100644 index 00000000..0cc5becb --- /dev/null +++ b/assets/dawnrpg/CMakeLists.txt @@ -0,0 +1,6 @@ +# Copyright (c) 2024 Dominic Masters +# +# This software is released under the MIT License. +# https://opensource.org/licenses/MIT + +tool_truetype(font_silver "${CMAKE_CURRENT_LIST_DIR}/Silver.ttf") \ No newline at end of file diff --git a/assets/dawnrpg/Silver.ttf b/assets/dawnrpg/Silver.ttf new file mode 100644 index 00000000..a6cbc52a Binary files /dev/null and b/assets/dawnrpg/Silver.ttf differ diff --git a/lib/CMakeLists.txt b/lib/CMakeLists.txt index c29b117d..af869a30 100644 --- a/lib/CMakeLists.txt +++ b/lib/CMakeLists.txt @@ -9,6 +9,7 @@ if(DAWN_TARGET STREQUAL "linux-x64-glfw") add_subdirectory(glad) add_subdirectory(glfw) add_subdirectory(libarchive) + add_subdirectory(freetype) else() message(FATAL_ERROR "Unknown target: ${DAWN_TARGET}") endif() diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 9f4e3119..0167be64 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -14,7 +14,7 @@ if(DAWN_TARGET STREQUAL "linux-x64-glfw") add_subdirectory(dawnlinux) add_subdirectory(dawnglfw) add_subdirectory(dawnopengl) - # add_subdirectory(dawnpoker) + add_subdirectory(dawnrpg) else() message(FATAL_ERROR "You need to define an entry target") endif() diff --git a/src/dawn/asset/AssetManager.cpp b/src/dawn/asset/AssetManager.cpp index f70d7bbd..f8d55315 100644 --- a/src/dawn/asset/AssetManager.cpp +++ b/src/dawn/asset/AssetManager.cpp @@ -5,6 +5,7 @@ #include "AssetManager.hpp" #include "loaders/TextureLoader.hpp" +#include "loaders/TrueTypeLoader.hpp" using namespace Dawn; @@ -59,6 +60,27 @@ bool_t AssetManager::isLoaded(const std::string filename) { return false; } +template<> +std::shared_ptr AssetManager::get( + const std::string filename, + const uint32_t fontSize +) { + auto existing = this->getExisting(filename); + if(existing) { + // Check pointer hasn't gone stale, if it has remove it and create new. + auto texture = existing->getTexture(fontSize); + if(texture) return texture; + this->removeExisting(filename); + } + + std::shared_ptr loader = std::make_shared( + filename + ); + pendingAssetLoaders.push_back(std::static_pointer_cast(loader)); + return loader->getTexture(fontSize); +} + + AssetManager::~AssetManager() { } \ No newline at end of file diff --git a/src/dawn/asset/loaders/CMakeLists.txt b/src/dawn/asset/loaders/CMakeLists.txt index b4f5f2b4..e7c90da1 100644 --- a/src/dawn/asset/loaders/CMakeLists.txt +++ b/src/dawn/asset/loaders/CMakeLists.txt @@ -7,4 +7,5 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE TextureLoader.cpp + TrueTypeLoader.cpp ) \ No newline at end of file diff --git a/src/dawn/asset/loaders/TrueTypeLoader.cpp b/src/dawn/asset/loaders/TrueTypeLoader.cpp new file mode 100644 index 00000000..d1138efa --- /dev/null +++ b/src/dawn/asset/loaders/TrueTypeLoader.cpp @@ -0,0 +1,97 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "TrueTypeLoader.hpp" +#include "assert/assert.hpp" + +using namespace Dawn; + +TrueTypeLoader::TrueTypeLoader(const std::string name) : + AssetLoader(name), + loader(name + ".ttf") +{ + // Init the font. + auto ret = FT_Init_FreeType(&fontLibrary); + assertTrue(ret == 0, "Failed to initialize FreeType library."); +} + +void TrueTypeLoader::updateSync() { + if(state != TrueTypeLoaderState::ASYNC_DONE) return; + state = TrueTypeLoaderState::SYNC_LOADING; + + // Init all the textures. + auto it = textures.begin(); + while(it != textures.end()) { + auto texture = it->second.lock(); + + if(texture) { + texture->setFace(face); + it++; + continue; + } + + it = textures.erase(it); + } + + // Done + state = TrueTypeLoaderState::SYNC_DONE; + this->loaded = true; +} + +void TrueTypeLoader::updateAsync() { + if(state != TrueTypeLoaderState::INITIAL) return; + state = TrueTypeLoaderState::ASYNC_LOADING; + + // Load the data. + this->loader.open(); + size_t size = loader.getSize(); + buffer = new uint8_t[size]; + + // Read the data. + size_t readSize = loader.read(buffer, size); + assertTrue(readSize == size, "Failed to read all data from TrueTypeLoader."); + + // Init the font. + auto ret = FT_New_Memory_Face(fontLibrary, buffer, size, 0, &face); + assertTrue(ret == 0, "Failed to load font face."); + + // Now close the asset loader + loader.close(); + state = TrueTypeLoaderState::ASYNC_DONE; +} + +std::shared_ptr TrueTypeLoader::getTexture( + const uint32_t fontSize +) { + // Check if we have the texture already and it hasn't gone stale. + auto it = textures.find(fontSize); + if(it != textures.end()) { + if(!it->second.expired()) return it->second.lock(); + textures.erase(it); + } + + // Create the texture. + auto texture = std::make_shared(fontSize); + textures[fontSize] = texture; + if(this->loaded) texture->setFace(face); + return texture; +} + +TrueTypeLoader::~TrueTypeLoader() { + if( + this->state == TrueTypeLoaderState::SYNC_DONE || + this->state == TrueTypeLoaderState::SYNC_LOADING || + this->state == TrueTypeLoaderState::ASYNC_DONE + ) { + FT_Done_Face(face); + } + + FT_Done_FreeType(fontLibrary); + + if(buffer != nullptr) { + delete[] buffer; + buffer = nullptr; + } +} diff --git a/src/dawn/asset/loaders/TrueTypeLoader.hpp b/src/dawn/asset/loaders/TrueTypeLoader.hpp new file mode 100644 index 00000000..18e08977 --- /dev/null +++ b/src/dawn/asset/loaders/TrueTypeLoader.hpp @@ -0,0 +1,57 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "asset/AssetLoader.hpp" +#include "asset/AssetDataLoader.hpp" +#include "display/font/TrueTypeTexture.hpp" + +namespace Dawn { + enum class TrueTypeLoaderState { + INITIAL, + ASYNC_LOADING, + ASYNC_DONE, + SYNC_LOADING, + SYNC_DONE + }; + + class TrueTypeLoader : public AssetLoader { + protected: + FT_Library fontLibrary; + FT_Face face; + AssetDataLoader loader; + std::unordered_map> textures; + enum TrueTypeLoaderState state = TrueTypeLoaderState::INITIAL; + uint8_t *buffer = nullptr; + + public: + /** + * Constructs a TrueTypeLoader. You should instead use the parent + * asset managers' abstracted load method + * + * @param name File name asset to load, omitting the extension. + */ + TrueTypeLoader(const std::string name); + + void updateSync() override; + void updateAsync() override; + + /** + * Returns the texture for the given font size. + * + * @param fontSize Font size to get the texture for. + * @return Texture for the given character. + */ + std::shared_ptr getTexture( + const uint32_t fontSize + ); + + /** + * Dispose / Cleanup the truetype asset. Will also dispose the underlying + * truetype itself. + */ + ~TrueTypeLoader(); + }; +} diff --git a/src/dawn/display/CMakeLists.txt b/src/dawn/display/CMakeLists.txt index 3a6929fb..c33affdc 100644 --- a/src/dawn/display/CMakeLists.txt +++ b/src/dawn/display/CMakeLists.txt @@ -12,5 +12,6 @@ target_sources(${DAWN_TARGET_NAME} ) # Subdirs +add_subdirectory(font) add_subdirectory(mesh) add_subdirectory(shader) \ No newline at end of file diff --git a/src/dawn/ui/elements/CMakeLists.txt b/src/dawn/ui/elements/CMakeLists.txt index 8e377a9b..bff5eaf1 100644 --- a/src/dawn/ui/elements/CMakeLists.txt +++ b/src/dawn/ui/elements/CMakeLists.txt @@ -5,5 +5,6 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE + UILabel.cpp UIRectangle.cpp ) \ No newline at end of file diff --git a/src/dawn/ui/elements/UILabel.cpp b/src/dawn/ui/elements/UILabel.cpp new file mode 100644 index 00000000..31bb630b --- /dev/null +++ b/src/dawn/ui/elements/UILabel.cpp @@ -0,0 +1,175 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "UILabel.hpp" + +using namespace Dawn; + +void UILabel::getSelfQuads(UICanvas &ctx) { + std::vector quads; + if(this->texture == nullptr || this->text.empty()) return; + + glm::vec4 quad; + glm::vec2 pos = glm::vec2(0, this->texture->fontSize); + bool_t lastCharWasSpace = false; + + for(size_t i = 0; i < text.size(); i++) { + wchar_t c = text[i]; + auto info = texture->getCharacterData(c); + + // Newline(s) + if(c == L'\n') { + pos.x = 0; + pos.y += this->texture->fontSize; + continue; + } + + // Spaces + if(c == L' ') { + pos.x += info.advance.x; + lastCharWasSpace = true; + continue; + } + + // Word Wrap + if(wordWrap) { + if(lastCharWasSpace) { + // Scan ahead to next space + float_t wordWidth = pos.x;// Start at current position and scan ahead. + for(size_t j = i; j < text.size(); j++) { + wchar_t c2 = text[j]; + if(c2 == L' ' || c2 == L'\n') { + break;// If we hit another space, we are OK. + } + + // Will this character fit on the row? If not the whole word will wrap. + auto info2 = texture->getCharacterData(c); + wordWidth += info.advance.x; + if(wordWidth > size.x) { + pos.x = 0; + pos.y += this->texture->fontSize; + break; + } + } + + lastCharWasSpace = false; + } + // } else if(pos.x + info.size.x > subAlignedPosition.x + size.x) { + // // Not word wrap, but instead just overflow characters. + // pos.x = 0; + // pos.y += this->texture->fontSize; + } + + ctx.addQuad( + { + subAlignedPosition.x + pos.x + info.offset.x, + subAlignedPosition.y + pos.y + info.offset.y, + subAlignedPosition.x + pos.x + info.size.x + info.offset.x, + subAlignedPosition.y + pos.y + info.size.y + info.offset.y + }, + { + info.quad.x, + info.quad.y, + info.quad.z, + info.quad.w + }, + this->color, + UIShaderQuadStyle::FONT, + texture->texture + ); + pos += info.advance; + } +} + +float_t UILabel::getContentWidth() { + if(this->texture == nullptr || this->text.empty()) return 0.0f; + + float_t lineWidth = 0.0f; + float_t width = 0.0f; + for(wchar_t c : text) { + if(c == L'\n') { + width = Math::max(width, lineWidth); + lineWidth = 0.0f; + continue; + } + + auto info = texture->getCharacterData(c); + lineWidth += info.advance.x; + if( + this->hasExplicitWidth() && + lineWidth >= size.x + ) return size.x; + } + width = Math::max(width, lineWidth); + return width; +} + +float_t UILabel::getContentHeight() { + if(this->texture == nullptr || this->text.empty()) return 0.0f; + + float_t height = this->texture->fontSize; + float_t lineWidth = 0.0f; + bool_t lastCharWasSpace = false; + + for(wchar_t c : text) { + if(c == L'\n') { + height += this->texture->fontSize; + continue; + } + + auto info = texture->getCharacterData(c); + + if(c == L' ') { + lineWidth += info.advance.x; + lastCharWasSpace = true; + continue; + } + + if(wordWrap) { + if(lastCharWasSpace) { + // Scan ahead to next space + float_t wordWidth = lineWidth;// Start at current position and scan ahead. + for(size_t j = 0; j < text.size(); j++) { + wchar_t c2 = text[j]; + if(c2 == L' ' || c2 == L'\n') { + break;// If we hit another space, we are OK. + } + + // Will this character fit on the row? If not the whole word will wrap. + auto info2 = texture->getCharacterData(c); + wordWidth += info.advance.x; + if(wordWidth > size.x) { + height += this->texture->fontSize; + lineWidth = 0.0f; + break; + } + } + + lastCharWasSpace = false; + } + // } else if(lineWidth + info.size.x > size.x) { + // height += this->texture->fontSize; + // lineWidth = 0.0f; + } + } + + return height; +} + +std::shared_ptr UILabel::getFont() { + return this->texture; +} + +std::wstring UILabel::getText() { + return this->text; +} + +void UILabel::setFont(std::shared_ptr texture) { + this->texture = texture; +} + +void UILabel::setText(const std::wstring &text) { + this->text = text; +} \ No newline at end of file diff --git a/src/dawn/ui/elements/UILabel.hpp b/src/dawn/ui/elements/UILabel.hpp index 0b28d959..0205aa82 100644 --- a/src/dawn/ui/elements/UILabel.hpp +++ b/src/dawn/ui/elements/UILabel.hpp @@ -1,17 +1,54 @@ -// Copyright (c) 2024 Dominic Masters +// Copyright (c) 2023 Dominic Masters // // This software is released under the MIT License. // https://opensource.org/licenses/MIT #pragma once -#include "ui/UIAlignableElement.hpp" +#include "ui/UISubAlignableElement.hpp" +#include "display/font/TrueTypeTexture.hpp" namespace Dawn { - class UILabel final : public UIAlignableElement { + class UILabel final : public UISubAlignableElement { + private: + std::shared_ptr texture = nullptr; + std::wstring text = L"Hello World"; + protected: void getSelfQuads(UICanvas &ctx) override; public: - std::shared_ptr font; + bool_t wordWrap = true; + struct Color color = COLOR_WHITE; + + float_t getContentWidth() override; + float_t getContentHeight() override; + + /** + * Returns the font used for this label. + * + * @return The font used for this label. + */ + std::shared_ptr getFont(); + + /** + * Returns the text used for this label. + * + * @return The text used for this label. + */ + std::wstring getText(); + + /** + * Sets the font to use for this label. + * + * @param texture TrueType texture to use for this label. + */ + void setFont(std::shared_ptr texture); + + /** + * Sets the text to use for this label. + * + * @param text The text to use for this label. + */ + void setText(const std::wstring &text); }; -} \ No newline at end of file +} diff --git a/src/dawnpoker/CMakeLists.txt b/src/dawnrpg/CMakeLists.txt similarity index 75% rename from src/dawnpoker/CMakeLists.txt rename to src/dawnrpg/CMakeLists.txt index 9ae6262e..cc5d12a9 100644 --- a/src/dawnpoker/CMakeLists.txt +++ b/src/dawnrpg/CMakeLists.txt @@ -12,12 +12,11 @@ target_include_directories(${DAWN_TARGET_NAME} # Sources target_sources(${DAWN_TARGET_NAME} PRIVATE - dawnpoker.cpp + dawnrpg.cpp ) # Subdirs -add_subdirectory(poker) add_subdirectory(scenes) # Assets -# include("${DAWN_ASSETS_SOURCE_DIR}/games/dawnpoker/CMakeLists.txt") \ No newline at end of file +include("${DAWN_ASSETS_SOURCE_DIR}/dawnrpg/CMakeLists.txt") \ No newline at end of file diff --git a/src/dawnpoker/dawnpoker.cpp b/src/dawnrpg/dawnrpg.cpp similarity index 100% rename from src/dawnpoker/dawnpoker.cpp rename to src/dawnrpg/dawnrpg.cpp diff --git a/src/dawnpoker/scenes/CMakeLists.txt b/src/dawnrpg/scenes/CMakeLists.txt similarity index 100% rename from src/dawnpoker/scenes/CMakeLists.txt rename to src/dawnrpg/scenes/CMakeLists.txt diff --git a/src/dawnpoker/scenes/SceneList.hpp b/src/dawnrpg/scenes/SceneList.hpp similarity index 100% rename from src/dawnpoker/scenes/SceneList.hpp rename to src/dawnrpg/scenes/SceneList.hpp diff --git a/src/dawnpoker/scenes/TestScene.cpp b/src/dawnrpg/scenes/TestScene.cpp similarity index 85% rename from src/dawnpoker/scenes/TestScene.cpp rename to src/dawnrpg/scenes/TestScene.cpp index 3e38799d..2b5c90e9 100644 --- a/src/dawnpoker/scenes/TestScene.cpp +++ b/src/dawnrpg/scenes/TestScene.cpp @@ -12,7 +12,7 @@ #include "component/ui/UICanvas.hpp" #include "ui/elements/UIRectangle.hpp" -// #include "ui/elements/UILabel.hpp" +#include "ui/elements/UILabel.hpp" #include "ui/UIMenu.hpp" #include "ui/container/UIRowContainer.hpp" #include "ui/container/UIPaddingContainer.hpp" @@ -51,6 +51,19 @@ void Dawn::testScene(Scene &s) { rect->alignY = UIAlignmentType::START; rect->color = COLOR_MAGENTA; container->appendChild(rect); + + auto texture = s.getGame()->assetManager.get("font_silver", 128); + while(!s.getGame()->assetManager.isEverythingLoaded()) { + s.getGame()->assetManager.update(); + } + + auto label = std::make_shared(); + label->align = { 0, 0, UI_ALIGN_SIZE_AUTO, UI_ALIGN_SIZE_AUTO }; + label->alignX = UIAlignmentType::START; + label->alignY = UIAlignmentType::START; + label->setFont(texture); + label->setText(L"Hello World!"); + container->appendChild(label); // auto game = std::make_shared(); // auto player0 = game->addNewPlayer();