From dd1004257e86e4b62d69e6b73a409fd550932e7f Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Mon, 12 Jun 2023 08:32:12 -0700 Subject: [PATCH] Pretty much everything Label wise is working --- src/dawn/scene/components/ui/UILabelNew.cpp | 139 ++++++++---------- src/dawn/scene/components/ui/UILabelNew.hpp | 2 +- src/dawnliminal/CMakeLists.txt | 17 ++- src/dawnliminal/scenes/HelloWorldScene.hpp | 1 + src/dawnshared/CMakeLists.txt | 1 + src/dawnshared/display/Color.cpp | 40 +++++ src/{dawn => dawnshared}/display/Color.hpp | 12 +- src/dawnshared/util/Xml.hpp | 2 - src/dawnshared/util/parser/TypeParsers.hpp | 155 ++++++++++++++++++++ src/dawnshared/util/string.hpp | 30 ++++ 10 files changed, 306 insertions(+), 93 deletions(-) create mode 100644 src/dawnshared/display/Color.cpp rename src/{dawn => dawnshared}/display/Color.hpp (85%) create mode 100644 src/dawnshared/util/parser/TypeParsers.hpp diff --git a/src/dawn/scene/components/ui/UILabelNew.cpp b/src/dawn/scene/components/ui/UILabelNew.cpp index 6ba93053..72a0d8ee 100644 --- a/src/dawn/scene/components/ui/UILabelNew.cpp +++ b/src/dawn/scene/components/ui/UILabelNew.cpp @@ -17,88 +17,66 @@ UILabelNew::UILabelNew(SceneItem *item) : void UILabelNew::onStart() { this->shaderBuffer.init(); - auto font = this->getGame()->assetManager.get("font_arial"); - + std::vector styleStack; + struct UILabelStyle current; + styleStack.push_back(current); std::vector texts; - texts.push_back({ - .text = "Hello", - .style = { - .color = COLOR_RED, - .style = 0, - .size = 32, - .font = font + + std::function parseChildren = [&](Xml *node) { + if(node->children.empty()) { + struct UILabelText text; + text.style = current; + text.text = node->value; + texts.push_back(text); + } else { + auto itNode = node->children.begin(); + while(itNode != node->children.end()) { + auto child = *itNode; + assertTrue(child->node == "font"); + + struct UILabelStyle style; + if(child->attributes.contains("font")) { + style.font = this->getGame()->assetManager.get(child->attributes["font"]); + } else { + style.font = current.font; + } + + if(child->attributes.contains("size")) { + style.size = std::stoi(child->attributes["size"]); + } else { + style.size = current.size; + } + + if(child->attributes.contains("style")) { + std::string s = child->attributes["style"]; + style.style = 0; + if(s.find("bold") != std::string::npos) style.style |= NEW_TRUETYPE_VARIANT_BOLD; + if(s.find("italic") != std::string::npos) style.style |= NEW_TRUETYPE_VARIANT_ITALICS; + } else { + style.style = current.style; + } + + if(child->attributes.contains("color")) { + style.color = Color::fromString(child->attributes["color"]); + } else { + style.color = current.color; + } + + styleStack.push_back(style); + current = style; + + parseChildren(child); + + styleStack.pop_back(); + current = styleStack.back(); + ++itNode; + } } - }); - texts.push_back({ - .text = "World", - .style = { - .color = COLOR_BLUE, - .style = 1, - .size = 64, - .font = font - } - }); + }; + + auto root = Xml::load("" + this->test + ""); + parseChildren(&root); this->rebufferQuads(texts); - - // std::vector styleStack; - // struct UILabelStyle current; - // styleStack.push_back(current); - // std::vector texts; - - // std::function parseChildren = [&](Xml *node) { - // if(node->children.empty()) { - // struct UILabelText text; - // text.style = current; - // text.text = node->value; - // (node->value) - // } else { - // auto itNode = node->children.begin(); - // while(itNode != node->children.end()) { - // auto child = *itNode; - // std::cout << "Node: " << child->node << std::endl; - - // assertTrue(child->node == "font"); - - // struct UILabelStyle style; - // if(child->attributes.contains("font")) { - // style.font = this->getGame()->assetManager.get(child->attributes["font"]); - // } else { - // style.font = current.font; - // } - - // if(child->attributes.contains("size")) { - // style.size = std::stoi(child->attributes["size"]); - // } else { - // style.size = current.size; - // } - - // if(child->attributes.contains("style")) { - // style.style = std::stoi(child->attributes["style"]); - // } else { - // style.style = current.style; - // } - - // if(child->attributes.contains("color")) { - // style.color = std::stoi(child->attributes["color"]); - // } else { - // style.color = current.color; - // } - - // styleStack.push_back(style); - // current = style; - - // parseChildren(child); - - // styleStack.pop_back(); - // current = styleStack.back(); - // ++itNode; - // } - // } - // }; - - // auto root = Xml::load("" + this->test + ""); - // parseChildren(&root); - // this->rebufferQuads(texts); } std::vector UILabelNew::getUIRenderPasses() { @@ -165,11 +143,10 @@ float_t UILabelNew::getContentHeight() { } void UILabelNew::rebufferQuads(std::vector texts) { - std::cout << "Rebuffering" << std::endl; auto oldTexts = this->texts; textureMap.clear(); - glm::vec2 position(32, 32); + glm::vec2 position(0, 0); struct FontShaderBufferData fontData; int32_t quadIndex = 0; int32_t partIndex = 0; diff --git a/src/dawn/scene/components/ui/UILabelNew.hpp b/src/dawn/scene/components/ui/UILabelNew.hpp index efa93883..10ccebca 100644 --- a/src/dawn/scene/components/ui/UILabelNew.hpp +++ b/src/dawn/scene/components/ui/UILabelNew.hpp @@ -11,7 +11,7 @@ namespace Dawn { struct UILabelStyle { - struct Color color = COLOR_MAGENTA; + struct Color color = COLOR_WHITE; flag_t style = 0; uint32_t size = 16; NewTrueTypeAsset *font = nullptr; diff --git a/src/dawnliminal/CMakeLists.txt b/src/dawnliminal/CMakeLists.txt index 90fba1c8..33052676 100644 --- a/src/dawnliminal/CMakeLists.txt +++ b/src/dawnliminal/CMakeLists.txt @@ -24,14 +24,15 @@ tool_texture(texture_eth FILE=${LIMINAL_ASSETS_DIR}/textures/eth.png) tool_texture(texture_border FILE=${LIMINAL_ASSETS_DIR}/textures/texture_test.png) tool_newtruetype(font_arial - # REGULAR="/usr/share/fonts/TTF/arial.ttf" - # BOLD="/usr/share/fonts/TTF/arialbd.ttf" - # ITALICS="/usr/share/fonts/TTF/ariali.ttf" - # BOLD_ITALICS="/usr/share/fonts/TTF/arialbi.ttf" - REGULAR="C:\\Windows\\Fonts\\arial.ttf" - BOLD="C:\\Windows\\Fonts\\arialbd.ttf" - ITALICS="C:\\Windows\\Fonts\\ariali.ttf" - BOLD_ITALICS="C:\\Windows\\Fonts\\arialbi.ttf" + REGULAR="/usr/share/fonts/TTF/arial.ttf" + BOLD="/usr/share/fonts/TTF/arialbd.ttf" + ITALICS="/usr/share/fonts/TTF/ariali.ttf" + BOLD_ITALICS="/usr/share/fonts/TTF/arialbi.ttf" + + # REGULAR="C:\\Windows\\Fonts\\arial.ttf" + # BOLD="C:\\Windows\\Fonts\\arialbd.ttf" + # ITALICS="C:\\Windows\\Fonts\\ariali.ttf" + # BOLD_ITALICS="C:\\Windows\\Fonts\\arialbi.ttf" ) tool_scene(${LIMINAL_ASSETS_DIR}/scenes/SceneBase.xml) diff --git a/src/dawnliminal/scenes/HelloWorldScene.hpp b/src/dawnliminal/scenes/HelloWorldScene.hpp index 61f86e93..7df4e12e 100644 --- a/src/dawnliminal/scenes/HelloWorldScene.hpp +++ b/src/dawnliminal/scenes/HelloWorldScene.hpp @@ -27,6 +27,7 @@ namespace Dawn { auto newLabelItem = this->createSceneItem(); newLabelItem->transform.setParent(canvas->transform); auto newLabel = newLabelItem->addComponent(); + newLabel->test = "HelloWorld"; } std::vector getRequiredAssets() override { diff --git a/src/dawnshared/CMakeLists.txt b/src/dawnshared/CMakeLists.txt index dfc3c7de..c3b024a2 100644 --- a/src/dawnshared/CMakeLists.txt +++ b/src/dawnshared/CMakeLists.txt @@ -16,6 +16,7 @@ set( set(D ${CMAKE_CURRENT_LIST_DIR}) set( DAWN_SHARED_SOURCES + ${D}/display/Color.cpp ${D}/assert/assert.cpp ${D}/util/Xml.cpp ${D}/util/UsageLock.cpp diff --git a/src/dawnshared/display/Color.cpp b/src/dawnshared/display/Color.cpp new file mode 100644 index 00000000..75947ef7 --- /dev/null +++ b/src/dawnshared/display/Color.cpp @@ -0,0 +1,40 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "Color.hpp" +#include "util/parser/TypeParsers.hpp" + +using namespace Dawn; + +struct Color Color::fromString(std::string str) { + // Convert to lowercase + auto lower = stringToLowercase(str); + + if(stringIncludes(lower, "cornflower")) { + return COLOR_CORNFLOWER_BLUE; + + } else if(stringIncludes(lower, "magenta")) { + return COLOR_MAGENTA; + + } else if(stringIncludes(lower, "white")) { + return COLOR_WHITE; + + } else if(stringIncludes(lower, "black")) { + return COLOR_BLACK; + + } else if(stringIncludes(lower, "red")) { + return COLOR_RED; + + } else if(stringIncludes(lower, "green")) { + return COLOR_GREEN; + + } else if(stringIncludes(lower, "blue")) { + return COLOR_BLUE; + } + + // TODO: Parse other kinds of colors + assertUnreachable(); + return {}; +} \ No newline at end of file diff --git a/src/dawn/display/Color.hpp b/src/dawnshared/display/Color.hpp similarity index 85% rename from src/dawn/display/Color.hpp rename to src/dawnshared/display/Color.hpp index 2ed11cdf..31613be5 100644 --- a/src/dawn/display/Color.hpp +++ b/src/dawnshared/display/Color.hpp @@ -4,7 +4,8 @@ // https://opensource.org/licenses/MIT #pragma once -#include "dawnlibs.hpp" +#include "dawnsharedlibs.hpp" +#include "util/string.hpp" namespace Dawn { struct ColorU8 { @@ -13,6 +14,15 @@ namespace Dawn { #pragma pack(push, 4) struct Color { + /** + * Returns a color from a string. + * + * @param str String to parse. + * @return Color parsed. + */ + static struct Color fromString(std::string str); + + float_t r, g, b, a; struct Color operator * (const float_t &x) { diff --git a/src/dawnshared/util/Xml.hpp b/src/dawnshared/util/Xml.hpp index eb279a77..44fa415b 100644 --- a/src/dawnshared/util/Xml.hpp +++ b/src/dawnshared/util/Xml.hpp @@ -30,8 +30,6 @@ namespace Dawn { static Xml load(std::string data); static void load(Xml *xml, std::string data, size_t *j); - - std::string node; std::string value; std::map attributes; diff --git a/src/dawnshared/util/parser/TypeParsers.hpp b/src/dawnshared/util/parser/TypeParsers.hpp new file mode 100644 index 00000000..ee93d732 --- /dev/null +++ b/src/dawnshared/util/parser/TypeParsers.hpp @@ -0,0 +1,155 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "util/string.hpp" + +namespace Dawn { + static inline std::string rawParser(std::string v, std::string *error) { + return v; + }; + + static inline std::string stringParser(std::string v, std::string *error) { + return "\"" + v + "\""; + }; + + static inline std::string floatParser(std::string v, std::string *error) { + v = stringTrim(v); + + // Make sure number contains decimal + if(v.find(".") == std::string::npos) { + v += ".0"; + } + // Make sure number contains a number before the decimal + if(v.find(".") == 0) { + v = "0" + v; + } + // Make sure ends with f + if(v.find("f") == std::string::npos) { + v += "f"; + } + return v; + }; + + static inline std::string intParser(std::string v, std::string *error) { + v = stringTrim(v); + return v; + } + + static inline std::string boolParser(std::string v, std::string *error) { + v = stringTrim(v); + if(v == "true") return "true"; + if(v == "false") return "false"; + *error = "Invalid bool value: " + v; + return std::string(""); + } + + static inline std::string vec2Parser(std::string v, std::string *error) { + // Split string by comma into two strings that we pass into float + auto split = stringSplit(v, ","); + if(split.size() != 2) { + *error = "Invalid vec2 value: " + v; + return std::string(""); + } + return std::string( + "glm::vec2(" + + floatParser(split[0], error) + ", " + + floatParser(split[1], error) + + ")" + ); + }; + + + static inline std::string vec3Parser(std::string v, std::string *error) { + // Split string by comma into two strings that we pass into float + auto split = stringSplit(v, ","); + if(split.size() != 3) { + *error = "Invalid vec3 value: " + v; + return std::string(""); + } + return std::string( + "glm::vec3(" + + floatParser(split[0], error) + ", " + + floatParser(split[1], error) + ", " + + floatParser(split[2], error) + + ")" + ); + }; + + + static inline std::string vec6Parser(std::string v, std::string *error) { + // Split string by comma into two strings that we pass into float + auto split = stringSplit(v, ","); + if(split.size() != 6) { + *error = "Invalid vec6 value: " + v; + return std::string(""); + } + return std::string( + "glm::vec6(" + + floatParser(split[0], error) + ", " + + floatParser(split[1], error) + ", " + + floatParser(split[2], error) + ", " + + floatParser(split[3], error) + ", " + + floatParser(split[4], error) + ", " + + floatParser(split[5], error) + + ")" + ); + }; + + + static inline std::string vec4Parser(std::string v, std::string *error) { + // Split string by comma into two strings that we pass into float + auto split = stringSplit(v, ","); + if(split.size() != 4) { + *error = "Invalid vec4 value: " + v; + return std::string(""); + } + return std::string( + "glm::vec4(" + + floatParser(split[0], error) + ", " + + floatParser(split[1], error) + ", " + + floatParser(split[2], error) + ", " + + floatParser(split[3], error) + + ")" + ); + }; + + static inline std::string colorParser(std::string v, std::string *error) { + return rawParser(v, error); + } + + static inline std::function parserFromTypeName(std::string type) { + std::function parser = rawParser; + + if(type.find("string") != std::string::npos) { + parser = stringParser; + } else if(type.find("float") != std::string::npos) { + parser = floatParser; + } else if(type.find("Color") != std::string::npos) { + parser = colorParser; + } else if(type.find("vec2") != std::string::npos) { + parser = vec2Parser; + } else if(type.find("vec3") != std::string::npos) { + parser = vec3Parser; + } else if(type.find("vec4") != std::string::npos) { + parser = vec4Parser; + } else if(type == "int32_t" || type == "int") { + parser = intParser; + } else if(type == "bool_t") { + parser = boolParser; + } else if(type == "flag_t") { + parser = rawParser; + } else if(type.starts_with("enum")) { + parser = rawParser; + } else if(type.find("*") == (type.size() - 1)) { + type = type.substr(0, type.size() - 1); + parser = rawParser; + } else { + throw std::string("Invalid parser type"); + } + + return parser; + } +} \ No newline at end of file diff --git a/src/dawnshared/util/string.hpp b/src/dawnshared/util/string.hpp index 9894927b..13092e31 100644 --- a/src/dawnshared/util/string.hpp +++ b/src/dawnshared/util/string.hpp @@ -95,4 +95,34 @@ static inline std::string stringRTrim(const std::string &i) { */ static inline std::string stringTrim(const std::string &s) { return stringLTrim(stringRTrim(s)); +} + +/** + * Checks if a string contains another string. + * + * @param haystack String to scan. + * @param needle String to search for. + * @return True if the string is found, false otherwise. + */ +static inline bool_t stringIncludes(const std::string &haystack, const std::string &needle) { + return haystack.find(needle) != std::string::npos; +} + +/** + * Converts an entire string to lowercase. + * + * @param str String to convert. + * @return A new string with all lowercase characters. + */ +static inline std::string stringToLowercase(const std::string &str) { + std::string data = str; + std::transform( + data.begin(), + data.end(), + data.begin(), + [](char c) { + return std::tolower(c); + } + ); + return data; } \ No newline at end of file