2014-01-24 19:16:50 +01:00
# include "aifollow.hpp"
2012-11-15 22:32:15 +01:00
# include <iostream>
2014-01-11 12:06:36 +01:00
# include "../mwbase/world.hpp"
# include "../mwbase/environment.hpp"
# include "../mwworld/class.hpp"
2014-03-29 18:36:32 +01:00
# include "../mwworld/cellstore.hpp"
2014-04-23 02:57:48 -04:00
# include "creaturestats.hpp"
2014-01-11 12:06:36 +01:00
# include "movement.hpp"
2014-01-24 19:13:23 +01:00
# include <OgreMath.h>
2014-01-29 20:29:07 +01:00
# include "steering.hpp"
2012-11-16 20:28:20 +01:00
MWMechanics : : AiFollow : : AiFollow ( const std : : string & actorId , float duration , float x , float y , float z )
2014-03-05 11:24:39 +01:00
: mAlwaysFollow ( false ) , mDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z ) , mActorId ( actorId ) , mCellId ( " " ) , mTimer ( 0 ) , mStuckTimer ( 0 )
2014-01-29 20:29:07 +01:00
{
}
MWMechanics : : AiFollow : : AiFollow ( const std : : string & actorId , const std : : string & cellId , float duration , float x , float y , float z )
2014-03-05 11:24:39 +01:00
: mAlwaysFollow ( false ) , mDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z ) , mActorId ( actorId ) , mCellId ( cellId ) , mTimer ( 0 ) , mStuckTimer ( 0 )
{
}
MWMechanics : : AiFollow : : AiFollow ( const std : : string & actorId )
: mAlwaysFollow ( true ) , mDuration ( 0 ) , mX ( 0 ) , mY ( 0 ) , mZ ( 0 ) , mActorId ( actorId ) , mCellId ( " " ) , mTimer ( 0 ) , mStuckTimer ( 0 )
2014-01-29 20:29:07 +01:00
{
}
2012-11-15 22:32:15 +01:00
2014-01-12 11:38:58 +01:00
bool MWMechanics : : AiFollow : : execute ( const MWWorld : : Ptr & actor , float duration )
2012-11-15 22:32:15 +01:00
{
2014-04-23 02:57:48 -04:00
const MWWorld : : Ptr target = MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtr ( mActorId , false ) ; //The target to follow
2014-01-11 12:06:36 +01:00
2014-04-23 02:57:48 -04:00
if ( target = = MWWorld : : Ptr ( ) ) return true ; //Target doesn't exist
2014-01-11 12:06:36 +01:00
2014-04-23 02:57:48 -04:00
mTimer = mTimer + duration ; //Update timer
mStuckTimer = mStuckTimer + duration ; //Update stuck timer
mTotalTime = mTotalTime + duration ; //Update total time following
2014-01-11 20:32:38 +01:00
2014-04-23 02:57:48 -04:00
ESM : : Position pos = actor . getRefData ( ) . getPosition ( ) ; //position of the actor
if ( ! mAlwaysFollow ) //Update if you only follow for a bit
2014-01-12 11:38:58 +01:00
{
2014-04-23 02:57:48 -04:00
if ( mTotalTime > mDuration & & mDuration ! = 0 ) //Check if we've run out of time
2014-03-05 11:24:39 +01:00
return true ;
if ( ( pos . pos [ 0 ] - mX ) * ( pos . pos [ 0 ] - mX ) +
( pos . pos [ 1 ] - mY ) * ( pos . pos [ 1 ] - mY ) +
2014-04-23 02:57:48 -04:00
( pos . pos [ 2 ] - mZ ) * ( pos . pos [ 2 ] - mZ ) < 100 * 100 ) //Close-ish to final position
2014-01-12 11:38:58 +01:00
{
2014-04-23 02:57:48 -04:00
if ( actor . getCell ( ) - > isExterior ( ) ) //Outside?
2014-03-05 11:24:39 +01:00
{
2014-04-23 02:57:48 -04:00
if ( mCellId = = " " ) //No cell to travel to
2014-03-05 11:24:39 +01:00
return true ;
}
else
{
2014-04-23 02:57:48 -04:00
if ( mCellId = = actor . getCell ( ) - > getCell ( ) - > mName ) //Cell to travel to
2014-03-05 11:24:39 +01:00
return true ;
}
2014-01-12 11:38:58 +01:00
}
}
2014-01-11 12:06:36 +01:00
2014-04-23 02:57:48 -04:00
//Set the target desition from the actor
2014-01-29 20:29:07 +01:00
ESM : : Pathgrid : : Point dest ;
dest . mX = target . getRefData ( ) . getPosition ( ) . pos [ 0 ] ;
dest . mY = target . getRefData ( ) . getPosition ( ) . pos [ 1 ] ;
2014-01-11 12:06:36 +01:00
dest . mZ = target . getRefData ( ) . getPosition ( ) . pos [ 2 ] ;
2014-04-23 02:57:48 -04:00
//Current position, for pathfilding stuff
2014-01-29 20:29:07 +01:00
ESM : : Pathgrid : : Point start ;
start . mX = pos . pos [ 0 ] ;
start . mY = pos . pos [ 1 ] ;
2014-01-24 19:13:23 +01:00
start . mZ = pos . pos [ 2 ] ;
2014-04-23 02:57:48 -04:00
//Build the path to get to the destination
2014-01-24 19:13:23 +01:00
if ( mPathFinder . getPath ( ) . empty ( ) )
mPathFinder . buildPath ( start , dest , actor . getCell ( ) , true ) ;
2014-04-23 02:57:48 -04:00
//***********************
// Checks if you can't get to the end position at all
//***********************
2014-01-11 12:06:36 +01:00
if ( mTimer > 0.25 )
{
2014-04-23 02:57:48 -04:00
if ( ! mPathFinder . getPath ( ) . empty ( ) ) //Path has points in it
2014-01-07 21:10:57 +01:00
{
2014-04-23 02:57:48 -04:00
ESM : : Pathgrid : : Point lastPos = mPathFinder . getPath ( ) . back ( ) ; //Get the end of the proposed path
2014-01-11 12:06:36 +01:00
2014-04-23 02:57:48 -04:00
if ( ( dest . mX - lastPos . mX ) * ( dest . mX - lastPos . mX )
2014-01-07 21:10:57 +01:00
+ ( dest . mY - lastPos . mY ) * ( dest . mY - lastPos . mY )
+ ( dest . mZ - lastPos . mZ ) * ( dest . mZ - lastPos . mZ )
2014-04-23 02:57:48 -04:00
> 100 * 100 ) //End of the path is far from the destination
mPathFinder . addPointToPath ( dest ) ; //Adds the final destination to the path, to try to get to where you want to go
2014-01-07 21:10:57 +01:00
}
2014-01-11 12:06:36 +01:00
mTimer = 0 ;
}
2014-04-23 02:57:48 -04:00
//************************
// Checks if you aren't moving; you're stuck
//************************
if ( mStuckTimer > 0.5 ) //Checks every half of a second
2014-01-07 21:10:57 +01:00
{
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 ] )
2014-01-24 19:13:23 +01:00
+ ( mStuckPos . pos [ 2 ] - pos . pos [ 2 ] ) * ( mStuckPos . pos [ 2 ] - pos . pos [ 2 ] ) < 100 ) //NPC is stuck
2014-01-12 18:42:31 +01:00
mPathFinder . buildPath ( start , dest , actor . getCell ( ) , true ) ;
2014-01-24 19:13:23 +01:00
2014-01-07 21:10:57 +01:00
mStuckTimer = 0 ;
mStuckPos = pos ;
}
2014-04-23 02:57:48 -04:00
//Checks if the path isn't over, turn tomards the direction that you're going
2014-01-07 21:10:57 +01:00
if ( ! mPathFinder . checkPathCompleted ( pos . pos [ 0 ] , pos . pos [ 1 ] , pos . pos [ 2 ] ) )
{
2014-01-29 20:29:07 +01:00
zTurn ( actor , Ogre : : Degree ( mPathFinder . getZAngleToNext ( pos . pos [ 0 ] , pos . pos [ 1 ] ) ) ) ;
}
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 )
actor . getClass ( ) . getMovementSettings ( actor ) . mPosition [ 1 ] = 0 ;
2014-01-11 12:06:36 +01:00
else
2014-01-29 20:29:07 +01:00
actor . getClass ( ) . getMovementSettings ( actor ) . mPosition [ 1 ] = 1 ;
2014-01-11 12:06:36 +01:00
2014-04-23 02:57:48 -04:00
//Check if you're far away
if ( ( dest . mX - start . mX ) * ( dest . mX - start . mX )
+ ( dest . mY - start . mY ) * ( dest . mY - start . mY )
+ ( dest . mZ - start . mZ ) * ( dest . mZ - start . mZ ) > 1000 * 1000 )
actor . getClass ( ) . getCreatureStats ( actor ) . setMovementFlag ( MWMechanics : : CreatureStats : : Flag_Run , true ) ; //Make NPC run
else if ( ( dest . mX - start . mX ) * ( dest . mX - start . mX ) //Have a bit of a dead zone, otherwise npc will constantly flip between running and not when right on the edge of the running threshhold
+ ( dest . mY - start . mY ) * ( dest . mY - start . mY )
+ ( dest . mZ - start . mZ ) * ( dest . mZ - start . mZ ) < 800 * 800 )
actor . getClass ( ) . getCreatureStats ( actor ) . setMovementFlag ( MWMechanics : : CreatureStats : : Flag_Run , false ) ; //make NPC walk
2014-01-11 12:06:36 +01:00
return false ;
2012-11-15 22:32:15 +01:00
}
2014-01-12 14:02:40 +01:00
std : : string MWMechanics : : AiFollow : : getFollowedActor ( )
{
return mActorId ;
}
2014-01-24 19:16:50 +01:00
2014-01-29 20:29:07 +01:00
MWMechanics : : AiFollow * MWMechanics : : AiFollow : : clone ( ) const
{
return new AiFollow ( * this ) ;
}
int MWMechanics : : AiFollow : : getTypeId ( ) const
{
return TypeIdFollow ;
2014-01-24 19:16:50 +01:00
}