1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-10-25 12:56:36 +00:00

Autocalculate enchantment costs and charges

This commit is contained in:
Evil Eye 2023-06-17 22:25:47 +02:00
parent 0755954b78
commit e9bcad4e05
17 changed files with 114 additions and 41 deletions

View file

@ -55,6 +55,7 @@
Bug #7307: Alchemy "Magic Effect" search string does not match on tool tip for effects related to attributes Bug #7307: Alchemy "Magic Effect" search string does not match on tool tip for effects related to attributes
Bug #7413: Generated wilderness cells don't spawn fish Bug #7413: Generated wilderness cells don't spawn fish
Bug #7415: Unbreakable lock discrepancies Bug #7415: Unbreakable lock discrepancies
Bug #7428: AutoCalc flag is not used to calculate enchantment costs
Feature #3537: Shader-based water ripples Feature #3537: Shader-based water ripples
Feature #5492: Let rain and snow collide with statics Feature #5492: Let rain and snow collide with statics
Feature #6447: Add LOD support to Object Paging Feature #6447: Add LOD support to Object Paging

View file

@ -12,6 +12,8 @@
#include "../mwbase/environment.hpp" #include "../mwbase/environment.hpp"
#include "../mwbase/world.hpp" #include "../mwbase/world.hpp"
#include "../mwmechanics/spellutil.hpp"
#include "../mwworld/class.hpp" #include "../mwworld/class.hpp"
#include "../mwworld/esmstore.hpp" #include "../mwworld/esmstore.hpp"
@ -199,8 +201,8 @@ namespace MWGui
break; break;
line.mCharge->setVisible(true); line.mCharge->setVisible(true);
line.mCharge->setValue( line.mCharge->setValue(static_cast<int>(line.mItemPtr.getCellRef().getEnchantmentCharge()),
static_cast<int>(line.mItemPtr.getCellRef().getEnchantmentCharge()), ench->mData.mCharge); MWMechanics::getEnchantmentCharge(*ench));
break; break;
} }
} }

View file

@ -24,6 +24,7 @@
#include "../mwworld/nullaction.hpp" #include "../mwworld/nullaction.hpp"
#include "../mwmechanics/alchemy.hpp" #include "../mwmechanics/alchemy.hpp"
#include "../mwmechanics/spellutil.hpp"
namespace namespace
{ {
@ -112,8 +113,8 @@ namespace
if (ench->mData.mType == ESM::Enchantment::ConstantEffect) if (ench->mData.mType == ESM::Enchantment::ConstantEffect)
leftChargePercent = 101; leftChargePercent = 101;
else else
leftChargePercent = static_cast<int>( leftChargePercent
left.mBase.getCellRef().getNormalizedEnchantmentCharge(ench->mData.mCharge) * 100); = static_cast<int>(left.mBase.getCellRef().getNormalizedEnchantmentCharge(*ench) * 100);
} }
} }
@ -126,8 +127,8 @@ namespace
if (ench->mData.mType == ESM::Enchantment::ConstantEffect) if (ench->mData.mType == ESM::Enchantment::ConstantEffect)
rightChargePercent = 101; rightChargePercent = 101;
else else
rightChargePercent = static_cast<int>( rightChargePercent
right.mBase.getCellRef().getNormalizedEnchantmentCharge(ench->mData.mCharge) * 100); = static_cast<int>(right.mBase.getCellRef().getNormalizedEnchantmentCharge(*ench) * 100);
} }
} }
@ -304,8 +305,8 @@ namespace MWGui
return false; return false;
} }
if (base.getCellRef().getEnchantmentCharge() >= ench->mData.mCharge if (base.getCellRef().getEnchantmentCharge() == -1
|| base.getCellRef().getEnchantmentCharge() == -1) || base.getCellRef().getEnchantmentCharge() >= MWMechanics::getEnchantmentCharge(*ench))
return false; return false;
} }

