2014-01-24 19:16:50 +01:00
# include "aifollow.hpp"
2014-06-12 23:27:04 +02:00
2012-11-15 22:32:15 +01:00
# include <iostream>
2014-06-12 23:27:04 +02:00
# include <components/esm/aisequence.hpp>
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"
2014-05-16 12:11:34 +02:00
MWMechanics : : AiFollow : : AiFollow ( const std : : string & actorId , float duration , float x , float y , float z )
2014-08-13 16:00:32 +02:00
: mAlwaysFollow ( false ) , mCommanded ( false ) , mRemainingDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z )
, mActorRefId ( actorId ) , mCellId ( " " ) , mActorId ( - 1 )
2014-01-29 20:29:07 +01:00
{
}
2014-05-16 12:11:34 +02:00
MWMechanics : : AiFollow : : AiFollow ( const std : : string & actorId , const std : : string & cellId , float duration , float x , float y , float z )
2014-08-13 16:00:32 +02:00
: mAlwaysFollow ( false ) , mCommanded ( false ) , mRemainingDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z )
, mActorRefId ( actorId ) , mCellId ( cellId ) , mActorId ( - 1 )
2014-03-05 11:24:39 +01:00
{
}
2014-08-06 21:16:14 +02:00
MWMechanics : : AiFollow : : AiFollow ( const std : : string & actorId , bool commanded )
2014-08-13 16:00:32 +02:00
: mAlwaysFollow ( true ) , mCommanded ( commanded ) , mRemainingDuration ( 0 ) , mX ( 0 ) , mY ( 0 ) , mZ ( 0 )
, mActorRefId ( actorId ) , mCellId ( " " ) , mActorId ( - 1 )
2014-01-29 20:29:07 +01:00
{
}
2012-11-15 22:32:15 +01:00
2014-08-17 17:01:04 +02:00
MWMechanics : : AiFollow : : AiFollow ( const ESM : : AiSequence : : AiFollow * follow )
: mAlwaysFollow ( follow - > mAlwaysFollow ) , mRemainingDuration ( follow - > mRemainingDuration )
, mX ( follow - > mData . mX ) , mY ( follow - > mData . mY ) , mZ ( follow - > mData . mZ )
, mActorRefId ( follow - > mTargetId ) , mActorId ( - 1 ) , mCellId ( follow - > mCellId )
, mCommanded ( follow - > mCommanded )
{
}
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-08-13 16:00:32 +02:00
MWWorld : : Ptr target = getTarget ( ) ;
2014-01-11 12:06:36 +01:00
2014-08-13 16:00:32 +02:00
if ( target . isEmpty ( ) )
return true ; //Target doesn't exist
2014-01-11 12:06:36 +01:00
2014-08-14 01:20:12 +02:00
// Only the player can be actively followed. AiFollow packages with targets other than the player
// are only used for defining combat alliances, since NPCs will defend whoever they are following or being followed by.
if ( target ! = MWBase : : Environment : : get ( ) . getWorld ( ) - > getPlayerPtr ( ) )
return false ;
2014-07-28 17:28:00 +02:00
actor . getClass ( ) . getCreatureStats ( actor ) . setDrawState ( DrawState_Nothing ) ;
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-06-12 23:27:04 +02:00
//Check if we've run out of time
if ( mRemainingDuration ! = 0 )
{
mRemainingDuration - = duration ;
if ( duration < = 0 )
return true ;
}
2014-03-05 11:24:39 +01:00
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-06-12 23:27:04 +02:00
//Set the target destination from the actor
2014-05-12 21:05:32 -04:00
ESM : : Pathgrid : : Point dest = target . getRefData ( ) . getPosition ( ) . pos ;
2014-01-11 12:06:36 +01:00
2014-05-12 21:05:32 -04:00
if ( distance ( dest , pos . pos [ 0 ] , pos . pos [ 1 ] , pos . pos [ 2 ] ) < 100 ) //Stop when you get close
2014-01-29 20:29:07 +01:00
actor . getClass ( ) . getMovementSettings ( actor ) . mPosition [ 1 ] = 0 ;
2014-05-14 14:11:34 -04:00
else {
pathTo ( actor , dest , duration ) ; //Go to the destination
}
2014-01-11 12:06:36 +01:00
2014-04-23 02:57:48 -04:00
//Check if you're far away
2014-07-28 17:11:46 +02:00
if ( distance ( dest , pos . pos [ 0 ] , pos . pos [ 1 ] , pos . pos [ 2 ] ) > 450 )
2014-04-23 02:57:48 -04:00
actor . getClass ( ) . getCreatureStats ( actor ) . setMovementFlag ( MWMechanics : : CreatureStats : : Flag_Run , true ) ; //Make NPC run
2014-07-28 17:11:46 +02:00
else if ( distance ( dest , pos . pos [ 0 ] , pos . pos [ 1 ] , pos . pos [ 2 ] ) < 325 ) //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
2014-04-23 02:57:48 -04:00
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 ( )
{
2014-08-13 16:00:32 +02:00
return mActorRefId ;
2014-01-12 14:02:40 +01:00
}
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 ) ;
}
2014-06-12 23:27:04 +02:00
int MWMechanics : : AiFollow : : getTypeId ( ) const
2014-01-29 20:29:07 +01:00
{
return TypeIdFollow ;
2014-01-24 19:16:50 +01:00
}
2014-06-12 23:27:04 +02:00
2014-08-06 21:16:14 +02:00
bool MWMechanics : : AiFollow : : isCommanded ( ) const
{
return mCommanded ;
}
2014-06-12 23:27:04 +02:00
void MWMechanics : : AiFollow : : writeState ( ESM : : AiSequence : : AiSequence & sequence ) const
{
std : : auto_ptr < ESM : : AiSequence : : AiFollow > follow ( new ESM : : AiSequence : : AiFollow ( ) ) ;
follow - > mData . mX = mX ;
follow - > mData . mY = mY ;
follow - > mData . mZ = mZ ;
2014-08-13 16:00:32 +02:00
follow - > mTargetId = mActorRefId ;
2014-06-12 23:27:04 +02:00
follow - > mRemainingDuration = mRemainingDuration ;
follow - > mCellId = mCellId ;
follow - > mAlwaysFollow = mAlwaysFollow ;
2014-08-06 21:16:14 +02:00
follow - > mCommanded = mCommanded ;
2014-06-12 23:27:04 +02:00
ESM : : AiSequence : : AiPackageContainer package ;
package . mType = ESM : : AiSequence : : Ai_Follow ;
package . mPackage = follow . release ( ) ;
sequence . mPackages . push_back ( package ) ;
}
2014-08-13 16:00:32 +02:00
MWWorld : : Ptr MWMechanics : : AiFollow : : getTarget ( )
2014-07-27 20:30:52 +02:00
{
2014-08-14 01:08:09 +02:00
if ( mActorId = = - 2 )
return MWWorld : : Ptr ( ) ;
2014-08-13 16:00:32 +02:00
if ( mActorId = = - 1 )
{
MWWorld : : Ptr target = MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtr ( mActorRefId , false ) ;
2014-08-14 01:08:09 +02:00
if ( target . isEmpty ( ) )
{
mActorId = - 2 ;
return target ;
}
else
mActorId = target . getClass ( ) . getCreatureStats ( target ) . getActorId ( ) ;
2014-08-13 16:00:32 +02:00
}
if ( mActorId ! = - 1 )
return MWBase : : Environment : : get ( ) . getWorld ( ) - > searchPtrViaActorId ( mActorId ) ;
else
return MWWorld : : Ptr ( ) ;
2014-07-27 20:30:52 +02:00
}