2012-05-29 15:13:44 +02:00
# ifndef MWMECHANICS_SPELLSUCCESS_H
# define MWMECHANICS_SPELLSUCCESS_H
2017-06-30 16:27:18 +04:00
# include <components/esm/effectlist.hpp>
2014-02-23 20:11:05 +01:00
# include <components/esm/loadskil.hpp>
2018-08-27 13:38:53 +04:00
# include <components/esm/loadmgef.hpp>
2014-02-23 20:11:05 +01:00
2016-06-17 23:07:16 +09:00
# include "../mwworld/ptr.hpp"
2018-03-17 13:41:13 +04:00
# include "magiceffects.hpp"
2014-02-23 20:11:05 +01:00
namespace ESM
{
struct Spell ;
struct Ingredient ;
struct Potion ;
struct EffectList ;
}
2012-05-29 15:13:44 +02:00
namespace MWMechanics
{
2015-03-06 21:36:42 +13:00
struct EffectKey ;
2014-06-17 22:24:56 +02:00
class MagicEffects ;
2015-07-18 20:39:45 +02:00
class CreatureStats ;
2014-01-18 10:52:16 +01:00
ESM : : Skill : : SkillEnum spellSchoolToSkill ( int school ) ;
2012-05-29 18:33:01 +02:00
2017-06-30 16:27:18 +04:00
float calcEffectCost ( const ESM : : ENAMstruct & effect ) ;
2018-08-27 13:38:53 +04:00
float calcEffectCost ( const ESM : : ENAMstruct & effect , const ESM : : MagicEffect * magicEffect ) ;
2017-06-30 16:27:18 +04:00
2015-01-22 04:12:08 +01:00
bool isSummoningEffect ( int effectId ) ;
2012-05-29 18:33:01 +02:00
/**
2012-10-23 11:42:38 +02:00
* @ param spell spell to cast
2012-05-29 18:33:01 +02:00
* @ param actor calculate spell success chance for this actor ( depends on actor ' s skills )
2013-11-14 19:19:32 +01:00
* @ param effectiveSchool the spell ' s effective school ( relevant for skill progress ) will be written here
2014-07-31 01:14:44 +02:00
* @ param cap cap the result to 100 % ?
2017-11-28 23:10:07 +01:00
* @ param checkMagicka check magicka ?
2014-07-31 01:14:44 +02:00
* @ note actor can be an NPC or a creature
* @ return success chance from 0 to 100 ( in percent ) , if cap = false then chance above 100 may be returned .
2012-05-29 18:33:01 +02:00
*/
2018-10-09 10:21:12 +04:00
float getSpellSuccessChance ( const ESM : : Spell * spell , const MWWorld : : Ptr & actor , int * effectiveSchool = nullptr , bool cap = true , bool checkMagicka = false ) ;
float getSpellSuccessChance ( const std : : string & spellId , const MWWorld : : Ptr & actor , int * effectiveSchool = nullptr , bool cap = true , bool checkMagicka = false ) ;
2013-11-14 19:19:32 +01:00
2014-01-18 10:52:16 +01:00
int getSpellSchool ( const std : : string & spellId , const MWWorld : : Ptr & actor ) ;
int getSpellSchool ( const ESM : : Spell * spell , const MWWorld : : Ptr & actor ) ;
2013-11-14 19:19:32 +01:00
2014-08-24 20:36:31 +02:00
/// Get whether or not the given spell contributes to skill progress.
bool spellIncreasesSkill ( const ESM : : Spell * spell ) ;
bool spellIncreasesSkill ( const std : : string & spellId ) ;
2014-07-15 22:06:46 +02:00
/// Get the resistance attribute against an effect for a given actor. This will add together
/// ResistX and Weakness to X effects relevant against the given effect.
float getEffectResistanceAttribute ( short effectId , const MagicEffects * actorEffects ) ;
/// Get the effective resistance against an effect casted by the given actor in the given spell (optional).
2013-11-16 02:34:43 +01:00
/// @return >=100 for fully resisted. can also return negative value for damage amplification.
2014-06-17 22:24:56 +02:00
/// @param effects Override the actor's current magicEffects. Useful if there are effects currently
/// being applied (but not applied yet) that should also be considered.
float getEffectResistance ( short effectId , const MWWorld : : Ptr & actor , const MWWorld : : Ptr & caster ,
2018-10-09 10:21:12 +04:00
const ESM : : Spell * spell = nullptr , const MagicEffects * effects = nullptr ) ;
2013-11-16 02:34:43 +01:00
2014-10-06 21:56:41 -05:00
/// Get an effect multiplier for applying an effect cast by the given actor in the given spell (optional).
/// @return effect multiplier from 0 to 2. (100% net resistance to 100% net weakness)
/// @param effects Override the actor's current magicEffects. Useful if there are effects currently
/// being applied (but not applied yet) that should also be considered.
2014-06-17 22:24:56 +02:00
float getEffectMultiplier ( short effectId , const MWWorld : : Ptr & actor , const MWWorld : : Ptr & caster ,
2018-10-09 10:21:12 +04:00
const ESM : : Spell * spell = nullptr , const MagicEffects * effects = nullptr ) ;
2013-11-17 23:15:57 +01:00
2016-11-15 00:20:17 +09:00
bool checkEffectTarget ( int effectId , const MWWorld : : Ptr & target , const MWWorld : : Ptr & caster , bool castByPlayer ) ;
2015-01-07 04:28:56 +01:00
int getEffectiveEnchantmentCastCost ( float castCost , const MWWorld : : Ptr & actor ) ;
2017-06-30 16:27:18 +04:00
float calcSpellBaseSuccessChance ( const ESM : : Spell * spell , const MWWorld : : Ptr & actor , int * effectiveSchool ) ;
2015-01-07 04:28:56 +01:00
2016-12-05 20:12:13 +09:00
/// Apply a magic effect that is applied in tick intervals until its remaining time ends or it is removed
/// @return Was the effect a tickable effect with a magnitude?
bool effectTick ( CreatureStats & creatureStats , const MWWorld : : Ptr & actor , const MWMechanics : : EffectKey & effectKey , float magnitude ) ;
2015-07-18 20:39:45 +02:00
2017-02-20 19:58:00 +01:00
std : : string getSummonedCreature ( int effectId ) ;
2013-11-17 23:15:57 +01:00
class CastSpell
{
private :
2014-06-18 01:41:07 +02:00
MWWorld : : Ptr mCaster ; // May be empty
MWWorld : : Ptr mTarget ; // May be empty
2019-04-08 12:07:44 +04:00
void playSpellCastingEffects ( const std : : vector < ESM : : ENAMstruct > & effects ) ;
2013-11-28 17:31:17 +01:00
public :
2013-11-17 23:15:57 +01:00
bool mStack ;
std : : string mId ; // ID of spell, potion, item etc
std : : string mSourceName ; // Display name for spell, potion, etc
2015-06-01 21:41:13 +02:00
osg : : Vec3f mHitPosition ; // Used for spawning area orb
2014-10-09 01:39:35 +02:00
bool mAlwaysSucceed ; // Always succeed spells casted by NPCs/creatures regardless of their chance (default: false)
2016-12-19 10:15:19 +01:00
bool mFromProjectile ; // True if spell is cast by enchantment of some projectile (arrow, bolt or thrown weapon)
2018-08-03 12:01:31 +04:00
bool mManualSpell ; // True if spell is casted from script and ignores some checks (mana level, success chance, etc.)
2013-11-17 23:15:57 +01:00
public :
2018-08-03 12:01:31 +04:00
CastSpell ( const MWWorld : : Ptr & caster , const MWWorld : : Ptr & target , const bool fromProjectile = false , const bool manualSpell = false ) ;
2013-11-17 23:15:57 +01:00
bool cast ( const ESM : : Spell * spell ) ;
2014-06-18 01:41:07 +02:00
/// @note mCaster must be an actor
2016-02-22 19:37:19 +01:00
/// @param launchProjectile If set to false, "on target" effects are directly applied instead of being launched as projectile originating from the caster.
bool cast ( const MWWorld : : Ptr & item , bool launchProjectile = true ) ;
2014-06-18 01:41:07 +02:00
/// @note mCaster must be an NPC
2013-11-17 23:15:57 +01:00
bool cast ( const ESM : : Ingredient * ingredient ) ;
2014-06-18 01:41:07 +02:00
2013-11-17 23:15:57 +01:00
bool cast ( const ESM : : Potion * potion ) ;
/// @note Auto detects if spell, ingredient or potion
bool cast ( const std : : string & id ) ;
2019-04-08 12:07:44 +04:00
void playSpellCastingEffects ( const std : : string & spellid , bool enchantment ) ;
2016-07-16 22:47:33 +09:00
2018-06-28 16:58:51 +04:00
bool spellIncreasesSkill ( ) ;
2016-09-05 04:22:57 +09:00
/// Launch a bolt with the given effects.
2017-09-20 18:56:32 +02:00
void launchMagicBolt ( ) ;
2016-08-31 19:03:28 +09:00
2014-05-14 05:33:18 +02:00
/// @note \a target can be any type of object, not just actors.
2014-05-15 02:37:20 +02:00
/// @note \a caster can be any type of object, or even an empty object.
2013-11-17 23:15:57 +01:00
void inflict ( const MWWorld : : Ptr & target , const MWWorld : : Ptr & caster ,
2014-01-20 15:48:06 +01:00
const ESM : : EffectList & effects , ESM : : RangeType range , bool reflected = false , bool exploded = false ) ;
2013-11-17 23:15:57 +01:00
2014-05-15 02:37:20 +02:00
/// @note \a caster can be any type of object, or even an empty object.
2015-08-04 17:55:38 +02:00
/// @return was the target suitable for the effect?
bool applyInstantEffect ( const MWWorld : : Ptr & target , const MWWorld : : Ptr & caster , const MWMechanics : : EffectKey & effect , float magnitude ) ;
2013-11-17 23:15:57 +01:00
} ;
2018-03-17 13:41:13 +04:00
class ApplyLoopingParticlesVisitor : public EffectSourceVisitor
{
private :
MWWorld : : Ptr mActor ;
public :
ApplyLoopingParticlesVisitor ( const MWWorld : : Ptr & actor )
: mActor ( actor )
{
}
virtual void visit ( MWMechanics : : EffectKey key ,
const std : : string & /*sourceName*/ , const std : : string & /*sourceId*/ , int /*casterActorId*/ ,
float /*magnitude*/ , float /*remainingTime*/ = - 1 , float /*totalTime*/ = - 1 ) ;
} ;
2012-05-29 15:13:44 +02:00
}
# endif