mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-25 15:35:23 +00:00
203 lines
7.7 KiB
C++
203 lines
7.7 KiB
C++
#include "recastmeshbuilder.hpp"
|
|
#include "chunkytrimesh.hpp"
|
|
#include "debug.hpp"
|
|
#include "settings.hpp"
|
|
#include "settingsutils.hpp"
|
|
#include "exceptions.hpp"
|
|
|
|
#include <components/bullethelpers/transformboundingbox.hpp>
|
|
#include <components/bullethelpers/processtrianglecallback.hpp>
|
|
#include <components/misc/convert.hpp>
|
|
|
|
#include <BulletCollision/CollisionShapes/btBoxShape.h>
|
|
#include <BulletCollision/CollisionShapes/btCompoundShape.h>
|
|
#include <BulletCollision/CollisionShapes/btConcaveShape.h>
|
|
#include <BulletCollision/CollisionShapes/btHeightfieldTerrainShape.h>
|
|
#include <LinearMath/btTransform.h>
|
|
#include <LinearMath/btAabbUtil2.h>
|
|
|
|
#include <algorithm>
|
|
|
|
namespace DetourNavigator
|
|
{
|
|
using BulletHelpers::makeProcessTriangleCallback;
|
|
|
|
RecastMeshBuilder::RecastMeshBuilder(const Settings& settings, const TileBounds& bounds)
|
|
: mSettings(settings)
|
|
, mBounds(bounds)
|
|
{
|
|
mBounds.mMin /= mSettings.get().mRecastScaleFactor;
|
|
mBounds.mMax /= mSettings.get().mRecastScaleFactor;
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btCollisionShape& shape, const btTransform& transform,
|
|
const AreaType areaType)
|
|
{
|
|
if (shape.isCompound())
|
|
return addObject(static_cast<const btCompoundShape&>(shape), transform, areaType);
|
|
else if (shape.getShapeType() == TERRAIN_SHAPE_PROXYTYPE)
|
|
return addObject(static_cast<const btHeightfieldTerrainShape&>(shape), transform, areaType);
|
|
else if (shape.isConcave())
|
|
return addObject(static_cast<const btConcaveShape&>(shape), transform, areaType);
|
|
else if (shape.getShapeType() == BOX_SHAPE_PROXYTYPE)
|
|
return addObject(static_cast<const btBoxShape&>(shape), transform, areaType);
|
|
std::ostringstream message;
|
|
message << "Unsupported shape type: " << BroadphaseNativeTypes(shape.getShapeType());
|
|
throw InvalidArgument(message.str());
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btCompoundShape& shape, const btTransform& transform,
|
|
const AreaType areaType)
|
|
{
|
|
for (int i = 0, num = shape.getNumChildShapes(); i < num; ++i)
|
|
addObject(*shape.getChildShape(i), transform * shape.getChildTransform(i), areaType);
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btConcaveShape& shape, const btTransform& transform,
|
|
const AreaType areaType)
|
|
{
|
|
return addObject(shape, transform, makeProcessTriangleCallback([&] (btVector3* triangle, int, int)
|
|
{
|
|
for (std::size_t i = 3; i > 0; --i)
|
|
addTriangleVertex(triangle[i - 1]);
|
|
mAreaTypes.push_back(areaType);
|
|
}));
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btHeightfieldTerrainShape& shape, const btTransform& transform,
|
|
const AreaType areaType)
|
|
{
|
|
return addObject(shape, transform, makeProcessTriangleCallback([&] (btVector3* triangle, int, int)
|
|
{
|
|
for (std::size_t i = 0; i < 3; ++i)
|
|
addTriangleVertex(triangle[i]);
|
|
mAreaTypes.push_back(areaType);
|
|
}));
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btBoxShape& shape, const btTransform& transform, const AreaType areaType)
|
|
{
|
|
const auto indexOffset = static_cast<int>(mVertices.size() / 3);
|
|
|
|
for (int vertex = 0, count = shape.getNumVertices(); vertex < count; ++vertex)
|
|
{
|
|
btVector3 position;
|
|
shape.getVertex(vertex, position);
|
|
addVertex(transform(position));
|
|
}
|
|
|
|
const std::array<int, 36> indices {{
|
|
0, 2, 3,
|
|
3, 1, 0,
|
|
0, 4, 6,
|
|
6, 2, 0,
|
|
0, 1, 5,
|
|
5, 4, 0,
|
|
7, 5, 1,
|
|
1, 3, 7,
|
|
7, 3, 2,
|
|
2, 6, 7,
|
|
7, 6, 4,
|
|
4, 5, 7,
|
|
}};
|
|
|
|
std::transform(indices.begin(), indices.end(), std::back_inserter(mIndices),
|
|
[&] (int index) { return index + indexOffset; });
|
|
|
|
std::generate_n(std::back_inserter(mAreaTypes), 12, [=] { return areaType; });
|
|
}
|
|
|
|
void RecastMeshBuilder::addWater(const int cellSize, const btTransform& transform)
|
|
{
|
|
mWater.push_back(RecastMesh::Water {cellSize, transform});
|
|
}
|
|
|
|
std::shared_ptr<RecastMesh> RecastMeshBuilder::create(std::size_t generation, std::size_t revision) const
|
|
{
|
|
return std::make_shared<RecastMesh>(generation, revision, mIndices, mVertices, mAreaTypes,
|
|
mWater, mSettings.get().mTrianglesPerChunk);
|
|
}
|
|
|
|
void RecastMeshBuilder::reset()
|
|
{
|
|
mIndices.clear();
|
|
mVertices.clear();
|
|
mAreaTypes.clear();
|
|
mWater.clear();
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btConcaveShape& shape, const btTransform& transform,
|
|
btTriangleCallback&& callback)
|
|
{
|
|
btVector3 aabbMin;
|
|
btVector3 aabbMax;
|
|
|
|
shape.getAabb(btTransform::getIdentity(), aabbMin, aabbMax);
|
|
|
|
const btVector3 boundsMin(mBounds.mMin.x(), mBounds.mMin.y(),
|
|
-std::numeric_limits<btScalar>::max() * std::numeric_limits<btScalar>::epsilon());
|
|
const btVector3 boundsMax(mBounds.mMax.x(), mBounds.mMax.y(),
|
|
std::numeric_limits<btScalar>::max() * std::numeric_limits<btScalar>::epsilon());
|
|
|
|
auto wrapper = makeProcessTriangleCallback([&] (btVector3* triangle, int partId, int triangleIndex)
|
|
{
|
|
std::array<btVector3, 3> transformed;
|
|
for (std::size_t i = 0; i < 3; ++i)
|
|
transformed[i] = transform(triangle[i]);
|
|
if (TestTriangleAgainstAabb2(transformed.data(), boundsMin, boundsMax))
|
|
callback.processTriangle(transformed.data(), partId, triangleIndex);
|
|
});
|
|
|
|
shape.processAllTriangles(&wrapper, aabbMin, aabbMax);
|
|
}
|
|
|
|
void RecastMeshBuilder::addObject(const btHeightfieldTerrainShape& shape, const btTransform& transform,
|
|
btTriangleCallback&& callback)
|
|
{
|
|
using BulletHelpers::transformBoundingBox;
|
|
|
|
btVector3 aabbMin;
|
|
btVector3 aabbMax;
|
|
|
|
shape.getAabb(btTransform::getIdentity(), aabbMin, aabbMax);
|
|
|
|
transformBoundingBox(transform, aabbMin, aabbMax);
|
|
|
|
aabbMin.setX(std::max(mBounds.mMin.x(), aabbMin.x()));
|
|
aabbMin.setX(std::min(mBounds.mMax.x(), aabbMin.x()));
|
|
aabbMin.setY(std::max(mBounds.mMin.y(), aabbMin.y()));
|
|
aabbMin.setY(std::min(mBounds.mMax.y(), aabbMin.y()));
|
|
|
|
aabbMax.setX(std::max(mBounds.mMin.x(), aabbMax.x()));
|
|
aabbMax.setX(std::min(mBounds.mMax.x(), aabbMax.x()));
|
|
aabbMax.setY(std::max(mBounds.mMin.y(), aabbMax.y()));
|
|
aabbMax.setY(std::min(mBounds.mMax.y(), aabbMax.y()));
|
|
|
|
transformBoundingBox(transform.inverse(), aabbMin, aabbMax);
|
|
|
|
auto wrapper = makeProcessTriangleCallback([&] (btVector3* triangle, int partId, int triangleIndex)
|
|
{
|
|
std::array<btVector3, 3> transformed;
|
|
for (std::size_t i = 0; i < 3; ++i)
|
|
transformed[i] = transform(triangle[i]);
|
|
callback.processTriangle(transformed.data(), partId, triangleIndex);
|
|
});
|
|
|
|
shape.processAllTriangles(&wrapper, aabbMin, aabbMax);
|
|
}
|
|
|
|
void RecastMeshBuilder::addTriangleVertex(const btVector3& worldPosition)
|
|
{
|
|
mIndices.push_back(static_cast<int>(mVertices.size() / 3));
|
|
addVertex(worldPosition);
|
|
}
|
|
|
|
void RecastMeshBuilder::addVertex(const btVector3& worldPosition)
|
|
{
|
|
const auto navMeshPosition = toNavMeshCoordinates(mSettings, Misc::Convert::makeOsgVec3f(worldPosition));
|
|
mVertices.push_back(navMeshPosition.x());
|
|
mVertices.push_back(navMeshPosition.y());
|
|
mVertices.push_back(navMeshPosition.z());
|
|
}
|
|
}
|