2010-02-10 04:56:00 +02:00
# define VCMI_DLL
# include "HeroBonus.h"
2010-04-03 06:33:46 +03:00
# include <boost/foreach.hpp>
2010-05-02 21:20:26 +03:00
# include "VCMI_Lib.h"
# include "../hch/CSpellHandler.h"
# include <sstream>
# include "../hch/CCreatureHandler.h"
2010-07-12 13:20:25 +03:00
# include <boost/assign/list_of.hpp>
# include "CCreatureSet.h"
2010-07-13 08:25:40 +03:00
# include <boost/algorithm/string/trim.hpp>
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
# define FOREACH_CONST_PARENT(pname, source) TCNodes parents; getParents(parents, source); BOOST_FOREACH(const CBonusSystemNode *pname, parents)
# define FOREACH_PARENT(pname, source) TNodes parents; getParents(parents, source); BOOST_FOREACH(CBonusSystemNode *pname, parents)
2010-02-10 04:56:00 +02:00
2010-07-12 13:20:25 +03:00
# define BONUS_NAME(x) ( #x, Bonus::x )
DLL_EXPORT const std : : map < std : : string , int > bonusNameMap = boost : : assign : : map_list_of BONUS_LIST ;
# undef BONUS_NAME
2010-05-02 21:20:26 +03:00
int DLL_EXPORT BonusList : : totalValue ( ) const
2010-02-10 04:56:00 +02:00
{
2010-05-02 21:20:26 +03:00
int base = 0 ;
int percentToBase = 0 ;
int percentToAll = 0 ;
int additive = 0 ;
2010-08-05 12:11:38 +03:00
int indepMax = 0 ;
bool hasIndepMax = false ;
2010-02-10 04:56:00 +02:00
2010-05-02 21:20:26 +03:00
for ( const_iterator i = begin ( ) ; i ! = end ( ) ; i + + )
2010-02-10 04:56:00 +02:00
{
2010-05-02 21:20:26 +03:00
switch ( i - > valType )
{
case Bonus : : BASE_NUMBER :
base + = i - > val ;
break ;
case Bonus : : PERCENT_TO_ALL :
percentToAll + = i - > val ;
break ;
case Bonus : : PERCENT_TO_BASE :
percentToBase + = i - > val ;
break ;
case Bonus : : ADDITIVE_VALUE :
additive + = i - > val ;
2010-08-05 12:11:38 +03:00
break ;
case Bonus : : INDEPENDENT_MAX :
if ( ! indepMax )
{
indepMax = i - > val ;
hasIndepMax = true ;
}
else
{
amax ( indepMax , i - > val ) ;
}
2010-05-02 21:20:26 +03:00
break ;
}
2010-02-10 04:56:00 +02:00
}
2010-05-02 21:20:26 +03:00
int modifiedBase = base + ( base * percentToBase ) / 100 ;
modifiedBase + = additive ;
2010-08-05 12:11:38 +03:00
int valFirst = ( modifiedBase * ( 100 + percentToAll ) ) / 100 ;
if ( hasIndepMax )
return std : : max ( valFirst , indepMax ) ;
else
return valFirst ;
2010-02-10 04:56:00 +02:00
}
2010-05-02 21:20:26 +03:00
const DLL_EXPORT Bonus * BonusList : : getFirst ( const CSelector & selector ) const
2010-02-10 04:56:00 +02:00
{
2010-05-02 21:20:26 +03:00
for ( const_iterator i = begin ( ) ; i ! = end ( ) ; i + + )
if ( selector ( * i ) )
return & * i ;
return NULL ;
}
2010-02-10 04:56:00 +02:00
2010-05-02 21:20:26 +03:00
DLL_EXPORT Bonus * BonusList : : getFirst ( const CSelector & select )
{
for ( iterator i = begin ( ) ; i ! = end ( ) ; i + + )
if ( select ( * i ) )
return & * i ;
return NULL ;
2010-02-10 04:56:00 +02:00
}
2010-05-02 21:20:26 +03:00
void DLL_EXPORT BonusList : : getModifiersWDescr ( TModDescr & out ) const
2010-02-10 04:56:00 +02:00
{
2010-05-02 21:20:26 +03:00
for ( const_iterator i = begin ( ) ; i ! = end ( ) ; i + + )
out . push_back ( std : : make_pair ( i - > val , i - > Description ( ) ) ) ;
}
void DLL_EXPORT BonusList : : getBonuses ( BonusList & out , const CSelector & selector , const CBonusSystemNode * source /*= NULL*/ ) const
{
for ( const_iterator i = begin ( ) ; i ! = end ( ) ; i + + )
if ( selector ( * i ) & & i - > effectRange = = Bonus : : NO_LIMIT )
out . push_back ( * i ) ;
}
void DLL_EXPORT BonusList : : getBonuses ( BonusList & out , const CSelector & selector , const CSelector & limit , const CBonusSystemNode * source /*= NULL*/ ) const
{
for ( const_iterator i = begin ( ) ; i ! = end ( ) ; i + + )
if ( selector ( * i ) & & ( ! limit | | limit ( * i ) ) )
out . push_back ( * i ) ;
}
2010-07-12 13:20:25 +03:00
void BonusList : : limit ( const CBonusSystemNode & node )
{
2010-07-31 03:26:34 +03:00
limit_start :
2010-07-17 04:19:16 +03:00
for ( iterator i = begin ( ) ; i ! = end ( ) ; i + + )
2010-07-12 13:20:25 +03:00
{
if ( i - > limiter & & i - > limiter - > limit ( * i , node ) )
{
2010-07-17 04:19:16 +03:00
iterator toErase = i ;
2010-07-31 03:26:34 +03:00
if ( i ! = begin ( ) )
{
i - - ;
erase ( toErase ) ;
}
else
{
erase ( toErase ) ;
goto limit_start ;
}
2010-07-12 13:20:25 +03:00
}
}
}
2010-05-02 21:20:26 +03:00
int CBonusSystemNode : : valOfBonuses ( Bonus : : BonusType type , int subtype /*= -1*/ ) const
{
CSelector s = Selector : : type ( type ) ;
if ( subtype ! = - 1 )
s = s & & Selector : : subtype ( subtype ) ;
return valOfBonuses ( s ) ;
}
int CBonusSystemNode : : valOfBonuses ( Bonus : : BonusType type , const CSelector & selector ) const
{
return valOfBonuses ( Selector : : type ( type ) & & selector ) ;
}
int CBonusSystemNode : : valOfBonuses ( const CSelector & selector , const CBonusSystemNode * root /* = NULL*/ ) const
{
BonusList hlp ;
getBonuses ( hlp , selector , root ) ;
return hlp . totalValue ( ) ;
}
bool CBonusSystemNode : : hasBonus ( const CSelector & selector , const CBonusSystemNode * root /* = NULL*/ ) const
{
return getBonuses ( selector ) . size ( ) > 0 ;
}
bool CBonusSystemNode : : hasBonusOfType ( Bonus : : BonusType type , int subtype /*= -1*/ ) const
{
CSelector s = Selector : : type ( type ) ;
if ( subtype ! = - 1 )
s = s & & Selector : : subtype ( subtype ) ;
return hasBonus ( s ) ;
}
Bonus * CBonusSystemNode : : getBonus ( const CSelector & selector )
{
Bonus * ret = bonuses . getFirst ( selector ) ;
if ( ret )
return ret ;
FOREACH_PARENT ( p , this )
if ( ret = p - > getBonus ( selector ) )
return ret ;
2010-02-10 04:56:00 +02:00
return NULL ;
}
2010-05-02 21:20:26 +03:00
void CBonusSystemNode : : getModifiersWDescr ( TModDescr & out , Bonus : : BonusType type , int subtype /*= -1 */ ) const
2010-02-10 04:56:00 +02:00
{
2010-05-02 21:20:26 +03:00
getModifiersWDescr ( out , Selector : : typeSybtype ( type , subtype ) ) ;
}
2010-02-10 04:56:00 +02:00
2010-05-02 21:20:26 +03:00
void CBonusSystemNode : : getModifiersWDescr ( TModDescr & out , const CSelector & selector , const CBonusSystemNode * root /*= NULL*/ ) const
{
getBonuses ( selector ) . getModifiersWDescr ( out ) ;
}
int CBonusSystemNode : : getBonusesCount ( int from , int id ) const
{
return getBonusesCount ( Selector : : source ( from , id ) ) ;
}
int CBonusSystemNode : : getBonusesCount ( const CSelector & selector , const CBonusSystemNode * root /* = NULL*/ ) const
{
return getBonuses ( selector , root ) . size ( ) ;
}
void CBonusSystemNode : : getParents ( TCNodes & out , const CBonusSystemNode * root ) const /*retreives list of parent nodes (nodes to inherit bonuses from) */
{
return ;
}
void CBonusSystemNode : : getParents ( TNodes & out , const CBonusSystemNode * root /*= NULL*/ )
{
//de-constify above
TCNodes hlp ;
getParents ( hlp , root ) ;
BOOST_FOREACH ( const CBonusSystemNode * pname , hlp )
out . insert ( const_cast < CBonusSystemNode * > ( pname ) ) ;
}
void CBonusSystemNode : : getBonuses ( BonusList & out , const CSelector & selector , const CBonusSystemNode * root /*= NULL*/ ) const
{
bonuses . getBonuses ( out , selector ) ;
FOREACH_CONST_PARENT ( p , root ? root : this )
p - > getBonuses ( out , selector , root ? root : this ) ;
2010-07-12 13:20:25 +03:00
if ( ! root )
out . limit ( * this ) ;
2010-04-03 06:33:46 +03:00
}
2010-05-02 21:20:26 +03:00
BonusList CBonusSystemNode : : getBonuses ( const CSelector & selector , const CBonusSystemNode * root /*= NULL*/ ) const
2010-04-03 06:33:46 +03:00
{
2010-05-02 21:20:26 +03:00
BonusList ret ;
getBonuses ( ret , selector , root ) ;
return ret ;
}
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
void CBonusSystemNode : : getBonuses ( BonusList & out , const CSelector & selector , const CSelector & limit , const CBonusSystemNode * root /*= NULL*/ ) const
{
bonuses . getBonuses ( out , selector , limit ) ;
FOREACH_CONST_PARENT ( p , root ? root : this )
p - > getBonuses ( out , selector , limit , root ? root : this ) ;
2010-07-12 13:20:25 +03:00
if ( ! root )
out . limit ( * this ) ;
2010-05-02 21:20:26 +03:00
}
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
BonusList CBonusSystemNode : : getBonuses ( const CSelector & selector , const CSelector & limit , const CBonusSystemNode * root /*= NULL*/ ) const
{
BonusList ret ;
getBonuses ( ret , selector , limit , root ) ;
2010-04-03 06:33:46 +03:00
return ret ;
}
2010-05-02 21:20:26 +03:00
bool CBonusSystemNode : : hasBonusFrom ( ui8 source , ui32 sourceID ) const
{
return hasBonus ( Selector : : source ( source , sourceID ) ) ;
}
int CBonusSystemNode : : MoraleVal ( ) const
2010-04-03 06:33:46 +03:00
{
2010-05-02 21:20:26 +03:00
if ( hasBonusOfType ( Bonus : : NON_LIVING ) | | hasBonusOfType ( Bonus : : UNDEAD ) | |
hasBonusOfType ( Bonus : : NO_MORALE ) | | hasBonusOfType ( Bonus : : SIEGE_WEAPON ) )
return 0 ;
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
int ret = valOfBonuses ( Selector : : type ( Bonus : : MORALE ) ) ;
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
if ( hasBonusOfType ( Bonus : : SELF_MORALE ) ) //eg. minotaur
amax ( ret , + 1 ) ;
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
return abetw ( ret , - 3 , + 3 ) ;
2010-04-03 06:33:46 +03:00
}
2010-05-02 21:20:26 +03:00
int CBonusSystemNode : : LuckVal ( ) const
2010-04-03 06:33:46 +03:00
{
2010-05-02 21:20:26 +03:00
if ( hasBonusOfType ( Bonus : : NO_LUCK ) )
return 0 ;
int ret = valOfBonuses ( Selector : : type ( Bonus : : LUCK ) ) ;
if ( hasBonusOfType ( Bonus : : SELF_LUCK ) ) //eg. halfling
amax ( ret , + 1 ) ;
return abetw ( ret , - 3 , + 3 ) ;
2010-04-03 06:33:46 +03:00
}
2010-05-02 21:20:26 +03:00
si32 CBonusSystemNode : : Attack ( ) const
2010-04-03 06:33:46 +03:00
{
2010-05-02 21:20:26 +03:00
si32 ret = valOfBonuses ( Bonus : : PRIMARY_SKILL , PrimarySkill : : ATTACK ) ;
if ( int frenzyPower = valOfBonuses ( Bonus : : IN_FRENZY ) ) //frenzy for attacker
{
ret + = frenzyPower * Defense ( false ) ;
}
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
return ret ;
2010-04-03 06:33:46 +03:00
}
2010-05-02 21:20:26 +03:00
si32 CBonusSystemNode : : Defense ( bool withFrenzy /*= true*/ ) const
2010-04-03 06:33:46 +03:00
{
2010-05-02 21:20:26 +03:00
si32 ret = valOfBonuses ( Bonus : : PRIMARY_SKILL , PrimarySkill : : DEFENSE ) ;
2010-04-03 06:33:46 +03:00
2010-05-02 21:20:26 +03:00
if ( withFrenzy & & hasBonusOfType ( Bonus : : IN_FRENZY ) ) //frenzy for defender
{
return 0 ;
}
2010-04-03 06:33:46 +03:00
return ret ;
}
2010-05-02 21:20:26 +03:00
ui16 CBonusSystemNode : : MaxHealth ( ) const
2010-04-03 06:33:46 +03:00
{
2010-05-02 21:20:26 +03:00
return valOfBonuses ( Bonus : : STACK_HEALTH ) ;
2010-04-03 06:33:46 +03:00
}
2010-07-12 13:20:25 +03:00
CBonusSystemNode : : CBonusSystemNode ( )
{
nodeType = UNKNOWN ;
}
CBonusSystemNode : : ~ CBonusSystemNode ( )
{
}
2010-05-02 21:20:26 +03:00
int NBonus : : valOf ( const CBonusSystemNode * obj , Bonus : : BonusType type , int subtype /*= -1*/ )
2010-04-03 06:33:46 +03:00
{
if ( obj )
return obj - > valOfBonuses ( type , subtype ) ;
return 0 ;
}
2010-05-02 21:20:26 +03:00
bool NBonus : : hasOfType ( const CBonusSystemNode * obj , Bonus : : BonusType type , int subtype /*= -1*/ )
2010-04-03 06:33:46 +03:00
{
if ( obj )
return obj - > hasBonusOfType ( type , subtype ) ;
return false ;
}
2010-05-02 21:20:26 +03:00
void NBonus : : getModifiersWDescr ( const CBonusSystemNode * obj , TModDescr & out , Bonus : : BonusType type , int subtype /*= -1 */ )
2010-04-03 06:33:46 +03:00
{
if ( obj )
return obj - > getModifiersWDescr ( out , type , subtype ) ;
}
int NBonus : : getCount ( const CBonusSystemNode * obj , int from , int id )
{
if ( obj )
return obj - > getBonusesCount ( from , id ) ;
return 0 ;
2010-05-02 21:20:26 +03:00
}
const CSpell * Bonus : : sourceSpell ( ) const
{
if ( source = = SPELL_EFFECT )
return & VLC - > spellh - > spells [ id ] ;
return NULL ;
}
std : : string Bonus : : Description ( ) const
{
if ( description . size ( ) )
return description ;
std : : ostringstream str ;
if ( val < 0 )
str < < ' - ' ;
else if ( val > 0 )
str < < ' + ' ;
str < < val < < " " ;
switch ( source )
{
case CREATURE_ABILITY :
str < < VLC - > creh - > creatures [ id ] - > namePl ;
break ;
}
return str . str ( ) ;
}
2010-07-13 08:25:40 +03:00
Bonus : : Bonus ( ui8 Dur , ui8 Type , ui8 Src , si32 Val , ui32 ID , std : : string Desc , si32 Subtype /*=-1*/ )
: duration ( Dur ) , type ( Type ) , subtype ( Subtype ) , source ( Src ) , val ( Val ) , id ( ID ) , description ( Desc )
{
additionalInfo = - 1 ;
turnsRemain = 0 ;
valType = ADDITIVE_VALUE ;
effectRange = NO_LIMIT ;
limiter = NULL ;
boost : : algorithm : : trim ( description ) ;
}
Bonus : : Bonus ( ui8 Dur , ui8 Type , ui8 Src , si32 Val , ui32 ID , si32 Subtype /*=-1*/ , ui8 ValType /*= ADDITIVE_VALUE*/ )
: duration ( Dur ) , type ( Type ) , subtype ( Subtype ) , source ( Src ) , val ( Val ) , id ( ID ) , valType ( ValType )
{
additionalInfo = - 1 ;
turnsRemain = 0 ;
effectRange = NO_LIMIT ;
limiter = NULL ;
}
Bonus : : Bonus ( )
{
subtype = - 1 ;
additionalInfo = - 1 ;
turnsRemain = 0 ;
valType = ADDITIVE_VALUE ;
effectRange = NO_LIMIT ;
limiter = NULL ;
}
2010-05-02 21:20:26 +03:00
CSelector DLL_EXPORT operator & & ( const CSelector & first , const CSelector & second )
{
return CSelectorsConjunction ( first , second ) ;
}
namespace Selector
{
DLL_EXPORT CSelectFieldEqual < TBonusType > type ( & Bonus : : type , 0 ) ;
DLL_EXPORT CSelectFieldEqual < TBonusSubtype > subtype ( & Bonus : : subtype , 0 ) ;
DLL_EXPORT CSelectFieldEqual < si32 > info ( & Bonus : : additionalInfo , 0 ) ;
DLL_EXPORT CSelectFieldEqual < ui8 > sourceType ( & Bonus : : source , 0 ) ;
DLL_EXPORT CSelectFieldEqual < ui8 > effectRange ( & Bonus : : effectRange , Bonus : : NO_LIMIT ) ;
DLL_EXPORT CWillLastTurns turns ; ;
CSelector DLL_EXPORT typeSybtype ( TBonusType Type , TBonusSubtype Subtype )
{
return type ( Type ) & & subtype ( Subtype ) ;
}
CSelector DLL_EXPORT typeSybtypeInfo ( TBonusType type , TBonusSubtype subtype , si32 info )
{
return CSelectFieldEqual < TBonusType > ( & Bonus : : type , type ) & & CSelectFieldEqual < TBonusSubtype > ( & Bonus : : subtype , subtype ) & & CSelectFieldEqual < si32 > ( & Bonus : : additionalInfo , info ) ;
}
CSelector DLL_EXPORT source ( ui8 source , ui32 sourceID )
{
return CSelectFieldEqual < ui8 > ( & Bonus : : source , source ) & & CSelectFieldEqual < ui32 > ( & Bonus : : id , sourceID ) ;
}
2010-05-14 05:18:37 +03:00
bool DLL_EXPORT matchesType ( const CSelector & sel , TBonusType type )
2010-05-02 21:20:26 +03:00
{
Bonus dummy ;
dummy . type = type ;
return sel ( dummy ) ;
}
2010-05-14 05:18:37 +03:00
bool DLL_EXPORT matchesTypeSubtype ( const CSelector & sel , TBonusType type , TBonusSubtype subtype )
{
Bonus dummy ;
dummy . type = type ;
dummy . subtype = subtype ;
return sel ( dummy ) ;
}
2010-07-12 13:20:25 +03:00
}
DLL_EXPORT std : : ostream & operator < < ( std : : ostream & out , const BonusList & bonusList )
{
int i = 0 ;
BOOST_FOREACH ( const Bonus & b , bonusList )
{
out < < " Bonus " < < i + + < < " \n " < < b < < std : : endl ;
}
return out ;
}
DLL_EXPORT std : : ostream & operator < < ( std : : ostream & out , const Bonus & bonus )
{
for ( std : : map < std : : string , int > : : const_iterator i = bonusNameMap . begin ( ) ; i ! = bonusNameMap . end ( ) ; i + + )
if ( i - > second = = bonus . type )
out < < " \t Type: " < < i - > first < < " \t " ;
# define printField(field) out << "\t" #field ": " << (int)bonus.field << "\n"
printField ( val ) ;
printField ( subtype ) ;
printField ( duration ) ;
printField ( source ) ;
printField ( id ) ;
printField ( additionalInfo ) ;
printField ( turnsRemain ) ;
printField ( valType ) ;
printField ( effectRange ) ;
# undef printField
return out ;
}
ILimiter : : ~ ILimiter ( )
{
}
bool ILimiter : : limit ( const Bonus & b , const CBonusSystemNode & node ) const /*return true to drop the bonus */
{
return false ;
}
bool CCreatureTypeLimiter : : limit ( const Bonus & b , const CBonusSystemNode & node ) const
{
if ( node . nodeType ! = CBonusSystemNode : : STACK )
return true ;
const CCreature * c = ( static_cast < const CStackInstance * > ( & node ) ) - > type ;
return c ! = creature & & ( ! includeUpgrades | | ! creature - > isMyUpgrade ( c ) ) ; //drop bonus if it's not our creature and (we dont check upgrades or its not our upgrade)
}
CCreatureTypeLimiter : : CCreatureTypeLimiter ( const CCreature & Creature , ui8 IncludeUpgrades /*= true*/ )
: creature ( & Creature ) , includeUpgrades ( IncludeUpgrades )
{
}
CCreatureTypeLimiter : : CCreatureTypeLimiter ( )
{
creature = NULL ;
includeUpgrades = false ;
2010-08-06 13:46:40 +03:00
}
HasAnotherBonusLimiter : : HasAnotherBonusLimiter ( TBonusType bonus )
: type ( bonus ) , subtype ( 0 ) , isSubtypeRelevant ( false )
{
}
HasAnotherBonusLimiter : : HasAnotherBonusLimiter ( TBonusType bonus , TBonusSubtype _subtype )
: type ( bonus ) , subtype ( _subtype ) , isSubtypeRelevant ( true )
{
}
bool HasAnotherBonusLimiter : : limit ( const Bonus & b , const CBonusSystemNode & node ) const
{
if ( isSubtypeRelevant )
{
return ! node . hasBonusOfType ( static_cast < Bonus : : BonusType > ( type ) , subtype ) ;
}
else
{
return ! node . hasBonusOfType ( static_cast < Bonus : : BonusType > ( type ) ) ;
}
}