Created first version of UI Generator
This commit is contained in:
@ -1,58 +1,58 @@
|
|||||||
// Copyright (c) 2022 Dominic Masters
|
// Copyright (c) 2022 Dominic Masters
|
||||||
//
|
//
|
||||||
// This software is released under the MIT License.
|
// This software is released under the MIT License.
|
||||||
// https://opensource.org/licenses/MIT
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
#include "SimpleVNScene.hpp"
|
#include "SimpleVNScene.hpp"
|
||||||
#include "prefabs/ui/VisualNovelTextboxPrefab.hpp"
|
#include "prefabs/ui/VisualNovelTextboxPrefab.hpp"
|
||||||
|
|
||||||
using namespace Dawn;
|
using namespace Dawn;
|
||||||
|
|
||||||
SimpleVNScene::SimpleVNScene(DawnGame *game) : Scene(game) {
|
SimpleVNScene::SimpleVNScene(DawnGame *game) : Scene(game) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
std::vector<Asset*> SimpleVNScene::getRequiredAssets() {
|
std::vector<Asset*> SimpleVNScene::getRequiredAssets() {
|
||||||
auto assMan = &this->game->assetManager;
|
auto assMan = &this->game->assetManager;
|
||||||
std::vector<Asset*> assets;
|
std::vector<Asset*> assets;
|
||||||
vectorAppend(&assets, VisualNovelTextboxPrefab::getRequiredAssets(assMan));
|
vectorAppend(&assets, VisualNovelTextboxPrefab::getRequiredAssets(assMan));
|
||||||
return assets;
|
return assets;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SimpleVNScene::stage() {
|
void SimpleVNScene::stage() {
|
||||||
auto assMan = &this->game->assetManager;
|
auto assMan = &this->game->assetManager;
|
||||||
|
|
||||||
// Camera
|
// Camera
|
||||||
this->camera = Camera::create(this);
|
this->camera = Camera::create(this);
|
||||||
this->camera->transform->lookAt(
|
this->camera->transform->lookAt(
|
||||||
glm::vec3(0, 0, 2),
|
glm::vec3(0, 0, 2),
|
||||||
glm::vec3(0, 0, 0)
|
glm::vec3(0, 0, 0)
|
||||||
);
|
);
|
||||||
|
|
||||||
this->background = SimpleVisualNovelBackground::create(this);
|
this->background = SimpleVisualNovelBackground::create(this);
|
||||||
|
this->canvas = UICanvas::create(this);
|
||||||
// Stage VN Items
|
|
||||||
this->vnStage();
|
// Stage VN Items
|
||||||
|
this->vnStage();
|
||||||
// UI
|
|
||||||
this->canvas = UICanvas::create(this);
|
// UI
|
||||||
this->textbox = VisualNovelTextboxPrefab::create(this->canvas);
|
this->textbox = VisualNovelTextboxPrefab::create(this->canvas);
|
||||||
|
|
||||||
// VN Manager
|
// VN Manager
|
||||||
auto vnManagerItem = this->createSceneItem();
|
auto vnManagerItem = this->createSceneItem();
|
||||||
this->vnManager = vnManagerItem->addComponent<VisualNovelManager>();
|
this->vnManager = vnManagerItem->addComponent<VisualNovelManager>();
|
||||||
|
|
||||||
// Audio
|
// Audio
|
||||||
auto listenerItem = this->createSceneItem();
|
auto listenerItem = this->createSceneItem();
|
||||||
this->audioListener = listenerItem->addComponent<AudioListener>();
|
this->audioListener = listenerItem->addComponent<AudioListener>();
|
||||||
|
|
||||||
auto audioBackgroundItem = this->createSceneItem();
|
auto audioBackgroundItem = this->createSceneItem();
|
||||||
vnManager->audioBackground = audioBackgroundItem->addComponent<AudioSource>();
|
vnManager->audioBackground = audioBackgroundItem->addComponent<AudioSource>();
|
||||||
|
|
||||||
auto audioCharacterItem = this->createSceneItem();
|
auto audioCharacterItem = this->createSceneItem();
|
||||||
vnManager->audioCharacter = audioCharacterItem->addComponent<AudioSource>();
|
vnManager->audioCharacter = audioCharacterItem->addComponent<AudioSource>();
|
||||||
|
|
||||||
// Fader (Drawn over the top of everything else)
|
// Fader (Drawn over the top of everything else)
|
||||||
this->vnFader = VisualNovelFader::create(canvas);
|
this->vnFader = VisualNovelFader::create(canvas);
|
||||||
}
|
}
|
@ -39,6 +39,8 @@ tool_truetype(truetype_alice ${DIR_GAME_ASSETS}/font/Alice-Regular.ttf truetype_
|
|||||||
|
|
||||||
tool_audio(audio_test borrowed/sample_short.wav)
|
tool_audio(audio_test borrowed/sample_short.wav)
|
||||||
|
|
||||||
|
tool_ui(ui_test ${DIR_GAME_ASSETS}/ui/uitest.xml)
|
||||||
|
|
||||||
add_dependencies(${DAWN_TARGET_NAME}
|
add_dependencies(${DAWN_TARGET_NAME}
|
||||||
language_en
|
language_en
|
||||||
|
|
||||||
@ -50,4 +52,6 @@ add_dependencies(${DAWN_TARGET_NAME}
|
|||||||
texture_test
|
texture_test
|
||||||
|
|
||||||
audio_test
|
audio_test
|
||||||
|
|
||||||
|
ui_test
|
||||||
)
|
)
|
@ -1,59 +1,64 @@
|
|||||||
// Copyright (c) 2022 Dominic Masters
|
// Copyright (c) 2022 Dominic Masters
|
||||||
//
|
//
|
||||||
// This software is released under the MIT License.
|
// This software is released under the MIT License.
|
||||||
// https://opensource.org/licenses/MIT
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "visualnovel/scene/SimpleVNScene.hpp"
|
#include "visualnovel/scene/SimpleVNScene.hpp"
|
||||||
#include "scenes/Scene_2.hpp"
|
#include "scenes/Scene_2.hpp"
|
||||||
#include "prefabs/characters/DeathPrefab.hpp"
|
#include "prefabs/characters/DeathPrefab.hpp"
|
||||||
#include "visualnovel/events/characters/VisualNovelFadeCharacterEvent.hpp"
|
#include "visualnovel/events/characters/VisualNovelFadeCharacterEvent.hpp"
|
||||||
#include "visualnovel/events/characters/VisualNovelTransformItemEvent.hpp"
|
#include "visualnovel/events/characters/VisualNovelTransformItemEvent.hpp"
|
||||||
#include "visualnovel/events/timing/VisualNovelBatchEvent.hpp"
|
#include "visualnovel/events/timing/VisualNovelBatchEvent.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
#include "ui_test.hpp"
|
||||||
class Scene_1 : public SimpleVNScene {
|
|
||||||
protected:
|
namespace Dawn {
|
||||||
DeathPrefab *death;
|
class Scene_1 : public SimpleVNScene {
|
||||||
|
protected:
|
||||||
void vnStage() override {
|
DeathPrefab *death;
|
||||||
|
|
||||||
this->death = DeathPrefab::create(this);
|
void vnStage() override {
|
||||||
this->death->material->color.a = 0;
|
|
||||||
// this->death->transform.setLocalPosition(glm::vec3(-100, 0, 0));
|
this->death = DeathPrefab::create(this);
|
||||||
}
|
this->death->material->color.a = 0;
|
||||||
|
// this->death->transform.setLocalPosition(glm::vec3(-100, 0, 0));
|
||||||
void onSceneEnded() {
|
|
||||||
auto scene = new Scene_2(this->game);
|
auto item = this->canvas->addElement<UITest>();
|
||||||
game->assetManager.queueSwap(
|
item->setTransform(UI_COMPONENT_ALIGN_STRETCH, UI_COMPONENT_ALIGN_STRETCH, glm::vec4(0, 0, 0, 0), 0.0f);
|
||||||
scene->getRequiredAssets(), this->getRequiredAssets()
|
}
|
||||||
);
|
|
||||||
game->assetManager.syncLoad();
|
void onSceneEnded() {
|
||||||
scene->stage();
|
auto scene = new Scene_2(this->game);
|
||||||
this->game->sceneCutover(scene);
|
game->assetManager.queueSwap(
|
||||||
}
|
scene->getRequiredAssets(), this->getRequiredAssets()
|
||||||
|
);
|
||||||
public:
|
game->assetManager.syncLoad();
|
||||||
Scene_1(DawnGame *game) : SimpleVNScene(game) {
|
scene->stage();
|
||||||
|
this->game->sceneCutover(scene);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<Asset*> getRequiredAssets() override {
|
public:
|
||||||
auto man = &this->game->assetManager;
|
Scene_1(DawnGame *game) : SimpleVNScene(game) {
|
||||||
std::vector<Asset*> assets = SimpleVNScene::getRequiredAssets();
|
|
||||||
vectorAppend(&assets, DeathPrefab::getRequiredAssets(man));
|
}
|
||||||
assets.push_back(man->get<AudioAsset>("audio_test"));
|
|
||||||
return assets;
|
std::vector<Asset*> getRequiredAssets() override {
|
||||||
}
|
auto man = &this->game->assetManager;
|
||||||
|
std::vector<Asset*> assets = SimpleVNScene::getRequiredAssets();
|
||||||
IVisualNovelEvent * getVNEvent() override {
|
vectorAppend(&assets, DeathPrefab::getRequiredAssets(man));
|
||||||
auto start = new VisualNovelPauseEvent(vnManager, 0.1f);
|
assets.push_back(man->get<AudioAsset>("audio_test"));
|
||||||
start
|
return assets;
|
||||||
->then(new VisualNovelTextboxEvent(vnManager, this->death->vnCharacter, this->death->emotionHappy, "scene.1.1"))
|
}
|
||||||
->then(new VisualNovelFadeCharacterEvent(vnManager, this->death->vnCharacter, true, &easeOutQuad, 1.0f))
|
|
||||||
// ->then(new VisualNovelCallbackEvent<Scene_1>(vnManager, this, &Scene_1::onSceneEnded))
|
IVisualNovelEvent * getVNEvent() override {
|
||||||
;
|
auto start = new VisualNovelPauseEvent(vnManager, 0.1f);
|
||||||
return start;
|
start
|
||||||
}
|
->then(new VisualNovelTextboxEvent(vnManager, this->death->vnCharacter, this->death->emotionHappy, "scene.1.1"))
|
||||||
};
|
->then(new VisualNovelFadeCharacterEvent(vnManager, this->death->vnCharacter, true, &easeOutQuad, 1.0f))
|
||||||
|
// ->then(new VisualNovelCallbackEvent<Scene_1>(vnManager, this, &Scene_1::onSceneEnded))
|
||||||
|
;
|
||||||
|
return start;
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
@ -1,36 +1,50 @@
|
|||||||
# Copyright (c) 2021 Dominic Msters
|
# Copyright (c) 2021 Dominic Msters
|
||||||
#
|
#
|
||||||
# This software is released under the MIT License.
|
# This software is released under the MIT License.
|
||||||
# https://opensource.org/licenses/MIT
|
# https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
add_subdirectory(texturegen)
|
add_subdirectory(texturegen)
|
||||||
add_subdirectory(tilesetgen)
|
add_subdirectory(tilesetgen)
|
||||||
add_subdirectory(truetypegen)
|
add_subdirectory(truetypegen)
|
||||||
|
add_subdirectory(uigen)
|
||||||
# Texture Tool
|
|
||||||
function(tool_texture target in)
|
# Texture Tool
|
||||||
add_custom_target(${target}
|
function(tool_texture target in)
|
||||||
COMMAND texturegen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${target}"
|
add_custom_target(${target}
|
||||||
COMMENT "Generating texture ${target} from ${in}"
|
COMMAND texturegen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${target}"
|
||||||
DEPENDS texturegen
|
COMMENT "Generating texture ${target} from ${in}"
|
||||||
)
|
DEPENDS texturegen
|
||||||
endfunction()
|
)
|
||||||
|
endfunction()
|
||||||
# Tileset Tool
|
|
||||||
function(tool_tileset targetTileset targetTexture in cols rows)
|
# Tileset Tool
|
||||||
tool_texture(${targetTexture} ${in})
|
function(tool_tileset targetTileset targetTexture in cols rows)
|
||||||
add_custom_target(${targetTileset}
|
tool_texture(${targetTexture} ${in})
|
||||||
COMMAND tilesetgen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${targetTileset}" "${cols}" "${rows}"
|
add_custom_target(${targetTileset}
|
||||||
COMMENT "Generating tileset ${target} from ${in}"
|
COMMAND tilesetgen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${targetTileset}" "${cols}" "${rows}"
|
||||||
DEPENDS tilesetgen ${targetTexture}
|
COMMENT "Generating tileset ${target} from ${in}"
|
||||||
)
|
DEPENDS tilesetgen ${targetTexture}
|
||||||
endfunction()
|
)
|
||||||
|
endfunction()
|
||||||
# TrueType Tool
|
|
||||||
function(tool_truetype target in out width height fontSize)
|
# TrueType Tool
|
||||||
add_custom_target(${target}
|
function(tool_truetype target in out width height fontSize)
|
||||||
COMMAND truetypegen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${out}" "${width}" "${height}" "${fontSize}"
|
add_custom_target(${target}
|
||||||
COMMENT "Generating truetype ${target} from ${in}"
|
COMMAND truetypegen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${out}" "${width}" "${height}" "${fontSize}"
|
||||||
DEPENDS truetypegen
|
COMMENT "Generating truetype ${target} from ${in}"
|
||||||
)
|
DEPENDS truetypegen
|
||||||
|
)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
# UI Tool
|
||||||
|
function(tool_ui target in)
|
||||||
|
add_custom_target(${target}
|
||||||
|
COMMAND uigen "${DAWN_ASSETS_SOURCE_DIR}/${in}" "${DAWN_ASSETS_BUILD_DIR}/${target}"
|
||||||
|
COMMENT "Generating ui ${target} from ${in}"
|
||||||
|
DEPENDS uigen
|
||||||
|
)
|
||||||
|
target_include_directories(${DAWN_TARGET_NAME}
|
||||||
|
PUBLIC
|
||||||
|
${DAWN_ASSETS_BUILD_DIR}
|
||||||
|
)
|
||||||
endfunction()
|
endfunction()
|
23
src/dawntools/display/uigen/CMakeLists.txt
Normal file
23
src/dawntools/display/uigen/CMakeLists.txt
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
# Copyright (c) 2021 Dominic Msters
|
||||||
|
#
|
||||||
|
# This software is released under the MIT License.
|
||||||
|
# https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
|
# Texture Build Tool
|
||||||
|
project(uigen VERSION 1.0)
|
||||||
|
add_executable(uigen)
|
||||||
|
target_sources(uigen
|
||||||
|
PRIVATE
|
||||||
|
main.cpp
|
||||||
|
../../utils/file.c
|
||||||
|
../../utils/xml.c
|
||||||
|
)
|
||||||
|
target_include_directories(uigen
|
||||||
|
PUBLIC
|
||||||
|
${CMAKE_CURRENT_LIST_DIR}/../../
|
||||||
|
${CMAKE_CURRENT_LIST_DIR}
|
||||||
|
)
|
||||||
|
target_link_libraries(uigen
|
||||||
|
PUBLIC
|
||||||
|
${DAWN_BUILD_HOST_LIBS}
|
||||||
|
)
|
347
src/dawntools/display/uigen/main.cpp
Normal file
347
src/dawntools/display/uigen/main.cpp
Normal file
@ -0,0 +1,347 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2023 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
#include "../../utils/xml.h"
|
||||||
|
#include "../../utils/file.h"
|
||||||
|
#include <memory.h>
|
||||||
|
}
|
||||||
|
#include <iostream>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
struct UIGenerated {
|
||||||
|
bool align = false;
|
||||||
|
bool alignGrid = false;
|
||||||
|
std::string alignX;
|
||||||
|
std::string alignY;
|
||||||
|
std::string align0;
|
||||||
|
std::string align1;
|
||||||
|
std::string align2;
|
||||||
|
std::string align3;
|
||||||
|
|
||||||
|
bool isGrid = false;
|
||||||
|
std::string rows;
|
||||||
|
std::string columns;
|
||||||
|
std::string gutterX;
|
||||||
|
std::string gutterY;
|
||||||
|
|
||||||
|
|
||||||
|
std::string type;
|
||||||
|
std::string name;
|
||||||
|
|
||||||
|
std::string parent;
|
||||||
|
|
||||||
|
// Optionals
|
||||||
|
bool hasColor = false;
|
||||||
|
std::string color;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<std::string> split(std::string s, std::string delimiter) {
|
||||||
|
size_t pos_start = 0, pos_end, delim_len = delimiter.length();
|
||||||
|
std::string token;
|
||||||
|
std::vector<std::string> res;
|
||||||
|
|
||||||
|
while ((pos_end = s.find (delimiter, pos_start)) != std::string::npos) {
|
||||||
|
token = s.substr (pos_start, pos_end - pos_start);
|
||||||
|
pos_start = pos_end + delim_len;
|
||||||
|
res.push_back(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
res.push_back (s.substr (pos_start));
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string alignmentFromRaw(std::string strRaw) {
|
||||||
|
if(strRaw == "start") return "UI_COMPONENT_ALIGN_START";
|
||||||
|
if(strRaw == "middle") return "UI_COMPONENT_ALIGN_MIDDLE";
|
||||||
|
if(strRaw == "end") return "UI_COMPONENT_ALIGN_END";
|
||||||
|
if(strRaw == "stretch") return "UI_COMPONENT_ALIGN_STRETCH";
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool parseChildren(
|
||||||
|
xml_t *currentNode,
|
||||||
|
std::string parent,
|
||||||
|
std::vector<struct UIGenerated> *items
|
||||||
|
) {
|
||||||
|
// Confirm attributes
|
||||||
|
auto attrName = xmlGetAttributeByName(currentNode, "name");
|
||||||
|
if(attrName == -1) {
|
||||||
|
std::cout << "Missing name attribute." << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UIGenerated item;
|
||||||
|
item.name = std::string(currentNode->attributeDatas[attrName] );
|
||||||
|
item.type = std::string(currentNode->node);
|
||||||
|
item.parent = parent;
|
||||||
|
|
||||||
|
// Standard Align
|
||||||
|
auto attrAlign = xmlGetAttributeByName(currentNode, "align");
|
||||||
|
if(attrAlign != -1) {
|
||||||
|
// Parse alignment
|
||||||
|
std::string alignRaw(currentNode->attributeDatas[attrAlign]);
|
||||||
|
std::vector<std::string> alignmentParts = split(alignRaw, " ");
|
||||||
|
if(alignmentParts.size() != 6) {
|
||||||
|
std::cout << "Alignment is invalid" << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
item.align = true;
|
||||||
|
item.alignX = alignmentFromRaw(alignmentParts[0]);
|
||||||
|
item.alignY = alignmentFromRaw(alignmentParts[1]);
|
||||||
|
item.align0 = alignmentParts[2];
|
||||||
|
item.align1 = alignmentParts[3];
|
||||||
|
item.align2 = alignmentParts[4];
|
||||||
|
item.align3 = alignmentParts[5];
|
||||||
|
|
||||||
|
if(item.alignX.size() == 0) {
|
||||||
|
std::cout << "X Align is invalid" << std::endl;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
if(item.alignY.size() == 0) {
|
||||||
|
std::cout << "Y Align is invalid" << std::endl;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Grid Align
|
||||||
|
auto attrGridAlign = xmlGetAttributeByName(currentNode, "grid");
|
||||||
|
if(attrGridAlign != -1) {
|
||||||
|
// Parse alignment
|
||||||
|
std::string alignRaw(currentNode->attributeDatas[attrGridAlign]);
|
||||||
|
std::vector<std::string> alignmentParts = split(alignRaw, " ");
|
||||||
|
if(alignmentParts.size() != 4) {
|
||||||
|
std::cout << "Grid alignment is invalid" << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
item.alignGrid = true;
|
||||||
|
item.alignX = alignmentFromRaw(alignmentParts[0]);
|
||||||
|
item.alignY = alignmentFromRaw(alignmentParts[1]);
|
||||||
|
item.align0 = alignmentParts[2];
|
||||||
|
item.align1 = alignmentParts[3];
|
||||||
|
|
||||||
|
if(item.alignX.size() == 0) {
|
||||||
|
std::cout << "X Align is invalid" << std::endl;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
if(item.alignY.size() == 0) {
|
||||||
|
std::cout << "Y Align is invalid" << std::endl;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse color
|
||||||
|
auto attrColor = xmlGetAttributeByName(currentNode, "color");
|
||||||
|
if(attrColor != -1) {
|
||||||
|
item.hasColor = true;
|
||||||
|
item.color = currentNode->attributeDatas[attrColor];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Grid
|
||||||
|
if(item.type == "UIGrid") {
|
||||||
|
auto attrRows = xmlGetAttributeByName(currentNode, "rows");
|
||||||
|
auto attrCols = xmlGetAttributeByName(currentNode, "columns");
|
||||||
|
auto attrGutter = xmlGetAttributeByName(currentNode, "gutter");
|
||||||
|
if(attrRows == -1 || attrCols == -1) {
|
||||||
|
std::cout << "Grid is invalid" << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
item.isGrid = true;
|
||||||
|
item.rows = currentNode->attributeDatas[attrRows];
|
||||||
|
item.columns = currentNode->attributeDatas[attrCols];
|
||||||
|
if(attrGutter != -1) {
|
||||||
|
auto gutterParts = split(currentNode->attributeDatas[attrGutter], " ");
|
||||||
|
if(gutterParts.size() != 2) {
|
||||||
|
std::cout << "Gutter is invalid" << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
item.gutterX = gutterParts[0];
|
||||||
|
item.gutterY = gutterParts[1];
|
||||||
|
} else {
|
||||||
|
item.gutterX = "0";
|
||||||
|
item.gutterY = "0";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Self
|
||||||
|
items->push_back(item);
|
||||||
|
|
||||||
|
// Children
|
||||||
|
for(int32_t i = 0; i < currentNode->childrenCount; i++) {
|
||||||
|
if(!parseChildren(currentNode->children + i, item.name, items)) return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *args[]) {
|
||||||
|
if(argc != 3) {
|
||||||
|
std::cout << "Invalid number of args for ui gen" << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char fileIn[FILENAME_MAX];
|
||||||
|
char fileOut[FILENAME_MAX];
|
||||||
|
|
||||||
|
sprintf(fileIn, "%s", args[1]);
|
||||||
|
sprintf(fileOut, "%s.hpp", args[2]);
|
||||||
|
fileNormalizeSlashes(args[1]);
|
||||||
|
fileNormalizeSlashes(args[2]);
|
||||||
|
|
||||||
|
// Open input file.
|
||||||
|
FILE *fin = fopen(fileIn, "rb");
|
||||||
|
if(fin == NULL) {
|
||||||
|
std::cout << "Failed to open input file " << fileIn << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tell file len
|
||||||
|
fseek(fin, 0, SEEK_END);
|
||||||
|
auto len = ftell(fin);
|
||||||
|
fseek(fin, 0, SEEK_SET);
|
||||||
|
|
||||||
|
// Read data.
|
||||||
|
char *buffer = (char *)malloc(sizeof(char) * (len + 1));
|
||||||
|
if(buffer == NULL) {
|
||||||
|
std::cout << "Failed to create temporary memory." << std::endl;
|
||||||
|
fclose(fin);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
assetReadString(fin, buffer);
|
||||||
|
fclose(fin);
|
||||||
|
|
||||||
|
// Parse XML
|
||||||
|
xml_t xml;
|
||||||
|
xmlLoad(&xml, buffer);
|
||||||
|
free(buffer);
|
||||||
|
|
||||||
|
// Begin output
|
||||||
|
std::string bufferOut = "";
|
||||||
|
|
||||||
|
// Imports
|
||||||
|
bufferOut += "#pragma once\n";
|
||||||
|
for(int32_t i = 0; i < xml.attributeCount; i++) {
|
||||||
|
std::string attrName = xml.attributeNames[i];
|
||||||
|
|
||||||
|
if(
|
||||||
|
attrName == "name"
|
||||||
|
) continue;
|
||||||
|
|
||||||
|
bufferOut += "#include \"";
|
||||||
|
bufferOut += xml.attributeDatas[i];
|
||||||
|
bufferOut += "\"\n";
|
||||||
|
}
|
||||||
|
bufferOut += "\n";
|
||||||
|
|
||||||
|
// Now prep class itself.
|
||||||
|
auto attrName = xmlGetAttributeByName(&xml, "name");
|
||||||
|
if(attrName == -1) {
|
||||||
|
std::cout << "Missing " << std::endl;
|
||||||
|
xmlDispose(&xml);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
std::string name = xml.attributeDatas[attrName];
|
||||||
|
|
||||||
|
// Children
|
||||||
|
std::vector<struct UIGenerated> items;
|
||||||
|
for(int32_t j = 0; j < xml.childrenCount; j++) {
|
||||||
|
if(parseChildren(xml.children + j, "", &items)) continue;
|
||||||
|
xmlDispose(&xml);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate strings.
|
||||||
|
bufferOut += "namespace Dawn {\n";
|
||||||
|
bufferOut += " class " + name + " : public UIEmpty {\n";
|
||||||
|
bufferOut += " public:\n";
|
||||||
|
auto it = items.begin();
|
||||||
|
while(it != items.end()) {
|
||||||
|
auto c = *it;
|
||||||
|
bufferOut += " " + c.type + " " + c.name + ";\n";
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
bufferOut += "\n";
|
||||||
|
bufferOut += " " + name + "(UICanvas *canvas) : UIEmpty(canvas),\n";
|
||||||
|
it = items.begin();
|
||||||
|
while(it != items.end()) {
|
||||||
|
auto c = *it;
|
||||||
|
bufferOut += " " + c.name + "(canvas)";
|
||||||
|
if(it != items.end() - 1) bufferOut += ",";
|
||||||
|
bufferOut += "\n";
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
bufferOut += " {\n";
|
||||||
|
|
||||||
|
it = items.begin();
|
||||||
|
while(it != items.end()) {
|
||||||
|
auto c = *it;
|
||||||
|
bufferOut += "\n";
|
||||||
|
|
||||||
|
// Transform
|
||||||
|
if(c.align) {
|
||||||
|
bufferOut += " " + c.name + ".setTransform(\n";
|
||||||
|
bufferOut += " " + c.alignX + ", " + c.alignY + ",\n";
|
||||||
|
bufferOut += " glm::vec4(" + c.align0 + ", " + c.align1 + ", " + c.align2 + ", " + c.align3 + "),\n";
|
||||||
|
bufferOut += " 0.0f\n";
|
||||||
|
bufferOut += " );\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Color
|
||||||
|
if(c.hasColor) {
|
||||||
|
bufferOut += " " + c.name + ".color = " + c.color + ";\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Grid
|
||||||
|
if(c.isGrid) {
|
||||||
|
bufferOut += " " + c.name + ".setGridSize(\n";
|
||||||
|
bufferOut += " " + c.rows + ", " + c.columns + ",\n";
|
||||||
|
bufferOut += " " + c.gutterX + ", " + c.gutterY + "\n";
|
||||||
|
bufferOut += " );\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parent setting
|
||||||
|
if(c.alignGrid) {
|
||||||
|
bufferOut += " " + c.parent + ".addToGrid(\n";
|
||||||
|
bufferOut += " &" + c.name + ",\n";
|
||||||
|
bufferOut += " " + c.align0 + ", " + c.align1 + ",\n";
|
||||||
|
bufferOut += " " + c.alignX + ", " + c.alignY + "\n";
|
||||||
|
bufferOut += " );\n";
|
||||||
|
} else {
|
||||||
|
if(c.parent == "") {
|
||||||
|
bufferOut += " this->addChild(&" + c.name + ");\n";
|
||||||
|
} else {
|
||||||
|
bufferOut += " " + c.parent + ".addChild(&" + c.name + ");\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
|
||||||
|
bufferOut += " }\n";
|
||||||
|
bufferOut += " };\n";
|
||||||
|
bufferOut += "}";
|
||||||
|
|
||||||
|
// Finished with XML data, now we can write data out.
|
||||||
|
xmlDispose(&xml);
|
||||||
|
|
||||||
|
FILE *fout = fopen(fileOut, "wb");
|
||||||
|
if(fout == NULL) {
|
||||||
|
std::cout << "Failed to open output file." << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Buffer out data.
|
||||||
|
const char *bufferOutStr = bufferOut.c_str();
|
||||||
|
fwrite(bufferOutStr, sizeof(char), strlen(bufferOutStr), fout);
|
||||||
|
fclose(fout);
|
||||||
|
std::cout << "Generated UI " << fileOut << std::endl;
|
||||||
|
|
||||||
|
// Cleanup
|
||||||
|
return 0;
|
||||||
|
}
|
@ -1,67 +1,67 @@
|
|||||||
/**
|
/**
|
||||||
* Copyright (c) 2021 Dominic Masters
|
* Copyright (c) 2021 Dominic Masters
|
||||||
*
|
*
|
||||||
* This software is released under the MIT License.
|
* This software is released under the MIT License.
|
||||||
* https://opensource.org/licenses/MIT
|
* https://opensource.org/licenses/MIT
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "file.h"
|
#include "file.h"
|
||||||
|
|
||||||
#define XML_DOING_NOTHING 0x00
|
#define XML_DOING_NOTHING 0x00
|
||||||
#define XML_PARSING_TAG_NAME 0x01
|
#define XML_PARSING_TAG_NAME 0x01
|
||||||
#define XML_LOOKING_FOR_ATTRIBUTE 0x02
|
#define XML_LOOKING_FOR_ATTRIBUTE 0x02
|
||||||
#define XML_PARSING_ATTRIBUTE_NAME 0x03
|
#define XML_PARSING_ATTRIBUTE_NAME 0x03
|
||||||
#define XML_LOOKING_FOR_ATTRIBUTE_VALUE 0x04
|
#define XML_LOOKING_FOR_ATTRIBUTE_VALUE 0x04
|
||||||
#define XML_PARSING_ATTRIBUTE_VALUE 0x05
|
#define XML_PARSING_ATTRIBUTE_VALUE 0x05
|
||||||
#define XML_PARSING_VALUE 0x06
|
#define XML_PARSING_VALUE 0x06
|
||||||
#define XML_PARSING_CHILD 0x07
|
#define XML_PARSING_CHILD 0x07
|
||||||
#define XML_PARSING_CLOSE 0x08
|
#define XML_PARSING_CLOSE 0x08
|
||||||
|
|
||||||
#define XML_TEXT_BUFFER_MAX 256
|
#define XML_TEXT_BUFFER_MAX 256
|
||||||
#define XML_CHILD_COUNT_MAX 16
|
#define XML_CHILD_COUNT_MAX 128
|
||||||
#define XML_ATTRIBUTE_MAX 16
|
#define XML_ATTRIBUTE_MAX 128
|
||||||
|
|
||||||
typedef struct _xml_t xml_t;
|
typedef struct _xml_t xml_t;
|
||||||
|
|
||||||
typedef struct _xml_t {
|
typedef struct _xml_t {
|
||||||
char *node;
|
char *node;
|
||||||
char *value;
|
char *value;
|
||||||
|
|
||||||
char *attributeNames[XML_ATTRIBUTE_MAX];
|
char *attributeNames[XML_ATTRIBUTE_MAX];
|
||||||
char *attributeDatas[XML_ATTRIBUTE_MAX];
|
char *attributeDatas[XML_ATTRIBUTE_MAX];
|
||||||
uint8_t attributeCount;
|
uint8_t attributeCount;
|
||||||
|
|
||||||
xml_t *children;
|
xml_t *children;
|
||||||
uint8_t childrenCount;
|
uint8_t childrenCount;
|
||||||
} xml_t;
|
} xml_t;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load an XML child from a string buffer.
|
* Load an XML child from a string buffer.
|
||||||
*
|
*
|
||||||
* @param xml XML to load.
|
* @param xml XML to load.
|
||||||
* @param data Data to parse
|
* @param data Data to parse
|
||||||
* @param i Character index within the data
|
* @param i Character index within the data
|
||||||
* @return The index in the data string this XML node ends.
|
* @return The index in the data string this XML node ends.
|
||||||
*/
|
*/
|
||||||
int32_t xmlLoadChild(xml_t *xml, char *data, int32_t i);
|
int32_t xmlLoadChild(xml_t *xml, char *data, int32_t i);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load an XML String into an XML memory.
|
* Load an XML String into an XML memory.
|
||||||
*
|
*
|
||||||
* @param xml XML to load into.
|
* @param xml XML to load into.
|
||||||
* @param data XML string.
|
* @param data XML string.
|
||||||
*/
|
*/
|
||||||
void xmlLoad(xml_t *xml, char *data);
|
void xmlLoad(xml_t *xml, char *data);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dispose a previously loaded XML.
|
* Dispose a previously loaded XML.
|
||||||
*
|
*
|
||||||
* @param xml XML to dispose.
|
* @param xml XML to dispose.
|
||||||
*/
|
*/
|
||||||
void xmlDispose(xml_t *xml);
|
void xmlDispose(xml_t *xml);
|
||||||
|
|
||||||
int16_t xmlGetAttributeByName(xml_t *xml, char *name);
|
int16_t xmlGetAttributeByName(xml_t *xml, char *name);
|
||||||
|
|
||||||
bool xmlIsWhitespace(char c);
|
bool xmlIsWhitespace(char c);
|
Reference in New Issue
Block a user