|
|
|
|
|
|
|
#include "spells.hpp"
|
|
|
|
|
|
|
|
#include <cstdlib>
|
|
|
|
|
|
|
|
#include <components/esm/loadspel.hpp>
|
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
|
|
|
|
#include "magiceffects.hpp"
|
|
|
|
|
|
|
|
namespace MWMechanics
|
|
|
|
{
|
|
|
|
Spells::TIterator Spells::begin() const
|
|
|
|
{
|
|
|
|
return mSpells.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
Spells::TIterator Spells::end() const
|
|
|
|
{
|
|
|
|
return mSpells.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::add (const std::string& spellId)
|
|
|
|
{
|
|
|
|
if (mSpells.find (spellId)==mSpells.end())
|
|
|
|
{
|
|
|
|
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(spellId);
|
|
|
|
|
|
|
|
std::vector<float> random;
|
|
|
|
random.resize(spell->mEffects.mList.size());
|
|
|
|
for (unsigned int i=0; i<random.size();++i)
|
|
|
|
random[i] = static_cast<float> (std::rand()) / RAND_MAX;
|
|
|
|
mSpells.insert (std::make_pair (Misc::StringUtils::lowerCase(spellId), random));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::remove (const std::string& spellId)
|
|
|
|
{
|
|
|
|
std::string lower = Misc::StringUtils::lowerCase(spellId);
|
|
|
|
TContainer::iterator iter = mSpells.find (lower);
|
|
|
|
|
|
|
|
if (iter!=mSpells.end())
|
|
|
|
mSpells.erase (iter);
|
|
|
|
|
|
|
|
if (spellId==mSelectedSpell)
|
|
|
|
mSelectedSpell.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
MagicEffects Spells::getMagicEffects() const
|
|
|
|
{
|
|
|
|
// TODO: These are recalculated every frame, no need to do that
|
|
|
|
|
|
|
|
MagicEffects effects;
|
|
|
|
|
|
|
|
for (TIterator iter = mSpells.begin(); iter!=mSpells.end(); ++iter)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (spell->mData.mType==ESM::Spell::ST_Ability || spell->mData.mType==ESM::Spell::ST_Blight ||
|
|
|
|
spell->mData.mType==ESM::Spell::ST_Disease || spell->mData.mType==ESM::Spell::ST_Curse)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator it = spell->mEffects.mList.begin(); it != spell->mEffects.mList.end(); ++it)
|
|
|
|
{
|
|
|
|
effects.add (*it, it->mMagnMin + (it->mMagnMax - it->mMagnMin) * iter->second[i]);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return effects;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::clear()
|
|
|
|
{
|
|
|
|
mSpells.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::setSelectedSpell (const std::string& spellId)
|
|
|
|
{
|
|
|
|
mSelectedSpell = spellId;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string Spells::getSelectedSpell() const
|
|
|
|
{
|
|
|
|
return mSelectedSpell;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Spells::hasCommonDisease() const
|
|
|
|
{
|
|
|
|
for (TIterator iter = mSpells.begin(); iter!=mSpells.end(); ++iter)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Disease)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Spells::hasBlightDisease() const
|
|
|
|
{
|
|
|
|
for (TIterator iter = mSpells.begin(); iter!=mSpells.end(); ++iter)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Blight)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::purgeCommonDisease()
|
|
|
|
{
|
|
|
|
for (TContainer::iterator iter = mSpells.begin(); iter!=mSpells.end();)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Disease)
|
|
|
|
mSpells.erase(iter++);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::purgeBlightDisease()
|
|
|
|
{
|
|
|
|
for (TContainer::iterator iter = mSpells.begin(); iter!=mSpells.end();)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Blight)
|
|
|
|
mSpells.erase(iter++);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::purgeCorprusDisease()
|
|
|
|
{
|
|
|
|
for (TContainer::iterator iter = mSpells.begin(); iter!=mSpells.end();)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (Misc::StringUtils::ciEqual(spell->mId, "corprus"))
|
|
|
|
mSpells.erase(iter++);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::purgeCurses()
|
|
|
|
{
|
|
|
|
for (TContainer::iterator iter = mSpells.begin(); iter!=mSpells.end();)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find (iter->first);
|
|
|
|
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Curse)
|
|
|
|
mSpells.erase(iter++);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spells::visitEffectSources(EffectSourceVisitor &visitor) const
|
|
|
|
{
|
|
|
|
for (TIterator it = begin(); it != end(); ++it)
|
|
|
|
{
|
|
|
|
const ESM::Spell* spell = MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(it->first);
|
|
|
|
|
|
|
|
// these are the spell types that are permanently in effect
|
|
|
|
if (!(spell->mData.mType == ESM::Spell::ST_Ability)
|
|
|
|
&& !(spell->mData.mType == ESM::Spell::ST_Disease)
|
|
|
|
&& !(spell->mData.mType == ESM::Spell::ST_Curse)
|
|
|
|
&& !(spell->mData.mType == ESM::Spell::ST_Blight))
|
|
|
|
continue;
|
|
|
|
const ESM::EffectList& list = spell->mEffects;
|
|
|
|
int i=0;
|
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator effectIt = list.mList.begin();
|
|
|
|
effectIt != list.mList.end(); ++effectIt, ++i)
|
|
|
|
{
|
|
|
|
float magnitude = effectIt->mMagnMin + (effectIt->mMagnMax - effectIt->mMagnMin) * it->second[i];
|
|
|
|
visitor.visit(MWMechanics::EffectKey(*effectIt), spell->mName, "", magnitude);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|