2009-04-15 17:03:31 +03: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
*
*/
2017-07-13 10:26:03 +02:00
# pragma once
# include "../lib/IGameCallback.h"
# include "../lib/battle/BattleAction.h"
# include "../lib/CStopWatch.h"
# include "../lib/int3.h"
2009-04-15 17:03:31 +03:00
2016-08-30 04:13:45 +02:00
struct CPack ;
2014-07-15 10:14:49 +03:00
class CCampaignState ;
class CBattleCallback ;
2011-05-10 01:20:47 +03:00
class IGameEventsReceiver ;
class IBattleEventsReceiver ;
2010-12-22 22:14:40 +02:00
class CBattleGameInterface ;
2008-12-27 03:01:59 +02:00
struct StartInfo ;
class CGameState ;
class CGameInterface ;
class CConnection ;
class CCallback ;
2009-03-07 00:11:17 +02:00
struct BattleAction ;
2017-06-04 14:59:11 +02:00
struct SharedMemory ;
2008-12-27 03:01:59 +02:00
class CClient ;
2011-06-11 02:50:32 +03:00
class CScriptingModule ;
2009-08-30 15:47:40 +03:00
struct CPathsInfo ;
2016-09-10 02:32:40 +02:00
class BinaryDeserializer ;
class BinarySerializer ;
2010-09-03 21:42:54 +03:00
namespace boost { class thread ; }
2009-08-30 15:47:40 +03:00
2011-02-22 13:52:36 +02:00
/// structure to handle running server and connecting to it
2010-09-03 21:42:54 +03:00
class CServerHandler
{
private :
void callServer ( ) ; //calls server via system(), should be called as thread
public :
2011-12-17 21:59:59 +03:00
CStopWatch th ;
2010-09-03 21:42:54 +03:00
boost : : thread * serverThread ; //thread that called system to run server
2017-06-04 14:59:11 +02:00
SharedMemory * shared ;
std : : string uuid ;
2010-09-03 21:42:54 +03:00
bool verbose ; //whether to print log msgs
2010-10-24 14:35:14 +03:00
//functions setting up local server
2010-09-03 21:42:54 +03:00
void startServer ( ) ; //creates a thread with callServer
void waitForServer ( ) ; //waits till server is ready
CConnection * connectToServer ( ) ; //connects to server
2010-10-24 14:35:14 +03:00
//////////////////////////////////////////////////////////////////////////
2017-06-04 07:49:23 +02:00
static CConnection * justConnectToServer ( const std : : string & host = " " , const ui16 port = 0 ) ; //connects to given host without taking any other actions (like setting up server)
static ui16 getDefaultPort ( ) ;
static std : : string getDefaultPortStr ( ) ;
2010-10-24 14:35:14 +03:00
CServerHandler ( bool runServer = false ) ;
2016-10-22 16:22:00 +02:00
virtual ~ CServerHandler ( ) ;
2010-09-03 21:42:54 +03:00
} ;
2012-03-26 01:46:14 +03: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 :
2017-07-12 16:02:25 +02:00
void clear ( )
{
TLock lock ( mx ) ;
items . clear ( ) ;
cond . notify_all ( ) ;
}
2012-03-26 01:46:14 +03:00
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 13:52:36 +02:00
/// Class which handles client - server logic
2008-12-27 03:01:59 +02:00
class CClient : public IGameCallback
{
2015-12-29 04:43:33 +02:00
std : : unique_ptr < CPathsInfo > pathInfo ;
2017-11-16 13:15:43 +02:00
std : : map < PlayerColor , std : : shared_ptr < boost : : thread > > playerActionThreads ;
2009-03-07 00:11:17 +02:00
public :
2015-12-29 04:43:33 +02:00
std : : map < PlayerColor , std : : shared_ptr < CCallback > > callbacks ; //callbacks given to player interfaces
std : : map < PlayerColor , std : : shared_ptr < CBattleCallback > > battleCallbacks ; //callbacks given to player interfaces
std : : vector < std : : shared_ptr < IGameEventsReceiver > > privilagedGameEventReceivers ; //scripting modules, spectator interfaces
std : : vector < std : : shared_ptr < IBattleEventsReceiver > > privilagedBattleEventReceivers ; //scripting modules, spectator interfaces
std : : map < PlayerColor , std : : shared_ptr < CGameInterface > > playerint ;
std : : map < PlayerColor , std : : shared_ptr < CBattleGameInterface > > battleints ;
2013-06-22 21:22:44 +03:00
2015-12-29 04:43:33 +02:00
std : : map < PlayerColor , std : : vector < std : : shared_ptr < IGameEventsReceiver > > > additionalPlayerInts ;
std : : map < PlayerColor , std : : vector < std : : shared_ptr < IBattleEventsReceiver > > > additionalBattleInts ;
2013-06-22 21:22:44 +03:00
2010-02-20 15:24:38 +02:00
bool hotSeat ;
2008-12-27 03:01:59 +02:00
CConnection * serv ;
2011-06-25 17:22:19 +03:00
2013-05-09 14:09:23 +03:00
boost : : optional < BattleAction > curbaction ;
2011-06-11 02:50:32 +03:00
CScriptingModule * erm ;
2008-12-27 03:01:59 +02:00
2017-07-12 16:02:25 +02:00
static ThreadSafeVector < int > waitingRequest ; //FIXME: make this normal field (need to join all threads before client destruction)
2009-08-04 02:53:18 +03:00
2017-11-16 13:15:43 +02:00
2009-08-04 02:53:18 +03:00
//void sendRequest(const CPackForServer *request, bool waitForRealization);
2008-12-27 03:01:59 +02:00
CClient ( void ) ;
CClient ( CConnection * con , StartInfo * si ) ;
~ CClient ( void ) ;
2009-01-30 23:28:02 +02:00
void init ( ) ;
2009-01-11 00:08:18 +02:00
void newGame ( CConnection * con , StartInfo * si ) ; //con - connection to server
2011-02-27 21:58:14 +02:00
void loadNeutralBattleAI ( ) ;
2017-06-03 07:25:10 +02:00
void installNewPlayerInterface ( std : : shared_ptr < CGameInterface > gameInterface , boost : : optional < PlayerColor > color , bool battlecb = false ) ;
2015-12-29 04:43:33 +02:00
void installNewBattleInterface ( std : : shared_ptr < CBattleGameInterface > battleInterface , boost : : optional < PlayerColor > color , bool needCallback = true ) ;
2013-06-22 17:47:20 +03:00
std : : string aiNameForPlayer ( const PlayerSettings & ps , bool battleAI ) ; //empty means no AI -> human
2017-03-14 01:40:39 +02:00
std : : string aiNameForPlayer ( bool battleAI ) ;
2013-06-22 17:47:20 +03:00
2010-08-20 16:34:39 +03:00
void endGame ( bool closeConnection = true ) ;
void stopConnection ( ) ;
2008-12-27 03:01:59 +02:00
void save ( const std : : string & fname ) ;
2017-06-04 07:49:23 +02: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 ui16 port = 0 ) ;
2008-12-27 03:01:59 +02:00
void run ( ) ;
2015-12-29 04:43:33 +02:00
void campaignMapFinished ( std : : shared_ptr < CCampaignState > camp ) ;
void finishCampaign ( std : : shared_ptr < CCampaignState > camp ) ;
void proposeNextMission ( std : : shared_ptr < CCampaignState > camp ) ;
2014-09-21 16:42:08 +03:00
void invalidatePaths ( ) ;
const CPathsInfo * getPathsInfo ( const CGHeroInstance * h ) ;
2009-11-01 03:15:16 +02:00
2009-12-28 06:08:24 +02:00
bool terminate ; // tell to terminate
2017-07-12 18:40:14 +02:00
std : : unique_ptr < boost : : thread > connectionHandler ; //thread running run() method
boost : : mutex connectionHandlerMutex ;
2009-11-01 03:15:16 +02:00
2011-06-11 02:50:32 +03:00
//////////////////////////////////////////////////////////////////////////
2013-06-26 14:18:27 +03:00
virtual PlayerColor getLocalPlayer ( ) const override ;
2011-06-11 02:50:32 +03:00
2008-12-27 03:01:59 +02:00
//////////////////////////////////////////////////////////////////////////
//not working yet, will be implement somewhen later with support for local-sim-based gameplay
2013-06-26 14:18:27 +03: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 { } ;
2016-08-30 04:13:45 +02:00
void changeSecSkill ( const CGHeroInstance * hero , SecondarySkill which , int val , bool abs = false ) override { } ;
2013-06-26 14:18:27 +03:00
2016-08-30 04:13:45 +02:00
void showBlockingDialog ( BlockingDialog * iw ) override { } ;
2013-06-26 14:18:27 +03:00
void showGarrisonDialog ( ObjectInstanceID upobj , ObjectInstanceID hid , bool removableUnits ) override { } ;
2015-03-08 15:52:50 +02:00
void showTeleportDialog ( TeleportDialog * iw ) override { } ;
2013-06-26 14:18:27 +03: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 17:08:25 +02:00
bool changeStackType ( const StackLocation & sl , const CCreature * c ) override { return false ; } ;
2013-06-26 14:18:27 +03: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 15:47:10 +02:00
bool eraseStack ( const StackLocation & sl , bool forceRemoval = false ) override { return false ; } ;
2013-06-26 14:18:27 +03: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 ; }
2016-08-30 04:13:45 +02:00
2013-06-26 14:18:27 +03: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 { } ;
2016-08-30 04:13:45 +02:00
void putArtifact ( const ArtifactLocation & al , const CArtifactInstance * a ) override { } ;
2013-06-26 14:18:27 +03:00
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 { } ;
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 16:04:09 +02:00
bool moveHero ( ObjectInstanceID hid , int3 dst , ui8 teleporting , bool transit = false , PlayerColor asker = PlayerColor : : NEUTRAL ) override { return false ; } ;
2013-06-26 14:18:27 +03: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 13:25:25 +02: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 03:01:59 +02:00
//////////////////////////////////////////////////////////////////////////
friend class CCallback ; //handling players actions
2011-08-25 23:02:38 +03:00
friend class CBattleCallback ; //handling players actions
2013-06-26 14:18:27 +03:00
2013-03-03 20:06:03 +03:00
int sendRequest ( const CPack * request , PlayerColor player ) ; //returns ID given to that request
2012-03-26 01:46:14 +03:00
2010-01-02 03:48:44 +02:00
void handlePack ( CPack * pack ) ; //applies the given pack and deletes it
2010-12-23 02:33:48 +02:00
void battleStarted ( const BattleInfo * info ) ;
2015-12-29 04:43:33 +02:00
void commenceTacticPhaseForInt ( std : : shared_ptr < CBattleGameInterface > battleInt ) ; //will be called as separate thread
2009-03-28 20:46:20 +02:00
2013-06-26 14:18:27 +03:00
void commitPackage ( CPackForClient * pack ) override ;
2011-05-22 21:46:52 +03:00
2009-03-28 20:46:20 +02:00
//////////////////////////////////////////////////////////////////////////
2016-09-10 02:32:40 +02:00
void serialize ( BinarySerializer & h , const int version ) ;
void serialize ( BinaryDeserializer & h , const int version ) ;
2016-08-30 04:13:45 +02:00
2016-09-10 02:32:40 +02:00
void serialize ( BinarySerializer & h , const int version , const std : : set < PlayerColor > & playerIDs ) ;
void serialize ( BinaryDeserializer & h , const int version , const std : : set < PlayerColor > & playerIDs ) ;
2012-08-26 12:07:48 +03:00
void battleFinished ( ) ;
2017-11-16 13:15:43 +02:00
void startPlayerBattleAction ( PlayerColor color ) ;
void stopPlayerBattleAction ( PlayerColor color ) ;
void stopAllBattleActions ( ) ;
private :
void waitForMoveAndSend ( PlayerColor color ) ;
2008-12-27 03:01:59 +02:00
} ;