// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdpch.h" #include "world_entity.h" #include "sheets.h" #include "world_position_manager.h" #include "gpm_service.h" using namespace std; using namespace NLMISC; using namespace NLPACS; #define DUMP_VAR(var) log->displayNL("%28s: %s", #var, toString(var).c_str()); #define DUMP_VARPTR(var) log->displayNL("%28s: %s", #var, toStringPtr(var).c_str()); /****************************************************************\ **************************************************************** CWorldEntity **************************************************************** \****************************************************************/ // Static entity allocator CBlockMemory CWorldEntity::_EntityAllocator; template void registerEntityProperty(const string &propertyName, CMirrorPropValue1DS *property, const CEntityId &id) { // TODO: optimize: use entity index instead of Id, prop index instead of name property->init( TheDataset, id, propertyName ); } /// Creates a new entity (new equivalent). This must be initialised later using init(); CWorldEntity* CWorldEntity::create() { CWorldEntity* entity = _EntityAllocator.allocate(); if (entity != NULL) { entity->ListIterator = CWorldPositionManager::_EntityList.insert(CWorldPositionManager::_EntityList.end(), entity); entity->PrimIterator = CWorldPositionManager::_PrimitivedList.end(); } return entity; } /// Removes an entity (delete equivalent). void CWorldEntity::remove(CWorldEntity* entity) { CWorldPositionManager::_EntityList.erase(entity->ListIterator); if (entity->PrimIterator != CWorldPositionManager::_PrimitivedList.end()) { CWorldPositionManager::_PrimitivedList.erase(entity->PrimIterator); } _EntityAllocator.free(entity); } /****************************************************************\ init \****************************************************************/ void CWorldEntity::init( const CEntityId& id, const TDataSetRow &index ) { Id = id; Index = index; // Keep properties local for invisible group entities (aiVision). // This is obsolete, as aiVision are not used anymore. Because these entities were not // in mirror, their property values always stayed in temp storage. All the code was // the same for aiVision and visible entities, i.e. we used the CMirrorPropValueAlice. // Now the 'Alice' version is not needed anymore, because all entities go into // registerEntityProperty(). // if ( Id.getType() != RYZOMID::aiVision ) // { registerEntityProperty("X", &X, id); registerEntityProperty("Y", &Y, id); registerEntityProperty("Z", &Z, id); registerEntityProperty("LocalX", &LocalX, id); registerEntityProperty("LocalY", &LocalY, id); registerEntityProperty("LocalZ", &LocalZ, id); registerEntityProperty("Theta", &Theta, id); registerEntityProperty("Sheet", &Sheet, id); registerEntityProperty("TickPos", &Tick, id); registerEntityProperty("Cell", &Cell, id); registerEntityProperty("VisionCounter", &VisionCounter, id); registerEntityProperty("WhoSeesMe", &WhoSeesMe, id); if (Cell > 0) { nlwarning("Entity %s Cell is preset to %d which should be 0 or negative, forced to 0", id.toString().c_str(), Cell()); Cell = 0; } // } // else // Cell = -1; //IsStaticObject = false; //IsInvisible = false; //IsAgent = false; //IsTrigger = false; //IsInvisibleToPlayer = false; RefCounter = 0; TickLock = 0; TempVisionState = false; TempControlInVision = false; TempParentInVision = false; HasVision = false; Previous = NULL; Next = NULL; PlayerInfos = NULL; Parent = NULL; Control = NULL; PosInitialised = false; Continent = INVALID_CONTINENT_INDEX; MoveContainer = NULL; Primitive = NULL; UsePrimitive = false; ForceUsePrimitive = false; ForceDontUsePrimitive = false; VisionCounter = (uint8)0x0; PlayersSeeingMe = 0; ClosestPlayer = NULL; CellPtr = NULL; PatatEntryIndex = 0; if (id.getType() == RYZOMID::object ) { _Type = Object; } else if (id.getType() == RYZOMID::player) { _Type = Player; } else if (id.getType() == RYZOMID::trigger) { _Type = Trigger; } else if ((id.getType() >= RYZOMID::bot_ai_begin) && (id.getType() <= RYZOMID::bot_ai_end)) { _Type = AI; } else { _Type = Unknown; } // commented, now set by the EGS //WhoSeesMe = 0xffffffff; CheckMotion = true; } // CWorldEntity constructor /****************************************************************\ CWorldEntity destructor \****************************************************************/ CWorldEntity::~CWorldEntity() { // the Primitive should have been removed in the UMoveContainer, and so is already deleted } // CWorldEntity destructor /** * Display debug */ void CWorldEntity::display(NLMISC::CLog *log) const { if (this == NULL) log->displayNL("CWorldEntity::display called on NULL"); log->displayNL("--- Display Entity %s E%u ---", Id.toString().c_str(), Index.getIndex()); DUMP_VAR(X()); DUMP_VAR(Y()); DUMP_VAR(Z()); DUMP_VAR(LocalX()); DUMP_VAR(LocalY()); DUMP_VAR(LocalZ()); DUMP_VAR(Theta()); DUMP_VAR(Sheet()); DUMP_VAR(Tick()); DUMP_VAR(Cell()); DUMP_VAR(VisionCounter()); DUMP_VAR(WhoSeesMe()); DUMP_VAR(PatatEntryIndex); DUMP_VARPTR(CellPtr); DUMP_VAR(PosInitialised); DUMP_VAR(Continent); DUMP_VARPTR(Primitive); DUMP_VARPTR(MoveContainer); DUMP_VAR(UsePrimitive); DUMP_VAR(ForceUsePrimitive); DUMP_VAR(ForceDontUsePrimitive); DUMP_VAR(TickLock); DUMP_VARPTR((CWorldEntity*)Previous); DUMP_VARPTR((CWorldEntity*)Next); DUMP_VARPTR((CWorldEntity*)Parent); DUMP_VARPTR((CWorldEntity*)Control); DUMP_VAR(HasVision); DUMP_VAR(CheckMotion); DUMP_VAR(TempVisionState); DUMP_VAR(RefCounter); log->displayNL("--- End of Display ---"); if (PlayerInfos != NULL) PlayerInfos->display(log); } /****************************************************************\ create primitive for fiche type entity \****************************************************************/ void CWorldEntity::createPrimitive(NLPACS::UMoveContainer *pMoveContainer, uint8 worldImage) { UMovePrimitive *primitive; if (PrimIterator != CWorldPositionManager::_PrimitivedList.end()) { CWorldPositionManager::_PrimitivedList.erase(PrimIterator); PrimIterator = CWorldPositionManager::_PrimitivedList.end(); } if (!UsePrimitive && !ForceUsePrimitive) { nlwarning("Entity %s asked to create a PACS primitive, whereas shouldn't not, abort", Id.toString().c_str()); return; } if (pMoveContainer == NULL) { nlwarning("pMoveContainer == NULL in createPrimitive()"); return; } // if a primitive already exists if (Primitive != NULL) { // previous move container must not be null if (MoveContainer == NULL) { nlwarning("MoveContainer == NULL in createPrimitive()"); return; } // check the previous move container is different (otherwise does not create a primitive) if (MoveContainer == pMoveContainer) return; // depending on the type of the previous primitive, creates a copy of it if (Primitive->isCollisionable()) { primitive = pMoveContainer->addCollisionablePrimitive(CWorldPositionManager::_FirstDynamicWorldImage, CWorldPositionManager::_NbDynamicWorldImages, Primitive); } else { primitive = pMoveContainer->addNonCollisionablePrimitive(Primitive); } // removes previous primitive MoveContainer->removePrimitive(Primitive); // and set new one as entity primitive Primitive = primitive; MoveContainer = pMoveContainer; if (Primitive == NULL) { MoveContainer = NULL; nlwarning("Can't create PACS primitive for entity %s", Id.toString().c_str()); return; } PrimIterator = CWorldPositionManager::_PrimitivedList.insert(CWorldPositionManager::_PrimitivedList.end(), this); // and leave... return; } Primitive = NULL; MoveContainer = NULL; const CGpmSheets::CSheet *sheet = CGpmSheets::lookup(CSheetId(Sheet())); float primRadius = 0.5f; float primHeight = 2.0f; if (sheet != NULL) { primRadius = sheet->Radius * sheet->Scale; primHeight = sheet->Height * sheet->Scale; } primitive = pMoveContainer->addNonCollisionablePrimitive(); if (primitive == NULL) { nlwarning("Can't create PACS primitive for entity %s", Id.toString().c_str()); return; } primitive->UserData = ((uint64)(Index.getIndex()) << 16); //nldebug("Set entity E%u to %"NL_I64"d", Index.getIndex(), primitive->UserData); primitive->setPrimitiveType( UMovePrimitive::_2DOrientedCylinder ); primitive->setReactionType( UMovePrimitive::Slide ); primitive->setTriggerType( UMovePrimitive::NotATrigger ); primitive->setCollisionMask( 0xffffffff ); primitive->setOcclusionMask( 0x00000000 ); primitive->setObstacle( false ); primitive->setAbsorbtion( 0 ); primitive->setHeight( primHeight ); primitive->setRadius( primRadius-0.1f ); // decrease primitive usable radius to lessen pacs load (avoid collision test) Primitive = primitive; MoveContainer = pMoveContainer; PrimIterator = CWorldPositionManager::_PrimitivedList.insert(CWorldPositionManager::_PrimitivedList.end(), this); } /****************************************************************\ removePrimitive \****************************************************************/ void CWorldEntity::removePrimitive() { if (PrimIterator != CWorldPositionManager::_PrimitivedList.end()) { CWorldPositionManager::_PrimitivedList.erase(PrimIterator); PrimIterator = CWorldPositionManager::_PrimitivedList.end(); } if (MoveContainer != NULL && Primitive != NULL) { MoveContainer->removePrimitive(Primitive); Primitive = NULL; MoveContainer = NULL; } else if (MoveContainer != NULL && Primitive == NULL || MoveContainer == NULL && Primitive != NULL) { nlwarning("Entity %s asked to remove PACS primitive, unable to continue (MoveContainer=%p, Primitive=%p)", Id.toString().c_str(), MoveContainer, Primitive); } } /****************************************************************\ updatePositionUsingMovePrimitive \****************************************************************/ void CWorldEntity::updatePositionUsingMovePrimitive(uint wi) { if (Primitive == NULL || Continent == INVALID_CONTINENT_INDEX || Continent == NO_CONTINENT_INDEX) return; NLPACS::UGlobalRetriever *retriever = CWorldPositionManager::getContinentContainer().getRetriever(Continent); if (retriever == NULL) return; NLPACS::UGlobalPosition gp; Primitive->getGlobalPosition(gp, wi); bool interior = retriever->isInterior(gp); bool local = localMotion(); float dummy; bool water = retriever->isWaterPosition(gp, dummy); NLMISC::CVectorD pos = Primitive->getFinalPosition(wi); setPosition((sint32)(pos.x*1000), (sint32)(pos.y*1000), (sint32)(pos.z*1000), local, interior, water); } /****************************************************************\ **************************************************************** CPlayerInfos **************************************************************** \****************************************************************/ // Static player allocator CBlockMemory CPlayerInfos::_PlayerAllocator; void CPlayerInfos::display(CLog *log) const { if (this == NULL) log->displayNL("CPlayerInfos::display called on NULL"); log->displayNL("--- Display PlayerInfos %s E%u ---", _PlayerId.toString().c_str(), (Entity != NULL ? Entity->Index.getIndex() : 0)); DUMP_VAR(FeId.get()); DUMP_VAR(LastVisionTick); DUMP_VAR(DelayVision); DUMP_VAR(ActivateSlot0); DUMP_VAR(DesactivateSlot0); DUMP_VAR(Slot0Active); DUMP_VAR(WhoICanSee); #ifdef RECORD_LAST_PLAYER_POSITIONS log->displayNL("Last positions recorded:"); uint i; for (i=0; idisplayNL("(%+9.3f,%+9.3f,%+9.3f) %d ticks", DistanceHistory[i].first.x, DistanceHistory[i].first.y, DistanceHistory[i].first.z, DistanceHistory[i].second); #endif log->displayNL("--- End of Display ---"); }