View file

@ -153,13 +153,12 @@ namespace MWGui
&& item.getClass().canBeEquipped(item, mActor).first == 0) && item.getClass().canBeEquipped(item, mActor).first == 0)
continue; continue;
int castCost int castCost = MWMechanics::getEffectiveEnchantmentCastCost(*enchant, mActor);
= MWMechanics::getEffectiveEnchantmentCastCost(static_cast<float>(enchant->mData.mCost), mActor);
std::string cost = std::to_string(castCost); std::string cost = std::to_string(castCost);
int currentCharge = int(item.getCellRef().getEnchantmentCharge()); int currentCharge = int(item.getCellRef().getEnchantmentCharge());
if (currentCharge == -1) if (currentCharge == -1)
currentCharge = enchant->mData.mCharge; currentCharge = MWMechanics::getEnchantmentCharge(*enchant);
std::string charge = std::to_string(currentCharge); std::string charge = std::to_string(currentCharge);
newSpell.mCostColumn = cost + "/" + charge; newSpell.mCostColumn = cost + "/" + charge;

View file

@ -539,7 +539,7 @@ namespace MWGui
if (enchant->mData.mType == ESM::Enchantment::WhenStrikes if (enchant->mData.mType == ESM::Enchantment::WhenStrikes
|| enchant->mData.mType == ESM::Enchantment::WhenUsed) || enchant->mData.mType == ESM::Enchantment::WhenUsed)
{ {
int maxCharge = enchant->mData.mCharge; const int maxCharge = MWMechanics::getEnchantmentCharge(*enchant);
int charge = (info.remainingEnchantCharge == -1) ? maxCharge : info.remainingEnchantCharge; int charge = (info.remainingEnchantCharge == -1) ? maxCharge : info.remainingEnchantCharge;
const int chargeWidth = 204; const int chargeWidth = 204;

View file

@ -1384,8 +1384,7 @@ namespace MWGui
mSelectedSpell = ESM::RefId(); mSelectedSpell = ESM::RefId();
const ESM::Enchantment* ench = mStore->get<ESM::Enchantment>().find(item.getClass().getEnchantment(item)); const ESM::Enchantment* ench = mStore->get<ESM::Enchantment>().find(item.getClass().getEnchantment(item));
int chargePercent int chargePercent = static_cast<int>(item.getCellRef().getNormalizedEnchantmentCharge(*ench) * 100);
= static_cast<int>(item.getCellRef().getNormalizedEnchantmentCharge(ench->mData.mCharge) * 100);
mHud->setSelectedEnchantItem(item, chargePercent); mHud->setSelectedEnchantItem(item, chargePercent);
mSpellWindow->setTitle(item.getClass().getName(item)); mSpellWindow->setTitle(item.getClass().getName(item));
} }

View file

