1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-09 21:42:13 +00:00
OpenMW/components/detournavigator/chunkytrimesh.hpp

100 lines
2.8 KiB
C++
Raw Normal View History

2018-03-13 22:49:08 +00:00
#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_CHUNKYTRIMESH_H
#define OPENMW_COMPONENTS_DETOURNAVIGATOR_CHUNKYTRIMESH_H
2018-07-18 19:09:50 +00:00
#include "areatype.hpp"
2018-03-13 22:49:08 +00:00
#include <osg/Vec2f>
#include <array>
#include <vector>
namespace DetourNavigator
{
struct Rect
{
osg::Vec2f mMinBound;
osg::Vec2f mMaxBound;
};
struct ChunkyTriMeshNode
{
Rect mBounds;
std::ptrdiff_t mOffset;
std::size_t mSize;
};
struct Chunk
{
const int* const mIndices;
2018-07-18 19:09:50 +00:00
const AreaType* const mAreaTypes;
2018-03-13 22:49:08 +00:00
const std::size_t mSize;
};
inline bool checkOverlapRect(const Rect& lhs, const Rect& rhs)
{
bool overlap = true;
overlap = (lhs.mMinBound.x() > rhs.mMaxBound.x() || lhs.mMaxBound.x() < rhs.mMinBound.x()) ? false : overlap;
overlap = (lhs.mMinBound.y() > rhs.mMaxBound.y() || lhs.mMaxBound.y() < rhs.mMinBound.y()) ? false : overlap;
return overlap;
}
class ChunkyTriMesh
{
public:
/// Creates partitioned triangle mesh (AABB tree),
/// where each node contains at max trisPerChunk triangles.
2018-07-12 08:44:11 +00:00
ChunkyTriMesh(const std::vector<float>& verts, const std::vector<int>& tris,
2018-07-18 19:09:50 +00:00
const std::vector<AreaType>& flags, const std::size_t trisPerChunk);
2018-03-13 22:49:08 +00:00
ChunkyTriMesh(const ChunkyTriMesh&) = delete;
ChunkyTriMesh& operator=(const ChunkyTriMesh&) = delete;
/// Returns the chunk indices which overlap the input rectable.
2018-11-04 15:34:31 +00:00
template <class Function>
void forEachChunksOverlappingRect(const Rect& rect, Function&& function) const
2018-03-13 22:49:08 +00:00
{
// Traverse tree
for (std::size_t i = 0; i < mNodes.size(); )
{
const ChunkyTriMeshNode* node = &mNodes[i];
const bool overlap = checkOverlapRect(rect, node->mBounds);
const bool isLeafNode = node->mOffset >= 0;
if (isLeafNode && overlap)
2018-11-04 15:34:31 +00:00
function(i);
2018-03-13 22:49:08 +00:00
if (overlap || isLeafNode)
i++;
else
{
const auto escapeIndex = -node->mOffset;
i += static_cast<std::size_t>(escapeIndex);
}
}
}
Chunk getChunk(const std::size_t chunkId) const
{
const auto& node = mNodes[chunkId];
return Chunk {
mIndices.data() + node.mOffset * 3,
2018-07-18 19:09:50 +00:00
mAreaTypes.data() + node.mOffset,
2018-03-13 22:49:08 +00:00
node.mSize
};
}
std::size_t getMaxTrisPerChunk() const
{
return mMaxTrisPerChunk;
}
private:
std::vector<ChunkyTriMeshNode> mNodes;
std::vector<int> mIndices;
2018-07-18 19:09:50 +00:00
std::vector<AreaType> mAreaTypes;
2018-03-13 22:49:08 +00:00
std::size_t mMaxTrisPerChunk;
};
}
#endif