2011-12-14 00:23:17 +03:00
# include "StdInc.h"
2013-07-28 17:49:50 +03:00
# include "../lib/filesystem/Filesystem.h"
2013-04-07 13:48:07 +03:00
# include "../lib/filesystem/CFileInfo.h"
2011-12-14 00:23:17 +03:00
# include "../lib/int3.h"
2013-04-07 13:48:07 +03:00
# include "../lib/mapping/CCampaignHandler.h"
2011-12-14 00:23:17 +03:00
# include "../lib/StartInfo.h"
2012-08-10 16:07:53 +03:00
# include "../lib/CModHandler.h"
2010-12-20 23:22:53 +02:00
# include "../lib/CArtHandler.h"
# include "../lib/CBuildingHandler.h"
# include "../lib/CDefObjInfoHandler.h"
# include "../lib/CHeroHandler.h"
# include "../lib/CObjectHandler.h"
# include "../lib/CSpellHandler.h"
# include "../lib/CGeneralTextHandler.h"
# include "../lib/CTownHandler.h"
# include "../lib/CCreatureHandler.h"
2009-05-20 13:08:56 +03:00
# include "../lib/CGameState.h"
2010-12-25 21:23:30 +02:00
# include "../lib/BattleState.h"
2008-08-04 18:56:36 +03:00
# include "../lib/CondSh.h"
2008-11-28 03:36:34 +02:00
# include "../lib/NetPacks.h"
# include "../lib/VCMI_Lib.h"
2013-04-07 13:48:07 +03:00
# include "../lib/mapping/CMap.h"
2009-10-10 08:47:59 +03:00
# include "../lib/VCMIDirs.h"
2013-08-19 14:50:53 +03:00
# include "../lib/ScopeGuard.h"
2010-12-20 23:22:53 +02:00
# include "../client/CSoundBase.h"
2008-11-28 03:36:34 +02:00
# include "CGameHandler.h"
2013-06-21 23:59:32 +03:00
# include "CVCMIServer.h"
2011-05-10 01:20:47 +03:00
# include "../lib/CCreatureSet.h"
2011-12-14 00:23:17 +03:00
# include "../lib/CThreadHelper.h"
# include "../lib/GameConstants.h"
# include "../lib/RegisterTypes.h"
2009-03-09 12:37:49 +02:00
2009-04-15 17:03:31 +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
*
*/
2011-12-14 00:23:17 +03:00
2008-08-09 02:02:32 +03:00
# ifndef _MSC_VER
2008-08-04 12:05:52 +03:00
# include <boost/thread/xtime.hpp>
2008-08-09 02:02:32 +03:00
# endif
2008-08-04 12:05:52 +03:00
extern bool end2 ;
2008-08-04 18:56:36 +03:00
# ifdef min
# undef min
# endif
# ifdef max
# undef max
# endif
2010-03-11 01:16:30 +02:00
2010-12-13 01:44:16 +02:00
# define COMPLAIN_RET_IF(cond, txt) do {if(cond){complain(txt); return;}} while(0)
2013-04-20 14:34:01 +03:00
# define COMPLAIN_RET_FALSE_IF(cond, txt) do {if(cond){complain(txt); return false;}} while(0)
2010-03-11 01:16:30 +02:00
# define COMPLAIN_RET(txt) {complain(txt); return false;}
2012-01-03 04:55:26 +03:00
# define COMPLAIN_RETF(txt, FORMAT) {complain(boost::str(boost::format(txt) % FORMAT)); return false;}
2008-08-04 18:56:36 +03:00
# define NEW_ROUND BattleNextRound bnr;\
bnr . round = gs - > curB - > round + 1 ; \
sendAndApply ( & bnr ) ;
2008-07-25 20:28:28 +03:00
2008-08-04 18:56:36 +03:00
CondSh < bool > battleMadeAction ;
2013-06-26 14:18:27 +03:00
CondSh < BattleResult * > battleResult ( nullptr ) ;
2010-09-04 17:47:39 +03:00
template < typename T > class CApplyOnGH ;
2010-09-03 21:42:54 +03:00
2009-03-09 12:37:49 +02:00
class CBaseForGHApply
{
public :
2013-03-03 20:06:03 +03:00
virtual bool applyOnGH ( CGameHandler * gh , CConnection * c , void * pack , PlayerColor player ) const = 0 ;
2010-09-03 21:42:54 +03:00
virtual ~ CBaseForGHApply ( ) { }
2013-06-26 14:18:27 +03:00
template < typename U > static CBaseForGHApply * getApplier ( const U * t = nullptr )
2010-09-03 21:42:54 +03:00
{
return new CApplyOnGH < U > ;
}
2009-03-09 12:37:49 +02:00
} ;
2010-09-04 17:47:39 +03:00
2009-03-09 12:37:49 +02:00
template < typename T > class CApplyOnGH : public CBaseForGHApply
{
public :
2013-03-03 20:06:03 +03:00
bool applyOnGH ( CGameHandler * gh , CConnection * c , void * pack , PlayerColor player ) const
2009-03-09 12:37:49 +02:00
{
T * ptr = static_cast < T * > ( pack ) ;
2009-03-09 21:40:43 +02:00
ptr - > c = c ;
2012-01-03 04:55:26 +03:00
ptr - > player = player ;
2009-04-16 03:28:54 +03:00
return ptr - > applyGh ( gh ) ;
2009-03-09 12:37:49 +02:00
}
} ;
2013-06-26 14:18:27 +03:00
static CApplier < CBaseForGHApply > * applier = nullptr ;
2009-03-09 12:37:49 +02:00
2009-09-07 05:29:44 +03:00
CMP_stack cmpst ;
2008-08-04 18:56:36 +03:00
2009-05-24 01:57:39 +03:00
static inline double distance ( int3 a , int3 b )
2008-07-28 15:44:08 +03:00
{
return std : : sqrt ( ( double ) ( a . x - b . x ) * ( a . x - b . x ) + ( a . y - b . y ) * ( a . y - b . y ) ) ;
}
2009-05-24 01:57:39 +03:00
static void giveExp ( BattleResult & r )
2008-09-29 13:16:02 +03:00
{
r . exp [ 0 ] = 0 ;
r . exp [ 1 ] = 0 ;
2013-06-29 16:05:48 +03:00
for ( auto i = r . casualties [ ! r . winner ] . begin ( ) ; i ! = r . casualties [ ! r . winner ] . end ( ) ; i + + )
2008-09-29 13:16:02 +03:00
{
2010-05-02 21:20:26 +03:00
r . exp [ r . winner ] + = VLC - > creh - > creatures [ i - > first ] - > valOfBonuses ( Bonus : : STACK_HEALTH ) * i - > second ;
2008-09-29 13:16:02 +03:00
}
}
2009-03-31 23:47:53 +03:00
2013-03-03 20:06:03 +03:00
PlayerStatus PlayerStatuses : : operator [ ] ( PlayerColor player )
2008-08-13 03:44:31 +03:00
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
if ( players . find ( player ) ! = players . end ( ) )
{
return players [ player ] ;
}
else
{
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " No such player! " ) ;
2008-08-13 03:44:31 +03:00
}
}
2013-03-03 20:06:03 +03:00
void PlayerStatuses : : addPlayer ( PlayerColor player )
2008-08-13 03:44:31 +03:00
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
players [ player ] ;
}
2012-01-03 04:55:26 +03:00
2013-03-03 20:06:03 +03:00
bool PlayerStatuses : : checkFlag ( PlayerColor player , bool PlayerStatus : : * flag )
2008-08-13 03:44:31 +03:00
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
if ( players . find ( player ) ! = players . end ( ) )
{
return players [ player ] . * flag ;
}
else
{
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " No such player! " ) ;
2008-08-13 03:44:31 +03:00
}
}
2013-04-20 14:34:01 +03:00
2013-03-03 20:06:03 +03:00
void PlayerStatuses : : setFlag ( PlayerColor player , bool PlayerStatus : : * flag , bool val )
2008-08-13 03:44:31 +03:00
{
boost : : unique_lock < boost : : mutex > l ( mx ) ;
if ( players . find ( player ) ! = players . end ( ) )
{
players [ player ] . * flag = val ;
}
else
{
2012-04-22 10:32:45 +03:00
throw std : : runtime_error ( " No such player! " ) ;
2008-08-13 03:44:31 +03:00
}
cv . notify_all ( ) ;
}
2009-03-31 23:47:53 +03:00
2008-08-13 03:44:31 +03:00
template < typename T >
2013-06-26 14:18:27 +03:00
void callWith ( std : : vector < T > args , std : : function < void ( T ) > fun , ui32 which )
2008-08-13 03:44:31 +03:00
{
fun ( args [ which ] ) ;
}
2013-02-12 22:49:40 +03:00
void CGameHandler : : levelUpHero ( const CGHeroInstance * hero , SecondarySkill skill )
2008-08-13 03:44:31 +03:00
{
2013-02-09 01:42:46 +03:00
changeSecSkill ( hero , skill , 1 , 0 ) ;
2013-04-20 14:34:01 +03:00
expGiven ( hero ) ;
2010-07-29 01:39:56 +03:00
}
2009-07-31 23:10:22 +03:00
2013-02-09 01:42:46 +03:00
void CGameHandler : : levelUpHero ( const CGHeroInstance * hero )
2010-07-29 01:39:56 +03:00
{
2011-07-30 18:04:34 +03:00
// required exp for at least 1 lvl-up hasn't been reached
2013-04-20 14:34:01 +03:00
if ( ! hero - > gainsLevel ( ) )
2011-07-30 21:02:31 +03:00
{
2010-07-29 01:39:56 +03:00
return ;
2011-07-30 21:02:31 +03:00
}
2012-07-23 13:23:43 +03:00
2010-07-29 01:39:56 +03:00
//give prim skill
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < hero - > name < < " got level " < < hero - > level ;
2010-07-29 01:39:56 +03:00
int r = rand ( ) % 100 , pom = 0 , x = 0 ;
2012-12-14 18:32:53 +03:00
auto & skillChances = ( hero - > level > 9 ) ? hero - > type - > heroClass - > primarySkillLowLevel : hero - > type - > heroClass - > primarySkillHighLevel ;
2011-12-14 00:23:17 +03:00
for ( ; x < GameConstants : : PRIMARY_SKILLS ; x + + )
2009-07-31 23:10:22 +03:00
{
2012-12-14 18:32:53 +03:00
pom + = skillChances [ x ] ;
2010-07-29 01:39:56 +03:00
if ( r < pom )
break ;
}
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " The hero gets the primary skill with the no. " < < x < < " with a probability of " < < r < < " %. " ;
2010-07-29 01:39:56 +03:00
SetPrimSkill sps ;
2013-02-09 01:42:46 +03:00
sps . id = hero - > id ;
2013-02-04 22:43:16 +03:00
sps . which = static_cast < PrimarySkill : : PrimarySkill > ( x ) ;
2010-07-29 01:39:56 +03:00
sps . abs = false ;
sps . val = 1 ;
sendAndApply ( & sps ) ;
HeroLevelUp hlu ;
2013-04-20 14:34:01 +03:00
hlu . hero = hero ;
2013-02-04 22:43:16 +03:00
hlu . primskill = static_cast < PrimarySkill : : PrimarySkill > ( x ) ;
2010-07-29 01:39:56 +03:00
hlu . level = hero - > level + 1 ;
2013-04-20 14:34:01 +03:00
hlu . skills = hero - > levelUpProposedSkills ( ) ;
2010-07-29 01:39:56 +03:00
2013-04-20 14:34:01 +03:00
if ( hlu . skills . size ( ) = = 0 )
2010-07-29 01:39:56 +03:00
{
2013-04-20 14:34:01 +03:00
sendAndApply ( & hlu ) ;
levelUpHero ( hero ) ;
2010-07-29 01:39:56 +03:00
}
2013-04-20 14:34:01 +03:00
else if ( hlu . skills . size ( ) = = 1 | | hero - > tempOwner = = PlayerColor : : NEUTRAL ) //choose skill automatically
2010-07-29 01:39:56 +03:00
{
2013-04-20 14:34:01 +03:00
sendAndApply ( & hlu ) ;
levelUpHero ( hero , vstd : : pickRandomElementOf ( hlu . skills , rand ) ) ;
2010-07-29 01:39:56 +03:00
}
2013-04-20 14:34:01 +03:00
else if ( hlu . skills . size ( ) > 1 )
2010-07-29 01:39:56 +03:00
{
2013-04-20 14:34:01 +03:00
auto levelUpQuery = make_shared < CHeroLevelUpDialogQuery > ( hlu ) ;
hlu . queryID = levelUpQuery - > queryID ;
queries . addQuery ( levelUpQuery ) ;
sendAndApply ( & hlu ) ;
//level up will be called on query reply
2009-07-31 23:10:22 +03:00
}
2008-08-13 03:44:31 +03:00
}
2009-01-11 00:08:18 +02:00
2012-05-18 17:02:27 +03:00
void CGameHandler : : levelUpCommander ( const CCommanderInstance * c , int skill )
2012-05-07 15:54:22 +03:00
{
2012-05-18 17:02:27 +03:00
SetCommanderProperty scp ;
auto hero = dynamic_cast < const CGHeroInstance * > ( c - > armyObj ) ;
if ( hero )
scp . heroid = hero - > id ;
else
2012-05-07 15:54:22 +03:00
{
2012-05-18 17:02:27 +03:00
complain ( " Commander is not led by hero! " ) ;
return ;
}
2012-05-19 21:27:23 +03:00
scp . accumulatedBonus . subtype = 0 ;
2012-05-18 17:02:27 +03:00
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 )
{
2012-05-18 22:44:15 +03:00
scp . which = SetCommanderProperty : : BONUS ;
2012-05-18 17:02:27 +03:00
auto difference = [ ] ( std : : vector < std : : vector < ui8 > > skillLevels , std : : vector < ui8 > secondarySkills , int skill ) - > int
{
2012-07-04 17:41:53 +03:00
int s = std : : min ( skill , ( int ) ECommander : : SPELL_POWER ) ; //spell power level controls also casts and resistance
return skillLevels [ skill ] [ secondarySkills [ s ] ] - ( secondarySkills [ s ] ? skillLevels [ skill ] [ secondarySkills [ s ] - 1 ] : 0 ) ;
2012-05-18 17:02:27 +03:00
} ;
2012-05-18 22:44:15 +03:00
2012-05-18 17:02:27 +03:00
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
2013-02-06 11:02:46 +03:00
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
2012-05-18 17:02:27 +03:00
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 [ skill ] + 1 ;
sendAndApply ( & scp ) ;
2012-05-07 15:54:22 +03:00
}
2012-05-18 17:02:27 +03:00
else if ( skill > = 100 )
2012-05-07 15:54:22 +03:00
{
2012-05-18 22:44:15 +03:00
scp . which = SetCommanderProperty : : SPECIAL_SKILL ;
2013-06-17 18:45:55 +03:00
scp . accumulatedBonus = * VLC - > creh - > skillRequirements [ skill - 100 ] . first ;
2012-05-18 22:44:15 +03:00
scp . additionalInfo = skill ; //unnormalized
2012-05-18 17:02:27 +03:00
sendAndApply ( & scp ) ;
2012-05-07 15:54:22 +03:00
}
2013-04-20 14:34:01 +03:00
expGiven ( hero ) ;
2012-05-07 15:54:22 +03:00
}
void CGameHandler : : levelUpCommander ( const CCommanderInstance * c )
{
2013-04-20 14:34:01 +03:00
if ( ! c - > gainsLevel ( ) )
2012-05-18 17:02:27 +03:00
{
return ;
}
2012-05-07 15:54:22 +03:00
CommanderLevelUp clu ;
2012-05-18 17:02:27 +03:00
auto hero = dynamic_cast < const CGHeroInstance * > ( c - > armyObj ) ;
if ( hero )
2013-04-20 14:34:01 +03:00
clu . hero = hero ;
2012-05-18 17:02:27 +03:00
else
{
complain ( " Commander is not led by hero! " ) ;
return ;
}
2012-05-07 15:54:22 +03:00
2012-05-18 17:02:27 +03:00
//picking sec. skills for choice
2012-05-07 15:54:22 +03:00
2012-05-18 17:02:27 +03:00
for ( int i = 0 ; i < = ECommander : : SPELL_POWER ; + + i )
{
if ( c - > secondarySkills [ i ] < ECommander : : MAX_SKILL_LEVEL )
clu . skills . push_back ( i ) ;
}
int i = 100 ;
2013-06-29 16:05:48 +03:00
for ( auto specialSkill : VLC - > creh - > skillRequirements )
2012-05-18 17:02:27 +03:00
{
2013-04-20 14:34:01 +03:00
if ( c - > secondarySkills [ specialSkill . second . first ] = = ECommander : : MAX_SKILL_LEVEL
& & c - > secondarySkills [ specialSkill . second . second ] = = ECommander : : MAX_SKILL_LEVEL
& & ! vstd : : contains ( c - > specialSKills , i ) )
2012-05-18 17:02:27 +03:00
clu . skills . push_back ( i ) ;
+ + i ;
}
int skillAmount = clu . skills . size ( ) ;
2012-05-07 15:54:22 +03:00
2013-04-20 14:34:01 +03:00
if ( ! skillAmount )
2012-05-07 15:54:22 +03:00
{
sendAndApply ( & clu ) ;
2013-04-20 14:34:01 +03:00
levelUpCommander ( c ) ;
2012-05-07 15:54:22 +03:00
}
2013-04-20 14:34:01 +03:00
else if ( skillAmount = = 1 | | hero - > tempOwner = = PlayerColor : : NEUTRAL ) //choose skill automatically
2012-05-07 15:54:22 +03:00
{
2013-04-20 14:34:01 +03:00
sendAndApply ( & clu ) ;
levelUpCommander ( c , vstd : : pickRandomElementOf ( clu . skills , rand ) ) ;
}
else if ( skillAmount > 1 ) //apply and ask for secondary skill
{
auto commanderLevelUp = make_shared < CCommanderLevelUpDialogQuery > ( clu ) ;
clu . queryID = commanderLevelUp - > queryID ;
queries . addQuery ( commanderLevelUp ) ;
sendAndApply ( & clu ) ;
2012-05-07 15:54:22 +03:00
}
}
2013-04-20 14:34:01 +03:00
void CGameHandler : : expGiven ( const CGHeroInstance * hero )
{
if ( hero - > gainsLevel ( ) )
levelUpHero ( hero ) ;
else if ( hero - > commander & & hero - > commander - > gainsLevel ( ) )
levelUpCommander ( hero - > commander ) ;
//if(hero->commander && hero->level > hero->commander->level && hero->commander->gainsLevel())
// levelUpCommander(hero->commander);
// else
// levelUpHero(hero);
}
2013-02-09 00:17:39 +03:00
void CGameHandler : : changePrimSkill ( const CGHeroInstance * hero , PrimarySkill : : PrimarySkill which , si64 val , bool abs )
2008-08-04 18:56:36 +03:00
{
SetPrimSkill sps ;
2013-02-09 00:17:39 +03:00
sps . id = hero - > id ;
2008-08-04 18:56:36 +03:00
sps . which = which ;
sps . abs = abs ;
sps . val = val ;
sendAndApply ( & sps ) ;
2012-07-23 13:23:43 +03:00
2011-07-30 18:04:34 +03:00
//only for exp - hero may level up
2013-02-04 22:43:16 +03:00
if ( which = = PrimarySkill : : EXPERIENCE )
2008-08-04 18:56:36 +03:00
{
2013-04-20 14:34:01 +03:00
if ( hero - > commander & & hero - > commander - > alive )
2012-05-18 22:44:15 +03:00
{
SetCommanderProperty scp ;
2013-02-09 00:17:39 +03:00
scp . heroid = hero - > id ;
2012-05-18 22:44:15 +03:00
scp . which = SetCommanderProperty : : EXPERIENCE ;
scp . amount = val ;
sendAndApply ( & scp ) ;
CBonusSystemNode : : treeHasChanged ( ) ;
}
2013-04-20 14:34:01 +03:00
expGiven ( hero ) ;
2010-07-29 01:39:56 +03:00
}
}
2009-02-14 17:00:29 +02:00
2013-02-12 22:49:40 +03:00
void CGameHandler : : changeSecSkill ( const CGHeroInstance * hero , SecondarySkill which , int val , bool abs /*=false*/ )
2010-07-29 01:39:56 +03:00
{
SetSecSkill sss ;
2013-02-09 01:42:46 +03:00
sss . id = hero - > id ;
2010-07-29 01:39:56 +03:00
sss . which = which ;
sss . val = val ;
sss . abs = abs ;
sendAndApply ( & sss ) ;
2009-02-14 17:00:29 +02:00
2013-02-04 22:43:16 +03:00
if ( which = = SecondarySkill : : WISDOM )
2010-07-29 01:39:56 +03:00
{
2013-02-09 01:42:46 +03:00
if ( hero & & hero - > visitedTown )
giveSpells ( hero - > visitedTown , hero ) ;
2008-08-04 18:56:36 +03:00
}
}
2009-09-28 17:21:48 +03:00
void CGameHandler : : endBattle ( int3 tile , const CGHeroInstance * hero1 , const CGHeroInstance * hero2 )
{
2013-04-20 14:34:01 +03:00
LOG_TRACE ( logGlobal ) ;
2008-09-20 21:30:37 +03:00
2013-04-20 14:34:01 +03:00
//Fill BattleResult structure with exp info
2008-09-29 13:16:02 +03:00
giveExp ( * battleResult . data ) ;
2013-07-21 21:27:33 +03:00
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 ;
}
2010-07-26 01:47:59 +03:00
if ( hero1 )
2011-02-12 20:48:11 +02:00
battleResult . data - > exp [ 0 ] = hero1 - > calculateXp ( battleResult . data - > exp [ 0 ] ) ; //scholar skill
2010-07-26 01:47:59 +03:00
if ( hero2 )
2011-02-12 20:48:11 +02:00
battleResult . data - > exp [ 1 ] = hero2 - > calculateXp ( battleResult . data - > exp [ 1 ] ) ;
2010-12-06 01:10:02 +02:00
2013-07-22 01:01:29 +03:00
const CArmedInstance * bEndArmy1 = gs - > curB - > sides [ 0 ] . armyObject ;
const CArmedInstance * bEndArmy2 = gs - > curB - > sides [ 1 ] . armyObject ;
2013-04-20 14:34:01 +03:00
const BattleResult : : EResult result = battleResult . get ( ) - > result ;
auto findBattleQuery = [ this ] ( ) - > shared_ptr < CBattleQuery >
2011-01-09 19:41:46 +02:00
{
2013-06-29 16:05:48 +03:00
for ( auto & q : queries . allQueries ( ) )
2013-04-20 14:34:01 +03:00
{
if ( auto bq = std : : dynamic_pointer_cast < CBattleQuery > ( q ) )
if ( bq - > bi = = gs - > curB )
return bq ;
}
2013-06-20 00:26:27 +03:00
return shared_ptr < CBattleQuery > ( ) ;
2013-04-20 14:34:01 +03:00
} ;
2013-06-17 18:45:55 +03:00
auto battleQuery = findBattleQuery ( ) ;
2013-04-20 14:34:01 +03:00
if ( ! battleQuery )
2013-06-17 18:45:55 +03:00
{
2013-04-20 14:34:01 +03:00
logGlobal - > errorStream ( ) < < " Cannot find battle query! " ;
2013-06-17 18:45:55 +03:00
if ( gs - > initialOpts - > mode = = StartInfo : : DUEL )
{
battleQuery = make_shared < CBattleQuery > ( gs - > curB ) ;
}
}
2013-07-22 01:01:29 +03:00
if ( battleQuery ! = queries . topQuery ( gs - > curB - > sides [ 0 ] . color ) )
complain ( " Player " + boost : : lexical_cast < std : : string > ( gs - > curB - > sides [ 0 ] . color ) + " although in battle has no battle query at the top! " ) ;
2013-06-17 18:45:55 +03:00
2013-04-20 14:34:01 +03:00
battleQuery - > result = * battleResult . data ;
//Check how many battle queries were created (number of players blocked by battle)
2013-06-17 18:45:55 +03:00
const int queriedPlayers = battleQuery ? boost : : count ( queries . allQueries ( ) , battleQuery ) : 0 ;
2013-04-20 14:34:01 +03:00
finishingBattle = make_unique < FinishingBattleHelper > ( battleQuery , gs - > initialOpts - > mode = = StartInfo : : DUEL , queriedPlayers ) ;
2010-12-12 01:11:26 +02:00
CasualtiesAfterBattle cab1 ( bEndArmy1 , gs - > curB ) , cab2 ( bEndArmy2 , gs - > curB ) ; //calculate casualties before deleting battle
2013-06-21 23:59:32 +03:00
if ( finishingBattle - > duel )
{
duelFinished ( ) ;
sendAndApply ( battleResult . data ) ; //after this point casualties objects are destroyed
return ;
}
2011-02-14 00:46:04 +02:00
ChangeSpells cs ; //for Eagle Eye
2011-02-12 18:12:48 +02:00
2013-04-20 14:34:01 +03:00
if ( finishingBattle - > winnerHero )
2011-02-14 00:46:04 +02:00
{
2013-04-20 14:34:01 +03:00
if ( int eagleEyeLevel = finishingBattle - > winnerHero - > getSecSkillLevel ( SecondarySkill : : EAGLE_EYE ) )
2011-02-14 00:46:04 +02:00
{
int maxLevel = eagleEyeLevel + 1 ;
2013-04-20 14:34:01 +03:00
double eagleEyeChance = finishingBattle - > winnerHero - > valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : EAGLE_EYE ) ;
2013-07-22 01:01:29 +03:00
for ( const CSpell * sp : gs - > curB - > sides [ ! battleResult . data - > winner ] . usedSpellsHistory )
2013-04-20 14:34:01 +03:00
if ( sp - > level < = maxLevel & & ! vstd : : contains ( finishingBattle - > winnerHero - > spells , sp - > id ) & & rand ( ) % 100 < eagleEyeChance )
2011-02-14 00:46:04 +02:00
cs . spells . insert ( sp - > id ) ;
}
}
2011-02-12 18:12:48 +02:00
2013-04-20 14:34:01 +03:00
2012-05-02 11:37:11 +03:00
std : : vector < ui32 > arts ; //display them in window
2012-05-01 17:38:53 +03:00
2013-04-20 14:34:01 +03:00
if ( result = = BattleResult : : NORMAL & & finishingBattle - > winnerHero )
2012-05-01 17:38:53 +03:00
{
2013-04-20 14:34:01 +03:00
if ( finishingBattle - > loserHero )
2012-05-01 17:38:53 +03:00
{
2013-04-20 14:34:01 +03:00
auto artifactsWorn = finishingBattle - > loserHero - > artifactsWorn ; //TODO: wrap it into a function, somehow (boost::variant -_-)
2013-06-29 16:05:48 +03:00
for ( auto artSlot : artifactsWorn )
2012-05-01 17:38:53 +03:00
{
2012-05-02 11:37:11 +03:00
MoveArtifact ma ;
2013-04-20 14:34:01 +03:00
ma . src = ArtifactLocation ( finishingBattle - > loserHero , artSlot . first ) ;
2012-05-01 17:38:53 +03:00
const CArtifactInstance * art = ma . src . getArt ( ) ;
2012-09-28 14:43:19 +03:00
if ( art & & ! art - > artType - > isBig ( ) & & art - > artType - > id ! = 0 ) // don't move war machines or locked arts (spellbook)
2012-05-01 17:38:53 +03:00
{
2012-05-02 11:37:11 +03:00
arts . push_back ( art - > artType - > id ) ;
2013-04-20 14:34:01 +03:00
ma . dst = ArtifactLocation ( finishingBattle - > winnerHero , art - > firstAvailableSlot ( finishingBattle - > winnerHero ) ) ;
2012-05-01 17:38:53 +03:00
sendAndApply ( & ma ) ;
}
}
2013-04-20 14:34:01 +03:00
while ( ! finishingBattle - > loserHero - > artifactsInBackpack . empty ( ) )
2012-05-01 17:38:53 +03:00
{
2012-05-02 11:37:11 +03:00
//we assume that no big artifacts can be found
2012-05-01 17:38:53 +03:00
MoveArtifact ma ;
2013-04-20 14:34:01 +03:00
ma . src = ArtifactLocation ( finishingBattle - > loserHero ,
2013-02-12 22:49:40 +03:00
ArtifactPosition ( GameConstants : : BACKPACK_START ) ) ; //backpack automatically shifts arts to beginning
2012-05-01 17:38:53 +03:00
const CArtifactInstance * art = ma . src . getArt ( ) ;
2012-05-02 11:37:11 +03:00
arts . push_back ( art - > artType - > id ) ;
2013-04-20 14:34:01 +03:00
ma . dst = ArtifactLocation ( finishingBattle - > winnerHero , art - > firstAvailableSlot ( finishingBattle - > winnerHero ) ) ;
2012-05-01 17:38:53 +03:00
sendAndApply ( & ma ) ;
}
2013-04-20 14:34:01 +03:00
if ( finishingBattle - > loserHero - > commander ) //TODO: what if commanders belong to no hero?
2012-05-02 11:37:11 +03:00
{
2013-04-20 14:34:01 +03:00
artifactsWorn = finishingBattle - > loserHero - > commander - > artifactsWorn ;
2013-06-29 16:05:48 +03:00
for ( auto artSlot : artifactsWorn )
2012-05-02 11:37:11 +03:00
{
MoveArtifact ma ;
2013-04-20 14:34:01 +03:00
ma . src = ArtifactLocation ( finishingBattle - > loserHero - > commander . get ( ) , artSlot . first ) ;
2012-05-02 11:37:11 +03:00
const CArtifactInstance * art = ma . src . getArt ( ) ;
if ( art & & ! art - > artType - > isBig ( ) )
{
arts . push_back ( art - > artType - > id ) ;
2013-04-20 14:34:01 +03:00
ma . dst = ArtifactLocation ( finishingBattle - > winnerHero , art - > firstAvailableSlot ( finishingBattle - > winnerHero ) ) ;
2012-05-02 11:37:11 +03:00
sendAndApply ( & ma ) ;
}
}
}
}
2013-07-22 01:01:29 +03:00
for ( auto armySlot : gs - > curB - > battleGetArmyObject ( ! battleResult . data - > winner ) - > stacks )
2012-05-02 11:37:11 +03:00
{
auto artifactsWorn = armySlot . second - > artifactsWorn ;
2013-06-29 16:05:48 +03:00
for ( auto artSlot : artifactsWorn )
2012-05-02 11:37:11 +03:00
{
MoveArtifact ma ;
ma . src = ArtifactLocation ( armySlot . second , artSlot . first ) ;
const CArtifactInstance * art = ma . src . getArt ( ) ;
if ( art & & ! art - > artType - > isBig ( ) )
{
arts . push_back ( art - > artType - > id ) ;
2013-04-20 14:34:01 +03:00
ma . dst = ArtifactLocation ( finishingBattle - > winnerHero , art - > firstAvailableSlot ( finishingBattle - > winnerHero ) ) ;
2012-05-02 11:37:11 +03:00
sendAndApply ( & ma ) ;
}
}
}
2012-05-01 17:38:53 +03:00
}
sendAndApply ( battleResult . data ) ; //after this point casualties objects are destroyed
2010-12-12 01:11:26 +02:00
2012-05-02 11:37:11 +03:00
if ( arts . size ( ) ) //display loot
{
InfoWindow iw ;
2013-04-20 14:34:01 +03:00
iw . player = finishingBattle - > winnerHero - > tempOwner ;
2012-05-02 11:37:11 +03:00
iw . text . addTxt ( MetaString : : GENERAL_TXT , 30 ) ; //You have captured enemy artifact
2013-06-29 16:05:48 +03:00
for ( auto id : arts ) //TODO; separate function to display loot for various ojects?
2012-05-02 11:37:11 +03:00
{
iw . components . push_back ( Component ( Component : : ARTIFACT , id , 0 , 0 ) ) ;
if ( iw . components . size ( ) > = 14 )
{
sendAndApply ( & iw ) ;
iw . components . clear ( ) ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 30 ) ; //repeat
}
}
if ( iw . components . size ( ) )
{
sendAndApply ( & iw ) ;
}
}
2011-02-11 21:12:08 +02:00
//Eagle Eye secondary skill handling
2011-02-14 11:29:35 +02:00
if ( cs . spells . size ( ) )
2011-02-11 21:12:08 +02:00
{
2011-02-14 00:46:04 +02:00
cs . learn = 1 ;
2013-04-20 14:34:01 +03:00
cs . hid = finishingBattle - > winnerHero - > id ;
2011-02-14 00:46:04 +02:00
InfoWindow iw ;
2013-04-20 14:34:01 +03:00
iw . player = finishingBattle - > winnerHero - > tempOwner ;
2011-02-14 00:46:04 +02:00
iw . text . addTxt ( MetaString : : GENERAL_TXT , 221 ) ; //Through eagle-eyed observation, %s is able to learn %s
2013-04-20 14:34:01 +03:00
iw . text . addReplacement ( finishingBattle - > winnerHero - > name ) ;
2011-02-14 00:46:04 +02:00
std : : ostringstream names ;
for ( int i = 0 ; i < cs . spells . size ( ) ; i + + )
2011-02-11 21:12:08 +02:00
{
2011-02-14 00:46:04 +02:00
names < < " %s " ;
if ( i < cs . spells . size ( ) - 2 )
names < < " , " ;
else if ( i < cs . spells . size ( ) - 1 )
names < < " %s " ;
}
2011-02-14 11:29:35 +02:00
names < < " . " ;
2011-02-11 21:12:08 +02:00
2011-02-14 00:46:04 +02:00
iw . text . addReplacement ( names . str ( ) ) ;
2011-02-11 21:12:08 +02:00
2013-02-11 02:24:57 +03:00
auto it = cs . spells . begin ( ) ;
2011-02-14 00:46:04 +02:00
for ( int i = 0 ; i < cs . spells . size ( ) ; i + + , it + + )
{
2013-02-11 02:24:57 +03:00
iw . text . addReplacement ( MetaString : : SPELL_NAME , it - > toEnum ( ) ) ;
2011-02-14 00:46:04 +02:00
if ( i = = cs . spells . size ( ) - 2 ) //we just added pre-last name
iw . text . addReplacement ( MetaString : : GENERAL_TXT , 141 ) ; // " and "
iw . components . push_back ( Component ( Component : : SPELL , * it , 0 , 0 ) ) ;
2011-02-11 21:12:08 +02:00
}
2011-02-14 00:46:04 +02:00
sendAndApply ( & iw ) ;
sendAndApply ( & cs ) ;
2011-02-14 11:29:35 +02:00
}
2013-04-20 14:34:01 +03:00
cab1 . takeFromArmy ( this ) ; cab2 . takeFromArmy ( this ) ; //take casualties after battle is deleted
2011-02-11 21:12:08 +02:00
2013-04-20 14:34:01 +03:00
//if one hero has lost we will erase him
if ( battleResult . data - > winner ! = 0 & & hero1 )
2008-09-20 21:30:37 +03:00
{
2013-04-20 14:34:01 +03:00
RemoveObject ro ( hero1 - > id ) ;
sendAndApply ( & ro ) ;
}
if ( battleResult . data - > winner ! = 1 & & hero2 )
{
RemoveObject ro ( hero2 - > id ) ;
sendAndApply ( & ro ) ;
}
2011-01-08 21:38:42 +02:00
2013-04-20 14:34:01 +03:00
//give exp
2013-08-18 20:49:31 +03:00
if ( battleResult . data - > exp [ 0 ] & & hero1 & & battleResult . get ( ) - > winner = = 0 )
2013-04-20 14:34:01 +03:00
changePrimSkill ( hero1 , PrimarySkill : : EXPERIENCE , battleResult . data - > exp [ 0 ] ) ;
2013-08-18 20:49:31 +03:00
else if ( battleResult . data - > exp [ 1 ] & & hero2 & & battleResult . get ( ) - > winner = = 1 )
2013-04-20 14:34:01 +03:00
changePrimSkill ( hero2 , PrimarySkill : : EXPERIENCE , battleResult . data - > exp [ 1 ] ) ;
2008-09-20 21:30:37 +03:00
2013-04-20 14:34:01 +03:00
queries . popIfTop ( battleQuery ) ;
//--> continuation (battleAfterLevelUp) occurs after level-up queries are handled or on removing query (above)
}
void CGameHandler : : battleAfterLevelUp ( const BattleResult & result )
{
LOG_TRACE ( logGlobal ) ;
finishingBattle - > remainingBattleQueriesCount - - ;
logGlobal - > traceStream ( ) < < " Decremented queries count to " < < finishingBattle - > remainingBattleQueriesCount ;
if ( finishingBattle - > remainingBattleQueriesCount > 0 )
2013-07-22 01:01:29 +03:00
//Battle results will be handled when all battle queries are closed
2013-04-20 14:34:01 +03:00
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,
2013-07-22 01:01:29 +03:00
// units will be given after casualties are taken
2013-04-20 14:34:01 +03:00
const SlotID necroSlot = raisedStack . type ? finishingBattle - > winnerHero - > getSlotFor ( raisedStack . type ) : SlotID ( ) ;
2008-09-29 13:16:02 +03:00
2013-02-16 17:03:47 +03:00
if ( necroSlot ! = SlotID ( ) )
2011-09-04 04:44:50 +03:00
{
2013-04-20 14:34:01 +03:00
finishingBattle - > winnerHero - > showNecromancyDialog ( raisedStack ) ;
addToSlot ( StackLocation ( finishingBattle - > winnerHero , necroSlot ) , raisedStack . type , raisedStack . count ) ;
}
BattleResultsApplied resultsApplied ;
resultsApplied . player1 = finishingBattle - > victor ;
resultsApplied . player2 = finishingBattle - > loser ;
2010-08-18 16:42:46 +03:00
sendAndApply ( & resultsApplied ) ;
2013-04-20 14:34:01 +03:00
2012-08-26 12:07:48 +03:00
setBattle ( nullptr ) ;
2010-08-18 16:42:46 +03:00
2013-04-20 14:34:01 +03:00
if ( visitObjectAfterVictory & & result . winner = = 0 )
2010-05-27 00:59:58 +03:00
{
2013-04-20 14:34:01 +03:00
logGlobal - > traceStream ( ) < < " post-victory visit " ;
visitObjectOnTile ( * getTile ( finishingBattle - > winnerHero - > getPosition ( ) ) , finishingBattle - > winnerHero ) ;
2010-05-27 00:59:58 +03:00
}
2012-07-23 13:23:43 +03:00
visitObjectAfterVictory = false ;
2010-05-27 00:59:58 +03:00
2013-04-20 14:34:01 +03:00
winLoseHandle ( 1 < < finishingBattle - > loser . getNum ( ) | 1 < < finishingBattle - > victor . getNum ( ) ) ; //handle victory/loss of engaged players
2010-07-08 08:52:11 +03:00
2013-04-20 14:34:01 +03:00
if ( result . result = = BattleResult : : SURRENDER | | result . result = = BattleResult : : ESCAPE ) //loser has escaped or surrendered
2010-07-08 08:52:11 +03:00
{
SetAvailableHeroes sah ;
2013-04-20 14:34:01 +03:00
sah . player = finishingBattle - > loser ;
sah . hid [ 0 ] = finishingBattle - > loserHero - > subID ;
if ( result . result = = BattleResult : : ESCAPE ) //retreat
2010-07-08 08:52:11 +03:00
{
2011-01-28 04:11:58 +02:00
sah . army [ 0 ] . clear ( ) ;
2013-04-20 14:34:01 +03:00
sah . army [ 0 ] . setCreature ( SlotID ( 0 ) , finishingBattle - > loserHero - > type - > initialArmy [ 0 ] . creature , 1 ) ;
2010-07-08 08:52:11 +03:00
}
2013-04-20 14:34:01 +03:00
if ( const CGHeroInstance * another = getPlayer ( finishingBattle - > loser ) - > availableHeroes [ 1 ] )
2010-07-08 08:52:11 +03:00
sah . hid [ 1 ] = another - > subID ;
else
sah . hid [ 1 ] = - 1 ;
sendAndApply ( & sah ) ;
}
2011-07-17 15:53:06 +03:00
}
2013-04-20 14:34:01 +03:00
2011-07-05 09:07:36 +03:00
void CGameHandler : : prepareAttack ( BattleAttack & bat , const CStack * att , const CStack * def , int distance , int targetHex )
2008-10-18 14:41:24 +03:00
{
2009-03-28 02:38:48 +02:00
bat . bsa . clear ( ) ;
2008-10-18 14:41:24 +03:00
bat . stackAttacking = att - > ID ;
2013-07-22 01:01:29 +03:00
const int attackerLuck = att - > LuckVal ( ) ;
auto sideHeroBlocksLuck = [ ] ( const SideInBattle & side ) { return NBonus : : hasOfType ( side . hero , Bonus : : BLOCK_LUCK ) ; } ;
2010-05-20 14:06:40 +03:00
2013-07-22 01:01:29 +03:00
if ( ! vstd : : contains_if ( gs - > curB - > sides , sideHeroBlocksLuck ) )
2009-02-05 11:49:45 +02:00
{
2013-05-04 16:14:23 +03:00
if ( attackerLuck > 0 & & rand ( ) % 24 < attackerLuck )
{
bat . flags | = BattleAttack : : LUCKY ;
}
if ( VLC - > modh - > settings . data [ " hardcodedFeatures " ] [ " NEGATIVE_LUCK " ] . Bool ( ) ) // negative luck enabled
{
if ( attackerLuck < 0 & & rand ( ) % 24 < abs ( attackerLuck ) )
{
bat . flags | = BattleAttack : : UNLUCKY ;
}
}
2009-02-05 11:49:45 +02:00
}
2013-05-04 16:14:23 +03:00
2011-07-06 20:00:45 +03:00
if ( rand ( ) % 100 < att - > valOfBonuses ( Bonus : : DOUBLE_DAMAGE_CHANCE ) )
{
bat . flags | = BattleAttack : : DEATH_BLOW ;
}
2010-04-06 16:19:54 +03:00
2013-02-09 00:17:39 +03:00
if ( att - > getCreature ( ) - > idNumber = = CreatureID : : BALLISTA )
2011-02-24 17:33:03 +02:00
{
static const int artilleryLvlToChance [ ] = { 0 , 50 , 75 , 100 } ;
const CGHeroInstance * owner = gs - > curB - > getHero ( att - > owner ) ;
2013-02-04 22:43:16 +03:00
int chance = artilleryLvlToChance [ owner - > getSecSkillLevel ( SecondarySkill : : ARTILLERY ) ] ;
2011-02-24 17:33:03 +02:00
if ( chance > rand ( ) % 100 )
{
bat . flags | = BattleAttack : : BALLISTA_DOUBLE_DMG ;
}
}
2012-07-23 13:23:43 +03:00
// only primary target
2011-07-04 22:34:49 +03:00
applyBattleEffects ( bat , att , def , distance , false ) ;
2011-02-24 17:33:03 +02:00
2011-07-04 22:34:49 +03:00
if ( ! bat . shot ( ) ) //multiple-hex attack - only in meele
{
2012-11-30 17:06:22 +03:00
std : : set < const CStack * > attackedCreatures = gs - > curB - > getAttackedCreatures ( att , targetHex ) ; //creatures other than primary target
2011-07-04 22:34:49 +03:00
2013-06-29 16:05:48 +03:00
for ( const CStack * stack : attackedCreatures )
2011-07-04 22:34:49 +03:00
{
if ( stack ! = def ) //do not hit same stack twice
{
applyBattleEffects ( bat , att , stack , distance , true ) ;
}
}
}
2010-02-26 17:52:48 +02:00
2012-09-20 19:55:21 +03:00
const Bonus * bonus = att - > getBonusLocalFirst ( Selector : : type ( Bonus : : SPELL_LIKE_ATTACK ) ) ;
2011-07-04 22:34:49 +03:00
if ( bonus & & ( bat . shot ( ) ) ) //TODO: make it work in meele?
2010-05-07 15:29:41 +03:00
{
2011-07-04 22:34:49 +03:00
bat . bsa . front ( ) . flags | = BattleStackAttacked : : EFFECT ;
bat . bsa . front ( ) . effect = VLC - > spellh - > spells [ bonus - > subtype ] - > mainEffectAnim ; //hopefully it does not interfere with any other effect?
2011-07-02 20:40:33 +03:00
2013-02-13 22:35:43 +03:00
std : : set < const CStack * > attackedCreatures = gs - > curB - > getAffectedCreatures ( SpellID ( bonus - > subtype ) . toSpell ( ) , bonus - > val , att - > owner , targetHex ) ;
2011-07-03 08:55:57 +03:00
//TODO: get exact attacked hex for defender
2010-05-07 15:29:41 +03:00
2013-06-29 16:05:48 +03:00
for ( const CStack * stack : attackedCreatures )
2010-05-07 15:29:41 +03:00
{
2011-07-02 19:49:22 +03:00
if ( stack ! = def ) //do not hit same stack twice
{
2011-07-04 22:34:49 +03:00
applyBattleEffects ( bat , att , stack , distance , true ) ;
2011-07-02 19:49:22 +03:00
}
}
2010-02-26 17:52:48 +02:00
}
2011-07-04 22:34:49 +03: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 ;
2013-05-04 16:14:23 +03:00
bsa . damageAmount = gs - > curB - > calculateDmg ( att , def , gs - > curB - > battleGetOwner ( att ) , gs - > curB - > battleGetOwner ( def ) , bat . shot ( ) , distance , bat . lucky ( ) , bat . unlucky ( ) , bat . deathBlow ( ) , bat . ballistaDoubleDmg ( ) ) ;
2011-07-04 22:34:49 +03:00
def - > prepareAttacked ( bsa ) ; //calculate casualties
//life drain handling
if ( att - > hasBonusOfType ( Bonus : : LIFE_DRAIN ) & & def - > isLiving ( ) )
{
StacksHealedOrResurrected shi ;
shi . lifeDrain = ( ui8 ) true ;
shi . tentHealing = ( ui8 ) false ;
shi . drainedFrom = def - > ID ;
StacksHealedOrResurrected : : HealInfo hi ;
hi . stackID = att - > ID ;
2013-02-08 12:22:10 +03:00
hi . healedHP = std : : min < int > ( bsa . damageAmount * att - > valOfBonuses ( Bonus : : LIFE_DRAIN ) / 100 ,
att - > MaxHealth ( ) - att - > firstHPleft + att - > MaxHealth ( ) * ( att - > baseAmount - att - > count ) ) ;
2011-07-04 22:34:49 +03:00
hi . lowLevelResurrection = false ;
shi . healedStacks . push_back ( hi ) ;
if ( hi . healedHP > 0 )
2011-07-02 19:49:22 +03:00
{
2011-07-04 22:34:49 +03:00
bsa . healedStacks . push_back ( shi ) ;
2011-07-02 19:49:22 +03:00
}
2011-09-05 09:37:24 +03:00
}
bat . bsa . push_back ( bsa ) ; //add this stack to the list of victims after drain life has been calculated
2011-07-04 22:34:49 +03:00
//fire shield handling
if ( ! bat . shot ( ) & & def - > hasBonusOfType ( Bonus : : FIRE_SHIELD ) & & ! att - > hasBonusOfType ( Bonus : : FIRE_IMMUNITY ) & & ! bsa . killed ( ) )
{
2013-05-30 10:43:42 +03:00
BattleStackAttacked bsa2 ;
bsa2 . stackAttacked = att - > ID ; //invert
bsa2 . attackerID = def - > ID ;
bsa2 . flags | = BattleStackAttacked : : EFFECT ; //FIXME: play anmation upon efreet and not attacker
bsa2 . effect = 11 ;
2011-07-04 22:34:49 +03:00
2013-05-30 10:43:42 +03:00
bsa2 . damageAmount = ( bsa . damageAmount * def - > valOfBonuses ( Bonus : : FIRE_SHIELD ) ) / 100 ; //TODO: scale with attack/defense
att - > prepareAttacked ( bsa2 ) ;
bat . bsa . push_back ( bsa2 ) ;
2011-07-02 19:49:22 +03:00
}
2008-10-18 14:41:24 +03:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : handleConnection ( std : : set < PlayerColor > players , CConnection & c )
2008-07-25 20:28:28 +03:00
{
2012-06-27 23:44:01 +03:00
setThreadName ( " CGameHandler::handleConnection " ) ;
2013-06-26 14:18:27 +03:00
srand ( time ( nullptr ) ) ;
2012-03-26 01:46:14 +03:00
2008-07-27 20:07:37 +03:00
try
2008-07-25 20:28:28 +03:00
{
2010-01-29 22:52:45 +02:00
while ( 1 ) //server should never shut connection first //was: while(!end2)
2008-07-25 20:28:28 +03:00
{
2013-06-26 14:18:27 +03:00
CPack * pack = nullptr ;
2013-03-03 20:06:03 +03:00
PlayerColor player = PlayerColor : : NEUTRAL ;
2012-03-26 01:46:14 +03:00
si32 requestID = - 999 ;
int packType = 0 ;
2009-04-16 03:28:54 +03:00
{
boost : : unique_lock < boost : : mutex > lock ( * c . rmx ) ;
2012-03-26 01:46:14 +03:00
c > > player > > requestID > > pack ; //get the package
2013-06-17 18:45:55 +03:00
if ( ! pack )
{
logGlobal - > errorStream ( ) < < boost : : format ( " Received a null package marked as request %d from player %d " ) % requestID % player ;
}
2012-03-26 01:46:14 +03:00
packType = typeList . getTypeID ( pack ) ; //get the id of type
2012-07-23 13:23:43 +03:00
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < boost : : format ( " Received client message (request %d by player %d) of type with ID=%d (%s). \n " )
2013-03-03 20:06:03 +03:00
% requestID % player . getNum ( ) % packType % typeid ( * pack ) . name ( ) ;
2009-04-16 03:28:54 +03:00
}
2012-03-26 01:46:14 +03:00
//prepare struct informing that action was applied
2013-04-20 14:34:01 +03:00
auto sendPackageResponse = [ & ] ( bool succesfullyApplied )
{
PackageApplied applied ;
applied . player = player ;
applied . result = succesfullyApplied ;
applied . packType = packType ;
applied . requestID = requestID ;
boost : : unique_lock < boost : : mutex > lock ( * c . wmx ) ;
c < < & applied ;
} ;
2012-03-26 01:46:14 +03:00
2009-04-16 03:28:54 +03:00
CBaseForGHApply * apply = applier - > apps [ packType ] ; //and appropriae applier object
2013-04-20 14:34:01 +03:00
if ( isBlockedByQueries ( pack , player ) )
2011-09-06 09:00:32 +03:00
{
2013-04-20 14:34:01 +03:00
sendPackageResponse ( false ) ;
2011-09-06 09:00:32 +03:00
}
else if ( apply )
2008-07-27 20:07:37 +03:00
{
2013-04-20 14:34:01 +03:00
const bool result = apply - > applyOnGH ( this , & c , pack , player ) ;
if ( ! result )
complain ( " Got false in applying... that request must have been fishy! " ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Message successfully applied (result= " < < result < < " )! " ;
2013-04-20 14:34:01 +03:00
sendPackageResponse ( true ) ;
2008-07-27 20:07:37 +03:00
}
2009-03-09 12:37:49 +02:00
else
{
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Message cannot be applied, cannot find applier (unregistered type)! " ;
2013-04-20 14:34:01 +03:00
sendPackageResponse ( false ) ;
2009-03-09 12:37:49 +02:00
}
2012-03-26 01:46:14 +03:00
vstd : : clear_pointer ( pack ) ;
2008-07-25 20:28:28 +03:00
}
}
2009-09-20 15:47:40 +03:00
catch ( boost : : system : : system_error & e ) //for boost errors just log, not crash - probably client shut down connection
{
2010-01-29 22:52:45 +02:00
assert ( ! c . connected ) ; //make sure that connection has been marked as broken
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < e . what ( ) ;
2009-09-20 15:47:40 +03:00
end2 = true ;
}
2012-07-23 13:23:43 +03:00
HANDLE_EXCEPTION ( end2 = true ) ;
2010-01-29 22:52:45 +02:00
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Ended handling connection " ;
2008-07-25 20:28:28 +03:00
}
2009-08-22 18:29:30 +03:00
2011-12-22 16:05:19 +03:00
int CGameHandler : : moveStack ( int stack , BattleHex dest )
2009-08-22 18:29:30 +03:00
{
int ret = 0 ;
2012-08-26 12:07:48 +03:00
const CStack * curStack = gs - > curB - > battleGetStackByID ( stack ) ,
* stackAtEnd = gs - > curB - > battleGetStackByPos ( dest ) ;
2008-08-09 02:02:32 +03:00
2009-09-20 15:47:40 +03:00
assert ( curStack ) ;
2011-12-14 00:23:17 +03:00
assert ( dest < GameConstants : : BFIELD_SIZE ) ;
2009-09-20 15:47:40 +03:00
2011-02-12 18:12:48 +02:00
if ( gs - > curB - > tacticDistance )
{
assert ( gs - > curB - > isInTacticRange ( dest ) ) ;
}
2012-08-26 22:13:57 +03:00
if ( curStack - > position = = dest )
return 0 ;
2008-08-09 02:02:32 +03:00
//initing necessary tables
2012-08-28 15:28:13 +03:00
auto accessibility = getAccesibility ( curStack ) ;
2008-08-09 02:02:32 +03:00
2009-02-05 16:44:27 +02:00
//shifting destination (if we have double wide stack and we can occupy dest but not be exactly there)
2012-08-26 12:07:48 +03:00
if ( ! stackAtEnd & & curStack - > doubleWide ( ) & & ! accessibility . accessible ( dest , curStack ) )
2009-02-05 16:44:27 +02:00
{
if ( curStack - > attackerOwned )
{
2012-08-26 12:07:48 +03:00
if ( accessibility . accessible ( dest + 1 , curStack ) )
2011-12-22 16:05:19 +03:00
dest + = BattleHex : : RIGHT ;
2009-02-05 16:44:27 +02:00
}
else
{
2012-08-26 12:07:48 +03:00
if ( accessibility . accessible ( dest - 1 , curStack ) )
2011-12-22 16:05:19 +03:00
dest + = BattleHex : : LEFT ;
2009-02-05 16:44:27 +02:00
}
}
2012-08-26 12:07:48 +03:00
if ( ( stackAtEnd & & stackAtEnd ! = curStack & & stackAtEnd - > alive ( ) ) | | ! accessibility . accessible ( dest , curStack ) )
2009-08-03 17:29:29 +03:00
{
2012-08-26 12:07:48 +03:00
complain ( " Given destination is not accessible! " ) ;
return 0 ;
2009-08-03 17:29:29 +03:00
}
2013-01-12 21:08:33 +03:00
2012-08-26 12:07:48 +03:00
std : : pair < std : : vector < BattleHex > , int > path = gs - > curB - > getPath ( curStack - > position , dest , curStack ) ;
2009-08-22 18:29:30 +03:00
ret = path . second ;
2011-12-14 00:23:17 +03:00
int creSpeed = gs - > curB - > tacticDistance ? GameConstants : : BFIELD_SIZE : curStack - > Speed ( ) ;
2011-02-12 18:12:48 +02:00
2010-05-02 21:20:26 +03:00
if ( curStack - > hasBonusOfType ( Bonus : : FLYING ) )
2008-08-09 02:02:32 +03:00
{
2011-02-12 18:12:48 +02:00
if ( path . second < = creSpeed & & path . first . size ( ) > 0 )
2009-03-07 18:05:53 +02:00
{
//inform clients about move
BattleStackMoved sm ;
sm . stack = curStack - > ID ;
2011-12-22 16:05:19 +03:00
std : : vector < BattleHex > tiles ;
2011-08-01 20:36:18 +03:00
tiles . push_back ( path . first [ 0 ] ) ;
sm . tilesToMove = tiles ;
2009-03-07 18:05:53 +02:00
sm . distance = path . second ;
2010-05-07 17:05:48 +03:00
sm . teleporting = false ;
2009-03-07 18:05:53 +02:00
sendAndApply ( & sm ) ;
}
}
else //for non-flying creatures
{
2011-08-01 20:36:18 +03:00
// send one package with the creature path information
2012-05-05 00:16:39 +03:00
2012-06-23 20:19:50 +03:00
shared_ptr < const CObstacleInstance > obstacle ; //obstacle that interrupted movement
2011-12-22 16:05:19 +03:00
std : : vector < BattleHex > tiles ;
2011-02-12 18:12:48 +02:00
int tilesToMove = std : : max ( ( int ) ( path . first . size ( ) - creSpeed ) , 0 ) ;
2012-05-05 00:16:39 +03:00
int v = path . first . size ( ) - 1 ;
startWalking :
for ( ; v > = tilesToMove ; - - v )
2009-03-07 18:05:53 +02:00
{
2012-05-05 00:16:39 +03:00
BattleHex hex = path . first [ v ] ;
2012-07-23 13:23:43 +03:00
tiles . push_back ( hex ) ;
2012-05-05 00:16:39 +03:00
2012-05-27 22:06:35 +03:00
if ( ( obstacle = battleGetObstacleOnPos ( hex , false ) ) )
2012-05-05 00:16:39 +03:00
{
2012-05-18 23:50:16 +03:00
//we walked onto something, so we finalize this portion of stack movement check into obstacle
2012-07-23 13:23:43 +03:00
break ;
2012-05-05 00:16:39 +03:00
}
2011-08-01 20:36:18 +03:00
}
2012-07-23 13:23:43 +03:00
2011-08-01 20:36:18 +03:00
if ( tiles . size ( ) > 0 )
{
2012-05-18 23:50:16 +03:00
//commit movement
2009-03-07 18:05:53 +02:00
BattleStackMoved sm ;
sm . stack = curStack - > ID ;
sm . distance = path . second ;
2010-05-07 17:05:48 +03:00
sm . teleporting = false ;
2011-08-01 20:36:18 +03:00
sm . tilesToMove = tiles ;
2009-03-07 18:05:53 +02:00
sendAndApply ( & sm ) ;
}
2012-05-05 00:16:39 +03:00
2012-05-18 23:50:16 +03:00
//we don't handle obstacle at the destination tile -> it's handled separately in the if at the end
if ( obstacle & & curStack - > position ! = dest )
2012-05-05 00:16:39 +03:00
{
2012-05-18 23:50:16 +03:00
handleDamageFromObstacle ( * obstacle , curStack ) ;
2012-05-05 00:16:39 +03:00
//if stack didn't die in explosion, continue movement
2012-05-18 23:50:16 +03:00
if ( ! obstacle - > stopsMovement ( ) & & curStack - > alive ( ) )
2012-05-05 00:16:39 +03:00
{
2012-06-23 20:19:50 +03:00
obstacle . reset ( ) ;
2012-05-05 00:16:39 +03:00
tiles . clear ( ) ;
2012-05-18 23:50:16 +03:00
v - - ;
2012-05-05 00:16:39 +03:00
goto startWalking ; //TODO it's so evil
}
}
2008-08-09 02:02:32 +03:00
}
2012-05-18 23:50:16 +03:00
//handling obstacle on the final field (separate, because it affects both flying and walking stacks)
if ( curStack - > alive ( ) )
{
if ( auto theLastObstacle = battleGetObstacleOnPos ( curStack - > position , false ) )
{
handleDamageFromObstacle ( * theLastObstacle , curStack ) ;
}
}
2009-08-22 18:29:30 +03:00
return ret ;
2008-08-09 02:02:32 +03:00
}
2010-11-27 22:17:28 +02:00
2008-07-01 11:01:02 +03:00
CGameHandler : : CGameHandler ( void )
{
2009-04-12 03:58:41 +03:00
QID = 1 ;
2013-06-26 14:18:27 +03:00
//gs = nullptr;
2009-01-12 22:05:56 +02:00
IObjectInterface : : cb = this ;
2010-09-03 21:42:54 +03:00
applier = new CApplier < CBaseForGHApply > ;
registerTypes3 ( * applier ) ;
2012-07-23 13:23:43 +03:00
visitObjectAfterVictory = false ;
2013-04-20 14:34:01 +03:00
queries . gh = this ;
2008-07-01 11:01:02 +03:00
}
CGameHandler : : ~ CGameHandler ( void )
{
2009-03-09 12:37:49 +02:00
delete applier ;
2013-06-26 14:18:27 +03:00
applier = nullptr ;
2008-07-25 20:28:28 +03:00
delete gs ;
}
2010-01-29 18:19:12 +02:00
2012-03-27 23:08:54 +03:00
void CGameHandler : : init ( StartInfo * si )
2008-07-25 20:28:28 +03:00
{
2013-06-26 14:18:27 +03:00
//extern DLL_LINKAGE std::minstd_rand ran;
2012-04-14 05:20:22 +03:00
if ( ! si - > seedToBeUsed )
2013-06-26 14:18:27 +03:00
si - > seedToBeUsed = std : : time ( nullptr ) ;
2012-03-27 23:08:54 +03:00
2008-07-25 20:28:28 +03:00
gs = new CGameState ( ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < " Gamestate created! " ;
2012-04-14 05:20:22 +03:00
gs - > init ( si ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < " Gamestate initialized! " ;
2008-12-22 19:48:41 +02:00
2013-06-29 16:05:48 +03:00
for ( auto & elem : gs - > players )
states . addPlayer ( elem . first ) ;
2008-07-01 11:01:02 +03:00
}
2008-09-28 16:29:37 +03:00
2013-02-19 01:37:22 +03:00
static bool evntCmp ( const CMapEvent & a , const CMapEvent & b )
2009-03-09 21:40:43 +02:00
{
2013-02-19 01:37:22 +03:00
return a . earlierThan ( b ) ;
2009-03-09 21:40:43 +02:00
}
2010-08-24 17:26:57 +03:00
void CGameHandler : : setPortalDwelling ( const CGTownInstance * town , bool forced = false , bool clear = false )
2010-07-10 19:50:23 +03:00
{ // bool forced = true - if creature should be replaced, if false - only if no creature was set
2011-05-30 02:49:25 +03:00
const PlayerState * p = gs - > getPlayer ( town - > tempOwner ) ;
if ( ! p )
{
2013-04-10 20:18:01 +03:00
logGlobal - > warnStream ( ) < < " There is no player owner of town " < < town - > name < < " at " < < town - > pos ;
2011-05-30 02:49:25 +03:00
return ;
}
2010-07-10 19:50:23 +03:00
2011-12-14 00:23:17 +03:00
if ( forced | | town - > creatures [ GameConstants : : CREATURES_PER_TOWN ] . second . empty ( ) ) //we need to change creature
2010-07-10 19:50:23 +03:00
{
SetAvailableCreatures ssi ;
ssi . tid = town - > id ;
ssi . creatures = town - > creatures ;
2011-12-14 00:23:17 +03:00
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . second . clear ( ) ; //remove old one
2012-07-23 13:23:43 +03:00
2011-05-30 02:49:25 +03:00
const std : : vector < ConstTransitivePtr < CGDwelling > > & dwellings = p - > dwellings ;
2010-07-10 19:50:23 +03:00
if ( dwellings . empty ( ) ) //no dwellings - just remove
{
sendAndApply ( & ssi ) ;
return ;
}
2012-07-23 13:23:43 +03:00
2010-07-10 19:50:23 +03:00
ui32 dwellpos = rand ( ) % dwellings . size ( ) ; //take random dwelling
ui32 creapos = rand ( ) % dwellings [ dwellpos ] - > creatures . size ( ) ; //for multi-creature dwellings like Golem Factory
2013-02-11 02:24:57 +03:00
CreatureID creature = dwellings [ dwellpos ] - > creatures [ creapos ] . second [ 0 ] ;
2012-07-23 13:23:43 +03:00
2010-08-24 17:26:57 +03:00
if ( clear )
2011-12-14 00:23:17 +03:00
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . first = std : : max ( ( ui32 ) 1 , ( VLC - > creh - > creatures [ creature ] - > growth ) / 2 ) ;
2010-08-24 17:26:57 +03:00
else
2011-12-14 00:23:17 +03:00
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . first = VLC - > creh - > creatures [ creature ] - > growth ;
ssi . creatures [ GameConstants : : CREATURES_PER_TOWN ] . second . push_back ( creature ) ;
2010-07-10 19:50:23 +03:00
sendAndApply ( & ssi ) ;
}
}
2008-07-25 20:28:28 +03:00
void CGameHandler : : newTurn ( )
{
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Turn " < < gs - > day + 1 ;
2008-07-26 16:57:32 +03:00
NewTurn n ;
2011-08-26 23:32:05 +03:00
n . specialWeek = NewTurn : : NO_ACTION ;
2013-02-07 20:34:50 +03:00
n . creatureid = CreatureID : : NONE ;
2008-07-26 16:57:32 +03:00
n . day = gs - > day + 1 ;
2011-08-26 23:32:05 +03:00
2013-02-02 11:29:57 +03:00
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 ;
2012-07-23 13:23:43 +03:00
2013-03-03 20:06:03 +03:00
std : : map < PlayerColor , si32 > hadGold ; //starting gold - for buildings like dwarven treasury
2013-06-26 14:18:27 +03:00
srand ( time ( nullptr ) ) ;
2008-07-26 16:57:32 +03:00
2013-02-04 15:32:53 +03:00
if ( firstTurn )
{
2013-06-29 16:05:48 +03:00
for ( auto obj : gs - > map - > objects )
2013-02-04 15:32:53 +03:00
{
2013-02-06 02:16:13 +03:00
if ( obj & & obj - > ID = = Obj : : PRISON ) //give imprisoned hero 0 exp to level him up. easiest to do at this point
2013-02-04 15:32:53 +03:00
{
2013-02-09 00:17:39 +03:00
changePrimSkill ( getHero ( obj - > id ) , PrimarySkill : : EXPERIENCE , 0 ) ;
2013-02-04 15:32:53 +03:00
}
}
}
2011-08-26 23:32:05 +03:00
if ( newWeek & & ! firstTurn )
2010-08-24 17:26:57 +03:00
{
2011-08-26 23:32:05 +03:00
n . specialWeek = NewTurn : : NORMAL ;
bool deityOfFireBuilt = false ;
2013-06-29 16:05:48 +03:00
for ( const CGTownInstance * t : gs - > map - > towns )
2010-08-24 17:26:57 +03:00
{
2013-02-11 02:24:57 +03:00
if ( t - > hasBuilt ( BuildingID : : GRAIL , ETownType : : INFERNO ) )
2010-08-24 17:26:57 +03:00
{
2011-08-26 23:32:05 +03:00
deityOfFireBuilt = true ;
break ;
2010-08-24 17:26:57 +03:00
}
}
2011-08-26 23:32:05 +03:00
if ( deityOfFireBuilt )
2010-08-24 17:26:57 +03:00
{
2011-08-26 23:32:05 +03:00
n . specialWeek = NewTurn : : DEITYOFFIRE ;
2013-02-07 20:34:50 +03:00
n . creatureid = CreatureID : : IMP ;
2011-08-26 23:32:05 +03:00
}
else
{
int monthType = rand ( ) % 100 ;
if ( newMonth ) //new month
2010-08-24 17:26:57 +03:00
{
2011-08-26 23:32:05 +03:00
if ( monthType < 40 ) //double growth
{
n . specialWeek = NewTurn : : DOUBLE_GROWTH ;
2012-08-11 12:06:23 +03:00
if ( VLC - > modh - > settings . ALL_CREATURES_GET_DOUBLE_MONTHS )
2011-08-26 23:32:05 +03:00
{
2013-07-02 15:08:30 +03:00
std : : pair < int , CreatureID > newMonster ( 54 , VLC - > creh - > pickRandomMonster ( [ ] { return rand ( ) ; } ) ) ;
2011-08-26 23:32:05 +03:00
n . creatureid = newMonster . second ;
}
2013-02-27 06:11:45 +03:00
else if ( VLC - > creh - > doubledCreatures . size ( ) )
2011-08-26 23:32:05 +03:00
{
2013-04-23 20:29:16 +03:00
const std : : vector < CreatureID > doubledCreatures ( VLC - > creh - > doubledCreatures . begin ( ) , VLC - > creh - > doubledCreatures . end ( ) ) ;
2013-07-02 15:08:30 +03:00
n . creatureid = vstd : : pickRandomElementOf ( doubledCreatures , [ ] { return rand ( ) ; } ) ;
2011-08-26 23:32:05 +03:00
}
2013-02-27 06:11:45 +03:00
else
{
complain ( " Cannot find creature that can be spawned! " ) ;
n . specialWeek = NewTurn : : NORMAL ;
}
2011-08-26 23:32:05 +03:00
}
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
2013-07-02 15:08:30 +03:00
std : : pair < int , CreatureID > newMonster ( 54 , VLC - > creh - > pickRandomMonster ( [ ] { return rand ( ) ; } ) ) ;
2011-08-26 23:32:05 +03:00
//TODO do not pick neutrals
n . creatureid = newMonster . second ;
}
2010-08-24 17:26:57 +03:00
}
}
}
2010-12-20 15:04:24 +02:00
bmap < ui32 , ConstTransitivePtr < CGHeroInstance > > pool = gs - > hpool . heroesPool ;
2009-08-05 03:05:37 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & elem : gs - > players )
2008-07-25 20:28:28 +03:00
{
2013-06-29 16:05:48 +03:00
if ( elem . first = = PlayerColor : : NEUTRAL )
2010-07-08 08:52:11 +03:00
continue ;
2013-06-29 16:05:48 +03:00
else if ( elem . first > = PlayerColor : : PLAYER_LIMIT )
2010-07-08 08:52:11 +03:00
assert ( 0 ) ; //illegal player number!
2009-11-13 21:04:36 +02:00
2013-06-29 16:05:48 +03:00
std : : pair < PlayerColor , si32 > playerGold ( elem . first , elem . second . resources [ Res : : GOLD ] ) ;
2012-07-23 13:23:43 +03:00
hadGold . insert ( playerGold ) ;
2010-01-25 23:25:14 +02:00
2011-08-26 23:32:05 +03:00
if ( newWeek ) //new heroes in tavern
2008-10-26 22:58:34 +02:00
{
SetAvailableHeroes sah ;
2013-06-29 16:05:48 +03:00
sah . player = elem . first ;
2010-07-08 08:52:11 +03:00
//pick heroes and their armies
2013-06-26 14:18:27 +03:00
CHeroClass * banned = nullptr ;
2011-12-14 00:23:17 +03:00
for ( int j = 0 ; j < GameConstants : : AVAILABLE_HEROES_PER_PLAYER ; j + + )
2010-07-08 08:52:11 +03:00
{
2013-06-29 16:05:48 +03:00
if ( CGHeroInstance * h = gs - > hpool . pickHeroFor ( j = = 0 , elem . first , getNativeTown ( elem . first ) , pool , banned ) ) //first hero - native if possible, second hero -> any other class
2010-07-08 08:52:11 +03:00
{
sah . hid [ j ] = h - > subID ;
2011-01-28 04:11:58 +02:00
h - > initArmy ( & sah . army [ j ] ) ;
2010-07-08 08:52:11 +03:00
banned = h - > type - > heroClass ;
}
else
sah . hid [ j ] = - 1 ;
}
2008-10-26 22:58:34 +02:00
sendAndApply ( & sah ) ;
}
2009-11-28 03:42:08 +02:00
2013-06-29 16:05:48 +03:00
n . res [ elem . first ] = elem . second . resources ;
2011-07-05 09:14:07 +03:00
2013-06-29 16:05:48 +03:00
for ( CGHeroInstance * h : ( elem ) . second . heroes )
2008-07-25 20:28:28 +03:00
{
2009-07-31 23:10:22 +03:00
if ( h - > visitedTown )
giveSpells ( h - > visitedTown , h ) ;
2008-10-18 14:41:24 +03:00
NewTurn : : Hero hth ;
hth . id = h - > id ;
2012-11-06 19:39:29 +03:00
hth . move = h - > maxMovePoints ( gs - > map - > getTile ( h - > getPosition ( false ) ) . terType ! = ETerrainType : : WATER ) ;
2009-04-04 01:34:31 +03:00
2013-02-11 02:24:57 +03:00
if ( h - > visitedTown & & h - > visitedTown - > hasBuilt ( BuildingID : : MAGES_GUILD_1 ) ) //if hero starts turn in town with mage guild
2009-12-24 08:48:45 +02:00
hth . mana = std : : max ( h - > mana , h - > manaLimit ( ) ) ; //restore all mana
2009-04-04 01:34:31 +03:00
else
2012-07-23 13:23:43 +03:00
hth . mana = std : : max ( ( si32 ) ( 0 ) , std : : max ( h - > mana , std : : min ( ( si32 ) ( h - > mana + h - > manaRegain ( ) ) , h - > manaLimit ( ) ) ) ) ;
2009-04-04 01:34:31 +03:00
2008-10-18 14:41:24 +03:00
n . heroes . insert ( hth ) ;
2012-07-23 13:23:43 +03:00
2011-08-26 23:32:05 +03:00
if ( ! firstTurn ) //not first day
2008-09-28 16:29:37 +03:00
{
2013-06-29 16:05:48 +03:00
n . res [ elem . first ] [ Res : : GOLD ] + = h - > valOfBonuses ( Selector : : typeSubtype ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : ESTATES ) ) ; //estates
2010-08-24 17:26:57 +03:00
2011-12-14 00:23:17 +03:00
for ( int k = 0 ; k < GameConstants : : RESOURCE_QUANTITY ; k + + )
2010-07-06 13:02:38 +03:00
{
2013-06-29 16:05:48 +03:00
n . res [ elem . first ] [ k ] + = h - > valOfBonuses ( Bonus : : GENERATE_RESOURCE , k ) ;
2010-07-06 13:02:38 +03:00
}
2009-06-19 04:01:43 +03:00
}
2008-07-25 20:28:28 +03:00
}
2009-11-07 22:23:39 +02:00
}
2013-06-29 16:05:48 +03:00
for ( CGTownInstance * t : gs - > map - > towns )
2009-11-13 21:04:36 +02:00
{
2013-03-03 20:06:03 +03:00
PlayerColor player = t - > tempOwner ;
2013-03-14 23:44:00 +03:00
handleTownEvents ( t , n ) ;
2011-08-26 23:32:05 +03:00
if ( newWeek ) //first day of week
2008-07-25 20:28:28 +03:00
{
2013-02-11 02:24:57 +03:00
if ( t - > hasBuilt ( BuildingID : : PORTAL_OF_SUMMON , ETownType : : DUNGEON ) )
2011-08-26 23:32:05 +03:00
setPortalDwelling ( t , true , ( n . specialWeek = = NewTurn : : PLAGUE ? true : false ) ) ; //set creatures for Portal of Summoning
if ( ! firstTurn )
2013-03-03 20:06:03 +03:00
if ( t - > hasBuilt ( BuildingID : : TREASURY , ETownType : : RAMPART ) & & player < PlayerColor : : PLAYER_LIMIT )
2011-08-26 23:32:05 +03:00
n . res [ player ] [ Res : : GOLD ] + = hadGold [ player ] / 10 ; //give 10% of starting gold
2013-03-14 23:44:00 +03:00
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 [ t - > id ] ;
2011-12-14 00:23:17 +03:00
for ( int k = 0 ; k < GameConstants : : CREATURES_PER_TOWN ; k + + ) //creature growths
2011-08-26 23:32:05 +03:00
{
2012-08-08 00:46:24 +03:00
if ( t - > creatureDwellingLevel ( k ) > = 0 ) //there is dwelling (k-level)
2011-08-26 23:32:05 +03:00
{
ui32 & availableCount = sac . creatures [ k ] . first ;
2012-01-13 17:18:32 +03:00
const CCreature * cre = VLC - > creh - > creatures [ t - > creatures [ k ] . second . back ( ) ] ;
2011-08-26 23:32:05 +03:00
if ( n . specialWeek = = NewTurn : : PLAGUE )
availableCount = t - > creatures [ k ] . first / 2 ; //halve their number, no growth
else
{
if ( firstTurn ) //first day of game: use only basic growths
availableCount = cre - > growth ;
else
availableCount + = t - > creatureGrowth ( k ) ;
2012-01-19 17:33:22 +03:00
//Deity of fire week - upgrade both imps and upgrades
if ( n . specialWeek = = NewTurn : : DEITYOFFIRE & & vstd : : contains ( t - > creatures [ k ] . second , n . creatureid ) )
availableCount + = 15 ;
if ( cre - > idNumber = = n . creatureid ) //bonus week, effect applies only to identical creatures
2011-08-26 23:32:05 +03:00
{
if ( n . specialWeek = = NewTurn : : DOUBLE_GROWTH )
availableCount * = 2 ;
else if ( n . specialWeek = = NewTurn : : BONUS_GROWTH )
availableCount + = 5 ;
}
}
}
}
2009-11-13 21:04:36 +02:00
}
2013-03-03 20:06:03 +03:00
if ( ! firstTurn & & player < PlayerColor : : PLAYER_LIMIT ) //not the first day and town not neutral
2009-11-13 21:04:36 +02:00
{
2013-02-11 02:24:57 +03:00
if ( t - > hasBuilt ( BuildingID : : RESOURCE_SILO ) ) //there is resource silo
2009-11-07 22:23:39 +02:00
{
2013-04-21 15:49:26 +03:00
if ( t - > town - > primaryRes = = Res : : WOOD_AND_ORE ) //we'll give wood and ore
2009-11-07 22:23:39 +02:00
{
2011-06-24 21:16:28 +03:00
n . res [ player ] [ Res : : WOOD ] + + ;
n . res [ player ] [ Res : : ORE ] + + ;
2009-11-07 22:23:39 +02:00
}
2009-11-13 21:04:36 +02:00
else
{
2011-08-26 23:32:05 +03:00
n . res [ player ] [ t - > town - > primaryRes ] + + ;
2009-11-13 21:04:36 +02:00
}
}
2011-08-26 23:32:05 +03:00
n . res [ player ] [ Res : : GOLD ] + = t - > dailyIncome ( ) ;
2010-06-07 08:28:12 +03:00
}
2013-02-11 02:24:57 +03:00
if ( t - > hasBuilt ( BuildingID : : GRAIL , ETownType : : TOWER ) )
2010-08-24 17:26:57 +03:00
{
2011-08-26 23:32:05 +03:00
// Skyship, probably easier to handle same as Veil of darkness
//do it every new day after veils apply
2013-03-03 20:06:03 +03:00
if ( player ! = PlayerColor : : NEUTRAL ) //do not reveal fow for neutral player
2012-09-28 09:10:01 +03:00
{
FoWChange fw ;
fw . mode = 1 ;
fw . player = player ;
// find all hidden tiles
auto & fow = gs - > getPlayerTeam ( player ) - > fogOfWarMap ;
for ( size_t i = 0 ; i < fow . size ( ) ; i + + )
for ( size_t j = 0 ; j < fow [ i ] . size ( ) ; j + + )
for ( size_t k = 0 ; k < fow [ i ] [ j ] . size ( ) ; k + + )
if ( ! fow [ i ] [ j ] [ k ] )
fw . tiles . insert ( int3 ( i , j , k ) ) ;
sendAndApply ( & fw ) ;
}
2010-06-07 08:28:12 +03:00
}
2011-08-26 23:32:05 +03:00
if ( t - > hasBonusOfType ( Bonus : : DARKNESS ) )
2010-06-07 08:28:12 +03:00
{
2012-09-20 19:55:21 +03:00
t - > hideTiles ( t - > getOwner ( ) , t - > getBonusLocalFirst ( Selector : : type ( Bonus : : DARKNESS ) ) - > val ) ;
2010-06-07 08:28:12 +03:00
}
2010-06-19 12:13:10 +03:00
//unhiding what shouldn't be hidden? //that's handled in netpacks client
2009-11-13 21:04:36 +02:00
}
2009-11-07 22:23:39 +02:00
2011-08-26 23:32:05 +03:00
if ( newMonth )
2010-06-27 19:03:01 +03:00
{
2012-07-23 13:23:43 +03:00
SetAvailableArtifacts saa ;
2010-06-27 19:03:01 +03:00
saa . id = - 1 ;
pickAllowedArtsSet ( saa . arts ) ;
sendAndApply ( & saa ) ;
}
2010-08-24 17:26:57 +03:00
sendAndApply ( & n ) ;
2011-08-26 23:32:05 +03:00
if ( newWeek )
{
//spawn wandering monsters
if ( newMonth & & ( n . specialWeek = = NewTurn : : DOUBLE_GROWTH | | n . specialWeek = = NewTurn : : DEITYOFFIRE ) )
2010-08-22 21:21:45 +03:00
{
2011-08-26 23:32:05 +03:00
spawnWanderingMonsters ( n . creatureid ) ;
2010-08-22 21:21:45 +03:00
}
2011-08-26 23:32:05 +03:00
//new week info popup
if ( ! firstTurn )
2010-08-22 21:21:45 +03:00
{
2011-08-26 23:32:05 +03:00
InfoWindow iw ;
2010-08-24 17:26:57 +03:00
switch ( n . specialWeek )
2010-08-22 21:21:45 +03:00
{
2010-08-24 17:26:57 +03:00
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 ;
2010-08-26 10:23:08 +03:00
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 ;
2010-08-24 17:26:57 +03:00
default :
2012-01-19 17:33:22 +03:00
if ( newMonth )
{
iw . text . addTxt ( MetaString : : ARRAY_TXT , ( 130 ) ) ;
iw . text . addReplacement ( MetaString : : ARRAY_TXT , 32 + rand ( ) % 10 ) ;
}
else
{
iw . text . addTxt ( MetaString : : ARRAY_TXT , ( 133 ) ) ;
iw . text . addReplacement ( MetaString : : ARRAY_TXT , 43 + rand ( ) % 15 ) ;
}
2010-08-22 21:21:45 +03:00
}
2013-06-29 16:05:48 +03:00
for ( auto & elem : gs - > players )
2010-08-24 17:26:57 +03:00
{
2013-06-29 16:05:48 +03:00
iw . player = elem . first ;
2010-08-24 17:26:57 +03:00
sendAndApply ( & iw ) ;
}
2010-08-22 21:21:45 +03:00
}
}
2011-08-26 23:32:05 +03:00
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Info about turn " < < n . day < < " has been sent! " ;
2009-03-09 21:40:43 +02:00
handleTimeEvents ( ) ;
//call objects
2013-06-29 16:05:48 +03:00
for ( auto & elem : gs - > map - > objects )
2010-08-22 10:54:24 +03:00
{
2013-06-29 16:05:48 +03:00
if ( elem )
elem - > newTurn ( ) ;
2010-08-22 10:54:24 +03:00
}
2010-01-29 22:52:45 +02:00
winLoseHandle ( 0xff ) ;
2010-02-02 01:30:03 +02:00
//warn players without town
if ( gs - > day )
{
2012-09-24 19:14:53 +03:00
for ( auto i = gs - > players . cbegin ( ) ; i ! = gs - > players . cend ( ) ; i + + )
2010-02-02 01:30:03 +02:00
{
2013-03-03 20:06:03 +03:00
if ( i - > second . status | | i - > second . towns . size ( ) | | i - > second . color > = PlayerColor : : PLAYER_LIMIT )
2010-02-02 01:30:03 +02:00
continue ;
InfoWindow iw ;
iw . player = i - > first ;
2013-03-03 20:06:03 +03:00
iw . components . push_back ( Component ( Component : : FLAG , i - > first . getNum ( ) , 0 , 0 ) ) ;
2010-02-02 01:30:03 +02:00
if ( ! i - > second . daysWithoutCastle )
{
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.
2013-03-03 20:06:03 +03:00
iw . text . addReplacement ( MetaString : : COLOR , i - > first . getNum ( ) ) ;
2010-02-02 01:30:03 +02:00
}
else if ( i - > second . daysWithoutCastle = = 6 )
{
iw . text . addTxt ( MetaString : : ARRAY_TXT , 129 ) ; //%s, this is your last day to capture a town or you will be banished from this land.
2013-03-03 20:06:03 +03:00
iw . text . addReplacement ( MetaString : : COLOR , i - > first . getNum ( ) ) ;
2010-02-02 01:30:03 +02:00
}
else
{
iw . text . addTxt ( MetaString : : ARRAY_TXT , 128 ) ; //%s, you only have %d days left to capture a town or you will be banished from this land.
2013-03-03 20:06:03 +03:00
iw . text . addReplacement ( MetaString : : COLOR , i - > first . getNum ( ) ) ;
2010-02-02 01:30:03 +02:00
iw . text . addReplacement ( 7 - i - > second . daysWithoutCastle ) ;
}
sendAndApply ( & iw ) ;
}
}
2013-02-19 01:37:22 +03:00
synchronizeArtifactHandlerLists ( ) ; //new day events may have changed them. TODO better of managing that
2008-07-25 20:28:28 +03:00
}
2010-10-24 14:35:14 +03:00
void CGameHandler : : run ( bool resume )
2010-05-02 21:20:26 +03:00
{
2013-06-17 18:45:55 +03:00
LOG_TRACE_PARAMS ( logGlobal , " resume=%d " , resume ) ;
2010-05-02 21:20:26 +03:00
using namespace boost : : posix_time ;
2013-06-29 16:05:48 +03:00
for ( CConnection * cc : conns )
2013-01-21 23:49:19 +03:00
{
2009-01-11 00:08:18 +02:00
if ( ! resume )
2010-12-22 22:14:40 +02:00
{
2012-04-14 05:20:22 +03:00
( * cc ) < < gs - > initialOpts ; // gs->scenarioOps
2010-12-22 22:14:40 +02:00
}
2009-01-11 00:08:18 +02:00
2013-03-03 20:06:03 +03:00
std : : set < PlayerColor > players ;
2013-01-21 23:49:19 +03:00
( * cc ) > > players ; //how many players will be handled at that client
2010-10-24 14:35:14 +03:00
2013-04-10 20:18:01 +03:00
std : : stringstream sbuffer ;
sbuffer < < " Connection " < < cc - > connectionID < < " will handle " < < players . size ( ) < < " player: " ;
2013-06-29 16:05:48 +03:00
for ( PlayerColor color : players )
2008-07-25 20:28:28 +03:00
{
2013-04-10 20:18:01 +03:00
sbuffer < < color < < " " ;
2009-04-12 04:48:50 +03:00
{
boost : : unique_lock < boost : : recursive_mutex > lock ( gsm ) ;
2013-01-21 23:49:19 +03:00
connections [ color ] = cc ;
2009-04-12 04:48:50 +03:00
}
2012-07-23 13:23:43 +03:00
}
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < sbuffer . str ( ) ;
2012-09-25 21:00:55 +03:00
cc - > addStdVecItems ( gs ) ;
cc - > enableStackSendingByID ( ) ;
cc - > disableSmartPointerSerialization ( ) ;
2008-07-25 20:28:28 +03:00
}
2012-07-23 13:23:43 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & elem : conns )
2008-07-25 20:28:28 +03:00
{
2013-03-03 20:06:03 +03:00
std : : set < PlayerColor > pom ;
for ( auto j = connections . cbegin ( ) ; j ! = connections . cend ( ) ; j + + )
2013-06-29 16:05:48 +03:00
if ( j - > second = = elem )
2008-07-25 20:28:28 +03:00
pom . insert ( j - > first ) ;
2013-07-02 18:23:32 +03:00
boost : : thread ( boost : : bind ( & CGameHandler : : handleConnection , this , pom , boost : : ref ( * elem ) ) ) ;
2008-07-25 20:28:28 +03:00
}
2008-07-31 00:27:15 +03:00
2011-09-28 01:03:43 +03:00
if ( gs - > scenarioOps - > mode = = StartInfo : : DUEL )
{
runBattle ( ) ;
2013-06-17 18:45:55 +03:00
end2 = true ;
while ( conns . size ( ) & & ( * conns . begin ( ) ) - > isOpen ( ) )
boost : : this_thread : : sleep ( boost : : posix_time : : milliseconds ( 5 ) ) ; //give time client to close socket
2011-09-28 01:03:43 +03:00
return ;
}
2008-08-04 12:05:52 +03:00
while ( ! end2 )
2008-07-25 20:28:28 +03:00
{
2009-01-25 18:19:34 +02:00
if ( ! resume )
newTurn ( ) ;
2009-01-11 00:08:18 +02:00
2013-03-03 20:06:03 +03:00
std : : map < PlayerColor , PlayerState > : : iterator i ;
2009-01-11 00:08:18 +02:00
if ( ! resume )
i = gs - > players . begin ( ) ;
else
i = gs - > players . find ( gs - > currentPlayer ) ;
2010-04-02 05:07:40 +03:00
resume = false ;
2009-01-11 00:08:18 +02:00
for ( ; i ! = gs - > players . end ( ) ; i + + )
2008-07-25 20:28:28 +03:00
{
2011-05-30 22:20:14 +03:00
if ( ( i - > second . towns . size ( ) = = 0 & & i - > second . heroes . size ( ) = = 0 )
2013-03-03 20:06:03 +03:00
| | i - > first > = PlayerColor : : PLAYER_LIMIT
2012-07-23 13:23:43 +03:00
| | i - > second . status )
2010-01-29 22:52:45 +02:00
{
2012-07-23 13:23:43 +03:00
continue ;
2010-01-29 22:52:45 +02:00
}
2008-08-13 03:44:31 +03:00
states . setFlag ( i - > first , & PlayerStatus : : makingTurn , true ) ;
2010-04-02 05:07:40 +03:00
2009-03-07 17:54:12 +02:00
{
YourTurn yt ;
yt . player = i - > first ;
2011-09-06 12:59:06 +03:00
applyAndSend ( & yt ) ;
2009-03-07 17:54:12 +02:00
}
2008-08-13 03:44:31 +03:00
2008-07-25 20:28:28 +03:00
//wait till turn is done
2008-08-13 03:44:31 +03:00
boost : : unique_lock < boost : : mutex > lock ( states . mx ) ;
while ( states . players [ i - > first ] . makingTurn & & ! end2 )
2008-07-25 20:28:28 +03:00
{
2010-12-22 22:14:40 +02:00
static time_duration p = milliseconds ( 200 ) ;
2012-07-23 13:23:43 +03:00
states . cv . timed_wait ( lock , p ) ;
2012-02-16 20:10:58 +03:00
2010-02-04 17:50:59 +02:00
}
2010-02-06 15:27:58 +02:00
}
2010-02-04 17:50:59 +02:00
}
2010-12-22 22:14:40 +02:00
while ( conns . size ( ) & & ( * conns . begin ( ) ) - > isOpen ( ) )
boost : : this_thread : : sleep ( boost : : posix_time : : milliseconds ( 5 ) ) ; //give time client to close socket
}
2009-07-21 16:53:26 +03:00
2010-12-22 22:14:40 +02:00
void CGameHandler : : setupBattle ( int3 tile , const CArmedInstance * armies [ 2 ] , const CGHeroInstance * heroes [ 2 ] , bool creatureBank , const CGTownInstance * town )
{
2013-06-26 14:18:27 +03:00
battleResult . set ( nullptr ) ;
2008-08-15 15:11:42 +03:00
//send info about battles
BattleStart bs ;
2010-12-22 22:14:40 +02:00
bs . info = gs - > setupBattle ( tile , armies , heroes , creatureBank , town ) ;
2008-08-15 15:11:42 +03:00
sendAndApply ( & bs ) ;
2008-08-17 17:09:30 +03:00
}
2008-09-09 10:05:02 +03:00
2012-08-27 15:34:43 +03:00
void CGameHandler : : checkForBattleEnd ( )
2008-09-09 10:05:02 +03:00
{
2012-08-27 15:34:43 +03:00
auto & stacks = gs - > curB - > stacks ;
2008-09-09 10:05:02 +03:00
//checking winning condition
2011-09-24 04:15:36 +03:00
bool hasStack [ 2 ] ; //hasStack[0] - true if attacker has a living stack; defender similarly
2008-09-09 10:05:02 +03:00
hasStack [ 0 ] = hasStack [ 1 ] = false ;
2013-06-29 16:05:48 +03:00
for ( auto & stack : stacks )
2008-09-09 10:05:02 +03:00
{
2013-06-29 16:05:48 +03:00
if ( stack - > alive ( ) & & ! stack - > hasBonusOfType ( Bonus : : SIEGE_WEAPON ) )
2008-09-09 10:05:02 +03:00
{
2013-06-29 16:05:48 +03:00
hasStack [ 1 - stack - > attackerOwned ] = true ;
2008-09-09 10:05:02 +03:00
}
}
if ( ! hasStack [ 0 ] | | ! hasStack [ 1 ] ) //somebody has won
{
2013-02-04 22:43:16 +03:00
setBattleResult ( BattleResult : : NORMAL , hasStack [ 1 ] ) ;
2008-09-09 10:05:02 +03:00
}
2008-09-12 11:51:46 +03:00
}
void CGameHandler : : giveSpells ( const CGTownInstance * t , const CGHeroInstance * h )
{
2011-01-18 20:56:14 +02:00
if ( ! h - > hasSpellbook ( ) )
2013-02-04 22:43:16 +03:00
return ; //hero hasn't spellbook
2008-09-12 11:51:46 +03:00
ChangeSpells cs ;
cs . hid = h - > id ;
cs . learn = true ;
2013-02-04 22:43:16 +03:00
for ( int i = 0 ; i < std : : min ( t - > mageGuildLevel ( ) , h - > getSecSkillLevel ( SecondarySkill : : WISDOM ) + 2 ) ; i + + )
2008-09-12 11:51:46 +03:00
{
2013-02-11 02:24:57 +03:00
if ( t - > hasBuilt ( BuildingID : : GRAIL , ETownType : : CONFLUX ) ) //Aurora Borealis
2008-09-12 11:51:46 +03:00
{
2013-02-11 02:24:57 +03:00
std : : vector < SpellID > spells ;
2010-06-19 13:42:30 +03:00
getAllowedSpells ( spells , i ) ;
2013-06-29 16:05:48 +03:00
for ( auto & spell : spells )
cs . spells . insert ( spell ) ;
2010-06-19 13:42:30 +03:00
}
else
{
for ( int j = 0 ; j < t - > spellsAtLevel ( i + 1 , true ) & & j < t - > spells [ i ] . size ( ) ; j + + )
{
if ( ! vstd : : contains ( h - > spells , t - > spells [ i ] [ j ] ) )
cs . spells . insert ( t - > spells [ i ] [ j ] ) ;
}
2008-09-12 11:51:46 +03:00
}
}
if ( cs . spells . size ( ) )
sendAndApply ( & cs ) ;
2008-09-18 23:24:53 +03:00
}
2008-12-22 19:48:41 +02:00
2013-02-14 02:55:42 +03:00
void CGameHandler : : setBlockVis ( ObjectInstanceID objid , bool bv )
2008-12-22 19:48:41 +02:00
{
SetObjectProperty sop ( objid , 2 , bv ) ;
sendAndApply ( & sop ) ;
}
2009-04-16 03:28:54 +03:00
2013-02-09 00:17:39 +03:00
bool CGameHandler : : removeObject ( const CGObjectInstance * obj )
2008-12-22 19:48:41 +02:00
{
2013-02-09 00:17:39 +03:00
if ( ! obj | | ! getObj ( obj - > id ) )
2009-04-16 03:28:54 +03:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Something wrong, that object already has been removed or hasn't existed! " ;
2009-04-16 03:28:54 +03:00
return false ;
}
2008-12-22 19:48:41 +02:00
RemoveObject ro ;
2013-02-09 00:17:39 +03:00
ro . id = obj - > id ;
2008-12-22 19:48:41 +02:00
sendAndApply ( & ro ) ;
2010-01-29 22:52:45 +02:00
winLoseHandle ( 255 ) ; //eg if monster escaped (removing objs after battle is done dircetly by endBattle, not this function)
2009-04-16 03:28:54 +03:00
return true ;
2008-12-22 19:48:41 +02:00
}
2013-02-14 02:55:42 +03:00
void CGameHandler : : setAmount ( ObjectInstanceID objid , ui32 val )
2008-12-22 19:48:41 +02:00
{
SetObjectProperty sop ( objid , 3 , val ) ;
sendAndApply ( & sop ) ;
}
2013-04-20 14:34:01 +03:00
bool CGameHandler : : moveHero ( ObjectInstanceID hid , int3 dst , ui8 teleporting , PlayerColor asker /*= 255*/ )
2008-12-22 19:48:41 +02:00
{
2009-03-09 12:37:49 +02:00
const CGHeroInstance * h = getHero ( hid ) ;
2013-04-20 14:34:01 +03:00
if ( ! h | | ( asker ! = PlayerColor : : NEUTRAL & & ( teleporting | | h - > getOwner ( ) ! = gs - > currentPlayer ) ) //not turn of that hero or player can't simply teleport hero (at least not with this function)
2009-03-09 12:37:49 +02:00
)
2008-12-22 19:48:41 +02:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Illegal call to move hero! " ;
2009-04-16 03:28:54 +03:00
return false ;
2008-12-22 19:48:41 +02:00
}
2009-03-09 12:37:49 +02:00
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Player " < < asker < < " wants to move hero " < < hid . getNum ( ) < < " from " < < h - > pos < < " to " < < dst ;
2013-04-20 14:34:01 +03:00
const int3 hmpos = dst + int3 ( - 1 , 0 , 0 ) ;
2009-07-31 23:10:22 +03:00
if ( ! gs - > map - > isInTheMap ( hmpos ) )
{
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Destination tile is outside the map! " ;
2009-07-31 23:10:22 +03:00
return false ;
}
2013-04-20 14:34:01 +03:00
const TerrainTile t = * gs - > getTile ( hmpos ) ;
const int cost = gs - > getMovementCost ( h , h - > getPosition ( false ) , hmpos , h - > movement ) ;
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 ;
2009-03-09 12:37:49 +02:00
//result structure for start - movement failed, no move points used
TryMoveHero tmh ;
tmh . id = hid ;
tmh . start = h - > pos ;
tmh . end = dst ;
2009-07-03 22:57:14 +03:00
tmh . result = TryMoveHero : : FAILED ;
2009-03-09 12:37:49 +02:00
tmh . movePoints = h - > movement ;
//check if destination tile is available
2009-07-19 04:00:19 +03:00
2012-07-23 13:23:43 +03:00
//it's a rock or blocked and not visitable tile
2009-07-19 04:00:19 +03:00
//OR hero is on land and dest is water and (there is not present only one object - boat)
2012-11-16 00:29:22 +03:00
if ( ( ( t . terType = = ETerrainType : : ROCK | | ( t . blocked & & ! t . visitable & & ! h - > hasBonusOfType ( Bonus : : FLYING_MOVEMENT ) ) )
2011-07-05 14:31:26 +03:00
& & complain ( " Cannot move hero, destination tile is blocked! " ) )
2013-01-31 23:11:25 +03:00
| | ( ( ! h - > boat & & ! h - > canWalkOnSea ( ) & & 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)
2011-07-05 14:31:26 +03:00
& & complain ( " Cannot move hero, destination tile is on water! " ) )
2012-11-16 00:29:22 +03:00
| | ( ( h - > boat & & t . terType ! = ETerrainType : : WATER & & t . blocked )
2011-07-05 14:31:26 +03:00
& & complain ( " Cannot disembark hero, tile is blocked! " ) )
2013-04-20 14:34:01 +03:00
| | ( ( distance ( h - > pos , dst ) > = 1.5 & & ! teleporting )
2012-11-16 00:29:22 +03:00
& & complain ( " Tiles are not neighboring! " ) )
2013-02-23 15:22:23 +03:00
| | ( ( h - > inTownGarrison )
& & complain ( " Can not move garrisoned hero! " ) )
2013-04-20 14:34:01 +03:00
| | ( ( h - > movement < cost & & dst ! = h - > pos & & ! teleporting )
2011-07-05 14:31:26 +03:00
& & complain ( " Hero doesn't have any movement points left! " ) )
2013-04-20 14:34:01 +03:00
/*|| (states.checkFlag(h->tempOwner, &PlayerStatus::engagedIntoBattle)
& & complain ( " Cannot move hero during the battle " ) ) */ )
2009-03-09 12:37:49 +02:00
{
2009-07-19 04:00:19 +03:00
//send info about movement failure
2009-03-09 12:37:49 +02:00
sendAndApply ( & tmh ) ;
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
2012-11-16 00:29:22 +03:00
//several generic blocks of code
// should be called if hero changes tile but before applying TryMoveHero package
auto leaveTile = [ & ] ( )
{
2013-06-29 16:05:48 +03:00
for ( CGObjectInstance * obj : gs - > map - > getTile ( int3 ( h - > pos . x - 1 , h - > pos . y , h - > pos . z ) ) . visitableObjects )
2012-11-16 00:29:22 +03:00
{
obj - > onHeroLeave ( h ) ;
}
2012-12-14 18:32:53 +03:00
this - > getTilesInRange ( tmh . fowRevealed , h - > getSightCenter ( ) + ( tmh . end - tmh . start ) , h - > getSightRadious ( ) , h - > tempOwner , 1 ) ;
2012-11-16 00:29:22 +03:00
} ;
2013-04-20 14:34:01 +03:00
auto doMove = [ & ] ( TryMoveHero : : EResult result , EGuardLook lookForGuards ,
EVisitDest visitDest , ELEaveTile leavingTile ) - > bool
2012-12-05 13:46:53 +03:00
{
2013-04-20 14:34:01 +03:00
LOG_TRACE_PARAMS ( logGlobal , " Hero %s starts movement from %s to %s " , h - > name % tmh . start % tmh . end ) ;
auto moveQuery = make_shared < CHeroMovementQuery > ( tmh , h ) ;
queries . addQuery ( moveQuery ) ;
if ( leavingTile = = LEAVING_TILE )
leaveTile ( ) ;
2012-12-05 13:46:53 +03:00
tmh . result = result ;
sendAndApply ( & tmh ) ;
2013-04-20 14:34:01 +03:00
2013-06-20 00:26:27 +03:00
if ( lookForGuards = = CHECK_FOR_GUARDS & & this - > isInTheMap ( guardPos ) )
2013-04-20 14:34:01 +03:00
{
tmh . attackedFrom = guardPos ;
const TerrainTile & guardTile = * gs - > getTile ( guardPos ) ;
objectVisited ( guardTile . visitableObjects . back ( ) , h ) ;
moveQuery - > visitDestAfterVictory = visitDest = = VISIT_DEST ;
}
else if ( visitDest = = VISIT_DEST )
{
visitObjectOnTile ( t , h ) ;
}
queries . popIfTop ( moveQuery ) ;
logGlobal - > traceStream ( ) < < " Hero " < < h - > name < < " ends movement " ;
2012-12-05 13:46:53 +03:00
return result ! = TryMoveHero : : FAILED ;
} ;
2012-11-16 00:29:22 +03:00
//interaction with blocking object (like resources)
auto blockingVisit = [ & ] ( ) - > bool
{
2013-06-29 16:05:48 +03:00
for ( CGObjectInstance * obj : t . visitableObjects )
2012-11-16 00:29:22 +03:00
{
2013-04-20 14:34:01 +03:00
if ( obj ! = h & & obj - > blockVisit & & ! obj - > passableFor ( h - > tempOwner ) )
2012-11-16 00:29:22 +03:00
{
2013-04-20 19:01:58 +03:00
return doMove ( TryMoveHero : : BLOCKING_VISIT , this - > IGNORE_GUARDS , VISIT_DEST , REMAINING_ON_TILE ) ;
//this-> is needed for MVS2010 to recognize scope (?)
2012-11-16 00:29:22 +03:00
}
}
return false ;
} ;
2013-04-20 14:34:01 +03:00
if ( embarking )
2009-07-19 04:00:19 +03:00
{
2011-09-19 23:50:25 +03:00
tmh . movePoints = h - > movementPointsAfterEmbark ( h - > movement , cost , false ) ;
2013-04-20 14:34:01 +03:00
return doMove ( TryMoveHero : : EMBARK , IGNORE_GUARDS , DONT_VISIT_DEST , LEAVING_TILE ) ;
2012-11-16 00:29:22 +03:00
//attack guards on embarking? In H3 creatures on water had no zone of control at all
2009-07-19 04:00:19 +03:00
}
2012-11-16 00:29:22 +03:00
2013-04-20 14:34:01 +03:00
if ( disembarking )
2009-07-19 06:10:24 +03:00
{
2011-09-19 23:50:25 +03:00
tmh . movePoints = h - > movementPointsAfterEmbark ( h - > movement , cost , true ) ;
2013-04-20 14:34:01 +03:00
return doMove ( TryMoveHero : : DISEMBARK , CHECK_FOR_GUARDS , VISIT_DEST , LEAVING_TILE ) ;
2009-07-19 06:10:24 +03:00
}
2009-07-19 04:00:19 +03:00
2013-04-20 14:34:01 +03:00
if ( teleporting )
2009-03-09 12:37:49 +02:00
{
2012-11-16 00:29:22 +03:00
if ( blockingVisit ( ) ) // e.g. hero on the other side of teleporter
2012-12-05 13:46:53 +03:00
return true ;
2009-07-21 02:34:06 +03:00
2013-04-20 14:34:01 +03:00
doMove ( TryMoveHero : : TELEPORTATION , IGNORE_GUARDS , DONT_VISIT_DEST , LEAVING_TILE ) ;
2012-11-16 00:29:22 +03:00
// 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 ) ;
2009-03-09 12:37:49 +02:00
}
2012-11-16 00:29:22 +03:00
2009-04-16 03:28:54 +03:00
return true ;
2008-12-22 19:48:41 +02:00
}
2013-04-20 14:34:01 +03:00
//still here? it is standard movement!
{
tmh . movePoints = h - > movement > = cost
? h - > movement - cost
: 0 ;
if ( blockingVisit ( ) )
return true ;
doMove ( TryMoveHero : : SUCCESS , CHECK_FOR_GUARDS , VISIT_DEST , LEAVING_TILE ) ;
return true ;
}
2008-12-22 19:48:41 +02:00
}
2010-06-30 22:27:35 +03:00
2013-03-03 20:06:03 +03:00
bool CGameHandler : : teleportHero ( ObjectInstanceID hid , ObjectInstanceID dstid , ui8 source , PlayerColor asker /* = 255*/ )
2010-06-30 22:27:35 +03:00
{
const CGHeroInstance * h = getHero ( hid ) ;
const CGTownInstance * t = getTown ( dstid ) ;
2012-07-23 13:23:43 +03:00
2010-06-30 22:27:35 +03:00
if ( ! h | | ! t | | h - > getOwner ( ) ! = gs - > currentPlayer )
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Invalid call to teleportHero! " ;
2012-07-23 13:23:43 +03:00
2010-06-30 22:27:35 +03:00
const CGTownInstance * from = h - > visitedTown ;
2012-07-23 13:23:43 +03:00
if ( ( ( h - > getOwner ( ) ! = t - > getOwner ( ) )
2011-07-05 14:31:26 +03:00
& & complain ( " Cannot teleport hero to another player " ) )
2013-02-11 02:24:57 +03:00
| | ( ( ! from | | from - > hasBuilt ( BuildingID : : CASTLE_GATE , ETownType : : INFERNO ) )
2011-07-05 14:31:26 +03:00
& & complain ( " Hero must be in town with Castle gate for teleporting " ) )
2013-02-11 02:24:57 +03:00
| | ( t - > hasBuilt ( BuildingID : : CASTLE_GATE , ETownType : : INFERNO )
2011-07-05 14:31:26 +03:00
& & complain ( " Cannot teleport hero to town without Castle gate in it " ) ) )
2010-06-30 22:27:35 +03:00
return false ;
int3 pos = t - > visitablePos ( ) ;
2010-07-12 13:42:51 +03:00
pos + = h - > getVisitableOffset ( ) ;
2010-06-30 22:27:35 +03:00
moveHero ( hid , pos , 1 ) ;
2010-07-10 19:50:23 +03:00
return true ;
2010-06-30 22:27:35 +03:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : setOwner ( const CGObjectInstance * obj , PlayerColor owner )
2008-12-22 19:48:41 +02:00
{
2013-03-03 20:06:03 +03:00
PlayerColor oldOwner = getOwner ( obj - > id ) ;
SetObjectProperty sop ( obj - > id , 1 , owner . getNum ( ) ) ;
2008-12-22 19:48:41 +02:00
sendAndApply ( & sop ) ;
2010-01-29 22:52:45 +02:00
2013-03-03 20:06:03 +03:00
winLoseHandle ( 1 < < owner . getNum ( ) | 1 < < oldOwner . getNum ( ) ) ;
if ( owner < PlayerColor : : PLAYER_LIMIT & & dynamic_cast < const CGTownInstance * > ( obj ) ) //town captured
2010-02-02 01:30:03 +02:00
{
2013-02-09 00:17:39 +03:00
const CGTownInstance * town = dynamic_cast < const CGTownInstance * > ( obj ) ;
2013-02-11 02:24:57 +03:00
if ( town - > hasBuilt ( BuildingID : : PORTAL_OF_SUMMON , ETownType : : DUNGEON ) )
2010-08-24 17:26:57 +03:00
setPortalDwelling ( town , true , false ) ;
2012-07-23 13:23:43 +03:00
2010-07-10 19:50:23 +03:00
if ( ! gs - > getPlayer ( owner ) - > towns . size ( ) ) //player lost last town
{
InfoWindow iw ;
iw . player = oldOwner ;
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.
sendAndApply ( & iw ) ;
}
2010-02-02 01:30:03 +02:00
}
2012-07-23 13:23:43 +03:00
2010-08-01 17:46:19 +03:00
const PlayerState * p = gs - > getPlayer ( owner ) ;
2012-09-23 21:01:04 +03:00
if ( ( obj - > ID = = Obj : : CREATURE_GENERATOR1 | | obj - > ID = = Obj : : CREATURE_GENERATOR4 ) & & p & & p - > dwellings . size ( ) = = 1 ) //first dwelling captured
2011-07-05 14:31:26 +03:00
{
2013-06-29 16:05:48 +03:00
for ( const CGTownInstance * t : gs - > getPlayer ( owner ) - > towns )
2011-07-05 14:31:26 +03:00
{
2013-02-11 02:24:57 +03:00
if ( t - > hasBuilt ( BuildingID : : PORTAL_OF_SUMMON , ETownType : : DUNGEON ) )
2010-07-10 19:50:23 +03:00
setPortalDwelling ( t ) ; //set initial creatures for all portals of summoning
2011-07-05 14:31:26 +03:00
}
}
2008-12-22 19:48:41 +02:00
}
2010-07-10 19:50:23 +03:00
2013-02-09 01:42:46 +03:00
void CGameHandler : : setHoverName ( const CGObjectInstance * obj , MetaString * name )
2008-12-22 19:48:41 +02:00
{
2013-02-09 01:42:46 +03:00
SetHoverName shn ( obj - > id , * name ) ;
2008-12-22 19:48:41 +02:00
sendAndApply ( & shn ) ;
}
2011-05-22 21:46:52 +03:00
2013-04-20 14:34:01 +03:00
void CGameHandler : : showBlockingDialog ( BlockingDialog * iw )
2008-12-22 19:48:41 +02:00
{
2013-04-20 14:34:01 +03:00
auto dialogQuery = make_shared < CBlockingDialogQuery > ( * iw ) ;
queries . addQuery ( dialogQuery ) ;
iw - > queryID = dialogQuery - > queryID ;
sendToAllClients ( iw ) ;
2008-12-22 19:48:41 +02:00
}
2009-04-11 04:32:50 +03:00
2013-03-03 20:06:03 +03:00
void CGameHandler : : giveResource ( PlayerColor player , Res : : ERes which , int val ) //TODO: cap according to Bersy's suggestion
2008-12-22 19:48:41 +02:00
{
2009-07-19 06:10:24 +03:00
if ( ! val ) return ; //don't waste time on empty call
2008-12-22 19:48:41 +02:00
SetResource sr ;
sr . player = player ;
sr . resid = which ;
2011-04-23 00:51:10 +03:00
sr . val = gs - > players . find ( player ) - > second . resources [ which ] + val ;
2008-12-22 19:48:41 +02:00
sendAndApply ( & sr ) ;
}
2010-12-13 01:44:16 +02:00
2013-04-20 14:34:01 +03:00
void CGameHandler : : giveResources ( PlayerColor player , TResources resources )
{
for ( TResources : : nziterator i ( resources ) ; i . valid ( ) ; i + + )
giveResource ( player , i - > resType , i - > resVal ) ;
}
2010-12-13 01:44:16 +02: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! " ) ;
2011-12-14 00:23:17 +03:00
COMPLAIN_RET_IF ( creatures . stacksCount ( ) > GameConstants : : ARMY_SIZE , " Too many stacks to give! " ) ;
2010-12-13 01:44:16 +02:00
//first we move creatures to give to make them army of object-source
2013-06-29 16:05:48 +03:00
for ( auto & elem : creatures . Slots ( ) )
2010-12-13 01:44:16 +02:00
{
2013-06-29 16:05:48 +03:00
addToSlot ( StackLocation ( obj , obj - > getSlotFor ( elem . second - > type ) ) , elem . second - > type , elem . second - > count ) ;
2012-07-23 13:23:43 +03:00
}
2010-12-13 01:44:16 +02:00
2011-03-08 15:27:32 +02:00
tryJoiningArmy ( obj , h , remove , true ) ;
2009-08-11 10:50:29 +03:00
}
2010-12-06 01:10:02 +02:00
2013-02-14 02:55:42 +03:00
void CGameHandler : : takeCreatures ( ObjectInstanceID objid , const std : : vector < CStackBasicDescriptor > & creatures )
2010-02-07 18:05:27 +02:00
{
2011-05-10 01:20:47 +03:00
std : : vector < CStackBasicDescriptor > cres = creatures ;
if ( cres . size ( ) < = 0 )
2010-02-07 18:05:27 +02:00
return ;
const CArmedInstance * obj = static_cast < const CArmedInstance * > ( getObj ( objid ) ) ;
2010-11-22 02:34:46 +02:00
2013-06-29 16:05:48 +03:00
for ( CStackBasicDescriptor & sbd : cres )
2010-12-06 01:10:02 +02:00
{
TQuantity collected = 0 ;
while ( collected < sbd . count )
{
2013-01-06 07:48:05 +03:00
bool foundSth = false ;
for ( auto i = obj - > Slots ( ) . begin ( ) ; i ! = obj - > Slots ( ) . end ( ) ; i + + )
2010-12-06 01:10:02 +02:00
{
if ( i - > second - > type = = sbd . type )
{
2011-05-10 01:20:47 +03:00
TQuantity take = std : : min ( sbd . count - collected , i - > second - > count ) ; //collect as much cres as we can
2011-09-24 03:32:55 +03:00
changeStackCount ( StackLocation ( obj , i - > first ) , - take , false ) ;
2010-12-06 01:10:02 +02:00
collected + = take ;
2013-01-06 07:48:05 +03:00
foundSth = true ;
2010-12-06 01:10:02 +02:00
break ;
}
}
2010-11-22 02:34:46 +02:00
2013-01-06 07:48:05 +03:00
if ( ! foundSth ) //we went through the whole loop and haven't found appropriate cres
2010-12-06 01:10:02 +02:00
{
complain ( " Unexpected failure during taking creatures! " ) ;
return ;
}
}
}
2010-11-22 02:34:46 +02:00
}
2008-12-22 19:48:41 +02:00
void CGameHandler : : showCompInfo ( ShowInInfobox * comp )
{
sendToAllClients ( comp ) ;
}
2013-02-09 00:17:39 +03:00
void CGameHandler : : heroVisitCastle ( const CGTownInstance * obj , const CGHeroInstance * hero )
2008-12-22 19:48:41 +02:00
{
HeroVisitCastle vc ;
2013-02-09 00:17:39 +03:00
vc . hid = hero - > id ;
vc . tid = obj - > id ;
2008-12-22 19:48:41 +02:00
vc . flags | = 1 ;
sendAndApply ( & vc ) ;
2013-02-09 00:17:39 +03:00
vistiCastleObjects ( obj , hero ) ;
giveSpells ( obj , hero ) ;
2010-01-30 14:46:15 +02:00
2011-12-14 00:23:17 +03:00
if ( gs - > map - > victoryCondition . condition = = EVictoryConditionType : : TRANSPORTITEM )
2013-02-09 00:17:39 +03:00
checkLossVictory ( hero - > tempOwner ) ; //transported artifact?
2009-10-04 17:20:19 +03:00
}
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 ) ;
2008-12-22 19:48:41 +02:00
}
2013-02-09 00:17:39 +03:00
void CGameHandler : : stopHeroVisitCastle ( const CGTownInstance * obj , const CGHeroInstance * hero )
2008-12-22 19:48:41 +02:00
{
HeroVisitCastle vc ;
2013-02-09 00:17:39 +03:00
vc . hid = hero - > id ;
vc . tid = obj - > id ;
2008-12-22 19:48:41 +02:00
sendAndApply ( & vc ) ;
}
2009-04-04 01:34:31 +03:00
2010-12-26 16:34:11 +02:00
void CGameHandler : : removeArtifact ( const ArtifactLocation & al )
{
2010-12-29 23:04:22 +02:00
assert ( al . getArt ( ) ) ;
EraseArtifact ea ;
ea . al = al ;
sendAndApply ( & ea ) ;
2010-12-26 16:34:11 +02:00
}
2013-04-20 14:34:01 +03:00
void CGameHandler : : startBattlePrimary ( const CArmedInstance * army1 , const CArmedInstance * army2 , int3 tile ,
const CGHeroInstance * hero1 , const CGHeroInstance * hero2 , bool creatureBank ,
2013-06-26 14:18:27 +03:00
const CGTownInstance * town ) //use hero=nullptr for no hero
2009-07-21 02:34:06 +03:00
{
2009-08-04 02:53:18 +03:00
engageIntoBattle ( army1 - > tempOwner ) ;
engageIntoBattle ( army2 - > tempOwner ) ;
2010-12-29 23:04:22 +02:00
static const CArmedInstance * armies [ 2 ] ;
2010-12-22 22:14:40 +02:00
armies [ 0 ] = army1 ;
armies [ 1 ] = army2 ;
2010-12-29 23:04:22 +02:00
static const CGHeroInstance * heroes [ 2 ] ;
2010-12-22 22:14:40 +02:00
heroes [ 0 ] = hero1 ;
heroes [ 1 ] = hero2 ;
2013-04-20 14:34:01 +03:00
setupBattle ( tile , armies , heroes , creatureBank , town ) ; //initializes stacks, places creatures on battlefield, blocks and informs player interfaces
auto battleQuery = make_shared < CBattleQuery > ( gs - > curB ) ;
queries . addQuery ( battleQuery ) ;
boost : : thread ( & CGameHandler : : runBattle , this ) ;
2009-07-21 02:34:06 +03:00
}
2013-04-20 14:34:01 +03:00
void CGameHandler : : startBattleI ( const CArmedInstance * army1 , const CArmedInstance * army2 , int3 tile , bool creatureBank )
2008-12-22 19:48:41 +02:00
{
2013-04-20 14:34:01 +03:00
startBattlePrimary ( army1 , army2 , tile ,
2013-06-26 14:18:27 +03:00
army1 - > ID = = Obj : : HERO ? static_cast < const CGHeroInstance * > ( army1 ) : nullptr ,
army2 - > ID = = Obj : : HERO ? static_cast < const CGHeroInstance * > ( army2 ) : nullptr ,
2013-04-20 14:34:01 +03:00
creatureBank ) ;
2008-12-22 19:48:41 +02:00
}
2009-07-21 02:34:06 +03:00
2013-04-20 14:34:01 +03:00
void CGameHandler : : startBattleI ( const CArmedInstance * army1 , const CArmedInstance * army2 , bool creatureBank )
2008-12-22 19:48:41 +02:00
{
2013-04-20 14:34:01 +03:00
startBattleI ( army1 , army2 , army2 - > visitablePos ( ) , creatureBank ) ;
2008-12-22 19:48:41 +02:00
}
2013-02-11 02:24:57 +03:00
void CGameHandler : : changeSpells ( const CGHeroInstance * hero , bool give , const std : : set < SpellID > & spells )
2008-12-22 19:48:41 +02:00
{
ChangeSpells cs ;
2013-02-09 00:17:39 +03:00
cs . hid = hero - > id ;
2008-12-22 19:48:41 +02:00
cs . spells = spells ;
cs . learn = give ;
sendAndApply ( & cs ) ;
}
2009-03-07 17:54:12 +02:00
void CGameHandler : : sendMessageTo ( CConnection & c , const std : : string & message )
2009-02-01 16:11:41 +02:00
{
2009-03-07 17:54:12 +02:00
SystemMessage sm ;
sm . text = message ;
2011-07-17 21:49:05 +03:00
boost : : unique_lock < boost : : mutex > lock ( * c . wmx ) ;
2009-03-07 17:54:12 +02:00
c < < & sm ;
2009-02-04 15:40:54 +02:00
}
void CGameHandler : : giveHeroBonus ( GiveBonus * bonus )
{
sendAndApply ( bonus ) ;
}
void CGameHandler : : setMovePoints ( SetMovePoints * smp )
{
sendAndApply ( smp ) ;
2009-02-06 13:15:39 +02:00
}
2013-02-14 02:55:42 +03:00
void CGameHandler : : setManaPoints ( ObjectInstanceID hid , int val )
2009-02-06 13:15:39 +02:00
{
SetMana sm ;
sm . hid = hid ;
sm . val = val ;
sendAndApply ( & sm ) ;
2009-02-14 21:12:40 +02:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : giveHero ( ObjectInstanceID id , PlayerColor player )
2009-02-14 21:12:40 +02:00
{
GiveHero gh ;
gh . id = id ;
gh . player = player ;
sendAndApply ( & gh ) ;
2009-02-20 12:36:15 +02:00
}
2013-02-14 02:55:42 +03:00
void CGameHandler : : changeObjPos ( ObjectInstanceID objid , int3 newPos , ui8 flags )
2009-02-20 12:36:15 +02:00
{
ChangeObjPos cop ;
cop . objid = objid ;
cop . nPos = newPos ;
cop . flags = flags ;
sendAndApply ( & cop ) ;
2009-03-07 00:11:17 +02:00
}
2013-02-14 02:55:42 +03:00
void CGameHandler : : useScholarSkill ( ObjectInstanceID fromHero , ObjectInstanceID toHero )
2010-02-28 12:01:36 +02:00
{
2010-02-06 15:27:58 +02:00
const CGHeroInstance * h1 = getHero ( fromHero ) ;
const CGHeroInstance * h2 = getHero ( toHero ) ;
2013-02-04 22:43:16 +03:00
if ( h1 - > getSecSkillLevel ( SecondarySkill : : SCHOLAR ) < h2 - > getSecSkillLevel ( SecondarySkill : : SCHOLAR ) )
2010-02-28 12:01:36 +02:00
{
std : : swap ( h1 , h2 ) ; //1st hero need to have higher scholar level for correct message
std : : swap ( fromHero , toHero ) ;
}
2010-02-06 15:27:58 +02:00
2013-02-04 22:43:16 +03:00
int ScholarLevel = h1 - > getSecSkillLevel ( SecondarySkill : : SCHOLAR ) ; //heroes can trade up to this level
2011-01-18 20:56:14 +02:00
if ( ! ScholarLevel | | ! h1 - > hasSpellbook ( ) | | ! h2 - > hasSpellbook ( ) )
2010-02-28 12:01:36 +02:00
return ; //no scholar skill or no spellbook
2010-02-06 15:27:58 +02:00
2013-02-04 22:43:16 +03:00
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
2010-02-06 15:27:58 +02:00
2010-02-28 12:01:36 +02:00
ChangeSpells cs1 ;
cs1 . learn = true ;
cs1 . hid = toHero ; //giving spells to first hero
2013-06-29 16:05:48 +03:00
for ( auto it : h1 - > spells )
2013-02-11 02:24:57 +03:00
if ( h2Lvl > = it . toSpell ( ) - > level & & ! vstd : : contains ( h2 - > spells , it ) ) //hero can learn it and don't have it yet
cs1 . spells . insert ( it ) ; //spell to learn
2010-02-06 15:27:58 +02:00
2010-02-28 12:01:36 +02:00
ChangeSpells cs2 ;
cs2 . learn = true ;
cs2 . hid = fromHero ;
2010-02-06 15:27:58 +02:00
2013-06-29 16:05:48 +03:00
for ( auto it : h2 - > spells )
2013-02-11 02:24:57 +03:00
if ( h1Lvl > = it . toSpell ( ) - > level & & ! vstd : : contains ( h1 - > spells , it ) )
cs2 . spells . insert ( it ) ;
2012-07-23 13:23:43 +03:00
2010-02-28 12:01:36 +02:00
if ( cs1 . spells . size ( ) | | cs2 . spells . size ( ) ) //create a message
2012-07-23 13:23:43 +03:00
{
2010-02-28 12:01:36 +02:00
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 ) ;
2012-07-23 13:23:43 +03:00
2010-02-28 12:01:36 +02:00
if ( cs2 . spells . size ( ) ) //if found new spell - apply
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 140 ) ; //learns
int size = cs2 . spells . size ( ) ;
2013-06-29 16:05:48 +03:00
for ( auto it : cs2 . spells )
2010-02-28 12:01:36 +02:00
{
2013-02-11 02:24:57 +03:00
iw . components . push_back ( Component ( Component : : SPELL , it , 1 , 0 ) ) ;
iw . text . addTxt ( MetaString : : SPELL_NAME , it . toEnum ( ) ) ;
2010-02-28 12:01:36 +02:00
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 ) ;
}
if ( cs1 . spells . size ( ) & & cs2 . spells . size ( ) )
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 141 ) ; //and
}
if ( cs1 . spells . size ( ) )
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 147 ) ; //teaches
int size = cs1 . spells . size ( ) ;
2013-06-29 16:05:48 +03:00
for ( auto it : cs1 . spells )
2010-02-28 12:01:36 +02:00
{
2013-02-11 02:24:57 +03:00
iw . components . push_back ( Component ( Component : : SPELL , it , 1 , 0 ) ) ;
iw . text . addTxt ( MetaString : : SPELL_NAME , it . toEnum ( ) ) ;
2010-02-28 12:01:36 +02:00
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 ) ;
}
2010-02-06 15:27:58 +02:00
}
2013-02-14 02:55:42 +03:00
void CGameHandler : : heroExchange ( ObjectInstanceID hero1 , ObjectInstanceID hero2 )
2009-06-16 14:18:14 +03:00
{
2013-05-27 13:53:28 +03:00
auto h1 = getHero ( hero1 ) , h2 = getHero ( hero2 ) ;
2009-06-16 14:18:14 +03:00
2013-05-27 13:53:28 +03:00
if ( gameState ( ) - > getPlayerRelations ( h1 - > getOwner ( ) , h2 - > getOwner ( ) ) )
2009-06-16 14:18:14 +03:00
{
2013-05-27 13:53:28 +03:00
auto exchange = make_shared < CGarrisonDialogQuery > ( h1 , h2 ) ;
ExchangeDialog hex ;
hex . queryID = exchange - > queryID ;
hex . heroes [ 0 ] = getHero ( hero1 ) ;
hex . heroes [ 1 ] = getHero ( hero2 ) ;
2009-06-16 14:18:14 +03:00
sendAndApply ( & hex ) ;
2010-02-28 12:01:36 +02:00
useScholarSkill ( hero1 , hero2 ) ;
2013-05-27 13:53:28 +03:00
queries . addQuery ( exchange ) ;
2009-06-16 14:18:14 +03:00
}
}
2009-03-09 12:37:49 +02:00
void CGameHandler : : sendToAllClients ( CPackForClient * info )
2009-03-07 17:54:12 +02:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Sending to all clients a package of type " < < typeid ( * info ) . name ( ) ;
2013-06-29 16:05:48 +03:00
for ( auto & elem : conns )
2009-03-07 17:54:12 +02:00
{
2013-06-29 16:05:48 +03:00
boost : : unique_lock < boost : : mutex > lock ( * ( elem ) - > wmx ) ;
* elem < < info ;
2009-03-07 17:54:12 +02:00
}
}
2011-09-06 12:59:06 +03:00
void CGameHandler : : sendAndApply ( CPackForClient * info )
2009-03-07 17:54:12 +02:00
{
sendToAllClients ( info ) ;
2010-12-06 01:10:02 +02:00
gs - > apply ( info ) ;
2009-03-09 12:37:49 +02:00
}
2011-09-06 12:59:06 +03:00
void CGameHandler : : applyAndSend ( CPackForClient * info )
{
gs - > apply ( info ) ;
sendToAllClients ( info ) ;
}
2010-12-06 01:10:02 +02:00
void CGameHandler : : sendAndApply ( CGarrisonOperationPack * info )
2010-01-30 14:46:15 +02:00
{
sendAndApply ( ( CPackForClient * ) info ) ;
2011-12-14 00:23:17 +03:00
if ( gs - > map - > victoryCondition . condition = = EVictoryConditionType : : GATHERTROOP )
2012-07-23 13:23:43 +03:00
winLoseHandle ( ) ;
2010-01-30 14:46:15 +02:00
}
void CGameHandler : : sendAndApply ( SetResource * info )
{
sendAndApply ( ( CPackForClient * ) info ) ;
2011-12-14 00:23:17 +03:00
if ( gs - > map - > victoryCondition . condition = = EVictoryConditionType : : GATHERRESOURCE )
2010-01-30 14:46:15 +02:00
checkLossVictory ( info - > player ) ;
}
void CGameHandler : : sendAndApply ( SetResources * info )
{
sendAndApply ( ( CPackForClient * ) info ) ;
2011-12-14 00:23:17 +03:00
if ( gs - > map - > victoryCondition . condition = = EVictoryConditionType : : GATHERRESOURCE )
2010-01-30 14:46:15 +02:00
checkLossVictory ( info - > player ) ;
}
void CGameHandler : : sendAndApply ( NewStructures * info )
{
sendAndApply ( ( CPackForClient * ) info ) ;
2011-12-14 00:23:17 +03:00
if ( gs - > map - > victoryCondition . condition = = EVictoryConditionType : : BUILDCITY )
2010-01-30 14:46:15 +02:00
checkLossVictory ( getTown ( info - > tid ) - > tempOwner ) ;
}
2010-12-06 01:10:02 +02:00
2012-08-07 14:28:52 +03:00
void CGameHandler : : save ( const std : : string & filename )
2009-03-09 12:37:49 +02:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Saving to " < < filename ;
2012-08-07 14:28:52 +03:00
CFileInfo info ( filename ) ;
2013-04-20 21:44:55 +03:00
//CResourceHandler::get()->createResource(info.getStem() + ".vlgm1");
2012-08-07 14:28:52 +03:00
CResourceHandler : : get ( ) - > createResource ( info . getStem ( ) + " .vsgm1 " ) ;
2009-03-28 20:46:20 +02:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < " Ordering clients to serialize... " ;
2012-08-07 14:28:52 +03:00
SaveGame sg ( info . getStem ( ) + " .vcgm1 " ) ;
2009-07-18 06:13:13 +03:00
sendToAllClients ( & sg ) ;
2009-03-28 20:46:20 +02:00
}
2012-06-09 22:58:17 +03:00
try
2009-03-09 12:37:49 +02:00
{
2013-02-19 01:37:22 +03:00
// {
2013-04-10 20:18:01 +03:00
// logGlobal->infoStream() << "Serializing game info...";
2013-02-19 01:37:22 +03:00
// CSaveFile save(CResourceHandler::get()->getResourceName(ResourceID(info.getStem(), EResType::LIB_SAVEGAME)));
// // char hlp[8] = "VCMISVG";
// // save << hlp;
// saveCommonState(save);
// }
2009-03-09 12:37:49 +02:00
2012-06-09 22:58:17 +03:00
{
2013-07-28 17:49:50 +03:00
CSaveFile save ( * CResourceHandler : : get ( ) - > getResourceName ( ResourceID ( info . getStem ( ) , EResType : : SERVER_SAVEGAME ) ) ) ;
2013-02-19 01:37:22 +03:00
saveCommonState ( save ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < " Saving server state " ;
2012-06-09 22:58:17 +03:00
save < < * this ;
}
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < " Game has been successfully saved! " ;
2012-06-09 22:58:17 +03:00
}
catch ( std : : exception & e )
2009-03-09 12:37:49 +02:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < " Failed to save game: " < < e . what ( ) ;
2009-03-09 12:37:49 +02:00
}
}
void CGameHandler : : close ( )
{
2013-04-10 20:18:01 +03:00
logGlobal - > infoStream ( ) < < " We have been requested to close. " ;
2011-01-08 21:38:42 +02:00
if ( gs - > initialOpts - > mode = = StartInfo : : DUEL )
{
exit ( 0 ) ;
}
2013-06-29 16:05:48 +03:00
//for(CConnection *cc : conns)
2009-04-11 04:32:50 +03:00
// if(cc && cc->socket && cc->socket->is_open())
// cc->socket->close();
//exit(0);
2009-03-09 12:37:49 +02:00
}
2013-03-03 20:06:03 +03:00
bool CGameHandler : : arrangeStacks ( ObjectInstanceID id1 , ObjectInstanceID id2 , ui8 what , SlotID p1 , SlotID p2 , si32 val , PlayerColor player )
2009-03-09 12:37:49 +02:00
{
2013-02-14 02:55:42 +03:00
const CArmedInstance * s1 = static_cast < CArmedInstance * > ( gs - > getObjInstance ( id1 ) ) ,
* s2 = static_cast < CArmedInstance * > ( gs - > getObjInstance ( id2 ) ) ;
2011-07-05 22:05:41 +03:00
const CCreatureSet & S1 = * s1 , & S2 = * s2 ;
2010-11-27 22:17:28 +02:00
StackLocation sl1 ( s1 , p1 ) , sl2 ( s2 , p2 ) ;
2013-02-16 17:03:47 +03:00
if ( ! sl1 . slot . validSlot ( ) | | ! sl2 . slot . validSlot ( ) )
2012-08-30 19:01:19 +03:00
{
complain ( " Invalid slot accessed! " ) ;
return false ;
}
2009-03-09 12:37:49 +02:00
2009-04-12 03:58:41 +03:00
if ( ! isAllowedExchange ( id1 , id2 ) )
{
complain ( " Cannot exchange stacks between these two objects! \n " ) ;
2009-04-16 03:28:54 +03:00
return false ;
2009-04-12 03:58:41 +03:00
}
2009-03-09 12:37:49 +02:00
if ( what = = 1 ) //swap
{
2013-03-03 20:06:03 +03:00
if ( ( ( s1 - > tempOwner ! = player & & s1 - > tempOwner ! = PlayerColor : : UNFLAGGABLE ) & & s1 - > getStackCount ( p1 ) ) //why 254??
| | ( ( s2 - > tempOwner ! = player & & s2 - > tempOwner ! = PlayerColor : : UNFLAGGABLE ) & & s2 - > getStackCount ( p2 ) ) )
2010-08-12 18:54:25 +03:00
{
complain ( " Can't take troops from another player! " ) ;
return false ;
}
2009-03-09 12:37:49 +02:00
2010-11-27 22:17:28 +02:00
swapStacks ( sl1 , sl2 ) ;
2009-03-09 12:37:49 +02:00
}
else if ( what = = 2 ) //merge
{
2011-07-05 22:05:41 +03:00
if ( ( s1 - > getCreature ( p1 ) ! = s2 - > getCreature ( p2 ) & & complain ( " Cannot merge different creatures stacks! " ) )
2013-03-03 20:06:03 +03:00
| | ( ( ( s1 - > tempOwner ! = player & & s1 - > tempOwner ! = PlayerColor : : UNFLAGGABLE ) & & s2 - > getStackCount ( p2 ) ) & & complain ( " Can't take troops from another player! " ) ) )
2012-07-23 13:23:43 +03:00
return false ;
2009-03-09 12:37:49 +02:00
2010-11-27 22:17:28 +02:00
moveStack ( sl1 , sl2 ) ;
2009-03-09 12:37:49 +02:00
}
else if ( what = = 3 ) //split
{
2011-07-05 22:05:41 +03:00
if ( ( s1 - > tempOwner ! = player & & s1 - > getStackCount ( p1 ) < s1 - > getStackCount ( p1 ) )
| | ( s2 - > tempOwner ! = player & & s2 - > getStackCount ( p2 ) < s2 - > getStackCount ( p2 ) ) )
2010-11-27 22:17:28 +02:00
{
complain ( " Can't move troops of another player! " ) ;
return false ;
}
2009-03-27 01:05:40 +02:00
//general conditions checking
2011-01-21 04:36:30 +02:00
if ( ( ! vstd : : contains ( S1 . stacks , p1 ) & & complain ( " no creatures to split " ) )
2009-03-27 01:05:40 +02:00
| | ( val < 1 & & complain ( " no creatures to split " ) ) )
2009-03-09 12:37:49 +02:00
{
2009-04-16 03:28:54 +03:00
return false ;
2009-03-27 01:05:40 +02:00
}
2011-01-21 04:36:30 +02:00
if ( vstd : : contains ( S2 . stacks , p2 ) ) //dest. slot not free - it must be "rebalancing"...
2009-03-27 01:05:40 +02:00
{
2011-07-05 22:05:41 +03:00
int total = s1 - > getStackCount ( p1 ) + s2 - > getStackCount ( p2 ) ;
2009-03-27 01:05:40 +02:00
if ( ( total < val & & complain ( " Cannot split that stack, not enough creatures! " ) )
2011-07-05 22:05:41 +03:00
| | ( s1 - > getCreature ( p1 ) ! = s2 - > getCreature ( p2 ) & & complain ( " Cannot rebalance different creatures stacks! " ) )
2009-03-27 01:05:40 +02:00
)
{
2012-07-23 13:23:43 +03:00
return false ;
2009-03-27 01:05:40 +02:00
}
2012-07-23 13:23:43 +03:00
2011-07-05 22:05:41 +03:00
moveStack ( sl1 , sl2 , val - s2 - > getStackCount ( p2 ) ) ;
2010-11-27 22:17:28 +02:00
//S2.slots[p2]->count = val;
//S1.slots[p1]->count = total - val;
2009-03-27 01:05:40 +02:00
}
else //split one stack to the two
{
2011-07-05 22:05:41 +03:00
if ( s1 - > getStackCount ( p1 ) < val ) //not enough creatures
2009-03-27 01:05:40 +02:00
{
complain ( " Cannot split that stack, not enough creatures! " ) ;
2012-07-23 13:23:43 +03:00
return false ;
2009-03-27 01:05:40 +02:00
}
2009-03-09 12:37:49 +02:00
2010-11-27 22:17:28 +02:00
moveStack ( sl1 , sl2 , val ) ;
2010-08-12 18:54:25 +03:00
}
2009-03-09 12:37:49 +02:00
}
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-03-03 20:06:03 +03:00
PlayerColor CGameHandler : : getPlayerAt ( CConnection * c ) const
2009-03-09 12:37:49 +02:00
{
2013-03-03 20:06:03 +03:00
std : : set < PlayerColor > all ;
for ( auto i = connections . cbegin ( ) ; i ! = connections . cend ( ) ; i + + )
2009-03-09 12:37:49 +02:00
if ( i - > second = = c )
all . insert ( i - > first ) ;
switch ( all . size ( ) )
{
case 0 :
2013-03-03 20:06:03 +03:00
return PlayerColor : : NEUTRAL ;
2009-03-09 12:37:49 +02:00
case 1 :
return * all . begin ( ) ;
default :
{
//if we have more than one player at this connection, try to pick active one
2013-03-03 20:06:03 +03:00
if ( vstd : : contains ( all , gs - > currentPlayer ) )
2009-03-09 12:37:49 +02:00
return gs - > currentPlayer ;
else
2013-03-03 20:06:03 +03:00
return PlayerColor : : CANNOT_DETERMINE ; //cannot say which player is it
2009-03-09 12:37:49 +02:00
}
}
}
2013-02-16 17:03:47 +03:00
bool CGameHandler : : disbandCreature ( ObjectInstanceID id , SlotID pos )
2009-03-09 12:37:49 +02:00
{
2013-02-14 02:55:42 +03:00
CArmedInstance * s1 = static_cast < CArmedInstance * > ( gs - > getObjInstance ( id ) ) ;
2011-01-21 04:36:30 +02:00
if ( ! vstd : : contains ( s1 - > stacks , pos ) )
2009-03-09 12:37:49 +02:00
{
2009-03-27 01:05:40 +02:00
complain ( " Illegal call to disbandCreature - no such stack in army! " ) ;
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
2010-11-27 22:17:28 +02:00
eraseStack ( StackLocation ( s1 , pos ) ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-05-22 18:16:36 +03:00
bool CGameHandler : : buildStructure ( ObjectInstanceID tid , BuildingID requestedID , bool force /*=false*/ )
2009-03-09 12:37:49 +02:00
{
2013-05-22 18:16:36 +03:00
const CGTownInstance * t = getTown ( tid ) ;
if ( ! t )
COMPLAIN_RETF ( " No such town (ID=%s)! " , tid ) ;
if ( ! t - > town - > buildings . count ( requestedID ) )
COMPLAIN_RETF ( " Town of faction %s does not have info about building ID=%s! " , t - > town - > faction - > name % tid ) ;
const CBuilding * requestedBuilding = t - > town - > buildings [ requestedID ] ;
//Vector with future list of built building and buildings in auto-mode that are not yet built.
std : : vector < const CBuilding * > buildingsThatWillBe , remainingAutoBuildings ;
2009-03-09 12:37:49 +02:00
2013-05-22 18:16:36 +03:00
//Check validity of request
2012-08-08 00:46:24 +03:00
if ( ! force )
2009-03-09 12:37:49 +02:00
{
2013-05-22 18:16:36 +03:00
switch ( requestedBuilding - > mode )
2010-02-24 15:03:36 +02:00
{
2012-09-05 15:49:23 +03:00
case CBuilding : : BUILD_NORMAL :
case CBuilding : : BUILD_AUTO :
2013-05-22 18:16:36 +03:00
if ( gs - > canBuildStructure ( t , requestedID ) ! = EBuildingState : : ALLOWED )
2012-09-05 15:49:23 +03:00
COMPLAIN_RET ( " Cannot build that building! " ) ;
break ;
case CBuilding : : BUILD_SPECIAL :
COMPLAIN_RET ( " This building can not be constructed! " ) ;
break ;
case CBuilding : : BUILD_GRAIL :
2013-05-22 18:16:36 +03:00
if ( requestedBuilding - > mode = = CBuilding : : BUILD_GRAIL ) //needs grail
2012-09-05 15:49:23 +03:00
{
2013-05-22 18:16:36 +03:00
if ( ! t - > visitingHero | | ! t - > visitingHero - > hasArt ( ArtifactID : : GRAIL ) )
2012-09-05 15:49:23 +03:00
COMPLAIN_RET ( " Cannot build this without grail! " )
else
2013-05-22 18:16:36 +03:00
removeArtifact ( ArtifactLocation ( t - > visitingHero , t - > visitingHero - > getArtPos ( ArtifactID : : GRAIL , false ) ) ) ;
2012-09-05 15:49:23 +03:00
}
break ;
2010-02-24 15:03:36 +02:00
}
}
2009-03-09 12:37:49 +02:00
2013-05-22 18:16:36 +03:00
//Performs stuff that has to be done after new building is built
auto processBuiltStructure = [ t , this ] ( const BuildingID buildingID )
{
if ( buildingID > = BuildingID : : DWELL_FIRST ) //dwelling
{
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 [ 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 [ t - > town - > creatures [ level ] [ upgradeNumber ] ] ;
2009-07-11 02:40:10 +03:00
2013-05-22 18:16:36 +03:00
SetAvailableCreatures ssi ;
ssi . tid = t - > id ;
ssi . creatures = t - > creatures ;
if ( buildingID < = BuildingID : : DWELL_LAST )
ssi . creatures [ level ] . first = crea - > growth ;
ssi . creatures [ level ] . second . push_back ( crea - > idNumber ) ;
sendAndApply ( & ssi ) ;
}
2013-07-21 21:27:33 +03:00
if ( t - > subID = = ETownType : : DUNGEON & & buildingID = = BuildingID : : PORTAL_OF_SUMMON )
2013-05-22 18:16:36 +03:00
{
setPortalDwelling ( t ) ;
}
if ( buildingID < = BuildingID : : MAGES_GUILD_5 ) //it's mage guild
{
if ( t - > visitingHero )
giveSpells ( t , t - > visitingHero ) ;
if ( t - > garrisonHero )
giveSpells ( t , t - > garrisonHero ) ;
}
} ;
//Checks if all requirements will be met with expected building list "buildingsThatWillBe"
auto allRequirementsFullfilled = [ & buildingsThatWillBe , t ] ( const CBuilding * b )
2009-03-09 12:37:49 +02:00
{
2013-06-29 16:05:48 +03:00
for ( auto requirementID : b - > requirements )
2013-05-22 18:16:36 +03:00
if ( ! vstd : : contains ( buildingsThatWillBe , t - > town - > buildings [ requirementID ] ) )
return false ;
2012-08-08 00:46:24 +03:00
2013-05-22 18:16:36 +03:00
return true ;
} ;
2012-08-08 00:46:24 +03:00
2013-05-22 18:16:36 +03:00
//Init the vectors
2013-06-29 16:05:48 +03:00
for ( auto & build : t - > town - > buildings )
2010-07-10 19:50:23 +03:00
{
2013-05-22 18:16:36 +03:00
if ( t - > hasBuilt ( build . first ) )
buildingsThatWillBe . push_back ( build . second ) ;
else if ( build . second - > mode = = CBuilding : : BUILD_AUTO ) //not built auto building
remainingAutoBuildings . push_back ( build . second ) ;
2010-07-10 19:50:23 +03:00
}
2009-03-09 12:37:49 +02:00
2013-05-22 18:16:36 +03:00
//Prepare structure (list of building ids will be filled later)
NewStructures ns ;
ns . tid = tid ;
ns . builded = force ? t - > builded : ( t - > builded + 1 ) ;
2012-09-05 15:49:23 +03:00
2013-05-22 18:16:36 +03:00
std : : queue < const CBuilding * > buildingsToAdd ;
buildingsToAdd . push ( requestedBuilding ) ;
while ( buildingsToAdd . size ( ) )
2012-09-05 15:49:23 +03:00
{
2013-05-22 18:16:36 +03:00
auto b = buildingsToAdd . front ( ) ;
buildingsToAdd . pop ( ) ;
ns . bid . insert ( b - > bid ) ;
buildingsThatWillBe . push_back ( b ) ;
remainingAutoBuildings - = b ;
2013-06-29 16:05:48 +03:00
for ( auto autoBuilding : remainingAutoBuildings )
2012-09-05 15:49:23 +03:00
{
2013-05-22 18:16:36 +03:00
if ( allRequirementsFullfilled ( autoBuilding ) )
buildingsToAdd . push ( autoBuilding ) ;
2012-09-05 15:49:23 +03:00
}
}
2013-05-22 18:16:36 +03:00
//Other post-built events
2013-06-29 16:05:48 +03:00
for ( auto builtID : ns . bid )
2013-05-22 18:16:36 +03:00
processBuiltStructure ( builtID ) ;
//Take cost
2010-08-18 17:24:30 +03:00
if ( ! force )
{
SetResources sr ;
sr . player = t - > tempOwner ;
2013-05-22 18:16:36 +03:00
sr . res = gs - > getPlayer ( t - > tempOwner ) - > resources - requestedBuilding - > resources ;
2010-08-18 17:24:30 +03:00
sendAndApply ( & sr ) ;
}
2009-03-09 12:37:49 +02:00
2013-05-30 21:43:45 +03:00
//We know what has been built, appluy changes. Do this as final step to properly update town window
sendAndApply ( & ns ) ;
2013-05-22 18:16:36 +03:00
2013-07-21 21:27:33 +03:00
// now when everything is built - reveal tiles for lookout tower
FoWChange fw ;
fw . player = t - > tempOwner ;
fw . mode = 1 ;
t - > getSightTiles ( fw . tiles ) ;
sendAndApply ( & fw ) ;
2009-10-06 09:15:56 +03:00
if ( t - > visitingHero )
vistiCastleObjects ( t , t - > visitingHero ) ;
if ( t - > garrisonHero )
vistiCastleObjects ( t , t - > garrisonHero ) ;
2010-05-14 05:18:37 +03:00
checkLossVictory ( t - > tempOwner ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-02-14 02:55:42 +03:00
bool CGameHandler : : razeStructure ( ObjectInstanceID tid , BuildingID bid )
2009-09-22 17:27:46 +03:00
{
///incomplete, simply erases target building
2013-02-14 02:55:42 +03:00
const CGTownInstance * t = getTown ( tid ) ;
if ( ! vstd : : contains ( t - > builtBuildings , bid ) )
2009-09-22 17:27:46 +03:00
return false ;
RazeStructures rs ;
rs . tid = tid ;
rs . bid . insert ( bid ) ;
2009-09-24 20:54:02 +03:00
rs . destroyed = t - > destroyed + 1 ;
2009-09-22 17:27:46 +03:00
sendAndApply ( & rs ) ;
2010-06-07 08:28:12 +03:00
//TODO: Remove dwellers
2011-02-04 16:58:14 +02:00
// 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);
// }
2009-09-22 17:27:46 +03:00
return true ;
}
2009-03-09 12:37:49 +02:00
void CGameHandler : : sendMessageToAll ( const std : : string & message )
{
SystemMessage sm ;
sm . text = message ;
sendToAllClients ( & sm ) ;
}
2013-02-14 02:55:42 +03:00
bool CGameHandler : : recruitCreatures ( ObjectInstanceID objid , CreatureID crid , ui32 cram , si32 fromLvl )
2009-03-09 12:37:49 +02:00
{
2013-02-14 02:55:42 +03:00
const CGDwelling * dw = static_cast < const CGDwelling * > ( gs - > getObj ( objid ) ) ;
2013-06-26 14:18:27 +03:00
const CArmedInstance * dst = nullptr ;
2010-05-15 11:33:32 +03:00
const CCreature * c = VLC - > creh - > creatures [ crid ] ;
bool warMachine = c - > hasBonusOfType ( Bonus : : SIEGE_WEAPON ) ;
//TODO: test for owning
2009-07-06 22:41:27 +03:00
2012-09-23 21:01:04 +03:00
if ( dw - > ID = = Obj : : TOWN )
2011-02-25 19:57:30 +02:00
dst = ( static_cast < const CGTownInstance * > ( dw ) ) - > getUpperArmy ( ) ;
2013-01-12 21:08:33 +03:00
else if ( dw - > ID = = Obj : : CREATURE_GENERATOR1 | | dw - > ID = = Obj : : CREATURE_GENERATOR4
2012-09-23 21:01:04 +03:00
| | dw - > ID = = Obj : : REFUGEE_CAMP ) //advmap dwelling
2009-07-06 22:41:27 +03:00
dst = getHero ( gs - > getPlayer ( dw - > tempOwner ) - > currentSelection ) ; //TODO: check if current hero is really visiting dwelling
2012-09-23 21:01:04 +03:00
else if ( dw - > ID = = Obj : : WAR_MACHINE_FACTORY )
2010-05-15 11:33:32 +03:00
dst = dynamic_cast < const CGHeroInstance * > ( getTile ( dw - > visitablePos ( ) ) - > visitableObjects . back ( ) ) ;
2009-07-06 22:41:27 +03:00
assert ( dw & & dst ) ;
2009-03-09 12:37:49 +02:00
//verify
bool found = false ;
2010-07-10 19:50:23 +03:00
int level = 0 ;
2009-07-06 22:41:27 +03:00
2010-07-10 19:50:23 +03:00
for ( ; level < dw - > creatures . size ( ) ; level + + ) //iterate through all levels
2009-03-09 12:37:49 +02:00
{
2010-07-10 19:50:23 +03:00
if ( ( fromLvl ! = - 1 ) & & ( level ! = fromLvl ) )
continue ;
2013-02-07 20:34:50 +03:00
const auto & cur = dw - > creatures [ level ] ; //current level info <amount, list of cr. ids>
2009-07-06 22:41:27 +03:00
int i = 0 ;
for ( ; i < cur . second . size ( ) ; i + + ) //look for crid among available creatures list on current level
if ( cur . second [ i ] = = crid )
break ;
if ( i < cur . second . size ( ) )
2009-03-09 12:37:49 +02:00
{
2009-07-06 22:41:27 +03:00
found = true ;
cram = std : : min ( cram , cur . first ) ; //reduce recruited amount up to available amount
2009-03-09 12:37:49 +02:00
break ;
}
}
2013-02-16 17:03:47 +03:00
SlotID slot = dst - > getSlotFor ( crid ) ;
2009-03-09 12:37:49 +02:00
2011-07-05 14:31:26 +03:00
if ( ( ! found & & complain ( " Cannot recruit: no such creatures! " ) )
| | ( cram > VLC - > creh - > creatures [ crid ] - > maxAmount ( gs - > getPlayer ( dst - > tempOwner ) - > resources ) & & complain ( " Cannot recruit: lack of resources! " ) )
| | ( cram < = 0 & & complain ( " Cannot recruit: cram <= 0! " ) )
2013-02-16 17:03:47 +03:00
| | ( ! slot . validSlot ( ) & & ! warMachine & & complain ( " Cannot recruit: no available slot! " ) ) )
2009-04-16 03:28:54 +03:00
{
return false ;
}
2009-03-09 12:37:49 +02:00
//recruit
SetResources sr ;
2009-07-06 22:41:27 +03:00
sr . player = dst - > tempOwner ;
2011-07-05 09:14:07 +03:00
sr . res = gs - > getPlayer ( dst - > tempOwner ) - > resources - ( c - > cost * cram ) ;
2009-03-09 12:37:49 +02:00
SetAvailableCreatures sac ;
sac . tid = objid ;
2009-07-06 22:41:27 +03:00
sac . creatures = dw - > creatures ;
sac . creatures [ level ] . first - = cram ;
2009-03-09 12:37:49 +02:00
2010-05-15 11:33:32 +03:00
sendAndApply ( & sr ) ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & sac ) ;
2012-07-23 13:23:43 +03:00
2010-05-15 11:33:32 +03:00
if ( warMachine )
{
2010-12-26 16:34:11 +02:00
const CGHeroInstance * h = dynamic_cast < const CGHeroInstance * > ( dst ) ;
if ( ! h )
COMPLAIN_RET ( " Only hero can buy war machines " ) ;
2010-05-15 11:33:32 +03:00
switch ( crid )
{
case 146 :
2011-12-14 00:23:17 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ 4 ] , ArtifactPosition : : MACH1 ) ;
2010-05-15 11:33:32 +03:00
break ;
case 147 :
2011-12-14 00:23:17 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ 6 ] , ArtifactPosition : : MACH3 ) ;
2010-05-15 11:33:32 +03:00
break ;
case 148 :
2011-12-14 00:23:17 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ 5 ] , ArtifactPosition : : MACH2 ) ;
2010-05-15 11:33:32 +03:00
break ;
default :
complain ( " This war machine cannot be recruited! " ) ;
return false ;
}
}
else
{
2010-11-27 22:17:28 +02:00
addToSlot ( StackLocation ( dst , slot ) , c , cram ) ;
2010-05-15 11:33:32 +03:00
}
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-02-16 17:03:47 +03:00
bool CGameHandler : : upgradeCreature ( ObjectInstanceID objid , SlotID pos , CreatureID upgID )
2009-03-09 12:37:49 +02:00
{
2013-02-14 02:55:42 +03:00
CArmedInstance * obj = static_cast < CArmedInstance * > ( gs - > getObjInstance ( objid ) ) ;
2010-11-27 22:17:28 +02:00
assert ( obj - > hasStackAtSlot ( pos ) ) ;
2010-07-15 06:04:57 +03:00
UpgradeInfo ui = gs - > getUpgradeInfo ( obj - > getStack ( pos ) ) ;
2013-03-03 20:06:03 +03:00
PlayerColor player = obj - > tempOwner ;
2011-07-05 09:14:07 +03:00
const PlayerState * p = getPlayer ( player ) ;
2011-01-21 04:36:30 +02:00
int crQuantity = obj - > stacks [ pos ] - > count ;
2011-12-14 00:23:17 +03:00
int newIDpos = vstd : : find_pos ( ui . newID , upgID ) ; //get position of new id in UpgradeInfo
2009-03-09 12:37:49 +02:00
//check if upgrade is possible
2012-07-23 13:23:43 +03:00
if ( ( ui . oldID < 0 | | newIDpos = = - 1 ) & & complain ( " That upgrade is not possible! " ) )
2009-04-16 03:28:54 +03:00
{
return false ;
}
2012-10-01 21:25:43 +03:00
TResources totalCost = ui . cost [ newIDpos ] * crQuantity ;
2009-03-09 12:37:49 +02:00
//check if player has enough resources
2011-07-05 09:14:07 +03:00
if ( ! p - > resources . canAfford ( totalCost ) )
COMPLAIN_RET ( " Cannot upgrade, not enough resources! " ) ;
2012-07-23 13:23:43 +03:00
2009-03-09 12:37:49 +02:00
//take resources
2011-07-05 09:14:07 +03:00
SetResources sr ;
sr . player = player ;
sr . res = p - > resources - totalCost ;
sendAndApply ( & sr ) ;
2012-07-23 13:23:43 +03:00
2009-03-09 12:37:49 +02:00
//upgrade creature
2010-11-27 22:17:28 +02:00
changeStackType ( StackLocation ( obj , pos ) , VLC - > creh - > creatures [ upgID ] ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2010-11-27 22:17:28 +02:00
bool CGameHandler : : changeStackType ( const StackLocation & sl , CCreature * c )
2010-08-17 18:50:17 +03:00
{
2010-11-27 22:17:28 +02:00
if ( ! sl . army - > hasStackAtSlot ( sl . slot ) )
COMPLAIN_RET ( " Cannot find a stack to change type " ) ;
2010-11-27 03:46:19 +02:00
SetStackType sst ;
sst . sl = sl ;
sst . type = c ;
2012-07-23 13:23:43 +03:00
sendAndApply ( & sst ) ;
2010-11-27 22:17:28 +02:00
return true ;
2010-08-17 18:50:17 +03:00
}
2012-07-23 13:23:43 +03:00
void CGameHandler : : moveArmy ( const CArmedInstance * src , const CArmedInstance * dst , bool allowMerging )
2010-12-13 01:44:16 +02:00
{
assert ( src - > canBeMergedWith ( * dst , allowMerging ) ) ;
2011-02-04 16:58:14 +02:00
while ( src - > stacksCount ( ) ) //while there are unmoved creatures
2010-12-13 01:44:16 +02:00
{
2013-06-29 16:05:48 +03:00
auto i = src - > Slots ( ) . begin ( ) ; //iterator to stack to move
2010-12-13 01:44:16 +02:00
StackLocation sl ( src , i - > first ) ; //location of stack to move
2013-02-16 17:03:47 +03:00
SlotID pos = dst - > getSlotFor ( i - > second - > type ) ;
if ( ! pos . validSlot ( ) )
2010-12-13 01:44:16 +02:00
{
//try to merge two other stacks to make place
2013-02-16 17:03:47 +03:00
std : : pair < SlotID , SlotID > toMerge ;
2010-12-13 01:44:16 +02:00
if ( dst - > mergableStacks ( toMerge , i - > first ) & & allowMerging )
{
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 ) ) ;
}
}
}
2013-02-14 02:55:42 +03:00
bool CGameHandler : : garrisonSwap ( ObjectInstanceID tid )
2009-03-09 12:37:49 +02:00
{
CGTownInstance * town = gs - > getTown ( tid ) ;
2009-12-19 14:31:57 +02:00
if ( ! town - > garrisonHero & & town - > visitingHero ) //visiting => garrison, merge armies: town army => hero army
2009-03-09 12:37:49 +02:00
{
2010-11-27 22:17:28 +02:00
if ( ! town - > visitingHero - > canBeMergedWith ( * town ) )
{
complain ( " Cannot make garrison swap, not enough free slots! " ) ;
return false ;
}
2012-07-23 13:23:43 +03:00
2010-12-13 01:44:16 +02:00
moveArmy ( town , town - > visitingHero , true ) ;
2012-07-23 13:23:43 +03:00
2009-03-09 12:37:49 +02:00
SetHeroesInTown intown ;
intown . tid = tid ;
2013-02-14 02:55:42 +03:00
intown . visiting = ObjectInstanceID ( ) ;
2009-03-09 12:37:49 +02:00
intown . garrison = town - > visitingHero - > id ;
sendAndApply ( & intown ) ;
2009-04-16 03:28:54 +03:00
return true ;
2012-07-23 13:23:43 +03:00
}
2009-03-09 12:37:49 +02:00
else if ( town - > garrisonHero & & ! town - > visitingHero ) //move hero out of the garrison
{
2009-04-09 18:05:20 +03:00
//check if moving hero out of town will break 8 wandering heroes limit
2009-06-24 06:14:46 +03:00
if ( getHeroCount ( town - > garrisonHero - > tempOwner , false ) > = 8 )
2009-04-09 18:05:20 +03:00
{
complain ( " Cannot move hero out of the garrison, there are already 8 wandering heroes! " ) ;
2009-04-16 03:28:54 +03:00
return false ;
2009-04-09 18:05:20 +03:00
}
2009-03-09 12:37:49 +02:00
SetHeroesInTown intown ;
intown . tid = tid ;
2013-02-14 02:55:42 +03:00
intown . garrison = ObjectInstanceID ( ) ;
2009-03-09 12:37:49 +02:00
intown . visiting = town - > garrisonHero - > id ;
sendAndApply ( & intown ) ;
2010-12-12 01:11:26 +02:00
return true ;
2009-03-09 12:37:49 +02:00
}
2011-02-04 16:58:14 +02:00
else if ( ! ! town - > garrisonHero & & town - > visitingHero ) //swap visiting and garrison hero
2009-03-09 12:37:49 +02:00
{
SetHeroesInTown intown ;
intown . tid = tid ;
intown . garrison = town - > visitingHero - > id ;
intown . visiting = town - > garrisonHero - > id ;
sendAndApply ( & intown ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
else
{
2009-03-27 01:05:40 +02:00
complain ( " Cannot swap garrison hero! " ) ;
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
}
2010-02-08 23:17:22 +02:00
// With the amount of changes done to the function, it's more like transferArtifacts.
2010-12-29 23:04:22 +02:00
// 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.
2012-04-14 05:20:22 +03:00
bool CGameHandler : : moveArtifact ( const ArtifactLocation & al1 , const ArtifactLocation & al2 )
2009-03-09 12:37:49 +02:00
{
2012-04-14 05:20:22 +03:00
ArtifactLocation src = al1 , dst = al2 ;
2013-03-03 20:06:03 +03:00
const PlayerColor srcPlayer = src . owningPlayer ( ) , dstPlayer = dst . owningPlayer ( ) ;
2012-04-14 05:20:22 +03:00
const CArmedInstance * srcObj = src . relatedObj ( ) , * dstObj = dst . relatedObj ( ) ;
2012-07-23 13:23:43 +03:00
2009-11-10 05:10:14 +02:00
// Make sure exchange is even possible between the two heroes.
2012-04-14 05:20:22 +03:00
if ( ! isAllowedExchange ( srcObj - > id , dstObj - > id ) )
2010-12-29 23:04:22 +02:00
COMPLAIN_RET ( " That heroes cannot make any exchange! " ) ;
2009-04-16 03:28:54 +03:00
2010-12-29 23:04:22 +02:00
const CArtifactInstance * srcArtifact = src . getArt ( ) ;
const CArtifactInstance * destArtifact = dst . getArt ( ) ;
2009-04-04 01:34:31 +03:00
2013-06-26 14:18:27 +03:00
if ( srcArtifact = = nullptr )
2010-12-29 23:04:22 +02:00
COMPLAIN_RET ( " No artifact to move! " ) ;
2012-04-14 05:20:22 +03:00
if ( destArtifact & & srcPlayer ! = dstPlayer )
2010-12-29 23:04:22 +02:00
COMPLAIN_RET ( " Can't touch artifact on hero of another player! " ) ;
2012-04-14 05:20:22 +03:00
2009-11-10 05:10:14 +02:00
// Check if src/dest slots are appropriate for the artifacts exchanged.
// Moving to the backpack is always allowed.
2012-04-14 05:20:22 +03:00
if ( ( ! srcArtifact | | dst . slot < GameConstants : : BACKPACK_START )
2011-01-06 22:00:19 +02:00
& & srcArtifact & & ! srcArtifact - > canBePutAt ( dst , true ) )
COMPLAIN_RET ( " Cannot move artifact! " ) ;
2009-11-10 05:10:14 +02:00
2013-02-11 17:42:09 +03:00
if ( ( srcArtifact & & srcArtifact - > artType - > id = = ArtifactID : : ART_LOCK ) | | ( destArtifact & & destArtifact - > artType - > id = = ArtifactID : : ART_LOCK ) )
2010-12-29 23:04:22 +02:00
COMPLAIN_RET ( " Cannot move artifact locks. " ) ;
2010-02-08 23:17:22 +02:00
2012-07-23 13:23:43 +03:00
if ( dst . slot > = GameConstants : : BACKPACK_START & & srcArtifact - > artType - > isBig ( ) )
2010-12-29 23:04:22 +02:00
COMPLAIN_RET ( " Cannot put big artifacts in backpack! " ) ;
2012-07-23 13:23:43 +03:00
if ( src . slot = = ArtifactPosition : : MACH4 | | dst . slot = = ArtifactPosition : : MACH4 )
2010-12-29 23:04:22 +02:00
COMPLAIN_RET ( " Cannot move catapult! " ) ;
2009-12-30 17:33:28 +02:00
2011-12-14 00:23:17 +03:00
if ( dst . slot > = GameConstants : : BACKPACK_START )
2013-02-12 22:49:40 +03:00
vstd : : amin ( dst . slot , ArtifactPosition ( GameConstants : : BACKPACK_START + dst . getHolderArtSet ( ) - > artifactsInBackpack . size ( ) ) ) ;
2011-01-06 22:00:19 +02:00
2012-04-14 05:20:22 +03:00
if ( src . slot = = dst . slot & & src . artHolder = = dst . artHolder )
2011-01-06 22:00:19 +02:00
COMPLAIN_RET ( " Won't move artifact: Dest same as source! " ) ;
2012-04-14 05:20:22 +03:00
if ( dst . slot < GameConstants : : BACKPACK_START & & destArtifact ) //moving art to another slot
2010-07-20 21:34:32 +03:00
{
2012-04-14 05:20:22 +03:00
//old artifact must be removed first
2013-02-12 22:49:40 +03:00
moveArtifact ( dst , ArtifactLocation ( dst . artHolder , ArtifactPosition (
2013-02-07 02:24:43 +03:00
dst . getHolderArtSet ( ) - > artifactsInBackpack . size ( ) + GameConstants : : BACKPACK_START ) ) ) ;
2010-12-29 23:04:22 +02:00
}
2012-01-30 19:07:52 +03:00
MoveArtifact ma ;
2012-04-14 05:20:22 +03:00
ma . src = src ;
ma . dst = dst ;
2012-01-30 19:07:52 +03:00
sendAndApply ( & ma ) ;
2012-04-30 15:13:36 +03:00
return true ;
2012-01-30 19:07:52 +03:00
}
2010-02-16 16:39:56 +02:00
/**
* 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 .
*/
2013-02-16 17:03:47 +03:00
bool CGameHandler : : assembleArtifacts ( ObjectInstanceID heroID , ArtifactPosition artifactSlot , bool assemble , ArtifactID assembleTo )
2010-02-16 16:39:56 +02:00
{
CGHeroInstance * hero = gs - > getHero ( heroID ) ;
2010-12-29 23:04:22 +02:00
const CArtifactInstance * destArtifact = hero - > getArt ( artifactSlot ) ;
2011-01-22 05:43:20 +02:00
if ( ! destArtifact )
COMPLAIN_RET ( " assembleArtifacts: there is no such artifact instance! " ) ;
2011-01-24 01:49:17 +02:00
if ( assemble )
{
CArtifact * combinedArt = VLC - > arth - > artifacts [ assembleTo ] ;
if ( ! combinedArt - > constituents )
COMPLAIN_RET ( " assembleArtifacts: Artifact being attempted to assemble is not a combined artifacts! " ) ;
if ( ! vstd : : contains ( destArtifact - > assemblyPossibilities ( hero ) , combinedArt ) )
COMPLAIN_RET ( " assembleArtifacts: It's impossible to assemble requested artifact! " ) ;
2011-01-22 05:43:20 +02:00
2011-01-24 01:49:17 +02:00
AssembledArtifact aa ;
aa . al = ArtifactLocation ( hero , artifactSlot ) ;
aa . builtArt = combinedArt ;
sendAndApply ( & aa ) ;
}
else
{
if ( ! destArtifact - > artType - > constituents )
COMPLAIN_RET ( " assembleArtifacts: Artifact being attempted to disassemble is not a combined artifact! " ) ;
2011-01-22 05:43:20 +02:00
2011-01-24 01:49:17 +02:00
DisassembledArtifact da ;
da . al = ArtifactLocation ( hero , artifactSlot ) ;
sendAndApply ( & da ) ;
}
2010-02-16 16:39:56 +02:00
2010-12-29 23:04:22 +02:00
return false ;
2010-02-16 16:39:56 +02:00
}
2013-02-14 02:55:42 +03:00
bool CGameHandler : : buyArtifact ( ObjectInstanceID hid , ArtifactID aid )
2009-03-09 12:37:49 +02:00
{
CGHeroInstance * hero = gs - > getHero ( hid ) ;
2011-02-22 11:47:25 +02:00
CGTownInstance * town = hero - > visitedTown ;
2013-02-07 02:24:43 +03:00
if ( aid = = ArtifactID : : SPELLBOOK )
2009-03-09 12:37:49 +02:00
{
2013-02-11 02:24:57 +03:00
if ( ( ! town - > hasBuilt ( BuildingID : : MAGES_GUILD_1 ) & & complain ( " Cannot buy a spellbook, no mage guild in the town! " ) )
2011-12-14 00:23:17 +03:00
| | ( getResource ( hero - > getOwner ( ) , Res : : GOLD ) < GameConstants : : SPELLBOOK_GOLD_COST & & complain ( " Cannot buy a spellbook, not enough gold! " ) )
| | ( hero - > getArt ( ArtifactPosition : : SPELLBOOK ) & & complain ( " Cannot buy a spellbook, hero already has a one! " ) )
2011-07-05 14:31:26 +03:00
)
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
2011-12-14 00:23:17 +03:00
giveResource ( hero - > getOwner ( ) , Res : : GOLD , - GameConstants : : SPELLBOOK_GOLD_COST ) ;
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ 0 ] , ArtifactPosition : : SPELLBOOK ) ;
assert ( hero - > getArt ( ArtifactPosition : : SPELLBOOK ) ) ;
2009-03-09 12:37:49 +02:00
giveSpells ( town , hero ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
else if ( aid < 7 & & aid > 3 ) //war machine
{
2010-06-26 19:02:10 +03:00
int price = VLC - > arth - > artifacts [ aid ] - > price ;
2013-01-12 21:08:33 +03:00
2013-02-12 22:49:40 +03:00
if ( ( hero - > getArt ( ArtifactPosition ( 9 + aid ) ) & & complain ( " Hero already has this machine! " ) )
2012-09-05 15:49:23 +03:00
| | ( gs - > getPlayer ( hero - > getOwner ( ) ) - > resources [ Res : : GOLD ] < price & & complain ( " Not enough gold! " ) ) )
2009-03-09 12:37:49 +02:00
{
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
2013-02-11 02:24:57 +03:00
if ( ( town - > hasBuilt ( BuildingID : : BLACKSMITH ) & & town - > town - > warMachine = = aid )
| | ( ( town - > hasBuilt ( BuildingID : : BALLISTA_YARD , ETownType : : STRONGHOLD ) ) & & aid = = ArtifactID : : BALLISTA ) )
2012-09-05 15:49:23 +03:00
{
giveResource ( hero - > getOwner ( ) , Res : : GOLD , - price ) ;
2013-02-12 22:49:40 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ aid ] , ArtifactPosition ( 9 + aid ) ) ;
2012-09-05 15:49:23 +03:00
return true ;
}
else
COMPLAIN_RET ( " This machine is unavailable here! " ) ;
2009-03-09 12:37:49 +02:00
}
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
2013-02-11 02:24:57 +03:00
bool CGameHandler : : buyArtifact ( const IMarket * m , const CGHeroInstance * h , Res : : ERes rid , ArtifactID aid )
2010-06-27 19:03:01 +03:00
{
2011-12-14 00:23:17 +03:00
if ( ! vstd : : contains ( m - > availableItemsIds ( EMarketMode : : RESOURCE_ARTIFACT ) , aid ) )
2010-06-27 19:03:01 +03:00
COMPLAIN_RET ( " That artifact is unavailable! " ) ;
int b1 , b2 ;
2011-12-14 00:23:17 +03:00
m - > getOffer ( rid , aid , b1 , b2 , EMarketMode : : RESOURCE_ARTIFACT ) ;
2012-07-23 13:23:43 +03:00
2010-06-27 19:03:01 +03:00
if ( getResource ( h - > tempOwner , rid ) < b1 )
COMPLAIN_RET ( " You can't afford to buy this artifact! " ) ;
SetResource sr ;
sr . player = h - > tempOwner ;
sr . resid = rid ;
sr . val = getResource ( h - > tempOwner , rid ) - b1 ;
sendAndApply ( & sr ) ;
SetAvailableArtifacts saa ;
2012-09-23 21:01:04 +03:00
if ( m - > o - > ID = = Obj : : TOWN )
2010-06-27 19:03:01 +03:00
{
saa . id = - 1 ;
saa . arts = CGTownInstance : : merchantArtifacts ;
}
else if ( const CGBlackMarket * bm = dynamic_cast < const CGBlackMarket * > ( m - > o ) ) //black market
{
2013-02-14 02:55:42 +03:00
saa . id = bm - > id . getNum ( ) ;
2010-06-27 19:03:01 +03:00
saa . arts = bm - > artifacts ;
}
else
COMPLAIN_RET ( " Wrong marktet... " ) ;
bool found = false ;
2013-06-29 16:05:48 +03:00
for ( const CArtifact * & art : saa . arts )
2010-06-27 19:03:01 +03:00
{
if ( art & & art - > id = = aid )
{
2013-06-26 14:18:27 +03:00
art = nullptr ;
2010-06-27 19:03:01 +03:00
found = true ;
break ;
}
}
if ( ! found )
COMPLAIN_RET ( " Cannot find selected artifact on the list " ) ;
sendAndApply ( & saa ) ;
2013-02-07 02:24:43 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ aid ] , ArtifactPosition : : FIRST_AVAILABLE ) ;
2010-08-02 14:06:49 +03:00
return true ;
2010-06-27 19:03:01 +03:00
}
2010-05-18 10:01:54 +03:00
2013-02-14 02:55:42 +03:00
bool CGameHandler : : sellArtifact ( const IMarket * m , const CGHeroInstance * h , ArtifactInstanceID aid , Res : : ERes rid )
2011-04-23 00:51:10 +03:00
{
const CArtifactInstance * art = h - > getArtByInstanceId ( aid ) ;
if ( ! art )
COMPLAIN_RET ( " There is no artifact to sell! " ) ;
if ( art - > artType - > id < 7 )
COMPLAIN_RET ( " Cannot sell a war machine or spellbook! " ) ;
int resVal = 0 , dump = 1 ;
2011-12-14 00:23:17 +03:00
m - > getOffer ( art - > artType - > id , rid , dump , resVal , EMarketMode : : ARTIFACT_RESOURCE ) ;
2011-04-23 00:51:10 +03:00
removeArtifact ( ArtifactLocation ( h , h - > getArtPos ( art ) ) ) ;
giveResource ( h - > tempOwner , rid , resVal ) ;
return true ;
}
2012-05-02 11:37:11 +03:00
//void CGameHandler::lootArtifacts (TArtHolder source, TArtHolder dest, std::vector<ui32> &arts)
//{
// //const CGHeroInstance * h1 = dynamic_cast<CGHeroInstance *> source;
// //auto s = boost::apply_visitor(GetArtifactSetPtr(), source);
// {
// }
//}
2013-02-12 22:49:40 +03:00
bool CGameHandler : : buySecSkill ( const IMarket * m , const CGHeroInstance * h , SecondarySkill skill )
2010-07-20 17:08:13 +03:00
{
if ( ! h )
COMPLAIN_RET ( " You need hero to buy a skill! " ) ;
2012-07-23 13:23:43 +03:00
2013-02-12 22:49:40 +03:00
if ( h - > getSecSkillLevel ( SecondarySkill ( skill ) ) )
2010-07-20 17:08:13 +03:00
COMPLAIN_RET ( " Hero already know this skill " ) ;
2012-07-23 13:23:43 +03:00
2012-12-18 14:24:13 +03:00
if ( ! h - > canLearnSkill ( ) )
2010-07-20 17:08:13 +03:00
COMPLAIN_RET ( " Hero can't learn any more skills " ) ;
2012-07-23 13:23:43 +03:00
2012-12-14 18:32:53 +03:00
if ( h - > type - > heroClass - > secSkillProbability [ skill ] = = 0 ) //can't learn this skill (like necromancy for most of non-necros)
2010-07-22 03:32:45 +03:00
COMPLAIN_RET ( " The hero can't learn this skill! " ) ;
2010-07-20 17:08:13 +03:00
2011-12-14 00:23:17 +03:00
if ( ! vstd : : contains ( m - > availableItemsIds ( EMarketMode : : RESOURCE_SKILL ) , skill ) )
2010-07-20 17:08:13 +03:00
COMPLAIN_RET ( " That skill is unavailable! " ) ;
2011-07-05 14:31:26 +03:00
if ( getResource ( h - > tempOwner , Res : : GOLD ) < 2000 ) //TODO: remove hardcoded resource\summ?
2010-07-20 17:08:13 +03:00
COMPLAIN_RET ( " You can't afford to buy this skill " ) ;
SetResource sr ;
sr . player = h - > tempOwner ;
2011-07-05 14:31:26 +03:00
sr . resid = Res : : GOLD ;
sr . val = getResource ( h - > tempOwner , Res : : GOLD ) - 2000 ;
2010-07-20 17:08:13 +03:00
sendAndApply ( & sr ) ;
2013-02-09 01:42:46 +03:00
changeSecSkill ( h , skill , 1 , true ) ;
2010-08-02 14:06:49 +03:00
return true ;
2010-07-20 17:08:13 +03:00
}
2013-03-03 20:06:03 +03:00
bool CGameHandler : : tradeResources ( const IMarket * market , ui32 val , PlayerColor player , ui32 id1 , ui32 id2 )
2009-03-09 12:37:49 +02:00
{
2012-07-23 13:23:43 +03:00
int r1 = gs - > getPlayer ( player ) - > resources [ id1 ] ,
2010-05-18 10:01:54 +03:00
r2 = gs - > getPlayer ( player ) - > resources [ id2 ] ;
2011-12-14 00:23:17 +03:00
vstd : : amin ( val , r1 ) ; //can't trade more resources than have
2010-05-18 10:01:54 +03:00
int b1 , b2 ; //base quantities for trade
2011-12-14 00:23:17 +03:00
market - > getOffer ( id1 , id2 , b1 , b2 , EMarketMode : : RESOURCE_RESOURCE ) ;
2010-05-18 10:01:54 +03:00
int units = val / b1 ; //how many base quantities we trade
if ( val % b1 ) //all offered units of resource should be used, if not -> somewhere in calculations must be an error
{
//TODO: complain?
assert ( 0 ) ;
}
2009-03-09 12:37:49 +02:00
SetResource sr ;
sr . player = player ;
2013-02-04 22:43:16 +03:00
sr . resid = static_cast < Res : : ERes > ( id1 ) ;
2010-05-18 10:01:54 +03:00
sr . val = r1 - b1 * units ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & sr ) ;
2013-02-04 22:43:16 +03:00
sr . resid = static_cast < Res : : ERes > ( id2 ) ;
2010-05-18 10:01:54 +03:00
sr . val = r2 + b2 * units ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & sr ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-02-16 17:03:47 +03:00
bool CGameHandler : : sellCreatures ( ui32 count , const IMarket * market , const CGHeroInstance * hero , SlotID slot , Res : : ERes resourceID )
2010-05-26 12:47:53 +03:00
{
if ( ! vstd : : contains ( hero - > Slots ( ) , slot ) )
COMPLAIN_RET ( " Hero doesn't have any creature in that slot! " ) ;
const CStackInstance & s = hero - > getStack ( slot ) ;
2011-07-05 14:31:26 +03:00
if ( s . count < count //can't sell more creatures than have
| | ( hero - > Slots ( ) . size ( ) = = 1 & & hero - > needsLastStack ( ) & & s . count = = count ) ) //can't sell last stack
2010-05-26 12:47:53 +03:00
{
COMPLAIN_RET ( " Not enough creatures in army! " ) ;
}
int b1 , b2 ; //base quantities for trade
2011-12-14 00:23:17 +03:00
market - > getOffer ( s . type - > idNumber , resourceID , b1 , b2 , EMarketMode : : CREATURE_RESOURCE ) ;
2010-05-26 12:47:53 +03:00
int units = count / b1 ; //how many base quantities we trade
2012-07-23 13:23:43 +03:00
2010-05-26 12:47:53 +03: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 ) ;
}
2012-07-23 13:23:43 +03:00
2010-11-27 22:17:28 +02:00
changeStackCount ( StackLocation ( hero , slot ) , - count ) ;
2010-05-26 12:47:53 +03:00
SetResource sr ;
sr . player = hero - > tempOwner ;
sr . resid = resourceID ;
sr . val = getResource ( hero - > tempOwner , resourceID ) + b2 * units ;
sendAndApply ( & sr ) ;
return true ;
}
2010-05-18 10:01:54 +03:00
2013-02-16 17:03:47 +03:00
bool CGameHandler : : transformInUndead ( const IMarket * market , const CGHeroInstance * hero , SlotID slot )
2010-07-03 15:00:53 +03:00
{
2013-06-26 14:18:27 +03:00
const CArmedInstance * army = nullptr ;
2010-07-03 15:00:53 +03:00
if ( hero )
army = hero ;
else
army = dynamic_cast < const CGTownInstance * > ( market - > o ) ;
2010-11-27 22:17:28 +02:00
2010-07-03 15:00:53 +03:00
if ( ! army )
COMPLAIN_RET ( " Incorrect call to transform in undead! " ) ;
2010-11-27 22:17:28 +02:00
if ( ! army - > hasStackAtSlot ( slot ) )
2010-07-03 15:00:53 +03:00
COMPLAIN_RET ( " Army doesn't have any creature in that slot! " ) ;
2010-11-27 22:17:28 +02:00
2010-07-03 15:00:53 +03:00
const CStackInstance & s = army - > getStack ( slot ) ;
int resCreature ; //resulting creature - bone dragons or skeletons
2012-07-23 13:23:43 +03:00
2010-08-06 16:14:10 +03:00
if ( s . hasBonusOfType ( Bonus : : DRAGON_NATURE ) )
2010-07-03 15:00:53 +03:00
resCreature = 68 ;
else
resCreature = 56 ;
2010-11-27 22:17:28 +02:00
changeStackType ( StackLocation ( army , slot ) , VLC - > creh - > creatures [ resCreature ] ) ;
2010-07-03 15:00:53 +03:00
return true ;
}
2013-03-03 20:06:03 +03:00
bool CGameHandler : : sendResources ( ui32 val , PlayerColor player , Res : : ERes r1 , PlayerColor r2 )
2010-05-08 01:10:32 +03:00
{
const PlayerState * p2 = gs - > getPlayer ( r2 , false ) ;
2013-02-09 15:56:35 +03:00
if ( ! p2 | | p2 - > status ! = EPlayerStatus : : INGAME )
2010-05-08 01:10:32 +03:00
{
complain ( " Dest player must be in game! " ) ;
return false ;
}
si32 curRes1 = gs - > getPlayer ( player ) - > resources [ r1 ] , curRes2 = gs - > getPlayer ( r2 ) - > resources [ r1 ] ;
val = std : : min ( si32 ( val ) , curRes1 ) ;
SetResource sr ;
sr . player = player ;
sr . resid = r1 ;
sr . val = curRes1 - val ;
sendAndApply ( & sr ) ;
sr . player = r2 ;
sr . val = curRes2 + val ;
sendAndApply ( & sr ) ;
return true ;
}
2013-02-14 02:55:42 +03:00
bool CGameHandler : : setFormation ( ObjectInstanceID hid , ui8 formation )
2009-03-09 12:37:49 +02:00
{
2010-05-02 21:20:26 +03:00
gs - > getHero ( hid ) - > formation = formation ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-03-03 20:06:03 +03:00
bool CGameHandler : : hireHero ( const CGObjectInstance * obj , ui8 hid , PlayerColor player )
2009-03-09 12:37:49 +02:00
{
2010-07-09 02:03:27 +03:00
const PlayerState * p = gs - > getPlayer ( player ) ;
const CGTownInstance * t = gs - > getTown ( obj - > id ) ;
2013-03-03 20:06:03 +03:00
static const int GOLD_NEEDED = 2500 ;
2010-07-08 08:52:11 +03:00
2011-07-05 09:14:07 +03:00
//common preconditions
2013-03-03 20:06:03 +03:00
if ( ( p - > resources [ Res : : GOLD ] < GOLD_NEEDED & & complain ( " Not enough gold for buying hero! " ) )
2012-03-13 15:47:47 +03:00
| | ( getHeroCount ( player , false ) > = GameConstants : : MAX_HEROES_PER_PLAYER & & complain ( " Cannot hire hero, only 8 wandering heroes are allowed! " ) ) )
2009-04-16 03:28:54 +03:00
return false ;
2010-07-09 02:03:27 +03:00
if ( t ) //tavern in town
{
2013-02-11 02:24:57 +03:00
if ( ( ! t - > hasBuilt ( BuildingID : : TAVERN ) & & complain ( " No tavern! " ) )
2011-07-05 14:31:26 +03:00
| | ( t - > visitingHero & & complain ( " There is visiting hero - no place! " ) ) )
2010-07-09 02:03:27 +03:00
return false ;
}
2012-09-23 21:01:04 +03:00
else if ( obj - > ID = = Obj : : TAVERN )
2010-07-09 02:03:27 +03:00
{
if ( getTile ( obj - > visitablePos ( ) ) - > visitableObjects . back ( ) ! = obj & & complain ( " Tavern entry must be unoccupied! " ) )
return false ;
}
2010-12-20 15:04:24 +02:00
const CGHeroInstance * nh = p - > availableHeroes [ hid ] ;
2012-10-01 21:25:43 +03:00
if ( ! nh )
{
complain ( " Hero is not available for hiring! " ) ;
return false ;
}
2009-03-09 12:37:49 +02:00
HeroRecruited hr ;
2010-07-09 02:03:27 +03:00
hr . tid = obj - > id ;
2009-03-09 12:37:49 +02:00
hr . hid = nh - > subID ;
2010-07-09 02:03:27 +03:00
hr . player = player ;
hr . tile = obj - > visitablePos ( ) + nh - > getVisitableOffset ( ) ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & hr ) ;
2009-12-01 23:05:57 +02:00
2010-12-20 15:04:24 +02:00
bmap < ui32 , ConstTransitivePtr < CGHeroInstance > > pool = gs - > unusedHeroesFromPool ( ) ;
2010-07-08 08:52:11 +03:00
const CGHeroInstance * theOtherHero = p - > availableHeroes [ ! hid ] ;
2013-06-26 14:18:27 +03:00
const CGHeroInstance * newHero = nullptr ;
2012-10-01 21:25:43 +03:00
if ( theOtherHero ) //on XXL maps all heroes can be imprisoned :(
newHero = gs - > hpool . pickHeroFor ( false , player , getNativeTown ( player ) , pool , theOtherHero - > type - > heroClass ) ;
2009-08-05 03:05:37 +03:00
2009-03-09 12:37:49 +02:00
SetAvailableHeroes sah ;
2010-07-09 02:03:27 +03:00
sah . player = player ;
2010-07-08 08:52:11 +03:00
if ( newHero )
{
sah . hid [ hid ] = newHero - > subID ;
2011-01-28 04:11:58 +02:00
sah . army [ hid ] . clear ( ) ;
2013-02-16 17:03:47 +03:00
sah . army [ hid ] . setCreature ( SlotID ( 0 ) , newHero - > type - > initialArmy [ 0 ] . creature , 1 ) ;
2010-07-08 08:52:11 +03:00
}
else
sah . hid [ hid ] = - 1 ;
sah . hid [ ! hid ] = theOtherHero ? theOtherHero - > subID : - 1 ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & sah ) ;
SetResource sr ;
2010-07-09 02:03:27 +03:00
sr . player = player ;
2011-07-05 09:14:07 +03:00
sr . resid = Res : : GOLD ;
2013-03-03 20:06:03 +03:00
sr . val = p - > resources [ Res : : GOLD ] - GOLD_NEEDED ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & sr ) ;
2009-03-20 20:51:48 +02:00
2010-07-09 02:03:27 +03:00
if ( t )
{
vistiCastleObjects ( t , nh ) ;
giveSpells ( t , nh ) ;
}
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2013-05-27 13:53:28 +03:00
bool CGameHandler : : queryReply ( QueryID qid , ui32 answer , PlayerColor player )
2009-03-09 12:37:49 +02:00
{
2009-04-12 04:48:50 +03:00
boost : : unique_lock < boost : : recursive_mutex > lock ( gsm ) ;
2013-04-20 14:34:01 +03:00
logGlobal - > traceStream ( ) < < boost : : format ( " Player %s attempts answering query %d with answer %d " ) % player % qid % answer ;
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! " ) ;
if ( auto dialogQuery = std : : dynamic_pointer_cast < CDialogQuery > ( topQuery ) )
dialogQuery - > answer = answer ;
queries . popQuery ( topQuery ) ;
2009-04-16 03:28:54 +03:00
return true ;
2009-03-09 12:37:49 +02:00
}
2011-06-11 07:54:41 +03:00
static EndAction end_action ;
2009-04-16 03:28:54 +03:00
bool CGameHandler : : makeBattleAction ( BattleAction & ba )
2009-03-09 12:37:49 +02:00
{
2009-04-16 03:28:54 +03:00
bool ok = true ;
2013-09-27 22:42:17 +03:00
2012-08-26 12:07:48 +03:00
const CStack * stack = battleGetStackByID ( ba . stackNumber ) ; //may be nullptr if action is not about stack
2013-09-27 22:42:17 +03:00
const CStack * destinationStack = ba . actionType = = Battle : : WALK_AND_ATTACK ? gs - > curB - > battleGetStackByPos ( ba . additionalInfo )
: ba . actionType = = Battle : : SHOOT ? gs - > curB - > battleGetStackByPos ( ba . destinationTile )
: nullptr ;
2013-01-12 21:08:33 +03:00
const bool isAboutActiveStack = stack & & ( stack = = battleActiveStack ( ) ) ;
2013-09-27 22:42:17 +03:00
logGlobal - > traceStream ( ) < < boost : : format (
" 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 " ) ) ;
2012-08-28 15:28:13 +03:00
2012-08-26 12:07:48 +03:00
switch ( ba . actionType )
{
2013-02-04 00:05:44 +03:00
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 :
2012-08-26 12:07:48 +03:00
if ( ! stack )
{
complain ( " No such stack! " ) ;
return false ;
}
if ( ! stack - > alive ( ) )
{
complain ( " This stack is dead: " + stack - > nodeName ( ) ) ;
return false ;
}
2012-08-28 15:28:13 +03:00
if ( battleTacticDist ( ) )
{
if ( stack & & ! stack - > attackerOwned ! = battleGetTacticsSide ( ) )
{
complain ( " This is not a stack of side that has tactics! " ) ;
return false ;
}
}
2013-01-12 21:08:33 +03:00
else if ( ! isAboutActiveStack )
2012-08-26 12:07:48 +03:00
{
complain ( " Action has to be about active stack! " ) ;
return false ;
}
}
2009-03-09 12:37:49 +02:00
switch ( ba . actionType )
{
2013-02-04 00:05:44 +03:00
case Battle : : END_TACTIC_PHASE : //wait
case Battle : : BAD_MORALE :
case Battle : : NO_ACTION :
2011-02-12 18:12:48 +02:00
{
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ;
sendAndApply ( & end_action ) ;
2011-02-12 18:12:48 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : WALK :
2009-03-09 12:37:49 +02:00
{
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ; //start movement
2012-08-26 22:13:57 +03:00
int walkedTiles = moveStack ( ba . stackNumber , ba . destinationTile ) ; //move
if ( ! walkedTiles )
complain ( " Stack failed movement! " ) ;
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-03-09 12:37:49 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : DEFEND :
2011-01-18 19:23:31 +02:00
{
2011-07-16 19:40:38 +03:00
//defensive stance //TODO: remove this bonus when stack becomes active
2011-01-18 19:23:31 +02:00
SetStackEffect sse ;
2011-01-20 21:57:12 +02:00
sse . effect . push_back ( Bonus ( Bonus : : STACK_GETS_TURN , Bonus : : PRIMARY_SKILL , Bonus : : OTHER , 20 , - 1 , PrimarySkill : : DEFENSE , Bonus : : PERCENT_TO_ALL ) ) ;
2012-07-23 13:23:43 +03:00
sse . effect . push_back ( Bonus ( Bonus : : STACK_GETS_TURN , Bonus : : PRIMARY_SKILL , Bonus : : OTHER , gs - > curB - > stacks [ ba . stackNumber ] - > valOfBonuses ( Bonus : : DEFENSIVE_STANCE ) ,
2011-07-16 19:40:38 +03:00
- 1 , PrimarySkill : : DEFENSE , Bonus : : ADDITIVE_VALUE ) ) ;
2011-01-18 19:23:31 +02:00
sse . stacks . push_back ( ba . stackNumber ) ;
sendAndApply ( & sse ) ;
//don't break - we share code with next case
}
2013-02-04 00:05:44 +03:00
case Battle : : WAIT :
2009-03-09 12:37:49 +02:00
{
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ;
sendAndApply ( & end_action ) ;
2009-03-09 12:37:49 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : RETREAT : //retreat/flee
2009-03-09 12:37:49 +02:00
{
2013-07-22 01:01:29 +03:00
if ( ! gs - > curB - > battleCanFlee ( gs - > curB - > sides [ ba . side ] . color ) )
2011-03-05 18:38:22 +02:00
complain ( " Cannot retreat! " ) ;
else
2013-02-04 22:43:16 +03:00
setBattleResult ( BattleResult : : ESCAPE , ! ba . side ) ; //surrendering side loses
2009-03-09 12:37:49 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : SURRENDER :
2011-03-05 18:38:22 +02:00
{
2013-07-22 01:01:29 +03:00
PlayerColor player = gs - > curB - > sides [ ba . side ] . color ;
2012-08-26 12:07:48 +03:00
int cost = gs - > curB - > battleGetSurrenderCost ( player ) ;
2011-03-05 18:38:22 +02:00
if ( cost < 0 )
complain ( " Cannot surrender! " ) ;
else if ( getResource ( player , Res : : GOLD ) < cost )
complain ( " Not enough gold to surrender! " ) ;
else
{
giveResource ( player , Res : : GOLD , - cost ) ;
2013-02-04 22:43:16 +03:00
setBattleResult ( BattleResult : : SURRENDER , ! ba . side ) ; //surrendering side loses
2011-03-05 18:38:22 +02:00
}
break ;
}
break ;
2013-02-04 00:05:44 +03:00
case Battle : : WALK_AND_ATTACK : //walk or attack
2009-03-09 12:37:49 +02:00
{
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ; //start movement and attack
2009-03-09 12:37:49 +02:00
2013-09-27 22:42:17 +03:00
if ( ! stack | | ! destinationStack )
2012-05-18 23:50:16 +03:00
{
sendAndApply ( & end_action ) ;
break ;
}
2013-01-12 21:08:33 +03:00
2012-09-18 18:06:29 +03:00
BattleHex startingPos = stack - > position ;
2012-09-16 17:36:31 +03:00
int distance = moveStack ( ba . stackNumber , ba . destinationTile ) ;
2013-01-12 21:08:33 +03:00
2013-09-27 22:42:17 +03:00
logGlobal - > traceStream ( ) < < stack - > nodeName ( ) < < " will attack " < < destinationStack - > nodeName ( ) ;
2012-08-26 12:07:48 +03:00
2012-08-26 22:13:57 +03:00
if ( stack - > position ! = ba . destinationTile //we wasn't able to reach destination tile
& & ! ( stack - > doubleWide ( )
& & ( stack - > position = = ba . destinationTile + ( stack - > attackerOwned ? + 1 : - 1 ) )
2009-08-18 14:49:34 +03:00
) //nor occupy specified hex
2012-07-23 13:23:43 +03:00
)
2009-03-09 12:37:49 +02:00
{
2012-08-26 22:13:57 +03:00
std : : string problem = " We cannot move this stack to its destination " + stack - > getCreature ( ) - > namePl ;
2013-04-10 20:18:01 +03:00
logGlobal - > warnStream ( ) < < problem ;
2009-08-03 17:29:29 +03:00
complain ( problem ) ;
2009-04-16 03:28:54 +03:00
ok = false ;
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-08-03 17:29:29 +03:00
break ;
2009-03-09 12:37:49 +02:00
}
2013-09-27 22:42:17 +03:00
if ( destinationStack & & stack - > ID = = destinationStack - > ID ) //we should just move, it will be handled by following check
2009-08-02 17:21:18 +03:00
{
2013-09-27 22:42:17 +03:00
destinationStack = nullptr ;
2009-08-02 17:21:18 +03:00
}
2013-09-27 22:42:17 +03:00
if ( ! destinationStack )
2009-03-09 12:37:49 +02:00
{
2011-01-08 20:33:40 +02:00
complain ( boost : : str ( boost : : format ( " walk and attack error: no stack at additionalInfo tile (%d)! \n " ) % ba . additionalInfo ) ) ;
2009-04-16 03:28:54 +03:00
ok = false ;
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-03-09 12:37:49 +02:00
break ;
}
2013-09-27 22:42:17 +03:00
if ( ! CStack : : isMeleeAttackPossible ( stack , destinationStack ) )
2009-03-09 12:37:49 +02:00
{
2011-01-15 04:17:56 +02:00
complain ( " Attack cannot be performed! " ) ;
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-04-16 03:28:54 +03:00
ok = false ;
2011-01-15 04:17:56 +02:00
break ;
2009-03-09 12:37:49 +02:00
}
//attack
2013-02-08 12:22:10 +03:00
int totalAttacks = 1 + stack - > getBonuses ( Selector : : type ( Bonus : : ADDITIONAL_ATTACK ) ,
2013-07-02 15:08:30 +03:00
( Selector : : effectRange ( Bonus : : NO_LIMIT ) . Or ( Selector : : effectRange ( Bonus : : ONLY_MELEE_FIGHT ) ) ) ) - > totalValue ( ) ; //all unspicified attacks + melee attacks
2009-03-09 12:37:49 +02:00
2013-02-08 12:22:10 +03:00
for ( int i = 0 ; i < totalAttacks ; + + i )
2009-03-09 12:37:49 +02:00
{
2013-02-19 17:33:30 +03:00
if ( stack & &
stack - > alive ( ) & & //move can cause death, eg. by walking into the moat
2013-09-27 22:42:17 +03:00
destinationStack - > alive ( ) )
2013-02-08 12:22:10 +03:00
{
BattleAttack bat ;
2013-09-27 22:42:17 +03:00
prepareAttack ( bat , stack , destinationStack , ( i ? 0 : distance ) , ba . additionalInfo ) ; //no distance travelled on second attack
2013-02-19 17:33:30 +03:00
//prepareAttack(bat, stack, stackAtEnd, 0, ba.additionalInfo);
2013-02-08 12:22:10 +03:00
handleAttackBeforeCasting ( bat ) ; //only before first attack
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
//counterattack
2013-09-27 22:42:17 +03:00
if ( destinationStack
2013-09-08 20:43:10 +03:00
& & ! stack - > hasBonusOfType ( Bonus : : BLOCKS_RETALIATION )
2013-09-27 22:42:17 +03:00
& & destinationStack - > ableToRetaliate ( )
2013-02-08 12:22:10 +03:00
& & stack - > alive ( ) ) //attacker may have died (fire shield)
{
BattleAttack bat ;
2013-09-27 22:42:17 +03:00
prepareAttack ( bat , destinationStack , stack , 0 , stack - > position ) ;
2013-02-08 12:22:10 +03:00
bat . flags | = BattleAttack : : COUNTER ;
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
2009-03-09 12:37:49 +02:00
}
2009-09-16 13:59:56 +03:00
//return
2012-08-26 22:13:57 +03:00
if ( stack - > hasBonusOfType ( Bonus : : RETURN_AFTER_STRIKE ) & & startingPos ! = stack - > position & & stack - > alive ( ) )
2009-09-16 13:59:56 +03:00
{
moveStack ( ba . stackNumber , startingPos ) ;
2010-01-29 20:17:07 +02:00
//NOTE: curStack->ID == ba.stackNumber (rev 1431)
2009-09-16 13:59:56 +03:00
}
2013-09-08 20:43:10 +03:00
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-03-09 12:37:49 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : SHOOT :
2009-03-09 12:37:49 +02:00
{
2012-08-26 22:13:57 +03:00
if ( ! gs - > curB - > battleCanShoot ( stack , ba . destinationTile ) )
{
complain ( " Cannot shoot! " ) ;
2009-05-16 17:49:06 +03:00
break ;
2012-08-26 22:13:57 +03:00
}
2009-03-09 12:37:49 +02:00
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ; //start shooting
2009-03-09 12:37:49 +02:00
2011-09-01 04:40:46 +03:00
{
BattleAttack bat ;
bat . flags | = BattleAttack : : SHOT ;
2013-09-27 22:42:17 +03:00
prepareAttack ( bat , stack , destinationStack , 0 , ba . destinationTile ) ;
2011-09-01 04:40:46 +03:00
handleAttackBeforeCasting ( bat ) ;
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
2009-03-09 12:37:49 +02:00
2013-07-06 19:10:20 +03:00
//second shot for ballista, only if hero has advanced artillery
2013-07-22 01:01:29 +03:00
const CGHeroInstance * attackingHero = gs - > curB - > battleGetFightingHero ( ba . side ) ;
2013-07-06 19:10:20 +03:00
2013-09-27 22:42:17 +03:00
if ( destinationStack - > alive ( )
2013-07-06 19:10:20 +03:00
& & ( stack - > getCreature ( ) - > idNumber = = CreatureID : : BALLISTA )
& & ( attackingHero - > getSecSkillLevel ( SecondarySkill : : ARTILLERY ) > = SecSkillLevel : : ADVANCED )
)
2011-02-23 19:10:35 +02:00
{
2011-02-24 17:33:03 +02:00
BattleAttack bat2 ;
bat2 . flags | = BattleAttack : : SHOT ;
2013-09-27 22:42:17 +03:00
prepareAttack ( bat2 , stack , destinationStack , 0 , ba . destinationTile ) ;
2011-02-24 17:33:03 +02:00
sendAndApply ( & bat2 ) ;
2011-02-23 19:10:35 +02:00
}
2013-02-08 12:22:10 +03:00
//allow more than one additional attack
int additionalAttacks = stack - > getBonuses ( Selector : : type ( Bonus : : ADDITIONAL_ATTACK ) ,
2013-07-02 15:08:30 +03:00
( Selector : : effectRange ( Bonus : : NO_LIMIT ) . Or ( Selector : : effectRange ( Bonus : : ONLY_DISTANCE_FIGHT ) ) ) ) - > totalValue ( ) ;
2013-02-08 12:22:10 +03:00
for ( int i = 0 ; i < additionalAttacks ; + + i )
2009-03-09 12:37:49 +02:00
{
2013-02-08 12:22:10 +03:00
if (
stack - > alive ( )
2013-09-27 22:42:17 +03:00
& & destinationStack - > alive ( )
2013-02-08 12:22:10 +03:00
& & stack - > shots
)
{
BattleAttack bat ;
bat . flags | = BattleAttack : : SHOT ;
2013-09-27 22:42:17 +03:00
prepareAttack ( bat , stack , destinationStack , 0 , ba . destinationTile ) ;
2013-02-08 12:22:10 +03:00
sendAndApply ( & bat ) ;
handleAfterAttackCasting ( bat ) ;
}
2009-03-09 12:37:49 +02:00
}
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-03-09 12:37:49 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : CATAPULT :
2009-09-01 16:54:13 +03:00
{
2013-08-06 14:20:28 +03:00
auto getCatapultHitChance = [ & ] ( EWallParts : : EWallParts part , const CHeroHandler : : SBallisticsLevelInfo & sbi ) - > int
{
switch ( part )
{
case EWallParts : : GATE :
return sbi . gate ;
case EWallParts : : KEEP :
return sbi . keep ;
case EWallParts : : BOTTOM_TOWER :
case EWallParts : : UPPER_TOWER :
return sbi . tower ;
case EWallParts : : BOTTOM_WALL :
case EWallParts : : BELOW_GATE :
case EWallParts : : OVER_GATE :
case EWallParts : : UPPER_WALL :
return sbi . wall ;
default :
return 0 ;
}
} ;
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ;
2013-08-19 14:50:53 +03:00
auto onExit = vstd : : makeScopeGuard ( [ & ] { sendAndApply ( & end_action ) ; } ) ; //if we started than we have to finish
2013-08-18 20:49:31 +03:00
2013-07-22 01:01:29 +03:00
const CGHeroInstance * attackingHero = gs - > curB - > battleGetFightingHero ( ba . side ) ;
2013-02-04 22:43:16 +03:00
CHeroHandler : : SBallisticsLevelInfo sbi = VLC - > heroh - > ballistics [ attackingHero - > getSecSkillLevel ( SecondarySkill : : BALLISTICS ) ] ;
2012-07-23 13:23:43 +03:00
2013-08-06 14:20:28 +03:00
EWallParts : : EWallParts desiredTarget = gs - > curB - > battleHexToWallPart ( ba . destinationTile ) ;
if ( desiredTarget < 0 )
2009-09-01 16:54:13 +03:00
{
complain ( " catapult tried to attack non-catapultable hex! " ) ;
break ;
}
2013-08-06 14:20:28 +03:00
//in successive iterations damage is dealt but not yet subtracted from wall's HPs
2013-08-18 20:49:31 +03:00
auto & currentHP = gs - > curB - > si . wallState ;
2013-08-06 14:20:28 +03:00
2013-08-18 20:49:31 +03:00
if ( currentHP [ desiredTarget ] = = EWallState : : DESTROYED | | currentHP [ desiredTarget ] = = EWallState : : NONE )
{
complain ( " catapult tried to attack already destroyed wall part! " ) ;
break ;
}
2013-08-06 14:20:28 +03:00
2009-09-01 16:54:13 +03:00
for ( int g = 0 ; g < sbi . shots ; + + g )
{
2013-08-06 14:20:28 +03:00
bool hitSuccessfull = false ;
EWallParts : : EWallParts attackedPart = desiredTarget ;
do // catapult has chance to attack desired target. Othervice - attacks randomly
{
if ( currentHP [ attackedPart ] ! = EWallState : : DESTROYED & & // this part can be hit
currentHP [ attackedPart ] ! = EWallState : : NONE & &
rand ( ) % 100 < getCatapultHitChance ( attackedPart , sbi ) ) //hit is successful
{
hitSuccessfull = true ;
}
else // select new target
{
std : : vector < EWallParts : : EWallParts > allowedTargets ;
for ( size_t i = 0 ; i < currentHP . size ( ) ; i + + )
{
if ( currentHP [ i ] ! = EWallState : : DESTROYED & &
currentHP [ i ] ! = EWallState : : NONE )
allowedTargets . push_back ( EWallParts : : EWallParts ( i ) ) ;
}
if ( allowedTargets . empty ( ) )
break ;
attackedPart = allowedTargets [ rand ( ) % allowedTargets . size ( ) ] ;
}
}
while ( ! hitSuccessfull ) ;
if ( ! hitSuccessfull ) // break triggered - no target to shoot at
break ;
2012-07-23 13:23:43 +03:00
2009-09-01 16:54:13 +03:00
CatapultAttack ca ; //package for clients
2013-08-06 14:20:28 +03:00
CatapultAttack : : AttackInfo attack ;
attack . attackedPart = attackedPart ;
attack . destinationTile = ba . destinationTile ;
attack . damageDealt = 0 ;
2009-09-01 16:54:13 +03:00
2011-12-14 00:23:17 +03:00
int dmgChance [ ] = { sbi . noDmg , sbi . oneDmg , sbi . twoDmg } ; //dmgChance[i] - chance for doing i dmg when hit is successful
2013-08-06 14:20:28 +03:00
int dmgRand = rand ( ) % 100 ;
//accumulating dmgChance
dmgChance [ 1 ] + = dmgChance [ 0 ] ;
dmgChance [ 2 ] + = dmgChance [ 1 ] ;
//calculating dealt damage
for ( int damage = 0 ; damage < ARRAY_COUNT ( dmgChance ) ; + + damage )
2009-09-01 16:54:13 +03:00
{
2013-08-06 14:20:28 +03:00
if ( dmgRand < = dmgChance [ damage ] )
{
currentHP [ attackedPart ] = SiegeInfo : : applyDamage ( EWallState : : EWallState ( currentHP [ attackedPart ] ) , damage ) ;
attack . damageDealt = damage ;
break ;
}
2009-09-01 16:54:13 +03:00
}
2013-08-06 14:20:28 +03:00
// attacked tile may have changed - update destination
attack . destinationTile = gs - > curB - > wallPartToBattleHex ( EWallParts : : EWallParts ( attack . attackedPart ) ) ;
2009-09-01 16:54:13 +03:00
2013-08-06 14:20:28 +03:00
logGlobal - > traceStream ( ) < < " Catapult attacks " < < ( int ) attack . attackedPart
< < " dealing " < < ( int ) attack . damageDealt < < " damage " ;
//removing creatures in turrets / keep if one is destroyed
if ( attack . damageDealt > 0 & & ( attackedPart = = EWallParts : : KEEP | |
attackedPart = = EWallParts : : BOTTOM_TOWER | | attackedPart = = EWallParts : : UPPER_TOWER ) )
2009-09-01 16:54:13 +03:00
{
2013-08-06 14:20:28 +03:00
int posRemove = - 1 ;
switch ( attackedPart )
2009-09-01 16:54:13 +03:00
{
2013-08-06 14:20:28 +03:00
case EWallParts : : KEEP :
posRemove = - 2 ;
break ;
case EWallParts : : BOTTOM_TOWER :
posRemove = - 3 ;
break ;
case EWallParts : : UPPER_TOWER :
posRemove = - 4 ;
break ;
2009-09-01 16:54:13 +03:00
}
2010-01-28 19:23:01 +02:00
2013-08-06 14:20:28 +03:00
BattleStacksRemoved bsr ;
for ( auto & elem : gs - > curB - > stacks )
2009-09-05 17:10:26 +03:00
{
2013-08-06 14:20:28 +03:00
if ( elem - > position = = posRemove )
2009-09-05 17:10:26 +03:00
{
2013-08-06 14:20:28 +03:00
bsr . stackIDs . insert ( elem - > ID ) ;
2009-09-05 17:10:26 +03:00
break ;
}
}
2013-08-06 14:20:28 +03:00
sendAndApply ( & bsr ) ;
2009-09-01 16:54:13 +03:00
}
2009-09-24 16:44:55 +03:00
ca . attacker = ba . stackNumber ;
2013-08-06 14:20:28 +03:00
ca . attackedParts . push_back ( attack ) ;
2009-09-01 16:54:13 +03:00
sendAndApply ( & ca ) ;
}
2013-08-18 20:49:31 +03:00
//finish by scope guard
2010-03-07 19:40:33 +02:00
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : STACK_HEAL : //healing with First Aid Tent
2010-03-07 19:40:33 +02:00
{
2011-06-11 07:54:41 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ;
2013-07-22 01:01:29 +03:00
const CGHeroInstance * attackingHero = gs - > curB - > battleGetFightingHero ( ba . side ) ;
2012-08-26 12:07:48 +03:00
const CStack * healer = gs - > curB - > battleGetStackByID ( ba . stackNumber ) ,
* destStack = gs - > curB - > battleGetStackByPos ( ba . destinationTile ) ;
2010-03-07 19:40:33 +02:00
2013-06-26 14:18:27 +03:00
if ( healer = = nullptr | | destStack = = nullptr | | ! healer - > hasBonusOfType ( Bonus : : HEALER ) )
2010-03-07 19:40:33 +02:00
{
complain ( " There is either no healer, no destination, or healer cannot heal :P " ) ;
}
int maxHealable = destStack - > MaxHealth ( ) - destStack - > firstHPleft ;
2013-02-04 22:43:16 +03:00
int maxiumHeal = healer - > count * std : : max ( 10 , attackingHero - > valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : FIRST_AID ) ) ;
2010-03-07 19:40:33 +02:00
int healed = std : : min ( maxHealable , maxiumHeal ) ;
if ( healed = = 0 )
{
//nothing to heal.. should we complain?
}
else
{
StacksHealedOrResurrected shr ;
2013-02-12 22:49:40 +03:00
shr . lifeDrain = false ;
shr . tentHealing = true ;
2011-06-21 15:45:57 +03:00
shr . drainedFrom = ba . stackNumber ;
2010-03-07 19:40:33 +02:00
2012-06-10 16:07:08 +03:00
StacksHealedOrResurrected : : HealInfo hi ;
2010-03-07 19:40:33 +02:00
hi . healedHP = healed ;
hi . lowLevelResurrection = 0 ;
hi . stackID = destStack - > ID ;
shr . healedStacks . push_back ( hi ) ;
sendAndApply ( & shr ) ;
}
2011-10-01 22:56:54 +03:00
sendAndApply ( & end_action ) ;
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : DAEMON_SUMMONING :
2011-10-01 22:56:54 +03:00
//TODO: From Strategija:
//Summon Demon is a level 2 spell.
{
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ;
2012-08-26 12:07:48 +03:00
const CStack * summoner = gs - > curB - > battleGetStackByID ( ba . stackNumber ) ,
* destStack = gs - > curB - > battleGetStackByPos ( ba . destinationTile , false ) ;
2011-10-01 22:56:54 +03:00
BattleStackAdded bsa ;
bsa . attacker = summoner - > attackerOwned ;
2013-02-11 02:24:57 +03:00
bsa . creID = CreatureID ( summoner - > getBonusLocalFirst ( Selector : : type ( Bonus : : DAEMON_SUMMONING ) ) - > subtype ) ; //in case summoner can summon more than one type of monsters... scream!
ui64 risedHp = summoner - > count * summoner - > valOfBonuses ( Bonus : : DAEMON_SUMMONING , bsa . creID . toEnum ( ) ) ;
2013-07-21 21:27:33 +03:00
ui64 targetHealth = destStack - > getCreature ( ) - > MaxHealth ( ) * destStack - > baseAmount ;
ui64 canRiseHp = std : : min ( targetHealth , risedHp ) ;
ui32 canRiseAmount = canRiseHp / VLC - > creh - > creatures [ bsa . creID ] - > MaxHealth ( ) ;
bsa . amount = std : : min ( canRiseAmount , destStack - > baseAmount ) ;
2011-10-01 22:56:54 +03:00
bsa . pos = gs - > curB - > getAvaliableHex ( bsa . creID , bsa . attacker , destStack - > position ) ;
bsa . summoned = false ;
2011-10-02 10:16:34 +03:00
if ( bsa . amount ) //there's rare possibility single creature cannot rise desired type
{
BattleStacksRemoved bsr ; //remove body
bsr . stackIDs . insert ( destStack - > ID ) ;
sendAndApply ( & bsr ) ;
sendAndApply ( & bsa ) ;
2011-10-02 16:59:12 +03:00
BattleSetStackProperty ssp ;
ssp . stackID = ba . stackNumber ;
2011-10-08 12:11:36 +03:00
ssp . which = BattleSetStackProperty : : CASTS ; //reduce number of casts
2011-10-02 16:59:12 +03:00
ssp . val = - 1 ;
ssp . absolute = false ;
sendAndApply ( & ssp ) ;
2011-10-02 10:16:34 +03:00
}
2011-10-01 22:56:54 +03:00
2011-10-08 12:11:36 +03:00
sendAndApply ( & end_action ) ;
break ;
}
2013-02-04 00:05:44 +03:00
case Battle : : MONSTER_SPELL :
2011-10-08 12:11:36 +03:00
{
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ;
2012-08-26 12:07:48 +03:00
const CStack * stack = gs - > curB - > battleGetStackByID ( ba . stackNumber ) ;
2013-02-14 02:55:42 +03:00
SpellID spellID = SpellID ( ba . additionalInfo ) ;
2011-12-22 16:05:19 +03:00
BattleHex destination ( ba . destinationTile ) ;
2011-10-08 12:11:36 +03:00
2012-09-20 19:55:21 +03:00
const Bonus * randSpellcaster = stack - > getBonusLocalFirst ( Selector : : type ( Bonus : : RANDOM_SPELLCASTER ) ) ;
const Bonus * spellcaster = stack - > getBonusLocalFirst ( Selector : : typeSubtype ( Bonus : : SPELLCASTER , spellID ) ) ;
2012-03-31 00:36:07 +03:00
//TODO special bonus for genies ability
if ( randSpellcaster & & battleGetRandomStackSpell ( stack , CBattleInfoCallback : : RANDOM_AIMED ) < 0 )
spellID = battleGetRandomStackSpell ( stack , CBattleInfoCallback : : RANDOM_GENIE ) ;
if ( spellID < 0 )
complain ( " That stack can't cast spells! " ) ;
else
{
int spellLvl = 0 ;
if ( spellcaster )
vstd : : amax ( spellLvl , spellcaster - > val ) ;
if ( randSpellcaster )
vstd : : amax ( spellLvl , randSpellcaster - > val ) ;
vstd : : amin ( spellLvl , 3 ) ;
int casterSide = gs - > curB - > whatSide ( stack - > owner ) ;
const CGHeroInstance * secHero = gs - > curB - > getHero ( gs - > curB - > theOtherPlayer ( stack - > owner ) ) ;
2013-06-26 14:18:27 +03:00
handleSpellCasting ( spellID , spellLvl , destination , casterSide , stack - > owner , nullptr , secHero , 0 , ECastingMode : : CREATURE_ACTIVE_CASTING , stack ) ;
2012-03-31 00:36:07 +03:00
}
2011-06-11 07:54:41 +03:00
sendAndApply ( & end_action ) ;
2009-09-01 16:54:13 +03:00
break ;
}
2009-03-09 12:37:49 +02:00
}
2010-02-02 01:30:03 +02:00
if ( ba . stackNumber = = gs - > curB - > activeStack | | battleResult . get ( ) ) //active stack has moved or battle has finished
2010-02-01 21:19:42 +02:00
battleMadeAction . setn ( true ) ;
2009-04-16 03:28:54 +03:00
return ok ;
2009-03-09 12:37:49 +02:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : playerMessage ( PlayerColor player , const std : : string & message )
2009-03-09 12:37:49 +02:00
{
bool cheated = true ;
2013-03-03 20:06:03 +03:00
PlayerMessage temp_message ( player , message ) ;
2011-06-11 07:54:41 +03:00
sendAndApply ( & temp_message ) ;
2009-03-09 12:37:49 +02:00
if ( message = = " vcmiistari " ) //give all spells and 999 mana
{
SetMana sm ;
ChangeSpells cs ;
2009-08-04 02:53:18 +03:00
CGHeroInstance * h = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
if ( ! h & & complain ( " Cannot realize cheat, no hero selected! " ) ) return ;
sm . hid = cs . hid = h - > id ;
//give all spells
2009-03-09 12:37:49 +02:00
cs . learn = 1 ;
2013-06-29 16:05:48 +03:00
for ( auto spell : VLC - > spellh - > spells )
2009-03-09 12:37:49 +02:00
{
2013-02-11 02:24:57 +03:00
if ( ! spell - > creatureAbility )
cs . spells . insert ( spell - > id ) ;
2009-03-09 12:37:49 +02:00
}
2009-08-04 02:53:18 +03:00
//give mana
2009-03-09 12:37:49 +02:00
sm . val = 999 ;
2009-08-04 02:53:18 +03:00
2011-01-18 20:56:14 +02:00
if ( ! h - > hasSpellbook ( ) ) //hero doesn't have spellbook
2011-12-14 00:23:17 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ 0 ] , ArtifactPosition : : SPELLBOOK ) ; //give spellbook
2009-08-04 02:53:18 +03:00
sendAndApply ( & cs ) ;
sendAndApply ( & sm ) ;
2009-03-09 12:37:49 +02:00
}
2012-02-16 20:10:58 +03:00
else if ( message = = " vcmiarmenelos " ) //build all buildings in selected town
{
2013-07-08 23:55:22 +03:00
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
CGTownInstance * town ;
if ( hero )
town = hero - > visitedTown ;
else
town = gs - > getTown ( gs - > getPlayer ( player ) - > currentSelection ) ;
2012-02-16 20:10:58 +03:00
if ( town )
{
2013-06-29 16:05:48 +03:00
for ( auto & build : town - > town - > buildings )
2012-02-16 20:10:58 +03:00
{
2013-07-08 23:55:22 +03:00
if ( ! town - > hasBuilt ( build . first )
& & ! build . second - > Name ( ) . empty ( )
& & build . first ! = BuildingID : : SHIP )
2012-02-16 20:10:58 +03:00
{
buildStructure ( town - > id , build . first , true ) ;
}
}
}
}
2009-03-09 12:37:49 +02:00
else if ( message = = " vcmiainur " ) //gives 5 archangels into each slot
{
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
2010-11-27 22:17:28 +02:00
const CCreature * archangel = VLC - > creh - > creatures [ 13 ] ;
2009-03-09 12:37:49 +02:00
if ( ! hero ) return ;
2010-05-02 21:20:26 +03:00
2011-12-14 00:23:17 +03:00
for ( int i = 0 ; i < GameConstants : : ARMY_SIZE ; i + + )
2013-02-16 17:03:47 +03:00
if ( ! hero - > hasStackAtSlot ( SlotID ( i ) ) )
insertNewStack ( StackLocation ( hero , SlotID ( i ) ) , archangel , 5 ) ;
2009-03-09 12:37:49 +02:00
}
2009-08-04 02:53:18 +03:00
else if ( message = = " vcmiangband " ) //gives 10 black knight into each slot
2009-03-09 12:37:49 +02:00
{
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
2010-11-27 22:17:28 +02:00
const CCreature * blackKnight = VLC - > creh - > creatures [ 66 ] ;
2009-03-09 12:37:49 +02:00
if ( ! hero ) return ;
2010-05-02 21:20:26 +03:00
2011-12-14 00:23:17 +03:00
for ( int i = 0 ; i < GameConstants : : ARMY_SIZE ; i + + )
2013-02-16 17:03:47 +03:00
if ( ! hero - > hasStackAtSlot ( SlotID ( i ) ) )
insertNewStack ( StackLocation ( hero , SlotID ( i ) ) , blackKnight , 10 ) ;
2009-03-09 12:37:49 +02:00
}
else if ( message = = " vcminoldor " ) //all war machines
{
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
if ( ! hero ) return ;
2011-01-18 20:56:14 +02:00
2011-12-14 00:23:17 +03:00
if ( ! hero - > getArt ( ArtifactPosition : : MACH1 ) )
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ 4 ] , ArtifactPosition : : MACH1 ) ;
if ( ! hero - > getArt ( ArtifactPosition : : MACH2 ) )
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ 5 ] , ArtifactPosition : : MACH2 ) ;
if ( ! hero - > getArt ( ArtifactPosition : : MACH3 ) )
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ 6 ] , ArtifactPosition : : MACH3 ) ;
2009-03-09 12:37:49 +02:00
}
else if ( message = = " vcminahar " ) //1000000 movement points
{
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
if ( ! hero ) return ;
SetMovePoints smp ;
smp . hid = hero - > id ;
smp . val = 1000000 ;
sendAndApply ( & smp ) ;
}
else if ( message = = " vcmiformenos " ) //give resources
{
SetResources sr ;
sr . player = player ;
sr . res = gs - > getPlayer ( player ) - > resources ;
2012-10-07 17:58:48 +03:00
for ( int i = 0 ; i < Res : : GOLD ; i + + )
2009-03-09 12:37:49 +02:00
sr . res [ i ] + = 100 ;
2012-10-07 17:58:48 +03:00
sr . res [ Res : : GOLD ] + = 100000 ; //100k
2009-03-09 12:37:49 +02:00
sendAndApply ( & sr ) ;
}
else if ( message = = " vcmieagles " ) //reveal FoW
{
FoWChange fc ;
2009-08-04 02:53:18 +03:00
fc . mode = 1 ;
2009-03-09 12:37:49 +02:00
fc . player = player ;
2013-06-29 16:05:48 +03:00
auto hlp_tab = new int3 [ gs - > map - > width * gs - > map - > height * ( gs - > map - > twoLevel ? 2 : 1 ) ] ;
2011-01-20 19:25:15 +02:00
int lastUnc = 0 ;
2009-03-09 12:37:49 +02:00
for ( int i = 0 ; i < gs - > map - > width ; i + + )
for ( int j = 0 ; j < gs - > map - > height ; j + + )
2012-11-20 20:53:45 +03:00
for ( int k = 0 ; k < ( gs - > map - > twoLevel ? 2 : 1 ) ; k + + )
2010-08-03 15:34:06 +03:00
if ( ! gs - > getPlayerTeam ( fc . player ) - > fogOfWarMap [ i ] [ j ] [ k ] )
2011-01-20 19:25:15 +02:00
hlp_tab [ lastUnc + + ] = int3 ( i , j , k ) ;
fc . tiles . insert ( hlp_tab , hlp_tab + lastUnc ) ;
delete [ ] hlp_tab ;
2009-03-09 12:37:49 +02:00
sendAndApply ( & fc ) ;
}
2010-07-30 14:29:42 +03:00
else if ( message = = " vcmiglorfindel " ) //selected hero gains a new level
2009-03-09 12:37:49 +02:00
{
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
2013-02-09 00:17:39 +03:00
changePrimSkill ( hero , PrimarySkill : : EXPERIENCE , VLC - > heroh - > reqExp ( hero - > level + 1 ) - VLC - > heroh - > reqExp ( hero - > level ) ) ;
2009-03-09 12:37:49 +02:00
}
2010-07-30 14:29:42 +03:00
else if ( message = = " vcmisilmaril " ) //player wins
{
gs - > getPlayer ( player ) - > enteredWinningCheatCode = 1 ;
}
else if ( message = = " vcmimelkor " ) //player looses
{
gs - > getPlayer ( player ) - > enteredLosingCheatCode = 1 ;
}
2010-08-05 12:47:12 +03:00
else if ( message = = " vcmiforgeofnoldorking " ) //hero gets all artifacts except war machines, spell scrolls and spell book
{
CGHeroInstance * hero = gs - > getHero ( gs - > getPlayer ( player ) - > currentSelection ) ;
if ( ! hero ) return ;
2013-05-25 10:06:00 +03:00
for ( int g = 7 ; g < VLC - > arth - > artifacts . size ( ) ; + + g ) //including artifacts from mods
2013-02-07 02:24:43 +03:00
giveHeroNewArtifact ( hero , VLC - > arth - > artifacts [ g ] , ArtifactPosition : : PRE_FIRST ) ;
2010-08-05 12:47:12 +03:00
}
2009-03-09 12:37:49 +02:00
else
cheated = false ;
if ( cheated )
{
2011-06-11 07:54:41 +03:00
SystemMessage temp_message ( VLC - > generaltexth - > allTexts [ 260 ] ) ;
sendAndApply ( & temp_message ) ;
2012-02-16 20:10:58 +03:00
checkLossVictory ( player ) ; //Player enter win code or got required art\creature
2009-03-09 12:37:49 +02:00
}
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : handleSpellCasting ( SpellID spellID , int spellLvl , BattleHex destination , ui8 casterSide , PlayerColor casterColor , const CGHeroInstance * caster , const CGHeroInstance * secHero ,
2012-04-28 22:40:27 +03:00
int usedSpellPower , ECastingMode : : ECastingMode mode , const CStack * stack , si32 selectedStack )
2010-02-23 17:39:31 +02:00
{
2013-02-13 22:35:43 +03:00
const CSpell * spell = SpellID ( spellID ) . toSpell ( ) ;
2010-02-23 17:39:31 +02:00
2012-05-18 23:50:16 +03:00
//Helper local function that creates obstacle on given position. Obstacle type is inferred from spell type.
//It creates, sends and applies needed package.
auto placeObstacle = [ & ] ( BattleHex pos )
{
2012-07-23 13:23:43 +03:00
static int obstacleIdToGive = gs - > curB - > obstacles . size ( )
? ( gs - > curB - > obstacles . back ( ) - > uniqueID + 1 )
2012-05-18 23:50:16 +03:00
: 0 ;
auto obstacle = make_shared < SpellCreatedObstacle > ( ) ;
2013-02-14 02:55:42 +03:00
switch ( spellID . toEnum ( ) ) // :/
2012-05-18 23:50:16 +03:00
{
2013-02-11 02:24:57 +03:00
case SpellID : : QUICKSAND :
2012-05-18 23:50:16 +03:00
obstacle - > obstacleType = CObstacleInstance : : QUICKSAND ;
obstacle - > turnsRemaining = - 1 ;
obstacle - > visibleForAnotherSide = false ;
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : LAND_MINE :
2012-05-18 23:50:16 +03:00
obstacle - > obstacleType = CObstacleInstance : : LAND_MINE ;
obstacle - > turnsRemaining = - 1 ;
obstacle - > visibleForAnotherSide = false ;
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : FIRE_WALL :
2012-05-18 23:50:16 +03:00
obstacle - > obstacleType = CObstacleInstance : : FIRE_WALL ;
obstacle - > turnsRemaining = 2 ;
obstacle - > visibleForAnotherSide = true ;
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : FORCE_FIELD :
2012-05-18 23:50:16 +03:00
obstacle - > obstacleType = CObstacleInstance : : FORCE_FIELD ;
obstacle - > turnsRemaining = 2 ;
obstacle - > visibleForAnotherSide = true ;
break ;
default :
//this function cannot be used with spells that do not create obstacles
assert ( 0 ) ;
}
obstacle - > pos = pos ;
obstacle - > casterSide = casterSide ;
obstacle - > ID = spellID ;
obstacle - > spellLevel = spellLvl ;
obstacle - > casterSpellPower = usedSpellPower ;
obstacle - > uniqueID = obstacleIdToGive + + ;
BattleObstaclePlaced bop ;
bop . obstacle = obstacle ;
sendAndApply ( & bop ) ;
} ;
2010-05-16 16:42:19 +03:00
BattleSpellCast sc ;
2010-02-23 17:39:31 +02:00
sc . side = casterSide ;
sc . id = spellID ;
sc . skill = spellLvl ;
sc . tile = destination ;
sc . dmgToDisplay = 0 ;
sc . castedByHero = ( bool ) caster ;
2013-02-11 23:34:31 +03:00
sc . attackerType = ( stack ? stack - > type - > idNumber : CreatureID ( CreatureID : : NONE ) ) ;
2011-10-09 14:23:24 +03:00
sc . manaGained = 0 ;
sc . spellCost = 0 ;
if ( caster ) //calculate spell cost
{
2013-02-13 22:35:43 +03:00
sc . spellCost = gs - > curB - > battleGetSpellCost ( SpellID ( spellID ) . toSpell ( ) , caster ) ;
2011-10-09 14:23:24 +03:00
2011-12-14 00:23:17 +03:00
if ( secHero & & mode = = ECastingMode : : HERO_CASTING ) //handle mana channel
2011-10-09 14:23:24 +03:00
{
int manaChannel = 0 ;
2013-06-29 16:05:48 +03:00
for ( CStack * stack : gs - > curB - > stacks ) //TODO: shouldn't bonus system handle it somehow?
2011-10-09 14:23:24 +03:00
{
if ( stack - > owner = = secHero - > tempOwner )
{
2011-12-14 00:23:17 +03:00
vstd : : amax ( manaChannel , stack - > valOfBonuses ( Bonus : : MANA_CHANNELING ) ) ;
2011-10-09 14:23:24 +03:00
}
}
sc . manaGained = ( manaChannel * sc . spellCost ) / 100 ;
}
}
2010-02-23 17:39:31 +02:00
//calculating affected creatures for all spells
2013-07-20 11:54:43 +03:00
//must be vector, as in Chain Lightning order matters
std : : vector < const CStack * > attackedCres ; //what is that and what is sc.afectedCres?
2011-12-14 00:23:17 +03:00
if ( mode ! = ECastingMode : : ENCHANTER_CASTING )
2011-10-09 10:20:23 +03:00
{
2013-07-20 11:54:43 +03:00
auto creatures = gs - > curB - > getAffectedCreatures ( spell , spellLvl , casterColor , destination ) ;
std : : copy ( creatures . begin ( ) , creatures . end ( ) , std : : back_inserter ( attackedCres ) ) ;
2011-10-09 10:20:23 +03:00
}
else //enchanter - hit all possible stacks
2010-02-23 17:39:31 +02:00
{
2013-06-29 16:05:48 +03:00
for ( const CStack * stack : gs - > curB - > stacks )
2011-10-09 10:20:23 +03:00
{
/*if it's non negative spell and our unit or non positive spell and hostile unit */
2012-02-17 00:19:07 +03:00
if ( ( ! spell - > isNegative ( ) & & stack - > owner = = casterColor )
| | ( ! spell - > isPositive ( ) & & stack - > owner ! = casterColor ) )
2011-10-09 10:20:23 +03:00
{
2012-02-17 00:19:07 +03:00
if ( stack - > isValidTarget ( ) ) //TODO: allow dead targets somewhere in the future
2012-04-18 18:57:49 +03:00
{
2013-07-20 11:54:43 +03:00
attackedCres . push_back ( stack ) ;
2012-04-18 18:57:49 +03:00
}
2011-10-09 10:20:23 +03:00
}
}
2010-02-23 17:39:31 +02:00
}
2013-07-20 11:54:43 +03:00
2013-06-29 16:05:48 +03:00
for ( auto cre : attackedCres )
2013-02-17 21:53:01 +03:00
{
sc . affectedCres . insert ( cre - > ID ) ;
}
2010-02-23 17:39:31 +02:00
//checking if creatures resist
2012-01-03 04:55:26 +03:00
sc . resisted = gs - > curB - > calculateResistedStacks ( spell , caster , secHero , attackedCres , casterColor , mode , usedSpellPower , spellLvl ) ;
2010-02-23 17:39:31 +02:00
//calculating dmg to display
2013-02-11 02:24:57 +03:00
if ( spellID = = SpellID : : DEATH_STARE | | spellID = = SpellID : : ACID_BREATH_DAMAGE )
2011-04-23 20:10:54 +03:00
{
sc . dmgToDisplay = usedSpellPower ;
2013-02-11 02:24:57 +03:00
if ( spellID = = SpellID : : DEATH_STARE )
2011-12-14 00:23:17 +03:00
vstd : : amin ( sc . dmgToDisplay , ( * attackedCres . begin ( ) ) - > count ) ; //stack is already reduced after attack
2011-04-23 20:10:54 +03:00
}
2011-08-12 22:38:30 +03:00
StacksInjured si ;
2013-02-11 23:34:31 +03:00
2010-02-23 17:39:31 +02:00
//applying effects
2013-02-11 23:34:31 +03:00
2013-01-16 14:19:04 +03:00
if ( spell - > isOffensiveSpell ( ) )
2012-07-23 13:23:43 +03:00
{
2011-07-08 10:00:11 +03:00
int spellDamage = 0 ;
2011-12-14 00:23:17 +03:00
if ( stack & & mode ! = ECastingMode : : MAGIC_MIRROR )
2011-07-08 10:00:11 +03:00
{
2013-02-14 02:55:42 +03:00
int unitSpellPower = stack - > valOfBonuses ( Bonus : : SPECIFIC_SPELL_POWER , spellID . toEnum ( ) ) ;
2011-07-08 10:00:11 +03:00
if ( unitSpellPower )
sc . dmgToDisplay = spellDamage = stack - > count * unitSpellPower ; //TODO: handle immunities
2011-07-08 16:17:05 +03:00
else //Faerie Dragon
2011-10-08 12:11:36 +03:00
{
2011-07-08 16:17:05 +03:00
usedSpellPower = stack - > valOfBonuses ( Bonus : : CREATURE_SPELL_POWER ) * stack - > count / 100 ;
2011-10-08 12:11:36 +03:00
sc . dmgToDisplay = 0 ;
}
2011-07-08 10:00:11 +03:00
}
2012-05-20 14:40:23 +03:00
int chainLightningModifier = 0 ;
2013-06-29 16:05:48 +03:00
for ( auto & attackedCre : attackedCres )
2010-02-23 17:39:31 +02:00
{
2013-06-29 16:05:48 +03:00
if ( vstd : : contains ( sc . resisted , ( attackedCre ) - > ID ) ) //this creature resisted the spell
2010-02-23 17:39:31 +02:00
continue ;
BattleStackAttacked bsa ;
2013-06-29 16:05:48 +03:00
if ( ( destination > - 1 & & ( attackedCre ) - > coversPos ( destination ) ) | | ( spell - > range [ spellLvl ] = = " X " | | mode = = ECastingMode : : ENCHANTER_CASTING ) )
2012-04-18 18:57:49 +03:00
//display effect only upon primary target of area spell
2013-02-17 21:53:01 +03:00
//FIXME: if no stack is attacked, ther eis no animation and interface freezes
2011-07-03 08:55:57 +03:00
{
bsa . flags | = BattleStackAttacked : : EFFECT ;
bsa . effect = spell - > mainEffectAnim ;
}
2011-07-08 10:00:11 +03:00
if ( spellDamage )
2012-05-20 14:40:23 +03:00
bsa . damageAmount = spellDamage > > chainLightningModifier ;
2011-07-08 10:00:11 +03:00
else
2013-06-29 16:05:48 +03:00
bsa . damageAmount = gs - > curB - > calculateSpellDmg ( spell , caster , attackedCre , spellLvl , usedSpellPower ) > > chainLightningModifier ;
2013-02-18 10:12:19 +03:00
sc . dmgToDisplay + = bsa . damageAmount ;
2013-06-29 16:05:48 +03:00
bsa . stackAttacked = ( attackedCre ) - > ID ;
2012-04-18 18:57:49 +03:00
if ( mode = = ECastingMode : : ENCHANTER_CASTING ) //multiple damage spells cast
bsa . attackerID = stack - > ID ;
else
bsa . attackerID = - 1 ;
2013-06-29 16:05:48 +03:00
( attackedCre ) - > prepareAttacked ( bsa ) ;
2010-02-23 17:39:31 +02:00
si . stacks . push_back ( bsa ) ;
2012-05-20 14:40:23 +03:00
2013-02-11 02:24:57 +03:00
if ( spellID = = SpellID : : CHAIN_LIGHTNING )
2012-05-20 14:40:23 +03:00
+ + chainLightningModifier ;
2013-02-11 23:34:31 +03:00
}
2013-01-16 14:19:04 +03:00
}
else if ( spell - > hasEffects ( ) )
{
2011-07-08 16:17:05 +03:00
int stackSpellPower = 0 ;
2011-12-14 00:23:17 +03:00
if ( stack & & mode ! = ECastingMode : : MAGIC_MIRROR )
2011-07-08 16:17:05 +03:00
{
stackSpellPower = stack - > valOfBonuses ( Bonus : : CREATURE_ENCHANT_POWER ) ;
}
2010-02-23 17:39:31 +02:00
SetStackEffect sse ;
2011-04-23 12:57:51 +03:00
Bonus pseudoBonus ;
pseudoBonus . sid = spellID ;
pseudoBonus . val = spellLvl ;
2011-07-08 16:17:05 +03:00
pseudoBonus . turnsRemain = gs - > curB - > calculateSpellDuration ( spell , caster , stackSpellPower ? stackSpellPower : usedSpellPower ) ;
2011-04-23 12:57:51 +03:00
CStack : : stackEffectToFeature ( sse . effect , pseudoBonus ) ;
2013-02-11 02:24:57 +03:00
if ( spellID = = SpellID : : SHIELD | | spellID = = SpellID : : AIR_SHIELD )
2013-02-06 23:43:17 +03:00
{
sse . effect . back ( ) . val = ( 100 - sse . effect . back ( ) . val ) ; //fix to original config: shiled should display damage reduction
}
2013-02-11 02:24:57 +03:00
if ( spellID = = SpellID : : BIND & & stack ) //bind
2011-10-17 11:24:51 +03:00
{
sse . effect . back ( ) . additionalInfo = stack - > ID ; //we need to know who casted Bind
}
2013-06-26 14:18:27 +03:00
const Bonus * bonus = nullptr ;
2011-04-25 12:03:13 +03:00
if ( caster )
2012-09-20 19:55:21 +03:00
bonus = caster - > getBonusLocalFirst ( Selector : : typeSubtype ( Bonus : : SPECIAL_PECULIAR_ENCHANT , spellID ) ) ;
2013-01-17 21:15:00 +03:00
//TODO does hero specialty should affects his stack casting spells?
2011-04-23 12:57:51 +03:00
2011-06-24 10:12:52 +03:00
si32 power = 0 ;
2013-06-29 16:05:48 +03:00
for ( const CStack * affected : attackedCres )
2010-02-23 17:39:31 +02:00
{
2012-09-20 19:55:21 +03:00
if ( vstd : : contains ( sc . resisted , affected - > ID ) ) //this creature resisted the spell
2010-02-23 17:39:31 +02:00
continue ;
2012-09-20 19:55:21 +03:00
sse . stacks . push_back ( affected - > ID ) ;
2012-07-23 13:23:43 +03:00
2011-04-23 12:57:51 +03:00
//Apply hero specials - peculiar enchants
2013-01-17 21:15:00 +03:00
const ui8 tier = std : : max ( ( ui8 ) 1 , affected - > getCreature ( ) - > level ) ; //don't divide by 0 for certain creatures (commanders, war machines)
2012-09-20 19:55:21 +03:00
if ( bonus )
{
switch ( bonus - > additionalInfo )
2011-04-23 12:57:51 +03:00
{
2012-09-20 19:55:21 +03:00
case 0 : //normal
{
switch ( tier )
{
case 1 : case 2 :
power = 3 ;
break ;
case 3 : case 4 :
power = 2 ;
break ;
case 5 : case 6 :
power = 1 ;
break ;
}
Bonus specialBonus ( sse . effect . back ( ) ) ;
specialBonus . val = power ; //it doesn't necessarily make sense for some spells, use it wisely
sse . uniqueBonuses . push_back ( std : : pair < ui32 , Bonus > ( affected - > ID , specialBonus ) ) ; //additional premy to given effect
}
break ;
case 1 : //only Coronius as yet
{
power = std : : max ( 5 - tier , 0 ) ;
Bonus specialBonus = CStack : : featureGenerator ( Bonus : : PRIMARY_SKILL , PrimarySkill : : ATTACK , power , pseudoBonus . turnsRemain ) ;
specialBonus . sid = spellID ;
sse . uniqueBonuses . push_back ( std : : pair < ui32 , Bonus > ( affected - > ID , specialBonus ) ) ; //additional attack to Slayer effect
}
break ;
2011-04-23 12:57:51 +03:00
}
2012-09-20 19:55:21 +03:00
}
if ( caster & & caster - > hasBonusOfType ( Bonus : : SPECIAL_BLESS_DAMAGE , spellID ) ) //TODO: better handling of bonus percentages
{
2013-02-14 02:55:42 +03:00
int damagePercent = caster - > level * caster - > valOfBonuses ( Bonus : : SPECIAL_BLESS_DAMAGE , spellID . toEnum ( ) ) / tier ;
2012-09-20 19:55:21 +03:00
Bonus specialBonus = CStack : : featureGenerator ( Bonus : : CREATURE_DAMAGE , 0 , damagePercent , pseudoBonus . turnsRemain ) ;
specialBonus . valType = Bonus : : PERCENT_TO_ALL ;
specialBonus . sid = spellID ;
sse . uniqueBonuses . push_back ( std : : pair < ui32 , Bonus > ( affected - > ID , specialBonus ) ) ;
}
2010-02-23 17:39:31 +02:00
}
2011-04-23 12:57:51 +03:00
2010-02-23 17:39:31 +02:00
if ( ! sse . stacks . empty ( ) )
sendAndApply ( & sse ) ;
2013-02-11 23:34:31 +03:00
2013-01-16 14:19:04 +03:00
}
2013-02-11 02:24:57 +03:00
else if ( spell - > isRisingSpell ( ) | | spell - > id = = SpellID : : CURE )
2013-01-16 14:19:04 +03:00
{
2011-07-08 16:17:05 +03:00
int hpGained = 0 ;
if ( stack )
{
2013-02-14 02:55:42 +03:00
int unitSpellPower = stack - > valOfBonuses ( Bonus : : SPECIFIC_SPELL_POWER , spellID . toEnum ( ) ) ;
2011-07-08 16:17:05 +03:00
if ( unitSpellPower )
hpGained = stack - > count * unitSpellPower ; //Archangel
else //Faerie Dragon-like effect - unused fo far
usedSpellPower = stack - > valOfBonuses ( Bonus : : CREATURE_SPELL_POWER ) * stack - > count / 100 ;
}
2010-02-23 17:39:31 +02:00
StacksHealedOrResurrected shr ;
2011-06-21 15:45:57 +03:00
shr . lifeDrain = ( ui8 ) false ;
shr . tentHealing = ( ui8 ) false ;
2013-06-29 16:05:48 +03:00
for ( auto & attackedCre : attackedCres )
2010-02-23 17:39:31 +02:00
{
2013-06-29 16:05:48 +03:00
if ( vstd : : contains ( sc . resisted , ( attackedCre ) - > ID ) //this creature resisted the spell
| | ( spellID = = SpellID : : ANIMATE_DEAD & & ! ( attackedCre ) - > hasBonusOfType ( Bonus : : UNDEAD ) ) //we try to cast animate dead on living stack, TODO: showuld be not affected earlier
2012-07-23 13:23:43 +03:00
)
2010-02-23 17:39:31 +02:00
continue ;
StacksHealedOrResurrected : : HealInfo hi ;
2013-06-29 16:05:48 +03:00
hi . stackID = ( attackedCre ) - > ID ;
2013-02-06 11:02:46 +03:00
if ( stack ) //casted by creature
2011-07-08 16:17:05 +03:00
{
if ( hpGained )
{
2013-06-29 16:05:48 +03:00
hi . healedHP = gs - > curB - > calculateHealedHP ( hpGained , spell , attackedCre ) ; //archangel
2011-07-08 16:17:05 +03:00
}
else
2013-06-29 16:05:48 +03:00
hi . healedHP = gs - > curB - > calculateHealedHP ( spell , usedSpellPower , spellLvl , attackedCre ) ; //any typical spell (commander's cure or animate dead)
2011-07-08 16:17:05 +03:00
}
else
2013-06-29 16:05:48 +03:00
hi . healedHP = gs - > curB - > calculateHealedHP ( caster , spell , attackedCre , gs - > curB - > battleGetStackByID ( selectedStack ) ) ; //Casted by hero
2010-02-23 17:39:31 +02:00
hi . lowLevelResurrection = spellLvl < = 1 ;
shr . healedStacks . push_back ( hi ) ;
}
if ( ! shr . healedStacks . empty ( ) )
sendAndApply ( & shr ) ;
2013-02-11 02:24:57 +03:00
if ( spellID = = SpellID : : SACRIFICE ) //remove victim
2012-04-28 22:40:27 +03:00
{
BattleStacksRemoved bsr ;
bsr . stackIDs . insert ( selectedStack ) ; //somehow it works for teleport?
sendAndApply ( & bsr ) ;
2013-02-11 23:34:31 +03:00
}
2013-01-16 14:19:04 +03:00
}
else
switch ( spellID )
{
2013-02-11 02:24:57 +03:00
case SpellID : : QUICKSAND :
case SpellID : : LAND_MINE :
2013-01-16 14:19:04 +03:00
{
std : : vector < BattleHex > availableTiles ;
for ( int i = 0 ; i < GameConstants : : BFIELD_SIZE ; i + = 1 )
{
BattleHex hex = i ;
if ( hex . getX ( ) > 2 & & hex . getX ( ) < 14 & & ! battleGetStackByPos ( hex , false ) & ! battleGetObstacleOnPos ( hex , false ) )
availableTiles . push_back ( hex ) ;
2012-04-28 22:40:27 +03:00
}
2013-01-16 14:19:04 +03:00
boost : : range : : random_shuffle ( availableTiles ) ;
const int patchesForSkill [ ] = { 4 , 4 , 6 , 8 } ;
const int patchesToPut = std : : min < int > ( patchesForSkill [ spellLvl ] , availableTiles . size ( ) ) ;
//land mines or quicksand patches are handled as spell created obstacles
for ( int i = 0 ; i < patchesToPut ; i + + )
placeObstacle ( availableTiles [ i ] ) ;
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : FORCE_FIELD :
2013-01-16 14:19:04 +03:00
placeObstacle ( destination ) ;
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : FIRE_WALL :
2013-01-16 14:19:04 +03:00
{
//fire wall is build from multiple obstacles - one fire piece for each affected hex
auto affectedHexes = spell - > rangeInHexes ( destination , spellLvl , casterSide ) ;
2013-06-29 16:05:48 +03:00
for ( BattleHex hex : affectedHexes )
2013-01-16 14:19:04 +03:00
placeObstacle ( hex ) ;
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : TELEPORT :
2013-01-16 14:19:04 +03:00
{
BattleStackMoved bsm ;
bsm . distance = - 1 ;
bsm . stack = selectedStack ;
std : : vector < BattleHex > tiles ;
tiles . push_back ( destination ) ;
bsm . tilesToMove = tiles ;
bsm . teleporting = true ;
sendAndApply ( & bsm ) ;
2010-02-23 17:39:31 +02:00
break ;
}
2013-02-11 02:24:57 +03:00
case SpellID : : SUMMON_FIRE_ELEMENTAL :
case SpellID : : SUMMON_EARTH_ELEMENTAL :
case SpellID : : SUMMON_WATER_ELEMENTAL :
case SpellID : : SUMMON_AIR_ELEMENTAL :
2011-10-01 22:56:54 +03:00
{ //elemental summoning
2013-02-11 02:24:57 +03:00
CreatureID creID ;
2011-10-01 22:56:54 +03:00
switch ( spellID )
{
2013-02-11 02:24:57 +03:00
case SpellID : : SUMMON_FIRE_ELEMENTAL :
2013-02-07 20:34:50 +03:00
creID = CreatureID : : FIRE_ELEMENTAL ;
2011-10-01 22:56:54 +03:00
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : SUMMON_EARTH_ELEMENTAL :
2013-02-07 20:34:50 +03:00
creID = CreatureID : : EARTH_ELEMENTAL ;
2011-10-01 22:56:54 +03:00
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : SUMMON_WATER_ELEMENTAL :
2013-02-07 20:34:50 +03:00
creID = CreatureID : : WATER_ELEMENTAL ;
2011-10-01 22:56:54 +03:00
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : SUMMON_AIR_ELEMENTAL :
2013-02-07 20:34:50 +03:00
creID = CreatureID : : AIR_ELEMENTAL ;
2011-10-01 22:56:54 +03:00
break ;
}
BattleStackAdded bsa ;
bsa . creID = creID ;
bsa . attacker = ! ( bool ) casterSide ;
bsa . summoned = true ;
2012-01-03 04:55:26 +03:00
bsa . pos = gs - > curB - > getAvaliableHex ( creID , ! ( bool ) casterSide ) ; //TODO: unify it
//TODO stack casting -> probably power will be zero; set the proper number of creatures manually
2013-02-14 02:55:42 +03:00
int percentBonus = caster ? caster - > valOfBonuses ( Bonus : : SPECIFIC_SPELL_DAMAGE , spellID . toEnum ( ) ) : 0 ;
2012-01-03 04:55:26 +03:00
2013-02-13 22:35:43 +03:00
bsa . amount = usedSpellPower
* SpellID ( spellID ) . toSpell ( ) - > powers [ spellLvl ]
* ( 100 + percentBonus ) / 100.0 ; //new feature - percentage bonus
2012-01-03 04:55:26 +03:00
if ( bsa . amount )
sendAndApply ( & bsa ) ;
else
complain ( " Summoning elementals didn't summon any! " ) ;
2011-10-01 22:56:54 +03:00
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : CLONE :
2012-02-10 16:13:24 +03:00
{
2013-06-26 14:18:27 +03:00
const CStack * clonedStack = nullptr ;
2012-02-10 16:13:24 +03:00
if ( attackedCres . size ( ) )
clonedStack = * attackedCres . begin ( ) ;
if ( ! clonedStack )
{
complain ( " No target stack to clone! " ) ;
return ;
}
BattleStackAdded bsa ;
bsa . creID = clonedStack - > type - > idNumber ;
bsa . attacker = ! ( bool ) casterSide ;
bsa . summoned = true ;
bsa . pos = gs - > curB - > getAvaliableHex ( bsa . creID , ! ( bool ) casterSide ) ; //TODO: unify it
bsa . amount = clonedStack - > count ;
sendAndApply ( & bsa ) ;
BattleSetStackProperty ssp ;
ssp . stackID = gs - > curB - > stacks . back ( ) - > ID ; //how to get recent stack?
ssp . which = BattleSetStackProperty : : CLONED ; //using enum values
ssp . val = 0 ;
ssp . absolute = 1 ;
sendAndApply ( & ssp ) ;
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : REMOVE_OBSTACLE :
2010-02-23 17:39:31 +02:00
{
2013-07-21 13:25:12 +03:00
if ( auto obstacleToRemove = battleGetObstacleOnPos ( destination , false ) )
2010-02-23 17:39:31 +02:00
{
2013-07-21 13:25:12 +03:00
ObstaclesRemoved obr ;
obr . obstacles . insert ( obstacleToRemove - > uniqueID ) ;
2010-02-23 17:39:31 +02:00
sendAndApply ( & obr ) ;
2013-07-21 13:25:12 +03:00
}
2012-04-23 22:56:37 +03:00
else
complain ( " There's no obstacle to remove! " ) ;
2010-02-23 17:39:31 +02:00
}
2011-04-25 12:03:13 +03:00
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : DEATH_STARE : //handled in a bit different way
2011-04-23 20:10:54 +03:00
{
2013-06-29 16:05:48 +03:00
for ( auto & attackedCre : attackedCres )
2011-04-23 20:10:54 +03:00
{
2013-06-29 16:05:48 +03:00
if ( ( attackedCre ) - > hasBonusOfType ( Bonus : : UNDEAD ) | | ( attackedCre ) - > hasBonusOfType ( Bonus : : NON_LIVING ) ) //this creature is immune
2011-04-25 12:03:13 +03:00
{
sc . dmgToDisplay = 0 ; //TODO: handle Death Stare for multiple targets (?)
2011-04-23 20:10:54 +03:00
continue ;
2011-04-25 12:03:13 +03:00
}
2011-04-23 20:10:54 +03:00
BattleStackAttacked bsa ;
bsa . flags | = BattleStackAttacked : : EFFECT ;
2011-04-25 12:03:13 +03:00
bsa . effect = spell - > mainEffectAnim ; //from config\spell-Info.txt
2013-06-29 16:05:48 +03:00
bsa . damageAmount = usedSpellPower * ( attackedCre ) - > valOfBonuses ( Bonus : : STACK_HEALTH ) ;
bsa . stackAttacked = ( attackedCre ) - > ID ;
2011-04-23 20:10:54 +03:00
bsa . attackerID = - 1 ;
2013-06-29 16:05:48 +03:00
( attackedCre ) - > prepareAttacked ( bsa ) ;
2011-04-23 20:10:54 +03:00
si . stacks . push_back ( bsa ) ;
}
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : ACID_BREATH_DAMAGE : //new effect, separate from acid breath defense reduction
2011-05-13 12:02:16 +03:00
{
2013-06-29 16:05:48 +03:00
for ( auto & attackedCre : attackedCres ) //no immunities
2011-05-13 12:02:16 +03:00
{
BattleStackAttacked bsa ;
bsa . flags | = BattleStackAttacked : : EFFECT ;
2013-01-15 17:20:48 +03:00
bsa . effect = spell - > mainEffectAnim ;
2011-05-13 12:02:16 +03:00
bsa . damageAmount = usedSpellPower ; //damage times the number of attackers
2013-06-29 16:05:48 +03:00
bsa . stackAttacked = ( attackedCre ) - > ID ;
2011-05-13 12:02:16 +03:00
bsa . attackerID = - 1 ;
2013-06-29 16:05:48 +03:00
( attackedCre ) - > prepareAttacked ( bsa ) ;
2011-05-13 12:02:16 +03:00
si . stacks . push_back ( bsa ) ;
}
}
break ;
2010-02-23 17:39:31 +02:00
}
2011-04-25 12:03:13 +03:00
sendAndApply ( & sc ) ;
2011-08-12 22:38:30 +03:00
if ( ! si . stacks . empty ( ) ) //after spellcast info shows
sendAndApply ( & si ) ;
2011-10-01 22:56:54 +03:00
2011-12-14 00:23:17 +03:00
if ( mode = = ECastingMode : : CREATURE_ACTIVE_CASTING | | mode = = ECastingMode : : ENCHANTER_CASTING ) //reduce number of casts remaining
2011-10-09 10:20:23 +03:00
{
BattleSetStackProperty ssp ;
ssp . stackID = stack - > ID ;
ssp . which = BattleSetStackProperty : : CASTS ;
ssp . val = - 1 ;
ssp . absolute = false ;
sendAndApply ( & ssp ) ;
}
2011-07-16 11:28:01 +03:00
//Magic Mirror effect
2012-02-17 00:19:07 +03:00
if ( spell - > isNegative ( ) & & mode ! = ECastingMode : : MAGIC_MIRROR & & spell - > level & & spell - > range [ 0 ] = = " 0 " ) //it is actual spell and can be reflected to single target, no recurrence
2011-07-16 11:28:01 +03:00
{
2013-06-29 16:05:48 +03:00
for ( auto & attackedCre : attackedCres )
2011-07-16 11:28:01 +03:00
{
2013-06-29 16:05:48 +03:00
int mirrorChance = ( attackedCre ) - > valOfBonuses ( Bonus : : MAGIC_MIRROR ) ;
2011-07-16 11:28:01 +03:00
if ( mirrorChance > rand ( ) % 100 )
{
2011-07-16 19:40:38 +03:00
std : : vector < CStack * > mirrorTargets ;
std : : vector < CStack * > & battleStacks = gs - > curB - > stacks ;
2013-06-29 16:05:48 +03:00
for ( auto & battleStack : battleStacks )
2011-07-16 19:40:38 +03:00
{
2013-07-22 01:01:29 +03:00
if ( battleStack - > owner = = gs - > curB - > sides [ casterSide ] . color ) //get enemy stacks which can be affected by this spell
2011-07-16 19:40:38 +03:00
{
2013-06-29 16:05:48 +03:00
if ( ! gs - > curB - > battleIsImmune ( nullptr , spell , ECastingMode : : MAGIC_MIRROR , battleStack - > position ) )
mirrorTargets . push_back ( battleStack ) ;
2011-07-16 19:40:38 +03:00
}
}
if ( mirrorTargets . size ( ) )
2011-07-16 11:28:01 +03:00
{
2011-07-16 19:40:38 +03:00
int targetHex = mirrorTargets [ rand ( ) % mirrorTargets . size ( ) ] - > position ;
2013-06-29 16:05:48 +03:00
handleSpellCasting ( spellID , 0 , targetHex , 1 - casterSide , ( attackedCre ) - > owner , nullptr , ( caster ? caster : nullptr ) , usedSpellPower , ECastingMode : : MAGIC_MIRROR , ( attackedCre ) ) ;
2011-07-16 11:28:01 +03:00
}
}
}
}
2010-02-23 17:39:31 +02:00
}
2009-04-16 03:28:54 +03:00
bool CGameHandler : : makeCustomAction ( BattleAction & ba )
2009-03-09 12:37:49 +02:00
{
switch ( ba . actionType )
{
2013-02-04 00:05:44 +03:00
case Battle : : HERO_SPELL :
2009-03-09 12:37:49 +02:00
{
2013-06-22 17:47:20 +03:00
COMPLAIN_RET_FALSE_IF ( ba . side > 1 , " Side must be 0 or 1! " ) ;
2013-07-22 01:01:29 +03:00
const CGHeroInstance * h = gs - > curB - > battleGetFightingHero ( ba . side ) ;
const CGHeroInstance * secondHero = gs - > curB - > battleGetFightingHero ( ! ba . side ) ;
2009-03-09 12:37:49 +02:00
if ( ! h )
{
2013-04-10 20:18:01 +03:00
logGlobal - > warnStream ( ) < < " Wrong caster! " ;
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
if ( ba . additionalInfo > = VLC - > spellh - > spells . size ( ) )
{
2013-04-10 20:18:01 +03:00
logGlobal - > warnStream ( ) < < " Wrong spell id ( " < < ba . additionalInfo < < " )! " ;
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 12:37:49 +02:00
}
2013-02-13 22:35:43 +03:00
const CSpell * s = SpellID ( ba . additionalInfo ) . toSpell ( ) ;
2013-02-11 02:24:57 +03:00
if ( s - > mainEffectAnim > - 1 | | ( s - > id > = 66 | | s - > id < = 69 ) | | s - > id = = SpellID : : CLONE ) //allow summon elementals
2011-10-01 22:56:54 +03:00
//TODO: special effects, like Clone
2009-03-09 12:37:49 +02:00
{
2011-06-19 12:07:15 +03:00
ui8 skill = h - > getSpellSchoolLevel ( s ) ; //skill level
2009-03-09 12:37:49 +02:00
2011-12-14 00:23:17 +03:00
ESpellCastProblem : : ESpellCastProblem escp = gs - > curB - > battleCanCastThisSpell ( h - > tempOwner , s , ECastingMode : : HERO_CASTING ) ;
if ( escp ! = ESpellCastProblem : : OK )
2011-06-19 12:07:15 +03:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > warnStream ( ) < < " Spell cannot be cast! " ;
logGlobal - > warnStream ( ) < < " Problem : " < < escp ;
2011-06-19 12:07:15 +03:00
return false ;
}
2009-03-09 12:37:49 +02:00
2011-06-19 12:07:15 +03:00
StartAction start_action ( ba ) ;
sendAndApply ( & start_action ) ; //start spell casting
2009-04-21 20:32:43 +03:00
2013-02-14 02:55:42 +03:00
handleSpellCasting ( SpellID ( ba . additionalInfo ) , skill , ba . destinationTile , ba . side , h - > tempOwner ,
2013-02-06 13:16:44 +03:00
h , secondHero , h - > getPrimSkillLevel ( PrimarySkill : : SPELL_POWER ) ,
2013-06-26 14:18:27 +03:00
ECastingMode : : HERO_CASTING , nullptr , ba . selectedStack ) ;
2011-06-19 12:07:15 +03:00
sendAndApply ( & end_action ) ;
2012-08-26 12:07:48 +03:00
if ( ! gs - > curB - > battleGetStackByID ( gs - > curB - > activeStack , false ) - > alive ( ) )
2011-06-19 12:07:15 +03:00
{
battleMadeAction . setn ( true ) ;
}
2012-08-27 15:34:43 +03:00
checkForBattleEnd ( ) ;
2011-06-19 12:07:15 +03:00
if ( battleResult . get ( ) )
{
battleMadeAction . setn ( true ) ;
//battle will be ended by startBattle function
//endBattle(gs->curB->tile, gs->curB->heroes[0], gs->curB->heroes[1]);
}
return true ;
2009-09-28 18:05:35 +03:00
}
2011-06-19 12:07:15 +03:00
else
2009-09-28 17:21:48 +03:00
{
2013-04-10 20:18:01 +03:00
logGlobal - > warnStream ( ) < < " Spell " < < s - > name < < " is not yet supported! " ;
2011-06-19 12:07:15 +03:00
return false ;
2009-09-28 17:21:48 +03:00
}
2009-03-09 12:37:49 +02:00
}
}
2009-04-16 03:28:54 +03:00
return false ;
2009-03-09 21:40:43 +02:00
}
2011-10-08 16:02:58 +03:00
void CGameHandler : : stackTurnTrigger ( const CStack * st )
{
BattleTriggerEffect bte ;
bte . stackID = st - > ID ;
bte . effect = - 1 ;
bte . val = 0 ;
bte . additionalInfo = 0 ;
if ( st - > alive ( ) )
{
2011-10-17 11:24:51 +03:00
//unbind
2013-02-19 13:31:13 +03:00
if ( st - > getEffect ( SpellID : : BIND ) )
2011-10-17 11:24:51 +03:00
{
bool unbind = true ;
BonusList bl = * ( st - > getBonuses ( Selector : : type ( Bonus : : BIND_EFFECT ) ) ) ;
2012-08-26 12:07:48 +03:00
std : : set < const CStack * > stacks = gs - > curB - > batteAdjacentCreatures ( st ) ;
2011-10-17 11:24:51 +03:00
2013-06-29 16:05:48 +03:00
for ( Bonus * b : bl )
2011-10-17 11:24:51 +03:00
{
2012-08-26 12:07:48 +03:00
const CStack * stack = gs - > curB - > battleGetStackByID ( b - > additionalInfo ) ; //binding stack must be alive and adjacent
2011-10-17 11:24:51 +03:00
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 ) ;
}
}
2011-10-08 16:02:58 +03:00
//regeneration
if ( st - > hasBonusOfType ( Bonus : : HP_REGENERATION ) )
{
bte . effect = Bonus : : HP_REGENERATION ;
bte . val = std : : min ( ( int ) ( st - > MaxHealth ( ) - st - > firstHPleft ) , st - > valOfBonuses ( Bonus : : HP_REGENERATION ) ) ;
}
if ( st - > hasBonusOfType ( Bonus : : FULL_HP_REGENERATION ) )
{
bte . effect = Bonus : : HP_REGENERATION ;
bte . val = st - > MaxHealth ( ) - st - > firstHPleft ;
}
if ( bte . val ) //anything to heal
sendAndApply ( & bte ) ;
if ( st - > hasBonusOfType ( Bonus : : POISON ) )
{
2013-07-02 15:08:30 +03:00
const Bonus * b = st - > getBonusLocalFirst ( Selector : : source ( Bonus : : SPELL_EFFECT , SpellID : : POISON ) . And ( Selector : : type ( Bonus : : STACK_HEALTH ) ) ) ;
2011-10-08 16:02:58 +03: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 ) ;
}
}
}
2012-04-17 17:50:23 +03:00
if ( st - > hasBonusOfType ( Bonus : : MANA_DRAIN ) & & ! vstd : : contains ( st - > state , EBattleStackState : : DRAINED_MANA ) )
2011-10-08 16:02:58 +03:00
{
const CGHeroInstance * enemy = gs - > curB - > getHero ( gs - > curB - > theOtherPlayer ( st - > owner ) ) ;
2013-07-28 17:49:50 +03:00
//const CGHeroInstance * owner = gs->curB->getHero(st->owner);
2011-10-08 16:02:58 +03:00
if ( enemy )
{
ui32 manaDrained = st - > valOfBonuses ( Bonus : : MANA_DRAIN ) ;
2013-07-22 01:01:29 +03:00
vstd : : amin ( manaDrained , gs - > curB - > battleGetFightingHero ( 0 ) - > mana ) ;
2011-10-08 16:02:58 +03:00
if ( manaDrained )
{
bte . effect = Bonus : : MANA_DRAIN ;
bte . val = manaDrained ;
2013-02-14 02:55:42 +03:00
bte . additionalInfo = enemy - > id . getNum ( ) ; //for sanity
2011-10-08 16:02:58 +03:00
sendAndApply ( & bte ) ;
}
}
}
2013-05-30 10:43:42 +03:00
if ( st - > isLiving ( ) & & ! st - > hasBonusOfType ( Bonus : : FEARLESS ) )
2011-10-08 20:10:43 +03:00
{
bool fearsomeCreature = false ;
2013-06-29 16:05:48 +03:00
for ( CStack * stack : gs - > curB - > stacks )
2011-10-08 20:10:43 +03:00
{
2013-05-30 10:43:42 +03:00
if ( stack - > owner ! = st - > owner & & stack - > alive ( ) & & stack - > hasBonusOfType ( Bonus : : FEAR ) )
2011-10-08 20:10:43 +03:00
{
fearsomeCreature = true ;
break ;
}
}
if ( fearsomeCreature )
{
if ( rand ( ) % 100 < 10 ) //fixed 10%
{
bte . effect = Bonus : : FEAR ;
sendAndApply ( & bte ) ;
}
}
}
2011-10-09 10:20:23 +03:00
BonusList bl = * ( st - > getBonuses ( Selector : : type ( Bonus : : ENCHANTER ) ) ) ;
int side = gs - > curB - > whatSide ( st - > owner ) ;
2013-07-22 01:01:29 +03:00
if ( bl . size ( ) & & st - > casts & & ! gs - > curB - > sides [ side ] . enchanterCounter )
2011-10-08 16:02:58 +03:00
{
2011-10-09 10:20:23 +03:00
int index = rand ( ) % bl . size ( ) ;
2013-02-14 02:55:42 +03:00
SpellID spellID = SpellID ( bl [ index ] - > subtype ) ;
2013-02-19 13:31:13 +03:00
if ( gs - > curB - > battleCanCastThisSpell ( st - > owner , SpellID ( spellID ) . toSpell ( ) , ECastingMode : : ENCHANTER_CASTING ) = = ESpellCastProblem : : OK ) //TODO: select another available?
2011-10-09 10:20:23 +03:00
{
int spellLeveL = bl [ index ] - > val ; //spell level
const CGHeroInstance * enemyHero = gs - > curB - > getHero ( gs - > curB - > theOtherPlayer ( st - > owner ) ) ;
2013-06-26 14:18:27 +03:00
handleSpellCasting ( spellID , spellLeveL , - 1 , side , st - > owner , nullptr , enemyHero , 0 , ECastingMode : : ENCHANTER_CASTING , st ) ;
2011-10-09 10:20:23 +03:00
BattleSetStackProperty ssp ;
ssp . which = BattleSetStackProperty : : ENCHANTER_COUNTER ;
ssp . absolute = false ;
ssp . val = bl [ index ] - > additionalInfo ; //increase cooldown counter
ssp . stackID = st - > ID ;
sendAndApply ( & ssp ) ;
}
2011-10-08 16:02:58 +03:00
}
2012-12-01 21:30:03 +03:00
bl = * ( st - > getBonuses ( Selector : : type ( Bonus : : ENCHANTED ) ) ) ;
2013-06-29 16:05:48 +03:00
for ( auto b : bl )
2012-12-01 21:30:03 +03:00
{
SetStackEffect sse ;
int val = bl . valOfBonuses ( Selector : : typeSubtype ( b - > type , b - > subtype ) ) ;
if ( val > 3 )
{
2013-06-29 16:05:48 +03:00
for ( auto s : gs - > curB - > battleGetAllStacks ( ) )
2012-12-01 21:30:03 +03:00
{
if ( st - > owner = = s - > owner & & s - > isValidTarget ( ) ) //all allied
sse . stacks . push_back ( s - > ID ) ;
}
}
else
sse . stacks . push_back ( st - > ID ) ;
Bonus pseudoBonus ;
pseudoBonus . sid = b - > subtype ;
pseudoBonus . val = ( ( val > 3 ) ? ( val - 3 ) : val ) ;
pseudoBonus . turnsRemain = 50 ;
st - > stackEffectToFeature ( sse . effect , pseudoBonus ) ;
2013-04-23 12:54:57 +03:00
if ( sse . effect . size ( ) )
sendAndApply ( & sse ) ;
2012-12-01 21:30:03 +03:00
}
2011-10-08 16:02:58 +03:00
}
}
2012-08-26 12:07:48 +03:00
void CGameHandler : : handleDamageFromObstacle ( const CObstacleInstance & obstacle , const CStack * curStack )
2012-05-18 23:50:16 +03:00
{
//we want to determine following vars depending on obstacle type
int damage = - 1 ;
int effect = - 1 ;
bool oneTimeObstacle = false ;
//helper info
const SpellCreatedObstacle * spellObstacle = dynamic_cast < const SpellCreatedObstacle * > ( & obstacle ) ; //not nice but we may need spell params
2013-08-04 09:16:17 +03:00
const ui8 side = ! curStack - > attackerOwned ; //if enemy is defending (false = 0), side of enemy hero is 1 (true)
2013-07-22 01:01:29 +03:00
const CGHeroInstance * hero = gs - > curB - > battleGetFightingHero ( side ) ;
2012-05-18 23:50:16 +03:00
if ( obstacle . obstacleType = = CObstacleInstance : : MOAT )
{
damage = battleGetMoatDmg ( ) ;
}
else if ( obstacle . obstacleType = = CObstacleInstance : : LAND_MINE )
{
//You don't get hit by a Mine you can see.
2012-08-26 12:07:48 +03:00
if ( gs - > curB - > battleIsObstacleVisibleForSide ( obstacle , ( BattlePerspective : : BattlePerspective ) side ) )
2012-05-18 23:50:16 +03:00
return ;
oneTimeObstacle = true ;
2012-07-23 13:23:43 +03:00
effect = 82 ; //makes
2013-02-13 22:35:43 +03:00
damage = gs - > curB - > calculateSpellDmg ( SpellID ( SpellID : : LAND_MINE ) . toSpell ( ) , hero , curStack ,
2012-05-18 23:50:16 +03:00
spellObstacle - > spellLevel , spellObstacle - > casterSpellPower ) ;
//TODO even if obstacle wasn't created by hero (Tower "moat") it should deal dmg as if casted by hero,
//if it is bigger than default dmg. Or is it just irrelevant H3 implementation quirk
}
else if ( obstacle . obstacleType = = CObstacleInstance : : FIRE_WALL )
{
2013-02-13 22:35:43 +03:00
damage = gs - > curB - > calculateSpellDmg ( SpellID ( SpellID : : FIRE_WALL ) . toSpell ( ) , hero , curStack ,
2012-05-18 23:50:16 +03:00
spellObstacle - > spellLevel , spellObstacle - > casterSpellPower ) ;
}
else
{
//no other obstacle does damage to stack
return ;
}
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 ) ;
StacksInjured si ;
si . stacks . push_back ( bsa ) ;
sendAndApply ( & si ) ;
if ( oneTimeObstacle )
removeObstacle ( obstacle ) ;
}
2009-03-09 21:40:43 +02:00
void CGameHandler : : handleTimeEvents ( )
{
2009-07-18 00:53:28 +03:00
gs - > map - > events . sort ( evntCmp ) ;
2013-02-19 01:37:22 +03:00
while ( gs - > map - > events . size ( ) & & gs - > map - > events . front ( ) . firstOccurence + 1 = = gs - > day )
2009-03-09 21:40:43 +02:00
{
2013-02-19 01:37:22 +03:00
CMapEvent ev = gs - > map - > events . front ( ) ;
2013-03-03 20:06:03 +03:00
for ( int player = 0 ; player < PlayerColor : : PLAYER_LIMIT_I ; player + + )
2009-03-09 21:40:43 +02:00
{
2013-03-03 20:06:03 +03:00
PlayerState * pinfo = gs - > getPlayer ( PlayerColor ( player ) ) ;
2009-03-09 21:40:43 +02:00
if ( pinfo //player exists
2013-02-19 01:37:22 +03:00
& & ( ev . players & 1 < < player ) //event is enabled to this player
& & ( ( ev . computerAffected & & ! pinfo - > human )
| | ( ev . humanAffected & & pinfo - > human )
2009-03-09 21:40:43 +02:00
)
2009-03-14 13:25:25 +02:00
)
2009-03-09 21:40:43 +02:00
{
//give resources
SetResources sr ;
2013-03-03 20:06:03 +03:00
sr . player = PlayerColor ( player ) ;
2013-02-19 01:37:22 +03:00
sr . res = pinfo - > resources + ev . resources ;
2009-03-09 21:40:43 +02:00
//prepare dialog
InfoWindow iw ;
2013-03-03 20:06:03 +03:00
iw . player = PlayerColor ( player ) ;
2013-02-19 01:37:22 +03:00
iw . text < < ev . message ;
2011-07-05 09:14:07 +03:00
2013-02-19 01:37:22 +03:00
for ( int i = 0 ; i < ev . resources . size ( ) ; i + + )
2009-03-09 21:40:43 +02:00
{
2013-02-19 01:37:22 +03:00
if ( ev . resources [ i ] ) //if resource is changed, we add it to the dialog
iw . components . push_back ( Component ( Component : : RESOURCE , i , ev . resources [ i ] , 0 ) ) ;
2009-03-09 21:40:43 +02:00
}
2011-07-05 09:14:07 +03:00
2009-03-09 21:40:43 +02:00
if ( iw . components . size ( ) )
{
2011-07-05 09:14:07 +03:00
sr . res . amax ( 0 ) ; // If removing too much resources, adjust the amount so the total doesn't become negative.
2009-03-09 21:40:43 +02:00
sendAndApply ( & sr ) ; //update player resources if changed
}
sendAndApply ( & iw ) ; //show dialog
}
} //PLAYERS LOOP
2013-02-19 01:37:22 +03:00
if ( ev . nextOccurence )
2009-03-09 21:40:43 +02:00
{
2011-01-01 22:26:39 +02:00
gs - > map - > events . pop_front ( ) ;
2013-02-19 01:37:22 +03:00
ev . firstOccurence + = ev . nextOccurence ;
auto it = gs - > map - > events . begin ( ) ;
while ( it ! = gs - > map - > events . end ( ) & & it - > earlierThanOrEqual ( ev ) )
2011-01-01 22:26:39 +02:00
it + + ;
gs - > map - > events . insert ( it , ev ) ;
2009-03-09 21:40:43 +02:00
}
else
{
gs - > map - > events . pop_front ( ) ;
}
}
2013-02-19 01:37:22 +03:00
//TODO send only if changed
UpdateMapEvents ume ;
ume . events = gs - > map - > events ;
sendAndApply ( & ume ) ;
2009-03-27 01:05:40 +02:00
}
2013-03-14 23:44:00 +03:00
void CGameHandler : : handleTownEvents ( CGTownInstance * town , NewTurn & n )
2010-08-18 17:24:30 +03:00
{
town - > events . sort ( evntCmp ) ;
2013-02-19 01:37:22 +03:00
while ( town - > events . size ( ) & & town - > events . front ( ) . firstOccurence = = gs - > day )
2010-08-18 17:24:30 +03:00
{
2013-03-03 20:06:03 +03:00
PlayerColor player = town - > tempOwner ;
2013-02-19 01:37:22 +03:00
CCastleEvent ev = town - > events . front ( ) ;
2010-08-18 17:24:30 +03:00
PlayerState * pinfo = gs - > getPlayer ( player ) ;
if ( pinfo //player exists
2013-03-03 20:06:03 +03:00
& & ( ev . players & 1 < < player . getNum ( ) ) //event is enabled to this player
2013-02-19 01:37:22 +03:00
& & ( ( ev . computerAffected & & ! pinfo - > human )
| | ( ev . humanAffected & & pinfo - > human ) ) )
2010-08-18 17:24:30 +03:00
{
2011-07-05 09:14:07 +03:00
2010-08-18 17:24:30 +03:00
// dialog
InfoWindow iw ;
iw . player = player ;
2013-02-19 01:37:22 +03:00
iw . text < < ev . message ;
2010-08-18 17:24:30 +03:00
2013-02-19 01:37:22 +03:00
if ( ev . resources . nonZero ( ) )
2011-07-05 09:14:07 +03:00
{
TResources was = n . res [ player ] ;
2013-02-19 01:37:22 +03:00
n . res [ player ] + = ev . resources ;
2011-07-05 09:14:07 +03:00
n . res [ player ] . amax ( 0 ) ;
2013-02-19 01:37:22 +03:00
for ( int i = 0 ; i < ev . resources . size ( ) ; i + + )
if ( ev . resources [ i ] & & pinfo - > resources [ i ] ! = n . res [ player ] [ i ] ) //if resource had changed, we add it to the dialog
2011-07-05 09:14:07 +03:00
iw . components . push_back ( Component ( Component : : RESOURCE , i , n . res [ player ] [ i ] - was [ i ] , 0 ) ) ;
}
2010-08-18 17:24:30 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & i : ev . buildings )
2013-02-11 22:11:34 +03:00
{
if ( town - > hasBuilt ( i ) )
2010-08-18 17:24:30 +03:00
{
2013-02-11 22:11:34 +03:00
buildStructure ( town - > id , i , true ) ;
iw . components . push_back ( Component ( Component : : BUILDING , town - > subID , i , 0 ) ) ;
2010-08-18 17:24:30 +03:00
}
2013-02-11 22:11:34 +03:00
}
2010-08-18 17:24:30 +03:00
2013-03-14 23:44:00 +03:00
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 ] ;
2013-02-19 01:37:22 +03:00
for ( si32 i = 0 ; i < ev . creatures . size ( ) ; i + + ) //creature growths
2010-08-18 17:24:30 +03:00
{
2013-02-19 01:37:22 +03:00
if ( town - > creatureDwellingLevel ( i ) > = 0 & & ev . creatures [ i ] ) //there is dwelling
2010-08-18 17:24:30 +03:00
{
2013-03-14 23:44:00 +03:00
sac . creatures [ i ] . first + = ev . creatures [ i ] ;
2012-07-23 13:23:43 +03:00
iw . components . push_back ( Component ( Component : : CREATURE ,
2013-02-19 01:37:22 +03:00
town - > creatures [ i ] . second . back ( ) , ev . creatures [ i ] , 0 ) ) ;
2010-08-18 17:24:30 +03:00
}
}
sendAndApply ( & iw ) ; //show dialog
}
2013-02-19 01:37:22 +03:00
if ( ev . nextOccurence )
2011-01-18 18:24:06 +02:00
{
town - > events . pop_front ( ) ;
2011-01-01 22:26:39 +02:00
2013-02-19 01:37:22 +03:00
ev . firstOccurence + = ev . nextOccurence ;
auto it = town - > events . begin ( ) ;
while ( it ! = town - > events . end ( ) & & it - > earlierThanOrEqual ( ev ) )
2011-01-01 22:26:39 +02:00
it + + ;
town - > events . insert ( it , ev ) ;
2010-08-18 17:24:30 +03:00
}
else
{
town - > events . pop_front ( ) ;
}
}
2013-02-19 01:37:22 +03:00
//TODO send only if changed
UpdateCastleEvents uce ;
uce . town = town - > id ;
uce . events = town - > events ;
sendAndApply ( & uce ) ;
2010-08-18 17:24:30 +03:00
}
2009-03-27 01:05:40 +02:00
bool CGameHandler : : complain ( const std : : string & problem )
{
sendMessageToAll ( " Server encountered a problem: " + problem ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > errorStream ( ) < < problem ;
2009-03-27 01:05:40 +02:00
return true ;
2009-04-11 04:32:50 +03:00
}
2013-05-14 16:42:52 +03:00
void CGameHandler : : showGarrisonDialog ( ObjectInstanceID upobj , ObjectInstanceID hid , bool removableUnits )
2009-04-12 03:58:41 +03:00
{
2013-04-21 15:49:26 +03:00
//PlayerColor player = getOwner(hid);
2013-04-20 14:34:01 +03:00
auto upperArmy = dynamic_cast < const CArmedInstance * > ( getObj ( upobj ) ) ;
auto lowerArmy = dynamic_cast < const CArmedInstance * > ( getObj ( hid ) ) ;
2013-04-21 15:49:26 +03:00
assert ( lowerArmy ) ;
assert ( upperArmy ) ;
2013-04-20 14:34:01 +03:00
auto garrisonQuery = make_shared < CGarrisonDialogQuery > ( upperArmy , lowerArmy ) ;
queries . addQuery ( garrisonQuery ) ;
2009-04-12 03:58:41 +03:00
GarrisonDialog gd ;
gd . hid = hid ;
gd . objid = upobj ;
2012-07-15 18:34:00 +03:00
gd . removableUnits = removableUnits ;
2013-04-20 14:34:01 +03:00
gd . queryID = garrisonQuery - > queryID ;
sendAndApply ( & gd ) ;
2009-04-12 03:58:41 +03:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : showThievesGuildWindow ( PlayerColor player , ObjectInstanceID requestingObjId )
2010-02-06 15:49:14 +02:00
{
2010-02-07 17:06:14 +02:00
OpenWindow ow ;
ow . window = OpenWindow : : THIEVES_GUILD ;
2013-03-03 20:06:03 +03:00
ow . id1 = player . getNum ( ) ;
2013-02-14 02:55:42 +03:00
ow . id2 = requestingObjId . getNum ( ) ;
2010-02-07 17:06:14 +02:00
sendAndApply ( & ow ) ;
2010-02-06 15:49:14 +02:00
}
2013-02-14 02:55:42 +03:00
bool CGameHandler : : isAllowedExchange ( ObjectInstanceID id1 , ObjectInstanceID id2 )
2009-04-12 03:58:41 +03:00
{
if ( id1 = = id2 )
return true ;
const CGObjectInstance * o1 = getObj ( id1 ) , * o2 = getObj ( id2 ) ;
2013-04-20 14:34:01 +03:00
if ( ! o1 | | ! o2 )
return true ; //arranging stacks within an object should be always allowed
2012-02-04 19:34:29 +03:00
if ( o1 & & o2 )
2009-04-12 03:58:41 +03:00
{
2012-09-23 21:01:04 +03:00
if ( o1 - > ID = = Obj : : TOWN )
2012-02-04 19:34:29 +03:00
{
const CGTownInstance * t = static_cast < const CGTownInstance * > ( o1 ) ;
if ( t - > visitingHero = = o2 | | t - > garrisonHero = = o2 )
return true ;
}
2012-09-23 21:01:04 +03:00
if ( o2 - > ID = = Obj : : TOWN )
2012-02-04 19:34:29 +03:00
{
const CGTownInstance * t = static_cast < const CGTownInstance * > ( o2 ) ;
if ( t - > visitingHero = = o1 | | t - > garrisonHero = = o1 )
return true ;
}
2013-04-20 14:34:01 +03:00
2013-04-21 15:49:26 +03:00
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 ;
}
2013-04-20 14:34:01 +03:00
//Ongoing garrison exchange
if ( auto dialog = std : : dynamic_pointer_cast < CGarrisonDialogQuery > ( queries . topQuery ( o1 - > tempOwner ) ) )
2012-02-04 19:34:29 +03:00
{
2013-04-20 14:34:01 +03:00
if ( dialog - > exchangingArmies [ 0 ] = = o1 & & dialog - > exchangingArmies [ 1 ] = = o2 )
return true ;
if ( dialog - > exchangingArmies [ 1 ] = = o1 & & dialog - > exchangingArmies [ 0 ] = = o2 )
return true ;
2012-02-04 19:34:29 +03:00
}
2009-04-12 03:58:41 +03:00
}
2013-04-20 14:34:01 +03:00
2009-04-12 03:58:41 +03:00
return false ;
2009-05-12 06:13:07 +03:00
}
2009-07-06 22:41:27 +03:00
void CGameHandler : : objectVisited ( const CGObjectInstance * obj , const CGHeroInstance * h )
{
2013-04-20 14:34:01 +03:00
logGlobal - > traceStream ( ) < < h - > nodeName ( ) < < " visits " < < obj - > getHoverText ( ) ;
auto visitQuery = make_shared < CObjectVisitQuery > ( obj , h , obj - > visitablePos ( ) ) ;
queries . addQuery ( visitQuery ) ; //TODO real visit pos
2011-05-10 01:20:47 +03:00
HeroVisit hv ;
hv . obj = obj ;
hv . hero = h ;
hv . starting = true ;
sendAndApply ( & hv ) ;
2009-07-06 22:41:27 +03:00
obj - > onHeroVisit ( h ) ;
2011-05-10 01:20:47 +03:00
2013-04-20 14:34:01 +03:00
queries . popIfTop ( visitQuery ) ; //visit ends here if no queries were created
}
void CGameHandler : : objectVisitEnded ( const CObjectVisitQuery & query )
{
logGlobal - > traceStream ( ) < < query . visitingHero - > nodeName ( ) < < " visit ends. \n " ;
HeroVisit hv ;
hv . obj = nullptr ; //not necessary, moreover may have been deleted in the meantime
hv . hero = query . visitingHero ;
2013-05-19 01:30:48 +03:00
assert ( hv . hero ) ;
2011-05-10 01:20:47 +03:00
hv . starting = false ;
sendAndApply ( & hv ) ;
2009-07-07 06:48:00 +03:00
}
2009-07-26 06:33:13 +03:00
2013-02-14 02:55:42 +03:00
bool CGameHandler : : buildBoat ( ObjectInstanceID objid )
2009-07-26 06:33:13 +03:00
{
const IShipyard * obj = IShipyard : : castFrom ( getObj ( objid ) ) ;
2013-07-21 13:08:32 +03:00
if ( obj - > shipyardStatus ( ) ! = IBoatGenerator : : GOOD )
2009-07-26 06:33:13 +03:00
{
complain ( " Cannot build boat in this shipyard! " ) ;
return false ;
}
2012-09-23 21:01:04 +03:00
else if ( obj - > o - > ID = = Obj : : TOWN
2013-02-11 02:24:57 +03:00
& & ! static_cast < const CGTownInstance * > ( obj ) - > hasBuilt ( BuildingID : : SHIPYARD ) )
2009-07-26 06:33:13 +03:00
{
complain ( " Cannot build boat in the town - no shipyard! " ) ;
return false ;
}
2013-03-03 20:06:03 +03:00
const PlayerColor playerID = obj - > o - > tempOwner ;
2013-01-12 21:08:33 +03:00
TResources boatCost ;
obj - > getBoatCost ( boatCost ) ;
TResources aviable = gs - > getPlayer ( playerID ) - > resources ;
if ( ! aviable . canAfford ( boatCost ) )
2009-07-26 06:33:13 +03:00
{
complain ( " Not enough resources to build a boat! " ) ;
return false ;
}
int3 tile = obj - > bestLocation ( ) ;
if ( ! gs - > map - > isInTheMap ( tile ) )
{
complain ( " Cannot find appropriate tile for a boat! " ) ;
return false ;
}
//take boat cost
SetResources sr ;
2013-01-12 21:08:33 +03:00
sr . player = playerID ;
sr . res = ( aviable - boatCost ) ;
2009-07-26 06:33:13 +03:00
sendAndApply ( & sr ) ;
//create boat
NewObject no ;
2013-01-12 21:08:33 +03:00
no . ID = Obj : : BOAT ;
2010-02-13 06:47:31 +02:00
no . subID = obj - > getBoatType ( ) ;
2009-07-26 06:33:13 +03:00
no . pos = tile + int3 ( 1 , 0 , 0 ) ;
sendAndApply ( & no ) ;
2009-07-31 23:10:22 +03:00
return true ;
2009-08-04 02:53:18 +03:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : engageIntoBattle ( PlayerColor player )
2009-08-04 02:53:18 +03:00
{
//notify interfaces
PlayerBlocked pb ;
pb . player = player ;
pb . reason = PlayerBlocked : : UPCOMING_BATTLE ;
sendAndApply ( & pb ) ;
2009-10-04 05:02:45 +03:00
}
2010-01-29 22:52:45 +02:00
void CGameHandler : : winLoseHandle ( ui8 players )
{
2012-02-16 20:10:58 +03:00
2013-03-03 20:06:03 +03:00
for ( size_t i = 0 ; i < PlayerColor : : PLAYER_LIMIT_I ; i + + )
2010-01-29 22:52:45 +02:00
{
2013-03-03 20:06:03 +03:00
if ( players & 1 < < i & & gs - > getPlayer ( PlayerColor ( i ) ) )
2010-01-29 22:52:45 +02:00
{
2013-03-03 20:06:03 +03:00
checkLossVictory ( PlayerColor ( i ) ) ;
2010-01-29 22:52:45 +02:00
}
}
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : checkLossVictory ( PlayerColor player )
2010-01-29 22:52:45 +02:00
{
2010-02-01 21:19:42 +02:00
const PlayerState * p = gs - > getPlayer ( player ) ;
if ( p - > status ) //player already won / lost
return ;
2010-01-29 22:52:45 +02:00
int loss = gs - > lossCheck ( player ) ;
int vic = gs - > victoryCheck ( player ) ;
if ( ! loss & & ! vic )
return ;
2012-02-16 20:10:58 +03:00
2010-01-29 22:52:45 +02:00
InfoWindow iw ;
2010-02-02 01:30:03 +02:00
getLossVicMessage ( player , vic ? vic : loss , vic , iw ) ;
2010-01-29 22:52:45 +02:00
sendAndApply ( & iw ) ;
2013-02-09 21:18:55 +03:00
2010-01-29 22:52:45 +02:00
PlayerEndsGame peg ;
peg . player = player ;
peg . victory = vic ;
sendAndApply ( & peg ) ;
2012-04-14 10:39:32 +03:00
if ( vic ) //one player won -> all enemies lost
2010-02-01 19:07:46 +02:00
{
iw . text . localStrings . front ( ) . second + + ; //message about losing because enemy won first is just after victory message
2012-09-24 19:14:53 +03:00
for ( auto i = gs - > players . cbegin ( ) ; i ! = gs - > players . cend ( ) ; i + + )
2010-02-01 19:07:46 +02:00
{
2013-03-03 20:06:03 +03:00
if ( i - > first < PlayerColor : : PLAYER_LIMIT & & i - > first ! = player ) //FIXME: skip already eliminated players?
2010-02-01 19:07:46 +02:00
{
iw . player = i - > first ;
sendAndApply ( & iw ) ;
peg . player = i - > first ;
2013-02-04 00:05:44 +03:00
peg . victory = gameState ( ) - > getPlayerRelations ( player , i - > first ) = = PlayerRelations : : ALLIES ; // ally of winner
2010-02-01 19:07:46 +02:00
sendAndApply ( & peg ) ;
}
}
}
2010-02-01 21:19:42 +02:00
else //player lost -> all his objects become unflagged (neutral)
{
2013-02-06 02:16:13 +03:00
auto hlp = p - > heroes ;
for ( auto i = hlp . cbegin ( ) ; i ! = hlp . cend ( ) ; i + + ) //eliminate heroes
2013-02-09 00:17:39 +03:00
removeObject ( * i ) ;
2010-02-02 01:30:03 +02:00
2013-02-06 02:16:13 +03:00
for ( auto i = gs - > map - > objects . cbegin ( ) ; i ! = gs - > map - > objects . cend ( ) ; i + + ) //unflag objs
2010-02-01 21:19:42 +02:00
{
if ( * i & & ( * i ) - > tempOwner = = player )
2013-03-03 20:06:03 +03:00
setOwner ( * i , PlayerColor : : NEUTRAL ) ;
2010-02-01 21:19:42 +02:00
}
2013-09-02 01:55:57 +03:00
2010-02-07 17:06:14 +02:00
//eliminating one player may cause victory of another:
2013-03-03 20:06:03 +03:00
winLoseHandle ( GameConstants : : ALL_PLAYERS & ~ ( 1 < < player . getNum ( ) ) ) ;
2012-04-14 10:39:32 +03:00
}
2010-02-01 19:07:46 +02:00
2013-02-06 02:16:13 +03:00
if ( vic & & p - > human )
2012-04-14 10:39:32 +03:00
{
2010-01-29 22:52:45 +02:00
end2 = true ;
2010-08-20 16:34:39 +03:00
2012-09-21 20:59:54 +03:00
if ( gs - > scenarioOps - > campState )
2012-04-14 10:39:32 +03:00
{
2010-09-04 17:49:15 +03:00
std : : vector < CGHeroInstance * > hes ;
2013-06-29 16:05:48 +03:00
for ( CGHeroInstance * ghi : gs - > map - > heroesOnMap )
2012-02-16 20:10:58 +03:00
{
2013-02-06 02:16:13 +03:00
if ( ghi - > tempOwner = = player )
2012-04-14 10:39:32 +03:00
{
2010-09-04 17:49:15 +03:00
hes . push_back ( ghi ) ;
2012-04-14 10:39:32 +03:00
}
2010-09-04 17:49:15 +03:00
}
2012-09-21 20:59:54 +03:00
gs - > scenarioOps - > campState - > mapConquered ( hes ) ;
2010-08-20 16:34:39 +03:00
2013-02-09 21:18:55 +03:00
//Request clients to change connection mode
PrepareForAdvancingCampaign pfac ;
sendAndApply ( & pfac ) ;
//Change connection mode
if ( getPlayer ( player ) - > human & & getStartInfo ( ) - > campState )
{
2013-06-29 16:05:48 +03:00
for ( auto connection : conns )
2013-02-09 21:18:55 +03:00
connection - > prepareForSendingHeroes ( ) ;
}
2010-08-20 16:34:39 +03:00
UpdateCampaignState ucs ;
2012-09-21 20:59:54 +03:00
ucs . camp = gs - > scenarioOps - > campState ;
2010-08-20 16:34:39 +03:00
sendAndApply ( & ucs ) ;
2012-04-14 10:39:32 +03:00
}
2010-08-20 16:34:39 +03:00
}
2013-09-02 01:55:57 +03:00
//If player making turn has lost his turn must be over as well
if ( gs - > getPlayer ( gs - > currentPlayer ) - > status ! = EPlayerStatus : : INGAME )
{
states . setFlag ( gs - > currentPlayer , & PlayerStatus : : makingTurn , false ) ;
}
2010-01-29 22:52:45 +02:00
}
2013-03-03 20:06:03 +03:00
void CGameHandler : : getLossVicMessage ( PlayerColor player , si8 standard , bool victory , InfoWindow & out ) const
2010-01-29 22:52:45 +02:00
{
2010-10-31 00:53:41 +03:00
// const PlayerState *p = gs->getPlayer(player);
2010-02-01 19:07:46 +02:00
// if(!p->human)
// return; //AI doesn't need text info of loss
2010-01-29 22:52:45 +02:00
out . player = player ;
if ( victory )
{
2012-05-28 22:29:32 +03:00
if ( standard > 0 ) //not std loss
2010-01-29 22:52:45 +02:00
{
switch ( gs - > map - > victoryCondition . condition )
{
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : ARTIFACT :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 280 ) ; //Congratulations! You have found the %s, and can claim victory!
2012-11-06 19:39:29 +03:00
out . text . addReplacement ( MetaString : : ART_NAMES , gs - > map - > victoryCondition . objectId ) ; //artifact name
2010-01-29 22:52:45 +02:00
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : GATHERTROOP :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 276 ) ; //Congratulations! You have over %d %s in your armies. Your enemies have no choice but to bow down before your power!
out . text . addReplacement ( gs - > map - > victoryCondition . count ) ;
2012-11-06 19:39:29 +03:00
out . text . addReplacement ( MetaString : : CRE_PL_NAMES , gs - > map - > victoryCondition . objectId ) ;
2010-01-29 22:52:45 +02:00
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : GATHERRESOURCE :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 278 ) ; //Congratulations! You have collected over %d %s in your treasury. Victory is yours!
out . text . addReplacement ( gs - > map - > victoryCondition . count ) ;
2012-11-06 19:39:29 +03:00
out . text . addReplacement ( MetaString : : RES_NAMES , gs - > map - > victoryCondition . objectId ) ;
2010-01-29 22:52:45 +02:00
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : BUILDCITY :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 282 ) ; //Congratulations! You have successfully upgraded your town, and can claim victory!
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : BUILDGRAIL :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 284 ) ; //Congratulations! You have constructed a permanent home for the Grail, and can claim victory!
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : BEATHERO :
2010-01-29 22:52:45 +02:00
{
out . text . addTxt ( MetaString : : GENERAL_TXT , 252 ) ; //Congratulations! You have completed your quest to defeat the enemy hero %s. Victory is yours!
const CGHeroInstance * h = dynamic_cast < const CGHeroInstance * > ( gs - > map - > victoryCondition . obj ) ;
assert ( h ) ;
out . text . addReplacement ( h - > name ) ;
}
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : CAPTURECITY :
2010-01-29 22:52:45 +02:00
{
out . text . addTxt ( MetaString : : GENERAL_TXT , 249 ) ; //Congratulations! You captured %s, and are victorious!
const CGTownInstance * t = dynamic_cast < const CGTownInstance * > ( gs - > map - > victoryCondition . obj ) ;
assert ( t ) ;
out . text . addReplacement ( t - > name ) ;
}
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : BEATMONSTER :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 286 ) ; //Congratulations! You have completed your quest to kill the fearsome beast, and can claim victory!
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : TAKEDWELLINGS :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 288 ) ; //Congratulations! Your flag flies on the dwelling of every creature. Victory is yours!
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : TAKEMINES :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 290 ) ; //Congratulations! Your flag flies on every mine. Victory is yours!
break ;
2011-12-14 00:23:17 +03:00
case EVictoryConditionType : : TRANSPORTITEM :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 292 ) ; //Congratulations! You have reached your destination, precious cargo intact, and can claim victory!
break ;
}
}
else
{
2010-02-02 01:30:03 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 659 ) ; //Congratulations! You have reached your destination, precious cargo intact, and can claim victory!
2010-01-29 22:52:45 +02:00
}
}
else
{
2012-05-28 22:29:32 +03:00
if ( standard > 0 ) //not std loss
2010-01-29 22:52:45 +02:00
{
switch ( gs - > map - > lossCondition . typeOfLossCon )
{
2011-12-14 00:23:17 +03:00
case ELossConditionType : : LOSSCASTLE :
2010-01-29 22:52:45 +02:00
{
out . text . addTxt ( MetaString : : GENERAL_TXT , 251 ) ; //The town of %s has fallen - all is lost!
const CGTownInstance * t = dynamic_cast < const CGTownInstance * > ( gs - > map - > lossCondition . obj ) ;
assert ( t ) ;
out . text . addReplacement ( t - > name ) ;
}
break ;
2011-12-14 00:23:17 +03:00
case ELossConditionType : : LOSSHERO :
2010-01-29 22:52:45 +02:00
{
out . text . addTxt ( MetaString : : GENERAL_TXT , 253 ) ; //The hero, %s, has suffered defeat - your quest is over!
const CGHeroInstance * h = dynamic_cast < const CGHeroInstance * > ( gs - > map - > lossCondition . obj ) ;
assert ( h ) ;
out . text . addReplacement ( h - > name ) ;
}
break ;
2011-12-14 00:23:17 +03:00
case ELossConditionType : : TIMEEXPIRES :
2010-01-29 22:52:45 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 254 ) ; //Alas, time has run out on your quest. All is lost.
break ;
}
}
2010-02-02 01:30:03 +02:00
else if ( standard = = 2 )
{
out . text . addTxt ( MetaString : : GENERAL_TXT , 7 ) ; //%s, your heroes abandon you, and you are banished from this land.
2013-03-03 20:06:03 +03:00
out . text . addReplacement ( MetaString : : COLOR , player . getNum ( ) ) ;
out . components . push_back ( Component ( Component : : FLAG , player . getNum ( ) , 0 , 0 ) ) ;
2010-02-02 01:30:03 +02:00
}
2010-01-29 22:52:45 +02:00
else //lost all towns and heroes
{
2010-01-30 17:00:05 +02:00
out . text . addTxt ( MetaString : : GENERAL_TXT , 660 ) ; //All your forces have been defeated, and you are banished from this land!
2010-01-29 22:52:45 +02:00
}
}
2010-02-21 17:03:30 +02:00
}
bool CGameHandler : : dig ( const CGHeroInstance * h )
{
2013-02-06 02:16:13 +03:00
for ( auto i = gs - > map - > objects . cbegin ( ) ; i ! = gs - > map - > objects . cend ( ) ; i + + ) //unflag objs
2010-02-21 17:03:30 +02:00
{
2012-09-23 21:01:04 +03:00
if ( * i & & ( * i ) - > ID = = Obj : : HOLE & & ( * i ) - > pos = = h - > getPosition ( ) )
2010-02-21 17:03:30 +02:00
{
complain ( " Cannot dig - there is already a hole under the hero! " ) ;
return false ;
}
}
2012-01-03 04:55:26 +03:00
if ( h - > diggingStatus ( ) ! = CGHeroInstance : : CAN_DIG ) //checks for terrain and movement
COMPLAIN_RETF ( " Hero cannot dig (error code %d)! " , h - > diggingStatus ( ) ) ;
//create a hole
2010-02-21 17:03:30 +02:00
NewObject no ;
2012-09-23 21:01:04 +03:00
no . ID = Obj : : HOLE ;
2010-02-24 15:03:36 +02:00
no . pos = h - > getPosition ( ) ;
2012-11-06 19:39:29 +03:00
no . subID = getTile ( no . pos ) - > terType ;
2010-02-21 17:03:30 +02:00
sendAndApply ( & no ) ;
2012-07-23 13:23:43 +03:00
2012-01-03 04:55:26 +03:00
//take MPs
2010-02-24 15:03:36 +02:00
SetMovePoints smp ;
smp . hid = h - > id ;
smp . val = 0 ;
sendAndApply ( & smp ) ;
InfoWindow iw ;
iw . player = h - > tempOwner ;
if ( gs - > map - > grailPos = = h - > getPosition ( ) )
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 58 ) ; //"Congratulations! After spending many hours digging here, your hero has uncovered the "
2010-03-01 20:22:22 +02:00
iw . text . addTxt ( MetaString : : ART_NAMES , 2 ) ;
2010-02-24 15:03:36 +02:00
iw . soundID = soundBase : : ULTIMATEARTIFACT ;
2013-02-07 02:24:43 +03:00
giveHeroNewArtifact ( h , VLC - > arth - > artifacts [ 2 ] , ArtifactPosition : : PRE_FIRST ) ; //give grail
2010-03-01 20:22:22 +02:00
sendAndApply ( & iw ) ;
2012-02-20 00:03:43 +03:00
iw . soundID = soundBase : : invalid ;
2010-03-01 20:22:22 +02:00
iw . text . clear ( ) ;
iw . text . addTxt ( MetaString : : ART_DESCR , 2 ) ;
sendAndApply ( & iw ) ;
2010-02-24 15:03:36 +02:00
}
else
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 59 ) ; //"Nothing here. \n Where could it be?"
iw . soundID = soundBase : : Dig ;
2010-03-01 20:22:22 +02:00
sendAndApply ( & iw ) ;
2010-02-24 15:03:36 +02:00
}
2010-02-21 17:03:30 +02:00
return true ;
2010-02-23 17:39:31 +02:00
}
2011-07-16 09:42:44 +03:00
void CGameHandler : : attackCasting ( const BattleAttack & bat , Bonus : : BonusType attackMode , const CStack * attacker )
2010-02-23 17:39:31 +02:00
{
2011-07-16 09:42:44 +03:00
if ( attacker - > hasBonusOfType ( attackMode ) )
2010-02-23 17:39:31 +02:00
{
2013-02-14 02:55:42 +03:00
std : : set < SpellID > spellsToCast ;
2011-09-06 16:59:26 +03:00
TBonusListPtr spells = attacker - > getBonuses ( Selector : : type ( attackMode ) ) ;
2013-06-29 16:05:48 +03:00
for ( const Bonus * sf : * spells )
2010-02-23 17:39:31 +02:00
{
2013-02-14 02:55:42 +03:00
spellsToCast . insert ( SpellID ( sf - > subtype ) ) ;
2011-04-25 12:03:13 +03:00
}
2013-06-29 16:05:48 +03:00
for ( SpellID spellID : spellsToCast )
2011-04-25 12:03:13 +03:00
{
2013-06-26 14:18:27 +03:00
const CStack * oneOfAttacked = nullptr ;
2013-06-29 16:05:48 +03:00
for ( auto & elem : bat . bsa )
2010-02-23 17:39:31 +02:00
{
2013-06-29 16:05:48 +03:00
if ( elem . newAmount > 0 & & ! elem . isSecondary ( ) ) //apply effects only to first target stack if it's alive
2010-02-23 17:39:31 +02:00
{
2013-06-29 16:05:48 +03:00
oneOfAttacked = gs - > curB - > battleGetStackByID ( elem . stackAttacked ) ;
2011-04-25 12:03:13 +03:00
break ;
2010-02-23 17:39:31 +02:00
}
2011-04-25 12:03:13 +03:00
}
bool castMe = false ;
int meleeRanged ;
2013-06-26 14:18:27 +03:00
if ( oneOfAttacked = = nullptr ) //all attacked creatures have been killed
2011-04-25 12:03:13 +03:00
return ;
int spellLevel = 0 ;
2011-09-06 16:59:26 +03:00
TBonusListPtr spellsByType = attacker - > getBonuses ( Selector : : typeSubtype ( attackMode , spellID ) ) ;
2013-06-29 16:05:48 +03:00
for ( const Bonus * sf : * spellsByType )
2011-04-25 12:03:13 +03:00
{
2011-12-14 00:23:17 +03:00
vstd : : amax ( spellLevel , sf - > additionalInfo % 1000 ) ; //pick highest level
2011-04-25 12:03:13 +03:00
meleeRanged = sf - > additionalInfo / 1000 ;
if ( meleeRanged = = 0 | | ( meleeRanged = = 1 & & bat . shot ( ) ) | | ( meleeRanged = = 2 & & ! bat . shot ( ) ) )
castMe = true ;
}
2011-07-16 09:42:44 +03:00
int chance = attacker - > valOfBonuses ( ( Selector : : typeSubtype ( attackMode , spellID ) ) ) ;
2011-12-14 00:23:17 +03:00
vstd : : amin ( chance , 100 ) ;
2011-04-25 12:03:13 +03:00
int destination = oneOfAttacked - > position ;
2011-02-20 20:32:39 +02:00
2013-02-13 22:35:43 +03:00
const CSpell * spell = SpellID ( spellID ) . toSpell ( ) ;
2011-12-14 00:23:17 +03:00
if ( gs - > curB - > battleCanCastThisSpellHere ( attacker - > owner , spell , ECastingMode : : AFTER_ATTACK_CASTING , oneOfAttacked - > position ) ! = ESpellCastProblem : : OK )
2011-04-25 12:03:13 +03:00
continue ;
2010-02-23 17:39:31 +02:00
2011-04-25 12:03:13 +03:00
//check if spell should be casted (probability handling)
if ( rand ( ) % 100 > = chance )
continue ;
2011-07-16 09:42:44 +03:00
//casting //TODO: check if spell can be blocked or target is immune
2011-07-08 17:54:20 +03:00
if ( castMe ) //stacks use 0 spell power. If needed, default = 3 or custom value is used
2013-06-26 14:18:27 +03:00
handleSpellCasting ( spellID , spellLevel , destination , ! attacker - > attackerOwned , attacker - > owner , nullptr , nullptr , 0 , ECastingMode : : AFTER_ATTACK_CASTING , attacker ) ;
2010-02-23 17:39:31 +02:00
}
}
2011-07-16 09:42:44 +03:00
}
void CGameHandler : : handleAttackBeforeCasting ( const BattleAttack & bat )
{
2012-08-26 12:07:48 +03:00
const CStack * attacker = gs - > curB - > battleGetStackByID ( bat . stackAttacking ) ;
2011-07-16 09:42:44 +03:00
attackCasting ( bat , Bonus : : SPELL_BEFORE_ATTACK , attacker ) ; //no detah stare / acid bretah needed?
}
void CGameHandler : : handleAfterAttackCasting ( const BattleAttack & bat )
{
2012-08-26 12:07:48 +03:00
const CStack * attacker = gs - > curB - > battleGetStackByID ( bat . stackAttacking ) ;
2012-04-18 18:57:49 +03:00
if ( ! attacker ) //could be already dead
return ;
2011-07-16 09:42:44 +03:00
attackCasting ( bat , Bonus : : SPELL_AFTER_ATTACK , attacker ) ;
2012-03-26 01:46:14 +03:00
if ( bat . bsa [ 0 ] . newAmount < = 0 )
{
//don't try death stare or acid breath on dead stack (crash!)
return ;
}
2013-03-30 23:09:50 +03:00
if ( attacker - > hasBonusOfType ( Bonus : : DEATH_STARE ) & & bat . bsa . size ( ) )
2011-04-23 20:10:54 +03:00
{
2012-08-27 13:47:02 +03:00
// mechanics of Death Stare as in H3:
2012-09-26 16:13:39 +03:00
// each gorgon have 10% chance to kill (counted separately in H3) -> binomial distribution
2013-07-28 11:48:29 +03:00
//original formula x = min(x, (gorgons_count + 9)/10);
2012-08-27 13:47:02 +03:00
2013-07-28 11:48:29 +03:00
double chanceToKill = attacker - > valOfBonuses ( Bonus : : DEATH_STARE , 0 ) / 100.0f ;
2012-09-26 16:13:39 +03:00
vstd : : amin ( chanceToKill , 1 ) ; //cap at 100%
2012-08-27 13:47:02 +03:00
2013-06-26 14:18:27 +03:00
std : : binomial_distribution < > distr ( attacker - > count , chanceToKill ) ;
std : : mt19937 rng ( rand ( ) ) ;
int staredCreatures = distr ( rng ) ;
2012-08-27 13:47:02 +03:00
2013-07-28 11:48:29 +03:00
double cap = 1 / std : : max ( chanceToKill , ( double ) ( 0.01 ) ) ; //don't divide by 0
int maxToKill = ( attacker - > count + cap - 1 ) / cap ; //not much more than chance * count
2012-08-27 13:47:02 +03:00
vstd : : amin ( staredCreatures , maxToKill ) ;
2012-07-23 13:23:43 +03:00
2013-03-30 23:09:50 +03:00
staredCreatures + = ( attacker - > level ( ) * attacker - > valOfBonuses ( Bonus : : DEATH_STARE , 1 ) ) / gs - > curB - > battleGetStackByID ( bat . bsa [ 0 ] . stackAttacked ) - > level ( ) ;
2011-04-23 20:10:54 +03:00
if ( staredCreatures )
{
2013-03-30 23:09:50 +03:00
if ( bat . bsa [ 0 ] . newAmount > 0 ) //TODO: death stare was not originally available for multiple-hex attacks, but...
2013-02-11 02:24:57 +03:00
handleSpellCasting ( SpellID : : DEATH_STARE , 0 , gs - > curB - > battleGetStackByID ( bat . bsa [ 0 ] . stackAttacked ) - > position ,
2013-06-26 14:18:27 +03:00
! attacker - > attackerOwned , attacker - > owner , nullptr , nullptr , staredCreatures , ECastingMode : : AFTER_ATTACK_CASTING , attacker ) ;
2011-04-23 20:10:54 +03:00
}
}
2012-03-26 01:46:14 +03:00
2011-05-13 12:02:16 +03:00
int acidDamage = 0 ;
2011-09-06 16:59:26 +03:00
TBonusListPtr acidBreath = attacker - > getBonuses ( Selector : : type ( Bonus : : ACID_BREATH ) ) ;
2013-06-29 16:05:48 +03:00
for ( const Bonus * b : * acidBreath )
2011-05-13 12:02:16 +03:00
{
if ( b - > additionalInfo > rand ( ) % 100 )
acidDamage + = b - > val ;
}
if ( acidDamage )
{
2013-02-11 02:24:57 +03:00
handleSpellCasting ( SpellID : : ACID_BREATH_DAMAGE , 0 , gs - > curB - > battleGetStackByID ( bat . bsa [ 0 ] . stackAttacked ) - > position ,
2013-06-26 14:18:27 +03:00
! attacker - > attackerOwned , attacker - > owner , nullptr , nullptr ,
2011-12-14 00:23:17 +03:00
acidDamage * attacker - > count , ECastingMode : : AFTER_ATTACK_CASTING , attacker ) ;
2011-05-13 12:02:16 +03:00
}
2010-02-23 17:39:31 +02:00
}
2010-03-11 01:16:30 +02:00
2013-02-11 02:24:57 +03:00
bool CGameHandler : : castSpell ( const CGHeroInstance * h , SpellID spellID , const int3 & pos )
2010-03-11 01:16:30 +02:00
{
2013-02-11 02:24:57 +03:00
const CSpell * s = spellID . toSpell ( ) ;
2010-03-11 01:16:30 +02:00
int cost = h - > getSpellCost ( s ) ;
int schoolLevel = h - > getSpellSchoolLevel ( s ) ;
if ( ! h - > canCastThisSpell ( s ) )
COMPLAIN_RET ( " Hero cannot cast this spell! " ) ;
2012-07-23 13:23:43 +03:00
if ( h - > mana < cost )
2010-03-11 01:16:30 +02:00
COMPLAIN_RET ( " Hero doesn't have enough spell points to cast this spell! " ) ;
if ( s - > combatSpell )
COMPLAIN_RET ( " This function can be used only for adventure map spells! " ) ;
2010-05-16 16:42:19 +03:00
AdvmapSpellCast asc ;
asc . caster = h ;
asc . spellID = spellID ;
sendAndApply ( & asc ) ;
2010-03-11 01:16:30 +02:00
switch ( spellID )
{
2013-02-11 02:24:57 +03:00
case SpellID : : SUMMON_BOAT :
2010-03-11 01:16:30 +02:00
{
//check if spell works at all
if ( rand ( ) % 100 > = s - > powers [ schoolLevel ] ) //power is % chance of success
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 336 ) ; //%s tried to summon a boat, but failed.
iw . text . addReplacement ( h - > name ) ;
sendAndApply ( & iw ) ;
2012-05-15 11:47:11 +03:00
break ;
2010-03-11 01:16:30 +02:00
}
//try to find unoccupied boat to summon
2013-06-26 14:18:27 +03:00
const CGBoat * nearest = nullptr ;
2010-03-11 01:16:30 +02:00
double dist = 0 ;
int3 summonPos = h - > bestLocation ( ) ;
if ( summonPos . x < 0 )
COMPLAIN_RET ( " There is no water tile available! " ) ;
2013-06-29 16:05:48 +03:00
for ( const CGObjectInstance * obj : gs - > map - > objects )
2010-03-11 01:16:30 +02:00
{
2012-09-23 21:01:04 +03:00
if ( obj & & obj - > ID = = Obj : : BOAT )
2010-03-11 01:16:30 +02:00
{
const CGBoat * b = static_cast < const CGBoat * > ( obj ) ;
if ( b - > hero ) continue ; //we're looking for unoccupied boat
double nDist = distance ( b - > pos , h - > getPosition ( ) ) ;
if ( ! nearest | | nDist < dist ) //it's first boat or closer than previous
{
nearest = b ;
dist = nDist ;
}
}
}
if ( nearest ) //we found boat to summon
{
ChangeObjPos cop ;
cop . objid = nearest - > id ;
cop . nPos = summonPos + int3 ( 1 , 0 , 0 ) ; ;
cop . flags = 1 ;
sendAndApply ( & cop ) ;
}
else if ( schoolLevel < 2 ) //none or basic level -> cannot create boat :(
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 335 ) ; //There are no boats to summon.
sendAndApply ( & iw ) ;
}
else //create boat
{
NewObject no ;
2013-02-07 20:34:50 +03:00
no . ID = Obj : : BOAT ;
2010-03-11 01:16:30 +02:00
no . subID = h - > getBoatType ( ) ;
no . pos = summonPos + int3 ( 1 , 0 , 0 ) ; ;
sendAndApply ( & no ) ;
}
break ;
}
2013-02-11 02:24:57 +03:00
case SpellID : : SCUTTLE_BOAT :
2010-03-21 00:17:19 +02:00
{
//check if spell works at all
if ( rand ( ) % 100 > = s - > powers [ schoolLevel ] ) //power is % chance of success
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 337 ) ; //%s tried to scuttle the boat, but failed
iw . text . addReplacement ( h - > name ) ;
sendAndApply ( & iw ) ;
2012-05-15 11:47:11 +03:00
break ;
2010-03-21 00:17:19 +02:00
}
2010-06-01 00:14:15 +03:00
if ( ! gs - > map - > isInTheMap ( pos ) )
COMPLAIN_RET ( " Invalid dst tile for scuttle! " ) ;
2010-03-21 00:17:19 +02:00
//TODO: test range, visibility
const TerrainTile * t = & gs - > map - > getTile ( pos ) ;
2012-09-23 21:01:04 +03:00
if ( ! t - > visitableObjects . size ( ) | | t - > visitableObjects . back ( ) - > ID ! = Obj : : BOAT )
2010-03-21 00:17:19 +02:00
COMPLAIN_RET ( " There is no boat to scuttle! " ) ;
RemoveObject ro ;
ro . id = t - > visitableObjects . back ( ) - > id ;
sendAndApply ( & ro ) ;
break ;
}
2013-02-11 02:24:57 +03:00
case SpellID : : DIMENSION_DOOR :
2010-03-21 00:17:19 +02:00
{
const TerrainTile * dest = getTile ( pos ) ;
const TerrainTile * curr = getTile ( h - > getSightCenter ( ) ) ;
if ( ! dest )
COMPLAIN_RET ( " Destination tile doesn't exist! " ) ;
if ( ! h - > movement )
COMPLAIN_RET ( " Hero needs movement points to cast Dimension Door! " ) ;
2013-02-11 02:24:57 +03:00
if ( h - > getBonusesCount ( Bonus : : SPELL_EFFECT , SpellID : : DIMENSION_DOOR ) > = s - > powers [ schoolLevel ] ) //limit casts per turn
2010-03-21 00:17:19 +02:00
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 338 ) ; //%s is not skilled enough to cast this spell again today.
iw . text . addReplacement ( h - > name ) ;
sendAndApply ( & iw ) ;
break ;
}
GiveBonus gb ;
2013-02-14 02:55:42 +03:00
gb . id = h - > id . getNum ( ) ;
2013-02-11 02:24:57 +03:00
gb . bonus = Bonus ( Bonus : : ONE_DAY , Bonus : : NONE , Bonus : : SPELL_EFFECT , 0 , SpellID : : DIMENSION_DOOR ) ;
2010-03-21 00:17:19 +02:00
sendAndApply ( & gb ) ;
2012-07-23 13:23:43 +03:00
2010-03-21 00:17:19 +02:00
if ( ! dest - > isClear ( curr ) ) //wrong dest tile
{
InfoWindow iw ;
iw . player = h - > tempOwner ;
iw . text . addTxt ( MetaString : : GENERAL_TXT , 70 ) ; //Dimension Door failed!
sendAndApply ( & iw ) ;
break ;
}
2012-11-16 00:29:22 +03:00
if ( moveHero ( h - > id , pos + h - > getVisitableOffset ( ) , true ) )
{
SetMovePoints smp ;
smp . hid = h - > id ;
smp . val = std : : max < ui32 > ( 0 , h - > movement - 300 ) ;
sendAndApply ( & smp ) ;
}
2010-03-21 00:17:19 +02:00
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : FLY :
2010-05-15 18:00:19 +03:00
{
int subtype = schoolLevel > = 2 ? 1 : 2 ; //adv or expert
GiveBonus gb ;
2013-02-14 02:55:42 +03:00
gb . id = h - > id . getNum ( ) ;
2013-02-11 02:24:57 +03:00
gb . bonus = Bonus ( Bonus : : ONE_DAY , Bonus : : FLYING_MOVEMENT , Bonus : : SPELL_EFFECT , 0 , SpellID : : FLY , subtype ) ;
2010-05-15 18:00:19 +03:00
sendAndApply ( & gb ) ;
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : WATER_WALK :
2010-05-15 18:00:19 +03:00
{
int subtype = schoolLevel > = 2 ? 1 : 2 ; //adv or expert
GiveBonus gb ;
2013-02-14 02:55:42 +03:00
gb . id = h - > id . getNum ( ) ;
2013-02-11 02:24:57 +03:00
gb . bonus = Bonus ( Bonus : : ONE_DAY , Bonus : : WATER_WALKING , Bonus : : SPELL_EFFECT , 0 , SpellID : : WATER_WALK , subtype ) ;
2010-05-15 18:00:19 +03:00
sendAndApply ( & gb ) ;
}
break ;
2012-07-23 13:23:43 +03:00
2013-02-11 02:24:57 +03:00
case SpellID : : TOWN_PORTAL :
2010-08-04 16:41:01 +03:00
{
2011-01-15 19:30:46 +02:00
if ( ! gs - > map - > isInTheMap ( pos ) )
COMPLAIN_RET ( " Destination tile not present! " )
TerrainTile tile = gs - > map - > getTile ( pos ) ;
2012-09-23 21:01:04 +03:00
if ( tile . visitableObjects . empty ( ) | | tile . visitableObjects . back ( ) - > ID ! = Obj : : TOWN )
2011-01-15 19:30:46 +02:00
COMPLAIN_RET ( " Town not found for Town Portal! " ) ;
2012-07-23 13:23:43 +03:00
2011-01-15 19:30:46 +02:00
CGTownInstance * town = static_cast < CGTownInstance * > ( tile . visitableObjects . back ( ) ) ;
if ( town - > tempOwner ! = h - > tempOwner )
COMPLAIN_RET ( " Can't teleport to another player! " ) ;
if ( town - > visitingHero )
COMPLAIN_RET ( " Can't teleport to occupied town! " ) ;
2012-07-23 13:23:43 +03:00
2011-01-15 19:30:46 +02:00
if ( h - > getSpellSchoolLevel ( s ) < 2 )
{
double dist = town - > pos . dist2d ( h - > pos ) ;
2013-02-14 02:55:42 +03:00
ObjectInstanceID nearest = town - > id ; //nearest town's ID
2013-06-29 16:05:48 +03:00
for ( const CGTownInstance * currTown : gs - > getPlayer ( h - > tempOwner ) - > towns )
2011-01-15 19:30:46 +02:00
{
double curDist = currTown - > pos . dist2d ( h - > pos ) ;
2013-02-14 02:55:42 +03:00
if ( nearest = = ObjectInstanceID ( ) | | curDist < dist )
2011-01-15 19:30:46 +02:00
{
nearest = town - > id ;
dist = curDist ;
}
}
if ( town - > id ! = nearest )
COMPLAIN_RET ( " This hero can only teleport to nearest town! " )
}
2012-11-16 00:29:22 +03:00
moveHero ( h - > id , town - > visitablePos ( ) + h - > getVisitableOffset ( ) , 1 ) ;
2010-08-04 16:41:01 +03:00
}
break ;
2013-02-11 02:24:57 +03:00
case SpellID : : VISIONS :
case SpellID : : VIEW_EARTH :
case SpellID : : DISGUISE :
case SpellID : : VIEW_AIR :
2010-03-11 01:16:30 +02:00
default :
COMPLAIN_RET ( " This spell is not implemented yet! " ) ;
2010-08-04 16:41:01 +03:00
break ;
2010-03-11 01:16:30 +02:00
}
SetMana sm ;
sm . hid = h - > id ;
sm . val = h - > mana - cost ;
sendAndApply ( & sm ) ;
return true ;
2010-05-08 01:10:32 +03:00
}
2010-05-27 00:59:58 +03:00
void CGameHandler : : visitObjectOnTile ( const TerrainTile & t , const CGHeroInstance * h )
{
2012-11-16 00:29:22 +03:00
if ( ! t . visitableObjects . empty ( ) )
{
//to prevent self-visiting heroes on space press
if ( t . visitableObjects . back ( ) ! = h )
objectVisited ( t . visitableObjects . back ( ) , h ) ;
else if ( t . visitableObjects . size ( ) > 1 )
objectVisited ( * ( t . visitableObjects . end ( ) - 2 ) , h ) ;
}
2010-06-01 00:14:15 +03:00
}
2013-02-16 17:03:47 +03:00
bool CGameHandler : : sacrificeCreatures ( const IMarket * market , const CGHeroInstance * hero , SlotID slot , ui32 count )
2010-07-20 09:05:45 +03:00
{
2010-11-27 03:46:19 +02:00
int oldCount = hero - > getStackCount ( slot ) ;
2010-07-20 09:05:45 +03:00
if ( oldCount < count )
COMPLAIN_RET ( " Not enough creatures to sacrifice! " )
else if ( oldCount = = count & & hero - > Slots ( ) . size ( ) = = 1 & & hero - > needsLastStack ( ) )
COMPLAIN_RET ( " Cannot sacrifice last creature! " ) ;
int crid = hero - > getStack ( slot ) . type - > idNumber ;
2012-07-23 13:23:43 +03:00
2010-11-27 22:17:28 +02:00
changeStackCount ( StackLocation ( hero , slot ) , - count ) ;
2010-07-20 09:05:45 +03:00
int dump , exp ;
2011-12-14 00:23:17 +03:00
market - > getOffer ( crid , 0 , dump , exp , EMarketMode : : CREATURE_EXP ) ;
2010-07-20 09:05:45 +03:00
exp * = count ;
2013-02-09 00:17:39 +03:00
changePrimSkill ( hero , PrimarySkill : : EXPERIENCE , hero - > calculateXp ( exp ) ) ;
2010-07-20 09:05:45 +03:00
2010-07-23 15:02:15 +03:00
return true ;
}
2013-02-12 22:49:40 +03:00
bool CGameHandler : : sacrificeArtifact ( const IMarket * m , const CGHeroInstance * hero , ArtifactPosition slot )
2010-07-23 15:02:15 +03:00
{
2010-12-29 23:04:22 +02:00
ArtifactLocation al ( hero , slot ) ;
const CArtifactInstance * a = al . getArt ( ) ;
if ( ! a )
2010-07-23 15:02:15 +03:00
COMPLAIN_RET ( " Cannot find artifact to sacrifice! " ) ;
2010-12-29 23:04:22 +02:00
2010-07-23 15:02:15 +03:00
int dmp , expToGive ;
2011-12-14 00:23:17 +03:00
m - > getOffer ( hero - > getArtTypeId ( slot ) , 0 , dmp , expToGive , EMarketMode : : ARTIFACT_EXP ) ;
2010-12-29 23:04:22 +02:00
removeArtifact ( al ) ;
2013-02-09 00:17:39 +03:00
changePrimSkill ( hero , PrimarySkill : : EXPERIENCE , expToGive ) ;
2010-06-01 00:14:15 +03:00
return true ;
2010-09-04 17:47:39 +03:00
}
2010-11-27 03:46:19 +02:00
2010-12-04 21:15:20 +02:00
void CGameHandler : : makeStackDoNothing ( const CStack * next )
{
BattleAction doNothing ;
2013-02-04 00:05:44 +03:00
doNothing . actionType = Battle : : NO_ACTION ;
2010-12-04 21:15:20 +02:00
doNothing . additionalInfo = 0 ;
doNothing . destinationTile = - 1 ;
doNothing . side = ! next - > attackerOwned ;
doNothing . stackNumber = next - > ID ;
2011-06-11 07:54:41 +03:00
2012-08-27 15:34:43 +03:00
makeAutomaticAction ( next , doNothing ) ;
2010-12-04 21:15:20 +02:00
}
2010-12-06 01:26:56 +02:00
2010-11-27 22:17:28 +02:00
bool CGameHandler : : insertNewStack ( const StackLocation & sl , const CCreature * c , TQuantity count )
2010-11-27 03:46:19 +02:00
{
2010-11-27 22:17:28 +02:00
if ( sl . army - > hasStackAtSlot ( sl . slot ) )
COMPLAIN_RET ( " Slot is already taken! " ) ;
2013-02-16 17:03:47 +03:00
if ( ! sl . slot . validSlot ( ) )
2012-08-30 19:01:19 +03:00
COMPLAIN_RET ( " Cannot insert stack to that slot! " ) ;
2010-11-27 03:46:19 +02:00
InsertNewStack ins ;
ins . sl = sl ;
2010-12-06 01:10:02 +02:00
ins . stack = CStackBasicDescriptor ( c , count ) ;
2010-11-27 03:46:19 +02:00
sendAndApply ( & ins ) ;
2010-11-27 22:17:28 +02:00
return true ;
2010-11-27 03:46:19 +02:00
}
2010-12-12 01:11:26 +02:00
bool CGameHandler : : eraseStack ( const StackLocation & sl , bool forceRemoval /* = false*/ )
2010-11-27 03:46:19 +02:00
{
2010-11-27 22:17:28 +02:00
if ( ! sl . army - > hasStackAtSlot ( sl . slot ) )
COMPLAIN_RET ( " Cannot find a stack to erase " ) ;
if ( sl . army - > Slots ( ) . size ( ) = = 1 //from the last stack
2010-12-12 01:11:26 +02:00
& & sl . army - > needsLastStack ( ) //that must be left
& & ! forceRemoval ) //ignore above conditions if we are forcing removal
2010-11-27 22:17:28 +02:00
{
COMPLAIN_RET ( " Cannot erase the last stack! " ) ;
}
2010-11-27 03:46:19 +02:00
EraseStack es ;
es . sl = sl ;
sendAndApply ( & es ) ;
2010-11-27 22:17:28 +02:00
return true ;
}
bool CGameHandler : : changeStackCount ( const StackLocation & sl , TQuantity count , bool absoluteValue /*= false*/ )
{
TQuantity currentCount = sl . army - > getStackCount ( sl . slot ) ;
2011-05-30 22:20:14 +03:00
if ( ( absoluteValue & & count < 0 )
| | ( ! absoluteValue & & - count > currentCount ) )
2010-11-27 22:17:28 +02:00
{
COMPLAIN_RET ( " Cannot take more stacks than present! " ) ;
}
2011-05-30 22:20:14 +03:00
if ( ( currentCount = = - count & & ! absoluteValue )
| | ( ! count & & absoluteValue ) )
2010-11-27 22:17:28 +02:00
{
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 ) ;
if ( ! slotC ) //slot is empty
insertNewStack ( sl , c , count ) ;
else if ( c = = slotC )
changeStackCount ( sl , count ) ;
else
{
2010-12-13 01:44:16 +02:00
COMPLAIN_RET ( " Cannot add " + c - > namePl + " to slot " + boost : : lexical_cast < std : : string > ( sl . slot ) + " ! " ) ;
2010-11-27 22:17:28 +02:00
}
return true ;
}
2010-12-13 01:44:16 +02:00
void CGameHandler : : tryJoiningArmy ( const CArmedInstance * src , const CArmedInstance * dst , bool removeObjWhenFinished , bool allowMerging )
2010-11-27 22:17:28 +02:00
{
2013-05-14 16:42:52 +03:00
if ( removeObjWhenFinished )
removeAfterVisit ( src ) ;
2011-07-18 18:21:16 +03:00
if ( ! src - > canBeMergedWith ( * dst , allowMerging ) )
2010-11-27 22:17:28 +02:00
{
2011-03-08 15:27:32 +02:00
if ( allowMerging ) //do that, add all matching creatures.
{
bool cont = true ;
while ( cont )
{
2013-06-29 16:05:48 +03:00
for ( auto i = src - > stacks . begin ( ) ; i ! = src - > stacks . end ( ) ; i + + ) //while there are unmoved creatures
2011-03-08 15:27:32 +02:00
{
2013-02-16 17:03:47 +03:00
SlotID pos = dst - > getSlotFor ( i - > second - > type ) ;
if ( pos . validSlot ( ) )
2011-03-08 15:27:32 +02:00
{
moveStack ( StackLocation ( src , i - > first ) , StackLocation ( dst , pos ) ) ;
cont = true ;
break ; //or iterator crashes
}
cont = false ;
}
}
2012-07-23 13:23:43 +03:00
}
2013-05-14 16:42:52 +03:00
showGarrisonDialog ( src - > id , dst - > id , true ) ; //show garrison window and optionally remove ourselves from map when player ends
2010-11-27 22:17:28 +02:00
}
else //merge
{
2010-12-13 01:44:16 +02:00
moveArmy ( src , dst , allowMerging ) ;
2010-11-27 22:17:28 +02:00
}
2010-11-27 03:46:19 +02:00
}
2010-11-27 22:17:28 +02:00
bool CGameHandler : : moveStack ( const StackLocation & src , const StackLocation & dst , TQuantity count )
2010-11-27 03:46:19 +02:00
{
2010-11-27 22:17:28 +02:00
if ( ! src . army - > hasStackAtSlot ( src . slot ) )
COMPLAIN_RET ( " No stack to move! " ) ;
if ( dst . army - > hasStackAtSlot ( dst . slot ) & & dst . army - > getCreature ( dst . slot ) ! = src . army - > getCreature ( src . slot ) )
COMPLAIN_RET ( " Cannot move: stack of different type at destination pos! " ) ;
2013-02-16 17:03:47 +03:00
if ( ! dst . slot . validSlot ( ) )
2012-08-30 19:01:19 +03:00
COMPLAIN_RET ( " Cannot move stack to that slot! " ) ;
2010-11-27 22:17:28 +02:00
if ( count = = - 1 )
{
count = src . army - > getStackCount ( src . slot ) ;
}
if ( src . army ! = dst . army //moving away
& & count = = src . army - > getStackCount ( src . slot ) //all creatures
& & src . army - > Slots ( ) . size ( ) = = 1 //from the last stack
& & src . army - > needsLastStack ( ) ) //that must be left
{
2012-08-30 19:01:19 +03:00
COMPLAIN_RET ( " Cannot move away the last creature! " ) ;
2010-11-27 22:17:28 +02:00
}
RebalanceStacks rs ;
rs . src = src ;
rs . dst = dst ;
rs . count = count ;
sendAndApply ( & rs ) ;
return true ;
}
bool CGameHandler : : swapStacks ( const StackLocation & sl1 , const StackLocation & sl2 )
{
if ( ! sl1 . army - > hasStackAtSlot ( sl1 . slot ) )
return moveStack ( sl2 , sl1 ) ;
else if ( ! sl2 . army - > hasStackAtSlot ( sl2 . slot ) )
return moveStack ( sl1 , sl2 ) ;
else
{
SwapStacks ss ;
ss . sl1 = sl1 ;
ss . sl2 = sl2 ;
sendAndApply ( & ss ) ;
return true ;
}
2010-12-12 01:11:26 +02:00
}
2010-12-22 22:14:40 +02:00
void CGameHandler : : runBattle ( )
{
2012-08-26 12:07:48 +03:00
setBattle ( gs - > curB ) ;
2010-12-22 22:14:40 +02:00
assert ( gs - > curB ) ;
//TODO: pre-tactic stuff, call scripts etc.
//tactic round
{
2011-08-26 00:08:53 +03:00
while ( gs - > curB - > tacticDistance & & ! battleResult . get ( ) )
2011-02-12 18:12:48 +02:00
boost : : this_thread : : sleep ( boost : : posix_time : : milliseconds ( 50 ) ) ;
2010-12-22 22:14:40 +02:00
}
//spells opening battle
2013-07-22 01:01:29 +03:00
for ( int i = 0 ; i < 2 ; + + i )
2010-12-22 22:14:40 +02:00
{
2013-07-22 01:01:29 +03:00
auto h = gs - > curB - > battleGetFightingHero ( i ) ;
if ( h & & h - > hasBonusOfType ( Bonus : : OPENING_BATTLE_SPELL ) )
2010-12-22 22:14:40 +02:00
{
2013-07-22 01:01:29 +03:00
TBonusListPtr bl = h - > getBonuses ( Selector : : type ( Bonus : : OPENING_BATTLE_SPELL ) ) ;
2013-06-29 16:05:48 +03:00
for ( Bonus * b : * bl )
2011-02-22 13:52:36 +02:00
{
2013-07-22 01:01:29 +03:00
handleSpellCasting ( SpellID ( b - > subtype ) , 3 , - 1 , 0 , h - > tempOwner , nullptr , gs - > curB - > battleGetFightingHero ( 1 - i ) , b - > val , ECastingMode : : HERO_CASTING , nullptr ) ;
2011-02-22 13:52:36 +02:00
}
2010-12-22 22:14:40 +02:00
}
}
//main loop
while ( ! battleResult . get ( ) ) //till the end of the battle ;]
{
NEW_ROUND ;
2012-05-18 23:50:16 +03:00
auto obstacles = gs - > curB - > obstacles ; //we copy container, because we're going to modify it
2013-06-29 16:05:48 +03:00
for ( auto & obstPtr : obstacles )
2012-05-18 23:50:16 +03:00
{
if ( const SpellCreatedObstacle * sco = dynamic_cast < const SpellCreatedObstacle * > ( obstPtr . get ( ) ) )
if ( sco - > turnsRemaining = = 0 )
removeObstacle ( * obstPtr ) ;
2012-07-23 13:23:43 +03:00
}
2012-05-18 23:50:16 +03:00
2010-12-22 22:14:40 +02:00
const BattleInfo & curB = * gs - > curB ;
2013-09-08 20:43:10 +03:00
//remove clones after all mechanics and animations are handled!
std : : set < const CStack * > stacksToRemove ;
for ( auto stack : curB . stacks )
{
if ( stack - > idDeadClone ( ) )
stacksToRemove . insert ( stack ) ;
}
for ( auto stack : stacksToRemove )
{
BattleStacksRemoved bsr ;
bsr . stackIDs . insert ( stack - > ID ) ;
sendAndApply ( & bsr ) ;
}
2010-12-22 22:14:40 +02:00
//stack loop
2013-09-08 20:43:10 +03:00
2010-12-22 22:14:40 +02:00
const CStack * next ;
while ( ! battleResult . get ( ) & & ( next = curB . getNextStack ( ) ) & & next - > willMove ( ) )
{
//check for bad morale => freeze
int nextStackMorale = next - > MoraleVal ( ) ;
if ( nextStackMorale < 0 & &
2013-07-22 01:01:29 +03: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)
2010-12-22 22:14:40 +02:00
)
{
if ( rand ( ) % 24 < - 2 * nextStackMorale )
{
//unit loses its turn - empty freeze action
BattleAction ba ;
2013-02-04 00:05:44 +03:00
ba . actionType = Battle : : BAD_MORALE ;
2010-12-22 22:14:40 +02:00
ba . additionalInfo = 1 ;
ba . side = ! next - > attackerOwned ;
ba . stackNumber = next - > ID ;
2011-06-11 07:54:41 +03:00
2012-08-27 15:34:43 +03:00
makeAutomaticAction ( next , ba ) ;
2010-12-22 22:14:40 +02:00
continue ;
}
}
if ( next - > hasBonusOfType ( Bonus : : ATTACKS_NEAREST_CREATURE ) ) //while in berserk
2013-04-27 09:39:59 +03:00
{ //fixme: stack should not attack itself
2013-05-25 10:06:00 +03:00
std : : pair < const CStack * , int > attackInfo = curB . getNearestStack ( next , boost : : logic : : indeterminate ) ;
2013-06-26 14:18:27 +03:00
if ( attackInfo . first ! = nullptr )
2010-12-22 22:14:40 +02:00
{
BattleAction attack ;
2013-02-04 00:05:44 +03:00
attack . actionType = Battle : : WALK_AND_ATTACK ;
2010-12-22 22:14:40 +02:00
attack . side = ! next - > attackerOwned ;
attack . stackNumber = next - > ID ;
attack . additionalInfo = attackInfo . first - > position ;
attack . destinationTile = attackInfo . second ;
2012-08-27 15:34:43 +03:00
makeAutomaticAction ( next , attack ) ;
2010-12-22 22:14:40 +02:00
}
else
{
makeStackDoNothing ( next ) ;
}
continue ;
}
2011-01-07 12:48:31 +02:00
const CGHeroInstance * curOwner = gs - > curB - > battleGetOwner ( next ) ;
2010-12-22 22:14:40 +02:00
2013-02-09 00:17:39 +03:00
if ( ( next - > position < 0 | | next - > getCreature ( ) - > idNumber = = CreatureID : : BALLISTA ) //arrow turret or ballista
2013-02-04 22:43:16 +03:00
& & ( ! curOwner | | curOwner - > getSecSkillLevel ( SecondarySkill : : ARTILLERY ) = = 0 ) ) //hero has no artillery
2010-12-22 22:14:40 +02:00
{
BattleAction attack ;
2013-02-04 00:05:44 +03:00
attack . actionType = Battle : : SHOOT ;
2010-12-22 22:14:40 +02:00
attack . side = ! next - > attackerOwned ;
attack . stackNumber = next - > ID ;
2013-06-29 16:05:48 +03:00
for ( auto & elem : gs - > curB - > stacks )
2010-12-22 22:14:40 +02:00
{
2013-06-29 16:05:48 +03:00
if ( elem - > owner ! = next - > owner & & elem - > isValidTarget ( ) )
2010-12-22 22:14:40 +02:00
{
2013-06-29 16:05:48 +03:00
attack . destinationTile = elem - > position ;
2010-12-22 22:14:40 +02:00
break ;
}
}
2012-08-27 15:34:43 +03:00
makeAutomaticAction ( next , attack ) ;
2010-12-22 22:14:40 +02:00
continue ;
}
2013-02-09 00:17:39 +03:00
if ( next - > getCreature ( ) - > idNumber = = CreatureID : : CATAPULT & & ( ! curOwner | | curOwner - > getSecSkillLevel ( SecondarySkill : : BALLISTICS ) = = 0 ) ) //catapult, hero has no ballistics
2010-12-22 22:14:40 +02:00
{
BattleAction attack ;
static const int wallHexes [ ] = { 50 , 183 , 182 , 130 , 62 , 29 , 12 , 95 } ;
attack . destinationTile = wallHexes [ rand ( ) % ARRAY_COUNT ( wallHexes ) ] ;
2013-02-04 00:05:44 +03:00
attack . actionType = Battle : : CATAPULT ;
2010-12-22 22:14:40 +02:00
attack . additionalInfo = 0 ;
attack . side = ! next - > attackerOwned ;
attack . stackNumber = next - > ID ;
2012-08-27 15:34:43 +03:00
makeAutomaticAction ( next , attack ) ;
2010-12-22 22:14:40 +02:00
continue ;
}
2012-06-10 16:07:08 +03:00
2013-02-09 00:17:39 +03:00
if ( next - > getCreature ( ) - > idNumber = = CreatureID : : FIRST_AID_TENT )
2010-12-22 22:14:40 +02:00
{
2012-06-10 16:07:08 +03:00
std : : vector < const CStack * > possibleStacks ;
2010-12-22 22:14:40 +02:00
2012-08-28 19:28:21 +03:00
//is there any clean algorithm for that? (boost.range seems to lack copy_if) -> remove_copy_if?
2013-06-29 16:05:48 +03:00
for ( const CStack * s : battleGetAllStacks ( ) )
2012-06-10 16:07:08 +03:00
if ( s - > owner = = next - > owner & & s - > canBeHealed ( ) )
possibleStacks . push_back ( s ) ;
2010-12-22 22:14:40 +02:00
2012-06-10 16:07:08 +03:00
if ( ! possibleStacks . size ( ) )
2010-12-22 22:14:40 +02:00
{
makeStackDoNothing ( next ) ;
continue ;
}
2012-06-10 16:07:08 +03:00
2013-02-04 22:43:16 +03:00
if ( ! curOwner | | curOwner - > getSecSkillLevel ( SecondarySkill : : FIRST_AID ) = = 0 ) //no hero or hero has no first aid
2010-12-22 22:14:40 +02:00
{
2012-06-10 16:07:08 +03:00
range : : random_shuffle ( possibleStacks ) ;
const CStack * toBeHealed = possibleStacks . front ( ) ;
BattleAction heal ;
2013-02-04 00:05:44 +03:00
heal . actionType = Battle : : STACK_HEAL ;
2010-12-22 22:14:40 +02:00
heal . additionalInfo = 0 ;
heal . destinationTile = toBeHealed - > position ;
heal . side = ! next - > attackerOwned ;
heal . stackNumber = next - > ID ;
2012-08-27 15:34:43 +03:00
makeAutomaticAction ( next , heal ) ;
2012-06-10 16:07:08 +03:00
continue ;
2010-12-22 22:14:40 +02:00
}
}
int numberOfAsks = 1 ;
bool breakOuter = false ;
2012-07-23 13:23:43 +03:00
do
2010-12-22 22:14:40 +02:00
{ //ask interface and wait for answer
if ( ! battleResult . get ( ) )
{
2011-10-08 16:02:58 +03:00
stackTurnTrigger ( next ) ; //various effects
2011-12-14 00:23:17 +03:00
if ( vstd : : contains ( next - > state , EBattleStackState : : FEAR ) )
2011-10-08 20:10:43 +03:00
{
makeStackDoNothing ( next ) ; //end immediately if stack was affected by fear
}
else
{
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Activating " < < next - > nodeName ( ) ;
2011-10-08 20:10:43 +03:00
BattleSetActiveStack sas ;
sas . stack = next - > ID ;
sendAndApply ( & sas ) ;
boost : : unique_lock < boost : : mutex > lock ( battleMadeAction . mx ) ;
battleMadeAction . data = false ;
while ( next - > alive ( ) & &
( ! battleMadeAction . data & & ! battleResult . get ( ) ) ) //active stack hasn't made its action and battle is still going
battleMadeAction . cond . wait ( lock ) ;
}
2010-12-22 22:14:40 +02:00
}
if ( battleResult . get ( ) ) //don't touch it, battle could be finished while waiting got action
{
breakOuter = true ;
break ;
}
//we're after action, all results applied
2012-08-27 15:34:43 +03:00
checkForBattleEnd ( ) ; //check if this action ended the battle
2010-12-22 22:14:40 +02:00
//check for good morale
nextStackMorale = next - > MoraleVal ( ) ;
2011-12-14 00:23:17 +03:00
if ( ! vstd : : contains ( next - > state , EBattleStackState : : HAD_MORALE ) //only one extra move per turn possible
& & ! vstd : : contains ( next - > state , EBattleStackState : : DEFENDING )
2012-09-20 19:55:21 +03:00
& & ! next - > waited ( )
2011-12-14 00:23:17 +03:00
& & ! vstd : : contains ( next - > state , EBattleStackState : : FEAR )
2010-12-22 22:14:40 +02:00
& & next - > alive ( )
& & nextStackMorale > 0
2013-07-22 01:01:29 +03: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
2010-12-22 22:14:40 +02:00
)
{
if ( rand ( ) % 24 < nextStackMorale ) //this stack hasn't got morale this turn
2012-01-26 19:48:53 +03: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
}
2010-12-22 22:14:40 +02:00
}
- - numberOfAsks ;
} while ( numberOfAsks > 0 ) ;
if ( breakOuter )
{
break ;
}
}
}
2013-07-22 01:01:29 +03:00
endBattle ( gs - > curB - > tile , gs - > curB - > battleGetFightingHero ( 0 ) , gs - > curB - > battleGetFightingHero ( 1 ) ) ;
2010-12-22 22:14:40 +02:00
}
2010-12-26 16:34:11 +02:00
2012-08-27 15:34:43 +03:00
bool CGameHandler : : makeAutomaticAction ( const CStack * stack , BattleAction & ba )
{
BattleSetActiveStack bsa ;
bsa . stack = stack - > ID ;
bsa . askPlayerInterface = false ;
sendAndApply ( & bsa ) ;
bool ret = makeBattleAction ( ba ) ;
checkForBattleEnd ( ) ;
return ret ;
}
2013-02-12 22:49:40 +03:00
void CGameHandler : : giveHeroArtifact ( const CGHeroInstance * h , const CArtifactInstance * a , ArtifactPosition pos )
2010-12-26 16:34:11 +02:00
{
assert ( a - > artType ) ;
ArtifactLocation al ;
2012-04-14 05:20:22 +03:00
al . artHolder = const_cast < CGHeroInstance * > ( h ) ;
2010-12-26 16:34:11 +02:00
2013-02-12 22:49:40 +03:00
ArtifactPosition slot = ArtifactPosition : : PRE_FIRST ;
2010-12-26 16:34:11 +02:00
if ( pos < 0 )
{
2013-02-07 02:24:43 +03:00
if ( pos = = ArtifactPosition : : FIRST_AVAILABLE )
2010-12-26 16:34:11 +02:00
slot = a - > firstAvailableSlot ( h ) ;
else
slot = a - > firstBackpackSlot ( h ) ;
}
else
{
slot = pos ;
}
al . slot = slot ;
if ( slot < 0 | | ! a - > canBePutAt ( al ) )
2012-07-23 13:23:43 +03:00
{
2010-12-26 16:34:11 +02: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 ) ;
}
2013-02-12 22:49:40 +03:00
void CGameHandler : : giveHeroNewArtifact ( const CGHeroInstance * h , const CArtifact * artType , ArtifactPosition pos )
2010-12-26 16:34:11 +02:00
{
2013-06-26 14:18:27 +03:00
CArtifactInstance * a = nullptr ;
2011-01-28 04:16:38 +02:00
if ( ! artType - > constituents )
2012-01-30 19:07:52 +03:00
{
2012-04-14 05:20:22 +03:00
a = new CArtifactInstance ( ) ;
2012-01-30 19:07:52 +03:00
}
2011-01-18 20:56:14 +02:00
else
2012-01-30 19:07:52 +03:00
{
2011-01-18 20:56:14 +02:00
a = new CCombinedArtifactInstance ( ) ;
2012-01-30 19:07:52 +03:00
}
2011-01-15 04:17:56 +02:00
a - > artType = artType ; //*NOT* via settype -> all bonus-related stuff must be done by NewArtifact apply
2012-07-23 13:23:43 +03:00
2010-12-26 16:34:11 +02:00
NewArtifact na ;
na . art = a ;
2011-01-15 04:17:56 +02:00
sendAndApply ( & na ) ; // -> updates a!!!, will create a on other machines
2010-12-26 16:34:11 +02:00
giveHeroArtifact ( h , a , pos ) ;
}
2013-02-04 22:43:16 +03:00
void CGameHandler : : setBattleResult ( BattleResult : : EResult resultType , int victoriusSide )
2011-03-05 18:38:22 +02:00
{
2011-08-26 00:08:53 +03:00
if ( battleResult . get ( ) )
{
complain ( " There is already set result? " ) ;
return ;
}
2013-06-29 16:05:48 +03:00
auto br = new BattleResult ;
2011-03-05 18:38:22 +02:00
br - > result = resultType ;
br - > winner = victoriusSide ; //surrendering side loses
gs - > curB - > calculateCasualties ( br - > casualties ) ;
battleResult . set ( br ) ;
}
2011-05-22 21:46:52 +03:00
void CGameHandler : : commitPackage ( CPackForClient * pack )
{
sendAndApply ( pack ) ;
}
2013-02-11 02:24:57 +03:00
void CGameHandler : : spawnWanderingMonsters ( CreatureID creatureID )
2011-08-26 23:32:05 +03:00
{
std : : vector < int3 > : : iterator tile ;
std : : vector < int3 > tiles ;
getFreeTiles ( tiles ) ;
2012-01-19 17:33:22 +03:00
ui32 amount = tiles . size ( ) / 200 ; //Chance is 0.5% for each tile
2012-03-27 23:08:54 +03:00
std : : random_shuffle ( tiles . begin ( ) , tiles . end ( ) ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " Spawning wandering monsters. Found " < < tiles . size ( ) < < " free tiles. Creature type: " < < creatureID ;
2011-08-26 23:32:05 +03:00
const CCreature * cre = VLC - > creh - > creatures [ creatureID ] ;
for ( int i = 0 ; i < amount ; + + i )
{
tile = tiles . begin ( ) ;
2013-04-10 20:18:01 +03:00
logGlobal - > traceStream ( ) < < " \t Spawning monster at " < < * tile ;
2011-08-26 23:32:05 +03:00
putNewMonster ( creatureID , cre - > getRandomAmount ( std : : rand ) , * tile ) ;
tiles . erase ( tile ) ; //not use it again
}
}
2011-09-06 12:59:06 +03:00
2012-05-18 23:50:16 +03:00
void CGameHandler : : removeObstacle ( const CObstacleInstance & obstacle )
{
ObstaclesRemoved obsRem ;
obsRem . obstacles . insert ( obstacle . uniqueID ) ;
sendAndApply ( & obsRem ) ;
}
2013-02-19 01:37:22 +03:00
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 ) ;
}
2013-04-20 14:34:01 +03:00
bool CGameHandler : : isValidObject ( const CGObjectInstance * obj ) const
{
return vstd : : contains ( gs - > map - > objects , obj ) ;
}
bool CGameHandler : : isBlockedByQueries ( const CPack * pack , PlayerColor player )
{
if ( dynamic_cast < const PlayerMessage * > ( pack ) )
return false ;
auto query = queries . topQuery ( player ) ;
if ( query & & query - > blocksPack ( pack ) )
{
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 ;
}
2013-05-14 16:42:52 +03:00
void CGameHandler : : removeAfterVisit ( const CGObjectInstance * object )
{
//If the object is being visited, there must be a matching query
2013-06-29 16:05:48 +03:00
for ( const auto & query : queries . allQueries ( ) )
2013-05-14 16:42:52 +03:00
{
if ( auto someVistQuery = std : : dynamic_pointer_cast < CObjectVisitQuery > ( query ) )
{
if ( someVistQuery - > visitedObject = = object )
{
someVistQuery - > removeObjectAfterVisit = true ;
return ;
}
}
} ;
//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! " ) ;
}
2013-05-28 00:46:04 +03:00
bool CGameHandler : : isVisitCoveredByAnotherQuery ( const CGObjectInstance * obj , const CGHeroInstance * hero )
{
if ( auto topQuery = queries . topQuery ( hero - > getOwner ( ) ) )
if ( auto visit = std : : dynamic_pointer_cast < const CObjectVisitQuery > ( topQuery ) )
return ! ( visit - > visitedObject = = obj & & visit - > visitingHero = = hero ) ;
return true ;
}
2013-06-21 23:59:32 +03:00
void CGameHandler : : duelFinished ( )
{
auto si = getStartInfo ( ) ;
2013-07-22 01:01:29 +03:00
auto getName = [ & ] ( int i ) { return si - > getIthPlayersSettings ( gs - > curB - > sides [ i ] . color ) . name ; } ;
2013-06-21 23:59:32 +03:00
int casualtiesPoints = 0 ;
logGlobal - > debugStream ( ) < < boost : : format ( " Winner side %d \n Winner casualties: " )
% ( int ) battleResult . data - > winner ;
2013-06-29 16:05:48 +03:00
for ( auto & elem : battleResult . data - > casualties [ battleResult . data - > winner ] )
2013-06-21 23:59:32 +03:00
{
2013-06-29 16:05:48 +03:00
const CCreature * c = VLC - > creh - > creatures [ elem . first ] ;
logGlobal - > debugStream ( ) < < boost : : format ( " \t * %d of %s " ) % elem . second % c - > namePl ;
casualtiesPoints + = c - > AIValue * elem . second ;
2013-06-21 23:59:32 +03:00
}
logGlobal - > debugStream ( ) < < boost : : format ( " Total casualties points: %d " ) % casualtiesPoints ;
time_t timeNow ;
time ( & timeNow ) ;
std : : ofstream out ( cmdLineOptions [ " resultsFile " ] . as < std : : string > ( ) , std : : ios : : app ) ;
if ( out )
{
out < < boost : : format ( " %s \t %s \t %s \t %d \t %d \t %d \t %s \n " ) % si - > mapname % getName ( 0 ) % getName ( 1 )
% battleResult . data - > winner % battleResult . data - > result % casualtiesPoints
% asctime ( localtime ( & timeNow ) ) ;
}
else
{
logGlobal - > errorStream ( ) < < " Cannot open to write " < < cmdLineOptions [ " resultsFile " ] . as < std : : string > ( ) ;
}
CSaveFile resultFile ( " result.vdrst " ) ;
resultFile < < * battleResult . data ;
BattleResultsApplied resultsApplied ;
resultsApplied . player1 = finishingBattle - > victor ;
resultsApplied . player2 = finishingBattle - > loser ;
sendAndApply ( & resultsApplied ) ;
return ;
}
2010-12-12 01:11:26 +02:00
CasualtiesAfterBattle : : CasualtiesAfterBattle ( const CArmedInstance * army , BattleInfo * bat )
{
2013-02-14 02:55:42 +03:00
heroWithDeadCommander = ObjectInstanceID ( ) ;
2012-07-17 11:52:27 +03:00
2013-03-03 20:06:03 +03:00
PlayerColor color = army - > tempOwner ;
if ( color = = PlayerColor : : UNFLAGGABLE )
color = PlayerColor : : NEUTRAL ;
2010-12-12 01:11:26 +02:00
2013-06-29 16:05:48 +03:00
for ( CStack * st : bat - > stacks )
2010-12-12 01:11:26 +02:00
{
2011-12-14 00:23:17 +03:00
if ( vstd : : contains ( st - > state , EBattleStackState : : SUMMONED ) ) //don't take into account summoned stacks
2010-12-12 01:11:26 +02:00
continue ;
2013-07-27 22:01:31 +03:00
//FIXME: this info is also used in BattleInfo::calculateCasualties, refactor
st - > count = std : : max ( 0 , st - > count - st - > resurrected ) ;
2010-12-12 01:11:26 +02:00
if ( st - > owner = = color & & ! army - > slotEmpty ( st - > slot ) & & st - > count < army - > getStackCount ( st - > slot ) )
{
StackLocation sl ( army , st - > slot ) ;
if ( st - > alive ( ) )
newStackCounts . push_back ( std : : pair < StackLocation , int > ( sl , st - > count ) ) ;
else
newStackCounts . push_back ( std : : pair < StackLocation , int > ( sl , 0 ) ) ;
}
2012-07-17 11:52:27 +03:00
if ( st - > base & & ! st - > count )
{
auto c = dynamic_cast < const CCommanderInstance * > ( st - > base ) ;
if ( c ) //switch commander status to dead
2012-07-23 13:23:43 +03:00
{
2012-07-17 11:52:27 +03:00
auto h = dynamic_cast < const CGHeroInstance * > ( army ) ;
if ( h & & h - > commander = = c )
heroWithDeadCommander = army - > id ; //TODO: unify commander handling
}
}
2010-12-12 01:11:26 +02:00
}
}
void CasualtiesAfterBattle : : takeFromArmy ( CGameHandler * gh )
{
2013-06-29 16:05:48 +03:00
for ( TStackAndItsNewCount & ncount : newStackCounts )
2010-12-12 01:11:26 +02:00
{
if ( ncount . second > 0 )
gh - > changeStackCount ( ncount . first , ncount . second , true ) ;
else
gh - > eraseStack ( ncount . first , true ) ;
}
2013-02-14 02:55:42 +03:00
if ( heroWithDeadCommander ! = ObjectInstanceID ( ) )
2012-07-17 11:52:27 +03:00
{
SetCommanderProperty scp ;
scp . heroid = heroWithDeadCommander ;
scp . which = SetCommanderProperty : : ALIVE ;
scp . amount = 0 ;
gh - > sendAndApply ( & scp ) ;
}
2011-05-30 22:20:14 +03:00
}
2013-04-20 14:34:01 +03:00
CGameHandler : : FinishingBattleHelper : : FinishingBattleHelper ( shared_ptr < const CBattleQuery > Query , bool Duel , int RemainingBattleQueriesCount )
{
assert ( Query - > result ) ;
assert ( Query - > bi ) ;
auto & result = * Query - > result ;
auto & info = * Query - > bi ;
2013-07-22 01:01:29 +03:00
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 ;
2013-04-20 14:34:01 +03:00
duel = Duel ;
remainingBattleQueriesCount = RemainingBattleQueriesCount ;
}
CGameHandler : : FinishingBattleHelper : : FinishingBattleHelper ( )
{
winnerHero = loserHero = nullptr ;
}