2012-05-29 15:13:44 +02:00
|
|
|
#ifndef MWMECHANICS_SPELLSUCCESS_H
|
|
|
|
#define MWMECHANICS_SPELLSUCCESS_H
|
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
#include <cfloat>
|
|
|
|
|
2012-07-03 12:30:50 +02:00
|
|
|
#include "../mwbase/world.hpp"
|
2012-05-29 15:13:44 +02:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-07-03 12:30:50 +02:00
|
|
|
|
|
|
|
#include "../mwworld/ptr.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
2012-05-29 18:33:01 +02:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
2012-05-29 15:13:44 +02:00
|
|
|
|
2012-10-01 19:17:04 +04:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2012-08-27 20:44:14 +02:00
|
|
|
|
2012-05-29 15:13:44 +02:00
|
|
|
#include "npcstats.hpp"
|
|
|
|
|
|
|
|
namespace MWMechanics
|
|
|
|
{
|
2012-05-29 18:33:01 +02:00
|
|
|
inline int spellSchoolToSkill(int school)
|
2012-05-29 15:13:44 +02:00
|
|
|
{
|
|
|
|
std::map<int, int> schoolSkillMap; // maps spell school to skill id
|
|
|
|
schoolSkillMap[0] = 11; // alteration
|
|
|
|
schoolSkillMap[1] = 13; // conjuration
|
|
|
|
schoolSkillMap[3] = 12; // illusion
|
|
|
|
schoolSkillMap[2] = 10; // destruction
|
|
|
|
schoolSkillMap[4] = 14; // mysticism
|
|
|
|
schoolSkillMap[5] = 15; // restoration
|
2012-05-29 18:33:01 +02:00
|
|
|
assert(schoolSkillMap.find(school) != schoolSkillMap.end());
|
|
|
|
return schoolSkillMap[school];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
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
|
2012-05-29 18:33:01 +02:00
|
|
|
* @attention actor has to be an NPC and not a creature!
|
|
|
|
* @return success chance from 0 to 100 (in percent)
|
|
|
|
*/
|
2013-11-14 19:19:32 +01:00
|
|
|
inline float getSpellSuccessChance (const ESM::Spell* spell, const MWWorld::Ptr& actor, int* effectiveSchool = NULL)
|
2012-05-29 18:33:01 +02:00
|
|
|
{
|
|
|
|
NpcStats& stats = MWWorld::Class::get(actor).getNpcStats(actor);
|
|
|
|
CreatureStats& creatureStats = MWWorld::Class::get(actor).getCreatureStats(actor);
|
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
if (creatureStats.getMagicEffects().get(ESM::MagicEffect::Silence).mMagnitude)
|
|
|
|
return 0;
|
2012-05-29 18:33:01 +02:00
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
float y = FLT_MAX;
|
|
|
|
float lowestSkill = 0;
|
2012-05-29 15:13:44 +02:00
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator it = spell->mEffects.mList.begin(); it != spell->mEffects.mList.end(); ++it)
|
|
|
|
{
|
|
|
|
float x = it->mDuration;
|
|
|
|
const ESM::MagicEffect* magicEffect = MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find(
|
|
|
|
it->mEffectID);
|
|
|
|
if (!(magicEffect->mData.mFlags & ESM::MagicEffect::UncappedDamage))
|
|
|
|
x = std::max(1.f, x);
|
|
|
|
x *= 0.1 * magicEffect->mData.mBaseCost;
|
|
|
|
x *= 0.5 * (it->mMagnMin + it->mMagnMax);
|
|
|
|
x *= it->mArea * 0.05 * magicEffect->mData.mBaseCost;
|
2013-11-15 20:29:47 +01:00
|
|
|
if (magicEffect->mData.mFlags & ESM::MagicEffect::CastTarget)
|
2013-11-14 19:19:32 +01:00
|
|
|
x *= 1.5;
|
|
|
|
static const float fEffectCostMult = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find(
|
|
|
|
"fEffectCostMult")->getFloat();
|
|
|
|
x *= fEffectCostMult;
|
|
|
|
|
|
|
|
float s = 2 * stats.getSkill(spellSchoolToSkill(magicEffect->mData.mSchool)).getModified();
|
|
|
|
if (s - x < y)
|
|
|
|
{
|
|
|
|
y = s - x;
|
|
|
|
if (effectiveSchool)
|
|
|
|
*effectiveSchool = magicEffect->mData.mSchool;
|
|
|
|
lowestSkill = s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-17 23:15:57 +01:00
|
|
|
if (spell->mData.mType != ESM::Spell::ST_Spell)
|
|
|
|
return 100;
|
|
|
|
|
|
|
|
if (spell->mData.mFlags & ESM::Spell::F_Always)
|
|
|
|
return 100;
|
2013-11-14 19:19:32 +01:00
|
|
|
|
|
|
|
int castBonus = -stats.getMagicEffects().get(ESM::MagicEffect::Sound).mMagnitude;
|
2013-11-17 23:15:57 +01:00
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
int actorWillpower = stats.getAttribute(ESM::Attribute::Willpower).getModified();
|
|
|
|
int actorLuck = stats.getAttribute(ESM::Attribute::Luck).getModified();
|
2012-05-29 15:13:44 +02:00
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
float castChance = (lowestSkill - spell->mData.mCost + castBonus + 0.2 * actorWillpower + 0.1 * actorLuck) * stats.getFatigueTerm();
|
|
|
|
if (MWBase::Environment::get().getWorld()->getGodModeState() && actor.getRefData().getHandle() == "player")
|
|
|
|
castChance = 100;
|
|
|
|
|
|
|
|
return std::max(0.f, std::min(100.f, castChance));
|
2012-05-29 15:13:44 +02:00
|
|
|
}
|
2012-10-23 11:42:38 +02:00
|
|
|
|
2013-11-14 19:19:32 +01:00
|
|
|
inline float getSpellSuccessChance (const std::string& spellId, const MWWorld::Ptr& actor, int* effectiveSchool = NULL)
|
2012-10-23 11:42:38 +02:00
|
|
|
{
|
2012-11-05 21:19:22 +04:00
|
|
|
const ESM::Spell* spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(spellId);
|
2013-11-14 19:19:32 +01:00
|
|
|
return getSpellSuccessChance(spell, actor, effectiveSchool);
|
2012-10-23 11:42:38 +02:00
|
|
|
}
|
2013-11-14 19:19:32 +01:00
|
|
|
|
|
|
|
inline int getSpellSchool(const std::string& spellId, const MWWorld::Ptr& actor)
|
|
|
|
{
|
|
|
|
int school = 0;
|
|
|
|
getSpellSuccessChance(spellId, actor, &school);
|
|
|
|
return school;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int getSpellSchool(const ESM::Spell* spell, const MWWorld::Ptr& actor)
|
|
|
|
{
|
|
|
|
int school = 0;
|
|
|
|
getSpellSuccessChance(spell, actor, &school);
|
|
|
|
return school;
|
|
|
|
}
|
|
|
|
|
2013-11-16 02:34:43 +01:00
|
|
|
/// @return >=100 for fully resisted. can also return negative value for damage amplification.
|
|
|
|
inline float getEffectResistance (short effectId, const MWWorld::Ptr& actor, const MWWorld::Ptr& caster, const ESM::Spell* spell = NULL)
|
|
|
|
{
|
|
|
|
const ESM::MagicEffect *magicEffect =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find (
|
|
|
|
effectId);
|
|
|
|
|
|
|
|
const MWMechanics::CreatureStats& stats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
float resisted = 0;
|
|
|
|
if (magicEffect->mData.mFlags & ESM::MagicEffect::Harmful)
|
|
|
|
{
|
|
|
|
|
|
|
|
short resistanceEffect = ESM::MagicEffect::getResistanceEffect(effectId);
|
|
|
|
short weaknessEffect = ESM::MagicEffect::getWeaknessEffect(effectId);
|
|
|
|
|
|
|
|
float resistance = 0;
|
|
|
|
if (resistanceEffect != -1)
|
|
|
|
resistance += stats.getMagicEffects().get(resistanceEffect).mMagnitude;
|
|
|
|
if (weaknessEffect != -1)
|
|
|
|
resistance -= stats.getMagicEffects().get(weaknessEffect).mMagnitude;
|
|
|
|
|
|
|
|
|
|
|
|
float willpower = stats.getAttribute(ESM::Attribute::Willpower).getModified();
|
|
|
|
float luck = stats.getAttribute(ESM::Attribute::Luck).getModified();
|
|
|
|
float x = (willpower + 0.1 * luck) * stats.getFatigueTerm();
|
|
|
|
|
|
|
|
// This makes spells that are easy to cast harder to resist and vice versa
|
2013-11-19 16:52:26 +01:00
|
|
|
if (spell != NULL && caster.getClass().isActor())
|
2013-11-16 02:34:43 +01:00
|
|
|
{
|
|
|
|
float castChance = getSpellSuccessChance(spell, caster);
|
|
|
|
if (castChance > 0)
|
|
|
|
x *= 50 / castChance;
|
|
|
|
}
|
|
|
|
|
|
|
|
float roll = static_cast<float>(std::rand()) / RAND_MAX * 100;
|
|
|
|
if (magicEffect->mData.mFlags & ESM::MagicEffect::NoMagnitude)
|
|
|
|
roll -= resistance;
|
|
|
|
|
|
|
|
if (x <= roll)
|
|
|
|
x = 0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (magicEffect->mData.mFlags & ESM::MagicEffect::NoMagnitude)
|
|
|
|
x = 100;
|
|
|
|
else
|
|
|
|
x = roll / std::min(x, 100.f);
|
|
|
|
}
|
|
|
|
|
|
|
|
x = std::min(x + resistance, 100.f);
|
|
|
|
|
|
|
|
resisted = x;
|
|
|
|
}
|
|
|
|
|
|
|
|
return resisted;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline float getEffectMultiplier(short effectId, const MWWorld::Ptr& actor, const MWWorld::Ptr& caster, const ESM::Spell* spell = NULL)
|
|
|
|
{
|
|
|
|
float resistance = getEffectResistance(effectId, actor, caster, spell);
|
|
|
|
if (resistance >= 0)
|
|
|
|
return 1 - resistance / 100.f;
|
|
|
|
else
|
|
|
|
return -(resistance-100) / 100.f;
|
|
|
|
}
|
|
|
|
|
2013-11-17 23:15:57 +01:00
|
|
|
|
|
|
|
class CastSpell
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
MWWorld::Ptr mCaster;
|
|
|
|
MWWorld::Ptr mTarget;
|
|
|
|
|
|
|
|
bool mStack;
|
|
|
|
std::string mId; // ID of spell, potion, item etc
|
|
|
|
std::string mSourceName; // Display name for spell, potion, etc
|
|
|
|
|
|
|
|
public:
|
|
|
|
CastSpell(const MWWorld::Ptr& caster, const MWWorld::Ptr& target);
|
|
|
|
|
|
|
|
bool cast (const ESM::Spell* spell);
|
|
|
|
bool cast (const MWWorld::Ptr& item);
|
|
|
|
bool cast (const ESM::Ingredient* ingredient);
|
|
|
|
bool cast (const ESM::Potion* potion);
|
|
|
|
|
|
|
|
/// @note Auto detects if spell, ingredient or potion
|
|
|
|
bool cast (const std::string& id);
|
|
|
|
|
|
|
|
void inflict (const MWWorld::Ptr& target, const MWWorld::Ptr& caster,
|
|
|
|
const ESM::EffectList& effects, ESM::RangeType range, bool reflected=false);
|
|
|
|
|
|
|
|
void applyInstantEffect (const MWWorld::Ptr& target, short effectId, float magnitude);
|
|
|
|
};
|
|
|
|
|
2012-05-29 15:13:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|