123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333 |
- /*
- EQ2Emulator: Everquest II Server Emulator
- Copyright (C) 2007 EQ2EMulator Development Team (http://www.eq2emulator.net)
- This file is part of EQ2Emulator.
- EQ2Emulator is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
- EQ2Emulator is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with EQ2Emulator. If not, see <http://www.gnu.org/licenses/>.
- */
- #include "Spawn.h"
- #include <stdio.h>
- #include "../common/timer.h"
- #include <time.h>
- #include <math.h>
- #include "Entity.h"
- #include "Widget.h"
- #include "Sign.h"
- #include "../common/MiscFunctions.h"
- #include "../common/Log.h"
- #include "Rules/Rules.h"
- #include "World.h"
- #include "LuaInterface.h"
- #include "Zone/SPGrid.h"
- #include "Bots/Bot.h"
- #include "Zone/raycast_mesh.h"
- extern ConfigReader configReader;
- extern RuleManager rule_manager;
- extern World world;
- Spawn::Spawn(){
- group_id = 0;
- size_offset = 0;
- merchant_id = 0;
- memset(&appearance, 0, sizeof(AppearanceData));
- memset(&basic_info, 0, sizeof(BasicInfoStruct));
- appearance.encounter_level =6;
- size = 32;
- appearance.pos.collision_radius = 32;
- id = Spawn::NextID();
- oversized_packet = 0xFF;
- zone = 0;
- spawn_location_id = 0;
- spawn_entry_id = 0;
- spawn_location_spawns_id = 0;
- respawn = 0;
- expire_time = 0;
- expire_offset = 0;
- x_offset = 0;
- y_offset = 0;
- z_offset = 0;
- database_id = 0;
- packet_num = 1;
- changed = false;
- vis_changed = false;
- position_changed = false;
- send_spawn_changes = true;
- info_changed = false;
- appearance.pos.Speed1 = 0;
- last_attacker = 0;
- faction_id = 0;
- running_to = 0;
- tmp_visual_state = -1;
- tmp_action_state = -1;
- transporter_id = 0;
- invulnerable = false;
- spawn_group_list = 0;
- MSpawnGroup = 0;
- movement_locations = 0;
- MMovementLocations = 0;
- target = 0;
- primary_command_list_id = 0;
- secondary_command_list_id = 0;
- is_pet = false;
- m_followTarget = 0;
- following = false;
- req_quests_continued_access = false;
- req_quests_override = 0;
- req_quests_private = false;
- m_illusionModel = 0;
- Cell_Info.CurrentCell = nullptr;
- Cell_Info.CellListIndex = -1;
- m_addedToWorldTimestamp = 0;
- m_spawnAnim = 0;
- m_spawnAnimLeeway = 0;
- m_Update.SetName("Spawn::m_Update");
- m_requiredHistory.SetName("Spawn::m_requiredHistory");
- m_requiredQuests.SetName("Spawn::m_requiredQuests");
- last_heading_angle = 0.0;
- last_grid_update = 0;
- last_location_update = 0.0;
- last_movement_update = Timer::GetCurrentTime2();
- forceMapCheck = false;
- m_followDistance = 0;
- MCommandMutex.SetName("Entity::MCommandMutex");
- has_spawn_proximities = false;
- pickup_item_id = 0;
- pickup_unique_item_id = 0;
- disable_sounds = false;
- }
- Spawn::~Spawn(){
- RemovePrimaryCommands();
- for(int32 i=0;i<secondary_command_list.size();i++){
- safe_delete(secondary_command_list[i]);
- }
- RemoveSpawnFromGroup();
- if (MMovementLocations)
- MMovementLocations->writelock(__FUNCTION__, __LINE__);
- if(movement_locations){
- while(movement_locations->size()){
- safe_delete(movement_locations->front());
- movement_locations->pop_front();
- }
- safe_delete(movement_locations);
- }
- if (MMovementLocations)
- MMovementLocations->releasewritelock(__FUNCTION__, __LINE__);
- safe_delete(MMovementLocations);
- MMovementLoop.lock();
- for (int32 i = 0; i < movement_loop.size(); i++)
- safe_delete(movement_loop.at(i));
- MMovementLoop.unlock();
- map<int32, vector<int16>* >::iterator rq_itr;
- m_requiredQuests.writelock(__FUNCTION__, __LINE__);
- for (rq_itr = required_quests.begin(); rq_itr != required_quests.end(); rq_itr++){
- safe_delete(rq_itr->second);
- }
- m_requiredQuests.releasewritelock(__FUNCTION__, __LINE__);
- // just in case to make sure data is destroyed
- RemoveSpawnProximities();
- }
- void Spawn::RemovePrimaryCommands()
- {
- for (int32 i = 0; i < primary_command_list.size(); i++) {
- safe_delete(primary_command_list[i]);
- }
- primary_command_list.clear();
- }
- void Spawn::InitializeHeaderPacketData(Player* player, PacketStruct* header, int16 index) {
- header->setDataByName("index", index);
- if (GetSpawnAnim() > 0 && Timer::GetCurrentTime2() < (GetAddedToWorldTimestamp() + GetSpawnAnimLeeway())) {
- if (header->GetVersion() >= 57080)
- header->setDataByName("spawn_anim", GetSpawnAnim());
- else
- header->setDataByName("spawn_anim", (int16)GetSpawnAnim());
- }
- else {
- if (header->GetVersion() >= 57080)
- header->setDataByName("spawn_anim", 0xFFFFFFFF);
- else
- header->setDataByName("spawn_anim", 0xFFFF);
- }
- if (primary_command_list.size() > 0){
- if (primary_command_list.size() > 1) {
- header->setArrayLengthByName("command_list", primary_command_list.size());
- for (int32 i = 0; i < primary_command_list.size(); i++) {
- header->setArrayDataByName("command_list_name", primary_command_list[i]->name.c_str(), i);
- header->setArrayDataByName("command_list_max_distance", primary_command_list[i]->distance, i);
- header->setArrayDataByName("command_list_error", primary_command_list[i]->error_text.c_str(), i);
- header->setArrayDataByName("command_list_command", primary_command_list[i]->command.c_str(), i);
- }
- }
- header->setMediumStringByName("default_command", primary_command_list[0]->command.c_str());
- header->setDataByName("max_distance", primary_command_list[0]->distance);
- }
- if (spawn_group_list && MSpawnGroup){
- MSpawnGroup->readlock(__FUNCTION__, __LINE__);
- header->setArrayLengthByName("group_size", spawn_group_list->size());
- vector<Spawn*>::iterator itr;
- int i = 0;
- for (itr = spawn_group_list->begin(); itr != spawn_group_list->end(); itr++, i++){
- header->setArrayDataByName("group_spawn_id", player->GetIDWithPlayerSpawn((*itr)), i);
- }
- MSpawnGroup->releasereadlock(__FUNCTION__, __LINE__);
- }
- header->setDataByName("spawn_id", player->GetIDWithPlayerSpawn(this));
- header->setDataByName("crc", 1);
- header->setDataByName("time_stamp", Timer::GetCurrentTime2());
- }
- void Spawn::InitializeVisPacketData(Player* player, PacketStruct* vis_packet) {
- int16 version = vis_packet->GetVersion();
- if (IsPlayer())
- appearance.pos.grid_id = 0xFFFFFFFF;
- if (appearance.targetable == 1 || appearance.show_level == 1 || appearance.display_name == 1){
- if (!IsGroundSpawn()){
- int8 arrow_color = ARROW_COLOR_WHITE;
- sint8 npc_con = player->GetFactions()->GetCon(faction_id);
- if (IsPlayer() && !((Player*)this)->CanSeeInvis(player))
- npc_con = 0;
- else if (!IsPlayer() && IsEntity() && !((Entity*)this)->CanSeeInvis(player))
- npc_con = 0;
- if (appearance.attackable == 1)
- arrow_color = player->GetArrowColor(GetLevel());
- vis_packet->setDataByName("arrow_color", arrow_color);
- vis_packet->setDataByName("locked_no_loot", appearance.locked_no_loot);
- if (player->GetArrowColor(GetLevel()) == ARROW_COLOR_GRAY)
- if (npc_con == -4)
- npc_con = -3;
- vis_packet->setDataByName("npc_con", npc_con);
- if (appearance.attackable == 1 && IsNPC() && (player->GetFactions()->GetCon(faction_id) <= -4 || ((NPC*)this)->Brain()->GetHate(player) > 1))
- vis_packet->setDataByName("npc_hate", ((NPC*)this)->Brain()->GetHatePercentage(player));
- int8 quest_flag = player->CheckQuestFlag(this);
- if (version < 1188 && quest_flag >= 16)
- quest_flag = 1;
- vis_packet->setDataByName("quest_flag",quest_flag);
- if( player->CheckQuestsKillUpdate(this, false)){
- vis_packet->setDataByName("name_quest_icon", 1);
- }
- }
- }
- int8 vis_flags = 0;
- if (MeetsSpawnAccessRequirements(player)) {
- if (appearance.attackable == 1)
- vis_flags += 64; //attackable icon
- if (appearance.show_level == 1)
- vis_flags += 32;
- if (appearance.display_name == 1)
- vis_flags += 16;
- if (IsPlayer() || appearance.targetable == 1)
- vis_flags += 4;
- if (appearance.show_command_icon == 1)
- vis_flags += 2;
- if (this == player) {
- vis_flags += 1;
- }
- }
- else if (req_quests_override > 0)
- {
- //Check to see if there's an override value set
- vis_flags = req_quests_override & 0xFF;
- }
- if (player->HasGMVision())
- {
- if ((vis_flags & 16) == 0 && appearance.display_name == 0)
- vis_flags += 16;
- if ((vis_flags & 4) == 0)
- vis_flags += 4;
- }
- vis_packet->setDataByName("vis_flags", vis_flags);
- if (MeetsSpawnAccessRequirements(player))
- vis_packet->setDataByName("hand_flag", appearance.display_hand_icon);
- else {
- if ((req_quests_override & 256) > 0)
- vis_packet->setDataByName("hand_flag", 1);
- }
- }
- void Spawn::InitializeFooterPacketData(Player* player, PacketStruct* footer) {
- if (IsWidget()){
- Widget* widget = (Widget*)this;
- if (widget->GetMultiFloorLift()) {
- footer->setDataByName("widget_x", widget->GetX());
- footer->setDataByName("widget_y", widget->GetY());
- footer->setDataByName("widget_z", widget->GetZ());
- }
- else {
- footer->setDataByName("widget_x", widget->GetWidgetX());
- footer->setDataByName("widget_y", widget->GetWidgetY());
- footer->setDataByName("widget_z", widget->GetWidgetZ());
- }
- footer->setDataByName("widget_id", widget->GetWidgetID());
- footer->setDataByName("unknown3c", 6);
- }
- else if (IsSign()){
- Sign* sign = (Sign*)this;
- footer->setDataByName("widget_id", sign->GetWidgetID());
- footer->setDataByName("widget_x", sign->GetWidgetX());
- footer->setDataByName("widget_y", sign->GetWidgetY());
- footer->setDataByName("widget_z", sign->GetWidgetZ());
- footer->setDataByName("unknown2b", 6);
- if (sign->GetSignTitle())
- footer->setMediumStringByName("title", sign->GetSignTitle());
- if (sign->GetSignDescription())
- footer->setMediumStringByName("description", sign->GetSignDescription());
- footer->setDataByName("sign_distance", sign->GetSignDistance());
- footer->setDataByName("show", 1);
- }
- if ( IsPlayer())
- footer->setDataByName("is_player", 1);
- if (strlen(appearance.name) < 1)
- strncpy(appearance.name,to_string(GetID()).c_str(),128);
- footer->setMediumStringByName("name", appearance.name);
- footer->setMediumStringByName("guild", appearance.sub_title);
- footer->setMediumStringByName("prefix", appearance.prefix_title);
- footer->setMediumStringByName("suffix", appearance.suffix_title);
- footer->setMediumStringByName("last_name", appearance.last_name);
- if (appearance.attackable == 0 && GetLevel() > 0)
- footer->setDataByName("spawn_type", 1);
- else if (appearance.attackable == 0)
- footer->setDataByName("spawn_type", 6);
- else
- footer->setDataByName("spawn_type", 3);
- }
- EQ2Packet* Spawn::spawn_serialize(Player* player, int16 version){
- // If spawn is NPC AND is pet && owner is a player && owner is the player passed to this function && player's char sheet pet id is 0
- if (IsNPC() && ((NPC*)this)->IsPet() && ((NPC*)this)->GetOwner()->IsPlayer() && player == ((NPC*)this)->GetOwner() && player->GetInfoStruct()->pet_id == 0) {
- ((Player*)((NPC*)this)->GetOwner())->GetInfoStruct()->pet_id = player->spawn_id;
- player->SetCharSheetChanged(true);
- }
- int16 index;
- if (player->player_spawn_index_map.count(this) > 0) {
- index = player->player_spawn_index_map[this];
- player->player_spawn_map[index] = this;
- }
- else {
- player->spawn_index++;
- if(player->spawn_index == 255)
- player->spawn_index++; //just so we dont have to worry about overloading
- index = player->spawn_index;
- player->player_spawn_index_map[this] = index;
- player->player_spawn_map[index] = this;
- }
- // Jabantiz - [Bug] Client Crash on Revive
- if (player->player_spawn_reverse_id_map.count(this) == 0) {
- int32 spawn_id = ++player->spawn_id;
- player->player_spawn_id_map[spawn_id] = this;
- player->player_spawn_reverse_id_map[this] = spawn_id;
- }
- m_Update.writelock(__FUNCTION__, __LINE__);
- PacketStruct* header = player->GetSpawnHeaderStruct();
- header->ResetData();
- InitializeHeaderPacketData(player, header, index);
- PacketStruct* footer;
- if(IsWidget())
- footer = player->GetWidgetFooterStruct();
- else if(IsSign())
- footer = player->GetSignFooterStruct();
- else
- footer = player->GetSpawnFooterStruct();
- footer->ResetData();
- InitializeFooterPacketData(player, footer);
- PacketStruct* vis_struct = player->GetSpawnVisStruct();
- PacketStruct* info_struct = player->GetSpawnInfoStruct();
- PacketStruct* pos_struct = player->GetSpawnPosStruct();
- player->vis_mutex.writelock(__FUNCTION__, __LINE__);
- vis_struct->ResetData();
- InitializeVisPacketData(player, vis_struct);
- player->info_mutex.writelock(__FUNCTION__, __LINE__);
- info_struct->ResetData();
- InitializeInfoPacketData(player, info_struct);
- player->pos_mutex.writelock(__FUNCTION__, __LINE__);
- pos_struct->ResetData();
- InitializePosPacketData(player, pos_struct);
- string* vis_data= vis_struct->serializeString();
- string* pos_data = pos_struct->serializeString();
- string* info_data = info_struct->serializeString();
- int16 part2_size = pos_data->length() + vis_data->length() + info_data->length();
- uchar* part2 = new uchar[part2_size];
- player->AddSpawnPosPacketForXOR(id, (uchar*)pos_data->c_str(), pos_data->length());
- player->AddSpawnVisPacketForXOR(id, (uchar*)vis_data->c_str(), vis_data->length());
- player->AddSpawnInfoPacketForXOR(id, (uchar*)info_data->c_str(), info_data->length());
-
- uchar* ptr = part2;
- memcpy(ptr, pos_data->c_str(), pos_data->length());
- ptr += pos_data->length();
- memcpy(ptr, vis_data->c_str(), vis_data->length());
- ptr += vis_data->length();
- memcpy(ptr, info_data->c_str(), info_data->length());
- player->pos_mutex.releasewritelock(__FUNCTION__, __LINE__);
- player->info_mutex.releasewritelock(__FUNCTION__, __LINE__);
- player->vis_mutex.releasewritelock(__FUNCTION__, __LINE__);
- string* part1 = header->serializeString();
- string* part3 = footer->serializeString();
- uchar tmp[900];
- int32 origPart2Size = part2_size;
- part2_size = Pack(tmp, part2, part2_size, 900, version);
- int32 total_size = part1->length() + part2_size + part3->length() + 3;
- uchar* final_packet = new uchar[total_size + 4];
- ptr = final_packet;
- memcpy(ptr, &total_size, sizeof(total_size));
- ptr += sizeof(total_size);
- memcpy(ptr, &oversized_packet, sizeof(oversized_packet));
- ptr += sizeof(oversized_packet);
- memcpy(ptr, &opcode, sizeof(opcode));
- ptr += sizeof(opcode);
-
- memcpy(ptr, part1->c_str(), part1->length());
- ptr += part1->length();
- memcpy(ptr, tmp, part2_size);
- ptr += part2_size;
-
- memcpy(ptr, part3->c_str(), part3->length());
- delete[] part2;
- // printf("%s (%i): p1: %i, p2:% i (%i), p3:% i, ts: %i\n", GetName(), GetID(), part1->length(), part2_size, origPart2Size, part3->length(), total_size);
- EQ2Packet* ret = new EQ2Packet(OP_ClientCmdMsg, final_packet, total_size + 4);
- delete[] final_packet;
- m_Update.releasewritelock(__FUNCTION__, __LINE__);
- return ret;
- }
- uchar* Spawn::spawn_info_changes(Player* player, int16 version){
- int16 index = player->player_spawn_index_map[this];
- PacketStruct* packet = player->GetSpawnInfoStruct();
- player->info_mutex.writelock(__FUNCTION__, __LINE__);
- packet->ResetData();
- InitializeInfoPacketData(player, packet);
- string* data = packet->serializeString();
- int32 size = data->length();
- uchar* xor_info_packet = player->GetTempInfoPacketForXOR();
- if (!xor_info_packet || size != player->GetTempInfoXorSize())
- {
- LogWrite(ZONE__DEBUG, 0, "Zone", "InstantiateInfoPacket: %i, %i", size, player->GetTempInfoXorSize());
- safe_delete(xor_info_packet);
- xor_info_packet = player->SetTempInfoPacketForXOR(size);
- }
- uchar* orig_packet = player->GetSpawnInfoPacketForXOR(id);
- if(orig_packet){
- memcpy(xor_info_packet, (uchar*)data->c_str(), size);
- Encode(xor_info_packet, orig_packet, size);
- }
- bool changed = false;
- for (int i = 0; i < size; ++i) {
- if (xor_info_packet[i]) {
- changed = true;
- break;
- }
- }
- if (!changed) {
- player->info_mutex.releasewritelock(__FUNCTION__, __LINE__);
- return nullptr;
- }
- uchar* tmp = new uchar[size + 10];
- size = Pack(tmp, xor_info_packet, size, size, version);
- player->info_mutex.releasewritelock(__FUNCTION__, __LINE__);
- int32 orig_size = size;
- size-=sizeof(int32);
- size+=CheckOverLoadSize(index);
- info_packet_size = size + CheckOverLoadSize(size);
- uchar* tmp2 = new uchar[info_packet_size];
- uchar* ptr = tmp2;
- ptr += DoOverLoad(size, ptr);
- ptr += DoOverLoad(index, ptr);
- memcpy(ptr, tmp+sizeof(int32), orig_size - sizeof(int32));
- delete[] tmp;
- return tmp2;
- }
- uchar* Spawn::spawn_vis_changes(Player* player, int16 version){
- PacketStruct* vis_struct = player->GetSpawnVisStruct();
- int16 index = player->player_spawn_index_map[this];
- player->vis_mutex.writelock(__FUNCTION__, __LINE__);
- uchar* orig_packet = player->GetSpawnVisPacketForXOR(id);
- vis_struct->ResetData();
- InitializeVisPacketData(player, vis_struct);
- string* data = vis_struct->serializeString();
- int32 size = data->length();
- uchar* xor_vis_packet = player->GetTempVisPacketForXOR();
- if (!xor_vis_packet || size != player->GetTempVisXorSize())
- {
- LogWrite(ZONE__DEBUG, 0, "Zone", "InstantiateVisPacket: %i, %i", size, player->GetTempVisXorSize());
- safe_delete(xor_vis_packet);
- xor_vis_packet = player->SetTempVisPacketForXOR(size);
- }
- if(orig_packet){
- memcpy(xor_vis_packet, (uchar*)data->c_str(), size);
- Encode(xor_vis_packet, orig_packet, size);
- }
- bool changed = false;
- for (int i = 0; i < size; ++i) {
- if (xor_vis_packet[i]) {
- changed = true;
- break;
- }
- }
- if (!changed) {
- player->vis_mutex.releasewritelock(__FUNCTION__, __LINE__);
- return nullptr;
- }
- uchar* tmp = new uchar[size + 10];
- size = Pack(tmp, xor_vis_packet, size, size, version);
- player->vis_mutex.releasewritelock(__FUNCTION__, __LINE__);
- int32 orig_size = size;
- size-=sizeof(int32);
- size+=CheckOverLoadSize(index);
- vis_packet_size = size + CheckOverLoadSize(size);
- uchar* tmp2 = new uchar[vis_packet_size];
- uchar* ptr = tmp2;
- ptr += DoOverLoad(size, ptr);
- ptr += DoOverLoad(index, ptr);
- memcpy(ptr, tmp+sizeof(int32), orig_size - sizeof(int32));
- delete[] tmp;
- return tmp2;
- }
- uchar* Spawn::spawn_pos_changes(Player* player, int16 version) {
- int16 index = player->GetIndexForSpawn(this);
- PacketStruct* packet = player->GetSpawnPosStruct();
- player->pos_mutex.writelock(__FUNCTION__, __LINE__);
- uchar* orig_packet = player->GetSpawnPosPacketForXOR(id);
- packet->ResetData();
- InitializePosPacketData(player, packet, true);
- string* data = packet->serializeString();
- int32 size = data->length();
- uchar* xor_pos_packet = player->GetTempPosPacketForXOR();
- if (!xor_pos_packet || size != player->GetTempPosXorSize())
- {
- LogWrite(ZONE__DEBUG, 0, "Zone", "InstantiatePosPacket: %i, %i", size, player->GetTempPosXorSize());
- safe_delete(xor_pos_packet);
- xor_pos_packet = player->SetTempPosPacketForXOR(size);
- }
- if(orig_packet){
- memcpy(xor_pos_packet, (uchar*)data->c_str(), size);
- Encode(xor_pos_packet, orig_packet, size);
- }
- bool changed = false;
- for (int i = 0; i < size; ++i) {
- if (xor_pos_packet[i]) {
- changed = true;
- break;
- }
- }
- if (!changed) {
- player->pos_mutex.releasewritelock(__FUNCTION__, __LINE__);
- return nullptr;
- }
- uchar* tmp;
- if (IsPlayer())
- tmp = new uchar[size + 14];
- else
- tmp = new uchar[size + 10];
- size = Pack(tmp, xor_pos_packet, size, size, version);
- player->pos_mutex.releasewritelock(__FUNCTION__, __LINE__);
- int32 orig_size = size;
- // Needed for CoE+ clients
- if (version >= 1188)
- size += 1;
- if(IsPlayer())
- size += 4;
- size-=sizeof(int32);
- size+=CheckOverLoadSize(index);
- pos_packet_size = size + CheckOverLoadSize(size);
- uchar* tmp2 = new uchar[pos_packet_size];
- uchar* ptr = tmp2;
- ptr += DoOverLoad(size, ptr);
- ptr += DoOverLoad(index, ptr);
- // extra byte in coe+ clients, 0 for NPC's 1 for Players
- int8 x = 0;
- if(IsPlayer()){
- if (version >= 1188) {
- // set x to 1 and add it to the packet
- x = 1;
- memcpy(ptr, &x, sizeof(int8));
- ptr += sizeof(int8);
- }
- int32 now = Timer::GetCurrentTime2();
- memcpy(ptr, &now, sizeof(int32));
- ptr += sizeof(int32);
- }
- else if (version >= 1188) {
- // add x to packet
- memcpy(ptr, &x, sizeof(int8));
- ptr += sizeof(int8);
- }
- memcpy(ptr, tmp+sizeof(int32), orig_size - sizeof(int32));
- delete[] tmp;
- return tmp2;
- }
- EQ2Packet* Spawn::player_position_update_packet(Player* player, int16 version){
- if(!player || player->IsPlayer() == false){
- LogWrite(SPAWN__ERROR, 0, "Spawn", "Error: Called player_position_update_packet without player!");
- return 0;
- }
- else if(IsPlayer() == false){
- LogWrite(SPAWN__ERROR, 0, "Spawn", "Error: Called player_position_update_packet from spawn!");
- return 0;
- }
- static const int8 info_size = 1;
- static const int8 vis_size = 1;
- m_Update.writelock(__FUNCTION__, __LINE__);
- uchar* pos_changes = spawn_pos_changes(player, version);
- if (pos_changes == NULL )
- {
- m_Update.releasewritelock(__FUNCTION__, __LINE__);
- return NULL;
- }
- int32 tmp_pos_packet_size = pos_packet_size;
- m_Update.releasewritelock(__FUNCTION__, __LINE__);
- int32 size = info_size + tmp_pos_packet_size + vis_size + 11;
- static const int8 oversized = 255;
- int16 opcode_val = EQOpcodeManager[GetOpcodeVersion(version)]->EmuToEQ(OP_EqUpdateGhostCmd);
- uchar* tmp = new uchar[size];
- memset(tmp, 0, size);
- uchar* ptr = tmp;
- size -=4;
- memcpy(ptr, &size, sizeof(int32));
- size +=4;
- ptr += sizeof(int32);
- memcpy(ptr, &oversized, sizeof(int8));
- ptr += sizeof(int8);
- memcpy(ptr, &opcode_val, sizeof(int16));
- ptr += sizeof(int16);
- ptr += sizeof(int32);
- ptr += info_size;
- memcpy(ptr, pos_changes, tmp_pos_packet_size);
- EQ2Packet* ret_packet = new EQ2Packet(OP_ClientCmdMsg, tmp, size);
- delete[] tmp;
- delete[] pos_changes;
- return ret_packet;
- }
- EQ2Packet* Spawn::spawn_update_packet(Player* player, int16 version, bool override_changes, bool override_vis_changes){
- if(!player || player->IsPlayer() == false){
- LogWrite(SPAWN__ERROR, 0, "Spawn", "Error: Called spawn_update_packet without player!");
- return 0;
- }
- else if((IsPlayer() && info_changed == false && vis_changed == false) || (info_changed == false && vis_changed == false && position_changed == false)){
- if(!override_changes && !override_vis_changes)
- return 0;
- }
- static const uchar null_byte = 0;
- uchar* info_changes = 0;
- uchar* pos_changes = 0;
- uchar* vis_changes = 0;
- static const int8 oversized = 255;
- int16 opcode_val = EQOpcodeManager[GetOpcodeVersion(version)]->EmuToEQ(OP_EqUpdateGhostCmd);
- int16 tmp_info_packet_size;
- int16 tmp_vis_packet_size;
- int16 tmp_pos_packet_size;
- //We need to lock these variables up to make this thread safe
- m_Update.writelock(__FUNCTION__, __LINE__);
- //These variables are set in the spawn_info_changes, pos and vis changes functions
- info_packet_size = 1;
- pos_packet_size = 1;
- vis_packet_size = 1;
- if (info_changed || override_changes)
- info_changes = spawn_info_changes(player, version);
- if ((position_changed || override_changes) && IsPlayer() == false)
- pos_changes = spawn_pos_changes(player, version);
- if (vis_changed || override_changes || override_vis_changes)
- vis_changes = spawn_vis_changes(player, version);
- tmp_info_packet_size = info_packet_size;
- tmp_pos_packet_size = pos_packet_size;
- tmp_vis_packet_size = vis_packet_size;
- int32 size = info_packet_size + pos_packet_size + vis_packet_size + 11;
- uchar* tmp = new uchar[size];
- memset(tmp, 0, size);
- uchar* ptr = tmp;
- size -=4;
- memcpy(ptr, &size, sizeof(int32));
- size +=4;
- ptr += sizeof(int32);
- memcpy(ptr, &oversized, sizeof(int8));
- ptr += sizeof(int8);
- memcpy(ptr, &opcode_val, sizeof(int16));
- ptr += sizeof(int16);
- if (IsPlayer() == false){ //this isnt sent for player updates, it is sent on position update
- //int32 time = Timer::GetCurrentTime2();
- packet_num = Timer::GetCurrentTime2();
- memcpy(ptr, &packet_num, sizeof(int32));
- }
- ptr += sizeof(int32);
- memcpy(ptr, info_changes ? info_changes : &null_byte, tmp_info_packet_size);
- ptr += info_packet_size;
- memcpy(ptr, pos_changes ? pos_changes : &null_byte, tmp_pos_packet_size);
- ptr += pos_packet_size;
- memcpy(ptr, vis_changes ? vis_changes : &null_byte, tmp_vis_packet_size);
- EQ2Packet* ret_packet = new EQ2Packet(OP_ClientCmdMsg, tmp, size);
- delete[] tmp;
- safe_delete_array(info_changes);
- safe_delete_array(vis_changes);
- safe_delete_array(pos_changes);
- m_Update.releasewritelock(__FUNCTION__, __LINE__);
- return ret_packet;
- }
- uchar* Spawn::spawn_info_changes_ex(Player* player, int16 version) {
- int16 index = player->player_spawn_index_map[this];
- PacketStruct* packet = player->GetSpawnInfoStruct();
- player->info_mutex.writelock(__FUNCTION__, __LINE__);
- packet->ResetData();
- InitializeInfoPacketData(player, packet);
- string* data = packet->serializeString();
- int32 size = data->length();
- uchar* xor_info_packet = player->GetTempInfoPacketForXOR();
- if (!xor_info_packet || size != player->GetTempInfoXorSize()) {
- LogWrite(ZONE__DEBUG, 0, "Zone", "InstantiateInfoExPacket: %i, %i", size, player->GetTempInfoXorSize());
- safe_delete(xor_info_packet);
- xor_info_packet = player->SetTempInfoPacketForXOR(size);
- }
- uchar* orig_packet = player->GetSpawnInfoPacketForXOR(id);
- if (orig_packet) {
- memcpy(xor_info_packet, (uchar*)data->c_str(), size);
- Encode(xor_info_packet, orig_packet, size);
- }
- bool changed = false;
- for (int i = 0; i < size; ++i) {
- if (xor_info_packet[i]) {
- changed = true;
- break;
- }
- }
- if (!changed) {
- player->info_mutex.releasewritelock(__FUNCTION__, __LINE__);
- return nullptr;
- }
- uchar* tmp = new uchar[size + 10];
- size = Pack(tmp, xor_info_packet, size, size, version);
- player->info_mutex.releasewritelock(__FUNCTION__, __LINE__);
- int32 orig_size = size;
- size -= sizeof(int32);
- size += CheckOverLoadSize(index);
- info_packet_size = size;
- uchar* tmp2 = new uchar[size];
- uchar* ptr = tmp2;
- ptr += DoOverLoad(index, ptr);
- memcpy(ptr, tmp + sizeof(int32), orig_size - sizeof(int32));
- delete[] tmp;
- return move(tmp2);
- }
- uchar* Spawn::spawn_vis_changes_ex(Player* player, int16 version) {
- PacketStruct* vis_struct = player->GetSpawnVisStruct();
- int16 index = player->player_spawn_index_map[this];
- player->vis_mutex.writelock(__FUNCTION__, __LINE__);
- uchar* orig_packet = player->GetSpawnVisPacketForXOR(id);
- vis_struct->ResetData();
- InitializeVisPacketData(player, vis_struct);
- string* data = vis_struct->serializeString();
- int32 size = data->length();
- uchar* xor_vis_packet = player->GetTempVisPacketForXOR();
- if (!xor_vis_packet || size != player->GetTempVisXorSize()) {
- LogWrite(ZONE__DEBUG, 0, "Zone", "InstantiateVisExPacket: %i, %i", size, player->GetTempVisXorSize());
- safe_delete(xor_vis_packet);
- xor_vis_packet = player->SetTempVisPacketForXOR(size);
- }
- if (orig_packet) {
- memcpy(xor_vis_packet, (uchar*)data->c_str(), size);
- Encode(xor_vis_packet, orig_packet, size);
- }
- bool changed = false;
- for (int i = 0; i < size; ++i) {
- if (xor_vis_packet[i]) {
- changed = true;
- break;
- }
- }
- if (!changed) {
- player->vis_mutex.releasewritelock(__FUNCTION__, __LINE__);
- return nullptr;
- }
- uchar* tmp = new uchar[size + 10];
- size = Pack(tmp, xor_vis_packet, size, size, version);
- player->vis_mutex.releasewritelock(__FUNCTION__, __LINE__);
- int32 orig_size = size;
- size -= sizeof(int32);
- size += CheckOverLoadSize(index);
- vis_packet_size = size;
- uchar* tmp2 = new uchar[size];
- uchar* ptr = tmp2;
- ptr += DoOverLoad(index, ptr);
- memcpy(ptr, tmp + sizeof(int32), orig_size - sizeof(int32));
- delete[] tmp;
- return move(tmp2);
- }
- uchar* Spawn::spawn_pos_changes_ex(Player* player, int16 version) {
- int16 index = player->GetIndexForSpawn(this);
- PacketStruct* packet = player->GetSpawnPosStruct();
- player->pos_mutex.writelock(__FUNCTION__, __LINE__);
- uchar* orig_packet = player->GetSpawnPosPacketForXOR(id);
- packet->ResetData();
- InitializePosPacketData(player, packet);
- string* data = packet->serializeString();
- int32 size = data->length();
- uchar* xor_pos_packet = player->GetTempPosPacketForXOR();
- if (!xor_pos_packet || size != player->GetTempPosXorSize()) {
- LogWrite(ZONE__DEBUG, 0, "Zone", "InstantiatePosExPacket: %i, %i", size, player->GetTempPosXorSize());
- safe_delete(xor_pos_packet);
- xor_pos_packet = player->SetTempPosPacketForXOR(size);
- }
- if (orig_packet) {
- memcpy(xor_pos_packet, (uchar*)data->c_str(), size);
- Encode(xor_pos_packet, orig_packet, size);
- }
- bool changed = false;
- for (int i = 0; i < size; ++i) {
- if (xor_pos_packet[i]) {
- changed = true;
- break;
- }
- }
- if (!changed) {
- player->pos_mutex.releasewritelock(__FUNCTION__, __LINE__);
- return nullptr;
- }
- uchar* tmp = new uchar[size + 10];
- size = Pack(tmp, xor_pos_packet, size, size, version);
- player->pos_mutex.releasewritelock(__FUNCTION__, __LINE__);
- int32 orig_size = size;
- if (version >= 1188) {
- size += 1;
- }
- if (IsPlayer()) {
- size += 4;
- }
- size -= sizeof(int32);
- size += CheckOverLoadSize(index);
- pos_packet_size = size;
- uchar* tmp2 = new uchar[size];
- uchar* ptr = tmp2;
- ptr += DoOverLoad(index, ptr);
- // extra byte in coe+ clients, 0 for NPC's 1 for Players
- int8 x = 0;
- if (version >= 1188) {
- if (IsPlayer()) {
- x = 1;
- memcpy(ptr, &x, sizeof(int8));
- ptr += sizeof(int8);
- int32 now = Timer::GetCurrentTime2();
- memcpy(ptr, &now, sizeof(int32));
- ptr += sizeof(int32);
- }
- else {
- memcpy(ptr, &x, sizeof(int8));
- ptr += sizeof(int8);
- }
- }
- memcpy(ptr, tmp + sizeof(int32), orig_size - sizeof(int32));
- delete[] tmp;
- return move(tmp2);
- }
- EQ2Packet* Spawn::serialize(Player* player, int16 version){
- return 0;
- }
- Spawn* Spawn::GetTarget(){
- Spawn* ret = 0;
-
- // only attempt to get a spawn if we had a target stored
- if (target != 0)
- {
- ret = GetZone()->GetSpawnByID(target);
-
- if (!ret)
- target = 0;
- }
- return ret;
- }
- void Spawn::SetTarget(Spawn* spawn){
- SetInfo(&target, spawn ? spawn->GetID() : 0);
- }
- Spawn* Spawn::GetLastAttacker() {
- Spawn* ret = 0;
- ret = GetZone()->GetSpawnByID(last_attacker);
- if (!ret)
- last_attacker = 0;
- return ret;
- }
- void Spawn::SetLastAttacker(Spawn* spawn){
- last_attacker = spawn->GetID();
- }
- void Spawn::SetInvulnerable(bool val){
- invulnerable = val;
- }
- bool Spawn::GetInvulnerable(){
- return invulnerable;
- }
- bool Spawn::TakeDamage(int32 damage){
- if(invulnerable)
- return false;
- if (IsEntity()) {
- if (((Entity*)this)->IsMezzed())
- ((Entity*)this)->RemoveAllMezSpells();
- if (damage == 0)
- return true;
- }
- int32 hp = GetHP();
- if(damage >= hp) {
- SetHP(0);
- if (IsPlayer()) {
- ((Player*)this)->InCombat(false);
- ((Player*)this)->SetRangeAttack(false);
- GetZone()->TriggerCharSheetTimer(); // force char sheet updates now
- }
- }
- else {
- SetHP(hp - damage);
- // if player flag the char sheet as changed so the ui updates properly
- if (IsPlayer())
- ((Player*)this)->SetCharSheetChanged(true);
- }
- return true;
- }
- void Spawn::TakeDamage(Spawn* attacker, int32 damage){
- if (TakeDamage(damage))
- GetZone()->CallSpawnScript(this, SPAWN_SCRIPT_HEALTHCHANGED, attacker);
- SetLastAttacker(attacker);
- }
- ZoneServer* Spawn::GetZone(){
- return zone;
- }
- void Spawn::SetZone(ZoneServer* in_zone){
- zone = in_zone;
- }
- /*** HIT POINT ***/
- void Spawn::SetHP(sint32 new_val, bool setUpdateFlags){
- if(new_val == 0){
- ClearRunningLocations();
- CalculateRunningLocation(true);
- }
- if(new_val > basic_info.max_hp)
- SetInfo(&basic_info.max_hp, new_val, setUpdateFlags);
- SetInfo(&basic_info.cur_hp, new_val, setUpdateFlags);
- if(/*IsPlayer() &&*/ GetZone() && basic_info.cur_hp > 0 && basic_info.cur_hp < basic_info.max_hp)
- GetZone()->AddDamagedSpawn(this);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- if (IsNPC() && ((NPC*)this)->IsPet() && ((NPC*)this)->GetOwner()->IsPlayer()) {
- Player* player = (Player*)((NPC*)this)->GetOwner();
- if (player->GetPet() && player->GetCharmedPet()) {
- if (this == player->GetPet()) {
- player->GetInfoStruct()->pet_health_pct = (float)basic_info.cur_hp / (float)basic_info.max_hp;
- player->SetCharSheetChanged(true);
- }
- }
- else {
- player->GetInfoStruct()->pet_health_pct = (float)basic_info.cur_hp / (float)basic_info.max_hp;
- player->SetCharSheetChanged(true);
- }
- }
- }
- void Spawn::SetTotalHP(sint32 new_val){
- if(basic_info.hp_base == 0)
- SetTotalHPBase(new_val);
- SetInfo(&basic_info.max_hp, new_val);
- if(GetZone() && basic_info.cur_hp > 0 && basic_info.cur_hp < basic_info.max_hp)
- GetZone()->AddDamagedSpawn(this);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- if (IsNPC() && ((NPC*)this)->IsPet() && ((NPC*)this)->GetOwner()->IsPlayer()) {
- Player* player = (Player*)((NPC*)this)->GetOwner();
- if (player->GetPet() && player->GetCharmedPet()) {
- if (this == player->GetPet()) {
- player->GetInfoStruct()->pet_health_pct = (float)basic_info.cur_hp / (float)basic_info.max_hp;
- player->SetCharSheetChanged(true);
- }
- }
- else {
- player->GetInfoStruct()->pet_health_pct = (float)basic_info.cur_hp / (float)basic_info.max_hp;
- player->SetCharSheetChanged(true);
- }
- }
- }
- void Spawn::SetTotalHPBase(sint32 new_val)
- {
- SetInfo(&basic_info.hp_base, new_val);
- if(GetZone() && basic_info.cur_hp > 0 && basic_info.cur_hp < basic_info.max_hp)
- GetZone()->AddDamagedSpawn(this);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- }
- sint32 Spawn::GetHP()
- {
- return basic_info.cur_hp;
- }
- sint32 Spawn::GetTotalHP()
- {
- return basic_info.max_hp;
- }
- sint32 Spawn::GetTotalHPBase()
- {
- return basic_info.hp_base;
- }
- /*** POWER ***/
- void Spawn::SetPower(sint32 power, bool setUpdateFlags){
- if(power > basic_info.max_power)
- SetInfo(&basic_info.max_power, power, setUpdateFlags);
- SetInfo(&basic_info.cur_power, power, setUpdateFlags);
- if(/*IsPlayer() &&*/ GetZone() && basic_info.cur_power < basic_info.max_power)
- GetZone()->AddDamagedSpawn(this);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- if (IsNPC() && ((NPC*)this)->IsPet() && ((NPC*)this)->GetOwner()->IsPlayer()) {
- Player* player = (Player*)((NPC*)this)->GetOwner();
- if (player->GetPet() && player->GetCharmedPet()) {
- if (this == player->GetPet()) {
- player->GetInfoStruct()->pet_power_pct = (float)basic_info.cur_power / (float)basic_info.max_power;
- player->SetCharSheetChanged(true);
- }
- }
- else {
- player->GetInfoStruct()->pet_power_pct = (float)basic_info.cur_power / (float)basic_info.max_power;
- player->SetCharSheetChanged(true);
- }
- }
- }
- void Spawn::SetTotalPower(sint32 new_val)
- {
- if(basic_info.power_base == 0)
- SetTotalPowerBase(new_val);
- SetInfo(&basic_info.max_power, new_val);
- if(GetZone() && basic_info.cur_power < basic_info.max_power)
- GetZone()->AddDamagedSpawn(this);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- if (IsNPC() && ((NPC*)this)->IsPet() && ((NPC*)this)->GetOwner()->IsPlayer()) {
- Player* player = (Player*)((NPC*)this)->GetOwner();
- if (player->GetPet() && player->GetCharmedPet()) {
- if (this == player->GetPet()) {
- player->GetInfoStruct()->pet_power_pct = (float)basic_info.cur_power / (float)basic_info.max_power;
- player->SetCharSheetChanged(true);
- }
- }
- else {
- player->GetInfoStruct()->pet_power_pct = (float)basic_info.cur_power / (float)basic_info.max_power;
- player->SetCharSheetChanged(true);
- }
- }
- }
- void Spawn::SetTotalPowerBase(sint32 new_val)
- {
- SetInfo(&basic_info.power_base, new_val);
- if(GetZone() && basic_info.cur_power < basic_info.max_power)
- GetZone()->AddDamagedSpawn(this);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- }
- sint32 Spawn::GetPower()
- {
- return basic_info.cur_power;
- }
- sint32 Spawn::GetTotalPower(){
- return basic_info.max_power;
- }
- sint32 Spawn::GetTotalPowerBase()
- {
- return basic_info.power_base;
- }
- /*** SAVAGERY ***/
- void Spawn::SetSavagery(sint32 savagery, bool setUpdateFlags)
- {
- /* JA: extremely limited functionality until we better understand Savagery */
- if(savagery > basic_info.max_savagery)
- SetInfo(&basic_info.max_savagery, savagery, setUpdateFlags);
- SetInfo(&basic_info.cur_savagery, savagery, setUpdateFlags);
- }
- void Spawn::SetTotalSavagery(sint32 new_val)
- {
- /* JA: extremely limited functionality until we better understand Savagery */
- if(basic_info.savagery_base == 0)
- SetTotalSavageryBase(new_val);
- SetInfo(&basic_info.max_savagery, new_val);
- }
- void Spawn::SetTotalSavageryBase(sint32 new_val){
- SetInfo(&basic_info.savagery_base, new_val);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- }
- sint32 Spawn::GetTotalSavagery()
- {
- return basic_info.max_savagery;
- }
- sint32 Spawn::GetSavagery()
- {
- return basic_info.cur_savagery;
- }
- /*** DISSONANCE ***/
- void Spawn::SetDissonance(sint32 dissonance, bool setUpdateFlags)
- {
- /* JA: extremely limited functionality until we better understand Dissonance */
- if(dissonance > basic_info.max_dissonance)
- SetInfo(&basic_info.max_dissonance, dissonance, setUpdateFlags);
- SetInfo(&basic_info.cur_dissonance, dissonance, setUpdateFlags);
- }
- void Spawn::SetTotalDissonance(sint32 new_val)
- {
- /* JA: extremely limited functionality until we better understand Dissonance */
- if(basic_info.dissonance_base == 0)
- SetTotalDissonanceBase(new_val);
- SetInfo(&basic_info.max_dissonance, new_val);
- }
- void Spawn::SetTotalDissonanceBase(sint32 new_val)
- {
- SetInfo(&basic_info.dissonance_base, new_val);
- if (IsEntity() && ((Entity*)this)->GetGroupMemberInfo()) {
- ((Entity*)this)->UpdateGroupMemberInfo();
- if (IsPlayer())
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id, GetZone()->GetClientBySpawn(this));
- else
- world.GetGroupManager()->SendGroupUpdate(((Entity*)this)->GetGroupMemberInfo()->group_id);
- }
- }
- sint32 Spawn::GetTotalDissonance()
- {
- return basic_info.max_dissonance;
- }
- sint32 Spawn::GetDissonance()
- {
- return basic_info.cur_dissonance;
- }
- /* --< Alternate Advancement Points >-- */
- void Spawn::SetAssignedAA(sint16 new_val)
- {
- SetInfo(&basic_info.assigned_aa, new_val);
- }
- void Spawn::SetUnassignedAA(sint16 new_val)
- {
- SetInfo(&basic_info.unassigned_aa, new_val);
- }
- void Spawn::SetTradeskillAA(sint16 new_val)
- {
- SetInfo(&basic_info.tradeskill_aa, new_val);
- }
- void Spawn::SetUnassignedTradeskillAA(sint16 new_val)
- {
- SetInfo(&basic_info.unassigned_tradeskill_aa, new_val);
- }
- void Spawn::SetPrestigeAA(sint16 new_val)
- {
- SetInfo(&basic_info.prestige_aa, new_val);
- }
- void Spawn::SetUnassignedPrestigeAA(sint16 new_val)
- {
- SetInfo(&basic_info.unassigned_prestige_aa, new_val);
- }
- void Spawn::SetTradeskillPrestigeAA(sint16 new_val)
- {
- SetInfo(&basic_info.tradeskill_prestige_aa, new_val);
- }
- void Spawn::SetUnassignedTradeskillPrestigeAA(sint16 new_val)
- {
- SetInfo(&basic_info.unassigned_tradeskill_prestige_aa, new_val);
- }
- sint16 Spawn::GetAssignedAA()
- {
- return basic_info.assigned_aa;
- }
- sint16 Spawn::GetUnassignedAA()
- {
- return basic_info.unassigned_aa;
- }
- sint16 Spawn::GetTradeskillAA()
- {
- return basic_info.tradeskill_aa;
- }
- sint16 Spawn::GetUnassignedTradeskillAA()
- {
- return basic_info.unassigned_tradeskill_aa;
- }
- sint16 Spawn::GetPrestigeAA()
- {
- return basic_info.prestige_aa;
- }
- sint16 Spawn::GetUnassignedPretigeAA()
- {
- return basic_info.unassigned_prestige_aa;
- }
- sint16 Spawn::GetTradeskillPrestigeAA()
- {
- return basic_info.tradeskill_prestige_aa;
- }
- sint16 Spawn::GetUnassignedTradeskillPrestigeAA()
- {
- return basic_info.unassigned_tradeskill_prestige_aa;
- }
- float Spawn::GetDistance(float x1, float y1, float z1, float x2, float y2, float z2){
- x1 = x1 - x2;
- y1 = y1 - y2;
- z1 = z1 - z2;
- return sqrt(x1*x1 + y1*y1 + z1*z1);
- }
- float Spawn::GetDistance(float x, float y, float z, float radius, bool ignore_y) {
- if (ignore_y)
- return GetDistance(x, y, z, GetX(), y, GetZ()) - radius;
- else
- return GetDistance(x, y, z, GetX(), GetY(), GetZ()) - radius;
- }
- float Spawn::GetDistance(float x, float y, float z, bool ignore_y) {
- return GetDistance(x, y, z, 0.0f, ignore_y);
- }
- float Spawn::GetDistance(Spawn* spawn, bool ignore_y, bool includeRadius){
- float ret = 0;
- if (spawn)
- {
- float radius = 0.0f;
- if (includeRadius)
- radius = CalculateRadius(spawn);
- ret = GetDistance(spawn->GetX(), spawn->GetY(), spawn->GetZ(), radius, ignore_y);
- }
- // maybe distance against ourselves, in that case we want to nullify the radius check
- if (ret < 0)
- ret = 0.0f;
- return ret;
- }
- float Spawn::GetDistance(Spawn* spawn, float x1, float y1, float z1, bool includeRadius) {
- float ret = 0;
- if (spawn)
- {
- float radius = 0.0f;
- if (includeRadius)
- radius = CalculateRadius(spawn);
- ret = GetDistance(x1, y1, z1, spawn->GetX(), spawn->GetY(), spawn->GetZ()) - radius;
- }
- // maybe distance against ourselves, in that case we want to nullify the radius check
- if (ret < 0)
- ret = 0.0f;
- return ret;
- }
- float Spawn::CalculateRadius(Spawn* target)
- {
- float srcRadius = short_to_float(appearance.pos.collision_radius);
- if (target)
- {
- float targRadius = short_to_float(target->appearance.pos.collision_radius);
- return (targRadius / 32.0f) + (srcRadius / 32.0f);
- }
- else
- return (srcRadius / 32.0f);
- }
- int32 Spawn::GetRespawnTime(){
- return respawn;
- }
- void Spawn::SetRespawnTime(int32 time){
- respawn = time;
- }
- int32 Spawn::GetExpireOffsetTime(){
- return expire_offset;
- }
- void Spawn::SetExpireOffsetTime(int32 time){
- expire_offset = time;
- }
- int32 Spawn::GetSpawnLocationID(){
- return spawn_location_id;
- }
- void Spawn::SetSpawnLocationID(int32 id){
- spawn_location_id = id;
- }
- int32 Spawn::GetSpawnEntryID(){
- return spawn_entry_id;
- }
- void Spawn::SetSpawnEntryID(int32 id){
- spawn_entry_id = id;
- }
- int32 Spawn::GetSpawnLocationPlacementID(){
- return spawn_location_spawns_id;
- }
- void Spawn::SetSpawnLocationPlacementID(int32 id){
- spawn_location_spawns_id = id;
- }
- const char* Spawn::GetSpawnScript(){
- if(spawn_script.length() > 0)
- return spawn_script.c_str();
- else
- return 0;
- }
- void Spawn::SetSpawnScript(string name){
- spawn_script = name;
- }
- void Spawn::SetPrimaryCommand(const char* name, const char* command, float distance){
- EntityCommand* entity_command = CreateEntityCommand(name, distance, command, "", 0, 0);
- if(primary_command_list.size() > 0 && primary_command_list[0]){
- safe_delete(primary_command_list[0]);
- primary_command_list[0] = entity_command;
- }
- else
- primary_command_list.push_back(entity_command);
- }
- void Spawn::SetSecondaryCommands(vector<EntityCommand*>* commands){
- if(commands && commands->size() > 0){
- vector<EntityCommand*>::iterator itr;
- if(secondary_command_list.size() > 0){
- for(itr = secondary_command_list.begin(); itr != secondary_command_list.end(); itr++){
- safe_delete(*itr);
- }
- secondary_command_list.clear();
- }
- EntityCommand* command = 0;
- for(itr = commands->begin(); itr != commands->end(); itr++){
- command = CreateEntityCommand(*itr);
- secondary_command_list.push_back(command);
- }
- }
- }
- void Spawn::SetPrimaryCommands(vector<EntityCommand*>* commands){
- if(commands && commands->size() > 0){
- vector<EntityCommand*>::iterator itr;
- if(primary_command_list.size() > 0){
- for(itr = primary_command_list.begin(); itr != primary_command_list.end(); itr++){
- safe_delete(*itr);
- }
- primary_command_list.clear();
- }
- EntityCommand* command = 0;
- for(itr = commands->begin(); itr != commands->end(); itr++){
- command = CreateEntityCommand(*itr);
- primary_command_list.push_back(command);
- }
- }
- }
- EntityCommand* Spawn::FindEntityCommand(string command, bool primaryOnly) {
- EntityCommand* entity_command = 0;
- if (primary_command_list.size() > 0) {
- vector<EntityCommand*>::iterator itr;
- for (itr = primary_command_list.begin(); itr != primary_command_list.end(); itr++) {
- if ((*itr)->command.compare(command) == 0) {
- entity_command = *itr;
- break;
- }
- }
- }
- if (primaryOnly)
- return entity_command;
- if (!entity_command && secondary_command_list.size() > 0) {
- vector<EntityCommand*>::iterator itr;
- for (itr = secondary_command_list.begin(); itr != secondary_command_list.end(); itr++) {
- if ((*itr)->command == command) {
- entity_command = *itr;
- break;
- }
- }
- }
- return entity_command;
- }
- void Spawn::SetSizeOffset(int8 offset){
- size_offset = offset;
- }
- int8 Spawn::GetSizeOffset(){
- return size_offset;
- }
- void Spawn::SetMerchantID(int32 val){
- merchant_id = val;
- }
- int32 Spawn::GetMerchantID(){
- return merchant_id;
- }
- void Spawn::SetMerchantType(int8 val){
- merchant_type = val;
- }
- int8 Spawn::GetMerchantType(){
- return merchant_type;
- }
- void Spawn::SetQuestsRequired(map<int32, vector<int16>* >* quests){
- if(quests){
- map<int32, vector<int16>* >::iterator itr;
- for(itr = quests->begin(); itr != quests->end(); itr++){
- vector<int16>* quest_steps = itr->second;
- for (int32 i = 0; i < quest_steps->size(); i++)
- SetQuestsRequired(itr->first, quest_steps->at(i));
- }
- }
- }
- void Spawn::SetQuestsRequired(int32 quest_id, int16 quest_step){
- m_requiredQuests.writelock(__FUNCTION__, __LINE__);
- if (required_quests.count(quest_id) == 0)
- required_quests[quest_id] = new vector<int16>;
- else{
- for (int32 i = 0; i < required_quests[quest_id]->size(); i++){
- if (required_quests[quest_id]->at(i) == quest_step){
- m_requiredQuests.releasewritelock(__FUNCTION__, __LINE__);
- return;
- }
- }
- }
- required_quests[quest_id]->push_back(quest_step);
- m_requiredQuests.releasewritelock(__FUNCTION__, __LINE__);
- }
- void Spawn::SetRequiredHistory(int32 event_id, int32 value1, int32 value2){
- LUAHistory set_value;
- set_value.Value = value1;
- set_value.Value2 = value2;
- set_value.SaveNeeded = false;
- m_requiredHistory.writelock(__FUNCTION__, __LINE__);
- required_history[event_id] = set_value;
- m_requiredHistory.releasewritelock(__FUNCTION__, __LINE__);
- }
- map<int32, vector<int16>* >* Spawn::GetQuestsRequired(){
- return &required_quests;
- }
- void Spawn::SetTransporterID(int32 id){
- transporter_id = id;
- }
- int32 Spawn::GetTransporterID(){
- return transporter_id;
- }
- void Spawn::InitializePosPacketData(Player* player, PacketStruct* packet, bool bSpawnUpdate) {
- int16 version = packet->GetVersion();
- packet->setDataByName("pos_grid_id", appearance.pos.grid_id);
- bool include_heading = true;
- if (IsWidget() && ((Widget*)this)->GetIncludeHeading() == false)
- include_heading = false;
- else if (IsSign() && ((Sign*)this)->GetIncludeHeading() == false)
- include_heading = false;
- else if (IsGroundSpawn())
- include_heading = false;
- if (include_heading){
- packet->setDataByName("pos_heading1", appearance.pos.Dir1);
- packet->setDataByName("pos_heading2", appearance.pos.Dir2);
- }
- if (version <= 910) {
- packet->setDataByName("pos_collision_radius", appearance.pos.collision_radius > 0 ? appearance.pos.collision_radius : 32);
- packet->setDataByName("pos_size", size > 0 ? size : 32);
- packet->setDataByName("pos_size_multiplier", 32); //32 is normal
- }
- else {
- if (size == 0)
- size = 32;
- packet->setDataByName("pos_collision_radius", appearance.pos.collision_radius > 0 ? appearance.pos.collision_radius : 32);
- if (!IsPlayer())
- packet->setDataByName("pos_size", size > 0 ? (((float)size) / 32.0f) : 1.0f);
- else
- packet->setDataByName("pos_size", 1.0f);
- packet->setDataByName("pos_size_ratio", 1.0f);
- }
- packet->setDataByName("pos_state", appearance.pos.state);
- bool include_location = true;
- if (IsWidget() && ((Widget*)this)->GetIncludeLocation() == false)
- include_location = false;
- else if (IsSign() && ((Sign*)this)->GetIncludeLocation() == false)
- include_location = false;
- if (include_location){
- if (IsWidget() && ((Widget*)this)->GetMultiFloorLift()) {
- Widget* widget = (Widget*)this;
- float x = appearance.pos.X - widget->GetWidgetX();
- float y = appearance.pos.Y - widget->GetWidgetY();
- float z = appearance.pos.Z - widget->GetWidgetZ();
- packet->setDataByName("pos_x", x);
- packet->setDataByName("pos_y", y);
- packet->setDataByName("pos_z", z);
- }
- else {
- packet->setDataByName("pos_x", appearance.pos.X);
- packet->setDataByName("pos_y", appearance.pos.Y);
- packet->setDataByName("pos_z", appearance.pos.Z);
- }
- if (IsSign())
- packet->setDataByName("pos_unknown6", 3, 2);
- }
- if (IsPlayer()) {
- packet->setDataByName("pos_x_velocity", static_cast<sint16>(GetSpeedX() * 32));
- packet->setDataByName("pos_y_velocity", static_cast<sint16>(GetSpeedY() * 32));
- packet->setDataByName("pos_z_velocity", static_cast<sint16>(GetSpeedZ() * 32));
- }
- bool bSendSpeed = true;
- if (IsWidget() && ((Widget*)this)->GetMultiFloorLift()) {
- Widget* widget = (Widget*)this;
- float x;
- float y;
- float z;
- if (IsRunning()){
- x = appearance.pos.X2 - widget->GetWidgetX();
- y = appearance.pos.Y2 - widget->GetWidgetY();
- z = appearance.pos.Z2- widget->GetWidgetZ();
- }
- else {
- x = appearance.pos.X - widget->GetWidgetX();
- y = appearance.pos.Y - widget->GetWidgetY();
- z = appearance.pos.Z - widget->GetWidgetZ();
- }
- packet->setDataByName("pos_next_x", x);
- packet->setDataByName("pos_next_y", y);
- packet->setDataByName("pos_next_z", z);
- packet->setDataByName("pos_x3", x);
- packet->setDataByName("pos_y3", y);
- packet->setDataByName("pos_z3", z);
- }
- //If this is a spawn update or this spawn is currently moving we can send these values, otherwise set speed and next_xyz to 0
- //This fixes the bug where spawns with movement scripts face south when initially spawning if they are at their target location.
- else if (bSpawnUpdate || memcmp(&appearance.pos.X, &appearance.pos.X2, sizeof(float) * 3) != 0) {
- packet->setDataByName("pos_next_x", appearance.pos.X2);
- packet->setDataByName("pos_next_y", appearance.pos.Y2);
- packet->setDataByName("pos_next_z", appearance.pos.Z2);
- packet->setDataByName("pos_x3", appearance.pos.X3);
- packet->setDataByName("pos_y3", appearance.pos.Y3);
- packet->setDataByName("pos_z3", appearance.pos.Z3);
- }
- else
- {
- bSendSpeed = false;
- }
- //packet->setDataByName("pos_unknown2", 4, 2);
- int16 speed_multiplier = rule_manager.GetGlobalRule(R_Spawn, SpeedMultiplier)->GetInt16(); // was 1280, 600 and now 300... investigating why
- if (IsPlayer()) {
- Player* player = static_cast<Player*>(this);
- packet->setDataByName("pos_speed", player->GetPosPacketSpeed() * speed_multiplier);
- packet->setDataByName("pos_side_speed", player->GetSideSpeed() * speed_multiplier);
- }
- else if (bSendSpeed) {
- packet->setDataByName("pos_speed", GetSpeed() * speed_multiplier);
- }
-
-
- if (IsNPC() || IsPlayer()) {
- packet->setDataByName("pos_move_type", 25);
- }
- else if (IsWidget() || IsSign()) {
- packet->setDataByName("pos_move_type", 11);
- }
- else if(IsGroundSpawn()) {
- packet->setDataByName("pos_move_type", 16);
- }
- if (!IsPlayer())
- packet->setDataByName("pos_movement_mode", 2);
-
- if(version <= 910)
- packet->setDataByName("pos_unknown10", 0xFFFF, 1);
- else if (version >= 1119)
- packet->setDataByName("face_actor_id", 0xFFFFFFFF);
- else
- packet->setDataByName("pos_unknown10", 0xFFFF);
- if(version <= 910)
- packet->setDataByName("pos_unknown10", 0xFFFF, 2);
- else if (version >= 1119)
- packet->setDataByName("face_actor_id", 0xFFFFFFFF);
- else
- packet->setDataByName("pos_unknown10", 0XFFFF, 1);
- packet->setDataByName("pos_pitch1", appearance.pos.Pitch1);
- packet->setDataByName("pos_pitch2", appearance.pos.Pitch2);
- packet->setDataByName("pos_roll", appearance.pos.Roll);
- }
- void Spawn::InitializeInfoPacketData(Player* spawn, PacketStruct* packet){
- int16 version = packet->GetVersion();
- bool spawnHiddenFromClient = false;
- // radius of 0 is always seen, -1 is never seen (unless items/spells override), larger than 0 is a defined radius to restrict visibility
- sint32 radius = rule_manager.GetGlobalRule(R_PVP, InvisPlayerDiscoveryRange)->GetSInt32();
- if (radius != 0 && (Spawn*)spawn != this && this->IsPlayer() && !spawn->CanSeeInvis((Entity*)this))
- spawnHiddenFromClient = true;
- if(!spawnHiddenFromClient && (appearance.targetable == 1 || appearance.show_level == 1 || appearance.display_name == 1)){
- appearance.locked_no_loot = 1; //for now
- if(!IsObject() && !IsGroundSpawn() && !IsWidget() && !IsSign()){
- int8 percent = 0;
- if(GetHP() > 0)
- percent = (int8)(((float)GetHP()/GetTotalHP()) * 100);
- if(percent < 100){
- packet->setDataByName("hp_remaining", 100 ^ percent);
- }
- else
- packet->setDataByName("hp_remaining", 0);
- if(GetTotalPower() > 0){
- percent = (int8)(((float)GetPower()/GetTotalPower()) * 100);
- if(percent > 0)
- packet->setDataByName("power_percent", percent);
- else
- packet->setDataByName("power_percent", 0);
- }
- }
- }
- packet->setDataByName("level", (int8)GetLevel());
- packet->setDataByName("unknown4", (int8)GetLevel());
- packet->setDataByName("difficulty", appearance.encounter_level); //6);
- packet->setDataByName("heroic_flag", appearance.heroic_flag);
- if(!IsObject() && !IsGroundSpawn() && !IsWidget() && !IsSign())
- packet->setDataByName("interaction_flag", 12); //this makes NPCs head turn to look at you
- packet->setDataByName("class", appearance.adventure_class);
- int16 model_type = appearance.model_type;
- if (GetIllusionModel() != 0) {
- if (IsPlayer()) {
- if (((Player*)this)->get_character_flag(CF_SHOW_ILLUSION)) {
- model_type = GetIllusionModel();
- }
- }
- else
- model_type = GetIllusionModel();
- }
- int16 sogaModelType = appearance.soga_model_type;
- if (spawnHiddenFromClient)
- {
- model_type = 0;
- sogaModelType = 0;
- }
- packet->setDataByName("model_type", model_type);
- if(appearance.soga_model_type == 0)
- packet->setDataByName("soga_model_type", model_type);
- else
- packet->setDataByName("soga_model_type", sogaModelType);
- if(GetTempActionState() >= 0)
- packet->setDataByName("action_state", GetTempActionState());
- else
- packet->setDataByName("action_state", appearance.action_state);
- if(GetTempVisualState() >= 0)
- packet->setDataByName("visual_state", GetTempVisualState());
- else
- packet->setDataByName("visual_state", appearance.visual_state);
- packet->setDataByName("emote_state", appearance.emote_state);
- packet->setDataByName("mood_state", appearance.mood_state);
- packet->setDataByName("gender", appearance.gender);
- packet->setDataByName("race", appearance.race);
- packet->setDataByName("gender", appearance.gender);
- if(IsEntity()){
- Entity* entity = ((Entity*)this);
- packet->setDataByName("combat_voice", entity->GetCombatVoice());
- packet->setDataByName("emote_voice", entity->GetEmoteVoice());
- for(int i=0;i<25;i++){
- if(i == 2){ //don't send helm if hidden flag
- if(IsPlayer()){
- if(((Player*)this)->get_character_flag(CF_HIDE_HELM)){
- packet->setDataByName("equipment_types", 0, i);
- packet->setColorByName("equipment_colors", 0, i);
- packet->setColorByName("equipment_highlights", 0, i);
- continue;
- }
- }
- if (IsBot()) {
- if (!((Bot*)this)->ShowHelm) {
- packet->setDataByName("equipment_types", 0, i);
- packet->setColorByName("equipment_colors", 0, i);
- packet->setColorByName("equipment_highlights", 0, i);
- continue;
- }
- }
- }
- else if(i == 19){ //don't send cloak if hidden
- if(IsPlayer()){
- if(!((Player*)this)->get_character_flag(CF_SHOW_CLOAK)){
- packet->setDataByName("equipment_types", 0, i);
- packet->setColorByName("equipment_colors", 0, i);
- packet->setColorByName("equipment_highlights", 0, i);
- continue;
- }
- }
- if (IsBot()) {
- if (!((Bot*)this)->ShowCloak) {
- packet->setDataByName("equipment_types", 0, i);
- packet->setColorByName("equipment_colors", 0, i);
- packet->setColorByName("equipment_highlights", 0, i);
- continue;
- }
- }
- }
- packet->setDataByName("equipment_types", entity->equipment.equip_id[i], i);
- packet->setColorByName("equipment_colors", entity->equipment.color[i], i);
- packet->setColorByName("equipment_highlights", entity->equipment.highlight[i], i);
- }
- packet->setDataByName("mount_type", entity->GetMount());
- // find the visual flags
- int8 vis_flag = 0;
- //Invis + crouch flag check
- if (entity->IsStealthed())
- vis_flag += (INFO_VIS_FLAG_INVIS + INFO_VIS_FLAG_CROUCH);
- //Invis flag check
- else if (entity->IsInvis())
- vis_flag += INFO_VIS_FLAG_INVIS;
- //Mount flag check
- if (entity->GetMount() > 0)
- vis_flag += INFO_VIS_FLAG_MOUNTED;
- //Hide hood check
- if ((IsPlayer() && ((Player*)this)->get_character_flag(CF_HIDE_HOOD)) || appearance.hide_hood)
- vis_flag += INFO_VIS_FLAG_HIDE_HOOD;
- packet->setDataByName("visual_flag", vis_flag);
- packet->setColorByName("mount_saddle_color", entity->GetMountSaddleColor());
- packet->setColorByName("mount_color", entity->GetMountColor());
- packet->setDataByName("hair_type_id", entity->features.hair_type);
- packet->setDataByName("chest_type_id", entity->features.chest_type);
- packet->setDataByName("wing_type_id", entity->features.wing_type);
- packet->setDataByName("legs_type_id", entity->features.legs_type);
- packet->setDataByName("soga_hair_type_id", entity->features.soga_hair_type);
- packet->setDataByName("facial_hair_type_id", entity->features.hair_face_type);
- packet->setDataByName("soga_facial_hair_type_id", entity->features.soga_hair_face_type);
- for(int i=0;i<3;i++){
- packet->setDataByName("eye_type", entity->features.eye_type[i], i);
- packet->setDataByName("ear_type", entity->features.ear_type[i], i);
- packet->setDataByName("eye_brow_type", entity->features.eye_brow_type[i], i);
- packet->setDataByName("cheek_type", entity->features.cheek_type[i], i);
- packet->setDataByName("lip_type", entity->features.lip_type[i], i);
- packet->setDataByName("chin_type", entity->features.chin_type[i], i);
- packet->setDataByName("nose_type", entity->features.nose_type[i], i);
- packet->setDataByName("soga_eye_type", entity->features.soga_eye_type[i], i);
- packet->setDataByName("soga_ear_type", entity->features.soga_ear_type[i], i);
- packet->setDataByName("soga_eye_brow_type", entity->features.soga_eye_brow_type[i], i);
- packet->setDataByName("soga_cheek_type", entity->features.soga_cheek_type[i], i);
- packet->setDataByName("soga_lip_type", entity->features.soga_lip_type[i], i);
- packet->setDataByName("soga_chin_type", entity->features.soga_chin_type[i], i);
- packet->setDataByName("soga_nose_type", entity->features.soga_nose_type[i], i);
- }
- packet->setColorByName("skin_color", entity->features.skin_color);
- packet->setColorByName("eye_color", entity->features.eye_color);
- packet->setColorByName("hair_type_color", entity->features.hair_type_color);
- packet->setColorByName("hair_type_highlight_color", entity->features.hair_type_highlight_color);
- packet->setColorByName("hair_face_color", entity->features.hair_face_color);
- packet->setColorByName("hair_face_highlight_color", entity->features.hair_face_highlight_color);
- packet->setColorByName("hair_highlight", entity->features.hair_highlight_color);
- packet->setColorByName("wing_color1", entity->features.wing_color1);
- packet->setColorByName("wing_color2", entity->features.wing_color2);
- packet->setColorByName("hair_color1", entity->features.hair_color1);
- packet->setColorByName("hair_color2", entity->features.hair_color2);
- packet->setColorByName("soga_skin_color", entity->features.soga_skin_color);
- packet->setColorByName("soga_eye_color", entity->features.soga_eye_color);
- packet->setColorByName("soga_hair_color1", entity->features.soga_hair_color1);
- packet->setColorByName("soga_hair_color2", entity->features.soga_hair_color2);
- packet->setColorByName("soga_hair_type_color", entity->features.soga_hair_type_color);
- packet->setColorByName("soga_hair_type_highlight_color", entity->features.soga_hair_type_highlight_color);
- packet->setColorByName("soga_hair_face_color", entity->features.soga_hair_face_color);
- packet->setColorByName("soga_hair_face_highlight_color", entity->features.soga_hair_face_highlight_color);
- packet->setColorByName("soga_hair_highlight", entity->features.soga_hair_highlight_color);
- packet->setDataByName("body_age", entity->features.body_age);
- }
- else{
- EQ2_Color empty;
- empty.red = 255;
- empty.blue = 255;
- empty.green = 255;
- packet->setColorByName("skin_color", empty);
- packet->setColorByName("eye_color", empty);
- packet->setColorByName("soga_skin_color", empty);
- packet->setColorByName("soga_eye_color", empty);
- }
- if(appearance.icon == 0){
- if(appearance.attackable == 1)
- appearance.icon = 0;
- else if(appearance.encounter_level > 0)
- appearance.icon = 4;
- else
- appearance.icon = 6;
- }
-
- // If Coe+ clients modify the values before we send
- // if not then just send the value we have.
- int8 temp_icon = appearance.icon;
- //Check if we need to add the hand icon..
- if ((temp_icon & 6) != 6 && appearance.display_hand_icon) {
- temp_icon |= 6;
- }
- //Icon value 28 for boats, set this without modifying the value
- if (version >= 1188 && temp_icon != 28) {
- if ((temp_icon & 64) > 0) {
- temp_icon -= 64; // remove the DoV value;
- temp_icon += 128; // add the CoE value
- }
- if ((temp_icon & 32) > 0) {
- temp_icon -= 32; // remove the DoV value;
- temp_icon += 64; // add the CoE value
- }
- if ((temp_icon & 4) > 0) {
- temp_icon -= 4; // remove DoV value
- temp_icon += 8; // add the CoE icon
- }
- if ((temp_icon & 6) > 0) {
- temp_icon -= 10; // remove DoV value
- temp_icon += 12; // add the CoE icon
- }
- }
- packet->setDataByName("icon", temp_icon);//appearance.icon);
- int32 temp_activity_status = 0;
-
- if (!Alive() && GetTotalHP() > 0 && !IsObject() && !IsGroundSpawn())
- temp_activity_status = 1;
- temp_activity_status += (IsNPC() || IsObject() || IsGroundSpawn()) ? 1 << 1 : 0;
- if (version >= 1188) {
- if (IsGroundSpawn() || GetShowHandIcon())
- temp_activity_status += ACTIVITY_STATUS_INTERACTABLE_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_ROLEPLAYING) > 0)
- temp_activity_status += ACTIVITY_STATUS_ROLEPLAYING_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_ANONYMOUS) > 0)
- temp_activity_status += ACTIVITY_STATUS_ANONYMOUS_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_LINKDEAD) > 0)
- temp_activity_status += ACTIVITY_STATUS_LINKDEAD_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_CAMPING) > 0)
- temp_activity_status += ACTIVITY_STATUS_CAMPING_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_LFG) > 0)
- temp_activity_status += ACTIVITY_STATUS_LFG_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_LFW) > 0)
- temp_activity_status += ACTIVITY_STATUS_LFW_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_SOLID) > 0)
- temp_activity_status += ACTIVITY_STATUS_SOLID_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_IMMUNITY_GAINED) > 0)
- temp_activity_status += ACTIVITY_STATUS_IMMUNITY_GAINED_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_IMMUNITY_REMAINING) > 0)
- temp_activity_status += ACTIVITY_STATUS_IMMUNITY_REMAINING_1188;
- if ((appearance.activity_status & ACTIVITY_STATUS_AFK) > 0)
- temp_activity_status += ACTIVITY_STATUS_AFK_1188;
- if (EngagedInCombat())
- temp_activity_status += ACTIVITY_STATUS_INCOMBAT_1188;
- // if this is either a boat or lift let the client be manipulated by the object
- if (appearance.icon == 28 || appearance.icon == 12)
- temp_activity_status += ACTIVITY_STATUS_ISTRANSPORT_1188;
- }
- else
- {
- temp_activity_status = appearance.activity_status;
- // WE ARE UNSURE OF THESE OLD CLIENT VALUES USED AS TEMP PLACEHOLDERS FOR NEWER CLIENTS
- if ((appearance.activity_status & ACTIVITY_STATUS_AFK) > 0)
- temp_activity_status -= ACTIVITY_STATUS_AFK;
- }
- packet->setDataByName("activity_status", temp_activity_status); //appearance.activity_status);
- // If player and player has a follow target
- if (IsPlayer()) {
- if (((Player*)this)->GetFollowTarget())
- packet->setDataByName("follow_target", ((((Player*)this)->GetIDWithPlayerSpawn(((Player*)this)->GetFollowTarget()) * -1) - 1));
- else
- packet->setDataByName("follow_target", 0);
- }
- if (GetTarget() && GetTarget()->GetTargetable())
- packet->setDataByName("target_id", ((spawn->GetIDWithPlayerSpawn(GetTarget()) * -1) - 1));
- else
- packet->setDataByName("target_id", 0);
- //Send spell effects for target window
- if(IsEntity()){
- InfoStruct* info = ((Entity*)this)->GetInfoStruct();
- int8 i = 0;
- int16 backdrop = 0;
- int16 spell_icon = 0;
- int32 spell_id = 0;
- LuaSpell* spell = 0;
- ((Entity*)this)->GetSpellEffectMutex()->readlock(__FUNCTION__, __LINE__);
- while(i < 30){
- //Change value of spell id for this packet if spell exists
- spell_id = info->spell_effects[i].spell_id;
- if(spell_id > 0)
- spell_id = 0xFFFFFFFF - spell_id;
- else
- spell_id = 0;
- packet->setSubstructDataByName("spell_effects", "spell_id", spell_id, i);
- //Change value of spell icon for this packet if spell exists
- spell_icon = info->spell_effects[i].icon;
- if(spell_icon > 0){
- if(!(spell_icon == 0xFFFF))
- spell_icon = 0xFFFF - spell_icon;
- }
- else
- spell_icon = 0;
- packet->setSubstructDataByName("spell_effects", "spell_icon", spell_icon, i);
- //Change backdrop values to match values in this packet
- backdrop = info->spell_effects[i].icon_backdrop;
- switch(backdrop){
- case 312:
- backdrop = 33080;
- break;
- case 313:
- backdrop = 33081;
- break;
- case 314:
- backdrop = 33082;
- break;
- case 315:
- backdrop = 33083;
- break;
- case 316:
- backdrop = 33084;
- break;
- case 317:
- backdrop = 33085;
- break;
- case (318 || 319):
- backdrop = 33086;
- break;
- default:
- break;
- }
- packet->setSubstructDataByName("spell_effects", "spell_icon_backdrop", backdrop, i);
- spell = info->spell_effects[i].spell;
- if (spell)
- packet->setSubstructDataByName("spell_effects", "spell_triggercount", spell->num_triggers, i);
- i++;
- }
- ((Entity*)this)->GetSpellEffectMutex()->releasereadlock(__FUNCTION__, __LINE__);
- }
- }
- void Spawn::MoveToLocation(Spawn* spawn, float distance, bool immediate, bool mapped){
- if(!spawn)
- return;
- SetRunningTo(spawn);
- FaceTarget(spawn);
- if (!IsPlayer() && distance > 0.0f)
- {
- if (IsFlying() && CheckLoS(spawn))
- {
- if (immediate)
- ClearRunningLocations();
- AddRunningLocation(spawn->GetX(), spawn->GetY(), spawn->GetZ(), GetSpeed(), distance, true, true, "", true);
- }
- else if (/*!mapped && */GetZone())
- {
- GetZone()->movementMgr->NavigateTo((Entity*)this, spawn->GetX(), spawn->GetY(), spawn->GetZ());
- last_grid_update = Timer::GetCurrentTime2();
- }
- else
- {
- if (immediate)
- ClearRunningLocations();
- AddRunningLocation(spawn->GetX(), spawn->GetY(), spawn->GetZ(), GetSpeed(), distance, true, true, "", mapped);
- }
- }
- }
- void Spawn::ProcessMovement(bool isSpawnListLocked){
- CheckProximities();
- if(IsPlayer()){
- //Check if player is riding a boat, if so update pos (boat's current location + XYZ offsets)
- Player* player = ((Player*)this);
- int32 boat_id = player->GetBoatSpawn();
- Spawn* boat = 0;
- if(boat_id > 0)
- boat = GetZone()->GetSpawnByID(boat_id, isSpawnListLocked);
- if(boat){
- SetX(boat->GetX() + player->GetBoatX());
- SetY(boat->GetY() + player->GetBoatY());
- SetZ(boat->GetZ() + player->GetBoatZ());
- }
- return;
- }
- if (forceMapCheck && GetZone() != nullptr && zone->zonemap != nullptr && zone->zonemap->IsMapLoaded())
- {
- FixZ(true);
- int32 newGrid = GetZone()->Grid->GetGridID(this);
- if (!IsFlying() && newGrid != 0 && newGrid != appearance.pos.grid_id)
- SetPos(&(appearance.pos.grid_id), newGrid);
- forceMapCheck = false;
- }
- if (GetHP() <= 0 && !IsWidget())
- return;
- if (EngagedInCombat())
- {
- int locations = 0;
- if (movement_locations && MMovementLocations)
- {
- MMovementLocations->readlock(__FUNCTION__, __LINE__);
- locations = movement_locations->size();
- MMovementLocations->releasereadlock(__FUNCTION__, __LINE__);
- }
- if (locations < 1 && GetZone() && ((Entity*)this)->IsFeared())
- {
- CalculateNewFearpoint();
- }
- }
- MMovementLoop.lock();
- Spawn* followTarget = GetZone()->GetSpawnByID(m_followTarget, isSpawnListLocked);
- if (!followTarget && m_followTarget > 0)
- m_followTarget = 0;
- if (following && followTarget && !((Entity*)this)->IsFeared()) {
- // Need to clear m_followTarget before the zoneserver deletes it
- if (followTarget->GetHP() <= 0) {
- followTarget = 0;
- MMovementLoop.unlock();
- return;
- }
- if (!IsEntity() || (!((Entity*)this)->IsCasting() && !((Entity*)this)->IsMezzedOrStunned() && !((Entity*)this)->IsRooted())) {
- if (GetBaseSpeed() > 0) {
- CalculateRunningLocation();
- }
- else {
- float speed = 4.0f;
- if (IsEntity())
- speed = ((Entity*)this)->GetMaxSpeed();
- SetSpeed(speed);
- }
- MovementLocation* loc = GetCurrentRunningLocation();
- float dist = GetDistance(followTarget, true);
- if ((!EngagedInCombat() && m_followDistance > 0 && dist <= m_followDistance) ||
- (dist <= rule_manager.GetGlobalRule(R_Combat, MaxCombatRange)->GetFloat())) {
- ClearRunningLocations();
- CalculateRunningLocation(true);
- }
- else if (loc) {
- float distance = GetDistance(followTarget, loc->x, loc->y, loc->z);
- if ( (!EngagedInCombat() && m_followDistance > 0 && distance > m_followDistance) ||
- ( EngagedInCombat() && distance > rule_manager.GetGlobalRule(R_Combat, MaxCombatRange)->GetFloat())) {
- MoveToLocation(followTarget, rule_manager.GetGlobalRule(R_Combat, MaxCombatRange)->GetFloat(), true, loc->mapped);
- CalculateRunningLocation();
- }
- }
- else {
- MoveToLocation(followTarget, rule_manager.GetGlobalRule(R_Combat, MaxCombatRange)->GetFloat());
- CalculateRunningLocation();
- }
- }
- }
- // Movement loop is only for scripted paths
- else if(!EngagedInCombat() && !NeedsToResumeMovement() && movement_loop.size() > 0 && movement_index < movement_loop.size() && (!IsNPC() || !((NPC*)this)->m_runningBack)){
- // Get the target location
- MovementData* data = movement_loop[movement_index];
- // need to resume our movement
- if(resume_movement){
- if (movement_locations){
- while (movement_locations->size()){
- safe_delete(movement_locations->front());
- movement_locations->pop_front();
- }
- movement_locations->clear();
- }
- data = movement_loop[movement_index];
- ((Entity*)this)->SetSpeed(data->speed);
- SetSpeed(data->speed);
- if(!IsWidget())
- FaceTarget(data->x, data->z);
- // 0 delay at target location, need to set multiple locations
- if(data->delay == 0 && movement_loop.size() > 0) {
- int16 tmp_index = movement_index+1;
- MovementData* data2 = 0;
- if(tmp_index < movement_loop.size())
- data2 = movement_loop[tmp_index];
- else
- data2 = movement_loop[0];
- AddRunningLocation(data->x, data->y, data->z, data->speed, 0, true, false, "", true);
- AddRunningLocation(data2->x, data2->y, data2->z, data2->speed, 0, true, true, "", true);
- }
- // delay at target location, only need to set 1 location
- else
- AddRunningLocation(data->x, data->y, data->z, data->speed);
- movement_start_time = 0;
- resume_movement = false;
- }
- // If we are not moving or we have arrived at our destination
- else if(!IsRunning() || (data && data->x == GetX() && data->y == GetY() && data->z == GetZ())){
- // If we were moving remove the last running location (the point we just arrived at)
- if(IsRunning())
- RemoveRunningLocation();
- // If this waypoint has a delay and we just arrived here (movement_start_time == 0)
- if(data->delay > 0 && movement_start_time == 0){
- // Set the current time
- movement_start_time = Timer::GetCurrentTime2();
- // If this waypoint had a lua function then call it
- if(data->lua_function.length() > 0)
- GetZone()->CallSpawnScript(this, SPAWN_SCRIPT_CUSTOM, 0, data->lua_function.c_str());
- int16 nextMove;
- if ((int16)(movement_index + 1) < movement_loop.size())
- nextMove = movement_index + 1;
- else
- nextMove = 0;
- // Get the next target location
- data = movement_loop[nextMove];
-
- //Go ahead and face the next location
- FaceTarget(data->x, data->z);
- }
- // If this waypoint has no delay or we have waited the required time (current time >= delay + movement_start_time)
- else if(data->delay == 0 || (data->delay > 0 && Timer::GetCurrentTime2() >= (data->delay+movement_start_time))) {
- // if no delay at this waypoint but a lua function for it then call the function
- if(data->delay == 0 && data->lua_function.length() > 0)
- GetZone()->CallSpawnScript(this, SPAWN_SCRIPT_CUSTOM, 0, data->lua_function.c_str());
- // Advance the current movement loop index
- if((int16)(movement_index+1) < movement_loop.size())
- movement_index++;
- else
- movement_index = 0;
- // Get the next target location
- data = movement_loop[movement_index];
- // set the speed for that location
- SetSpeed(data->speed);
- if(!IsWidget())
- // turn towards the location
- FaceTarget(data->x, data->z);
- // If 0 delay at location get and set data for the point after it
- if(data->delay == 0 && movement_loop.size() > 0){
- while (movement_locations->size()){
- safe_delete(movement_locations->front());
- movement_locations->pop_front();
- }
- // clear current target locations
- movement_locations->clear();
- // get the data for the location after out new location
- int16 tmp_index = movement_index+1;
- MovementData* data2 = 0;
- if(tmp_index < movement_loop.size())
- data2 = movement_loop[tmp_index];
- else
- data2 = movement_loop[0];
- // set the first location (adds it to movement_locations that we just cleared)
- AddRunningLocation(data->x, data->y, data->z, data->speed, 0, true, false, "", true);
- // set the location after that
- AddRunningLocation(data2->x, data2->y, data2->z, data2->speed, 0, true, true, "", true);
- }
- // there is a delay at the next location so we only need to set it
- else
- AddRunningLocation(data->x, data->y, data->z, data->speed);
- // reset this timer to 0 now that we are moving again
- movement_start_time = 0;
- }
- }
- // moving and not at target location yet
- else if(GetBaseSpeed() > 0)
- CalculateRunningLocation();
- // not moving, have a target location but not at it yet
- else if (data) {
- SetSpeed(data->speed);
- AddRunningLocation(data->x, data->y, data->z, data->speed);
- }
- }
- else if (IsRunning()) {
- CalculateRunningLocation();
- }
- /*else if (IsNPC() && !IsRunning() && !EngagedInCombat() && ((NPC*)this)->GetRunbackLocation()) {
- // Is an npc that is not moving and not engaged in combat but has a run back location set then clear the runback location
- LogWrite(NPC_AI__DEBUG, 7, "NPC_AI", "Clear runback location for %s", GetName());
- ((NPC*)this)->ClearRunback();
- resume_movement = true;
- NeedsToResumeMovement(false);
- }*/
- MMovementLoop.unlock();
- }
- void Spawn::ResetMovement(){
- MMovementLoop.lock();
- vector<MovementData*>::iterator itr;
- for(itr = movement_loop.begin(); itr != movement_loop.end(); itr++){
- safe_delete(*itr);
- }
- MMovementLoop.unlock();
- resume_movement = true;
- movement_index = 0;
- }
- void Spawn::AddMovementLocation(float x, float y, float z, float speed, int16 delay, const char* lua_function){
- LogWrite(LUA__DEBUG, 5, "LUA", "AddMovementLocation: x: %.2f, y: %.2f, z: %.2f, speed: %.2f, delay: %i, lua: %s",
- x, y, z, speed, delay, string(lua_function).c_str());
- MovementData* data = new MovementData;
- data->x = x;
- data->y = y;
- data->z = z;
- data->speed = speed;
- data->delay = delay*1000;
- if(lua_function)
- data->lua_function = string(lua_function);
- MMovementLoop.lock();
- movement_loop.push_back(data);
- MMovementLoop.unlock();
- }
- bool Spawn::IsRunning(){
- if(movement_locations && movement_locations->size() > 0)
- return true;
- else
- return false;
- }
- void Spawn::RunToLocation(float x, float y, float z, float following_x, float following_y, float following_z){
- if(!IsWidget())
- FaceTarget(x, z);
- SetPos(&appearance.pos.X2, x, false);
- SetPos(&appearance.pos.Z2, z, false);
- SetPos(&appearance.pos.Y2, y, false);
- if(following_x == 0 && following_y == 0 && following_z == 0){
- SetPos(&appearance.pos.X3, x, false);
- SetPos(&appearance.pos.Z3, z, false);
- SetPos(&appearance.pos.Y3, y, false);
- }
- else{
- SetPos(&appearance.pos.X3, following_x, false);
- SetPos(&appearance.pos.Y3, following_y, false);
- SetPos(&appearance.pos.Z3, following_z, false);
- }
- position_changed = true;
- changed = true;
- GetZone()->AddChangedSpawn(this);
- }
- MovementLocation* Spawn::GetCurrentRunningLocation(){
- MovementLocation* ret = 0;
- if(movement_locations && movement_locations->size() > 0){
- MMovementLocations->readlock(__FUNCTION__, __LINE__);
- ret = movement_locations->front();
- MMovementLocations->releasereadlock(__FUNCTION__, __LINE__);
- }
- return ret;
- }
- MovementLocation* Spawn::GetLastRunningLocation(){
- MovementLocation* ret = 0;
- if(movement_locations && movement_locations->size() > 0){
- MMovementLocations->readlock(__FUNCTION__, __LINE__);
- ret = movement_locations->back();
- MMovementLocations->releasereadlock(__FUNCTION__, __LINE__);
- }
- return ret;
- }
- void Spawn::AddRunningLocation(float x, float y, float z, float speed, float distance_away, bool attackable, bool finished_adding_locations, string lua_function, bool isMapped){
- if(speed == 0)
- return;
- ((Entity*)this)->SetSpeed(speed);
- MovementLocation* current_location = 0;
- float distance = GetDistance(x, y, z, distance_away != 0);
- if(distance_away != 0){
- distance -= distance_away;
- x = x - (GetX() - x)*distance_away/distance;
- z = z - (GetZ() - z)*distance_away/distance;
- }
- if(!movement_locations){
- movement_locations = new deque<MovementLocation*>();
- MMovementLocations = new Mutex();
- }
- MovementLocation* data = new MovementLocation;
- data->mapped = isMapped;
- data->x = x;
- data->y = y;
- data->z = z;
- data->speed = speed;
- data->attackable = attackable;
- data->lua_function = lua_function;
- data->gridid = 0; // used for runback defaults
- MMovementLocations->writelock(__FUNCTION__, __LINE__);
- if(movement_locations->size() > 0)
- current_location = movement_locations->back();
- if(!current_location){
- SetSpawnOrigX(GetX());
- SetSpawnOrigY(GetY());
- SetSpawnOrigZ(GetZ());
- SetSpawnOrigHeading(GetHeading());
- }
- movement_locations->push_back(data);
- if(finished_adding_locations){
- current_location = movement_locations->front();
- SetSpeed(current_location->speed);
- if(movement_locations->size() > 1){
- data = movement_locations->at(1);
- RunToLocation(current_location->x, current_location->y, current_location->z, data->x, data->y, data->z);
- }
- else
- RunToLocation(current_location->x, current_location->y, current_location->z, 0, 0, 0);
- }
- MMovementLocations->releasewritelock(__FUNCTION__, __LINE__);
- }
- bool Spawn::RemoveRunningLocation(){
- bool ret = false;
- if(movement_locations){
- MMovementLocations->writelock(__FUNCTION__, __LINE__);
- if(movement_locations->size() > 0){
- delete movement_locations->front();
- movement_locations->pop_front();
- ret = true;
- }
- MMovementLocations->releasewritelock(__FUNCTION__, __LINE__);
- }
- return ret;
- }
- void Spawn::ClearRunningLocations(){
- while(RemoveRunningLocation()){}
- }
- bool Spawn::CalculateChange(){
- bool remove_needed = false;
- if(movement_locations && MMovementLocations){
- MovementLocation* data = 0;
- MMovementLocations->readlock(__FUNCTION__, __LINE__);
- if(movement_locations->size() > 0){
- // Target location
- data = movement_locations->front();
- // If no target or we are at the target location need to remove this point
- if(!data || (data->x == GetX() && data->y == GetY() && data->z == GetZ()))
- remove_needed = true;
- if(data){
- if(NeedsToResumeMovement()){
- resume_movement = true;
- NeedsToResumeMovement(false);
- }
- if(!data->attackable)
- SetHeading(GetSpawnOrigHeading());
- }
- }
- MMovementLocations->releasereadlock(__FUNCTION__, __LINE__);
- if(remove_needed) {
- if (data && data->lua_function.length() > 0)
- GetZone()->CallSpawnScript(this, SPAWN_SCRIPT_CUSTOM, 0, data->lua_function.c_str());
- RemoveRunningLocation();
- //CalculateChange();
- }
- else if(data){
- // Speed is per second so we need a time_step (amount of time since the last update) to modify movement by
- float time_step = (Timer::GetCurrentTime2() - last_movement_update) * 0.001; // * 0.001 is the same as / 1000, float muliplications is suppose to be faster though
- // Get current location
- float nx = GetX();
- float ny = GetY();
- float nz = GetZ();
-
- // Get Forward vecotr
- float tar_vx = data->x - nx;
- float tar_vy = data->y - ny;
- float tar_vz = data->z - nz;
- // Multiply speed by the time_step to get how much should have changed over the last tick
- float speed = GetSpeed() * time_step;
- // Normalize the forward vector and multiply by speed, this gives us our change in coords, just need to add them to our current coords
- float len = sqrtf(tar_vx * tar_vx + tar_vy * tar_vy + tar_vz * tar_vz);
- tar_vx = (tar_vx / len) * speed;
- tar_vy = (tar_vy / len) * speed;
- tar_vz = (tar_vz / len) * speed;
- // Distance less then 0.5 just set the npc to the target location
- if (GetDistance(data->x, data->y, data->z, IsWidget() ? false : true) <= 0.5f) {
- SetX(data->x, false);
- SetZ(data->z, false);
- SetY(data->y, false, true);
- }
- else {
- SetX(nx + tar_vx, false);
- SetZ(nz + tar_vz, false);
- if ( IsWidget() )
- SetY(ny + tar_vy, false, true);
- else
- SetY(ny + tar_vy, false);
- }
- if (GetZone()->Grid != nullptr) {
- Cell* newCell = GetZone()->Grid->GetCell(GetX(), GetZ());
- if (newCell != Cell_Info.CurrentCell) {
- GetZone()->Grid->RemoveSpawnFromCell(this);
- GetZone()->Grid->AddSpawn(this, newCell);
- }
- int32 newGrid = GetZone()->Grid->GetGridID(this);
- if (!IsFlying() && newGrid != 0 && newGrid != appearance.pos.grid_id)
- SetPos(&(appearance.pos.grid_id), newGrid);
- }
- }
- }
- return remove_needed;
- }
- void Spawn::CalculateRunningLocation(bool stop){
- if (!stop && (last_location_update + 100) > Timer::GetCurrentTime2())
- return;
- else if (!stop)
- last_location_update = Timer::GetCurrentTime2();
- bool removed = CalculateChange();
- if (stop) {
- //following = false;
- SetPos(&appearance.pos.X2, GetX(), false);
- SetPos(&appearance.pos.Y2, GetY(), false);
- SetPos(&appearance.pos.Z2, GetZ(), false);
- SetPos(&appearance.pos.X3, GetX(), false);
- SetPos(&appearance.pos.Y3, GetY(), false);
- SetPos(&appearance.pos.Z3, GetZ(), false);
- }
- else if (removed && movement_locations && movement_locations->size() > 0) {
- MovementLocation* current_location = movement_locations->at(0);
- if (movement_locations->size() > 1) {
- MovementLocation* data = movement_locations->at(1);
- RunToLocation(current_location->x, current_location->y, current_location->z, data->x, data->y, data->z);
- }
- else
- RunToLocation(current_location->x, current_location->y, current_location->z, 0, 0, 0);
- }
- else if (GetZone() && GetTarget() != NULL && EngagedInCombat())
- {
- if (GetDistance(GetTarget()) > rule_manager.GetGlobalRule(R_Combat, MaxCombatRange)->GetFloat())
- {
- if (IsFlying() && CheckLoS(GetTarget()))
- AddRunningLocation(GetTarget()->GetX(), GetTarget()->GetY(), GetTarget()->GetZ(), GetSpeed(), 0, false);
- else
- GetZone()->movementMgr->NavigateTo((Entity*)this, GetTarget()->GetX(), GetTarget()->GetY(), GetTarget()->GetZ());
- }
- else
- ((Entity*)this)->HaltMovement();
- }
- }
- float Spawn::GetFaceTarget(float x, float z) {
- float angle;
- double diff_x = x - GetX();
- double diff_z = z - GetZ();
- //If we're very close to the same spot don't bother changing heading
- if (sqrt(diff_x * diff_x * diff_z * diff_z) < .1) {
- return GetHeading();
- }
- if (diff_z == 0) {
- if (diff_x > 0)
- angle = 90;
- else
- angle = 270;
- }
- else
- angle = ((atan(diff_x / diff_z)) * 180) / 3.14159265358979323846;
- if (angle < 0)
- angle = angle + 360;
- else
- angle = angle + 180;
- if (diff_x < 0)
- angle = angle + 180;
- if (last_heading_angle == angle) return angle;
- return angle;
- }
- void Spawn::FaceTarget(float x, float z){
- float angle;
- double diff_x = x - GetX();
- double diff_z = z - GetZ();
- //If we're very close to the same spot don't bother changing heading
- if (sqrt(diff_x * diff_x * diff_z * diff_z) < .1) {
- return;
- }
- if(diff_z==0){
- if(diff_x > 0)
- angle = 90;
- else
- angle = 270;
- }
- else
- angle = ((atan(diff_x / diff_z)) * 180) / 3.14159265358979323846;
- if(angle < 0)
- angle = angle + 360;
- else
- angle = angle + 180;
- if(diff_x < 0)
- angle = angle + 180;
- if (last_heading_angle == angle) return;
- SetHeading(angle);
- }
- void Spawn::FaceTarget(Spawn* target){
- if(!target)
- return;
- FaceTarget(target->GetX(), target->GetZ());
- if(GetHP() > 0 && target->IsPlayer() && !EngagedInCombat()){
- GetZone()->AddHeadingTimer(this);
- SetTempActionState(0);
- }
- }
- bool Spawn::MeetsSpawnAccessRequirements(Player* player){
- bool ret = false;
- Quest* quest = 0;
- //Check if we meet all quest requirements first..
- m_requiredQuests.readlock(__FUNCTION__, __LINE__);
- if (player && required_quests.size() > 0) {
- map<int32, vector<int16>* >::iterator itr;
- for (itr = required_quests.begin(); itr != required_quests.end(); itr++) {
- player->AddQuestRequiredSpawn(this, itr->first);
- vector<int16>* quest_steps = itr->second;
- for (int32 i = 0; i < quest_steps->size(); i++) {
- quest = player->GetQuest(itr->first);
- if (req_quests_continued_access) {
- if (quest) {
- if (quest->GetQuestStepCompleted(quest_steps->at(i))) {
- ret = true;
- break;
- }
- }
- else if (player->GetCompletedQuest(itr->first)) {
- ret = true;
- break;
- }
- }
- if (quest && quest->QuestStepIsActive(quest_steps->at(i))) {
- ret = true;
- break;
- }
- }
- }
- }
- else
- ret = true;
- m_requiredQuests.releasereadlock(__FUNCTION__, __LINE__);
- if (!ret)
- return ret;
- //Now check if the player meets all history requirements
- m_requiredHistory.readlock(__FUNCTION__, __LINE__);
- if (required_history.size() > 0){
- map<int32, LUAHistory>::iterator itr;
- for (itr = required_history.begin(); itr != required_history.end(); itr++){
- player->AddHistoryRequiredSpawn(this, itr->first);
- LUAHistory* player_history = player->GetLUAHistory(itr->first);
- if (player_history){
- if (player_history->Value != itr->second.Value || player_history->Value2 != itr->second.Value2)
- ret = false;
- }
- else
- ret = false;
- if (!ret)
- break;
- }
- }
- m_requiredHistory.releasereadlock(__FUNCTION__, __LINE__);
- return ret;
- }
- vector<Spawn*>* Spawn::GetSpawnGroup(){
- vector<Spawn*>* ret_list = 0;
- if(spawn_group_list){
- ret_list = new vector<Spawn*>();
- if(MSpawnGroup)
- MSpawnGroup->readlock(__FUNCTION__, __LINE__);
- ret_list->insert(ret_list->begin(), spawn_group_list->begin(), spawn_group_list->end());
- if(MSpawnGroup)
- MSpawnGroup->releasereadlock(__FUNCTION__, __LINE__);
- }
- return ret_list;
- }
- bool Spawn::HasSpawnGroup() {
- return spawn_group_list && spawn_group_list->size() > 0;
- }
- bool Spawn::IsInSpawnGroup(Spawn* spawn) {
- bool ret = false;
- if (HasSpawnGroup() && spawn) {
- vector<Spawn*>::iterator itr;
- for (itr = spawn_group_list->begin(); itr != spawn_group_list->end(); itr++) {
- if ((*itr) == spawn) {
- ret = true;
- break;
- }
- }
- }
- return ret;
- }
- void Spawn::AddSpawnToGroup(Spawn* spawn){
- if(!spawn)
- return;
- if(!spawn_group_list){
- spawn_group_list = new vector<Spawn*>();
- spawn_group_list->push_back(this);
- safe_delete(MSpawnGroup);
- MSpawnGroup = new Mutex();
- MSpawnGroup->SetName("Spawn::MSpawnGroup");
- }
- vector<Spawn*>::iterator itr;
- MSpawnGroup->writelock(__FUNCTION__, __LINE__);
- for(itr = spawn_group_list->begin(); itr != spawn_group_list->end(); itr++){
- if((*itr) == spawn){
- MSpawnGroup->releasewritelock(__FUNCTION__, __LINE__);
- return;
- }
- }
- spawn_group_list->push_back(spawn);
- spawn->SetSpawnGroupList(spawn_group_list, MSpawnGroup);
- MSpawnGroup->releasewritelock(__FUNCTION__, __LINE__);
- }
- void Spawn::SetSpawnGroupList(vector<Spawn*>* list, Mutex* mutex){
- spawn_group_list = list;
- MSpawnGroup = mutex;
- }
- void Spawn::RemoveSpawnFromGroup(bool erase_all){
- SetSpawnGroupID(0);
- bool del = false;
- if(MSpawnGroup){
- MSpawnGroup->writelock(__FUNCTION__, __LINE__);
- if(spawn_group_list){
- vector<Spawn*>::iterator itr;
- Spawn* spawn = 0;
- if(spawn_group_list->size() == 1)
- erase_all = true;
- for(itr = spawn_group_list->begin(); itr != spawn_group_list->end(); itr++){
- spawn = *itr;
- if (spawn) {
- if(!erase_all){
- if(spawn == this){
- spawn_group_list->erase(itr);
- MSpawnGroup->releasewritelock(__FUNCTION__, __LINE__);
- spawn_group_list = 0;
- MSpawnGroup = 0;
- return;
- }
- }
- else{
- if (spawn != this)
- spawn->SetSpawnGroupList(0, 0);
- }
- }
- }
- if (erase_all)
- spawn_group_list->clear();
- del = (spawn_group_list->size() == 0);
- }
- MSpawnGroup->releasewritelock(__FUNCTION__, __LINE__);
- if (del){
- safe_delete(MSpawnGroup);
- safe_delete(spawn_group_list);
- }
- }
- }
- void Spawn::SetSpawnGroupID(int32 id){
- group_id = id;
- }
- int32 Spawn::GetSpawnGroupID(){
- return group_id;
- }
- void Spawn::AddChangedZoneSpawn(){
- if(send_spawn_changes && GetZone())
- GetZone()->AddChangedSpawn(this);
- }
- void Spawn::RemoveSpawnAccess(Spawn* spawn) {
- if (allowed_access.count(spawn->GetID()) > 0) {
- allowed_access.erase(spawn->GetID());
- GetZone()->HidePrivateSpawn(this);
- }
- }
- void Spawn::SetFollowTarget(Spawn* spawn, int32 follow_distance) {
- if (spawn && spawn != this) {
- m_followTarget = spawn->GetID();
- m_followDistance = follow_distance;
- }
- else {
- m_followTarget = 0;
- if (following)
- following = false;
- m_followDistance = 0;
- }
- }
- void Spawn::AddTempVariable(string var, string val) {
- m_tempVariableTypes[var] = 5;
- m_tempVariables[var] = val;
- }
- void Spawn::AddTempVariable(string var, Spawn* val) {
- m_tempVariableTypes[var] = 1;
- m_tempVariableSpawn[var] = val->GetID();
- }
- void Spawn::AddTempVariable(string var, ZoneServer* val) {
- m_tempVariableTypes[var] = 2;
- m_tempVariableZone[var] = val;
- }
- void Spawn::AddTempVariable(string var, Item* val) {
- m_tempVariableTypes[var] = 3;
- m_tempVariableItem[var] = val;
- }
- void Spawn::AddTempVariable(string var, Quest* val) {
- m_tempVariableTypes[var] = 4;
- m_tempVariableQuest[var] = val;
- }
- string Spawn::GetTempVariable(string var) {
- string ret = "";
- if (m_tempVariables.count(var) > 0)
- ret = m_tempVariables[var];
- return ret;
- }
- Spawn* Spawn::GetTempVariableSpawn(string var) {
- Spawn* ret = 0;
-
- if (m_tempVariableSpawn.count(var) > 0)
- ret = GetZone()->GetSpawnByID(m_tempVariableSpawn[var]);
- return ret;
- }
- ZoneServer* Spawn::GetTempVariableZone(string var) {
- ZoneServer* ret = 0;
- if (m_tempVariableZone.count(var) > 0)
- ret = m_tempVariableZone[var];
- return ret;
- }
- Item* Spawn::GetTempVariableItem(string var) {
- Item* ret = 0;
- if (m_tempVariableItem.count(var) > 0)
- ret = m_tempVariableItem[var];
- return ret;
- }
- Quest* Spawn::GetTempVariableQuest(string var) {
- Quest* ret = 0;
- if (m_tempVariableQuest.count(var) > 0)
- ret = m_tempVariableQuest[var];
- return ret;
- }
- int8 Spawn::GetTempVariableType(string var) {
- int8 ret = 0;
- if (m_tempVariableTypes.count(var) > 0)
- ret = m_tempVariableTypes[var];
- return ret;
- }
- void Spawn::DeleteTempVariable(string var) {
- int8 type = GetTempVariableType(var);
- switch (type) {
- case 1:
- m_tempVariableSpawn.erase(var);
- break;
- case 2:
- m_tempVariableZone.erase(var);
- break;
- case 3:
- m_tempVariableItem.erase(var);
- break;
- case 4:
- m_tempVariableQuest.erase(var);
- break;
- case 5:
- m_tempVariables.erase(var);
- break;
- }
- m_tempVariableTypes.erase(var);
- }
- Spawn* Spawn::GetRunningTo() {
- return GetZone()->GetSpawnByID(running_to);
- }
- Spawn* Spawn::GetFollowTarget() {
- return GetZone()->GetSpawnByID(m_followTarget);
- }
- void Spawn::CopySpawnAppearance(Spawn* spawn){
- if (!spawn)
- return;
- //This function copies the appearace of the provided spawn to this one
- if (spawn->IsEntity() && IsEntity()){
- memcpy(&((Entity*)this)->features, &((Entity*)spawn)->features, sizeof(CharFeatures));
- memcpy(&((Entity*)this)->equipment, &((Entity*)spawn)->equipment, sizeof(EQ2_Equipment));
- }
- SetSize(spawn->GetSize());
- SetModelType(spawn->GetModelType());
- }
- void Spawn::SetY(float y, bool updateFlags, bool disableYMapCheck)
- {
- SetPos(&appearance.pos.Y, y, updateFlags);
- if (!disableYMapCheck)
- FixZ();
- }
- float Spawn::FindDestGroundZ(glm::vec3 dest, float z_offset)
- {
- float best_z = BEST_Z_INVALID;
- if (GetZone() != nullptr && GetZone()->zonemap != nullptr)
- {
- dest.z += z_offset;
- best_z = zone->zonemap->FindBestZ(dest, nullptr);
- }
- return best_z;
- }
- float Spawn::GetFixedZ(const glm::vec3& destination, int32 z_find_offset) {
- BenchTimer timer;
- timer.reset();
- float new_z = destination.z;
- if (GetZone() != nullptr && zone->zonemap != nullptr) {
- /* if (flymode == GravityBehavior::Flying)
- return new_z;
- */
- /* if (zone->HasWaterMap() && zone->watermap->InLiquid(glm::vec3(m_Position)))
- return new_z;
- */
- /*
- * Any more than 5 in the offset makes NPC's hop/snap to ceiling in small corridors
- */
- new_z = this->FindDestGroundZ(destination, z_find_offset);
- if (new_z != BEST_Z_INVALID) {
- if (new_z < -2000) {
- new_z = GetY();
- }
- }
- auto duration = timer.elapsed();
- LogWrite(MAP__DEBUG, 0, "Map", "Mob::GetFixedZ() ([{%s}]) returned [{%f}] at [{%f}], [{%f}], [{%f}] - Took [{%f}]",
- this->GetName(),
- new_z,
- destination.x,
- destination.y,
- destination.z,
- duration);
- }
- return new_z;
- }
- void Spawn::FixZ(bool forceUpdate) {
- if (IsPlayer() || IsFlying() || !GetZone() || (IsObject() && GetZone()->GetInstanceType() == Instance_Type::PERSONAL_HOUSE_INSTANCE)) {
- return;
- }
- /*
- if (flymode == GravityBehavior::Flying) {
- return;
- }*/
- /*
- if (zone->watermap && zone->watermap->InLiquid(m_Position)) {
- return;
- }*/
- glm::vec3 current_loc(GetX(), GetZ(), GetY());
- float new_z = GetFixedZ(current_loc, 1);
- if (new_z == GetY())
- return;
- if ((new_z > -2000) && new_z != BEST_Z_INVALID) {
- SetY(new_z, forceUpdate, true);
- }
- else {
- LogWrite(MAP__DEBUG, 0, "Map", "[{%s}] is failing to find Z [{%f}]", this->GetName(), std::abs(GetY() - new_z));
- }
- }
- bool Spawn::CheckLoS(Spawn* target)
- {
- glm::vec3 targpos(target->GetX(), target->GetZ(), target->GetY()+1.0f);
- glm::vec3 pos(GetX(), GetZ(), GetY()+1.0f);
- return CheckLoS(pos, targpos);
- }
- bool Spawn::CheckLoS(glm::vec3 myloc, glm::vec3 oloc)
- {
- ZoneServer* zone = GetZone();
- if (zone == NULL || zone->zonemap == NULL || !zone->zonemap->IsMapLoaded())
- return true;
- else
- return zone->zonemap->CheckLoS(myloc, oloc);
- return false;
- }
- void Spawn::CalculateNewFearpoint()
- {
- if (GetZone() && GetZone()->pathing) {
- auto Node = zone->pathing->GetRandomLocation(glm::vec3(GetX(), GetZ(), GetY()));
- if (Node.x != 0.0f || Node.y != 0.0f || Node.z != 0.0f) {
- AddRunningLocation(Node.x, Node.y, Node.z, GetSpeed(), 0, true, true, "", true);
- }
- }
- }
- void Spawn::CheckProximities()
- {
- if (!has_spawn_proximities)
- return;
- if (spawn_proximities.size() > 0)
- {
- MutexList<SpawnProximity*>::iterator itr = spawn_proximities.begin();
- while (itr.Next()) {
- SpawnProximity* prox = itr.value;
- map<int32, bool>::iterator spawnsItr;
- for (spawnsItr = prox->spawns_in_proximity.begin(); spawnsItr != prox->spawns_in_proximity.end(); spawnsItr++) {
- Spawn* tmpSpawn = 0;
- if (spawnsItr->first &&
- ((prox->spawn_type == SPAWNPROXIMITY_DATABASE_ID && (tmpSpawn = GetZone()->GetSpawnByDatabaseID(spawnsItr->first)) != 0) ||
- (prox->spawn_type == SPAWNPROXIMITY_LOCATION_ID && (tmpSpawn = GetZone()->GetSpawnByLocationID(spawnsItr->first)) != 0)))
- {
- if (!spawnsItr->second && tmpSpawn->GetDistance(this) <= prox->distance)
- {
- if (prox->in_range_lua_function.size() > 0)
- GetZone()->CallSpawnScript(this, SPAWN_SCRIPT_CUSTOM, tmpSpawn, prox->in_range_lua_function.c_str());
- spawnsItr->second = true;
- }
- else if (spawnsItr->second && tmpSpawn->GetDistance(this) > prox->distance)
- {
- if (prox->leaving_range_lua_function.size() > 0)
- GetZone()->CallSpawnScript(this, SPAWN_SCRIPT_CUSTOM, tmpSpawn, prox->leaving_range_lua_function.c_str());
- spawnsItr->second = false;
- }
- }
- }
- }
- }
- }
- void Spawn::AddSpawnToProximity(int32 spawnValue, SpawnProximityType type)
- {
- if (!has_spawn_proximities)
- return;
- if (spawn_proximities.size() > 0)
- {
- MutexList<SpawnProximity*>::iterator itr = spawn_proximities.begin();
- while (itr.Next()) {
- SpawnProximity* prox = itr->value;
- if (prox->spawn_value == spawnValue && prox->spawn_type == type)
- prox->spawns_in_proximity.insert(make_pair(spawnValue, false));
- }
- }
- }
- void Spawn::RemoveSpawnFromProximity(int32 spawnValue, SpawnProximityType type)
- {
- if (!has_spawn_proximities)
- return;
- if (spawn_proximities.size() > 0)
- {
- MutexList<SpawnProximity*>::iterator itr = spawn_proximities.begin();
- while (itr.Next()) {
- SpawnProximity* prox = itr->value;
- if (prox->spawn_value == spawnValue && prox->spawn_type == type &&
- prox->spawns_in_proximity.count(spawnValue) > 0)
- prox->spawns_in_proximity.erase(spawnValue);
- }
- }
- }
- void Spawn::AddPrimaryEntityCommand(const char* name, float distance, const char* command, const char* error_text, int16 cast_time, int32 spell_visual, bool defaultDenyList, Player* player) {
- EntityCommand* cmd = FindEntityCommand(string(command), true);
- bool newCommand = false;
- if (!cmd)
- {
- newCommand = true;
- cmd = CreateEntityCommand(name, distance, command, error_text, cast_time, spell_visual, !defaultDenyList);
- }
- if (defaultDenyList)
- SetPermissionToEntityCommand(cmd, player, true);
- if (newCommand)
- primary_command_list.push_back(cmd);
- }
- void Spawn::RemovePrimaryEntityCommand(const char* command) {
- vector<EntityCommand*>::iterator itr;
- string tmpStr(command);
- for (itr = primary_command_list.begin(); itr != primary_command_list.end(); itr++) {
- EntityCommand* cmd = *itr;
- if (cmd->command.compare(tmpStr) == 0)
- {
- primary_command_list.erase(itr);
- delete cmd;
- break;
- }
- }
- }
- bool Spawn::SetPermissionToEntityCommand(EntityCommand* command, Player* player, bool permissionValue)
- {
- if (player != NULL)
- {
- map<int32, bool>::iterator itr = command->allow_or_deny.find(player->GetCharacterID());
- if (itr == command->allow_or_deny.end())
- command->allow_or_deny.insert(make_pair(player->GetCharacterID(), permissionValue));
- else if (itr->second != permissionValue)
- itr->second = permissionValue;
- return true;
- }
- return false;
- }
|