2017-04-08 05:59:21 +00:00
|
|
|
#include <components/openmw-mp/Log.hpp>
|
|
|
|
|
2017-04-06 12:11:51 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
2017-04-16 13:42:07 +00:00
|
|
|
|
2017-05-26 01:37:49 +00:00
|
|
|
#include "../mwmechanics/mechanicsmanagerimp.hpp"
|
2017-04-06 12:11:51 +00:00
|
|
|
#include "../mwmechanics/movement.hpp"
|
2017-04-16 13:42:07 +00:00
|
|
|
|
2017-04-06 12:11:51 +00:00
|
|
|
#include "../mwrender/animation.hpp"
|
2017-04-16 13:42:07 +00:00
|
|
|
|
2017-04-06 12:11:51 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
2017-05-26 01:37:49 +00:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2017-04-06 12:11:51 +00:00
|
|
|
#include "../mwworld/worldimp.hpp"
|
2017-04-06 01:00:50 +00:00
|
|
|
|
2017-04-05 09:52:27 +00:00
|
|
|
#include "LocalActor.hpp"
|
2017-04-10 14:10:18 +00:00
|
|
|
#include "Main.hpp"
|
|
|
|
#include "Networking.hpp"
|
|
|
|
#include "ActorList.hpp"
|
2017-04-19 19:06:04 +00:00
|
|
|
#include "MechanicsHelper.hpp"
|
2017-04-05 09:52:27 +00:00
|
|
|
|
|
|
|
using namespace mwmp;
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
LocalActor::LocalActor()
|
|
|
|
{
|
2017-04-10 14:10:18 +00:00
|
|
|
posWasChanged = false;
|
2017-05-26 01:37:49 +00:00
|
|
|
equipmentChanged = false;
|
2017-04-13 12:26:48 +00:00
|
|
|
|
|
|
|
wasRunning = false;
|
|
|
|
wasSneaking = false;
|
|
|
|
wasForceJumping = false;
|
|
|
|
wasForceMoveJumping = false;
|
|
|
|
wasFlying = false;
|
2017-04-16 13:42:07 +00:00
|
|
|
|
2017-05-05 21:33:48 +00:00
|
|
|
positionTimer = 0;
|
2017-04-16 13:42:07 +00:00
|
|
|
statTimer = 0;
|
|
|
|
|
2017-04-19 19:06:04 +00:00
|
|
|
attack.type = Attack::MELEE;
|
|
|
|
attack.shouldSend = false;
|
2017-05-09 11:19:25 +00:00
|
|
|
|
|
|
|
creatureStats.mDead = false;
|
2017-04-05 09:52:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LocalActor::~LocalActor()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-04-10 14:10:18 +00:00
|
|
|
void LocalActor::update(bool forceUpdate)
|
2017-04-05 09:52:27 +00:00
|
|
|
{
|
2017-04-16 13:42:07 +00:00
|
|
|
updateStatsDynamic(forceUpdate);
|
2017-05-26 01:37:49 +00:00
|
|
|
updateEquipment(forceUpdate);
|
2017-05-08 23:47:29 +00:00
|
|
|
|
|
|
|
if (forceUpdate || !creatureStats.mDead)
|
|
|
|
{
|
|
|
|
updatePosition(forceUpdate);
|
|
|
|
updateAnimFlags(forceUpdate);
|
|
|
|
updateAnimPlay();
|
|
|
|
updateSpeech();
|
|
|
|
updateAttack();
|
|
|
|
}
|
2017-04-06 12:11:51 +00:00
|
|
|
}
|
|
|
|
|
2017-04-23 10:59:15 +00:00
|
|
|
void LocalActor::updateCell()
|
|
|
|
{
|
2017-05-05 01:46:40 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_ACTOR_CELL_CHANGE about %s-%i-%i to server",
|
2017-04-23 10:59:15 +00:00
|
|
|
refId.c_str(), refNumIndex, mpNum);
|
|
|
|
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Moved from %s to %s", cell.getDescription().c_str(), ptr.getCell()->getCell()->getDescription().c_str());
|
|
|
|
|
|
|
|
cell = *ptr.getCell()->getCell();
|
2017-05-27 22:11:32 +00:00
|
|
|
position = ptr.getRefData().getPosition();
|
2017-04-23 10:59:15 +00:00
|
|
|
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addCellChangeActor(*this);
|
|
|
|
}
|
|
|
|
|
2017-04-10 14:10:18 +00:00
|
|
|
void LocalActor::updatePosition(bool forceUpdate)
|
2017-04-06 12:11:51 +00:00
|
|
|
{
|
2017-05-05 21:33:48 +00:00
|
|
|
const float timeoutSec = 0.03;
|
2017-04-10 14:10:18 +00:00
|
|
|
|
2017-05-05 21:33:48 +00:00
|
|
|
if (forceUpdate || (positionTimer += MWBase::Environment::get().getFrameDuration()) >= timeoutSec)
|
2017-04-10 14:10:18 +00:00
|
|
|
{
|
2017-05-05 21:33:48 +00:00
|
|
|
bool posIsChanging = (direction.pos[0] != 0 || direction.pos[1] != 0 || direction.pos[2] != 0 ||
|
|
|
|
direction.rot[0] != 0 || direction.rot[1] != 0 || direction.rot[2] != 0);
|
2017-04-10 14:10:18 +00:00
|
|
|
|
2017-05-05 21:33:48 +00:00
|
|
|
if (forceUpdate || posIsChanging || posWasChanged)
|
|
|
|
{
|
|
|
|
posWasChanged = posIsChanging;
|
|
|
|
|
|
|
|
position = ptr.getRefData().getPosition();
|
2017-04-10 14:10:18 +00:00
|
|
|
|
2017-05-05 21:33:48 +00:00
|
|
|
positionTimer = 0;
|
|
|
|
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addPositionActor(*this);
|
|
|
|
}
|
2017-04-10 14:10:18 +00:00
|
|
|
}
|
2017-04-06 12:11:51 +00:00
|
|
|
}
|
|
|
|
|
2017-04-14 13:00:34 +00:00
|
|
|
void LocalActor::updateAnimFlags(bool forceUpdate)
|
2017-04-06 12:11:51 +00:00
|
|
|
{
|
2017-04-13 12:26:48 +00:00
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-04-24 08:58:39 +00:00
|
|
|
MWMechanics::CreatureStats ptrCreatureStats = ptr.getClass().getCreatureStats(ptr);
|
2017-04-13 12:26:48 +00:00
|
|
|
|
|
|
|
using namespace MWMechanics;
|
|
|
|
|
2017-04-24 08:58:39 +00:00
|
|
|
bool isRunning = ptrCreatureStats.getMovementFlag(CreatureStats::Flag_Run);
|
|
|
|
bool isSneaking = ptrCreatureStats.getMovementFlag(CreatureStats::Flag_Sneak);
|
|
|
|
bool isForceJumping = ptrCreatureStats.getMovementFlag(CreatureStats::Flag_ForceJump);
|
|
|
|
bool isForceMoveJumping = ptrCreatureStats.getMovementFlag(CreatureStats::Flag_ForceMoveJump);
|
2017-04-13 12:26:48 +00:00
|
|
|
|
|
|
|
isFlying = world->isFlying(ptr);
|
|
|
|
|
2017-04-24 08:58:39 +00:00
|
|
|
MWMechanics::DrawState_ currentDrawState = ptr.getClass().getCreatureStats(ptr).getDrawState();
|
2017-04-13 12:26:48 +00:00
|
|
|
|
2017-05-09 09:27:59 +00:00
|
|
|
if (wasRunning != isRunning || wasSneaking != isSneaking ||
|
|
|
|
wasForceJumping != isForceJumping || wasForceMoveJumping != isForceMoveJumping ||
|
|
|
|
lastDrawState != currentDrawState || wasFlying != isFlying ||
|
2017-04-24 07:58:39 +00:00
|
|
|
forceUpdate)
|
2017-04-13 12:26:48 +00:00
|
|
|
{
|
2017-05-09 09:27:59 +00:00
|
|
|
|
2017-04-13 12:26:48 +00:00
|
|
|
wasRunning = isRunning;
|
|
|
|
wasSneaking = isSneaking;
|
|
|
|
wasForceJumping = isForceJumping;
|
|
|
|
wasForceMoveJumping = isForceMoveJumping;
|
|
|
|
lastDrawState = currentDrawState;
|
|
|
|
|
|
|
|
wasFlying = isFlying;
|
|
|
|
|
|
|
|
movementFlags = 0;
|
|
|
|
|
|
|
|
#define __SETFLAG(flag, value) (value) ? (movementFlags | flag) : (movementFlags & ~flag)
|
|
|
|
|
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_Sneak, isSneaking);
|
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_Run, isRunning);
|
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_ForceJump, isForceJumping);
|
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_ForceMoveJump, isForceMoveJumping);
|
|
|
|
|
|
|
|
#undef __SETFLAG
|
|
|
|
|
2017-05-06 01:49:49 +00:00
|
|
|
drawState = currentDrawState;
|
2017-04-13 12:26:48 +00:00
|
|
|
|
2017-04-15 08:45:20 +00:00
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addAnimFlagsActor(*this);
|
2017-04-13 12:26:48 +00:00
|
|
|
}
|
2017-04-06 12:11:51 +00:00
|
|
|
}
|
|
|
|
|
2017-04-15 10:42:30 +00:00
|
|
|
void LocalActor::updateAnimPlay()
|
|
|
|
{
|
|
|
|
if (!animation.groupname.empty())
|
|
|
|
{
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addAnimPlayActor(*this);
|
|
|
|
animation.groupname.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-16 15:43:13 +00:00
|
|
|
void LocalActor::updateSpeech()
|
|
|
|
{
|
|
|
|
if (!sound.empty())
|
|
|
|
{
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addSpeechActor(*this);
|
|
|
|
sound.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-16 13:42:07 +00:00
|
|
|
void LocalActor::updateStatsDynamic(bool forceUpdate)
|
|
|
|
{
|
|
|
|
MWMechanics::CreatureStats *ptrCreatureStats = &ptr.getClass().getCreatureStats(ptr);
|
|
|
|
MWMechanics::DynamicStat<float> health(ptrCreatureStats->getHealth());
|
|
|
|
MWMechanics::DynamicStat<float> magicka(ptrCreatureStats->getMagicka());
|
|
|
|
MWMechanics::DynamicStat<float> fatigue(ptrCreatureStats->getFatigue());
|
|
|
|
|
|
|
|
const float timeoutSec = 0.5;
|
|
|
|
|
2017-05-05 20:17:15 +00:00
|
|
|
if (forceUpdate || (statTimer += MWBase::Environment::get().getFrameDuration()) >= timeoutSec)
|
2017-04-16 13:42:07 +00:00
|
|
|
{
|
2017-05-05 20:17:15 +00:00
|
|
|
// Update stats when they become 0 or they have changed enough
|
|
|
|
//
|
|
|
|
// Also check for an oldHealth of 0 changing to something else for resurrected NPCs
|
|
|
|
bool shouldUpdateHealth = oldHealth != health && (health.getCurrent() == 0 || oldHealth.getCurrent() == 0 || abs(oldHealth.getCurrent() - health.getCurrent()) > 5);
|
|
|
|
bool shouldUpdateMagicka = false;
|
|
|
|
bool shouldUpdateFatigue = false;
|
|
|
|
|
|
|
|
if (!shouldUpdateHealth)
|
|
|
|
shouldUpdateMagicka = oldMagicka != magicka && (magicka.getCurrent() == 0 || abs(oldMagicka.getCurrent() - magicka.getCurrent()) > 10);
|
|
|
|
|
|
|
|
if (!shouldUpdateMagicka)
|
|
|
|
shouldUpdateFatigue = oldFatigue != fatigue && (fatigue.getCurrent() == 0 || abs(oldFatigue.getCurrent() - fatigue.getCurrent()) > 10);
|
|
|
|
|
|
|
|
if (forceUpdate || shouldUpdateHealth || shouldUpdateMagicka || shouldUpdateFatigue)
|
2017-04-16 13:42:07 +00:00
|
|
|
{
|
|
|
|
oldHealth = health;
|
|
|
|
oldMagicka = magicka;
|
|
|
|
oldFatigue = fatigue;
|
|
|
|
|
2017-04-30 15:44:59 +00:00
|
|
|
health.writeState(creatureStats.mDynamic[0]);
|
|
|
|
magicka.writeState(creatureStats.mDynamic[1]);
|
|
|
|
fatigue.writeState(creatureStats.mDynamic[2]);
|
2017-04-16 13:42:07 +00:00
|
|
|
|
2017-05-08 23:47:29 +00:00
|
|
|
creatureStats.mDead = ptrCreatureStats->isDead();
|
|
|
|
|
2017-04-16 13:42:07 +00:00
|
|
|
statTimer = 0;
|
|
|
|
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addStatsDynamicActor(*this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-26 01:37:49 +00:00
|
|
|
void LocalActor::updateEquipment(bool forceUpdate)
|
|
|
|
{
|
|
|
|
if (!ptr.getClass().hasInventoryStore(ptr))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (forceUpdate)
|
|
|
|
equipmentChanged = true;
|
|
|
|
|
|
|
|
MWWorld::InventoryStore &invStore = ptr.getClass().getInventoryStore(ptr);
|
|
|
|
for (int slot = 0; slot < MWWorld::InventoryStore::Slots; slot++)
|
|
|
|
{
|
|
|
|
MWWorld::ContainerStoreIterator it = invStore.getSlot(slot);
|
|
|
|
if (it != invStore.end() && !::Misc::StringUtils::ciEqual(it->getCellRef().getRefId(), equipedItems[slot].refId))
|
|
|
|
{
|
|
|
|
equipmentChanged = true;
|
|
|
|
|
|
|
|
equipedItems[slot].refId = it->getCellRef().getRefId();
|
|
|
|
equipedItems[slot].charge = it->getCellRef().getCharge();
|
|
|
|
if (slot == MWWorld::InventoryStore::Slot_CarriedRight)
|
|
|
|
{
|
|
|
|
MWMechanics::WeaponType weaptype;
|
|
|
|
MWMechanics::getActiveWeapon(ptr.getClass().getCreatureStats(ptr), ptr.getClass().getInventoryStore(ptr), &weaptype);
|
|
|
|
if (weaptype != MWMechanics::WeapType_Thrown)
|
|
|
|
equipedItems[slot].count = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
equipedItems[slot].count = invStore.count(it->getCellRef().getRefId());
|
|
|
|
}
|
|
|
|
else if (it == invStore.end() && !equipedItems[slot].refId.empty())
|
|
|
|
{
|
|
|
|
equipmentChanged = true;
|
|
|
|
equipedItems[slot].refId = "";
|
|
|
|
equipedItems[slot].count = 0;
|
|
|
|
equipedItems[slot].charge = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (equipmentChanged)
|
|
|
|
{
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addEquipmentActor(*this);
|
2017-05-26 19:53:28 +00:00
|
|
|
equipmentChanged = false;
|
2017-05-26 01:37:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-19 19:06:04 +00:00
|
|
|
void LocalActor::updateAttack()
|
|
|
|
{
|
|
|
|
if (attack.shouldSend)
|
|
|
|
{
|
|
|
|
if (attack.type == Attack::MAGIC)
|
|
|
|
{
|
2017-04-24 08:58:39 +00:00
|
|
|
MWMechanics::CreatureStats &attackerStats = ptr.getClass().getCreatureStats(ptr);
|
2017-04-19 19:06:04 +00:00
|
|
|
attack.spellId = attackerStats.getSpells().getSelectedSpell();
|
|
|
|
attack.success = mwmp::Main::get().getMechanicsHelper()->getSpellSuccess(attack.spellId, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
mwmp::Main::get().getNetworking()->getActorList()->addAttackActor(*this);
|
|
|
|
attack.shouldSend = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-06 01:00:50 +00:00
|
|
|
MWWorld::Ptr LocalActor::getPtr()
|
|
|
|
{
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LocalActor::setPtr(const MWWorld::Ptr& newPtr)
|
|
|
|
{
|
|
|
|
ptr = newPtr;
|
2017-04-10 04:22:03 +00:00
|
|
|
|
2017-04-13 12:26:48 +00:00
|
|
|
refId = ptr.getCellRef().getRefId();
|
|
|
|
refNumIndex = ptr.getCellRef().getRefNum().mIndex;
|
|
|
|
mpNum = ptr.getCellRef().getMpNum();
|
|
|
|
|
2017-04-24 08:58:39 +00:00
|
|
|
lastDrawState = ptr.getClass().getCreatureStats(ptr).getDrawState();
|
2017-04-16 14:09:47 +00:00
|
|
|
oldHealth = ptr.getClass().getCreatureStats(ptr).getHealth();
|
|
|
|
oldMagicka = ptr.getClass().getCreatureStats(ptr).getMagicka();
|
|
|
|
oldFatigue = ptr.getClass().getCreatureStats(ptr).getFatigue();
|
2017-04-06 01:00:50 +00:00
|
|
|
}
|