1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 21:53:51 +00:00
openmw-tes3mp/apps/openmw/mwmechanics/pathfinding.cpp

320 lines
12 KiB
C++
Raw Normal View History

2013-03-31 17:30:03 +00:00
#include "pathfinding.hpp"
2014-02-23 19:11:05 +00:00
#include "../mwbase/world.hpp"
#include "../mwbase/environment.hpp"
2014-02-23 19:11:05 +00:00
#include "../mwworld/esmstore.hpp"
#include "../mwworld/cellstore.hpp"
2013-03-31 17:30:03 +00:00
2013-04-11 17:02:12 +00:00
namespace
2013-03-31 17:30:03 +00:00
{
// Slightly cheaper version for comparisons.
// Caller needs to be careful for very short distances (i.e. less than 1)
// or when accumuating the results i.e. (a + b)^2 != a^2 + b^2
//
2015-06-03 17:41:19 +00:00
float distanceSquared(ESM::Pathgrid::Point point, const osg::Vec3f& pos)
{
2015-06-03 17:41:19 +00:00
return (MWMechanics::PathFinder::MakeOsgVec3(point) - pos).length2();
}
// Return the closest pathgrid point index from the specified position co
// -ordinates. NOTE: Does not check if there is a sensible way to get there
// (e.g. a cliff in front).
//
// NOTE: pos is expected to be in local co-ordinates, as is grid->mPoints
//
2015-06-03 17:41:19 +00:00
int getClosestPoint(const ESM::Pathgrid* grid, const osg::Vec3f& pos)
2013-03-31 17:30:03 +00:00
{
if(!grid || grid->mPoints.empty())
return -1;
2013-03-31 17:30:03 +00:00
float distanceBetween = distanceSquared(grid->mPoints[0], pos);
int closestIndex = 0;
2013-03-31 17:30:03 +00:00
// TODO: if this full scan causes performance problems mapping pathgrid
// points to a quadtree may help
for(unsigned int counter = 1; counter < grid->mPoints.size(); counter++)
2013-03-31 17:30:03 +00:00
{
float potentialDistBetween = distanceSquared(grid->mPoints[counter], pos);
if(potentialDistBetween < distanceBetween)
2013-03-31 17:30:03 +00:00
{
distanceBetween = potentialDistBetween;
closestIndex = counter;
2013-03-31 17:30:03 +00:00
}
}
return closestIndex;
}
2013-03-31 17:30:03 +00:00
// Chooses a reachable end pathgrid point. start is assumed reachable.
std::pair<int, bool> getClosestReachablePoint(const ESM::Pathgrid* grid,
const MWWorld::CellStore *cell,
2015-06-03 17:41:19 +00:00
const osg::Vec3f pos, int start)
{
if(!grid || grid->mPoints.empty())
return std::pair<int, bool> (-1, false);
float distanceBetween = distanceSquared(grid->mPoints[0], pos);
int closestIndex = 0;
int closestReachableIndex = 0;
// TODO: if this full scan causes performance problems mapping pathgrid
// points to a quadtree may help
for(unsigned int counter = 1; counter < grid->mPoints.size(); counter++)
{
float potentialDistBetween = distanceSquared(grid->mPoints[counter], pos);
if(potentialDistBetween < distanceBetween)
{
// found a closer one
distanceBetween = potentialDistBetween;
closestIndex = counter;
if (cell->isPointConnected(start, counter))
{
closestReachableIndex = counter;
}
2013-03-31 17:30:03 +00:00
}
}
// AiWander has logic that depends on whether a path was created, deleting
// allowed nodes if not. Hence a path needs to be created even if the start
// and the end points are the same.
//if(start == closestReachableIndex)
//closestReachableIndex = -1; // couldn't find anyting other than start
return std::pair<int, bool>
(closestReachableIndex, closestReachableIndex == closestIndex);
}
}
namespace MWMechanics
{
2015-08-04 06:14:36 +00:00
float sqrDistanceIgnoreZ(const ESM::Pathgrid::Point& point, float x, float y)
{
x -= point.mX;
y -= point.mY;
return (x * x + y * y);
}
2015-08-04 06:14:36 +00:00
float distance(const ESM::Pathgrid::Point& point, float x, float y, float z)
{
x -= point.mX;
y -= point.mY;
z -= point.mZ;
return sqrt(x * x + y * y + z * z);
}
2015-08-04 06:14:36 +00:00
float distance(const ESM::Pathgrid::Point& a, const ESM::Pathgrid::Point& b)
{
float x = static_cast<float>(a.mX - b.mX);
float y = static_cast<float>(a.mY - b.mY);
float z = static_cast<float>(a.mZ - b.mZ);
return sqrt(x * x + y * y + z * z);
}
2013-03-31 17:30:03 +00:00
PathFinder::PathFinder()
: mPathgrid(NULL),
2014-02-04 06:51:18 +00:00
mCell(NULL)
2013-03-31 17:30:03 +00:00
{
}
void PathFinder::clearPath()
{
if(!mPath.empty())
mPath.clear();
}
/*
* NOTE: This method may fail to find a path. The caller must check the
* result before using it. If there is no path the AI routies need to
* implement some other heuristics to reach the target.
*
* NOTE: It may be desirable to simply go directly to the endPoint if for
* example there are no pathgrids in this cell.
*
* NOTE: startPoint & endPoint are in world co-ordinates
*
* Updates mPath using aStarSearch() or ray test (if shortcut allowed).
* mPath consists of pathgrid points, except the last element which is
* endPoint. This may be useful where the endPoint is not on a pathgrid
* point (e.g. combat). However, if the caller has already chosen a
* pathgrid point (e.g. wander) then it may be worth while to call
* pop_back() to remove the redundant entry.
*
* mPathConstructed is set true if successful, false if not
*
* NOTE: co-ordinates must be converted prior to calling getClosestPoint()
*
* |
* | cell
* | +-----------+
* | | |
* | | |
* | | @ |
* | i | j |
* |<--->|<---->| |
* | +-----------+
* | k
* |<---------->| world
* +-----------------------------
*
* i = x value of cell itself (multiply by ESM::Land::REAL_SIZE to convert)
* j = @.x in local co-ordinates (i.e. within the cell)
* k = @.x in world co-ordinates
*/
void PathFinder::buildPath(const ESM::Pathgrid::Point &startPoint,
const ESM::Pathgrid::Point &endPoint,
const MWWorld::CellStore* cell,
bool allowShortcuts)
2014-01-26 20:53:55 +00:00
{
2014-01-11 10:57:07 +00:00
mPath.clear();
2014-01-26 20:53:55 +00:00
if(allowShortcuts)
2014-01-26 20:53:55 +00:00
{
// if there's a ray cast hit, can't take a direct path
if (!MWBase::Environment::get().getWorld()->castRay(
static_cast<float>(startPoint.mX), static_cast<float>(startPoint.mY), static_cast<float>(startPoint.mZ),
static_cast<float>(endPoint.mX), static_cast<float>(endPoint.mY), static_cast<float>(endPoint.mZ)))
2014-01-26 20:53:55 +00:00
{
mPath.push_back(endPoint);
return;
2014-01-26 20:53:55 +00:00
}
}
if(mCell != cell || !mPathgrid)
2014-01-26 20:53:55 +00:00
{
mCell = cell;
mPathgrid = MWBase::Environment::get().getWorld()->getStore().get<ESM::Pathgrid>().search(*mCell->getCell());
2014-01-26 20:53:55 +00:00
}
2014-01-26 21:06:54 +00:00
// Refer to AiWander reseach topic on openmw forums for some background.
// Maybe there is no pathgrid for this cell. Just go to destination and let
// physics take care of any blockages.
if(!mPathgrid || mPathgrid->mPoints.empty())
2014-01-26 21:06:54 +00:00
{
mPath.push_back(endPoint);
return;
2014-01-26 21:06:54 +00:00
}
// NOTE: getClosestPoint expects local co-ordinates
float xCell = 0;
float yCell = 0;
if (mCell->isExterior())
{
xCell = static_cast<float>(mCell->getCell()->mData.mX * ESM::Land::REAL_SIZE);
yCell = static_cast<float>(mCell->getCell()->mData.mY * ESM::Land::REAL_SIZE);
}
// NOTE: It is possible that getClosestPoint returns a pathgrind point index
// that is unreachable in some situations. e.g. actor is standing
// outside an area enclosed by walls, but there is a pathgrid
// point right behind the wall that is closer than any pathgrid
// point outside the wall
int startNode = getClosestPoint(mPathgrid,
2015-06-03 17:41:19 +00:00
osg::Vec3f(startPoint.mX - xCell, startPoint.mY - yCell, static_cast<float>(startPoint.mZ)));
// Some cells don't have any pathgrids at all
if(startNode != -1)
2013-03-31 17:30:03 +00:00
{
std::pair<int, bool> endNode = getClosestReachablePoint(mPathgrid, cell,
2015-06-03 17:41:19 +00:00
osg::Vec3f(endPoint.mX - xCell, endPoint.mY - yCell, static_cast<float>(endPoint.mZ)),
startNode);
// this shouldn't really happen, but just in case
if(endNode.first != -1)
{
// AiWander has logic that depends on whether a path was created,
// deleting allowed nodes if not. Hence a path needs to be created
// even if the start and the end points are the same.
// NOTE: aStarSearch will return an empty path if the start and end
2014-04-18 05:19:22 +00:00
// nodes are the same
if(startNode == endNode.first)
{
mPath.push_back(endPoint);
return;
}
mPath = mCell->aStarSearch(startNode, endNode.first);
if(!mPath.empty())
{
// Add the destination (which may be different to the closest
// pathgrid point). However only add if endNode was the closest
// point to endPoint.
//
// This logic can fail in the opposite situate, e.g. endPoint may
// have been reachable but happened to be very close to an
// unreachable pathgrid point.
//
// The AI routines will have to deal with such situations.
if(endNode.second)
mPath.push_back(endPoint);
}
}
}
return;
2013-03-31 17:30:03 +00:00
}
2013-08-30 02:17:27 +00:00
float PathFinder::getZAngleToNext(float x, float y) const
2013-03-31 17:30:03 +00:00
{
// This should never happen (programmers should have an if statement checking
// isPathConstructed that prevents this call if otherwise).
2013-03-31 17:30:03 +00:00
if(mPath.empty())
2013-10-07 08:20:02 +00:00
return 0.;
2013-08-30 02:17:27 +00:00
const ESM::Pathgrid::Point &nextPoint = *mPath.begin();
float directionX = nextPoint.mX - x;
float directionY = nextPoint.mY - y;
return std::atan2(directionX, directionY);
2013-03-31 17:30:03 +00:00
}
bool PathFinder::checkPathCompleted(float x, float y, float tolerance)
2013-03-31 17:30:03 +00:00
{
if(mPath.empty())
return true;
2015-08-04 06:14:36 +00:00
const ESM::Pathgrid::Point& nextPoint = *mPath.begin();
if (sqrDistanceIgnoreZ(nextPoint, x, y) < tolerance*tolerance)
2013-03-31 17:30:03 +00:00
{
mPath.pop_front();
if(mPath.empty())
{
2013-03-31 17:30:03 +00:00
return true;
}
2013-03-31 17:30:03 +00:00
}
2013-03-31 17:30:03 +00:00
return false;
}
2014-01-19 20:09:51 +00:00
// see header for the rationale
void PathFinder::buildSyncedPath(const ESM::Pathgrid::Point &startPoint,
const ESM::Pathgrid::Point &endPoint,
const MWWorld::CellStore* cell,
bool allowShortcuts)
2014-01-19 20:09:51 +00:00
{
if (mPath.size() < 2)
2014-01-19 20:09:51 +00:00
{
// if path has one point, then it's the destination.
// don't need to worry about bad path for this case
buildPath(startPoint, endPoint, cell, allowShortcuts);
}
else
{
const ESM::Pathgrid::Point oldStart(*getPath().begin());
buildPath(startPoint, endPoint, cell, allowShortcuts);
if (mPath.size() >= 2)
{
// if 2nd waypoint of new path == 1st waypoint of old,
// delete 1st waypoint of new path.
std::list<ESM::Pathgrid::Point>::iterator iter = ++mPath.begin();
if (iter->mX == oldStart.mX
&& iter->mY == oldStart.mY
&& iter->mZ == oldStart.mZ)
{
mPath.pop_front();
}
}
2014-01-19 20:09:51 +00:00
}
}
2014-01-27 20:38:01 +00:00
}