// 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 . #ifndef NL_ENTITY_CONTAINER_H #define NL_ENTITY_CONTAINER_H #include "nel/misc/types_nl.h" #include "game_share/entity_types.h" #include "game_share/mirror_prop_value.h" #include "game_share/tick_event_handler.h" #include "game_share/ryzom_mirror_properties.h" #include "client_id_lookup.h" #include #include // made to allow template method to support TDataSetRow without specialization. template inline uint32 asUInt32 (const T& obj) { return (uint32)obj; } template <> inline uint32 asUInt32 (const TDataSetRow& obj) { return obj.getIndex(); } extern bool verboseVision; extern NLMISC::CLog *TmpDebugLogger; #ifdef NL_RELEASE #define LOG_VISION ; #else //#define LOG_VISION if (!verboseVision) {} else nldebug #define LOG_VISION TmpDebugLogger->displayNL #endif #define STORE_MIRROR_VP_IN_CLASS //#undef STORE_MIRROR_VP_IN_CLASS extern TPropertyIndex DSPropertyTickPos; extern TPropertyIndex DSPropertyLocalX; extern TPropertyIndex DSPropertyLocalY; extern TPropertyIndex DSPropertyLocalZ; //extern TPropertyIndex DSPropertyStunned; extern TPropertyIndex DSFirstPropertyAvailableImpulseBitSize; typedef uint64 TPropertiesValue; extern const TDataSetIndex LAST_VISION_CHANGE; /// Vision associations: entityindex, slot and 'replace' bool typedef std::map< TEntityIndex, std::pair > TMapOfVisionAssociations; /// Information for vision disassociations: slots typedef std::set< CLFECOMMON::TCLEntityId > TSetOfRemovedEntities; /// Vision associations received before an entity is in mirror (TEMP workaround) //typedef std::multimap< NLMISC::CEntityId, std::pair< TEntityIndex, std::pair > > TMapOfEarlyVisionAssociations; /** * CEntity */ struct CEntity { /// Cache for property initialized bool PropInitialized[CLFECOMMON::NB_VISUAL_PROPERTIES]; /// Property X direct access + previous value CMirrorPropValueROCF X; /// Property Y direct access + previous value CMirrorPropValueROCF Y; #ifdef STORE_MIRROR_VP_IN_CLASS CMirrorPropValueRO VP_SHEET; CMirrorPropValueRO VP_BEHAVIOUR; CMirrorPropValueRO VP_NAME_STRING_ID; CMirrorPropValueRO VP_TARGET_ID; CMirrorPropValueRO VP_CONTEXTUAL; CMirrorPropValueRO VP_MODE; CMirrorPropValueRO VP_BARS; CMirrorPropValueRO VP_VPA; CMirrorPropValueRO VP_VPB; CMirrorPropValueRO VP_VPC; CMirrorPropValueRO VP_ENTITY_MOUNTED_ID; CMirrorPropValueRO VP_RIDER_ENTITY_ID; CMirrorPropValueRO VP_TARGET_LIST; CMirrorPropValueRO VP_VISUAL_FX; CMirrorPropValueRO VP_GUILD_SYMBOL; CMirrorPropValueRO VP_GUILD_NAME_ID; CMirrorPropValueRO VP_EVENT_FACTION_ID; CMirrorPropValueRO VP_PVP_MODE; CMirrorPropValueRO VP_PVP_CLAN; CMirrorPropValueRO VP_OWNER_PEOPLE; CMirrorPropValueRO VP_OUTPOST_INFOS; #endif /// Manhattan distance accumulated when moving uint32 Mileage; /// Multi target has changed bool MultiTargetChanged; /// Game cycle of the latest change of pos on this entity CMirrorPropValueRO TickPosition; /// Index of the next updated entity (for the vision) in the chain TEntityIndex NextUpdatedEntityVision; /// New entities in the vision TMapOfVisionAssociations VisionIn; /// Old entities in the vision TSetOfRemovedEntities VisionOut; static CMirroredDataSet *DataSet; // Not used because there is only one gamecycle stored per entity //NLMISC::TGameCycle getGameCycleForProperty( TPropIndex ) { return TickPosition; } /// Initialize struct void initFields( const TDataSetRow& entityIndex, NLMISC::CEntityId entityId ) { if ( ! X.isReadable() ) X.init( *DataSet, entityIndex, DSPropertyPOSX ); if ( ! Y.isReadable() ) Y.init( *DataSet, entityIndex, DSPropertyPOSY ); TickPosition.init( *DataSet, entityIndex, DSPropertyTickPos ); Mileage = 0; NextUpdatedEntityVision.initToLastChanged(); // = LAST_VISION_CHANGE; #ifdef STORE_MIRROR_VP_IN_CLASS VP_SHEET.init( *DataSet, entityIndex, DSPropertySHEET ); VP_BEHAVIOUR.init( *DataSet, entityIndex, DSPropertyBEHAVIOUR ); VP_NAME_STRING_ID.init( *DataSet, entityIndex, DSPropertyNAME_STRING_ID ); VP_TARGET_ID.init( *DataSet, entityIndex, DSPropertyTARGET_ID ); VP_CONTEXTUAL.init( *DataSet, entityIndex, DSPropertyCONTEXTUAL ); VP_MODE.init( *DataSet, entityIndex, DSPropertyMODE ); VP_BARS.init( *DataSet, entityIndex, DSPropertyBARS ); VP_VPA.init( *DataSet, entityIndex, DSPropertyVPA ); VP_VPB.init( *DataSet, entityIndex, DSPropertyVPB ); VP_VPC.init( *DataSet, entityIndex, DSPropertyVPC ); VP_ENTITY_MOUNTED_ID.init( *DataSet, entityIndex, DSPropertyENTITY_MOUNTED_ID ); VP_RIDER_ENTITY_ID.init( *DataSet, entityIndex, DSPropertyRIDER_ENTITY_ID ); VP_TARGET_LIST.init( *DataSet, entityIndex, DSPropertyTARGET_LIST ); VP_VISUAL_FX.init( *DataSet, entityIndex, DSPropertyVISUAL_FX ); VP_GUILD_SYMBOL.init( *DataSet, entityIndex, DSPropertyGUILD_SYMBOL ); VP_GUILD_NAME_ID.init( *DataSet, entityIndex, DSPropertyGUILD_NAME_ID ); VP_EVENT_FACTION_ID.init( *DataSet, entityIndex, DSPropertyEVENT_FACTION_ID ); VP_PVP_MODE.init( *DataSet, entityIndex, DSPropertyPVP_MODE ); VP_PVP_CLAN.init( *DataSet, entityIndex, DSPropertyPVP_CLAN ); VP_OWNER_PEOPLE.init( *DataSet, entityIndex, DSPropertyOWNER_PEOPLE ); VP_OUTPOST_INFOS.init( *DataSet, entityIndex, DSPropertyOUTPOST_INFOS ); #endif invalidateProperties(); } /// Reinitialize Initialized state of properties void invalidateProperties() { for ( sint p=0; p!=CLFECOMMON::MAX_PROPERTIES_PER_ENTITY; ++p ) { PropInitialized[p] = false; } } /// Return the current cached state without checking the value bool propIsInitializedState( CLFECOMMON::TPropIndex propIndex ) const { return PropInitialized[propIndex]; } /** Return true if the property has been set (at least once) to non-zero in the mirror. * For positions, use positionIsInitialized() instead. */ template bool propertyIsInitialized( CLFECOMMON::TPropIndex propIndex, TPropertyIndex dsPropertyIndex, const TEntityIndex& entityIndex, T* pt ) { return PropInitialized[propIndex] || checkPropertyInitialized( propIndex, dsPropertyIndex, entityIndex, pt ); } /// Return if the position of the current processed entity is initialized bool positionIsInitialized( const TEntityIndex& entityIndex ) { return PropInitialized[CLFECOMMON::PROPERTY_POSITION] || checkPositionInitialized( entityIndex ); } /// Return if the position of the current processed entity is initialized bool positionIsInitialized() { return PropInitialized[CLFECOMMON::PROPERTY_POSITION] || checkPositionInitialized(); } /// Helper for propertyIsInitialized() template bool checkPropertyInitialized( CLFECOMMON::TPropIndex propIndex, TPropertyIndex dsPropertyIndex, const TEntityIndex& entityIndex, T* ) { CMirrorPropValueRO value( *DataSet, entityIndex, dsPropertyIndex ); return PropInitialized[propIndex] = ( value() != 0 ); /* if ( value() != 0 ) { PropInitialized[propIndex] = true; return true; } else return false; */ } /// Helper for propertyIsInitialized(): overload for row properties (specialization syntax is too much different among compilers) bool checkPropertyInitialized( CLFECOMMON::TPropIndex propIndex, TPropertyIndex dsPropertyIndex, const TEntityIndex& entityIndex, TDataSetRow* ) { CMirrorPropValueRO value( *DataSet, entityIndex, dsPropertyIndex ); return PropInitialized[propIndex] = ( asUInt32(value()) != 0 ); /* if ( asUInt32(value()) != 0 ) // test as integer { PropInitialized[propIndex] = true; return true; } else return false; */ } /// Helper for positionIsInitialized() bool checkPositionInitialized( const TEntityIndex& entityIndex ) { return PropInitialized[CLFECOMMON::PROPERTY_POSITION] = (X() != 0 || Y() != 0); /* if ( ! ((X() == 0) && (Y() == 0)) ) // not local pos because can be 0 0 { PropInitialized[PROPERTY_POSITION] = true; #ifdef NL_DEBUG nldebug( "%u: E%u: Position initialized to %d %d", CTickEventHandler::getGameCycle(), entityIndex.getIndex(), X(), Y() ); #endif return true; } else return false; */ } /// Helper for positionIsInitialized() bool checkPositionInitialized() { return PropInitialized[CLFECOMMON::PROPERTY_POSITION] = (X() != 0 || Y() != 0); } /// Insert entityId, slot into VisionIn //void insertIntoVisionIn( TEntityIndex iviewer, const NLMISC::CEntityId& entityId, CLFECOMMON::TCLEntityId slot, bool replace ); void insertIntoVisionIn( const TEntityIndex& iviewer, const TDataSetRow& entityIndex, CLFECOMMON::TCLEntityId slot, bool replace ); CLFECOMMON::TCoord posXm( const TEntityIndex& ) const { return X()/1000; } CLFECOMMON::TCoord posYm( const TEntityIndex& ) const { return Y()/1000; } CLFECOMMON::TCoord posZm( const TEntityIndex& entityIndex ) const { return z( entityIndex )/1000; } CLFECOMMON::TCoord z( const TEntityIndex& entityIndex ) const { CMirrorPropValueRO zValue( *DataSet, entityIndex, DSPropertyPOSZ ); return zValue(); } CLFECOMMON::TCoord posLocalXm( const TEntityIndex& entityIndex ) const { CMirrorPropValueRO prop( *DataSet, entityIndex, DSPropertyLocalX ); return prop()/1000; } CLFECOMMON::TCoord posLocalYm( const TEntityIndex& entityIndex ) const { CMirrorPropValueRO prop( *DataSet, entityIndex, DSPropertyLocalY ); return prop()/1000; } CLFECOMMON::TCoord posLocalZm( const TEntityIndex& entityIndex ) const { CMirrorPropValueRO prop( *DataSet, entityIndex, DSPropertyLocalZ ); return prop()/1000; } /// Compute the manhattan mileage with X void updateMileageX( const TEntityIndex& entityIndex ) { //H_BEFORE(TestX) if ( ! X.isReadable() ) X.init( *DataSet, entityIndex, DSPropertyPOSX ); //H_AFTER(TestX) #ifdef NL_DEBUG if ( X() == 0 ) nldebug( "X is zero" ); #endif //H_BEFORE(XMileage) Mileage += abs( X() - X.previousValue() ); X.resetChangeFlag(); //H_AFTER(XMileage) } /// Compute the manhattan mileage with Y void updateMileageY( const TEntityIndex& entityIndex ) { //H_BEFORE(TestY) if ( ! Y.isReadable() ) Y.init( *DataSet, entityIndex, DSPropertyPOSY ); //H_AFTER(TestY) #ifdef NL_DEBUG if ( Y() == 0 ) nldebug( "Y is zero" ); #endif //H_BEFORE(YMileage) Mileage += abs( Y() - Y.previousValue() ); Y.resetChangeFlag(); //H_AFTER(YMileage) } void displayProperties( const TEntityIndex& entityIndex, NLMISC::CLog *log=NLMISC::InfoLog, TClientId optClientId=INVALID_CLIENT, CLFECOMMON::TCLEntityId optSlot=CLFECOMMON::INVALID_SLOT ) const; static void fillVisualPropertiesFromMirror( uint64 properties[], const TEntityIndex& entityIndex ); }; /** * * \author Olivier Cado * \author Nevrax France * \date 2002 */ class CEntityContainer { public: /// Instance static CEntityContainer* I; /// Constructor CEntityContainer(); /// Initialisation void init( CClientIdLookup *cl, void (*cbUpdate)(), void (*cbSync)() = NULL ); /// Update (called every cycle by the mirror system) void updateMirror(); /** Get the entity structure corresponding to an entity index * Preconditions: * entityIndex != INVALID_ENTITY_INDEX * entityIndex < _Entities.size() */ CEntity *getEntity( const TEntityIndex& entityIndex ) { #ifdef NL_DEBUG nlassert( entityIndex.isValid() ); nlassert( entityIndex.getIndex() < (uint32)_Entities.size() ); #endif return &_Entities[entityIndex.getIndex()]; } /// Return the entity index corresponding to an entity id (INVALID_ENTITY_INDEX if failed) TEntityIndex entityIdToIndex( const NLMISC::CEntityId& entityId ) const { return _DataSet->getDataSetRow( entityId ); } /// Convert a visual property index into a dataset property index (access to a static array) static TPropertyIndex propertyIndexInDataSetToVisualPropIndex( CLFECOMMON::TPropIndex vPropIndex ) { return _VisualPropIndexToDataSet[vPropIndex]; } /// Initialisation of the properties (after mirror system is ready) void initMirror(); /// Return true when the mirror is ready bool mirrorIsReady() const { return _Mirror.mirrorIsReady(); } /// Return the dataset CMirroredDataSet& dataset() { return *_DataSet; } /// Return the mirror const CMirror& mirrorInstance() { return _Mirror; } CClientIdLookup *EntityToClient; /// Temporary set of entities entering vision, received before their entityIndex is known (workaround) // removed, there should not be any more early visions //TMapOfEarlyVisionAssociations EarlyVisionIns; protected: /// Set the mapping and return the dataset property index TPropertyIndex mapVisualPropIndex( const std::string& propName, CLFECOMMON::TPropIndex vPropIndex ); /// Return the property index at startup TPropertyIndex getPropertyIndex( const std::string& propName ) const; private: /// Vector of entity structures std::vector< CEntity > _Entities; /// Mirror CMirror _Mirror; /// The dataset used by the front-end service CMirroredDataSet *_DataSet; /// Vector to convert dataset property indices into visual property indices //std::vector< TPropertyIndex > _VisualPropIndexToDataSet; // CHANGED BEN static TPropertyIndex _VisualPropIndexToDataSet[CLFECOMMON::NB_VISUAL_PROPERTIES]; }; #define TheEntityContainer (CEntityContainer::I) #define TheDataset (CEntityContainer::I->dataset()) #endif // NL_ENTITY_CONTAINER_H /* End of entity_container.h */