Prefab tool more-or-less done

This commit is contained in:
2023-03-22 21:44:31 -07:00
parent 97b228e0a7
commit 3782e731b2
4 changed files with 264 additions and 3 deletions

View File

@ -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}
)

View File

@ -8,7 +8,7 @@
using namespace Dawn;
std::vector<std::string> PrefabTool::getRequiredFlags() {
return { "input" };
return { "input", "output" };
}
std::map<std::string, std::string> 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<std::string> 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<std::string> PrefabParser::getRequiredAttributes() {
return { "name", "type" };
}
std::map<std::string, std::string> PrefabParser::getOptionalAttributes() {
return { };
}
int32_t PrefabParser::onParse(
Xml *node,
std::map<std::string, std::string> 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<std::string> PrefabComponentParser::getRequiredAttributes() {
return { };
}
std::map<std::string, std::string> PrefabComponentParser::getOptionalAttributes() {
return { { "ref", "" } };
}
int32_t PrefabComponentParser::onParse(
Xml *node,
std::map<std::string, std::string> 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<std::string(std::string)> stringParser = [&](std::string str) {
return "\"" + str + "\"";
};
// Handle different node types!
std::map<std::string, struct PrefabComponentParserRuleset> 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<std::string> *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<Asset*>";
methodAssets.args = "AssetManager *man";
line(&methodAssets.body, "std::vector<Asset*> 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);
}

View File

@ -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<std::string, std::function<std::string(std::string)>> optional;
};
struct PrefabComponent {
std::string include;
std::string type;
std::map<std::string, std::string> values;
std::string ref;
};
struct Prefab {
std::string name;
std::string type;
std::vector<struct PrefabComponent> components;
};
class PrefabTool : public DawnTool {
protected:
std::vector<std::string> getRequiredFlags() override;
@ -16,4 +36,37 @@ namespace Dawn {
public:
int32_t start();
};
class PrefabParser : public XmlParser<struct Prefab> {
protected:
std::vector<std::string> getRequiredAttributes() override;
std::map<std::string, std::string> getOptionalAttributes() override;
int32_t onParse(
Xml *node,
std::map<std::string, std::string> values,
struct Prefab *out,
std::string *error
) override;
};
class PrefabComponentParser : public XmlParser<struct PrefabComponent> {
protected:
std::vector<std::string> getRequiredAttributes() override;
std::map<std::string, std::string> getOptionalAttributes() override;
int32_t onParse(
Xml *node,
std::map<std::string, std::string> values,
struct PrefabComponent *out,
std::string *error
) override;
};
class PrefabGen : public CodeGen {
public:
static void generate(
std::vector<std::string> *out,
struct Prefab *prefab,
std::string tabs
);
};
}

View File

@ -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<std::string> 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) {