#ifndef GAME_MWMECHANICS_AIWANDER_H #define GAME_MWMECHANICS_AIWANDER_H #include "aipackage.hpp" #include #include #include "pathfinding.hpp" #include "obstacle.hpp" #include "../mwworld/timestamp.hpp" namespace ESM { namespace AiSequence { struct AiWander; } } namespace MWMechanics { /// \brief Causes the Actor to wander within a specified range class AiWander : public AiPackage { public: /// Constructor /** \param distance Max distance the ACtor will wander \param duration Time, in hours, that this package will be preformed \param timeOfDay Start time of the package, if it has a duration. Currently unimplemented \param idle Chances of each idle to play (9 in total) \param repeat Repeat wander or not **/ AiWander(int distance, int duration, int timeOfDay, const std::vector& idle, bool repeat); AiWander (const ESM::AiSequence::AiWander* wander); // NOTE: mDistance and mDuration must be set already void init(); virtual AiPackage *clone() const; virtual bool execute (const MWWorld::Ptr& actor,float duration); virtual int getTypeId() const; /// Set the position to return to for a stationary (non-wandering) actor /** In case another AI package moved the actor elsewhere **/ void setReturnPosition (const Ogre::Vector3& position); virtual void writeState(ESM::AiSequence::AiSequence &sequence) const; private: void stopWalking(const MWWorld::Ptr& actor); void playIdle(const MWWorld::Ptr& actor, unsigned short idleSelect); bool checkIdle(const MWWorld::Ptr& actor, unsigned short idleSelect); void getRandomIdle(); int mDistance; // how far the actor can wander from the spawn point int mDuration; int mTimeOfDay; std::vector mIdle; bool mRepeat; enum GreetingState { Greet_None, Greet_InProgress, Greet_Done }; GreetingState mSaidGreeting; int mGreetingTimer; bool mHasReturnPosition; // NOTE: Could be removed if mReturnPosition was initialized to actor position, // if we had the actor in the AiWander constructor... Ogre::Vector3 mReturnPosition; // Cached current cell location int mCellX; int mCellY; // Cell location multiplied by ESM::Land::REAL_SIZE float mXCell; float mYCell; const MWWorld::CellStore* mCell; // for detecting cell change // if false triggers calculating allowed nodes based on mDistance bool mStoredAvailableNodes; // AiWander states bool mChooseAction; bool mIdleNow; bool mMoveNow; bool mWalking; unsigned short mPlayedIdle; MWWorld::TimeStamp mStartTime; // allowed pathgrid nodes based on mDistance from the spawn point std::vector mAllowedNodes; ESM::Pathgrid::Point mCurrentNode; bool mTrimCurrentNode; void trimAllowedNodes(std::vector& nodes, const PathFinder& pathfinder); PathFinder mPathFinder; ObstacleCheck mObstacleCheck; float mDoorCheckDuration; int mStuckCount; // the z rotation angle (degrees) we want to reach // used every frame when mRotate is true float mTargetAngle; bool mRotate; float mReaction; // update some actions infrequently }; } #endif