2015-12-02 21:05:10 +02:00
/*
2014-06-05 20:26:50 +03:00
* CGHeroInstance . cpp , part of VCMI engine
2014-06-05 19:52:14 +03:00
*
* 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
*
*/
# include "StdInc.h"
# include "CGHeroInstance.h"
2014-06-05 20:26:50 +03:00
# include "../NetPacks.h"
# include "../CGeneralTextHandler.h"
# include "../CHeroHandler.h"
2014-06-25 17:11:07 +03:00
# include "../CModHandler.h"
2014-06-05 23:51:24 +03:00
# include "../CSoundBase.h"
2015-02-02 10:25:26 +02:00
# include "../spells/CSpellHandler.h"
2014-06-05 19:52:14 +03:00
# include "CObjectClassesHandler.h"
2014-06-25 17:11:07 +03:00
# include "../IGameCallback.h"
# include "../CGameState.h"
# include "../CCreatureHandler.h"
2017-03-17 17:48:44 +02:00
# include "../CStack.h"
2015-12-02 21:05:10 +02:00
# include "../CTownHandler.h"
2016-01-27 12:47:42 +02:00
# include "../mapping/CMap.h"
2015-12-02 21:05:10 +02:00
# include "CGTownInstance.h"
2016-02-22 01:37:19 +02:00
# include "../serializer/JsonSerializeFormat.h"
2014-06-05 19:52:14 +03:00
///helpers
static void showInfoDialog ( const PlayerColor playerID , const ui32 txtID , const ui16 soundID )
{
InfoWindow iw ;
iw . soundID = soundID ;
iw . player = playerID ;
iw . text . addTxt ( MetaString : : ADVOB_TXT , txtID ) ;
IObjectInterface : : cb - > sendAndApply ( & iw ) ;
}
static void showInfoDialog ( const CGHeroInstance * h , const ui32 txtID , const ui16 soundID )
{
const PlayerColor playerID = h - > getOwner ( ) ;
showInfoDialog ( playerID , txtID , soundID ) ;
}
static int lowestSpeed ( const CGHeroInstance * chi )
{
2015-12-24 20:30:57 +02:00
if ( ! chi - > stacksCount ( ) )
2014-06-05 19:52:14 +03:00
{
2016-03-12 03:41:27 +02:00
logGlobal - > errorStream ( ) < < " Error! Hero " < < chi - > id . getNum ( ) < < " ( " < < chi - > name < < " ) has no army! " ;
2014-06-05 19:52:14 +03:00
return 20 ;
}
auto i = chi - > Slots ( ) . begin ( ) ;
//TODO? should speed modifiers (eg from artifacts) affect hero movement?
int ret = ( i + + ) - > second - > valOfBonuses ( Bonus : : STACKS_SPEED ) ;
for ( ; i ! = chi - > Slots ( ) . end ( ) ; i + + )
{
ret = std : : min ( ret , i - > second - > valOfBonuses ( Bonus : : STACKS_SPEED ) ) ;
}
return ret ;
}
2015-11-10 13:26:45 +02:00
ui32 CGHeroInstance : : getTileCost ( const TerrainTile & dest , const TerrainTile & from , const TurnInfo * ti ) const
2014-06-05 19:52:14 +03:00
{
2015-10-16 02:03:40 +02:00
unsigned ret = GameConstants : : BASE_MOVEMENT_COST ;
2014-06-05 19:52:14 +03:00
//if there is road both on dest and src tiles - use road movement cost
2015-10-15 14:17:21 +02:00
if ( dest . roadType ! = ERoadType : : NO_ROAD & & from . roadType ! = ERoadType : : NO_ROAD )
2014-06-05 19:52:14 +03:00
{
2015-10-15 14:17:21 +02:00
int road = std : : min ( dest . roadType , from . roadType ) ; //used road ID
2014-06-05 19:52:14 +03:00
switch ( road )
{
2015-10-15 14:17:21 +02:00
case ERoadType : : DIRT_ROAD :
2014-06-05 19:52:14 +03:00
ret = 75 ;
break ;
2015-10-15 14:17:21 +02:00
case ERoadType : : GRAVEL_ROAD :
2014-06-05 19:52:14 +03:00
ret = 65 ;
break ;
2015-10-15 14:17:21 +02:00
case ERoadType : : COBBLESTONE_ROAD :
2014-06-05 19:52:14 +03:00
ret = 50 ;
break ;
default :
2015-10-15 14:17:21 +02:00
logGlobal - > errorStream ( ) < < " Unknown road type: " < < road < < " ... Something wrong! " ;
2014-06-05 19:52:14 +03:00
break ;
}
}
2015-11-21 12:30:39 +02:00
else if ( ti - > nativeTerrain ! = from . terType & & ! ti - > hasBonusOfType ( Bonus : : NO_TERRAIN_PENALTY , from . terType ) )
2014-06-05 19:52:14 +03:00
{
2015-11-21 12:30:39 +02:00
ret = VLC - > heroh - > terrCosts [ from . terType ] ;
ret - = getSecSkillLevel ( SecondarySkill : : PATHFINDING ) * 25 ;
if ( ret < GameConstants : : BASE_MOVEMENT_COST )
ret = GameConstants : : BASE_MOVEMENT_COST ;
}
return ret ;
}
2014-06-05 19:52:14 +03:00
2015-11-21 12:30:39 +02:00
int CGHeroInstance : : getNativeTerrain ( ) const
{
// NOTE: in H3 neutral stacks will ignore terrain penalty only if placed as topmost stack(s) in hero army.
// This is clearly bug in H3 however intended behaviour is not clear.
// Current VCMI behaviour will ignore neutrals in calculations so army in VCMI
// will always have best penalty without any influence from player-defined stacks order
2015-10-16 02:03:40 +02:00
2015-11-21 12:30:39 +02:00
// TODO: What should we do if all hero stacks are neutral creatures?
int nativeTerrain = - 1 ;
for ( auto stack : stacks )
{
int stackNativeTerrain = VLC - > townh - > factions [ stack . second - > type - > faction ] - > nativeTerrain ;
if ( stackNativeTerrain = = - 1 )
continue ;
if ( nativeTerrain = = - 1 )
nativeTerrain = stackNativeTerrain ;
else if ( nativeTerrain ! = stackNativeTerrain )
return - 1 ;
2015-10-15 14:17:21 +02:00
}
2015-11-21 12:30:39 +02:00
return nativeTerrain ;
2014-06-05 19:52:14 +03:00
}
int3 CGHeroInstance : : convertPosition ( int3 src , bool toh3m ) //toh3m=true: manifest->h3m; toh3m=false: h3m->manifest
{
if ( toh3m )
{
src . x + = 1 ;
return src ;
}
else
{
src . x - = 1 ;
return src ;
}
}
int3 CGHeroInstance : : getPosition ( bool h3m ) const //h3m=true - returns position of hero object; h3m=false - returns position of hero 'manifestation'
{
if ( h3m )
{
return pos ;
}
else
{
return convertPosition ( pos , false ) ;
}
}
ui8 CGHeroInstance : : getSecSkillLevel ( SecondarySkill skill ) const
{
for ( auto & elem : secSkills )
if ( elem . first = = skill )
return elem . second ;
return 0 ;
}
void CGHeroInstance : : setSecSkillLevel ( SecondarySkill which , int val , bool abs )
{
if ( getSecSkillLevel ( which ) = = 0 )
{
secSkills . push_back ( std : : pair < SecondarySkill , ui8 > ( which , val ) ) ;
updateSkill ( which , val ) ;
}
else
{
for ( auto & elem : secSkills )
{
if ( elem . first = = which )
{
if ( abs )
elem . second = val ;
else
elem . second + = val ;
if ( elem . second > 3 ) //workaround to avoid crashes when same sec skill is given more than once
{
2016-03-12 03:41:27 +02:00
logGlobal - > warnStream ( ) < < " Warning: Skill " < < which < < " increased over limit! Decreasing to Expert. " ;
2014-06-05 19:52:14 +03:00
elem . second = 3 ;
}
updateSkill ( which , elem . second ) ; //when we know final value
}
}
}
}
bool CGHeroInstance : : canLearnSkill ( ) const
{
return secSkills . size ( ) < GameConstants : : SKILL_PER_HERO ;
}
2015-11-12 13:04:33 +02:00
int CGHeroInstance : : maxMovePoints ( bool onLand , const TurnInfo * ti ) const
2014-06-05 19:52:14 +03:00
{
2016-08-16 14:47:21 +02:00
bool localTi = false ;
2015-11-12 13:04:33 +02:00
if ( ! ti )
2016-08-16 14:47:21 +02:00
{
localTi = true ;
2015-11-12 13:04:33 +02:00
ti = new TurnInfo ( this ) ;
2016-08-16 14:47:21 +02:00
}
2015-11-12 13:04:33 +02:00
2014-06-05 19:52:14 +03:00
int base ;
if ( onLand )
{
// used function is f(x) = 66.6x + 1300, rounded to second digit, where x is lowest speed in army
static const int baseSpeed = 1300 ; // base speed from creature with 0 speed
int armySpeed = lowestSpeed ( this ) * 20 / 3 ;
base = armySpeed * 10 + baseSpeed ; // separate *10 is intentional to receive same rounding as in h3
vstd : : abetween ( base , 1500 , 2000 ) ; // base speed is limited by these values
}
else
{
base = 1500 ; //on water base movement is always 1500 (speed of army doesn't matter)
}
const Bonus : : BonusType bt = onLand ? Bonus : : LAND_MOVEMENT : Bonus : : SEA_MOVEMENT ;
2015-11-12 13:04:33 +02:00
const int bonus = ti - > valOfBonuses ( Bonus : : MOVEMENT ) + ti - > valOfBonuses ( bt ) ;
2014-06-05 19:52:14 +03:00
const int subtype = onLand ? SecondarySkill : : LOGISTICS : SecondarySkill : : NAVIGATION ;
2015-11-12 13:04:33 +02:00
const double modifier = ti - > valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , subtype ) / 100.0 ;
2014-06-05 19:52:14 +03:00
2016-08-16 14:47:21 +02:00
if ( localTi )
delete ti ;
2014-06-05 19:52:14 +03:00
return int ( base * ( 1 + modifier ) ) + bonus ;
}
CGHeroInstance : : CGHeroInstance ( )
: IBoatGenerator ( this )
{
setNodeType ( HERO ) ;
ID = Obj : : HERO ;
tacticFormationEnabled = inTownGarrison = false ;
2016-01-26 21:24:38 +02:00
mana = UNINITIALIZED_MANA ;
movement = UNINITIALIZED_MOVEMENT ;
portrait = UNINITIALIZED_PORTRAIT ;
2014-06-05 19:52:14 +03:00
isStanding = true ;
moveDir = 4 ;
2015-09-16 01:20:57 +02:00
level = 1 ;
2014-06-05 19:52:14 +03:00
exp = 0xffffffff ;
visitedTown = nullptr ;
type = nullptr ;
boat = nullptr ;
commander = nullptr ;
sex = 0xff ;
secSkills . push_back ( std : : make_pair ( SecondarySkill : : DEFAULT , - 1 ) ) ;
}
2016-09-09 19:30:36 +02:00
void CGHeroInstance : : initHero ( CRandomGenerator & rand , HeroTypeID SUBID )
2014-06-05 19:52:14 +03:00
{
subID = SUBID . getNum ( ) ;
2016-09-09 19:30:36 +02:00
initHero ( rand ) ;
2014-06-05 19:52:14 +03:00
}
void CGHeroInstance : : setType ( si32 ID , si32 subID )
{
assert ( ID = = Obj : : HERO ) ; // just in case
type = VLC - > heroh - > heroes [ subID ] ;
portrait = type - > imageIndex ;
2016-09-18 09:01:09 +02:00
CGObjectInstance : : setType ( ID , type - > heroClass - > id ) ; // to find object handler we must use heroClass->id
this - > subID = subID ; // after setType subID used to store unique hero identify id. Check issue 2277 for details
2014-06-05 19:52:14 +03:00
randomizeArmy ( type - > heroClass - > faction ) ;
}
2016-09-09 19:30:36 +02:00
void CGHeroInstance : : initHero ( CRandomGenerator & rand )
2014-06-05 19:52:14 +03:00
{
assert ( validTypes ( true ) ) ;
if ( ! type )
type = VLC - > heroh - > heroes [ subID ] ;
if ( ID = = Obj : : HERO )
appearance = VLC - > objtypeh - > getHandlerFor ( Obj : : HERO , type - > heroClass - > id ) - > getTemplates ( ) . front ( ) ;
if ( ! vstd : : contains ( spells , SpellID : : PRESET ) ) //hero starts with a spell
{
for ( auto spellID : type - > spells )
spells . insert ( spellID ) ;
}
else //remove placeholder
spells - = SpellID : : PRESET ;
if ( ! getArt ( ArtifactPosition : : MACH4 ) & & ! getArt ( ArtifactPosition : : SPELLBOOK ) & & type - > haveSpellBook ) //no catapult means we haven't read pre-existent set -> use default rules for spellbook
2015-11-07 10:46:58 +02:00
putArtifact ( ArtifactPosition : : SPELLBOOK , CArtifactInstance : : createNewArtifactInstance ( ArtifactID : : SPELLBOOK ) ) ;
2014-06-05 19:52:14 +03:00
if ( ! getArt ( ArtifactPosition : : MACH4 ) )
2015-11-07 10:46:58 +02:00
putArtifact ( ArtifactPosition : : MACH4 , CArtifactInstance : : createNewArtifactInstance ( ArtifactID : : CATAPULT ) ) ; //everyone has a catapult
2014-06-05 19:52:14 +03:00
if ( portrait < 0 | | portrait = = 255 )
portrait = type - > imageIndex ;
if ( ! hasBonus ( Selector : : sourceType ( Bonus : : HERO_BASE_SKILL ) ) )
{
for ( int g = 0 ; g < GameConstants : : PRIMARY_SKILLS ; + + g )
{
pushPrimSkill ( static_cast < PrimarySkill : : PrimarySkill > ( g ) , type - > heroClass - > primarySkillInitial [ g ] ) ;
}
}
if ( secSkills . size ( ) = = 1 & & secSkills [ 0 ] = = std : : pair < SecondarySkill , ui8 > ( SecondarySkill : : DEFAULT , - 1 ) ) //set secondary skills to default
secSkills = type - > secSkillsInit ;
if ( ! name . length ( ) )
name = type - > name ;
if ( sex = = 0xFF ) //sex is default
sex = type - > sex ;
setFormation ( false ) ;
if ( ! stacksCount ( ) ) //standard army//initial army
{
2016-09-09 19:30:36 +02:00
initArmy ( rand ) ;
2014-06-05 19:52:14 +03:00
}
assert ( validTypes ( ) ) ;
if ( exp = = 0xffffffff )
{
2016-09-09 19:30:36 +02:00
initExp ( rand ) ;
2014-06-05 19:52:14 +03:00
}
else
{
2016-09-09 19:30:36 +02:00
levelUpAutomatically ( rand ) ;
2014-06-05 19:52:14 +03:00
}
if ( VLC - > modh - > modules . COMMANDERS & & ! commander )
{
commander = new CCommanderInstance ( type - > heroClass - > commander - > idNumber ) ;
commander - > setArmyObj ( castToArmyObj ( ) ) ; //TODO: separate function for setting commanders
commander - > giveStackExp ( exp ) ; //after our exp is set
}
if ( mana < 0 )
mana = manaLimit ( ) ;
}
2016-08-23 07:13:52 +02:00
void CGHeroInstance : : initArmy ( CRandomGenerator & rand , IArmyDescriptor * dst /*= nullptr*/ )
2014-06-05 19:52:14 +03:00
{
if ( ! dst )
dst = this ;
int howManyStacks = 0 ; //how many stacks will hero receives <1 - 3>
2016-08-23 07:13:52 +02:00
int pom = rand . nextInt ( 99 ) ;
2014-06-05 19:52:14 +03:00
int warMachinesGiven = 0 ;
if ( pom < 9 )
howManyStacks = 1 ;
else if ( pom < 79 )
howManyStacks = 2 ;
else
howManyStacks = 3 ;
vstd : : amin ( howManyStacks , type - > initialArmy . size ( ) ) ;
for ( int stackNo = 0 ; stackNo < howManyStacks ; stackNo + + )
{
auto & stack = type - > initialArmy [ stackNo ] ;
2016-08-23 07:13:52 +02:00
int count = rand . nextInt ( stack . minAmount , stack . maxAmount ) ;
2014-06-05 19:52:14 +03:00
if ( stack . creature > = CreatureID : : CATAPULT & &
stack . creature < = CreatureID : : ARROW_TOWERS ) //war machine
{
warMachinesGiven + + ;
if ( dst ! = this )
continue ;
int slot = - 1 ;
ArtifactID aid = ArtifactID : : NONE ;
switch ( stack . creature )
{
case CreatureID : : CATAPULT :
slot = ArtifactPosition : : MACH4 ;
aid = ArtifactID : : CATAPULT ;
break ;
default :
aid = CArtHandler : : creatureToMachineID ( stack . creature ) ;
slot = 9 + aid ;
break ;
}
auto convSlot = ArtifactPosition ( slot ) ;
if ( ! getArt ( convSlot ) )
putArtifact ( convSlot , CArtifactInstance : : createNewArtifactInstance ( aid ) ) ;
else
2016-03-12 03:41:27 +02:00
logGlobal - > warnStream ( ) < < " Hero " < < name < < " already has artifact at " < < slot < < " , omitting giving " < < aid ;
2014-06-05 19:52:14 +03:00
}
else
dst - > setCreature ( SlotID ( stackNo - warMachinesGiven ) , stack . creature , count ) ;
}
}
CGHeroInstance : : ~ CGHeroInstance ( )
{
commander . dellNull ( ) ;
}
bool CGHeroInstance : : needsLastStack ( ) const
{
return true ;
}
void CGHeroInstance : : onHeroVisit ( const CGHeroInstance * h ) const
{
if ( h = = this ) return ; //exclude potential self-visiting
if ( ID = = Obj : : HERO )
{
if ( cb - > gameState ( ) - > getPlayerRelations ( tempOwner , h - > tempOwner ) ) //our or ally hero
{
//exchange
cb - > heroExchange ( h - > id , id ) ;
}
else //battle
{
if ( visitedTown ) //we're in town
visitedTown - > onHeroVisit ( h ) ; //town will handle attacking
else
cb - > startBattleI ( h , this ) ;
}
}
else if ( ID = = Obj : : PRISON )
{
int txt_id ;
if ( cb - > getHeroCount ( h - > tempOwner , false ) < VLC - > modh - > settings . MAX_HEROES_ON_MAP_PER_PLAYER ) //GameConstants::MAX_HEROES_PER_PLAYER) //free hero slot
{
cb - > changeObjPos ( id , pos + int3 ( 1 , 0 , 0 ) , 0 ) ;
//update hero parameters
SetMovePoints smp ;
smp . hid = id ;
smp . val = maxMovePoints ( true ) ; //TODO: hota prison on water?
cb - > setMovePoints ( & smp ) ;
cb - > setManaPoints ( id , manaLimit ( ) ) ;
cb - > setObjProperty ( id , ObjProperty : : ID , Obj : : HERO ) ; //set ID to 34
cb - > giveHero ( id , h - > tempOwner ) ; //recreates def and adds hero to player
txt_id = 102 ;
}
else //already 8 wandering heroes
{
txt_id = 103 ;
}
showInfoDialog ( h , txt_id , soundBase : : ROGUE ) ;
}
}
2014-06-24 20:39:36 +03:00
std : : string CGHeroInstance : : getObjectName ( ) const
2014-06-05 19:52:14 +03:00
{
if ( ID ! = Obj : : PRISON )
{
2014-06-24 20:39:36 +03:00
std : : string hoverName = VLC - > generaltexth - > allTexts [ 15 ] ;
2014-06-05 19:52:14 +03:00
boost : : algorithm : : replace_first ( hoverName , " %s " , name ) ;
boost : : algorithm : : replace_first ( hoverName , " %s " , type - > heroClass - > name ) ;
return hoverName ;
}
else
2014-06-24 20:39:36 +03:00
return CGObjectInstance : : getObjectName ( ) ;
2014-06-05 19:52:14 +03:00
}
const std : : string & CGHeroInstance : : getBiography ( ) const
{
if ( biography . length ( ) )
return biography ;
return type - > biography ;
}
ui8 CGHeroInstance : : maxlevelsToMagicSchool ( ) const
{
return type - > heroClass - > isMagicHero ( ) ? 3 : 4 ;
}
ui8 CGHeroInstance : : maxlevelsToWisdom ( ) const
{
return type - > heroClass - > isMagicHero ( ) ? 3 : 6 ;
}
2016-08-18 17:53:28 +02:00
CGHeroInstance : : SecondarySkillsInfo : : SecondarySkillsInfo ( )
{
2016-08-19 22:31:54 +02:00
rand . setSeed ( 0 ) ;
2016-08-18 17:53:28 +02:00
magicSchoolCounter = 1 ;
wisdomCounter = 1 ;
}
2014-06-05 19:52:14 +03:00
void CGHeroInstance : : SecondarySkillsInfo : : resetMagicSchoolCounter ( )
{
magicSchoolCounter = 1 ;
}
void CGHeroInstance : : SecondarySkillsInfo : : resetWisdomCounter ( )
{
wisdomCounter = 1 ;
}
2016-09-09 19:30:36 +02:00
void CGHeroInstance : : initObj ( CRandomGenerator & rand )
2014-06-05 19:52:14 +03:00
{
blockVisit = true ;
auto hs = new HeroSpecial ( ) ;
hs - > setNodeType ( CBonusSystemNode : : SPECIALTY ) ;
attachTo ( hs ) ; //do we ever need to detach it?
if ( ! type )
2016-09-09 19:30:36 +02:00
initHero ( rand ) ; //TODO: set up everything for prison before specialties are configured
2014-06-05 19:52:14 +03:00
2016-09-09 19:30:36 +02:00
skillsInfo . rand . setSeed ( rand . nextInt ( ) ) ;
2014-06-05 19:52:14 +03:00
skillsInfo . resetMagicSchoolCounter ( ) ;
skillsInfo . resetWisdomCounter ( ) ;
2014-06-28 17:19:53 +03:00
if ( ID ! = Obj : : PRISON )
{
auto customApp = VLC - > objtypeh - > getHandlerFor ( ID , type - > heroClass - > id ) - > getOverride ( cb - > gameState ( ) - > getTile ( visitablePos ( ) ) - > terType , this ) ;
if ( customApp )
appearance = customApp . get ( ) ;
}
2014-06-16 19:27:26 +03:00
2014-06-05 19:52:14 +03:00
for ( const auto & spec : type - > spec ) //TODO: unfity with bonus system
{
2016-09-19 23:36:35 +02:00
auto bonus = std : : make_shared < Bonus > ( ) ;
2014-06-05 19:52:14 +03:00
bonus - > val = spec . val ;
bonus - > sid = id . getNum ( ) ; //from the hero, specialty has no unique id
bonus - > duration = Bonus : : PERMANENT ;
bonus - > source = Bonus : : HERO_SPECIAL ;
switch ( spec . type )
{
case 1 : // creature specialty
{
hs - > growsWithLevel = true ;
const CCreature & specCreature = * VLC - > creh - > creatures [ spec . additionalinfo ] ; //creature in which we have specialty
//int creLevel = specCreature.level;
//if(!creLevel)
//{
// if(spec.additionalinfo == 146)
// creLevel = 5; //treat ballista as 5-level
// else
// {
// logGlobal->warnStream() << "Warning: unknown level of " << specCreature.namePl;
// continue;
// }
//}
//bonus->additionalInfo = spec.additionalinfo; //creature id, should not be used again - this works only with limiter
bonus - > limiter . reset ( new CCreatureTypeLimiter ( specCreature , true ) ) ; //with upgrades
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
bonus - > subtype = PrimarySkill : : ATTACK ;
hs - > addNewBonus ( bonus ) ;
2016-09-19 23:36:35 +02:00
bonus = std : : make_shared < Bonus > ( * bonus ) ;
2014-06-05 19:52:14 +03:00
bonus - > subtype = PrimarySkill : : DEFENSE ;
hs - > addNewBonus ( bonus ) ;
//values will be calculated later
2016-09-19 23:36:35 +02:00
bonus = std : : make_shared < Bonus > ( * bonus ) ;
2014-06-05 19:52:14 +03:00
bonus - > type = Bonus : : STACKS_SPEED ;
bonus - > val = 1 ; //+1 speed
hs - > addNewBonus ( bonus ) ;
}
break ;
case 2 : //secondary skill
hs - > growsWithLevel = true ;
bonus - > type = Bonus : : SPECIAL_SECONDARY_SKILL ; //needs to be recalculated with level, based on this value
bonus - > valType = Bonus : : BASE_NUMBER ; // to receive nonzero value
bonus - > subtype = spec . subtype ; //skill id
bonus - > val = spec . val ; //value per level, in percent
hs - > addNewBonus ( bonus ) ;
2016-09-19 23:36:35 +02:00
bonus = std : : make_shared < Bonus > ( * bonus ) ;
2014-06-05 19:52:14 +03:00
switch ( spec . additionalinfo )
{
case 0 : //normal
bonus - > valType = Bonus : : PERCENT_TO_BASE ;
break ;
case 1 : //when it's navigation or there's no 'base' at all
bonus - > valType = Bonus : : PERCENT_TO_ALL ;
break ;
}
bonus - > type = Bonus : : SECONDARY_SKILL_PREMY ; //value will be calculated later
hs - > addNewBonus ( bonus ) ;
break ;
case 3 : //spell damage bonus, level dependent but calculated elsewhere
bonus - > type = Bonus : : SPECIAL_SPELL_LEV ;
bonus - > subtype = spec . subtype ;
hs - > addNewBonus ( bonus ) ;
break ;
case 4 : //creature stat boost
switch ( spec . subtype )
{
case 1 : //attack
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > subtype = PrimarySkill : : ATTACK ;
break ;
case 2 : //defense
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > subtype = PrimarySkill : : DEFENSE ;
break ;
case 3 :
bonus - > type = Bonus : : CREATURE_DAMAGE ;
bonus - > subtype = 0 ; //both min and max
break ;
case 4 : //hp
bonus - > type = Bonus : : STACK_HEALTH ;
break ;
case 5 :
bonus - > type = Bonus : : STACKS_SPEED ;
break ;
default :
continue ;
}
bonus - > additionalInfo = spec . additionalinfo ; //creature id
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
bonus - > limiter . reset ( new CCreatureTypeLimiter ( * VLC - > creh - > creatures [ spec . additionalinfo ] , true ) ) ;
hs - > addNewBonus ( bonus ) ;
break ;
case 5 : //spell damage bonus in percent
bonus - > type = Bonus : : SPECIFIC_SPELL_DAMAGE ;
bonus - > valType = Bonus : : BASE_NUMBER ; // current spell system is screwed
bonus - > subtype = spec . subtype ; //spell id
hs - > addNewBonus ( bonus ) ;
break ;
case 6 : //damage bonus for bless (Adela)
bonus - > type = Bonus : : SPECIAL_BLESS_DAMAGE ;
bonus - > subtype = spec . subtype ; //spell id if you ever wanted to use it otherwise
bonus - > additionalInfo = spec . additionalinfo ; //damage factor
hs - > addNewBonus ( bonus ) ;
break ;
case 7 : //maxed mastery for spell
bonus - > type = Bonus : : MAXED_SPELL ;
bonus - > subtype = spec . subtype ; //spell i
hs - > addNewBonus ( bonus ) ;
break ;
case 8 : //peculiar spells - enchantments
bonus - > type = Bonus : : SPECIAL_PECULIAR_ENCHANT ;
bonus - > subtype = spec . subtype ; //spell id
bonus - > additionalInfo = spec . additionalinfo ; //0, 1 for Coronius
hs - > addNewBonus ( bonus ) ;
break ;
case 9 : //upgrade creatures
{
const auto & creatures = VLC - > creh - > creatures ;
bonus - > type = Bonus : : SPECIAL_UPGRADE ;
bonus - > subtype = spec . subtype ; //base id
bonus - > additionalInfo = spec . additionalinfo ; //target id
hs - > addNewBonus ( bonus ) ;
2016-09-19 23:36:35 +02:00
bonus = std : : make_shared < Bonus > ( * bonus ) ;
2014-06-05 19:52:14 +03:00
for ( auto cre_id : creatures [ spec . subtype ] - > upgrades )
{
bonus - > subtype = cre_id ; //propagate for regular upgrades of base creature
hs - > addNewBonus ( bonus ) ;
2016-09-19 23:36:35 +02:00
bonus = std : : make_shared < Bonus > ( * bonus ) ;
2014-06-05 19:52:14 +03:00
}
break ;
}
case 10 : //resource generation
bonus - > type = Bonus : : GENERATE_RESOURCE ;
bonus - > subtype = spec . subtype ;
hs - > addNewBonus ( bonus ) ;
break ;
case 11 : //starting skill with mastery (Adrienne)
setSecSkillLevel ( SecondarySkill ( spec . val ) , spec . additionalinfo , true ) ;
break ;
case 12 : //army speed
bonus - > type = Bonus : : STACKS_SPEED ;
hs - > addNewBonus ( bonus ) ;
break ;
case 13 : //Dragon bonuses (Mutare)
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
switch ( spec . subtype )
{
case 1 :
bonus - > subtype = PrimarySkill : : ATTACK ;
break ;
case 2 :
bonus - > subtype = PrimarySkill : : DEFENSE ;
break ;
}
bonus - > limiter . reset ( new HasAnotherBonusLimiter ( Bonus : : DRAGON_NATURE ) ) ;
hs - > addNewBonus ( bonus ) ;
break ;
default :
2016-03-12 03:41:27 +02:00
logGlobal - > warnStream ( ) < < " Unexpected hero specialty " < < type ;
2014-06-05 19:52:14 +03:00
}
}
specialty . push_back ( hs ) ; //will it work?
for ( auto hs2 : type - > specialty ) //copy active (probably growing) bonuses from hero prootype to hero object
{
auto hs = new HeroSpecial ( ) ;
attachTo ( hs ) ; //do we ever need to detach it?
hs - > setNodeType ( CBonusSystemNode : : SPECIALTY ) ;
for ( auto bonus : hs2 . bonuses )
{
hs - > addNewBonus ( bonus ) ;
}
hs - > growsWithLevel = hs2 . growsWithLevel ;
specialty . push_back ( hs ) ; //will it work?
}
//initialize bonuses
recreateSecondarySkillsBonuses ( ) ;
Updatespecialty ( ) ;
mana = manaLimit ( ) ; //after all bonuses are taken into account, make sure this line is the last one
type - > name = name ;
}
void CGHeroInstance : : Updatespecialty ( ) //TODO: calculate special value of bonuses on-the-fly?
{
for ( auto hs : specialty )
{
if ( hs - > growsWithLevel )
{
//const auto &creatures = VLC->creh->creatures;
2016-09-19 23:36:35 +02:00
for ( auto & b : hs - > getBonusList ( ) )
2014-06-05 19:52:14 +03:00
{
switch ( b - > type )
{
case Bonus : : SECONDARY_SKILL_PREMY :
b - > val = ( hs - > valOfBonuses ( Bonus : : SPECIAL_SECONDARY_SKILL , b - > subtype ) * level ) ;
break ; //use only hero skills as bonuses to avoid feedback loop
case Bonus : : PRIMARY_SKILL : //for creatures, that is
{
const CCreature * cre = nullptr ;
int creLevel = 0 ;
if ( auto creatureLimiter = std : : dynamic_pointer_cast < CCreatureTypeLimiter > ( b - > limiter ) ) //TODO: more general eveluation of bonuses?
{
cre = creatureLimiter - > creature ;
creLevel = cre - > level ;
if ( ! creLevel )
{
creLevel = 5 ; //treat ballista as tier 5
}
}
else //no creature found, can't calculate value
{
2016-03-12 03:41:27 +02:00
logGlobal - > warnStream ( ) < < " Primary skill specialty growth supported only with creature type limiters " ;
2014-06-05 19:52:14 +03:00
break ;
}
double primSkillModifier = ( int ) ( level / creLevel ) / 20.0 ;
int param ;
switch ( b - > subtype )
{
case PrimarySkill : : ATTACK :
param = cre - > Attack ( ) ;
break ;
case PrimarySkill : : DEFENSE :
param = cre - > Defense ( ) ;
break ;
default :
continue ;
}
b - > val = ceil ( param * ( 1 + primSkillModifier ) ) - param ; //yep, overcomplicated but matches original
break ;
}
}
}
}
}
}
void CGHeroInstance : : recreateSecondarySkillsBonuses ( )
{
auto secondarySkillsBonuses = getBonuses ( Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) ;
for ( auto bonus : * secondarySkillsBonuses )
removeBonus ( bonus ) ;
for ( auto skill_info : secSkills )
updateSkill ( SecondarySkill ( skill_info . first ) , skill_info . second ) ;
}
void CGHeroInstance : : updateSkill ( SecondarySkill which , int val )
{
if ( which = = SecondarySkill : : LEADERSHIP | | which = = SecondarySkill : : LUCK )
{ //luck-> VLC->generaltexth->arraytxt[73+luckSkill]; VLC->generaltexth->arraytxt[104+moraleSkill]
bool luck = which = = SecondarySkill : : LUCK ;
Bonus : : BonusType type [ ] = { Bonus : : MORALE , Bonus : : LUCK } ;
2016-09-19 23:36:35 +02:00
auto b = getBonusLocalFirst ( Selector : : type ( type [ luck ] ) . And ( Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) ) ;
2014-06-05 19:52:14 +03:00
if ( ! b )
{
2016-09-19 23:36:35 +02:00
b = std : : make_shared < Bonus > ( Bonus : : PERMANENT , type [ luck ] , Bonus : : SECONDARY_SKILL , + val , which , which , Bonus : : BASE_NUMBER ) ;
2014-06-05 19:52:14 +03:00
addNewBonus ( b ) ;
}
else
b - > val = + val ;
}
else if ( which = = SecondarySkill : : DIPLOMACY ) //surrender discount: 20% per level
{
2016-09-19 23:36:35 +02:00
if ( auto b = getBonusLocalFirst ( Selector : : type ( Bonus : : SURRENDER_DISCOUNT ) . And ( Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) ) )
2014-06-05 19:52:14 +03:00
b - > val = + val ;
else
2016-09-19 23:36:35 +02:00
addNewBonus ( std : : make_shared < Bonus > ( Bonus : : PERMANENT , Bonus : : SURRENDER_DISCOUNT , Bonus : : SECONDARY_SKILL , val * 20 , which ) ) ;
2014-06-05 19:52:14 +03:00
}
int skillVal = 0 ;
switch ( which )
{
case SecondarySkill : : ARCHERY :
switch ( val )
{
case 1 :
skillVal = 10 ; break ;
case 2 :
skillVal = 25 ; break ;
case 3 :
skillVal = 50 ; break ;
}
break ;
case SecondarySkill : : LOGISTICS :
skillVal = 10 * val ; break ;
case SecondarySkill : : NAVIGATION :
skillVal = 50 * val ; break ;
case SecondarySkill : : MYSTICISM :
skillVal = val ; break ;
case SecondarySkill : : EAGLE_EYE :
skillVal = 30 + 10 * val ; break ;
case SecondarySkill : : NECROMANCY :
skillVal = 10 * val ; break ;
case SecondarySkill : : LEARNING :
skillVal = 5 * val ; break ;
case SecondarySkill : : OFFENCE :
skillVal = 10 * val ; break ;
case SecondarySkill : : ARMORER :
skillVal = 5 * val ; break ;
case SecondarySkill : : INTELLIGENCE :
skillVal = 25 < < ( val - 1 ) ; break ;
case SecondarySkill : : SORCERY :
skillVal = 5 * val ; break ;
case SecondarySkill : : RESISTANCE :
skillVal = 5 < < ( val - 1 ) ; break ;
case SecondarySkill : : FIRST_AID :
skillVal = 25 + 25 * val ; break ;
case SecondarySkill : : ESTATES :
skillVal = 125 < < ( val - 1 ) ; break ;
}
Bonus : : ValueType skillValType = skillVal ? Bonus : : BASE_NUMBER : Bonus : : INDEPENDENT_MIN ;
2016-09-19 23:36:35 +02:00
if ( auto b = getExportedBonusList ( ) . getFirst ( Selector : : typeSubtype ( Bonus : : SECONDARY_SKILL_PREMY , which )
. And ( Selector : : sourceType ( Bonus : : SECONDARY_SKILL ) ) ) ) //only local hero bonus
2014-06-05 19:52:14 +03:00
{
b - > val = skillVal ;
b - > valType = skillValType ;
}
else
{
2016-09-19 23:36:35 +02:00
auto bonus = std : : make_shared < Bonus > ( Bonus : : PERMANENT , Bonus : : SECONDARY_SKILL_PREMY , Bonus : : SECONDARY_SKILL , skillVal , id . getNum ( ) , which , skillValType ) ;
2014-06-05 19:52:14 +03:00
bonus - > source = Bonus : : SECONDARY_SKILL ;
addNewBonus ( bonus ) ;
}
2015-12-11 15:13:18 +02:00
CBonusSystemNode : : treeHasChanged ( ) ;
2014-06-05 19:52:14 +03:00
}
void CGHeroInstance : : setPropertyDer ( ui8 what , ui32 val )
{
if ( what = = ObjProperty : : PRIMARY_STACK_COUNT )
setStackCount ( SlotID ( 0 ) , val ) ;
}
double CGHeroInstance : : getFightingStrength ( ) const
{
return sqrt ( ( 1.0 + 0.05 * getPrimSkillLevel ( PrimarySkill : : ATTACK ) ) * ( 1.0 + 0.05 * getPrimSkillLevel ( PrimarySkill : : DEFENSE ) ) ) ;
}
double CGHeroInstance : : getMagicStrength ( ) const
{
return sqrt ( ( 1.0 + 0.05 * getPrimSkillLevel ( PrimarySkill : : KNOWLEDGE ) ) * ( 1.0 + 0.05 * getPrimSkillLevel ( PrimarySkill : : SPELL_POWER ) ) ) ;
}
double CGHeroInstance : : getHeroStrength ( ) const
{
return sqrt ( pow ( getFightingStrength ( ) , 2.0 ) * pow ( getMagicStrength ( ) , 2.0 ) ) ;
}
ui64 CGHeroInstance : : getTotalStrength ( ) const
{
double ret = getFightingStrength ( ) * getArmyStrength ( ) ;
return ( ui64 ) ret ;
}
TExpType CGHeroInstance : : calculateXp ( TExpType exp ) const
{
return exp * ( 100 + valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : LEARNING ) ) / 100.0 ;
}
ui8 CGHeroInstance : : getSpellSchoolLevel ( const CSpell * spell , int * outSelectedSchool ) const
{
si16 skill = - 1 ; //skill level
2016-01-24 01:27:14 +02:00
2014-11-24 18:14:10 +02:00
spell - > forEachSchool ( [ & , this ] ( const SpellSchoolInfo & cnf , bool & stop )
2014-11-13 16:24:30 +02:00
{
2015-10-31 22:15:40 +02:00
int thisSchool = std : : max < int > ( getSecSkillLevel ( cnf . skill ) , valOfBonuses ( Bonus : : MAGIC_SCHOOL_SKILL , 1 < < ( ( ui8 ) cnf . id ) ) ) ; //FIXME: Bonus shouldn't be additive (Witchking Artifacts : Crown of Skies)
2016-01-24 01:27:14 +02:00
if ( thisSchool > skill )
{
skill = thisSchool ;
if ( outSelectedSchool )
* outSelectedSchool = ( ui8 ) cnf . id ;
}
2014-11-24 18:14:10 +02:00
} ) ;
2016-01-24 01:27:14 +02:00
2014-06-05 19:52:14 +03:00
vstd : : amax ( skill , valOfBonuses ( Bonus : : MAGIC_SCHOOL_SKILL , 0 ) ) ; //any school bonus
vstd : : amax ( skill , valOfBonuses ( Bonus : : SPELL , spell - > id . toEnum ( ) ) ) ; //given by artifact or other effect
2015-09-15 04:53:10 +02:00
2015-10-31 22:15:40 +02:00
vstd : : amax ( skill , 0 ) ; //in case we don't know any school
vstd : : amin ( skill , 3 ) ;
2014-06-05 19:52:14 +03:00
return skill ;
}
2015-03-18 21:22:52 +02:00
ui32 CGHeroInstance : : getSpellBonus ( const CSpell * spell , ui32 base , const CStack * affectedStack ) const
{
//applying sorcery secondary skill
base * = ( 100.0 + valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : SORCERY ) ) / 100.0 ;
base * = ( 100.0 + valOfBonuses ( Bonus : : SPELL_DAMAGE ) + valOfBonuses ( Bonus : : SPECIFIC_SPELL_DAMAGE , spell - > id . toEnum ( ) ) ) / 100.0 ;
spell - > forEachSchool ( [ & base , this ] ( const SpellSchoolInfo & cnf , bool & stop )
{
base * = ( 100.0 + valOfBonuses ( cnf . damagePremyBonus ) ) / 100.0 ;
stop = true ; //only bonus from one school is used
} ) ;
if ( affectedStack & & affectedStack - > getCreature ( ) - > level ) //Hero specials like Solmyr, Deemer
base * = ( 100. + ( ( valOfBonuses ( Bonus : : SPECIAL_SPELL_LEV , spell - > id . toEnum ( ) ) * level ) / affectedStack - > getCreature ( ) - > level ) ) / 100.0 ;
2016-01-24 01:27:14 +02:00
return base ;
2015-03-18 21:22:52 +02:00
}
2015-09-17 07:42:30 +02:00
int CGHeroInstance : : getEffectLevel ( const CSpell * spell ) const
{
if ( hasBonusOfType ( Bonus : : MAXED_SPELL , spell - > id ) )
return 3 ; //todo: recheck specialty from where this bonus is. possible bug
else
2016-01-24 01:27:14 +02:00
return getSpellSchoolLevel ( spell ) ;
2015-09-17 07:42:30 +02:00
}
int CGHeroInstance : : getEffectPower ( const CSpell * spell ) const
{
return getPrimSkillLevel ( PrimarySkill : : SPELL_POWER ) ;
}
int CGHeroInstance : : getEnchantPower ( const CSpell * spell ) const
{
2016-01-24 01:27:14 +02:00
return getPrimSkillLevel ( PrimarySkill : : SPELL_POWER ) + valOfBonuses ( Bonus : : SPELL_DURATION ) ;
2015-09-17 07:42:30 +02:00
}
int CGHeroInstance : : getEffectValue ( const CSpell * spell ) const
{
return 0 ;
}
2015-03-18 21:22:52 +02:00
2015-09-17 08:29:57 +02:00
const PlayerColor CGHeroInstance : : getOwner ( ) const
{
return tempOwner ;
}
2016-09-10 17:23:55 +02:00
void CGHeroInstance : : getCasterName ( MetaString & text ) const
{
//FIXME: use local name, MetaString need access to gamestate as hero name is part of map object
text . addReplacement ( name ) ;
}
2016-09-17 22:04:23 +02:00
void CGHeroInstance : : getCastDescription ( const CSpell * spell , const std : : vector < const CStack * > & attacked , MetaString & text ) const
{
const bool singleTarget = attacked . size ( ) = = 1 ;
const int textIndex = singleTarget ? 195 : 196 ;
text . addTxt ( MetaString : : GENERAL_TXT , textIndex ) ;
getCasterName ( text ) ;
text . addReplacement ( MetaString : : SPELL_NAME , spell - > id . toEnum ( ) ) ;
if ( singleTarget )
text . addReplacement ( MetaString : : CRE_PL_NAMES , attacked . at ( 0 ) - > getCreature ( ) - > idNumber . num ) ;
}
2014-06-05 19:52:14 +03:00
bool CGHeroInstance : : canCastThisSpell ( const CSpell * spell ) const
{
2015-09-16 03:39:44 +02:00
const bool isAllowed = IObjectInterface : : cb - > isAllowed ( 0 , spell - > id ) ;
2016-10-09 12:38:54 +02:00
const bool inSpellBook = vstd : : contains ( spells , spell - > id ) & & hasSpellbook ( ) ;
2015-09-16 03:39:44 +02:00
const bool specificBonus = hasBonusOfType ( Bonus : : SPELL , spell - > id ) ;
bool schoolBonus = false ;
spell - > forEachSchool ( [ this , & schoolBonus ] ( const SpellSchoolInfo & cnf , bool & stop )
{
if ( hasBonusOfType ( cnf . knoledgeBonus ) )
{
schoolBonus = stop = true ;
}
} ) ;
const bool levelBonus = hasBonusOfType ( Bonus : : SPELLS_OF_LEVEL , spell - > level ) ;
2016-03-12 03:41:27 +02:00
if ( spell - > isSpecialSpell ( ) )
{
if ( inSpellBook )
{ //hero has this spell in spellbook
2016-11-02 13:52:02 +02:00
logGlobal - > error ( " Special spell %s in spellbook. " , spell - > name ) ;
2016-03-12 03:41:27 +02:00
}
return specificBonus ;
}
else if ( ! isAllowed )
{
if ( inSpellBook )
2016-11-02 13:52:02 +02:00
{
//hero has this spell in spellbook
//it is normal if set in map editor, but trace it to possible debug of magic guild
logGlobal - > trace ( " Banned spell %s in spellbook. " , spell - > name ) ;
2016-03-12 03:41:27 +02:00
}
2016-11-02 13:52:02 +02:00
return inSpellBook | | specificBonus | | schoolBonus | | levelBonus ;
2016-03-12 03:41:27 +02:00
}
else
{
2015-09-16 03:39:44 +02:00
return inSpellBook | | schoolBonus | | specificBonus | | levelBonus ;
2016-03-12 03:41:27 +02:00
}
2014-06-05 19:52:14 +03:00
}
2016-10-09 12:38:54 +02:00
bool CGHeroInstance : : canLearnSpell ( const CSpell * spell ) const
{
if ( ! hasSpellbook ( ) )
return false ;
if ( spell - > level > getSecSkillLevel ( SecondarySkill : : WISDOM ) + 2 ) //not enough wisdom
return false ;
if ( vstd : : contains ( spells , spell - > id ) ) //already known
return false ;
if ( spell - > isSpecialSpell ( ) )
{
logGlobal - > warn ( " Hero %s try to learn special spell %s " , nodeName ( ) , spell - > name ) ;
return false ; //special spells can not be learned
}
if ( spell - > isCreatureAbility ( ) )
{
logGlobal - > warn ( " Hero %s try to learn creature spell %s " , nodeName ( ) , spell - > name ) ;
return false ; //creature abilities can not be learned
}
2016-10-11 20:25:22 +02:00
if ( ! IObjectInterface : : cb - > isAllowed ( 0 , spell - > id ) )
2016-10-09 12:38:54 +02:00
{
logGlobal - > warn ( " Hero %s try to learn banned spell %s " , nodeName ( ) , spell - > name ) ;
return false ; //banned spells should not be learned
}
return true ;
}
2014-06-05 19:52:14 +03:00
/**
* Calculates what creatures and how many to be raised from a battle .
* @ param battleResult The results of the battle .
* @ return Returns a pair with the first value indicating the ID of the creature
* type and second value the amount . Both values are returned as - 1 if necromancy
* could not be applied .
*/
CStackBasicDescriptor CGHeroInstance : : calculateNecromancy ( const BattleResult & battleResult ) const
{
const ui8 necromancyLevel = getSecSkillLevel ( SecondarySkill : : NECROMANCY ) ;
// Hero knows necromancy or has Necromancer Cloak
if ( necromancyLevel > 0 | | hasBonusOfType ( Bonus : : IMPROVED_NECROMANCY ) )
{
double necromancySkill = valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , SecondarySkill : : NECROMANCY ) / 100.0 ;
vstd : : amin ( necromancySkill , 1.0 ) ; //it's impossible to raise more creatures than all...
const std : : map < ui32 , si32 > & casualties = battleResult . casualties [ ! battleResult . winner ] ;
ui32 raisedUnits = 0 ;
// Figure out what to raise and how many.
const CreatureID creatureTypes [ ] = { CreatureID : : SKELETON , CreatureID : : WALKING_DEAD , CreatureID : : WIGHTS , CreatureID : : LICHES } ;
const bool improvedNecromancy = hasBonusOfType ( Bonus : : IMPROVED_NECROMANCY ) ;
const CCreature * raisedUnitType = VLC - > creh - > creatures [ creatureTypes [ improvedNecromancy ? necromancyLevel : 0 ] ] ;
const ui32 raisedUnitHP = raisedUnitType - > valOfBonuses ( Bonus : : STACK_HEALTH ) ;
//calculate creatures raised from each defeated stack
for ( auto & casualtie : casualties )
{
// Get lost enemy hit points convertible to units.
CCreature * c = VLC - > creh - > creatures [ casualtie . first ] ;
const ui32 raisedHP = c - > valOfBonuses ( Bonus : : STACK_HEALTH ) * casualtie . second * necromancySkill ;
raisedUnits + = std : : min < ui32 > ( raisedHP / raisedUnitHP , casualtie . second * necromancySkill ) ; //limit to % of HP and % of original stack count
}
// Make room for new units.
SlotID slot = getSlotFor ( raisedUnitType - > idNumber ) ;
if ( slot = = SlotID ( ) )
{
// If there's no room for unit, try it's upgraded version 2/3rds the size.
raisedUnitType = VLC - > creh - > creatures [ * raisedUnitType - > upgrades . begin ( ) ] ;
raisedUnits = ( raisedUnits * 2 ) / 3 ;
slot = getSlotFor ( raisedUnitType - > idNumber ) ;
}
if ( raisedUnits < = 0 )
raisedUnits = 1 ;
return CStackBasicDescriptor ( raisedUnitType - > idNumber , raisedUnits ) ;
}
return CStackBasicDescriptor ( ) ;
}
/**
* Show the necromancy dialog with information about units raised .
* @ param raisedStack Pair where the first element represents ID of the raised creature
* and the second element the amount .
*/
2016-09-09 19:30:36 +02:00
void CGHeroInstance : : showNecromancyDialog ( const CStackBasicDescriptor & raisedStack , CRandomGenerator & rand ) const
2014-06-05 19:52:14 +03:00
{
InfoWindow iw ;
2016-09-09 19:30:36 +02:00
iw . soundID = soundBase : : pickup01 + rand . nextInt ( 6 ) ;
2014-06-05 19:52:14 +03:00
iw . player = tempOwner ;
iw . components . push_back ( Component ( raisedStack ) ) ;
if ( raisedStack . count > 1 ) // Practicing the dark arts of necromancy, ... (plural)
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 145 ) ;
iw . text . addReplacement ( raisedStack . count ) ;
}
else // Practicing the dark arts of necromancy, ... (singular)
{
iw . text . addTxt ( MetaString : : GENERAL_TXT , 146 ) ;
}
iw . text . addReplacement ( raisedStack ) ;
cb - > showInfoDialog ( & iw ) ;
}
2014-06-24 02:26:36 +03:00
/*
2014-06-05 19:52:14 +03:00
int3 CGHeroInstance : : getSightCenter ( ) const
{
return getPosition ( false ) ;
2014-06-24 02:26:36 +03:00
} */
2014-06-05 19:52:14 +03:00
2016-01-31 17:01:58 +02:00
int CGHeroInstance : : getSightRadius ( ) const
2014-06-05 19:52:14 +03:00
{
return 5 + getSecSkillLevel ( SecondarySkill : : SCOUTING ) + valOfBonuses ( Bonus : : SIGHT_RADIOUS ) ; //default + scouting
}
si32 CGHeroInstance : : manaRegain ( ) const
{
if ( hasBonusOfType ( Bonus : : FULL_MANA_REGENERATION ) )
return manaLimit ( ) ;
return 1 + valOfBonuses ( Bonus : : SECONDARY_SKILL_PREMY , 8 ) + valOfBonuses ( Bonus : : MANA_REGENERATION ) ; //1 + Mysticism level
}
2016-01-30 09:20:49 +02:00
si32 CGHeroInstance : : getManaNewTurn ( ) const
{
if ( visitedTown & & visitedTown - > hasBuilt ( BuildingID : : MAGES_GUILD_1 ) )
{
//if hero starts turn in town with mage guild - restore all mana
return std : : max ( mana , manaLimit ( ) ) ;
}
si32 res = mana + manaRegain ( ) ;
res = std : : min ( res , manaLimit ( ) ) ;
res = std : : max ( res , mana ) ;
res = std : : max ( res , 0 ) ;
return res ;
}
2014-06-05 19:52:14 +03:00
// /**
// * Places an artifact in hero's backpack. If it's a big artifact equips it
// * or discards it if it cannot be equipped.
// */
// void CGHeroInstance::giveArtifact (ui32 aid) //use only for fixed artifacts
// {
// CArtifact * const artifact = VLC->arth->artifacts[aid]; //pointer to constant object
// CArtifactInstance *ai = CArtifactInstance::createNewArtifactInstance(artifact);
// ai->putAt(this, ai->firstAvailableSlot(this));
// }
int CGHeroInstance : : getBoatType ( ) const
{
switch ( type - > heroClass - > getAlignment ( ) )
{
case EAlignment : : GOOD :
return 1 ;
case EAlignment : : EVIL :
return 0 ;
case EAlignment : : NEUTRAL :
return 2 ;
default :
throw std : : runtime_error ( " Wrong alignment! " ) ;
}
}
void CGHeroInstance : : getOutOffsets ( std : : vector < int3 > & offsets ) const
{
2015-11-26 09:55:02 +02:00
// FIXME: Offsets need to be fixed once we get rid of convertPosition
// Check issue 515 for details
2016-01-24 01:27:14 +02:00
offsets =
{
2015-11-26 09:55:02 +02:00
int3 ( - 1 , 1 , 0 ) , int3 ( - 1 , - 1 , 0 ) , int3 ( - 2 , 0 , 0 ) , int3 ( 0 , 0 , 0 ) , int3 ( 0 , 1 , 0 ) , int3 ( - 2 , 1 , 0 ) , int3 ( 0 , - 1 , 0 ) , int3 ( - 2 , - 1 , 0 )
2014-10-03 23:34:13 +03:00
} ;
2014-06-05 19:52:14 +03:00
}
int CGHeroInstance : : getSpellCost ( const CSpell * sp ) const
{
return sp - > getCost ( getSpellSchoolLevel ( sp ) ) ;
}
void CGHeroInstance : : pushPrimSkill ( PrimarySkill : : PrimarySkill which , int val )
{
assert ( ! hasBonus ( Selector : : typeSubtype ( Bonus : : PRIMARY_SKILL , which )
. And ( Selector : : sourceType ( Bonus : : HERO_BASE_SKILL ) ) ) ) ;
2016-09-19 23:36:35 +02:00
addNewBonus ( std : : make_shared < Bonus > ( Bonus : : PERMANENT , Bonus : : PRIMARY_SKILL , Bonus : : HERO_BASE_SKILL , val , id . getNum ( ) , which ) ) ;
2014-06-05 19:52:14 +03:00
}
EAlignment : : EAlignment CGHeroInstance : : getAlignment ( ) const
{
return type - > heroClass - > getAlignment ( ) ;
}
2016-09-09 19:30:36 +02:00
void CGHeroInstance : : initExp ( CRandomGenerator & rand )
2014-06-05 19:52:14 +03:00
{
2016-09-09 19:30:36 +02:00
exp = rand . nextInt ( 40 , 89 ) ;
2014-06-05 19:52:14 +03:00
}
std : : string CGHeroInstance : : nodeName ( ) const
{
return " Hero " + name ;
}
void CGHeroInstance : : putArtifact ( ArtifactPosition pos , CArtifactInstance * art )
{
assert ( ! getArt ( pos ) ) ;
art - > putAt ( ArtifactLocation ( this , pos ) ) ;
}
void CGHeroInstance : : putInBackpack ( CArtifactInstance * art )
{
putArtifact ( art - > firstBackpackSlot ( this ) , art ) ;
}
bool CGHeroInstance : : hasSpellbook ( ) const
{
return getArt ( ArtifactPosition : : SPELLBOOK ) ;
}
void CGHeroInstance : : deserializationFix ( )
{
artDeserializationFix ( this ) ;
for ( auto hs : specialty )
{
attachTo ( hs ) ;
}
}
CBonusSystemNode * CGHeroInstance : : whereShouldBeAttached ( CGameState * gs )
{
if ( visitedTown )
{
if ( inTownGarrison )
return visitedTown ;
else
return & visitedTown - > townAndVis ;
}
else
return CArmedInstance : : whereShouldBeAttached ( gs ) ;
}
2015-11-12 13:04:33 +02:00
int CGHeroInstance : : movementPointsAfterEmbark ( int MPsBefore , int basicCost , bool disembark /*= false*/ , const TurnInfo * ti ) const
2014-06-05 19:52:14 +03:00
{
2016-11-25 14:34:38 +02:00
int ret = 0 ; //take all MPs by default
2016-08-16 14:47:21 +02:00
bool localTi = false ;
2015-11-12 13:04:33 +02:00
if ( ! ti )
2016-08-16 14:47:21 +02:00
{
localTi = true ;
2015-11-12 13:04:33 +02:00
ti = new TurnInfo ( this ) ;
2016-08-16 14:47:21 +02:00
}
2015-11-12 13:04:33 +02:00
2015-11-21 11:46:59 +02:00
int mp1 = ti - > getMaxMovePoints ( disembark ? EPathfindingLayer : : LAND : EPathfindingLayer : : SAIL ) ;
int mp2 = ti - > getMaxMovePoints ( disembark ? EPathfindingLayer : : SAIL : EPathfindingLayer : : LAND ) ;
2015-11-12 13:04:33 +02:00
if ( ti - > hasBonusOfType ( Bonus : : FREE_SHIP_BOARDING ) )
2016-11-25 14:34:38 +02:00
ret = ( MPsBefore - basicCost ) * static_cast < double > ( mp1 ) / mp2 ;
2014-06-05 19:52:14 +03:00
2016-08-16 14:47:21 +02:00
if ( localTi )
delete ti ;
2016-11-25 14:34:38 +02:00
return ret ;
2014-06-05 19:52:14 +03:00
}
2015-11-29 11:32:06 +02:00
EDiggingStatus CGHeroInstance : : diggingStatus ( ) const
2014-06-05 19:52:14 +03:00
{
if ( movement < maxMovePoints ( true ) )
2015-11-29 11:32:06 +02:00
return EDiggingStatus : : LACK_OF_MOVEMENT ;
return cb - > getTile ( getPosition ( false ) ) - > getDiggingStatus ( ) ;
2014-06-05 19:52:14 +03:00
}
ArtBearer : : ArtBearer CGHeroInstance : : bearerType ( ) const
{
return ArtBearer : : HERO ;
}
std : : vector < SecondarySkill > CGHeroInstance : : getLevelUpProposedSecondarySkills ( ) const
{
std : : vector < SecondarySkill > obligatorySkills ; //hero is offered magic school or wisdom if possible
if ( ! skillsInfo . wisdomCounter )
{
if ( cb - > isAllowed ( 2 , SecondarySkill : : WISDOM ) & & ! getSecSkillLevel ( SecondarySkill : : WISDOM ) )
obligatorySkills . push_back ( SecondarySkill : : WISDOM ) ;
}
if ( ! skillsInfo . magicSchoolCounter )
{
2014-10-03 23:34:13 +03:00
std : : vector < SecondarySkill > ss =
{
SecondarySkill : : FIRE_MAGIC , SecondarySkill : : AIR_MAGIC , SecondarySkill : : WATER_MAGIC , SecondarySkill : : EARTH_MAGIC
} ;
2014-06-05 19:52:14 +03:00
std : : shuffle ( ss . begin ( ) , ss . end ( ) , skillsInfo . rand . getStdGenerator ( ) ) ;
for ( auto skill : ss )
{
if ( cb - > isAllowed ( 2 , skill ) & & ! getSecSkillLevel ( skill ) ) //only schools hero doesn't know yet
{
obligatorySkills . push_back ( skill ) ;
break ; //only one
}
}
}
std : : vector < SecondarySkill > skills ;
//picking sec. skills for choice
std : : set < SecondarySkill > basicAndAdv , expert , none ;
for ( int i = 0 ; i < GameConstants : : SKILL_QUANTITY ; i + + )
if ( cb - > isAllowed ( 2 , i ) )
none . insert ( SecondarySkill ( i ) ) ;
for ( auto & elem : secSkills )
{
if ( elem . second < SecSkillLevel : : EXPERT )
basicAndAdv . insert ( elem . first ) ;
else
expert . insert ( elem . first ) ;
none . erase ( elem . first ) ;
}
for ( auto s : obligatorySkills ) //don't duplicate them
{
none . erase ( s ) ;
basicAndAdv . erase ( s ) ;
expert . erase ( s ) ;
}
//first offered skill:
// 1) give obligatory skill
// 2) give any other new skill
// 3) upgrade existing
if ( canLearnSkill ( ) & & obligatorySkills . size ( ) > 0 )
{
skills . push_back ( obligatorySkills [ 0 ] ) ;
}
else if ( none . size ( ) & & canLearnSkill ( ) ) //hero have free skill slot
{
skills . push_back ( type - > heroClass - > chooseSecSkill ( none , skillsInfo . rand ) ) ; //new skill
none . erase ( skills . back ( ) ) ;
}
else if ( ! basicAndAdv . empty ( ) )
{
skills . push_back ( type - > heroClass - > chooseSecSkill ( basicAndAdv , skillsInfo . rand ) ) ; //upgrade existing
basicAndAdv . erase ( skills . back ( ) ) ;
}
//second offered skill:
//1) upgrade existing
//2) give obligatory skill
//3) give any other new skill
if ( ! basicAndAdv . empty ( ) )
{
SecondarySkill s = type - > heroClass - > chooseSecSkill ( basicAndAdv , skillsInfo . rand ) ; //upgrade existing
skills . push_back ( s ) ;
basicAndAdv . erase ( s ) ;
}
else if ( canLearnSkill ( ) & & obligatorySkills . size ( ) > 1 )
{
skills . push_back ( obligatorySkills [ 1 ] ) ;
}
else if ( none . size ( ) & & canLearnSkill ( ) )
{
skills . push_back ( type - > heroClass - > chooseSecSkill ( none , skillsInfo . rand ) ) ; //give new skill
none . erase ( skills . back ( ) ) ;
}
2014-08-19 13:15:40 +03:00
if ( skills . size ( ) = = 2 ) // Fix for #1868 to avoid changing logic (possibly causing bugs in process)
std : : swap ( skills [ 0 ] , skills [ 1 ] ) ;
2014-06-05 19:52:14 +03:00
return skills ;
}
2016-08-23 18:12:10 +02:00
PrimarySkill : : PrimarySkill CGHeroInstance : : nextPrimarySkill ( CRandomGenerator & rand ) const
2014-06-05 19:52:14 +03:00
{
assert ( gainsLevel ( ) ) ;
2016-08-23 18:12:10 +02:00
int randomValue = rand . nextInt ( 99 ) , pom = 0 , primarySkill = 0 ;
2014-06-05 19:52:14 +03:00
const auto & skillChances = ( level > 9 ) ? type - > heroClass - > primarySkillLowLevel : type - > heroClass - > primarySkillHighLevel ;
for ( ; primarySkill < GameConstants : : PRIMARY_SKILLS ; + + primarySkill )
{
pom + = skillChances [ primarySkill ] ;
if ( randomValue < pom )
{
break ;
}
}
logGlobal - > traceStream ( ) < < " The hero gets the primary skill " < < primarySkill < < " with a probability of " < < randomValue < < " %. " ;
return static_cast < PrimarySkill : : PrimarySkill > ( primarySkill ) ;
}
2016-09-09 19:30:36 +02:00
boost : : optional < SecondarySkill > CGHeroInstance : : nextSecondarySkill ( CRandomGenerator & rand ) const
2014-06-05 19:52:14 +03:00
{
assert ( gainsLevel ( ) ) ;
boost : : optional < SecondarySkill > chosenSecondarySkill ;
const auto proposedSecondarySkills = getLevelUpProposedSecondarySkills ( ) ;
if ( ! proposedSecondarySkills . empty ( ) )
{
std : : vector < SecondarySkill > learnedSecondarySkills ;
for ( auto secondarySkill : proposedSecondarySkills )
{
if ( getSecSkillLevel ( secondarySkill ) > 0 )
{
learnedSecondarySkills . push_back ( secondarySkill ) ;
}
}
if ( learnedSecondarySkills . empty ( ) )
{
// there are only new skills to learn, so choose anyone of them
chosenSecondarySkill = * RandomGeneratorUtil : : nextItem ( proposedSecondarySkills , rand ) ;
}
else
{
// preferably upgrade a already learned secondary skill
chosenSecondarySkill = * RandomGeneratorUtil : : nextItem ( learnedSecondarySkills , rand ) ;
}
}
return chosenSecondarySkill ;
}
void CGHeroInstance : : setPrimarySkill ( PrimarySkill : : PrimarySkill primarySkill , si64 value , ui8 abs )
{
if ( primarySkill < PrimarySkill : : EXPERIENCE )
{
2016-09-19 23:36:35 +02:00
auto skill = getBonusLocalFirst ( Selector : : type ( Bonus : : PRIMARY_SKILL )
. And ( Selector : : subtype ( primarySkill ) )
. And ( Selector : : sourceType ( Bonus : : HERO_BASE_SKILL ) ) ) ;
2014-06-05 19:52:14 +03:00
assert ( skill ) ;
if ( abs )
{
skill - > val = value ;
}
else
{
skill - > val + = value ;
}
2015-12-11 15:13:18 +02:00
CBonusSystemNode : : treeHasChanged ( ) ;
2014-06-05 19:52:14 +03:00
}
else if ( primarySkill = = PrimarySkill : : EXPERIENCE )
{
if ( abs )
{
exp = value ;
}
else
{
exp + = value ;
}
}
}
bool CGHeroInstance : : gainsLevel ( ) const
{
return exp > = VLC - > heroh - > reqExp ( level + 1 ) ;
}
void CGHeroInstance : : levelUp ( std : : vector < SecondarySkill > skills )
{
+ + level ;
//deterministic secondary skills
skillsInfo . magicSchoolCounter = ( skillsInfo . magicSchoolCounter + 1 ) % maxlevelsToMagicSchool ( ) ;
skillsInfo . wisdomCounter = ( skillsInfo . wisdomCounter + 1 ) % maxlevelsToWisdom ( ) ;
if ( vstd : : contains ( skills , SecondarySkill : : WISDOM ) )
{
skillsInfo . resetWisdomCounter ( ) ;
}
SecondarySkill spellSchools [ ] = {
SecondarySkill : : FIRE_MAGIC , SecondarySkill : : AIR_MAGIC , SecondarySkill : : WATER_MAGIC , SecondarySkill : : EARTH_MAGIC } ;
for ( auto skill : spellSchools )
{
if ( vstd : : contains ( skills , skill ) )
{
skillsInfo . resetMagicSchoolCounter ( ) ;
break ;
}
}
//specialty
Updatespecialty ( ) ;
}
2016-09-09 19:30:36 +02:00
void CGHeroInstance : : levelUpAutomatically ( CRandomGenerator & rand )
2014-06-05 19:52:14 +03:00
{
while ( gainsLevel ( ) )
{
2016-09-09 19:30:36 +02:00
const auto primarySkill = nextPrimarySkill ( rand ) ;
2014-06-05 19:52:14 +03:00
setPrimarySkill ( primarySkill , 1 , false ) ;
auto proposedSecondarySkills = getLevelUpProposedSecondarySkills ( ) ;
2016-09-09 19:30:36 +02:00
const auto secondarySkill = nextSecondarySkill ( rand ) ;
2014-06-05 19:52:14 +03:00
if ( secondarySkill )
{
setSecSkillLevel ( * secondarySkill , 1 , false ) ;
}
//TODO why has the secondary skills to be passed to the method?
levelUp ( proposedSecondarySkills ) ;
}
}
2015-02-06 16:36:09 +02:00
bool CGHeroInstance : : hasVisions ( const CGObjectInstance * target , const int subtype ) const
{
//VISIONS spell support
2016-01-24 01:27:14 +02:00
const std : : string cached = boost : : to_string ( ( boost : : format ( " type_%d__subtype_%d " ) % Bonus : : VISIONS % subtype ) ) ;
2015-02-06 16:36:09 +02:00
const int visionsMultiplier = valOfBonuses ( Selector : : typeSubtype ( Bonus : : VISIONS , subtype ) , cached ) ;
2016-01-24 01:27:14 +02:00
2015-02-06 16:36:09 +02:00
int visionsRange = visionsMultiplier * getPrimSkillLevel ( PrimarySkill : : SPELL_POWER ) ;
2016-01-24 01:27:14 +02:00
if ( visionsMultiplier > 0 )
2015-02-06 16:36:09 +02:00
vstd : : amax ( visionsRange , 3 ) ; //minimum range is 3 tiles, but only if VISIONS bonus present
2016-01-24 01:27:14 +02:00
2015-02-06 16:36:09 +02:00
const int distance = target - > pos . dist2d ( getPosition ( false ) ) ;
2016-01-24 01:27:14 +02:00
2015-03-28 23:17:45 +02:00
//logGlobal->debug(boost::to_string(boost::format("Visions: dist %d, mult %d, range %d") % distance % visionsMultiplier % visionsRange));
2016-01-24 01:27:14 +02:00
return ( distance < visionsRange ) & & ( target - > pos . z = = pos . z ) ;
}
2016-02-22 01:37:19 +02:00
void CGHeroInstance : : serializeJsonOptions ( JsonSerializeFormat & handler )
2016-01-24 01:27:14 +02:00
{
2016-02-22 01:37:19 +02:00
serializeJsonOwner ( handler ) ;
if ( handler . saving )
2016-02-14 10:25:01 +02:00
{
2016-02-22 01:37:19 +02:00
if ( type )
{
handler . serializeString ( " type " , type - > identifier ) ;
}
else
{
auto temp = VLC - > heroh - > heroes [ subID ] - > identifier ;
handler . serializeString ( " type " , temp ) ;
}
2016-02-14 10:25:01 +02:00
}
else
{
2016-02-22 01:37:19 +02:00
if ( ID = = Obj : : HERO | | ID = = Obj : : PRISON )
{
std : : string typeName ;
handler . serializeString ( " type " , typeName ) ;
2016-02-14 11:13:30 +02:00
2016-02-22 01:37:19 +02:00
auto rawId = VLC - > modh - > identifiers . getIdentifier ( " core " , " hero " , typeName ) ;
2016-02-14 10:25:01 +02:00
2016-02-22 01:37:19 +02:00
if ( rawId )
subID = rawId . get ( ) ;
else
subID = 0 ; //fallback to Orrin, throw error instead?
}
}
CCreatureSet : : serializeJson ( handler , " army " ) ;
2015-02-06 16:36:09 +02:00
2016-02-14 10:25:01 +02:00
{
2016-02-22 01:37:19 +02:00
auto artifacts = handler . enterStruct ( " artifacts " ) ;
if ( handler . saving )
CArtifactSet : : writeJson ( handler . getCurrent ( ) ) ;
2016-02-14 10:25:01 +02:00
else
2016-02-22 01:37:19 +02:00
CArtifactSet : : readJson ( handler . getCurrent ( ) ) ;
2016-02-14 10:25:01 +02:00
}
2016-01-24 01:27:14 +02:00
}
2016-02-14 11:13:30 +02:00
2016-01-27 12:47:42 +02:00
bool CGHeroInstance : : isMissionCritical ( ) const
{
for ( const TriggeredEvent & event : IObjectInterface : : cb - > getMapHeader ( ) - > triggeredEvents )
{
if ( event . trigger . test ( [ & ] ( const EventCondition & condition )
{
if ( condition . condition = = EventCondition : : CONTROL & & condition . object )
{
auto hero = dynamic_cast < const CGHeroInstance * > ( condition . object ) ;
return ( hero ! = this ) ;
}
else if ( condition . condition = = EventCondition : : IS_HUMAN )
{
return true ;
}
return false ;
} ) )
{
return true ;
}
}
return false ;
}