Updated locale gen
This commit is contained in:
@ -4,9 +4,9 @@
|
|||||||
# https://opensource.org/licenses/MIT
|
# https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
cmake_minimum_required(VERSION 3.13)
|
cmake_minimum_required(VERSION 3.13)
|
||||||
set(CMAKE_C_STANDARD 99)
|
set(CMAKE_C_STANDARD 20)
|
||||||
set(CMAKE_C_STANDARD_REQUIRED ON)
|
set(CMAKE_C_STANDARD_REQUIRED ON)
|
||||||
set(CMAKE_CXX_STANDARD 14)
|
set(CMAKE_CXX_STANDARD 20)
|
||||||
set(CMAKE_CXX_STANDARD_REQUIRED True)
|
set(CMAKE_CXX_STANDARD_REQUIRED True)
|
||||||
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/modules/")
|
set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/modules/")
|
||||||
|
|
||||||
|
@ -17,8 +17,13 @@ target_include_directories(${DAWN_TARGET_NAME}
|
|||||||
${CMAKE_CURRENT_LIST_DIR}
|
${CMAKE_CURRENT_LIST_DIR}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
target_sources(${DAWN_TARGET_NAME}
|
||||||
|
PRIVATE
|
||||||
|
${DAWN_SHARED_SOURCES}
|
||||||
|
)
|
||||||
|
|
||||||
# Subdirs
|
# Subdirs
|
||||||
add_subdirectory(assert)
|
|
||||||
add_subdirectory(asset)
|
add_subdirectory(asset)
|
||||||
add_subdirectory(display)
|
add_subdirectory(display)
|
||||||
add_subdirectory(input)
|
add_subdirectory(input)
|
||||||
|
@ -11,3 +11,12 @@ set(
|
|||||||
CACHE INTERNAL
|
CACHE INTERNAL
|
||||||
${DAWN_CACHE_TARGET}
|
${DAWN_CACHE_TARGET}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
set(D ${CMAKE_CURRENT_LIST_DIR})
|
||||||
|
set(
|
||||||
|
DAWN_SHARED_SOURCES
|
||||||
|
${D}/assert/assert.cpp
|
||||||
|
|
||||||
|
CACHE INTERNAL
|
||||||
|
${DAWN_CACHE_TARGET}
|
||||||
|
)
|
@ -6,7 +6,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "dawnlibs.hpp"
|
#include "dawnsharedlibs.hpp"
|
||||||
|
|
||||||
#define ASSERTS_ENABLED 1
|
#define ASSERTS_ENABLED 1
|
||||||
|
|
@ -19,6 +19,7 @@ extern "C" {
|
|||||||
#include <float.h>
|
#include <float.h>
|
||||||
|
|
||||||
typedef bool bool_t;
|
typedef bool bool_t;
|
||||||
|
typedef char char_t;
|
||||||
}
|
}
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
@ -29,3 +30,4 @@ extern "C" {
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
#include <string>
|
@ -4,7 +4,7 @@
|
|||||||
// https://opensource.org/licenses/MIT
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "dawnlibs.hpp"
|
#include "dawnsharedlibs.hpp"
|
||||||
#include "assert/assert.hpp"
|
#include "assert/assert.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
|
@ -8,18 +8,23 @@ project(languagegen VERSION 2.0)
|
|||||||
add_executable(languagegen)
|
add_executable(languagegen)
|
||||||
target_sources(languagegen
|
target_sources(languagegen
|
||||||
PRIVATE
|
PRIVATE
|
||||||
|
${DAWN_SHARED_SOURCES}
|
||||||
LanguageGen.cpp
|
LanguageGen.cpp
|
||||||
../../util/DawnTool.cpp
|
../../util/DawnTool.cpp
|
||||||
|
../../util/XmlNew.cpp
|
||||||
../../util/file.cpp
|
../../util/file.cpp
|
||||||
../../util/csv.cpp
|
../../util/csv.cpp
|
||||||
../../util/xml.cpp
|
../../util/xml.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
target_include_directories(languagegen
|
target_include_directories(languagegen
|
||||||
PUBLIC
|
PUBLIC
|
||||||
${DAWN_SHARED_INCLUDES}
|
${DAWN_SHARED_INCLUDES}
|
||||||
${CMAKE_CURRENT_LIST_DIR}/../../
|
${CMAKE_CURRENT_LIST_DIR}/../../
|
||||||
${CMAKE_CURRENT_LIST_DIR}
|
${CMAKE_CURRENT_LIST_DIR}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
target_link_libraries(languagegen
|
target_link_libraries(languagegen
|
||||||
PUBLIC
|
PUBLIC
|
||||||
${DAWN_BUILD_HOST_LIBS}
|
${DAWN_BUILD_HOST_LIBS}
|
||||||
|
@ -37,48 +37,42 @@ int32_t LanguageGen::start() {
|
|||||||
|
|
||||||
assetReadString(fileIn, buffer);
|
assetReadString(fileIn, buffer);
|
||||||
fclose(fileIn);
|
fclose(fileIn);
|
||||||
|
auto xml = Xml::load(std::string(buffer));
|
||||||
xml_t xml;
|
|
||||||
xmlLoad(&xml, buffer);
|
|
||||||
free(buffer);
|
free(buffer);
|
||||||
|
|
||||||
// Begin parsing. Start by looking for the <language> tags
|
// Begin parsing. Start by looking for the <language> tags
|
||||||
std::vector<std::string> languages;
|
std::vector<std::string> languages;
|
||||||
for(int32_t i = 0; i < xml.childrenCount; i++) {
|
auto itChildren = xml.children.begin();
|
||||||
auto c = xml.children + i;
|
while(itChildren != xml.children.end()) {
|
||||||
if(std::string(c->node) != "language") continue;
|
auto child = *itChildren;
|
||||||
auto attrName = xmlGetAttributeByName(c, "name");
|
if(child->node == "language") {
|
||||||
|
auto attrName = child->attributes.find("name");
|
||||||
if(attrName == -1) {
|
if(attrName == child->attributes.end()) {
|
||||||
std::cout << "Missing name param on language node" << std::endl;
|
std::cout << "Missing name param on language node" << std::endl;
|
||||||
xmlDispose(&xml);
|
return 1;
|
||||||
return 1;
|
}
|
||||||
|
languages.push_back(attrName->second);
|
||||||
}
|
}
|
||||||
languages.push_back(std::string(c->attributeDatas[attrName]));
|
++itChildren;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now begin actually parsing
|
// Now begin actually parsing
|
||||||
std::map<std::string, std::vector<struct LanguageString>> strings;
|
std::map<std::string, std::vector<struct LanguageString>> strings;
|
||||||
for(int32_t i = 0; i < xml.childrenCount; i++) {
|
itChildren = xml.children.begin();
|
||||||
auto c = xml.children + i;
|
while(itChildren != xml.children.end()) {
|
||||||
if(std::string(c->node) == "group") {
|
auto child = *itChildren;
|
||||||
auto ret = this->parseGroup(c, "", &strings);
|
if(child->node == "group") {
|
||||||
if(ret != 0) {
|
auto ret = this->parseGroup(child, "", &strings);
|
||||||
xmlDispose(&xml);
|
if(ret != 0) return ret;
|
||||||
return ret;
|
} else if(child->node == "string") {
|
||||||
}
|
|
||||||
} else if(std::string(c->node) == "string") {
|
|
||||||
std::cout << "String cannot be a root node" << std::endl;
|
std::cout << "String cannot be a root node" << std::endl;
|
||||||
xmlDispose(&xml);
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
++itChildren;
|
||||||
}
|
}
|
||||||
|
|
||||||
xmlDispose(&xml);
|
|
||||||
|
|
||||||
// Now we validate each lang has each key.
|
// Now we validate each lang has each key.
|
||||||
std::vector<std::string> keys;
|
std::vector<std::string> keys;
|
||||||
|
|
||||||
auto it = strings.begin();
|
auto it = strings.begin();
|
||||||
while(it != strings.end()) {
|
while(it != strings.end()) {
|
||||||
auto it2 = it->second.begin();
|
auto it2 = it->second.begin();
|
||||||
@ -98,7 +92,7 @@ int32_t LanguageGen::start() {
|
|||||||
while(it != strings.end()) {
|
while(it != strings.end()) {
|
||||||
std::vector<std::string> itKeys;
|
std::vector<std::string> itKeys;
|
||||||
|
|
||||||
std::string bufferOut = "";
|
std::string bufferOut;
|
||||||
|
|
||||||
auto it2 = it->second.begin();
|
auto it2 = it->second.begin();
|
||||||
while(it2 != it->second.end()) {
|
while(it2 != it->second.end()) {
|
||||||
@ -119,8 +113,8 @@ int32_t LanguageGen::start() {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
const char *strOut = bufferOut.c_str();
|
const char_t *strOut = bufferOut.c_str();
|
||||||
fwrite(strOut, sizeof(char), strlen(strOut), fileOut);
|
fwrite(strOut, sizeof(char_t), strlen(strOut), fileOut);
|
||||||
fclose(fileOut);
|
fclose(fileOut);
|
||||||
|
|
||||||
auto it3 = keys.begin();
|
auto it3 = keys.begin();
|
||||||
@ -144,54 +138,55 @@ int32_t LanguageGen::start() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int32_t LanguageGen::parseString(
|
int32_t LanguageGen::parseString(
|
||||||
xml_t *stringNode,
|
Xml *stringNode,
|
||||||
std::string key,
|
std::string key,
|
||||||
std::map<std::string,std::vector<struct LanguageString>> *strings
|
std::map<std::string,std::vector<struct LanguageString>> *strings
|
||||||
) {
|
) {
|
||||||
auto attrLang = xmlGetAttributeByName(stringNode, "lang");
|
auto attrLang = stringNode->attributes.find("lang");
|
||||||
if(attrLang == -1) {
|
if(attrLang == stringNode->attributes.end()) {
|
||||||
std::cout << "String is missing lang parameter." << std::endl;
|
std::cout << "String is missing lang parameter." << std::endl;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string lang(stringNode->attributeDatas[attrLang]);
|
|
||||||
struct LanguageString str;
|
struct LanguageString str;
|
||||||
str.key = key;
|
str.key = key;
|
||||||
str.value = std::string(stringNode->value);
|
str.value = stringNode->value;
|
||||||
|
|
||||||
auto existing = (*strings).find(lang);
|
auto existing = (*strings).find(attrLang->second);
|
||||||
if(existing == (*strings).end()) {
|
if(existing == (*strings).end()) {
|
||||||
(*strings).insert(std::make_pair(lang, std::vector<struct LanguageString>()));
|
(*strings).insert(std::make_pair(attrLang->second, std::vector<struct LanguageString>()));
|
||||||
}
|
}
|
||||||
(*strings)[lang].push_back(str);
|
(*strings)[attrLang->second].push_back(str);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int32_t LanguageGen::parseGroup(
|
int32_t LanguageGen::parseGroup(
|
||||||
xml_t *groupNode,
|
Xml *groupNode,
|
||||||
std::string key,
|
std::string key,
|
||||||
std::map<std::string,std::vector<struct LanguageString>> *strings
|
std::map<std::string, std::vector<struct LanguageString>> *strings
|
||||||
) {
|
) {
|
||||||
int32_t ret;
|
int32_t ret;
|
||||||
|
|
||||||
auto attrKey = xmlGetAttributeByName(groupNode, "key");
|
auto attrKey = groupNode->attributes.find("key");
|
||||||
if(attrKey == -1) {
|
if(attrKey == groupNode->attributes.end()) {
|
||||||
std::cout << "Group node is missing key" << std::endl;
|
std::cout << "Group node is missing key" << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(key.size() > 0) key += ".";
|
if(key.size() > 0) key += ".";
|
||||||
key += std::string(groupNode->attributeDatas[attrKey]);
|
key += attrKey->second;
|
||||||
|
|
||||||
for(int32_t i = 0; i < groupNode->childrenCount; i++) {
|
auto itChildren = groupNode->children.begin();
|
||||||
auto c = groupNode->children + i;
|
while(itChildren != groupNode->children.end()) {
|
||||||
if(std::string(c->node) == "string") {
|
auto child = *itChildren;
|
||||||
ret = this->parseString(c, key, strings);
|
if(child->node == "string") {
|
||||||
|
ret = this->parseString(child, key, strings);
|
||||||
if(ret != 0) return ret;
|
if(ret != 0) return ret;
|
||||||
} else if(std::string(c->node) == "group") {
|
} else if(child->node == "group") {
|
||||||
ret = this->parseGroup(c, key, strings);
|
ret = this->parseGroup(child, key, strings);
|
||||||
if(ret != 0) return ret;
|
if(ret != 0) return ret;
|
||||||
}
|
}
|
||||||
|
++itChildren;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "util/DawnTool.hpp"
|
#include "util/DawnTool.hpp"
|
||||||
|
#include "util/XmlNew.hpp"
|
||||||
|
|
||||||
namespace Dawn {
|
namespace Dawn {
|
||||||
struct LanguageString {
|
struct LanguageString {
|
||||||
@ -15,13 +16,13 @@ namespace Dawn {
|
|||||||
class LanguageGen : public DawnTool {
|
class LanguageGen : public DawnTool {
|
||||||
protected:
|
protected:
|
||||||
int32_t parseGroup(
|
int32_t parseGroup(
|
||||||
xml_t *node,
|
Xml *node,
|
||||||
std::string key,
|
std::string key,
|
||||||
std::map<std::string, std::vector<struct LanguageString>> *strings
|
std::map<std::string, std::vector<struct LanguageString>> *strings
|
||||||
);
|
);
|
||||||
|
|
||||||
int32_t parseString(
|
int32_t parseString(
|
||||||
xml_t *node,
|
Xml *node,
|
||||||
std::string key,
|
std::string key,
|
||||||
std::map<std::string, std::vector<struct LanguageString>> *strings
|
std::map<std::string, std::vector<struct LanguageString>> *strings
|
||||||
);
|
);
|
||||||
|
223
src/dawntools/util/XmlNew.cpp
Normal file
223
src/dawntools/util/XmlNew.cpp
Normal file
@ -0,0 +1,223 @@
|
|||||||
|
// Copyright (c) 2023 Dominic Masters
|
||||||
|
//
|
||||||
|
// This software is released under the MIT License.
|
||||||
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
|
#include "XmlNew.hpp"
|
||||||
|
#include "util/array.hpp"
|
||||||
|
|
||||||
|
using namespace Dawn;
|
||||||
|
|
||||||
|
bool_t Xml::isWhitespace(char_t c) {
|
||||||
|
return c == ' ' || c == '\r' || c == '\n' || c == '\t';
|
||||||
|
}
|
||||||
|
|
||||||
|
Xml Xml::load(std::string data) {
|
||||||
|
size_t j = 0;
|
||||||
|
Xml xml;
|
||||||
|
Xml::load(&xml, data, &j);
|
||||||
|
return xml;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Xml::load(Xml *xml, std::string data, size_t *j) {
|
||||||
|
char_t c;
|
||||||
|
int32_t level = 0;
|
||||||
|
enum XmlParseState doing = XML_PARSE_STATE_DOING_NOTHING;
|
||||||
|
enum XmlParseState doingBeforeComment;
|
||||||
|
bool_t insideTag = false;
|
||||||
|
std::string buffer = "";
|
||||||
|
std::string attrKey = "";
|
||||||
|
size_t i = *j;
|
||||||
|
|
||||||
|
while(c = data[i++]) {
|
||||||
|
switch(doing) {
|
||||||
|
case XML_PARSE_STATE_DOING_NOTHING:
|
||||||
|
if(c == '>') continue;
|
||||||
|
if(c == '<') {
|
||||||
|
if(data[i] == '!' && data[i+1] == '-' && data[i+2] == '-') {
|
||||||
|
doingBeforeComment = doing;
|
||||||
|
doing = XML_PARSE_STATE_PARSING_COMMENT;
|
||||||
|
i += 3;
|
||||||
|
} else if(insideTag) {
|
||||||
|
i -= 1;
|
||||||
|
auto child = new Xml();
|
||||||
|
Xml::load(child, data, &i);
|
||||||
|
xml->children.push_back(child);
|
||||||
|
doing = XML_PARSE_STATE_PARSING_CHILD;
|
||||||
|
} else {
|
||||||
|
doing = XML_PARSE_STATE_PARSING_TAG_NAME;
|
||||||
|
level++;
|
||||||
|
insideTag = true;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Xml::isWhitespace(c)) continue;
|
||||||
|
doing = XML_PARSE_STATE_PARSING_VALUE;
|
||||||
|
buffer += c;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_PARSING_TAG_NAME:
|
||||||
|
// Just keep reading until we either hit a space (end of the tag name)
|
||||||
|
// or a closing tag value, either / or >
|
||||||
|
if(Xml::isWhitespace(c) || c == '>' || c == '/') {
|
||||||
|
xml->node = buffer;
|
||||||
|
buffer = "";
|
||||||
|
if(c == '/') {
|
||||||
|
level--;
|
||||||
|
insideTag = false;
|
||||||
|
doing = XML_PARSE_STATE_PARSING_CLOSE;
|
||||||
|
} else {
|
||||||
|
doing = c == '>' ? XML_PARSE_STATE_DOING_NOTHING : XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
buffer += c;
|
||||||
|
break;
|
||||||
|
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE:
|
||||||
|
// Look until we hit either the end of a tag, or the attribute itself
|
||||||
|
if(Xml::isWhitespace(c) || c == '>' || c == '/' || c == '=') {
|
||||||
|
if(c == '>' || c == '/') {
|
||||||
|
doing = XML_PARSE_STATE_DOING_NOTHING;
|
||||||
|
if(c == '/') {
|
||||||
|
level--;
|
||||||
|
insideTag = false;
|
||||||
|
doing = XML_PARSE_STATE_PARSING_CLOSE;
|
||||||
|
}
|
||||||
|
} else if(c == '=') {
|
||||||
|
doing = XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE_VALUE;
|
||||||
|
} else {
|
||||||
|
doing = XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(buffer.size() > 0) {
|
||||||
|
attrKey = buffer;
|
||||||
|
xml->attributes[buffer] = "";
|
||||||
|
buffer = "";
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
buffer += c;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE_VALUE:
|
||||||
|
// Keep looking until we find a quote mark
|
||||||
|
if(Xml::isWhitespace(c)) continue;
|
||||||
|
if(c == '>' || c == '/') {
|
||||||
|
doing = XML_PARSE_STATE_DOING_NOTHING;
|
||||||
|
insideTag = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(c != '"') continue;
|
||||||
|
doing = XML_PARSE_STATE_PARSING_ATTRIBUTE_VALUE;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_PARSING_ATTRIBUTE_VALUE:
|
||||||
|
// Parse the attribute value until we find a quote mark.
|
||||||
|
if(c == '"') {
|
||||||
|
doing = XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE;
|
||||||
|
xml->attributes[attrKey] = buffer;
|
||||||
|
buffer = "";
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
buffer += c;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_PARSING_VALUE:
|
||||||
|
// Keep parsing child until we find a < for an opening/closing tag.
|
||||||
|
if(c == '<') {
|
||||||
|
// In HTML Spec there could be a child here but not in XML spec.
|
||||||
|
doing = XML_PARSE_STATE_PARSING_CLOSE;
|
||||||
|
xml->value = buffer;
|
||||||
|
buffer = "";
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
buffer += c;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_PARSING_CHILD:
|
||||||
|
if(c == '<') {
|
||||||
|
// Read ahead and confirm this is a close or not
|
||||||
|
if(data[i] == '/') {
|
||||||
|
doing = XML_PARSE_STATE_PARSING_CLOSE;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(data[i] == '!' && data[i+1] == '-' && data[i+2] == '-') {
|
||||||
|
doingBeforeComment = doing;
|
||||||
|
doing = XML_PARSE_STATE_PARSING_COMMENT;
|
||||||
|
i += 3;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Likely another child.
|
||||||
|
auto child = new Xml();
|
||||||
|
i -= 1;
|
||||||
|
Xml::load(child, data, &i);
|
||||||
|
xml->children.push_back(child);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Xml::isWhitespace(c)) continue;
|
||||||
|
|
||||||
|
// In HTML Spec there's a chance for there to be a value here, but not
|
||||||
|
// in the XML spec.
|
||||||
|
break;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_PARSING_CLOSE:
|
||||||
|
// Just keep parsing until the tag closer finishes.
|
||||||
|
if(c != '>') continue;
|
||||||
|
doing = XML_PARSE_STATE_DOING_NOTHING;
|
||||||
|
|
||||||
|
//TODO: Return index or something?
|
||||||
|
*j = i;
|
||||||
|
return;
|
||||||
|
|
||||||
|
case XML_PARSE_STATE_PARSING_COMMENT:
|
||||||
|
if(c != '-') continue;
|
||||||
|
if(data[i] != '-') continue;
|
||||||
|
if(data[i+1] != '>') continue;
|
||||||
|
i += 2;
|
||||||
|
doing = doingBeforeComment;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*j = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
std::vector<Xml*> Xml::getChildrenOfType(std::string type) {
|
||||||
|
std::vector<Xml*> children;
|
||||||
|
auto itChildren = this->children.begin();
|
||||||
|
while(itChildren != this->children.end()) {
|
||||||
|
auto child = *itChildren;
|
||||||
|
if(child->node == type) children.push_back(child);
|
||||||
|
++itChildren;
|
||||||
|
}
|
||||||
|
return children;
|
||||||
|
}
|
||||||
|
|
||||||
|
Xml * Xml::getFirstChildOfType(std::string type) {
|
||||||
|
auto itChildren = this->children.begin();
|
||||||
|
while(itChildren != this->children.end()) {
|
||||||
|
auto child = *itChildren;
|
||||||
|
if(child->node == type) return child;
|
||||||
|
++itChildren;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Xml::~Xml() {
|
||||||
|
auto it = this->children.begin();
|
||||||
|
while(it != this->children.end()) {
|
||||||
|
delete *it;
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
}
|
43
src/dawntools/util/XmlNew.hpp
Normal file
43
src/dawntools/util/XmlNew.hpp
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
// Copyright (c) 2023 Dominic Masters
|
||||||
|
//
|
||||||
|
// This software is released under the MIT License.
|
||||||
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include "dawnsharedlibs.hpp"
|
||||||
|
|
||||||
|
namespace Dawn {
|
||||||
|
enum XmlParseState {
|
||||||
|
XML_PARSE_STATE_DOING_NOTHING,
|
||||||
|
XML_PARSE_STATE_PARSING_TAG_NAME,
|
||||||
|
XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE,
|
||||||
|
XML_PARSE_STATE_PARSING_ATTRIBUTE_NAME,
|
||||||
|
XML_PARSE_STATE_LOOKING_FOR_ATTRIBUTE_VALUE,
|
||||||
|
XML_PARSE_STATE_PARSING_ATTRIBUTE_VALUE,
|
||||||
|
XML_PARSE_STATE_PARSING_VALUE,
|
||||||
|
XML_PARSE_STATE_PARSING_CHILD,
|
||||||
|
XML_PARSE_STATE_PARSING_CLOSE,
|
||||||
|
XML_PARSE_STATE_PARSING_COMMENT
|
||||||
|
};
|
||||||
|
|
||||||
|
class Xml {
|
||||||
|
protected:
|
||||||
|
static bool_t isWhitespace(char_t c);
|
||||||
|
|
||||||
|
public:
|
||||||
|
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<std::string, std::string> attributes;
|
||||||
|
std::vector<Xml*> children;
|
||||||
|
|
||||||
|
std::vector<Xml*> getChildrenOfType(std::string type);
|
||||||
|
Xml * getFirstChildOfType(std::string type);
|
||||||
|
|
||||||
|
~Xml();
|
||||||
|
};
|
||||||
|
}
|
Reference in New Issue
Block a user