2012-03-30 16:18:58 +02:00
|
|
|
|
|
|
|
#include "actors.hpp"
|
|
|
|
|
2012-03-30 17:01:55 +02:00
|
|
|
#include <typeinfo>
|
|
|
|
|
2012-07-03 15:32:38 +02:00
|
|
|
#include <OgreVector3.h>
|
|
|
|
|
2012-03-30 17:01:55 +02:00
|
|
|
#include <components/esm/loadnpc.hpp>
|
|
|
|
|
2012-10-01 19:17:04 +04:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2012-09-21 17:53:16 +02:00
|
|
|
|
2012-03-30 16:18:58 +02:00
|
|
|
#include "../mwworld/class.hpp"
|
2012-03-31 17:26:15 +02:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2012-03-30 16:18:58 +02:00
|
|
|
|
2012-09-21 17:53:16 +02:00
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/windowmanager.hpp"
|
|
|
|
|
2012-05-17 13:15:31 +02:00
|
|
|
#include "creaturestats.hpp"
|
2013-03-31 00:13:56 -07:00
|
|
|
#include "movement.hpp"
|
2012-05-17 13:15:31 +02:00
|
|
|
|
2012-03-30 16:18:58 +02:00
|
|
|
namespace MWMechanics
|
|
|
|
{
|
2012-03-30 17:01:55 +02:00
|
|
|
void Actors::updateActor (const MWWorld::Ptr& ptr, float duration)
|
|
|
|
{
|
2012-05-17 13:21:49 +02:00
|
|
|
// magic effects
|
|
|
|
adjustMagicEffects (ptr);
|
2012-07-17 16:44:55 +02:00
|
|
|
calculateDynamicStats (ptr);
|
2012-10-23 13:54:36 +02:00
|
|
|
calculateCreatureStatModifiers (ptr);
|
2012-09-21 17:53:16 +02:00
|
|
|
|
2012-09-04 13:32:35 +02:00
|
|
|
// AI
|
2013-05-26 19:33:45 -07:00
|
|
|
if(!MWBase::Environment::get().getWindowManager()->isGuiMode())
|
2013-05-26 19:54:59 -07:00
|
|
|
{
|
|
|
|
CreatureStats& creatureStats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
2013-05-26 19:33:45 -07:00
|
|
|
creatureStats.getAiSequence().execute (ptr);
|
2013-05-26 19:54:59 -07:00
|
|
|
}
|
2012-03-30 17:01:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actors::updateNpc (const MWWorld::Ptr& ptr, float duration, bool paused)
|
|
|
|
{
|
2012-04-08 13:01:03 +02:00
|
|
|
if (!paused && ptr.getRefData().getHandle()!="player")
|
2013-04-05 15:42:05 +02:00
|
|
|
MWWorld::Class::get (ptr).getInventoryStore (ptr).autoEquip (ptr);
|
2012-03-30 17:01:55 +02:00
|
|
|
}
|
|
|
|
|
2012-05-17 13:21:49 +02:00
|
|
|
void Actors::adjustMagicEffects (const MWWorld::Ptr& creature)
|
|
|
|
{
|
|
|
|
CreatureStats& creatureStats = MWWorld::Class::get (creature).getCreatureStats (creature);
|
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
MagicEffects now = creatureStats.getSpells().getMagicEffects();
|
2012-05-17 13:21:49 +02:00
|
|
|
|
2012-05-18 15:48:55 +02:00
|
|
|
if (creature.getTypeName()==typeid (ESM::NPC).name())
|
|
|
|
{
|
|
|
|
MWWorld::InventoryStore& store = MWWorld::Class::get (creature).getInventoryStore (creature);
|
|
|
|
now += store.getMagicEffects();
|
|
|
|
}
|
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
now += creatureStats.getActiveSpells().getMagicEffects();
|
2012-05-17 13:21:49 +02:00
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
MagicEffects diff = MagicEffects::diff (creatureStats.getMagicEffects(), now);
|
2012-05-17 13:21:49 +02:00
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
creatureStats.setMagicEffects(now);
|
2012-05-17 13:21:49 +02:00
|
|
|
|
|
|
|
// TODO apply diff to other stats
|
|
|
|
}
|
|
|
|
|
2012-07-17 12:18:43 +02:00
|
|
|
void Actors::calculateDynamicStats (const MWWorld::Ptr& ptr)
|
|
|
|
{
|
|
|
|
CreatureStats& creatureStats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
int strength = creatureStats.getAttribute(0).getBase();
|
|
|
|
int intelligence = creatureStats.getAttribute(1).getBase();
|
|
|
|
int willpower = creatureStats.getAttribute(2).getBase();
|
|
|
|
int agility = creatureStats.getAttribute(3).getBase();
|
|
|
|
int endurance = creatureStats.getAttribute(5).getBase();
|
2012-07-17 12:18:43 +02:00
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
double magickaFactor =
|
2013-03-16 23:28:26 +01:00
|
|
|
creatureStats.getMagicEffects().get (EffectKey (ESM::MagicEffect::FortifyMaximumMagicka)).mMagnitude * 0.1 + 0.5;
|
2012-07-17 12:18:43 +02:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
DynamicStat<float> magicka = creatureStats.getMagicka();
|
|
|
|
magicka.setBase (static_cast<int> (intelligence + magickaFactor * intelligence));
|
|
|
|
creatureStats.setMagicka (magicka);
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
DynamicStat<float> fatigue = creatureStats.getFatigue();
|
|
|
|
fatigue.setBase (strength+willpower+agility+endurance);
|
|
|
|
creatureStats.setFatigue (fatigue);
|
2012-07-17 12:18:43 +02:00
|
|
|
}
|
|
|
|
|
2012-09-21 17:53:16 +02:00
|
|
|
void Actors::calculateRestoration (const MWWorld::Ptr& ptr, float duration)
|
|
|
|
{
|
|
|
|
CreatureStats& stats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
|
|
|
|
|
|
if (duration == 3600)
|
|
|
|
{
|
2013-03-16 23:28:26 +01:00
|
|
|
bool stunted = stats.getMagicEffects ().get(MWMechanics::EffectKey(ESM::MagicEffect::StuntedMagicka)).mMagnitude > 0;
|
2012-09-21 17:53:16 +02:00
|
|
|
|
|
|
|
int endurance = stats.getAttribute (ESM::Attribute::Endurance).getModified ();
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
DynamicStat<float> health = stats.getHealth();
|
|
|
|
health.setCurrent (health.getCurrent() + 0.1 * endurance);
|
|
|
|
stats.setHealth (health);
|
2012-09-21 17:53:16 +02:00
|
|
|
|
2012-10-01 19:17:04 +04:00
|
|
|
const MWWorld::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
2012-09-21 17:53:16 +02:00
|
|
|
|
2012-11-05 21:19:22 +04:00
|
|
|
float fFatigueReturnBase = store.get<ESM::GameSetting>().find("fFatigueReturnBase")->getFloat ();
|
|
|
|
float fFatigueReturnMult = store.get<ESM::GameSetting>().find("fFatigueReturnMult")->getFloat ();
|
|
|
|
float fEndFatigueMult = store.get<ESM::GameSetting>().find("fEndFatigueMult")->getFloat ();
|
2012-09-21 17:53:16 +02:00
|
|
|
|
|
|
|
float capacity = MWWorld::Class::get(ptr).getCapacity(ptr);
|
|
|
|
float encumbrance = MWWorld::Class::get(ptr).getEncumbrance(ptr);
|
|
|
|
float normalizedEncumbrance = (capacity == 0 ? 1 : encumbrance/capacity);
|
2012-09-21 17:56:15 +02:00
|
|
|
if (normalizedEncumbrance > 1)
|
|
|
|
normalizedEncumbrance = 1;
|
2012-09-21 17:53:16 +02:00
|
|
|
|
|
|
|
float x = fFatigueReturnBase + fFatigueReturnMult * (1 - normalizedEncumbrance);
|
|
|
|
x *= fEndFatigueMult * endurance;
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
DynamicStat<float> fatigue = stats.getFatigue();
|
|
|
|
fatigue.setCurrent (fatigue.getCurrent() + 3600 * x);
|
|
|
|
stats.setFatigue (fatigue);
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-09-21 17:53:16 +02:00
|
|
|
if (!stunted)
|
|
|
|
{
|
2012-11-05 21:19:22 +04:00
|
|
|
float fRestMagicMult = store.get<ESM::GameSetting>().find("fRestMagicMult")->getFloat ();
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
DynamicStat<float> magicka = stats.getMagicka();
|
|
|
|
magicka.setCurrent (magicka.getCurrent()
|
|
|
|
+ fRestMagicMult * stats.getAttribute(ESM::Attribute::Intelligence).getModified());
|
|
|
|
stats.setMagicka (magicka);
|
2012-09-21 17:53:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-17 15:49:37 +02:00
|
|
|
void Actors::calculateCreatureStatModifiers (const MWWorld::Ptr& ptr)
|
|
|
|
{
|
|
|
|
CreatureStats& creatureStats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
|
|
|
|
|
|
// attributes
|
2012-10-19 10:07:27 +02:00
|
|
|
for (int i=0; i<8; ++i)
|
2012-07-17 15:49:37 +02:00
|
|
|
{
|
2012-07-22 18:29:54 +04:00
|
|
|
int modifier =
|
2013-03-16 23:28:26 +01:00
|
|
|
creatureStats.getMagicEffects().get (EffectKey (ESM::MagicEffect::FortifyAttribute, i)).mMagnitude;
|
2012-07-22 18:29:54 +04:00
|
|
|
|
2013-03-16 23:28:26 +01:00
|
|
|
modifier -= creatureStats.getMagicEffects().get (EffectKey (ESM::MagicEffect::DrainAttribute, i)).mMagnitude;
|
2012-07-17 15:49:37 +02:00
|
|
|
|
2012-07-22 18:29:54 +04:00
|
|
|
creatureStats.getAttribute(i).setModifier (modifier);
|
2012-07-17 15:49:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// dynamic stats
|
2012-07-22 18:29:54 +04:00
|
|
|
MagicEffects effects = creatureStats.getMagicEffects();
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
for (int i=0; i<3; ++i)
|
|
|
|
{
|
|
|
|
DynamicStat<float> stat = creatureStats.getDynamic (i);
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
stat.setModifier (
|
|
|
|
effects.get (EffectKey(80+i)).mMagnitude - effects.get (EffectKey(18+i)).mMagnitude);
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-19 13:10:06 +02:00
|
|
|
creatureStats.setDynamic (i, stat);
|
2013-03-18 10:54:47 +01:00
|
|
|
}
|
2012-07-17 15:49:37 +02:00
|
|
|
}
|
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
Actors::Actors() : mDuration (0) {}
|
2012-03-30 16:18:58 +02:00
|
|
|
|
|
|
|
void Actors::addActor (const MWWorld::Ptr& ptr)
|
|
|
|
{
|
2013-03-18 10:54:47 +01:00
|
|
|
// erase previous death events since we are currently only tracking them while in an active cell
|
2013-07-15 23:43:33 -07:00
|
|
|
MWWorld::Class::get(ptr).getCreatureStats(ptr).clearHasDied();
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2013-07-17 21:39:21 -07:00
|
|
|
removeActor(ptr);
|
|
|
|
|
2013-01-16 10:16:37 -08:00
|
|
|
MWRender::Animation *anim = MWBase::Environment::get().getWorld()->getAnimation(ptr);
|
2013-07-15 23:43:33 -07:00
|
|
|
mActors.insert(std::make_pair(ptr, new CharacterController(ptr, anim)));
|
2012-03-30 16:18:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actors::removeActor (const MWWorld::Ptr& ptr)
|
|
|
|
{
|
2013-01-12 07:12:12 -08:00
|
|
|
PtrControllerMap::iterator iter = mActors.find(ptr);
|
|
|
|
if(iter != mActors.end())
|
2013-07-08 14:05:53 -07:00
|
|
|
{
|
|
|
|
delete iter->second;
|
2013-01-12 07:12:12 -08:00
|
|
|
mActors.erase(iter);
|
2013-07-08 14:05:53 -07:00
|
|
|
}
|
2012-03-30 16:18:58 +02:00
|
|
|
}
|
|
|
|
|
2013-02-25 09:57:34 -08:00
|
|
|
void Actors::updateActor(const MWWorld::Ptr &old, const MWWorld::Ptr &ptr)
|
2013-01-29 00:19:24 -08:00
|
|
|
{
|
2013-02-25 09:57:34 -08:00
|
|
|
PtrControllerMap::iterator iter = mActors.find(old);
|
2013-01-29 00:19:24 -08:00
|
|
|
if(iter != mActors.end())
|
|
|
|
{
|
2013-07-08 14:05:53 -07:00
|
|
|
CharacterController *ctrl = iter->second;
|
2013-01-29 00:19:24 -08:00
|
|
|
mActors.erase(iter);
|
2013-02-25 09:57:34 -08:00
|
|
|
|
2013-07-08 14:05:53 -07:00
|
|
|
ctrl->updatePtr(ptr);
|
2013-01-29 00:19:24 -08:00
|
|
|
mActors.insert(std::make_pair(ptr, ctrl));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-30 16:18:58 +02:00
|
|
|
void Actors::dropActors (const MWWorld::Ptr::CellStore *cellStore)
|
|
|
|
{
|
2013-01-12 07:12:12 -08:00
|
|
|
PtrControllerMap::iterator iter = mActors.begin();
|
|
|
|
while(iter != mActors.end())
|
|
|
|
{
|
|
|
|
if(iter->first.getCell()==cellStore)
|
2013-07-08 14:05:53 -07:00
|
|
|
{
|
|
|
|
delete iter->second;
|
2013-01-12 07:12:12 -08:00
|
|
|
mActors.erase(iter++);
|
2013-07-08 14:05:53 -07:00
|
|
|
}
|
2012-03-30 16:18:58 +02:00
|
|
|
else
|
|
|
|
++iter;
|
2013-01-12 07:12:12 -08:00
|
|
|
}
|
2012-03-30 16:18:58 +02:00
|
|
|
}
|
|
|
|
|
2013-01-16 08:22:38 -08:00
|
|
|
void Actors::update (float duration, bool paused)
|
2012-03-30 16:18:58 +02:00
|
|
|
{
|
2012-03-30 17:01:55 +02:00
|
|
|
mDuration += duration;
|
|
|
|
|
2013-04-11 17:57:58 +01:00
|
|
|
//if (mDuration>=0.25)
|
2012-03-30 17:01:55 +02:00
|
|
|
{
|
2012-07-17 18:37:20 +02:00
|
|
|
float totalDuration = mDuration;
|
|
|
|
mDuration = 0;
|
|
|
|
|
2013-02-02 23:39:43 -08:00
|
|
|
for(PtrControllerMap::iterator iter(mActors.begin());iter != mActors.end();iter++)
|
2012-03-30 17:01:55 +02:00
|
|
|
{
|
2013-07-26 08:08:52 -07:00
|
|
|
const MWWorld::Class &cls = MWWorld::Class::get(iter->first);
|
|
|
|
CreatureStats &stats = cls.getCreatureStats(iter->first);
|
|
|
|
|
|
|
|
stats.setLastHitObject(std::string());
|
|
|
|
if(!stats.isDead())
|
2012-10-20 10:54:51 +02:00
|
|
|
{
|
2013-07-15 23:43:33 -07:00
|
|
|
if(iter->second->isDead())
|
|
|
|
iter->second->resurrect();
|
2013-01-12 10:10:27 -08:00
|
|
|
|
2013-01-12 07:12:12 -08:00
|
|
|
updateActor(iter->first, totalDuration);
|
|
|
|
if(iter->first.getTypeName() == typeid(ESM::NPC).name())
|
|
|
|
updateNpc(iter->first, totalDuration, paused);
|
2012-10-20 10:54:51 +02:00
|
|
|
|
2013-07-26 08:08:52 -07:00
|
|
|
if(!stats.isDead())
|
2013-01-12 07:12:12 -08:00
|
|
|
continue;
|
2012-10-20 10:54:51 +02:00
|
|
|
}
|
|
|
|
|
2013-01-12 07:12:12 -08:00
|
|
|
// workaround: always keep player alive for now
|
|
|
|
// \todo remove workaround, once player death can be handled
|
|
|
|
if(iter->first.getRefData().getHandle()=="player")
|
2012-10-25 12:28:45 +02:00
|
|
|
{
|
2013-07-26 08:08:52 -07:00
|
|
|
MWMechanics::DynamicStat<float> stat(stats.getHealth());
|
2013-01-12 07:12:12 -08:00
|
|
|
|
2013-07-26 08:08:52 -07:00
|
|
|
if(stat.getModified()<1)
|
2012-10-25 12:28:45 +02:00
|
|
|
{
|
2013-07-26 08:08:52 -07:00
|
|
|
stat.setModified(1, 0);
|
|
|
|
stats.setHealth(stat);
|
2012-10-25 12:28:45 +02:00
|
|
|
}
|
2012-10-27 10:36:42 +02:00
|
|
|
|
2013-07-26 08:08:52 -07:00
|
|
|
stats.resurrect();
|
2013-01-12 07:12:12 -08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-07-15 23:43:33 -07:00
|
|
|
if(iter->second->isDead())
|
2013-01-12 10:10:27 -08:00
|
|
|
continue;
|
2012-10-27 11:15:52 +02:00
|
|
|
|
2013-07-15 23:43:33 -07:00
|
|
|
iter->second->kill();
|
2012-10-27 10:36:42 +02:00
|
|
|
|
2013-07-26 08:08:52 -07:00
|
|
|
++mDeathCount[cls.getId(iter->first)];
|
2013-01-12 10:10:27 -08:00
|
|
|
|
2013-07-26 08:08:52 -07:00
|
|
|
if(cls.isEssential(iter->first))
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sKilledEssential}");
|
2012-03-30 17:01:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-16 18:03:39 -08:00
|
|
|
if(!paused)
|
2012-03-30 16:18:58 +02:00
|
|
|
{
|
2013-02-05 06:59:01 -08:00
|
|
|
mMovement.reserve(mActors.size());
|
|
|
|
|
2013-01-16 18:03:39 -08:00
|
|
|
for(PtrControllerMap::iterator iter(mActors.begin());iter != mActors.end();++iter)
|
|
|
|
{
|
2013-03-31 00:13:56 -07:00
|
|
|
Movement movement;
|
2013-07-08 14:05:53 -07:00
|
|
|
iter->second->update(duration, movement);
|
2013-02-05 06:59:01 -08:00
|
|
|
mMovement.push_back(std::make_pair(iter->first, movement));
|
2013-01-16 18:03:39 -08:00
|
|
|
}
|
2013-02-05 12:45:10 -08:00
|
|
|
MWBase::Environment::get().getWorld()->doPhysics(mMovement, duration);
|
2013-02-05 06:59:01 -08:00
|
|
|
|
|
|
|
mMovement.clear();
|
2013-01-16 18:03:39 -08:00
|
|
|
}
|
2012-03-30 16:18:58 +02:00
|
|
|
}
|
2012-09-21 17:53:16 +02:00
|
|
|
|
|
|
|
void Actors::restoreDynamicStats()
|
|
|
|
{
|
2013-01-12 07:12:12 -08:00
|
|
|
for(PtrControllerMap::iterator iter(mActors.begin());iter != mActors.end();++iter)
|
2013-01-28 23:39:11 -08:00
|
|
|
calculateRestoration(iter->first, 3600);
|
2012-09-21 17:53:16 +02:00
|
|
|
}
|
2013-03-18 10:54:47 +01:00
|
|
|
|
2012-10-27 11:33:18 +02:00
|
|
|
int Actors::countDeaths (const std::string& id) const
|
|
|
|
{
|
2013-01-12 07:12:12 -08:00
|
|
|
std::map<std::string, int>::const_iterator iter = mDeathCount.find(id);
|
|
|
|
if(iter != mDeathCount.end())
|
2012-10-27 11:33:18 +02:00
|
|
|
return iter->second;
|
|
|
|
return 0;
|
|
|
|
}
|
2013-01-16 17:53:18 -08:00
|
|
|
|
2013-04-25 07:08:11 -07:00
|
|
|
void Actors::forceStateUpdate(const MWWorld::Ptr & ptr)
|
|
|
|
{
|
|
|
|
PtrControllerMap::iterator iter = mActors.find(ptr);
|
|
|
|
if(iter != mActors.end())
|
2013-07-08 14:05:53 -07:00
|
|
|
iter->second->forceStateUpdate();
|
2013-04-25 07:08:11 -07:00
|
|
|
}
|
|
|
|
|
2013-01-16 17:53:18 -08:00
|
|
|
void Actors::playAnimationGroup(const MWWorld::Ptr& ptr, const std::string& groupName, int mode, int number)
|
|
|
|
{
|
|
|
|
PtrControllerMap::iterator iter = mActors.find(ptr);
|
|
|
|
if(iter != mActors.end())
|
2013-07-08 14:05:53 -07:00
|
|
|
iter->second->playGroup(groupName, mode, number);
|
2013-01-16 17:53:18 -08:00
|
|
|
}
|
|
|
|
void Actors::skipAnimation(const MWWorld::Ptr& ptr)
|
|
|
|
{
|
|
|
|
PtrControllerMap::iterator iter = mActors.find(ptr);
|
|
|
|
if(iter != mActors.end())
|
2013-07-08 14:05:53 -07:00
|
|
|
iter->second->skipAnim();
|
2013-01-16 17:53:18 -08:00
|
|
|
}
|
2013-05-24 20:10:07 -07:00
|
|
|
|
|
|
|
bool Actors::checkAnimationPlaying(const MWWorld::Ptr& ptr, const std::string& groupName)
|
|
|
|
{
|
|
|
|
PtrControllerMap::iterator iter = mActors.find(ptr);
|
|
|
|
if(iter != mActors.end())
|
2013-07-08 14:05:53 -07:00
|
|
|
return iter->second->isAnimPlaying(groupName);
|
2013-05-24 20:10:07 -07:00
|
|
|
return false;
|
|
|
|
}
|
2012-03-30 16:18:58 +02:00
|
|
|
}
|