|
|
@ -50,11 +50,6 @@ namespace MWMechanics
|
|
|
|
|
|
|
|
|
|
|
|
float getSpellSuccessChance (const ESM::Spell* spell, const MWWorld::Ptr& actor, int* effectiveSchool, bool cap)
|
|
|
|
float getSpellSuccessChance (const ESM::Spell* spell, const MWWorld::Ptr& actor, int* effectiveSchool, bool cap)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
CreatureStats& stats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (stats.getMagicEffects().get(ESM::MagicEffect::Silence).getMagnitude())
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float y = std::numeric_limits<float>::max();
|
|
|
|
float y = std::numeric_limits<float>::max();
|
|
|
|
float lowestSkill = 0;
|
|
|
|
float lowestSkill = 0;
|
|
|
|
|
|
|
|
|
|
|
@ -84,6 +79,13 @@ namespace MWMechanics
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool godmode = actor == MWMechanics::getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CreatureStats& stats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (stats.getMagicEffects().get(ESM::MagicEffect::Silence).getMagnitude()&& !godmode)
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Power)
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Power)
|
|
|
|
return stats.getSpells().canUsePower(spell) ? 100 : 0;
|
|
|
|
return stats.getSpells().canUsePower(spell) ? 100 : 0;
|
|
|
|
|
|
|
|
|
|
|
@ -93,6 +95,11 @@ namespace MWMechanics
|
|
|
|
if (spell->mData.mFlags & ESM::Spell::F_Always)
|
|
|
|
if (spell->mData.mFlags & ESM::Spell::F_Always)
|
|
|
|
return 100;
|
|
|
|
return 100;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return 100;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float castBonus = -stats.getMagicEffects().get(ESM::MagicEffect::Sound).getMagnitude();
|
|
|
|
float castBonus = -stats.getMagicEffects().get(ESM::MagicEffect::Sound).getMagnitude();
|
|
|
|
|
|
|
|
|
|
|
|
int actorWillpower = stats.getAttribute(ESM::Attribute::Willpower).getModified();
|
|
|
|
int actorWillpower = stats.getAttribute(ESM::Attribute::Willpower).getModified();
|
|
|
@ -713,14 +720,19 @@ namespace MWMechanics
|
|
|
|
|
|
|
|
|
|
|
|
mStack = false;
|
|
|
|
mStack = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool godmode = mCaster == MWMechanics::getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
|
|
|
|
|
|
|
|
|
|
|
// Check if there's enough charge left
|
|
|
|
// Check if there's enough charge left
|
|
|
|
if (enchantment->mData.mType == ESM::Enchantment::WhenUsed || enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
|
|
|
|
if (enchantment->mData.mType == ESM::Enchantment::WhenUsed || enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const int castCost = getEffectiveEnchantmentCastCost(static_cast<float>(enchantment->mData.mCost), mCaster);
|
|
|
|
int castCost = getEffectiveEnchantmentCastCost(static_cast<float>(enchantment->mData.mCost), mCaster);
|
|
|
|
|
|
|
|
|
|
|
|
if (item.getCellRef().getEnchantmentCharge() == -1)
|
|
|
|
if (item.getCellRef().getEnchantmentCharge() == -1)
|
|
|
|
item.getCellRef().setEnchantmentCharge(static_cast<float>(enchantment->mData.mCharge));
|
|
|
|
item.getCellRef().setEnchantmentCharge(static_cast<float>(enchantment->mData.mCharge));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (godmode)
|
|
|
|
|
|
|
|
castCost = 0;
|
|
|
|
|
|
|
|
|
|
|
|
if (item.getCellRef().getEnchantmentCharge() < castCost)
|
|
|
|
if (item.getCellRef().getEnchantmentCharge() < castCost)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (mCaster == getPlayer())
|
|
|
|
if (mCaster == getPlayer())
|
|
|
@ -750,8 +762,10 @@ namespace MWMechanics
|
|
|
|
if (mCaster == getPlayer())
|
|
|
|
if (mCaster == getPlayer())
|
|
|
|
mCaster.getClass().skillUsageSucceeded (mCaster, ESM::Skill::Enchant, 1);
|
|
|
|
mCaster.getClass().skillUsageSucceeded (mCaster, ESM::Skill::Enchant, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (enchantment->mData.mType == ESM::Enchantment::CastOnce)
|
|
|
|
if (enchantment->mData.mType == ESM::Enchantment::CastOnce && !godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
item.getContainerStore()->remove(item, 1, mCaster);
|
|
|
|
item.getContainerStore()->remove(item, 1, mCaster);
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (enchantment->mData.mType != ESM::Enchantment::WhenStrikes)
|
|
|
|
else if (enchantment->mData.mType != ESM::Enchantment::WhenStrikes)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (mCaster == getPlayer())
|
|
|
|
if (mCaster == getPlayer())
|
|
|
@ -801,37 +815,45 @@ namespace MWMechanics
|
|
|
|
|
|
|
|
|
|
|
|
int school = 0;
|
|
|
|
int school = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool godmode = mCaster == MWMechanics::getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
|
|
|
|
|
|
|
|
|
|
|
if (mCaster.getClass().isActor() && !mAlwaysSucceed)
|
|
|
|
if (mCaster.getClass().isActor() && !mAlwaysSucceed)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
school = getSpellSchool(spell, mCaster);
|
|
|
|
school = getSpellSchool(spell, mCaster);
|
|
|
|
|
|
|
|
|
|
|
|
CreatureStats& stats = mCaster.getClass().getCreatureStats(mCaster);
|
|
|
|
CreatureStats& stats = mCaster.getClass().getCreatureStats(mCaster);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
// Reduce fatigue (note that in the vanilla game, both GMSTs are 0, and there's no fatigue loss)
|
|
|
|
// Reduce fatigue (note that in the vanilla game, both GMSTs are 0, and there's no fatigue loss)
|
|
|
|
static const float fFatigueSpellBase = store.get<ESM::GameSetting>().find("fFatigueSpellBase")->getFloat();
|
|
|
|
static const float fFatigueSpellBase = store.get<ESM::GameSetting>().find("fFatigueSpellBase")->getFloat();
|
|
|
|
static const float fFatigueSpellMult = store.get<ESM::GameSetting>().find("fFatigueSpellMult")->getFloat();
|
|
|
|
static const float fFatigueSpellMult = store.get<ESM::GameSetting>().find("fFatigueSpellMult")->getFloat();
|
|
|
|
DynamicStat<float> fatigue = stats.getFatigue();
|
|
|
|
DynamicStat<float> fatigue = stats.getFatigue();
|
|
|
|
const float normalizedEncumbrance = mCaster.getClass().getNormalizedEncumbrance(mCaster);
|
|
|
|
const float normalizedEncumbrance = mCaster.getClass().getNormalizedEncumbrance(mCaster);
|
|
|
|
|
|
|
|
|
|
|
|
float fatigueLoss = spell->mData.mCost * (fFatigueSpellBase + normalizedEncumbrance * fFatigueSpellMult);
|
|
|
|
float fatigueLoss = spell->mData.mCost * (fFatigueSpellBase + normalizedEncumbrance * fFatigueSpellMult);
|
|
|
|
fatigue.setCurrent(fatigue.getCurrent() - fatigueLoss); stats.setFatigue(fatigue);
|
|
|
|
fatigue.setCurrent(fatigue.getCurrent() - fatigueLoss); stats.setFatigue(fatigue);
|
|
|
|
|
|
|
|
|
|
|
|
bool fail = false;
|
|
|
|
bool fail = false;
|
|
|
|
|
|
|
|
|
|
|
|
// Check success
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Major change done by tes3mp:
|
|
|
|
// Major change done by tes3mp:
|
|
|
|
|
|
|
|
//
|
|
|
|
// Instead of checking whether the caster is a player or an NPC,
|
|
|
|
// Instead of checking whether the caster is a player or an NPC,
|
|
|
|
// check whether it's the LocalPlayer or a DedicatedPlayer and calculate
|
|
|
|
// first check whether it's the LocalPlayer or a DedicatedPlayer and calculate
|
|
|
|
// calculate the success chance in clients' LocalPlayer::prepareAttack()
|
|
|
|
// calculate the success chance in clients' LocalPlayer::prepareAttack()
|
|
|
|
|
|
|
|
//
|
|
|
|
// TODO: Make this make sense for NPCs too
|
|
|
|
// TODO: Make this make sense for NPCs too
|
|
|
|
|
|
|
|
//
|
|
|
|
// TODO: See if LocalPlayer being the target and having godmode on
|
|
|
|
// TODO: See if LocalPlayer being the target and having godmode on
|
|
|
|
// can be accounted for like it is in OpenMW's corresponding code
|
|
|
|
// can be accounted for like it is in OpenMW's corresponding code
|
|
|
|
|
|
|
|
|
|
|
|
mwmp::DedicatedPlayer *dedicatedPlayer = mwmp::Players::getPlayer(mCaster);
|
|
|
|
mwmp::DedicatedPlayer *dedicatedPlayer = mwmp::Players::getPlayer(mCaster);
|
|
|
|
bool isDedicated = dedicatedPlayer != NULL;
|
|
|
|
bool isDedicated = dedicatedPlayer != NULL;
|
|
|
|
|
|
|
|
|
|
|
|
if (isDedicated)
|
|
|
|
if (isDedicated)
|
|
|
|
dedicatedPlayer->attack.pressed = false;
|
|
|
|
dedicatedPlayer->attack.pressed = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Check success
|
|
|
|
if ((!isDedicated && !mwmp::Main::get().getLocalPlayer()->attack.success) ||
|
|
|
|
if ((!isDedicated && !mwmp::Main::get().getLocalPlayer()->attack.success) ||
|
|
|
|
(isDedicated && dedicatedPlayer->attack.success == 0))
|
|
|
|
(isDedicated && dedicatedPlayer->attack.success == 0))
|
|
|
|
{
|
|
|
|
{
|
|
|
@ -841,6 +863,16 @@ namespace MWMechanics
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fail = true;
|
|
|
|
fail = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (!(mCaster == getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState()))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
float successChance = getSpellSuccessChance(spell, mCaster);
|
|
|
|
|
|
|
|
if (Misc::Rng::roll0to99() >= successChance)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (mCaster == getPlayer())
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sMagicSkillFail}");
|
|
|
|
|
|
|
|
fail = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (fail)
|
|
|
|
if (fail)
|
|
|
|
{
|
|
|
|
{
|
|
|
@ -853,6 +885,7 @@ namespace MWMechanics
|
|
|
|
sndMgr->playSound3D(mCaster, "Spell Failure " + schools[school], 1.0f, 1.0f);
|
|
|
|
sndMgr->playSound3D(mCaster, "Spell Failure " + schools[school], 1.0f, 1.0f);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// A power can be used once per 24h
|
|
|
|
// A power can be used once per 24h
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Power)
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Power)
|
|
|
@ -1059,6 +1092,8 @@ namespace MWMechanics
|
|
|
|
|
|
|
|
|
|
|
|
bool receivedMagicDamage = false;
|
|
|
|
bool receivedMagicDamage = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool godmode = actor == MWMechanics::getPlayer() && MWBase::Environment::get().getWorld()->getGodModeState();
|
|
|
|
|
|
|
|
|
|
|
|
switch (effectKey.mId)
|
|
|
|
switch (effectKey.mId)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
case ESM::MagicEffect::DamageAttribute:
|
|
|
|
case ESM::MagicEffect::DamageAttribute:
|
|
|
@ -1081,21 +1116,34 @@ namespace MWMechanics
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::RestoreHealth, magnitude);
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::RestoreHealth, magnitude);
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
case ESM::MagicEffect::DamageHealth:
|
|
|
|
case ESM::MagicEffect::DamageHealth:
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::DamageHealth, -magnitude);
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::DamageHealth, -magnitude);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM::MagicEffect::DamageMagicka:
|
|
|
|
case ESM::MagicEffect::DamageMagicka:
|
|
|
|
case ESM::MagicEffect::DamageFatigue:
|
|
|
|
case ESM::MagicEffect::DamageFatigue:
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::DamageHealth, -magnitude);
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::DamageHealth, -magnitude);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case ESM::MagicEffect::AbsorbHealth:
|
|
|
|
case ESM::MagicEffect::AbsorbHealth:
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
if (magnitude > 0.f)
|
|
|
|
if (magnitude > 0.f)
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::AbsorbHealth, -magnitude);
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::AbsorbHealth, -magnitude);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ESM::MagicEffect::AbsorbMagicka:
|
|
|
|
case ESM::MagicEffect::AbsorbMagicka:
|
|
|
|
case ESM::MagicEffect::AbsorbFatigue:
|
|
|
|
case ESM::MagicEffect::AbsorbFatigue:
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::AbsorbHealth, -magnitude);
|
|
|
|
adjustDynamicStat(creatureStats, effectKey.mId-ESM::MagicEffect::AbsorbHealth, -magnitude);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case ESM::MagicEffect::DisintegrateArmor:
|
|
|
|
case ESM::MagicEffect::DisintegrateArmor:
|
|
|
@ -1140,9 +1188,13 @@ namespace MWMechanics
|
|
|
|
if (weather > 1)
|
|
|
|
if (weather > 1)
|
|
|
|
damageScale *= fMagicSunBlockedMult;
|
|
|
|
damageScale *= fMagicSunBlockedMult;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
|
|
|
|
{
|
|
|
|
adjustDynamicStat(creatureStats, 0, -magnitude * damageScale);
|
|
|
|
adjustDynamicStat(creatureStats, 0, -magnitude * damageScale);
|
|
|
|
if (magnitude * damageScale > 0.f)
|
|
|
|
if (magnitude * damageScale > 0.f)
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -1150,9 +1202,13 @@ namespace MWMechanics
|
|
|
|
case ESM::MagicEffect::ShockDamage:
|
|
|
|
case ESM::MagicEffect::ShockDamage:
|
|
|
|
case ESM::MagicEffect::FrostDamage:
|
|
|
|
case ESM::MagicEffect::FrostDamage:
|
|
|
|
case ESM::MagicEffect::Poison:
|
|
|
|
case ESM::MagicEffect::Poison:
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!godmode)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
adjustDynamicStat(creatureStats, 0, -magnitude);
|
|
|
|
adjustDynamicStat(creatureStats, 0, -magnitude);
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
receivedMagicDamage = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|