2020-09-09 17:06:22 +00:00
|
|
|
#include "TableData.hpp"
|
|
|
|
#include "NPCManager.hpp"
|
2020-09-13 20:26:16 +00:00
|
|
|
#include "TransportManager.hpp"
|
2020-09-13 22:54:47 +00:00
|
|
|
#include "ItemManager.hpp"
|
2020-09-09 17:06:22 +00:00
|
|
|
#include "settings.hpp"
|
2020-09-09 19:09:01 +00:00
|
|
|
#include "MissionManager.hpp"
|
2020-09-16 23:43:48 +00:00
|
|
|
#include "MobManager.hpp"
|
2020-09-17 22:45:43 +00:00
|
|
|
#include "ChunkManager.hpp"
|
2020-09-09 19:09:01 +00:00
|
|
|
|
2020-09-09 17:06:22 +00:00
|
|
|
#include "contrib/JSON.hpp"
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
|
2020-09-09 19:09:01 +00:00
|
|
|
void TableData::init() {
|
2020-09-09 17:06:22 +00:00
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
// load NPCs from NPC.json
|
|
|
|
try {
|
|
|
|
std::ifstream inFile(settings::NPCJSON);
|
|
|
|
nlohmann::json npcData;
|
|
|
|
|
|
|
|
// read file into json
|
|
|
|
inFile >> npcData;
|
|
|
|
|
2020-09-16 23:43:48 +00:00
|
|
|
for (nlohmann::json::iterator _npc = npcData.begin(); _npc != npcData.end(); _npc++) {
|
|
|
|
auto npc = _npc.value();
|
|
|
|
BaseNPC *tmp = new BaseNPC(npc["x"], npc["y"], npc["z"], npc["id"]);
|
2020-09-14 13:53:48 +00:00
|
|
|
|
2020-09-09 17:06:22 +00:00
|
|
|
// Temporary fix, IDs will be pulled from json later
|
2020-09-16 23:43:48 +00:00
|
|
|
tmp->appearanceData.iNPC_ID = i;
|
2020-09-14 13:53:48 +00:00
|
|
|
|
2020-09-17 22:45:43 +00:00
|
|
|
NPCManager::NPCs[i] = tmp;
|
|
|
|
ChunkManager::addNPC(npc["x"], npc["y"], i);
|
|
|
|
i++;
|
2020-09-09 17:06:22 +00:00
|
|
|
|
2020-09-16 23:43:48 +00:00
|
|
|
if (npc["id"] == 641 || npc["id"] == 642)
|
|
|
|
NPCManager::RespawnPoints.push_back({ npc["x"], npc["y"], ((int)npc["z"]) + RESURRECT_HEIGHT });
|
2020-09-09 17:06:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
catch (const std::exception& err) {
|
|
|
|
std::cerr << "[WARN] Malformed NPCs.json file! Reason:" << err.what() << std::endl;
|
|
|
|
}
|
|
|
|
|
2020-09-17 02:27:21 +00:00
|
|
|
// load paths
|
|
|
|
try {
|
|
|
|
std::ifstream inFile(settings::PATHJSON);
|
|
|
|
nlohmann::json pathData;
|
|
|
|
|
|
|
|
// read file into json
|
|
|
|
inFile >> pathData;
|
|
|
|
|
|
|
|
nlohmann::json pathDataSkyway = pathData["skyway"];
|
|
|
|
for (nlohmann::json::iterator skywayPath = pathDataSkyway.begin(); skywayPath != pathDataSkyway.end(); skywayPath++) {
|
2020-09-17 09:04:00 +00:00
|
|
|
constructPath(skywayPath);
|
2020-09-17 02:27:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "[INFO] Loaded " << TransportManager::SkywayPaths.size() << " skyway paths" << std::endl;
|
|
|
|
}
|
|
|
|
catch (const std::exception& err) {
|
|
|
|
std::cerr << "[WARN] Malformed paths.json file! Reason:" << err.what() << std::endl;
|
|
|
|
}
|
2020-09-09 17:06:22 +00:00
|
|
|
// load everything else from xdttable
|
2020-09-09 22:31:09 +00:00
|
|
|
std::cout << "[INFO] Parsing xdt.json..." << std::endl;
|
2020-09-09 19:09:01 +00:00
|
|
|
std::ifstream infile(settings::XDTJSON);
|
|
|
|
nlohmann::json xdtData;
|
2020-09-09 17:06:22 +00:00
|
|
|
|
2020-09-09 19:09:01 +00:00
|
|
|
// read file into json
|
|
|
|
infile >> xdtData;
|
2020-09-09 17:06:22 +00:00
|
|
|
|
2020-09-09 19:09:01 +00:00
|
|
|
try {
|
2020-09-09 22:31:09 +00:00
|
|
|
// load warps
|
2020-09-09 17:06:22 +00:00
|
|
|
nlohmann::json warpData = xdtData["m_pInstanceTable"]["m_pWarpData"];
|
|
|
|
|
2020-09-16 23:43:48 +00:00
|
|
|
for (nlohmann::json::iterator _warp = warpData.begin(); _warp != warpData.end(); _warp++) {
|
|
|
|
auto warp = _warp.value();
|
|
|
|
WarpLocation warpLoc = { warp["m_iToX"], warp["m_iToY"], warp["m_iToZ"] };
|
|
|
|
int warpID = warp["m_iWarpNumber"];
|
2020-09-09 17:06:22 +00:00
|
|
|
NPCManager::Warps[warpID] = warpLoc;
|
|
|
|
}
|
|
|
|
|
2020-09-10 16:40:38 +00:00
|
|
|
std::cout << "[INFO] Populated " << NPCManager::Warps.size() << " Warps" << std::endl;
|
2020-09-09 19:09:01 +00:00
|
|
|
|
2020-09-13 20:26:16 +00:00
|
|
|
// load transport routes and locations
|
|
|
|
nlohmann::json transRouteData = xdtData["m_pTransportationTable"]["m_pTransportationData"];
|
|
|
|
nlohmann::json transLocData = xdtData["m_pTransportationTable"]["m_pTransportationWarpLocation"];
|
|
|
|
|
2020-09-16 23:43:48 +00:00
|
|
|
for (nlohmann::json::iterator _tLoc = transLocData.begin(); _tLoc != transLocData.end(); _tLoc++) {
|
|
|
|
auto tLoc = _tLoc.value();
|
|
|
|
TransportLocation transLoc = { tLoc["m_iNPCID"], tLoc["m_iXpos"], tLoc["m_iYpos"], tLoc["m_iZpos"] };
|
|
|
|
TransportManager::Locations[tLoc["m_iLocationID"]] = transLoc;
|
2020-09-13 20:26:16 +00:00
|
|
|
}
|
|
|
|
std::cout << "[INFO] Loaded " << TransportManager::Locations.size() << " S.C.A.M.P.E.R. locations" << std::endl; // TODO: Skyway operates differently
|
|
|
|
|
2020-09-16 23:43:48 +00:00
|
|
|
for (nlohmann::json::iterator _tRoute = transRouteData.begin(); _tRoute != transRouteData.end(); _tRoute++) {
|
|
|
|
auto tRoute = _tRoute.value();
|
|
|
|
TransportRoute transRoute = { tRoute["m_iMoveType"], tRoute["m_iStartLocation"], tRoute["m_iEndLocation"],
|
|
|
|
tRoute["m_iCost"] , tRoute["m_iSpeed"], tRoute["m_iRouteNum"] };
|
|
|
|
TransportManager::Routes[tRoute["m_iVehicleID"]] = transRoute;
|
2020-09-13 20:26:16 +00:00
|
|
|
}
|
|
|
|
std::cout << "[INFO] Loaded " << TransportManager::Routes.size() << " transportation routes" << std::endl;
|
|
|
|
|
2020-09-10 13:01:35 +00:00
|
|
|
// load mission-related data
|
2020-09-09 19:09:01 +00:00
|
|
|
nlohmann::json tasks = xdtData["m_pMissionTable"]["m_pMissionData"];
|
|
|
|
|
|
|
|
for (auto _task = tasks.begin(); _task != tasks.end(); _task++) {
|
|
|
|
auto task = _task.value();
|
|
|
|
|
|
|
|
// rewards
|
|
|
|
if (task["m_iSUReward"] != 0) {
|
2020-09-10 16:40:38 +00:00
|
|
|
auto _rew = xdtData["m_pMissionTable"]["m_pRewardData"][(int)task["m_iSUReward"]];
|
|
|
|
Reward *rew = new Reward(_rew["m_iMissionRewardID"], _rew["m_iMissionRewarItemType"],
|
|
|
|
_rew["m_iMissionRewardItemID"], _rew["m_iCash"], _rew["m_iFusionMatter"]);
|
2020-09-09 19:09:01 +00:00
|
|
|
|
|
|
|
MissionManager::Rewards[task["m_iHTaskID"]] = rew;
|
|
|
|
}
|
2020-09-09 22:31:09 +00:00
|
|
|
|
2020-09-10 16:40:38 +00:00
|
|
|
// everything else lol. see TaskData comment.
|
|
|
|
MissionManager::Tasks[task["m_iHTaskID"]] = new TaskData(task);
|
2020-09-09 19:09:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "[INFO] Loaded mission-related data" << std::endl;
|
2020-09-13 22:54:47 +00:00
|
|
|
|
2020-09-14 04:25:14 +00:00
|
|
|
// load all item data. i'm sorry. it has to be done
|
2020-09-14 05:27:20 +00:00
|
|
|
const char* setNames[12] = { "m_pBackItemTable", "m_pFaceItemTable", "m_pGlassItemTable", "m_pHatItemTable",
|
2020-09-14 04:25:14 +00:00
|
|
|
"m_pHeadItemTable", "m_pPantsItemTable", "m_pShirtsItemTable", "m_pShoesItemTable", "m_pWeaponItemTable",
|
2020-09-14 05:27:20 +00:00
|
|
|
"m_pVehicleItemTable", "m_pGeneralItemTable", "m_pChestItemTable" };
|
2020-09-14 04:25:14 +00:00
|
|
|
nlohmann::json itemSet;
|
2020-09-14 05:27:20 +00:00
|
|
|
for (int i = 0; i < 12; i++) {
|
2020-09-14 04:25:14 +00:00
|
|
|
itemSet = xdtData[setNames[i]]["m_pItemData"];
|
2020-09-16 23:43:48 +00:00
|
|
|
for (nlohmann::json::iterator _item = itemSet.begin(); _item != itemSet.end(); _item++) {
|
|
|
|
auto item = _item.value();
|
2020-09-19 20:22:28 +00:00
|
|
|
int typeOverride = getItemType(i); // used for special cases where iEquipLoc doesn't indicate item type
|
2020-09-17 16:59:30 +00:00
|
|
|
ItemManager::ItemData[std::pair<int32_t, int32_t>(item["m_iItemNumber"], typeOverride != -1 ? typeOverride : (int)item["m_iEquipLoc"])]
|
2020-09-16 00:30:01 +00:00
|
|
|
= { item["m_iTradeAble"] == 1, item["m_iSellAble"] == 1, item["m_iItemPrice"], item["m_iItemSellPrice"], item["m_iStackNumber"], i > 9 ? 0 : (int)item["m_iMinReqLev"],
|
|
|
|
i > 9 ? 1 : (int)item["m_iRarity"] };
|
2020-09-16 23:43:48 +00:00
|
|
|
}
|
2020-09-14 04:25:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "[INFO] Loaded " << ItemManager::ItemData.size() << " items" << std::endl;
|
|
|
|
|
2020-09-13 22:54:47 +00:00
|
|
|
// load vendor listings
|
|
|
|
nlohmann::json listings = xdtData["m_pVendorTable"]["m_pItemData"];
|
|
|
|
|
2020-09-16 23:43:48 +00:00
|
|
|
for (nlohmann::json::iterator _lst = listings.begin(); _lst != listings.end(); _lst++) {
|
|
|
|
auto lst = _lst.value();
|
|
|
|
VendorListing vListing = { lst["m_iSortNumber"], lst["m_iItemType"], lst["m_iitemID"] };
|
|
|
|
ItemManager::VendorTables[lst["m_iNpcNumber"]].push_back(vListing);
|
2020-09-13 22:54:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "[INFO] Loaded " << ItemManager::VendorTables.size() << " vendor tables" << std::endl;
|
2020-09-16 00:30:01 +00:00
|
|
|
|
|
|
|
// load crocpot entries
|
|
|
|
nlohmann::json crocs = xdtData["m_pCombiningTable"]["m_pCombiningData"];
|
|
|
|
|
|
|
|
for (nlohmann::json::iterator croc = crocs.begin(); croc != crocs.end(); croc++) {
|
|
|
|
CrocPotEntry crocEntry = { croc.value()["m_iStatConstant"], croc.value()["m_iLookConstant"], croc.value()["m_fLevelGapStandard"],
|
|
|
|
croc.value()["m_fSameGrade"], croc.value()["m_fOneGrade"], croc.value()["m_fTwoGrade"], croc.value()["m_fThreeGrade"] };
|
|
|
|
ItemManager::CrocPotTable[croc.value()["m_iLevelGap"]] = crocEntry;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "[INFO] Loaded " << ItemManager::CrocPotTable.size() << " croc pot value sets" << std::endl;
|
2020-09-09 17:06:22 +00:00
|
|
|
}
|
|
|
|
catch (const std::exception& err) {
|
|
|
|
std::cerr << "[WARN] Malformed xdt.json file! Reason:" << err.what() << std::endl;
|
|
|
|
}
|
2020-09-16 23:43:48 +00:00
|
|
|
|
|
|
|
// load temporary mob dump
|
|
|
|
try {
|
|
|
|
std::ifstream inFile(settings::MOBJSON);
|
|
|
|
nlohmann::json npcData;
|
|
|
|
|
|
|
|
// read file into json
|
|
|
|
inFile >> npcData;
|
|
|
|
|
|
|
|
nlohmann::json npcTableData = xdtData["m_pNpcTable"]["m_pNpcData"];
|
|
|
|
|
|
|
|
for (nlohmann::json::iterator _npc = npcData.begin(); _npc != npcData.end(); _npc++) {
|
|
|
|
auto npc = _npc.value();
|
|
|
|
auto td = npcTableData[(int)npc["iNPCType"]];
|
|
|
|
Mob *tmp = new Mob(npc["iX"], npc["iY"], npc["iZ"], npc["iNPCType"], npc["iHP"], npc["iAngle"], td["m_iRegenTime"]);
|
|
|
|
|
|
|
|
// Temporary fix, IDs will be pulled from json later
|
|
|
|
tmp->appearanceData.iNPC_ID = i;
|
|
|
|
|
|
|
|
NPCManager::NPCs[i] = tmp;
|
|
|
|
MobManager::Mobs[i] = (Mob*)NPCManager::NPCs[i];
|
2020-09-17 22:45:43 +00:00
|
|
|
ChunkManager::addNPC(npc["iX"], npc["iY"], i);
|
2020-09-16 23:43:48 +00:00
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "[INFO] Populated " << NPCManager::NPCs.size() << " NPCs" << std::endl;
|
|
|
|
}
|
|
|
|
catch (const std::exception& err) {
|
|
|
|
std::cerr << "[WARN] Malformed mobs.json file! Reason:" << err.what() << std::endl;
|
|
|
|
}
|
2020-09-09 19:09:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TableData::cleanup() {
|
|
|
|
/*
|
|
|
|
* This is just to shut the address sanitizer up. Dynamically allocated data
|
|
|
|
* doesn't need to be cleaned up if it's supposed to last the program's full runtime.
|
|
|
|
*/
|
|
|
|
for (auto& pair : MissionManager::Rewards)
|
|
|
|
delete pair.second;
|
2020-09-10 16:40:38 +00:00
|
|
|
for (auto& pair : MissionManager::Tasks)
|
2020-09-10 13:01:35 +00:00
|
|
|
delete pair.second;
|
2020-09-16 23:43:48 +00:00
|
|
|
for (auto& pair : NPCManager::NPCs)
|
|
|
|
delete pair.second;
|
2020-09-09 19:09:01 +00:00
|
|
|
}
|
2020-09-17 16:59:30 +00:00
|
|
|
|
2020-09-19 20:22:28 +00:00
|
|
|
/*
|
|
|
|
* Some item categories either don't possess iEquipLoc or use a different value for item type.
|
|
|
|
*/
|
2020-09-17 16:59:30 +00:00
|
|
|
int TableData::getItemType(int itemSet) {
|
|
|
|
int overriden;
|
|
|
|
switch (itemSet)
|
|
|
|
{
|
2020-09-19 20:22:28 +00:00
|
|
|
case 11: // Chest items don't have iEquipLoc and are type 9.
|
2020-09-17 16:59:30 +00:00
|
|
|
overriden = 9;
|
|
|
|
break;
|
2020-09-19 20:22:28 +00:00
|
|
|
case 10: // General items don't have iEquipLoc and are type 7.
|
2020-09-17 16:59:30 +00:00
|
|
|
overriden = 7;
|
|
|
|
break;
|
2020-09-19 20:22:28 +00:00
|
|
|
case 9: // Vehicles have iEquipLoc 8, but type 10.
|
2020-09-17 16:59:30 +00:00
|
|
|
overriden = 10;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
overriden = -1;
|
|
|
|
}
|
|
|
|
return overriden;
|
|
|
|
}
|
2020-09-17 09:04:00 +00:00
|
|
|
|
|
|
|
void TableData::constructPath(nlohmann::json::iterator pathData) {
|
|
|
|
// Interpolate
|
|
|
|
nlohmann::json pathPoints = pathData.value()["points"];
|
|
|
|
std::queue<WarpLocation> points;
|
|
|
|
nlohmann::json::iterator point = pathPoints.begin();
|
|
|
|
WarpLocation last = { point.value()["iX"] , point.value()["iY"] , point.value()["iZ"] }; // start pos
|
|
|
|
// use some for loop trickery
|
|
|
|
for (point++; point != pathPoints.end(); point++) {
|
|
|
|
WarpLocation coords = { point.value()["iX"] , point.value()["iY"] , point.value()["iZ"] };
|
|
|
|
// avoiding pow here
|
|
|
|
int distanceBetween = sqrt((last.x - coords.x) * (last.x - coords.x) + (last.y - coords.y) * (last.y - coords.y) + (last.z - coords.z) * (last.z - coords.z));
|
|
|
|
int lerps = distanceBetween / (int)pathData.value()["iMaxGapSize"]; // integer division to ensure a whole number
|
|
|
|
for (int i = 0; i < lerps; i++) {
|
|
|
|
WarpLocation lerp;
|
|
|
|
float frac = (i + 1) * 1.0f / (lerps + 1);
|
|
|
|
lerp.x = (last.x * (1.0f - frac)) + (coords.x * frac);
|
|
|
|
lerp.y = (last.y * (1.0f - frac)) + (coords.y * frac);
|
|
|
|
lerp.z = (last.z * (1.0f - frac)) + (coords.z * frac);
|
|
|
|
points.push(lerp); // add lerp'd point to the queue
|
|
|
|
}
|
|
|
|
points.push(coords);
|
|
|
|
last = coords; // update start pos
|
|
|
|
}
|
|
|
|
TransportManager::SkywayPaths[pathData.value()["iRouteID"]] = points;
|
|
|
|
}
|