|
|
|
// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
|
|
|
|
// Copyright (C) 2010-2019 Winch Gate Property Limited
|
|
|
|
//
|
|
|
|
// This source file has been modified by the following contributors:
|
|
|
|
// Copyright (C) 2013 Jan BOON (Kaetemi) <jan.boon@kaetemi.be>
|
|
|
|
//
|
|
|
|
// 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 <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef CL_ENTITIES_H
|
|
|
|
#define CL_ENTITIES_H
|
|
|
|
|
|
|
|
|
|
|
|
/////////////
|
|
|
|
// INCLUDE //
|
|
|
|
/////////////
|
|
|
|
|
|
|
|
// Client
|
|
|
|
#include "ground_fx_manager.h"
|
|
|
|
#include "projectile_manager.h"
|
|
|
|
#include "user_entity.h"
|
|
|
|
// Some constants
|
|
|
|
#include "game_share/mission_desc.h"
|
|
|
|
#include "game_share/inventories.h"
|
|
|
|
// Misc.
|
|
|
|
#include "nel/misc/types_nl.h"
|
|
|
|
#include "nel/misc/stream.h"
|
|
|
|
#include "nel/misc/time_nl.h"
|
|
|
|
#include "nel/misc/vector.h"
|
|
|
|
#include "nel/misc/file.h"
|
|
|
|
#include "nel/misc/aabbox.h"
|
|
|
|
#include "nel/misc/cdb_leaf.h"
|
|
|
|
// 3D
|
|
|
|
#include "nel/3d/u_instance.h"
|
|
|
|
// Std.
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
|
|
///////////
|
|
|
|
// CLASS //
|
|
|
|
///////////
|
|
|
|
class CEntityCL;
|
|
|
|
class CUserEntity;
|
|
|
|
struct TNewEntityInfo;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Enum for entity selection
|
|
|
|
*/
|
|
|
|
class CEntityFilterFlag
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
enum TFlag
|
|
|
|
{
|
|
|
|
NotUser = 1 << 0,
|
|
|
|
Friend = 1 << 1,
|
|
|
|
Enemy = 1 << 2,
|
|
|
|
Alive = 1 << 3,
|
|
|
|
Dead = 1 << 4,
|
|
|
|
Player = 1 << 5, // ok if entity is a Player (not user)
|
|
|
|
NonPlayer = 1 << 6, // ok if entity is a non-player (ie a bot, whaterver Npc fauna etc...)
|
|
|
|
|
|
|
|
NoFilter = 0 // no filter
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class to make cache entities
|
|
|
|
*/
|
|
|
|
class CEntityReference
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CEntityReference (uint slot, CEntityCL *entity)
|
|
|
|
{
|
|
|
|
Slot = slot;
|
|
|
|
Entity = entity;
|
|
|
|
}
|
|
|
|
|
|
|
|
CEntityCL *Entity;
|
|
|
|
uint Slot;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class to make cache shape instances
|
|
|
|
*/
|
|
|
|
class CShapeInstanceReference
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CShapeInstanceReference (NL3D::UInstance instance, const string &text, const string &url, bool bbox_active=true, bool in_ig_zone = false)
|
|
|
|
{
|
|
|
|
Instance = instance;
|
|
|
|
ContextText = text;
|
|
|
|
ContextURL = url;
|
|
|
|
BboxActive = bbox_active;
|
|
|
|
Deleted = false;
|
|
|
|
LastDeleted = -1;
|
|
|
|
InIGZone = in_ig_zone;
|
|
|
|
Primitive = NULL;
|
|
|
|
PrimSize = CVector(1.f, 1.f, 1.f);
|
|
|
|
PrimHeight = 1.f;
|
|
|
|
PrimRelativePos = CVector(0.f, 0.f, 0.f);
|
|
|
|
}
|
|
|
|
|
|
|
|
NL3D::UInstance Instance;
|
|
|
|
NLPACS::UMovePrimitive *Primitive;
|
|
|
|
CVector PrimSize;
|
|
|
|
float PrimHeight;
|
|
|
|
CVector PrimRelativePos;
|
|
|
|
string ContextText;
|
|
|
|
string ContextURL;
|
|
|
|
bool BboxActive;
|
|
|
|
bool Deleted;
|
|
|
|
bool InIGZone;
|
|
|
|
sint32 LastDeleted;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class to manage entities and shapes instances.
|
|
|
|
* \author Guillaume PUZIN
|
|
|
|
* \author Nevrax France
|
|
|
|
* \date 2001
|
|
|
|
*/
|
|
|
|
class CEntityManager : public NLMISC::IStreamable
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
typedef std::vector<CEntityCL *> TEntities;
|
|
|
|
// Contain all entities.
|
|
|
|
TEntities _Entities;
|
|
|
|
// maximum number of entities.
|
|
|
|
uint32 _NbMaxEntity;
|
|
|
|
/// Nb Entities really allocated.
|
|
|
|
uint32 _EntitiesAllocated;
|
|
|
|
|
|
|
|
/// Entity caches
|
|
|
|
std::vector<CEntityReference> _ActiveEntities;
|
|
|
|
std::vector<CEntityReference> _VisibleEntities;
|
|
|
|
|
|
|
|
/// Shapes Instances caches
|
|
|
|
typedef std::map<uint16, std::vector<uint32>> TIGZoneShapes;
|
|
|
|
TIGZoneShapes _IgZoneShapes;
|
|
|
|
std::vector<CShapeInstanceReference> _ShapeInstances;
|
|
|
|
sint32 _LastRemovedInstance;
|
|
|
|
bool _InstancesRemoved;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
NLMISC::TGameCycle GC;
|
|
|
|
sint64 Value;
|
|
|
|
} TProperty;
|
|
|
|
typedef std::map<uint32, TProperty> TProperties;
|
|
|
|
typedef std::map<uint32, TProperties> TBackupedChanges;
|
|
|
|
TBackupedChanges _BackupedChanges;
|
|
|
|
|
|
|
|
// ground FXs
|
|
|
|
CGroundFXManager _GroundFXManager;
|
|
|
|
// Handle of each entity in the ground fx manager
|
|
|
|
std::vector<CGroundFXManager::TEntityHandle> _EntityGroundFXHandle;
|
|
|
|
|
|
|
|
// For selection. NB: the pointer is just a cache. Must not be accessed
|
|
|
|
CEntityCL *_LastEntityUnderPos;
|
|
|
|
|
|
|
|
NL3D::UInstance _LastInstanceUnderPos;
|
|
|
|
|
|
|
|
// DB node pointers used to update some entity flags
|
|
|
|
NLMISC::CRefPtr<NLMISC::CCDBNodeLeaf> _MissionTargetTitleDB[MAX_NUM_MISSIONS][MAX_NUM_MISSION_TARGETS];
|
|
|
|
NLMISC::CRefPtr<NLMISC::CCDBNodeLeaf> _GroupMemberUidDB[8]; // MaxNumPeopleInTeam in people_interaction.h
|
|
|
|
NLMISC::CRefPtr<NLMISC::CCDBNodeLeaf> _GroupMemberNameDB[8]; // MaxNumPeopleInTeam in people_interaction.h
|
|
|
|
NLMISC::CRefPtr<NLMISC::CCDBNodeLeaf> _BeastUidDB[MAX_INVENTORY_ANIMAL];
|
|
|
|
NLMISC::CRefPtr<NLMISC::CCDBNodeLeaf> _BeastStatusDB[MAX_INVENTORY_ANIMAL];
|
|
|
|
NLMISC::CRefPtr<NLMISC::CCDBNodeLeaf> _BeastTypeDB[MAX_INVENTORY_ANIMAL];
|
|
|
|
|
|
|
|
//////////////
|
|
|
|
//// DEBUG ///
|
|
|
|
uint _NbUser;
|
|
|
|
uint _NbPlayer;
|
|
|
|
uint _NbChar;
|
|
|
|
// For loging the property stage changes of the watched entity
|
|
|
|
struct CStageSetLog
|
|
|
|
{
|
|
|
|
bool Enabled;
|
|
|
|
CLFECOMMON::TCLEntityId LastEntityLoged;
|
|
|
|
// start time of recording for the last entity
|
|
|
|
sint32 StartGameCycle;
|
|
|
|
sint64 StartLocalTime;
|
|
|
|
CStageSet::TStageSet StageSet;
|
|
|
|
|
|
|
|
CStageSetLog()
|
|
|
|
{
|
|
|
|
Enabled= false;
|
|
|
|
LastEntityLoged= CLFECOMMON::INVALID_SLOT;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
CStageSetLog _LogStageChange;
|
|
|
|
|
|
|
|
private:
|
|
|
|
/// Reset Counters
|
|
|
|
void resetCounters() {_EntitiesAllocated = 0; _NbUser = 0; _NbPlayer = 0; _NbChar = 0;}
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
NLMISC_DECLARE_CLASS(CEntityManager);
|
|
|
|
|
|
|
|
/// Constructor.
|
|
|
|
CEntityManager();
|
|
|
|
/// Destructor.
|
|
|
|
~CEntityManager();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize some dynamic parameters.
|
|
|
|
* \param uint nbMaxEntity : maximum number of entities allocated.
|
|
|
|
*/
|
|
|
|
void initialize(uint nbMaxEntity);
|
|
|
|
|
|
|
|
// get ground fx manager
|
|
|
|
CGroundFXManager &getGroundFXManager() { return _GroundFXManager; }
|
|
|
|
|
|
|
|
/// Free the class and all the components.
|
|
|
|
void release();
|
|
|
|
|
|
|
|
/// Release + initialize
|
|
|
|
void reinit();
|
|
|
|
|
|
|
|
|
|
|
|
CShapeInstanceReference createInstance(const string& shape, const CVector &pos, const string &text, const string &url, bool haveCollisions, uint16 inIgZone, sint32 &idx);
|
|
|
|
bool deleteInstance(uint32 idx);
|
|
|
|
bool removeInstances();
|
|
|
|
void removeInstancesInIgZone(uint16 igZone);
|
|
|
|
CVector getInstancePos(uint32 idx);
|
|
|
|
bool setInstancePos(uint32 idx, CVector pos);
|
|
|
|
CVector getInstanceRot(uint32 idx);
|
|
|
|
CVector getInstanceScale(uint32 idx);
|
|
|
|
CVector getInstanceColPos(uint32 idx);
|
|
|
|
CVector getInstanceColScale(uint32 idx);
|
|
|
|
double getInstanceColOrient(uint32 idx);
|
|
|
|
CVector getInstanceBBoxMin(uint32 idx);
|
|
|
|
CVector getInstanceBBoxMax(uint32 idx);
|
|
|
|
bool setInstanceRot(uint32 idx, CVector pos);
|
|
|
|
bool setupInstance(uint32 idx, const std::vector<std::string> &keys, const std::vector<std::string> &values);
|
|
|
|
CShapeInstanceReference getShapeInstanceUnderPos(float x, float y, sint32 &idx);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create an entity according to the slot and the form.
|
|
|
|
* \param uint slot : slot for the entity.
|
|
|
|
* \param uint32 form : form to create the entity.
|
|
|
|
* \param TClientDataSetIndex : persitent id while the entity is connected.
|
|
|
|
* \return CEntityCL * : pointer on the new entity.
|
|
|
|
*/
|
|
|
|
CEntityCL *create(uint slot, uint32 form, const TNewEntityInfo& newEntityInfo);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete an entity.
|
|
|
|
* \return bool : 'true' if the entity has been correctly removed.
|
|
|
|
*/
|
|
|
|
bool remove(uint slot, bool warning);
|
|
|
|
/// Remove the collision for all entities.
|
|
|
|
void removeCollision();
|
|
|
|
/// Re-load animations (remove and load).
|
|
|
|
void reloadAnims();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a pointer on an entity according to the asked slot.
|
|
|
|
* \param uint slot : the asked slot.
|
|
|
|
*/
|
|
|
|
CEntityCL *entity(uint slot);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return if there is an entity near a door.
|
|
|
|
* \param float openingDist : near is when you are under the 'openingDist'.
|
|
|
|
* \param const CVector& posDoor1 : first door position.
|
|
|
|
* \param const CVector& posDoor2 : second door position.
|
|
|
|
* \return bool ; 'true' if any entity is near one of the door.
|
|
|
|
*/
|
|
|
|
bool entitiesNearDoors(float openingDist, const NLMISC::CVector& posDoor1, const NLMISC::CVector& posDoor2);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the entity under the (2d) position. Return NULL if no entity under this position.
|
|
|
|
* NB: the UserEntity can NOT be returned. But if the code has find it, isPlayerUnderCursor is set to true
|
|
|
|
* NB: somewhat slow. should be called ONLY ONCE per frame.
|
|
|
|
* NB: unselectable entities are not returned
|
|
|
|
*/
|
|
|
|
CEntityCL *getEntityUnderPos(float x, float y, float distSelection, bool &isPlayerUnderCursor);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the entity (not user) in the camera. Return NULL if not entity under this position.
|
|
|
|
* \param flags a ORed of CEntityFilterFlag::TFlag
|
|
|
|
* \param distSelection don't go beyond
|
|
|
|
* \param precEntity is used for Cyclic management. If found in the select list, get the next one, else get the first.
|
|
|
|
*/
|
|
|
|
CEntityCL *getEntityInCamera(uint flags, float distSelection, CLFECOMMON::TCLEntityId precEntity);
|
|
|
|
|
|
|
|
/// Get an entity by name. Returns NULL if the entity is not found.
|
|
|
|
CEntityCL *getEntityByName (uint32 stringId) const;
|
|
|
|
|
|
|
|
/** Get an entity by name. Returns NULL if the entity is not found.
|
|
|
|
* \param name of the entity to find
|
|
|
|
* \param caseSensitive type of test to perform
|
|
|
|
* \param complete : if true, the name must match the full name of the entity.
|
|
|
|
*/
|
|
|
|
CEntityCL *getEntityByName (const std::string &name, bool caseSensitive, bool complete) const;
|
|
|
|
/**
|
|
|
|
* Case insensitive match against entity name. All listed keywords must match.
|
|
|
|
* \param keywords to match
|
|
|
|
* \param onlySelectable : if true, match only entity that can be selected
|
|
|
|
*/
|
|
|
|
CEntityCL *getEntityByKeywords (const std::vector<std::string> &keywords, bool onlySelectable) const;
|
|
|
|
CEntityCL *getEntityBySheetName (const std::string &sheet) const;
|
|
|
|
/// Get an entity by dataset index. Returns NULL if the entity is not found.
|
|
|
|
CEntityCL *getEntityByCompressedIndex(TDataSetIndex compressedIndex) const;
|
|
|
|
|
|
|
|
/// Return number of entities allocated.
|
|
|
|
uint nbEntitiesAllocated() const {return _EntitiesAllocated;}
|
|
|
|
/// Return the number of user allocated.
|
|
|
|
uint nbUser() const {return _NbUser;}
|
|
|
|
/// Return the number of player allocated.
|
|
|
|
uint nbPlayer() const {return _NbPlayer;}
|
|
|
|
/// Return the number of character allocated.
|
|
|
|
uint nbChar() const {return _NbChar;}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// Continent has changed.
|
|
|
|
void changeContinent();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update the entity (position\animation).
|
|
|
|
*/
|
|
|
|
void updatePreCamera();
|
|
|
|
void updatePostCamera(uint clippedUpdateTime, const std::vector<NLMISC::CPlane> &clippingPlanes, const NLMISC::CVector &camPos);
|
|
|
|
void updatePostRender();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Method to update the visual property 'prop' for the entity in 'slot'.
|
|
|
|
* \param uint slot : slot of the entity to update.
|
|
|
|
* \param uint prop : the property to udapte.
|
|
|
|
*/
|
|
|
|
void updateVisualProperty(const NLMISC::TGameCycle &gameCycle, const uint &slot, const uint &prop, const NLMISC::TGameCycle &predictedInterval = 0);
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void applyBackupedProperties(uint slot);
|
|
|
|
|
|
|
|
/// Return the reference for all entities in the vision. An entity object, for a slot, is not allocated if not visible.
|
|
|
|
TEntities &entities() {return _Entities;}
|
|
|
|
|
|
|
|
/// Manage PACS Triggers.
|
|
|
|
void managePACSTriggers();
|
|
|
|
///
|
|
|
|
void removeColUserOther();
|
|
|
|
void restoreColUserOther();
|
|
|
|
|
|
|
|
// For Sound Reset/Reload
|
|
|
|
void resetAllSoundAnimId();
|
|
|
|
|
|
|
|
///////////
|
|
|
|
// DEBUG //
|
|
|
|
///////////
|
|
|
|
/// Write a file with the position of all entities.
|
|
|
|
void writeEntities();
|
|
|
|
/// Dump entities state.
|
|
|
|
void dump(class NLMISC::IStream &f);
|
|
|
|
/// Dump entities state (XML Format).
|
|
|
|
void dumpXML(class NLMISC::IStream &f);
|
|
|
|
/// Log Watched Entity Stages Change
|
|
|
|
void startLogStageChange(sint32 currentGameCycle, sint64 currentLocalTime);
|
|
|
|
void logStageChange(sint64 currentLocalTime);
|
|
|
|
void stopLogStageChange();
|
|
|
|
bool isLogingStageChange() const;
|
|
|
|
sint32 getLogStageChangeStartCycle() const;
|
|
|
|
sint64 getLogStageChangeStartLocalTime() const;
|
|
|
|
|
|
|
|
/// Serialize entities.
|
|
|
|
virtual void serial(NLMISC::IStream &f);
|
|
|
|
|
|
|
|
// remove all attached fx of all entities (so that they can be reloaded)
|
|
|
|
virtual void removeAllAttachedFX();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If the slot leads to a CCharacterCL compatible with mission icon, and
|
|
|
|
* who isFriend(), perform releaseInSceneInterfaces() and buildInSceneInterface().
|
|
|
|
* Otherwise, do nothing.
|
|
|
|
*/
|
|
|
|
void refreshInsceneInterfaceOfFriendNPC(uint slot);
|
|
|
|
|
|
|
|
inline NLMISC::CCDBNodeLeaf *getMissionTargetTitleDB(int mission, int target) { return _MissionTargetTitleDB[mission][target]; }
|
|
|
|
inline NLMISC::CCDBNodeLeaf *getGroupMemberUidDB(int member) { return _GroupMemberUidDB[member]; }
|
|
|
|
inline NLMISC::CCDBNodeLeaf *getGroupMemberNameDB(int member) { return _GroupMemberNameDB[member]; }
|
|
|
|
inline NLMISC::CCDBNodeLeaf *getBeastUidDB(int beast) { return _BeastUidDB[beast]; }
|
|
|
|
inline NLMISC::CCDBNodeLeaf *getBeastStatusDB(int beast) { return _BeastStatusDB[beast]; }
|
|
|
|
inline NLMISC::CCDBNodeLeaf *getBeastTypeDB(int beast) { return _BeastTypeDB[beast]; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
// NB: don't return unselectable entities
|
|
|
|
void getEntityListForSelection(std::vector<CEntityCL*> &entities, uint flags);
|
|
|
|
|
|
|
|
void updateEntitiesIsInTeam();
|
|
|
|
|
|
|
|
/// Log entities Property Stages Change
|
|
|
|
void logPropertyChange(CLFECOMMON::TCLEntityId who, const CStage &oldStage, const CStage &newStage,
|
|
|
|
const NLMISC::CVectorD &precOldPos, const NLMISC::CVectorD &precNewPos, sint32 relGameCycle, sint64 relLocalTime);
|
|
|
|
};
|
|
|
|
|
|
|
|
extern CEntityManager EntitiesMngr;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#endif // CL_ENTITIES_H
|
|
|
|
|
|
|
|
/* End of entities.h */
|