2017-07-13 11:26:03 +03:00
/*
* CGameHandler . cpp , 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-03-17 19:51:07 +00:00
# include "StdInc.h"
# include "../lib/filesystem/Filesystem.h"
2016-01-26 14:51:38 +01:00
# include "../lib/filesystem/FileInfo.h"
2014-03-17 19:51:07 +00:00
# include "../lib/int3.h"
# include "../lib/mapping/CCampaignHandler.h"
# include "../lib/StartInfo.h"
# include "../lib/CModHandler.h"
# include "../lib/CArtHandler.h"
# include "../lib/CBuildingHandler.h"
# include "../lib/CHeroHandler.h"
2015-02-02 11:25:26 +03:00
# include "../lib/spells/CSpellHandler.h"
2015-09-16 10:50:33 +03:00
# include "../lib/spells/ISpellMechanics.h"
2014-03-17 19:51:07 +00:00
# include "../lib/CGeneralTextHandler.h"
# include "../lib/CTownHandler.h"
# include "../lib/CCreatureHandler.h"
# include "../lib/CGameState.h"
2017-03-17 16:48:44 +01:00
# include "../lib/CStack.h"
2017-06-24 16:42:05 +02:00
# include "../lib/battle/BattleInfo.h"
2014-03-17 19:51:07 +00:00
# include "../lib/CondSh.h"
# include "../lib/NetPacks.h"
# include "../lib/VCMI_Lib.h"
# include "../lib/mapping/CMap.h"
2016-09-10 03:32:40 +03:00
# include "../lib/rmg/CMapGenOptions.h"
2014-03-17 19:51:07 +00:00
# include "../lib/VCMIDirs.h"
# include "../lib/ScopeGuard.h"
2014-06-05 23:51:24 +03:00
# include "../lib/CSoundBase.h"
2014-03-17 19:51:07 +00:00
# include "CGameHandler.h"
# include "CVCMIServer.h"
# include "../lib/CCreatureSet.h"
# include "../lib/CThreadHelper.h"
# include "../lib/GameConstants.h"
# include "../lib/registerTypes/RegisterTypes.h"
2016-09-10 03:32:40 +03:00
# include "../lib/serializer/CTypeList.h"
# include "../lib/serializer/Connection.h"
2014-03-17 19:51:07 +00:00
# ifndef _MSC_VER
# include <boost/thread/xtime.hpp>
# endif
2017-06-04 11:59:26 +03:00
extern std : : atomic < bool > serverShuttingDown ;
2014-03-17 19:51:07 +00:00
# ifdef min
# undef min
# endif
# ifdef max
# undef max
# endif
2016-10-12 17:16:26 +02:00
# define COMPLAIN_RET_IF(cond, txt) do {if (cond){complain(txt); return;}} while(0)
# define COMPLAIN_RET_FALSE_IF(cond, txt) do {if (cond){complain(txt); return false;}} while(0)
2014-03-17 19:51:07 +00:00
# define COMPLAIN_RET(txt) {complain(txt); return false;}
# define COMPLAIN_RETF(txt, FORMAT) {complain(boost::str(boost::format(txt) % FORMAT)); return false;}
2014-11-25 18:16:14 +03:00
class ServerSpellCastEnvironment : public SpellCastEnvironment
{
public :
ServerSpellCastEnvironment ( CGameHandler * gh ) ;
~ ServerSpellCastEnvironment ( ) { } ;
2015-10-28 23:53:44 +03:00
void sendAndApply ( CPackForClient * info ) const override ;
2014-11-25 18:16:14 +03:00
CRandomGenerator & getRandomGenerator ( ) const override ;
void complain ( const std : : string & problem ) const override ;
2014-12-26 00:02:50 +03:00
const CMap * getMap ( ) const override ;
const CGameInfoCallback * getCb ( ) const override ;
2017-07-03 21:09:27 +03:00
bool moveHero ( ObjectInstanceID hid , int3 dst , bool teleporting ) const override ;
void genericQuery ( Query * request , PlayerColor color , std : : function < void ( const JsonNode & ) > callback ) const override ;
2014-11-25 18:16:14 +03:00
private :
2015-10-28 23:53:44 +03:00
mutable CGameHandler * gh ;
2014-11-25 18:16:14 +03:00
} ;
2016-11-27 19:13:40 +03:00
CondSh < bool > battleMadeAction ( false ) ;
2014-03-17 19:51:07 +00:00
CondSh < BattleResult * > battleResult ( nullptr ) ;
template < typename T > class CApplyOnGH ;
class CBaseForGHApply
{
public :
virtual bool applyOnGH ( CGameHandler * gh , CConnection * c , void * pack , PlayerColor player ) const = 0 ;
virtual ~ CBaseForGHApply ( ) { }
template < typename U > static CBaseForGHApply * getApplier ( const U * t = nullptr )
{
2017-07-16 12:58:05 +03:00
return new CApplyOnGH < U > ( ) ;
2014-03-17 19:51:07 +00:00
}
} ;
template < typename T > class CApplyOnGH : public CBaseForGHApply
{
public :
2017-02-17 10:16:17 +01:00
bool applyOnGH ( CGameHandler * gh , CConnection * c , void * pack , PlayerColor player ) const override
2014-03-17 19:51:07 +00:00
{
T * ptr = static_cast < T * > ( pack ) ;
ptr - > c = c ;
ptr - > player = player ;
return ptr - > applyGh ( gh ) ;
}
} ;
2015-10-28 23:53:44 +03:00
template < >
2014-03-17 19:51:07 +00:00
class CApplyOnGH < CPack > : public CBaseForGHApply
{
public :
2017-02-17 10:16:17 +01:00
bool applyOnGH ( CGameHandler * gh , CConnection * c , void * pack , PlayerColor player ) const override
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Cannot apply on GH plain CPack! " ) ;
2014-03-17 19:51:07 +00:00
assert ( 0 ) ;
return false ;
}
} ;
static CApplier < CBaseForGHApply > * applier = nullptr ;
CMP_stack cmpst ;
static inline double distance ( int3 a , int3 b )
{
2016-10-12 17:16:26 +02:00
return std : : sqrt ( ( double ) ( a . x - b . x ) * ( a . x - b . x ) + ( a . y - b . y ) * ( a . y - b . y ) ) ;
2014-03-17 19:51:07 +00:00
}
static void giveExp ( BattleResult & r )
{
2016-10-12 17:16:26 +02:00
if ( r . winner > 1 )
2016-01-25 15:36:34 +03:00
{
// draw
return ;
}
2014-03-17 19:51:07 +00:00
r . exp [ 0 ] = 0 ;
r . exp [ 1 ] = 0 ;
2016-10-12 17:16:26 +02:00
for ( auto i = r . casualties [ ! r . winner ] . begin ( ) ; i ! = r . casualties [ ! r . winner ] . end ( ) ; i + + )
2014-03-17 19:51:07 +00:00
{
r . exp [ r . winner ] + = VLC - > creh - > creatures . at ( i - > first ) - > valOfBonuses ( Bonus : : STACK_HEALTH ) * i - > second ;
}
}
2017-07-01 11:34:00 +03:00
static void summonGuardiansHelper ( std : : vector < BattleHex > & output , const BattleHex & targetPosition , ui8 side , bool targetIsTwoHex ) //return hexes for summoning two hex monsters in output, target = unit to guard
2017-01-26 21:24:01 +01:00
{
2017-01-29 11:50:37 +01:00
int x = targetPosition . getX ( ) ;
int y = targetPosition . getY ( ) ;
2017-07-01 11:34:00 +03:00
const bool targetIsAttacker = side = = BattleSide : : ATTACKER ;
2017-01-26 21:24:01 +01:00
if ( targetIsAttacker ) //handle front guardians, TODO: should we handle situation when units start battle near opposite side of the battlefield? Cannot happen in normal H3...
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
else
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
//guardian spawn locations for four default position cases for attacker and defender, non-default starting location for att and def is handled in first two if's
2017-01-29 11:50:37 +01:00
if ( targetIsAttacker & & ( ( y % 2 = = 0 ) | | ( x > 1 ) ) )
2017-01-26 21:24:01 +01:00
{
2017-01-29 11:50:37 +01:00
if ( targetIsTwoHex & & ( y % 2 = = 1 ) & & ( x = = 2 ) ) //handle exceptional case
2017-01-26 21:24:01 +01:00
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : TOP_RIGHT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : BOTTOM_RIGHT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
}
else
{ //add back-side guardians for two-hex target, side guardians for one-hex
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( targetIsTwoHex ? BattleHex : : EDir : : TOP_LEFT : BattleHex : : EDir : : TOP_RIGHT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( targetIsTwoHex ? BattleHex : : EDir : : BOTTOM_LEFT : BattleHex : : EDir : : BOTTOM_RIGHT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
2017-01-29 11:50:37 +01:00
if ( ! targetIsTwoHex & & x > 2 ) //back guard for one-hex
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
else if ( targetIsTwoHex ) //front-side guardians for two-hex target
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) . cloneInDirection ( BattleHex : : EDir : : TOP_RIGHT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) . cloneInDirection ( BattleHex : : EDir : : BOTTOM_RIGHT , false ) , output ) ;
2017-01-29 11:50:37 +01:00
if ( x > 3 ) //back guard for two-hex
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
}
}
}
2017-01-29 11:50:37 +01:00
else if ( ! targetIsAttacker & & ( ( y % 2 = = 1 ) | | ( x < GameConstants : : BFIELD_WIDTH - 2 ) ) )
2017-01-26 21:24:01 +01:00
{
2017-01-29 11:50:37 +01:00
if ( targetIsTwoHex & & ( y % 2 = = 0 ) & & ( x = = GameConstants : : BFIELD_WIDTH - 3 ) ) //handle exceptional case... equivalent for above for defender side
2017-01-26 21:24:01 +01:00
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : TOP_LEFT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : BOTTOM_LEFT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
}
else
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( targetIsTwoHex ? BattleHex : : EDir : : TOP_RIGHT : BattleHex : : EDir : : TOP_LEFT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( targetIsTwoHex ? BattleHex : : EDir : : BOTTOM_RIGHT : BattleHex : : EDir : : BOTTOM_LEFT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
2017-01-29 11:50:37 +01:00
if ( ! targetIsTwoHex & & x < GameConstants : : BFIELD_WIDTH - 3 )
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
else if ( targetIsTwoHex )
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) . cloneInDirection ( BattleHex : : EDir : : TOP_LEFT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) . cloneInDirection ( BattleHex : : EDir : : BOTTOM_LEFT , false ) , output ) ;
2017-01-29 11:50:37 +01:00
if ( x < GameConstants : : BFIELD_WIDTH - 4 )
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
}
}
}
2017-01-29 11:50:37 +01:00
else if ( ! targetIsAttacker & & y % 2 = = 0 )
2017-01-26 21:24:01 +01:00
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) . cloneInDirection ( BattleHex : : EDir : : TOP_LEFT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : LEFT , false ) . cloneInDirection ( BattleHex : : EDir : : BOTTOM_LEFT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
}
2017-01-29 11:50:37 +01:00
else if ( targetIsAttacker & & y % 2 = = 1 )
2017-01-26 21:24:01 +01:00
{
2017-05-29 09:33:34 +02:00
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) . cloneInDirection ( BattleHex : : EDir : : TOP_RIGHT , false ) , output ) ;
BattleHex : : checkAndPush ( targetPosition . cloneInDirection ( BattleHex : : EDir : : RIGHT , false ) . cloneInDirection ( BattleHex : : EDir : : BOTTOM_RIGHT , false ) , output ) ;
2017-01-26 21:24:01 +01:00
}
}
2014-03-17 19:51:07 +00:00
PlayerStatus PlayerStatuses : : operator [ ] ( PlayerColor player )
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
2016-10-12 17:16:26 +02:00
if ( players . find ( player ) ! = players . end ( ) )
2014-03-17 19:51:07 +00:00
{
return players . at ( player ) ;
}
else
{
throw std : : runtime_error ( " No such player! " ) ;
}
}
void PlayerStatuses : : addPlayer ( PlayerColor player )
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
players [ player ] ;
}
bool PlayerStatuses : : checkFlag ( PlayerColor player , bool PlayerStatus : : * flag )
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
2016-10-12 17:16:26 +02:00
if ( players . find ( player ) ! = players . end ( ) )
2014-03-17 19:51:07 +00:00
{
return players [ player ] . * flag ;
}
else
{
throw std : : runtime_error ( " No such player! " ) ;
}
}
void PlayerStatuses : : setFlag ( PlayerColor player , bool PlayerStatus : : * flag , bool val )
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
2016-10-12 17:16:26 +02:00
if ( players . find ( player ) ! = players . end ( ) )
2014-03-17 19:51:07 +00:00
{
players [ player ] . * flag = val ;
}
else
{
throw std : : runtime_error ( " No such player! " ) ;
}
cv . notify_all ( ) ;
}
template < typename T >
void callWith ( std : : vector < T > args , std : : function < void ( T ) > fun , ui32 which )
{
fun ( args [ which ] ) ;
}
void CGameHandler : : levelUpHero ( const CGHeroInstance * hero , SecondarySkill skill )
{
changeSecSkill ( hero , skill , 1 , 0 ) ;
expGiven ( hero ) ;
}
void CGameHandler : : levelUpHero ( const CGHeroInstance * hero )
{
// required exp for at least 1 lvl-up hasn't been reached
2016-10-12 17:16:26 +02:00
if ( ! hero - > gainsLevel ( ) )
2014-03-17 19:51:07 +00:00
{
return ;
}
2014-04-10 19:11:09 +02:00
// give primary skill
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " %s got level %d " , hero - > name , hero - > level ) ;
2016-08-23 19:12:10 +03:00
auto primarySkill = hero - > nextPrimarySkill ( getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
SetPrimSkill sps ;
sps . id = hero - > id ;
2014-04-10 19:11:09 +02:00
sps . which = primarySkill ;
2014-03-17 19:51:07 +00:00
sps . abs = false ;
sps . val = 1 ;
sendAndApply ( & sps ) ;
2016-09-08 00:24:05 +03:00
PrepareHeroLevelUp pre ;
pre . hero = hero ;
sendAndApply ( & pre ) ;
2014-03-17 19:51:07 +00:00
HeroLevelUp hlu ;
hlu . hero = hero ;
2014-04-10 19:11:09 +02:00
hlu . primskill = primarySkill ;
2016-09-08 00:24:05 +03:00
hlu . skills = pre . skills ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( hlu . skills . size ( ) = = 0 )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( & hlu ) ;
levelUpHero ( hero ) ;
}
2016-10-12 17:16:26 +02:00
else if ( hlu . skills . size ( ) = = 1 )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( & hlu ) ;
2016-09-08 00:24:05 +03:00
levelUpHero ( hero , pre . skills . front ( ) ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
else if ( hlu . skills . size ( ) > 1 )
2014-03-17 19:51:07 +00:00
{
2017-06-06 07:53:51 +03:00
auto levelUpQuery = std : : make_shared < CHeroLevelUpDialogQuery > ( this , hlu ) ;
2014-03-17 19:51:07 +00:00
hlu . queryID = levelUpQuery - > queryID ;
queries . addQuery ( levelUpQuery ) ;
sendAndApply ( & hlu ) ;
//level up will be called on query reply
}
}
void CGameHandler : : levelUpCommander ( const CCommanderInstance * c , int skill )
{
SetCommanderProperty scp ;
auto hero = dynamic_cast < const CGHeroInstance * > ( c - > armyObj ) ;
if ( hero )
scp . heroid = hero - > id ;
else
{
complain ( " Commander is not led by hero! " ) ;
return ;
}
scp . accumulatedBonus . subtype = 0 ;
scp . accumulatedBonus . additionalInfo = 0 ;
scp . accumulatedBonus . duration = Bonus : : PERMANENT ;
scp . accumulatedBonus . turnsRemain = 0 ;
scp . accumulatedBonus . source = Bonus : : COMMANDER ;
scp . accumulatedBonus . valType = Bonus : : BASE_NUMBER ;
if ( skill < = ECommander : : SPELL_POWER )
{
scp . which = SetCommanderProperty : : BONUS ;
auto difference = [ ] ( std : : vector < std : : vector < ui8 > > skillLevels , std : : vector < ui8 > secondarySkills , int skill ) - > int
{
int s = std : : min ( skill , ( int ) ECommander : : SPELL_POWER ) ; //spell power level controls also casts and resistance
return skillLevels . at ( skill ) . at ( secondarySkills . at ( s ) ) - ( secondarySkills . at ( s ) ? skillLevels . at ( skill ) . at ( secondarySkills . at ( s ) - 1 ) : 0 ) ;
} ;
switch ( skill )
{
case ECommander : : ATTACK :
scp . accumulatedBonus . type = Bonus : : PRIMARY_SKILL ;
scp . accumulatedBonus . subtype = PrimarySkill : : ATTACK ;
break ;
case ECommander : : DEFENSE :
scp . accumulatedBonus . type = Bonus : : PRIMARY_SKILL ;
scp . accumulatedBonus . subtype = PrimarySkill : : DEFENSE ;
break ;
case ECommander : : HEALTH :
scp . accumulatedBonus . type = Bonus : : STACK_HEALTH ;
scp . accumulatedBonus . valType = Bonus : : PERCENT_TO_BASE ;
break ;
case ECommander : : DAMAGE :
scp . accumulatedBonus . type = Bonus : : CREATURE_DAMAGE ;
scp . accumulatedBonus . subtype = 0 ;
scp . accumulatedBonus . valType = Bonus : : PERCENT_TO_BASE ;
break ;
case ECommander : : SPEED :
scp . accumulatedBonus . type = Bonus : : STACKS_SPEED ;
break ;
case ECommander : : SPELL_POWER :
scp . accumulatedBonus . type = Bonus : : MAGIC_RESISTANCE ;
scp . accumulatedBonus . val = difference ( VLC - > creh - > skillLevels , c - > secondarySkills , ECommander : : RESISTANCE ) ;
sendAndApply ( & scp ) ; //additional pack
scp . accumulatedBonus . type = Bonus : : CREATURE_SPELL_POWER ;
scp . accumulatedBonus . val = difference ( VLC - > creh - > skillLevels , c - > secondarySkills , ECommander : : SPELL_POWER ) * 100 ; //like hero with spellpower = ability level
sendAndApply ( & scp ) ; //additional pack
scp . accumulatedBonus . type = Bonus : : CASTS ;
scp . accumulatedBonus . val = difference ( VLC - > creh - > skillLevels , c - > secondarySkills , ECommander : : CASTS ) ;
sendAndApply ( & scp ) ; //additional pack
scp . accumulatedBonus . type = Bonus : : CREATURE_ENCHANT_POWER ; //send normally
break ;
}
scp . accumulatedBonus . val = difference ( VLC - > creh - > skillLevels , c - > secondarySkills , skill ) ;
sendAndApply ( & scp ) ;
scp . which = SetCommanderProperty : : SECONDARY_SKILL ;
scp . additionalInfo = skill ;
scp . amount = c - > secondarySkills . at ( skill ) + 1 ;
sendAndApply ( & scp ) ;
}
else if ( skill > = 100 )
{
scp . which = SetCommanderProperty : : SPECIAL_SKILL ;
scp . accumulatedBonus = * VLC - > creh - > skillRequirements . at ( skill - 100 ) . first ;
scp . additionalInfo = skill ; //unnormalized
sendAndApply ( & scp ) ;
}
expGiven ( hero ) ;
}
void CGameHandler : : levelUpCommander ( const CCommanderInstance * c )
{
if ( ! c - > gainsLevel ( ) )
{
return ;
}
CommanderLevelUp clu ;
auto hero = dynamic_cast < const CGHeroInstance * > ( c - > armyObj ) ;
if ( hero )
clu . hero = hero ;
else
{
complain ( " Commander is not led by hero! " ) ;
return ;
}
//picking sec. skills for choice
for ( int i = 0 ; i < = ECommander : : SPELL_POWER ; + + i )
{
if ( c - > secondarySkills . at ( i ) < ECommander : : MAX_SKILL_LEVEL )
clu . skills . push_back ( i ) ;
}
int i = 100 ;
for ( auto specialSkill : VLC - > creh - > skillRequirements )
{
if ( c - > secondarySkills . at ( specialSkill . second . first ) = = ECommander : : MAX_SKILL_LEVEL
& & c - > secondarySkills . at ( specialSkill . second . second ) = = ECommander : : MAX_SKILL_LEVEL
& & ! vstd : : contains ( c - > specialSKills , i ) )
clu . skills . push_back ( i ) ;
+ + i ;
}
int skillAmount = clu . skills . size ( ) ;
2016-10-12 17:16:26 +02:00
if ( ! skillAmount )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( & clu ) ;
levelUpCommander ( c ) ;
}
2016-10-12 17:16:26 +02:00
else if ( skillAmount = = 1 | | hero - > tempOwner = = PlayerColor : : NEUTRAL ) //choose skill automatically
2014-03-17 19:51:07 +00:00
{
sendAndApply ( & clu ) ;
2016-09-08 00:54:26 +03:00
levelUpCommander ( c , * RandomGeneratorUtil : : nextItem ( clu . skills , getRandomGenerator ( ) ) ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
else if ( skillAmount > 1 ) //apply and ask for secondary skill
2014-03-17 19:51:07 +00:00
{
2017-06-06 07:53:51 +03:00
auto commanderLevelUp = std : : make_shared < CCommanderLevelUpDialogQuery > ( this , clu ) ;
2014-03-17 19:51:07 +00:00
clu . queryID = commanderLevelUp - > queryID ;
queries . addQuery ( commanderLevelUp ) ;
sendAndApply ( & clu ) ;
}
}
void CGameHandler : : expGiven ( const CGHeroInstance * hero )
{
2016-10-12 17:16:26 +02:00
if ( hero - > gainsLevel ( ) )
2014-03-17 19:51:07 +00:00
levelUpHero ( hero ) ;
2016-10-12 17:16:26 +02:00
else if ( hero - > commander & & hero - > commander - > gainsLevel ( ) )
2014-03-17 19:51:07 +00:00
levelUpCommander ( hero - > commander ) ;
2016-10-12 17:16:26 +02:00
//if (hero->commander && hero->level > hero->commander->level && hero->commander->gainsLevel())
2014-03-17 19:51:07 +00:00
// levelUpCommander(hero->commander);
// else
// levelUpHero(hero);
}
void CGameHandler : : changePrimSkill ( const CGHeroInstance * hero , PrimarySkill : : PrimarySkill which , si64 val , bool abs )
{
if ( which = = PrimarySkill : : EXPERIENCE ) // Check if scenario limit reached
{
if ( gs - > map - > levelLimit ! = 0 )
{
TExpType expLimit = VLC - > heroh - > reqExp ( gs - > map - > levelLimit ) ;
TExpType resultingExp = abs ? val : hero - > exp + val ;
if ( resultingExp > expLimit )
{
// set given experience to max possible, but don't decrease if hero already over top
abs = true ;
val = std : : max ( expLimit , hero - > exp ) ;
InfoWindow iw ;
iw . player = hero - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 1 ) ; //can gain no more XP
iw . text . addReplacement ( hero - > name ) ;
sendAndApply ( & iw ) ;
}
}
}
SetPrimSkill sps ;
sps . id = hero - > id ;
sps . which = which ;
sps . abs = abs ;
sps . val = val ;
sendAndApply ( & sps ) ;
//only for exp - hero may level up
if ( which = = PrimarySkill : : EXPERIENCE )
{
2016-10-12 17:16:26 +02:00
if ( hero - > commander & & hero - > commander - > alive )
2014-03-17 19:51:07 +00:00
{
//FIXME: trim experience according to map limit?
SetCommanderProperty scp ;
scp . heroid = hero - > id ;
scp . which = SetCommanderProperty : : EXPERIENCE ;
scp . amount = val ;
sendAndApply ( & scp ) ;
CBonusSystemNode : : treeHasChanged ( ) ;
}
expGiven ( hero ) ;
}
}
2017-07-15 14:08:20 +03:00
void CGameHandler : : changeSecSkill ( const CGHeroInstance * hero , SecondarySkill which , int val , bool abs )
2014-03-17 19:51:07 +00:00
{
2016-11-25 21:17:24 +03:00
if ( ! hero )
{
logGlobal - > error ( " changeSecSkill provided no hero " ) ;
return ;
}
2014-03-17 19:51:07 +00:00
SetSecSkill sss ;
sss . id = hero - > id ;
sss . which = which ;
sss . val = val ;
sss . abs = abs ;
sendAndApply ( & sss ) ;
2016-10-12 17:16:26 +02:00
if ( which = = SecondarySkill : : WISDOM )
2014-03-17 19:51:07 +00:00
{
2016-11-25 21:17:24 +03:00
if ( hero - > visitedTown )
2014-03-17 19:51:07 +00:00
giveSpells ( hero - > visitedTown , hero ) ;
}
}
void CGameHandler : : endBattle ( int3 tile , const CGHeroInstance * hero1 , const CGHeroInstance * hero2 )
{
LOG_TRACE ( logGlobal ) ;
//Fill BattleResult structure with exp info
giveExp ( * battleResult . data ) ;
if ( battleResult . get ( ) - > result = = BattleResult : : NORMAL ) // give 500 exp for defeating hero, unless he escaped
{
if ( hero1 )
battleResult . data - > exp [ 1 ] + = 500 ;
if ( hero2 )
battleResult . data - > exp [ 0 ] + = 500 ;
}
if ( hero1 )
battleResult . data - > exp [ 0 ] = hero1 - > calculateXp ( battleResult . data - > exp [ 0 ] ) ; //scholar skill
if ( hero2 )
battleResult . data - > exp [ 1 ] = hero2 - > calculateXp ( battleResult . data - > exp [ 1 ] ) ;
const CArmedInstance * bEndArmy1 = gs - > curB - > sides . at ( 0 ) . armyObject ;
const CArmedInstance * bEndArmy2 = gs - > curB - > sides . at ( 1 ) . armyObject ;
const BattleResult : : EResult result = battleResult . get ( ) - > result ;
2016-01-21 20:23:45 +03:00
auto findBattleQuery = [ this ] ( ) - > std : : shared_ptr < CBattleQuery >
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
for ( auto & q : queries . allQueries ( ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( auto bq = std : : dynamic_pointer_cast < CBattleQuery > ( q ) )
if ( bq - > bi = = gs - > curB )
2014-03-17 19:51:07 +00:00
return bq ;
}
2015-12-29 05:43:33 +03:00
return std : : shared_ptr < CBattleQuery > ( ) ;
2014-03-17 19:51:07 +00:00
} ;
auto battleQuery = findBattleQuery ( ) ;
2016-10-12 17:16:26 +02:00
if ( ! battleQuery )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Cannot find battle query! " ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
if ( battleQuery ! = queries . topQuery ( gs - > curB - > sides [ 0 ] . color ) )
2014-03-17 19:51:07 +00:00
complain ( " Player " + boost : : lexical_cast < std : : string > ( gs - > curB - > sides [ 0 ] . color ) + " although in battle has no battle query at the top! " ) ;
2017-08-05 16:09:29 +03:00
battleQuery - > result = boost : : make_optional ( * battleResult . data ) ;
2014-03-17 19:51:07 +00:00
//Check how many battle queries were created (number of players blocked by battle)
const int queriedPlayers = battleQuery ? boost : : count ( queries . allQueries ( ) , battleQuery ) : 0 ;
2017-07-01 16:30:13 +03:00
finishingBattle = make_unique < FinishingBattleHelper > ( battleQuery , queriedPlayers ) ;
2014-03-17 19:51:07 +00:00
CasualtiesAfterBattle cab1 ( bEndArmy1 , gs - > curB ) , cab2 ( bEndArmy2 , gs - > curB ) ; //calculate casualties before deleting battle
ChangeSpells cs ; //for Eagle Eye
2016-10-12 17:16:26 +02:00
if ( finishingBattle - > winnerHero )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( int eagleEyeLevel = finishingBattle - > winnerHero - > getSecSkillLevel ( SecondarySkill : : EAGLE_EYE ) )
2014-03-17 19:51:07 +00:00
{
int maxLevel = eagleEyeLevel + 1 ;
double eagleEyeChance = finishingBattle - > winnerHero - > valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : EAGLE_EYE ) ;
2016-10-12 17:16:26 +02:00
for ( const CSpell * sp : gs - > curB - > sides . at ( ! battleResult . data - > winner ) . usedSpellsHistory )
if ( sp - > level < = maxLevel & & ! vstd : : contains ( finishingBattle - > winnerHero - > spells , sp - > id ) & & getRandomGenerator ( ) . nextInt ( 99 ) < eagleEyeChance )
2014-03-17 19:51:07 +00:00
cs . spells . insert ( sp - > id ) ;
}
}
2016-01-21 20:23:45 +03:00
std : : vector < const CArtifactInstance * > arts ; //display them in window
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( result = = BattleResult : : NORMAL & & finishingBattle - > winnerHero )
2014-03-17 19:51:07 +00:00
{
2016-01-21 20:23:45 +03:00
auto sendMoveArtifact = [ & ] ( const CArtifactInstance * art , MoveArtifact * ma )
2014-03-17 19:51:07 +00:00
{
2016-01-21 20:23:45 +03:00
arts . push_back ( art ) ;
ma - > dst = ArtifactLocation ( finishingBattle - > winnerHero , art - > firstAvailableSlot ( finishingBattle - > winnerHero ) ) ;
sendAndApply ( ma ) ;
} ;
2016-10-12 17:16:26 +02:00
if ( finishingBattle - > loserHero )
2016-01-21 20:23:45 +03:00
{
//TODO: wrap it into a function, somehow (boost::variant -_-)
auto artifactsWorn = finishingBattle - > loserHero - > artifactsWorn ;
2014-03-17 19:51:07 +00:00
for ( auto artSlot : artifactsWorn )
{
MoveArtifact ma ;
2016-01-21 20:23:45 +03:00
ma . src = ArtifactLocation ( finishingBattle - > loserHero , artSlot . first ) ;
2014-03-17 19:51:07 +00:00
const CArtifactInstance * art = ma . src . getArt ( ) ;
2016-10-12 17:16:26 +02:00
if ( art & & ! art - > artType - > isBig ( ) & &
2016-01-21 20:23:45 +03:00
art - > artType - > id ! = ArtifactID : : SPELLBOOK )
// don't move war machines or locked arts (spellbook)
2014-03-17 19:51:07 +00:00
{
2016-01-21 20:23:45 +03:00
sendMoveArtifact ( art , & ma ) ;
2014-03-17 19:51:07 +00:00
}
}
2016-01-21 20:23:45 +03:00
while ( ! finishingBattle - > loserHero - > artifactsInBackpack . empty ( ) )
2014-03-17 19:51:07 +00:00
{
//we assume that no big artifacts can be found
MoveArtifact ma ;
2016-01-21 20:23:45 +03:00
ma . src = ArtifactLocation ( finishingBattle - > loserHero ,
2014-03-17 19:51:07 +00:00
ArtifactPosition ( GameConstants : : BACKPACK_START ) ) ; //backpack automatically shifts arts to beginning
const CArtifactInstance * art = ma . src . getArt ( ) ;
2016-10-12 17:16:26 +02:00
if ( art - > artType - > id ! = ArtifactID : : GRAIL ) //grail may not be won
2016-01-21 20:23:45 +03:00
{
sendMoveArtifact ( art , & ma ) ;
}
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
if ( finishingBattle - > loserHero - > commander ) //TODO: what if commanders belong to no hero?
2014-03-17 19:51:07 +00:00
{
artifactsWorn = finishingBattle - > loserHero - > commander - > artifactsWorn ;
2016-10-12 17:16:26 +02:00
for ( auto artSlot : artifactsWorn )
2014-03-17 19:51:07 +00:00
{
MoveArtifact ma ;
2016-01-21 20:23:45 +03:00
ma . src = ArtifactLocation ( finishingBattle - > loserHero - > commander . get ( ) , artSlot . first ) ;
2014-03-17 19:51:07 +00:00
const CArtifactInstance * art = ma . src . getArt ( ) ;
if ( art & & ! art - > artType - > isBig ( ) )
{
2016-01-21 20:23:45 +03:00
sendMoveArtifact ( art , & ma ) ;
2014-03-17 19:51:07 +00:00
}
}
}
}
2016-10-12 17:16:26 +02:00
for ( auto armySlot : gs - > curB - > battleGetArmyObject ( ! battleResult . data - > winner ) - > stacks )
2014-03-17 19:51:07 +00:00
{
auto artifactsWorn = armySlot . second - > artifactsWorn ;
for ( auto artSlot : artifactsWorn )
{
MoveArtifact ma ;
2016-01-21 20:23:45 +03:00
ma . src = ArtifactLocation ( armySlot . second , artSlot . first ) ;
2014-03-17 19:51:07 +00:00
const CArtifactInstance * art = ma . src . getArt ( ) ;
if ( art & & ! art - > artType - > isBig ( ) )
{
2016-01-21 20:23:45 +03:00
sendMoveArtifact ( art , & ma ) ;
2014-03-17 19:51:07 +00:00
}
}
}
}
sendAndApply ( battleResult . data ) ; //after this point casualties objects are destroyed
2016-10-12 17:16:26 +02:00
if ( arts . size ( ) ) //display loot
2014-03-17 19:51:07 +00:00
{
InfoWindow iw ;
iw . player = finishingBattle - > winnerHero - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 30 ) ; //You have captured enemy artifact
2016-10-12 17:16:26 +02:00
for ( auto art : arts ) //TODO; separate function to display loot for various ojects?
2014-03-17 19:51:07 +00:00
{
2016-01-21 20:23:45 +03:00
iw . components . push_back ( Component (
Component : : ARTIFACT , art - > artType - > id ,
art - > artType - > id = = ArtifactID : : SPELL_SCROLL ? art - > getGivenSpellID ( ) : 0 , 0 ) ) ;
2016-10-12 17:16:26 +02:00
if ( iw . components . size ( ) > = 14 )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( & iw ) ;
iw . components . clear ( ) ;
}
}
2016-10-12 17:16:26 +02:00
if ( iw . components . size ( ) )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( & iw ) ;
}
}
//Eagle Eye secondary skill handling
2016-10-12 17:16:26 +02:00
if ( ! cs . spells . empty ( ) )
2014-03-17 19:51:07 +00:00
{
cs . learn = 1 ;
cs . hid = finishingBattle - > winnerHero - > id ;
InfoWindow iw ;
iw . player = finishingBattle - > winnerHero - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 221 ) ; //Through eagle-eyed observation, %s is able to learn %s
iw . text . addReplacement ( finishingBattle - > winnerHero - > name ) ;
std : : ostringstream names ;
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < cs . spells . size ( ) ; i + + )
2014-03-17 19:51:07 +00:00
{
names < < " %s " ;
2016-10-12 17:16:26 +02:00
if ( i < cs . spells . size ( ) - 2 )
2014-03-17 19:51:07 +00:00
names < < " , " ;
2016-10-12 17:16:26 +02:00
else if ( i < cs . spells . size ( ) - 1 )
2014-03-17 19:51:07 +00:00
names < < " %s " ;
}
names < < " . " ;
iw . text . addReplacement ( names . str ( ) ) ;
auto it = cs . spells . begin ( ) ;
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < cs . spells . size ( ) ; i + + , it + + )
2014-03-17 19:51:07 +00:00
{
iw . text . addReplacement ( MetaString : : SPELL_NAME , it - > toEnum ( ) ) ;
2016-10-12 17:16:26 +02:00
if ( i = = cs . spells . size ( ) - 2 ) //we just added pre-last name
2014-03-17 19:51:07 +00:00
iw . text . addReplacement ( MetaString : : GENERAL_TXT , 141 ) ; // " and "
iw . components . push_back ( Component ( Component : : SPELL , * it , 0 , 0 ) ) ;
}
sendAndApply ( & iw ) ;
sendAndApply ( & cs ) ;
}
2015-11-07 11:42:06 +03:00
cab1 . updateArmy ( this ) ;
cab2 . updateArmy ( this ) ; //take casualties after battle is deleted
2014-03-17 19:51:07 +00:00
//if one hero has lost we will erase him
2016-10-12 17:16:26 +02:00
if ( battleResult . data - > winner ! = 0 & & hero1 )
2014-03-17 19:51:07 +00:00
{
RemoveObject ro ( hero1 - > id ) ;
sendAndApply ( & ro ) ;
}
2016-10-12 17:16:26 +02:00
if ( battleResult . data - > winner ! = 1 & & hero2 )
2014-03-17 19:51:07 +00:00
{
2016-02-22 03:35:38 +03:00
auto town = hero2 - > visitedTown ;
2014-03-17 19:51:07 +00:00
RemoveObject ro ( hero2 - > id ) ;
sendAndApply ( & ro ) ;
2016-02-22 03:35:38 +03:00
2016-10-12 17:16:26 +02:00
if ( town & & ! town - > garrisonHero ) // TODO: that must be called from CGHeroInstance or CGTownInstance
2016-02-22 03:35:38 +03:00
town - > battleFinished ( hero1 , * battleResult . get ( ) ) ;
2014-03-17 19:51:07 +00:00
}
//give exp
if ( battleResult . data - > exp [ 0 ] & & hero1 & & battleResult . get ( ) - > winner = = 0 )
changePrimSkill ( hero1 , PrimarySkill : : EXPERIENCE , battleResult . data - > exp [ 0 ] ) ;
else if ( battleResult . data - > exp [ 1 ] & & hero2 & & battleResult . get ( ) - > winner = = 1 )
changePrimSkill ( hero2 , PrimarySkill : : EXPERIENCE , battleResult . data - > exp [ 1 ] ) ;
queries . popIfTop ( battleQuery ) ;
//--> continuation (battleAfterLevelUp) occurs after level-up queries are handled or on removing query (above)
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : battleAfterLevelUp ( const BattleResult & result )
2014-03-17 19:51:07 +00:00
{
LOG_TRACE ( logGlobal ) ;
finishingBattle - > remainingBattleQueriesCount - - ;
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Decremented queries count to %d " , finishingBattle - > remainingBattleQueriesCount ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( finishingBattle - > remainingBattleQueriesCount > 0 )
2014-03-17 19:51:07 +00:00
//Battle results will be handled when all battle queries are closed
return ;
//TODO consider if we really want it to work like above. ATM each player as unblocked as soon as possible
// but the battle consequences are applied after final player is unblocked. Hard to abuse...
// Still, it looks like a hole.
// Necromancy if applicable.
const CStackBasicDescriptor raisedStack = finishingBattle - > winnerHero ? finishingBattle - > winnerHero - > calculateNecromancy ( * battleResult . data ) : CStackBasicDescriptor ( ) ;
// Give raised units to winner and show dialog, if any were raised,
// units will be given after casualties are taken
const SlotID necroSlot = raisedStack . type ? finishingBattle - > winnerHero - > getSlotFor ( raisedStack . type ) : SlotID ( ) ;
if ( necroSlot ! = SlotID ( ) )
{
2016-09-09 20:30:36 +03:00
finishingBattle - > winnerHero - > showNecromancyDialog ( raisedStack , getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
addToSlot ( StackLocation ( finishingBattle - > winnerHero , necroSlot ) , raisedStack . type , raisedStack . count ) ;
}
BattleResultsApplied resultsApplied ;
resultsApplied . player1 = finishingBattle - > victor ;
resultsApplied . player2 = finishingBattle - > loser ;
sendAndApply ( & resultsApplied ) ;
setBattle ( nullptr ) ;
2016-10-12 17:16:26 +02:00
if ( visitObjectAfterVictory & & result . winner = = 0 & & ! finishingBattle - > winnerHero - > stacks . empty ( ) )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " post-victory visit " ) ;
2014-03-17 19:51:07 +00:00
visitObjectOnTile ( * getTile ( finishingBattle - > winnerHero - > getPosition ( ) ) , finishingBattle - > winnerHero ) ;
}
visitObjectAfterVictory = false ;
//handle victory/loss of engaged players
2014-10-02 19:43:46 +04:00
std : : set < PlayerColor > playerColors = { finishingBattle - > loser , finishingBattle - > victor } ;
2014-03-17 19:51:07 +00:00
checkVictoryLossConditions ( playerColors ) ;
2016-10-12 17:16:26 +02:00
if ( result . result = = BattleResult : : SURRENDER | | result . result = = BattleResult : : ESCAPE ) //loser has escaped or surrendered
2014-03-17 19:51:07 +00:00
{
SetAvailableHeroes sah ;
sah . player = finishingBattle - > loser ;
sah . hid [ 0 ] = finishingBattle - > loserHero - > subID ;
2016-10-12 17:16:26 +02:00
if ( result . result = = BattleResult : : ESCAPE ) //retreat
2014-03-17 19:51:07 +00:00
{
sah . army [ 0 ] . clear ( ) ;
sah . army [ 0 ] . setCreature ( SlotID ( 0 ) , finishingBattle - > loserHero - > type - > initialArmy . at ( 0 ) . creature , 1 ) ;
}
2016-10-12 17:16:26 +02:00
if ( const CGHeroInstance * another = getPlayer ( finishingBattle - > loser ) - > availableHeroes . at ( 0 ) )
2014-03-17 19:51:07 +00:00
sah . hid [ 1 ] = another - > subID ;
else
sah . hid [ 1 ] = - 1 ;
sendAndApply ( & sah ) ;
}
2016-10-12 17:16:26 +02:00
if ( result . winner ! = 2 & & finishingBattle - > winnerHero & & finishingBattle - > winnerHero - > stacks . empty ( ) )
2016-01-22 22:29:53 +03:00
{
RemoveObject ro ( finishingBattle - > winnerHero - > id ) ;
sendAndApply ( & ro ) ;
2016-01-27 11:38:35 +03:00
if ( VLC - > modh - > settings . WINNING_HERO_WITH_NO_TROOPS_RETREATS )
{
SetAvailableHeroes sah ;
sah . player = finishingBattle - > victor ;
sah . hid [ 0 ] = finishingBattle - > winnerHero - > subID ;
sah . army [ 0 ] . clear ( ) ;
sah . army [ 0 ] . setCreature ( SlotID ( 0 ) , finishingBattle - > winnerHero - > type - > initialArmy . at ( 0 ) . creature , 1 ) ;
2016-01-22 22:29:53 +03:00
2016-10-12 17:16:26 +02:00
if ( const CGHeroInstance * another = getPlayer ( finishingBattle - > victor ) - > availableHeroes . at ( 0 ) )
2016-01-27 11:38:35 +03:00
sah . hid [ 1 ] = another - > subID ;
else
sah . hid [ 1 ] = - 1 ;
2014-03-17 19:51:07 +00:00
2016-01-27 11:38:35 +03:00
sendAndApply ( & sah ) ;
}
2014-03-17 19:51:07 +00:00
}
}
void CGameHandler : : prepareAttack ( BattleAttack & bat , const CStack * att , const CStack * def , int distance , int targetHex )
{
bat . bsa . clear ( ) ;
bat . stackAttacking = att - > ID ;
const int attackerLuck = att - > LuckVal ( ) ;
auto sideHeroBlocksLuck = [ ] ( const SideInBattle & side ) { return NBonus : : hasOfType ( side . hero , Bonus : : BLOCK_LUCK ) ; } ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains_if ( gs - > curB - > sides , sideHeroBlocksLuck ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( attackerLuck > 0 & & getRandomGenerator ( ) . nextInt ( 23 ) < attackerLuck )
2014-03-17 19:51:07 +00:00
{
bat . flags | = BattleAttack : : LUCKY ;
}
if ( VLC - > modh - > settings . data [ " hardcodedFeatures " ] [ " NEGATIVE_LUCK " ] . Bool ( ) ) // negative luck enabled
{
2016-09-08 00:54:26 +03:00
if ( attackerLuck < 0 & & getRandomGenerator ( ) . nextInt ( 23 ) < abs ( attackerLuck ) )
2014-03-17 19:51:07 +00:00
{
bat . flags | = BattleAttack : : UNLUCKY ;
}
}
}
2016-10-12 17:16:26 +02:00
if ( getRandomGenerator ( ) . nextInt ( 99 ) < att - > valOfBonuses ( Bonus : : DOUBLE_DAMAGE_CHANCE ) )
2014-03-17 19:51:07 +00:00
{
bat . flags | = BattleAttack : : DEATH_BLOW ;
}
2016-10-12 17:16:26 +02:00
if ( att - > getCreature ( ) - > idNumber = = CreatureID : : BALLISTA )
2014-03-17 19:51:07 +00:00
{
static const int artilleryLvlToChance [ ] = { 0 , 50 , 75 , 100 } ;
const CGHeroInstance * owner = gs - > curB - > getHero ( att - > owner ) ;
int chance = artilleryLvlToChance [ owner - > getSecSkillLevel ( SecondarySkill : : ARTILLERY ) ] ;
2016-10-12 17:16:26 +02:00
if ( chance > getRandomGenerator ( ) . nextInt ( 99 ) )
2014-03-17 19:51:07 +00:00
{
bat . flags | = BattleAttack : : BALLISTA_DOUBLE_DMG ;
}
}
// only primary target
applyBattleEffects ( bat , att , def , distance , false ) ;
if ( ! bat . shot ( ) ) //multiple-hex attack - only in meele
{
std : : set < const CStack * > attackedCreatures = gs - > curB - > getAttackedCreatures ( att , targetHex ) ; //creatures other than primary target
2016-10-12 17:16:26 +02:00
for ( const CStack * stack : attackedCreatures )
2014-03-17 19:51:07 +00:00
{
if ( stack ! = def ) //do not hit same stack twice
{
applyBattleEffects ( bat , att , stack , distance , true ) ;
}
}
}
2016-09-19 23:36:35 +02:00
const std : : shared_ptr < Bonus > bonus = att - > getBonusLocalFirst ( Selector : : type ( Bonus : : SPELL_LIKE_ATTACK ) ) ;
2014-11-28 00:36:14 +03:00
if ( bonus & & ( bat . shot ( ) ) ) //TODO: make it work in melee?
2015-10-28 23:53:44 +03:00
{
2014-11-28 00:36:14 +03:00
//this is need for displaying hit animation
bat . flags | = BattleAttack : : SPELL_LIKE ;
bat . spellID = SpellID ( bonus - > subtype ) ;
2015-10-28 23:53:44 +03:00
2014-11-28 00:36:14 +03:00
//TODO: should spell override creature`s projectile?
2015-10-28 23:53:44 +03:00
2016-09-06 06:40:23 +03:00
auto attackedCreatures = SpellID ( bonus - > subtype ) . toSpell ( ) - > getAffectedStacks ( gs - > curB , ECastingMode : : SPELL_LIKE_ATTACK , att , bonus - > val , targetHex ) ;
2015-10-28 23:53:44 +03:00
2014-03-17 19:51:07 +00:00
//TODO: get exact attacked hex for defender
2016-10-12 17:16:26 +02:00
for ( const CStack * stack : attackedCreatures )
2014-03-17 19:51:07 +00:00
{
if ( stack ! = def ) //do not hit same stack twice
{
applyBattleEffects ( bat , att , stack , distance , true ) ;
}
}
2015-10-28 23:53:44 +03:00
2014-11-28 00:36:14 +03:00
//now add effect info for all attacked stacks
2016-10-12 17:16:26 +02:00
for ( BattleStackAttacked & bsa : bat . bsa )
2014-11-28 00:36:14 +03:00
{
2016-10-12 17:16:26 +02:00
if ( bsa . attackerID = = att - > ID ) //this is our attack and not f.e. fire shield
2014-11-28 00:36:14 +03:00
{
//this is need for displaying affect animation
bsa . flags | = BattleStackAttacked : : SPELL_EFFECT ;
bsa . spellID = SpellID ( bonus - > subtype ) ;
}
}
2015-10-28 23:53:44 +03:00
2014-03-17 19:51:07 +00:00
}
}
void CGameHandler : : applyBattleEffects ( BattleAttack & bat , const CStack * att , const CStack * def , int distance , bool secondary ) //helper function for prepareAttack
{
BattleStackAttacked bsa ;
if ( secondary )
bsa . flags | = BattleStackAttacked : : SECONDARY ; //all other targets do not suffer from spells & spell-like abilities
bsa . attackerID = att - > ID ;
bsa . stackAttacked = def - > ID ;
2016-09-29 21:47:41 +03:00
bsa . damageAmount = gs - > curB - > calculateDmg ( att , def , bat . shot ( ) , distance , bat . lucky ( ) , bat . unlucky ( ) , bat . deathBlow ( ) , bat . ballistaDoubleDmg ( ) , getRandomGenerator ( ) ) ;
2016-09-08 00:54:26 +03:00
def - > prepareAttacked ( bsa , getRandomGenerator ( ) ) ; //calculate casualties
2014-03-17 19:51:07 +00:00
//life drain handling
2017-07-04 14:24:46 +03:00
if ( att - > hasBonusOfType ( Bonus : : LIFE_DRAIN ) & & def - > isLiving ( ) )
2014-03-17 19:51:07 +00:00
{
StacksHealedOrResurrected shi ;
2015-09-16 10:08:40 +03:00
shi . lifeDrain = true ;
shi . tentHealing = false ;
2016-09-24 10:06:24 +03:00
shi . cure = false ;
2014-03-17 19:51:07 +00:00
shi . drainedFrom = def - > ID ;
2017-07-04 14:24:46 +03:00
int32_t toHeal = bsa . damageAmount * att - > valOfBonuses ( Bonus : : LIFE_DRAIN ) / 100 ;
CHealth health = att - > healthAfterHealed ( toHeal , EHealLevel : : RESURRECT , EHealPower : : PERMANENT ) ;
CHealthInfo hi ;
health . toInfo ( hi ) ;
2017-07-09 19:49:52 +03:00
hi . stackId = att - > ID ;
2017-07-04 14:24:46 +03:00
hi . delta = toHeal ;
2014-03-17 19:51:07 +00:00
shi . healedStacks . push_back ( hi ) ;
2017-07-04 14:24:46 +03:00
if ( hi . delta > 0 )
2014-03-17 19:51:07 +00:00
bsa . healedStacks . push_back ( shi ) ;
}
2017-01-20 15:48:45 +01:00
//soul steal handling
2017-07-04 14:24:46 +03:00
if ( att - > hasBonusOfType ( Bonus : : SOUL_STEAL ) & & def - > isLiving ( ) )
2017-01-20 15:48:45 +01:00
{
StacksHealedOrResurrected shi ;
shi . lifeDrain = true ;
shi . tentHealing = false ;
shi . cure = false ;
shi . drainedFrom = def - > ID ;
2017-07-04 14:24:46 +03:00
for ( int i = 0 ; i < 2 ; i + + ) //we can have two bonuses - one with subtype 0 and another with subtype 1
2017-01-29 14:45:36 +01:00
{
2017-07-04 14:24:46 +03:00
if ( att - > hasBonusOfType ( Bonus : : SOUL_STEAL , i ) )
2017-01-29 14:45:36 +01:00
{
2017-07-04 14:24:46 +03:00
int32_t toHeal = bsa . killedAmount * att - > valOfBonuses ( Bonus : : SOUL_STEAL , i ) * att - > MaxHealth ( ) ;
CHealth health = att - > healthAfterHealed ( toHeal , EHealLevel : : OVERHEAL , ( ( i = = 0 ) ? EHealPower : : ONE_BATTLE : EHealPower : : PERMANENT ) ) ;
CHealthInfo hi ;
health . toInfo ( hi ) ;
2017-07-09 19:49:52 +03:00
hi . stackId = att - > ID ;
2017-07-04 14:24:46 +03:00
hi . delta = toHeal ;
if ( hi . delta > 0 )
shi . healedStacks . push_back ( hi ) ;
2017-01-29 14:45:36 +01:00
}
2017-01-20 15:48:45 +01:00
}
2017-07-04 14:24:46 +03:00
if ( ! shi . healedStacks . empty ( ) )
2017-02-01 22:58:31 +01:00
bsa . healedStacks . push_back ( shi ) ;
2017-01-20 15:48:45 +01:00
}
2017-03-18 13:25:12 +03:00
bat . bsa . push_back ( bsa ) ; //add this stack to the list of victims after drain life has been calculated
2014-03-17 19:51:07 +00:00
//fire shield handling
2017-07-04 14:24:46 +03:00
if ( ! bat . shot ( ) & & ! def - > isClone ( ) & &
2016-03-14 15:28:17 +03:00
def - > hasBonusOfType ( Bonus : : FIRE_SHIELD ) & & ! att - > hasBonusOfType ( Bonus : : FIRE_IMMUNITY ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
// TODO: Fire shield damage should be calculated separately after BattleAttack applied.
// Currently it looks like attacking stack damage itself with defenders fire shield.
// So no separate message on spell damage in log and experience calculation is likely wrong too.
2014-03-17 19:51:07 +00:00
BattleStackAttacked bsa2 ;
bsa2 . stackAttacked = att - > ID ; //invert
bsa2 . attackerID = def - > ID ;
2015-04-13 17:27:47 +03:00
bsa2 . flags | = BattleStackAttacked : : EFFECT ; //FIXME: play animation upon efreet and not attacker
2014-03-17 19:51:07 +00:00
bsa2 . effect = 11 ;
2017-07-09 20:23:51 +03:00
bsa2 . damageAmount = ( std : : min < int64_t > ( def - > health . available ( ) , bsa . damageAmount ) * def - > valOfBonuses ( Bonus : : FIRE_SHIELD ) ) / 100 ; //TODO: scale with attack/defense
2016-09-08 00:54:26 +03:00
att - > prepareAttacked ( bsa2 , getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
bat . bsa . push_back ( bsa2 ) ;
}
}
void CGameHandler : : handleConnection ( std : : set < PlayerColor > players , CConnection & c )
{
setThreadName ( " CGameHandler::handleConnection " ) ;
2017-05-31 09:45:26 +03:00
auto handleDisconnection = [ & ] ( const std : : exception & e )
{
2017-06-04 11:59:26 +03:00
boost : : unique_lock < boost : : mutex > lock ( * c . wmx ) ;
2017-05-31 09:45:26 +03:00
assert ( ! c . connected ) ; //make sure that connection has been marked as broken
logGlobal - > error ( e . what ( ) ) ;
conns - = & c ;
for ( auto playerConn : connections )
{
2017-06-04 11:59:26 +03:00
if ( ! serverShuttingDown & & playerConn . second = = & c )
2017-05-31 09:45:26 +03:00
{
2017-06-02 03:34:50 +03:00
PlayerCheated pc ;
pc . player = playerConn . first ;
pc . losingCheatCode = true ;
sendAndApply ( & pc ) ;
2017-05-31 09:45:26 +03:00
checkVictoryLossConditionsForPlayer ( playerConn . first ) ;
}
}
} ;
2014-03-17 19:51:07 +00:00
try
{
while ( 1 ) //server should never shut connection first //was: while(!end2)
{
CPack * pack = nullptr ;
PlayerColor player = PlayerColor : : NEUTRAL ;
si32 requestID = - 999 ;
int packType = 0 ;
{
boost : : unique_lock < boost : : mutex > lock ( * c . rmx ) ;
2017-05-31 09:45:26 +03:00
if ( ! c . connected )
throw clientDisconnectedException ( ) ;
2014-03-17 19:51:07 +00:00
c > > player > > requestID > > pack ; //get the package
2016-10-12 17:16:26 +02:00
if ( ! pack )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Received a null package marked as request %d from player %d " , requestID , player ) ;
2014-03-17 19:51:07 +00:00
}
2016-11-27 22:50:37 +03:00
else
{
packType = typeList . getTypeID ( pack ) ; //get the id of type
2014-03-17 19:51:07 +00:00
2016-11-27 22:50:37 +03:00
logGlobal - > trace ( " Received client message (request %d by player %d (%s)) of type with ID=%d (%s). \n " ,
requestID , player , player . getStr ( ) , packType , typeid ( * pack ) . name ( ) ) ;
}
2014-03-17 19:51:07 +00:00
}
//prepare struct informing that action was applied
auto sendPackageResponse = [ & ] ( bool succesfullyApplied )
{
2017-05-31 09:45:26 +03:00
//dont reply to disconnected client
//TODO: this must be implemented as option of CPackForServer
if ( dynamic_cast < LeaveGame * > ( pack ) | | dynamic_cast < CloseServer * > ( pack ) )
return ;
2014-03-17 19:51:07 +00:00
PackageApplied applied ;
applied . player = player ;
applied . result = succesfullyApplied ;
applied . packType = packType ;
applied . requestID = requestID ;
boost : : unique_lock < boost : : mutex > lock ( * c . wmx ) ;
c < < & applied ;
} ;
2016-09-10 03:32:40 +03:00
CBaseForGHApply * apply = applier - > getApplier ( packType ) ; //and appropriate applier object
if ( isBlockedByQueries ( pack , player ) )
2014-03-17 19:51:07 +00:00
{
sendPackageResponse ( false ) ;
}
2016-10-12 17:16:26 +02:00
else if ( apply )
2014-03-17 19:51:07 +00:00
{
2016-10-02 22:41:57 +02:00
const bool result = apply - > applyOnGH ( this , & c , pack , player ) ;
2016-10-12 17:16:26 +02:00
if ( result )
2016-10-03 20:32:01 +02:00
logGlobal - > trace ( " Message %s successfully applied! " , typeid ( * pack ) . name ( ) ) ;
2016-08-30 01:11:54 +03:00
else
2016-01-23 15:20:51 +03:00
complain ( ( boost : : format ( " Got false in applying %s... that request must have been fishy! " )
2016-03-12 04:41:27 +03:00
% typeid ( * pack ) . name ( ) ) . str ( ) ) ;
2016-08-30 01:11:54 +03:00
2014-03-17 19:51:07 +00:00
sendPackageResponse ( true ) ;
}
else
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Message cannot be applied, cannot find applier (unregistered type)! " ) ;
2014-03-17 19:51:07 +00:00
sendPackageResponse ( false ) ;
}
vstd : : clear_pointer ( pack ) ;
}
}
catch ( boost : : system : : system_error & e ) //for boost errors just log, not crash - probably client shut down connection
{
2017-05-31 09:45:26 +03:00
handleDisconnection ( e ) ;
}
catch ( clientDisconnectedException & e )
{
handleDisconnection ( e ) ;
2014-03-17 19:51:07 +00:00
}
2015-02-14 22:42:47 +03:00
catch ( . . . )
{
2017-06-04 11:59:26 +03:00
serverShuttingDown = true ;
2015-02-14 22:42:47 +03:00
handleException ( ) ;
throw ;
}
2014-03-17 19:51:07 +00:00
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Ended handling connection " ) ;
2014-03-17 19:51:07 +00:00
}
int CGameHandler : : moveStack ( int stack , BattleHex dest )
{
int ret = 0 ;
const CStack * curStack = gs - > curB - > battleGetStackByID ( stack ) ,
* stackAtEnd = gs - > curB - > battleGetStackByPos ( dest ) ;
assert ( curStack ) ;
assert ( dest < GameConstants : : BFIELD_SIZE ) ;
if ( gs - > curB - > tacticDistance )
{
assert ( gs - > curB - > isInTacticRange ( dest ) ) ;
}
2016-02-09 10:45:59 +03:00
auto start = curStack - > position ;
2016-10-12 17:16:26 +02:00
if ( start = = dest )
2014-03-17 19:51:07 +00:00
return 0 ;
//initing necessary tables
auto accessibility = getAccesibility ( curStack ) ;
//shifting destination (if we have double wide stack and we can occupy dest but not be exactly there)
2017-07-01 11:34:00 +03:00
if ( ! stackAtEnd & & curStack - > doubleWide ( ) & & ! accessibility . accessible ( dest , curStack ) )
2014-03-17 19:51:07 +00:00
{
2017-07-03 12:59:33 +03:00
BattleHex shifted = dest . cloneInDirection ( curStack - > destShiftDir ( ) , false ) ;
2017-07-01 11:34:00 +03:00
if ( accessibility . accessible ( shifted , curStack ) )
dest = shifted ;
2014-03-17 19:51:07 +00:00
}
2017-07-01 11:34:00 +03:00
if ( ( stackAtEnd & & stackAtEnd ! = curStack & & stackAtEnd - > alive ( ) ) | | ! accessibility . accessible ( dest , curStack ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Given destination is not accessible! " ) ;
return 0 ;
}
2016-02-13 17:40:31 +03:00
bool canUseGate = false ;
auto dbState = gs - > curB - > si . gateState ;
2017-07-01 11:34:00 +03:00
if ( battleGetSiegeLevel ( ) > 0 & & curStack - > side = = BattleSide : : DEFENDER & &
2016-02-13 17:40:31 +03:00
dbState ! = EGateState : : DESTROYED & &
dbState ! = EGateState : : BLOCKED )
2016-02-09 11:30:09 +03:00
{
2016-02-13 17:40:31 +03:00
canUseGate = true ;
2016-02-09 11:30:09 +03:00
}
2016-02-09 10:45:59 +03:00
std : : pair < std : : vector < BattleHex > , int > path = gs - > curB - > getPath ( start , dest , curStack ) ;
2014-03-17 19:51:07 +00:00
ret = path . second ;
int creSpeed = gs - > curB - > tacticDistance ? GameConstants : : BFIELD_SIZE : curStack - > Speed ( ) ;
2016-02-09 11:59:20 +03:00
auto isGateDrawbridgeHex = [ & ] ( BattleHex hex ) - > bool
{
2016-10-12 17:16:26 +02:00
if ( gs - > curB - > town - > subID = = ETownType : : FORTRESS & & hex = = ESiegeHex : : GATE_BRIDGE )
2016-02-09 11:59:20 +03:00
return true ;
2016-10-12 17:16:26 +02:00
if ( hex = = ESiegeHex : : GATE_OUTER )
2016-02-09 11:59:20 +03:00
return true ;
2016-10-12 17:16:26 +02:00
if ( hex = = ESiegeHex : : GATE_INNER )
2016-02-09 11:59:20 +03:00
return true ;
return false ;
} ;
auto occupyGateDrawbridgeHex = [ & ] ( BattleHex hex ) - > bool
{
2016-10-12 17:16:26 +02:00
if ( isGateDrawbridgeHex ( hex ) )
2016-02-09 11:59:20 +03:00
return true ;
2016-10-12 17:16:26 +02:00
if ( curStack - > doubleWide ( ) )
2016-02-09 11:59:20 +03:00
{
BattleHex otherHex = curStack - > occupiedHex ( hex ) ;
2016-10-12 17:16:26 +02:00
if ( otherHex . isValid ( ) & & isGateDrawbridgeHex ( otherHex ) )
2016-02-09 11:59:20 +03:00
return true ;
}
return false ;
} ;
2016-10-12 17:16:26 +02:00
if ( curStack - > hasBonusOfType ( Bonus : : FLYING ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( path . second < = creSpeed & & path . first . size ( ) > 0 )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( canUseGate & & dbState ! = EGateState : : OPENED & &
2016-02-09 11:59:20 +03:00
occupyGateDrawbridgeHex ( dest ) )
2016-02-09 11:30:09 +03:00
{
2016-02-13 17:40:31 +03:00
BattleUpdateGateState db ;
db . state = EGateState : : OPENED ;
2016-02-09 11:30:09 +03:00
sendAndApply ( & db ) ;
}
2014-03-17 19:51:07 +00:00
//inform clients about move
BattleStackMoved sm ;
sm . stack = curStack - > ID ;
std : : vector < BattleHex > tiles ;
tiles . push_back ( path . first [ 0 ] ) ;
sm . tilesToMove = tiles ;
sm . distance = path . second ;
sm . teleporting = false ;
sendAndApply ( & sm ) ;
}
}
else //for non-flying creatures
{
std : : vector < BattleHex > tiles ;
2015-04-12 09:26:50 +03:00
const int tilesToMove = std : : max ( ( int ) ( path . first . size ( ) - creSpeed ) , 0 ) ;
2014-03-17 19:51:07 +00:00
int v = path . first . size ( ) - 1 ;
2016-02-09 10:45:59 +03:00
path . first . push_back ( start ) ;
2014-03-17 19:51:07 +00:00
2016-02-08 12:15:07 +03:00
// check if gate need to be open or closed at some point
BattleHex openGateAtHex , gateMayCloseAtHex ;
2016-10-12 17:16:26 +02:00
if ( canUseGate )
2016-02-08 12:15:07 +03:00
{
2016-10-12 17:16:26 +02:00
for ( int i = path . first . size ( ) - 1 ; i > = 0 ; i - - )
2016-02-08 12:15:07 +03:00
{
2016-02-09 15:41:37 +03:00
auto needOpenGates = [ & ] ( BattleHex hex ) - > bool
2016-02-08 12:15:07 +03:00
{
2016-10-12 17:16:26 +02:00
if ( gs - > curB - > town - > subID = = ETownType : : FORTRESS & & hex = = ESiegeHex : : GATE_BRIDGE )
2016-02-09 15:41:37 +03:00
return true ;
2016-10-12 17:16:26 +02:00
if ( hex = = ESiegeHex : : GATE_BRIDGE & & i - 1 > = 0 & & path . first [ i - 1 ] = = ESiegeHex : : GATE_OUTER )
2016-02-09 15:41:37 +03:00
return true ;
2016-10-12 17:16:26 +02:00
else if ( hex = = ESiegeHex : : GATE_OUTER | | hex = = ESiegeHex : : GATE_INNER )
2016-02-09 15:41:37 +03:00
return true ;
return false ;
} ;
auto hex = path . first [ i ] ;
2016-10-12 17:16:26 +02:00
if ( ! openGateAtHex . isValid ( ) & & dbState ! = EGateState : : OPENED )
2016-02-09 15:41:37 +03:00
{
2016-10-12 17:16:26 +02:00
if ( needOpenGates ( hex ) )
2016-02-08 12:15:07 +03:00
openGateAtHex = path . first [ i + 1 ] ;
2016-02-09 15:41:37 +03:00
//TODO we need find batter way to handle double-wide stacks
//currently if only second occupied stack part is standing on gate / bridge hex then stack will start to wait for bridge to lower before it's needed. Though this is just a visual bug.
2016-10-12 17:16:26 +02:00
if ( curStack - > doubleWide ( ) )
2016-02-09 15:41:37 +03:00
{
BattleHex otherHex = curStack - > occupiedHex ( hex ) ;
2016-10-12 17:16:26 +02:00
if ( otherHex . isValid ( ) & & needOpenGates ( otherHex ) )
2016-02-09 15:41:37 +03:00
openGateAtHex = path . first [ i + 2 ] ;
2016-02-08 12:15:07 +03:00
}
2016-02-09 10:45:59 +03:00
//gate may be opened and then closed during stack movement, but not other way around
2016-10-12 17:16:26 +02:00
if ( openGateAtHex . isValid ( ) )
2016-02-13 17:40:31 +03:00
dbState = EGateState : : OPENED ;
2016-02-08 12:15:07 +03:00
}
2016-10-12 17:16:26 +02:00
if ( ! gateMayCloseAtHex . isValid ( ) & & dbState ! = EGateState : : CLOSED )
2016-02-08 12:15:07 +03:00
{
2016-10-12 17:16:26 +02:00
if ( hex = = ESiegeHex : : GATE_INNER & & i - 1 > = 0 & & path . first [ i - 1 ] ! = ESiegeHex : : GATE_OUTER )
2016-02-08 12:15:07 +03:00
{
gateMayCloseAtHex = path . first [ i - 1 ] ;
}
2016-10-12 17:16:26 +02:00
if ( gs - > curB - > town - > subID = = ETownType : : FORTRESS )
2016-02-08 12:15:07 +03:00
{
2016-10-12 17:16:26 +02:00
if ( hex = = ESiegeHex : : GATE_BRIDGE & & i - 1 > = 0 & & path . first [ i - 1 ] ! = ESiegeHex : : GATE_OUTER )
2016-02-08 12:15:07 +03:00
{
gateMayCloseAtHex = path . first [ i - 1 ] ;
}
2016-10-12 17:16:26 +02:00
else if ( hex = = ESiegeHex : : GATE_OUTER & & i - 1 > = 0 & &
2016-02-09 17:38:59 +03:00
path . first [ i - 1 ] ! = ESiegeHex : : GATE_INNER & &
path . first [ i - 1 ] ! = ESiegeHex : : GATE_BRIDGE )
2016-02-08 12:15:07 +03:00
{
gateMayCloseAtHex = path . first [ i - 1 ] ;
}
}
2016-10-12 17:16:26 +02:00
else if ( hex = = ESiegeHex : : GATE_OUTER & & i - 1 > = 0 & & path . first [ i - 1 ] ! = ESiegeHex : : GATE_INNER )
2016-02-08 12:15:07 +03:00
{
gateMayCloseAtHex = path . first [ i - 1 ] ;
}
}
}
}
2014-03-17 19:51:07 +00:00
2015-04-12 08:12:07 +03:00
bool stackIsMoving = true ;
2015-10-28 23:53:44 +03:00
2015-04-12 08:12:07 +03:00
while ( stackIsMoving )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( v < tilesToMove )
2014-03-17 19:51:07 +00:00
{
2015-04-12 08:12:07 +03:00
logGlobal - > error ( " Movement terminated abnormally " ) ;
2014-03-17 19:51:07 +00:00
break ;
}
2015-04-12 09:26:50 +03:00
2016-02-08 12:15:07 +03:00
bool gateStateChanging = false ;
2016-02-09 10:45:59 +03:00
//special handling for opening gate on from starting hex
2016-10-12 17:16:26 +02:00
if ( openGateAtHex . isValid ( ) & & openGateAtHex = = start )
2016-02-09 10:45:59 +03:00
gateStateChanging = true ;
else
2015-04-12 08:12:07 +03:00
{
2016-10-12 17:16:26 +02:00
for ( bool obstacleHit = false ; ( ! obstacleHit ) & & ( ! gateStateChanging ) & & ( v > = tilesToMove ) ; - - v )
2015-04-12 08:12:07 +03:00
{
2016-02-09 10:45:59 +03:00
BattleHex hex = path . first [ v ] ;
tiles . push_back ( hex ) ;
2015-04-12 09:26:50 +03:00
2016-10-12 17:16:26 +02:00
if ( ( openGateAtHex . isValid ( ) & & openGateAtHex = = hex ) | |
2016-02-09 10:45:59 +03:00
( gateMayCloseAtHex . isValid ( ) & & gateMayCloseAtHex = = hex ) )
{
gateStateChanging = true ;
}
2015-04-12 09:26:50 +03:00
2016-02-09 10:45:59 +03:00
//if we walked onto something, finalize this portion of stack movement check into obstacle
2017-07-09 10:18:46 +02:00
if ( ! battleGetAllObstaclesOnPos ( hex , false ) . empty ( ) )
2015-04-12 09:26:50 +03:00
obstacleHit = true ;
2016-02-09 10:45:59 +03:00
2016-10-12 17:16:26 +02:00
if ( curStack - > doubleWide ( ) )
2016-02-09 10:45:59 +03:00
{
BattleHex otherHex = curStack - > occupiedHex ( hex ) ;
//two hex creature hit obstacle by backside
2017-07-09 10:18:46 +02:00
auto obstacle2 = battleGetAllObstaclesOnPos ( otherHex , false ) ;
2017-07-01 17:59:53 +02:00
if ( otherHex . isValid ( ) & & ! obstacle2 . empty ( ) )
2016-02-09 10:45:59 +03:00
obstacleHit = true ;
}
2015-04-12 08:12:07 +03:00
}
}
2016-10-12 17:16:26 +02:00
if ( tiles . size ( ) > 0 )
2014-03-17 19:51:07 +00:00
{
2015-04-12 08:12:07 +03:00
//commit movement
BattleStackMoved sm ;
sm . stack = curStack - > ID ;
sm . distance = path . second ;
sm . teleporting = false ;
sm . tilesToMove = tiles ;
sendAndApply ( & sm ) ;
2014-03-17 19:51:07 +00:00
tiles . clear ( ) ;
}
2015-04-12 08:12:07 +03:00
//we don't handle obstacle at the destination tile -> it's handled separately in the if at the end
2016-10-12 17:16:26 +02:00
if ( curStack - > position ! = dest )
2015-04-12 08:12:07 +03:00
{
2017-07-09 10:18:46 +02:00
if ( stackIsMoving & & start ! = curStack - > position )
stackIsMoving = handleDamageFromObstacle ( curStack , stackIsMoving ) ;
2016-10-12 17:16:26 +02:00
if ( gateStateChanging )
2016-02-08 12:15:07 +03:00
{
2016-10-12 17:16:26 +02:00
if ( curStack - > position = = openGateAtHex )
2016-02-08 12:15:07 +03:00
{
2016-02-09 10:45:59 +03:00
openGateAtHex = BattleHex ( ) ;
//only open gate if stack is still alive
2016-10-12 17:16:26 +02:00
if ( curStack - > alive ( ) )
2016-02-09 10:45:59 +03:00
{
2016-02-13 17:40:31 +03:00
BattleUpdateGateState db ;
db . state = EGateState : : OPENED ;
2016-02-09 10:45:59 +03:00
sendAndApply ( & db ) ;
}
2016-02-08 12:15:07 +03:00
}
2016-10-12 17:16:26 +02:00
else if ( curStack - > position = = gateMayCloseAtHex )
2016-02-09 10:45:59 +03:00
{
2016-02-09 15:41:37 +03:00
gateMayCloseAtHex = BattleHex ( ) ;
2016-02-13 17:40:31 +03:00
updateGateState ( ) ;
2016-02-09 10:45:59 +03:00
}
2016-02-08 12:15:07 +03:00
}
2015-04-12 08:12:07 +03:00
}
else
2015-04-12 09:26:50 +03:00
//movement finished normally: we reached destination
2015-04-12 08:12:07 +03:00
stackIsMoving = false ;
2014-03-17 19:51:07 +00:00
}
}
//handling obstacle on the final field (separate, because it affects both flying and walking stacks)
2017-07-09 10:18:46 +02:00
handleDamageFromObstacle ( curStack ) ;
2015-04-12 08:12:07 +03:00
2014-03-17 19:51:07 +00:00
return ret ;
}
CGameHandler : : CGameHandler ( void )
{
QID = 1 ;
//gs = nullptr;
IObjectInterface : : cb = this ;
2017-07-16 12:58:05 +03:00
applier = new CApplier < CBaseForGHApply > ( ) ;
2014-03-17 19:51:07 +00:00
registerTypesServerPacks ( * applier ) ;
visitObjectAfterVictory = false ;
2015-10-28 23:53:44 +03:00
2014-11-25 22:00:04 +03:00
spellEnv = new ServerSpellCastEnvironment ( this ) ;
2014-03-17 19:51:07 +00:00
}
CGameHandler : : ~ CGameHandler ( void )
{
2014-11-25 22:00:04 +03:00
delete spellEnv ;
2014-03-17 19:51:07 +00:00
delete applier ;
applier = nullptr ;
delete gs ;
}
void CGameHandler : : init ( StartInfo * si )
{
2016-10-12 17:16:26 +02:00
if ( si - > seedToBeUsed = = 0 )
2014-03-17 19:51:07 +00:00
{
si - > seedToBeUsed = std : : time ( nullptr ) ;
}
gs = new CGameState ( ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " Gamestate created! " ) ;
2014-03-17 19:51:07 +00:00
gs - > init ( si ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " Gamestate initialized! " ) ;
2014-03-17 19:51:07 +00:00
2014-04-10 19:11:09 +02:00
// reset seed, so that clients can't predict any following random values
2016-09-08 00:54:26 +03:00
getRandomGenerator ( ) . resetSeed ( ) ;
2014-04-10 19:11:09 +02:00
2016-10-12 17:16:26 +02:00
for ( auto & elem : gs - > players )
2014-03-17 19:51:07 +00:00
{
states . addPlayer ( elem . first ) ;
}
}
static bool evntCmp ( const CMapEvent & a , const CMapEvent & b )
{
2016-03-12 04:41:27 +03:00
return a . earlierThan ( b ) ;
2014-03-17 19:51:07 +00:00
}
void CGameHandler : : setPortalDwelling ( const CGTownInstance * town , bool forced = false , bool clear = false )
{ // bool forced = true - if creature should be replaced, if false - only if no creature was set
2016-09-18 11:53:51 +03:00
const PlayerState * p = getPlayer ( town - > tempOwner ) ;
2016-10-12 17:16:26 +02:00
if ( ! p )
2014-03-17 19:51:07 +00:00
{
2017-08-11 20:03:05 +03:00
logGlobal - > warn ( " There is no player owner of town %s at %s " , town - > name , town - > pos . toString ( ) ) ;
2014-03-17 19:51:07 +00:00
return ;
}
if ( forced | | town - > creatures . at ( GameConstants : : CREATURES_PER_TOWN ) . second . empty ( ) ) //we need to change creature
{
SetAvailableCreatures ssi ;
ssi . tid = town - > id ;
ssi . creatures = town - > creatures ;
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . second . clear ( ) ; //remove old one
const std : : vector < ConstTransitivePtr < CGDwelling > > & dwellings = p - > dwellings ;
if ( dwellings . empty ( ) ) //no dwellings - just remove
{
sendAndApply ( & ssi ) ;
return ;
}
2016-09-08 00:54:26 +03:00
auto dwelling = * RandomGeneratorUtil : : nextItem ( dwellings , getRandomGenerator ( ) ) ;
2014-04-10 19:11:09 +02:00
// for multi-creature dwellings like Golem Factory
2016-09-08 00:54:26 +03:00
auto creatureId = RandomGeneratorUtil : : nextItem ( dwelling - > creatures , getRandomGenerator ( ) ) - > second [ 0 ] ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( clear )
2014-04-10 19:11:09 +02:00
{
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . first = std : : max ( ( ui32 ) 1 , ( VLC - > creh - > creatures . at ( creatureId ) - > growth ) / 2 ) ;
}
2014-03-17 19:51:07 +00:00
else
2014-04-10 19:11:09 +02:00
{
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . first = VLC - > creh - > creatures . at ( creatureId ) - > growth ;
}
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . second . push_back ( creatureId ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & ssi ) ;
}
}
void CGameHandler : : newTurn ( )
{
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Turn %d " , gs - > day + 1 ) ;
2014-03-17 19:51:07 +00:00
NewTurn n ;
n . specialWeek = NewTurn : : NO_ACTION ;
n . creatureid = CreatureID : : NONE ;
n . day = gs - > day + 1 ;
bool firstTurn = ! getDate ( Date : : DAY ) ;
bool newWeek = getDate ( Date : : DAY_OF_WEEK ) = = 7 ; //day numbers are confusing, as day was not yet switched
bool newMonth = getDate ( Date : : DAY_OF_MONTH ) = = 28 ;
std : : map < PlayerColor , si32 > hadGold ; //starting gold - for buildings like dwarven treasury
if ( firstTurn )
{
for ( auto obj : gs - > map - > objects )
{
if ( obj & & obj - > ID = = Obj : : PRISON ) //give imprisoned hero 0 exp to level him up. easiest to do at this point
{
changePrimSkill ( getHero ( obj - > id ) , PrimarySkill : : EXPERIENCE , 0 ) ;
}
}
}
if ( newWeek & & ! firstTurn )
{
n . specialWeek = NewTurn : : NORMAL ;
bool deityOfFireBuilt = false ;
2016-10-12 17:16:26 +02:00
for ( const CGTownInstance * t : gs - > map - > towns )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( t - > hasBuilt ( BuildingID : : GRAIL , ETownType : : INFERNO ) )
2014-03-17 19:51:07 +00:00
{
deityOfFireBuilt = true ;
break ;
}
}
2016-10-12 17:16:26 +02:00
if ( deityOfFireBuilt )
2014-03-17 19:51:07 +00:00
{
n . specialWeek = NewTurn : : DEITYOFFIRE ;
n . creatureid = CreatureID : : IMP ;
}
else
{
2016-09-08 00:54:26 +03:00
int monthType = getRandomGenerator ( ) . nextInt ( 99 ) ;
2016-10-12 17:16:26 +02:00
if ( newMonth ) //new month
2014-03-17 19:51:07 +00:00
{
if ( monthType < 40 ) //double growth
{
n . specialWeek = NewTurn : : DOUBLE_GROWTH ;
if ( VLC - > modh - > settings . ALL_CREATURES_GET_DOUBLE_MONTHS )
{
2016-09-08 00:54:26 +03:00
std : : pair < int , CreatureID > newMonster ( 54 , VLC - > creh - > pickRandomMonster ( getRandomGenerator ( ) ) ) ;
2014-03-17 19:51:07 +00:00
n . creatureid = newMonster . second ;
}
2016-10-12 17:16:26 +02:00
else if ( VLC - > creh - > doubledCreatures . size ( ) )
2014-03-17 19:51:07 +00:00
{
const std : : vector < CreatureID > doubledCreatures ( VLC - > creh - > doubledCreatures . begin ( ) , VLC - > creh - > doubledCreatures . end ( ) ) ;
2016-09-08 00:54:26 +03:00
n . creatureid = * RandomGeneratorUtil : : nextItem ( doubledCreatures , getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
}
else
{
complain ( " Cannot find creature that can be spawned! " ) ;
n . specialWeek = NewTurn : : NORMAL ;
}
}
else if ( monthType < 50 )
n . specialWeek = NewTurn : : PLAGUE ;
}
else //it's a week, but not full month
{
if ( monthType < 25 )
{
n . specialWeek = NewTurn : : BONUS_GROWTH ; //+5
2016-09-08 00:54:26 +03:00
std : : pair < int , CreatureID > newMonster ( 54 , VLC - > creh - > pickRandomMonster ( getRandomGenerator ( ) ) ) ;
2014-03-17 19:51:07 +00:00
//TODO do not pick neutrals
n . creatureid = newMonster . second ;
}
}
}
}
std : : map < ui32 , ConstTransitivePtr < CGHeroInstance > > pool = gs - > hpool . heroesPool ;
2016-10-12 17:16:26 +02:00
for ( auto & hp : pool )
2016-01-28 22:48:51 +03:00
{
auto hero = hp . second ;
2016-10-12 17:16:26 +02:00
if ( hero - > isInitialized ( ) & & hero - > stacks . size ( ) )
2016-01-28 22:48:51 +03:00
{
// reset retreated or surrendered heroes
auto maxmove = hero - > maxMovePoints ( true ) ;
// if movement is greater than maxmove, we should decrease it
2016-10-12 17:16:26 +02:00
if ( hero - > movement ! = maxmove | | hero - > mana < hero - > manaLimit ( ) )
2016-01-28 22:48:51 +03:00
{
NewTurn : : Hero hth ;
hth . id = hero - > id ;
hth . move = maxmove ;
2016-01-30 10:20:49 +03:00
hth . mana = hero - > getManaNewTurn ( ) ;
2016-01-28 22:48:51 +03:00
n . heroes . insert ( hth ) ;
}
}
}
2014-03-17 19:51:07 +00:00
for ( auto & elem : gs - > players )
{
2016-10-12 17:16:26 +02:00
if ( elem . first = = PlayerColor : : NEUTRAL )
2014-03-17 19:51:07 +00:00
continue ;
2016-10-12 17:16:26 +02:00
else if ( elem . first > = PlayerColor : : PLAYER_LIMIT )
2014-03-17 19:51:07 +00:00
assert ( 0 ) ; //illegal player number!
std : : pair < PlayerColor , si32 > playerGold ( elem . first , elem . second . resources . at ( Res : : GOLD ) ) ;
hadGold . insert ( playerGold ) ;
2016-10-12 17:16:26 +02:00
if ( newWeek ) //new heroes in tavern
2014-03-17 19:51:07 +00:00
{
SetAvailableHeroes sah ;
sah . player = elem . first ;
//pick heroes and their armies
CHeroClass * banned = nullptr ;
for ( int j = 0 ; j < GameConstants : : AVAILABLE_HEROES_PER_PLAYER ; j + + )
{
//first hero - native if possible, second hero -> any other class
2016-10-12 17:16:26 +02:00
if ( CGHeroInstance * h = gs - > hpool . pickHeroFor ( j = = 0 , elem . first , getNativeTown ( elem . first ) , pool , getRandomGenerator ( ) , banned ) )
2014-03-17 19:51:07 +00:00
{
sah . hid [ j ] = h - > subID ;
2016-08-23 08:13:52 +03:00
h - > initArmy ( getRandomGenerator ( ) , & sah . army [ j ] ) ;
2014-03-17 19:51:07 +00:00
banned = h - > type - > heroClass ;
}
else
2016-01-28 22:48:51 +03:00
{
2014-03-17 19:51:07 +00:00
sah . hid [ j ] = - 1 ;
2016-01-28 22:48:51 +03:00
}
2014-03-17 19:51:07 +00:00
}
sendAndApply ( & sah ) ;
}
n . res [ elem . first ] = elem . second . resources ;
2016-10-12 17:16:26 +02:00
for ( CGHeroInstance * h : ( elem ) . second . heroes )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( h - > visitedTown )
2014-03-17 19:51:07 +00:00
giveSpells ( h - > visitedTown , h ) ;
NewTurn : : Hero hth ;
hth . id = h - > id ;
2016-08-17 08:53:36 +03:00
auto ti = make_unique < TurnInfo > ( h , 1 ) ;
2015-12-25 11:09:06 +03:00
// TODO: this code executed when bonuses of previous day not yet updated (this happen in NewTurn::applyGs). See issue 2356
2016-08-17 08:53:36 +03:00
hth . move = h - > maxMovePoints ( gs - > map - > getTile ( h - > getPosition ( false ) ) . terType ! = ETerrainType : : WATER , ti . get ( ) ) ;
2016-01-30 10:20:49 +03:00
hth . mana = h - > getManaNewTurn ( ) ;
2014-03-17 19:51:07 +00:00
n . heroes . insert ( hth ) ;
2016-10-12 17:16:26 +02:00
if ( ! firstTurn ) //not first day
2014-03-17 19:51:07 +00:00
{
n . res [ elem . first ] [ Res : : GOLD ] + = h - > valOfBonuses ( Selector : : typeSubtype ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : ESTATES ) ) ; //estates
for ( int k = 0 ; k < GameConstants : : RESOURCE_QUANTITY ; k + + )
{
n . res [ elem . first ] [ k ] + = h - > valOfBonuses ( Bonus : : GENERATE_RESOURCE , k ) ;
}
}
}
}
2016-10-12 17:16:26 +02:00
for ( CGTownInstance * t : gs - > map - > towns )
2014-03-17 19:51:07 +00:00
{
PlayerColor player = t - > tempOwner ;
handleTownEvents ( t , n ) ;
2016-10-12 17:16:26 +02:00
if ( newWeek ) //first day of week
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( t - > hasBuilt ( BuildingID : : PORTAL_OF_SUMMON , ETownType : : DUNGEON ) )
2014-03-17 19:51:07 +00:00
setPortalDwelling ( t , true , ( n . specialWeek = = NewTurn : : PLAGUE ? true : false ) ) ; //set creatures for Portal of Summoning
2016-10-12 17:16:26 +02:00
if ( ! firstTurn )
2014-03-17 19:51:07 +00:00
if ( t - > hasBuilt ( BuildingID : : TREASURY , ETownType : : RAMPART ) & & player < PlayerColor : : PLAYER_LIMIT )
n . res [ player ] [ Res : : GOLD ] + = hadGold . at ( player ) / 10 ; //give 10% of starting gold
if ( ! vstd : : contains ( n . cres , t - > id ) )
{
n . cres [ t - > id ] . tid = t - > id ;
n . cres [ t - > id ] . creatures = t - > creatures ;
}
auto & sac = n . cres . at ( t - > id ) ;
for ( int k = 0 ; k < GameConstants : : CREATURES_PER_TOWN ; k + + ) //creature growths
{
if ( ! t - > creatures . at ( k ) . second . empty ( ) ) // there are creatures at this level
{
ui32 & availableCount = sac . creatures . at ( k ) . first ;
const CCreature * cre = VLC - > creh - > creatures . at ( t - > creatures . at ( k ) . second . back ( ) ) ;
if ( n . specialWeek = = NewTurn : : PLAGUE )
availableCount = t - > creatures . at ( k ) . first / 2 ; //halve their number, no growth
else
{
2016-10-12 17:16:26 +02:00
if ( firstTurn ) //first day of game: use only basic growths
2014-03-17 19:51:07 +00:00
availableCount = cre - > growth ;
else
availableCount + = t - > creatureGrowth ( k ) ;
//Deity of fire week - upgrade both imps and upgrades
if ( n . specialWeek = = NewTurn : : DEITYOFFIRE & & vstd : : contains ( t - > creatures . at ( k ) . second , n . creatureid ) )
availableCount + = 15 ;
2016-10-12 17:16:26 +02:00
if ( cre - > idNumber = = n . creatureid ) //bonus week, effect applies only to identical creatures
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( n . specialWeek = = NewTurn : : DOUBLE_GROWTH )
2014-03-17 19:51:07 +00:00
availableCount * = 2 ;
2016-10-12 17:16:26 +02:00
else if ( n . specialWeek = = NewTurn : : BONUS_GROWTH )
2014-03-17 19:51:07 +00:00
availableCount + = 5 ;
}
}
}
}
}
2016-10-12 17:16:26 +02:00
if ( ! firstTurn & & player < PlayerColor : : PLAYER_LIMIT ) //not the first day and town not neutral
2014-03-17 19:51:07 +00:00
{
2014-04-26 18:23:35 +04:00
n . res [ player ] = n . res [ player ] + t - > dailyIncome ( ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
if ( t - > hasBuilt ( BuildingID : : GRAIL , ETownType : : TOWER ) )
2014-03-17 19:51:07 +00:00
{
// Skyship, probably easier to handle same as Veil of darkness
//do it every new day after veils apply
if ( player ! = PlayerColor : : NEUTRAL ) //do not reveal fow for neutral player
{
FoWChange fw ;
fw . mode = 1 ;
fw . player = player ;
// find all hidden tiles
2016-09-18 11:53:51 +03:00
const auto & fow = getPlayerTeam ( player ) - > fogOfWarMap ;
2014-03-17 19:51:07 +00:00
for ( size_t i = 0 ; i < fow . size ( ) ; i + + )
for ( size_t j = 0 ; j < fow . at ( i ) . size ( ) ; j + + )
for ( size_t k = 0 ; k < fow . at ( i ) . at ( j ) . size ( ) ; k + + )
if ( ! fow . at ( i ) . at ( j ) . at ( k ) )
fw . tiles . insert ( int3 ( i , j , k ) ) ;
sendAndApply ( & fw ) ;
}
}
if ( t - > hasBonusOfType ( Bonus : : DARKNESS ) )
{
2016-10-12 17:16:26 +02:00
for ( auto & player : gs - > players )
2014-06-24 14:50:27 +03:00
{
if ( getPlayerStatus ( player . first ) = = EPlayerStatus : : INGAME & &
getPlayerRelations ( player . first , t - > tempOwner ) = = PlayerRelations : : ENEMIES )
changeFogOfWar ( t - > visitablePos ( ) , t - > getBonusLocalFirst ( Selector : : type ( Bonus : : DARKNESS ) ) - > val , player . first , true ) ;
}
2014-03-17 19:51:07 +00:00
}
}
2016-10-12 17:16:26 +02:00
if ( newMonth )
2014-03-17 19:51:07 +00:00
{
SetAvailableArtifacts saa ;
saa . id = - 1 ;
2016-09-08 00:54:26 +03:00
pickAllowedArtsSet ( saa . arts , getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & saa ) ;
}
sendAndApply ( & n ) ;
2016-10-12 17:16:26 +02:00
if ( newWeek )
2014-03-17 19:51:07 +00:00
{
//spawn wandering monsters
if ( newMonth & & ( n . specialWeek = = NewTurn : : DOUBLE_GROWTH | | n . specialWeek = = NewTurn : : DEITYOFFIRE ) )
{
spawnWanderingMonsters ( n . creatureid ) ;
}
//new week info popup
2016-10-12 17:16:26 +02:00
if ( ! firstTurn )
2014-03-17 19:51:07 +00:00
{
InfoWindow iw ;
switch ( n . specialWeek )
{
case NewTurn : : DOUBLE_GROWTH :
iw . text . addTxt ( MetaString : : ARRAY_TXT , 131 ) ;
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , n . creatureid ) ;
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , n . creatureid ) ;
break ;
case NewTurn : : PLAGUE :
iw . text . addTxt ( MetaString : : ARRAY_TXT , 132 ) ;
break ;
case NewTurn : : BONUS_GROWTH :
iw . text . addTxt ( MetaString : : ARRAY_TXT , 134 ) ;
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , n . creatureid ) ;
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , n . creatureid ) ;
break ;
case NewTurn : : DEITYOFFIRE :
iw . text . addTxt ( MetaString : : ARRAY_TXT , 135 ) ;
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , 42 ) ; //%s imp
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , 42 ) ; //%s imp
iw . text . addReplacement2 ( 15 ) ; //%+d 15
iw . text . addReplacement ( MetaString : : CRE_SING_NAMES , 43 ) ; //%s familiar
iw . text . addReplacement2 ( 15 ) ; //%+d 15
break ;
default :
if ( newMonth )
{
iw . text . addTxt ( MetaString : : ARRAY_TXT , ( 130 ) ) ;
2016-09-08 00:54:26 +03:00
iw . text . addReplacement ( MetaString : : ARRAY_TXT , getRandomGenerator ( ) . nextInt ( 32 , 41 ) ) ;
2014-03-17 19:51:07 +00:00
}
else
{
iw . text . addTxt ( MetaString : : ARRAY_TXT , ( 133 ) ) ;
2016-09-08 00:54:26 +03:00
iw . text . addReplacement ( MetaString : : ARRAY_TXT , getRandomGenerator ( ) . nextInt ( 43 , 57 ) ) ;
2014-03-17 19:51:07 +00:00
}
}
for ( auto & elem : gs - > players )
{
iw . player = elem . first ;
sendAndApply ( & iw ) ;
}
}
}
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Info about turn %d has been sent! " , n . day ) ;
2014-03-17 19:51:07 +00:00
handleTimeEvents ( ) ;
//call objects
2016-10-12 17:16:26 +02:00
for ( auto & elem : gs - > map - > objects )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( elem )
2016-09-09 20:30:36 +03:00
elem - > newTurn ( getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
}
synchronizeArtifactHandlerLists ( ) ; //new day events may have changed them. TODO better of managing that
}
void CGameHandler : : run ( bool resume )
{
LOG_TRACE_PARAMS ( logGlobal , " resume=%d " , resume ) ;
using namespace boost : : posix_time ;
2016-10-12 17:16:26 +02:00
for ( CConnection * cc : conns )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! resume )
2014-03-17 19:51:07 +00:00
{
( * cc ) < < gs - > initialOpts ; // gs->scenarioOps
}
std : : set < PlayerColor > players ;
( * cc ) > > players ; //how many players will be handled at that client
2016-03-12 04:41:27 +03:00
std : : stringstream sbuffer ;
sbuffer < < " Connection " < < cc - > connectionID < < " will handle " < < players . size ( ) < < " player: " ;
2016-10-12 17:16:26 +02:00
for ( PlayerColor color : players )
2014-03-17 19:51:07 +00:00
{
2016-03-12 04:41:27 +03:00
sbuffer < < color < < " " ;
2014-03-17 19:51:07 +00:00
{
boost : : unique_lock < boost : : recursive_mutex > lock ( gsm ) ;
2017-06-03 08:25:10 +03:00
if ( ! color . isSpectator ( ) ) // there can be more than one spectator
connections [ color ] = cc ;
2014-03-17 19:51:07 +00:00
}
}
2016-08-30 01:11:54 +03:00
logGlobal - > info ( sbuffer . str ( ) ) ;
2014-03-17 19:51:07 +00:00
cc - > addStdVecItems ( gs ) ;
cc - > enableStackSendingByID ( ) ;
cc - > disableSmartPointerSerialization ( ) ;
}
2016-10-12 17:16:26 +02:00
for ( auto & elem : conns )
2014-03-17 19:51:07 +00:00
{
std : : set < PlayerColor > pom ;
2016-10-12 17:16:26 +02:00
for ( auto j = connections . cbegin ( ) ; j ! = connections . cend ( ) ; j + + )
if ( j - > second = = elem )
2014-03-17 19:51:07 +00:00
pom . insert ( j - > first ) ;
2014-08-04 20:33:59 +02:00
boost : : thread ( std : : bind ( & CGameHandler : : handleConnection , this , pom , std : : ref ( * elem ) ) ) ;
2014-03-17 19:51:07 +00:00
}
auto playerTurnOrder = generatePlayerTurnOrder ( ) ;
2017-06-04 11:59:26 +03:00
while ( ! serverShuttingDown )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! resume ) newTurn ( ) ;
2014-03-17 19:51:07 +00:00
std : : list < PlayerColor > : : iterator it ;
2016-10-12 17:16:26 +02:00
if ( resume )
2014-03-17 19:51:07 +00:00
{
it = std : : find ( playerTurnOrder . begin ( ) , playerTurnOrder . end ( ) , gs - > currentPlayer ) ;
}
else
{
it = playerTurnOrder . begin ( ) ;
}
resume = false ;
2015-10-24 16:21:30 +02:00
for ( ; it ! = playerTurnOrder . end ( ) ; it + + )
2014-03-17 19:51:07 +00:00
{
auto playerColor = * it ;
2015-10-24 16:21:30 +02:00
2015-10-24 17:02:00 +02:00
PlayerState * playerState = & gs - > players [ playerColor ] ; //can't copy CBonusSystemNode by value
if ( playerState - > status = = EPlayerStatus : : INGAME )
2014-03-17 19:51:07 +00:00
{
2015-10-24 16:21:30 +02:00
//if player runs out of time, he shouldn't get the turn (especially AI)
checkVictoryLossConditionsForAll ( ) ;
if ( gs - > players [ playerColor ] . status ! = EPlayerStatus : : INGAME )
{ //player lost at the beginning of his turn
continue ;
}
2015-10-24 17:02:00 +02:00
else //give normal turn
{
states . setFlag ( playerColor , & PlayerStatus : : makingTurn , true ) ;
2014-03-17 19:51:07 +00:00
2015-10-24 17:02:00 +02:00
YourTurn yt ;
yt . player = playerColor ;
//Change local daysWithoutCastle counter for local interface message //TODO: needed?
yt . daysWithoutCastle = playerState - > daysWithoutCastle ;
applyAndSend ( & yt ) ;
2014-03-17 19:51:07 +00:00
2015-10-24 17:02:00 +02:00
//wait till turn is done
boost : : unique_lock < boost : : mutex > lock ( states . mx ) ;
2017-06-04 11:59:26 +03:00
while ( states . players . at ( playerColor ) . makingTurn & & ! serverShuttingDown )
2015-10-24 17:02:00 +02:00
{
static time_duration p = milliseconds ( 100 ) ;
states . cv . timed_wait ( lock , p ) ;
}
2014-03-17 19:51:07 +00:00
}
}
}
2015-08-22 16:22:10 +03:00
//additional check that game is not finished
bool activePlayer = false ;
2016-10-12 17:16:26 +02:00
for ( auto player : playerTurnOrder )
2015-08-22 16:22:10 +03:00
{
2016-10-12 17:16:26 +02:00
if ( gs - > players [ player ] . status = = EPlayerStatus : : INGAME )
2015-08-22 16:22:10 +03:00
activePlayer = true ;
}
2016-10-12 17:16:26 +02:00
if ( ! activePlayer )
2017-06-04 11:59:26 +03:00
serverShuttingDown = true ;
2014-03-17 19:51:07 +00:00
}
while ( conns . size ( ) & & ( * conns . begin ( ) ) - > isOpen ( ) )
boost : : this_thread : : sleep ( boost : : posix_time : : milliseconds ( 5 ) ) ; //give time client to close socket
}
std : : list < PlayerColor > CGameHandler : : generatePlayerTurnOrder ( ) const
{
// Generate player turn order
std : : list < PlayerColor > playerTurnOrder ;
2016-10-12 17:16:26 +02:00
for ( const auto & player : gs - > players ) // add human players first
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( player . second . human )
2014-03-17 19:51:07 +00:00
playerTurnOrder . push_back ( player . first ) ;
}
2016-10-12 17:16:26 +02:00
for ( const auto & player : gs - > players ) // then add non-human players
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! player . second . human )
2014-03-17 19:51:07 +00:00
playerTurnOrder . push_back ( player . first ) ;
}
2015-12-04 01:06:02 +02:00
return playerTurnOrder ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : setupBattle ( int3 tile , const CArmedInstance * armies [ 2 ] , const CGHeroInstance * heroes [ 2 ] , bool creatureBank , const CGTownInstance * town )
2014-03-17 19:51:07 +00:00
{
battleResult . set ( nullptr ) ;
2016-09-18 11:53:51 +03:00
const auto t = getTile ( tile ) ;
2016-08-29 04:46:01 +03:00
ETerrainType terrain = t - > terType ;
2016-10-12 17:16:26 +02:00
if ( gs - > map - > isCoastalTile ( tile ) ) //coastal tile is always ground
2016-08-29 04:46:01 +03:00
terrain = ETerrainType : : SAND ;
2016-09-07 20:00:20 +03:00
BFieldType terType = gs - > battleGetBattlefieldType ( tile , getRandomGenerator ( ) ) ;
2016-08-29 04:46:01 +03:00
if ( heroes [ 0 ] & & heroes [ 0 ] - > boat & & heroes [ 1 ] & & heroes [ 1 ] - > boat )
terType = BFieldType : : SHIP_TO_SHIP ;
2014-03-17 19:51:07 +00:00
//send info about battles
BattleStart bs ;
2016-08-29 04:46:01 +03:00
bs . info = BattleInfo : : setupBattle ( tile , terrain , terType , armies , heroes , creatureBank , town ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & bs ) ;
}
2016-02-09 10:59:33 +03:00
void CGameHandler : : checkBattleStateChanges ( )
2014-03-17 19:51:07 +00:00
{
2016-02-09 10:59:33 +03:00
//check if drawbridge state need to be changes
2016-10-12 17:16:26 +02:00
if ( battleGetSiegeLevel ( ) > 0 )
2016-02-13 17:40:31 +03:00
updateGateState ( ) ;
2016-02-09 10:59:33 +03:00
//check if battle ended
2016-10-12 17:16:26 +02:00
if ( auto result = battleIsFinished ( ) )
2014-03-17 19:51:07 +00:00
{
setBattleResult ( BattleResult : : NORMAL , * result ) ;
}
}
2016-03-13 13:24:17 +03:00
void CGameHandler : : giveSpells ( const CGTownInstance * t , const CGHeroInstance * h )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! h - > hasSpellbook ( ) )
2014-03-17 19:51:07 +00:00
return ; //hero hasn't spellbook
ChangeSpells cs ;
cs . hid = h - > id ;
cs . learn = true ;
2016-10-12 17:16:26 +02:00
if ( t - > hasBuilt ( BuildingID : : GRAIL , ETownType : : CONFLUX ) & & t - > hasBuilt ( BuildingID : : MAGES_GUILD_1 ) )
2014-03-17 19:51:07 +00:00
{
2016-03-13 13:24:17 +03:00
// Aurora Borealis give spells of all levels even if only level 1 mages guild built
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < h - > getSecSkillLevel ( SecondarySkill : : WISDOM ) + 2 ; i + + )
2014-03-17 19:51:07 +00:00
{
std : : vector < SpellID > spells ;
2016-03-13 13:24:17 +03:00
getAllowedSpells ( spells , i + 1 ) ;
2016-10-12 17:16:26 +02:00
for ( auto & spell : spells )
2014-03-17 19:51:07 +00:00
cs . spells . insert ( spell ) ;
}
2016-03-13 13:24:17 +03:00
}
else
{
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < std : : min ( t - > mageGuildLevel ( ) , h - > getSecSkillLevel ( SecondarySkill : : WISDOM ) + 2 ) ; i + + )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
for ( int j = 0 ; j < t - > spellsAtLevel ( i + 1 , true ) & & j < t - > spells . at ( i ) . size ( ) ; j + + )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( h - > spells , t - > spells . at ( i ) . at ( j ) ) )
2014-03-17 19:51:07 +00:00
cs . spells . insert ( t - > spells . at ( i ) . at ( j ) ) ;
}
}
}
2016-10-12 17:16:26 +02:00
if ( ! cs . spells . empty ( ) )
2014-03-17 19:51:07 +00:00
sendAndApply ( & cs ) ;
}
void CGameHandler : : setBlockVis ( ObjectInstanceID objid , bool bv )
{
2016-09-09 17:42:16 +03:00
SetObjectProperty sop ( objid , ObjProperty : : BLOCKVIS , bv ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & sop ) ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : removeObject ( const CGObjectInstance * obj )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! obj | | ! getObj ( obj - > id ) )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Something wrong, that object already has been removed or hasn't existed! " ) ;
2014-03-17 19:51:07 +00:00
return false ;
}
RemoveObject ro ;
ro . id = obj - > id ;
sendAndApply ( & ro ) ;
checkVictoryLossConditionsForAll ( ) ; //eg if monster escaped (removing objs after battle is done dircetly by endBattle, not this function)
return true ;
}
void CGameHandler : : setAmount ( ObjectInstanceID objid , ui32 val )
{
2016-09-09 17:42:16 +03:00
SetObjectProperty sop ( objid , ObjProperty : : PRIMARY_STACK_COUNT , val ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & sop ) ;
}
2017-07-15 14:08:20 +03:00
bool CGameHandler : : moveHero ( ObjectInstanceID hid , int3 dst , ui8 teleporting , bool transit , PlayerColor asker )
2014-03-17 19:51:07 +00:00
{
const CGHeroInstance * h = getHero ( hid ) ;
2016-10-12 17:16:26 +02:00
// not turn of that hero or player can't simply teleport hero (at least not with this function)
if ( ! h | | ( asker ! = PlayerColor : : NEUTRAL & & ( teleporting | | h - > getOwner ( ) ! = gs - > currentPlayer ) ) )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Illegal call to move hero! " ) ;
2014-03-17 19:51:07 +00:00
return false ;
}
2017-08-11 20:03:05 +03:00
logGlobal - > trace ( " Player %d (%s) wants to move hero %d from %s to %s " , asker , asker . getStr ( ) , hid . getNum ( ) , h - > pos . toString ( ) , dst . toString ( ) ) ;
2015-12-03 17:20:03 +03:00
const int3 hmpos = CGHeroInstance : : convertPosition ( dst , false ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( ! gs - > map - > isInTheMap ( hmpos ) )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Destination tile is outside the map! " ) ;
2014-03-17 19:51:07 +00:00
return false ;
}
2016-09-18 11:53:51 +03:00
const TerrainTile t = * getTile ( hmpos ) ;
2014-03-17 19:51:07 +00:00
const int3 guardPos = gs - > guardingCreaturePosition ( hmpos ) ;
const bool embarking = ! h - > boat & & ! t . visitableObjects . empty ( ) & & t . visitableObjects . back ( ) - > ID = = Obj : : BOAT ;
const bool disembarking = h - > boat & & t . terType ! = ETerrainType : : WATER & & ! t . blocked ;
//result structure for start - movement failed, no move points used
TryMoveHero tmh ;
tmh . id = hid ;
tmh . start = h - > pos ;
tmh . end = dst ;
tmh . result = TryMoveHero : : FAILED ;
tmh . movePoints = h - > movement ;
//check if destination tile is available
2016-08-17 08:53:36 +03:00
auto ti = make_unique < TurnInfo > ( h ) ;
2015-11-12 14:04:33 +03:00
const bool canFly = ti - > hasBonusOfType ( Bonus : : FLYING_MOVEMENT ) ;
const bool canWalkOnSea = ti - > hasBonusOfType ( Bonus : : WATER_WALKING ) ;
2016-08-17 08:53:36 +03:00
const int cost = CPathfinderHelper : : getMovementCost ( h , h - > getPosition ( ) , hmpos , nullptr , nullptr , h - > movement , ti . get ( ) ) ;
2014-03-17 19:51:07 +00:00
//it's a rock or blocked and not visitable tile
//OR hero is on land and dest is water and (there is not present only one object - boat)
2016-10-12 17:16:26 +02:00
if ( ( ( t . terType = = ETerrainType : : ROCK | | ( t . blocked & & ! t . visitable & & ! canFly ) )
2014-03-17 19:51:07 +00:00
& & complain ( " Cannot move hero, destination tile is blocked! " ) )
2015-11-09 19:57:26 +03:00
| | ( ( ! h - > boat & & ! canWalkOnSea & & ! canFly & & t . terType = = ETerrainType : : WATER & & ( t . visitableObjects . size ( ) < 1 | | ( t . visitableObjects . back ( ) - > ID ! = Obj : : BOAT & & t . visitableObjects . back ( ) - > ID ! = Obj : : HERO ) ) ) //hero is not on boat/water walking and dst water tile doesn't contain boat/hero (objs visitable from land) -> we test back cause boat may be on top of another object (#276)
2014-03-17 19:51:07 +00:00
& & complain ( " Cannot move hero, destination tile is on water! " ) )
| | ( ( h - > boat & & t . terType ! = ETerrainType : : WATER & & t . blocked )
& & complain ( " Cannot disembark hero, tile is blocked! " ) )
2016-10-12 17:16:26 +02:00
| | ( ( distance ( h - > pos , dst ) > = 1.5 & & ! teleporting )
2014-03-17 19:51:07 +00:00
& & complain ( " Tiles are not neighboring! " ) )
2016-10-12 17:16:26 +02:00
| | ( ( h - > inTownGarrison )
2014-03-17 19:51:07 +00:00
& & complain ( " Can not move garrisoned hero! " ) )
| | ( ( h - > movement < cost & & dst ! = h - > pos & & ! teleporting )
& & complain ( " Hero doesn't have any movement points left! " ) )
2015-11-09 19:57:26 +03:00
| | ( ( transit & & ! canFly & & ! CGTeleport : : isTeleport ( t . topVisitableObj ( ) ) )
2015-11-05 10:50:47 +03:00
& & complain ( " Hero cannot transit over this tile! " ) )
2014-03-17 19:51:07 +00:00
/*|| (states.checkFlag(h->tempOwner, &PlayerStatus::engagedIntoBattle)
& & complain ( " Cannot move hero during the battle " ) ) */ )
{
//send info about movement failure
sendAndApply ( & tmh ) ;
return false ;
}
//several generic blocks of code
// should be called if hero changes tile but before applying TryMoveHero package
auto leaveTile = [ & ] ( )
{
2016-10-12 17:16:26 +02:00
for ( CGObjectInstance * obj : gs - > map - > getTile ( int3 ( h - > pos . x - 1 , h - > pos . y , h - > pos . z ) ) . visitableObjects )
2014-03-17 19:51:07 +00:00
{
obj - > onHeroLeave ( h ) ;
}
2016-01-31 18:01:58 +03:00
this - > getTilesInRange ( tmh . fowRevealed , h - > getSightCenter ( ) + ( tmh . end - tmh . start ) , h - > getSightRadius ( ) , h - > tempOwner , 1 ) ;
2014-03-17 19:51:07 +00:00
} ;
auto doMove = [ & ] ( TryMoveHero : : EResult result , EGuardLook lookForGuards ,
EVisitDest visitDest , ELEaveTile leavingTile ) - > bool
{
2017-08-11 20:03:05 +03:00
LOG_TRACE_PARAMS ( logGlobal , " Hero %s starts movement from %s to %s " , h - > name % tmh . start . toString ( ) % tmh . end . toString ( ) ) ;
2014-03-17 19:51:07 +00:00
2017-06-06 07:53:51 +03:00
auto moveQuery = std : : make_shared < CHeroMovementQuery > ( this , tmh , h ) ;
2014-03-17 19:51:07 +00:00
queries . addQuery ( moveQuery ) ;
2016-10-12 17:16:26 +02:00
if ( leavingTile = = LEAVING_TILE )
2014-03-17 19:51:07 +00:00
leaveTile ( ) ;
tmh . result = result ;
sendAndApply ( & tmh ) ;
2015-02-22 20:12:49 +03:00
if ( visitDest = = VISIT_DEST & & t . topVisitableObj ( ) & & t . topVisitableObj ( ) - > id = = h - > id )
{ // Hero should be always able to visit any object he staying on even if there guards around
visitObjectOnTile ( t , h ) ;
}
2016-10-12 17:16:26 +02:00
else if ( lookForGuards = = CHECK_FOR_GUARDS & & this - > isInTheMap ( guardPos ) )
2014-03-17 19:51:07 +00:00
{
2017-08-05 16:09:29 +03:00
tmh . attackedFrom = boost : : make_optional ( guardPos ) ;
2014-03-17 19:51:07 +00:00
const TerrainTile & guardTile = * gs - > getTile ( guardPos ) ;
objectVisited ( guardTile . visitableObjects . back ( ) , h ) ;
moveQuery - > visitDestAfterVictory = visitDest = = VISIT_DEST ;
}
2016-10-12 17:16:26 +02:00
else if ( visitDest = = VISIT_DEST )
2014-03-17 19:51:07 +00:00
{
2015-11-05 10:02:13 +03:00
visitObjectOnTile ( t , h ) ;
2014-03-17 19:51:07 +00:00
}
queries . popIfTop ( moveQuery ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Hero %s ends movement " , h - > name ) ;
2014-03-17 19:51:07 +00:00
return result ! = TryMoveHero : : FAILED ;
} ;
//interaction with blocking object (like resources)
auto blockingVisit = [ & ] ( ) - > bool
{
2016-10-12 17:16:26 +02:00
for ( CGObjectInstance * obj : t . visitableObjects )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( obj ! = h & & obj - > blockVisit & & ! obj - > passableFor ( h - > tempOwner ) )
2014-03-17 19:51:07 +00:00
{
return doMove ( TryMoveHero : : BLOCKING_VISIT , this - > IGNORE_GUARDS , VISIT_DEST , REMAINING_ON_TILE ) ;
//this-> is needed for MVS2010 to recognize scope (?)
}
}
return false ;
} ;
2016-10-12 17:16:26 +02:00
if ( ! transit & & embarking )
2014-03-17 19:51:07 +00:00
{
2016-08-17 08:53:36 +03:00
tmh . movePoints = h - > movementPointsAfterEmbark ( h - > movement , cost , false , ti . get ( ) ) ;
2014-03-17 19:51:07 +00:00
return doMove ( TryMoveHero : : EMBARK , IGNORE_GUARDS , DONT_VISIT_DEST , LEAVING_TILE ) ;
2015-12-15 19:13:55 +03:00
// In H3 embark ignore guards
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
if ( disembarking )
2014-03-17 19:51:07 +00:00
{
2016-08-17 08:53:36 +03:00
tmh . movePoints = h - > movementPointsAfterEmbark ( h - > movement , cost , true , ti . get ( ) ) ;
2014-03-17 19:51:07 +00:00
return doMove ( TryMoveHero : : DISEMBARK , CHECK_FOR_GUARDS , VISIT_DEST , LEAVING_TILE ) ;
}
2016-10-12 17:16:26 +02:00
if ( teleporting )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( blockingVisit ( ) ) // e.g. hero on the other side of teleporter
2014-03-17 19:51:07 +00:00
return true ;
doMove ( TryMoveHero : : TELEPORTATION , IGNORE_GUARDS , DONT_VISIT_DEST , LEAVING_TILE ) ;
// visit town for town portal \ castle gates
// do not use generic visitObjectOnTile to avoid double-teleporting
// if this moveHero call was triggered by teleporter
if ( ! t . visitableObjects . empty ( ) )
{
if ( CGTownInstance * town = dynamic_cast < CGTownInstance * > ( t . visitableObjects . back ( ) ) )
town - > onHeroVisit ( h ) ;
}
return true ;
}
//still here? it is standard movement!
{
tmh . movePoints = h - > movement > = cost
? h - > movement - cost
: 0 ;
2015-11-05 10:02:13 +03:00
EGuardLook lookForGuards = CHECK_FOR_GUARDS ;
EVisitDest visitDest = VISIT_DEST ;
2016-10-12 17:16:26 +02:00
if ( transit )
2015-11-05 10:02:13 +03:00
{
2016-10-12 17:16:26 +02:00
if ( CGTeleport : : isTeleport ( t . topVisitableObj ( ) ) )
2015-11-05 10:02:13 +03:00
visitDest = DONT_VISIT_DEST ;
2016-10-12 17:16:26 +02:00
if ( canFly )
2015-11-05 10:02:13 +03:00
{
lookForGuards = IGNORE_GUARDS ;
visitDest = DONT_VISIT_DEST ;
}
}
2016-10-12 17:16:26 +02:00
else if ( blockingVisit ( ) )
2014-03-17 19:51:07 +00:00
return true ;
2015-11-05 10:02:13 +03:00
doMove ( TryMoveHero : : SUCCESS , lookForGuards , visitDest , LEAVING_TILE ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
}
2017-07-15 14:08:20 +03:00
bool CGameHandler : : teleportHero ( ObjectInstanceID hid , ObjectInstanceID dstid , ui8 source , PlayerColor asker )
2014-03-17 19:51:07 +00:00
{
const CGHeroInstance * h = getHero ( hid ) ;
const CGTownInstance * t = getTown ( dstid ) ;
2016-10-12 17:16:26 +02:00
if ( ! h | | ! t | | h - > getOwner ( ) ! = gs - > currentPlayer )
2016-11-27 22:50:37 +03:00
COMPLAIN_RET ( " Invalid call to teleportHero! " ) ;
2014-03-17 19:51:07 +00:00
const CGTownInstance * from = h - > visitedTown ;
2016-10-12 17:16:26 +02:00
if ( ( ( h - > getOwner ( ) ! = t - > getOwner ( ) )
2014-03-17 19:51:07 +00:00
& & complain ( " Cannot teleport hero to another player " ) )
| | ( ( ! from | | ! from - > hasBuilt ( BuildingID : : CASTLE_GATE , ETownType : : INFERNO ) )
& & complain ( " Hero must be in town with Castle gate for teleporting " ) )
| | ( ! t - > hasBuilt ( BuildingID : : CASTLE_GATE , ETownType : : INFERNO )
& & complain ( " Cannot teleport hero to town without Castle gate in it " ) ) )
return false ;
int3 pos = t - > visitablePos ( ) ;
pos + = h - > getVisitableOffset ( ) ;
moveHero ( hid , pos , 1 ) ;
return true ;
}
void CGameHandler : : setOwner ( const CGObjectInstance * obj , PlayerColor owner )
{
PlayerColor oldOwner = getOwner ( obj - > id ) ;
2016-09-09 17:42:16 +03:00
SetObjectProperty sop ( obj - > id , ObjProperty : : OWNER , owner . getNum ( ) ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & sop ) ;
2014-10-02 19:43:46 +04:00
std : : set < PlayerColor > playerColors = { owner , oldOwner } ;
2014-03-17 19:51:07 +00:00
checkVictoryLossConditions ( playerColors ) ;
2016-11-27 22:50:37 +03:00
const CGTownInstance * town = dynamic_cast < const CGTownInstance * > ( obj ) ;
if ( town ) //town captured
2014-03-17 19:51:07 +00:00
{
2015-10-24 18:03:00 +02:00
if ( owner < PlayerColor : : PLAYER_LIMIT ) //new owner is real player
{
if ( town - > hasBuilt ( BuildingID : : PORTAL_OF_SUMMON , ETownType : : DUNGEON ) )
setPortalDwelling ( town , true , false ) ;
}
2014-03-17 19:51:07 +00:00
2015-10-24 18:03:00 +02:00
if ( oldOwner < PlayerColor : : PLAYER_LIMIT ) //old owner is real player
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( getPlayer ( oldOwner ) - > towns . empty ( ) ) //previous player lost last last town
2015-10-24 18:03:00 +02:00
{
InfoWindow iw ;
iw . player = oldOwner ;
2015-11-09 00:54:59 +03:00
iw . text . addTxt ( MetaString : : GENERAL_TXT , 6 ) ; //%s, you have lost your last town. If you do not conquer another town in the next week, you will be eliminated.
iw . text . addReplacement ( MetaString : : COLOR , oldOwner . getNum ( ) ) ;
2015-10-24 18:03:00 +02:00
sendAndApply ( & iw ) ;
}
2014-03-17 19:51:07 +00:00
}
}
2016-09-18 11:53:51 +03:00
const PlayerState * p = getPlayer ( owner ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( ( obj - > ID = = Obj : : CREATURE_GENERATOR1 | | obj - > ID = = Obj : : CREATURE_GENERATOR4 ) & & p & & p - > dwellings . size ( ) = = 1 ) //first dwelling captured
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
for ( const CGTownInstance * t : getPlayer ( owner ) - > towns )
2014-03-17 19:51:07 +00:00
{
if ( t - > hasBuilt ( BuildingID : : PORTAL_OF_SUMMON , ETownType : : DUNGEON ) )
setPortalDwelling ( t ) ; //set initial creatures for all portals of summoning
}
}
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : showBlockingDialog ( BlockingDialog * iw )
2014-03-17 19:51:07 +00:00
{
2017-06-06 07:53:51 +03:00
auto dialogQuery = std : : make_shared < CBlockingDialogQuery > ( this , * iw ) ;
2014-03-17 19:51:07 +00:00
queries . addQuery ( dialogQuery ) ;
iw - > queryID = dialogQuery - > queryID ;
sendToAllClients ( iw ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : showTeleportDialog ( TeleportDialog * iw )
2015-03-08 16:52:50 +03:00
{
2017-06-06 07:53:51 +03:00
auto dialogQuery = std : : make_shared < CTeleportDialogQuery > ( this , * iw ) ;
2015-03-08 16:52:50 +03:00
queries . addQuery ( dialogQuery ) ;
iw - > queryID = dialogQuery - > queryID ;
2014-03-17 19:51:07 +00:00
sendToAllClients ( iw ) ;
}
void CGameHandler : : giveResource ( PlayerColor player , Res : : ERes which , int val ) //TODO: cap according to Bersy's suggestion
{
2016-10-12 17:16:26 +02:00
if ( ! val ) return ; //don't waste time on empty call
2016-11-26 15:14:43 +03:00
TResources resources ;
resources . at ( which ) = val ;
giveResources ( player , resources ) ;
2014-03-17 19:51:07 +00:00
}
void CGameHandler : : giveResources ( PlayerColor player , TResources resources )
{
2016-11-26 15:14:43 +03:00
SetResources sr ;
sr . abs = false ;
sr . player = player ;
sr . res = resources ;
2016-11-28 05:19:44 +03:00
sendAndApply ( & sr ) ;
2014-03-17 19:51:07 +00:00
}
void CGameHandler : : giveCreatures ( const CArmedInstance * obj , const CGHeroInstance * h , const CCreatureSet & creatures , bool remove )
{
COMPLAIN_RET_IF ( ! creatures . stacksCount ( ) , " Strange, giveCreatures called without args! " ) ;
COMPLAIN_RET_IF ( obj - > stacksCount ( ) , " Cannot give creatures from not-cleared object! " ) ;
COMPLAIN_RET_IF ( creatures . stacksCount ( ) > GameConstants : : ARMY_SIZE , " Too many stacks to give! " ) ;
//first we move creatures to give to make them army of object-source
for ( auto & elem : creatures . Slots ( ) )
{
addToSlot ( StackLocation ( obj , obj - > getSlotFor ( elem . second - > type ) ) , elem . second - > type , elem . second - > count ) ;
}
tryJoiningArmy ( obj , h , remove , true ) ;
}
void CGameHandler : : takeCreatures ( ObjectInstanceID objid , const std : : vector < CStackBasicDescriptor > & creatures )
{
std : : vector < CStackBasicDescriptor > cres = creatures ;
if ( cres . size ( ) < = 0 )
return ;
const CArmedInstance * obj = static_cast < const CArmedInstance * > ( getObj ( objid ) ) ;
2016-10-12 17:16:26 +02:00
for ( CStackBasicDescriptor & sbd : cres )
2014-03-17 19:51:07 +00:00
{
TQuantity collected = 0 ;
while ( collected < sbd . count )
{
bool foundSth = false ;
2016-10-12 17:16:26 +02:00
for ( auto i = obj - > Slots ( ) . begin ( ) ; i ! = obj - > Slots ( ) . end ( ) ; i + + )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( i - > second - > type = = sbd . type )
2014-03-17 19:51:07 +00:00
{
TQuantity take = std : : min ( sbd . count - collected , i - > second - > count ) ; //collect as much cres as we can
changeStackCount ( StackLocation ( obj , i - > first ) , - take , false ) ;
collected + = take ;
foundSth = true ;
break ;
}
}
2016-10-12 17:16:26 +02:00
if ( ! foundSth ) //we went through the whole loop and haven't found appropriate cres
2014-03-17 19:51:07 +00:00
{
complain ( " Unexpected failure during taking creatures! " ) ;
return ;
}
}
}
}
void CGameHandler : : showCompInfo ( ShowInInfobox * comp )
{
sendToAllClients ( comp ) ;
}
void CGameHandler : : heroVisitCastle ( const CGTownInstance * obj , const CGHeroInstance * hero )
{
HeroVisitCastle vc ;
vc . hid = hero - > id ;
vc . tid = obj - > id ;
vc . flags | = 1 ;
sendAndApply ( & vc ) ;
vistiCastleObjects ( obj , hero ) ;
giveSpells ( obj , hero ) ;
checkVictoryLossConditionsForPlayer ( hero - > tempOwner ) ; //transported artifact?
}
void CGameHandler : : vistiCastleObjects ( const CGTownInstance * t , const CGHeroInstance * h )
{
std : : vector < CGTownBuilding * > : : const_iterator i ;
for ( i = t - > bonusingBuildings . begin ( ) ; i ! = t - > bonusingBuildings . end ( ) ; i + + )
( * i ) - > onHeroVisit ( h ) ;
}
void CGameHandler : : stopHeroVisitCastle ( const CGTownInstance * obj , const CGHeroInstance * hero )
{
HeroVisitCastle vc ;
vc . hid = hero - > id ;
vc . tid = obj - > id ;
sendAndApply ( & vc ) ;
}
void CGameHandler : : removeArtifact ( const ArtifactLocation & al )
{
EraseArtifact ea ;
ea . al = al ;
sendAndApply ( & ea ) ;
}
void CGameHandler : : startBattlePrimary ( const CArmedInstance * army1 , const CArmedInstance * army2 , int3 tile ,
const CGHeroInstance * hero1 , const CGHeroInstance * hero2 , bool creatureBank ,
const CGTownInstance * town ) //use hero=nullptr for no hero
{
engageIntoBattle ( army1 - > tempOwner ) ;
engageIntoBattle ( army2 - > tempOwner ) ;
static const CArmedInstance * armies [ 2 ] ;
armies [ 0 ] = army1 ;
armies [ 1 ] = army2 ;
static const CGHeroInstance * heroes [ 2 ] ;
heroes [ 0 ] = hero1 ;
heroes [ 1 ] = hero2 ;
setupBattle ( tile , armies , heroes , creatureBank , town ) ; //initializes stacks, places creatures on battlefield, blocks and informs player interfaces
2017-06-06 07:53:51 +03:00
auto battleQuery = std : : make_shared < CBattleQuery > ( this , gs - > curB ) ;
2014-03-17 19:51:07 +00:00
queries . addQuery ( battleQuery ) ;
boost : : thread ( & CGameHandler : : runBattle , this ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : startBattleI ( const CArmedInstance * army1 , const CArmedInstance * army2 , int3 tile , bool creatureBank )
2014-03-17 19:51:07 +00:00
{
startBattlePrimary ( army1 , army2 , tile ,
army1 - > ID = = Obj : : HERO ? static_cast < const CGHeroInstance * > ( army1 ) : nullptr ,
army2 - > ID = = Obj : : HERO ? static_cast < const CGHeroInstance * > ( army2 ) : nullptr ,
creatureBank ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : startBattleI ( const CArmedInstance * army1 , const CArmedInstance * army2 , bool creatureBank )
2014-03-17 19:51:07 +00:00
{
startBattleI ( army1 , army2 , army2 - > visitablePos ( ) , creatureBank ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : changeSpells ( const CGHeroInstance * hero , bool give , const std : : set < SpellID > & spells )
2014-03-17 19:51:07 +00:00
{
ChangeSpells cs ;
cs . hid = hero - > id ;
cs . spells = spells ;
cs . learn = give ;
sendAndApply ( & cs ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : sendMessageTo ( CConnection & c , const std : : string & message )
2014-03-17 19:51:07 +00:00
{
SystemMessage sm ;
sm . text = message ;
boost : : unique_lock < boost : : mutex > lock ( * c . wmx ) ;
c < < & sm ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : giveHeroBonus ( GiveBonus * bonus )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( bonus ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : setMovePoints ( SetMovePoints * smp )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( smp ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : setManaPoints ( ObjectInstanceID hid , int val )
2014-03-17 19:51:07 +00:00
{
SetMana sm ;
sm . hid = hid ;
sm . val = val ;
2014-11-26 13:30:55 +03:00
sm . absolute = true ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & sm ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : giveHero ( ObjectInstanceID id , PlayerColor player )
2014-03-17 19:51:07 +00:00
{
GiveHero gh ;
gh . id = id ;
gh . player = player ;
sendAndApply ( & gh ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : changeObjPos ( ObjectInstanceID objid , int3 newPos , ui8 flags )
2014-03-17 19:51:07 +00:00
{
ChangeObjPos cop ;
cop . objid = objid ;
cop . nPos = newPos ;
cop . flags = flags ;
sendAndApply ( & cop ) ;
}
void CGameHandler : : useScholarSkill ( ObjectInstanceID fromHero , ObjectInstanceID toHero )
{
const CGHeroInstance * h1 = getHero ( fromHero ) ;
const CGHeroInstance * h2 = getHero ( toHero ) ;
2016-10-12 17:16:26 +02:00
if ( h1 - > getSecSkillLevel ( SecondarySkill : : SCHOLAR ) < h2 - > getSecSkillLevel ( SecondarySkill : : SCHOLAR ) )
2014-03-17 19:51:07 +00:00
{
std : : swap ( h1 , h2 ) ; //1st hero need to have higher scholar level for correct message
std : : swap ( fromHero , toHero ) ;
}
int ScholarLevel = h1 - > getSecSkillLevel ( SecondarySkill : : SCHOLAR ) ; //heroes can trade up to this level
2016-10-12 17:16:26 +02:00
if ( ! ScholarLevel | | ! h1 - > hasSpellbook ( ) | | ! h2 - > hasSpellbook ( ) )
2014-03-17 19:51:07 +00:00
return ; //no scholar skill or no spellbook
int h1Lvl = std : : min ( ScholarLevel + 1 , h1 - > getSecSkillLevel ( SecondarySkill : : WISDOM ) + 2 ) ,
h2Lvl = std : : min ( ScholarLevel + 1 , h2 - > getSecSkillLevel ( SecondarySkill : : WISDOM ) + 2 ) ; //heroes can receive this levels
ChangeSpells cs1 ;
cs1 . learn = true ;
cs1 . hid = toHero ; //giving spells to first hero
2016-10-12 17:16:26 +02:00
for ( auto it : h1 - > spells )
if ( h2Lvl > = it . toSpell ( ) - > level & & ! vstd : : contains ( h2 - > spells , it ) ) //hero can learn it and don't have it yet
2014-03-17 19:51:07 +00:00
cs1 . spells . insert ( it ) ; //spell to learn
ChangeSpells cs2 ;
cs2 . learn = true ;
cs2 . hid = fromHero ;
2016-10-12 17:16:26 +02:00
for ( auto it : h2 - > spells )
if ( h1Lvl > = it . toSpell ( ) - > level & & ! vstd : : contains ( h1 - > spells , it ) )
2014-03-17 19:51:07 +00:00
cs2 . spells . insert ( it ) ;
if ( ! cs1 . spells . empty ( ) | | ! cs2 . spells . empty ( ) ) //create a message
{
InfoWindow iw ;
iw . player = h1 - > tempOwner ;
iw . components . push_back ( Component ( Component : : SEC_SKILL , 18 , ScholarLevel , 0 ) ) ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 139 ) ; //"%s, who has studied magic extensively,
iw . text . addReplacement ( h1 - > name ) ;
if ( ! cs2 . spells . empty ( ) ) //if found new spell - apply
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 140 ) ; //learns
int size = cs2 . spells . size ( ) ;
2016-10-12 17:16:26 +02:00
for ( auto it : cs2 . spells )
2014-03-17 19:51:07 +00:00
{
iw . components . push_back ( Component ( Component : : SPELL , it , 1 , 0 ) ) ;
iw . text . addTxt ( MetaString : : SPELL_NAME , it . toEnum ( ) ) ;
switch ( size - - )
{
case 2 : iw . text . addTxt ( MetaString : : GENERAL_TXT , 141 ) ;
case 1 : break ;
default : iw . text < < " , " ;
}
}
iw . text . addTxt ( MetaString : : GENERAL_TXT , 142 ) ; //from %s
iw . text . addReplacement ( h2 - > name ) ;
sendAndApply ( & cs2 ) ;
}
2016-10-12 17:16:26 +02:00
if ( ! cs1 . spells . empty ( ) & & ! cs2 . spells . empty ( ) )
2014-03-17 19:51:07 +00:00
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 141 ) ; //and
}
if ( ! cs1 . spells . empty ( ) )
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 147 ) ; //teaches
int size = cs1 . spells . size ( ) ;
2016-10-12 17:16:26 +02:00
for ( auto it : cs1 . spells )
2014-03-17 19:51:07 +00:00
{
iw . components . push_back ( Component ( Component : : SPELL , it , 1 , 0 ) ) ;
iw . text . addTxt ( MetaString : : SPELL_NAME , it . toEnum ( ) ) ;
switch ( size - - )
{
case 2 : iw . text . addTxt ( MetaString : : GENERAL_TXT , 141 ) ;
case 1 : break ;
default : iw . text < < " , " ;
} }
iw . text . addTxt ( MetaString : : GENERAL_TXT , 148 ) ; //from %s
iw . text . addReplacement ( h2 - > name ) ;
sendAndApply ( & cs1 ) ;
}
sendAndApply ( & iw ) ;
}
}
void CGameHandler : : heroExchange ( ObjectInstanceID hero1 , ObjectInstanceID hero2 )
{
auto h1 = getHero ( hero1 ) , h2 = getHero ( hero2 ) ;
2016-10-12 17:16:26 +02:00
if ( getPlayerRelations ( h1 - > getOwner ( ) , h2 - > getOwner ( ) ) )
2014-03-17 19:51:07 +00:00
{
2017-06-06 07:53:51 +03:00
auto exchange = std : : make_shared < CGarrisonDialogQuery > ( this , h1 , h2 ) ;
2014-03-17 19:51:07 +00:00
ExchangeDialog hex ;
hex . queryID = exchange - > queryID ;
hex . heroes [ 0 ] = getHero ( hero1 ) ;
hex . heroes [ 1 ] = getHero ( hero2 ) ;
sendAndApply ( & hex ) ;
useScholarSkill ( hero1 , hero2 ) ;
queries . addQuery ( exchange ) ;
}
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : sendToAllClients ( CPackForClient * info )
2014-03-17 19:51:07 +00:00
{
2016-09-22 21:35:37 +03:00
logNetwork - > trace ( " Sending to all clients a package of type %s " , typeid ( * info ) . name ( ) ) ;
2016-10-12 17:16:26 +02:00
for ( auto & elem : conns )
2014-03-17 19:51:07 +00:00
{
2016-10-30 02:43:06 +03:00
if ( ! elem - > isOpen ( ) )
continue ;
2014-03-17 19:51:07 +00:00
boost : : unique_lock < boost : : mutex > lock ( * ( elem ) - > wmx ) ;
* elem < < info ;
}
}
void CGameHandler : : sendAndApply ( CPackForClient * info )
{
sendToAllClients ( info ) ;
gs - > apply ( info ) ;
}
void CGameHandler : : applyAndSend ( CPackForClient * info )
{
gs - > apply ( info ) ;
sendToAllClients ( info ) ;
}
void CGameHandler : : sendAndApply ( CGarrisonOperationPack * info )
{
sendAndApply ( static_cast < CPackForClient * > ( info ) ) ;
checkVictoryLossConditionsForAll ( ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : sendAndApply ( SetResources * info )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( static_cast < CPackForClient * > ( info ) ) ;
checkVictoryLossConditionsForPlayer ( info - > player ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : sendAndApply ( NewStructures * info )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( static_cast < CPackForClient * > ( info ) ) ;
checkVictoryLossConditionsForPlayer ( getTown ( info - > tid ) - > tempOwner ) ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : save ( const std : : string & filename )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " Saving to %s " , filename ) ;
2016-01-26 14:51:38 +01:00
const auto stem = FileInfo : : GetPathStem ( filename ) ;
const auto savefname = stem . to_string ( ) + " .vsgm1 " ;
CResourceHandler : : get ( " local " ) - > createResource ( savefname ) ;
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " Ordering clients to serialize... " ) ;
2016-01-26 14:51:38 +01:00
SaveGame sg ( savefname ) ;
2014-03-17 19:51:07 +00:00
sendToAllClients ( & sg ) ;
}
try
{
{
2016-01-26 14:51:38 +01:00
CSaveFile save ( * CResourceHandler : : get ( " local " ) - > getResourceName ( ResourceID ( stem . to_string ( ) , EResType : : SERVER_SAVEGAME ) ) ) ;
2014-03-17 19:51:07 +00:00
saveCommonState ( save ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " Saving server state " ) ;
2014-03-17 19:51:07 +00:00
save < < * this ;
}
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " Game has been successfully saved! " ) ;
2014-03-17 19:51:07 +00:00
}
catch ( std : : exception & e )
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Failed to save game: %s " , e . what ( ) ) ;
2014-03-17 19:51:07 +00:00
}
}
void CGameHandler : : close ( )
{
2016-08-30 01:11:54 +03:00
logGlobal - > info ( " We have been requested to close. " ) ;
2017-06-04 11:59:26 +03:00
serverShuttingDown = true ;
2014-03-17 19:51:07 +00:00
2016-12-04 12:54:26 +03:00
for ( auto & elem : conns )
{
if ( ! elem - > isOpen ( ) )
continue ;
boost : : unique_lock < boost : : mutex > lock ( * ( elem ) - > wmx ) ;
elem - > close ( ) ;
elem - > connected = false ;
}
2014-03-17 19:51:07 +00:00
}
2017-05-31 09:45:26 +03:00
void CGameHandler : : playerLeftGame ( int cid )
{
for ( auto & elem : conns )
{
if ( elem - > isOpen ( ) & & elem - > connectionID = = cid )
{
boost : : unique_lock < boost : : mutex > lock ( * ( elem ) - > wmx ) ;
elem - > close ( ) ;
elem - > connected = false ;
break ;
}
}
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : arrangeStacks ( ObjectInstanceID id1 , ObjectInstanceID id2 , ui8 what , SlotID p1 , SlotID p2 , si32 val , PlayerColor player )
2014-03-17 19:51:07 +00:00
{
2016-09-18 11:53:51 +03:00
const CArmedInstance * s1 = static_cast < const CArmedInstance * > ( getObjInstance ( id1 ) ) ,
* s2 = static_cast < const CArmedInstance * > ( getObjInstance ( id2 ) ) ;
2014-03-17 19:51:07 +00:00
const CCreatureSet & S1 = * s1 , & S2 = * s2 ;
StackLocation sl1 ( s1 , p1 ) , sl2 ( s2 , p2 ) ;
2016-10-12 17:16:26 +02:00
if ( ! sl1 . slot . validSlot ( ) | | ! sl2 . slot . validSlot ( ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Invalid slot accessed! " ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
if ( ! isAllowedExchange ( id1 , id2 ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot exchange stacks between these two objects! \n " ) ;
return false ;
}
2016-09-19 03:20:44 +03:00
// We can always put stacks into locked garrison, but not take them out of it
auto notRemovable = [ & ] ( const CArmedInstance * army )
{
2016-10-12 17:16:26 +02:00
if ( id1 ! = id2 ) // Stack arrangement inside locked garrison is allowed
2016-09-19 03:20:44 +03:00
{
auto g = dynamic_cast < const CGGarrison * > ( army ) ;
2016-10-12 17:16:26 +02:00
if ( g & & ! g - > removableUnits )
2016-09-19 03:20:44 +03:00
{
complain ( " Stacks in this garrison are not removable! \n " ) ;
return true ;
}
}
return false ;
} ;
2016-10-12 17:16:26 +02:00
if ( what = = 1 ) //swap
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ( ( s1 - > tempOwner ! = player & & s1 - > tempOwner ! = PlayerColor : : UNFLAGGABLE ) & & s1 - > getStackCount ( p1 ) )
2014-03-17 19:51:07 +00:00
| | ( ( s2 - > tempOwner ! = player & & s2 - > tempOwner ! = PlayerColor : : UNFLAGGABLE ) & & s2 - > getStackCount ( p2 ) ) )
{
complain ( " Can't take troops from another player! " ) ;
return false ;
}
2015-02-22 18:36:13 +01:00
if ( sl1 . army = = sl2 . army & & sl1 . slot = = sl2 . slot )
{
complain ( " Cannot swap stacks - slots are the same! " ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
if ( ! s1 - > slotEmpty ( p1 ) & & ! s2 - > slotEmpty ( p2 ) )
2016-09-19 03:20:44 +03:00
{
2016-10-12 17:16:26 +02:00
if ( notRemovable ( sl1 . army ) | | notRemovable ( sl2 . army ) )
2016-09-19 03:20:44 +03:00
return false ;
}
2016-10-12 17:16:26 +02:00
if ( s1 - > slotEmpty ( p1 ) & & notRemovable ( sl2 . army ) )
2016-09-19 03:20:44 +03:00
return false ;
2016-10-12 17:16:26 +02:00
else if ( s2 - > slotEmpty ( p2 ) & & notRemovable ( sl1 . army ) )
2016-09-19 03:20:44 +03:00
return false ;
2014-03-17 19:51:07 +00:00
swapStacks ( sl1 , sl2 ) ;
}
2016-10-12 17:16:26 +02:00
else if ( what = = 2 ) //merge
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ( s1 - > getCreature ( p1 ) ! = s2 - > getCreature ( p2 ) & & complain ( " Cannot merge different creatures stacks! " ) )
2014-03-17 19:51:07 +00:00
| | ( ( ( s1 - > tempOwner ! = player & & s1 - > tempOwner ! = PlayerColor : : UNFLAGGABLE ) & & s2 - > getStackCount ( p2 ) ) & & complain ( " Can't take troops from another player! " ) ) )
return false ;
2016-10-12 17:16:26 +02:00
if ( s1 - > slotEmpty ( p1 ) | | s2 - > slotEmpty ( p2 ) )
2016-09-19 03:20:44 +03:00
{
complain ( " Cannot merge empty stack! " ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
else if ( notRemovable ( sl1 . army ) )
2016-09-19 03:20:44 +03:00
return false ;
2014-03-17 19:51:07 +00:00
moveStack ( sl1 , sl2 ) ;
}
2016-10-12 17:16:26 +02:00
else if ( what = = 3 ) //split
2014-03-17 19:51:07 +00:00
{
const int countToMove = val - s2 - > getStackCount ( p2 ) ;
const int countLeftOnSrc = s1 - > getStackCount ( p1 ) - countToMove ;
2016-10-12 17:16:26 +02:00
if ( ( s1 - > tempOwner ! = player & & countLeftOnSrc < s1 - > getStackCount ( p1 ) )
| | ( s2 - > tempOwner ! = player & & val < s2 - > getStackCount ( p2 ) ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Can't move troops of another player! " ) ;
return false ;
}
//general conditions checking
2016-10-12 17:16:26 +02:00
if ( ( ! vstd : : contains ( S1 . stacks , p1 ) & & complain ( " no creatures to split " ) )
| | ( val < 1 & & complain ( " no creatures to split " ) ) )
2014-03-17 19:51:07 +00:00
{
return false ;
}
2016-10-12 17:16:26 +02:00
if ( vstd : : contains ( S2 . stacks , p2 ) ) //dest. slot not free - it must be "rebalancing"...
2014-03-17 19:51:07 +00:00
{
int total = s1 - > getStackCount ( p1 ) + s2 - > getStackCount ( p2 ) ;
2016-10-12 17:16:26 +02:00
if ( ( total < val & & complain ( " Cannot split that stack, not enough creatures! " ) )
2014-03-17 19:51:07 +00:00
| | ( s1 - > getCreature ( p1 ) ! = s2 - > getCreature ( p2 ) & & complain ( " Cannot rebalance different creatures stacks! " ) )
)
{
return false ;
}
2016-10-12 17:16:26 +02:00
if ( notRemovable ( sl1 . army ) )
2016-09-19 03:20:44 +03:00
{
2016-10-12 17:16:26 +02:00
if ( s1 - > getStackCount ( p1 ) > countLeftOnSrc )
2016-09-19 03:20:44 +03:00
return false ;
}
2016-10-12 17:16:26 +02:00
else if ( notRemovable ( sl2 . army ) )
2016-09-19 03:20:44 +03:00
{
2016-10-12 17:16:26 +02:00
if ( s2 - > getStackCount ( p1 ) < countLeftOnSrc )
2016-09-19 03:20:44 +03:00
return false ;
}
2014-03-17 19:51:07 +00:00
moveStack ( sl1 , sl2 , countToMove ) ;
//S2.slots[p2]->count = val;
//S1.slots[p1]->count = total - val;
}
else //split one stack to the two
{
2016-10-12 17:16:26 +02:00
if ( s1 - > getStackCount ( p1 ) < val ) //not enough creatures
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot split that stack, not enough creatures! " ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
if ( notRemovable ( sl1 . army ) )
2016-09-19 03:20:44 +03:00
return false ;
2014-03-17 19:51:07 +00:00
moveStack ( sl1 , sl2 , val ) ;
}
}
return true ;
}
2016-10-12 17:16:26 +02:00
PlayerColor CGameHandler : : getPlayerAt ( CConnection * c ) const
2014-03-17 19:51:07 +00:00
{
std : : set < PlayerColor > all ;
2016-10-12 17:16:26 +02:00
for ( auto i = connections . cbegin ( ) ; i ! = connections . cend ( ) ; i + + )
if ( i - > second = = c )
2014-03-17 19:51:07 +00:00
all . insert ( i - > first ) ;
switch ( all . size ( ) )
{
case 0 :
return PlayerColor : : NEUTRAL ;
case 1 :
return * all . begin ( ) ;
default :
{
//if we have more than one player at this connection, try to pick active one
2016-10-12 17:16:26 +02:00
if ( vstd : : contains ( all , gs - > currentPlayer ) )
2014-03-17 19:51:07 +00:00
return gs - > currentPlayer ;
else
return PlayerColor : : CANNOT_DETERMINE ; //cannot say which player is it
}
}
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : disbandCreature ( ObjectInstanceID id , SlotID pos )
2014-03-17 19:51:07 +00:00
{
2016-09-18 11:53:51 +03:00
const CArmedInstance * s1 = static_cast < const CArmedInstance * > ( getObjInstance ( id ) ) ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( s1 - > stacks , pos ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Illegal call to disbandCreature - no such stack in army! " ) ;
return false ;
}
eraseStack ( StackLocation ( s1 , pos ) ) ;
return true ;
}
2017-07-15 14:08:20 +03:00
bool CGameHandler : : buildStructure ( ObjectInstanceID tid , BuildingID requestedID , bool force )
2014-03-17 19:51:07 +00:00
{
const CGTownInstance * t = getTown ( tid ) ;
2016-10-12 17:16:26 +02:00
if ( ! t )
2014-03-17 19:51:07 +00:00
COMPLAIN_RETF ( " No such town (ID=%s)! " , tid ) ;
2016-10-12 17:16:26 +02:00
if ( ! t - > town - > buildings . count ( requestedID ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RETF ( " Town of faction %s does not have info about building ID=%s! " , t - > town - > faction - > name % tid ) ;
if ( t - > hasBuilt ( requestedID ) )
COMPLAIN_RETF ( " Building %s is already built in %s " , t - > town - > buildings . at ( requestedID ) - > Name ( ) % t - > name ) ;
const CBuilding * requestedBuilding = t - > town - > buildings . at ( requestedID ) ;
//Vector with future list of built building and buildings in auto-mode that are not yet built.
std : : vector < const CBuilding * > remainingAutoBuildings ;
std : : set < BuildingID > buildingsThatWillBe ;
//Check validity of request
2016-10-12 17:16:26 +02:00
if ( ! force )
2014-03-17 19:51:07 +00:00
{
switch ( requestedBuilding - > mode )
{
case CBuilding : : BUILD_NORMAL :
2016-10-12 17:16:26 +02:00
if ( canBuildStructure ( t , requestedID ) ! = EBuildingState : : ALLOWED )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot build that building! " ) ;
break ;
case CBuilding : : BUILD_AUTO :
case CBuilding : : BUILD_SPECIAL :
COMPLAIN_RET ( " This building can not be constructed normally! " ) ;
case CBuilding : : BUILD_GRAIL :
2016-10-12 17:16:26 +02:00
if ( requestedBuilding - > mode = = CBuilding : : BUILD_GRAIL ) //needs grail
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! t - > visitingHero | | ! t - > visitingHero - > hasArt ( ArtifactID : : GRAIL ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot build this without grail! " )
else
removeArtifact ( ArtifactLocation ( t - > visitingHero , t - > visitingHero - > getArtPos ( ArtifactID : : GRAIL , false ) ) ) ;
}
break ;
}
}
2016-03-13 11:31:09 +03:00
//Performs stuff that has to be done before new building is built
auto processBeforeBuiltStructure = [ t , this ] ( const BuildingID buildingID )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( buildingID > = BuildingID : : DWELL_FIRST ) //dwelling
2014-03-17 19:51:07 +00:00
{
int level = ( buildingID - BuildingID : : DWELL_FIRST ) % GameConstants : : CREATURES_PER_TOWN ;
int upgradeNumber = ( buildingID - BuildingID : : DWELL_FIRST ) / GameConstants : : CREATURES_PER_TOWN ;
if ( upgradeNumber > = t - > town - > creatures . at ( level ) . size ( ) )
{
complain ( boost : : str ( boost : : format ( " Error ecountered when building dwelling (bid=%s): "
" no creature found (upgrade number %d, level %d! " )
% buildingID % upgradeNumber % level ) ) ;
return ;
}
CCreature * crea = VLC - > creh - > creatures . at ( t - > town - > creatures . at ( level ) . at ( upgradeNumber ) ) ;
SetAvailableCreatures ssi ;
ssi . tid = t - > id ;
ssi . creatures = t - > creatures ;
if ( ssi . creatures [ level ] . second . empty ( ) ) // first creature in a dwelling
ssi . creatures [ level ] . first = crea - > growth ;
ssi . creatures [ level ] . second . push_back ( crea - > idNumber ) ;
sendAndApply ( & ssi ) ;
}
2016-10-12 17:16:26 +02:00
if ( t - > subID = = ETownType : : DUNGEON & & buildingID = = BuildingID : : PORTAL_OF_SUMMON )
2014-03-17 19:51:07 +00:00
{
setPortalDwelling ( t ) ;
}
2016-03-13 11:31:09 +03:00
} ;
2014-03-17 19:51:07 +00:00
2016-03-13 11:31:09 +03:00
//Performs stuff that has to be done after new building is built
auto processAfterBuiltStructure = [ t , this ] ( const BuildingID buildingID )
{
2016-10-12 17:16:26 +02:00
if ( buildingID < = BuildingID : : MAGES_GUILD_5 | | //it's mage guild
2016-03-13 11:31:09 +03:00
( t - > subID = = ETownType : : TOWER & & buildingID = = BuildingID : : LIBRARY ) | |
( t - > subID = = ETownType : : CONFLUX & & buildingID = = BuildingID : : GRAIL ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( t - > visitingHero )
2014-03-17 19:51:07 +00:00
giveSpells ( t , t - > visitingHero ) ;
2016-10-12 17:16:26 +02:00
if ( t - > garrisonHero )
2014-03-17 19:51:07 +00:00
giveSpells ( t , t - > garrisonHero ) ;
}
} ;
//Checks if all requirements will be met with expected building list "buildingsThatWillBe"
auto areRequirementsFullfilled = [ & ] ( const BuildingID & buildID )
{
return buildingsThatWillBe . count ( buildID ) ;
} ;
//Init the vectors
2016-10-12 17:16:26 +02:00
for ( auto & build : t - > town - > buildings )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( t - > hasBuilt ( build . first ) )
2014-03-17 19:51:07 +00:00
buildingsThatWillBe . insert ( build . first ) ;
else
{
2016-10-12 17:16:26 +02:00
if ( build . second - > mode = = CBuilding : : BUILD_AUTO ) //not built auto building
2014-03-17 19:51:07 +00:00
remainingAutoBuildings . push_back ( build . second ) ;
}
}
//Prepare structure (list of building ids will be filled later)
NewStructures ns ;
ns . tid = tid ;
ns . builded = force ? t - > builded : ( t - > builded + 1 ) ;
std : : queue < const CBuilding * > buildingsToAdd ;
buildingsToAdd . push ( requestedBuilding ) ;
while ( ! buildingsToAdd . empty ( ) )
{
auto b = buildingsToAdd . front ( ) ;
buildingsToAdd . pop ( ) ;
ns . bid . insert ( b - > bid ) ;
buildingsThatWillBe . insert ( b - > bid ) ;
remainingAutoBuildings - = b ;
2016-10-12 17:16:26 +02:00
for ( auto autoBuilding : remainingAutoBuildings )
2014-03-17 19:51:07 +00:00
{
if ( autoBuilding - > requirements . test ( areRequirementsFullfilled ) )
buildingsToAdd . push ( autoBuilding ) ;
}
}
2016-03-13 11:31:09 +03:00
// FIXME: it's done before NewStructures applied because otherwise town window wont be properly updated on client. That should be actually fixed on client and not on server.
2016-10-12 17:16:26 +02:00
for ( auto builtID : ns . bid )
2016-03-13 11:31:09 +03:00
processBeforeBuiltStructure ( builtID ) ;
2014-03-17 19:51:07 +00:00
//Take cost
if ( ! force )
{
2016-11-26 15:14:43 +03:00
giveResources ( t - > tempOwner , - requestedBuilding - > resources ) ;
2014-03-17 19:51:07 +00:00
}
//We know what has been built, appluy changes. Do this as final step to properly update town window
sendAndApply ( & ns ) ;
2016-03-13 11:31:09 +03:00
//Other post-built events. To some logic like giving spells to work gamestate changes for new building must be already in place!
2016-10-12 17:16:26 +02:00
for ( auto builtID : ns . bid )
2016-03-13 11:31:09 +03:00
processAfterBuiltStructure ( builtID ) ;
2014-03-17 19:51:07 +00:00
// now when everything is built - reveal tiles for lookout tower
FoWChange fw ;
fw . player = t - > tempOwner ;
fw . mode = 1 ;
2016-01-31 18:01:58 +03:00
getTilesInRange ( fw . tiles , t - > getSightCenter ( ) , t - > getSightRadius ( ) , t - > tempOwner , 1 ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & fw ) ;
2016-10-12 17:16:26 +02:00
if ( t - > visitingHero )
2014-03-17 19:51:07 +00:00
vistiCastleObjects ( t , t - > visitingHero ) ;
2016-10-12 17:16:26 +02:00
if ( t - > garrisonHero )
2014-03-17 19:51:07 +00:00
vistiCastleObjects ( t , t - > garrisonHero ) ;
checkVictoryLossConditionsForPlayer ( t - > tempOwner ) ;
return true ;
}
bool CGameHandler : : razeStructure ( ObjectInstanceID tid , BuildingID bid )
{
///incomplete, simply erases target building
const CGTownInstance * t = getTown ( tid ) ;
if ( ! vstd : : contains ( t - > builtBuildings , bid ) )
return false ;
RazeStructures rs ;
rs . tid = tid ;
rs . bid . insert ( bid ) ;
rs . destroyed = t - > destroyed + 1 ;
sendAndApply ( & rs ) ;
//TODO: Remove dwellers
// if (t->subID == 4 && bid == 17) //Veil of Darkness
// {
// RemoveBonus rb(RemoveBonus::TOWN);
// rb.whoID = t->id;
// rb.source = Bonus::TOWN_STRUCTURE;
// rb.id = 17;
// sendAndApply(&rb);
// }
return true ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : sendMessageToAll ( const std : : string & message )
2014-03-17 19:51:07 +00:00
{
SystemMessage sm ;
sm . text = message ;
sendToAllClients ( & sm ) ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : recruitCreatures ( ObjectInstanceID objid , ObjectInstanceID dstid , CreatureID crid , ui32 cram , si32 fromLvl )
2014-03-17 19:51:07 +00:00
{
2016-09-18 11:53:51 +03:00
const CGDwelling * dw = static_cast < const CGDwelling * > ( getObj ( objid ) ) ;
2014-03-17 19:51:07 +00:00
const CArmedInstance * dst = nullptr ;
const CCreature * c = VLC - > creh - > creatures . at ( crid ) ;
2017-05-26 19:51:45 +03:00
const bool warMachine = c - > warMachine ! = ArtifactID : : NONE ;
2014-03-17 19:51:07 +00:00
//TODO: test for owning
2014-09-19 00:18:49 +03:00
//TODO: check if dst can recruit objects (e.g. hero is actually visiting object, town and source are same, etc)
dst = dynamic_cast < const CArmedInstance * > ( getObj ( dstid ) ) ;
2014-03-17 19:51:07 +00:00
assert ( dw & & dst ) ;
//verify
bool found = false ;
int level = 0 ;
2016-10-12 17:16:26 +02:00
for ( ; level < dw - > creatures . size ( ) ; level + + ) //iterate through all levels
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ( fromLvl ! = - 1 ) & & ( level ! = fromLvl ) )
2014-03-17 19:51:07 +00:00
continue ;
const auto & cur = dw - > creatures . at ( level ) ; //current level info <amount, list of cr. ids>
int i = 0 ;
2016-10-12 17:16:26 +02:00
for ( ; i < cur . second . size ( ) ; i + + ) //look for crid among available creatures list on current level
if ( cur . second . at ( i ) = = crid )
2014-03-17 19:51:07 +00:00
break ;
2016-10-12 17:16:26 +02:00
if ( i < cur . second . size ( ) )
2014-03-17 19:51:07 +00:00
{
found = true ;
cram = std : : min ( cram , cur . first ) ; //reduce recruited amount up to available amount
break ;
}
}
SlotID slot = dst - > getSlotFor ( crid ) ;
2016-10-12 17:16:26 +02:00
if ( ( ! found & & complain ( " Cannot recruit: no such creatures! " ) )
2016-09-18 11:53:51 +03:00
| | ( cram > VLC - > creh - > creatures . at ( crid ) - > maxAmount ( getPlayer ( dst - > tempOwner ) - > resources ) & & complain ( " Cannot recruit: lack of resources! " ) )
2014-03-17 19:51:07 +00:00
| | ( cram < = 0 & & complain ( " Cannot recruit: cram <= 0! " ) )
| | ( ! slot . validSlot ( ) & & ! warMachine & & complain ( " Cannot recruit: no available slot! " ) ) )
{
return false ;
}
//recruit
2016-11-26 15:14:43 +03:00
giveResources ( dst - > tempOwner , - ( c - > cost * cram ) ) ;
2014-03-17 19:51:07 +00:00
SetAvailableCreatures sac ;
sac . tid = objid ;
sac . creatures = dw - > creatures ;
sac . creatures [ level ] . first - = cram ;
sendAndApply ( & sac ) ;
2016-10-12 17:16:26 +02:00
if ( warMachine )
2014-03-17 19:51:07 +00:00
{
const CGHeroInstance * h = dynamic_cast < const CGHeroInstance * > ( dst ) ;
2017-05-26 19:51:45 +03:00
COMPLAIN_RET_FALSE_IF ( ! h , " Only hero can buy war machines " ) ;
ArtifactID artId = c - > warMachine ;
COMPLAIN_RET_FALSE_IF ( artId = = ArtifactID : : CATAPULT , " Catapult cannot be recruited! " ) ;
const CArtifact * art = artId . toArtifact ( ) ;
COMPLAIN_RET_FALSE_IF ( nullptr = = art , " Invalid war machine artifact " ) ;
return giveHeroNewArtifact ( h , art ) ;
2014-03-17 19:51:07 +00:00
}
else
{
addToSlot ( StackLocation ( dst , slot ) , c , cram ) ;
}
return true ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : upgradeCreature ( ObjectInstanceID objid , SlotID pos , CreatureID upgID )
2014-03-17 19:51:07 +00:00
{
2016-09-18 11:53:51 +03:00
const CArmedInstance * obj = static_cast < const CArmedInstance * > ( getObjInstance ( objid ) ) ;
2015-02-14 19:44:04 +01:00
if ( ! obj - > hasStackAtSlot ( pos ) )
{
COMPLAIN_RET ( " Cannot upgrade, no stack at slot " + boost : : to_string ( pos ) ) ;
}
2016-09-18 11:53:51 +03:00
UpgradeInfo ui ;
getUpgradeInfo ( obj , pos , ui ) ;
2014-03-17 19:51:07 +00:00
PlayerColor player = obj - > tempOwner ;
const PlayerState * p = getPlayer ( player ) ;
int crQuantity = obj - > stacks . at ( pos ) - > count ;
int newIDpos = vstd : : find_pos ( ui . newID , upgID ) ; //get position of new id in UpgradeInfo
//check if upgrade is possible
2016-10-12 17:16:26 +02:00
if ( ( ui . oldID < 0 | | newIDpos = = - 1 ) & & complain ( " That upgrade is not possible! " ) )
2014-03-17 19:51:07 +00:00
{
return false ;
}
TResources totalCost = ui . cost . at ( newIDpos ) * crQuantity ;
//check if player has enough resources
2016-10-12 17:16:26 +02:00
if ( ! p - > resources . canAfford ( totalCost ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot upgrade, not enough resources! " ) ;
//take resources
2016-11-26 15:14:43 +03:00
giveResources ( player , - totalCost ) ;
2014-03-17 19:51:07 +00:00
//upgrade creature
changeStackType ( StackLocation ( obj , pos ) , VLC - > creh - > creatures . at ( upgID ) ) ;
return true ;
}
2015-09-04 18:08:25 +03:00
bool CGameHandler : : changeStackType ( const StackLocation & sl , const CCreature * c )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! sl . army - > hasStackAtSlot ( sl . slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot find a stack to change type " ) ;
SetStackType sst ;
sst . sl = sl ;
sst . type = c ;
sendAndApply ( & sst ) ;
return true ;
}
void CGameHandler : : moveArmy ( const CArmedInstance * src , const CArmedInstance * dst , bool allowMerging )
{
assert ( src - > canBeMergedWith ( * dst , allowMerging ) ) ;
while ( src - > stacksCount ( ) ) //while there are unmoved creatures
{
auto i = src - > Slots ( ) . begin ( ) ; //iterator to stack to move
StackLocation sl ( src , i - > first ) ; //location of stack to move
SlotID pos = dst - > getSlotFor ( i - > second - > type ) ;
2016-10-12 17:16:26 +02:00
if ( ! pos . validSlot ( ) )
2014-03-17 19:51:07 +00:00
{
//try to merge two other stacks to make place
std : : pair < SlotID , SlotID > toMerge ;
2016-10-12 17:16:26 +02:00
if ( dst - > mergableStacks ( toMerge , i - > first ) & & allowMerging )
2014-03-17 19:51:07 +00:00
{
moveStack ( StackLocation ( dst , toMerge . first ) , StackLocation ( dst , toMerge . second ) ) ; //merge toMerge.first into toMerge.second
assert ( ! dst - > hasStackAtSlot ( toMerge . first ) ) ; //we have now a new free slot
moveStack ( sl , StackLocation ( dst , toMerge . first ) ) ; //move stack to freed slot
}
else
{
complain ( " Unexpected failure during an attempt to move army from " + src - > nodeName ( ) + " to " + dst - > nodeName ( ) + " ! " ) ;
return ;
}
}
else
{
moveStack ( sl , StackLocation ( dst , pos ) ) ;
}
}
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : garrisonSwap ( ObjectInstanceID tid )
2014-03-17 19:51:07 +00:00
{
2016-09-18 11:53:51 +03:00
const CGTownInstance * town = getTown ( tid ) ;
2016-10-12 17:16:26 +02:00
if ( ! town - > garrisonHero & & town - > visitingHero ) //visiting => garrison, merge armies: town army => hero army
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! town - > visitingHero - > canBeMergedWith ( * town ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot make garrison swap, not enough free slots! " ) ;
return false ;
}
moveArmy ( town , town - > visitingHero , true ) ;
SetHeroesInTown intown ;
intown . tid = tid ;
intown . visiting = ObjectInstanceID ( ) ;
intown . garrison = town - > visitingHero - > id ;
sendAndApply ( & intown ) ;
return true ;
}
else if ( town - > garrisonHero & & ! town - > visitingHero ) //move hero out of the garrison
{
//check if moving hero out of town will break 8 wandering heroes limit
2016-10-12 17:16:26 +02:00
if ( getHeroCount ( town - > garrisonHero - > tempOwner , false ) > = 8 )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot move hero out of the garrison, there are already 8 wandering heroes! " ) ;
return false ;
}
SetHeroesInTown intown ;
intown . tid = tid ;
intown . garrison = ObjectInstanceID ( ) ;
intown . visiting = town - > garrisonHero - > id ;
sendAndApply ( & intown ) ;
return true ;
}
2016-10-12 17:16:26 +02:00
else if ( ! ! town - > garrisonHero & & town - > visitingHero ) //swap visiting and garrison hero
2014-03-17 19:51:07 +00:00
{
SetHeroesInTown intown ;
intown . tid = tid ;
intown . garrison = town - > visitingHero - > id ;
intown . visiting = town - > garrisonHero - > id ;
sendAndApply ( & intown ) ;
return true ;
}
else
{
complain ( " Cannot swap garrison hero! " ) ;
return false ;
}
}
// With the amount of changes done to the function, it's more like transferArtifacts.
// Function moves artifact from src to dst. If dst is not a backpack and is already occupied, old dst art goes to backpack and is replaced.
bool CGameHandler : : moveArtifact ( const ArtifactLocation & al1 , const ArtifactLocation & al2 )
{
ArtifactLocation src = al1 , dst = al2 ;
const PlayerColor srcPlayer = src . owningPlayer ( ) , dstPlayer = dst . owningPlayer ( ) ;
const CArmedInstance * srcObj = src . relatedObj ( ) , * dstObj = dst . relatedObj ( ) ;
// Make sure exchange is even possible between the two heroes.
2016-10-12 17:16:26 +02:00
if ( ! isAllowedExchange ( srcObj - > id , dstObj - > id ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " That heroes cannot make any exchange! " ) ;
const CArtifactInstance * srcArtifact = src . getArt ( ) ;
const CArtifactInstance * destArtifact = dst . getArt ( ) ;
if ( srcArtifact = = nullptr )
COMPLAIN_RET ( " No artifact to move! " ) ;
if ( destArtifact & & srcPlayer ! = dstPlayer )
COMPLAIN_RET ( " Can't touch artifact on hero of another player! " ) ;
// Check if src/dest slots are appropriate for the artifacts exchanged.
// Moving to the backpack is always allowed.
if ( ( ! srcArtifact | | dst . slot < GameConstants : : BACKPACK_START )
& & srcArtifact & & ! srcArtifact - > canBePutAt ( dst , true ) )
COMPLAIN_RET ( " Cannot move artifact! " ) ;
2016-08-18 06:35:29 +03:00
auto srcSlot = src . getSlot ( ) ;
auto dstSlot = dst . getSlot ( ) ;
if ( ( srcSlot & & srcSlot - > locked ) | | ( dstSlot & & dstSlot - > locked ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot move artifact locks. " ) ;
if ( dst . slot > = GameConstants : : BACKPACK_START & & srcArtifact - > artType - > isBig ( ) )
COMPLAIN_RET ( " Cannot put big artifacts in backpack! " ) ;
if ( src . slot = = ArtifactPosition : : MACH4 | | dst . slot = = ArtifactPosition : : MACH4 )
COMPLAIN_RET ( " Cannot move catapult! " ) ;
2016-10-12 17:16:26 +02:00
if ( dst . slot > = GameConstants : : BACKPACK_START )
2014-03-17 19:51:07 +00:00
vstd : : amin ( dst . slot , ArtifactPosition ( GameConstants : : BACKPACK_START + dst . getHolderArtSet ( ) - > artifactsInBackpack . size ( ) ) ) ;
if ( src . slot = = dst . slot & & src . artHolder = = dst . artHolder )
COMPLAIN_RET ( " Won't move artifact: Dest same as source! " ) ;
2016-10-12 17:16:26 +02:00
if ( dst . slot < GameConstants : : BACKPACK_START & & destArtifact ) //moving art to another slot
2014-03-17 19:51:07 +00:00
{
//old artifact must be removed first
moveArtifact ( dst , ArtifactLocation ( dst . artHolder , ArtifactPosition (
dst . getHolderArtSet ( ) - > artifactsInBackpack . size ( ) + GameConstants : : BACKPACK_START ) ) ) ;
}
MoveArtifact ma ;
ma . src = src ;
ma . dst = dst ;
sendAndApply ( & ma ) ;
return true ;
}
/**
* Assembles or disassembles a combination artifact .
* @ param heroID ID of hero holding the artifact ( s ) .
* @ param artifactSlot The worn slot ID of the combination - or constituent artifact .
* @ param assemble True for assembly operation , false for disassembly .
* @ param assembleTo If assemble is true , this represents the artifact ID of the combination
* artifact to assemble to . Otherwise it ' s not used .
*/
bool CGameHandler : : assembleArtifacts ( ObjectInstanceID heroID , ArtifactPosition artifactSlot , bool assemble , ArtifactID assembleTo )
{
2016-09-18 11:53:51 +03:00
const CGHeroInstance * hero = getHero ( heroID ) ;
2014-03-17 19:51:07 +00:00
const CArtifactInstance * destArtifact = hero - > getArt ( artifactSlot ) ;
2016-10-12 17:16:26 +02:00
if ( ! destArtifact )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " assembleArtifacts: there is no such artifact instance! " ) ;
2016-10-12 17:16:26 +02:00
if ( assemble )
2014-03-17 19:51:07 +00:00
{
2015-11-06 21:54:51 +03:00
CArtifact * combinedArt = VLC - > arth - > artifacts [ assembleTo ] ;
2016-10-12 17:16:26 +02:00
if ( ! combinedArt - > constituents )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " assembleArtifacts: Artifact being attempted to assemble is not a combined artifacts! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( destArtifact - > assemblyPossibilities ( hero ) , combinedArt ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " assembleArtifacts: It's impossible to assemble requested artifact! " ) ;
AssembledArtifact aa ;
aa . al = ArtifactLocation ( hero , artifactSlot ) ;
aa . builtArt = combinedArt ;
sendAndApply ( & aa ) ;
}
else
{
2016-10-12 17:16:26 +02:00
if ( ! destArtifact - > artType - > constituents )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " assembleArtifacts: Artifact being attempted to disassemble is not a combined artifact! " ) ;
DisassembledArtifact da ;
da . al = ArtifactLocation ( hero , artifactSlot ) ;
sendAndApply ( & da ) ;
}
2016-01-23 15:20:51 +03:00
return true ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : buyArtifact ( ObjectInstanceID hid , ArtifactID aid )
2014-03-17 19:51:07 +00:00
{
2016-09-18 11:53:51 +03:00
const CGHeroInstance * hero = getHero ( hid ) ;
2017-05-26 19:51:45 +03:00
COMPLAIN_RET_FALSE_IF ( nullptr = = hero , " Invalid hero index " ) ;
2016-09-18 11:53:51 +03:00
const CGTownInstance * town = hero - > visitedTown ;
2017-05-26 19:51:45 +03:00
COMPLAIN_RET_FALSE_IF ( nullptr = = town , " Hero not in town " ) ;
2016-10-12 17:16:26 +02:00
if ( aid = = ArtifactID : : SPELLBOOK )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ( ! town - > hasBuilt ( BuildingID : : MAGES_GUILD_1 ) & & complain ( " Cannot buy a spellbook, no mage guild in the town! " ) )
| | ( getResource ( hero - > getOwner ( ) , Res : : GOLD ) < GameConstants : : SPELLBOOK_GOLD_COST & & complain ( " Cannot buy a spellbook, not enough gold! " ) )
2014-03-17 19:51:07 +00:00
| | ( hero - > getArt ( ArtifactPosition : : SPELLBOOK ) & & complain ( " Cannot buy a spellbook, hero already has a one! " ) )
2016-10-12 17:16:26 +02:00
)
2014-03-17 19:51:07 +00:00
return false ;
giveResource ( hero - > getOwner ( ) , Res : : GOLD , - GameConstants : : SPELLBOOK_GOLD_COST ) ;
2014-12-21 13:56:32 +01:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ ArtifactID : : SPELLBOOK ] , ArtifactPosition : : SPELLBOOK ) ;
2014-03-17 19:51:07 +00:00
assert ( hero - > getArt ( ArtifactPosition : : SPELLBOOK ) ) ;
giveSpells ( town , hero ) ;
return true ;
}
2017-05-26 19:51:45 +03:00
else
2014-03-17 19:51:07 +00:00
{
2017-05-26 19:51:45 +03:00
const CArtifact * art = aid . toArtifact ( ) ;
COMPLAIN_RET_FALSE_IF ( nullptr = = art , " Invalid artifact index to buy " ) ;
COMPLAIN_RET_FALSE_IF ( art - > warMachine = = CreatureID : : NONE , " War machine artifact required " ) ;
COMPLAIN_RET_FALSE_IF ( hero - > hasArt ( aid ) , " Hero already has this machine! " ) ;
const int price = art - > price ;
COMPLAIN_RET_FALSE_IF ( getPlayer ( hero - > getOwner ( ) ) - > resources . at ( Res : : GOLD ) < price , " Not enough gold! " ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( ( town - > hasBuilt ( BuildingID : : BLACKSMITH ) & & town - > town - > warMachine = = aid )
2014-03-17 19:51:07 +00:00
| | ( ( town - > hasBuilt ( BuildingID : : BALLISTA_YARD , ETownType : : STRONGHOLD ) ) & & aid = = ArtifactID : : BALLISTA ) )
{
giveResource ( hero - > getOwner ( ) , Res : : GOLD , - price ) ;
2017-05-26 19:51:45 +03:00
return giveHeroNewArtifact ( hero , art ) ;
2014-03-17 19:51:07 +00:00
}
else
COMPLAIN_RET ( " This machine is unavailable here! " ) ;
}
}
bool CGameHandler : : buyArtifact ( const IMarket * m , const CGHeroInstance * h , Res : : ERes rid , ArtifactID aid )
{
2016-11-27 22:50:37 +03:00
if ( ! h )
COMPLAIN_RET ( " Only hero can buy artifacts! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( m - > availableItemsIds ( EMarketMode : : RESOURCE_ARTIFACT ) , aid ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " That artifact is unavailable! " ) ;
int b1 , b2 ;
m - > getOffer ( rid , aid , b1 , b2 , EMarketMode : : RESOURCE_ARTIFACT ) ;
2016-10-12 17:16:26 +02:00
if ( getResource ( h - > tempOwner , rid ) < b1 )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " You can't afford to buy this artifact! " ) ;
2016-11-26 15:14:43 +03:00
giveResource ( h - > tempOwner , rid , - b1 ) ;
2014-03-17 19:51:07 +00:00
SetAvailableArtifacts saa ;
2016-10-12 17:16:26 +02:00
if ( m - > o - > ID = = Obj : : TOWN )
2014-03-17 19:51:07 +00:00
{
saa . id = - 1 ;
saa . arts = CGTownInstance : : merchantArtifacts ;
}
2016-10-12 17:16:26 +02:00
else if ( const CGBlackMarket * bm = dynamic_cast < const CGBlackMarket * > ( m - > o ) ) //black market
2014-03-17 19:51:07 +00:00
{
saa . id = bm - > id . getNum ( ) ;
saa . arts = bm - > artifacts ;
}
else
COMPLAIN_RET ( " Wrong marktet... " ) ;
bool found = false ;
2016-10-12 17:16:26 +02:00
for ( const CArtifact * & art : saa . arts )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( art & & art - > id = = aid )
2014-03-17 19:51:07 +00:00
{
art = nullptr ;
found = true ;
break ;
}
}
2016-10-12 17:16:26 +02:00
if ( ! found )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot find selected artifact on the list " ) ;
sendAndApply ( & saa ) ;
2015-11-06 21:54:51 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ aid ] , ArtifactPosition : : FIRST_AVAILABLE ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : sellArtifact ( const IMarket * m , const CGHeroInstance * h , ArtifactInstanceID aid , Res : : ERes rid )
2014-03-17 19:51:07 +00:00
{
2016-11-27 22:50:37 +03:00
COMPLAIN_RET_FALSE_IF ( ( ! h ) , " Only hero can sell artifacts! " ) ;
2014-03-17 19:51:07 +00:00
const CArtifactInstance * art = h - > getArtByInstanceId ( aid ) ;
2016-11-27 22:50:37 +03:00
COMPLAIN_RET_FALSE_IF ( ( ! art ) , " There is no artifact to sell! " ) ;
COMPLAIN_RET_FALSE_IF ( ( ! art - > artType - > isTradable ( ) ) , " Cannot sell a war machine or spellbook! " ) ;
2014-03-17 19:51:07 +00:00
int resVal = 0 , dump = 1 ;
m - > getOffer ( art - > artType - > id , rid , dump , resVal , EMarketMode : : ARTIFACT_RESOURCE ) ;
removeArtifact ( ArtifactLocation ( h , h - > getArtPos ( art ) ) ) ;
giveResource ( h - > tempOwner , rid , resVal ) ;
return true ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : buySecSkill ( const IMarket * m , const CGHeroInstance * h , SecondarySkill skill )
2014-03-17 19:51:07 +00:00
{
if ( ! h )
COMPLAIN_RET ( " You need hero to buy a skill! " ) ;
if ( h - > getSecSkillLevel ( SecondarySkill ( skill ) ) )
COMPLAIN_RET ( " Hero already know this skill " ) ;
if ( ! h - > canLearnSkill ( ) )
COMPLAIN_RET ( " Hero can't learn any more skills " ) ;
if ( h - > type - > heroClass - > secSkillProbability . at ( skill ) = = 0 ) //can't learn this skill (like necromancy for most of non-necros)
COMPLAIN_RET ( " The hero can't learn this skill! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( m - > availableItemsIds ( EMarketMode : : RESOURCE_SKILL ) , skill ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " That skill is unavailable! " ) ;
2016-11-26 15:14:43 +03:00
if ( getResource ( h - > tempOwner , Res : : GOLD ) < GameConstants : : SKILL_GOLD_COST ) //TODO: remove hardcoded resource\summ?
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " You can't afford to buy this skill " ) ;
2016-11-26 15:14:43 +03:00
giveResource ( h - > tempOwner , Res : : GOLD , - GameConstants : : SKILL_GOLD_COST ) ;
2014-03-17 19:51:07 +00:00
changeSecSkill ( h , skill , 1 , true ) ;
return true ;
}
bool CGameHandler : : tradeResources ( const IMarket * market , ui32 val , PlayerColor player , ui32 id1 , ui32 id2 )
{
2016-11-26 15:14:43 +03:00
TResourceCap r1 = getPlayer ( player ) - > resources . at ( id1 ) ;
2014-03-17 19:51:07 +00:00
vstd : : amin ( val , r1 ) ; //can't trade more resources than have
int b1 , b2 ; //base quantities for trade
market - > getOffer ( id1 , id2 , b1 , b2 , EMarketMode : : RESOURCE_RESOURCE ) ;
int units = val / b1 ; //how many base quantities we trade
2016-10-12 17:16:26 +02:00
if ( val % b1 ) //all offered units of resource should be used, if not -> somewhere in calculations must be an error
2014-03-17 19:51:07 +00:00
{
2016-11-26 15:14:43 +03:00
COMPLAIN_RET ( " Invalid deal, not all offered units of resource were used. " ) ;
2014-03-17 19:51:07 +00:00
}
2016-11-26 15:14:43 +03:00
giveResource ( player , static_cast < Res : : ERes > ( id1 ) , - b1 * units ) ;
giveResource ( player , static_cast < Res : : ERes > ( id2 ) , b2 * units ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
bool CGameHandler : : sellCreatures ( ui32 count , const IMarket * market , const CGHeroInstance * hero , SlotID slot , Res : : ERes resourceID )
{
2016-11-27 22:50:37 +03:00
if ( ! hero )
COMPLAIN_RET ( " Only hero can sell creatures! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( hero - > Slots ( ) , slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Hero doesn't have any creature in that slot! " ) ;
const CStackInstance & s = hero - > getStack ( slot ) ;
2016-10-12 17:16:26 +02:00
if ( s . count < count //can't sell more creatures than have
2015-12-24 21:30:57 +03:00
| | ( hero - > stacksCount ( ) = = 1 & & hero - > needsLastStack ( ) & & s . count = = count ) ) //can't sell last stack
2014-03-17 19:51:07 +00:00
{
COMPLAIN_RET ( " Not enough creatures in army! " ) ;
}
int b1 , b2 ; //base quantities for trade
2016-10-12 17:16:26 +02:00
market - > getOffer ( s . type - > idNumber , resourceID , b1 , b2 , EMarketMode : : CREATURE_RESOURCE ) ;
int units = count / b1 ; //how many base quantities we trade
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( count % b1 ) //all offered units of resource should be used, if not -> somewhere in calculations must be an error
{
//TODO: complain?
assert ( 0 ) ;
}
2014-03-17 19:51:07 +00:00
changeStackCount ( StackLocation ( hero , slot ) , - count ) ;
2016-11-26 15:14:43 +03:00
giveResource ( hero - > tempOwner , resourceID , b2 * units ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
bool CGameHandler : : transformInUndead ( const IMarket * market , const CGHeroInstance * hero , SlotID slot )
{
const CArmedInstance * army = nullptr ;
if ( hero )
army = hero ;
else
army = dynamic_cast < const CGTownInstance * > ( market - > o ) ;
if ( ! army )
COMPLAIN_RET ( " Incorrect call to transform in undead! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! army - > hasStackAtSlot ( slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Army doesn't have any creature in that slot! " ) ;
const CStackInstance & s = army - > getStack ( slot ) ;
2015-09-04 18:08:25 +03:00
//resulting creature - bone dragons or skeletons
CreatureID resCreature = CreatureID : : SKELETON ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( s . hasBonusOfType ( Bonus : : DRAGON_NATURE )
2015-09-04 18:08:25 +03:00
| | ( s . getCreatureID ( ) = = CreatureID : : HYDRA )
| | ( s . getCreatureID ( ) = = CreatureID : : CHAOS_HYDRA ) )
resCreature = CreatureID : : BONE_DRAGON ;
changeStackType ( StackLocation ( army , slot ) , resCreature . toCreature ( ) ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
bool CGameHandler : : sendResources ( ui32 val , PlayerColor player , Res : : ERes r1 , PlayerColor r2 )
{
2016-09-18 11:53:51 +03:00
const PlayerState * p2 = getPlayer ( r2 , false ) ;
2016-10-12 17:16:26 +02:00
if ( ! p2 | | p2 - > status ! = EPlayerStatus : : INGAME )
2014-03-17 19:51:07 +00:00
{
complain ( " Dest player must be in game! " ) ;
return false ;
}
2016-11-26 15:14:43 +03:00
TResourceCap curRes1 = getPlayer ( player ) - > resources . at ( r1 ) ;
2014-03-17 19:51:07 +00:00
2016-11-26 15:14:43 +03:00
vstd : : amin ( val , curRes1 ) ;
2014-03-17 19:51:07 +00:00
2016-11-26 15:14:43 +03:00
giveResource ( player , r1 , - val ) ;
giveResource ( r2 , r1 , val ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
2016-09-08 19:29:15 +03:00
bool CGameHandler : : setFormation ( ObjectInstanceID hid , ui8 formation )
2014-03-17 19:51:07 +00:00
{
2016-09-08 19:29:15 +03:00
const CGHeroInstance * h = getHero ( hid ) ;
2016-10-12 17:16:26 +02:00
if ( ! h )
2016-09-08 19:29:15 +03:00
{
logGlobal - > error ( " Hero doesn't exist! " ) ;
return false ;
}
ChangeFormation cf ;
cf . hid = hid ;
cf . formation = formation ;
sendAndApply ( & cf ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
bool CGameHandler : : hireHero ( const CGObjectInstance * obj , ui8 hid , PlayerColor player )
{
2016-09-18 11:53:51 +03:00
const PlayerState * p = getPlayer ( player ) ;
const CGTownInstance * t = getTown ( obj - > id ) ;
2014-03-17 19:51:07 +00:00
//common preconditions
2016-10-12 17:16:26 +02:00
// if ((p->resources.at(Res::GOLD)<GOLD_NEEDED && complain("Not enough gold for buying hero!"))
2014-04-27 10:43:42 +04:00
// || (getHeroCount(player, false) >= GameConstants::MAX_HEROES_PER_PLAYER && complain("Cannot hire hero, only 8 wandering heroes are allowed!")))
2016-10-12 17:16:26 +02:00
if ( ( p - > resources . at ( Res : : GOLD ) < GameConstants : : HERO_GOLD_COST & & complain ( " Not enough gold for buying hero! " ) )
2014-04-27 10:43:42 +04:00
| | ( ( ! t ) & & ( getHeroCount ( player , false ) > = VLC - > modh - > settings . MAX_HEROES_ON_MAP_PER_PLAYER & & complain ( " Cannot hire hero, too many wandering heroes already! " ) ) )
2016-10-12 17:16:26 +02:00
| | ( ( t ) & & ( getHeroCount ( player , true ) > = VLC - > modh - > settings . MAX_HEROES_AVAILABLE_PER_PLAYER & & complain ( " Cannot hire hero, too many heroes garrizoned and wandering already! " ) ) ) )
2016-01-26 22:24:38 +03:00
{
return false ;
}
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( t ) //tavern in town
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ( ! t - > hasBuilt ( BuildingID : : TAVERN ) & & complain ( " No tavern! " ) )
2016-01-26 22:24:38 +03:00
| | ( t - > visitingHero & & complain ( " There is visiting hero - no place! " ) ) )
{
2014-03-17 19:51:07 +00:00
return false ;
2016-01-26 22:24:38 +03:00
}
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
else if ( obj - > ID = = Obj : : TAVERN )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( getTile ( obj - > visitablePos ( ) ) - > visitableObjects . back ( ) ! = obj & & complain ( " Tavern entry must be unoccupied! " ) )
2016-01-26 22:24:38 +03:00
{
2014-03-17 19:51:07 +00:00
return false ;
2016-01-26 22:24:38 +03:00
}
2014-03-17 19:51:07 +00:00
}
const CGHeroInstance * nh = p - > availableHeroes . at ( hid ) ;
if ( ! nh )
{
complain ( " Hero is not available for hiring! " ) ;
return false ;
}
HeroRecruited hr ;
hr . tid = obj - > id ;
hr . hid = nh - > subID ;
hr . player = player ;
hr . tile = obj - > visitablePos ( ) + nh - > getVisitableOffset ( ) ;
sendAndApply ( & hr ) ;
std : : map < ui32 , ConstTransitivePtr < CGHeroInstance > > pool = gs - > unusedHeroesFromPool ( ) ;
const CGHeroInstance * theOtherHero = p - > availableHeroes . at ( ! hid ) ;
const CGHeroInstance * newHero = nullptr ;
if ( theOtherHero ) //on XXL maps all heroes can be imprisoned :(
2016-01-26 22:24:38 +03:00
{
2016-09-08 00:54:26 +03:00
newHero = gs - > hpool . pickHeroFor ( false , player , getNativeTown ( player ) , pool , getRandomGenerator ( ) , theOtherHero - > type - > heroClass ) ;
2016-01-26 22:24:38 +03:00
}
2014-03-17 19:51:07 +00:00
SetAvailableHeroes sah ;
sah . player = player ;
2016-10-12 17:16:26 +02:00
if ( newHero )
2014-03-17 19:51:07 +00:00
{
sah . hid [ hid ] = newHero - > subID ;
sah . army [ hid ] . clear ( ) ;
sah . army [ hid ] . setCreature ( SlotID ( 0 ) , newHero - > type - > initialArmy [ 0 ] . creature , 1 ) ;
}
else
2016-01-26 22:24:38 +03:00
{
2014-03-17 19:51:07 +00:00
sah . hid [ hid ] = - 1 ;
2016-01-26 22:24:38 +03:00
}
2014-03-17 19:51:07 +00:00
sah . hid [ ! hid ] = theOtherHero ? theOtherHero - > subID : - 1 ;
sendAndApply ( & sah ) ;
2016-11-26 15:14:43 +03:00
giveResource ( player , Res : : GOLD , - GameConstants : : HERO_GOLD_COST ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( t )
2014-03-17 19:51:07 +00:00
{
vistiCastleObjects ( t , nh ) ;
giveSpells ( t , nh ) ;
}
return true ;
}
2017-06-06 07:53:51 +03:00
bool CGameHandler : : queryReply ( QueryID qid , const JsonNode & answer , PlayerColor player )
2014-03-17 19:51:07 +00:00
{
boost : : unique_lock < boost : : recursive_mutex > lock ( gsm ) ;
2017-06-06 07:53:51 +03:00
logGlobal - > trace ( " Player %s attempts answering query %d with answer: " , player , qid ) ;
2017-08-11 16:27:42 +03:00
logGlobal - > trace ( answer . toJson ( ) ) ;
2014-03-17 19:51:07 +00:00
auto topQuery = queries . topQuery ( player ) ;
COMPLAIN_RET_FALSE_IF ( ! topQuery , " This player doesn't have any queries! " ) ;
COMPLAIN_RET_FALSE_IF ( topQuery - > queryID ! = qid , " This player top query has different ID! " ) ;
COMPLAIN_RET_FALSE_IF ( ! topQuery - > endsByPlayerAnswer ( ) , " This query cannot be ended by player's answer! " ) ;
2017-06-06 07:53:51 +03:00
topQuery - > setReply ( answer ) ;
2014-03-17 19:51:07 +00:00
queries . popQuery ( topQuery ) ;
return true ;
}
static EndAction end_action ;
2016-02-13 17:40:31 +03:00
void CGameHandler : : updateGateState ( )
2016-01-29 22:43:35 +03:00
{
2016-02-13 17:40:31 +03:00
BattleUpdateGateState db ;
db . state = gs - > curB - > si . gateState ;
2016-10-12 17:16:26 +02:00
if ( gs - > curB - > si . wallState [ EWallPart : : GATE ] = = EWallState : : DESTROYED )
2016-01-29 22:43:35 +03:00
{
2016-02-13 17:40:31 +03:00
db . state = EGateState : : DESTROYED ;
2016-01-29 22:43:35 +03:00
}
2016-10-12 17:16:26 +02:00
else if ( db . state = = EGateState : : OPENED )
2016-01-29 22:43:35 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! gs - > curB - > battleGetStackByPos ( BattleHex ( ESiegeHex : : GATE_OUTER ) , false ) & &
2016-02-09 17:38:59 +03:00
! gs - > curB - > battleGetStackByPos ( BattleHex ( ESiegeHex : : GATE_INNER ) , false ) )
2016-01-29 22:43:35 +03:00
{
2016-10-12 17:16:26 +02:00
if ( gs - > curB - > town - > subID = = ETownType : : FORTRESS )
2016-02-09 10:45:59 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! gs - > curB - > battleGetStackByPos ( BattleHex ( ESiegeHex : : GATE_BRIDGE ) , false ) )
2016-02-13 17:40:31 +03:00
db . state = EGateState : : CLOSED ;
2016-02-09 10:45:59 +03:00
}
2016-10-12 17:16:26 +02:00
else if ( gs - > curB - > battleGetStackByPos ( BattleHex ( ESiegeHex : : GATE_BRIDGE ) ) )
2016-02-13 17:40:31 +03:00
db . state = EGateState : : BLOCKED ;
2016-02-09 10:45:59 +03:00
else
2016-02-13 17:40:31 +03:00
db . state = EGateState : : CLOSED ;
2016-01-29 22:43:35 +03:00
}
}
2016-10-12 17:16:26 +02:00
else if ( gs - > curB - > battleGetStackByPos ( BattleHex ( ESiegeHex : : GATE_BRIDGE ) , false ) )
2016-02-13 17:40:31 +03:00
db . state = EGateState : : BLOCKED ;
2016-02-08 12:15:07 +03:00
else
2016-02-13 17:40:31 +03:00
db . state = EGateState : : CLOSED ;
2016-02-08 12:15:07 +03:00
2016-10-12 17:16:26 +02:00
if ( db . state ! = gs - > curB - > si . gateState )
2016-02-08 12:15:07 +03:00
sendAndApply ( & db ) ;
2016-01-29 22:43:35 +03:00
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : makeBattleAction ( BattleAction & ba )
2014-03-17 19:51:07 +00:00
{
bool ok = true ;
const CStack * stack = battleGetStackByID ( ba . stackNumber ) ; //may be nullptr if action is not about stack
const CStack * destinationStack = ba . actionType = = Battle : : WALK_AND_ATTACK ? gs - > curB - > battleGetStackByPos ( ba . additionalInfo )
: ba . actionType = = Battle : : SHOOT ? gs - > curB - > battleGetStackByPos ( ba . destinationTile )
: nullptr ;
const bool isAboutActiveStack = stack & & ( stack = = battleActiveStack ( ) ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > trace (
" Making action: type=%d; side=%d; stack=%s; dst=%s; additionalInfo=%d; stackAtDst=%s " ,
ba . actionType , ( int ) ba . side , ( stack ? stack - > getName ( ) : std : : string ( " none " ) ) ,
ba . destinationTile , ba . additionalInfo , ( destinationStack ? destinationStack - > getName ( ) : std : : string ( " none " ) ) ) ;
2014-03-17 19:51:07 +00:00
switch ( ba . actionType )
{
case Battle : : WALK : //walk
case Battle : : DEFEND : //defend
case Battle : : WAIT : //wait
case Battle : : WALK_AND_ATTACK : //walk or attack
case Battle : : SHOOT : //shoot
case Battle : : CATAPULT : //catapult
case Battle : : STACK_HEAL : //healing with First Aid Tent
case Battle : : DAEMON_SUMMONING :
case Battle : : MONSTER_SPELL :
2016-10-12 17:16:26 +02:00
if ( ! stack )
2014-03-17 19:51:07 +00:00
{
complain ( " No such stack! " ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
if ( ! stack - > alive ( ) )
2014-03-17 19:51:07 +00:00
{
complain ( " This stack is dead: " + stack - > nodeName ( ) ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
if ( battleTacticDist ( ) )
2014-03-17 19:51:07 +00:00
{
2017-07-01 11:34:00 +03:00
if ( stack & & stack - > side ! = battleGetTacticsSide ( ) )
2014-03-17 19:51:07 +00:00
{
complain ( " This is not a stack of side that has tactics! " ) ;
return false ;
}
}
2016-10-12 17:16:26 +02:00
else if ( ! isAboutActiveStack )
2014-03-17 19:51:07 +00:00
{
complain ( " Action has to be about active stack! " ) ;
return false ;
}
}
2016-11-26 19:46:34 +03:00
auto wrapAction = [ this ] ( BattleAction & ba )
{
StartAction startAction ( ba ) ;
sendAndApply ( & startAction ) ;
2017-07-19 02:06:05 +03:00
return vstd : : makeScopeGuard ( [ & ] ( )
2016-11-26 19:46:34 +03:00
{
sendAndApply ( & end_action ) ;
} ) ;
} ;
2014-03-17 19:51:07 +00:00
switch ( ba . actionType )
{
case Battle : : END_TACTIC_PHASE : //wait
case Battle : : BAD_MORALE :
case Battle : : NO_ACTION :
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
break ;
}
case Battle : : WALK :
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
int walkedTiles = moveStack ( ba . stackNumber , ba . destinationTile ) ; //move
2016-10-12 17:16:26 +02:00
if ( ! walkedTiles )
2014-03-17 19:51:07 +00:00
complain ( " Stack failed movement! " ) ;
break ;
}
case Battle : : DEFEND :
{
2017-07-10 04:05:36 +03:00
//defensive stance
2014-03-17 19:51:07 +00:00
SetStackEffect sse ;
2017-07-10 04:05:36 +03:00
Bonus bonus1 ( Bonus : : STACK_GETS_TURN , Bonus : : PRIMARY_SKILL , Bonus : : OTHER , 20 , - 1 , PrimarySkill : : DEFENSE , Bonus : : PERCENT_TO_ALL ) ;
Bonus bonus2 ( Bonus : : STACK_GETS_TURN , Bonus : : PRIMARY_SKILL , Bonus : : OTHER , stack - > valOfBonuses ( Bonus : : DEFENSIVE_STANCE ) ,
- 1 , PrimarySkill : : DEFENSE , Bonus : : ADDITIVE_VALUE ) ;
2017-07-10 08:53:29 +03:00
BonusList defence = * stack - > getBonuses ( Selector : : typeSubtype ( Bonus : : PRIMARY_SKILL , PrimarySkill : : DEFENSE ) ) ;
int oldDefenceValue = defence . totalValue ( ) ;
2017-07-10 04:05:36 +03:00
2017-07-10 08:53:29 +03:00
defence . push_back ( std : : make_shared < Bonus > ( bonus1 ) ) ;
defence . push_back ( std : : make_shared < Bonus > ( bonus2 ) ) ;
2017-07-10 04:05:36 +03:00
2017-07-10 08:53:29 +03:00
int difference = defence . totalValue ( ) - oldDefenceValue ;
2017-07-10 04:05:36 +03:00
MetaString text ;
stack - > addText ( text , MetaString : : GENERAL_TXT , 120 ) ;
stack - > addNameReplacement ( text ) ;
text . addReplacement ( difference ) ;
sse . battleLog . push_back ( text ) ;
sse . effect . push_back ( bonus1 ) ;
sse . effect . push_back ( bonus2 ) ;
2014-03-17 19:51:07 +00:00
sse . stacks . push_back ( ba . stackNumber ) ;
sendAndApply ( & sse ) ;
//don't break - we share code with next case
}
case Battle : : WAIT :
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
break ;
}
case Battle : : RETREAT : //retreat/flee
{
2016-10-12 17:16:26 +02:00
if ( ! gs - > curB - > battleCanFlee ( gs - > curB - > sides . at ( ba . side ) . color ) )
2014-03-17 19:51:07 +00:00
complain ( " Cannot retreat! " ) ;
else
setBattleResult ( BattleResult : : ESCAPE , ! ba . side ) ; //surrendering side loses
break ;
}
case Battle : : SURRENDER :
{
PlayerColor player = gs - > curB - > sides . at ( ba . side ) . color ;
int cost = gs - > curB - > battleGetSurrenderCost ( player ) ;
2016-10-12 17:16:26 +02:00
if ( cost < 0 )
2014-03-17 19:51:07 +00:00
complain ( " Cannot surrender! " ) ;
2016-10-12 17:16:26 +02:00
else if ( getResource ( player , Res : : GOLD ) < cost )
2014-03-17 19:51:07 +00:00
complain ( " Not enough gold to surrender! " ) ;
else
{
giveResource ( player , Res : : GOLD , - cost ) ;
setBattleResult ( BattleResult : : SURRENDER , ! ba . side ) ; //surrendering side loses
}
break ;
}
case Battle : : WALK_AND_ATTACK : //walk or attack
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2016-10-12 17:16:26 +02:00
if ( ! stack | | ! destinationStack )
2014-03-17 19:51:07 +00:00
{
break ;
}
BattleHex startingPos = stack - > position ;
int distance = moveStack ( ba . stackNumber , ba . destinationTile ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " %s will attack %s " , stack - > nodeName ( ) , destinationStack - > nodeName ( ) ) ;
2014-03-17 19:51:07 +00:00
2017-07-01 11:34:00 +03:00
if ( stack - > position ! = ba . destinationTile //we wasn't able to reach destination tile
2017-07-03 12:59:33 +03:00
& & ! ( stack - > doubleWide ( ) & & ( stack - > position = = ba . destinationTile . cloneInDirection ( stack - > destShiftDir ( ) , false ) ) ) //nor occupy specified hex
2014-03-17 19:51:07 +00:00
)
{
2016-08-30 01:11:54 +03:00
complain ( " We cannot move this stack to its destination " + stack - > getCreature ( ) - > namePl ) ;
2014-03-17 19:51:07 +00:00
ok = false ;
break ;
}
2016-10-12 17:16:26 +02:00
if ( destinationStack & & stack & & stack - > ID = = destinationStack - > ID ) //we should just move, it will be handled by following check
2014-03-17 19:51:07 +00:00
{
destinationStack = nullptr ;
}
2016-10-12 17:16:26 +02:00
if ( ! destinationStack )
2014-03-17 19:51:07 +00:00
{
complain ( boost : : str ( boost : : format ( " walk and attack error: no stack at additionalInfo tile (%d)! \n " ) % ba . additionalInfo ) ) ;
ok = false ;
break ;
}
2016-10-12 17:16:26 +02:00
if ( ! CStack : : isMeleeAttackPossible ( stack , destinationStack ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Attack cannot be performed! " ) ;
ok = false ;
break ;
}
//attack
int totalAttacks = 1 + stack - > getBonuses ( Selector : : type ( Bonus : : ADDITIONAL_ATTACK ) ,
( Selector : : effectRange ( Bonus : : NO_LIMIT ) . Or ( Selector : : effectRange ( Bonus : : ONLY_MELEE_FIGHT ) ) ) ) - > totalValue ( ) ; //all unspicified attacks + melee attacks
for ( int i = 0 ; i < totalAttacks ; + + i )
{
if ( stack & &
stack - > alive ( ) & & //move can cause death, eg. by walking into the moat
destinationStack - > alive ( ) )
{
BattleAttack bat ;
prepareAttack ( bat , stack , destinationStack , ( i ? 0 : distance ) , ba . additionalInfo ) ; //no distance travelled on second attack
//prepareAttack(bat, stack, stackAtEnd, 0, ba.additionalInfo);
2016-10-12 17:19:02 +02:00
handleAttackBeforeCasting ( & bat ) ; //only before first attack
2014-03-17 19:51:07 +00:00
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
//counterattack
2017-02-10 00:36:15 +01:00
if ( i = = 0 & & destinationStack
2014-03-17 19:51:07 +00:00
& & ! stack - > hasBonusOfType ( Bonus : : BLOCKS_RETALIATION )
& & destinationStack - > ableToRetaliate ( )
& & stack - > alive ( ) ) //attacker may have died (fire shield)
{
BattleAttack bat ;
prepareAttack ( bat , destinationStack , stack , 0 , stack - > position ) ;
bat . flags | = BattleAttack : : COUNTER ;
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
}
//return
2016-10-12 17:16:26 +02:00
if ( stack - > hasBonusOfType ( Bonus : : RETURN_AFTER_STRIKE ) & & startingPos ! = stack - > position & & stack - > alive ( ) )
2014-03-17 19:51:07 +00:00
{
moveStack ( ba . stackNumber , startingPos ) ;
//NOTE: curStack->ID == ba.stackNumber (rev 1431)
}
break ;
}
case Battle : : SHOOT :
{
2016-10-12 17:16:26 +02:00
if ( ! gs - > curB - > battleCanShoot ( stack , ba . destinationTile ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot shoot! " ) ;
break ;
}
2017-05-28 15:00:55 +03:00
if ( ! destinationStack )
{
complain ( " No target to shoot! " ) ;
break ;
}
2014-03-17 19:51:07 +00:00
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
{
BattleAttack bat ;
bat . flags | = BattleAttack : : SHOT ;
prepareAttack ( bat , stack , destinationStack , 0 , ba . destinationTile ) ;
2016-10-12 17:19:02 +02:00
handleAttackBeforeCasting ( & bat ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
2017-02-04 10:33:45 +01:00
//ranged counterattack
2017-05-28 15:00:55 +03:00
if ( destinationStack - > hasBonusOfType ( Bonus : : RANGED_RETALIATION )
2017-02-04 10:33:45 +01:00
& & ! stack - > hasBonusOfType ( Bonus : : BLOCKS_RANGED_RETALIATION )
& & destinationStack - > ableToRetaliate ( )
& & stack - > alive ( ) ) //attacker may have died (fire shield)
{
BattleAttack bat ;
prepareAttack ( bat , destinationStack , stack , 0 , stack - > position ) ;
bat . flags | = BattleAttack : : COUNTER | BattleAttack : : SHOT ;
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
2014-03-17 19:51:07 +00:00
//second shot for ballista, only if hero has advanced artillery
const CGHeroInstance * attackingHero = gs - > curB - > battleGetFightingHero ( ba . side ) ;
2017-07-09 10:18:46 +02:00
if ( destinationStack - > alive ( )
& & ( stack - > getCreature ( ) - > idNumber = = CreatureID : : BALLISTA )
& & ( attackingHero - > getSecSkillLevel ( SecondarySkill : : ARTILLERY ) > = SecSkillLevel : : ADVANCED )
)
2014-03-17 19:51:07 +00:00
{
BattleAttack bat2 ;
bat2 . flags | = BattleAttack : : SHOT ;
prepareAttack ( bat2 , stack , destinationStack , 0 , ba . destinationTile ) ;
sendAndApply ( & bat2 ) ;
}
//allow more than one additional attack
int additionalAttacks = stack - > getBonuses ( Selector : : type ( Bonus : : ADDITIONAL_ATTACK ) ,
( Selector : : effectRange ( Bonus : : NO_LIMIT ) . Or ( Selector : : effectRange ( Bonus : : ONLY_DISTANCE_FIGHT ) ) ) ) - > totalValue ( ) ;
2017-07-04 14:24:46 +03:00
for ( int i = 0 ; i < additionalAttacks ; + + i )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if (
2014-03-17 19:51:07 +00:00
stack - > alive ( )
& & destinationStack - > alive ( )
2017-07-04 14:24:46 +03:00
& & stack - > shots . canUse ( )
2014-03-17 19:51:07 +00:00
)
{
BattleAttack bat ;
bat . flags | = BattleAttack : : SHOT ;
prepareAttack ( bat , stack , destinationStack , 0 , ba . destinationTile ) ;
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
}
break ;
}
case Battle : : CATAPULT :
{
auto getCatapultHitChance = [ & ] ( EWallPart : : EWallPart part , const CHeroHandler : : SBallisticsLevelInfo & sbi ) - > int
{
switch ( part )
{
case EWallPart : : GATE :
return sbi . gate ;
case EWallPart : : KEEP :
return sbi . keep ;
case EWallPart : : BOTTOM_TOWER :
case EWallPart : : UPPER_TOWER :
return sbi . tower ;
case EWallPart : : BOTTOM_WALL :
case EWallPart : : BELOW_GATE :
case EWallPart : : OVER_GATE :
case EWallPart : : UPPER_WALL :
return sbi . wall ;
default :
return 0 ;
}
} ;
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
const CGHeroInstance * attackingHero = gs - > curB - > battleGetFightingHero ( ba . side ) ;
2017-02-01 11:25:57 +01:00
CHeroHandler : : SBallisticsLevelInfo sbi ;
if ( stack - > getCreature ( ) - > idNumber = = CreatureID : : CATAPULT )
sbi = VLC - > heroh - > ballistics . at ( attackingHero - > getSecSkillLevel ( SecondarySkill : : BALLISTICS ) ) ;
else //may need to use higher ballistics level for creatures in future for some cases to match original H3 (upgraded cyclops etc)
{
sbi = VLC - > heroh - > ballistics . at ( 1 ) ;
2017-02-01 13:38:28 +01:00
sbi . shots + = std : : max ( stack - > valOfBonuses ( Bonus : : CATAPULT_EXTRA_SHOTS ) , 0 ) ;
2017-02-01 11:25:57 +01:00
}
2014-03-17 19:51:07 +00:00
auto wallPart = gs - > curB - > battleHexToWallPart ( ba . destinationTile ) ;
2016-10-12 17:16:26 +02:00
if ( ! gs - > curB - > isWallPartPotentiallyAttackable ( wallPart ) )
2014-03-17 19:51:07 +00:00
{
complain ( " catapult tried to attack non-catapultable hex! " ) ;
break ;
}
//in successive iterations damage is dealt but not yet subtracted from wall's HPs
auto & currentHP = gs - > curB - > si . wallState ;
if ( currentHP . at ( wallPart ) = = EWallState : : DESTROYED | | currentHP . at ( wallPart ) = = EWallState : : NONE )
{
complain ( " catapult tried to attack already destroyed wall part! " ) ;
break ;
}
2016-10-12 17:16:26 +02:00
for ( int g = 0 ; g < sbi . shots ; + + g )
2014-03-17 19:51:07 +00:00
{
bool hitSuccessfull = false ;
auto attackedPart = wallPart ;
do // catapult has chance to attack desired target. Othervice - attacks randomly
{
2016-10-12 17:16:26 +02:00
if ( currentHP . at ( attackedPart ) ! = EWallState : : DESTROYED & & // this part can be hit
2014-03-17 19:51:07 +00:00
currentHP . at ( attackedPart ) ! = EWallState : : NONE & &
2016-09-08 00:54:26 +03:00
getRandomGenerator ( ) . nextInt ( 99 ) < getCatapultHitChance ( attackedPart , sbi ) ) //hit is successful
2014-03-17 19:51:07 +00:00
{
hitSuccessfull = true ;
}
else // select new target
{
std : : vector < EWallPart : : EWallPart > allowedTargets ;
for ( size_t i = 0 ; i < currentHP . size ( ) ; i + + )
{
2017-07-09 10:18:46 +02:00
if ( currentHP . at ( i ) ! = EWallState : : DESTROYED & &
currentHP . at ( i ) ! = EWallState : : NONE )
2014-03-17 19:51:07 +00:00
allowedTargets . push_back ( EWallPart : : EWallPart ( i ) ) ;
}
if ( allowedTargets . empty ( ) )
break ;
2016-09-08 00:54:26 +03:00
attackedPart = * RandomGeneratorUtil : : nextItem ( allowedTargets , getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
}
}
while ( ! hitSuccessfull ) ;
if ( ! hitSuccessfull ) // break triggered - no target to shoot at
break ;
CatapultAttack ca ; //package for clients
CatapultAttack : : AttackInfo attack ;
attack . attackedPart = attackedPart ;
attack . destinationTile = ba . destinationTile ;
attack . damageDealt = 0 ;
int dmgChance [ ] = { sbi . noDmg , sbi . oneDmg , sbi . twoDmg } ; //dmgChance[i] - chance for doing i dmg when hit is successful
2016-09-08 00:54:26 +03:00
int dmgRand = getRandomGenerator ( ) . nextInt ( 99 ) ;
2014-03-17 19:51:07 +00:00
//accumulating dmgChance
dmgChance [ 1 ] + = dmgChance [ 0 ] ;
dmgChance [ 2 ] + = dmgChance [ 1 ] ;
//calculating dealt damage
2016-10-12 17:16:26 +02:00
for ( int damage = 0 ; damage < ARRAY_COUNT ( dmgChance ) ; + + damage )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( dmgRand < = dmgChance [ damage ] )
2014-03-17 19:51:07 +00:00
{
attack . damageDealt = damage ;
break ;
}
}
// attacked tile may have changed - update destination
attack . destinationTile = gs - > curB - > wallPartToBattleHex ( EWallPart : : EWallPart ( attack . attackedPart ) ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Catapult attacks %d dealing %d damage " , ( int ) attack . attackedPart , ( int ) attack . damageDealt ) ;
2014-03-17 19:51:07 +00:00
//removing creatures in turrets / keep if one is destroyed
2016-10-12 17:16:26 +02:00
if ( attack . damageDealt > 0 & & ( attackedPart = = EWallPart : : KEEP | |
2014-03-17 19:51:07 +00:00
attackedPart = = EWallPart : : BOTTOM_TOWER | | attackedPart = = EWallPart : : UPPER_TOWER ) )
{
int posRemove = - 1 ;
switch ( attackedPart )
{
case EWallPart : : KEEP :
posRemove = - 2 ;
break ;
case EWallPart : : BOTTOM_TOWER :
posRemove = - 3 ;
break ;
case EWallPart : : UPPER_TOWER :
posRemove = - 4 ;
break ;
}
BattleStacksRemoved bsr ;
2016-10-12 17:16:26 +02:00
for ( auto & elem : gs - > curB - > stacks )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( elem - > position = = posRemove )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
bsr . stackIDs . insert ( elem - > ID ) ;
2014-03-17 19:51:07 +00:00
break ;
}
}
sendAndApply ( & bsr ) ;
}
ca . attacker = ba . stackNumber ;
ca . attackedParts . push_back ( attack ) ;
sendAndApply ( & ca ) ;
}
//finish by scope guard
break ;
}
case Battle : : STACK_HEAL : //healing with First Aid Tent
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
const CGHeroInstance * attackingHero = gs - > curB - > battleGetFightingHero ( ba . side ) ;
const CStack * healer = gs - > curB - > battleGetStackByID ( ba . stackNumber ) ,
* destStack = gs - > curB - > battleGetStackByPos ( ba . destinationTile ) ;
2015-09-16 10:08:40 +03:00
2017-07-04 14:24:46 +03:00
if ( healer = = nullptr | | destStack = = nullptr | | ! healer - > hasBonusOfType ( Bonus : : HEALER ) )
2014-03-17 19:51:07 +00:00
{
complain ( " There is either no healer, no destination, or healer cannot heal :P " ) ;
}
2015-09-16 10:08:40 +03:00
else
{
2017-07-04 14:24:46 +03:00
int32_t toHeal = healer - > getCount ( ) * std : : max ( 10 , attackingHero - > valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : FIRST_AID ) ) ;
2014-03-17 19:51:07 +00:00
2017-07-04 14:24:46 +03:00
//TODO: allow resurrection for mods
CHealth health = destStack - > healthAfterHealed ( toHeal , EHealLevel : : HEAL , EHealPower : : PERMANENT ) ;
2014-03-17 19:51:07 +00:00
2017-07-04 14:24:46 +03:00
if ( toHeal = = 0 )
{
logGlobal - > warn ( " Nothing to heal " ) ;
}
else
{
StacksHealedOrResurrected shr ;
shr . lifeDrain = false ;
shr . tentHealing = true ;
shr . cure = false ;
shr . drainedFrom = ba . stackNumber ;
2014-03-17 19:51:07 +00:00
2017-07-04 14:24:46 +03:00
CHealthInfo hi ;
health . toInfo ( hi ) ;
2017-07-09 19:49:52 +03:00
hi . stackId = destStack - > ID ;
hi . delta = toHeal ;
2017-07-04 14:24:46 +03:00
shr . healedStacks . push_back ( hi ) ;
sendAndApply ( & shr ) ;
}
2014-03-17 19:51:07 +00:00
}
break ;
}
case Battle : : DAEMON_SUMMONING :
//TODO: From Strategija:
//Summon Demon is a level 2 spell.
{
const CStack * summoner = gs - > curB - > battleGetStackByID ( ba . stackNumber ) ,
* destStack = gs - > curB - > battleGetStackByPos ( ba . destinationTile , false ) ;
2015-11-07 11:42:06 +03:00
CreatureID summonedType ( summoner - > getBonusLocalFirst ( Selector : : type ( Bonus : : DAEMON_SUMMONING ) ) - > subtype ) ; //in case summoner can summon more than one type of monsters... scream!
2014-03-17 19:51:07 +00:00
BattleStackAdded bsa ;
2017-07-01 11:34:00 +03:00
bsa . side = summoner - > side ;
2014-03-17 19:51:07 +00:00
2016-01-30 00:53:53 +03:00
bsa . creID = summonedType ;
2017-07-04 14:24:46 +03:00
ui64 risedHp = summoner - > getCount ( ) * summoner - > valOfBonuses ( Bonus : : DAEMON_SUMMONING , bsa . creID . toEnum ( ) ) ;
2016-09-10 09:37:18 +03:00
ui64 targetHealth = destStack - > getCreature ( ) - > MaxHealth ( ) * destStack - > baseAmount ;
2014-03-17 19:51:07 +00:00
ui64 canRiseHp = std : : min ( targetHealth , risedHp ) ;
ui32 canRiseAmount = canRiseHp / VLC - > creh - > creatures . at ( bsa . creID ) - > MaxHealth ( ) ;
bsa . amount = std : : min ( canRiseAmount , destStack - > baseAmount ) ;
2017-07-01 11:34:00 +03:00
bsa . pos = gs - > curB - > getAvaliableHex ( bsa . creID , bsa . side , destStack - > position ) ;
2014-03-17 19:51:07 +00:00
bsa . summoned = false ;
if ( bsa . amount ) //there's rare possibility single creature cannot rise desired type
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2015-11-07 11:42:06 +03:00
2014-03-17 19:51:07 +00:00
BattleStacksRemoved bsr ; //remove body
bsr . stackIDs . insert ( destStack - > ID ) ;
sendAndApply ( & bsr ) ;
sendAndApply ( & bsa ) ;
BattleSetStackProperty ssp ;
ssp . stackID = ba . stackNumber ;
ssp . which = BattleSetStackProperty : : CASTS ; //reduce number of casts
ssp . val = - 1 ;
ssp . absolute = false ;
sendAndApply ( & ssp ) ;
2015-11-07 11:42:06 +03:00
}
2014-03-17 19:51:07 +00:00
break ;
}
case Battle : : MONSTER_SPELL :
{
2016-11-26 19:46:34 +03:00
auto wrapper = wrapAction ( ba ) ;
2014-03-17 19:51:07 +00:00
const CStack * stack = gs - > curB - > battleGetStackByID ( ba . stackNumber ) ;
SpellID spellID = SpellID ( ba . additionalInfo ) ;
BattleHex destination ( ba . destinationTile ) ;
2016-09-19 23:36:35 +02:00
const std : : shared_ptr < Bonus > randSpellcaster = stack - > getBonusLocalFirst ( Selector : : type ( Bonus : : RANDOM_SPELLCASTER ) ) ;
const std : : shared_ptr < Bonus > spellcaster = stack - > getBonusLocalFirst ( Selector : : typeSubtype ( Bonus : : SPELLCASTER , spellID ) ) ;
2014-03-17 19:51:07 +00:00
//TODO special bonus for genies ability
2016-10-12 17:16:26 +02:00
if ( randSpellcaster & & battleGetRandomStackSpell ( getRandomGenerator ( ) , stack , CBattleInfoCallback : : RANDOM_AIMED ) < 0 )
2016-09-09 20:30:36 +03:00
spellID = battleGetRandomStackSpell ( getRandomGenerator ( ) , stack , CBattleInfoCallback : : RANDOM_GENIE ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( spellID < 0 )
2014-03-17 19:51:07 +00:00
complain ( " That stack can't cast spells! " ) ;
else
{
2015-10-28 23:53:44 +03:00
const CSpell * spell = SpellID ( spellID ) . toSpell ( ) ;
BattleSpellCastParameters parameters ( gs - > curB , stack , spell ) ;
2015-09-17 05:47:32 +03:00
parameters . spellLvl = 0 ;
2014-03-17 19:51:07 +00:00
if ( spellcaster )
2015-09-17 05:47:32 +03:00
vstd : : amax ( parameters . spellLvl , spellcaster - > val ) ;
2014-03-17 19:51:07 +00:00
if ( randSpellcaster )
2015-09-17 05:47:32 +03:00
vstd : : amax ( parameters . spellLvl , randSpellcaster - > val ) ;
2016-10-12 17:16:26 +02:00
vstd : : amin ( parameters . spellLvl , 3 ) ;
2015-09-17 08:42:30 +03:00
parameters . effectLevel = parameters . spellLvl ;
2015-09-17 05:47:32 +03:00
parameters . mode = ECastingMode : : CREATURE_ACTIVE_CASTING ;
2015-09-29 21:47:04 +03:00
parameters . aimToHex ( destination ) ; //todo: allow multiple destinations
2016-09-06 13:33:11 +03:00
parameters . cast ( spellEnv ) ;
2014-03-17 19:51:07 +00:00
}
break ;
}
}
2017-07-09 10:18:46 +02:00
if ( ba . actionType = = Battle : : DAEMON_SUMMONING | | ba . actionType = = Battle : : WAIT | | ba . actionType = = Battle : : DEFEND
| | ba . actionType = = Battle : : SHOOT | | ba . actionType = = Battle : : MONSTER_SPELL )
handleDamageFromObstacle ( stack ) ;
if ( ba . stackNumber = = gs - > curB - > activeStack | | battleResult . get ( ) ) //active stack has moved or battle has finished
2014-03-17 19:51:07 +00:00
battleMadeAction . setn ( true ) ;
return ok ;
}
2016-10-02 16:21:46 +03:00
void CGameHandler : : playerMessage ( PlayerColor player , const std : : string & message , ObjectInstanceID currObj )
2014-09-21 16:42:08 +03:00
{
2016-10-02 16:21:46 +03:00
bool cheated = true ;
2014-09-21 20:35:53 +03:00
PlayerMessage temp_message ( player , message , ObjectInstanceID ( - 1 ) ) ; // don't inform other client on selected object
2014-03-17 19:51:07 +00:00
sendAndApply ( & temp_message ) ;
2016-10-02 16:21:46 +03:00
std : : vector < std : : string > cheat ;
boost : : split ( cheat , message , boost : : is_any_of ( " " ) ) ;
int obj = 0 ;
2016-10-12 17:16:26 +02:00
if ( cheat . size ( ) = = 2 )
2014-03-17 19:51:07 +00:00
{
2016-10-02 16:21:46 +03:00
obj = std : : atoi ( cheat [ 1 ] . c_str ( ) ) ;
2016-10-12 17:16:26 +02:00
if ( obj )
2016-10-02 16:21:46 +03:00
currObj = ObjectInstanceID ( obj ) ;
}
2014-03-17 19:51:07 +00:00
2016-10-02 16:21:46 +03:00
const CGHeroInstance * hero = getHero ( currObj ) ;
const CGTownInstance * town = getTown ( currObj ) ;
2016-10-12 17:16:26 +02:00
if ( ! town & & hero )
2016-10-02 16:21:46 +03:00
town = hero - > visitedTown ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( cheat . size ( ) = = 1 | | obj )
2016-10-02 16:21:46 +03:00
handleCheatCode ( cheat [ 0 ] , player , hero , town , cheated ) ;
else
{
2016-10-12 17:16:26 +02:00
for ( const auto & i : gs - > players )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( i . first = = PlayerColor : : NEUTRAL )
2016-10-02 16:21:46 +03:00
continue ;
2016-10-12 17:16:26 +02:00
if ( cheat [ 1 ] = = " ai " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( i . second . human )
2016-10-02 16:21:46 +03:00
continue ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat [ 1 ] ! = " all " & & cheat [ 1 ] ! = i . first . getStr ( ) )
2016-10-02 16:21:46 +03:00
continue ;
2016-10-12 17:16:26 +02:00
if ( cheat [ 0 ] = = " vcmiformenos " | | cheat [ 0 ] = = " vcmieagles " | | cheat [ 0 ] = = " vcmiungoliant " )
2016-10-02 16:21:46 +03:00
{
handleCheatCode ( cheat [ 0 ] , i . first , nullptr , nullptr , cheated ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat [ 0 ] = = " vcmiarmenelos " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
for ( const auto & t : i . second . towns )
2016-10-02 16:21:46 +03:00
{
handleCheatCode ( cheat [ 0 ] , i . first , nullptr , t , cheated ) ;
}
}
else
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
for ( const auto & h : i . second . heroes )
2014-03-17 19:51:07 +00:00
{
2016-10-02 16:21:46 +03:00
handleCheatCode ( cheat [ 0 ] , i . first , h , nullptr , cheated ) ;
2014-03-17 19:51:07 +00:00
}
}
}
}
2016-10-12 17:16:26 +02:00
if ( cheated )
2014-03-17 19:51:07 +00:00
{
SystemMessage temp_message ( VLC - > generaltexth - > allTexts . at ( 260 ) ) ;
sendAndApply ( & temp_message ) ;
2017-06-03 08:25:10 +03:00
if ( ! player . isSpectator ( ) )
checkVictoryLossConditionsForPlayer ( player ) ; //Player enter win code or got required art\creature
2014-09-21 16:42:08 +03:00
}
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : makeCustomAction ( BattleAction & ba )
2014-03-17 19:51:07 +00:00
{
switch ( ba . actionType )
{
case Battle : : HERO_SPELL :
{
COMPLAIN_RET_FALSE_IF ( ba . side > 1 , " Side must be 0 or 1! " ) ;
const CGHeroInstance * h = gs - > curB - > battleGetFightingHero ( ba . side ) ;
2016-08-30 01:11:54 +03:00
COMPLAIN_RET_FALSE_IF ( ( ! h ) , " Wrong caster! " ) ;
2016-10-02 15:22:55 +03:00
const CSpell * s = SpellID ( ba . additionalInfo ) . toSpell ( ) ;
2016-10-12 17:16:26 +02:00
if ( ! s )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Wrong spell id (%d)! " , ba . additionalInfo ) ;
2014-03-17 19:51:07 +00:00
return false ;
}
2015-09-17 08:42:30 +03:00
BattleSpellCastParameters parameters ( gs - > curB , h , s ) ;
2015-09-29 21:47:04 +03:00
parameters . aimToHex ( ba . destinationTile ) ; //todo: allow multiple destinations
2014-11-25 22:00:04 +03:00
parameters . mode = ECastingMode : : HERO_CASTING ;
2016-10-12 17:16:26 +02:00
if ( ba . selectedStack > = 0 )
2016-09-06 12:26:01 +03:00
parameters . aimToStack ( gs - > curB - > battleGetStackByID ( ba . selectedStack , false ) ) ;
2014-03-17 19:51:07 +00:00
2017-06-05 23:46:55 +03:00
ESpellCastProblem : : ESpellCastProblem escp = s - > canBeCast ( gs - > curB , ECastingMode : : HERO_CASTING , h ) ; //todo: should we check aimed cast?
2016-10-12 17:16:26 +02:00
if ( escp ! = ESpellCastProblem : : OK )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > warn ( " Spell cannot be cast! Problem: %d " , escp ) ;
2014-11-25 22:00:04 +03:00
return false ;
}
2014-03-17 19:51:07 +00:00
2014-11-25 22:00:04 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ; //start spell casting
2015-10-28 23:53:44 +03:00
2016-09-06 13:33:11 +03:00
parameters . cast ( spellEnv ) ;
2015-10-28 23:53:44 +03:00
2014-11-25 22:00:04 +03:00
sendAndApply ( & end_action ) ;
2016-10-12 17:16:26 +02:00
if ( ! gs - > curB - > battleGetStackByID ( gs - > curB - > activeStack ) )
2014-11-25 22:00:04 +03:00
{
battleMadeAction . setn ( true ) ;
2014-03-17 19:51:07 +00:00
}
2016-02-09 10:59:33 +03:00
checkBattleStateChanges ( ) ;
2016-10-12 17:16:26 +02:00
if ( battleResult . get ( ) )
2014-03-17 19:51:07 +00:00
{
2014-11-25 22:00:04 +03:00
battleMadeAction . setn ( true ) ;
//battle will be ended by startBattle function
//endBattle(gs->curB->tile, gs->curB->heroes[0], gs->curB->heroes[1]);
2014-03-17 19:51:07 +00:00
}
2014-11-25 22:00:04 +03:00
return true ;
2014-03-17 19:51:07 +00:00
}
}
return false ;
}
2016-10-28 23:37:45 +02:00
2016-11-02 20:11:01 +03:00
void CGameHandler : : stackEnchantedTrigger ( const CStack * st )
2016-10-28 23:37:45 +02:00
{
2016-10-29 13:45:08 +02:00
auto bl = * ( st - > getBonuses ( Selector : : type ( Bonus : : ENCHANTED ) ) ) ;
2016-11-02 20:11:01 +03:00
for ( auto b : bl )
2016-10-29 13:45:08 +02:00
{
SetStackEffect sse ;
int val = bl . valOfBonuses ( Selector : : typeSubtype ( b - > type , b - > subtype ) ) ;
2016-11-02 20:11:01 +03:00
if ( val > 3 )
2016-10-29 13:45:08 +02:00
{
2016-11-02 20:11:01 +03:00
for ( auto s : gs - > curB - > battleGetAllStacks ( ) )
2016-10-29 13:45:08 +02:00
{
2016-11-02 20:11:01 +03:00
if ( battleMatchOwner ( st , s , true ) & & s - > isValidTarget ( ) ) //all allied
2016-10-29 13:45:08 +02:00
sse . stacks . push_back ( s - > ID ) ;
}
}
else
sse . stacks . push_back ( st - > ID ) ;
2016-11-02 20:11:01 +03:00
const CSpell * sp = SpellID ( b - > subtype ) . toSpell ( ) ;
const int level = ( ( val > 3 ) ? ( val - 3 ) : val ) ;
sp - > getEffects ( sse . effect , level , false , 50 ) ;
//this makes effect accumulate for at most 50 turns by default, but effect may be permanent and last till the end of battle
sp - > getEffects ( sse . cumulativeEffects , level , true , 50 ) ;
if ( ! sse . effect . empty ( ) | | ! sse . cumulativeEffects . empty ( ) )
2016-10-29 13:45:08 +02:00
sendAndApply ( & sse ) ;
}
2016-10-28 23:37:45 +02:00
}
void CGameHandler : : stackTurnTrigger ( const CStack * st )
2014-03-17 19:51:07 +00:00
{
BattleTriggerEffect bte ;
bte . stackID = st - > ID ;
bte . effect = - 1 ;
bte . val = 0 ;
bte . additionalInfo = 0 ;
if ( st - > alive ( ) )
{
//unbind
2016-10-12 17:16:26 +02:00
if ( st - > hasBonus ( Selector : : type ( Bonus : : BIND_EFFECT ) ) )
2014-03-17 19:51:07 +00:00
{
bool unbind = true ;
BonusList bl = * ( st - > getBonuses ( Selector : : type ( Bonus : : BIND_EFFECT ) ) ) ;
std : : set < const CStack * > stacks = gs - > curB - > batteAdjacentCreatures ( st ) ;
2016-10-12 17:16:26 +02:00
for ( auto b : bl )
2014-03-17 19:51:07 +00:00
{
const CStack * stack = gs - > curB - > battleGetStackByID ( b - > additionalInfo ) ; //binding stack must be alive and adjacent
if ( stack )
{
if ( vstd : : contains ( stacks , stack ) ) //binding stack is still present
{
unbind = false ;
}
}
}
if ( unbind )
{
BattleSetStackProperty ssp ;
ssp . which = BattleSetStackProperty : : UNBIND ;
ssp . stackID = st - > ID ;
sendAndApply ( & ssp ) ;
}
}
//regeneration
2016-10-12 17:16:26 +02:00
if ( st - > hasBonusOfType ( Bonus : : HP_REGENERATION ) )
2014-03-17 19:51:07 +00:00
{
bte . effect = Bonus : : HP_REGENERATION ;
2017-07-04 14:24:46 +03:00
bte . val = std : : min ( ( int ) ( st - > MaxHealth ( ) - st - > getFirstHPleft ( ) ) , st - > valOfBonuses ( Bonus : : HP_REGENERATION ) ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
if ( st - > hasBonusOfType ( Bonus : : FULL_HP_REGENERATION ) )
2014-03-17 19:51:07 +00:00
{
bte . effect = Bonus : : HP_REGENERATION ;
2017-07-04 14:24:46 +03:00
bte . val = st - > MaxHealth ( ) - st - > getFirstHPleft ( ) ;
2014-03-17 19:51:07 +00:00
}
if ( bte . val ) //anything to heal
sendAndApply ( & bte ) ;
2016-10-12 17:16:26 +02:00
if ( st - > hasBonusOfType ( Bonus : : POISON ) )
2014-03-17 19:51:07 +00:00
{
2016-09-19 23:36:35 +02:00
const std : : shared_ptr < Bonus > b = st - > getBonusLocalFirst ( Selector : : source ( Bonus : : SPELL_EFFECT , SpellID : : POISON ) . And ( Selector : : type ( Bonus : : STACK_HEALTH ) ) ) ;
2014-03-17 19:51:07 +00:00
if ( b ) //TODO: what if not?...
{
bte . val = std : : max ( b - > val - 10 , - ( st - > valOfBonuses ( Bonus : : POISON ) ) ) ;
if ( bte . val < b - > val ) //(negative) poison effect increases - update it
{
bte . effect = Bonus : : POISON ;
sendAndApply ( & bte ) ;
}
}
}
if ( st - > hasBonusOfType ( Bonus : : MANA_DRAIN ) & & ! vstd : : contains ( st - > state , EBattleStackState : : DRAINED_MANA ) )
{
2016-09-29 21:47:41 +03:00
const PlayerColor opponent = gs - > curB - > theOtherPlayer ( gs - > curB - > battleGetOwner ( st ) ) ;
2015-09-16 18:28:14 +03:00
const CGHeroInstance * opponentHero = gs - > curB - > getHero ( opponent ) ;
if ( opponentHero )
2014-03-17 19:51:07 +00:00
{
ui32 manaDrained = st - > valOfBonuses ( Bonus : : MANA_DRAIN ) ;
2015-09-16 18:28:14 +03:00
vstd : : amin ( manaDrained , opponentHero - > mana ) ;
2014-03-17 19:51:07 +00:00
if ( manaDrained )
{
bte . effect = Bonus : : MANA_DRAIN ;
bte . val = manaDrained ;
2015-09-16 18:28:14 +03:00
bte . additionalInfo = opponentHero - > id . getNum ( ) ; //for sanity
2014-03-17 19:51:07 +00:00
sendAndApply ( & bte ) ;
}
}
}
if ( st - > isLiving ( ) & & ! st - > hasBonusOfType ( Bonus : : FEARLESS ) )
{
bool fearsomeCreature = false ;
2016-10-12 17:16:26 +02:00
for ( CStack * stack : gs - > curB - > stacks )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( battleMatchOwner ( st , stack ) & & stack - > alive ( ) & & stack - > hasBonusOfType ( Bonus : : FEAR ) )
2014-03-17 19:51:07 +00:00
{
fearsomeCreature = true ;
break ;
}
}
if ( fearsomeCreature )
{
2016-09-08 00:54:26 +03:00
if ( getRandomGenerator ( ) . nextInt ( 99 ) < 10 ) //fixed 10%
2014-03-17 19:51:07 +00:00
{
bte . effect = Bonus : : FEAR ;
sendAndApply ( & bte ) ;
}
}
}
BonusList bl = * ( st - > getBonuses ( Selector : : type ( Bonus : : ENCHANTER ) ) ) ;
int side = gs - > curB - > whatSide ( st - > owner ) ;
2017-07-04 14:24:46 +03:00
if ( st - > canCast ( ) & & ! gs - > curB - > sides . at ( side ) . enchanterCounter )
2014-03-17 19:51:07 +00:00
{
2015-09-21 12:19:35 +03:00
bool cast = false ;
while ( ! bl . empty ( ) & & ! cast )
2014-03-17 19:51:07 +00:00
{
2016-09-08 00:54:26 +03:00
auto bonus = * RandomGeneratorUtil : : nextItem ( bl , getRandomGenerator ( ) ) ;
2015-06-21 21:27:58 +03:00
auto spellID = SpellID ( bonus - > subtype ) ;
const CSpell * spell = SpellID ( spellID ) . toSpell ( ) ;
2016-09-19 23:36:35 +02:00
bl . remove_if ( [ & bonus ] ( const Bonus * b ) { return b = = bonus . get ( ) ; } ) ;
2015-10-28 23:53:44 +03:00
2017-06-05 21:41:27 +03:00
BattleSpellCastParameters parameters ( gs - > curB , st , spell ) ;
parameters . spellLvl = bonus - > val ;
parameters . effectLevel = bonus - > val ; //todo: recheck
parameters . mode = ECastingMode : : ENCHANTER_CASTING ;
2015-06-21 21:27:58 +03:00
2017-06-05 21:41:27 +03:00
cast = parameters . castIfPossible ( spellEnv ) ;
if ( cast )
{
2015-09-17 05:47:32 +03:00
//todo: move to mechanics
2015-06-21 21:27:58 +03:00
BattleSetStackProperty ssp ;
ssp . which = BattleSetStackProperty : : ENCHANTER_COUNTER ;
ssp . absolute = false ;
ssp . val = bonus - > additionalInfo ; //increase cooldown counter
ssp . stackID = st - > ID ;
sendAndApply ( & ssp ) ;
2015-10-28 23:53:44 +03:00
}
2014-03-17 19:51:07 +00:00
}
}
}
}
2017-07-09 10:18:46 +02:00
bool CGameHandler : : handleDamageFromObstacle ( const CStack * curStack , bool stackIsMoving )
2014-03-17 19:51:07 +00:00
{
2017-07-09 10:18:46 +02:00
if ( ! curStack - > alive ( ) )
return false ;
bool containDamageFromMoat = false ;
for ( auto & obstacle : getAllAffectedObstaclesByStack ( curStack ) )
{
if ( ! curStack - > alive ( ) | | obstacle - > stopsMovement ( ) & & stackIsMoving = = true )
return false ;
//we want to determine following vars depending on obstacle type
int damage = - 1 ;
int effect = - 1 ;
bool oneTimeObstacle = false ;
2014-03-17 19:51:07 +00:00
2017-07-09 10:18:46 +02:00
//helper info
const SpellCreatedObstacle * spellObstacle = dynamic_cast < const SpellCreatedObstacle * > ( obstacle . get ( ) ) ; //not nice but we may need spell params
2016-11-27 22:50:37 +03:00
2017-07-09 10:18:46 +02:00
const ui8 side = curStack - > side ; //if enemy is defending (false = 0), side of enemy hero is 1 (true)
const CGHeroInstance * hero = gs - > curB - > battleGetFightingHero ( side ) ; //FIXME: there may be no hero - landmines in Tower
2014-03-17 19:51:07 +00:00
2017-07-09 10:18:46 +02:00
if ( obstacle - > obstacleType = = CObstacleInstance : : MOAT )
{
damage = battleGetMoatDmg ( ) ;
if ( ! containDamageFromMoat )
containDamageFromMoat = true ;
else
continue ;
}
else if ( obstacle - > obstacleType = = CObstacleInstance : : LAND_MINE )
{
if ( ! spellObstacle )
COMPLAIN_RET ( " Invalid obstacle instance " ) ;
//You don't get hit by a Mine you can see.
if ( gs - > curB - > battleIsObstacleVisibleForSide ( * obstacle , ( BattlePerspective : : BattlePerspective ) side ) )
continue ;
oneTimeObstacle = true ;
effect = 82 ;
const CSpell * sp = SpellID ( SpellID : : LAND_MINE ) . toSpell ( ) ;
2014-03-17 19:51:07 +00:00
2017-07-09 10:18:46 +02:00
if ( sp - > isImmuneByStack ( hero , curStack ) )
continue ;
2015-03-29 16:50:23 +03:00
2017-07-09 10:18:46 +02:00
damage = sp - > calculateDamage ( hero , curStack , spellObstacle - > spellLevel , spellObstacle - > casterSpellPower ) ;
//TODO even if obstacle wasn't created by hero (Tower "moat") it should deal dmg as if cast by hero,
//if it is bigger than default dmg. Or is it just irrelevant H3 implementation quirk
}
else if ( obstacle - > obstacleType = = CObstacleInstance : : FIRE_WALL )
{
if ( ! spellObstacle )
COMPLAIN_RET ( " Invalid obstacle instance " ) ;
const CSpell * sp = SpellID ( SpellID : : FIRE_WALL ) . toSpell ( ) ;
2015-03-29 16:50:23 +03:00
2017-07-09 10:18:46 +02:00
if ( sp - > isImmuneByStack ( hero , curStack ) )
continue ;
2015-03-29 16:50:23 +03:00
2017-07-09 10:18:46 +02:00
damage = sp - > calculateDamage ( hero , curStack ,
spellObstacle - > spellLevel , spellObstacle - > casterSpellPower ) ;
}
else
continue ;
2015-03-29 16:50:23 +03:00
2017-07-09 10:18:46 +02:00
BattleStackAttacked bsa ;
if ( effect > = 0 )
{
bsa . flags | = BattleStackAttacked : : EFFECT ;
bsa . effect = effect ; //makes POOF
}
bsa . damageAmount = damage ;
bsa . stackAttacked = curStack - > ID ;
bsa . attackerID = - 1 ;
curStack - > prepareAttacked ( bsa , getRandomGenerator ( ) ) ;
2015-03-29 16:50:23 +03:00
2017-07-09 10:18:46 +02:00
StacksInjured si ;
si . stacks . push_back ( bsa ) ;
sendAndApply ( & si ) ;
2014-03-17 19:51:07 +00:00
2017-07-09 10:18:46 +02:00
if ( oneTimeObstacle )
removeObstacle ( * obstacle ) ;
2014-03-17 19:51:07 +00:00
}
2017-07-09 10:18:46 +02:00
if ( ! curStack - > alive ( ) )
return false ;
return true ;
2014-03-17 19:51:07 +00:00
}
void CGameHandler : : handleTimeEvents ( )
{
gs - > map - > events . sort ( evntCmp ) ;
while ( gs - > map - > events . size ( ) & & gs - > map - > events . front ( ) . firstOccurence + 1 = = gs - > day )
{
CMapEvent ev = gs - > map - > events . front ( ) ;
2015-10-28 23:53:44 +03:00
2014-09-17 11:03:53 +02:00
for ( int player = 0 ; player < PlayerColor : : PLAYER_LIMIT_I ; player + + )
2014-03-17 19:51:07 +00:00
{
2014-09-17 11:03:53 +02:00
auto color = PlayerColor ( player ) ;
2016-09-18 11:53:51 +03:00
const PlayerState * pinfo = getPlayer ( color , false ) ; //do not output error if player does not exist
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( pinfo //player exists
2014-03-17 19:51:07 +00:00
& & ( ev . players & 1 < < player ) //event is enabled to this player
& & ( ( ev . computerAffected & & ! pinfo - > human )
| | ( ev . humanAffected & & pinfo - > human )
)
)
{
//give resources
2016-11-26 15:14:43 +03:00
giveResources ( color , ev . resources ) ;
2014-03-17 19:51:07 +00:00
//prepare dialog
InfoWindow iw ;
2014-09-17 11:03:53 +02:00
iw . player = color ;
2014-03-17 19:51:07 +00:00
iw . text < < ev . message ;
for ( int i = 0 ; i < ev . resources . size ( ) ; i + + )
{
2016-10-12 17:16:26 +02:00
if ( ev . resources . at ( i ) ) //if resource is changed, we add it to the dialog
2014-03-17 19:51:07 +00:00
iw . components . push_back ( Component ( Component : : RESOURCE , i , ev . resources . at ( i ) , 0 ) ) ;
}
sendAndApply ( & iw ) ; //show dialog
}
} //PLAYERS LOOP
2016-10-12 17:16:26 +02:00
if ( ev . nextOccurence )
2014-03-17 19:51:07 +00:00
{
gs - > map - > events . pop_front ( ) ;
ev . firstOccurence + = ev . nextOccurence ;
auto it = gs - > map - > events . begin ( ) ;
2016-03-12 04:41:27 +03:00
while ( it ! = gs - > map - > events . end ( ) & & it - > earlierThanOrEqual ( ev ) )
2014-03-17 19:51:07 +00:00
it + + ;
gs - > map - > events . insert ( it , ev ) ;
}
else
{
gs - > map - > events . pop_front ( ) ;
}
}
//TODO send only if changed
UpdateMapEvents ume ;
ume . events = gs - > map - > events ;
sendAndApply ( & ume ) ;
}
void CGameHandler : : handleTownEvents ( CGTownInstance * town , NewTurn & n )
{
town - > events . sort ( evntCmp ) ;
while ( town - > events . size ( ) & & town - > events . front ( ) . firstOccurence = = gs - > day )
{
PlayerColor player = town - > tempOwner ;
CCastleEvent ev = town - > events . front ( ) ;
2016-09-18 11:53:51 +03:00
const PlayerState * pinfo = getPlayer ( player , false ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( pinfo //player exists
2014-03-17 19:51:07 +00:00
& & ( ev . players & 1 < < player . getNum ( ) ) //event is enabled to this player
& & ( ( ev . computerAffected & & ! pinfo - > human )
2016-10-12 17:16:26 +02:00
| | ( ev . humanAffected & & pinfo - > human ) ) )
2014-03-17 19:51:07 +00:00
{
// dialog
InfoWindow iw ;
iw . player = player ;
iw . text < < ev . message ;
2016-10-12 17:16:26 +02:00
if ( ev . resources . nonZero ( ) )
2014-03-17 19:51:07 +00:00
{
TResources was = n . res [ player ] ;
n . res [ player ] + = ev . resources ;
n . res [ player ] . amax ( 0 ) ;
for ( int i = 0 ; i < ev . resources . size ( ) ; i + + )
2016-10-12 17:16:26 +02:00
if ( ev . resources . at ( i ) & & pinfo - > resources . at ( i ) ! = n . res . at ( player ) . at ( i ) ) //if resource had changed, we add it to the dialog
2014-03-17 19:51:07 +00:00
iw . components . push_back ( Component ( Component : : RESOURCE , i , n . res . at ( player ) . at ( i ) - was . at ( i ) , 0 ) ) ;
}
2016-10-12 17:16:26 +02:00
for ( auto & i : ev . buildings )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! town - > hasBuilt ( i ) )
2014-03-17 19:51:07 +00:00
{
buildStructure ( town - > id , i , true ) ;
iw . components . push_back ( Component ( Component : : BUILDING , town - > subID , i , 0 ) ) ;
}
}
if ( ! ev . creatures . empty ( ) & & ! vstd : : contains ( n . cres , town - > id ) )
{
n . cres [ town - > id ] . tid = town - > id ;
n . cres [ town - > id ] . creatures = town - > creatures ;
}
auto & sac = n . cres [ town - > id ] ;
2016-10-12 17:16:26 +02:00
for ( si32 i = 0 ; i < ev . creatures . size ( ) ; i + + ) //creature growths
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! town - > creatures . at ( i ) . second . empty ( ) & & ev . creatures . at ( i ) > 0 ) //there is dwelling
2014-03-17 19:51:07 +00:00
{
sac . creatures [ i ] . first + = ev . creatures . at ( i ) ;
iw . components . push_back ( Component ( Component : : CREATURE ,
town - > creatures . at ( i ) . second . back ( ) , ev . creatures . at ( i ) , 0 ) ) ;
}
}
sendAndApply ( & iw ) ; //show dialog
}
2016-10-12 17:16:26 +02:00
if ( ev . nextOccurence )
2014-03-17 19:51:07 +00:00
{
town - > events . pop_front ( ) ;
ev . firstOccurence + = ev . nextOccurence ;
auto it = town - > events . begin ( ) ;
2016-03-12 04:41:27 +03:00
while ( it ! = town - > events . end ( ) & & it - > earlierThanOrEqual ( ev ) )
2014-03-17 19:51:07 +00:00
it + + ;
town - > events . insert ( it , ev ) ;
}
else
{
town - > events . pop_front ( ) ;
}
}
//TODO send only if changed
UpdateCastleEvents uce ;
uce . town = town - > id ;
uce . events = town - > events ;
sendAndApply ( & uce ) ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : complain ( const std : : string & problem )
2014-03-17 19:51:07 +00:00
{
sendMessageToAll ( " Server encountered a problem: " + problem ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > error ( problem ) ;
2014-03-17 19:51:07 +00:00
return true ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : showGarrisonDialog ( ObjectInstanceID upobj , ObjectInstanceID hid , bool removableUnits )
2014-03-17 19:51:07 +00:00
{
//PlayerColor player = getOwner(hid);
auto upperArmy = dynamic_cast < const CArmedInstance * > ( getObj ( upobj ) ) ;
auto lowerArmy = dynamic_cast < const CArmedInstance * > ( getObj ( hid ) ) ;
assert ( lowerArmy ) ;
assert ( upperArmy ) ;
2017-06-06 07:53:51 +03:00
auto garrisonQuery = std : : make_shared < CGarrisonDialogQuery > ( this , upperArmy , lowerArmy ) ;
2014-03-17 19:51:07 +00:00
queries . addQuery ( garrisonQuery ) ;
GarrisonDialog gd ;
gd . hid = hid ;
gd . objid = upobj ;
gd . removableUnits = removableUnits ;
gd . queryID = garrisonQuery - > queryID ;
sendAndApply ( & gd ) ;
}
void CGameHandler : : showThievesGuildWindow ( PlayerColor player , ObjectInstanceID requestingObjId )
{
OpenWindow ow ;
ow . window = OpenWindow : : THIEVES_GUILD ;
ow . id1 = player . getNum ( ) ;
ow . id2 = requestingObjId . getNum ( ) ;
sendAndApply ( & ow ) ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : isAllowedExchange ( ObjectInstanceID id1 , ObjectInstanceID id2 )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( id1 = = id2 )
2014-03-17 19:51:07 +00:00
return true ;
const CGObjectInstance * o1 = getObj ( id1 ) , * o2 = getObj ( id2 ) ;
2016-10-12 17:16:26 +02:00
if ( ! o1 | | ! o2 )
2014-03-17 19:51:07 +00:00
return true ; //arranging stacks within an object should be always allowed
if ( o1 & & o2 )
{
2016-10-12 17:16:26 +02:00
if ( o1 - > ID = = Obj : : TOWN )
2014-03-17 19:51:07 +00:00
{
const CGTownInstance * t = static_cast < const CGTownInstance * > ( o1 ) ;
2016-10-12 17:16:26 +02:00
if ( t - > visitingHero = = o2 | | t - > garrisonHero = = o2 )
2014-03-17 19:51:07 +00:00
return true ;
}
2016-10-12 17:16:26 +02:00
if ( o2 - > ID = = Obj : : TOWN )
2014-03-17 19:51:07 +00:00
{
const CGTownInstance * t = static_cast < const CGTownInstance * > ( o2 ) ;
2016-10-12 17:16:26 +02:00
if ( t - > visitingHero = = o1 | | t - > garrisonHero = = o1 )
2014-03-17 19:51:07 +00:00
return true ;
}
if ( o1 - > ID = = Obj : : HERO & & o2 - > ID = = Obj : : HERO )
{
const CGHeroInstance * h1 = static_cast < const CGHeroInstance * > ( o1 ) ;
const CGHeroInstance * h2 = static_cast < const CGHeroInstance * > ( o2 ) ;
// two heroes in same town (garrisoned and visiting)
if ( h1 - > visitedTown ! = nullptr & & h2 - > visitedTown ! = nullptr & & h1 - > visitedTown = = h2 - > visitedTown )
return true ;
}
//Ongoing garrison exchange
2016-10-12 17:16:26 +02:00
if ( auto dialog = std : : dynamic_pointer_cast < CGarrisonDialogQuery > ( queries . topQuery ( o1 - > tempOwner ) ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( dialog - > exchangingArmies . at ( 0 ) = = o1 & & dialog - > exchangingArmies . at ( 1 ) = = o2 )
2014-03-17 19:51:07 +00:00
return true ;
2016-10-12 17:16:26 +02:00
if ( dialog - > exchangingArmies . at ( 1 ) = = o1 & & dialog - > exchangingArmies . at ( 0 ) = = o2 )
2014-03-17 19:51:07 +00:00
return true ;
}
}
return false ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : objectVisited ( const CGObjectInstance * obj , const CGHeroInstance * h )
2014-03-17 19:51:07 +00:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > debug ( " %s visits %s (%d:%d) " , h - > nodeName ( ) , obj - > getObjectName ( ) , obj - > ID , obj - > subID ) ;
2017-06-06 07:53:51 +03:00
auto visitQuery = std : : make_shared < CObjectVisitQuery > ( this , obj , h , obj - > visitablePos ( ) ) ;
2014-03-17 19:51:07 +00:00
queries . addQuery ( visitQuery ) ; //TODO real visit pos
HeroVisit hv ;
hv . obj = obj ;
hv . hero = h ;
hv . player = h - > tempOwner ;
hv . starting = true ;
sendAndApply ( & hv ) ;
obj - > onHeroVisit ( h ) ;
queries . popIfTop ( visitQuery ) ; //visit ends here if no queries were created
}
void CGameHandler : : objectVisitEnded ( const CObjectVisitQuery & query )
{
2016-08-30 01:11:54 +03:00
logGlobal - > debug ( " %s visit ends. \n " , query . visitingHero - > nodeName ( ) ) ;
2014-03-17 19:51:07 +00:00
HeroVisit hv ;
hv . player = query . players . front ( ) ;
hv . obj = nullptr ; //not necessary, moreover may have been deleted in the meantime
hv . hero = query . visitingHero ;
assert ( hv . hero ) ;
hv . starting = false ;
sendAndApply ( & hv ) ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : buildBoat ( ObjectInstanceID objid )
2014-03-17 19:51:07 +00:00
{
const IShipyard * obj = IShipyard : : castFrom ( getObj ( objid ) ) ;
2016-10-12 17:16:26 +02:00
if ( obj - > shipyardStatus ( ) ! = IBoatGenerator : : GOOD )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot build boat in this shipyard! " ) ;
return false ;
}
2016-10-12 17:16:26 +02:00
else if ( obj - > o - > ID = = Obj : : TOWN
2014-03-17 19:51:07 +00:00
& & ! static_cast < const CGTownInstance * > ( obj ) - > hasBuilt ( BuildingID : : SHIPYARD ) )
{
complain ( " Cannot build boat in the town - no shipyard! " ) ;
return false ;
}
const PlayerColor playerID = obj - > o - > tempOwner ;
TResources boatCost ;
obj - > getBoatCost ( boatCost ) ;
2016-09-18 11:53:51 +03:00
TResources aviable = getPlayer ( playerID ) - > resources ;
2014-03-17 19:51:07 +00:00
if ( ! aviable . canAfford ( boatCost ) )
{
complain ( " Not enough resources to build a boat! " ) ;
return false ;
}
int3 tile = obj - > bestLocation ( ) ;
2016-10-12 17:16:26 +02:00
if ( ! gs - > map - > isInTheMap ( tile ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot find appropriate tile for a boat! " ) ;
return false ;
}
//take boat cost
2016-11-26 15:14:43 +03:00
giveResources ( playerID , - boatCost ) ;
2014-03-17 19:51:07 +00:00
//create boat
NewObject no ;
no . ID = Obj : : BOAT ;
no . subID = obj - > getBoatType ( ) ;
no . pos = tile + int3 ( 1 , 0 , 0 ) ;
sendAndApply ( & no ) ;
return true ;
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : engageIntoBattle ( PlayerColor player )
2014-03-17 19:51:07 +00:00
{
//notify interfaces
PlayerBlocked pb ;
pb . player = player ;
pb . reason = PlayerBlocked : : UPCOMING_BATTLE ;
pb . startOrEnd = PlayerBlocked : : BLOCKADE_STARTED ;
sendAndApply ( & pb ) ;
}
void CGameHandler : : checkVictoryLossConditions ( const std : : set < PlayerColor > & playerColors )
{
2016-10-12 17:16:26 +02:00
for ( auto playerColor : playerColors )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( getPlayer ( playerColor , false ) )
2014-03-17 19:51:07 +00:00
checkVictoryLossConditionsForPlayer ( playerColor ) ;
}
}
void CGameHandler : : checkVictoryLossConditionsForAll ( )
{
std : : set < PlayerColor > playerColors ;
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < PlayerColor : : PLAYER_LIMIT_I ; + + i )
2014-03-17 19:51:07 +00:00
{
playerColors . insert ( PlayerColor ( i ) ) ;
}
checkVictoryLossConditions ( playerColors ) ;
}
void CGameHandler : : checkVictoryLossConditionsForPlayer ( PlayerColor player )
{
2016-09-18 11:53:51 +03:00
const PlayerState * p = getPlayer ( player ) ;
2016-10-12 17:16:26 +02:00
if ( p - > status ! = EPlayerStatus : : INGAME ) return ;
2014-03-17 19:51:07 +00:00
auto victoryLossCheckResult = gs - > checkForVictoryAndLoss ( player ) ;
2016-10-12 17:16:26 +02:00
if ( victoryLossCheckResult . victory ( ) | | victoryLossCheckResult . loss ( ) )
2014-03-17 19:51:07 +00:00
{
InfoWindow iw ;
getVictoryLossMessage ( player , victoryLossCheckResult , iw ) ;
sendAndApply ( & iw ) ;
PlayerEndsGame peg ;
peg . player = player ;
peg . victoryLossCheckResult = victoryLossCheckResult ;
sendAndApply ( & peg ) ;
2016-10-12 17:16:26 +02:00
if ( victoryLossCheckResult . victory ( ) )
2014-03-17 19:51:07 +00:00
{
//one player won -> all enemies lost
for ( auto i = gs - > players . cbegin ( ) ; i ! = gs - > players . cend ( ) ; i + + )
{
2016-10-12 17:16:26 +02:00
if ( i - > first ! = player & & getPlayer ( i - > first ) - > status = = EPlayerStatus : : INGAME )
2014-03-17 19:51:07 +00:00
{
peg . player = i - > first ;
2016-09-18 11:53:51 +03:00
peg . victoryLossCheckResult = getPlayerRelations ( player , i - > first ) = = PlayerRelations : : ALLIES ?
2014-03-17 19:51:07 +00:00
victoryLossCheckResult : victoryLossCheckResult . invert ( ) ; // ally of winner
InfoWindow iw ;
getVictoryLossMessage ( player , peg . victoryLossCheckResult , iw ) ;
iw . player = i - > first ;
sendAndApply ( & iw ) ;
sendAndApply ( & peg ) ;
}
}
2016-10-12 17:16:26 +02:00
if ( p - > human )
2014-03-17 19:51:07 +00:00
{
2017-06-04 11:59:26 +03:00
serverShuttingDown = true ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( gs - > scenarioOps - > campState )
2014-03-17 19:51:07 +00:00
{
std : : vector < CGHeroInstance * > crossoverHeroes ;
2016-10-12 17:16:26 +02:00
for ( CGHeroInstance * hero : gs - > map - > heroesOnMap )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( hero - > tempOwner = = player )
2014-03-17 19:51:07 +00:00
{
// keep all heroes from the winning player
crossoverHeroes . push_back ( hero ) ;
}
2016-10-12 17:16:26 +02:00
else if ( vstd : : contains ( gs - > scenarioOps - > campState - > getCurrentScenario ( ) . keepHeroes , HeroTypeID ( hero - > subID ) ) )
2014-03-17 19:51:07 +00:00
{
// keep hero whether lost or won (like Xeron in AB campaign)
crossoverHeroes . push_back ( hero ) ;
}
}
// keep lost heroes which are in heroes pool
2016-10-12 17:16:26 +02:00
for ( auto & heroPair : gs - > hpool . heroesPool )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( vstd : : contains ( gs - > scenarioOps - > campState - > getCurrentScenario ( ) . keepHeroes , HeroTypeID ( heroPair . first ) ) )
2014-03-17 19:51:07 +00:00
{
crossoverHeroes . push_back ( heroPair . second . get ( ) ) ;
}
}
gs - > scenarioOps - > campState - > setCurrentMapAsConquered ( crossoverHeroes ) ;
//Request clients to change connection mode
PrepareForAdvancingCampaign pfac ;
sendAndApply ( & pfac ) ;
//Change connection mode
2016-10-12 17:16:26 +02:00
if ( getPlayer ( player ) - > human & & getStartInfo ( ) - > campState )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
for ( auto connection : conns )
2014-03-17 19:51:07 +00:00
connection - > prepareForSendingHeroes ( ) ;
}
UpdateCampaignState ucs ;
ucs . camp = gs - > scenarioOps - > campState ;
sendAndApply ( & ucs ) ;
}
}
}
else
{
2015-12-10 12:43:55 +03:00
//copy heroes vector to avoid iterator invalidation as removal change PlayerState
auto hlp = p - > heroes ;
2016-10-12 17:16:26 +02:00
for ( auto h : hlp ) //eliminate heroes
2015-12-10 12:43:55 +03:00
{
2016-10-12 17:16:26 +02:00
if ( h . get ( ) )
2015-10-24 15:09:46 +02:00
removeObject ( h ) ;
2015-12-10 12:43:55 +03:00
}
2014-03-17 19:51:07 +00:00
2015-12-10 12:43:55 +03:00
//player lost -> all his objects become unflagged (neutral)
2015-10-24 15:09:46 +02:00
for ( auto obj : gs - > map - > objects ) //unflag objs
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( obj . get ( ) & & obj - > tempOwner = = player )
2015-10-24 15:09:46 +02:00
setOwner ( obj , PlayerColor : : NEUTRAL ) ;
2014-03-17 19:51:07 +00:00
}
//eliminating one player may cause victory of another:
std : : set < PlayerColor > playerColors ;
2015-10-24 17:15:21 +02:00
//do not copy player state (CBonusSystemNode) by value
2015-10-25 07:07:01 +01:00
for ( auto & p : gs - > players ) //players may have different colors, iterate over players and not integers
2014-03-17 19:51:07 +00:00
{
2015-10-25 07:07:01 +01:00
if ( p . first ! = player )
playerColors . insert ( p . first ) ;
2014-03-17 19:51:07 +00:00
}
//notify all players
2015-10-24 15:09:46 +02:00
for ( auto pc : playerColors )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( getPlayer ( pc ) - > status = = EPlayerStatus : : INGAME )
2014-03-17 19:51:07 +00:00
{
InfoWindow iw ;
getVictoryLossMessage ( player , victoryLossCheckResult . invert ( ) , iw ) ;
2015-10-24 15:09:46 +02:00
iw . player = pc ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & iw ) ;
}
}
checkVictoryLossConditions ( playerColors ) ;
}
2016-09-18 11:53:51 +03:00
auto playerInfo = getPlayer ( gs - > currentPlayer , false ) ;
2014-03-17 19:51:07 +00:00
// If we are called before the actual game start, there might be no current player
2015-10-24 15:09:46 +02:00
if ( playerInfo & & playerInfo - > status ! = EPlayerStatus : : INGAME )
2014-03-17 19:51:07 +00:00
{
// If player making turn has lost his turn must be over as well
states . setFlag ( gs - > currentPlayer , & PlayerStatus : : makingTurn , false ) ;
}
}
}
2014-06-25 17:11:07 +03:00
void CGameHandler : : getVictoryLossMessage ( PlayerColor player , const EVictoryLossCheckResult & victoryLossCheckResult , InfoWindow & out ) const
2014-03-17 19:51:07 +00:00
{
out . player = player ;
out . text . clear ( ) ;
out . text < < victoryLossCheckResult . messageToSelf ;
// hackish, insert one player-specific string, if applicable
if ( victoryLossCheckResult . messageToSelf . find ( " %s " ) ! = std : : string : : npos )
out . text . addReplacement ( MetaString : : COLOR , player . getNum ( ) ) ;
out . components . push_back ( Component ( Component : : FLAG , player . getNum ( ) , 0 , 0 ) ) ;
}
2016-10-12 17:16:26 +02:00
bool CGameHandler : : dig ( const CGHeroInstance * h )
2014-03-17 19:51:07 +00:00
{
for ( auto i = gs - > map - > objects . cbegin ( ) ; i ! = gs - > map - > objects . cend ( ) ; i + + ) //unflag objs
{
2016-10-12 17:16:26 +02:00
if ( * i & & ( * i ) - > ID = = Obj : : HOLE & & ( * i ) - > pos = = h - > getPosition ( ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot dig - there is already a hole under the hero! " ) ;
return false ;
}
}
2016-10-12 17:16:26 +02:00
if ( h - > diggingStatus ( ) ! = EDiggingStatus : : CAN_DIG ) //checks for terrain and movement
2014-03-17 19:51:07 +00:00
COMPLAIN_RETF ( " Hero cannot dig (error code %d)! " , h - > diggingStatus ( ) ) ;
//create a hole
NewObject no ;
no . ID = Obj : : HOLE ;
no . pos = h - > getPosition ( ) ;
no . subID = 0 ;
sendAndApply ( & no ) ;
//take MPs
SetMovePoints smp ;
smp . hid = h - > id ;
smp . val = 0 ;
sendAndApply ( & smp ) ;
InfoWindow iw ;
iw . player = h - > tempOwner ;
2016-10-12 17:16:26 +02:00
if ( gs - > map - > grailPos = = h - > getPosition ( ) )
2014-03-17 19:51:07 +00:00
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 58 ) ; //"Congratulations! After spending many hours digging here, your hero has uncovered the "
2015-11-06 21:54:51 +03:00
iw . text . addTxt ( MetaString : : ART_NAMES , ArtifactID : : GRAIL ) ;
2014-03-17 19:51:07 +00:00
iw . soundID = soundBase : : ULTIMATEARTIFACT ;
2015-11-06 21:54:51 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ ArtifactID : : GRAIL ] , ArtifactPosition : : PRE_FIRST ) ; //give grail
2014-03-17 19:51:07 +00:00
sendAndApply ( & iw ) ;
iw . soundID = soundBase : : invalid ;
iw . text . clear ( ) ;
2015-11-06 21:54:51 +03:00
iw . text . addTxt ( MetaString : : ART_DESCR , ArtifactID : : GRAIL ) ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & iw ) ;
}
else
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 59 ) ; //"Nothing here. \n Where could it be?"
iw . soundID = soundBase : : Dig ;
sendAndApply ( & iw ) ;
}
return true ;
}
void CGameHandler : : attackCasting ( const BattleAttack & bat , Bonus : : BonusType attackMode , const CStack * attacker )
{
2016-10-12 17:16:26 +02:00
if ( attacker - > hasBonusOfType ( attackMode ) )
2014-03-17 19:51:07 +00:00
{
std : : set < SpellID > spellsToCast ;
TBonusListPtr spells = attacker - > getBonuses ( Selector : : type ( attackMode ) ) ;
2016-10-12 17:16:26 +02:00
for ( const std : : shared_ptr < Bonus > sf : * spells )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
spellsToCast . insert ( SpellID ( sf - > subtype ) ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
for ( SpellID spellID : spellsToCast )
2014-03-17 19:51:07 +00:00
{
const CStack * oneOfAttacked = nullptr ;
for ( auto & elem : bat . bsa )
{
2017-07-04 14:24:46 +03:00
if ( ( elem . newHealth . fullUnits > 0 | | elem . newHealth . firstHPleft > 0 ) & & ! elem . isSecondary ( ) ) //apply effects only to first target stack if it's alive
2014-03-17 19:51:07 +00:00
{
oneOfAttacked = gs - > curB - > battleGetStackByID ( elem . stackAttacked ) ;
break ;
}
}
bool castMe = false ;
2016-10-12 17:16:26 +02:00
if ( oneOfAttacked = = nullptr )
{
logGlobal - > debug ( " attackCasting: all attacked creatures have been killed " ) ;
2014-03-17 19:51:07 +00:00
return ;
2016-10-12 17:16:26 +02:00
}
2014-03-17 19:51:07 +00:00
int spellLevel = 0 ;
TBonusListPtr spellsByType = attacker - > getBonuses ( Selector : : typeSubtype ( attackMode , spellID ) ) ;
2016-10-12 17:16:26 +02:00
for ( const std : : shared_ptr < Bonus > sf : * spellsByType )
2014-03-17 19:51:07 +00:00
{
vstd : : amax ( spellLevel , sf - > additionalInfo % 1000 ) ; //pick highest level
int meleeRanged = sf - > additionalInfo / 1000 ;
if ( meleeRanged = = 0 | | ( meleeRanged = = 1 & & bat . shot ( ) ) | | ( meleeRanged = = 2 & & ! bat . shot ( ) ) )
castMe = true ;
}
int chance = attacker - > valOfBonuses ( ( Selector : : typeSubtype ( attackMode , spellID ) ) ) ;
2016-10-12 17:16:26 +02:00
vstd : : amin ( chance , 100 ) ;
2014-03-17 19:51:07 +00:00
const CSpell * spell = SpellID ( spellID ) . toSpell ( ) ;
2017-07-16 00:01:33 +03:00
if ( spell - > canBeCastAt ( gs - > curB , ECastingMode : : AFTER_ATTACK_CASTING , attacker , oneOfAttacked - > position ) ! = ESpellCastProblem : : OK )
2014-03-17 19:51:07 +00:00
continue ;
2015-09-21 12:19:35 +03:00
//check if spell should be cast (probability handling)
2016-10-12 17:16:26 +02:00
if ( getRandomGenerator ( ) . nextInt ( 99 ) > = chance )
2014-03-17 19:51:07 +00:00
continue ;
2014-11-25 22:00:04 +03:00
//casting
2014-03-17 19:51:07 +00:00
if ( castMe ) //stacks use 0 spell power. If needed, default = 3 or custom value is used
2014-11-25 22:00:04 +03:00
{
2016-10-12 17:16:26 +02:00
logGlobal - > debug ( " battle spell cast " ) ;
2015-09-17 08:42:30 +03:00
BattleSpellCastParameters parameters ( gs - > curB , attacker , spell ) ;
2014-11-25 22:00:04 +03:00
parameters . spellLvl = spellLevel ;
2015-09-17 08:42:30 +03:00
parameters . effectLevel = spellLevel ;
2015-09-29 21:47:04 +03:00
parameters . aimToStack ( oneOfAttacked ) ;
2014-11-25 22:00:04 +03:00
parameters . mode = ECastingMode : : AFTER_ATTACK_CASTING ;
2016-09-06 13:33:11 +03:00
parameters . cast ( spellEnv ) ;
2014-11-25 22:00:04 +03:00
}
2014-03-17 19:51:07 +00:00
}
}
}
2016-10-12 17:19:02 +02:00
void CGameHandler : : handleAttackBeforeCasting ( BattleAttack * bat )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:19:02 +02:00
const CStack * attacker = gs - > curB - > battleGetStackByID ( bat - > stackAttacking ) ;
attackCasting ( * bat , Bonus : : SPELL_BEFORE_ATTACK , attacker ) ; //no death stare / acid breath needed?
// filter possibly dead stacks
bat - > bsa . erase ( std : : remove_if ( bat - > bsa . begin ( ) , bat - > bsa . end ( ) ,
[ this ] ( const BattleStackAttacked & bsa )
{
return battleGetStackByID ( bsa . stackAttacked ) = = nullptr ;
} ) ,
bat - > bsa . end ( ) ) ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : handleAfterAttackCasting ( const BattleAttack & bat )
2014-03-17 19:51:07 +00:00
{
const CStack * attacker = gs - > curB - > battleGetStackByID ( bat . stackAttacking ) ;
2016-10-12 17:19:02 +02:00
if ( ! attacker | | bat . bsa . empty ( ) ) // can be already dead
2014-03-17 19:51:07 +00:00
return ;
2015-10-28 23:53:44 +03:00
2017-07-04 14:24:46 +03:00
const CStack * defender = gs - > curB - > battleGetStackByID ( bat . bsa . at ( 0 ) . stackAttacked ) ;
if ( ! defender )
return ; //already dead
2017-01-22 20:28:08 +01:00
2014-11-25 22:00:04 +03:00
auto cast = [ = ] ( SpellID spellID , int power )
{
const CSpell * spell = SpellID ( spellID ) . toSpell ( ) ;
2015-09-17 08:42:30 +03:00
BattleSpellCastParameters parameters ( gs - > curB , attacker , spell ) ;
2014-11-25 22:00:04 +03:00
parameters . spellLvl = 0 ;
2015-09-17 08:42:30 +03:00
parameters . effectLevel = 0 ;
2017-07-04 14:24:46 +03:00
parameters . aimToStack ( defender ) ;
2015-10-28 23:53:44 +03:00
parameters . effectPower = power ;
2014-11-25 22:00:04 +03:00
parameters . mode = ECastingMode : : AFTER_ATTACK_CASTING ;
2016-09-06 13:33:11 +03:00
parameters . cast ( spellEnv ) ;
2015-10-28 23:53:44 +03:00
} ;
2014-03-17 19:51:07 +00:00
attackCasting ( bat , Bonus : : SPELL_AFTER_ATTACK , attacker ) ;
2017-07-04 14:24:46 +03:00
if ( ! defender - > alive ( ) )
2014-03-17 19:51:07 +00:00
{
//don't try death stare or acid breath on dead stack (crash!)
return ;
}
2017-07-04 14:24:46 +03:00
if ( attacker - > hasBonusOfType ( Bonus : : DEATH_STARE ) )
2014-03-17 19:51:07 +00:00
{
// mechanics of Death Stare as in H3:
// each gorgon have 10% chance to kill (counted separately in H3) -> binomial distribution
//original formula x = min(x, (gorgons_count + 9)/10);
double chanceToKill = attacker - > valOfBonuses ( Bonus : : DEATH_STARE , 0 ) / 100.0f ;
vstd : : amin ( chanceToKill , 1 ) ; //cap at 100%
2017-07-04 14:24:46 +03:00
std : : binomial_distribution < > distribution ( attacker - > getCount ( ) , chanceToKill ) ;
2014-03-17 19:51:07 +00:00
2016-09-18 13:39:02 +03:00
int staredCreatures = distribution ( getRandomGenerator ( ) . getStdGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
double cap = 1 / std : : max ( chanceToKill , ( double ) ( 0.01 ) ) ; //don't divide by 0
2017-07-04 14:24:46 +03:00
int maxToKill = ( attacker - > getCount ( ) + cap - 1 ) / cap ; //not much more than chance * count
2014-03-17 19:51:07 +00:00
vstd : : amin ( staredCreatures , maxToKill ) ;
2017-07-04 14:24:46 +03:00
staredCreatures + = ( attacker - > level ( ) * attacker - > valOfBonuses ( Bonus : : DEATH_STARE , 1 ) ) / defender - > level ( ) ;
if ( staredCreatures )
2014-03-17 19:51:07 +00:00
{
2017-07-04 14:24:46 +03:00
//TODO: death stare was not originally available for multiple-hex attacks, but...
cast ( SpellID : : DEATH_STARE , staredCreatures ) ;
2014-03-17 19:51:07 +00:00
}
}
2017-07-04 14:24:46 +03:00
if ( ! defender - > alive ( ) )
return ;
2014-03-17 19:51:07 +00:00
int acidDamage = 0 ;
TBonusListPtr acidBreath = attacker - > getBonuses ( Selector : : type ( Bonus : : ACID_BREATH ) ) ;
2017-07-04 14:24:46 +03:00
for ( const std : : shared_ptr < Bonus > b : * acidBreath )
2014-03-17 19:51:07 +00:00
{
2017-07-04 14:24:46 +03:00
if ( b - > additionalInfo > getRandomGenerator ( ) . nextInt ( 99 ) )
2014-03-17 19:51:07 +00:00
acidDamage + = b - > val ;
}
2017-01-22 20:28:08 +01:00
2017-07-04 14:24:46 +03:00
if ( acidDamage )
cast ( SpellID : : ACID_BREATH_DAMAGE , acidDamage * attacker - > getCount ( ) ) ;
if ( ! defender - > alive ( ) )
return ;
if ( attacker - > hasBonusOfType ( Bonus : : TRANSMUTATION ) & & defender - > isLiving ( ) ) //transmutation mechanics, similar to WoG werewolf ability
2017-01-22 20:28:08 +01:00
{
double chanceToTrigger = attacker - > valOfBonuses ( Bonus : : TRANSMUTATION ) / 100.0f ;
vstd : : amin ( chanceToTrigger , 1 ) ; //cap at 100%
2017-07-04 14:24:46 +03:00
if ( getRandomGenerator ( ) . getDoubleRange ( 0 , 1 ) ( ) > chanceToTrigger )
2017-01-22 20:28:08 +01:00
return ;
2017-03-18 13:25:12 +03:00
2017-01-22 20:28:08 +01:00
int bonusAdditionalInfo = attacker - > getBonus ( Selector : : type ( Bonus : : TRANSMUTATION ) ) - > additionalInfo ;
2017-07-04 14:24:46 +03:00
if ( defender - > getCreature ( ) - > idNumber = = bonusAdditionalInfo | |
2017-01-22 20:28:08 +01:00
( bonusAdditionalInfo = = - 1 & & defender - > getCreature ( ) - > idNumber = = attacker - > getCreature ( ) - > idNumber ) )
return ;
BattleStackAdded resurrectInfo ;
resurrectInfo . pos = defender - > position ;
2017-07-01 11:34:00 +03:00
resurrectInfo . side = defender - > side ;
2017-01-22 20:28:08 +01:00
2017-07-04 14:24:46 +03:00
if ( bonusAdditionalInfo ! = - 1 )
2017-01-22 20:28:08 +01:00
resurrectInfo . creID = ( CreatureID ) bonusAdditionalInfo ;
else
resurrectInfo . creID = attacker - > getCreature ( ) - > idNumber ;
2017-03-18 13:25:12 +03:00
2017-07-04 14:24:46 +03:00
if ( attacker - > hasBonusOfType ( ( Bonus : : TRANSMUTATION ) , 0 ) )
resurrectInfo . amount = std : : max ( ( defender - > getCount ( ) * defender - > MaxHealth ( ) ) / resurrectInfo . creID . toCreature ( ) - > MaxHealth ( ) , 1u ) ;
2017-01-22 20:28:08 +01:00
else if ( attacker - > hasBonusOfType ( ( Bonus : : TRANSMUTATION ) , 1 ) )
2017-07-04 14:24:46 +03:00
resurrectInfo . amount = defender - > getCount ( ) ;
2017-01-22 20:28:08 +01:00
else
return ; //wrong subtype
BattleStacksRemoved victimInfo ;
victimInfo . stackIDs . insert ( bat . bsa . at ( 0 ) . stackAttacked ) ;
sendAndApply ( & victimInfo ) ;
sendAndApply ( & resurrectInfo ) ;
}
2014-03-17 19:51:07 +00:00
}
void CGameHandler : : visitObjectOnTile ( const TerrainTile & t , const CGHeroInstance * h )
{
if ( ! t . visitableObjects . empty ( ) )
{
//to prevent self-visiting heroes on space press
2016-10-12 17:16:26 +02:00
if ( t . visitableObjects . back ( ) ! = h )
2014-03-17 19:51:07 +00:00
objectVisited ( t . visitableObjects . back ( ) , h ) ;
2016-10-12 17:16:26 +02:00
else if ( t . visitableObjects . size ( ) > 1 )
2014-03-17 19:51:07 +00:00
objectVisited ( * ( t . visitableObjects . end ( ) - 2 ) , h ) ;
}
}
bool CGameHandler : : sacrificeCreatures ( const IMarket * market , const CGHeroInstance * hero , SlotID slot , ui32 count )
{
2016-11-27 22:50:37 +03:00
if ( ! hero )
COMPLAIN_RET ( " You need hero to sacrifice creature! " ) ;
2014-03-17 19:51:07 +00:00
int oldCount = hero - > getStackCount ( slot ) ;
2016-10-12 17:16:26 +02:00
if ( oldCount < count )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Not enough creatures to sacrifice! " )
2016-10-12 17:16:26 +02:00
else if ( oldCount = = count & & hero - > stacksCount ( ) = = 1 & & hero - > needsLastStack ( ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot sacrifice last creature! " ) ;
int crid = hero - > getStack ( slot ) . type - > idNumber ;
changeStackCount ( StackLocation ( hero , slot ) , - count ) ;
int dump , exp ;
market - > getOffer ( crid , 0 , dump , exp , EMarketMode : : CREATURE_EXP ) ;
exp * = count ;
changePrimSkill ( hero , PrimarySkill : : EXPERIENCE , hero - > calculateXp ( exp ) ) ;
return true ;
}
bool CGameHandler : : sacrificeArtifact ( const IMarket * m , const CGHeroInstance * hero , ArtifactPosition slot )
{
2016-11-27 22:50:37 +03:00
if ( ! hero )
COMPLAIN_RET ( " You need hero to sacrifice artifact! " ) ;
2014-03-17 19:51:07 +00:00
ArtifactLocation al ( hero , slot ) ;
const CArtifactInstance * a = al . getArt ( ) ;
2016-11-26 22:06:12 +03:00
COMPLAIN_RET_FALSE_IF ( ! a , " Cannot find artifact to sacrifice! " ) ;
2014-03-17 19:51:07 +00:00
int dmp , expToGive ;
2016-11-28 02:17:29 +03:00
const CArtifactInstance * art = hero - > getArt ( slot ) ;
COMPLAIN_RET_FALSE_IF ( ( ! art ) , " No artifact at position to sacrifice! " ) ;
2016-11-26 22:06:12 +03:00
2016-11-28 02:17:29 +03:00
si32 typId = art - > artType - > id ;
2016-11-26 22:06:12 +03:00
m - > getOffer ( typId , 0 , dmp , expToGive , EMarketMode : : ARTIFACT_EXP ) ;
2014-03-17 19:51:07 +00:00
removeArtifact ( al ) ;
changePrimSkill ( hero , PrimarySkill : : EXPERIENCE , expToGive ) ;
return true ;
}
void CGameHandler : : makeStackDoNothing ( const CStack * next )
{
BattleAction doNothing ;
doNothing . actionType = Battle : : NO_ACTION ;
doNothing . additionalInfo = 0 ;
doNothing . destinationTile = - 1 ;
2017-07-01 11:34:00 +03:00
doNothing . side = next - > side ;
2014-03-17 19:51:07 +00:00
doNothing . stackNumber = next - > ID ;
makeAutomaticAction ( next , doNothing ) ;
}
bool CGameHandler : : insertNewStack ( const StackLocation & sl , const CCreature * c , TQuantity count )
{
2016-10-12 17:16:26 +02:00
if ( sl . army - > hasStackAtSlot ( sl . slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Slot is already taken! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! sl . slot . validSlot ( ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot insert stack to that slot! " ) ;
InsertNewStack ins ;
ins . sl = sl ;
ins . stack = CStackBasicDescriptor ( c , count ) ;
sendAndApply ( & ins ) ;
return true ;
}
2017-07-15 14:08:20 +03:00
bool CGameHandler : : eraseStack ( const StackLocation & sl , bool forceRemoval )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ! sl . army - > hasStackAtSlot ( sl . slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot find a stack to erase " ) ;
2016-10-12 17:16:26 +02:00
if ( sl . army - > stacksCount ( ) = = 1 //from the last stack
2014-03-17 19:51:07 +00:00
& & sl . army - > needsLastStack ( ) //that must be left
& & ! forceRemoval ) //ignore above conditions if we are forcing removal
{
COMPLAIN_RET ( " Cannot erase the last stack! " ) ;
}
EraseStack es ;
es . sl = sl ;
sendAndApply ( & es ) ;
return true ;
}
2017-07-15 14:08:20 +03:00
bool CGameHandler : : changeStackCount ( const StackLocation & sl , TQuantity count , bool absoluteValue )
2014-03-17 19:51:07 +00:00
{
TQuantity currentCount = sl . army - > getStackCount ( sl . slot ) ;
2016-10-12 17:16:26 +02:00
if ( ( absoluteValue & & count < 0 )
2014-03-17 19:51:07 +00:00
| | ( ! absoluteValue & & - count > currentCount ) )
{
COMPLAIN_RET ( " Cannot take more stacks than present! " ) ;
}
2016-10-12 17:16:26 +02:00
if ( ( currentCount = = - count & & ! absoluteValue )
2014-03-17 19:51:07 +00:00
| | ( ! count & & absoluteValue ) )
{
eraseStack ( sl ) ;
}
else
{
ChangeStackCount csc ;
csc . sl = sl ;
csc . count = count ;
csc . absoluteValue = absoluteValue ;
sendAndApply ( & csc ) ;
}
return true ;
}
bool CGameHandler : : addToSlot ( const StackLocation & sl , const CCreature * c , TQuantity count )
{
const CCreature * slotC = sl . army - > getCreature ( sl . slot ) ;
2016-10-12 17:16:26 +02:00
if ( ! slotC ) //slot is empty
2014-03-17 19:51:07 +00:00
insertNewStack ( sl , c , count ) ;
2016-10-12 17:16:26 +02:00
else if ( c = = slotC )
2014-03-17 19:51:07 +00:00
changeStackCount ( sl , count ) ;
else
{
COMPLAIN_RET ( " Cannot add " + c - > namePl + " to slot " + boost : : lexical_cast < std : : string > ( sl . slot ) + " ! " ) ;
}
return true ;
}
void CGameHandler : : tryJoiningArmy ( const CArmedInstance * src , const CArmedInstance * dst , bool removeObjWhenFinished , bool allowMerging )
{
2016-10-12 17:16:26 +02:00
if ( removeObjWhenFinished )
2014-03-17 19:51:07 +00:00
removeAfterVisit ( src ) ;
2016-10-12 17:16:26 +02:00
if ( ! src - > canBeMergedWith ( * dst , allowMerging ) )
2014-03-17 19:51:07 +00:00
{
if ( allowMerging ) //do that, add all matching creatures.
{
bool cont = true ;
while ( cont )
{
2016-10-12 17:16:26 +02:00
for ( auto i = src - > stacks . begin ( ) ; i ! = src - > stacks . end ( ) ; i + + ) //while there are unmoved creatures
2014-03-17 19:51:07 +00:00
{
SlotID pos = dst - > getSlotFor ( i - > second - > type ) ;
2016-10-12 17:16:26 +02:00
if ( pos . validSlot ( ) )
2014-03-17 19:51:07 +00:00
{
moveStack ( StackLocation ( src , i - > first ) , StackLocation ( dst , pos ) ) ;
cont = true ;
break ; //or iterator crashes
}
cont = false ;
}
}
}
showGarrisonDialog ( src - > id , dst - > id , true ) ; //show garrison window and optionally remove ourselves from map when player ends
}
else //merge
{
moveArmy ( src , dst , allowMerging ) ;
}
}
bool CGameHandler : : moveStack ( const StackLocation & src , const StackLocation & dst , TQuantity count )
{
2016-10-12 17:16:26 +02:00
if ( ! src . army - > hasStackAtSlot ( src . slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " No stack to move! " ) ;
2016-10-12 17:16:26 +02:00
if ( dst . army - > hasStackAtSlot ( dst . slot ) & & dst . army - > getCreature ( dst . slot ) ! = src . army - > getCreature ( src . slot ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot move: stack of different type at destination pos! " ) ;
2016-10-12 17:16:26 +02:00
if ( ! dst . slot . validSlot ( ) )
2014-03-17 19:51:07 +00:00
COMPLAIN_RET ( " Cannot move stack to that slot! " ) ;
2016-10-12 17:16:26 +02:00
if ( count = = - 1 )
2014-03-17 19:51:07 +00:00
{
count = src . army - > getStackCount ( src . slot ) ;
}
2016-10-12 17:16:26 +02:00
if ( src . army ! = dst . army //moving away
2014-03-17 19:51:07 +00:00
& & count = = src . army - > getStackCount ( src . slot ) //all creatures
2015-12-24 21:30:57 +03:00
& & src . army - > stacksCount ( ) = = 1 //from the last stack
2014-03-17 19:51:07 +00:00
& & src . army - > needsLastStack ( ) ) //that must be left
{
COMPLAIN_RET ( " Cannot move away the last creature! " ) ;
}
RebalanceStacks rs ;
rs . src = src ;
rs . dst = dst ;
rs . count = count ;
sendAndApply ( & rs ) ;
return true ;
}
bool CGameHandler : : swapStacks ( const StackLocation & sl1 , const StackLocation & sl2 )
{
2016-10-12 17:16:26 +02:00
if ( ! sl1 . army - > hasStackAtSlot ( sl1 . slot ) )
2014-03-17 19:51:07 +00:00
return moveStack ( sl2 , sl1 ) ;
2016-10-12 17:16:26 +02:00
else if ( ! sl2 . army - > hasStackAtSlot ( sl2 . slot ) )
2014-03-17 19:51:07 +00:00
return moveStack ( sl1 , sl2 ) ;
else
{
SwapStacks ss ;
ss . sl1 = sl1 ;
ss . sl2 = sl2 ;
sendAndApply ( & ss ) ;
return true ;
}
}
2016-10-29 13:45:08 +02:00
void CGameHandler : : runBattle ( )
{
2014-03-17 19:51:07 +00:00
setBattle ( gs - > curB ) ;
assert ( gs - > curB ) ;
//TODO: pre-tactic stuff, call scripts etc.
//tactic round
{
2016-10-28 23:37:45 +02:00
while ( gs - > curB - > tacticDistance & & ! battleResult . get ( ) )
2014-03-17 19:51:07 +00:00
boost : : this_thread : : sleep ( boost : : posix_time : : milliseconds ( 50 ) ) ;
}
2016-10-28 23:37:45 +02:00
//initial stacks appearance triggers, e.g. built-in bonus spells
2017-01-29 11:50:37 +01:00
auto initialStacks = gs - > curB - > stacks ; //use temporary variable to outclude summoned stacks added to gs->curB->stacks from processing
2017-01-26 20:53:28 +01:00
for ( CStack * stack : initialStacks )
2016-10-28 23:37:45 +02:00
{
2017-01-26 20:53:28 +01:00
if ( stack - > hasBonusOfType ( Bonus : : SUMMON_GUARDIANS ) )
{
const std : : shared_ptr < Bonus > summonInfo = stack - > getBonus ( Selector : : type ( Bonus : : SUMMON_GUARDIANS ) ) ;
auto accessibility = getAccesibility ( ) ;
CreatureID creatureData = CreatureID ( summonInfo - > subtype ) ;
std : : vector < BattleHex > targetHexes ;
2017-03-18 13:25:12 +03:00
const bool targetIsBig = stack - > getCreature ( ) - > isDoubleWide ( ) ; //target = creature to guard
const bool guardianIsBig = creatureData . toCreature ( ) - > isDoubleWide ( ) ;
2017-01-26 20:53:28 +01:00
2017-01-29 11:50:37 +01:00
/*Chosen idea for two hex units was to cover all possible surrounding hexes of target unit with as small number of stacks as possible.
For one - hex targets there are four guardians - front , back and one per side ( up + down ) .
Two - hex targets are wider and the difference is there are two guardians per side to cover 3 hexes + extra hex in the front
Additionally , there are special cases for starting positions etc . , where guardians would be outside of battlefield if spawned normally */
2017-03-18 13:25:12 +03:00
if ( ! guardianIsBig )
2017-01-26 20:53:28 +01:00
targetHexes = stack - > getSurroundingHexes ( ) ;
else
2017-07-01 11:34:00 +03:00
summonGuardiansHelper ( targetHexes , stack - > position , stack - > side , targetIsBig ) ;
2017-03-18 13:25:12 +03:00
2017-01-26 20:53:28 +01:00
for ( auto hex : targetHexes )
{
2017-07-01 11:34:00 +03:00
if ( accessibility . accessible ( hex , guardianIsBig , stack - > side ) ) //without this multiple creatures can occupy one hex
2017-01-26 20:53:28 +01:00
{
BattleStackAdded newStack ;
2017-07-04 14:24:46 +03:00
newStack . amount = std : : max ( 1 , ( int ) ( stack - > getCount ( ) * 0.01 * summonInfo - > val ) ) ;
2017-01-26 20:53:28 +01:00
newStack . creID = creatureData . num ;
2017-07-01 11:34:00 +03:00
newStack . side = stack - > side ;
2017-01-26 20:53:28 +01:00
newStack . summoned = true ;
newStack . pos = hex . hex ;
sendAndApply ( & newStack ) ;
}
}
}
2017-01-26 21:24:01 +01:00
2016-11-02 20:11:01 +03:00
stackEnchantedTrigger ( stack ) ;
2016-10-28 23:37:45 +02:00
}
2014-03-17 19:51:07 +00:00
//spells opening battle
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < 2 ; + + i )
2014-03-17 19:51:07 +00:00
{
auto h = gs - > curB - > battleGetFightingHero ( i ) ;
2016-10-12 17:16:26 +02:00
if ( h )
2014-03-17 19:51:07 +00:00
{
2015-10-28 23:53:44 +03:00
TBonusListPtr bl = h - > getBonuses ( Selector : : type ( Bonus : : OPENING_BATTLE_SPELL ) ) ;
2016-09-19 23:36:35 +02:00
for ( auto b : * bl )
2014-03-17 19:51:07 +00:00
{
2014-11-25 22:00:04 +03:00
const CSpell * spell = SpellID ( b - > subtype ) . toSpell ( ) ;
2016-09-18 17:21:56 +03:00
2015-09-17 08:42:30 +03:00
BattleSpellCastParameters parameters ( gs - > curB , h , spell ) ;
parameters . spellLvl = 3 ;
parameters . effectLevel = 3 ;
parameters . mode = ECastingMode : : PASSIVE_CASTING ;
parameters . enchantPower = b - > val ;
2017-03-18 14:08:02 +03:00
parameters . castIfPossible ( spellEnv ) ;
2014-03-17 19:51:07 +00:00
}
}
}
2017-06-13 15:30:21 +03:00
bool firstRound = true ; //FIXME: why first round is -1?
2014-03-17 19:51:07 +00:00
//main loop
2016-10-28 23:37:45 +02:00
while ( ! battleResult . get ( ) ) //till the end of the battle ;]
2014-03-17 19:51:07 +00:00
{
2016-09-29 12:13:06 +03:00
BattleNextRound bnr ;
bnr . round = gs - > curB - > round + 1 ;
2017-06-13 15:30:21 +03:00
logGlobal - > debug ( " Round %d " , bnr . round ) ;
2016-09-29 12:13:06 +03:00
sendAndApply ( & bnr ) ;
2014-03-17 19:51:07 +00:00
auto obstacles = gs - > curB - > obstacles ; //we copy container, because we're going to modify it
2016-10-12 17:16:26 +02:00
for ( auto & obstPtr : obstacles )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( const SpellCreatedObstacle * sco = dynamic_cast < const SpellCreatedObstacle * > ( obstPtr . get ( ) ) )
if ( sco - > turnsRemaining = = 0 )
2014-03-17 19:51:07 +00:00
removeObstacle ( * obstPtr ) ;
}
const BattleInfo & curB = * gs - > curB ;
2016-11-02 20:11:01 +03:00
for ( auto stack : curB . stacks )
{
2017-06-13 15:30:21 +03:00
if ( stack - > alive ( ) & & ! firstRound )
2016-11-02 20:11:01 +03:00
stackEnchantedTrigger ( stack ) ;
}
2014-03-17 19:51:07 +00:00
//stack loop
const CStack * next ;
2016-10-28 23:37:45 +02:00
while ( ! battleResult . get ( ) & & ( next = curB . getNextStack ( ) ) & & next - > willMove ( ) )
2014-03-17 19:51:07 +00:00
{
2016-03-01 11:07:45 +03:00
std : : set < const CStack * > stacksToRemove ;
2016-10-12 17:16:26 +02:00
for ( auto stack : curB . stacks )
2016-03-01 11:07:45 +03:00
{
2016-10-12 17:16:26 +02:00
if ( vstd : : contains ( stack - > state , EBattleStackState : : GHOST_PENDING ) )
2016-03-01 11:07:45 +03:00
stacksToRemove . insert ( stack ) ;
}
2016-10-12 17:16:26 +02:00
for ( auto stack : stacksToRemove )
2016-03-01 11:07:45 +03:00
{
BattleStacksRemoved bsr ;
bsr . stackIDs . insert ( stack - > ID ) ;
sendAndApply ( & bsr ) ;
}
2014-03-17 19:51:07 +00:00
//check for bad morale => freeze
int nextStackMorale = next - > MoraleVal ( ) ;
2016-10-12 17:16:26 +02:00
if ( nextStackMorale < 0 & &
2014-03-17 19:51:07 +00:00
! ( NBonus : : hasOfType ( gs - > curB - > battleGetFightingHero ( 0 ) , Bonus : : BLOCK_MORALE )
| | NBonus : : hasOfType ( gs - > curB - > battleGetFightingHero ( 1 ) , Bonus : : BLOCK_MORALE ) ) //checking if gs->curB->heroes have (or don't have) morale blocking bonuses)
)
{
2016-10-12 17:16:26 +02:00
if ( getRandomGenerator ( ) . nextInt ( 23 ) < - 2 * nextStackMorale )
2014-03-17 19:51:07 +00:00
{
//unit loses its turn - empty freeze action
BattleAction ba ;
ba . actionType = Battle : : BAD_MORALE ;
ba . additionalInfo = 1 ;
2017-07-01 11:34:00 +03:00
ba . side = next - > side ;
2014-03-17 19:51:07 +00:00
ba . stackNumber = next - > ID ;
makeAutomaticAction ( next , ba ) ;
continue ;
}
}
2016-10-12 17:16:26 +02:00
if ( next - > hasBonusOfType ( Bonus : : ATTACKS_NEAREST_CREATURE ) ) //while in berserk
2016-09-29 14:11:15 +03:00
{
logGlobal - > debug ( " Handle Berserk effect " ) ;
2017-07-01 11:34:00 +03:00
std : : pair < const CStack * , int > attackInfo = curB . getNearestStack ( next , boost : : none ) ;
2016-10-12 17:16:26 +02:00
if ( attackInfo . first ! = nullptr )
2014-03-17 19:51:07 +00:00
{
BattleAction attack ;
attack . actionType = Battle : : WALK_AND_ATTACK ;
2017-07-01 11:34:00 +03:00
attack . side = next - > side ;
2014-03-17 19:51:07 +00:00
attack . stackNumber = next - > ID ;
attack . additionalInfo = attackInfo . first - > position ;
attack . destinationTile = attackInfo . second ;
makeAutomaticAction ( next , attack ) ;
2016-09-29 14:11:15 +03:00
logGlobal - > debug ( " Attacked nearest target %s " , attackInfo . first - > nodeName ( ) ) ;
2014-03-17 19:51:07 +00:00
}
else
{
makeStackDoNothing ( next ) ;
2016-09-29 14:11:15 +03:00
logGlobal - > debug ( " No target found " ) ;
2014-03-17 19:51:07 +00:00
}
continue ;
}
2016-09-29 23:14:22 +03:00
const CGHeroInstance * curOwner = battleGetOwnerHero ( next ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( ( next - > position < 0 | | next - > getCreature ( ) - > idNumber = = CreatureID : : BALLISTA ) //arrow turret or ballista
2014-03-17 19:51:07 +00:00
& & ( ! curOwner | | curOwner - > getSecSkillLevel ( SecondarySkill : : ARTILLERY ) = = 0 ) ) //hero has no artillery
{
BattleAction attack ;
attack . actionType = Battle : : SHOOT ;
2017-07-01 11:34:00 +03:00
attack . side = next - > side ;
2014-03-17 19:51:07 +00:00
attack . stackNumber = next - > ID ;
2016-10-12 17:16:26 +02:00
for ( auto & elem : gs - > curB - > stacks )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( elem - > owner ! = next - > owner & & elem - > isValidTarget ( ) )
2014-03-17 19:51:07 +00:00
{
attack . destinationTile = elem - > position ;
break ;
}
}
makeAutomaticAction ( next , attack ) ;
continue ;
}
2016-10-12 17:16:26 +02:00
if ( next - > getCreature ( ) - > idNumber = = CreatureID : : CATAPULT )
2014-03-17 19:51:07 +00:00
{
const auto & attackableBattleHexes = curB . getAttackableBattleHexes ( ) ;
2016-10-12 17:16:26 +02:00
if ( attackableBattleHexes . empty ( ) )
2015-08-27 23:03:48 +02:00
{
makeStackDoNothing ( next ) ;
continue ;
}
2016-10-12 17:16:26 +02:00
if ( ! curOwner | | curOwner - > getSecSkillLevel ( SecondarySkill : : BALLISTICS ) = = 0 )
2014-03-17 19:51:07 +00:00
{
BattleAction attack ;
2015-08-27 23:03:48 +02:00
attack . destinationTile = * RandomGeneratorUtil : : nextItem ( attackableBattleHexes ,
2016-09-08 00:54:26 +03:00
getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
attack . actionType = Battle : : CATAPULT ;
attack . additionalInfo = 0 ;
2017-07-01 11:34:00 +03:00
attack . side = next - > side ;
2014-03-17 19:51:07 +00:00
attack . stackNumber = next - > ID ;
makeAutomaticAction ( next , attack ) ;
2015-08-26 20:54:24 +02:00
continue ;
}
}
2016-10-12 17:16:26 +02:00
if ( next - > getCreature ( ) - > idNumber = = CreatureID : : FIRST_AID_TENT )
2014-03-17 19:51:07 +00:00
{
2016-02-28 05:10:20 +03:00
TStacks possibleStacks = battleGetStacksIf ( [ = ] ( const CStack * s )
{
return s - > owner = = next - > owner & & s - > canBeHealed ( ) ;
2014-05-17 14:36:49 +04:00
} ) ;
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( ! possibleStacks . size ( ) )
2014-03-17 19:51:07 +00:00
{
makeStackDoNothing ( next ) ;
continue ;
}
2016-10-12 17:16:26 +02:00
if ( ! curOwner | | curOwner - > getSecSkillLevel ( SecondarySkill : : FIRST_AID ) = = 0 ) //no hero or hero has no first aid
2014-03-17 19:51:07 +00:00
{
2016-09-09 22:11:13 +03:00
RandomGeneratorUtil : : randomShuffle ( possibleStacks , getRandomGenerator ( ) ) ;
2014-03-17 19:51:07 +00:00
const CStack * toBeHealed = possibleStacks . front ( ) ;
BattleAction heal ;
heal . actionType = Battle : : STACK_HEAL ;
heal . additionalInfo = 0 ;
heal . destinationTile = toBeHealed - > position ;
2017-07-01 11:34:00 +03:00
heal . side = next - > side ;
2014-03-17 19:51:07 +00:00
heal . stackNumber = next - > ID ;
makeAutomaticAction ( next , heal ) ;
continue ;
}
}
int numberOfAsks = 1 ;
bool breakOuter = false ;
do
{ //ask interface and wait for answer
2016-10-12 17:16:26 +02:00
if ( ! battleResult . get ( ) )
2014-03-17 19:51:07 +00:00
{
stackTurnTrigger ( next ) ; //various effects
if ( vstd : : contains ( next - > state , EBattleStackState : : FEAR ) )
{
makeStackDoNothing ( next ) ; //end immediately if stack was affected by fear
}
else
{
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Activating %s " , next - > nodeName ( ) ) ;
2016-03-12 04:41:27 +03:00
auto nextId = next - > ID ;
2015-10-28 23:53:44 +03:00
BattleSetActiveStack sas ;
2015-10-06 02:04:25 +03:00
sas . stack = nextId ;
2014-03-17 19:51:07 +00:00
sendAndApply ( & sas ) ;
2015-10-05 21:09:00 +03:00
auto actionWasMade = [ & ] ( ) - > bool
{
2016-10-12 17:16:26 +02:00
if ( battleMadeAction . data ) //active stack has made its action
2015-10-05 21:09:00 +03:00
return true ;
2016-10-12 17:16:26 +02:00
if ( battleResult . get ( ) ) // battle is finished
2015-10-05 21:09:00 +03:00
return true ;
2016-10-12 17:16:26 +02:00
if ( next = = nullptr ) //active stack was been removed
2015-10-06 03:46:35 +03:00
return true ;
2015-10-05 21:09:00 +03:00
return ! next - > alive ( ) ; //active stack is dead
} ;
2014-03-17 19:51:07 +00:00
boost : : unique_lock < boost : : mutex > lock ( battleMadeAction . mx ) ;
battleMadeAction . data = false ;
2016-10-28 23:37:45 +02:00
while ( ! actionWasMade ( ) )
2015-10-05 21:09:00 +03:00
{
2014-03-17 19:51:07 +00:00
battleMadeAction . cond . wait ( lock ) ;
2016-10-12 17:16:26 +02:00
if ( battleGetStackByID ( nextId , false ) ! = next )
2015-10-06 02:04:25 +03:00
next = nullptr ; //it may be removed, while we wait
2015-10-05 21:09:00 +03:00
}
2014-03-17 19:51:07 +00:00
}
}
2016-10-12 17:16:26 +02:00
if ( battleResult . get ( ) ) //don't touch it, battle could be finished while waiting got action
2014-03-17 19:51:07 +00:00
{
breakOuter = true ;
break ;
}
//we're after action, all results applied
2016-02-09 10:59:33 +03:00
checkBattleStateChanges ( ) ; //check if this action ended the battle
2014-03-17 19:51:07 +00:00
2016-10-12 17:16:26 +02:00
if ( next ! = nullptr )
2014-03-17 19:51:07 +00:00
{
2015-10-06 02:04:25 +03:00
//check for good morale
nextStackMorale = next - > MoraleVal ( ) ;
2016-10-12 17:16:26 +02:00
if ( ! vstd : : contains ( next - > state , EBattleStackState : : HAD_MORALE ) //only one extra move per turn possible
2015-10-06 02:04:25 +03:00
& & ! vstd : : contains ( next - > state , EBattleStackState : : DEFENDING )
& & ! next - > waited ( )
& & ! vstd : : contains ( next - > state , EBattleStackState : : FEAR )
& & next - > alive ( )
& & nextStackMorale > 0
& & ! ( NBonus : : hasOfType ( gs - > curB - > battleGetFightingHero ( 0 ) , Bonus : : BLOCK_MORALE )
| | NBonus : : hasOfType ( gs - > curB - > battleGetFightingHero ( 1 ) , Bonus : : BLOCK_MORALE ) ) //checking if gs->curB->heroes have (or don't have) morale blocking bonuses
)
{
2016-10-12 17:16:26 +02:00
if ( getRandomGenerator ( ) . nextInt ( 23 ) < nextStackMorale ) //this stack hasn't got morale this turn
2016-10-28 23:37:45 +02:00
{
BattleTriggerEffect bte ;
bte . stackID = next - > ID ;
bte . effect = Bonus : : MORALE ;
bte . val = 1 ;
bte . additionalInfo = 0 ;
sendAndApply ( & bte ) ; //play animation
+ + numberOfAsks ; //move this stack once more
}
2015-10-06 02:04:25 +03:00
}
2014-03-17 19:51:07 +00:00
}
- - numberOfAsks ;
} while ( numberOfAsks > 0 ) ;
if ( breakOuter )
{
break ;
}
}
2017-06-13 15:30:21 +03:00
firstRound = false ;
2014-03-17 19:51:07 +00:00
}
endBattle ( gs - > curB - > tile , gs - > curB - > battleGetFightingHero ( 0 ) , gs - > curB - > battleGetFightingHero ( 1 ) ) ;
}
bool CGameHandler : : makeAutomaticAction ( const CStack * stack , BattleAction & ba )
{
BattleSetActiveStack bsa ;
bsa . stack = stack - > ID ;
bsa . askPlayerInterface = false ;
sendAndApply ( & bsa ) ;
bool ret = makeBattleAction ( ba ) ;
2016-02-09 10:59:33 +03:00
checkBattleStateChanges ( ) ;
2014-03-17 19:51:07 +00:00
return ret ;
}
void CGameHandler : : giveHeroArtifact ( const CGHeroInstance * h , const CArtifactInstance * a , ArtifactPosition pos )
{
assert ( a - > artType ) ;
ArtifactLocation al ;
al . artHolder = const_cast < CGHeroInstance * > ( h ) ;
ArtifactPosition slot = ArtifactPosition : : PRE_FIRST ;
2016-10-12 17:16:26 +02:00
if ( pos < 0 )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( pos = = ArtifactPosition : : FIRST_AVAILABLE )
2014-03-17 19:51:07 +00:00
slot = a - > firstAvailableSlot ( h ) ;
else
slot = a - > firstBackpackSlot ( h ) ;
}
else
{
slot = pos ;
}
al . slot = slot ;
2016-10-12 17:16:26 +02:00
if ( slot < 0 | | ! a - > canBePutAt ( al ) )
2014-03-17 19:51:07 +00:00
{
complain ( " Cannot put artifact in that slot! " ) ;
return ;
}
putArtifact ( al , a ) ;
}
void CGameHandler : : putArtifact ( const ArtifactLocation & al , const CArtifactInstance * a )
{
PutArtifact pa ;
pa . art = a ;
pa . al = al ;
sendAndApply ( & pa ) ;
}
2017-06-06 19:45:34 +03:00
bool CGameHandler : : giveHeroNewArtifact ( const CGHeroInstance * h , const CArtifact * art )
2017-05-26 19:51:45 +03:00
{
2017-06-06 19:45:34 +03:00
COMPLAIN_RET_FALSE_IF ( art - > possibleSlots . at ( ArtBearer : : HERO ) . empty ( ) , " Not a hero artifact! " ) ;
2017-05-26 19:51:45 +03:00
ArtifactPosition slot = art - > possibleSlots . at ( ArtBearer : : HERO ) . front ( ) ;
COMPLAIN_RET_FALSE_IF ( nullptr ! = h - > getArt ( slot , false ) , " Hero already has artifact in slot " ) ;
giveHeroNewArtifact ( h , art , slot ) ;
return true ;
}
2014-03-17 19:51:07 +00:00
void CGameHandler : : giveHeroNewArtifact ( const CGHeroInstance * h , const CArtifact * artType , ArtifactPosition pos )
{
CArtifactInstance * a = nullptr ;
2016-10-12 17:16:26 +02:00
if ( ! artType - > constituents )
2014-03-17 19:51:07 +00:00
{
a = new CArtifactInstance ( ) ;
}
else
{
a = new CCombinedArtifactInstance ( ) ;
}
a - > artType = artType ; //*NOT* via settype -> all bonus-related stuff must be done by NewArtifact apply
NewArtifact na ;
na . art = a ;
sendAndApply ( & na ) ; // -> updates a!!!, will create a on other machines
giveHeroArtifact ( h , a , pos ) ;
}
void CGameHandler : : setBattleResult ( BattleResult : : EResult resultType , int victoriusSide )
{
2016-01-25 15:36:34 +03:00
boost : : unique_lock < boost : : mutex > guard ( battleResult . mx ) ;
2016-10-12 17:16:26 +02:00
if ( battleResult . data )
2014-03-17 19:51:07 +00:00
{
2016-01-25 15:36:34 +03:00
complain ( ( boost : : format ( " The battle result has been already set (to %d, asked to %d) " )
% battleResult . data - > result % resultType ) . str ( ) ) ;
2014-03-17 19:51:07 +00:00
return ;
}
2017-07-16 12:58:05 +03:00
auto br = new BattleResult ( ) ;
2014-03-17 19:51:07 +00:00
br - > result = resultType ;
br - > winner = victoriusSide ; //surrendering side loses
gs - > curB - > calculateCasualties ( br - > casualties ) ;
2016-01-25 15:36:34 +03:00
battleResult . data = br ;
2014-03-17 19:51:07 +00:00
}
2016-10-12 17:16:26 +02:00
void CGameHandler : : commitPackage ( CPackForClient * pack )
2014-03-17 19:51:07 +00:00
{
sendAndApply ( pack ) ;
}
void CGameHandler : : spawnWanderingMonsters ( CreatureID creatureID )
{
std : : vector < int3 > : : iterator tile ;
std : : vector < int3 > tiles ;
getFreeTiles ( tiles ) ;
ui32 amount = tiles . size ( ) / 200 ; //Chance is 0.5% for each tile
2016-09-09 22:11:13 +03:00
RandomGeneratorUtil : : randomShuffle ( tiles , getRandomGenerator ( ) ) ;
2016-08-30 01:11:54 +03:00
logGlobal - > trace ( " Spawning wandering monsters. Found %d free tiles. Creature type: %d " , tiles . size ( ) , creatureID . num ) ;
2014-03-17 19:51:07 +00:00
const CCreature * cre = VLC - > creh - > creatures . at ( creatureID ) ;
for ( int i = 0 ; i < amount ; + + i )
{
tile = tiles . begin ( ) ;
2017-08-12 14:36:04 +03:00
logGlobal - > trace ( " \t Spawning monster at %s " , tile - > toString ( ) ) ;
2014-03-17 19:51:07 +00:00
putNewMonster ( creatureID , cre - > getRandomAmount ( std : : rand ) , * tile ) ;
tiles . erase ( tile ) ; //not use it again
}
}
2016-10-02 16:21:46 +03:00
void CGameHandler : : handleCheatCode ( std : : string & cheat , PlayerColor player , const CGHeroInstance * hero , const CGTownInstance * town , bool & cheated )
{
2016-10-12 17:16:26 +02:00
if ( cheat = = " vcmiistari " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! hero ) return ;
2016-10-02 16:21:46 +03:00
///Give hero spellbook
2016-10-12 17:16:26 +02:00
if ( ! hero - > hasSpellbook ( ) )
2016-10-02 16:21:46 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ ArtifactID : : SPELLBOOK ] , ArtifactPosition : : SPELLBOOK ) ;
///Give all spells with bonus (to allow banned spells)
GiveBonus giveBonus ( GiveBonus : : HERO ) ;
giveBonus . id = hero - > id . getNum ( ) ;
giveBonus . bonus = Bonus ( Bonus : : PERMANENT , Bonus : : SPELLS_OF_LEVEL , Bonus : : OTHER , 0 , 0 ) ;
//start with level 0 to skip abilities
2016-10-12 17:16:26 +02:00
for ( int level = 1 ; level < = GameConstants : : SPELL_LEVELS ; level + + )
2016-10-02 16:21:46 +03:00
{
giveBonus . bonus . subtype = level ;
sendAndApply ( & giveBonus ) ;
}
///Give mana
SetMana sm ;
sm . hid = hero - > id ;
sm . val = 999 ;
sm . absolute = true ;
sendAndApply ( & sm ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmiarmenelos " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! town ) return ;
2016-10-02 16:21:46 +03:00
///Build all buildings in selected town
2016-10-12 17:16:26 +02:00
for ( auto & build : town - > town - > buildings )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! town - > hasBuilt ( build . first )
2016-10-02 16:21:46 +03:00
& & ! build . second - > Name ( ) . empty ( )
& & build . first ! = BuildingID : : SHIP )
{
buildStructure ( town - > id , build . first , true ) ;
}
}
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmiainur " | | cheat = = " vcmiangband " | | cheat = = " vcmiglaurung " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! hero ) return ;
2016-10-02 16:21:46 +03:00
///Gives N creatures into each slot
std : : map < std : : string , std : : pair < int , int > > creatures ;
creatures . insert ( std : : make_pair ( " vcmiainur " , std : : make_pair ( 13 , 5 ) ) ) ; //5 archangels
creatures . insert ( std : : make_pair ( " vcmiangband " , std : : make_pair ( 66 , 10 ) ) ) ; //10 black knights
creatures . insert ( std : : make_pair ( " vcmiglaurung " , std : : make_pair ( 133 , 5000 ) ) ) ; //5000 crystal dragons
const CCreature * creature = VLC - > creh - > creatures . at ( creatures [ cheat ] . first ) ;
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < GameConstants : : ARMY_SIZE ; i + + )
if ( ! hero - > hasStackAtSlot ( SlotID ( i ) ) )
2016-10-02 16:21:46 +03:00
insertNewStack ( StackLocation ( hero , SlotID ( i ) ) , creature , creatures [ cheat ] . second ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcminoldor " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! hero ) return ;
2016-10-02 16:21:46 +03:00
///Give all war machines to hero
2016-10-12 17:16:26 +02:00
if ( ! hero - > getArt ( ArtifactPosition : : MACH1 ) )
2016-10-02 16:21:46 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ ArtifactID : : BALLISTA ] , ArtifactPosition : : MACH1 ) ;
2016-10-12 17:16:26 +02:00
if ( ! hero - > getArt ( ArtifactPosition : : MACH2 ) )
2016-10-02 16:21:46 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ ArtifactID : : AMMO_CART ] , ArtifactPosition : : MACH2 ) ;
2016-10-12 17:16:26 +02:00
if ( ! hero - > getArt ( ArtifactPosition : : MACH3 ) )
2016-10-02 16:21:46 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ ArtifactID : : FIRST_AID_TENT ] , ArtifactPosition : : MACH3 ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmiforgeofnoldorking " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! hero ) return ;
2016-10-02 16:21:46 +03:00
///Give hero all artifacts except war machines, spell scrolls and spell book
2016-10-12 17:16:26 +02:00
for ( int g = 7 ; g < VLC - > arth - > artifacts . size ( ) ; + + g ) //including artifacts from mods
2016-10-02 16:21:46 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ g ] , ArtifactPosition : : PRE_FIRST ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmiglorfindel " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! hero ) return ;
2016-10-02 16:21:46 +03:00
///selected hero gains a new level
changePrimSkill ( hero , PrimarySkill : : EXPERIENCE , VLC - > heroh - > reqExp ( hero - > level + 1 ) - VLC - > heroh - > reqExp ( hero - > level ) ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcminahar " )
2016-10-02 16:21:46 +03:00
{
2016-10-12 17:16:26 +02:00
if ( ! hero ) return ;
2016-10-02 16:21:46 +03:00
///Give 1000000 movement points to hero
SetMovePoints smp ;
smp . hid = hero - > id ;
smp . val = 1000000 ;
sendAndApply ( & smp ) ;
2017-06-14 11:41:36 +03:00
GiveBonus gb ( GiveBonus : : HERO ) ;
gb . bonus . type = Bonus : : FREE_SHIP_BOARDING ;
gb . bonus . duration = Bonus : : ONE_DAY ;
gb . bonus . source = Bonus : : OTHER ;
gb . id = hero - > id . getNum ( ) ;
giveHeroBonus ( & gb ) ;
2016-10-02 16:21:46 +03:00
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmiformenos " )
2016-10-02 16:21:46 +03:00
{
///Give resources to player
TResources resources ;
resources [ Res : : GOLD ] = 100000 ;
2016-10-12 17:16:26 +02:00
for ( Res : : ERes i = Res : : WOOD ; i < Res : : GOLD ; vstd : : advance ( i , 1 ) )
2016-10-02 16:21:46 +03:00
resources [ i ] = 100 ;
giveResources ( player , resources ) ;
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmisilmaril " )
2016-10-02 16:21:46 +03:00
{
///Player wins
2017-06-02 03:34:50 +03:00
PlayerCheated pc ;
pc . player = player ;
pc . winningCheatCode = true ;
sendAndApply ( & pc ) ;
2016-10-02 16:21:46 +03:00
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmimelkor " )
2016-10-02 16:21:46 +03:00
{
///Player looses
2017-06-02 03:34:50 +03:00
PlayerCheated pc ;
pc . player = player ;
pc . losingCheatCode = true ;
sendAndApply ( & pc ) ;
2016-10-02 16:21:46 +03:00
}
2016-10-12 17:16:26 +02:00
else if ( cheat = = " vcmieagles " | | cheat = = " vcmiungoliant " )
2016-10-02 16:21:46 +03:00
{
///Reveal or conceal FoW
FoWChange fc ;
fc . mode = ( cheat = = " vcmieagles " ? 1 : 0 ) ;
fc . player = player ;
const auto & fowMap = gs - > getPlayerTeam ( player ) - > fogOfWarMap ;
auto hlp_tab = new int3 [ gs - > map - > width * gs - > map - > height * ( gs - > map - > twoLevel ? 2 : 1 ) ] ;
int lastUnc = 0 ;
2016-10-12 17:16:26 +02:00
for ( int i = 0 ; i < gs - > map - > width ; i + + )
for ( int j = 0 ; j < gs - > map - > height ; j + + )
for ( int k = 0 ; k < ( gs - > map - > twoLevel ? 2 : 1 ) ; k + + )
if ( ! fowMap . at ( i ) . at ( j ) . at ( k ) | | ! fc . mode )
2016-10-02 16:21:46 +03:00
hlp_tab [ lastUnc + + ] = int3 ( i , j , k ) ;
fc . tiles . insert ( hlp_tab , hlp_tab + lastUnc ) ;
delete [ ] hlp_tab ;
sendAndApply ( & fc ) ;
}
else
cheated = false ;
}
2014-03-17 19:51:07 +00:00
void CGameHandler : : removeObstacle ( const CObstacleInstance & obstacle )
{
ObstaclesRemoved obsRem ;
obsRem . obstacles . insert ( obstacle . uniqueID ) ;
sendAndApply ( & obsRem ) ;
}
void CGameHandler : : synchronizeArtifactHandlerLists ( )
{
UpdateArtHandlerLists uahl ;
uahl . treasures = VLC - > arth - > treasures ;
uahl . minors = VLC - > arth - > minors ;
uahl . majors = VLC - > arth - > majors ;
uahl . relics = VLC - > arth - > relics ;
sendAndApply ( & uahl ) ;
}
bool CGameHandler : : isValidObject ( const CGObjectInstance * obj ) const
{
return vstd : : contains ( gs - > map - > objects , obj ) ;
}
bool CGameHandler : : isBlockedByQueries ( const CPack * pack , PlayerColor player )
{
2016-10-12 17:16:26 +02:00
if ( ! strcmp ( typeid ( * pack ) . name ( ) , typeid ( PlayerMessage ) . name ( ) ) )
2014-03-17 19:51:07 +00:00
return false ;
auto query = queries . topQuery ( player ) ;
2016-10-12 17:16:26 +02:00
if ( query & & query - > blocksPack ( pack ) )
2014-03-17 19:51:07 +00:00
{
complain ( boost : : str ( boost : : format ( " Player %s has to answer queries before attempting any further actions. Top query is %s! " ) % player % query - > toString ( ) ) ) ;
return true ;
}
return false ;
}
void CGameHandler : : removeAfterVisit ( const CGObjectInstance * object )
{
//If the object is being visited, there must be a matching query
2016-10-12 17:16:26 +02:00
for ( const auto & query : queries . allQueries ( ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( auto someVistQuery = std : : dynamic_pointer_cast < CObjectVisitQuery > ( query ) )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( someVistQuery - > visitedObject = = object )
2014-03-17 19:51:07 +00:00
{
someVistQuery - > removeObjectAfterVisit = true ;
return ;
}
}
2017-07-12 22:01:10 +03:00
}
2014-03-17 19:51:07 +00:00
//If we haven't returned so far, there is no query and no visit, call was wrong
assert ( " This function needs to be called during the object visit! " ) ;
}
2014-06-24 14:50:27 +03:00
void CGameHandler : : changeFogOfWar ( int3 center , ui32 radius , PlayerColor player , bool hide )
{
std : : unordered_set < int3 , ShashInt3 > tiles ;
getTilesInRange ( tiles , center , radius , player , hide ? - 1 : 1 ) ;
2014-07-01 09:13:19 +02:00
if ( hide )
{
std : : unordered_set < int3 , ShashInt3 > observedTiles ; //do not hide tiles observed by heroes. May lead to disastrous AI problems
2016-09-18 11:53:51 +03:00
auto p = getPlayer ( player ) ;
2014-07-01 09:13:19 +02:00
for ( auto h : p - > heroes )
{
2016-01-31 18:01:58 +03:00
getTilesInRange ( observedTiles , h - > getSightCenter ( ) , h - > getSightRadius ( ) , h - > tempOwner , - 1 ) ;
2014-07-01 09:13:19 +02:00
}
for ( auto t : p - > towns )
{
2016-01-31 18:01:58 +03:00
getTilesInRange ( observedTiles , t - > getSightCenter ( ) , t - > getSightRadius ( ) , t - > tempOwner , - 1 ) ;
2014-07-01 09:13:19 +02:00
}
for ( auto tile : observedTiles )
vstd : : erase_if_present ( tiles , tile ) ;
}
2014-06-24 14:50:27 +03:00
changeFogOfWar ( tiles , player , hide ) ;
}
void CGameHandler : : changeFogOfWar ( std : : unordered_set < int3 , ShashInt3 > & tiles , PlayerColor player , bool hide )
{
FoWChange fow ;
fow . tiles = tiles ;
fow . player = player ;
fow . mode = hide ? 0 : 1 ;
sendAndApply ( & fow ) ;
}
2014-03-17 19:51:07 +00:00
bool CGameHandler : : isVisitCoveredByAnotherQuery ( const CGObjectInstance * obj , const CGHeroInstance * hero )
{
2016-10-12 17:16:26 +02:00
if ( auto topQuery = queries . topQuery ( hero - > getOwner ( ) ) )
if ( auto visit = std : : dynamic_pointer_cast < const CObjectVisitQuery > ( topQuery ) )
2014-03-17 19:51:07 +00:00
return ! ( visit - > visitedObject = = obj & & visit - > visitingHero = = hero ) ;
return true ;
}
2015-11-07 11:42:06 +03:00
CasualtiesAfterBattle : : CasualtiesAfterBattle ( const CArmedInstance * _army , BattleInfo * bat ) :
army ( _army )
2014-03-17 19:51:07 +00:00
{
heroWithDeadCommander = ObjectInstanceID ( ) ;
PlayerColor color = army - > tempOwner ;
2017-07-04 14:24:46 +03:00
if ( color = = PlayerColor : : UNFLAGGABLE )
2014-03-17 19:51:07 +00:00
color = PlayerColor : : NEUTRAL ;
2017-07-04 14:24:46 +03:00
for ( CStack * st : bat - > stacks )
2014-03-17 19:51:07 +00:00
{
2017-07-04 14:24:46 +03:00
if ( vstd : : contains ( st - > state , EBattleStackState : : SUMMONED ) ) //don't take into account temporary summoned stacks
2014-03-17 19:51:07 +00:00
continue ;
2017-07-04 14:24:46 +03:00
if ( st - > owner ! = color ) //remove only our stacks
2014-03-17 19:51:07 +00:00
continue ;
2016-08-30 01:11:54 +03:00
logGlobal - > debug ( " Calculating casualties for %s " , st - > nodeName ( ) ) ;
2016-01-30 00:53:53 +03:00
2017-07-04 14:24:46 +03:00
st - > health . takeResurrected ( ) ;
2014-03-17 19:51:07 +00:00
2017-07-04 14:24:46 +03:00
if ( st - > slot = = SlotID : : ARROW_TOWERS_SLOT )
2016-01-30 00:53:53 +03:00
{
2016-03-01 04:06:32 +03:00
logGlobal - > debug ( " Ignored arrow towers stack. " ) ;
2016-01-30 00:53:53 +03:00
}
2017-07-04 14:24:46 +03:00
else if ( st - > slot = = SlotID : : WAR_MACHINES_SLOT )
2014-03-17 19:51:07 +00:00
{
2017-05-26 19:51:45 +03:00
auto warMachine = st - > type - > warMachine ;
2016-01-30 00:53:53 +03:00
2017-07-04 14:24:46 +03:00
if ( warMachine = = ArtifactID : : NONE )
2017-05-26 19:51:45 +03:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Invalid creature in war machine virtual slot. Stack: %s " , st - > nodeName ( ) ) ;
2017-05-26 19:51:45 +03:00
}
2015-12-10 13:31:03 +03:00
//catapult artifact remain even if "creature" killed in siege
2017-07-04 14:24:46 +03:00
else if ( warMachine ! = ArtifactID : : CATAPULT & & st - > getCount ( ) < = 0 )
2014-03-17 19:51:07 +00:00
{
2016-03-01 04:06:32 +03:00
logGlobal - > debug ( " War machine has been destroyed " ) ;
2015-10-31 18:04:06 +03:00
auto hero = dynamic_ptr_cast < CGHeroInstance > ( army ) ;
2014-03-17 19:51:07 +00:00
if ( hero )
removedWarMachines . push_back ( ArtifactLocation ( hero , hero - > getArtPos ( warMachine , true ) ) ) ;
2016-01-30 00:53:53 +03:00
else
2016-03-01 04:06:32 +03:00
logGlobal - > error ( " War machine in army without hero " ) ;
2014-03-17 19:51:07 +00:00
}
}
2017-07-04 14:24:46 +03:00
else if ( st - > slot = = SlotID : : SUMMONED_SLOT_PLACEHOLDER )
2014-03-17 19:51:07 +00:00
{
2017-07-04 14:24:46 +03:00
if ( st - > alive ( ) & & st - > getCount ( ) > 0 )
2015-11-07 11:42:06 +03:00
{
2017-07-04 14:24:46 +03:00
logGlobal - > debug ( " Permanently summoned %d units. " , st - > getCount ( ) ) ;
2015-11-07 11:42:06 +03:00
const CreatureID summonedType = st - > type - > idNumber ;
2017-07-04 14:24:46 +03:00
summoned [ summonedType ] + = st - > getCount ( ) ;
2016-01-30 00:53:53 +03:00
}
2014-03-17 19:51:07 +00:00
}
2017-07-04 14:24:46 +03:00
else if ( st - > slot = = SlotID : : COMMANDER_SLOT_PLACEHOLDER )
2016-03-01 04:06:32 +03:00
{
2016-10-12 17:16:26 +02:00
if ( nullptr = = st - > base )
2016-03-01 04:06:32 +03:00
{
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Stack with no base in commander slot. Stack: %s " , st - > nodeName ( ) ) ;
2016-03-01 04:06:32 +03:00
}
else
{
auto c = dynamic_cast < const CCommanderInstance * > ( st - > base ) ;
2017-07-04 14:24:46 +03:00
if ( c )
2016-03-01 04:06:32 +03:00
{
auto h = dynamic_cast < const CGHeroInstance * > ( army ) ;
2017-07-04 14:24:46 +03:00
if ( h & & h - > commander = = c & & ( st - > getCount ( ) = = 0 | | ! st - > alive ( ) ) )
2016-03-01 04:06:32 +03:00
{
logGlobal - > debug ( " Commander is dead. " ) ;
heroWithDeadCommander = army - > id ; //TODO: unify commander handling
}
}
else
2016-08-30 01:11:54 +03:00
logGlobal - > error ( " Stack with invalid instance in commander slot. Stack: %s " , st - > nodeName ( ) ) ;
2016-03-01 04:06:32 +03:00
}
}
2017-07-04 14:24:46 +03:00
else if ( st - > base & & ! army - > slotEmpty ( st - > slot ) )
2014-03-17 19:51:07 +00:00
{
2017-07-14 23:43:03 +03:00
logGlobal - > debug ( " Count: %d; base count: %d " , st - > getCount ( ) , army - > getStackCount ( st - > slot ) ) ;
2017-07-04 14:24:46 +03:00
if ( st - > getCount ( ) = = 0 | | ! st - > alive ( ) )
2014-03-17 19:51:07 +00:00
{
2016-03-01 04:06:32 +03:00
logGlobal - > debug ( " Stack has been destroyed. " ) ;
StackLocation sl ( army , st - > slot ) ;
newStackCounts . push_back ( TStackAndItsNewCount ( sl , 0 ) ) ;
2014-03-17 19:51:07 +00:00
}
2017-07-04 14:24:46 +03:00
else if ( st - > getCount ( ) < army - > getStackCount ( st - > slot ) )
2016-01-30 00:53:53 +03:00
{
2017-07-04 14:24:46 +03:00
logGlobal - > debug ( " Stack lost %d units. " , army - > getStackCount ( st - > slot ) - st - > getCount ( ) ) ;
2015-11-07 11:42:06 +03:00
StackLocation sl ( army , st - > slot ) ;
2017-07-04 14:24:46 +03:00
newStackCounts . push_back ( TStackAndItsNewCount ( sl , st - > getCount ( ) ) ) ;
2014-03-17 19:51:07 +00:00
}
2017-07-04 14:24:46 +03:00
else if ( st - > getCount ( ) > army - > getStackCount ( st - > slot ) )
2017-01-20 15:48:45 +01:00
{
2017-07-04 14:24:46 +03:00
logGlobal - > debug ( " Stack gained %d units. " , st - > getCount ( ) - army - > getStackCount ( st - > slot ) ) ;
2017-01-20 15:48:45 +01:00
StackLocation sl ( army , st - > slot ) ;
2017-07-04 14:24:46 +03:00
newStackCounts . push_back ( TStackAndItsNewCount ( sl , st - > getCount ( ) ) ) ;
2017-01-20 15:48:45 +01:00
}
2014-03-17 19:51:07 +00:00
}
2016-01-30 00:53:53 +03:00
else
{
2016-08-30 01:11:54 +03:00
logGlobal - > warn ( " Unable to process stack: %s " , st - > nodeName ( ) ) ;
2014-03-17 19:51:07 +00:00
}
}
}
2015-11-07 11:42:06 +03:00
void CasualtiesAfterBattle : : updateArmy ( CGameHandler * gh )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
for ( TStackAndItsNewCount & ncount : newStackCounts )
2014-03-17 19:51:07 +00:00
{
2016-10-12 17:16:26 +02:00
if ( ncount . second > 0 )
2014-03-17 19:51:07 +00:00
gh - > changeStackCount ( ncount . first , ncount . second , true ) ;
else
gh - > eraseStack ( ncount . first , true ) ;
}
2016-10-12 17:16:26 +02:00
for ( auto summoned_iter : summoned )
2015-11-07 11:42:06 +03:00
{
SlotID slot = army - > getSlotFor ( summoned_iter . first ) ;
2016-10-12 17:16:26 +02:00
if ( slot . validSlot ( ) )
2015-11-07 11:42:06 +03:00
{
StackLocation location ( army , slot ) ;
gh - > addToSlot ( location , summoned_iter . first . toCreature ( ) , summoned_iter . second ) ;
}
else
{
//even if it will be possible to summon anything permanently it should be checked for free slot
//necromancy is handled separately
gh - > complain ( " No free slot to put summoned creature " ) ;
}
}
2014-03-17 19:51:07 +00:00
for ( auto al : removedWarMachines )
{
gh - > removeArtifact ( al ) ;
}
if ( heroWithDeadCommander ! = ObjectInstanceID ( ) )
{
SetCommanderProperty scp ;
scp . heroid = heroWithDeadCommander ;
scp . which = SetCommanderProperty : : ALIVE ;
scp . amount = 0 ;
2016-10-29 13:45:08 +02:00
gh - > sendAndApply ( & scp ) ;
2014-03-17 19:51:07 +00:00
}
}
2017-07-01 16:30:13 +03:00
CGameHandler : : FinishingBattleHelper : : FinishingBattleHelper ( std : : shared_ptr < const CBattleQuery > Query , int RemainingBattleQueriesCount )
2014-03-17 19:51:07 +00:00
{
assert ( Query - > result ) ;
assert ( Query - > bi ) ;
auto & result = * Query - > result ;
auto & info = * Query - > bi ;
winnerHero = result . winner ! = 0 ? info . sides [ 1 ] . hero : info . sides [ 0 ] . hero ;
loserHero = result . winner ! = 0 ? info . sides [ 0 ] . hero : info . sides [ 1 ] . hero ;
victor = info . sides [ result . winner ] . color ;
loser = info . sides [ ! result . winner ] . color ;
remainingBattleQueriesCount = RemainingBattleQueriesCount ;
}
CGameHandler : : FinishingBattleHelper : : FinishingBattleHelper ( )
{
winnerHero = loserHero = nullptr ;
2016-11-27 22:37:41 +03:00
remainingBattleQueriesCount = 0 ;
2014-03-17 19:51:07 +00:00
}
2014-11-25 18:16:14 +03:00
2016-09-09 20:30:36 +03:00
CRandomGenerator & CGameHandler : : getRandomGenerator ( )
{
return CRandomGenerator : : getDefault ( ) ;
}
2014-12-26 00:02:50 +03:00
///ServerSpellCastEnvironment
2014-11-25 18:16:14 +03:00
ServerSpellCastEnvironment : : ServerSpellCastEnvironment ( CGameHandler * gh ) : gh ( gh )
{
2015-10-28 23:53:44 +03:00
2014-11-25 18:16:14 +03:00
}
void ServerSpellCastEnvironment : : sendAndApply ( CPackForClient * info ) const
{
gh - > sendAndApply ( info ) ;
}
CRandomGenerator & ServerSpellCastEnvironment : : getRandomGenerator ( ) const
{
2016-09-08 00:54:26 +03:00
return gh - > getRandomGenerator ( ) ;
2014-11-25 18:16:14 +03:00
}
void ServerSpellCastEnvironment : : complain ( const std : : string & problem ) const
{
gh - > complain ( problem ) ;
}
2014-12-26 00:02:50 +03:00
const CGameInfoCallback * ServerSpellCastEnvironment : : getCb ( ) const
{
return gh ;
}
const CMap * ServerSpellCastEnvironment : : getMap ( ) const
{
return gh - > gameState ( ) - > map ;
}
2017-07-03 21:09:27 +03:00
bool ServerSpellCastEnvironment : : moveHero ( ObjectInstanceID hid , int3 dst , bool teleporting ) const
{
return gh - > moveHero ( hid , dst , teleporting , false ) ;
}
void ServerSpellCastEnvironment : : genericQuery ( Query * request , PlayerColor color , std : : function < void ( const JsonNode & ) > callback ) const
2014-12-26 00:02:50 +03:00
{
2017-07-03 21:09:27 +03:00
auto query = std : : make_shared < CGenericQuery > ( & gh - > queries , color , callback ) ;
request - > queryID = query - > queryID ;
gh - > queries . addQuery ( query ) ;
gh - > sendAndApply ( request ) ;
2014-12-26 00:02:50 +03:00
}