Graceful error handling for missing spells/factions (Fixes #1825, Bug #2176, Bug #2203)

moveref
scrawl 10 years ago
parent 8f29f2667e
commit 88a2e4c043

@ -119,7 +119,13 @@ namespace MWClass
// spells // spells
for (std::vector<std::string>::const_iterator iter (ref->mBase->mSpells.mList.begin()); for (std::vector<std::string>::const_iterator iter (ref->mBase->mSpells.mList.begin());
iter!=ref->mBase->mSpells.mList.end(); ++iter) iter!=ref->mBase->mSpells.mList.end(); ++iter)
data->mCreatureStats.getSpells().add (*iter); {
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().search(*iter);
if (spell)
data->mCreatureStats.getSpells().add (spell);
else /// \todo add option to make this a fatal error message pop-up, but default to warning for vanilla compatibility
std::cerr << "Warning: ignoring nonexistent spell '" << spell->mId << "' on creature '" << ref->mBase->mId << "'" << std::endl;
}
// inventory // inventory
if (ref->mBase->mFlags & ESM::Creature::Weapon) if (ref->mBase->mFlags & ESM::Creature::Weapon)

@ -300,15 +300,20 @@ namespace MWClass
if (!ref->mBase->mFaction.empty()) if (!ref->mBase->mFaction.empty())
{ {
std::string faction = ref->mBase->mFaction; std::string faction = ref->mBase->mFaction;
Misc::StringUtils::toLower(faction); const ESM::Faction* fact = MWBase::Environment::get().getWorld()->getStore().get<ESM::Faction>().search(faction);
if(ref->mBase->mNpdtType != ESM::NPC::NPC_WITH_AUTOCALCULATED_STATS) if (fact)
{ {
data->mNpcStats.setFactionRank(faction, (int)ref->mBase->mNpdt52.mRank); if(ref->mBase->mNpdtType != ESM::NPC::NPC_WITH_AUTOCALCULATED_STATS)
{
data->mNpcStats.setFactionRank(fact->mId, (int)ref->mBase->mNpdt52.mRank);
}
else
{
data->mNpcStats.setFactionRank(fact->mId, (int)ref->mBase->mNpdt12.mRank);
}
} }
else else
{ std::cerr << "Warning: ignoring nonexistent faction '" << fact->mId << "' on NPC '" << ref->mBase->mId << "'" << std::endl;
data->mNpcStats.setFactionRank(faction, (int)ref->mBase->mNpdt12.mRank);
}
} }
// creature stats // creature stats
@ -361,7 +366,11 @@ namespace MWClass
for (std::vector<std::string>::const_iterator iter (race->mPowers.mList.begin()); for (std::vector<std::string>::const_iterator iter (race->mPowers.mList.begin());
iter!=race->mPowers.mList.end(); ++iter) iter!=race->mPowers.mList.end(); ++iter)
{ {
data->mNpcStats.getSpells().add (*iter); const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().search(*iter);
if (spell)
data->mNpcStats.getSpells().add (spell);
else
std::cerr << "Warning: ignoring nonexistent race power '" << *iter << "' on NPC '" << ref->mBase->mId << "'" << std::endl;
} }
if (data->mNpcStats.getFactionRanks().size()) if (data->mNpcStats.getFactionRanks().size())
@ -385,7 +394,16 @@ namespace MWClass
// spells // spells
for (std::vector<std::string>::const_iterator iter (ref->mBase->mSpells.mList.begin()); for (std::vector<std::string>::const_iterator iter (ref->mBase->mSpells.mList.begin());
iter!=ref->mBase->mSpells.mList.end(); ++iter) iter!=ref->mBase->mSpells.mList.end(); ++iter)
data->mNpcStats.getSpells().add (*iter); {
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().search(*iter);
if (spell)
data->mNpcStats.getSpells().add (spell);
else
{
/// \todo add option to make this a fatal error message pop-up, but default to warning for vanilla compatibility
std::cerr << "Warning: ignoring nonexistent spell '" << *iter << "' on NPC '" << ref->mBase->mId << "'" << std::endl;
}
}
// inventory // inventory
data->mInventoryStore.fill(ref->mBase->mInventory, getId(ptr), "", data->mInventoryStore.fill(ref->mBase->mInventory, getId(ptr), "",

@ -25,12 +25,10 @@ namespace MWMechanics
return mSpells.end(); return mSpells.end();
} }
void Spells::add (const std::string& spellId) void Spells::add (const ESM::Spell* spell)
{ {
if (mSpells.find (spellId)==mSpells.end()) if (mSpells.find (spell->mId)==mSpells.end())
{ {
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(spellId);
std::map<const int, float> random; std::map<const int, float> random;
// Determine the random magnitudes (unless this is a castable spell, in which case // Determine the random magnitudes (unless this is a castable spell, in which case
@ -50,13 +48,19 @@ namespace MWMechanics
corprus.mWorsenings = 0; corprus.mWorsenings = 0;
corprus.mNextWorsening = MWBase::Environment::get().getWorld()->getTimeStamp() + CorprusStats::sWorseningPeriod; corprus.mNextWorsening = MWBase::Environment::get().getWorld()->getTimeStamp() + CorprusStats::sWorseningPeriod;
mCorprusSpells[spellId] = corprus; mCorprusSpells[spell->mId] = corprus;
} }
mSpells.insert (std::make_pair (Misc::StringUtils::lowerCase(spellId), random)); mSpells.insert (std::make_pair (spell->mId, random));
} }
} }
void Spells::add (const std::string& spellId)
{
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(spellId);
add(spell);
}
void Spells::remove (const std::string& spellId) void Spells::remove (const std::string& spellId)
{ {
std::string lower = Misc::StringUtils::lowerCase(spellId); std::string lower = Misc::StringUtils::lowerCase(spellId);

@ -79,6 +79,9 @@ namespace MWMechanics
void add (const std::string& spell); void add (const std::string& spell);
///< Adding a spell that is already listed in *this is a no-op. ///< Adding a spell that is already listed in *this is a no-op.
void add (const ESM::Spell* spell);
///< Adding a spell that is already listed in *this is a no-op.
void remove (const std::string& spell); void remove (const std::string& spell);
///< If the spell to be removed is the selected spell, the selected spell will be changed to ///< If the spell to be removed is the selected spell, the selected spell will be changed to
/// no spell (empty string). /// no spell (empty string).

Loading…
Cancel
Save