#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_CHUNKYTRIMESH_H #define OPENMW_COMPONENTS_DETOURNAVIGATOR_CHUNKYTRIMESH_H #include #include #include 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; 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. ChunkyTriMesh(const std::vector& verts, const std::vector& tris, std::size_t trisPerChunk); ChunkyTriMesh(const ChunkyTriMesh&) = delete; ChunkyTriMesh& operator=(const ChunkyTriMesh&) = delete; /// Returns the chunk indices which overlap the input rectable. template void getChunksOverlappingRect(const Rect& rect, OutputIterator out) const { // 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) *out++ = i; if (overlap || isLeafNode) i++; else { const auto escapeIndex = -node->mOffset; i += static_cast(escapeIndex); } } } Chunk getChunk(const std::size_t chunkId) const { const auto& node = mNodes[chunkId]; return Chunk { mIndices.data() + node.mOffset * 3, node.mSize }; } std::size_t getMaxTrisPerChunk() const { return mMaxTrisPerChunk; } private: std::vector mNodes; std::vector mIndices; std::size_t mMaxTrisPerChunk; }; } #endif