2017-07-13 10:26:03 +02:00
/*
* CHeroHandler . 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
# include "StdInc.h"
2007-06-08 17:58:04 +03:00
# include "CHeroHandler.h"
2011-12-14 00:23:17 +03:00
2012-08-25 11:44:51 +03:00
# include "CGeneralTextHandler.h"
2013-07-28 17:49:50 +03:00
# include "filesystem/Filesystem.h"
2012-08-25 11:44:51 +03:00
# include "VCMI_Lib.h"
# include "JsonNode.h"
2012-12-14 18:32:53 +03:00
# include "StringConstants.h"
2017-06-24 16:42:05 +02:00
# include "battle/BattleHex.h"
2013-12-13 21:27:54 +03:00
# include "CCreatureHandler.h"
2012-12-03 19:00:17 +03:00
# include "CModHandler.h"
2012-12-15 11:47:02 +03:00
# include "CTownHandler.h"
2014-06-05 14:19:47 +03:00
# include "mapObjects/CObjectHandler.h" //for hero specialty
2013-11-10 00:29:46 +03:00
# include <math.h>
2009-08-31 18:57:15 +03:00
2014-06-05 19:52:14 +03:00
# include "mapObjects/CObjectClassesHandler.h"
2014-04-10 20:11:09 +03:00
SecondarySkill CHeroClass : : chooseSecSkill ( const std : : set < SecondarySkill > & possibles , CRandomGenerator & rand ) const //picks secondary skill out from given possibilities
2008-08-04 18:56:36 +03:00
{
int totalProb = 0 ;
2013-06-29 16:05:48 +03:00
for ( auto & possible : possibles )
2008-08-04 18:56:36 +03:00
{
2013-06-29 16:05:48 +03:00
totalProb + = secSkillProbability [ possible ] ;
2008-08-04 18:56:36 +03:00
}
2013-09-05 02:00:51 +03:00
if ( totalProb ! = 0 ) // may trigger if set contains only banned skills (0 probability)
2008-08-04 18:56:36 +03:00
{
2014-04-10 20:11:09 +03:00
auto ran = rand . nextInt ( totalProb - 1 ) ;
2013-09-05 02:00:51 +03:00
for ( auto & possible : possibles )
{
ran - = secSkillProbability [ possible ] ;
2014-04-10 20:11:09 +03:00
if ( ran < 0 )
{
2013-09-05 02:00:51 +03:00
return possible ;
2014-04-10 20:11:09 +03:00
}
2013-09-05 02:00:51 +03:00
}
2008-08-04 18:56:36 +03:00
}
2013-09-05 02:00:51 +03:00
// FIXME: select randomly? How H3 handles such rare situation?
return * possibles . begin ( ) ;
2008-08-04 18:56:36 +03:00
}
2013-09-09 18:23:59 +03:00
bool CHeroClass : : isMagicHero ( ) const
{
2013-12-13 21:27:54 +03:00
return affinity = = MAGIC ;
2013-09-09 18:23:59 +03:00
}
2012-09-23 21:01:04 +03:00
EAlignment : : EAlignment CHeroClass : : getAlignment ( ) const
2010-07-20 09:05:45 +03:00
{
2013-04-21 15:49:26 +03:00
return EAlignment : : EAlignment ( VLC - > townh - > factions [ faction ] - > alignment ) ;
2010-07-20 09:05:45 +03:00
}
2014-01-06 20:45:21 +03:00
CHeroClass : : CHeroClass ( )
2016-11-27 21:07:01 +02:00
: faction ( 0 ) , id ( 0 ) , affinity ( 0 ) , defaultTavernChance ( 0 ) , commander ( nullptr )
2014-01-06 20:45:21 +03:00
{
}
2012-04-23 22:56:37 +03:00
std : : vector < BattleHex > CObstacleInfo : : getBlocked ( BattleHex hex ) const
2009-02-09 16:50:32 +02:00
{
2012-04-23 22:56:37 +03:00
std : : vector < BattleHex > ret ;
if ( isAbsoluteObstacle )
2009-02-09 16:50:32 +02:00
{
2012-04-23 22:56:37 +03:00
assert ( ! hex . isValid ( ) ) ;
range : : copy ( blockedTiles , std : : back_inserter ( ret ) ) ;
return ret ;
2009-02-09 16:50:32 +02:00
}
2013-06-29 16:05:48 +03:00
for ( int offset : blockedTiles )
2009-02-09 16:50:32 +02:00
{
2012-04-23 22:56:37 +03:00
BattleHex toBlock = hex + offset ;
if ( ( hex . getY ( ) & 1 ) & & ! ( toBlock . getY ( ) & 1 ) )
toBlock + = BattleHex : : LEFT ;
if ( ! toBlock . isValid ( ) )
2016-08-12 11:10:27 +02:00
logGlobal - > error ( " Misplaced obstacle! " ) ;
2012-04-23 22:56:37 +03:00
else
ret . push_back ( toBlock ) ;
2009-02-09 16:50:32 +02:00
}
return ret ;
}
2017-07-15 13:08:20 +02:00
bool CObstacleInfo : : isAppropriate ( ETerrainType terrainType , int specialBattlefield ) const
2009-09-25 14:38:18 +03:00
{
2012-04-23 22:56:37 +03:00
if ( specialBattlefield ! = - 1 )
return vstd : : contains ( allowedSpecialBfields , specialBattlefield ) ;
return vstd : : contains ( allowedTerrains , terrainType ) ;
2009-09-25 14:38:18 +03:00
}
2015-08-24 10:55:45 +02:00
CHeroClass * CHeroClassHandler : : loadFromJson ( const JsonNode & node , const std : : string & identifier )
2009-01-11 00:08:18 +02:00
{
2013-12-13 21:27:54 +03:00
std : : string affinityStr [ 2 ] = { " might " , " magic " } ;
2013-06-29 16:05:48 +03:00
auto heroClass = new CHeroClass ( ) ;
2015-08-24 10:55:45 +02:00
heroClass - > identifier = identifier ;
2013-04-21 15:49:26 +03:00
heroClass - > imageBattleFemale = node [ " animation " ] [ " battle " ] [ " female " ] . String ( ) ;
heroClass - > imageBattleMale = node [ " animation " ] [ " battle " ] [ " male " ] . String ( ) ;
2014-06-16 19:27:26 +03:00
//MODS COMPATIBILITY FOR 0.96
2013-04-21 15:49:26 +03:00
heroClass - > imageMapFemale = node [ " animation " ] [ " map " ] [ " female " ] . String ( ) ;
heroClass - > imageMapMale = node [ " animation " ] [ " map " ] [ " male " ] . String ( ) ;
heroClass - > name = node [ " name " ] . String ( ) ;
2013-12-13 21:27:54 +03:00
heroClass - > affinity = vstd : : find_pos ( affinityStr , node [ " affinity " ] . String ( ) ) ;
2013-04-21 15:49:26 +03:00
2013-06-29 16:05:48 +03:00
for ( const std : : string & pSkill : PrimarySkill : : names )
2013-04-21 15:49:26 +03:00
{
heroClass - > primarySkillInitial . push_back ( node [ " primarySkills " ] [ pSkill ] . Float ( ) ) ;
heroClass - > primarySkillLowLevel . push_back ( node [ " lowLevelChance " ] [ pSkill ] . Float ( ) ) ;
heroClass - > primarySkillHighLevel . push_back ( node [ " highLevelChance " ] [ pSkill ] . Float ( ) ) ;
}
2013-06-29 16:05:48 +03:00
for ( const std : : string & secSkill : NSecondarySkill : : names )
2013-04-21 15:49:26 +03:00
{
heroClass - > secSkillProbability . push_back ( node [ " secondarySkills " ] [ secSkill ] . Float ( ) ) ;
}
2014-06-23 20:10:08 +03:00
VLC - > modh - > identifiers . requestIdentifier ( " creature " , node [ " commander " ] ,
[ = ] ( si32 commanderID )
2013-12-13 21:27:54 +03:00
{
2014-06-23 20:10:08 +03:00
heroClass - > commander = VLC - > creh - > creatures [ commanderID ] ;
} ) ;
2013-12-13 21:27:54 +03:00
2013-11-03 15:07:23 +03:00
heroClass - > defaultTavernChance = node [ " defaultTavern " ] . Float ( ) ;
2013-06-29 16:05:48 +03:00
for ( auto & tavern : node [ " tavern " ] . Struct ( ) )
2013-04-21 15:49:26 +03:00
{
int value = tavern . second . Float ( ) ;
2013-04-25 17:03:35 +03:00
VLC - > modh - > identifiers . requestIdentifier ( tavern . second . meta , " faction " , tavern . first ,
2013-04-21 15:49:26 +03:00
[ = ] ( si32 factionID )
{
heroClass - > selectionProbability [ factionID ] = value ;
} ) ;
}
2013-04-25 17:03:35 +03:00
VLC - > modh - > identifiers . requestIdentifier ( " faction " , node [ " faction " ] ,
2013-04-21 15:49:26 +03:00
[ = ] ( si32 factionID )
{
heroClass - > faction = factionID ;
} ) ;
return heroClass ;
}
std : : vector < JsonNode > CHeroClassHandler : : loadLegacyData ( size_t dataSize )
{
2013-04-26 00:50:55 +03:00
heroClasses . resize ( dataSize ) ;
2013-04-21 15:49:26 +03:00
std : : vector < JsonNode > h3Data ;
h3Data . reserve ( dataSize ) ;
2012-12-14 18:32:53 +03:00
CLegacyConfigParser parser ( " DATA/HCTRAITS.TXT " ) ;
parser . endLine ( ) ; // header
parser . endLine ( ) ;
2013-04-21 15:49:26 +03:00
for ( size_t i = 0 ; i < dataSize ; i + + )
2012-12-14 18:32:53 +03:00
{
2013-03-12 17:56:23 +03:00
JsonNode entry ;
entry [ " name " ] . String ( ) = parser . readString ( ) ;
parser . readNumber ( ) ; // unused aggression
2012-12-14 18:32:53 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & name : PrimarySkill : : names )
entry [ " primarySkills " ] [ name ] . Float ( ) = parser . readNumber ( ) ;
2012-12-14 18:32:53 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & name : PrimarySkill : : names )
entry [ " lowLevelChance " ] [ name ] . Float ( ) = parser . readNumber ( ) ;
2012-12-14 18:32:53 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & name : PrimarySkill : : names )
entry [ " highLevelChance " ] [ name ] . Float ( ) = parser . readNumber ( ) ;
2009-01-11 00:08:18 +02:00
2013-06-29 16:05:48 +03:00
for ( auto & name : NSecondarySkill : : names )
entry [ " secondarySkills " ] [ name ] . Float ( ) = parser . readNumber ( ) ;
2012-12-14 18:32:53 +03:00
2013-06-29 16:05:48 +03:00
for ( auto & name : ETownType : : names )
entry [ " tavern " ] [ name ] . Float ( ) = parser . readNumber ( ) ;
2012-12-14 18:32:53 +03:00
2013-04-21 15:49:26 +03:00
parser . endLine ( ) ;
2013-03-12 17:56:23 +03:00
h3Data . push_back ( entry ) ;
2012-12-14 18:32:53 +03:00
}
2013-04-21 15:49:26 +03:00
return h3Data ;
2012-12-14 18:32:53 +03:00
}
2013-04-21 15:49:26 +03:00
void CHeroClassHandler : : loadObject ( std : : string scope , std : : string name , const JsonNode & data )
2012-12-14 18:32:53 +03:00
{
2016-02-21 19:58:09 +02:00
auto object = loadFromJson ( data , normalizeIdentifier ( scope , " core " , name ) ) ;
2013-04-21 15:49:26 +03:00
object - > id = heroClasses . size ( ) ;
2012-12-16 16:47:53 +03:00
2013-04-21 15:49:26 +03:00
heroClasses . push_back ( object ) ;
2014-06-15 23:25:10 +03:00
VLC - > modh - > identifiers . requestIdentifier ( scope , " object " , " hero " , [ = ] ( si32 index )
{
2014-06-30 17:11:25 +03:00
JsonNode classConf = data [ " mapObject " ] ;
2014-06-15 23:25:10 +03:00
classConf [ " heroClass " ] . String ( ) = name ;
classConf . setMeta ( scope ) ;
VLC - > objtypeh - > loadSubObject ( name , classConf , index , object - > id ) ;
} ) ;
2013-04-21 15:49:26 +03:00
VLC - > modh - > identifiers . registerObject ( scope , " heroClass " , name , object - > id ) ;
2012-12-14 18:32:53 +03:00
}
2013-04-21 15:49:26 +03:00
void CHeroClassHandler : : loadObject ( std : : string scope , std : : string name , const JsonNode & data , size_t index )
2012-12-14 18:32:53 +03:00
{
2016-02-21 19:58:09 +02:00
auto object = loadFromJson ( data , normalizeIdentifier ( scope , " core " , name ) ) ;
2013-04-21 15:49:26 +03:00
object - > id = index ;
2012-12-16 16:47:53 +03:00
2013-04-21 15:49:26 +03:00
assert ( heroClasses [ index ] = = nullptr ) ; // ensure that this id was not loaded before
heroClasses [ index ] = object ;
2012-12-16 16:47:53 +03:00
2014-06-15 23:25:10 +03:00
VLC - > modh - > identifiers . requestIdentifier ( scope , " object " , " hero " , [ = ] ( si32 index )
{
2014-06-16 19:27:26 +03:00
JsonNode classConf = data [ " mapObject " ] ;
2014-06-15 23:25:10 +03:00
classConf [ " heroClass " ] . String ( ) = name ;
classConf . setMeta ( scope ) ;
VLC - > objtypeh - > loadSubObject ( name , classConf , index , object - > id ) ;
} ) ;
2013-04-21 15:49:26 +03:00
VLC - > modh - > identifiers . registerObject ( scope , " heroClass " , name , object - > id ) ;
}
2012-12-16 16:47:53 +03:00
2013-11-03 15:07:23 +03:00
void CHeroClassHandler : : afterLoadFinalization ( )
{
// for each pair <class, town> set selection probability if it was not set before in tavern entries
for ( CHeroClass * heroClass : heroClasses )
{
for ( CFaction * faction : VLC - > townh - > factions )
{
if ( ! faction - > town )
continue ;
if ( heroClass - > selectionProbability . count ( faction - > index ) )
continue ;
float chance = heroClass - > defaultTavernChance * faction - > town - > defaultTavernChance ;
2013-11-10 00:29:46 +03:00
heroClass - > selectionProbability [ faction - > index ] = static_cast < int > ( sqrt ( chance ) + 0.5 ) ; //FIXME: replace with std::round once MVS supports it
2013-11-03 15:07:23 +03:00
}
}
2014-01-03 02:48:38 +03:00
for ( CHeroClass * hc : heroClasses )
{
2014-06-04 11:25:13 +03:00
if ( ! hc - > imageMapMale . empty ( ) )
{
JsonNode templ ;
templ [ " animation " ] . String ( ) = hc - > imageMapMale ;
VLC - > objtypeh - > getHandlerFor ( Obj : : HERO , hc - > id ) - > addTemplate ( templ ) ;
}
2014-01-03 02:48:38 +03:00
}
2013-11-03 15:07:23 +03:00
}
2013-04-21 15:49:26 +03:00
std : : vector < bool > CHeroClassHandler : : getDefaultAllowed ( ) const
{
return std : : vector < bool > ( heroClasses . size ( ) , true ) ;
2012-12-14 18:32:53 +03:00
}
CHeroClassHandler : : ~ CHeroClassHandler ( )
{
2013-06-29 16:05:48 +03:00
for ( auto heroClass : heroClasses )
2012-12-14 18:32:53 +03:00
{
delete heroClass . get ( ) ;
}
}
CHeroHandler : : ~ CHeroHandler ( )
{
2013-06-29 16:05:48 +03:00
for ( auto hero : heroes )
2012-12-14 18:32:53 +03:00
delete hero . get ( ) ;
2009-01-11 00:08:18 +02:00
}
CHeroHandler : : CHeroHandler ( )
2012-12-26 12:46:09 +03:00
{
2013-04-21 15:49:26 +03:00
VLC - > heroh = this ;
2012-12-16 16:47:53 +03:00
2013-04-21 15:49:26 +03:00
for ( int i = 0 ; i < GameConstants : : SKILL_QUANTITY ; + + i )
{
VLC - > modh - > identifiers . registerObject ( " core " , " skill " , NSecondarySkill : : names [ i ] , i ) ;
2016-11-13 12:38:42 +02:00
VLC - > modh - > identifiers . registerObject ( " core " , " secondarySkill " , NSecondarySkill : : names [ i ] , i ) ;
2013-04-21 15:49:26 +03:00
}
loadObstacles ( ) ;
loadTerrains ( ) ;
2016-02-04 18:57:51 +02:00
for ( int i = 0 ; i < GameConstants : : TERRAIN_TYPES ; + + i )
{
VLC - > modh - > identifiers . registerObject ( " core " , " terrain " , GameConstants : : TERRAIN_NAMES [ i ] , i ) ;
}
2013-04-21 15:49:26 +03:00
loadBallistics ( ) ;
loadExperience ( ) ;
2012-12-15 11:47:02 +03:00
}
2015-08-24 10:55:45 +02:00
CHero * CHeroHandler : : loadFromJson ( const JsonNode & node , const std : : string & identifier )
2012-12-15 11:47:02 +03:00
{
2017-07-16 11:58:05 +02:00
auto hero = new CHero ( ) ;
2015-08-24 10:55:45 +02:00
hero - > identifier = identifier ;
2013-03-03 21:00:37 +03:00
hero - > sex = node [ " female " ] . Bool ( ) ;
hero - > special = node [ " special " ] . Bool ( ) ;
2012-12-16 16:47:53 +03:00
hero - > name = node [ " texts " ] [ " name " ] . String ( ) ;
hero - > biography = node [ " texts " ] [ " biography " ] . String ( ) ;
hero - > specName = node [ " texts " ] [ " specialty " ] [ " name " ] . String ( ) ;
hero - > specTooltip = node [ " texts " ] [ " specialty " ] [ " tooltip " ] . String ( ) ;
hero - > specDescr = node [ " texts " ] [ " specialty " ] [ " description " ] . String ( ) ;
hero - > iconSpecSmall = node [ " images " ] [ " specialtySmall " ] . String ( ) ;
hero - > iconSpecLarge = node [ " images " ] [ " specialtyLarge " ] . String ( ) ;
hero - > portraitSmall = node [ " images " ] [ " small " ] . String ( ) ;
hero - > portraitLarge = node [ " images " ] [ " large " ] . String ( ) ;
2013-03-03 21:00:37 +03:00
loadHeroArmy ( hero , node ) ;
loadHeroSkills ( hero , node ) ;
loadHeroSpecialty ( hero , node ) ;
2013-04-25 17:03:35 +03:00
VLC - > modh - > identifiers . requestIdentifier ( " heroClass " , node [ " class " ] ,
2013-03-03 21:00:37 +03:00
[ = ] ( si32 classID )
{
hero - > heroClass = classes . heroClasses [ classID ] ;
} ) ;
return hero ;
}
void CHeroHandler : : loadHeroArmy ( CHero * hero , const JsonNode & node )
{
2012-12-16 16:47:53 +03:00
assert ( node [ " army " ] . Vector ( ) . size ( ) < = 3 ) ; // anything bigger is useless - army initialization uses up to 3 slots
2013-03-03 21:00:37 +03:00
2012-12-16 16:47:53 +03:00
hero - > initialArmy . resize ( node [ " army " ] . Vector ( ) . size ( ) ) ;
for ( size_t i = 0 ; i < hero - > initialArmy . size ( ) ; i + + )
{
const JsonNode & source = node [ " army " ] . Vector ( ) [ i ] ;
hero - > initialArmy [ i ] . minAmount = source [ " min " ] . Float ( ) ;
hero - > initialArmy [ i ] . maxAmount = source [ " max " ] . Float ( ) ;
assert ( hero - > initialArmy [ i ] . minAmount < = hero - > initialArmy [ i ] . maxAmount ) ;
2013-04-25 17:03:35 +03:00
VLC - > modh - > identifiers . requestIdentifier ( " creature " , source [ " creature " ] , [ = ] ( si32 creature )
2012-12-16 16:47:53 +03:00
{
2013-02-11 02:24:57 +03:00
hero - > initialArmy [ i ] . creature = CreatureID ( creature ) ;
2012-12-16 16:47:53 +03:00
} ) ;
}
}
2013-03-03 21:00:37 +03:00
void CHeroHandler : : loadHeroSkills ( CHero * hero , const JsonNode & node )
2012-12-16 16:47:53 +03:00
{
2013-06-29 16:05:48 +03:00
for ( const JsonNode & set : node [ " skills " ] . Vector ( ) )
2012-12-16 16:47:53 +03:00
{
2013-06-29 16:05:48 +03:00
int skillLevel = boost : : range : : find ( NSecondarySkill : : levels , set [ " level " ] . String ( ) ) - std : : begin ( NSecondarySkill : : levels ) ;
2013-04-23 12:16:20 +03:00
if ( skillLevel < SecSkillLevel : : LEVELS_SIZE )
{
size_t currentIndex = hero - > secSkillsInit . size ( ) ;
2013-04-25 11:22:04 +03:00
hero - > secSkillsInit . push_back ( std : : make_pair ( SecondarySkill ( - 1 ) , skillLevel ) ) ;
2013-04-23 12:16:20 +03:00
2013-04-25 17:03:35 +03:00
VLC - > modh - > identifiers . requestIdentifier ( " skill " , set [ " skill " ] , [ = ] ( si32 id )
2013-04-23 12:16:20 +03:00
{
hero - > secSkillsInit [ currentIndex ] . first = SecondarySkill ( id ) ;
} ) ;
}
else
{
2017-08-30 23:23:19 +02:00
logMod - > error ( " Unknown skill level: %s " , set [ " level " ] . String ( ) ) ;
2013-04-23 12:16:20 +03:00
}
2012-12-16 16:47:53 +03:00
}
2013-03-02 19:55:51 +03:00
// spellbook is considered present if hero have "spellbook" entry even when this is an empty set (0 spells)
2013-03-03 21:00:37 +03:00
hero - > haveSpellBook = ! node [ " spellbook " ] . isNull ( ) ;
2013-03-02 19:55:51 +03:00
2013-06-29 16:05:48 +03:00
for ( const JsonNode & spell : node [ " spellbook " ] . Vector ( ) )
2012-12-16 16:47:53 +03:00
{
2013-08-19 14:50:53 +03:00
VLC - > modh - > identifiers . requestIdentifier ( " spell " , spell ,
[ = ] ( si32 spellID )
2013-03-02 19:55:51 +03:00
{
2013-08-19 14:50:53 +03:00
hero - > spells . insert ( SpellID ( spellID ) ) ;
} ) ;
2012-12-16 16:47:53 +03:00
}
2013-03-03 21:00:37 +03:00
}
2012-12-16 16:47:53 +03:00
2017-09-10 04:10:50 +02:00
// convert deprecated format
std : : vector < std : : shared_ptr < Bonus > > SpecialtyInfoToBonuses ( const SSpecialtyInfo & spec , int sid )
{
std : : vector < std : : shared_ptr < Bonus > > result ;
std : : shared_ptr < Bonus > bonus = std : : make_shared < Bonus > ( ) ;
bonus - > duration = Bonus : : PERMANENT ;
bonus - > source = Bonus : : HERO_SPECIAL ;
bonus - > sid = sid ;
bonus - > val = spec . val ;
switch ( spec . type )
{
case 1 : //creature specialty
{
const CCreature & specCreature = * VLC - > creh - > creatures [ spec . additionalinfo ] ; //creature in which we have specialty
bonus - > limiter . reset ( new CCreatureTypeLimiter ( specCreature , true ) ) ;
bonus - > type = Bonus : : STACKS_SPEED ;
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
bonus - > val = 1 ;
result . push_back ( bonus ) ;
bonus = std : : make_shared < Bonus > ( * bonus ) ;
bonus - > type = Bonus : : PRIMARY_SKILL ;
2017-09-12 01:56:38 +02:00
bonus - > val = 0 ;
2017-09-10 04:10:50 +02:00
int stepSize = specCreature . level ? specCreature . level : 5 ;
bonus - > subtype = PrimarySkill : : ATTACK ;
bonus - > updater . reset ( new ScalingUpdater ( specCreature . getAttack ( false ) , stepSize ) ) ;
result . push_back ( bonus ) ;
bonus = std : : make_shared < Bonus > ( * bonus ) ;
bonus - > subtype = PrimarySkill : : DEFENSE ;
bonus - > updater . reset ( new ScalingUpdater ( specCreature . getDefence ( false ) , stepSize ) ) ;
result . push_back ( bonus ) ;
}
break ;
case 2 : //secondary skill
bonus - > type = Bonus : : SECONDARY_SKILL_PREMY ;
bonus - > valType = Bonus : : PERCENT_TO_BASE ;
bonus - > subtype = spec . subtype ;
2017-09-12 01:56:38 +02:00
bonus - > val = 0 ;
2017-09-10 04:10:50 +02:00
bonus - > updater . reset ( new ScalingUpdater ( spec . val * 20 ) ) ;
result . push_back ( bonus ) ;
break ;
case 3 : //spell damage bonus, level dependent but calculated elsewhere
bonus - > type = Bonus : : SPECIAL_SPELL_LEV ;
bonus - > subtype = spec . subtype ;
result . push_back ( bonus ) ;
break ;
case 4 : //creature stat boost
switch ( spec . subtype )
{
case 1 :
bonus - > type = Bonus : : PRIMARY_SKILL ;
bonus - > subtype = PrimarySkill : : ATTACK ;
break ;
case 2 :
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 :
bonus - > type = Bonus : : STACK_HEALTH ;
break ;
case 5 :
bonus - > type = Bonus : : STACKS_SPEED ;
break ;
default :
logMod - > warn ( " Unknown subtype for specialty 4 " ) ;
return result ;
}
bonus - > valType = Bonus : : ADDITIVE_VALUE ;
bonus - > limiter . reset ( new CCreatureTypeLimiter ( * VLC - > creh - > creatures [ spec . additionalinfo ] , true ) ) ;
result . push_back ( 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
result . push_back ( 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
result . push_back ( bonus ) ;
break ;
case 7 : //maxed mastery for spell
bonus - > type = Bonus : : MAXED_SPELL ;
bonus - > subtype = spec . subtype ; //spell id
result . push_back ( 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
result . push_back ( 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
result . push_back ( bonus ) ;
//propagate for regular upgrades of base creature
for ( auto cre_id : creatures [ spec . subtype ] - > upgrades )
{
bonus = std : : make_shared < Bonus > ( * bonus ) ;
bonus - > subtype = cre_id ;
result . push_back ( bonus ) ;
}
}
break ;
case 10 : //resource generation
bonus - > type = Bonus : : GENERATE_RESOURCE ;
bonus - > subtype = spec . subtype ;
result . push_back ( bonus ) ;
break ;
case 11 : //starting skill with mastery (Adrienne)
logMod - > warn ( " Secondary skill mastery is no longer supported as specialty. " ) ;
break ;
case 12 : //army speed
bonus - > type = Bonus : : STACKS_SPEED ;
result . push_back ( 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 ) ) ;
result . push_back ( bonus ) ;
break ;
default :
logMod - > warn ( " Unknown hero specialty %d " , spec . type ) ;
break ;
}
return result ;
}
2013-03-03 21:00:37 +03:00
void CHeroHandler : : loadHeroSpecialty ( CHero * hero , const JsonNode & node )
{
2017-09-10 04:10:50 +02:00
int sid = hero - > ID . getNum ( ) ;
auto prepSpec = [ = ] ( std : : shared_ptr < Bonus > bonus )
{
bonus - > duration = Bonus : : PERMANENT ;
bonus - > source = Bonus : : HERO_SPECIAL ;
bonus - > sid = sid ;
return bonus ;
} ;
2017-09-11 08:21:24 +02:00
//deprecated, used only for original specialties
const JsonNode & specialties = node [ " specialties " ] ;
if ( ! specialties . isNull ( ) )
2012-12-16 16:47:53 +03:00
{
2017-09-12 14:59:50 +02:00
logMod - > warn ( " Hero %s has deprecated specialties format. " , hero - > identifier ) ;
2017-09-11 08:21:24 +02:00
for ( const JsonNode & specialty : node [ " specialties " ] . Vector ( ) )
{
SSpecialtyInfo spec ;
spec . type = specialty [ " type " ] . Float ( ) ;
spec . val = specialty [ " val " ] . Float ( ) ;
spec . subtype = specialty [ " subtype " ] . Float ( ) ;
spec . additionalinfo = specialty [ " info " ] . Float ( ) ;
2017-09-12 14:59:50 +02:00
//we convert after loading completes, to have all identifiers for json logging
hero - > specDeprecated . push_back ( spec ) ;
2017-09-11 08:21:24 +02:00
}
2012-12-16 16:47:53 +03:00
}
2013-01-19 20:38:37 +03:00
//new format, using bonus system
2017-09-10 04:10:50 +02:00
for ( const JsonNode & specialty : node [ " specialty " ] . Vector ( ) )
2013-01-19 20:38:37 +03:00
{
2017-09-10 04:10:50 +02:00
//deprecated new format
if ( ! specialty [ " bonuses " ] . isNull ( ) )
{
for ( const JsonNode & bonus : specialty [ " bonuses " ] . Vector ( ) )
hero - > specialty . push_back ( prepSpec ( JsonUtils : : parseBonus ( bonus ) ) ) ;
}
else //proper new format
2013-01-19 20:38:37 +03:00
{
2017-09-10 04:10:50 +02:00
hero - > specialty . push_back ( prepSpec ( JsonUtils : : parseBonus ( specialty ) ) ) ;
2013-01-19 20:38:37 +03:00
}
}
2012-12-15 11:47:02 +03:00
}
void CHeroHandler : : loadExperience ( )
{
expPerLevel . push_back ( 0 ) ;
expPerLevel . push_back ( 1000 ) ;
expPerLevel . push_back ( 2000 ) ;
expPerLevel . push_back ( 3200 ) ;
expPerLevel . push_back ( 4600 ) ;
expPerLevel . push_back ( 6200 ) ;
expPerLevel . push_back ( 8000 ) ;
expPerLevel . push_back ( 10000 ) ;
expPerLevel . push_back ( 12200 ) ;
expPerLevel . push_back ( 14700 ) ;
expPerLevel . push_back ( 17500 ) ;
expPerLevel . push_back ( 20600 ) ;
expPerLevel . push_back ( 24320 ) ;
expPerLevel . push_back ( 28784 ) ;
expPerLevel . push_back ( 34140 ) ;
while ( expPerLevel [ expPerLevel . size ( ) - 1 ] > expPerLevel [ expPerLevel . size ( ) - 2 ] )
{
int i = expPerLevel . size ( ) - 1 ;
expPerLevel . push_back ( expPerLevel [ i ] + ( expPerLevel [ i ] - expPerLevel [ i - 1 ] ) * 1.2 ) ;
}
expPerLevel . pop_back ( ) ; //last value is broken
2012-12-14 18:32:53 +03:00
}
2009-02-07 18:07:29 +02:00
void CHeroHandler : : loadObstacles ( )
{
2012-04-23 22:56:37 +03:00
auto loadObstacles = [ ] ( const JsonNode & node , bool absolute , std : : map < int , CObstacleInfo > & out )
{
2013-06-29 16:05:48 +03:00
for ( const JsonNode & obs : node . Vector ( ) )
2012-04-23 22:56:37 +03:00
{
int ID = obs [ " id " ] . Float ( ) ;
CObstacleInfo & obi = out [ ID ] ;
obi . ID = ID ;
obi . defName = obs [ " defname " ] . String ( ) ;
obi . width = obs [ " width " ] . Float ( ) ;
obi . height = obs [ " height " ] . Float ( ) ;
2013-02-13 01:24:48 +03:00
obi . allowedTerrains = obs [ " allowedTerrain " ] . convertTo < std : : vector < ETerrainType > > ( ) ;
obi . allowedSpecialBfields = obs [ " specialBattlefields " ] . convertTo < std : : vector < BFieldType > > ( ) ;
2012-12-02 15:21:44 +03:00
obi . blockedTiles = obs [ " blockedTiles " ] . convertTo < std : : vector < si16 > > ( ) ;
2012-04-23 22:56:37 +03:00
obi . isAbsoluteObstacle = absolute ;
}
} ;
2011-09-02 05:12:55 +03:00
2012-08-02 14:03:26 +03:00
const JsonNode config ( ResourceID ( " config/obstacles.json " ) ) ;
2012-04-23 22:56:37 +03:00
loadObstacles ( config [ " obstacles " ] , false , obstacles ) ;
loadObstacles ( config [ " absoluteObstacles " ] , true , absoluteObstacles ) ;
2012-05-18 23:50:16 +03:00
//loadObstacles(config["moats"], true, moats);
2009-02-07 18:07:29 +02:00
}
2013-03-03 21:00:37 +03:00
/// convert h3-style ID (e.g. Gobin Wolf Rider) to vcmi (e.g. goblinWolfRider)
static std : : string genRefName ( std : : string input )
{
boost : : algorithm : : replace_all ( input , " " , " " ) ; //remove spaces
input [ 0 ] = std : : tolower ( input [ 0 ] ) ; // to camelCase
return input ;
}
2013-04-21 15:49:26 +03:00
void CHeroHandler : : loadBallistics ( )
{
CLegacyConfigParser ballParser ( " DATA/BALLIST.TXT " ) ;
ballParser . endLine ( ) ; //header
ballParser . endLine ( ) ;
do
{
ballParser . readString ( ) ;
ballParser . readString ( ) ;
CHeroHandler : : SBallisticsLevelInfo bli ;
bli . keep = ballParser . readNumber ( ) ;
bli . tower = ballParser . readNumber ( ) ;
bli . gate = ballParser . readNumber ( ) ;
bli . wall = ballParser . readNumber ( ) ;
bli . shots = ballParser . readNumber ( ) ;
bli . noDmg = ballParser . readNumber ( ) ;
bli . oneDmg = ballParser . readNumber ( ) ;
bli . twoDmg = ballParser . readNumber ( ) ;
bli . sum = ballParser . readNumber ( ) ;
ballistics . push_back ( bli ) ;
2013-08-06 14:20:28 +03:00
assert ( bli . noDmg + bli . oneDmg + bli . twoDmg = = 100 & & bli . sum = = 100 ) ;
2013-04-21 15:49:26 +03:00
}
while ( ballParser . endLine ( ) ) ;
}
std : : vector < JsonNode > CHeroHandler : : loadLegacyData ( size_t dataSize )
2007-06-08 17:58:04 +03:00
{
2013-04-21 15:49:26 +03:00
heroes . resize ( dataSize ) ;
std : : vector < JsonNode > h3Data ;
h3Data . reserve ( dataSize ) ;
2013-03-02 19:55:51 +03:00
CLegacyConfigParser specParser ( " DATA/HEROSPEC.TXT " ) ;
CLegacyConfigParser bioParser ( " DATA/HEROBIOS.TXT " ) ;
2012-08-25 11:44:51 +03:00
CLegacyConfigParser parser ( " DATA/HOTRAITS.TXT " ) ;
parser . endLine ( ) ; //ignore header
parser . endLine ( ) ;
2007-08-04 22:01:22 +03:00
2013-03-02 19:55:51 +03:00
specParser . endLine ( ) ; //ignore header
specParser . endLine ( ) ;
2011-12-14 00:23:17 +03:00
for ( int i = 0 ; i < GameConstants : : HEROES_QUANTITY ; i + + )
2007-06-08 17:58:04 +03:00
{
2013-03-03 21:00:37 +03:00
JsonNode heroData ;
heroData [ " texts " ] [ " name " ] . String ( ) = parser . readString ( ) ;
heroData [ " texts " ] [ " biography " ] . String ( ) = bioParser . readString ( ) ;
heroData [ " texts " ] [ " specialty " ] [ " name " ] . String ( ) = specParser . readString ( ) ;
heroData [ " texts " ] [ " specialty " ] [ " tooltip " ] . String ( ) = specParser . readString ( ) ;
heroData [ " texts " ] [ " specialty " ] [ " description " ] . String ( ) = specParser . readString ( ) ;
2007-08-04 22:01:22 +03:00
2008-05-03 18:30:11 +03:00
for ( int x = 0 ; x < 3 ; x + + )
{
2013-03-03 21:00:37 +03:00
JsonNode armySlot ;
armySlot [ " min " ] . Float ( ) = parser . readNumber ( ) ;
armySlot [ " max " ] . Float ( ) = parser . readNumber ( ) ;
armySlot [ " creature " ] . String ( ) = genRefName ( parser . readString ( ) ) ;
2012-12-03 19:00:17 +03:00
2013-03-03 21:00:37 +03:00
heroData [ " army " ] . Vector ( ) . push_back ( armySlot ) ;
2008-05-03 18:30:11 +03:00
}
2012-08-25 11:44:51 +03:00
parser . endLine ( ) ;
2013-03-02 19:55:51 +03:00
specParser . endLine ( ) ;
bioParser . endLine ( ) ;
2012-08-25 11:44:51 +03:00
2013-03-03 21:00:37 +03:00
h3Data . push_back ( heroData ) ;
2007-06-08 17:58:04 +03:00
}
2013-04-21 15:49:26 +03:00
return h3Data ;
}
2011-08-27 18:53:45 +03:00
2013-04-21 15:49:26 +03:00
void CHeroHandler : : loadObject ( std : : string scope , std : : string name , const JsonNode & data )
{
2016-02-21 19:58:09 +02:00
auto object = loadFromJson ( data , normalizeIdentifier ( scope , " core " , name ) ) ;
2013-05-19 01:30:48 +03:00
object - > ID = HeroTypeID ( heroes . size ( ) ) ;
2016-11-13 12:38:42 +02:00
object - > imageIndex = heroes . size ( ) + GameConstants : : HERO_PORTRAIT_SHIFT ; // 2 special frames + some extra portraits
2013-03-03 21:00:37 +03:00
2013-04-21 15:49:26 +03:00
heroes . push_back ( object ) ;
2013-03-03 21:00:37 +03:00
2013-05-19 01:30:48 +03:00
VLC - > modh - > identifiers . registerObject ( scope , " hero " , name , object - > ID . getNum ( ) ) ;
2012-12-16 16:47:53 +03:00
}
2011-08-27 20:33:07 +03:00
2013-04-21 15:49:26 +03:00
void CHeroHandler : : loadObject ( std : : string scope , std : : string name , const JsonNode & data , size_t index )
2012-12-15 11:47:02 +03:00
{
2016-02-21 19:58:09 +02:00
auto object = loadFromJson ( data , normalizeIdentifier ( scope , " core " , name ) ) ;
2013-05-19 01:30:48 +03:00
object - > ID = HeroTypeID ( index ) ;
2013-04-22 22:51:22 +03:00
object - > imageIndex = index ;
2012-08-25 11:44:51 +03:00
2013-04-21 15:49:26 +03:00
assert ( heroes [ index ] = = nullptr ) ; // ensure that this id was not loaded before
heroes [ index ] = object ;
2012-08-25 11:44:51 +03:00
2013-05-19 01:30:48 +03:00
VLC - > modh - > identifiers . registerObject ( scope , " hero " , name , object - > ID . getNum ( ) ) ;
2007-06-09 16:28:03 +03:00
}
2011-08-27 20:33:07 +03:00
2017-09-12 14:59:50 +02:00
void CHeroHandler : : afterLoadFinalization ( )
{
for ( ConstTransitivePtr < CHero > hero : heroes )
{
if ( hero - > specDeprecated . size ( ) > 0 )
{
logMod - > debug ( " Converting specialties format for hero %s(%s) " , hero - > identifier , VLC - > townh - > encodeFaction ( hero - > heroClass - > faction ) ) ;
JsonNode specVec ( JsonNode : : JsonType : : DATA_VECTOR ) ;
for ( const SSpecialtyInfo & spec : hero - > specDeprecated )
{
for ( std : : shared_ptr < Bonus > bonus : SpecialtyInfoToBonuses ( spec , hero - > ID . getNum ( ) ) )
{
hero - > specialty . push_back ( bonus ) ;
specVec . Vector ( ) . push_back ( bonus - > toJsonNode ( ) ) ;
}
}
hero - > specDeprecated . clear ( ) ;
2017-09-12 15:35:57 +02:00
logMod - > trace ( " \" specialty \" : %s " , specVec . toJson ( ) ) ;
2017-09-12 14:59:50 +02:00
}
}
}
2011-12-14 00:23:17 +03:00
ui32 CHeroHandler : : level ( ui64 experience ) const
2007-08-12 20:48:05 +03:00
{
2013-06-29 16:05:48 +03:00
return boost : : range : : upper_bound ( expPerLevel , experience ) - std : : begin ( expPerLevel ) ;
2008-01-27 18:07:27 +02:00
}
2011-12-14 00:23:17 +03:00
ui64 CHeroHandler : : reqExp ( ui32 level ) const
2008-01-27 18:07:27 +02:00
{
2009-07-20 06:30:48 +03:00
if ( ! level )
return 0 ;
2009-10-28 12:45:45 +02:00
if ( level < = expPerLevel . size ( ) )
2009-07-20 06:30:48 +03:00
{
2009-10-28 12:45:45 +02:00
return expPerLevel [ level - 1 ] ;
2009-07-20 06:30:48 +03:00
}
2008-01-27 18:07:27 +02:00
else
2007-09-14 16:11:10 +03:00
{
2016-08-12 11:10:27 +02:00
logGlobal - > warn ( " A hero has reached unsupported amount of experience " ) ;
2010-08-16 12:54:09 +03:00
return expPerLevel [ expPerLevel . size ( ) - 1 ] ;
2007-09-14 16:11:10 +03:00
}
2007-08-12 20:48:05 +03:00
}
2011-09-02 06:39:49 +03:00
void CHeroHandler : : loadTerrains ( )
2007-08-15 18:13:11 +03:00
{
2012-08-02 14:03:26 +03:00
const JsonNode config ( ResourceID ( " config/terrains.json " ) ) ;
2009-06-30 15:28:22 +03:00
2012-09-05 15:49:23 +03:00
terrCosts . reserve ( GameConstants : : TERRAIN_TYPES ) ;
2013-06-29 16:05:48 +03:00
for ( const std : : string & name : GameConstants : : TERRAIN_NAMES )
2012-12-10 17:28:27 +03:00
terrCosts . push_back ( config [ name ] [ " moveCost " ] . Float ( ) ) ;
2009-06-30 15:28:22 +03:00
}
2009-07-03 21:40:36 +03:00
2013-04-21 15:49:26 +03:00
std : : vector < bool > CHeroHandler : : getDefaultAllowed ( ) const
2012-11-20 20:53:45 +03:00
{
// Look Data/HOTRAITS.txt for reference
2013-02-05 00:58:42 +03:00
std : : vector < bool > allowedHeroes ;
2012-12-17 15:55:29 +03:00
allowedHeroes . reserve ( heroes . size ( ) ) ;
2013-06-29 16:05:48 +03:00
for ( const CHero * hero : heroes )
2012-11-20 20:53:45 +03:00
{
2012-12-17 15:55:29 +03:00
allowedHeroes . push_back ( ! hero - > special ) ;
2012-11-20 20:53:45 +03:00
}
2012-12-17 15:55:29 +03:00
2012-11-20 20:53:45 +03:00
return allowedHeroes ;
2012-12-17 15:55:29 +03:00
}
2013-01-06 22:30:12 +03:00
2013-02-05 00:58:42 +03:00
std : : vector < bool > CHeroHandler : : getDefaultAllowedAbilities ( ) const
2013-01-06 22:30:12 +03:00
{
2013-02-05 00:58:42 +03:00
std : : vector < bool > allowedAbilities ;
allowedAbilities . resize ( GameConstants : : SKILL_QUANTITY , true ) ;
2013-01-06 22:30:12 +03:00
return allowedAbilities ;
}
2016-02-21 19:58:09 +02:00
2016-02-21 21:13:20 +02:00
si32 CHeroHandler : : decodeHero ( const std : : string & identifier )
{
auto rawId = VLC - > modh - > identifiers . getIdentifier ( " core " , " hero " , identifier ) ;
if ( rawId )
return rawId . get ( ) ;
else
return - 1 ;
}
std : : string CHeroHandler : : encodeHero ( const si32 index )
{
return VLC - > heroh - > heroes . at ( index ) - > identifier ;
}
si32 CHeroHandler : : decodeSkill ( const std : : string & identifier )
2016-02-21 19:58:09 +02:00
{
auto rawId = VLC - > modh - > identifiers . getIdentifier ( " core " , " skill " , identifier ) ;
if ( rawId )
return rawId . get ( ) ;
else
return - 1 ;
}
std : : string CHeroHandler : : encodeSkill ( const si32 index )
{
return NSecondarySkill : : names [ index ] ;
}