2011-12-13 21:23:17 +00:00
# pragma once
2008-12-27 01:01:59 +00:00
# include "../lib/IGameCallback.h"
2014-07-15 10:14:49 +03:00
# include "../lib/BattleAction.h"
2011-12-17 18:59:59 +00:00
# include "../lib/CStopWatch.h"
2014-07-15 10:14:49 +03:00
# include "../lib/int3.h"
2009-03-28 18:46:20 +00:00
2009-04-15 14:03:31 +00:00
/*
* Client . h , part of VCMI engine
*
* Authors : listed in file AUTHORS in main folder
*
* License : GNU General Public License v2 .0 or later
* Full text of license available in license . txt file , in main folder
*
*/
2014-07-15 10:14:49 +03:00
class CPack ;
class CCampaignState ;
class CBattleCallback ;
2011-05-09 22:20:47 +00:00
class IGameEventsReceiver ;
class IBattleEventsReceiver ;
2010-12-22 20:14:40 +00:00
class CBattleGameInterface ;
2008-12-27 01:01:59 +00:00
struct StartInfo ;
class CGameState ;
class CGameInterface ;
class CConnection ;
class CCallback ;
2009-03-06 22:11:17 +00:00
struct BattleAction ;
2009-01-30 21:28:02 +00:00
struct SharedMem ;
2008-12-27 01:01:59 +00:00
class CClient ;
2011-06-10 23:50:32 +00:00
class CScriptingModule ;
2009-08-30 12:47:40 +00:00
struct CPathsInfo ;
2014-12-21 22:33:20 +03:00
class CISer ;
class COSer ;
2010-09-03 18:42:54 +00:00
namespace boost { class thread ; }
2009-08-30 12:47:40 +00:00
2011-02-22 11:52:36 +00:00
/// structure to handle running server and connecting to it
2010-09-03 18:42:54 +00:00
class CServerHandler
{
private :
void callServer ( ) ; //calls server via system(), should be called as thread
public :
2011-12-17 18:59:59 +00:00
CStopWatch th ;
2010-09-03 18:42:54 +00:00
boost : : thread * serverThread ; //thread that called system to run server
SharedMem * shared ; //interprocess memory (for waiting for server)
bool verbose ; //whether to print log msgs
std : : string port ; //port number in text form
2010-10-24 11:35:14 +00:00
//functions setting up local server
2010-09-03 18:42:54 +00:00
void startServer ( ) ; //creates a thread with callServer
void waitForServer ( ) ; //waits till server is ready
CConnection * connectToServer ( ) ; //connects to server
2010-10-24 11:35:14 +00:00
//////////////////////////////////////////////////////////////////////////
static CConnection * justConnectToServer ( const std : : string & host = " " , const std : : string & port = " " ) ; //connects to given host without taking any other actions (like setting up server)
CServerHandler ( bool runServer = false ) ;
2010-09-03 18:42:54 +00:00
~ CServerHandler ( ) ;
} ;
2012-03-25 22:46:14 +00:00
template < typename T >
class ThreadSafeVector
{
typedef std : : vector < T > TVector ;
typedef boost : : unique_lock < boost : : mutex > TLock ;
TVector items ;
boost : : mutex mx ;
boost : : condition_variable cond ;
public :
void pushBack ( const T & item )
{
TLock lock ( mx ) ;
items . push_back ( item ) ;
cond . notify_all ( ) ;
}
// //to access list, caller must present a lock used to lock mx
// TVector &getList(TLock &lockedLock)
// {
// assert(lockedLock.owns_lock() && lockedLock.mutex() == &mx);
// return items;
// }
TLock getLock ( )
{
return TLock ( mx ) ;
}
void waitWhileContains ( const T & item )
{
auto lock = getLock ( ) ;
while ( vstd : : contains ( items , item ) )
cond . wait ( lock ) ;
}
bool tryRemovingElement ( const T & item ) //returns false if element was not present
{
auto lock = getLock ( ) ;
auto itr = vstd : : find ( items , item ) ;
if ( itr = = items . end ( ) ) //not in container
{
return false ;
}
items . erase ( itr ) ;
cond . notify_all ( ) ;
return true ;
}
} ;
2011-02-22 11:52:36 +00:00
/// Class which handles client - server logic
2008-12-27 01:01:59 +00:00
class CClient : public IGameCallback
{
2014-09-21 16:42:08 +03:00
unique_ptr < CPathsInfo > pathInfo ;
2009-03-06 22:11:17 +00:00
public :
2013-03-03 17:06:03 +00:00
std : : map < PlayerColor , shared_ptr < CCallback > > callbacks ; //callbacks given to player interfaces
std : : map < PlayerColor , shared_ptr < CBattleCallback > > battleCallbacks ; //callbacks given to player interfaces
2013-06-22 21:47:51 +00:00
std : : vector < shared_ptr < IGameEventsReceiver > > privilagedGameEventReceivers ; //scripting modules, spectator interfaces
std : : vector < shared_ptr < IBattleEventsReceiver > > privilagedBattleEventReceivers ; //scripting modules, spectator interfaces
std : : map < PlayerColor , shared_ptr < CGameInterface > > playerint ;
std : : map < PlayerColor , shared_ptr < CBattleGameInterface > > battleints ;
2013-06-22 18:22:44 +00:00
2013-06-23 16:09:15 +00:00
std : : map < PlayerColor , std : : vector < shared_ptr < IGameEventsReceiver > > > additionalPlayerInts ;
std : : map < PlayerColor , std : : vector < shared_ptr < IBattleEventsReceiver > > > additionalBattleInts ;
2013-06-22 18:22:44 +00:00
2010-02-20 13:24:38 +00:00
bool hotSeat ;
2008-12-27 01:01:59 +00:00
CConnection * serv ;
2011-06-25 14:22:19 +00:00
2013-05-09 11:09:23 +00:00
boost : : optional < BattleAction > curbaction ;
2011-06-10 23:50:32 +00:00
CScriptingModule * erm ;
2008-12-27 01:01:59 +00:00
2012-03-25 22:46:14 +00:00
ThreadSafeVector < int > waitingRequest ;
2009-08-03 23:53:18 +00:00
2013-03-03 17:06:03 +00:00
void waitForMoveAndSend ( PlayerColor color ) ;
2009-08-03 23:53:18 +00:00
//void sendRequest(const CPackForServer *request, bool waitForRealization);
2008-12-27 01:01:59 +00:00
CClient ( void ) ;
CClient ( CConnection * con , StartInfo * si ) ;
~ CClient ( void ) ;
2009-01-30 21:28:02 +00:00
void init ( ) ;
2009-01-10 22:08:18 +00:00
void newGame ( CConnection * con , StartInfo * si ) ; //con - connection to server
2011-02-27 19:58:14 +00:00
void loadNeutralBattleAI ( ) ;
2013-06-22 21:47:51 +00:00
void installNewPlayerInterface ( shared_ptr < CGameInterface > gameInterface , boost : : optional < PlayerColor > color ) ;
void installNewBattleInterface ( shared_ptr < CBattleGameInterface > battleInterface , boost : : optional < PlayerColor > color , bool needCallback = true ) ;
2013-06-22 14:47:20 +00:00
std : : string aiNameForPlayer ( const PlayerSettings & ps , bool battleAI ) ; //empty means no AI -> human
2010-08-20 13:34:39 +00:00
void endGame ( bool closeConnection = true ) ;
void stopConnection ( ) ;
2008-12-27 01:01:59 +00:00
void save ( const std : : string & fname ) ;
2014-10-19 07:47:09 -04:00
void loadGame ( const std : : string & fname , const bool server = true , const std : : vector < int > & humanplayerindices = std : : vector < int > ( ) , const int loadnumplayers = 1 , int player_ = - 1 , const std : : string & ipaddr = " " , const std : : string & port = " " ) ;
2008-12-27 01:01:59 +00:00
void run ( ) ;
2013-02-23 19:16:14 +00:00
void campaignMapFinished ( shared_ptr < CCampaignState > camp ) ;
2012-09-21 17:59:54 +00:00
void finishCampaign ( shared_ptr < CCampaignState > camp ) ;
void proposeNextMission ( shared_ptr < CCampaignState > camp ) ;
2014-09-21 16:42:08 +03:00
void invalidatePaths ( ) ;
const CPathsInfo * getPathsInfo ( const CGHeroInstance * h ) ;
2009-11-01 01:15:16 +00:00
2009-12-28 04:08:24 +00:00
bool terminate ; // tell to terminate
boost : : thread * connectionHandler ; //thread running run() method
2009-11-01 01:15:16 +00:00
2011-06-10 23:50:32 +00:00
//////////////////////////////////////////////////////////////////////////
2013-06-26 11:18:27 +00:00
virtual PlayerColor getLocalPlayer ( ) const override ;
2011-06-10 23:50:32 +00:00
2008-12-27 01:01:59 +00:00
//////////////////////////////////////////////////////////////////////////
//not working yet, will be implement somewhen later with support for local-sim-based gameplay
2013-06-26 11:18:27 +00:00
void changeSpells ( const CGHeroInstance * hero , bool give , const std : : set < SpellID > & spells ) override { } ;
bool removeObject ( const CGObjectInstance * obj ) override { return false ; } ;
void setBlockVis ( ObjectInstanceID objid , bool bv ) override { } ;
void setOwner ( const CGObjectInstance * obj , PlayerColor owner ) override { } ;
void changePrimSkill ( const CGHeroInstance * hero , PrimarySkill : : PrimarySkill which , si64 val , bool abs = false ) override { } ;
void changeSecSkill ( const CGHeroInstance * hero , SecondarySkill which , int val , bool abs = false ) override { } ;
void showBlockingDialog ( BlockingDialog * iw ) override { } ;
void showGarrisonDialog ( ObjectInstanceID upobj , ObjectInstanceID hid , bool removableUnits ) override { } ;
2015-03-08 16:52:50 +03:00
void showTeleportDialog ( TeleportDialog * iw ) override { } ;
2013-06-26 11:18:27 +00:00
void showThievesGuildWindow ( PlayerColor player , ObjectInstanceID requestingObjId ) override { } ;
void giveResource ( PlayerColor player , Res : : ERes which , int val ) override { } ;
virtual void giveResources ( PlayerColor player , TResources resources ) override { } ;
void giveCreatures ( const CArmedInstance * objid , const CGHeroInstance * h , const CCreatureSet & creatures , bool remove ) override { } ;
void takeCreatures ( ObjectInstanceID objid , const std : : vector < CStackBasicDescriptor > & creatures ) override { } ;
2015-09-04 18:08:25 +03:00
bool changeStackType ( const StackLocation & sl , const CCreature * c ) override { return false ; } ;
2013-06-26 11:18:27 +00:00
bool changeStackCount ( const StackLocation & sl , TQuantity count , bool absoluteValue = false ) override { return false ; } ;
bool insertNewStack ( const StackLocation & sl , const CCreature * c , TQuantity count ) override { return false ; } ;
2015-10-12 16:47:10 +03:00
bool eraseStack ( const StackLocation & sl , bool forceRemoval = false ) override { return false ; } ;
2013-06-26 11:18:27 +00:00
bool swapStacks ( const StackLocation & sl1 , const StackLocation & sl2 ) override { return false ; }
bool addToSlot ( const StackLocation & sl , const CCreature * c , TQuantity count ) override { return false ; }
void tryJoiningArmy ( const CArmedInstance * src , const CArmedInstance * dst , bool removeObjWhenFinished , bool allowMerging ) override { }
bool moveStack ( const StackLocation & src , const StackLocation & dst , TQuantity count = - 1 ) override { return false ; }
2013-05-14 13:42:52 +00:00
2013-06-26 11:18:27 +00:00
void removeAfterVisit ( const CGObjectInstance * object ) override { } ;
void giveHeroNewArtifact ( const CGHeroInstance * h , const CArtifact * artType , ArtifactPosition pos ) override { } ;
void giveHeroArtifact ( const CGHeroInstance * h , const CArtifactInstance * a , ArtifactPosition pos ) override { } ;
void putArtifact ( const ArtifactLocation & al , const CArtifactInstance * a ) override { } ;
void removeArtifact ( const ArtifactLocation & al ) override { } ;
bool moveArtifact ( const ArtifactLocation & al1 , const ArtifactLocation & al2 ) override { return false ; } ;
void synchronizeArtifactHandlerLists ( ) override { } ;
void showCompInfo ( ShowInInfobox * comp ) override { } ;
void heroVisitCastle ( const CGTownInstance * obj , const CGHeroInstance * hero ) override { } ;
void stopHeroVisitCastle ( const CGTownInstance * obj , const CGHeroInstance * hero ) override { } ;
2010-12-26 14:34:11 +00:00
//void giveHeroArtifact(int artid, int hid, int position){};
//void giveNewArtifact(int hid, int position){};
2013-06-26 11:18:27 +00:00
void startBattlePrimary ( const CArmedInstance * army1 , const CArmedInstance * army2 , int3 tile , const CGHeroInstance * hero1 , const CGHeroInstance * hero2 , bool creatureBank = false , const CGTownInstance * town = nullptr ) override { } ; //use hero=nullptr for no hero
void startBattleI ( const CArmedInstance * army1 , const CArmedInstance * army2 , int3 tile , bool creatureBank = false ) override { } ; //if any of armies is hero, hero will be used
void startBattleI ( const CArmedInstance * army1 , const CArmedInstance * army2 , bool creatureBank = false ) override { } ; //if any of armies is hero, hero will be used, visitable tile of second obj is place of battle
void setAmount ( ObjectInstanceID objid , ui32 val ) override { } ;
2015-03-08 17:04:09 +03:00
bool moveHero ( ObjectInstanceID hid , int3 dst , ui8 teleporting , bool transit = false , PlayerColor asker = PlayerColor : : NEUTRAL ) override { return false ; } ;
2013-06-26 11:18:27 +00:00
void giveHeroBonus ( GiveBonus * bonus ) override { } ;
void setMovePoints ( SetMovePoints * smp ) override { } ;
void setManaPoints ( ObjectInstanceID hid , int val ) override { } ;
void giveHero ( ObjectInstanceID id , PlayerColor player ) override { } ;
void changeObjPos ( ObjectInstanceID objid , int3 newPos , ui8 flags ) override { } ;
void sendAndApply ( CPackForClient * info ) override { } ;
void heroExchange ( ObjectInstanceID hero1 , ObjectInstanceID hero2 ) override { } ;
2009-03-14 11:25:25 +00:00
2014-06-24 14:50:27 +03:00
void changeFogOfWar ( int3 center , ui32 radius , PlayerColor player , bool hide ) override { }
void changeFogOfWar ( std : : unordered_set < int3 , ShashInt3 > & tiles , PlayerColor player , bool hide ) override { }
2008-12-27 01:01:59 +00:00
//////////////////////////////////////////////////////////////////////////
friend class CCallback ; //handling players actions
2011-08-25 20:02:38 +00:00
friend class CBattleCallback ; //handling players actions
2013-06-26 11:18:27 +00:00
2013-03-03 17:06:03 +00:00
int sendRequest ( const CPack * request , PlayerColor player ) ; //returns ID given to that request
2012-03-25 22:46:14 +00:00
2010-01-02 01:48:44 +00:00
void handlePack ( CPack * pack ) ; //applies the given pack and deletes it
2010-12-23 00:33:48 +00:00
void battleStarted ( const BattleInfo * info ) ;
2013-06-22 21:47:51 +00:00
void commenceTacticPhaseForInt ( shared_ptr < CBattleGameInterface > battleInt ) ; //will be called as separate thread
2009-03-28 18:46:20 +00:00
2013-06-26 11:18:27 +00:00
void commitPackage ( CPackForClient * pack ) override ;
2011-05-22 18:46:52 +00:00
2009-03-28 18:46:20 +00:00
//////////////////////////////////////////////////////////////////////////
2014-12-21 22:33:20 +03:00
void serialize ( COSer & h , const int version ) ;
void serialize ( CISer & h , const int version ) ;
void serialize ( COSer & h , const int version , const std : : set < PlayerColor > & playerIDs ) ;
void serialize ( CISer & h , const int version , const std : : set < PlayerColor > & playerIDs ) ;
2012-08-26 09:07:48 +00:00
void battleFinished ( ) ;
2008-12-27 01:01:59 +00:00
} ;