2012-11-15 21:32:15 +00:00
|
|
|
#include "aifollow.hpp"
|
|
|
|
#include <iostream>
|
2014-01-11 11:06:36 +00:00
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "movement.hpp"
|
2012-11-15 21:32:15 +00:00
|
|
|
|
2012-11-16 19:28:20 +00:00
|
|
|
MWMechanics::AiFollow::AiFollow(const std::string &actorId,float duration, float x, float y, float z)
|
2014-01-07 20:10:57 +00:00
|
|
|
: mDuration(duration), mX(x), mY(y), mZ(z), mActorId(actorId), mTimer(0), mStuckTimer(0)
|
2012-11-15 21:32:15 +00:00
|
|
|
{
|
|
|
|
}
|
2012-11-30 00:16:16 +00:00
|
|
|
MWMechanics::AiFollow::AiFollow(const std::string &actorId,const std::string &cellId,float duration, float x, float y, float z)
|
2014-01-11 11:06:36 +00:00
|
|
|
: mDuration(duration), mX(x), mY(y), mZ(z), mActorId(actorId), mCellId(cellId), mTimer(0), mStuckTimer(0)
|
2012-11-30 00:16:16 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2012-11-15 21:32:15 +00:00
|
|
|
MWMechanics::AiFollow *MWMechanics::AiFollow::clone() const
|
|
|
|
{
|
|
|
|
return new AiFollow(*this);
|
|
|
|
}
|
|
|
|
|
2013-10-30 19:42:50 +00:00
|
|
|
bool MWMechanics::AiFollow::execute (const MWWorld::Ptr& actor,float duration)
|
2012-11-15 21:32:15 +00:00
|
|
|
{
|
2014-01-11 11:06:36 +00:00
|
|
|
const MWWorld::Ptr target = MWBase::Environment::get().getWorld()->getPtr(mActorId, false);
|
|
|
|
|
|
|
|
mTimer = mTimer + duration;
|
|
|
|
mStuckTimer = mStuckTimer + duration;
|
|
|
|
mTotalTime = mTotalTime + duration;
|
|
|
|
|
2014-01-11 19:32:38 +00:00
|
|
|
ESM::Position pos = actor.getRefData().getPosition();
|
|
|
|
|
|
|
|
if(mTotalTime > mDuration ||
|
|
|
|
(pos.pos[0]-mX)*(pos.pos[0]-mX) +
|
|
|
|
(pos.pos[1]-mY)*(pos.pos[1]-mY) +
|
|
|
|
(pos.pos[2]-mZ)*(pos.pos[2]-mZ) < 100*100)
|
|
|
|
return true;
|
2014-01-11 11:06:36 +00:00
|
|
|
|
|
|
|
ESM::Pathgrid::Point dest;
|
|
|
|
dest.mX = target.getRefData().getPosition().pos[0];
|
|
|
|
dest.mY = target.getRefData().getPosition().pos[1];
|
|
|
|
dest.mZ = target.getRefData().getPosition().pos[2];
|
|
|
|
|
|
|
|
if(mTimer > 0.25)
|
|
|
|
{
|
2014-01-07 20:10:57 +00:00
|
|
|
if(!mPathFinder.getPath().empty())
|
|
|
|
{
|
|
|
|
ESM::Pathgrid::Point lastPos = mPathFinder.getPath().back();
|
2014-01-11 11:06:36 +00:00
|
|
|
|
2014-01-07 20:10:57 +00:00
|
|
|
if((dest.mX - lastPos.mX)*(dest.mX - lastPos.mX)
|
|
|
|
+(dest.mY - lastPos.mY)*(dest.mY - lastPos.mY)
|
|
|
|
+(dest.mZ - lastPos.mZ)*(dest.mZ - lastPos.mZ)
|
2014-01-11 11:06:36 +00:00
|
|
|
> 100*100)
|
2014-01-07 20:10:57 +00:00
|
|
|
mPathFinder.addPointToPath(dest);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
mPathFinder.addPointToPath(dest);
|
2014-01-11 11:06:36 +00:00
|
|
|
|
|
|
|
mTimer = 0;
|
|
|
|
}
|
|
|
|
|
2014-01-07 20:10:57 +00:00
|
|
|
if(mStuckTimer>0.5)
|
|
|
|
{
|
|
|
|
if((mStuckPos.pos[0] - pos.pos[0])*(mStuckPos.pos[0] - pos.pos[0])
|
|
|
|
+(mStuckPos.pos[1] - pos.pos[1])*(mStuckPos.pos[1] - pos.pos[1])
|
|
|
|
+(mStuckPos.pos[2] - pos.pos[2])*(mStuckPos.pos[2] - pos.pos[2])
|
|
|
|
< 100) //NPC is stuck
|
|
|
|
{
|
|
|
|
ESM::Pathgrid::Point start;
|
|
|
|
start.mX = pos.pos[0];
|
|
|
|
start.mY = pos.pos[1];
|
|
|
|
start.mZ = pos.pos[2];
|
|
|
|
|
|
|
|
const ESM::Pathgrid *pathgrid =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Pathgrid>().search(*actor.getCell()->mCell);
|
|
|
|
|
|
|
|
float xCell = 0;
|
|
|
|
float yCell = 0;
|
|
|
|
|
|
|
|
if (actor.getCell()->mCell->isExterior())
|
|
|
|
{
|
|
|
|
xCell = actor.getCell()->mCell->mData.mX * ESM::Land::REAL_SIZE;
|
|
|
|
yCell = actor.getCell()->mCell->mData.mY * ESM::Land::REAL_SIZE;
|
|
|
|
}
|
|
|
|
mPathFinder.buildPath(start, dest, pathgrid, xCell, yCell, true);
|
|
|
|
}
|
|
|
|
mStuckTimer = 0;
|
|
|
|
mStuckPos = pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(mPathFinder.getPath().empty())
|
|
|
|
mPathFinder.addPointToPath(dest);
|
|
|
|
|
|
|
|
if(!mPathFinder.checkPathCompleted(pos.pos[0],pos.pos[1],pos.pos[2]))
|
|
|
|
{
|
|
|
|
float zAngle = mPathFinder.getZAngleToNext(pos.pos[0], pos.pos[1]);
|
|
|
|
MWBase::Environment::get().getWorld()->rotateObject(actor, 0, 0, zAngle, false);
|
|
|
|
}
|
2014-01-11 11:06:36 +00:00
|
|
|
|
|
|
|
if((dest.mX - pos.pos[0])*(dest.mX - pos.pos[0])+(dest.mY - pos.pos[1])*(dest.mY - pos.pos[1])+(dest.mZ - pos.pos[2])*(dest.mZ - pos.pos[2])
|
|
|
|
< 100*100)
|
|
|
|
MWWorld::Class::get(actor).getMovementSettings(actor).mPosition[1] = 0;
|
|
|
|
else
|
|
|
|
MWWorld::Class::get(actor).getMovementSettings(actor).mPosition[1] = 1;
|
|
|
|
|
|
|
|
return false;
|
2012-11-15 21:32:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int MWMechanics::AiFollow::getTypeId() const
|
|
|
|
{
|
|
|
|
return 3;
|
|
|
|
}
|