2021-12-05 17:54:36 +00:00
|
|
|
|
#include "Zone.h"
|
|
|
|
|
#include "Level.h"
|
|
|
|
|
#include <fstream>
|
|
|
|
|
#include <sstream>
|
|
|
|
|
#include "Game.h"
|
2023-10-21 23:31:55 +00:00
|
|
|
|
#include "Logger.h"
|
2021-12-05 17:54:36 +00:00
|
|
|
|
#include "GeneralUtils.h"
|
|
|
|
|
#include "BinaryIO.h"
|
2023-01-07 05:17:05 +00:00
|
|
|
|
#include "LUTriggers.h"
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
#include "AssetManager.h"
|
2021-12-05 17:54:36 +00:00
|
|
|
|
#include "CDClientManager.h"
|
|
|
|
|
#include "CDZoneTableTable.h"
|
|
|
|
|
#include "Spawner.h"
|
|
|
|
|
#include "dZoneManager.h"
|
2023-08-12 14:20:00 +00:00
|
|
|
|
#include "dpWorld.h"
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
2023-02-10 08:29:53 +00:00
|
|
|
|
#include "eTriggerCommandType.h"
|
|
|
|
|
#include "eTriggerEventType.h"
|
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
|
Zone::Zone(const LWOMAPID& mapID, const LWOINSTANCEID& instanceID, const LWOCLONEID& cloneID) :
|
|
|
|
|
m_ZoneID(mapID, instanceID, cloneID) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
m_NumberOfScenesLoaded = 0;
|
|
|
|
|
m_NumberOfObjectsLoaded = 0;
|
|
|
|
|
m_NumberOfSceneTransitionsLoaded = 0;
|
|
|
|
|
m_CheckSum = 0;
|
|
|
|
|
m_WorldID = 0;
|
|
|
|
|
m_SceneCount = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Zone::~Zone() {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
LOG("Destroying zone %i", m_ZoneID.GetMapID());
|
2021-12-05 17:54:36 +00:00
|
|
|
|
for (std::map<LWOSCENEID, SceneRef>::iterator it = m_Scenes.begin(); it != m_Scenes.end(); ++it) {
|
|
|
|
|
if (it->second.level != nullptr) delete it->second.level;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
|
void Zone::Initalize() {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
LoadZoneIntoMemory();
|
|
|
|
|
LoadLevelsIntoMemory();
|
|
|
|
|
m_CheckSum = CalculateChecksum();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Zone::LoadZoneIntoMemory() {
|
|
|
|
|
m_ZoneFilePath = GetFilePathForZoneID();
|
|
|
|
|
m_ZonePath = m_ZoneFilePath.substr(0, m_ZoneFilePath.rfind('/') + 1);
|
|
|
|
|
if (m_ZoneFilePath == "ERR") return;
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
AssetMemoryBuffer buffer = Game::assetManager->GetFileAsBuffer(m_ZoneFilePath.c_str());
|
2022-11-10 18:59:31 +00:00
|
|
|
|
|
|
|
|
|
if (!buffer.m_Success) {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
LOG("Failed to load %s", m_ZoneFilePath.c_str());
|
2022-11-10 18:59:31 +00:00
|
|
|
|
throw std::runtime_error("Aborting Zone loading due to no Zone File.");
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
std::istream file(&buffer);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
if (file) {
|
2023-08-10 21:35:12 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, m_FileFormatVersion);
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
uint32_t mapRevision = 0;
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::Alpha) BinaryIO::BinaryRead(file, mapRevision);
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, m_WorldID);
|
2023-10-21 23:31:55 +00:00
|
|
|
|
if ((uint16_t)m_WorldID != m_ZoneID.GetMapID()) LOG("WorldID: %i doesn't match MapID %i! Is this intended?", m_WorldID, m_ZoneID.GetMapID());
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
|
|
|
|
AddRevision(LWOSCENEID_INVALID, mapRevision);
|
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::Beta) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, m_Spawnpoint);
|
|
|
|
|
BinaryIO::BinaryRead(file, m_SpawnpointRotation);
|
|
|
|
|
}
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion <= Zone::FileFormatVersion::LateAlpha) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
uint8_t sceneCount;
|
|
|
|
|
BinaryIO::BinaryRead(file, sceneCount);
|
|
|
|
|
m_SceneCount = sceneCount;
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else BinaryIO::BinaryRead(file, m_SceneCount);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < m_SceneCount; ++i) {
|
|
|
|
|
LoadScene(file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//Read generic zone info:
|
|
|
|
|
uint8_t stringLength;
|
|
|
|
|
BinaryIO::BinaryRead(file, stringLength);
|
|
|
|
|
m_ZonePath = BinaryIO::ReadString(file, stringLength);
|
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, stringLength);
|
|
|
|
|
m_ZoneRawPath = BinaryIO::ReadString(file, stringLength);
|
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, stringLength);
|
|
|
|
|
m_ZoneName = BinaryIO::ReadString(file, stringLength);
|
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, stringLength);
|
|
|
|
|
m_ZoneDesc = BinaryIO::ReadString(file, stringLength);
|
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::PreAlpha) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, m_NumberOfSceneTransitionsLoaded);
|
|
|
|
|
for (uint32_t i = 0; i < m_NumberOfSceneTransitionsLoaded; ++i) {
|
|
|
|
|
LoadSceneTransition(file);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::EarlyAlpha) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, m_PathDataLength);
|
2022-11-12 14:44:03 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, m_PathChunkVersion); // always should be 1
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
2022-11-12 14:44:03 +00:00
|
|
|
|
uint32_t pathCount;
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, pathCount);
|
|
|
|
|
|
2022-11-12 14:44:03 +00:00
|
|
|
|
for (uint32_t i = 0; i < pathCount; ++i) LoadPath(file);
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
for (Path path : m_Paths) {
|
|
|
|
|
if (path.pathType == PathType::Spawner) {
|
|
|
|
|
SpawnerInfo info = SpawnerInfo();
|
|
|
|
|
for (PathWaypoint waypoint : path.pathWaypoints) {
|
|
|
|
|
SpawnerNode* node = new SpawnerNode();
|
|
|
|
|
node->position = waypoint.position;
|
|
|
|
|
node->rotation = waypoint.rotation;
|
|
|
|
|
node->nodeID = 0;
|
|
|
|
|
node->config = waypoint.config;
|
|
|
|
|
|
|
|
|
|
for (LDFBaseData* data : waypoint.config) {
|
|
|
|
|
if (data) {
|
|
|
|
|
if (data->GetKey() == u"spawner_node_id") {
|
|
|
|
|
node->nodeID = std::stoi(data->GetValueAsString());
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (data->GetKey() == u"spawner_max_per_node") {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
node->nodeMax = std::stoi(data->GetValueAsString());
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (data->GetKey() == u"groupID") { // Load object group
|
2021-12-05 17:54:36 +00:00
|
|
|
|
std::string groupStr = data->GetValueAsString();
|
|
|
|
|
info.groups = GeneralUtils::SplitString(groupStr, ';');
|
|
|
|
|
info.groups.erase(info.groups.end() - 1);
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (data->GetKey() == u"grpNameQBShowBricks") {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
if (data->GetValueAsString() == "") continue;
|
|
|
|
|
/*std::string groupStr = data->GetValueAsString();
|
|
|
|
|
info.groups.push_back(groupStr);*/
|
|
|
|
|
info.grpNameQBShowBricks = data->GetValueAsString();
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (data->GetKey() == u"spawner_name") {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
info.name = data->GetValueAsString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
info.nodes.push_back(node);
|
|
|
|
|
}
|
|
|
|
|
info.templateID = path.spawner.spawnedLOT;
|
|
|
|
|
info.spawnerID = path.spawner.spawnerObjID;
|
|
|
|
|
info.respawnTime = path.spawner.respawnTime;
|
|
|
|
|
info.amountMaintained = path.spawner.amountMaintained;
|
|
|
|
|
info.maxToSpawn = path.spawner.maxToSpawn;
|
|
|
|
|
info.activeOnLoad = path.spawner.spawnerNetActive;
|
|
|
|
|
info.isNetwork = true;
|
|
|
|
|
Spawner* spawner = new Spawner(info);
|
2023-07-17 22:55:33 +00:00
|
|
|
|
Game::zoneManager->AddSpawner(info.spawnerID, spawner);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
LOG("Failed to open: %s", m_ZoneFilePath.c_str());
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
m_ZonePath = m_ZoneFilePath.substr(0, m_ZoneFilePath.rfind('/') + 1);
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
buffer.close();
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string Zone::GetFilePathForZoneID() {
|
|
|
|
|
//We're gonna go ahead and presume we've got the db loaded already:
|
2023-03-17 14:36:21 +00:00
|
|
|
|
CDZoneTableTable* zoneTable = CDClientManager::Instance().GetTable<CDZoneTableTable>();
|
2021-12-05 17:54:36 +00:00
|
|
|
|
const CDZoneTable* zone = zoneTable->Query(this->GetZoneID().GetMapID());
|
|
|
|
|
if (zone != nullptr) {
|
2022-11-01 18:21:26 +00:00
|
|
|
|
std::string toReturn = "maps/" + zone->zoneName;
|
2022-07-28 13:39:57 +00:00
|
|
|
|
std::transform(toReturn.begin(), toReturn.end(), toReturn.begin(), ::tolower);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
return toReturn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return std::string("ERR");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//Based off code from: https://www.liquisearch.com/fletchers_checksum/implementation/optimizations
|
|
|
|
|
uint32_t Zone::CalculateChecksum() {
|
|
|
|
|
uint32_t sum1 = 0xffff, sum2 = 0xffff;
|
|
|
|
|
|
|
|
|
|
for (std::map<LWOSCENEID, uint32_t>::const_iterator it = m_MapRevisions.cbegin(); it != m_MapRevisions.cend(); ++it) {
|
|
|
|
|
uint32_t sceneID = it->first.GetSceneID();
|
|
|
|
|
sum2 += sum1 += (sceneID >> 16);
|
|
|
|
|
sum2 += sum1 += (sceneID & 0xffff);
|
|
|
|
|
|
|
|
|
|
uint32_t layerID = it->first.GetLayerID();
|
|
|
|
|
sum2 += sum1 += (layerID >> 16);
|
|
|
|
|
sum2 += sum1 += (layerID & 0xffff);
|
|
|
|
|
|
|
|
|
|
uint32_t revision = it->second;
|
|
|
|
|
sum2 += sum1 += (revision >> 16);
|
|
|
|
|
sum2 += sum1 += (revision & 0xffff);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sum1 = (sum1 & 0xffff) + (sum1 >> 16);
|
|
|
|
|
sum2 = (sum2 & 0xffff) + (sum2 >> 16);
|
|
|
|
|
|
|
|
|
|
return sum2 << 16 | sum1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Zone::LoadLevelsIntoMemory() {
|
|
|
|
|
for (std::map<LWOSCENEID, SceneRef>::iterator it = m_Scenes.begin(); it != m_Scenes.end(); ++it) {
|
|
|
|
|
if (it->second.level == nullptr) {
|
|
|
|
|
it->second.level = new Level(this, m_ZonePath + it->second.filename);
|
|
|
|
|
|
|
|
|
|
if (it->second.level->m_ChunkHeaders.size() > 0) {
|
|
|
|
|
it->second.level->m_ChunkHeaders.begin()->second.lwoSceneID = it->first;
|
|
|
|
|
AddRevision(it->second.level->m_ChunkHeaders.begin()->second.lwoSceneID, it->second.level->m_ChunkHeaders.begin()->second.fileInfo->revision);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Zone::AddRevision(LWOSCENEID sceneID, uint32_t revision) {
|
|
|
|
|
for (std::pair<LWOSCENEID, uint32_t> item : m_MapRevisions) {
|
|
|
|
|
if (item.first == sceneID) return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_MapRevisions[LWOSCENEID(sceneID)] = revision;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const void Zone::PrintAllGameObjects() {
|
|
|
|
|
for (std::pair<LWOSCENEID, SceneRef> scene : m_Scenes) {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
LOG("In sceneID: %i", scene.first.GetSceneID());
|
2021-12-05 17:54:36 +00:00
|
|
|
|
scene.second.level->PrintAllObjects();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
void Zone::LoadScene(std::istream& file) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
SceneRef scene;
|
|
|
|
|
scene.level = nullptr;
|
|
|
|
|
LWOSCENEID lwoSceneID(LWOZONEID_INVALID, 0);
|
|
|
|
|
|
|
|
|
|
uint8_t sceneFilenameLength;
|
|
|
|
|
BinaryIO::BinaryRead(file, sceneFilenameLength);
|
|
|
|
|
scene.filename = BinaryIO::ReadString(file, sceneFilenameLength);
|
|
|
|
|
|
|
|
|
|
std::string luTriggersPath = scene.filename.substr(0, scene.filename.size() - 4) + ".lutriggers";
|
2022-11-12 14:44:37 +00:00
|
|
|
|
std::vector<LUTriggers::Trigger*> triggers;
|
2022-11-14 18:55:40 +00:00
|
|
|
|
if(Game::assetManager->HasFile((m_ZonePath + luTriggersPath).c_str())) triggers = LoadLUTriggers(luTriggersPath, scene.id);
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
for (LUTriggers::Trigger* trigger : triggers) {
|
|
|
|
|
scene.triggers.insert({ trigger->id, trigger });
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::LatePreAlpha || m_FileFormatVersion < Zone::FileFormatVersion::PrePreAlpha) {
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.id);
|
|
|
|
|
lwoSceneID.SetSceneID(scene.id);
|
|
|
|
|
}
|
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::LatePreAlpha) {
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.sceneType);
|
|
|
|
|
lwoSceneID.SetLayerID(scene.sceneType);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
uint8_t sceneNameLength;
|
|
|
|
|
BinaryIO::BinaryRead(file, sceneNameLength);
|
|
|
|
|
scene.name = BinaryIO::ReadString(file, sceneNameLength);
|
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion == Zone::FileFormatVersion::LatePreAlpha){
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.unknown1);
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.unknown2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (m_FileFormatVersion >= Zone::FileFormatVersion::LatePreAlpha) {
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.color_r);
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.color_b);
|
|
|
|
|
BinaryIO::BinaryRead(file, scene.color_g);
|
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
|
|
|
|
m_Scenes.insert(std::make_pair(lwoSceneID, scene));
|
|
|
|
|
m_NumberOfScenesLoaded++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::vector<LUTriggers::Trigger*> Zone::LoadLUTriggers(std::string triggerFile, LWOSCENEID sceneID) {
|
|
|
|
|
std::vector<LUTriggers::Trigger*> lvlTriggers;
|
2022-11-01 18:21:26 +00:00
|
|
|
|
|
|
|
|
|
auto buffer = Game::assetManager->GetFileAsBuffer((m_ZonePath + triggerFile).c_str());
|
|
|
|
|
|
2022-11-10 18:59:31 +00:00
|
|
|
|
if (!buffer.m_Success) {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
LOG("Failed to load %s from disk. Skipping loading triggers", (m_ZonePath + triggerFile).c_str());
|
2022-11-10 18:59:31 +00:00
|
|
|
|
return lvlTriggers;
|
|
|
|
|
}
|
2022-11-01 18:21:26 +00:00
|
|
|
|
|
|
|
|
|
std::istream file(&buffer);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
std::stringstream data;
|
|
|
|
|
data << file.rdbuf();
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
buffer.close();
|
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
if (data.str().size() == 0) return lvlTriggers;
|
|
|
|
|
|
|
|
|
|
tinyxml2::XMLDocument* doc = new tinyxml2::XMLDocument();
|
|
|
|
|
if (!doc) return lvlTriggers;
|
|
|
|
|
|
|
|
|
|
if (doc->Parse(data.str().c_str(), data.str().size()) == 0) {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
//LOG("Loaded LUTriggers from file %s!", triggerFile.c_str());
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else {
|
2023-10-21 23:31:55 +00:00
|
|
|
|
LOG("Failed to load LUTriggers from file %s", triggerFile.c_str());
|
2021-12-05 17:54:36 +00:00
|
|
|
|
return lvlTriggers;
|
|
|
|
|
}
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
tinyxml2::XMLElement* triggers = doc->FirstChildElement("triggers");
|
|
|
|
|
if (!triggers) return lvlTriggers;
|
|
|
|
|
|
|
|
|
|
auto currentTrigger = triggers->FirstChildElement("trigger");
|
|
|
|
|
while (currentTrigger) {
|
2022-07-28 13:39:57 +00:00
|
|
|
|
LUTriggers::Trigger* newTrigger = new LUTriggers::Trigger();
|
2021-12-05 17:54:36 +00:00
|
|
|
|
currentTrigger->QueryAttribute("enabled", &newTrigger->enabled);
|
|
|
|
|
currentTrigger->QueryAttribute("id", &newTrigger->id);
|
|
|
|
|
|
|
|
|
|
auto currentEvent = currentTrigger->FirstChildElement("event");
|
|
|
|
|
while (currentEvent) {
|
|
|
|
|
LUTriggers::Event* newEvent = new LUTriggers::Event();
|
2023-02-10 08:29:53 +00:00
|
|
|
|
newEvent->id = TriggerEventType::StringToTriggerEventType(currentEvent->Attribute("id"));
|
2021-12-05 17:54:36 +00:00
|
|
|
|
auto currentCommand = currentEvent->FirstChildElement("command");
|
|
|
|
|
while (currentCommand) {
|
|
|
|
|
LUTriggers::Command* newCommand = new LUTriggers::Command();
|
2023-02-10 08:29:53 +00:00
|
|
|
|
newCommand->id = TriggerCommandType::StringToTriggerCommandType(currentCommand->Attribute("id"));
|
2021-12-05 17:54:36 +00:00
|
|
|
|
newCommand->target = currentCommand->Attribute("target");
|
|
|
|
|
if (currentCommand->Attribute("targetName") != NULL) {
|
|
|
|
|
newCommand->targetName = currentCommand->Attribute("targetName");
|
|
|
|
|
}
|
|
|
|
|
if (currentCommand->Attribute("args") != NULL) {
|
|
|
|
|
newCommand->args = currentCommand->Attribute("args");
|
|
|
|
|
}
|
|
|
|
|
newEvent->commands.push_back(newCommand);
|
|
|
|
|
currentCommand = currentCommand->NextSiblingElement("command");
|
|
|
|
|
}
|
|
|
|
|
newTrigger->events.push_back(newEvent);
|
|
|
|
|
currentEvent = currentEvent->NextSiblingElement("event");
|
|
|
|
|
}
|
|
|
|
|
currentTrigger = currentTrigger->NextSiblingElement("trigger");
|
|
|
|
|
lvlTriggers.push_back(newTrigger);
|
|
|
|
|
}
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
delete doc;
|
|
|
|
|
|
|
|
|
|
return lvlTriggers;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
LUTriggers::Trigger* Zone::GetTrigger(uint32_t sceneID, uint32_t triggerID) {
|
|
|
|
|
if (m_Scenes.find(sceneID) == m_Scenes.end()) return nullptr;
|
|
|
|
|
if (m_Scenes[sceneID].triggers.find(triggerID) == m_Scenes[sceneID].triggers.end()) return nullptr;
|
|
|
|
|
|
|
|
|
|
return m_Scenes[sceneID].triggers[triggerID];
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
|
const Path* Zone::GetPath(std::string name) const {
|
|
|
|
|
for (const auto& path : m_Paths) {
|
|
|
|
|
if (name == path.pathName) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
return &path;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
void Zone::LoadSceneTransition(std::istream& file) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
SceneTransition sceneTrans;
|
2023-08-10 21:35:12 +00:00
|
|
|
|
if (m_FileFormatVersion < Zone::FileFormatVersion::Auramar) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
uint8_t length;
|
|
|
|
|
BinaryIO::BinaryRead(file, length);
|
|
|
|
|
sceneTrans.name = BinaryIO::ReadString(file, length);
|
2023-07-31 06:49:43 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, sceneTrans.width);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//BR<42>THER MAY I HAVE SOME L<><4C>PS?
|
2023-08-10 21:35:12 +00:00
|
|
|
|
uint8_t loops = (m_FileFormatVersion <= Zone::FileFormatVersion::LatePreAlpha || m_FileFormatVersion >= Zone::FileFormatVersion::Launch) ? 2 : 5;
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < loops; ++i) {
|
|
|
|
|
sceneTrans.points.push_back(LoadSceneTransitionInfo(file));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_SceneTransitions.push_back(sceneTrans);
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
SceneTransitionInfo Zone::LoadSceneTransitionInfo(std::istream& file) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
SceneTransitionInfo info;
|
|
|
|
|
BinaryIO::BinaryRead(file, info.sceneID);
|
|
|
|
|
BinaryIO::BinaryRead(file, info.position);
|
|
|
|
|
return info;
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-01 18:21:26 +00:00
|
|
|
|
void Zone::LoadPath(std::istream& file) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
Path path = Path();
|
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, path.pathVersion);
|
2022-11-12 14:44:03 +00:00
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
uint8_t stringLength;
|
|
|
|
|
BinaryIO::BinaryRead(file, stringLength);
|
|
|
|
|
for (uint8_t i = 0; i < stringLength; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
path.pathName.push_back(character);
|
|
|
|
|
}
|
2022-11-12 14:44:03 +00:00
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, path.pathType);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.flags);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.pathBehavior);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
|
|
|
|
|
if (path.pathType == PathType::MovingPlatform) {
|
|
|
|
|
if (path.pathVersion >= 18) {
|
2022-11-12 14:44:03 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.movingPlatform.timeBasedMovement);
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (path.pathVersion >= 13) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
uint8_t count;
|
|
|
|
|
BinaryIO::BinaryRead(file, count);
|
|
|
|
|
for (uint8_t i = 0; i < count; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
path.movingPlatform.platformTravelSound.push_back(character);
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (path.pathType == PathType::Property) {
|
2022-11-12 14:44:03 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.pathType);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.price);
|
2023-08-10 21:35:12 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.rentalTime);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.associatedZone);
|
2022-10-24 05:54:21 +00:00
|
|
|
|
|
|
|
|
|
if (path.pathVersion >= 5) {
|
|
|
|
|
uint8_t count1;
|
|
|
|
|
BinaryIO::BinaryRead(file, count1);
|
|
|
|
|
for (uint8_t i = 0; i < count1; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
path.property.displayName.push_back(character);
|
|
|
|
|
}
|
|
|
|
|
uint32_t count2;
|
|
|
|
|
BinaryIO::BinaryRead(file, count2);
|
|
|
|
|
for (uint8_t i = 0; i < count2; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
path.property.displayDesc.push_back(character);
|
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
2022-10-24 05:54:21 +00:00
|
|
|
|
|
2022-11-12 14:44:03 +00:00
|
|
|
|
if (path.pathVersion >= 6) BinaryIO::BinaryRead(file, path.property.type);
|
2022-10-24 05:54:21 +00:00
|
|
|
|
|
|
|
|
|
if (path.pathVersion >= 7) {
|
|
|
|
|
BinaryIO::BinaryRead(file, path.property.cloneLimit);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.property.repMultiplier);
|
2023-08-10 21:35:12 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.rentalPeriod);
|
2022-10-24 05:54:21 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (path.pathVersion >= 8) {
|
|
|
|
|
BinaryIO::BinaryRead(file, path.property.achievementRequired);
|
2023-08-10 21:35:12 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.playerZoneCoords);
|
2022-10-24 05:54:21 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.property.maxBuildHeight);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (path.pathType == PathType::Camera) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
uint8_t count;
|
|
|
|
|
BinaryIO::BinaryRead(file, count);
|
|
|
|
|
for (uint8_t i = 0; i < count; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
path.camera.nextPath.push_back(character);
|
|
|
|
|
}
|
|
|
|
|
if (path.pathVersion >= 14) {
|
2022-11-12 14:44:03 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, path.camera.rotatePlayer);
|
|
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
} else if (path.pathType == PathType::Spawner) {
|
|
|
|
|
BinaryIO::BinaryRead(file, path.spawner.spawnedLOT);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.spawner.respawnTime);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.spawner.maxToSpawn);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.spawner.amountMaintained);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.spawner.spawnerObjID);
|
|
|
|
|
BinaryIO::BinaryRead(file, path.spawner.spawnerNetActive);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Read waypoints
|
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, path.waypointCount);
|
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < path.waypointCount; ++i) {
|
|
|
|
|
PathWaypoint waypoint = PathWaypoint();
|
|
|
|
|
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.position.x);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.position.y);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.position.z);
|
2022-07-25 02:26:51 +00:00
|
|
|
|
|
|
|
|
|
|
2022-11-12 14:44:03 +00:00
|
|
|
|
if (path.pathType == PathType::Spawner || path.pathType == PathType::MovingPlatform || path.pathType == PathType::Race || path.pathType == PathType::Camera || path.pathType == PathType::Rail) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.rotation.w);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.rotation.x);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.rotation.y);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.rotation.z);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (path.pathType == PathType::MovingPlatform) {
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.movingPlatform.lockPlayer);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.movingPlatform.speed);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.movingPlatform.wait);
|
|
|
|
|
if (path.pathVersion >= 13) {
|
|
|
|
|
uint8_t count1;
|
|
|
|
|
BinaryIO::BinaryRead(file, count1);
|
|
|
|
|
for (uint8_t i = 0; i < count1; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
waypoint.movingPlatform.departSound.push_back(character);
|
|
|
|
|
}
|
|
|
|
|
uint8_t count2;
|
|
|
|
|
BinaryIO::BinaryRead(file, count2);
|
|
|
|
|
for (uint8_t i = 0; i < count2; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
waypoint.movingPlatform.arriveSound.push_back(character);
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (path.pathType == PathType::Camera) {
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.camera.time);
|
2022-11-12 14:44:03 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.camera.fov);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.camera.tension);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.camera.continuity);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.camera.bias);
|
2022-07-28 13:39:57 +00:00
|
|
|
|
} else if (path.pathType == PathType::Race) {
|
2022-11-12 14:44:03 +00:00
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.racing.isResetNode);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.racing.isNonHorizontalCamera);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.racing.planeWidth);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.racing.planeHeight);
|
|
|
|
|
BinaryIO::BinaryRead(file, waypoint.racing.shortestDistanceToEnd);
|
2022-11-14 19:57:49 +00:00
|
|
|
|
} else if (path.pathType == PathType::Rail) {
|
|
|
|
|
if (path.pathVersion > 16) BinaryIO::BinaryRead(file, waypoint.rail.speed);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// object LDF configs
|
|
|
|
|
if (path.pathType == PathType::Movement || path.pathType == PathType::Spawner || path.pathType == PathType::Rail) {
|
|
|
|
|
uint32_t count;
|
|
|
|
|
BinaryIO::BinaryRead(file, count);
|
|
|
|
|
for (uint32_t i = 0; i < count; ++i) {
|
|
|
|
|
uint8_t count1;
|
|
|
|
|
std::string parameter;
|
|
|
|
|
std::string value;
|
|
|
|
|
BinaryIO::BinaryRead(file, count1);
|
|
|
|
|
for (uint8_t i = 0; i < count1; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
parameter.push_back(character);
|
|
|
|
|
}
|
|
|
|
|
uint8_t count2;
|
|
|
|
|
BinaryIO::BinaryRead(file, count2);
|
|
|
|
|
for (uint8_t i = 0; i < count2; ++i) {
|
|
|
|
|
uint16_t character;
|
|
|
|
|
BinaryIO::BinaryRead(file, character);
|
|
|
|
|
value.push_back(character);
|
|
|
|
|
}
|
2022-11-12 14:44:03 +00:00
|
|
|
|
|
|
|
|
|
LDFBaseData* ldfConfig = nullptr;
|
2022-11-14 19:57:49 +00:00
|
|
|
|
if (path.pathType == PathType::Movement || path.pathType == PathType::Rail) {
|
2022-11-12 14:44:03 +00:00
|
|
|
|
ldfConfig = LDFBaseData::DataFromString(parameter + "=0:" + value);
|
|
|
|
|
} else {
|
|
|
|
|
ldfConfig = LDFBaseData::DataFromString(parameter + "=" + value);
|
|
|
|
|
}
|
|
|
|
|
if (ldfConfig) waypoint.config.push_back(ldfConfig);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-08-12 14:20:00 +00:00
|
|
|
|
// We verify the waypoint heights against the navmesh because in many movement paths,
|
|
|
|
|
// the waypoint is located near 0 height,
|
|
|
|
|
if (path.pathType == PathType::Movement) {
|
|
|
|
|
if (dpWorld::Instance().IsLoaded()) {
|
|
|
|
|
// 2000 should be large enough for every world.
|
|
|
|
|
waypoint.position.y = dpWorld::Instance().GetNavMesh()->GetHeightAtPoint(waypoint.position, 2000.0f);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
path.pathWaypoints.push_back(waypoint);
|
|
|
|
|
}
|
|
|
|
|
m_Paths.push_back(path);
|
|
|
|
|
}
|