From 3782e731b2ba114d0917248d7818423bd718f937 Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Wed, 22 Mar 2023 21:44:31 -0700 Subject: [PATCH] Prefab tool more-or-less done --- src/dawntools/prefabtool/CMakeLists.txt | 2 +- src/dawntools/prefabtool/PrefabTool.cpp | 195 +++++++++++++++++++++++- src/dawntools/prefabtool/PrefabTool.hpp | 53 +++++++ src/dawntools/util/CodeGen.cpp | 17 ++- 4 files changed, 264 insertions(+), 3 deletions(-) diff --git a/src/dawntools/prefabtool/CMakeLists.txt b/src/dawntools/prefabtool/CMakeLists.txt index 642a4577..77fc2b25 100644 --- a/src/dawntools/prefabtool/CMakeLists.txt +++ b/src/dawntools/prefabtool/CMakeLists.txt @@ -44,7 +44,7 @@ function(tool_prefab in) endif() add_custom_target(prefab_${in} - COMMAND prefabtool --input="${DAWN_ASSETS_SOURCE_DIR}/${in}" + COMMAND prefabtool --input="${DAWN_ASSETS_SOURCE_DIR}/${in}" --output="${DAWN_TEMP_DIR}" COMMENT "Generating prefab from ${in}" DEPENDS ${DEPS} ) diff --git a/src/dawntools/prefabtool/PrefabTool.cpp b/src/dawntools/prefabtool/PrefabTool.cpp index fa32555d..01df510a 100644 --- a/src/dawntools/prefabtool/PrefabTool.cpp +++ b/src/dawntools/prefabtool/PrefabTool.cpp @@ -8,7 +8,7 @@ using namespace Dawn; std::vector PrefabTool::getRequiredFlags() { - return { "input" }; + return { "input", "output" }; } std::map PrefabTool::getOptionalFlags() { @@ -22,5 +22,198 @@ int32_t PrefabTool::start() { return 1; } + std::string data; + if(!input.readString(&data)) return 1; + + auto xml = Xml::load(data); + std::string error; + struct Prefab prefab; + auto result = ((PrefabParser()).parse(&xml, &prefab, &error)); + if(result != 0) return result; + + // Generate output + std::vector outputData; + PrefabGen::generate(&outputData, &prefab, ""); + + // Load output file from name and type + File outputFile = File(flags["output"] + "/prefabs/" + prefab.type + "/" + prefab.name + ".hpp"); + if(!outputFile.mkdirp()) { + std::cout << "Failed to create output directory!" << std::endl; + return 1; + } + + // Combine vector into single string + std::string outputDataStr = ""; + auto it = outputData.begin(); + while(it != outputData.end()) { + outputDataStr += *it + "\n"; + ++it; + } + + if(!outputFile.writeString(outputDataStr)) { + std::cout << "Failed to write output file!" << std::endl; + return 1; + } + return 0; +} + +// + +std::vector PrefabParser::getRequiredAttributes() { + return { "name", "type" }; +} + +std::map PrefabParser::getOptionalAttributes() { + return { }; +} + +int32_t PrefabParser::onParse( + Xml *node, + std::map values, + struct Prefab *out, + std::string *error +) { + out->name = values["name"]; + out->type = values["type"]; + + auto itChildren = node->children.begin(); + while(itChildren != node->children.end()) { + // Parse children as components + struct PrefabComponent component; + auto ret = (PrefabComponentParser()).parse(*itChildren, &component, error); + if(ret != 0) return ret; + out->components.push_back(component); + ++itChildren; + } + + return 0; +} + +// + +std::vector PrefabComponentParser::getRequiredAttributes() { + return { }; +} + +std::map PrefabComponentParser::getOptionalAttributes() { + return { { "ref", "" } }; +} + +int32_t PrefabComponentParser::onParse( + Xml *node, + std::map values, + struct PrefabComponent *out, + std::string *error +) { + // Check if values contains key "ref" and has a string of size more than 0 + if(values.find("ref") != values.end() && values["ref"].size() > 0) { + out->ref = values["ref"]; + } + + std::function stringParser = [&](std::string str) { + return "\"" + str + "\""; + }; + + // Handle different node types! + std::map attributesForTypes = { + { "UILabel", { + .include = "scene/components/ui/UILabel.hpp", + .optional = { + { "text", stringParser } + } + } } + }; + + // Check if the node type is in the map + if(attributesForTypes.find(node->node) != attributesForTypes.end()) { + // Get the ruleset for this node type + auto ruleset = attributesForTypes[node->node]; + out->include = ruleset.include; + out->type = node->node; + + // Iterate all the optional attributes and store within the out if present + auto itOptional = ruleset.optional.begin(); + while(itOptional != ruleset.optional.end()) { + if(node->attributes.find(itOptional->first) != node->attributes.end()) { + auto raw = node->attributes[itOptional->first]; + auto parsed = itOptional->second(raw); + out->values[itOptional->first] = parsed; + } + ++itOptional; + } + + // TODO: Handle required attributes + } else { + *error = "Unknown node type: " + node->node; + return 1; + } + + return 0; +} + +// + +void PrefabGen::generate( + std::vector *out, + struct Prefab *info, + std::string tabs +) { + struct ClassGenInfo classInfo; + classInfo.clazz = info->name; + classInfo.extend = "SceneItemPrefab<" + info->name + ">"; + classInfo.constructorArgs = "Scene *scene, sceneid_t id"; + classInfo.extendArgs = "scene, id"; + + struct MethodGenInfo methodAssets; + methodAssets.name = "prefabAssets"; + methodAssets.isStatic = true; + methodAssets.type = "std::vector"; + methodAssets.args = "AssetManager *man"; + line(&methodAssets.body, "std::vector assets;", ""); + + struct MethodGenInfo methodInit; + methodInit.name = "prefabInit"; + methodInit.isOverride = true; + methodInit.args = "AssetManager *man"; + + classInfo.includes.push_back("prefab/SceneItemPrefab.hpp"); + + // Includes + int32_t componentNumber = 0; + auto itChildren = info->components.begin(); + while(itChildren != info->components.end()) { + auto c = *itChildren; + std::string name = "c" + std::to_string(componentNumber++); + bool_t init = true; + if(c.ref.size() > 0) { + init = false; + name = c.ref; + line(&classInfo.publicProperties, c.type + " *" + c.ref + ";", ""); + } + + // Initialize + line(&methodInit.body, (init ? "auto " : "") + name + " = this->addComponent<" + c.type + ">();", ""); + + // Now set each property + auto itValues = c.values.begin(); + while(itValues != c.values.end()) { + line(&methodInit.body, name + "->" + itValues->first + " = " + itValues->second + ";", ""); + ++itValues; + } + line(&methodInit.body, "", ""); + + classInfo.includes.push_back(c.include); + ++itChildren; + } + + // Seal methods + line(&methodAssets.body, "return assets;", ""); + + // Add in methods + CodeGen::methodGen(&classInfo.publicCode, methodAssets); + line(&classInfo.publicCode, "", ""); + CodeGen::methodGen(&classInfo.publicCode, methodInit); + + CodeGen::classGen(out, classInfo); } \ No newline at end of file diff --git a/src/dawntools/prefabtool/PrefabTool.hpp b/src/dawntools/prefabtool/PrefabTool.hpp index debbd0a4..46a8dbf2 100644 --- a/src/dawntools/prefabtool/PrefabTool.hpp +++ b/src/dawntools/prefabtool/PrefabTool.hpp @@ -6,8 +6,28 @@ #pragma once #include "util/DawnTool.hpp" #include "util/File.hpp" +#include "util/XmlParser.hpp" +#include "util/CodeGen.hpp" namespace Dawn { + struct PrefabComponentParserRuleset { + std::string include; + std::map> optional; + }; + + struct PrefabComponent { + std::string include; + std::string type; + std::map values; + std::string ref; + }; + + struct Prefab { + std::string name; + std::string type; + std::vector components; + }; + class PrefabTool : public DawnTool { protected: std::vector getRequiredFlags() override; @@ -16,4 +36,37 @@ namespace Dawn { public: int32_t start(); }; + + class PrefabParser : public XmlParser { + protected: + std::vector getRequiredAttributes() override; + std::map getOptionalAttributes() override; + int32_t onParse( + Xml *node, + std::map values, + struct Prefab *out, + std::string *error + ) override; + }; + + class PrefabComponentParser : public XmlParser { + protected: + std::vector getRequiredAttributes() override; + std::map getOptionalAttributes() override; + int32_t onParse( + Xml *node, + std::map values, + struct PrefabComponent *out, + std::string *error + ) override; + }; + + class PrefabGen : public CodeGen { + public: + static void generate( + std::vector *out, + struct Prefab *prefab, + std::string tabs + ); + }; } \ No newline at end of file diff --git a/src/dawntools/util/CodeGen.cpp b/src/dawntools/util/CodeGen.cpp index 1e98c9d8..a0d90dc6 100644 --- a/src/dawntools/util/CodeGen.cpp +++ b/src/dawntools/util/CodeGen.cpp @@ -35,10 +35,25 @@ void CodeGen::classGen( line(out, "#pragma once", ""); line(out, "", ""); + + // Includes if(info.includes.size() > 0) { - lines(out, info.includes, ""); + // iterate over info.includes + std::vector included; + auto itInclude = info.includes.begin(); + while(itInclude != info.includes.end()) { + // skip if already included + if(std::find(included.begin(), included.end(), *itInclude) != included.end()) { + ++itInclude; + continue; + } + line(out, "#include \"" + *itInclude + "\"", ""); + included.push_back(*itInclude); + ++itInclude; + } line(out, "", ""); } + line(out, "namespace Dawn {", ""); line(out, "class " + info.clazz + (info.extend.size() == 0 ? "{" : " : public " + info.extend + " {" ), " "); if(info.protectedCode.size() > 0) {