2019-12-12 17:32:41 +01:00
|
|
|
// Copyright (C) 2008-2012 Nikolaus Gebhardt
|
|
|
|
// This file is part of the "Irrlicht Engine".
|
|
|
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
|
|
|
|
2021-08-27 17:03:34 +02:00
|
|
|
#ifndef IRR_C_DYNAMIC_MESHBUFFER_H_INCLUDED
|
|
|
|
#define IRR_C_DYNAMIC_MESHBUFFER_H_INCLUDED
|
2019-12-12 17:32:41 +01:00
|
|
|
|
|
|
|
#include "IDynamicMeshBuffer.h"
|
|
|
|
|
|
|
|
#include "CVertexBuffer.h"
|
|
|
|
#include "CIndexBuffer.h"
|
|
|
|
|
|
|
|
namespace irr
|
|
|
|
{
|
|
|
|
namespace scene
|
|
|
|
{
|
2022-04-12 18:04:56 +02:00
|
|
|
//! Implementation of the IMeshBuffer interface for which can work with 16 and 32 bit indices as well as different vertex types
|
2019-12-12 17:32:41 +01:00
|
|
|
class CDynamicMeshBuffer: public IDynamicMeshBuffer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
//! constructor
|
|
|
|
CDynamicMeshBuffer(video::E_VERTEX_TYPE vertexType, video::E_INDEX_TYPE indexType)
|
|
|
|
: PrimitiveType(EPT_TRIANGLES)
|
|
|
|
{
|
|
|
|
VertexBuffer=new CVertexBuffer(vertexType);
|
|
|
|
IndexBuffer=new CIndexBuffer(indexType);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! destructor
|
|
|
|
virtual ~CDynamicMeshBuffer()
|
|
|
|
{
|
|
|
|
if (VertexBuffer)
|
|
|
|
VertexBuffer->drop();
|
|
|
|
if (IndexBuffer)
|
|
|
|
IndexBuffer->drop();
|
|
|
|
}
|
|
|
|
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual IVertexBuffer& getVertexBuffer() const IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
return *VertexBuffer;
|
|
|
|
}
|
|
|
|
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual IIndexBuffer& getIndexBuffer() const IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
return *IndexBuffer;
|
|
|
|
}
|
|
|
|
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual void setVertexBuffer(IVertexBuffer *newVertexBuffer) IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
if (newVertexBuffer)
|
|
|
|
newVertexBuffer->grab();
|
|
|
|
if (VertexBuffer)
|
|
|
|
VertexBuffer->drop();
|
|
|
|
|
|
|
|
VertexBuffer=newVertexBuffer;
|
|
|
|
}
|
|
|
|
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual void setIndexBuffer(IIndexBuffer *newIndexBuffer) IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
if (newIndexBuffer)
|
|
|
|
newIndexBuffer->grab();
|
|
|
|
if (IndexBuffer)
|
|
|
|
IndexBuffer->drop();
|
|
|
|
|
|
|
|
IndexBuffer=newIndexBuffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Get Material of this buffer.
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual const video::SMaterial& getMaterial() const IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
return Material;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Get Material of this buffer.
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual video::SMaterial& getMaterial() IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
return Material;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Get bounding box
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual const core::aabbox3d<f32>& getBoundingBox() const IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
return BoundingBox;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Set bounding box
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual void setBoundingBox( const core::aabbox3df& box) IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
BoundingBox = box;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Recalculate bounding box
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual void recalculateBoundingBox() IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
if (!getVertexBuffer().size())
|
|
|
|
BoundingBox.reset(0,0,0);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
BoundingBox.reset(getVertexBuffer()[0].Pos);
|
|
|
|
for (u32 i=1; i<getVertexBuffer().size(); ++i)
|
|
|
|
BoundingBox.addInternalPoint(getVertexBuffer()[i].Pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-22 16:22:41 +02:00
|
|
|
//! Append the vertices and indices to the current buffer
|
|
|
|
/** Only works for compatible vertex types.
|
|
|
|
\param vertices Pointer to a vertex array.
|
|
|
|
\param numVertices Number of vertices in the array.
|
|
|
|
\param indices Pointer to index array.
|
|
|
|
\param numIndices Number of indices in array. */
|
|
|
|
virtual void append(const void* const vertices, u32 numVertices, const u16* const indices, u32 numIndices) IRR_OVERRIDE
|
|
|
|
{
|
2023-05-15 19:28:42 +02:00
|
|
|
if (vertices == getVertices() || indices == getIndices()) // can't do that because we're doing reallocations on those blocks
|
|
|
|
return;
|
|
|
|
|
|
|
|
const u32 vertexCount = getVertexCount();
|
|
|
|
|
|
|
|
VertexBuffer->reallocate(vertexCount+numVertices);
|
|
|
|
switch ( VertexBuffer->getType() )
|
|
|
|
{
|
|
|
|
case video::EVT_STANDARD:
|
|
|
|
for (u32 i=0; i<numVertices; ++i)
|
|
|
|
{
|
|
|
|
VertexBuffer->push_back(static_cast<const video::S3DVertex*>(vertices)[i]);
|
|
|
|
BoundingBox.addInternalPoint(static_cast<const video::S3DVertex*>(vertices)[i].Pos);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case video::EVT_2TCOORDS:
|
|
|
|
for (u32 i=0; i<numVertices; ++i)
|
|
|
|
{
|
|
|
|
VertexBuffer->push_back(static_cast<const video::S3DVertex2TCoords*>(vertices)[i]);
|
|
|
|
BoundingBox.addInternalPoint(static_cast<const video::S3DVertex2TCoords*>(vertices)[i].Pos);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case video::EVT_TANGENTS:
|
|
|
|
for (u32 i=0; i<numVertices; ++i)
|
|
|
|
{
|
|
|
|
VertexBuffer->push_back(static_cast<const video::S3DVertexTangents*>(vertices)[i]);
|
|
|
|
BoundingBox.addInternalPoint(static_cast<const video::S3DVertexTangents*>(vertices)[i].Pos);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
IndexBuffer->reallocate(getIndexCount()+numIndices);
|
|
|
|
switch ( IndexBuffer->getType() )
|
|
|
|
{
|
|
|
|
case video::EIT_16BIT:
|
|
|
|
for (u32 i=0; i<numIndices; ++i)
|
|
|
|
{
|
|
|
|
IndexBuffer->push_back(indices[i]+vertexCount);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case video::EIT_32BIT:
|
|
|
|
for (u32 i=0; i<numIndices; ++i)
|
|
|
|
{
|
|
|
|
IndexBuffer->push_back(reinterpret_cast<const irr::u32*>(indices)[i]+vertexCount);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
setDirty();
|
2022-04-22 16:22:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//! Append the meshbuffer to the current buffer
|
2023-05-15 19:28:42 +02:00
|
|
|
/** Only works for compatible vertex and index types
|
2022-04-22 16:22:41 +02:00
|
|
|
\param other Buffer to append to this one. */
|
|
|
|
virtual void append(const IMeshBuffer* const other) IRR_OVERRIDE
|
|
|
|
{
|
2023-05-15 19:28:42 +02:00
|
|
|
if ( getVertexType() != other->getVertexType() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
append(other->getVertices(), other->getVertexCount(), other->getIndices(), other->getIndexCount());
|
2022-04-22 16:22:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-12 17:32:41 +01:00
|
|
|
//! Describe what kind of primitive geometry is used by the meshbuffer
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual void setPrimitiveType(E_PRIMITIVE_TYPE type) IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
PrimitiveType = type;
|
|
|
|
}
|
|
|
|
|
|
|
|
//! Get the kind of primitive geometry which is used by the meshbuffer
|
2021-08-27 14:55:10 +02:00
|
|
|
virtual E_PRIMITIVE_TYPE getPrimitiveType() const IRR_OVERRIDE
|
2019-12-12 17:32:41 +01:00
|
|
|
{
|
|
|
|
return PrimitiveType;
|
|
|
|
}
|
|
|
|
|
2022-04-11 16:36:49 +02:00
|
|
|
//! Returns type of the class implementing the IMeshBuffer
|
|
|
|
virtual EMESH_BUFFER_TYPE getType() const IRR_OVERRIDE
|
|
|
|
{
|
|
|
|
return EMBT_DYNAMIC;
|
|
|
|
}
|
|
|
|
|
2022-04-14 18:54:06 +02:00
|
|
|
//! Create copy of the meshbuffer
|
|
|
|
virtual IMeshBuffer* createClone(int cloneFlags) const IRR_OVERRIDE
|
|
|
|
{
|
|
|
|
CDynamicMeshBuffer* clone = new CDynamicMeshBuffer(VertexBuffer->getType(), IndexBuffer->getType());
|
|
|
|
|
|
|
|
if (cloneFlags & ECF_VERTICES)
|
|
|
|
{
|
|
|
|
const u32 numVertices = VertexBuffer->size();
|
|
|
|
clone->VertexBuffer->reallocate(numVertices);
|
|
|
|
for ( u32 i=0; i<numVertices; ++i )
|
|
|
|
{
|
|
|
|
clone->VertexBuffer->push_back((*VertexBuffer)[i]);
|
|
|
|
}
|
|
|
|
clone->BoundingBox = BoundingBox;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cloneFlags & ECF_INDICES)
|
|
|
|
{
|
|
|
|
const u32 numIndices = IndexBuffer->size();
|
|
|
|
clone->IndexBuffer->reallocate(numIndices);
|
|
|
|
for ( u32 i=0; i<numIndices; ++i )
|
|
|
|
{
|
|
|
|
clone->IndexBuffer->push_back((*IndexBuffer)[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
clone->VertexBuffer->setHardwareMappingHint(VertexBuffer->getHardwareMappingHint());
|
|
|
|
clone->IndexBuffer->setHardwareMappingHint(clone->IndexBuffer->getHardwareMappingHint());
|
|
|
|
clone->Material = Material;
|
|
|
|
clone->PrimitiveType = PrimitiveType;
|
|
|
|
|
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
2019-12-12 17:32:41 +01:00
|
|
|
video::SMaterial Material;
|
|
|
|
core::aabbox3d<f32> BoundingBox;
|
|
|
|
//! Primitive type used for rendering (triangles, lines, ...)
|
|
|
|
E_PRIMITIVE_TYPE PrimitiveType;
|
|
|
|
private:
|
|
|
|
CDynamicMeshBuffer(const CDynamicMeshBuffer&); // = delete in c++11, prevent copying
|
|
|
|
|
|
|
|
IVertexBuffer *VertexBuffer;
|
|
|
|
IIndexBuffer *IndexBuffer;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
} // end namespace scene
|
|
|
|
} // end namespace irr
|
|
|
|
|
|
|
|
#endif
|