477 lines
13 KiB
C
477 lines
13 KiB
C
|
// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
|
||
|
// 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 <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
|
||
|
|
||
|
#ifndef NL_WORLD_ENTITY_H
|
||
|
#define NL_WORLD_ENTITY_H
|
||
|
|
||
|
#include "nel/misc/types_nl.h"
|
||
|
#include "nel/misc/block_memory.h"
|
||
|
#include "nel/misc/debug.h"
|
||
|
#include "nel/misc/time_nl.h"
|
||
|
|
||
|
#include "nel/pacs/u_move_container.h"
|
||
|
#include "nel/pacs/u_move_primitive.h"
|
||
|
|
||
|
#include "gpm_utilities.h"
|
||
|
#include "gpm_defs.h"
|
||
|
|
||
|
#include "game_share/mirror_prop_value.h"
|
||
|
#include "game_share/ryzom_mirror_properties.h"
|
||
|
#include "server_share/msg_gpm_service.h"
|
||
|
|
||
|
#include <deque>
|
||
|
|
||
|
class CCell;
|
||
|
class CPlayerInfos;
|
||
|
class CWorldEntity;
|
||
|
|
||
|
typedef std::list<CWorldEntity*> TWorldEntityList;
|
||
|
|
||
|
struct CVisionEntry
|
||
|
{
|
||
|
|
||
|
CVisionEntry() { }
|
||
|
CVisionEntry(CWorldEntity* e, uint32 m, uint32 d) : Entity(e), Mask(m), Distance(d) { }
|
||
|
|
||
|
CWorldEntity* Entity;
|
||
|
uint32 Mask;
|
||
|
uint32 Distance;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* World entity contained all properties positions in world for an entity
|
||
|
* \author Alain Saffray
|
||
|
* \author Nevrax France
|
||
|
* \date 2002
|
||
|
*/
|
||
|
class CWorldEntity
|
||
|
{
|
||
|
friend class NLMISC::CBlockMemory<CWorldEntity>;
|
||
|
|
||
|
public:
|
||
|
typedef CSimpleSmartPointer<CWorldEntity> CWorldEntitySmartPointer;
|
||
|
|
||
|
enum TEntityType
|
||
|
{
|
||
|
Player = 0,
|
||
|
Object,
|
||
|
Trigger,
|
||
|
AI,
|
||
|
Unknown
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
enum TVisionState // Enum of vision state for this entity
|
||
|
{
|
||
|
Ready = 0, // ready to use
|
||
|
Checked, // in check
|
||
|
Seen, // entity is seen by another
|
||
|
};
|
||
|
*/
|
||
|
|
||
|
public:
|
||
|
|
||
|
NLMISC::CEntityId Id; // Id of entity
|
||
|
TDataSetRow Index;
|
||
|
|
||
|
CMirrorPropValue1DS<TYPE_POSX> X; // Coordinate X in world in unit
|
||
|
CMirrorPropValue1DS<TYPE_POSY> Y; // Coordinate Y in world in unit
|
||
|
CMirrorPropValue1DS<TYPE_POSZ> Z; // Coordinate Z in world in unit
|
||
|
CMirrorPropValue1DS<TYPE_POSX> LocalX; // Local Coordinate X in world in unit
|
||
|
CMirrorPropValue1DS<TYPE_POSY> LocalY; // Local Coordinate X in world in unit
|
||
|
CMirrorPropValue1DS<TYPE_POSZ> LocalZ; // Local Coordinate X in world in unit
|
||
|
CMirrorPropValue1DS<TYPE_ORIENTATION> Theta; // Heading in world
|
||
|
CMirrorPropValue1DS<TYPE_SHEET> Sheet; // Id sheet of entity
|
||
|
CMirrorPropValue1DS<NLMISC::TGameCycle> Tick; // GameCycle of other properties
|
||
|
CMirrorPropValue1DS<TYPE_CELL> Cell; // Current XY Cell where entity is
|
||
|
CMirrorPropValue1DS<TYPE_VISION_COUNTER> VisionCounter; // Number of times this entity is seen by players
|
||
|
uint32 PlayersSeeingMe;
|
||
|
CWorldEntity* ClosestPlayer;
|
||
|
|
||
|
CMirrorPropValue1DS<TYPE_WHO_SEES_ME> WhoSeesMe;
|
||
|
|
||
|
uint32 PatatEntryIndex; // The patat entry for the _PatatSubscribeManager
|
||
|
|
||
|
CCell *CellPtr; // pointer on cell where entity is
|
||
|
|
||
|
TWorldEntityList::iterator ListIterator; // Iterator on entity in world entity list
|
||
|
TWorldEntityList::iterator PrimIterator; // Iterator on entity in prmitived entity list
|
||
|
|
||
|
uint8 Continent; // index on the continent on which the player is located
|
||
|
bool PosInitialised; // Pos was initialised by mirror
|
||
|
bool UsePrimitive; // entity uses a primitive normally
|
||
|
bool ForceUsePrimitive; // forces entity to use a primitive temporarily (mount, etc.)
|
||
|
NLPACS::UMovePrimitive *Primitive; // Primitive for collision systeme (PACS)
|
||
|
NLPACS::UMoveContainer *MoveContainer; // MoveContainer entity is in
|
||
|
uint32 TickLock;
|
||
|
|
||
|
CWorldEntitySmartPointer Previous;
|
||
|
CWorldEntitySmartPointer Next;
|
||
|
|
||
|
CPlayerInfos *PlayerInfos; // The player infos associated to this entity (if there is some)
|
||
|
|
||
|
CWorldEntitySmartPointer Parent; // Which is entity we are in/on (ex: mount, ferry)
|
||
|
std::vector<CWorldEntitySmartPointer> Children; // Which are the child we contain
|
||
|
CWorldEntitySmartPointer Control; // Which entity is controlling us (ex: rider, pilot...)
|
||
|
|
||
|
bool ForceDontUsePrimitive; // forces entity not to use a primitive temporarily (mount, etc.)
|
||
|
bool CheckMotion;
|
||
|
bool HasVision; // Entity has vision
|
||
|
|
||
|
std::vector<CEntitySheetId> Content;
|
||
|
|
||
|
bool TempVisionState; // temporary flag for vision delta, telling if the entity is now visible
|
||
|
bool TempControlInVision; // temporary flag for vision delta, telling if the controller entity (if any) is in vision
|
||
|
bool TempParentInVision; // temporary flag for vision delta, telling if the parent (controlled) entity (if any) is in vision
|
||
|
|
||
|
sint32 RefCounter; // Number of references on this entity -- used by smart pointer
|
||
|
|
||
|
public:
|
||
|
/**
|
||
|
* destructor
|
||
|
*/
|
||
|
~CWorldEntity();
|
||
|
|
||
|
/**
|
||
|
* Init
|
||
|
* \param id is entity's CEntityId
|
||
|
*/
|
||
|
void init( const NLMISC::CEntityId& id, const TDataSetRow &index );
|
||
|
|
||
|
/**
|
||
|
* Display debug
|
||
|
*/
|
||
|
void display(NLMISC::CLog *log = NLMISC::InfoLog) const;
|
||
|
|
||
|
/**
|
||
|
* create primitive for fiche type entity
|
||
|
* \param ficheId is sheet type Id of entity
|
||
|
* \param pMoveContainer adress of the move container
|
||
|
* \param worldImage numvber of the world image in which the primitive is to be inserted
|
||
|
* \return pointer on PACS primitve
|
||
|
*/
|
||
|
void createPrimitive(NLPACS::UMoveContainer *pMoveContainer, uint8 worldImage);
|
||
|
|
||
|
/**
|
||
|
* Removes primitive allocated previously
|
||
|
*/
|
||
|
void removePrimitive();
|
||
|
|
||
|
|
||
|
/**
|
||
|
* removes entity from the cell it is in
|
||
|
*/
|
||
|
//void removeFromCellAsEntity();
|
||
|
|
||
|
/**
|
||
|
* removes object from the cell it is in
|
||
|
*/
|
||
|
//void removeFromCellAsObject();
|
||
|
|
||
|
|
||
|
/// Test if entity is linked in a cell
|
||
|
bool isLinked() const { return CellPtr != NULL; }
|
||
|
|
||
|
/// Get (const) CCell point in which entity is
|
||
|
const CCell* getCell() const { return CellPtr; }
|
||
|
|
||
|
|
||
|
/// Tests if entity uses a pacs primitive
|
||
|
bool hasPrimitive() const { return Primitive != NULL && !ForceDontUsePrimitive; }
|
||
|
|
||
|
/// local motion
|
||
|
bool localMotion() const { return Parent != NULL; }
|
||
|
|
||
|
/// has control ?
|
||
|
bool hasControl() const { return Parent != NULL && Parent->Control == this; }
|
||
|
|
||
|
/// is controlled ?
|
||
|
bool isControlled() const { return Control != NULL; }
|
||
|
|
||
|
/// has children
|
||
|
bool hasChildren() const { return !Children.empty(); }
|
||
|
|
||
|
/// remove from children
|
||
|
void removeFromChildren(CWorldEntity *entity)
|
||
|
{
|
||
|
std::vector<CWorldEntitySmartPointer>::iterator it;
|
||
|
for (it=Children.begin(); it!=Children.end(); ++it)
|
||
|
if ((CWorldEntity*)(*it) == entity)
|
||
|
it = Children.erase(it);
|
||
|
|
||
|
entity->Parent = NULL;
|
||
|
Control = NULL;
|
||
|
}
|
||
|
|
||
|
/// get controlled
|
||
|
CWorldEntity *getControlled()
|
||
|
{
|
||
|
if (!hasControl())
|
||
|
return NULL;
|
||
|
|
||
|
CWorldEntity *parent = Parent;
|
||
|
while (parent->hasControl())
|
||
|
parent = parent->Parent;
|
||
|
|
||
|
return parent;
|
||
|
}
|
||
|
|
||
|
/// update local or global position
|
||
|
void updatePosition(sint32 x, sint32 y, sint32 z, float theta, NLMISC::TGameCycle cycle, bool interior, bool water)
|
||
|
{
|
||
|
if (localMotion())
|
||
|
{
|
||
|
LocalX = x;
|
||
|
LocalY = y;
|
||
|
LocalZ = z;
|
||
|
|
||
|
setPosition(x + Parent->X(), y + Parent->Y(), z + Parent->Z(), true, interior, water);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
setPosition(x, y, z, false, interior, water);
|
||
|
}
|
||
|
|
||
|
Tick = cycle;
|
||
|
Theta = theta;
|
||
|
|
||
|
// force position as valid
|
||
|
PosInitialised = true;
|
||
|
}
|
||
|
|
||
|
/// update global position for local motion
|
||
|
void updatePosition(bool interior, bool water)
|
||
|
{
|
||
|
if (localMotion())
|
||
|
{
|
||
|
setPosition(LocalX() + Parent->X(), LocalY() + Parent->Y(), LocalZ() + Parent->Z(), true, interior, water);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Set position
|
||
|
void setPosition(sint32 x, sint32 y, sint32 z, bool local, bool interior, bool water)
|
||
|
{
|
||
|
X = x;
|
||
|
Y = y;
|
||
|
Z = (z&(~7)) + (local ? 1 : 0) + (interior ? 2 : 0) + (water ? 4 : 0);
|
||
|
}
|
||
|
|
||
|
/// update position using move primitive
|
||
|
void updatePositionUsingMovePrimitive(uint wi);
|
||
|
|
||
|
|
||
|
|
||
|
/// get Type of the entity
|
||
|
TEntityType getType() const
|
||
|
{
|
||
|
return _Type;
|
||
|
}
|
||
|
|
||
|
|
||
|
private:
|
||
|
|
||
|
/// Is in interior
|
||
|
bool interior() const
|
||
|
{
|
||
|
return (Z()&2) != 0;
|
||
|
}
|
||
|
|
||
|
/// Type of the entity
|
||
|
TEntityType _Type;
|
||
|
|
||
|
public:
|
||
|
|
||
|
/// Creates a new entity (new equivalent). This must be initialised later using init();
|
||
|
static CWorldEntity* create();
|
||
|
|
||
|
/// Removes an entity (delete equivalent).
|
||
|
static void remove(CWorldEntity *entity);
|
||
|
|
||
|
protected:
|
||
|
/**
|
||
|
* Default constructor, used because of CBlockMemory
|
||
|
*/
|
||
|
CWorldEntity() {}
|
||
|
|
||
|
private:
|
||
|
|
||
|
/// Static cell allocator
|
||
|
static NLMISC::CBlockMemory<CWorldEntity> _EntityAllocator;
|
||
|
};
|
||
|
|
||
|
//
|
||
|
typedef CWorldEntity::CWorldEntitySmartPointer CWorldEntityPtr;
|
||
|
|
||
|
/// A list of CWorldEntity, referred by smart pointers. First template param is the pointed type, second param is the pointer storage type (here smart pointer)
|
||
|
typedef CObjectList<CWorldEntity, CWorldEntityPtr> TEntityList;
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Player Infos : contains all information specific to players (like vision and original front end)
|
||
|
* \author David Fleury
|
||
|
* \author Nevrax France
|
||
|
* \date 2002
|
||
|
*/
|
||
|
class CPlayerInfos
|
||
|
{
|
||
|
friend class NLMISC::CBlockMemory<CPlayerInfos>;
|
||
|
|
||
|
public:
|
||
|
/// init
|
||
|
void init(const NLMISC::CEntityId &id, NLNET::TServiceId feId, CWorldEntity *entity)
|
||
|
{
|
||
|
WhoICanSee = 0xffffffff;
|
||
|
|
||
|
Next = NULL;
|
||
|
Previous = NULL;
|
||
|
ActivateSlot0 = false;
|
||
|
DesactivateSlot0 = false;
|
||
|
Slot0Active = false;
|
||
|
|
||
|
uint i;
|
||
|
for (i = MAX_SEEN_ENTITIES-1 ; i > 0; --i)
|
||
|
FreeSlots.push_back( i );
|
||
|
|
||
|
for (i = 0 ; i < MAX_SEEN_ENTITIES ; ++i)
|
||
|
Slots[i] = NULL;
|
||
|
|
||
|
LastVisionTick = 0;
|
||
|
|
||
|
_PlayerId = id;
|
||
|
FeId = feId;
|
||
|
Entity = entity;
|
||
|
|
||
|
CheckSpeed = true;
|
||
|
EnableVisionProcessing = true;
|
||
|
}
|
||
|
|
||
|
/// get playerId
|
||
|
inline const NLMISC::CEntityId &getPlayerId() const { return _PlayerId; }
|
||
|
|
||
|
/**
|
||
|
* Display debug
|
||
|
*/
|
||
|
void display(NLMISC::CLog *log = NLMISC::InfoLog) const;
|
||
|
|
||
|
private:
|
||
|
/// the player Id
|
||
|
NLMISC::CEntityId _PlayerId;
|
||
|
|
||
|
/// default constructor
|
||
|
CPlayerInfos() { }
|
||
|
|
||
|
public:
|
||
|
/// original front end Id
|
||
|
NLNET::TServiceId FeId;
|
||
|
|
||
|
/// front end datas
|
||
|
TMapFrontEndData::iterator ItFrontEnd;
|
||
|
|
||
|
/// iterator in the update player list
|
||
|
TPlayerList::iterator ItUpdatePlayer;
|
||
|
|
||
|
/// tick at last vision update
|
||
|
NLMISC::TGameCycle LastVisionTick;
|
||
|
|
||
|
/// Delay vision till cycle
|
||
|
NLMISC::TGameCycle DelayVision;
|
||
|
|
||
|
///
|
||
|
typedef CUnsafeConstantSizeStack<uint16, MAX_SEEN_ENTITIES+1> TSlotStack;
|
||
|
|
||
|
/// list of free slots for vision
|
||
|
TSlotStack FreeSlots;
|
||
|
|
||
|
/// The world entity for this player
|
||
|
CWorldEntityPtr Entity;
|
||
|
|
||
|
/// Previous player in list
|
||
|
CPlayerInfos* Previous;
|
||
|
/// Next player in list
|
||
|
CPlayerInfos* Next;
|
||
|
|
||
|
/// slots for this player
|
||
|
CWorldEntityPtr Slots[MAX_SEEN_ENTITIES];
|
||
|
|
||
|
bool ActivateSlot0;
|
||
|
bool DesactivateSlot0;
|
||
|
bool Slot0Active;
|
||
|
|
||
|
bool EnableVisionProcessing;
|
||
|
|
||
|
bool CheckSpeed;
|
||
|
|
||
|
/// Who I can see flag field
|
||
|
uint32 WhoICanSee;
|
||
|
|
||
|
#ifdef RECORD_LAST_PLAYER_POSITIONS
|
||
|
/// Distance history
|
||
|
std::deque< std::pair<NLMISC::CVectorD, uint> > DistanceHistory;
|
||
|
#endif
|
||
|
|
||
|
float meanSpeed() const
|
||
|
{
|
||
|
#ifdef RECORD_LAST_PLAYER_POSITIONS
|
||
|
float dist = 0.0f;
|
||
|
uint i;
|
||
|
for (i=0; i+1<DistanceHistory.size(); ++i)
|
||
|
{
|
||
|
double dx = DistanceHistory[i+1].first.x-DistanceHistory[i].first.x,
|
||
|
dy = DistanceHistory[i+1].first.y-DistanceHistory[i].first.y;
|
||
|
|
||
|
dist += (float)sqrt(dx*dx + dy*dy);
|
||
|
}
|
||
|
|
||
|
return DistanceHistory.size() > 1 ? (dist/(DistanceHistory.size()-1)) : 0.0f;
|
||
|
#else
|
||
|
return 0.0f;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
|
||
|
struct CPlayerPos
|
||
|
{
|
||
|
NLMISC::TGameCycle AtTick;
|
||
|
NLPACS::UGlobalPosition GPos;
|
||
|
NLMISC::CVectorD Motion;
|
||
|
float Theta;
|
||
|
};
|
||
|
|
||
|
std::deque<CPlayerPos> PosHistory;
|
||
|
|
||
|
public:
|
||
|
/// Creates a new entity (new equivalent). This must be initialised later using init();
|
||
|
static CPlayerInfos *create() { return _PlayerAllocator.allocate(); }
|
||
|
|
||
|
/// Removes an entity (delete equivalent).
|
||
|
static void remove(CPlayerInfos *player) { _PlayerAllocator.free(player); }
|
||
|
|
||
|
private:
|
||
|
|
||
|
/// Static cell allocator
|
||
|
static NLMISC::CBlockMemory<CPlayerInfos> _PlayerAllocator;
|
||
|
};
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#endif // NL_WORLD_ENTITY_H
|
||
|
|
||
|
/* End of world_entity.h */
|