2012-11-14 18:42:04 +01:00
|
|
|
#include "aiescort.hpp"
|
2013-05-08 15:27:20 -07:00
|
|
|
|
2014-06-12 23:27:04 +02:00
|
|
|
#include <components/esm/aisequence.hpp>
|
2015-12-06 23:32:20 +01:00
|
|
|
#include <components/esm/loadcell.hpp>
|
2014-06-12 23:27:04 +02:00
|
|
|
|
2013-05-08 20:02:24 -07:00
|
|
|
#include "../mwbase/world.hpp"
|
2013-05-08 15:27:20 -07:00
|
|
|
#include "../mwbase/environment.hpp"
|
2013-05-08 20:02:24 -07:00
|
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
2013-05-08 15:27:20 -07:00
|
|
|
|
2014-02-23 20:11:05 +01:00
|
|
|
#include "../mwworld/class.hpp"
|
2015-12-06 23:32:20 +01:00
|
|
|
#include "../mwworld/cellstore.hpp"
|
2014-02-23 20:11:05 +01:00
|
|
|
|
2014-07-28 17:28:00 +02:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
|
2014-01-29 20:29:07 +01:00
|
|
|
#include "steering.hpp"
|
2014-02-23 20:11:05 +01:00
|
|
|
#include "movement.hpp"
|
2014-01-29 20:29:07 +01:00
|
|
|
|
2013-05-08 20:02:24 -07:00
|
|
|
/*
|
|
|
|
TODO: Test vanilla behavior on passing x0, y0, and z0 with duration of anything including 0.
|
|
|
|
TODO: Different behavior for AIEscort a d x y z and AIEscortCell a c d x y z.
|
|
|
|
TODO: Take account for actors being in different cells.
|
|
|
|
*/
|
2013-05-08 15:27:20 -07:00
|
|
|
|
2013-05-31 17:49:52 -07:00
|
|
|
namespace MWMechanics
|
2013-05-08 15:27:20 -07:00
|
|
|
{
|
2014-05-16 12:11:34 +02:00
|
|
|
AiEscort::AiEscort(const std::string &actorId, int duration, float x, float y, float z)
|
2015-03-08 17:42:07 +13:00
|
|
|
: mActorId(actorId), mX(x), mY(y), mZ(z), mRemainingDuration(static_cast<float>(duration))
|
2014-01-12 22:47:22 +01:00
|
|
|
, mCellX(std::numeric_limits<int>::max())
|
|
|
|
, mCellY(std::numeric_limits<int>::max())
|
2013-05-31 17:49:52 -07:00
|
|
|
{
|
2014-09-19 05:46:59 +02:00
|
|
|
mMaxDist = 450;
|
2013-05-08 20:02:24 -07:00
|
|
|
|
2014-01-29 20:29:07 +01:00
|
|
|
// The CS Help File states that if a duration is given, the AI package will run for that long
|
2013-05-31 17:49:52 -07:00
|
|
|
// BUT if a location is givin, it "trumps" the duration so it will simply escort to that location.
|
|
|
|
if(mX != 0 || mY != 0 || mZ != 0)
|
2014-06-12 23:27:04 +02:00
|
|
|
mRemainingDuration = 0;
|
2013-05-08 15:27:20 -07:00
|
|
|
}
|
2012-11-30 02:16:16 +02:00
|
|
|
|
2014-05-16 12:11:34 +02:00
|
|
|
AiEscort::AiEscort(const std::string &actorId, const std::string &cellId,int duration, float x, float y, float z)
|
2015-03-08 17:42:07 +13:00
|
|
|
: mActorId(actorId), mCellId(cellId), mX(x), mY(y), mZ(z), mRemainingDuration(static_cast<float>(duration))
|
2014-01-12 22:47:22 +01:00
|
|
|
, mCellX(std::numeric_limits<int>::max())
|
|
|
|
, mCellY(std::numeric_limits<int>::max())
|
2013-05-31 17:49:52 -07:00
|
|
|
{
|
2014-09-19 05:46:59 +02:00
|
|
|
mMaxDist = 450;
|
2013-05-08 20:02:24 -07:00
|
|
|
|
2014-01-29 20:29:07 +01:00
|
|
|
// The CS Help File states that if a duration is given, the AI package will run for that long
|
2014-06-12 23:27:04 +02:00
|
|
|
// BUT if a location is given, it "trumps" the duration so it will simply escort to that location.
|
2013-05-31 17:49:52 -07:00
|
|
|
if(mX != 0 || mY != 0 || mZ != 0)
|
2014-06-12 23:27:04 +02:00
|
|
|
mRemainingDuration = 0;
|
|
|
|
}
|
2013-05-08 15:27:20 -07:00
|
|
|
|
2014-06-12 23:27:04 +02:00
|
|
|
AiEscort::AiEscort(const ESM::AiSequence::AiEscort *escort)
|
2015-04-30 19:24:27 -05:00
|
|
|
: mActorId(escort->mTargetId), mCellId(escort->mCellId), mX(escort->mData.mX), mY(escort->mData.mY), mZ(escort->mData.mZ)
|
|
|
|
, mMaxDist(450)
|
|
|
|
, mRemainingDuration(escort->mRemainingDuration)
|
2014-06-12 23:27:04 +02:00
|
|
|
, mCellX(std::numeric_limits<int>::max())
|
|
|
|
, mCellY(std::numeric_limits<int>::max())
|
|
|
|
{
|
2013-05-08 15:27:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-31 17:49:52 -07:00
|
|
|
AiEscort *MWMechanics::AiEscort::clone() const
|
2013-05-08 15:27:20 -07:00
|
|
|
{
|
2013-05-31 17:49:52 -07:00
|
|
|
return new AiEscort(*this);
|
2013-05-08 15:27:20 -07:00
|
|
|
}
|
|
|
|
|
2015-06-26 17:47:04 +02:00
|
|
|
bool AiEscort::execute (const MWWorld::Ptr& actor, CharacterController& characterController, AiState& state, float duration)
|
2013-05-31 17:49:52 -07:00
|
|
|
{
|
|
|
|
// If AiEscort has ran for as long or longer then the duration specified
|
|
|
|
// and the duration is not infinite, the package is complete.
|
2014-06-12 23:27:04 +02:00
|
|
|
if(mRemainingDuration != 0)
|
2013-05-31 17:49:52 -07:00
|
|
|
{
|
2014-06-12 23:27:04 +02:00
|
|
|
mRemainingDuration -= duration;
|
|
|
|
if (duration <= 0)
|
2016-05-29 01:02:22 +09:00
|
|
|
{
|
|
|
|
// Reset mStarted to false so that package can be repeated again
|
|
|
|
mStarted = false;
|
2013-05-31 17:49:52 -07:00
|
|
|
return true;
|
2016-05-29 01:02:22 +09:00
|
|
|
}
|
2013-05-31 17:49:52 -07:00
|
|
|
}
|
2013-05-08 15:27:20 -07:00
|
|
|
|
2015-12-06 23:32:20 +01:00
|
|
|
if (!mCellId.empty() && mCellId != actor.getCell()->getCell()->getCellId().mWorldspace)
|
|
|
|
return false; // Not in the correct cell, pause and rely on the player to go back through a teleport door
|
|
|
|
|
2014-07-28 17:28:00 +02:00
|
|
|
actor.getClass().getCreatureStats(actor).setDrawState(DrawState_Nothing);
|
2014-08-11 04:21:04 +02:00
|
|
|
actor.getClass().getCreatureStats(actor).setMovementFlag(CreatureStats::Flag_Run, false);
|
2014-07-28 17:28:00 +02:00
|
|
|
|
2014-05-16 12:11:34 +02:00
|
|
|
const MWWorld::Ptr follower = MWBase::Environment::get().getWorld()->getPtr(mActorId, false);
|
2013-05-31 17:49:52 -07:00
|
|
|
const float* const leaderPos = actor.getRefData().getPosition().pos;
|
|
|
|
const float* const followerPos = follower.getRefData().getPosition().pos;
|
|
|
|
double differenceBetween[3];
|
2013-05-08 15:27:20 -07:00
|
|
|
|
2013-05-31 17:49:52 -07:00
|
|
|
for (short counter = 0; counter < 3; counter++)
|
|
|
|
differenceBetween[counter] = (leaderPos[counter] - followerPos[counter]);
|
2013-05-08 17:19:47 -07:00
|
|
|
|
2015-03-08 17:42:07 +13:00
|
|
|
double distanceBetweenResult =
|
2013-05-31 17:49:52 -07:00
|
|
|
(differenceBetween[0] * differenceBetween[0]) + (differenceBetween[1] * differenceBetween[1]) + (differenceBetween[2] *
|
|
|
|
differenceBetween[2]);
|
2013-05-08 17:19:47 -07:00
|
|
|
|
2013-05-31 17:49:52 -07:00
|
|
|
if(distanceBetweenResult <= mMaxDist * mMaxDist)
|
|
|
|
{
|
2015-03-08 17:42:07 +13:00
|
|
|
ESM::Pathgrid::Point point(static_cast<int>(mX), static_cast<int>(mY), static_cast<int>(mZ));
|
2014-09-28 12:44:47 +02:00
|
|
|
point.mAutogenerated = 0;
|
|
|
|
point.mConnectionNum = 0;
|
|
|
|
point.mUnknown = 0;
|
|
|
|
if(pathTo(actor,point,duration)) //Returns true on path complete
|
2016-05-29 01:02:22 +09:00
|
|
|
{
|
|
|
|
// Reset mStarted to false so that package can be repeated again
|
|
|
|
mStarted = false;
|
2014-05-14 01:44:11 -04:00
|
|
|
return true;
|
2016-05-29 01:02:22 +09:00
|
|
|
}
|
2014-09-19 05:46:59 +02:00
|
|
|
mMaxDist = 450;
|
2013-05-31 17:49:52 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Stop moving if the player is to far away
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->playAnimationGroup(actor, "idle3", 0, 1);
|
2014-05-22 20:37:22 +02:00
|
|
|
actor.getClass().getMovementSettings(actor).mPosition[1] = 0;
|
2014-09-19 05:46:59 +02:00
|
|
|
mMaxDist = 250;
|
2013-05-31 17:49:52 -07:00
|
|
|
}
|
2013-05-08 17:19:47 -07:00
|
|
|
|
2013-05-31 17:49:52 -07:00
|
|
|
return false;
|
2013-05-08 15:27:20 -07:00
|
|
|
}
|
2013-05-31 17:49:52 -07:00
|
|
|
|
|
|
|
int AiEscort::getTypeId() const
|
2013-05-08 15:27:20 -07:00
|
|
|
{
|
2014-01-07 04:12:37 +04:00
|
|
|
return TypeIdEscort;
|
2013-05-08 15:27:20 -07:00
|
|
|
}
|
2014-06-12 23:27:04 +02:00
|
|
|
|
2016-02-16 19:17:04 +01:00
|
|
|
MWWorld::Ptr AiEscort::getTarget() const
|
2015-12-06 23:32:49 +01:00
|
|
|
{
|
|
|
|
return MWBase::Environment::get().getWorld()->getPtr(mActorId, false);
|
|
|
|
}
|
|
|
|
|
2014-06-12 23:27:04 +02:00
|
|
|
void AiEscort::writeState(ESM::AiSequence::AiSequence &sequence) const
|
|
|
|
{
|
|
|
|
std::auto_ptr<ESM::AiSequence::AiEscort> escort(new ESM::AiSequence::AiEscort());
|
|
|
|
escort->mData.mX = mX;
|
|
|
|
escort->mData.mY = mY;
|
|
|
|
escort->mData.mZ = mZ;
|
|
|
|
escort->mTargetId = mActorId;
|
|
|
|
escort->mRemainingDuration = mRemainingDuration;
|
|
|
|
escort->mCellId = mCellId;
|
|
|
|
|
|
|
|
ESM::AiSequence::AiPackageContainer package;
|
|
|
|
package.mType = ESM::AiSequence::Ai_Escort;
|
|
|
|
package.mPackage = escort.release();
|
|
|
|
sequence.mPackages.push_back(package);
|
|
|
|
}
|
2012-11-30 02:16:16 +02:00
|
|
|
}
|
|
|
|
|