mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-16 06:59:55 +00:00
Remove unnecessary loops from path
This prevents actors going back when a new shortest path includes a point behind them where they were right before. Such situation can happen when path includes off mesh connection. Resulting cost of such path can be lower than the real one because off mesh connections are straight lines and walking surface usually is not a plane but a surface. Skip to path point where distance from current position to the line between previous and this point is less than point tolerance. Which means actor is standing very close to the edge between those points. Additionally check by navmesh raycasting to make sure there is actually a valid path.
This commit is contained in:
parent
c263bbf0f6
commit
7ec7c57879
2 changed files with 35 additions and 4 deletions
|
@ -6,6 +6,7 @@
|
||||||
#include <components/detournavigator/navigatorutils.hpp>
|
#include <components/detournavigator/navigatorutils.hpp>
|
||||||
#include <components/debug/debuglog.hpp>
|
#include <components/debug/debuglog.hpp>
|
||||||
#include <components/misc/coordinateconverter.hpp>
|
#include <components/misc/coordinateconverter.hpp>
|
||||||
|
#include <components/misc/math.hpp>
|
||||||
|
|
||||||
#include "../mwbase/world.hpp"
|
#include "../mwbase/world.hpp"
|
||||||
#include "../mwbase/environment.hpp"
|
#include "../mwbase/environment.hpp"
|
||||||
|
@ -315,12 +316,13 @@ namespace MWMechanics
|
||||||
while (mPath.size() > 1 && sqrDistanceIgnoreZ(mPath.front(), position) < pointTolerance * pointTolerance)
|
while (mPath.size() > 1 && sqrDistanceIgnoreZ(mPath.front(), position) < pointTolerance * pointTolerance)
|
||||||
mPath.pop_front();
|
mPath.pop_front();
|
||||||
|
|
||||||
if (shortenIfAlmostStraight)
|
|
||||||
{
|
|
||||||
const IsValidShortcut isValidShortcut {
|
const IsValidShortcut isValidShortcut {
|
||||||
MWBase::Environment::get().getWorld()->getNavigator(),
|
MWBase::Environment::get().getWorld()->getNavigator(),
|
||||||
halfExtents, flags
|
halfExtents, flags
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (shortenIfAlmostStraight)
|
||||||
|
{
|
||||||
while (mPath.size() > 2 && isAlmostStraight(mPath[0], mPath[1], mPath[2], pointTolerance)
|
while (mPath.size() > 2 && isAlmostStraight(mPath[0], mPath[1], mPath[2], pointTolerance)
|
||||||
&& isValidShortcut(mPath[0], mPath[2]))
|
&& isValidShortcut(mPath[0], mPath[2]))
|
||||||
mPath.erase(mPath.begin() + 1);
|
mPath.erase(mPath.begin() + 1);
|
||||||
|
@ -329,6 +331,19 @@ namespace MWMechanics
|
||||||
mPath.pop_front();
|
mPath.pop_front();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mPath.size() > 1)
|
||||||
|
{
|
||||||
|
std::size_t begin = 0;
|
||||||
|
for (std::size_t i = 1; i < mPath.size(); ++i)
|
||||||
|
{
|
||||||
|
const float sqrDistance = Misc::getVectorToLine(position, mPath[i - 1], mPath[i]).length2();
|
||||||
|
if (sqrDistance < pointTolerance * pointTolerance && isValidShortcut(position, mPath[i]))
|
||||||
|
begin = i;
|
||||||
|
}
|
||||||
|
for (std::size_t i = 0; i < begin; ++i)
|
||||||
|
mPath.pop_front();
|
||||||
|
}
|
||||||
|
|
||||||
if (mPath.size() == 1)
|
if (mPath.size() == 1)
|
||||||
{
|
{
|
||||||
float distSqr;
|
float distSqr;
|
||||||
|
|
16
components/misc/math.hpp
Normal file
16
components/misc/math.hpp
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
#ifndef OPENMW_COMPONENTS_MISC_MATH_H
|
||||||
|
#define OPENMW_COMPONENTS_MISC_MATH_H
|
||||||
|
|
||||||
|
#include <osg/Vec3f>
|
||||||
|
|
||||||
|
namespace Misc
|
||||||
|
{
|
||||||
|
inline osg::Vec3f getVectorToLine(const osg::Vec3f& position, const osg::Vec3f& a, const osg::Vec3f& b)
|
||||||
|
{
|
||||||
|
osg::Vec3f direction = b - a;
|
||||||
|
direction.normalize();
|
||||||
|
return (position - a) ^ direction;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in a new issue