mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-27 12:35:46 +00:00
177 lines
6.3 KiB
C++
177 lines
6.3 KiB
C++
|
#include "obstacle.hpp"
|
||
|
|
||
|
#include <OgreVector3.h>
|
||
|
|
||
|
#include "../mwbase/world.hpp"
|
||
|
#include "../mwworld/class.hpp"
|
||
|
#include "../mwworld/cellstore.hpp"
|
||
|
|
||
|
namespace MWMechanics
|
||
|
{
|
||
|
// NOTE: determined empirically but probably need further tweaking
|
||
|
static const float DIST_SAME_SPOT = 1.8f;
|
||
|
static const float DURATION_SAME_SPOT = 1.0f;
|
||
|
static const float DURATION_TO_EVADE = 0.4f;
|
||
|
|
||
|
// Proximity check function for interior doors. Given that most interior cells
|
||
|
// do not have many doors performance shouldn't be too much of an issue.
|
||
|
//
|
||
|
// Limitation: there can be false detections, and does not test whether the
|
||
|
// actor is facing the door.
|
||
|
bool proximityToDoor(const MWWorld::Ptr& actor, float minSqr, bool closed)
|
||
|
{
|
||
|
MWWorld::CellStore *cell = actor.getCell();
|
||
|
|
||
|
if(cell->getCell()->isExterior())
|
||
|
return false; // check interior cells only
|
||
|
|
||
|
// Check all the doors in this cell
|
||
|
MWWorld::CellRefList<ESM::Door>& doors = cell->get<ESM::Door>();
|
||
|
MWWorld::CellRefList<ESM::Door>::List& refList = doors.mList;
|
||
|
MWWorld::CellRefList<ESM::Door>::List::iterator it = refList.begin();
|
||
|
Ogre::Vector3 pos(actor.getRefData().getPosition().pos);
|
||
|
|
||
|
// TODO: How to check whether the actor is facing a door? Below code is for
|
||
|
// the player, perhaps it can be adapted.
|
||
|
//MWWorld::Ptr ptr = MWBase::Environment::get().getWorld()->getFacedObject();
|
||
|
//if(!ptr.isEmpty())
|
||
|
//std::cout << "faced door " << ptr.getClass().getName(ptr) << std::endl;
|
||
|
|
||
|
// TODO: The in-game observation of rot[2] value seems to be the
|
||
|
// opposite of the code in World::activateDoor() ::confused::
|
||
|
for (; it != refList.end(); ++it)
|
||
|
{
|
||
|
MWWorld::LiveCellRef<ESM::Door>& ref = *it;
|
||
|
if(pos.squaredDistance(Ogre::Vector3(ref.mRef.mPos.pos)) < minSqr &&
|
||
|
ref.mData.getLocalRotation().rot[2] == (closed ? 0 : 1))
|
||
|
{
|
||
|
//#if 0
|
||
|
std::cout << "\""+actor.getClass().getName(actor)+"\" "
|
||
|
<<"next to door "+ref.mRef.mRefID
|
||
|
//+", enabled? "+std::to_string(ref.mData.isEnabled())
|
||
|
+", dist "+std::to_string(sqrt(pos.squaredDistance(Ogre::Vector3(ref.mRef.mPos.pos))))
|
||
|
<< std::endl;
|
||
|
//#endif
|
||
|
return true; // found, stop searching
|
||
|
}
|
||
|
}
|
||
|
return false; // none found
|
||
|
}
|
||
|
|
||
|
ObstacleCheck::ObstacleCheck():
|
||
|
mPrevX(0) // to see if the moved since last time
|
||
|
, mPrevY(0)
|
||
|
, mDistSameSpot(-1) // avoid calculating it each time
|
||
|
, mWalkState(State_Norm)
|
||
|
, mStuckDuration(0)
|
||
|
, mEvadeDuration(0)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void ObstacleCheck::clear()
|
||
|
{
|
||
|
mWalkState = State_Norm;
|
||
|
mStuckDuration = 0;
|
||
|
mEvadeDuration = 0;
|
||
|
}
|
||
|
|
||
|
bool ObstacleCheck::isNormalState() const
|
||
|
{
|
||
|
return mWalkState == State_Norm;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* input - actor, duration (time since last check)
|
||
|
* output - true if evasive action needs to be taken
|
||
|
*
|
||
|
* Walking state transitions (player greeting check not shown):
|
||
|
*
|
||
|
* MoveNow <------------------------------------+
|
||
|
* | d|
|
||
|
* | |
|
||
|
* +-> State_Norm <---> State_CheckStuck --> State_Evade
|
||
|
* ^ ^ | f ^ | t ^ | |
|
||
|
* | | | | | | | |
|
||
|
* | +---+ +---+ +---+ | u
|
||
|
* | any < t < u |
|
||
|
* +--------------------------------------------+
|
||
|
*
|
||
|
* f = one reaction time
|
||
|
* d = proximity to a closed door
|
||
|
* t = how long before considered stuck
|
||
|
* u = how long to move sideways
|
||
|
*
|
||
|
* DIST_SAME_SPOT is calibrated for movement speed of around 150.
|
||
|
* A rat has walking speed of around 30, so we need to adjust for
|
||
|
* that.
|
||
|
*/
|
||
|
bool ObstacleCheck::check(const MWWorld::Ptr& actor, float duration)
|
||
|
{
|
||
|
const MWWorld::Class& cls = actor.getClass();
|
||
|
ESM::Position pos = actor.getRefData().getPosition();
|
||
|
|
||
|
if(mDistSameSpot == -1)
|
||
|
mDistSameSpot = DIST_SAME_SPOT * (cls.getSpeed(actor) / 150);
|
||
|
|
||
|
bool samePosition = (abs(pos.pos[0] - mPrevX) < mDistSameSpot) &&
|
||
|
(abs(pos.pos[1] - mPrevY) < mDistSameSpot);
|
||
|
// update position
|
||
|
mPrevX = pos.pos[0];
|
||
|
mPrevY = pos.pos[1];
|
||
|
|
||
|
switch(mWalkState)
|
||
|
{
|
||
|
case State_Norm:
|
||
|
{
|
||
|
if(!samePosition)
|
||
|
break;
|
||
|
else
|
||
|
mWalkState = State_CheckStuck;
|
||
|
}
|
||
|
/* FALL THROUGH */
|
||
|
case State_CheckStuck:
|
||
|
{
|
||
|
if(!samePosition)
|
||
|
{
|
||
|
mWalkState = State_Norm;
|
||
|
mStuckDuration = 0;
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
mStuckDuration += duration;
|
||
|
// consider stuck only if position unchanges for a period
|
||
|
if(mStuckDuration < DURATION_SAME_SPOT)
|
||
|
break; // still checking, note duration added to timer
|
||
|
else
|
||
|
{
|
||
|
mStuckDuration = 0;
|
||
|
mWalkState = State_Evade;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
/* FALL THROUGH */
|
||
|
case State_Evade:
|
||
|
{
|
||
|
mEvadeDuration += duration;
|
||
|
if(mEvadeDuration < DURATION_TO_EVADE)
|
||
|
return true;
|
||
|
else
|
||
|
{
|
||
|
//#if 0
|
||
|
std::cout << "evade \""+actor.getClass().getName(actor)+"\" "
|
||
|
//<<"dist spot "+std::to_string(mDistSameSpot)
|
||
|
//+", speed "+std::to_string(cls.getSpeed(actor))
|
||
|
<< std::endl;
|
||
|
//#endif
|
||
|
// tried to evade, assume all is ok and start again
|
||
|
mWalkState = State_Norm;
|
||
|
mEvadeDuration = 0;
|
||
|
}
|
||
|
}
|
||
|
/* NO DEFAULT CASE */
|
||
|
}
|
||
|
return false; // no obstacles to evade (yet)
|
||
|
}
|
||
|
}
|