2015-02-02 10:25:26 +02:00
/*
* CSpellHandler . h , 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
*
*/
2015-02-26 19:59:18 +02:00
# pragma once
2015-02-02 10:25:26 +02:00
2015-02-26 19:59:18 +02:00
# include "../IHandlerBase.h"
# include "../ConstTransitivePtr.h"
# include "../int3.h"
# include "../GameConstants.h"
# include "../BattleHex.h"
# include "../HeroBonus.h"
class CGObjectInstance ;
2015-02-02 10:25:26 +02:00
class CSpell ;
class ISpellMechanics ;
class CLegacyConfigParser ;
class CGHeroInstance ;
class CStack ;
class CBattleInfoCallback ;
class BattleInfo ;
struct CPackForClient ;
struct BattleSpellCast ;
class CGameInfoCallback ;
class CRandomGenerator ;
class CMap ;
struct SpellSchoolInfo
{
ESpellSchool id ; //backlink
Bonus : : BonusType damagePremyBonus ;
2015-02-26 19:59:18 +02:00
Bonus : : BonusType immunityBonus ;
2015-02-02 10:25:26 +02:00
std : : string jsonName ;
SecondarySkill : : ESecondarySkill skill ;
2015-02-26 19:59:18 +02:00
Bonus : : BonusType knoledgeBonus ;
2015-02-02 10:25:26 +02:00
} ;
///callback to be provided by server
class DLL_LINKAGE SpellCastEnvironment
{
public :
virtual ~ SpellCastEnvironment ( ) { } ;
virtual void sendAndApply ( CPackForClient * info ) const = 0 ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
virtual CRandomGenerator & getRandomGenerator ( ) const = 0 ;
virtual void complain ( const std : : string & problem ) const = 0 ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
virtual const CMap * getMap ( ) const = 0 ;
2015-02-26 19:59:18 +02:00
virtual const CGameInfoCallback * getCb ( ) const = 0 ;
2015-02-02 10:25:26 +02:00
virtual bool moveHero ( ObjectInstanceID hid , int3 dst , ui8 teleporting , PlayerColor asker = PlayerColor : : NEUTRAL ) const = 0 ; //TODO: remove
} ;
///helper struct
struct DLL_LINKAGE BattleSpellCastParameters
{
public :
BattleSpellCastParameters ( const BattleInfo * cb ) ;
int spellLvl ;
BattleHex destination ;
ui8 casterSide ;
PlayerColor casterColor ;
const CGHeroInstance * caster ;
const CGHeroInstance * secHero ;
int usedSpellPower ;
ECastingMode : : ECastingMode mode ;
const CStack * casterStack ;
2015-02-26 19:59:18 +02:00
const CStack * selectedStack ;
const BattleInfo * cb ;
2015-02-02 10:25:26 +02:00
} ;
struct DLL_LINKAGE AdventureSpellCastParameters
{
const CGHeroInstance * caster ;
2015-02-26 19:59:18 +02:00
int3 pos ;
2015-02-02 10:25:26 +02:00
} ;
enum class VerticalPosition : ui8 { TOP , CENTER , BOTTOM } ;
class DLL_LINKAGE CSpell
{
public :
struct ProjectileInfo
{
///in radians. Only positive value. Negative angle is handled by vertical flip
2015-02-26 19:59:18 +02:00
double minimumAngle ;
2015-02-02 10:25:26 +02:00
///resource name
std : : string resourceName ;
2015-02-26 19:59:18 +02:00
template < typename Handler > void serialize ( Handler & h , const int version )
2015-02-02 10:25:26 +02:00
{
2015-02-26 19:59:18 +02:00
h & minimumAngle & resourceName ;
}
2015-02-02 10:25:26 +02:00
} ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
struct AnimationItem
{
std : : string resourceName ;
VerticalPosition verticalPosition ;
2015-02-26 19:59:18 +02:00
template < typename Handler > void serialize ( Handler & h , const int version )
2015-02-02 10:25:26 +02:00
{
2015-02-26 19:59:18 +02:00
h & resourceName & verticalPosition ;
}
2015-02-02 10:25:26 +02:00
} ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
typedef AnimationItem TAnimation ;
2015-02-26 19:59:18 +02:00
typedef std : : vector < TAnimation > TAnimationQueue ;
2015-02-02 10:25:26 +02:00
struct DLL_LINKAGE AnimationInfo
{
AnimationInfo ( ) ;
~ AnimationInfo ( ) ;
2015-02-26 19:59:18 +02:00
///displayed on all affected targets.
2015-02-02 10:25:26 +02:00
TAnimationQueue affect ;
///displayed on caster.
TAnimationQueue cast ;
///displayed on target hex. If spell was casted with no target selection displayed on entire battlefield (f.e. ARMAGEDDON)
TAnimationQueue hit ;
///displayed "between" caster and (first) target. Ignored if spell was casted with no target selection.
///use selectProjectile to access
std : : vector < ProjectileInfo > projectile ;
2015-02-26 19:59:18 +02:00
template < typename Handler > void serialize ( Handler & h , const int version )
2015-02-02 10:25:26 +02:00
{
h & projectile & hit & cast ;
}
std : : string selectProjectile ( const double angle ) const ;
} animationInfo ;
public :
struct LevelInfo
{
std : : string description ; //descriptions of spell for skill level
si32 cost ;
si32 power ;
si32 AIValue ;
bool smartTarget ;
bool clearTarget ;
bool clearAffected ;
std : : string range ;
std : : vector < Bonus > effects ;
LevelInfo ( ) ;
~ LevelInfo ( ) ;
template < typename Handler > void serialize ( Handler & h , const int version )
{
h & description & cost & power & AIValue & smartTarget & range & effects ;
h & clearTarget & clearAffected ;
}
} ;
/** \brief Low level accessor. Don`t use it if absolutely necessary
*
* \ param level . spell school level
* \ return Spell level info structure
*
*/
const CSpell : : LevelInfo & getLevelInfo ( const int level ) const ;
public :
enum ETargetType { NO_TARGET , CREATURE , OBSTACLE , LOCATION } ;
enum ESpellPositiveness { NEGATIVE = - 1 , NEUTRAL = 0 , POSITIVE = 1 } ;
struct TargetInfo
{
ETargetType type ;
bool smart ;
bool massive ;
bool onlyAlive ;
///no immunity on primary target (mostly spell-like attack)
bool alwaysHitDirectly ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
bool clearTarget ;
bool clearAffected ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
TargetInfo ( const CSpell * spell , const int level ) ;
TargetInfo ( const CSpell * spell , const int level , ECastingMode : : ECastingMode mode ) ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
private :
void init ( const CSpell * spell , const int level ) ;
} ;
SpellID id ;
std : : string identifier ; //???
std : : string name ;
si32 level ;
std : : map < ESpellSchool , bool > school ; //todo: use this instead of separate boolean fields
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
si32 power ; //spell's power
std : : map < TFaction , si32 > probabilities ; //% chance to gain for castles
bool combatSpell ; //is this spell combat (true) or adventure (false)
bool creatureAbility ; //if true, only creatures can use this spell
si8 positiveness ; //1 if spell is positive for influenced stacks, 0 if it is indifferent, -1 if it's negative
std : : vector < SpellID > counteredSpells ; //spells that are removed when effect of this spell is placed on creature (for bless-curse, haste-slow, and similar pairs)
CSpell ( ) ;
~ CSpell ( ) ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
bool isCastableBy ( const IBonusBearer * caster , bool hasSpellBook , const std : : set < SpellID > & spellBook ) const ;
2015-02-26 19:59:18 +02:00
std : : vector < BattleHex > rangeInHexes ( BattleHex centralHex , ui8 schoolLvl , ui8 side , bool * outDroppedHexes = nullptr ) const ; //convert range to specific hexes; last optional out parameter is set to true, if spell would cover unavailable hexes (that are not included in ret)
2015-02-02 10:25:26 +02:00
ETargetType getTargetType ( ) const ; //deprecated
CSpell : : TargetInfo getTargetInfo ( const int level ) const ;
bool isCombatSpell ( ) const ;
bool isAdventureSpell ( ) const ;
bool isCreatureAbility ( ) const ;
bool isPositive ( ) const ;
bool isNegative ( ) const ;
bool isNeutral ( ) const ;
bool isDamageSpell ( ) const ;
bool isHealingSpell ( ) const ;
2015-02-26 19:59:18 +02:00
bool isRisingSpell ( ) const ;
2015-02-02 10:25:26 +02:00
bool isOffensiveSpell ( ) const ;
bool isSpecialSpell ( ) const ;
bool hasEffects ( ) const ;
void getEffects ( std : : vector < Bonus > & lst , const int level ) const ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
///checks for creature immunity / anything that prevent casting *at given hex* - doesn't take into account general problems such as not having spellbook or mana points etc.
ESpellCastProblem : : ESpellCastProblem isImmuneAt ( const CBattleInfoCallback * cb , const CGHeroInstance * caster , ECastingMode : : ECastingMode mode , BattleHex destination ) const ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
//internal, for use only by Mechanics classes
ESpellCastProblem : : ESpellCastProblem isImmuneBy ( const IBonusBearer * obj ) const ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
//internal, for use only by Mechanics classes. applying secondary skills
ui32 calculateBonus ( ui32 baseDamage , const CGHeroInstance * caster , const CStack * affectedCreature ) const ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
///calculate spell damage on stack taking caster`s secondary skills and affectedCreature`s bonuses into account
ui32 calculateDamage ( const CGHeroInstance * caster , const CStack * affectedCreature , int spellSchoolLevel , int usedSpellPower ) const ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
///selects from allStacks actually affected stacks
std : : set < const CStack * > getAffectedStacks ( const CBattleInfoCallback * cb , ECastingMode : : ECastingMode mode , PlayerColor casterColor , int spellLvl , BattleHex destination , const CGHeroInstance * caster = nullptr ) const ;
si32 getCost ( const int skillLevel ) const ;
/**
* Returns spell level power , base power ignored
*/
si32 getPower ( const int skillLevel ) const ;
si32 getProbability ( const TFaction factionId ) const ;
/**
* Calls cb for each school this spell belongs to
*
* Set stop to true to abort looping
2015-02-26 19:59:18 +02:00
*/
2015-02-02 10:25:26 +02:00
void forEachSchool ( const std : : function < void ( const SpellSchoolInfo & , bool & ) > & cb ) const ;
/**
* Returns resource name of icon for SPELL_IMMUNITY bonus
*/
const std : : string & getIconImmune ( ) const ;
const std : : string & getCastSound ( ) const ;
template < typename Handler > void serialize ( Handler & h , const int version )
{
h & identifier & id & name & level & power
& probabilities & attributes & combatSpell & creatureAbility & positiveness & counteredSpells ;
h & isRising & isDamage & isOffensive ;
h & targetType ;
h & immunities & limiters & absoluteImmunities & absoluteLimiters ;
h & iconImmune ;
h & defaultProbability ;
h & isSpecial ;
h & castSound & iconBook & iconEffect & iconScenarioBonus & iconScroll ;
2015-02-26 19:59:18 +02:00
h & levels ;
h & school ;
2015-02-02 10:25:26 +02:00
h & animationInfo ;
if ( ! h . saving )
setup ( ) ;
}
friend class CSpellHandler ;
friend class Graphics ;
2015-03-18 15:39:07 +02:00
public :
///internal interface (for callbacks)
///Checks general but spell-specific problems for all casting modes. Use only during battle.
ESpellCastProblem : : ESpellCastProblem canBeCasted ( const CBattleInfoCallback * cb ) const ;
///checks for creature immunity / anything that prevent casting *at given target* - doesn't take into account general problems such as not having spellbook or mana points etc.
ESpellCastProblem : : ESpellCastProblem isImmuneByStack ( const CGHeroInstance * caster , const CStack * obj ) const ;
2015-02-02 10:25:26 +02:00
public :
///Server logic. Has write access to GameState via packets.
///May be executed on client side by (future) non-cheat-proof scripts.
2015-02-26 19:59:18 +02:00
bool adventureCast ( const SpellCastEnvironment * env , AdventureSpellCastParameters & parameters ) const ;
void battleCast ( const SpellCastEnvironment * env , BattleSpellCastParameters & parameters ) const ;
public :
2015-02-02 10:25:26 +02:00
///Client-server logic. Has direct write access to GameState.
///Shall be called (only) when applying packets on BOTH SIDES
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
///implementation of BattleSpellCast applying
void applyBattle ( BattleInfo * battle , const BattleSpellCast * packet ) const ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
private :
void setIsOffensive ( const bool val ) ;
void setIsRising ( const bool val ) ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
//call this after load or deserialization. cant be done in constructor.
void setup ( ) ;
void setupMechanics ( ) ;
private :
si32 defaultProbability ;
bool isRising ;
bool isDamage ;
bool isOffensive ;
bool isSpecial ;
std : : string attributes ; //reference only attributes //todo: remove or include in configuration format, currently unused
ETargetType targetType ;
std : : vector < Bonus : : BonusType > immunities ; //any of these grants immunity
std : : vector < Bonus : : BonusType > absoluteImmunities ; //any of these grants immunity, can't be negated
std : : vector < Bonus : : BonusType > limiters ; //all of them are required to be affected
std : : vector < Bonus : : BonusType > absoluteLimiters ; //all of them are required to be affected, can't be negated
///graphics related stuff
std : : string iconImmune ;
std : : string iconBook ;
std : : string iconEffect ;
std : : string iconScenarioBonus ;
std : : string iconScroll ;
///sound related stuff
std : : string castSound ;
std : : vector < LevelInfo > levels ;
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
ISpellMechanics * mechanics ; //(!) do not serialize
} ;
bool DLL_LINKAGE isInScreenRange ( const int3 & center , const int3 & pos ) ; //for spells like Dimension Door
class DLL_LINKAGE CSpellHandler : public CHandlerBase < SpellID , CSpell >
{
public :
CSpellHandler ( ) ;
virtual ~ CSpellHandler ( ) ;
///IHandler base
std : : vector < JsonNode > loadLegacyData ( size_t dataSize ) override ;
void afterLoadFinalization ( ) override ;
void beforeValidate ( JsonNode & object ) override ;
/**
* Gets a list of default allowed spells . OH3 spells are all allowed by default .
*
* @ return a list of allowed spells , the index is the spell id and the value either 0 for not allowed or 1 for allowed
*/
std : : vector < bool > getDefaultAllowed ( ) const override ;
const std : : string getTypeName ( ) const override ;
template < typename Handler > void serialize ( Handler & h , const int version )
{
h & objects ;
}
2015-02-26 19:59:18 +02:00
2015-02-02 10:25:26 +02:00
protected :
CSpell * loadFromJson ( const JsonNode & json ) override ;
} ;