2023-10-19 16:19:09 +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
*
*/
# include "StdInc.h"
# include "CHeroHandler.h"
2024-10-11 16:30:16 +00:00
# include "CHero.h"
# include "../../VCMI_Lib.h"
# include "../../constants/StringConstants.h"
# include "../../CCreatureHandler.h"
# include "../../IGameSettings.h"
# include "../../bonuses/Limiters.h"
# include "../../bonuses/Updaters.h"
# include "../../json/JsonBonus.h"
# include "../../json/JsonUtils.h"
# include "../../modding/IdentifierStorage.h"
# include "../../texts/CGeneralTextHandler.h"
# include "../../texts/CLegacyConfigParser.h"
2024-06-01 15:28:17 +00:00
2023-10-19 16:19:09 +02:00
VCMI_LIB_NAMESPACE_BEGIN
CHeroHandler : : ~ CHeroHandler ( ) = default ;
CHeroHandler : : CHeroHandler ( )
{
loadExperience ( ) ;
}
const std : : vector < std : : string > & CHeroHandler : : getTypeNames ( ) const
{
static const std : : vector < std : : string > typeNames = { " hero " } ;
return typeNames ;
}
2024-05-16 22:05:51 +00:00
std : : shared_ptr < CHero > CHeroHandler : : loadFromJson ( const std : : string & scope , const JsonNode & node , const std : : string & identifier , size_t index )
2023-10-19 16:19:09 +02:00
{
assert ( identifier . find ( ' : ' ) = = std : : string : : npos ) ;
assert ( ! scope . empty ( ) ) ;
2024-05-16 22:05:51 +00:00
auto hero = std : : make_shared < CHero > ( ) ;
2023-10-19 16:19:09 +02:00
hero - > ID = HeroTypeID ( index ) ;
hero - > identifier = identifier ;
hero - > modScope = scope ;
hero - > gender = node [ " female " ] . Bool ( ) ? EHeroGender : : FEMALE : EHeroGender : : MALE ;
hero - > special = node [ " special " ] . Bool ( ) ;
//Default - both false
hero - > onlyOnWaterMap = node [ " onlyOnWaterMap " ] . Bool ( ) ;
hero - > onlyOnMapWithoutWater = node [ " onlyOnMapWithoutWater " ] . Bool ( ) ;
2024-09-30 10:21:10 +00:00
VLC - > generaltexth - > registerString ( scope , hero - > getNameTextID ( ) , node [ " texts " ] [ " name " ] ) ;
VLC - > generaltexth - > registerString ( scope , hero - > getBiographyTextID ( ) , node [ " texts " ] [ " biography " ] ) ;
VLC - > generaltexth - > registerString ( scope , hero - > getSpecialtyNameTextID ( ) , node [ " texts " ] [ " specialty " ] [ " name " ] ) ;
VLC - > generaltexth - > registerString ( scope , hero - > getSpecialtyTooltipTextID ( ) , node [ " texts " ] [ " specialty " ] [ " tooltip " ] ) ;
VLC - > generaltexth - > registerString ( scope , hero - > getSpecialtyDescriptionTextID ( ) , node [ " texts " ] [ " specialty " ] [ " description " ] ) ;
2023-10-19 16:19:09 +02:00
hero - > iconSpecSmall = node [ " images " ] [ " specialtySmall " ] . String ( ) ;
hero - > iconSpecLarge = node [ " images " ] [ " specialtyLarge " ] . String ( ) ;
hero - > portraitSmall = node [ " images " ] [ " small " ] . String ( ) ;
hero - > portraitLarge = node [ " images " ] [ " large " ] . String ( ) ;
hero - > battleImage = AnimationPath : : fromJson ( node [ " battleImage " ] ) ;
2024-05-16 22:05:51 +00:00
loadHeroArmy ( hero . get ( ) , node ) ;
loadHeroSkills ( hero . get ( ) , node ) ;
loadHeroSpecialty ( hero . get ( ) , node ) ;
2023-10-19 16:19:09 +02:00
VLC - > identifiers ( ) - > requestIdentifier ( " heroClass " , node [ " class " ] ,
[ = ] ( si32 classID )
{
2023-12-31 23:43:35 +02:00
hero - > heroClass = HeroClassID ( classID ) . toHeroClass ( ) ;
2023-10-19 16:19:09 +02:00
} ) ;
return hero ;
}
void CHeroHandler : : loadHeroArmy ( CHero * hero , const JsonNode & node ) const
{
assert ( node [ " army " ] . Vector ( ) . size ( ) < = 3 ) ; // anything bigger is useless - army initialization uses up to 3 slots
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 = static_cast < ui32 > ( source [ " min " ] . Float ( ) ) ;
hero - > initialArmy [ i ] . maxAmount = static_cast < ui32 > ( source [ " max " ] . Float ( ) ) ;
2024-01-15 18:19:21 +02:00
if ( hero - > initialArmy [ i ] . minAmount > hero - > initialArmy [ i ] . maxAmount )
{
logMod - > error ( " Hero %s has minimal army size (%d) greater than maximal size (%d)! " , hero - > getJsonKey ( ) , hero - > initialArmy [ i ] . minAmount , hero - > initialArmy [ i ] . maxAmount ) ;
std : : swap ( hero - > initialArmy [ i ] . minAmount , hero - > initialArmy [ i ] . maxAmount ) ;
}
2023-10-19 16:19:09 +02:00
VLC - > identifiers ( ) - > requestIdentifier ( " creature " , source [ " creature " ] , [ = ] ( si32 creature )
{
hero - > initialArmy [ i ] . creature = CreatureID ( creature ) ;
} ) ;
}
}
void CHeroHandler : : loadHeroSkills ( CHero * hero , const JsonNode & node ) const
{
for ( const JsonNode & set : node [ " skills " ] . Vector ( ) )
{
int skillLevel = static_cast < int > ( boost : : range : : find ( NSecondarySkill : : levels , set [ " level " ] . String ( ) ) - std : : begin ( NSecondarySkill : : levels ) ) ;
2023-10-22 18:36:41 +03:00
if ( skillLevel < MasteryLevel : : LEVELS_SIZE )
2023-10-19 16:19:09 +02:00
{
size_t currentIndex = hero - > secSkillsInit . size ( ) ;
hero - > secSkillsInit . emplace_back ( SecondarySkill ( - 1 ) , skillLevel ) ;
VLC - > identifiers ( ) - > requestIdentifier ( " skill " , set [ " skill " ] , [ = ] ( si32 id )
{
hero - > secSkillsInit [ currentIndex ] . first = SecondarySkill ( id ) ;
} ) ;
}
else
{
logMod - > error ( " Unknown skill level: %s " , set [ " level " ] . String ( ) ) ;
}
}
// spellbook is considered present if hero have "spellbook" entry even when this is an empty set (0 spells)
hero - > haveSpellBook = ! node [ " spellbook " ] . isNull ( ) ;
for ( const JsonNode & spell : node [ " spellbook " ] . Vector ( ) )
{
VLC - > identifiers ( ) - > requestIdentifier ( " spell " , spell ,
[ = ] ( si32 spellID )
{
hero - > spells . insert ( SpellID ( spellID ) ) ;
} ) ;
}
}
/// creates standard H3 hero specialty for creatures
static std : : vector < std : : shared_ptr < Bonus > > createCreatureSpecialty ( CreatureID baseCreatureID )
{
std : : vector < std : : shared_ptr < Bonus > > result ;
std : : set < CreatureID > targets ;
targets . insert ( baseCreatureID ) ;
// go through entire upgrade chain and collect all creatures to which baseCreatureID can be upgraded
for ( ; ; )
{
std : : set < CreatureID > oldTargets = targets ;
2024-01-10 21:30:12 +02:00
for ( const auto & upgradeSourceID : oldTargets )
2023-10-19 16:19:09 +02:00
{
2023-11-05 18:58:07 +02:00
const CCreature * upgradeSource = upgradeSourceID . toCreature ( ) ;
2023-10-19 16:19:09 +02:00
targets . insert ( upgradeSource - > upgrades . begin ( ) , upgradeSource - > upgrades . end ( ) ) ;
}
if ( oldTargets . size ( ) = = targets . size ( ) )
break ;
}
for ( CreatureID cid : targets )
{
2024-01-10 21:30:12 +02:00
const auto & specCreature = * cid . toCreature ( ) ;
2023-10-19 16:19:09 +02:00
int stepSize = specCreature . getLevel ( ) ? specCreature . getLevel ( ) : 5 ;
{
2024-01-17 12:50:00 +00:00
auto bonus = std : : make_shared < Bonus > ( ) ;
2023-10-19 16:19:09 +02:00
bonus - > limiter . reset ( new CCreatureTypeLimiter ( specCreature , false ) ) ;
bonus - > type = BonusType : : STACKS_SPEED ;
bonus - > val = 1 ;
result . push_back ( bonus ) ;
}
{
2024-01-17 12:50:00 +00:00
auto bonus = std : : make_shared < Bonus > ( ) ;
2023-10-19 16:19:09 +02:00
bonus - > type = BonusType : : PRIMARY_SKILL ;
2023-10-22 18:36:41 +03:00
bonus - > subtype = BonusSubtypeID ( PrimarySkill : : ATTACK ) ;
2023-10-19 16:19:09 +02:00
bonus - > val = 0 ;
bonus - > limiter . reset ( new CCreatureTypeLimiter ( specCreature , false ) ) ;
bonus - > updater . reset ( new GrowsWithLevelUpdater ( specCreature . getAttack ( false ) , stepSize ) ) ;
result . push_back ( bonus ) ;
}
{
2024-01-17 12:50:00 +00:00
auto bonus = std : : make_shared < Bonus > ( ) ;
2023-10-19 16:19:09 +02:00
bonus - > type = BonusType : : PRIMARY_SKILL ;
2023-10-22 18:36:41 +03:00
bonus - > subtype = BonusSubtypeID ( PrimarySkill : : DEFENSE ) ;
2023-10-19 16:19:09 +02:00
bonus - > val = 0 ;
bonus - > limiter . reset ( new CCreatureTypeLimiter ( specCreature , false ) ) ;
bonus - > updater . reset ( new GrowsWithLevelUpdater ( specCreature . getDefense ( false ) , stepSize ) ) ;
result . push_back ( bonus ) ;
}
}
return result ;
}
void CHeroHandler : : beforeValidate ( JsonNode & object )
{
//handle "base" specialty info
JsonNode & specialtyNode = object [ " specialty " ] ;
if ( specialtyNode . getType ( ) = = JsonNode : : JsonType : : DATA_STRUCT )
{
const JsonNode & base = specialtyNode [ " base " ] ;
if ( ! base . isNull ( ) )
{
if ( specialtyNode [ " bonuses " ] . isNull ( ) )
{
logMod - > warn ( " specialty has base without bonuses " ) ;
}
else
{
JsonMap & bonuses = specialtyNode [ " bonuses " ] . Struct ( ) ;
for ( std : : pair < std : : string , JsonNode > keyValue : bonuses )
JsonUtils : : inherit ( bonuses [ keyValue . first ] , base ) ;
}
}
}
}
void CHeroHandler : : afterLoadFinalization ( )
{
2024-01-10 21:30:12 +02:00
for ( const auto & functor : callAfterLoadFinalization )
2023-10-19 16:19:09 +02:00
functor ( ) ;
callAfterLoadFinalization . clear ( ) ;
}
void CHeroHandler : : loadHeroSpecialty ( CHero * hero , const JsonNode & node )
{
auto prepSpec = [ = ] ( std : : shared_ptr < Bonus > bonus )
{
bonus - > duration = BonusDuration : : PERMANENT ;
bonus - > source = BonusSource : : HERO_SPECIAL ;
2023-10-22 18:36:41 +03:00
bonus - > sid = BonusSourceID ( hero - > getId ( ) ) ;
2023-10-19 16:19:09 +02:00
return bonus ;
} ;
//new format, using bonus system
const JsonNode & specialtyNode = node [ " specialty " ] ;
if ( specialtyNode . getType ( ) ! = JsonNode : : JsonType : : DATA_STRUCT )
{
logMod - > error ( " Unsupported speciality format for hero %s! " , hero - > getNameTranslated ( ) ) ;
return ;
}
//creature specialty - alias for simplicity
if ( ! specialtyNode [ " creature " ] . isNull ( ) )
{
JsonNode creatureNode = specialtyNode [ " creature " ] ;
std : : function < void ( ) > specialtyLoader = [ creatureNode , hero , prepSpec ]
{
VLC - > identifiers ( ) - > requestIdentifier ( " creature " , creatureNode , [ hero , prepSpec ] ( si32 creature )
{
for ( const auto & bonus : createCreatureSpecialty ( CreatureID ( creature ) ) )
hero - > specialty . push_back ( prepSpec ( bonus ) ) ;
} ) ;
} ;
callAfterLoadFinalization . push_back ( specialtyLoader ) ;
}
for ( const auto & keyValue : specialtyNode [ " bonuses " ] . Struct ( ) )
hero - > specialty . push_back ( prepSpec ( JsonUtils : : parseBonus ( keyValue . second ) ) ) ;
}
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 ) ;
2024-01-04 23:57:36 +02:00
for ( ; ; )
2023-10-19 16:19:09 +02:00
{
auto i = expPerLevel . size ( ) - 1 ;
2024-01-04 23:57:36 +02:00
auto currExp = expPerLevel [ i ] ;
auto prevExp = expPerLevel [ i - 1 ] ;
auto prevDiff = currExp - prevExp ;
auto nextDiff = prevDiff + prevDiff / 5 ;
auto maxExp = std : : numeric_limits < decltype ( currExp ) > : : max ( ) ;
if ( currExp > maxExp - nextDiff )
break ; // overflow point reached
expPerLevel . push_back ( currExp + nextDiff ) ;
2023-10-19 16:19:09 +02: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 ;
}
std : : vector < JsonNode > CHeroHandler : : loadLegacyData ( )
{
2024-08-31 11:00:36 +00:00
size_t dataSize = VLC - > engineSettings ( ) - > getInteger ( EGameSettings : : TEXTS_HERO ) ;
2023-10-19 16:19:09 +02:00
objects . resize ( dataSize ) ;
std : : vector < JsonNode > h3Data ;
h3Data . reserve ( dataSize ) ;
CLegacyConfigParser specParser ( TextPath : : builtin ( " DATA/HEROSPEC.TXT " ) ) ;
CLegacyConfigParser bioParser ( TextPath : : builtin ( " DATA/HEROBIOS.TXT " ) ) ;
CLegacyConfigParser parser ( TextPath : : builtin ( " DATA/HOTRAITS.TXT " ) ) ;
parser . endLine ( ) ; //ignore header
parser . endLine ( ) ;
specParser . endLine ( ) ; //ignore header
specParser . endLine ( ) ;
for ( int i = 0 ; i < GameConstants : : HEROES_QUANTITY ; i + + )
{
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 ( ) ;
for ( int x = 0 ; x < 3 ; x + + )
{
JsonNode armySlot ;
armySlot [ " min " ] . Float ( ) = parser . readNumber ( ) ;
armySlot [ " max " ] . Float ( ) = parser . readNumber ( ) ;
armySlot [ " creature " ] . String ( ) = genRefName ( parser . readString ( ) ) ;
heroData [ " army " ] . Vector ( ) . push_back ( armySlot ) ;
}
parser . endLine ( ) ;
specParser . endLine ( ) ;
bioParser . endLine ( ) ;
h3Data . push_back ( heroData ) ;
}
return h3Data ;
}
void CHeroHandler : : loadObject ( std : : string scope , std : : string name , const JsonNode & data )
{
size_t index = objects . size ( ) ;
static const int specialFramesCount = 2 ; // reserved for 2 special frames
2024-05-16 22:05:51 +00:00
auto object = loadFromJson ( scope , data , name , index ) ;
2023-10-19 16:19:09 +02:00
object - > imageIndex = static_cast < si32 > ( index ) + specialFramesCount ;
objects . emplace_back ( object ) ;
registerObject ( scope , " hero " , name , object - > getIndex ( ) ) ;
2024-02-05 21:56:06 +02:00
for ( const auto & compatID : data [ " compatibilityIdentifiers " ] . Vector ( ) )
registerObject ( scope , " hero " , compatID . String ( ) , object - > getIndex ( ) ) ;
2023-10-19 16:19:09 +02:00
}
void CHeroHandler : : loadObject ( std : : string scope , std : : string name , const JsonNode & data , size_t index )
{
2024-05-16 22:05:51 +00:00
auto object = loadFromJson ( scope , data , name , index ) ;
2023-10-19 16:19:09 +02:00
object - > imageIndex = static_cast < si32 > ( index ) ;
assert ( objects [ index ] = = nullptr ) ; // ensure that this id was not loaded before
objects [ index ] = object ;
registerObject ( scope , " hero " , name , object - > getIndex ( ) ) ;
2024-02-05 21:56:06 +02:00
for ( const auto & compatID : data [ " compatibilityIdentifiers " ] . Vector ( ) )
registerObject ( scope , " hero " , compatID . String ( ) , object - > getIndex ( ) ) ;
2023-10-19 16:19:09 +02:00
}
2024-01-04 23:57:36 +02:00
ui32 CHeroHandler : : level ( TExpType experience ) const
2023-10-19 16:19:09 +02:00
{
return static_cast < ui32 > ( boost : : range : : upper_bound ( expPerLevel , experience ) - std : : begin ( expPerLevel ) ) ;
}
2024-01-04 23:57:36 +02:00
TExpType CHeroHandler : : reqExp ( ui32 level ) const
2023-10-19 16:19:09 +02:00
{
if ( ! level )
return 0 ;
if ( level < = expPerLevel . size ( ) )
{
return expPerLevel [ level - 1 ] ;
}
else
{
logGlobal - > warn ( " A hero has reached unsupported amount of experience " ) ;
return expPerLevel [ expPerLevel . size ( ) - 1 ] ;
}
}
2024-01-04 23:57:36 +02:00
ui32 CHeroHandler : : maxSupportedLevel ( ) const
{
return expPerLevel . size ( ) ;
}
2023-11-05 15:24:26 +02:00
std : : set < HeroTypeID > CHeroHandler : : getDefaultAllowed ( ) const
2023-10-19 16:19:09 +02:00
{
2023-11-05 15:24:26 +02:00
std : : set < HeroTypeID > result ;
2023-10-19 16:19:09 +02:00
2023-12-31 23:43:35 +02:00
for ( auto & hero : objects )
2023-11-05 15:24:26 +02:00
if ( hero & & ! hero - > special )
result . insert ( hero - > getId ( ) ) ;
2023-10-19 16:19:09 +02:00
2023-11-05 15:24:26 +02:00
return result ;
2023-10-19 16:19:09 +02:00
}
VCMI_LIB_NAMESPACE_END