OpenFusion/src/NPCManager.cpp

106 lines
3.4 KiB
C++
Raw Normal View History

2020-08-19 22:21:35 +00:00
#include "NPCManager.hpp"
2020-08-20 21:43:48 +00:00
#include "settings.hpp"
#include <cmath>
#include <algorithm>
#include <list>
2020-08-21 22:14:11 +00:00
#include <fstream>
2020-08-22 02:03:12 +00:00
#include "contrib/JSON.hpp"
2020-08-20 21:43:48 +00:00
std::map<int32_t, BaseNPC> NPCManager::NPCs;
2020-08-19 22:21:35 +00:00
void NPCManager::init() {
2020-08-21 22:14:11 +00:00
// load NPCs from NPCs.json into our NPC manager
try {
std::ifstream inFile("NPCs.json");
nlohmann::json jsonData;
2020-08-21 22:14:11 +00:00
// read file into jsonData
inFile >> jsonData;
2020-08-21 22:14:11 +00:00
for (auto& npc : jsonData) {
BaseNPC tmp(npc["x"], npc["y"], npc["z"], npc["id"]);
NPCManager::NPCs[tmp.appearanceData.iNPC_ID] = tmp;
}
2020-08-21 22:14:11 +00:00
std::cout << "populated " << NPCs.size() << " NPCs" << std::endl;
}
catch (const std::exception& err) {
std::cerr << "[WARN] Malformed NPC.json file! Reason:" << std::endl << err.what() << std::endl;
2020-08-21 22:14:11 +00:00
}
REGISTER_SHARD_PACKET(P_CL2FE_REQ_PC_WARP_USE_NPC, npcWarpManager);
2020-08-20 21:43:48 +00:00
}
void NPCManager::updatePlayerNPCS(CNSocket* sock, PlayerView& view) {
std::list<int32_t> yesView;
std::list<int32_t> noView;
for (auto& pair : NPCs) {
2020-08-20 21:43:48 +00:00
int diffX = abs(view.plr.x - pair.second.appearanceData.iX);
int diffY = abs(view.plr.y - pair.second.appearanceData.iY);
if (diffX < settings::VIEWDISTANCE && diffY < settings::VIEWDISTANCE) {
yesView.push_back(pair.first);
} else {
noView.push_back(pair.first);
}
}
2020-08-23 00:26:18 +00:00
INITSTRUCT(sP_FE2CL_NPC_EXIT, exitData);
2020-08-20 21:43:48 +00:00
std::list<int32_t>::iterator i = view.viewableNPCs.begin();
while (i != view.viewableNPCs.end()) {
int32_t id = *i;
if (std::find(noView.begin(), noView.end(), id) != noView.end()) {
// it shouldn't be visible, send NPC_EXIT
2020-08-22 23:31:09 +00:00
exitData.iNPC_ID = id;
sock->sendPacket((void*)&exitData, P_FE2CL_NPC_EXIT, sizeof(sP_FE2CL_NPC_EXIT));
2020-08-20 21:43:48 +00:00
// remove from view
view.viewableNPCs.erase(i++);
}
++i;
}
2020-08-23 00:26:18 +00:00
INITSTRUCT(sP_FE2CL_NPC_ENTER, enterData);
2020-08-20 21:43:48 +00:00
for (int32_t id : yesView) {
if (std::find(view.viewableNPCs.begin(), view.viewableNPCs.end(), id) == view.viewableNPCs.end()) {
// needs to be added to viewableNPCs! send NPC_ENTER
2020-08-22 23:31:09 +00:00
enterData.NPCAppearanceData = NPCs[id].appearanceData;
sock->sendPacket((void*)&enterData, P_FE2CL_NPC_ENTER, sizeof(sP_FE2CL_NPC_ENTER));
2020-08-20 21:43:48 +00:00
2020-08-22 23:31:09 +00:00
// add to viewable
2020-08-20 21:43:48 +00:00
view.viewableNPCs.push_back(id);
}
}
PlayerManager::players[sock].viewableNPCs = view.viewableNPCs;
}
void NPCManager::npcWarpManager(CNSocket* sock, CNPacketData* data)
{
std::ifstream warp_file("warps.json", std::ifstream::binary);
nlohmann::json warp;
warp_file >> warp;
if (data->size != sizeof(sP_CL2FE_REQ_PC_WARP_USE_NPC))
return; // malformed packet
sP_CL2FE_REQ_PC_WARP_USE_NPC* warpNpc = (sP_CL2FE_REQ_PC_WARP_USE_NPC*)data->buf;
//Send to Client
INITSTRUCT(sP_FE2CL_REP_PC_WARP_USE_NPC_SUCC, resp);
resp.iX = warp[std::to_string(warpNpc->iWarpID)]["m_iToX"];
resp.iY = warp[std::to_string(warpNpc->iWarpID)]["m_iToY"];
resp.iZ = warp[std::to_string(warpNpc->iWarpID)]["m_iToZ"];
//Add Instance Stuff Later
std::cerr << "OpenFusion: Warp using " << "Warp ID" <<warpNpc->iWarpID << "Warp to Z:"<< warp[std::to_string(warpNpc->iWarpID)]["m_iToZ"] << std::endl;
sock->sendPacket((void*)&resp, P_FE2CL_REP_PC_WARP_USE_NPC_SUCC, sizeof(sP_FE2CL_REP_PC_WARP_USE_NPC_SUCC));
}