From c1d56d94c4b3329b50afd0863f0284ef7753a7bc Mon Sep 17 00:00:00 2001 From: Andrei Kortunov Date: Mon, 2 Nov 2020 17:51:36 +0400 Subject: [PATCH] Do not remove active effects in loop (bug #3789) --- CHANGELOG.md | 1 + apps/openmw/mwmechanics/spellcasting.cpp | 31 +++++++++++++++++++++ apps/openmw/mwmechanics/tickableeffects.cpp | 18 ------------ apps/openmw/mwmechanics/tickableeffects.hpp | 1 + 4 files changed, 33 insertions(+), 18 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 304892ca4..5bda1ff5b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -10,6 +10,7 @@ Bug #2976 [reopened]: Issues combining settings from the command line and both config files Bug #3676: NiParticleColorModifier isn't applied properly Bug #3714: Savegame fails to load due to conflict between SpellState and MagicEffects + Bug #3789: Crash in visitEffectSources while in battle Bug #3862: Random container contents behave differently than vanilla Bug #3929: Leveled list merchant containers respawn on barter Bug #4021: Attributes and skills are not stored as floats diff --git a/apps/openmw/mwmechanics/spellcasting.cpp b/apps/openmw/mwmechanics/spellcasting.cpp index 81b3a353d..beab36cbf 100644 --- a/apps/openmw/mwmechanics/spellcasting.cpp +++ b/apps/openmw/mwmechanics/spellcasting.cpp @@ -381,6 +381,37 @@ namespace MWMechanics target.getClass().getCreatureStats(target).getActiveSpells().purgeAll(magnitude, true); return true; } + else if (target.getClass().isActor() && effectId == ESM::MagicEffect::CurePoison) + { + target.getClass().getCreatureStats(target).getActiveSpells().purgeEffect(ESM::MagicEffect::Poison); + return true; + } + else if (target.getClass().isActor() && effectId == ESM::MagicEffect::CureParalyzation) + { + target.getClass().getCreatureStats(target).getActiveSpells().purgeEffect(ESM::MagicEffect::Paralyze); + return true; + } + else if (target.getClass().isActor() && effectId == ESM::MagicEffect::CureCommonDisease) + { + target.getClass().getCreatureStats(target).getSpells().purgeCommonDisease(); + return true; + } + else if (target.getClass().isActor() && effectId == ESM::MagicEffect::CureBlightDisease) + { + target.getClass().getCreatureStats(target).getSpells().purgeBlightDisease(); + return true; + } + else if (target.getClass().isActor() && effectId == ESM::MagicEffect::CureCorprusDisease) + { + target.getClass().getCreatureStats(target).getActiveSpells().purgeCorprusDisease(); + target.getClass().getCreatureStats(target).getSpells().purgeCorprusDisease(); + return true; + } + else if (target.getClass().isActor() && effectId == ESM::MagicEffect::RemoveCurse) + { + target.getClass().getCreatureStats(target).getSpells().purgeCurses(); + return true; + } else if (target.getClass().isActor() && target == getPlayer()) { MWRender::Animation* anim = MWBase::Environment::get().getWorld()->getAnimation(mCaster); diff --git a/apps/openmw/mwmechanics/tickableeffects.cpp b/apps/openmw/mwmechanics/tickableeffects.cpp index fa3b6ac20..5056179f8 100644 --- a/apps/openmw/mwmechanics/tickableeffects.cpp +++ b/apps/openmw/mwmechanics/tickableeffects.cpp @@ -205,24 +205,6 @@ namespace MWMechanics break; } - case ESM::MagicEffect::CurePoison: - actor.getClass().getCreatureStats(actor).getActiveSpells().purgeEffect(ESM::MagicEffect::Poison); - break; - case ESM::MagicEffect::CureParalyzation: - actor.getClass().getCreatureStats(actor).getActiveSpells().purgeEffect(ESM::MagicEffect::Paralyze); - break; - case ESM::MagicEffect::CureCommonDisease: - actor.getClass().getCreatureStats(actor).getSpells().purgeCommonDisease(); - break; - case ESM::MagicEffect::CureBlightDisease: - actor.getClass().getCreatureStats(actor).getSpells().purgeBlightDisease(); - break; - case ESM::MagicEffect::CureCorprusDisease: - actor.getClass().getCreatureStats(actor).getSpells().purgeCorprusDisease(); - break; - case ESM::MagicEffect::RemoveCurse: - actor.getClass().getCreatureStats(actor).getSpells().purgeCurses(); - break; default: return false; } diff --git a/apps/openmw/mwmechanics/tickableeffects.hpp b/apps/openmw/mwmechanics/tickableeffects.hpp index c4abed6a3..ccd42ca19 100644 --- a/apps/openmw/mwmechanics/tickableeffects.hpp +++ b/apps/openmw/mwmechanics/tickableeffects.hpp @@ -12,6 +12,7 @@ namespace MWMechanics struct EffectKey; /// Apply a magic effect that is applied in tick intervals until its remaining time ends or it is removed + /// Note: this function works in loop, so magic effects should not be removed here to avoid iterator invalidation. /// @return Was the effect a tickable effect with a magnitude? bool effectTick(CreatureStats& creatureStats, const MWWorld::Ptr& actor, const EffectKey& effectKey, float magnitude); }