mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-26 09:35:28 +00:00
675c0ab72f
This allows to distribute AI reaction calls over time. Before this change actors appearing at the same frame will react in the same frame over and over because AI reaction period is constant. It creates a non-uniform CPU usage over frames. If a single frame has too many AI reactions it may cause stuttering when there are too many actors on a scene for current system. Another concern is a synchronization of actions between creatures and NPC. They start to go or hit at the same frame that is unnatural.
146 lines
4.5 KiB
C++
146 lines
4.5 KiB
C++
#ifndef GAME_MWMECHANICS_AICOMBAT_H
|
|
#define GAME_MWMECHANICS_AICOMBAT_H
|
|
|
|
#include "typedaipackage.hpp"
|
|
|
|
#include "../mwworld/cellstore.hpp" // for Doors
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "pathfinding.hpp"
|
|
#include "movement.hpp"
|
|
#include "obstacle.hpp"
|
|
#include "aitimer.hpp"
|
|
|
|
namespace ESM
|
|
{
|
|
namespace AiSequence
|
|
{
|
|
struct AiCombat;
|
|
}
|
|
}
|
|
|
|
namespace MWMechanics
|
|
{
|
|
class Action;
|
|
|
|
/// \brief This class holds the variables AiCombat needs which are deleted if the package becomes inactive.
|
|
struct AiCombatStorage : AiTemporaryBase
|
|
{
|
|
float mAttackCooldown;
|
|
AiReactionTimer mReaction;
|
|
float mTimerCombatMove;
|
|
bool mReadyToAttack;
|
|
bool mAttack;
|
|
float mAttackRange;
|
|
bool mCombatMove;
|
|
osg::Vec3f mLastTargetPos;
|
|
const MWWorld::CellStore* mCell;
|
|
std::shared_ptr<Action> mCurrentAction;
|
|
float mActionCooldown;
|
|
float mStrength;
|
|
bool mForceNoShortcut;
|
|
ESM::Position mShortcutFailPos;
|
|
MWMechanics::Movement mMovement;
|
|
|
|
enum FleeState
|
|
{
|
|
FleeState_None,
|
|
FleeState_Idle,
|
|
FleeState_RunBlindly,
|
|
FleeState_RunToDestination
|
|
};
|
|
FleeState mFleeState;
|
|
bool mLOS;
|
|
float mUpdateLOSTimer;
|
|
float mFleeBlindRunTimer;
|
|
ESM::Pathgrid::Point mFleeDest;
|
|
|
|
bool mUseCustomDestination;
|
|
osg::Vec3f mCustomDestination;
|
|
|
|
AiCombatStorage():
|
|
mAttackCooldown(0.0f),
|
|
mTimerCombatMove(0.0f),
|
|
mReadyToAttack(false),
|
|
mAttack(false),
|
|
mAttackRange(0.0f),
|
|
mCombatMove(false),
|
|
mLastTargetPos(0,0,0),
|
|
mCell(nullptr),
|
|
mCurrentAction(),
|
|
mActionCooldown(0.0f),
|
|
mStrength(),
|
|
mForceNoShortcut(false),
|
|
mShortcutFailPos(),
|
|
mMovement(),
|
|
mFleeState(FleeState_None),
|
|
mLOS(false),
|
|
mUpdateLOSTimer(0.0f),
|
|
mFleeBlindRunTimer(0.0f),
|
|
mUseCustomDestination(false),
|
|
mCustomDestination()
|
|
{}
|
|
|
|
void startCombatMove(bool isDistantCombat, float distToTarget, float rangeAttack, const MWWorld::Ptr& actor, const MWWorld::Ptr& target);
|
|
void updateCombatMove(float duration);
|
|
void stopCombatMove();
|
|
void startAttackIfReady(const MWWorld::Ptr& actor, CharacterController& characterController,
|
|
const ESM::Weapon* weapon, bool distantCombat);
|
|
void updateAttack(CharacterController& characterController);
|
|
void stopAttack();
|
|
|
|
void startFleeing();
|
|
void stopFleeing();
|
|
bool isFleeing();
|
|
};
|
|
|
|
/// \brief Causes the actor to fight another actor
|
|
class AiCombat final : public TypedAiPackage<AiCombat>
|
|
{
|
|
public:
|
|
///Constructor
|
|
/** \param actor Actor to fight **/
|
|
explicit AiCombat(const MWWorld::Ptr& actor);
|
|
|
|
explicit AiCombat (const ESM::AiSequence::AiCombat* combat);
|
|
|
|
void init();
|
|
|
|
bool execute (const MWWorld::Ptr& actor, CharacterController& characterController, AiState& state, float duration) override;
|
|
|
|
static constexpr AiPackageTypeId getTypeId() { return AiPackageTypeId::Combat; }
|
|
|
|
static constexpr Options makeDefaultOptions()
|
|
{
|
|
AiPackage::Options options;
|
|
options.mPriority = 1;
|
|
options.mCanCancel = false;
|
|
options.mShouldCancelPreviousAi = false;
|
|
return options;
|
|
}
|
|
|
|
///Returns target ID
|
|
MWWorld::Ptr getTarget() const override;
|
|
|
|
void writeState(ESM::AiSequence::AiSequence &sequence) const override;
|
|
|
|
private:
|
|
/// Returns true if combat should end
|
|
bool attack(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, AiCombatStorage& storage, CharacterController& characterController);
|
|
|
|
void updateLOS(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, float duration, AiCombatStorage& storage);
|
|
|
|
void updateFleeing(const MWWorld::Ptr& actor, const MWWorld::Ptr& target, float duration, AiCombatStorage& storage);
|
|
|
|
/// Transfer desired movement (from AiCombatStorage) to Actor
|
|
void updateActorsMovement(const MWWorld::Ptr& actor, float duration, AiCombatStorage& storage);
|
|
void rotateActorOnAxis(const MWWorld::Ptr& actor, int axis,
|
|
MWMechanics::Movement& actorMovementSettings, AiCombatStorage& storage);
|
|
};
|
|
|
|
|
|
}
|
|
|
|
#endif
|