region_map_v1.cpp 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849
  1. #include "region_map_v1.h"
  2. #include "../../common/Log.h"
  3. #include "../client.h"
  4. #include "../Spawn.h"
  5. #include "../LuaInterface.h"
  6. #include "../World.h"
  7. #undef snprintf
  8. #include <boost/filesystem.hpp>
  9. extern LuaInterface* lua_interface;
  10. extern World world;
  11. RegionMapV1::RegionMapV1() {
  12. mVersion = 1;
  13. }
  14. RegionMapV1::~RegionMapV1() {
  15. std::unique_lock lock(MRegions);
  16. map<Region_Node*, ZBSP_Node*>::const_iterator itr;
  17. int region_num = 0;
  18. for (itr = Regions.begin(); itr != Regions.end();)
  19. {
  20. Region_Node* node = itr->first;
  21. ZBSP_Node* bsp_node = itr->second;
  22. map<Region_Node*, ZBSP_Node*>::const_iterator deleteItr = itr;
  23. itr++;
  24. Regions.erase(deleteItr);
  25. safe_delete(node);
  26. safe_delete_array(bsp_node);
  27. }
  28. Regions.clear();
  29. }
  30. WaterRegionType RegionMapV1::ReturnRegionType(const glm::vec3& location, int32 gridid) const {
  31. return BSPReturnRegionType(1, glm::vec3(location.x, location.y, location.z), gridid);
  32. }
  33. bool RegionMapV1::InWater(const glm::vec3& location, int32 gridid) const {
  34. return ReturnRegionType(location, gridid) == RegionTypeWater;
  35. }
  36. bool RegionMapV1::InLava(const glm::vec3& location, int32 gridid) const {
  37. return ReturnRegionType(location, gridid) == RegionTypeLava;
  38. }
  39. bool RegionMapV1::InLiquid(const glm::vec3& location) const {
  40. return InWater(location) || InLava(location);
  41. }
  42. bool RegionMapV1::InPvP(const glm::vec3& location) const {
  43. return ReturnRegionType(location) == RegionTypePVP;
  44. }
  45. bool RegionMapV1::InZoneLine(const glm::vec3& location) const {
  46. return ReturnRegionType(location) == RegionTypeZoneLine;
  47. }
  48. std::string RegionMapV1::TestFile(std::string testFile)
  49. {
  50. std::string tmpStr(testFile);
  51. std::size_t pos = tmpStr.find(".");
  52. if ( pos != testFile.npos )
  53. tmpStr = testFile.substr (0, pos);
  54. string tmpScript("RegionScripts/");
  55. tmpScript.append(mZoneNameLower);
  56. tmpScript.append("/" + tmpStr + ".lua");
  57. std::ifstream f(tmpScript.c_str());
  58. return f.good() ? tmpScript : string("");
  59. }
  60. bool RegionMapV1::Load(FILE* fp, std::string inZoneNameLwr, int32 version) {
  61. mZoneNameLower = string(inZoneNameLwr.c_str());
  62. uint32 region_size;
  63. if (fread(&region_size, sizeof(uint32), 1, fp) != 1) {
  64. return false;
  65. }
  66. LogWrite(REGION__DEBUG, 0, "RegionMap", "region count = %u", region_size);
  67. for (int i = 0; i < region_size; i++)
  68. {
  69. uint32 region_num;
  70. if (fread(&region_num, sizeof(uint32), 1, fp) != 1) {
  71. return false;
  72. }
  73. uint32 region_type;
  74. if (fread(&region_type, sizeof(uint32), 1, fp) != 1) {
  75. return false;
  76. }
  77. float x, y, z, dist;
  78. if (fread(&x, sizeof(float), 1, fp) != 1) {
  79. return false;
  80. }
  81. if (fread(&y, sizeof(float), 1, fp) != 1) {
  82. return false;
  83. }
  84. if (fread(&z, sizeof(float), 1, fp) != 1) {
  85. return false;
  86. }
  87. if (fread(&dist, sizeof(float), 1, fp) != 1) {
  88. return false;
  89. }
  90. int8 strSize;
  91. char envName[256] = {""};
  92. char regionName[256] = {""};
  93. uint32 grid_id = 0;
  94. if ( version > 1 )
  95. {
  96. fread(&strSize, sizeof(int8), 1, fp);
  97. LogWrite(REGION__DEBUG, 7, "Region", "Region environment strSize = %u", strSize);
  98. if(strSize)
  99. {
  100. size_t len = fread(&envName, sizeof(char), strSize, fp);
  101. envName[len] = '\0';
  102. }
  103. LogWrite(REGION__DEBUG, 7, "Region", "Region environment file name = %s", envName);
  104. fread(&strSize, sizeof(int8), 1, fp);
  105. LogWrite(REGION__DEBUG, 7, "Region", "Region name strSize = %u", strSize);
  106. if(strSize)
  107. {
  108. size_t len = fread(&regionName, sizeof(char), strSize, fp);
  109. regionName[len] = '\0';
  110. }
  111. LogWrite(REGION__DEBUG, 7, "Region", "Region name file name = %s", regionName);
  112. if (fread(&grid_id, sizeof(uint32), 1, fp) != 1) {
  113. return false;
  114. }
  115. }
  116. int32 bsp_tree_size;
  117. if (fread(&bsp_tree_size, sizeof(int32), 1, fp) != 1) {
  118. return false;
  119. }
  120. LogWrite(REGION__DEBUG, 7, "Region", "region x,y,z,dist = %f, %f, %f, %f, region bsp tree size: %i\n", x, y, z, dist, bsp_tree_size);
  121. ZBSP_Node* BSP_Root = new ZBSP_Node[bsp_tree_size];
  122. if (fread(BSP_Root, sizeof(ZBSP_Node), bsp_tree_size, fp) != bsp_tree_size) {
  123. LogWrite(REGION__ERROR, 0, "RegionMap", "Failed to load region.");
  124. return false;
  125. }
  126. Region_Node* tmpNode = new Region_Node;
  127. tmpNode->x = x;
  128. tmpNode->y = y;
  129. tmpNode->z = z;
  130. tmpNode->dist = dist;
  131. tmpNode->region_type = region_type;
  132. tmpNode->regionName = string(regionName);
  133. tmpNode->regionEnvFileName = string(envName);
  134. tmpNode->grid_id = grid_id;
  135. tmpNode->regionScriptName = string("");
  136. tmpNode->regionScriptName = TestFile(regionName);
  137. if ( tmpNode->regionScriptName.size() < 1 )
  138. {
  139. tmpNode->regionScriptName = TestFile(envName);
  140. }
  141. if ( tmpNode->regionScriptName.size() < 1 )
  142. {
  143. tmpNode->regionScriptName = TestFile("default");
  144. }
  145. tmpNode->vert_count = bsp_tree_size;
  146. MRegions.lock();
  147. Regions.insert(make_pair(tmpNode, BSP_Root));
  148. MRegions.unlock();
  149. }
  150. fclose(fp);
  151. LogWrite(REGION__DEBUG, 0, "RegionMap", "completed load!");
  152. return true;
  153. }
  154. void RegionMapV1::IdentifyRegionsInGrid(Client *client, const glm::vec3 &location) const
  155. {
  156. std::shared_lock lock(MRegions);
  157. map<Region_Node*, ZBSP_Node*>::const_iterator itr;
  158. int region_num = 0;
  159. int32 grid = 0;
  160. int32 widget_id = 0;
  161. float x =0.0f,y = 0.0f,z = 0.0f;
  162. if (client->GetPlayer()->GetMap() != nullptr && client->GetPlayer()->GetMap()->IsMapLoaded())
  163. {
  164. auto loc = glm::vec3(location.x, location.z, location.y);
  165. float new_z = client->GetPlayer()->GetMap()->FindBestZ(loc, nullptr, &grid, &widget_id);
  166. std::map<int32, glm::vec3>::iterator itr = client->GetPlayer()->GetMap()->widget_map.find(widget_id);
  167. if(itr != client->GetPlayer()->GetMap()->widget_map.end()) {
  168. x = itr->second.x;
  169. y = itr->second.y;
  170. z = itr->second.z;
  171. }
  172. }
  173. else
  174. client->SimpleMessage(CHANNEL_COLOR_RED, "No map to establish grid id, using grid id 0 (attempt match all).");
  175. client->Message(2, "Region check against location %f / %f / %f. Grid to try: %u, player grid is %u, widget id is %u. Widget location is %f %f %f.", location.x, location.y, location.z, grid, client->GetPlayer()->appearance.pos.grid_id, widget_id, x, y, z);
  176. for (itr = Regions.begin(); itr != Regions.end(); itr++)
  177. {
  178. Region_Node *node = itr->first;
  179. ZBSP_Node *BSP_Root = itr->second;
  180. if (grid == 0 || node->grid_id == grid)
  181. {
  182. float x1 = node->x - location.x;
  183. float y1 = node->y - location.y;
  184. float z1 = node->z - location.z;
  185. float dist = sqrt(x1 *x1 + y1 *y1 + z1 *z1);
  186. glm::vec3 testLoc(location.x, location.y, location.z);
  187. if(!BSP_Root) {
  188. if(client)
  189. client->Message(CHANNEL_COLOR_YELLOW, "[%s] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, Script: %s. X: %f, Y: %f, Z: %f, Distance: %f, Widget ID Marker: %u", (widget_id == node->trigger_widget_id) ? "IN REGION" : "WIDGET MARKER", region_num,
  190. node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(), node->regionScriptName.c_str(), node->x, node->y, node->z, node->dist, node->trigger_widget_id);
  191. }
  192. else if (dist <= node->dist)
  193. {
  194. WaterRegionType regionType = RegionTypeUntagged;
  195. if (node->region_type == ClassWaterRegion)
  196. regionType = BSPReturnRegionWaterRegion(node, BSP_Root, 1, testLoc, dist);
  197. else
  198. regionType = BSPReturnRegionTypeNode(node, BSP_Root, 1, testLoc, dist);
  199. if (regionType != RegionTypeNormal)
  200. {
  201. client->Message(CHANNEL_COLOR_YELLOW, "[DETECTED IN REGION %i] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", regionType, region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  202. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  203. }
  204. else
  205. {
  206. client->Message(CHANNEL_COLOR_RED, "[IN DIST RANGE] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  207. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  208. }
  209. }
  210. else
  211. client->Message(CHANNEL_COLOR_RED, "[OUT OF RANGE] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  212. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  213. }
  214. else
  215. client->Message(CHANNEL_COLOR_RED, "[OUT OF GRID] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  216. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  217. region_num++;
  218. }
  219. }
  220. void RegionMapV1::MapRegionsNearSpawn(Spawn *spawn, Client *client) const
  221. {
  222. std::shared_lock lock(MRegions);
  223. map<Region_Node*, ZBSP_Node*>::const_iterator itr;
  224. int region_num = 0;
  225. spawn->RegionMutex.writelock();
  226. glm::vec3 testLoc(spawn->GetX(), spawn->GetY(), spawn->GetZ());
  227. for (itr = Regions.begin(); itr != Regions.end(); itr++)
  228. {
  229. Region_Node *node = itr->first;
  230. ZBSP_Node *BSP_Root = itr->second;
  231. if (node->regionScriptName.size() < 1) // only track ones that are used with LUA scripting
  232. continue;
  233. if(!BSP_Root) {
  234. int32 currentGridID = spawn->appearance.pos.grid_id;
  235. bool inRegion = false;
  236. if(!(inRegion = spawn->InRegion(node, nullptr)) && currentGridID == node->grid_id &&
  237. ( node->trigger_widget_id == spawn->trigger_widget_id || (node->dist > 0.0f && spawn->GetDistance(node->x, node->y, node->z) < node->dist)) ) {
  238. int32 returnValue = spawn->InsertRegionToSpawn(node, nullptr, RegionTypeUntagged);
  239. if (client)
  240. client->Message(CHANNEL_COLOR_YELLOW, "[ENTER REGION %i %u] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", RegionTypeUntagged, returnValue, region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  241. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  242. }
  243. continue;
  244. }
  245. float x1 = node->x - testLoc.x;
  246. float y1 = node->y - testLoc.y;
  247. float z1 = node->z - testLoc.z;
  248. float dist = sqrt(x1 *x1 + y1 *y1 + z1 *z1);
  249. if (dist <= node->dist)
  250. {
  251. WaterRegionType regionType = RegionTypeUntagged;
  252. if (node->region_type == ClassWaterRegion)
  253. regionType = BSPReturnRegionWaterRegion(node, BSP_Root, 1, testLoc, dist);
  254. else
  255. regionType = BSPReturnRegionTypeNode(node, BSP_Root, 1, testLoc, dist);
  256. if (regionType != RegionTypeNormal)
  257. {
  258. if (!spawn->InRegion(node, BSP_Root))
  259. {
  260. spawn->DeleteRegion(node, BSP_Root);
  261. int32 returnValue = spawn->InsertRegionToSpawn(node, BSP_Root, regionType);
  262. if (client)
  263. client->Message(CHANNEL_COLOR_YELLOW, "[ENTER REGION %i %u] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", regionType, returnValue, region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  264. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  265. }
  266. }
  267. else
  268. {
  269. if(spawn->HasRegionTracked(node, BSP_Root, false)) {
  270. continue;
  271. } // UpdateRegionsNearSpawn will capture it for nodes that have BSP_Root's
  272. if (spawn->InRegion(node, BSP_Root))
  273. {
  274. if (client)
  275. client->Message(CHANNEL_COLOR_RED, "[LEAVE REGION] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  276. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  277. WaterRegionType whatWasRegionType = (WaterRegionType) spawn->GetRegionType(node, BSP_Root);
  278. lua_interface->RunRegionScript(node->regionScriptName, "LeaveRegion", spawn->GetZone(), spawn, whatWasRegionType);
  279. }
  280. spawn->DeleteRegion(node, BSP_Root);
  281. spawn->InsertRegionToSpawn(node, BSP_Root, RegionTypeNormal, false);
  282. if (client)
  283. client->Message(CHANNEL_COLOR_RED, "[NEAR REGION] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  284. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  285. }
  286. }
  287. region_num++;
  288. }
  289. spawn->RegionMutex.releasewritelock();
  290. }
  291. void RegionMapV1::UpdateRegionsNearSpawn(Spawn *spawn, Client *client) const
  292. {
  293. std::shared_lock lock(MRegions);
  294. map<map<Region_Node*, ZBSP_Node*>, Region_Status>::iterator testitr;
  295. int region_num = 0;
  296. spawn->RegionMutex.writelock();
  297. glm::vec3 testLoc(spawn->GetX(), spawn->GetY(), spawn->GetZ());
  298. map<Region_Node*, ZBSP_Node*> deleteNodes;
  299. for (testitr = spawn->Regions.begin(); testitr != spawn->Regions.end(); testitr++)
  300. {
  301. map<Region_Node*, ZBSP_Node*>::const_iterator actualItr = testitr->first.begin();
  302. Region_Node *node = actualItr->first;
  303. ZBSP_Node *BSP_Root = actualItr->second;
  304. std::map<Region_Node*, bool>::const_iterator dead_itr = dead_nodes.find(node);
  305. if(dead_itr != dead_nodes.end()) {
  306. deleteNodes.insert(make_pair(node, BSP_Root));
  307. continue;
  308. }
  309. if(!BSP_Root) {
  310. continue;
  311. }
  312. float x1 = node->x - testLoc.x;
  313. float y1 = node->y - testLoc.y;
  314. float z1 = node->z - testLoc.z;
  315. float dist = sqrt(x1 *x1 + y1 *y1 + z1 *z1);
  316. if (dist <= node->dist)
  317. {
  318. WaterRegionType regionType = RegionTypeUntagged;
  319. if (node->region_type == ClassWaterRegion)
  320. regionType = BSPReturnRegionWaterRegion(node, BSP_Root, 1, testLoc, dist);
  321. else
  322. regionType = BSPReturnRegionTypeNode(node, BSP_Root, 1, testLoc, dist);
  323. if (regionType != RegionTypeNormal)
  324. {
  325. if (!testitr->second.inRegion)
  326. {
  327. testitr->second.inRegion = true;
  328. int32 returnValue = 0;
  329. lua_interface->RunRegionScript(node->regionScriptName, "EnterRegion", spawn->GetZone(), spawn, regionType, &returnValue);
  330. if (client)
  331. client->Message(CHANNEL_COLOR_YELLOW, "[ENTER RANGE %i %u] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", regionType, returnValue, region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  332. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  333. testitr->second.timerTic = returnValue;
  334. testitr->second.lastTimerTic = returnValue ? Timer::GetCurrentTime2() : 0;
  335. }
  336. }
  337. else
  338. {
  339. if (testitr->second.inRegion)
  340. {
  341. testitr->second.inRegion = false;
  342. testitr->second.timerTic = 0;
  343. testitr->second.lastTimerTic = 0;
  344. if (client)
  345. client->Message(CHANNEL_COLOR_RED, "[LEAVE RANGE] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  346. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  347. WaterRegionType whatWasRegionType = (WaterRegionType) spawn->GetRegionType(node, BSP_Root);
  348. lua_interface->RunRegionScript(node->regionScriptName, "LeaveRegion", spawn->GetZone(), spawn, whatWasRegionType);
  349. }
  350. }
  351. }
  352. else
  353. {
  354. if (client)
  355. client->Message(CHANNEL_COLOR_RED, "[LEAVE RANGE - OOR] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  356. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  357. deleteNodes.insert(make_pair(node, BSP_Root));
  358. }
  359. region_num++;
  360. }
  361. map<Region_Node*, ZBSP_Node*>::const_iterator deleteItr;
  362. for (deleteItr = deleteNodes.begin(); deleteItr != deleteNodes.end(); deleteItr++)
  363. {
  364. Region_Node *tmpNode = deleteItr->first;
  365. ZBSP_Node *bspNode = deleteItr->second;
  366. spawn->DeleteRegion(tmpNode, bspNode);
  367. }
  368. spawn->RegionMutex.releasewritelock();
  369. }
  370. void RegionMapV1::TicRegionsNearSpawn(Spawn *spawn, Client *client) const
  371. {
  372. map<map<Region_Node*, ZBSP_Node*>, Region_Status>::iterator testitr;
  373. int region_num = 0;
  374. spawn->RegionMutex.writelock();
  375. for (testitr = spawn->Regions.begin(); testitr != spawn->Regions.end(); testitr++)
  376. {
  377. map<Region_Node*, ZBSP_Node*>::const_iterator actualItr = testitr->first.begin();
  378. Region_Node *node = actualItr->first;
  379. ZBSP_Node *BSP_Root = actualItr->second;
  380. std::map<Region_Node*, bool>::const_iterator dead_itr = dead_nodes.find(node);
  381. if(dead_itr != dead_nodes.end()) {
  382. continue;
  383. }
  384. if(!BSP_Root) {
  385. bool passDistCheck = false;
  386. int32 currentGridID = spawn->appearance.pos.grid_id;
  387. if(testitr->second.timerTic && currentGridID == node->grid_id && (node->trigger_widget_id == spawn->trigger_widget_id || (node->dist > 0.0f && spawn->GetDistance(node->x, node->y, node->z) <= node->dist && (passDistCheck = true)))
  388. && Timer::GetCurrentTime2() >= (testitr->second.lastTimerTic + testitr->second.timerTic)) {
  389. testitr->second.lastTimerTic = Timer::GetCurrentTime2();
  390. if (client)
  391. client->Message(CHANNEL_COLOR_RED, "[TICK] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  392. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  393. int32 returnValue = 0;
  394. lua_interface->RunRegionScript(node->regionScriptName, "Tick", spawn->GetZone(), spawn, RegionTypeUntagged, &returnValue);
  395. if (returnValue == 1)
  396. {
  397. testitr->second.lastTimerTic = 0;
  398. testitr->second.timerTic = 0;
  399. }
  400. }
  401. else if(currentGridID != node->grid_id || (node->trigger_widget_id != spawn->trigger_widget_id && !passDistCheck)) {
  402. if (client)
  403. client->Message(CHANNEL_COLOR_RED, "[LEAVE REGION] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  404. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  405. lua_interface->RunRegionScript(node->regionScriptName, "LeaveRegion", spawn->GetZone(), spawn, RegionTypeUntagged);
  406. spawn->DeleteRegion(node, nullptr);
  407. break;
  408. }
  409. }
  410. else if (testitr->second.timerTic && testitr->second.inRegion && Timer::GetCurrentTime2() >= (testitr->second.lastTimerTic + testitr->second.timerTic))
  411. {
  412. testitr->second.lastTimerTic = Timer::GetCurrentTime2();
  413. if (client)
  414. client->Message(CHANNEL_COLOR_RED, "[TICK] Region %i, GridID: %u, RegionName: %s, RegionEnvFileName: %s, distance: %f, X/Y/Z: %f / %f / %f. Script: %s", region_num, node->grid_id, node->regionName.c_str(), node->regionEnvFileName.c_str(),
  415. node->dist, node->x, node->y, node->z, node->regionScriptName.c_str());
  416. WaterRegionType whatWasRegionType = RegionTypeNormal; // default will be 0
  417. if (BSP_Root->special == SPECIAL_REGION_LAVA_OR_DEATH)
  418. whatWasRegionType = RegionTypeLava; // 2
  419. else if (BSP_Root->special == SPECIAL_REGION_WATER)
  420. whatWasRegionType = RegionTypeWater; // 1
  421. int32 returnValue = 0;
  422. lua_interface->RunRegionScript(node->regionScriptName, "Tick", spawn->GetZone(), spawn, whatWasRegionType, &returnValue);
  423. if (returnValue == 1)
  424. {
  425. testitr->second.lastTimerTic = 0;
  426. testitr->second.timerTic = 0;
  427. }
  428. }
  429. region_num++;
  430. }
  431. spawn->RegionMutex.releasewritelock();
  432. }
  433. WaterRegionType RegionMapV1::BSPReturnRegionType(int32 node_number, const glm::vec3& location, int32 gridid) const {
  434. std::shared_lock lock(MRegions);
  435. map<Region_Node*, ZBSP_Node*>::const_iterator itr;
  436. int region_num = 0;
  437. for (itr = Regions.begin(); itr != Regions.end(); itr++)
  438. {
  439. Region_Node* node = itr->first;
  440. // did not match grid id of current region, skip
  441. //if ( gridid > 0 && gridid != node->grid_id)
  442. // continue;
  443. ZBSP_Node* BSP_Root = itr->second;
  444. float x1 = node->x - location.x;
  445. float y1 = node->y - location.y;
  446. float z1 = node->z - location.z;
  447. float dist = sqrt(x1 * x1 + y1 * y1 + z1 * z1);
  448. #ifdef REGIONDEBUG
  449. printf("Region %i (%i) dist %f / node dist %f. NodeXYZ: %f %f %f, XYZ: %f %f %f.\n", region_num, node->region_type, dist, node->dist, node->x, node->y, node->z, location.x, location.y, location.z);
  450. #endif
  451. if (dist <= node->dist)
  452. {
  453. ZBSP_Node* BSP_Root = itr->second;
  454. WaterRegionType regionType = RegionTypeUntagged;
  455. if (node->region_type == ClassWaterRegion)
  456. regionType = BSPReturnRegionWaterRegion(node, BSP_Root, node_number, location, dist);
  457. else
  458. regionType = BSPReturnRegionTypeNode(node, BSP_Root, node_number, location, dist);
  459. if (regionType != RegionTypeNormal)
  460. return regionType;
  461. }
  462. region_num++;
  463. }
  464. return(RegionTypeNormal);
  465. }
  466. WaterRegionType RegionMapV1::BSPReturnRegionTypeNode(const Region_Node* region_node, const ZBSP_Node* BSP_Root, int32 node_number, const glm::vec3& location, float distToNode) const {
  467. if(node_number > region_node->vert_count)
  468. {
  469. LogWrite(REGION__DEBUG, 0, "Region", "Region %s grid %u (%s) - Node %u is out of range for region max vert count of %i. Hit at location %f %f %f.",
  470. region_node->regionName.c_str(), region_node->grid_id, region_node->regionScriptName.c_str(), node_number, region_node->vert_count,
  471. location.x, location.y, location.z);
  472. return (RegionTypeWater);
  473. }
  474. const ZBSP_Node* current_node = &BSP_Root[node_number - 1];
  475. float distance;
  476. #ifdef REGIONDEBUG
  477. printf("left = %u, right %u (Size: %i)\n", current_node->left, current_node->right, region_node->vert_count);
  478. #endif
  479. if (region_node->region_type == ClassWaterRegion2)
  480. {
  481. distance = (location.x * current_node->normal[0]) +
  482. (location.y * current_node->normal[1]) +
  483. (location.z * current_node->normal[2]) +
  484. current_node->splitdistance;
  485. }
  486. else {
  487. distance = (location.x * current_node->normal[0]) +
  488. (location.y * current_node->normal[1]) +
  489. (location.z * current_node->normal[2]) -
  490. current_node->splitdistance;
  491. }
  492. float absDistance = distance;
  493. if (absDistance < 0.0f)
  494. absDistance *= -1.0f;
  495. float absSplitDist = current_node->splitdistance;
  496. if (absSplitDist < 0.0f)
  497. absSplitDist *= -1.0f;
  498. #ifdef REGIONDEBUG
  499. printf("distance = %f, normals: %f %f %f, location: %f %f %f, split distance: %f\n", distance, current_node->left, current_node->right, current_node->normal[0], current_node->normal[1], current_node->normal[2],
  500. location.x, location.y, location.z, current_node->splitdistance);
  501. #endif
  502. if ((current_node->left == -2) &&
  503. (current_node->right == -1 || current_node->right == -2)) {
  504. if (region_node->region_type == ClassWaterOcean || region_node->region_type == ClassWaterOcean2)
  505. {
  506. if ( region_node->region_type == ClassWaterOcean && current_node->right == -1 &&
  507. current_node->normal[1] >= 0.9f && distance > 0 )
  508. return RegionTypeWater;
  509. else
  510. return EstablishDistanceAtAngle(region_node, current_node, distance, absDistance, absSplitDist, true);
  511. }
  512. else
  513. {
  514. if (distance > 0)
  515. return(RegionTypeWater);
  516. else
  517. return RegionTypeNormal;
  518. }
  519. }
  520. else if ((region_node->region_type == ClassWaterOcean || region_node->region_type == ClassWaterOcean2) && current_node->normal[1] != 1.0f && current_node->normal[1] != -1.0f)
  521. {
  522. float fraction = abs(current_node->normal[0] * current_node->normal[2]);
  523. float diff = distToNode / region_node->dist;
  524. if (distance > 0)
  525. diff = distance * diff;
  526. #ifdef REGIONDEBUG
  527. printf("Diff: %f (%f + %f), fraction %f\n", diff, distToNode, distance, fraction);
  528. #endif
  529. if ((abs(diff) / 2.0f) > (absSplitDist * (1.0f / fraction)) * 2.0f)
  530. return RegionTypeNormal;
  531. }
  532. if (distance == 0.0f) {
  533. return(RegionTypeNormal);
  534. }
  535. if (distance > 0.0f) {
  536. #ifdef REGIONDEBUG
  537. printf("to left node %i\n", current_node->left);
  538. #endif
  539. if (current_node->left == -2)
  540. {
  541. switch(region_node->region_type)
  542. {
  543. case ClassWaterVolume:
  544. case ClassWaterOcean:
  545. return RegionTypeWater;
  546. break;
  547. case ClassWaterOcean2:
  548. return EstablishDistanceAtAngle(region_node, current_node, distance, absDistance, absSplitDist, false);
  549. break;
  550. case ClassWaterCavern:
  551. return EstablishDistanceAtAngle(region_node, current_node, distance, absDistance, absSplitDist, true);
  552. break;
  553. default:
  554. return RegionTypeNormal;
  555. break;
  556. }
  557. }
  558. else if (current_node->left == -1) {
  559. return(RegionTypeNormal);
  560. }
  561. return BSPReturnRegionTypeNode(region_node, BSP_Root, current_node->left + 1, location, distToNode);
  562. }
  563. #ifdef REGIONDEBUG
  564. printf("to right node %i, sign bit %i\n", current_node->right, signbit(current_node->normal[1]));
  565. #endif
  566. if (current_node->right == -1) {
  567. if (region_node->region_type == ClassWaterOcean2 && signbit(current_node->normal[1]) == 0 && absDistance < absSplitDist)
  568. return RegionTypeWater;
  569. else if ((region_node->region_type == ClassWaterOcean || region_node->region_type == ClassWaterOcean2) &&
  570. (current_node->normal[1] > 0.0f && distance < 0.0f && absDistance < absSplitDist))
  571. {
  572. return(RegionTypeWater);
  573. }
  574. return(RegionTypeNormal);
  575. }
  576. return BSPReturnRegionTypeNode(region_node, BSP_Root, current_node->right + 1, location, distToNode);
  577. }
  578. WaterRegionType RegionMapV1::BSPReturnRegionWaterRegion(const Region_Node* region_node, const ZBSP_Node* BSP_Root, int32 node_number, const glm::vec3& location, float distToNode) const {
  579. if(node_number > region_node->vert_count)
  580. {
  581. LogWrite(REGION__DEBUG, 0, "Region", "Region %s grid %u (%s) - Node %u is out of range for region max vert count of %i. Hit at location %f %f %f.",
  582. region_node->regionName.c_str(), region_node->grid_id, region_node->regionScriptName.c_str(), node_number, region_node->vert_count,
  583. location.x, location.y, location.z);
  584. return (RegionTypeNormal);
  585. }
  586. const ZBSP_Node* current_node = &BSP_Root[node_number - 1];
  587. float distance;
  588. #ifdef REGIONDEBUG
  589. printf("left = %u, right %u\n", current_node->left, current_node->right);
  590. #endif
  591. distance = (location.x * current_node->normal[0]) +
  592. (location.y * current_node->normal[1]) +
  593. (location.z * current_node->normal[2]) -
  594. current_node->splitdistance;
  595. #ifdef REGIONDEBUG
  596. printf("distance = %f, normals: %f %f %f, location: %f %f %f, split distance: %f\n", distance, current_node->left, current_node->right, current_node->normal[0], current_node->normal[1], current_node->normal[2],
  597. location.x, location.y, location.z, current_node->splitdistance);
  598. #endif
  599. if (distance > 0.0f) {
  600. #ifdef REGIONDEBUG
  601. printf("to left node %i\n", current_node->left);
  602. #endif
  603. if (current_node->left == -1) {
  604. return(RegionTypeNormal);
  605. }
  606. else if (current_node->left == -2) {
  607. switch(current_node->special)
  608. {
  609. case SPECIAL_REGION_LAVA_OR_DEATH:
  610. return(RegionTypeLava);
  611. break;
  612. case SPECIAL_REGION_WATER:
  613. return(RegionTypeWater);
  614. break;
  615. default:
  616. return(RegionTypeUntagged);
  617. break;
  618. }
  619. }
  620. return BSPReturnRegionWaterRegion(region_node, BSP_Root, current_node->left + 1, location, distToNode);
  621. }
  622. #ifdef REGIONDEBUG
  623. printf("to right node %i, sign bit %i\n", current_node->right, signbit(current_node->normal[1]));
  624. #endif
  625. if (current_node->right == -1) {
  626. return(RegionTypeNormal);
  627. }
  628. return BSPReturnRegionWaterRegion(region_node, BSP_Root, current_node->right + 1, location, distToNode);
  629. }
  630. WaterRegionType RegionMapV1::EstablishDistanceAtAngle(const Region_Node* region_node, const ZBSP_Node* current_node, float distance, float absDistance, float absSplitDist, bool checkEdgedAngle) const {
  631. float fraction = abs(current_node->normal[0] * current_node->normal[2]);
  632. #ifdef REGIONDEBUG
  633. printf("Distcheck: %f < %f\n", absDistance, absSplitDist);
  634. #endif
  635. if (absDistance < absSplitDist &&
  636. (current_node->normal[0] >= 1.0f || current_node->normal[0] <= -1.0f ||
  637. (current_node->normal[1] >= .9f && distance < 0.0f) ||
  638. (current_node->normal[1] <= -.9f && distance > 0.0f)))
  639. {
  640. return RegionTypeWater;
  641. }
  642. else if (fraction > 0.0f && (region_node->region_type == ClassWaterOcean2 || checkEdgedAngle))
  643. {
  644. if (current_node->normal[2] >= 1.0f || current_node->normal[2] <= -1.0f)
  645. return RegionTypeNormal;
  646. else if (current_node->normal[1] == 0.0f && (current_node->normal[0] < -0.5f || current_node->normal[0] > 0.5f) &&
  647. ((abs(absDistance * current_node->normal[0]) / 2.0f) < ((abs(absSplitDist * (1.0f / fraction))))))
  648. {
  649. return RegionTypeWater;
  650. }
  651. else if (current_node->normal[1] == 0.0f && (current_node->normal[2] < -0.5f || current_node->normal[2] > 0.5f) &&
  652. ((abs(absDistance * current_node->normal[2]) / 2.0f) < ((abs(absSplitDist * (1.0f / fraction))))))
  653. {
  654. return RegionTypeWater;
  655. }
  656. }
  657. return RegionTypeNormal;
  658. }
  659. void RegionMapV1::InsertRegionNode(ZoneServer* zone, int32 version, std::string regionName, std::string envName, uint32 gridID, uint32 triggerWidgetID, float dist)
  660. {
  661. Region_Node* tmpNode = new Region_Node;
  662. tmpNode->x = 0.0f;
  663. tmpNode->y = 0.0f;
  664. tmpNode->z = 0.0f;
  665. if(!zone)
  666. return;
  667. Map* current_map = world.GetMap(std::string(zone->GetZoneFile()), version);
  668. if(current_map) {
  669. std::map<int32, glm::vec3>::iterator itr = current_map->widget_map.find(triggerWidgetID);
  670. if(itr != current_map->widget_map.end()) {
  671. tmpNode->x = itr->second.x;
  672. tmpNode->y = itr->second.y;
  673. tmpNode->z = itr->second.z;
  674. }
  675. }
  676. tmpNode->dist = dist;
  677. tmpNode->region_type = RegionTypeUntagged;
  678. tmpNode->regionName = string(regionName);
  679. tmpNode->regionEnvFileName = string(envName);
  680. tmpNode->grid_id = gridID;
  681. tmpNode->regionScriptName = string("");
  682. tmpNode->trigger_widget_id = triggerWidgetID;
  683. tmpNode->regionScriptName = TestFile(regionName);
  684. if ( tmpNode->regionScriptName.size() < 1 )
  685. {
  686. tmpNode->regionScriptName = TestFile(envName);
  687. }
  688. if ( tmpNode->regionScriptName.size() < 1 )
  689. {
  690. tmpNode->regionScriptName = TestFile("default");
  691. }
  692. tmpNode->vert_count = 0;
  693. ZBSP_Node* BSP_Root = nullptr;
  694. MRegions.lock();
  695. Regions.insert(make_pair(tmpNode, BSP_Root));
  696. MRegions.unlock();
  697. }
  698. void RegionMapV1::RemoveRegionNode(std::string name) {
  699. std::unique_lock lock(MRegions);
  700. map<Region_Node*, ZBSP_Node*>::const_iterator itr;
  701. for (itr = Regions.begin(); itr != Regions.end();)
  702. {
  703. Region_Node *node = itr->first;
  704. ZBSP_Node *BSP_Root = itr->second;
  705. if(node->regionName.find(name) != node->regionName.npos) {
  706. itr = Regions.erase(itr);
  707. dead_nodes.insert(make_pair(node, true));
  708. safe_delete(node);
  709. safe_delete_array(BSP_Root);
  710. }
  711. else {
  712. itr++;
  713. }
  714. }
  715. }