2014-01-24 19:16:50 +01:00
# include "aifollow.hpp"
2014-06-12 23:27:04 +02:00
# include <components/esm/aisequence.hpp>
2015-07-25 04:14:22 +02:00
# include <components/esm/loadcell.hpp>
2014-06-12 23:27:04 +02:00
2014-01-11 12:06:36 +01:00
# include "../mwbase/world.hpp"
# include "../mwbase/environment.hpp"
2014-12-09 16:02:07 +01:00
# include "../mwbase/mechanicsmanager.hpp"
2016-06-17 23:07:16 +09:00
2014-01-11 12:06:36 +01:00
# include "../mwworld/class.hpp"
2014-03-29 18:36:32 +01:00
# include "../mwworld/cellstore.hpp"
2016-06-17 23:07:16 +09:00
2014-04-23 02:57:48 -04:00
# include "creaturestats.hpp"
2014-01-11 12:06:36 +01:00
# include "movement.hpp"
2014-01-29 20:29:07 +01:00
# include "steering.hpp"
2014-12-09 22:25:28 +01:00
namespace MWMechanics
{
int AiFollow : : mFollowIndexCounter = 0 ;
2017-11-21 20:00:51 +04:00
AiFollow : : AiFollow ( const std : : string & actorId , float duration , float x , float y , float z )
2020-05-16 21:08:39 +02:00
: mAlwaysFollow ( false ) , mDuration ( duration ) , mRemainingDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z )
2017-11-21 20:00:51 +04:00
, mCellId ( " " ) , mActive ( false ) , mFollowIndex ( mFollowIndexCounter + + )
2014-01-29 20:29:07 +01:00
{
2017-11-21 20:00:51 +04:00
mTargetActorRefId = actorId ;
2014-01-29 20:29:07 +01:00
}
2016-06-11 22:34:49 +09:00
2017-11-21 20:00:51 +04:00
AiFollow : : AiFollow ( const std : : string & actorId , const std : : string & cellId , float duration , float x , float y , float z )
2020-05-16 21:08:39 +02:00
: mAlwaysFollow ( false ) , mDuration ( duration ) , mRemainingDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z )
2017-11-21 20:00:51 +04:00
, mCellId ( cellId ) , mActive ( false ) , mFollowIndex ( mFollowIndexCounter + + )
2014-03-05 11:24:39 +01:00
{
2017-11-21 20:00:51 +04:00
mTargetActorRefId = actorId ;
2014-03-05 11:24:39 +01:00
}
2017-11-21 20:00:51 +04:00
AiFollow : : AiFollow ( const MWWorld : : Ptr & actor , float duration , float x , float y , float z )
2020-05-16 21:08:39 +02:00
: mAlwaysFollow ( false ) , mDuration ( duration ) , mRemainingDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z )
2017-11-21 20:00:51 +04:00
, mCellId ( " " ) , mActive ( false ) , mFollowIndex ( mFollowIndexCounter + + )
{
mTargetActorRefId = actor . getCellRef ( ) . getRefId ( ) ;
mTargetActorId = actor . getClass ( ) . getCreatureStats ( actor ) . getActorId ( ) ;
}
AiFollow : : AiFollow ( const MWWorld : : Ptr & actor , const std : : string & cellId , float duration , float x , float y , float z )
2020-05-16 21:08:39 +02:00
: mAlwaysFollow ( false ) , mDuration ( duration ) , mRemainingDuration ( duration ) , mX ( x ) , mY ( y ) , mZ ( z )
2017-11-21 20:00:51 +04:00
, mCellId ( cellId ) , mActive ( false ) , mFollowIndex ( mFollowIndexCounter + + )
{
mTargetActorRefId = actor . getCellRef ( ) . getRefId ( ) ;
mTargetActorId = actor . getClass ( ) . getCreatureStats ( actor ) . getActorId ( ) ;
}
AiFollow : : AiFollow ( const MWWorld : : Ptr & actor , bool commanded )
2020-05-16 21:08:39 +02:00
: TypedAiPackage < AiFollow > ( makeDefaultOptions ( ) . withShouldCancelPreviousAi ( ! commanded ) )
, mAlwaysFollow ( true ) , mDuration ( 0 ) , mRemainingDuration ( 0 ) , mX ( 0 ) , mY ( 0 ) , mZ ( 0 )
2017-11-21 20:00:51 +04:00
, mCellId ( " " ) , mActive ( false ) , mFollowIndex ( mFollowIndexCounter + + )
2014-01-29 20:29:07 +01:00
{
2017-11-21 20:00:51 +04:00
mTargetActorRefId = actor . getCellRef ( ) . getRefId ( ) ;
mTargetActorId = actor . getClass ( ) . getCreatureStats ( actor ) . getActorId ( ) ;
2014-01-29 20:29:07 +01:00
}
2012-11-15 22:32:15 +01:00
2014-12-09 22:25:28 +01:00
AiFollow : : AiFollow ( const ESM : : AiSequence : : AiFollow * follow )
2020-05-16 21:08:39 +02:00
: TypedAiPackage < AiFollow > ( makeDefaultOptions ( ) . withShouldCancelPreviousAi ( ! follow - > mCommanded ) )
, mAlwaysFollow ( follow - > mAlwaysFollow )
2020-06-02 21:30:46 +02:00
// mDuration isn't saved in the save file, so just giving it "1" for now if the package had a duration.
// The exact value of mDuration only matters for repeating packages.
// Previously mRemainingDuration could be negative even when mDuration was 0. Checking for > 0 should fix old saves.
, mDuration ( follow - > mRemainingDuration )
, mRemainingDuration ( follow - > mRemainingDuration )
2014-08-17 17:01:04 +02:00
, mX ( follow - > mData . mX ) , mY ( follow - > mData . mY ) , mZ ( follow - > mData . mZ )
2015-04-30 19:24:27 -05:00
, mCellId ( follow - > mCellId ) , mActive ( follow - > mActive ) , mFollowIndex ( mFollowIndexCounter + + )
2014-08-17 17:01:04 +02:00
{
2017-11-21 20:00:51 +04:00
mTargetActorRefId = follow - > mTargetId ;
mTargetActorId = follow - > mTargetActorId ;
2014-08-17 17:01:04 +02:00
}
2015-06-26 17:47:04 +02:00
bool AiFollow : : execute ( const MWWorld : : Ptr & actor , CharacterController & characterController , AiState & state , float duration )
2012-11-15 22:32:15 +01:00
{
2018-11-02 14:24:43 +03:00
const MWWorld : : Ptr target = getTarget ( ) ;
2014-01-11 12:06:36 +01:00
2018-11-02 14:24:43 +03:00
// Target is not here right now, wait for it to return
// Really we should be checking whether the target is currently registered with the MechanicsManager
if ( target = = MWWorld : : Ptr ( ) | | ! target . getRefData ( ) . getCount ( ) | | ! target . getRefData ( ) . isEnabled ( ) )
return false ;
2014-01-11 12:06:36 +01:00
2014-07-28 17:28:00 +02:00
actor . getClass ( ) . getCreatureStats ( actor ) . setDrawState ( DrawState_Nothing ) ;
2015-09-26 01:55:12 +02:00
AiFollowStorage & storage = state . get < AiFollowStorage > ( ) ;
2017-08-13 18:05:35 +04:00
bool & rotate = storage . mTurnActorToTarget ;
if ( rotate )
{
if ( zTurn ( actor , storage . mTargetAngleRadians ) )
rotate = false ;
return false ;
}
2018-11-02 14:24:43 +03:00
const osg : : Vec3f actorPos ( actor . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
const osg : : Vec3f targetPos ( target . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
const osg : : Vec3f targetDir = targetPos - actorPos ;
2014-12-09 22:25:28 +01:00
// AiFollow requires the target to be in range and within sight for the initial activation
if ( ! mActive )
{
storage . mTimer - = duration ;
if ( storage . mTimer < 0 )
{
2018-11-02 14:24:43 +03:00
if ( targetDir . length2 ( ) < 500 * 500 & & MWBase : : Environment : : get ( ) . getWorld ( ) - > getLOS ( actor , target ) )
2014-12-09 22:25:28 +01:00
mActive = true ;
storage . mTimer = 0.5f ;
}
}
if ( ! mActive )
return false ;
2017-03-26 16:51:32 +09:00
// The distances below are approximations based on observations of the original engine.
// If only one actor is following the target, it uses 186.
// If there are multiple actors following the same target, they form a group with each group member at 313 + (130 * i) distance to the target.
short followDistance = 186 ;
2014-12-09 16:02:07 +01:00
std : : list < int > followers = MWBase : : Environment : : get ( ) . getMechanicsManager ( ) - > getActorsFollowingIndices ( target ) ;
2017-03-26 16:51:32 +09:00
if ( followers . size ( ) > = 2 )
2014-12-09 16:02:07 +01:00
{
2017-03-26 16:51:32 +09:00
followDistance = 313 ;
short i = 0 ;
followers . sort ( ) ;
for ( std : : list < int > : : iterator it = followers . begin ( ) ; it ! = followers . end ( ) ; + + it )
{
if ( * it = = mFollowIndex )
followDistance + = 130 * i ;
+ + i ;
}
2014-12-09 16:02:07 +01:00
}
2017-02-18 01:38:21 +09:00
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
2017-02-18 01:38:21 +09:00
if ( mDuration > 0 )
2014-06-12 23:27:04 +02:00
{
2016-06-11 22:34:49 +09:00
mRemainingDuration - = ( ( duration * MWBase : : Environment : : get ( ) . getWorld ( ) - > getTimeScaleFactor ( ) ) / 3600 ) ;
if ( mRemainingDuration < = 0 )
{
mRemainingDuration = mDuration ;
2014-06-12 23:27:04 +02:00
return true ;
2016-06-11 22:34:49 +09:00
}
2014-06-12 23:27:04 +02:00
}
2014-03-05 11:24:39 +01:00
2018-11-02 14:24:43 +03:00
osg : : Vec3f finalPos ( mX , mY , mZ ) ;
if ( ( actorPos - finalPos ) . length2 ( ) < followDistance * followDistance ) //Close-ish to final position
2014-01-12 11:38:58 +01:00
{
2017-02-18 01:38:21 +09:00
if ( actor . getCell ( ) - > isExterior ( ) ) //Outside?
2014-03-05 11:24:39 +01:00
{
2017-02-18 01:38:21 +09:00
if ( mCellId = = " " ) //No cell to travel to
2014-03-05 11:24:39 +01:00
return true ;
}
else
{
2017-02-18 01:38:21 +09: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
2017-08-13 18:05:35 +04:00
short baseFollowDistance = followDistance ;
short threshold = 30 ; // to avoid constant switching between moving/stopping
if ( storage . mMoving )
followDistance - = threshold ;
else
2016-08-19 22:15:26 +03:00
followDistance + = threshold ;
2017-08-13 18:05:35 +04:00
2018-11-02 14:24:43 +03:00
if ( targetDir . length2 ( ) < = followDistance * followDistance )
2017-08-13 18:05:35 +04:00
{
2018-11-02 14:24:43 +03:00
float faceAngleRadians = std : : atan2 ( targetDir . x ( ) , targetDir . y ( ) ) ;
2017-08-13 18:05:35 +04:00
if ( ! zTurn ( actor , faceAngleRadians , osg : : DegreesToRadians ( 45.f ) ) )
{
storage . mTargetAngleRadians = faceAngleRadians ;
storage . mTurnActorToTarget = true ;
}
return false ;
2016-01-03 18:20:34 +01:00
}
2015-09-26 01:55:12 +02:00
2018-11-02 14:24:43 +03:00
storage . mMoving = ! pathTo ( actor , targetPos , duration , baseFollowDistance ) ; // Go to the destination
2016-01-01 16:41:45 +03:00
2016-08-19 22:15:26 +03:00
if ( storage . mMoving )
2016-08-14 18:02:13 +02:00
{
2016-08-19 22:15:26 +03:00
//Check if you're far away
2018-11-02 14:24:43 +03:00
if ( targetDir . length2 ( ) > 450 * 450 )
2016-08-19 22:15:26 +03:00
actor . getClass ( ) . getCreatureStats ( actor ) . setMovementFlag ( MWMechanics : : CreatureStats : : Flag_Run , true ) ; //Make NPC run
2018-11-02 14:24:43 +03:00
else if ( targetDir . length2 ( ) < 325 * 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 threshold
2016-08-19 22:15:26 +03:00
actor . getClass ( ) . getCreatureStats ( actor ) . setMovementFlag ( MWMechanics : : CreatureStats : : Flag_Run , false ) ; //make NPC walk
}
2014-04-23 02:57:48 -04:00
2014-01-11 12:06:36 +01:00
return false ;
2012-11-15 22:32:15 +01:00
}
2014-12-09 22:25:28 +01:00
std : : string AiFollow : : getFollowedActor ( )
2014-01-12 14:02:40 +01:00
{
2017-11-21 20:00:51 +04:00
return mTargetActorRefId ;
2014-01-12 14:02:40 +01:00
}
2014-01-24 19:16:50 +01:00
2014-12-09 22:25:28 +01:00
bool AiFollow : : isCommanded ( ) const
2014-08-06 21:16:14 +02:00
{
2020-05-16 21:08:39 +02:00
return ! mOptions . mShouldCancelPreviousAi ;
2014-08-06 21:16:14 +02:00
}
2014-12-09 22:25:28 +01:00
void AiFollow : : writeState ( ESM : : AiSequence : : AiSequence & sequence ) const
2014-06-12 23:27:04 +02:00
{
2017-04-28 17:30:26 +02:00
std : : unique_ptr < ESM : : AiSequence : : AiFollow > follow ( new ESM : : AiSequence : : AiFollow ( ) ) ;
2014-06-12 23:27:04 +02:00
follow - > mData . mX = mX ;
follow - > mData . mY = mY ;
follow - > mData . mZ = mZ ;
2017-11-21 20:00:51 +04:00
follow - > mTargetId = mTargetActorRefId ;
follow - > mTargetActorId = mTargetActorId ;
2014-06-12 23:27:04 +02:00
follow - > mRemainingDuration = mRemainingDuration ;
follow - > mCellId = mCellId ;
follow - > mAlwaysFollow = mAlwaysFollow ;
2020-05-16 21:08:39 +02:00
follow - > mCommanded = isCommanded ( ) ;
2014-12-09 22:25:28 +01:00
follow - > mActive = mActive ;
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-12-09 22:25:28 +01:00
int AiFollow : : getFollowIndex ( ) const
2014-12-09 16:02:07 +01:00
{
return mFollowIndex ;
}
2014-12-09 22:25:28 +01:00
2016-06-11 22:34:49 +09:00
void AiFollow : : fastForward ( const MWWorld : : Ptr & actor , AiState & state )
{
2017-02-18 01:38:21 +09:00
// Update duration counter if this package has a duration
if ( mDuration > 0 )
mRemainingDuration - - ;
2016-06-11 22:34:49 +09:00
}
2014-12-09 22:25:28 +01:00
}