|
|
@ -27,6 +27,7 @@
|
|
|
|
#include "../mwrender/vismask.hpp"
|
|
|
|
#include "../mwrender/vismask.hpp"
|
|
|
|
|
|
|
|
|
|
|
|
#include "spellcasting.hpp"
|
|
|
|
#include "spellcasting.hpp"
|
|
|
|
|
|
|
|
#include "steering.hpp"
|
|
|
|
#include "npcstats.hpp"
|
|
|
|
#include "npcstats.hpp"
|
|
|
|
#include "creaturestats.hpp"
|
|
|
|
#include "creaturestats.hpp"
|
|
|
|
#include "movement.hpp"
|
|
|
|
#include "movement.hpp"
|
|
|
@ -141,6 +142,9 @@ void getRestorationPerHourOfSleep (const MWWorld::Ptr& ptr, float& health, float
|
|
|
|
|
|
|
|
|
|
|
|
namespace MWMechanics
|
|
|
|
namespace MWMechanics
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
static const int GREETING_SHOULD_START = 4; //how many updates should pass before NPC can greet player
|
|
|
|
|
|
|
|
static const int GREETING_SHOULD_END = 10;
|
|
|
|
|
|
|
|
|
|
|
|
class GetStuntedMagickaDuration : public MWMechanics::EffectSourceVisitor
|
|
|
|
class GetStuntedMagickaDuration : public MWMechanics::EffectSourceVisitor
|
|
|
|
{
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
public:
|
|
|
@ -397,6 +401,113 @@ namespace MWMechanics
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(actor, "idle");
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(actor, "idle");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void Actors::updateGreetingState(const MWWorld::Ptr& actor, bool turnOnly)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!actor.getClass().isActor() || actor == getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CreatureStats &stats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
int hello = stats.getAiSetting(CreatureStats::AI_Hello).getModified();
|
|
|
|
|
|
|
|
if (hello == 0)
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (MWBase::Environment::get().getWorld()->isSwimming(actor))
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MWWorld::Ptr player = getPlayer();
|
|
|
|
|
|
|
|
osg::Vec3f playerPos(player.getRefData().getPosition().asVec3());
|
|
|
|
|
|
|
|
osg::Vec3f actorPos(actor.getRefData().getPosition().asVec3());
|
|
|
|
|
|
|
|
osg::Vec3f dir = playerPos - actorPos;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const MWMechanics::AiSequence& seq = stats.getAiSequence();
|
|
|
|
|
|
|
|
int packageId = seq.getTypeId();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (seq.isInCombat() || (packageId != AiPackage::TypeIdWander && packageId != AiPackage::TypeIdTravel && packageId != -1))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
stats.setTurningToPlayer(false);
|
|
|
|
|
|
|
|
stats.setGreetingTimer(0);
|
|
|
|
|
|
|
|
stats.setGreetingState(Greet_None);
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (stats.isTurningToPlayer())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// Reduce the turning animation glitch by using a *HUGE* value of
|
|
|
|
|
|
|
|
// epsilon... TODO: a proper fix might be in either the physics or the
|
|
|
|
|
|
|
|
// animation subsystem
|
|
|
|
|
|
|
|
if (zTurn(actor, stats.getAngleToPlayer(), osg::DegreesToRadians(5.f)))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
stats.setTurningToPlayer(false);
|
|
|
|
|
|
|
|
// An original engine launches an endless idle2 when an actor greets player.
|
|
|
|
|
|
|
|
playAnimationGroup (actor, "idle2", 0, std::numeric_limits<int>::max(), false);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (turnOnly)
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Play a random voice greeting if the player gets too close
|
|
|
|
|
|
|
|
float helloDistance = static_cast<float>(hello);
|
|
|
|
|
|
|
|
static int iGreetDistanceMultiplier = MWBase::Environment::get().getWorld()->getStore()
|
|
|
|
|
|
|
|
.get<ESM::GameSetting>().find("iGreetDistanceMultiplier")->mValue.getInteger();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
helloDistance *= iGreetDistanceMultiplier;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int greetingTimer = stats.getGreetingTimer();
|
|
|
|
|
|
|
|
GreetingState greetingState = stats.getGreetingState();
|
|
|
|
|
|
|
|
if (greetingState == Greet_None)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if ((playerPos - actorPos).length2() <= helloDistance*helloDistance &&
|
|
|
|
|
|
|
|
!player.getClass().getCreatureStats(player).isDead() && !actor.getClass().getCreatureStats(actor).isParalyzed()
|
|
|
|
|
|
|
|
&& MWBase::Environment::get().getWorld()->getLOS(player, actor)
|
|
|
|
|
|
|
|
&& MWBase::Environment::get().getMechanicsManager()->awarenessCheck(player, actor))
|
|
|
|
|
|
|
|
greetingTimer++;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (greetingTimer >= GREETING_SHOULD_START)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
greetingState = Greet_InProgress;
|
|
|
|
|
|
|
|
MWBase::Environment::get().getDialogueManager()->say(actor, "hello");
|
|
|
|
|
|
|
|
greetingTimer = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (greetingState == Greet_InProgress)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
greetingTimer++;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
turnActorToFacePlayer(actor, dir);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (greetingTimer >= GREETING_SHOULD_END)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
greetingState = Greet_Done;
|
|
|
|
|
|
|
|
greetingTimer = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (greetingState == Greet_Done)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
float resetDist = 2 * helloDistance;
|
|
|
|
|
|
|
|
if ((playerPos - actorPos).length2() >= resetDist*resetDist)
|
|
|
|
|
|
|
|
greetingState = Greet_None;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
stats.setGreetingTimer(greetingTimer);
|
|
|
|
|
|
|
|
stats.setGreetingState(greetingState);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void Actors::turnActorToFacePlayer(const MWWorld::Ptr& actor, const osg::Vec3f& dir)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
actor.getClass().getMovementSettings(actor).mPosition[1] = 0;
|
|
|
|
|
|
|
|
actor.getClass().getMovementSettings(actor).mPosition[0] = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CreatureStats &stats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
if (!stats.isTurningToPlayer())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
stats.setAngleToPlayer(std::atan2(dir.x(), dir.y()));
|
|
|
|
|
|
|
|
stats.setTurningToPlayer(true);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Actors::engageCombat (const MWWorld::Ptr& actor1, const MWWorld::Ptr& actor2, std::map<const MWWorld::Ptr, const std::set<MWWorld::Ptr> >& cachedAllies, bool againstPlayer)
|
|
|
|
void Actors::engageCombat (const MWWorld::Ptr& actor1, const MWWorld::Ptr& actor2, std::map<const MWWorld::Ptr, const std::set<MWWorld::Ptr> >& cachedAllies, bool againstPlayer)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// No combat for totally static creatures
|
|
|
|
// No combat for totally static creatures
|
|
|
@ -1409,11 +1520,13 @@ namespace MWMechanics
|
|
|
|
static float timerUpdateAITargets = 0;
|
|
|
|
static float timerUpdateAITargets = 0;
|
|
|
|
static float timerUpdateHeadTrack = 0;
|
|
|
|
static float timerUpdateHeadTrack = 0;
|
|
|
|
static float timerUpdateEquippedLight = 0;
|
|
|
|
static float timerUpdateEquippedLight = 0;
|
|
|
|
|
|
|
|
static float timerUpdateHello = 0;
|
|
|
|
const float updateEquippedLightInterval = 1.0f;
|
|
|
|
const float updateEquippedLightInterval = 1.0f;
|
|
|
|
|
|
|
|
|
|
|
|
// target lists get updated once every 1.0 sec
|
|
|
|
// target lists get updated once every 1.0 sec
|
|
|
|
if (timerUpdateAITargets >= 1.0f) timerUpdateAITargets = 0;
|
|
|
|
if (timerUpdateAITargets >= 1.0f) timerUpdateAITargets = 0;
|
|
|
|
if (timerUpdateHeadTrack >= 0.3f) timerUpdateHeadTrack = 0;
|
|
|
|
if (timerUpdateHeadTrack >= 0.3f) timerUpdateHeadTrack = 0;
|
|
|
|
|
|
|
|
if (timerUpdateHello >= 0.25f) timerUpdateHello = 0;
|
|
|
|
if (mTimerDisposeSummonsCorpses >= 0.2f) mTimerDisposeSummonsCorpses = 0;
|
|
|
|
if (mTimerDisposeSummonsCorpses >= 0.2f) mTimerDisposeSummonsCorpses = 0;
|
|
|
|
if (timerUpdateEquippedLight >= updateEquippedLightInterval) timerUpdateEquippedLight = 0;
|
|
|
|
if (timerUpdateEquippedLight >= updateEquippedLightInterval) timerUpdateEquippedLight = 0;
|
|
|
|
|
|
|
|
|
|
|
@ -1532,6 +1645,7 @@ namespace MWMechanics
|
|
|
|
if (isConscious(iter->first))
|
|
|
|
if (isConscious(iter->first))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
stats.getAiSequence().execute(iter->first, *ctrl, duration);
|
|
|
|
stats.getAiSequence().execute(iter->first, *ctrl, duration);
|
|
|
|
|
|
|
|
updateGreetingState(iter->first, timerUpdateHello > 0);
|
|
|
|
playIdleDialogue(iter->first);
|
|
|
|
playIdleDialogue(iter->first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1554,6 +1668,7 @@ namespace MWMechanics
|
|
|
|
timerUpdateAITargets += duration;
|
|
|
|
timerUpdateAITargets += duration;
|
|
|
|
timerUpdateHeadTrack += duration;
|
|
|
|
timerUpdateHeadTrack += duration;
|
|
|
|
timerUpdateEquippedLight += duration;
|
|
|
|
timerUpdateEquippedLight += duration;
|
|
|
|
|
|
|
|
timerUpdateHello += duration;
|
|
|
|
mTimerDisposeSummonsCorpses += duration;
|
|
|
|
mTimerDisposeSummonsCorpses += duration;
|
|
|
|
|
|
|
|
|
|
|
|
// Animation/movement update
|
|
|
|
// Animation/movement update
|
|
|
|