From 90fa8dca3535a4312794e951d6e93335f16ec2c1 Mon Sep 17 00:00:00 2001 From: Evil Eye Date: Sun, 6 Jun 2021 18:10:55 +0200 Subject: [PATCH 1/2] Do not assume the bound item cache is valid after loading a save --- apps/openmw/mwmechanics/actors.cpp | 42 +++++++++++++++++------------- 1 file changed, 24 insertions(+), 18 deletions(-) diff --git a/apps/openmw/mwmechanics/actors.cpp b/apps/openmw/mwmechanics/actors.cpp index 0ffcef2f8b..218d4b4201 100644 --- a/apps/openmw/mwmechanics/actors.cpp +++ b/apps/openmw/mwmechanics/actors.cpp @@ -1239,7 +1239,7 @@ namespace MWMechanics // Update bound effects // Note: in vanilla MW multiple bound items of the same type can be created by different spells. // As these extra copies are kinda useless this may or may not be important. - static std::map boundItemsMap; + static std::map boundItemsMap; if (boundItemsMap.empty()) { boundItemsMap[ESM::MagicEffect::BoundBattleAxe] = "sMagicBoundBattleAxeID"; @@ -1255,30 +1255,36 @@ namespace MWMechanics boundItemsMap[ESM::MagicEffect::BoundSpear] = "sMagicBoundSpearID"; } - for (std::map::iterator it = boundItemsMap.begin(); it != boundItemsMap.end(); ++it) + if(ptr.getClass().hasInventoryStore(ptr)) { - bool found = creatureStats.mBoundItems.find(it->first) != creatureStats.mBoundItems.end(); - float magnitude = effects.get(it->first).getMagnitude(); - if (found != (magnitude > 0)) + for (const auto& [effect, itemGmst] : boundItemsMap) { - if (magnitude > 0) - creatureStats.mBoundItems.insert(it->first); - else - creatureStats.mBoundItems.erase(it->first); - - std::string itemGmst = it->second; - std::string item = MWBase::Environment::get().getWorld()->getStore().get().find( - itemGmst)->mValue.getString(); + bool found = creatureStats.mBoundItems.find(effect) != creatureStats.mBoundItems.end(); + float magnitude = effects.get(effect).getMagnitude(); + if (found != (magnitude > 0) || creatureStats.mBoundItems.empty()) + { + if (magnitude > 0) + creatureStats.mBoundItems.insert(effect); + else + creatureStats.mBoundItems.erase(effect); - magnitude > 0 ? addBoundItem(item, ptr) : removeBoundItem(item, ptr); + std::string item = MWBase::Environment::get().getWorld()->getStore().get().find( + itemGmst)->mValue.getString(); - if (it->first == ESM::MagicEffect::BoundGloves) - { - item = MWBase::Environment::get().getWorld()->getStore().get().find( - "sMagicBoundRightGauntletID")->mValue.getString(); magnitude > 0 ? addBoundItem(item, ptr) : removeBoundItem(item, ptr); + + if (effect == ESM::MagicEffect::BoundGloves) + { + item = MWBase::Environment::get().getWorld()->getStore().get().find( + "sMagicBoundRightGauntletID")->mValue.getString(); + magnitude > 0 ? addBoundItem(item, ptr) : removeBoundItem(item, ptr); + } } } + // mBoundItems does not get saved so the cache is out of date after loading a save. + // Use Length as a sentinel value to force an update. + if(creatureStats.mBoundItems.empty()) + creatureStats.mBoundItems.insert(ESM::MagicEffect::Length); } // Summoned creature update visitor assumes the actor belongs to a cell. From 7d756d997e94a86f10442e35eae8e88d9bff34ed Mon Sep 17 00:00:00 2001 From: unknown Date: Sat, 12 Jun 2021 18:18:52 +0200 Subject: [PATCH 2/2] Rebuild the cache in readState --- apps/openmw/mwmechanics/actors.cpp | 6 +----- apps/openmw/mwmechanics/creaturestats.cpp | 22 ++++++++++++++++++++++ 2 files changed, 23 insertions(+), 5 deletions(-) diff --git a/apps/openmw/mwmechanics/actors.cpp b/apps/openmw/mwmechanics/actors.cpp index 218d4b4201..87f462d6ff 100644 --- a/apps/openmw/mwmechanics/actors.cpp +++ b/apps/openmw/mwmechanics/actors.cpp @@ -1261,7 +1261,7 @@ namespace MWMechanics { bool found = creatureStats.mBoundItems.find(effect) != creatureStats.mBoundItems.end(); float magnitude = effects.get(effect).getMagnitude(); - if (found != (magnitude > 0) || creatureStats.mBoundItems.empty()) + if (found != (magnitude > 0)) { if (magnitude > 0) creatureStats.mBoundItems.insert(effect); @@ -1281,10 +1281,6 @@ namespace MWMechanics } } } - // mBoundItems does not get saved so the cache is out of date after loading a save. - // Use Length as a sentinel value to force an update. - if(creatureStats.mBoundItems.empty()) - creatureStats.mBoundItems.insert(ESM::MagicEffect::Length); } // Summoned creature update visitor assumes the actor belongs to a cell. diff --git a/apps/openmw/mwmechanics/creaturestats.cpp b/apps/openmw/mwmechanics/creaturestats.cpp index 4bc0b9a581..c6561af961 100644 --- a/apps/openmw/mwmechanics/creaturestats.cpp +++ b/apps/openmw/mwmechanics/creaturestats.cpp @@ -601,6 +601,28 @@ namespace MWMechanics mAiSequence.readState(state.mAiSequence); mMagicEffects.readState(state.mMagicEffects); + // Rebuild the bound item cache + for(int effectId = ESM::MagicEffect::BoundDagger; effectId <= ESM::MagicEffect::BoundGloves; effectId++) + { + if(mMagicEffects.get(effectId).getMagnitude() > 0) + mBoundItems.insert(effectId); + else + { + // Check active spell effects + // We can't use mActiveSpells::getMagicEffects here because it doesn't include expired effects + auto spell = std::find_if(mActiveSpells.begin(), mActiveSpells.end(), [&] (const auto& spell) + { + const auto& effects = spell.second.mEffects; + return std::find_if(effects.begin(), effects.end(), [&] (const auto& effect) + { + return effect.mEffectId == effectId; + }) != effects.end(); + }); + if(spell != mActiveSpells.end()) + mBoundItems.insert(effectId); + } + } + mSummonedCreatures = state.mSummonedCreatureMap; mSummonGraveyard = state.mSummonGraveyard;