2012-11-15 22:15:20 +01:00
|
|
|
#ifndef GAME_MWMECHANICS_AIWANDER_H
|
|
|
|
#define GAME_MWMECHANICS_AIWANDER_H
|
2012-11-14 18:42:04 +01:00
|
|
|
|
2020-05-17 22:10:36 +02:00
|
|
|
#include "typedaipackage.hpp"
|
2014-04-29 09:09:51 +02:00
|
|
|
|
2012-11-14 18:42:04 +01:00
|
|
|
#include <vector>
|
|
|
|
|
2013-05-26 08:49:44 -07:00
|
|
|
#include "../mwworld/timestamp.hpp"
|
|
|
|
|
2016-06-17 23:07:16 +09:00
|
|
|
#include "pathfinding.hpp"
|
|
|
|
#include "obstacle.hpp"
|
2014-10-08 10:58:52 +02:00
|
|
|
#include "aistate.hpp"
|
|
|
|
|
2014-06-12 23:27:04 +02:00
|
|
|
namespace ESM
|
|
|
|
{
|
2015-03-06 21:36:42 +13:00
|
|
|
struct Cell;
|
2014-06-12 23:27:04 +02:00
|
|
|
namespace AiSequence
|
|
|
|
{
|
|
|
|
struct AiWander;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-14 18:42:04 +01:00
|
|
|
namespace MWMechanics
|
2018-06-27 12:48:34 +04:00
|
|
|
{
|
|
|
|
/// \brief This class holds the variables AiWander needs which are deleted if the package becomes inactive.
|
|
|
|
struct AiWanderStorage : AiTemporaryBase
|
|
|
|
{
|
|
|
|
float mReaction; // update some actions infrequently
|
|
|
|
|
|
|
|
// AiWander states
|
|
|
|
enum WanderState
|
|
|
|
{
|
|
|
|
Wander_ChooseAction,
|
|
|
|
Wander_IdleNow,
|
|
|
|
Wander_MoveNow,
|
|
|
|
Wander_Walking
|
|
|
|
};
|
|
|
|
WanderState mState;
|
|
|
|
|
|
|
|
bool mIsWanderingManually;
|
|
|
|
bool mCanWanderAlongPathGrid;
|
|
|
|
|
|
|
|
unsigned short mIdleAnimation;
|
|
|
|
std::vector<unsigned short> mBadIdles; // Idle animations that when called cause errors
|
|
|
|
|
|
|
|
// do we need to calculate allowed nodes based on mDistance
|
|
|
|
bool mPopulateAvailableNodes;
|
|
|
|
|
|
|
|
// allowed pathgrid nodes based on mDistance from the spawn point
|
|
|
|
// in local coordinates of mCell
|
|
|
|
std::vector<ESM::Pathgrid::Point> mAllowedNodes;
|
|
|
|
|
|
|
|
ESM::Pathgrid::Point mCurrentNode;
|
|
|
|
bool mTrimCurrentNode;
|
|
|
|
|
2020-08-31 01:12:52 +02:00
|
|
|
float mCheckIdlePositionTimer;
|
2018-06-27 12:48:34 +04:00
|
|
|
int mStuckCount;
|
|
|
|
|
|
|
|
AiWanderStorage():
|
|
|
|
mReaction(0),
|
|
|
|
mState(Wander_ChooseAction),
|
|
|
|
mIsWanderingManually(false),
|
|
|
|
mCanWanderAlongPathGrid(true),
|
|
|
|
mIdleAnimation(0),
|
|
|
|
mBadIdles(),
|
|
|
|
mPopulateAvailableNodes(true),
|
|
|
|
mAllowedNodes(),
|
|
|
|
mTrimCurrentNode(false),
|
2020-08-31 01:12:52 +02:00
|
|
|
mCheckIdlePositionTimer(0),
|
2018-06-27 12:48:34 +04:00
|
|
|
mStuckCount(0)
|
|
|
|
{};
|
|
|
|
|
|
|
|
void setState(const WanderState wanderState, const bool isManualWander = false)
|
|
|
|
{
|
|
|
|
mState = wanderState;
|
|
|
|
mIsWanderingManually = isManualWander;
|
|
|
|
}
|
|
|
|
};
|
2014-12-01 13:34:42 +01:00
|
|
|
|
2014-04-29 23:40:59 -04:00
|
|
|
/// \brief Causes the Actor to wander within a specified range
|
2020-05-17 22:10:36 +02:00
|
|
|
class AiWander final : public TypedAiPackage<AiWander>
|
2012-11-16 18:38:15 +01:00
|
|
|
{
|
2013-05-24 04:49:20 -07:00
|
|
|
public:
|
2014-04-29 23:40:59 -04:00
|
|
|
/// Constructor
|
|
|
|
/** \param distance Max distance the ACtor will wander
|
|
|
|
\param duration Time, in hours, that this package will be preformed
|
2016-06-11 21:25:40 +02:00
|
|
|
\param timeOfDay Currently unimplemented. Not functional in the original engine.
|
2014-04-29 23:40:59 -04:00
|
|
|
\param idle Chances of each idle to play (9 in total)
|
|
|
|
\param repeat Repeat wander or not **/
|
2014-06-12 23:27:04 +02:00
|
|
|
AiWander(int distance, int duration, int timeOfDay, const std::vector<unsigned char>& idle, bool repeat);
|
|
|
|
|
2020-10-22 23:57:53 +02:00
|
|
|
explicit AiWander (const ESM::AiSequence::AiWander* wander);
|
2014-06-12 23:27:04 +02:00
|
|
|
|
2020-10-22 23:57:53 +02:00
|
|
|
bool execute(const MWWorld::Ptr& actor, CharacterController& characterController, AiState& state, float duration) override;
|
2014-04-29 23:40:59 -04:00
|
|
|
|
2020-05-16 21:52:16 +02:00
|
|
|
static constexpr AiPackageTypeId getTypeId() { return AiPackageTypeId::Wander; }
|
2012-11-14 18:42:04 +01:00
|
|
|
|
2020-05-16 21:08:39 +02:00
|
|
|
static constexpr Options makeDefaultOptions()
|
|
|
|
{
|
|
|
|
AiPackage::Options options;
|
|
|
|
options.mUseVariableSpeed = true;
|
|
|
|
return options;
|
|
|
|
}
|
2020-02-11 12:33:22 +03:00
|
|
|
|
2020-10-22 23:57:53 +02:00
|
|
|
void writeState(ESM::AiSequence::AiSequence &sequence) const override;
|
2014-06-12 23:27:04 +02:00
|
|
|
|
2020-10-22 23:57:53 +02:00
|
|
|
void fastForward(const MWWorld::Ptr& actor, AiState& state) override;
|
2017-12-01 18:53:31 +04:00
|
|
|
|
2020-10-22 23:57:53 +02:00
|
|
|
osg::Vec3f getDestination(const MWWorld::Ptr& actor) const override;
|
2017-12-01 18:53:31 +04:00
|
|
|
|
2020-10-22 23:57:53 +02:00
|
|
|
osg::Vec3f getDestination() const override
|
2020-02-11 12:33:22 +03:00
|
|
|
{
|
|
|
|
if (!mHasDestination)
|
|
|
|
return osg::Vec3f(0, 0, 0);
|
|
|
|
|
|
|
|
return mDestination;
|
|
|
|
}
|
|
|
|
|
2020-08-31 01:12:52 +02:00
|
|
|
bool isStationary() const { return mDistance == 0; }
|
|
|
|
|
2013-05-24 04:49:20 -07:00
|
|
|
private:
|
2020-05-16 14:57:20 +02:00
|
|
|
void stopWalking(const MWWorld::Ptr& actor);
|
2015-07-30 08:08:58 -04:00
|
|
|
|
2015-07-30 08:20:16 -04:00
|
|
|
/// Have the given actor play an idle animation
|
|
|
|
/// @return Success or error
|
2015-07-30 08:08:58 -04:00
|
|
|
bool playIdle(const MWWorld::Ptr& actor, unsigned short idleSelect);
|
2013-05-26 08:49:44 -07:00
|
|
|
bool checkIdle(const MWWorld::Ptr& actor, unsigned short idleSelect);
|
2015-07-19 18:04:42 +12:00
|
|
|
short unsigned getRandomIdle();
|
2015-07-12 16:35:15 +12:00
|
|
|
void setPathToAnAllowedNode(const MWWorld::Ptr& actor, AiWanderStorage& storage, const ESM::Position& actorPos);
|
2020-05-16 17:07:42 +02:00
|
|
|
void evadeObstacles(const MWWorld::Ptr& actor, AiWanderStorage& storage);
|
2015-07-19 18:00:49 +12:00
|
|
|
void turnActorToFacePlayer(const osg::Vec3f& actorPosition, const osg::Vec3f& playerPosition, AiWanderStorage& storage);
|
2018-08-18 18:48:34 +03:00
|
|
|
void doPerFrameActionsForState(const MWWorld::Ptr& actor, float duration, AiWanderStorage& storage);
|
2015-07-26 17:24:33 +12:00
|
|
|
void onIdleStatePerFrameActions(const MWWorld::Ptr& actor, float duration, AiWanderStorage& storage);
|
2018-08-18 18:48:34 +03:00
|
|
|
void onWalkingStatePerFrameActions(const MWWorld::Ptr& actor, float duration, AiWanderStorage& storage);
|
2015-07-26 17:25:44 +12:00
|
|
|
void onChooseActionStatePerFrameActions(const MWWorld::Ptr& actor, AiWanderStorage& storage);
|
2020-02-02 11:02:19 +04:00
|
|
|
bool reactionTimeActions(const MWWorld::Ptr& actor, AiWanderStorage& storage, ESM::Position& pos);
|
2020-05-16 14:58:50 +02:00
|
|
|
inline bool isPackageCompleted() const;
|
2016-04-15 21:56:41 -05:00
|
|
|
void wanderNearStart(const MWWorld::Ptr &actor, AiWanderStorage &storage, int wanderDistance);
|
2016-04-16 16:39:13 -05:00
|
|
|
bool destinationIsAtWater(const MWWorld::Ptr &actor, const osg::Vec3f& destination);
|
2016-04-16 20:38:58 -05:00
|
|
|
void completeManualWalking(const MWWorld::Ptr &actor, AiWanderStorage &storage);
|
2020-08-31 01:12:52 +02:00
|
|
|
bool isNearAllowedNode(const MWWorld::Ptr &actor, const AiWanderStorage& storage, float distance) const;
|
2013-05-26 08:49:44 -07:00
|
|
|
|
2020-06-02 21:30:46 +02:00
|
|
|
const int mDistance; // how far the actor can wander from the spawn point
|
|
|
|
const int mDuration;
|
2016-06-10 00:56:01 +09:00
|
|
|
float mRemainingDuration;
|
2020-06-02 21:30:46 +02:00
|
|
|
const int mTimeOfDay;
|
|
|
|
const std::vector<unsigned char> mIdle;
|
2014-01-19 17:03:25 +01:00
|
|
|
|
2014-12-31 21:27:19 +01:00
|
|
|
bool mStoredInitialActorPosition;
|
2020-02-02 11:02:19 +04:00
|
|
|
osg::Vec3f mInitialActorPosition; // Note: an original engine does not reset coordinates even when actor changes a cell
|
2017-04-30 22:29:59 +09:00
|
|
|
|
|
|
|
bool mHasDestination;
|
|
|
|
osg::Vec3f mDestination;
|
2018-08-23 00:20:25 +03:00
|
|
|
bool mUsePathgrid;
|
2014-10-08 22:11:45 +02:00
|
|
|
|
2017-11-11 12:31:18 +04:00
|
|
|
void getNeighbouringNodes(ESM::Pathgrid::Point dest, const MWWorld::CellStore* currentCell, ESM::Pathgrid::PointList& points);
|
|
|
|
|
2016-04-15 21:56:41 -05:00
|
|
|
void getAllowedNodes(const MWWorld::Ptr& actor, const ESM::Cell* cell, AiWanderStorage& storage);
|
2016-08-19 22:15:26 +03:00
|
|
|
|
2016-06-16 03:43:09 +09:00
|
|
|
void trimAllowedNodes(std::vector<ESM::Pathgrid::Point>& nodes, const PathFinder& pathfinder);
|
2014-04-20 10:06:03 +10:00
|
|
|
|
2015-03-23 20:57:36 +13:00
|
|
|
// constants for converting idleSelect values into groupNames
|
|
|
|
enum GroupIndex
|
|
|
|
{
|
|
|
|
GroupIndex_MinIdle = 2,
|
|
|
|
GroupIndex_MaxIdle = 9
|
|
|
|
};
|
2014-10-08 10:58:52 +02:00
|
|
|
|
2016-12-14 22:11:22 +01:00
|
|
|
/// convert point from local (i.e. cell) to world coordinates
|
2015-07-05 18:07:14 +12:00
|
|
|
void ToWorldCoordinates(ESM::Pathgrid::Point& point, const ESM::Cell * cell);
|
|
|
|
|
2017-04-20 20:36:14 +09:00
|
|
|
void SetCurrentNodeToClosestAllowedNode(const osg::Vec3f& npcPos, AiWanderStorage& storage);
|
2015-07-05 18:17:18 +12:00
|
|
|
|
2016-06-16 03:43:09 +09:00
|
|
|
void AddNonPathGridAllowedPoints(osg::Vec3f npcPos, const ESM::Pathgrid * pathGrid, int pointIndex, AiWanderStorage& storage);
|
2015-07-05 18:21:35 +12:00
|
|
|
|
2016-06-16 03:43:09 +09:00
|
|
|
void AddPointBetweenPathGridPoints(const ESM::Pathgrid::Point& start, const ESM::Pathgrid::Point& end, AiWanderStorage& storage);
|
2015-07-05 18:21:35 +12:00
|
|
|
|
2015-03-23 20:57:36 +13:00
|
|
|
/// lookup table for converting idleSelect value to groupName
|
|
|
|
static const std::string sIdleSelectToGroupName[GroupIndex_MaxIdle - GroupIndex_MinIdle + 1];
|
2015-07-08 18:41:03 +12:00
|
|
|
|
|
|
|
static int OffsetToPreventOvercrowding();
|
2020-05-16 21:08:39 +02:00
|
|
|
};
|
2013-05-24 04:49:20 -07:00
|
|
|
}
|
2012-11-14 18:42:04 +01:00
|
|
|
|
2012-11-16 20:28:20 +01:00
|
|
|
#endif
|