@ -15,6 +15,7 @@
#include "actorutil.hpp" #include "actorutil.hpp"
#include "creaturestats.hpp" #include "creaturestats.hpp"
#include "spellutil.hpp"
namespace MWMechanics namespace MWMechanics
{ {
@ -68,8 +69,9 @@ namespace MWMechanics
const ESM::Enchantment* enchantment const ESM::Enchantment* enchantment
= MWBase::Environment::get().getESMStore()->get<ESM::Enchantment>().find( = MWBase::Environment::get().getESMStore()->get<ESM::Enchantment>().find(
item.getClass().getEnchantment(item)); item.getClass().getEnchantment(item));
item.getCellRef().setEnchantmentCharge(std::min( const int maxCharge = MWMechanics::getEnchantmentCharge(*enchantment);
item.getCellRef().getEnchantmentCharge() + restored, static_cast<float>(enchantment->mData.mCharge))); item.getCellRef().setEnchantmentCharge(
std::min(item.getCellRef().getEnchantmentCharge() + restored, static_cast<float>(maxCharge)));
MWBase::Environment::get().getWindowManager()->playSound(ESM::RefId::stringRefId("Enchant Success")); MWBase::Environment::get().getWindowManager()->playSound(ESM::RefId::stringRefId("Enchant Success"));

View file

@ -319,10 +319,11 @@ namespace MWMechanics
if (!godmode if (!godmode
&& (type == ESM::Enchantment::WhenUsed || (!isProjectile && type == ESM::Enchantment::WhenStrikes))) && (type == ESM::Enchantment::WhenUsed || (!isProjectile && type == ESM::Enchantment::WhenStrikes)))
{ {
int castCost = getEffectiveEnchantmentCastCost(static_cast<float>(enchantment->mData.mCost), mCaster); int castCost = getEffectiveEnchantmentCastCost(*enchantment, 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>(MWMechanics::getEnchantmentCharge(*enchantment)));
if (item.getCellRef().getEnchantmentCharge() < castCost) if (item.getCellRef().getEnchantmentCharge() < castCost)
{ {

View file

@ -296,8 +296,7 @@ namespace
{ {
const ESM::Enchantment* enchantment = esmStore.get<ESM::Enchantment>().search(spellId); const ESM::Enchantment* enchantment = esmStore.get<ESM::Enchantment>().search(spellId);
if (enchantment) if (enchantment)
spellCost = MWMechanics::getEffectiveEnchantmentCastCost( spellCost = MWMechanics::getEffectiveEnchantmentCastCost(*enchantment, caster);
static_cast<float>(enchantment->mData.mCost), caster);
} }
// Magicka is increased by the cost of the spell // Magicka is increased by the cost of the spell

View file

@ -171,7 +171,7 @@ namespace MWMechanics
if (actor.getClass().isNpc() && !store.isEquipped(ptr)) if (actor.getClass().isNpc() && !store.isEquipped(ptr))
return 0.f; return 0.f;
int castCost = getEffectiveEnchantmentCastCost(static_cast<float>(enchantment->mData.mCost), actor); int castCost = getEffectiveEnchantmentCastCost(*enchantment, actor);
if (ptr.getCellRef().getEnchantmentCharge() != -1 && ptr.getCellRef().getEnchantmentCharge() < castCost) if (ptr.getCellRef().getEnchantmentCharge() != -1 && ptr.getCellRef().getEnchantmentCharge() < castCost)
return 0.f; return 0.f;

View file

@ -2,6 +2,7 @@
#include <limits> #include <limits>
#include <components/esm3/loadench.hpp>
#include <components/esm3/loadmgef.hpp> #include <components/esm3/loadmgef.hpp>
#include "../mwbase/environment.hpp" #include "../mwbase/environment.hpp"
@ -15,6 +16,27 @@
namespace MWMechanics namespace MWMechanics
{ {
namespace
{
float getTotalCost(const ESM::EffectList& list, const EffectCostMethod method = EffectCostMethod::GameSpell)
{
float cost = 0;
for (const ESM::ENAMstruct& effect : list.mList)
{
float effectCost = std::max(0.f, MWMechanics::calcEffectCost(effect, nullptr, method));
// This is applied to the whole spell cost for each effect when
// creating spells, but is only applied on the effect itself in TES:CS.
if (effect.mRange == ESM::RT_Target)
effectCost *= 1.5;
cost += effectCost;
}
return cost;
}
}
ESM::RefId spellSchoolToSkill(int school) ESM::RefId spellSchoolToSkill(int school)
{ {
static const std::array<ESM::RefId, 6> schoolSkillArray{ static const std::array<ESM::RefId, 6> schoolSkillArray{
@ -39,6 +61,11 @@ namespace MWMechanics
bool appliedOnce = magicEffect->mData.mFlags & ESM::MagicEffect::AppliedOnce; bool appliedOnce = magicEffect->mData.mFlags & ESM::MagicEffect::AppliedOnce;
int minMagn = hasMagnitude ? effect.mMagnMin : 1; int minMagn = hasMagnitude ? effect.mMagnMin : 1;
int maxMagn = hasMagnitude ? effect.mMagnMax : 1; int maxMagn = hasMagnitude ? effect.mMagnMax : 1;
if (method != EffectCostMethod::GameEnchantment)
{
minMagn = std::max(1, minMagn);
maxMagn = std::max(1, maxMagn);
}
int duration = hasDuration ? effect.mDuration : 1; int duration = hasDuration ? effect.mDuration : 1;
if (!appliedOnce) if (!appliedOnce)
duration = std::max(1, duration); duration = std::max(1, duration);
@ -52,7 +79,7 @@ namespace MWMechanics
minArea = 1; minArea = 1;
} }
float x = 0.5 * (std::max(1, minMagn) + std::max(1, maxMagn)); float x = 0.5 * (minMagn + maxMagn);
x *= 0.1 * magicEffect->mData.mBaseCost; x *= 0.1 * magicEffect->mData.mBaseCost;
x *= durationOffset + duration; x *= durationOffset + duration;
x += 0.05 * std::max(minArea, effect.mArea) * magicEffect->mData.mBaseCost; x += 0.05 * std::max(minArea, effect.mArea) * magicEffect->mData.mBaseCost;
@ -65,19 +92,7 @@ namespace MWMechanics
if (!(spell.mData.mFlags & ESM::Spell::F_Autocalc)) if (!(spell.mData.mFlags & ESM::Spell::F_Autocalc))
return spell.mData.mCost; return spell.mData.mCost;
float cost = 0; float cost = getTotalCost(spell.mEffects);
for (const ESM::ENAMstruct& effect : spell.mEffects.mList)
{
float effectCost = std::max(0.f, MWMechanics::calcEffectCost(effect));
// This is applied to the whole spell cost for each effect when
// creating spells, but is only applied on the effect itself in TES:CS.
if (effect.mRange == ESM::RT_Target)
effectCost *= 1.5;
cost += effectCost;
}
return std::round(cost); return std::round(cost);
} }
@ -94,6 +109,50 @@ namespace MWMechanics
return static_cast<int>((result < 1) ? 1 : result); return static_cast<int>((result < 1) ? 1 : result);
} }
int getEffectiveEnchantmentCastCost(const ESM::Enchantment& enchantment, const MWWorld::Ptr& actor)
{
float castCost;
if (enchantment.mData.mFlags & ESM::Enchantment::Autocalc)
castCost = getTotalCost(enchantment.mEffects, EffectCostMethod::GameEnchantment);
else
castCost = static_cast<float>(enchantment.mData.mCost);
return getEffectiveEnchantmentCastCost(castCost, actor);
}
int getEnchantmentCharge(const ESM::Enchantment& enchantment)
{
if (enchantment.mData.mFlags & ESM::Enchantment::Autocalc)
{
int charge
= static_cast<int>(std::round(getTotalCost(enchantment.mEffects, EffectCostMethod::GameEnchantment)));
const auto& store = MWBase::Environment::get().getESMStore()->get<ESM::GameSetting>();
switch (enchantment.mData.mType)
{
case ESM::Enchantment::CastOnce:
{
static const int iMagicItemChargeOnce = store.find("iMagicItemChargeOnce")->mValue.getInteger();
return charge * iMagicItemChargeOnce;
}
case ESM::Enchantment::WhenStrikes:
{
static const int iMagicItemChargeStrike = store.find("iMagicItemChargeStrike")->mValue.getInteger();
return charge * iMagicItemChargeStrike;
}
case ESM::Enchantment::WhenUsed:
{
static const int iMagicItemChargeUse = store.find("iMagicItemChargeUse")->mValue.getInteger();
return charge * iMagicItemChargeUse;
}
case ESM::Enchantment::ConstantEffect:
{
static const int iMagicItemChargeConst = store.find("iMagicItemChargeConst")->mValue.getInteger();
return charge * iMagicItemChargeConst;
}
}
}
return enchantment.mData.mCharge;
}
float calcSpellBaseSuccessChance(const ESM::Spell* spell, const MWWorld::Ptr& actor, int* effectiveSchool) float calcSpellBaseSuccessChance(const ESM::Spell* spell, const MWWorld::Ptr& actor, int* effectiveSchool)
{ {
// Morrowind for some reason uses a formula slightly different from magicka cost calculation // Morrowind for some reason uses a formula slightly different from magicka cost calculation

View file

@ -6,6 +6,7 @@
namespace ESM namespace ESM
{ {
struct ENAMstruct; struct ENAMstruct;
struct Enchantment;
struct MagicEffect; struct MagicEffect;
struct Spell; struct Spell;
} }
@ -23,6 +24,7 @@ namespace MWMechanics
{ {
GameSpell, GameSpell,
PlayerSpell, PlayerSpell,
GameEnchantment,
}; };
float calcEffectCost(const ESM::ENAMstruct& effect, const ESM::MagicEffect* magicEffect = nullptr, float calcEffectCost(const ESM::ENAMstruct& effect, const ESM::MagicEffect* magicEffect = nullptr,
@ -30,6 +32,8 @@ namespace MWMechanics
int calcSpellCost(const ESM::Spell& spell); int calcSpellCost(const ESM::Spell& spell);
int getEffectiveEnchantmentCastCost(float castCost, const MWWorld::Ptr& actor); int getEffectiveEnchantmentCastCost(float castCost, const MWWorld::Ptr& actor);
int getEffectiveEnchantmentCastCost(const ESM::Enchantment& enchantment, const MWWorld::Ptr& actor);
int getEnchantmentCharge(const ESM::Enchantment& enchantment);
/** /**
* @param spell spell to cast * @param spell spell to cast

View file

@ -108,7 +108,7 @@ namespace MWMechanics
const ESM::Enchantment* enchantment = world->getStore().get<ESM::Enchantment>().find(weapon->mEnchant); const ESM::Enchantment* enchantment = world->getStore().get<ESM::Enchantment>().find(weapon->mEnchant);
if (enchantment->mData.mType == ESM::Enchantment::WhenStrikes) if (enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
{ {
int castCost = getEffectiveEnchantmentCastCost(static_cast<float>(enchantment->mData.mCost), actor); int castCost = getEffectiveEnchantmentCastCost(*enchantment, actor);
float charge = item.getCellRef().getEnchantmentCharge(); float charge = item.getCellRef().getEnchantmentCharge();
if (charge == -1 || charge >= castCost || weapclass == ESM::WeaponType::Thrown if (charge == -1 || charge >= castCost || weapclass == ESM::WeaponType::Thrown

View file

@ -10,6 +10,7 @@
#include <apps/openmw/mwbase/environment.hpp> #include <apps/openmw/mwbase/environment.hpp>
#include <apps/openmw/mwbase/world.hpp> #include <apps/openmw/mwbase/world.hpp>
#include <apps/openmw/mwmechanics/spellutil.hpp>
#include <apps/openmw/mwworld/esmstore.hpp> #include <apps/openmw/mwworld/esmstore.hpp>
namespace MWWorld namespace MWWorld
@ -129,8 +130,9 @@ namespace MWWorld
mCellRef.mVariant); mCellRef.mVariant);
} }
float CellRef::getNormalizedEnchantmentCharge(int maxCharge) const float CellRef::getNormalizedEnchantmentCharge(const ESM::Enchantment& enchantment) const
{ {
const int maxCharge = MWMechanics::getEnchantmentCharge(enchantment);
if (maxCharge == 0) if (maxCharge == 0)
{ {
return 0; return 0;

View file

@ -9,6 +9,7 @@
namespace ESM namespace ESM
{ {
struct Enchantment;
struct ObjectState; struct ObjectState;
} }
@ -87,7 +88,7 @@ namespace MWWorld
float getEnchantmentCharge() const; float getEnchantmentCharge() const;
// Remaining enchantment charge rescaled to the supplied maximum charge (such as one of the enchantment). // Remaining enchantment charge rescaled to the supplied maximum charge (such as one of the enchantment).
float getNormalizedEnchantmentCharge(int maxCharge) const; float getNormalizedEnchantmentCharge(const ESM::Enchantment& enchantment) const;
void setEnchantmentCharge(float charge); void setEnchantmentCharge(float charge);

View file

@ -55,6 +55,7 @@
#include "../mwmechanics/creaturestats.hpp" #include "../mwmechanics/creaturestats.hpp"
#include "../mwmechanics/recharge.hpp" #include "../mwmechanics/recharge.hpp"
#include "../mwmechanics/spellutil.hpp"
#include "class.hpp" #include "class.hpp"
#include "containerstore.hpp" #include "containerstore.hpp"
@ -1248,7 +1249,8 @@ namespace MWWorld
if (enchantment->mData.mType == ESM::Enchantment::WhenUsed if (enchantment->mData.mType == ESM::Enchantment::WhenUsed
|| enchantment->mData.mType == ESM::Enchantment::WhenStrikes) || enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
mRechargingItems.emplace_back(ptr.getBase(), static_cast<float>(enchantment->mData.mCharge)); mRechargingItems.emplace_back(
ptr.getBase(), static_cast<float>(MWMechanics::getEnchantmentCharge(*enchantment)));
} }
Ptr MWWorld::CellStore::getMovedActor(int actorId) const Ptr MWWorld::CellStore::getMovedActor(int actorId) const

View file

@ -16,6 +16,7 @@
#include "../mwmechanics/levelledlist.hpp" #include "../mwmechanics/levelledlist.hpp"
#include "../mwmechanics/recharge.hpp" #include "../mwmechanics/recharge.hpp"
#include "../mwmechanics/spellutil.hpp"
#include "class.hpp" #include "class.hpp"
#include "esmstore.hpp" #include "esmstore.hpp"
@ -268,7 +269,7 @@ bool MWWorld::ContainerStore::stacks(const ConstPtr& ptr1, const ConstPtr& ptr2)
{ {
const ESM::Enchantment* enchantment = MWBase::Environment::get().getESMStore()->get<ESM::Enchantment>().find( const ESM::Enchantment* enchantment = MWBase::Environment::get().getESMStore()->get<ESM::Enchantment>().find(
ptr1.getClass().getEnchantment(ptr1)); ptr1.getClass().getEnchantment(ptr1));
float maxCharge = static_cast<float>(enchantment->mData.mCharge); const float maxCharge = static_cast<float>(MWMechanics::getEnchantmentCharge(*enchantment));
float enchantCharge1 float enchantCharge1
= ptr1.getCellRef().getEnchantmentCharge() == -1 ? maxCharge : ptr1.getCellRef().getEnchantmentCharge(); = ptr1.getCellRef().getEnchantmentCharge() == -1 ? maxCharge : ptr1.getCellRef().getEnchantmentCharge();
float enchantCharge2 float enchantCharge2
@ -507,7 +508,7 @@ void MWWorld::ContainerStore::updateRechargingItems()
if (enchantment->mData.mType == ESM::Enchantment::WhenUsed if (enchantment->mData.mType == ESM::Enchantment::WhenUsed
|| enchantment->mData.mType == ESM::Enchantment::WhenStrikes) || enchantment->mData.mType == ESM::Enchantment::WhenStrikes)
mRechargingItems.emplace_back(it, static_cast<float>(enchantment->mData.mCharge)); mRechargingItems.emplace_back(it, static_cast<float>(MWMechanics::getEnchantmentCharge(*enchantment)));
} }
} }
} }