mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-22 23:54:01 +00:00
137 lines
3.4 KiB
C++
137 lines
3.4 KiB
C++
//
|
|
// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org
|
|
//
|
|
// This software is provided 'as-is', without any express or implied
|
|
// warranty. In no event will the authors be held liable for any damages
|
|
// arising from the use of this software.
|
|
// Permission is granted to anyone to use this software for any purpose,
|
|
// including commercial applications, and to alter it and redistribute it
|
|
// freely, subject to the following restrictions:
|
|
// 1. The origin of this software must not be misrepresented; you must not
|
|
// claim that you wrote the original software. If you use this software
|
|
// in a product, an acknowledgment in the product documentation would be
|
|
// appreciated but is not required.
|
|
// 2. Altered source versions must be plainly marked as such, and must not be
|
|
// misrepresented as being the original software.
|
|
// 3. This notice may not be removed or altered from any source distribution.
|
|
//
|
|
|
|
#include <float.h>
|
|
#include <string.h>
|
|
#include "DetourLocalBoundary.h"
|
|
#include "DetourNavMeshQuery.h"
|
|
#include "DetourCommon.h"
|
|
#include "DetourAssert.h"
|
|
|
|
|
|
dtLocalBoundary::dtLocalBoundary() :
|
|
m_nsegs(0),
|
|
m_npolys(0)
|
|
{
|
|
dtVset(m_center, FLT_MAX,FLT_MAX,FLT_MAX);
|
|
}
|
|
|
|
dtLocalBoundary::~dtLocalBoundary()
|
|
{
|
|
}
|
|
|
|
void dtLocalBoundary::reset()
|
|
{
|
|
dtVset(m_center, FLT_MAX,FLT_MAX,FLT_MAX);
|
|
m_npolys = 0;
|
|
m_nsegs = 0;
|
|
}
|
|
|
|
void dtLocalBoundary::addSegment(const float dist, const float* s)
|
|
{
|
|
// Insert neighbour based on the distance.
|
|
Segment* seg = 0;
|
|
if (!m_nsegs)
|
|
{
|
|
// First, trivial accept.
|
|
seg = &m_segs[0];
|
|
}
|
|
else if (dist >= m_segs[m_nsegs-1].d)
|
|
{
|
|
// Further than the last segment, skip.
|
|
if (m_nsegs >= MAX_LOCAL_SEGS)
|
|
return;
|
|
// Last, trivial accept.
|
|
seg = &m_segs[m_nsegs];
|
|
}
|
|
else
|
|
{
|
|
// Insert inbetween.
|
|
int i;
|
|
for (i = 0; i < m_nsegs; ++i)
|
|
if (dist <= m_segs[i].d)
|
|
break;
|
|
const int tgt = i+1;
|
|
const int n = dtMin(m_nsegs-i, MAX_LOCAL_SEGS-tgt);
|
|
dtAssert(tgt+n <= MAX_LOCAL_SEGS);
|
|
if (n > 0)
|
|
memmove(&m_segs[tgt], &m_segs[i], sizeof(Segment)*n);
|
|
seg = &m_segs[i];
|
|
}
|
|
|
|
seg->d = dist;
|
|
memcpy(seg->s, s, sizeof(float)*6);
|
|
|
|
if (m_nsegs < MAX_LOCAL_SEGS)
|
|
m_nsegs++;
|
|
}
|
|
|
|
void dtLocalBoundary::update(dtPolyRef ref, const float* pos, const float collisionQueryRange,
|
|
dtNavMeshQuery* navquery, const dtQueryFilter* filter)
|
|
{
|
|
static const int MAX_SEGS_PER_POLY = DT_VERTS_PER_POLYGON*3;
|
|
|
|
if (!ref)
|
|
{
|
|
dtVset(m_center, FLT_MAX,FLT_MAX,FLT_MAX);
|
|
m_nsegs = 0;
|
|
m_npolys = 0;
|
|
return;
|
|
}
|
|
|
|
dtVcopy(m_center, pos);
|
|
|
|
// First query non-overlapping polygons.
|
|
navquery->findLocalNeighbourhood(ref, pos, collisionQueryRange,
|
|
filter, m_polys, 0, &m_npolys, MAX_LOCAL_POLYS);
|
|
|
|
// Secondly, store all polygon edges.
|
|
m_nsegs = 0;
|
|
float segs[MAX_SEGS_PER_POLY*6];
|
|
int nsegs = 0;
|
|
for (int j = 0; j < m_npolys; ++j)
|
|
{
|
|
navquery->getPolyWallSegments(m_polys[j], filter, segs, 0, &nsegs, MAX_SEGS_PER_POLY);
|
|
for (int k = 0; k < nsegs; ++k)
|
|
{
|
|
const float* s = &segs[k*6];
|
|
// Skip too distant segments.
|
|
float tseg;
|
|
const float distSqr = dtDistancePtSegSqr2D(pos, s, s+3, tseg);
|
|
if (distSqr > dtSqr(collisionQueryRange))
|
|
continue;
|
|
addSegment(distSqr, s);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool dtLocalBoundary::isValid(dtNavMeshQuery* navquery, const dtQueryFilter* filter)
|
|
{
|
|
if (!m_npolys)
|
|
return false;
|
|
|
|
// Check that all polygons still pass query filter.
|
|
for (int i = 0; i < m_npolys; ++i)
|
|
{
|
|
if (!navquery->isValidPolyRef(m_polys[i], filter))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|