2012-05-19 13:01:07 +00:00
|
|
|
#include "activespells.hpp"
|
|
|
|
|
2012-07-03 10:30:50 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
2012-09-13 09:30:59 +00:00
|
|
|
|
2012-05-19 13:01:07 +00:00
|
|
|
namespace MWMechanics
|
|
|
|
{
|
|
|
|
void ActiveSpells::update() const
|
|
|
|
{
|
|
|
|
bool rebuild = false;
|
|
|
|
|
|
|
|
MWWorld::TimeStamp now = MWBase::Environment::get().getWorld()->getTimeStamp();
|
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
// Erase no longer active spells
|
2012-05-19 13:01:07 +00:00
|
|
|
if (mLastUpdate!=now)
|
|
|
|
{
|
|
|
|
TContainer::iterator iter (mSpells.begin());
|
|
|
|
while (iter!=mSpells.end())
|
|
|
|
if (!timeToExpire (iter))
|
|
|
|
{
|
|
|
|
mSpells.erase (iter++);
|
|
|
|
rebuild = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
|
|
|
|
mLastUpdate = now;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mSpellsChanged)
|
|
|
|
{
|
|
|
|
mSpellsChanged = false;
|
|
|
|
rebuild = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rebuild)
|
2012-09-13 09:30:59 +00:00
|
|
|
rebuildEffects();
|
|
|
|
}
|
2012-05-19 13:01:07 +00:00
|
|
|
|
2012-09-13 09:30:59 +00:00
|
|
|
void ActiveSpells::rebuildEffects() const
|
|
|
|
{
|
|
|
|
MWWorld::TimeStamp now = MWBase::Environment::get().getWorld()->getTimeStamp();
|
|
|
|
|
|
|
|
mEffects = MagicEffects();
|
2012-05-19 13:01:07 +00:00
|
|
|
|
2012-09-13 09:30:59 +00:00
|
|
|
for (TIterator iter (begin()); iter!=end(); ++iter)
|
|
|
|
{
|
2013-11-09 06:51:46 +00:00
|
|
|
const MWWorld::TimeStamp& start = iter->second.mTimeStamp;
|
2012-09-13 09:30:59 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
const std::vector<Effect>& effects = iter->second.mEffects;
|
|
|
|
|
|
|
|
for (std::vector<Effect>::const_iterator effectIt = effects.begin(); effectIt != effects.end(); ++effectIt)
|
2012-09-13 09:30:59 +00:00
|
|
|
{
|
2013-11-17 22:15:57 +00:00
|
|
|
int duration = effectIt->mDuration;
|
|
|
|
MWWorld::TimeStamp end = start;
|
|
|
|
end += static_cast<double> (duration)*
|
|
|
|
MWBase::Environment::get().getWorld()->getTimeScaleFactor()/(60*60);
|
2013-11-09 06:51:46 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
if (end>now)
|
|
|
|
mEffects.add(effectIt->mKey, MWMechanics::EffectParam(effectIt->mMagnitude));
|
2012-05-19 13:01:07 +00:00
|
|
|
}
|
2012-09-13 09:30:59 +00:00
|
|
|
}
|
2012-05-19 13:01:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ActiveSpells::ActiveSpells()
|
2013-11-17 22:15:57 +00:00
|
|
|
: mSpellsChanged (false)
|
|
|
|
, mLastUpdate (MWBase::Environment::get().getWorld()->getTimeStamp())
|
2012-05-19 13:01:07 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
const MagicEffects& ActiveSpells::getMagicEffects() const
|
|
|
|
{
|
|
|
|
update();
|
|
|
|
return mEffects;
|
|
|
|
}
|
|
|
|
|
|
|
|
ActiveSpells::TIterator ActiveSpells::begin() const
|
|
|
|
{
|
|
|
|
return mSpells.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
ActiveSpells::TIterator ActiveSpells::end() const
|
|
|
|
{
|
|
|
|
return mSpells.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
double ActiveSpells::timeToExpire (const TIterator& iterator) const
|
|
|
|
{
|
2013-11-17 22:15:57 +00:00
|
|
|
const std::vector<Effect>& effects = iterator->second.mEffects;
|
2012-05-19 13:01:07 +00:00
|
|
|
|
|
|
|
int duration = 0;
|
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
for (std::vector<Effect>::const_iterator iter (effects.begin());
|
|
|
|
iter!=effects.end(); ++iter)
|
2012-05-19 13:01:07 +00:00
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
if (iter->mDuration > duration)
|
|
|
|
duration = iter->mDuration;
|
2012-05-19 13:01:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
double scaledDuration = duration *
|
|
|
|
MWBase::Environment::get().getWorld()->getTimeScaleFactor()/(60*60);
|
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
double usedUp = MWBase::Environment::get().getWorld()->getTimeStamp() - iterator->second.mTimeStamp;
|
2012-05-19 13:01:07 +00:00
|
|
|
|
|
|
|
if (usedUp>=scaledDuration)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return scaledDuration-usedUp;
|
|
|
|
}
|
2012-11-25 00:26:29 +00:00
|
|
|
|
|
|
|
bool ActiveSpells::isSpellActive(std::string id) const
|
|
|
|
{
|
2013-01-09 19:51:52 +00:00
|
|
|
Misc::StringUtils::toLower(id);
|
2012-11-25 00:26:29 +00:00
|
|
|
for (TContainer::iterator iter = mSpells.begin(); iter != mSpells.end(); ++iter)
|
|
|
|
{
|
|
|
|
std::string left = iter->first;
|
2013-01-09 19:51:52 +00:00
|
|
|
Misc::StringUtils::toLower(left);
|
2012-11-25 00:26:29 +00:00
|
|
|
|
|
|
|
if (iter->first == id)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2013-03-03 11:01:19 +00:00
|
|
|
|
|
|
|
const ActiveSpells::TContainer& ActiveSpells::getActiveSpells() const
|
|
|
|
{
|
|
|
|
return mSpells;
|
|
|
|
}
|
2013-11-15 19:29:47 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
void ActiveSpells::addSpell(const std::string &id, bool stack, std::vector<Effect> effects, const std::string &displayName)
|
2013-11-15 19:29:47 +00:00
|
|
|
{
|
2013-11-17 22:15:57 +00:00
|
|
|
bool exists = false;
|
2013-11-15 19:29:47 +00:00
|
|
|
for (TContainer::const_iterator it = begin(); it != end(); ++it)
|
|
|
|
{
|
2013-11-17 22:15:57 +00:00
|
|
|
if (id == it->first)
|
|
|
|
exists = true;
|
|
|
|
}
|
2013-11-15 19:29:47 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
ActiveSpellParams params;
|
|
|
|
params.mTimeStamp = MWBase::Environment::get().getWorld()->getTimeStamp();
|
|
|
|
params.mEffects = effects;
|
|
|
|
params.mDisplayName = displayName;
|
|
|
|
|
|
|
|
if (!exists || stack)
|
|
|
|
mSpells.insert (std::make_pair(id, params));
|
|
|
|
else
|
|
|
|
mSpells.find(id)->second = params;
|
|
|
|
|
|
|
|
mSpellsChanged = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ActiveSpells::visitEffectSources(EffectSourceVisitor &visitor) const
|
|
|
|
{
|
|
|
|
for (TContainer::const_iterator it = begin(); it != end(); ++it)
|
|
|
|
{
|
2013-11-15 19:29:47 +00:00
|
|
|
float timeScale = MWBase::Environment::get().getWorld()->getTimeScaleFactor();
|
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
for (std::vector<Effect>::const_iterator effectIt = it->second.mEffects.begin();
|
|
|
|
effectIt != it->second.mEffects.end(); ++effectIt)
|
2013-11-15 19:29:47 +00:00
|
|
|
{
|
2013-11-17 22:15:57 +00:00
|
|
|
std::string name = it->second.mDisplayName;
|
2013-11-15 19:29:47 +00:00
|
|
|
|
|
|
|
float remainingTime = effectIt->mDuration +
|
|
|
|
(it->second.mTimeStamp - MWBase::Environment::get().getWorld()->getTimeStamp())*3600/timeScale;
|
2013-11-17 22:15:57 +00:00
|
|
|
float magnitude = effectIt->mMagnitude;
|
2013-11-15 19:29:47 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
if (magnitude)
|
|
|
|
visitor.visit(effectIt->mKey, name, magnitude, remainingTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-11-15 19:29:47 +00:00
|
|
|
|
2014-01-02 19:15:07 +00:00
|
|
|
void ActiveSpells::purgeAll(float chance)
|
2013-11-17 22:15:57 +00:00
|
|
|
{
|
2014-01-02 19:15:07 +00:00
|
|
|
for (TContainer::iterator it = mSpells.begin(); it != mSpells.end(); )
|
|
|
|
{
|
|
|
|
int roll = std::rand()/ (static_cast<double> (RAND_MAX) + 1) * 100; // [0, 99]
|
|
|
|
if (roll < chance)
|
|
|
|
mSpells.erase(it++);
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
2014-01-02 19:02:28 +00:00
|
|
|
mSpellsChanged = true;
|
2013-11-17 22:15:57 +00:00
|
|
|
}
|
2013-11-15 19:29:47 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
void ActiveSpells::purgeEffect(short effectId)
|
|
|
|
{
|
|
|
|
for (TContainer::iterator it = mSpells.begin(); it != mSpells.end(); ++it)
|
|
|
|
{
|
|
|
|
for (std::vector<Effect>::iterator effectIt = it->second.mEffects.begin();
|
|
|
|
effectIt != it->second.mEffects.end();)
|
|
|
|
{
|
|
|
|
if (effectIt->mKey.mId == effectId)
|
|
|
|
effectIt = it->second.mEffects.erase(effectIt);
|
|
|
|
else
|
|
|
|
effectIt++;
|
2013-11-15 19:29:47 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-02 19:02:28 +00:00
|
|
|
mSpellsChanged = true;
|
2013-11-15 19:29:47 +00:00
|
|
|
}
|
2012-05-19 13:01:07 +00:00
|
|
|
}
|