// 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_DISTANCE_PRIORITIZER_H #define NL_DISTANCE_PRIORITIZER_H #include "nel/misc/types_nl.h" #include "fe_types.h" #include "vision_array.h" #include "processing_spreader.h" #include "entity_container.h" #include "history.h" #include extern TClientId verbosePropertiesSent; #ifdef NL_RELEASE #define LOG_WHAT_IS_SENT ; #else #define LOG_WHAT_IS_SENT if ( (verbosePropertiesSent!=0) && (verbosePropertiesSent!=TVPNodeServer::PrioContext.ClientId) ) {} else nldebug #endif /** * Comparison functor * Helps sorting in decreasing order the entities seen by a client, by priority */ struct TComparePairsByPriority { //TComparePairsByPriority( CVisionArray *va, TClientId clientId ) : _VisionArray(va), _ClientId(clientId) {} // CHANGED BEN TComparePairsByPriority( CVisionArray *va, TClientId clientId ) { _ClientStateArray = va->getClientStateArray(clientId); } bool operator() ( CLFECOMMON::TCLEntityId first, CLFECOMMON::TCLEntityId second ) { //return ( _VisionArray->getPairState( _ClientId, first ).Priority > _VisionArray->getPairState( _ClientId, second ).Priority ); // CHANGED BEN return _ClientStateArray[first].getPrio() > _ClientStateArray[second].getPrio(); } //CVisionArray *_VisionArray; //TClientId _ClientId; TPairState* _ClientStateArray; }; class CHistory; class CClientHost; class CVisionProvider; void fillSHEET( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillBEHAVIOUR( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillNAME_STRING_ID( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillTARGET_LIST( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillVISUAL_FX( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillBARS( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillVisualPropertyABC( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillCONTEXTUAL( TOutBox& outbox, CLFECOMMON::TPropIndex propIndex ); void fillMODE( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillGUILD_SYMBOL( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillGUILD_NAME_ID( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillEVENT_FACTION_ID( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillPVP_MODE( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillPVP_CLAN( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillOWNER_PEOPLE( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillOUTPOST_INFOS( TOutBox& outbox, CLFECOMMON::TPropIndex ); void fillRowProperty( TOutBox& outbox, CLFECOMMON::TPropIndex propIndex ); class CDistancePrioritizer; typedef void (*TFillFunc) (TOutBox&, CLFECOMMON::TPropIndex); /* * Visual property tree node */ class TVPNodeServer : public CLFECOMMON::TVPNodeBase { public: struct TPrioContext { CDistancePrioritizer *Prioritizer; CClientHost *ClientHost; TClientId ClientId; CLFECOMMON::TCLEntityId Slot; CEntity *Sentity; TEntityIndex EntityIndex; //TPairState *PairState; CLFECOMMON::TCoord DistanceCE; NLMISC::TGameCycle Timestamp; bool IsTarget; bool PositionAlreadySent; sint32 ZCache; }; // Static data static TPrioContext PrioContext; TFillFunc FillFunc; /// Flattened Node Tree (speed up server VP tree accesses) static TVPNodeServer* FlatVPTree[CLFECOMMON::NB_VISUAL_PROPERTIES]; /* * propagateBackBranchHasPayload used to walk through VPNode tree build. * Yet, this tree is invariable, so scanning processing is invariable too. * So I build a simple array of all nodes to be examined (all nodes except leaves.) */ static const bool FalseBoolPayLoad; struct CSortedFlatVPTreeItem { CSortedFlatVPTreeItem() : Node(NULL), APayLoad(&FalseBoolPayLoad), BPayLoad(&FalseBoolPayLoad) { } TVPNodeServer* Node; const bool* APayLoad; const bool* BPayLoad; }; struct CSortedFlatVPTreeFillItem { CSortedFlatVPTreeFillItem() : Node(NULL), NextIfNoPayload(0) { } TVPNodeServer* Node; uint NextIfNoPayload; }; /// Reordered Node tree (used by fastPropagateBackBranchHasPayload()) static std::vector SortedFlatVPTree; static void fastPropagateBackBranchHasPayload() { std::vector::iterator it; for (it=SortedFlatVPTree.begin(); it!=SortedFlatVPTree.end(); ++it) (*it).Node->BranchHasPayload = ( *((*it).APayLoad) || *((*it).BPayLoad) ); } /// Reordered Node tree (used by fastFillDiscreetProperties) static std::vector SortedFlatVPTreeFill; static void fastFillDiscreetProperties(TOutBox& outbox) { uint i = 0, size = (uint)SortedFlatVPTreeFill.size(); while (i < size) { CSortedFlatVPTreeFillItem& item = SortedFlatVPTreeFill[i++]; TVPNodeServer* node = item.Node; //nlinfo("examine node %p:%d", node, node->PropIndex); if (!node->isLeaf()) { outbox.serialBitAndLog( node->BranchHasPayload ); if (!node->BranchHasPayload) i = item.NextIfNoPayload; } else { // The payload bit depends on the success of fillDiscreetProperty, so it is serialized inside node->FillFunc( outbox, node->PropIndex ); } } } /// Constructor TVPNodeServer() : TVPNodeBase(), FillFunc(NULL) {} //bool isLeaf() const { return FillFunc != NULL; } // for discreet ones TVPNodeServer *a() { return (TVPNodeServer*)VPA; } TVPNodeServer *b() { return (TVPNodeServer*)VPB; } TVPNodeServer *parent() { return (TVPNodeServer*)VPParent; } /// Set the flags in the entire tree using the flags in the leaves bool propagateBackBranchHasPayload() { //nldebug( "Level: %u - Node=%p A=%p B=%p", getLevel(), this, VPA, VPB ); if ( isLeaf() ) { return BranchHasPayload; } else { bool leftBranch = (a() && a()->propagateBackBranchHasPayload()); bool rightBranch = (b() && b()->propagateBackBranchHasPayload()); BranchHasPayload = (leftBranch || rightBranch); return BranchHasPayload; } } /// Fill the values of the discreet properties the PropIndexes of which are in the leaves, using the BranchHasPayload flags void fillDiscreetProperties( TOutBox& outbox ) { //nlinfo("examine node %p:%d", this, PropIndex); if ( BranchHasPayload ) { //nldebug( "Level: %u - Node=%p A=%p B=%p", getLevel(), this, VPA, VPB ); if ( isLeaf() ) { // The payload bit depends on the success of fillDiscreetProperty, so it is serialized inside (*FillFunc)( outbox, PropIndex ); } else { outbox.serialBitAndLog( BranchHasPayload ); if ( a() ) a()->fillDiscreetProperties( outbox ); if ( a() ) b()->fillDiscreetProperties( outbox ); } } else { outbox.serialBitAndLog( BranchHasPayload ); } } #define INIT_FLAT_VP_TREE( vpname ) FlatVPTree[ CLFECOMMON::PROPERTY_##vpname ] = (TVPNodeServer*)(discreetRoot->get##vpname##node()) #define SET_FILLFUNC( name ) INIT_FLAT_VP_TREE(name); ((TVPNodeServer*)(discreetRoot->get##name##node()))->FillFunc = fill##name #define SET_FILLFUNC2( nameProp, nameFunc ) INIT_FLAT_VP_TREE(nameProp); ((TVPNodeServer*)(discreetRoot->get##nameProp##node()))->FillFunc = fill##nameFunc sint buildTree() { sint res = TVPNodeBase::buildTree(); TVPNodeBase *discreetRoot = VPB->VPB; // setup position a orientation manually FlatVPTree[ CLFECOMMON::PROPERTY_POSITION ] = (TVPNodeServer*)VPA; FlatVPTree[ CLFECOMMON::PROPERTY_POSX ] = FlatVPTree[ CLFECOMMON::PROPERTY_POSITION ]; FlatVPTree[ CLFECOMMON::PROPERTY_POSY ] = FlatVPTree[ CLFECOMMON::PROPERTY_POSITION ]; FlatVPTree[ CLFECOMMON::PROPERTY_POSZ ] = FlatVPTree[ CLFECOMMON::PROPERTY_POSITION ]; FlatVPTree[ CLFECOMMON::PROPERTY_ORIENTATION ] = (TVPNodeServer*)(VPB->VPA); SET_FILLFUNC( SHEET ); SET_FILLFUNC( BEHAVIOUR ); SET_FILLFUNC( NAME_STRING_ID ); SET_FILLFUNC( TARGET_LIST ); FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST_0 ] = FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST ]; FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST_1 ] = FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST ]; FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST_2 ] = FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST ]; FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST_3 ] = FlatVPTree[ CLFECOMMON::PROPERTY_TARGET_LIST ]; SET_FILLFUNC( VISUAL_FX ); SET_FILLFUNC( CONTEXTUAL ); SET_FILLFUNC( MODE ); SET_FILLFUNC( BARS ); SET_FILLFUNC2( VPA, VisualPropertyABC ); SET_FILLFUNC2( VPB, VisualPropertyABC ); SET_FILLFUNC2( VPC, VisualPropertyABC ); SET_FILLFUNC2( TARGET_ID, RowProperty ); SET_FILLFUNC2( ENTITY_MOUNTED_ID, RowProperty ); SET_FILLFUNC2( RIDER_ENTITY_ID, RowProperty ); SET_FILLFUNC( GUILD_SYMBOL ); SET_FILLFUNC( GUILD_NAME_ID ); SET_FILLFUNC( EVENT_FACTION_ID ); SET_FILLFUNC( PVP_MODE ); SET_FILLFUNC( PVP_CLAN ); SET_FILLFUNC( OWNER_PEOPLE ); SET_FILLFUNC( OUTPOST_INFOS ); for (uint i=0; iinitSortedFlatVPTreeFill(); return res; } void initSortedFlatVPTree(); void initSortedFlatVPTreeFill(); #define GET_VP_NODE(name) TVPNodeServer::getServerVPNode(PROPERTY_##name) static TVPNodeServer* getServerVPNode(CLFECOMMON::TPropIndex prop) { return FlatVPTree[prop]; } void deleteBranches() { if ( a() ) { a()->deleteBranches(); delete a(); } if ( b() ) { b()->deleteBranches(); delete b(); } } void deleteA() { if ( a() ) { a()->deleteBranches(); delete a(); } } void deleteB() { if ( b() ) { b()->deleteBranches(); delete b(); } } void displayStatesOfTreeLeaves() { std::string /*s1,*/ s2; displayStatesOfTreeLeaves( /*s1,*/ s2 ); if ( ! s2.empty() /*s1 != "000000000"*/ ) { //nldebug( "%u: Filled the properties for C%hu S%hu: %s", CTickEventHandler::getGameCycle(), PrioContext.ClientId, (uint16)PrioContext.Slot, s1.c_str() ); LOG_WHAT_IS_SENT( "%u: Filled props of C%hu S%hu: %s", CTickEventHandler::getGameCycle(), PrioContext.ClientId, (uint16)PrioContext.Slot, s2.c_str() ); } } void displayStatesOfTreeLeaves( /*std::string& s1,*/ std::string& s2 ) { if ( isLeaf() ) { //s1 += (BranchHasPayload?"1":"0"); if ( BranchHasPayload ) s2 += std::string(CLFECOMMON::getPropText( PropIndex )) + std::string(" "); } else { if ( a() ) a()->displayStatesOfTreeLeaves( /*s1,*/ s2 ); if ( b() ) b()->displayStatesOfTreeLeaves( /*s1,*/ s2 ); } } /// From the root, get the node mode (using hardcoded mapping as in CDistancePrioritizer::init()) TVPNodeServer *getModeNodeFromRoot() { //nlassert( b()->b()->b()->a()->a()->PropIndex == CLFECOMMON::PROPERTY_MODE ); return b()->b()->b()->a()->a(); } }; /** * * \author Olivier Cado * \author Nevrax France * \date 2002 */ class CDistancePrioritizer { public: /// Constructor CDistancePrioritizer() : _VisionArray(NULL), _VisionProvider(NULL), _DistanceDeltaRatio(10) { for ( uint c=0; c!=MAX_NB_CLIENTS; ++c ) { _CurrentEntitiesToStudy[c] = 0; } } /// Destructor ~CDistancePrioritizer() { _VisualPropertyTreeRoot->deleteBranches(); } /// Init void init( CVisionArray *va, CVisionProvider *vp, CHistory *hy ); /// Calculate the priorities void calculatePriorities(); /// Called when processing the vision received from the GPMS (and when a client connects, for slot 0) void addEntitySeenByClient( TClientId clientId, CLFECOMMON::TCLEntityId slot ) { _PrioritizedEntitiesByClient[clientId].push_back( slot ); if ( slot == 0 ) { _VisionArray->getPairState( clientId, slot ).setSteadyPrio( 100 ); } } /// Called when processing the vision received from the GPMS void removeEntitySeenByClient( TClientId clientId, CLFECOMMON::TCLEntityId slot ) { std::vector& vec = _PrioritizedEntitiesByClient[clientId]; std::vector::iterator is = std::find( vec.begin(), vec.end(), slot ); if ( is != vec.end() ) { (*is) = vec.back(); // it breaks the sorting, nevermind vec.pop_back(); } } /// Called when a client leaves void removeAllEntitiesSeenByClient( TClientId clientId ) { _PrioritizedEntitiesByClient[clientId].clear(); } /// void fillOutBox( CClientHost& client, TOutBox& outbox ); /// Set/change the distance/delta ratio that triggers the sending of a position void setDistanceDeltaRatioForPos( uint32 ddratio ) { _DistanceDeltaRatio = ddratio; } CProcessingSpreader SortSpreader; protected: /// Begin a browsing cycle void initDispatchingCycle( TClientId clientId ) { _CurrentEntitiesToStudy[clientId] = 0; } /// Browse the entities seen in order of priority; returns INVALID_SLOT if no more CLFECOMMON::TCLEntityId getNextEntityToStudy( TClientId clientId ) { if ( ! _DissassociationsToResend[clientId].empty() ) { CLFECOMMON::TCLEntityId slot = _DissassociationsToResend[clientId].front(); _DissassociationsToResend[clientId].pop_front(); #ifdef NL_DEBUG nldebug( "Returning dissassociation to resend for C%hu S%hu", clientId, (uint16)slot ); #endif return slot; } else { uint& current = _CurrentEntitiesToStudy[clientId]; if ( current < _PrioritizedEntitiesByClient[clientId].size() ) return _PrioritizedEntitiesByClient[clientId][current++]; else return CLFECOMMON::INVALID_SLOT; } } /// void serialSlotHeader( CClientHost& client, CEntity *sentity, TPairState& pairState, CLFECOMMON::TCLEntityId slot, TOutBox& outbox ); /// Called by calculatePriorities() void updatePriorityOfEntitiesSeenByClient( TClientId clientId ) { CLFECOMMON::TCLEntityId slot; TPairState* states = _VisionArray->getClientStateArray(clientId) + 1; for ( slot=1; slot!=MAX_SEEN_ENTITIES_PER_CLIENT; ++slot ) { states->updatePrio(); ++states; //_VisionArray->getPairState( clientId, slot ).updatePrio(); // CHANGED BEN } } /// Called by calculatePriorities() void sortEntitiesOfClient( TClientId clientId ) { std::sort( _PrioritizedEntitiesByClient[clientId].begin(), _PrioritizedEntitiesByClient[clientId].end(), TComparePairsByPriority(_VisionArray, clientId) ); /*#ifdef NL_DEBUG std::string s1 = "Sorted slots:", s2 = "Distances(m):", s3 = "Priorities:"; std::vector::iterator iv; for ( iv=_PrioritizedEntitiesByClient[clientId].begin(); iv!=_PrioritizedEntitiesByClient[clientId].end(); ++iv ) { s1 += NLMISC::toString( " %hu", (uint16)(*iv) ); //s2 += NLMISC::toString( " %d", _VisionArray->getPairState( clientId, *iv ).DistanceCE/1000 ); //s3 += NLMISC::toString( " %.1f", _VisionArray->getPairState( clientId, *iv ).Priority ); } nldebug( "%s", s1.c_str() ); #endif*/ } /// Test the criterion for the position of the entity 'slot' seen by 'clientId' bool positionHasChangedEnough(); /// Test the criterion for the orientation of the entity 'slot' seen by 'clientId' + initialized bool orientationHasChangedEnough(const CPropertyHistory::CPropertyEntry& entry, float angleRatio ); // Test the criterion for thetaIntMode bool thetaIntModehasChanged(const CPropertyHistory::CPropertyEntry& entry); /// Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent void arbitrateAllDiscreetProperties(const CPropertyHistory::CEntityEntry& entry); /// Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent void arbitrateNPCDiscreetProperties(const CPropertyHistory::CEntityEntry& entry); /// Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent void arbitrateCreatureDiscreetProperties(const CPropertyHistory::CEntityEntry& entry); /// Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent void arbitrateForageSourceDiscreetProperties(const CPropertyHistory::CEntityEntry& entry); /// Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent void arbitrateCommonPosAndMode(const CPropertyHistory::CEntityEntry& entry); /// Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent void arbitrateSlot0PosAndMode(const CPropertyHistory::CEntityEntry& entry); // Get distance threshold CLFECOMMON::TCoord getDistThreshold( CLFECOMMON::TPropIndex propIndex ) { //nlassert( propertyid < CLFECOMMON::NB_PROPERTIES ); return _DistThresholdTable[propIndex]; } void arbitrateDiscreetBehaviourProperty(const CPropertyHistory::CEntityEntry& entry, CEntity *sentity); public: #ifdef STORE_MIRROR_VP_IN_CLASS /* * Test the criterion for discreet properties + initialized */ template bool discreetPropertyHasChanged(const CPropertyHistory::CPropertyEntry& entry, const CMirrorPropValueRO& currentValue, CLFECOMMON::TPropIndex propIndex, T* ) const { // Although the client should already know the sheet id of the controlled player, let's send it //if ( (propIndex==PROPERTY_SHEET) && (TVPNodeServer::PrioContext.Slot == 0) ) // return false; if (entry.HasValue) { #ifdef NL_DEBUG if ( (propIndex==PROPERTY_SHEET) && (currentValue() != *((T*)&(entry.LastSent))) ) LOG_WHAT_IS_SENT( "C%hu S%hu: sheet changes from %u to %u", TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.Slot, (uint32)entry.LastSent, asUInt32(currentValue()) ); #endif return (currentValue() != *((T*)&(entry.LastSent))); } else { TPropertyIndex dsPropertyIndex = CEntityContainer::propertyIndexInDataSetToVisualPropIndex( propIndex ); #ifdef NL_DEBUG bool res = TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( propIndex, dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex, (T*)NULL ); if (res) { // Not sent yet //nldebug( "No history yet for C%hu - slot %hu - prop %hu", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex ); if ( (propIndex==PROPERTY_SHEET) ) { CMirrorPropValueRO currentValue( TheDataset, TVPNodeServer::PrioContext.EntityIndex, dsPropertyIndex ); LOG_WHAT_IS_SENT( "C%hu S%hu: sheet initializes to %u", TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.Slot, asUInt32(currentValue()) ); } } return res; #else return TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( propIndex, dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex, (T*)NULL ); #endif } } /* * Test the criterion + initialized: overload for row properties (specialization syntax is too much different among compilers) */ bool discreetPropertyHasChanged(const CPropertyHistory::CPropertyEntry& entry, const CMirrorPropValueRO& currentValue, CLFECOMMON::TPropIndex propIndex, TDataSetRow* ) const { if (entry.HasValue) { //nldebug( "History for C%hu - slot %hu - prop %hu: %"NL_I64"u", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex, lastsent_value ); return (currentValue().getIndex() != *((TDataSetIndex*)&(entry.LastSent))); } else { TPropertyIndex dsPropertyIndex = CEntityContainer::propertyIndexInDataSetToVisualPropIndex( propIndex ); return TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( propIndex, dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex, (TDataSetRow*)NULL ); } } #else // STORE_MIRROR_VP_IN_CLASS /* * Test the criterion for discreet properties + initialized */ template bool discreetPropertyHasChanged(const CPropertyHistory::CPropertyEntry& entry, TPropIndex propIndex, T* ) const { // Although the client should already know the sheet id of the controlled player, let's send it //if ( (propIndex==PROPERTY_SHEET) && (TVPNodeServer::PrioContext.Slot == 0) ) // return false; TPropertyIndex dsPropertyIndex = CEntityContainer::propertyIndexInDataSetToVisualPropIndex( propIndex ); if (entry.HasValue) { //nldebug( "History for C%hu - slot %hu - prop %hu: %"NL_I64"u", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex, lastsent_value ); CMirrorPropValueRO currentValue( TheDataset, TVPNodeServer::PrioContext.EntityIndex, dsPropertyIndex ); #ifdef NL_DEBUG if ( (propIndex==PROPERTY_SHEET) && (currentValue() != *((T*)&(entry.LastSent))) ) LOG_WHAT_IS_SENT( "C%hu S%hu: sheet changes from %u to %u", TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.Slot, (uint32)entry.LastSent, asUInt32(currentValue()) ); #endif return (currentValue() != *((T*)&(entry.LastSent))); } else { #ifdef NL_DEBUG bool res = TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( propIndex, dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex, (T*)NULL ); if (res) { // Not sent yet //nldebug( "No history yet for C%hu - slot %hu - prop %hu", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex ); if ( (propIndex==PROPERTY_SHEET) ) { CMirrorPropValueRO currentValue( TheDataset, TVPNodeServer::PrioContext.EntityIndex, dsPropertyIndex ); LOG_WHAT_IS_SENT( "C%hu S%hu: sheet initializes to %u", TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.Slot, asUInt32(currentValue()) ); } } return res; #else return TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( propIndex, dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex, (T*)NULL ); #endif } } /* * Test the criterion + initialized: overload for row properties (specialization syntax is too much different among compilers) */ bool discreetPropertyHasChanged(const CPropertyHistory::CPropertyEntry& entry, TPropIndex propIndex, TDataSetRow* ) const { TPropertyIndex dsPropertyIndex = CEntityContainer::propertyIndexInDataSetToVisualPropIndex( propIndex ); if (entry.HasValue) { //nldebug( "History for C%hu - slot %hu - prop %hu: %"NL_I64"u", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex, lastsent_value ); CMirrorPropValueRO currentValue( TheDataset, TVPNodeServer::PrioContext.EntityIndex, dsPropertyIndex ); return (currentValue().getIndex() != *((TDataSetIndex*)&(entry.LastSent))); } else { return TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( propIndex, dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex, (TDataSetRow*)NULL ); } } #endif // STORE_MIRROR_VP_IN_CLASS /* * Test the criterion for target list + initialized */ template bool targetListHasChanged(const CPropertyHistory::CPropertyEntry& entry, CLFECOMMON::TPropIndex propIndex, T* ) const { // Although the client should already know the sheet id of the controlled player, let's send it //if ( (propIndex==PROPERTY_SHEET) && (TVPNodeServer::PrioContext.Slot == 0) ) // return false; TPropertyIndex dsPropertyIndex = CEntityContainer::propertyIndexInDataSetToVisualPropIndex( propIndex ); if (entry.HasValue) { //nldebug( "History for C%hu - slot %hu - prop %hu: %"NL_I64"u", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex, lastsent_value ); NLMISC::TGameCycle mirrorCycle = TheDataset.getChangeTimestamp( dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex ); return (mirrorCycle != entry.LastSent); } else { return TheDataset.getChangeTimestamp( dsPropertyIndex, TVPNodeServer::PrioContext.EntityIndex ) != 0; } } /// Test the criterion for the specified property of the entity 'slot' seen by 'clientId' bool entityIsWithinDistanceThreshold( CLFECOMMON::TPropIndex propIndex ); /// Fill the specified dicreet property void fillDiscreetProperty( TOutBox& outbox, CLFECOMMON::TPropIndex propIndex ); /// void pushDissociationToResend( TClientId clientId, CLFECOMMON::TCLEntityId slot ) { _DissassociationsToResend[clientId].push_back( slot ); } CVisionArray* getVisionArray() { return _VisionArray; } private: std::vector _PrioritizedEntitiesByClient [MAX_NB_CLIENTS]; std::list _DissassociationsToResend [MAX_NB_CLIENTS]; uint _CurrentEntitiesToStudy [MAX_NB_CLIENTS]; TVPNodeServer *_VisualPropertyTreeRoot; CVisionArray *_VisionArray; CVisionProvider *_VisionProvider; CHistory *_History; /// Distance threshold CLFECOMMON::TCoord _DistThresholdTable [CLFECOMMON::NB_VISUAL_PROPERTIES]; uint32 _DistanceDeltaRatio; }; #define A a() #define B b() #define Parent parent() #endif // NL_DISTANCE_PRIORITIZER_H /* End of distance_prioritizer.h */