2021-12-05 17:54:36 +00:00
|
|
|
#include <sstream>
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
#include "BrickDatabase.h"
|
|
|
|
#include "Game.h"
|
2022-11-01 18:21:26 +00:00
|
|
|
#include "AssetManager.h"
|
2023-01-07 05:17:05 +00:00
|
|
|
#include "tinyxml2.h"
|
2023-07-17 22:55:25 +00:00
|
|
|
#include "Brick.h"
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2023-07-17 22:55:25 +00:00
|
|
|
const BrickList& BrickDatabase::GetBricks(const LxfmlPath& lxfmlPath) {
|
|
|
|
static std::unordered_map<LxfmlPath, BrickList> m_Cache;
|
|
|
|
static const BrickList emptyCache;
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
const auto cached = m_Cache.find(lxfmlPath);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
if (cached != m_Cache.end()) {
|
|
|
|
return cached->second;
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2023-12-23 17:24:16 +00:00
|
|
|
auto file = Game::assetManager->GetFile((lxfmlPath).c_str());
|
2022-11-10 18:59:31 +00:00
|
|
|
|
2023-12-23 17:24:16 +00:00
|
|
|
if (!file) {
|
2021-12-05 17:54:36 +00:00
|
|
|
return emptyCache;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
std::stringstream data;
|
|
|
|
data << file.rdbuf();
|
|
|
|
if (data.str().empty()) {
|
|
|
|
return emptyCache;
|
|
|
|
}
|
|
|
|
|
2024-04-08 09:34:36 +00:00
|
|
|
tinyxml2::XMLDocument doc;
|
|
|
|
if (doc.Parse(data.str().c_str(), data.str().size()) != 0) {
|
2021-12-05 17:54:36 +00:00
|
|
|
return emptyCache;
|
|
|
|
}
|
|
|
|
|
2023-07-17 22:55:25 +00:00
|
|
|
BrickList parts;
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2024-04-08 09:34:36 +00:00
|
|
|
auto* lxfml = doc.FirstChildElement("LXFML");
|
2021-12-05 17:54:36 +00:00
|
|
|
auto* bricks = lxfml->FirstChildElement("Bricks");
|
|
|
|
std::string searchTerm = "Brick";
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
if (!bricks) {
|
|
|
|
searchTerm = "Part";
|
|
|
|
bricks = lxfml->FirstChildElement("Scene")->FirstChildElement("Model")->FirstChildElement("Group");
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
if (!bricks) {
|
2022-07-28 13:39:57 +00:00
|
|
|
return emptyCache;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
auto* currentBrick = bricks->FirstChildElement(searchTerm.c_str());
|
|
|
|
while (currentBrick != nullptr) {
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
auto* part = currentBrick->FirstChildElement("Part");
|
2021-12-05 17:54:36 +00:00
|
|
|
if (part == nullptr) part = currentBrick;
|
|
|
|
|
|
|
|
if (part->Attribute("designID") != nullptr) {
|
2022-07-28 13:39:57 +00:00
|
|
|
Brick brick{ static_cast<uint32_t>(part->IntAttribute("designID")) };
|
|
|
|
|
|
|
|
// Depends on the file, some don't specify a list but just a single material
|
|
|
|
const auto* materialList = part->Attribute("materials");
|
|
|
|
const auto* materialID = part->Attribute("materialID");
|
|
|
|
|
|
|
|
if (materialList != nullptr) {
|
|
|
|
std::string materialString(materialList);
|
|
|
|
const auto materials = GeneralUtils::SplitString(materialString, ',');
|
|
|
|
|
|
|
|
if (!materials.empty()) {
|
|
|
|
brick.materialID = std::stoi(materials[0]);
|
|
|
|
} else {
|
|
|
|
brick.materialID = 0;
|
|
|
|
}
|
|
|
|
} else if (materialID != nullptr) {
|
|
|
|
brick.materialID = std::stoi(materialID);
|
|
|
|
} else {
|
|
|
|
brick.materialID = 0; // This is bad, makes it so the minigame can't be played
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
|
|
parts.push_back(brick);
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
currentBrick = currentBrick->NextSiblingElement(searchTerm.c_str());
|
|
|
|
}
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
m_Cache[lxfmlPath] = parts;
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
return m_Cache[lxfmlPath];
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|