2021-07-24 21:11:47 -07:00

322 lines
13 KiB
C++

//=========== Copyright © Valve Corporation, All rights reserved. ===========//
//
// Purpose: Mesh types for meshutils library
//
//===========================================================================//
#ifndef MESH_H
#define MESH_H
#ifdef _WIN32
#pragma once
#endif
#include "tier1/utlvector.h"
#include "tier1/utllinkedlist.h"
#include "mathlib/vector.h"
#include "materialsystem/imaterial.h"
#include "rendersystem/irenderdevice.h"
#include "bitvec.h"
class CDmeModel;
class CDmeMesh;
#define VERTEX_ELEMENT_TANGENT_WITH_FLIP VERTEX_ELEMENT_TEXCOORD4D_7
// our usual distance epsilon is 1/32 of an inch
const float ONE_32ND_UNIT = 0.03125f;
// UVChart for unique parameterization
struct UVChart_t
{
Vector4D m_vPlane;
CUtlVector<int> m_TriangleList;
Vector2D m_vMinUV;
Vector2D m_vMaxUV;
int m_nVertexStart;
int m_nVertexCount;
};
// Atlas chart data for atlasing
// TODO: we should probably pull atlasing out into a more visible library
struct AtlasChart_t
{
Vector2D m_vMaxTextureSize;
Vector2D m_vAtlasMin;
Vector2D m_vAtlasMax;
bool m_bAtlased;
};
// the mesh library only supports a single stream, with multiple attributes.
// NOTE: Each attribute must be an array of floats
class CMeshVertexAttribute
{
public:
int m_nOffsetFloats; // Offset is in # of floats
VertexElement_t m_nType;
inline bool IsJointWeight() const // { return !V_stricmp( m_name.Get(), "blendweight" ) || !V_stricmp( m_name.Get(), "blendweights" ); }
{
return m_nType == VERTEX_ELEMENT_BONEWEIGHTS1 || m_nType == VERTEX_ELEMENT_BONEWEIGHTS2 || m_nType == VERTEX_ELEMENT_BONEWEIGHTS3 || m_nType == VERTEX_ELEMENT_BONEWEIGHTS4;
}
inline bool IsJointIndices() const //{ return !V_stricmp( m_name.Get(), "blendindices" ); }
{
return m_nType == VERTEX_ELEMENT_BONEINDEX;
}
inline bool IsClothEnable()const //{ return !V_stricmp( m_name.Get(), "cloth_enable" ); }
{
return false; // needs to be string-based
}
inline bool IsPositionRemap()const //{ return !V_stricmp( m_name.Get(), "dme_position_map" ); }
{
return false; // needs to be string-based
}
};
class CMesh
{
public:
CMesh();
~CMesh();
// -----------------------------------------------------------------------------------------
// Allocate vertex and index space for a mesh, this memory will automatically be freed in the destructor
void AllocateMesh( int nVertexCount, int nIndexCount, int nVertexStride, CMeshVertexAttribute *pAttributes, int nAtrributeCount );
void AllocateAndCopyMesh( int nInputVertexCount, const float *pInputVerts, int nInputIndexCount, const uint32 *pInputIndices, int nVertexStride, CMeshVertexAttribute *pAttributes, int nAtrributeCount );
void FreeAllMemory();
// -----------------------------------------------------------------------------------------
// If you don't supply the attribute definition it assumes the vertices are just a position vector
// NOTE: VertexStrideFloats is 3 for a normal vector of 3 floats
// External meshes are allocated by external code and not freed in the destructor
void InitExternalMesh( float *pVerts, int nVertexCount, uint32 *pIndices, int nIndexCount, int nVertexStrideFloats = 3, CMeshVertexAttribute *pAttributes = NULL, int nAtrributeCount = 0 );
inline int IndexCount() const
{
return m_nIndexCount;
}
inline int VertexCount() const
{
return m_nVertexCount;
}
inline void SetVertexPosition( int nIndex, const Vector & v )
{
*( Vector * )GetVertex( nIndex ) = v ;
}
struct SkinningDataFields_t
{
SkinningDataFields_t ():m_nBoneWeights( -1 ), m_nBoneIndices( -1 ){}
bool HasSkinningData() const { return m_nBoneIndices >= 0; }
int m_nBoneWeights, m_nBoneIndices;
};
struct ClothDataFields_t
{
ClothDataFields_t() : m_nClothEnable( -1 ), m_nPositionRemap( -1 ) {}
bool HasClothData() const { return m_nClothEnable >= 0 && m_nPositionRemap >= 0; }
int m_nClothEnable, m_nPositionRemap;
};
// accessing a single attribute of a single vertex (attribute may be scalar or an array of T; scalar and array of size 1 are one and the same)
template <typename T>
class CSingleVertexFieldAccessor
{
public:
CSingleVertexFieldAccessor( T*pBase, int nCount ) : m_pBase( pBase ), m_nCount( nCount ) {}
protected:
T *m_pBase;
int m_nCount;
public:
T& operator [] ( int i ) { AssertDbg( i < m_nCount ); return m_pBase[ i ]; }
const T& operator [] ( int i ) const { AssertDbg( i < m_nCount ); return m_pBase[ i ]; }
T& operator *( ){ AssertDbg( m_nCount > 0 ); return *m_pBase; }
const T& operator *( ) const { AssertDbg( m_nCount > 0 ); return *m_pBase; }
T& Tail() { AssertDbg( m_nCount > 0 ); return m_pBase[ m_nCount - 1 ]; }
const T& Tail() const { AssertDbg( m_nCount > 0 ); return m_pBase[ m_nCount - 1 ]; }
int Count()const { return m_nCount; }
};
template <typename T>
class CScalarAttrArray;
// an accessor for a common attribute of all vertices (e.g. blendweights, or position, or normal)
template < typename T >
class CAttrArray
{
protected:
friend class CScalarAttrArray < T > ;
float *m_pBase; // this points to the first vertex field (first vertex + field offset)
int m_nStride; // this is the stride between fields
int m_nElementCount;
int m_nAttrCount;
public:
CAttrArray( float *pBase = NULL, int nStride = 0, int nVertexCount = 0, int nAttrCount = 0 )
: m_pBase( pBase )
, m_nStride( nStride )
, m_nElementCount( nVertexCount )
, m_nAttrCount( nAttrCount )
{ }
struct Range_t
{
T m_Min;
T m_Max;
Range_t();
void Append( T val )
{
if ( val < m_Min )
m_Min = val;
if ( val > m_Max )
m_Max = val;
}
};
Range_t GetRange()const
{
Range_t range;
for ( int v = 0; v < m_nElementCount; ++v )
{
const T *pValues = ( const T* )( m_pBase + m_nStride * v );
for ( int na = 0; na < m_nAttrCount; ++na )
{
range.Append( pValues[ na ] );
}
}
return range;
}
public:
// converting to bool to use this in if() to check for NULL
operator bool() const { return m_pBase != NULL; }
CSingleVertexFieldAccessor< T > operator []( int nVertex ) { AssertDbg( nVertex < m_nElementCount ); return CSingleVertexFieldAccessor< T >( ( T* )( m_pBase + m_nStride * nVertex ), m_nAttrCount ); }
CSingleVertexFieldAccessor< const T > operator []( int nVertex ) const { AssertDbg( nVertex < m_nElementCount ); return CSingleVertexFieldAccessor< const T >( ( const T* )( m_pBase + m_nStride * nVertex ), m_nAttrCount ); }
int GetElementCount()const { return m_nElementCount; }
int GetAttrCount()const { return m_nAttrCount; }
};
bool HasSkinningData() const;
SkinningDataFields_t GetSkinningDataFields() const;
ClothDataFields_t GetClothDataFields() const;
float GetVertexJointSumWeight( const SkinningDataFields_t &skinData, int nVertex, const CVarBitVec &jointSet ); // get the weight of this vertex at this joint; 1.0 for vertices rigdly bound to the given bone, 0.0 for vertices not bound to this bone
void AppendMesh( const CMesh &inputMesh );
inline int TriangleCount() const { return m_nIndexCount / 3; }
inline float *GetVertex(int nIndex) { return m_pVerts + (nIndex * m_nVertexStrideFloats); }
inline const float *GetVertex(int nIndex) const { return m_pVerts + (nIndex * m_nVertexStrideFloats); }
inline const Vector &GetVertexPosition(int nIndex) const { return *(Vector *)(m_pVerts + (nIndex * m_nVertexStrideFloats)); }
inline size_t GetVertexSizeInBytes() const { return m_nVertexStrideFloats * sizeof(float); }
inline size_t GetTotalVertexSizeInBytes() const { return GetVertexSizeInBytes() * m_nVertexCount; }
inline size_t GetTotalIndexSizeInBytes() const { return sizeof(uint32) * m_nIndexCount; }
Vector4D PlaneFromTriangle( int nTriangle ) const;
bool CalculateBounds( Vector *pMinOut, Vector *pMaxOut, int nStartVertex = 0, int nVertexCount = 0 ) const;
bool CalculateAdjacency( int *pAdjacencyOut, int nSizeAdjacencyOut ) const;
bool CalculateIndicentFacesForVertices( CUtlLinkedList<int> *pFacesPerVertex, int nFacesPerVertexSize ) const;
bool CalculateInputLayoutFromAttributes( RenderInputLayoutField_t *pOutFields, int *pInOutNumFields ) const;
int FindFirstAttributeOffset( VertexElement_t nType ) const;
void AddAttributes( CMeshVertexAttribute *pAttributes, int nAttributeCount );
void CalculateTangents();
bool CalculateTangentSpaceWorldLengthsPerFace( Vector2D *pLengthsOut, int nLengthsOut, float flMaxWorldPerUV = 256.0f );
bool CalculateFaceCenters( Vector *pCentersOut, int nCentersOut );
int GetAttrSizeFloats( int nAttribute );
float *m_pVerts;
CMeshVertexAttribute *m_pAttributes;
uint32 *m_pIndices;
int m_nVertexCount;
int m_nVertexStrideFloats; // Stride is in # of floats
int m_nAttributeCount;
int m_nIndexCount;
CUtlString m_materialName;
private:
void RestrideVertexBuffer( int nNewStrideFloats );
bool m_bAllocatedMeshData;
};
struct GridVolume_t
{
Vector m_vMinBounds;
Vector m_vMaxBounds;
};
struct IndexRange_t
{
int m_nStartIndex;
int m_nIndexCount;
};
bool ClipMeshToHalfSpace( CMesh *pMeshBack, CMesh *pMeshFront, const CMesh &inputMesh, Vector4D &vClipPlane );
void DuplicateMesh( CMesh *pMeshOut, const CMesh &inputMesh );
bool RationalizeUVsInPlace( CMesh *pMesh );
bool RationalizeUVs( CMesh *pRationalMeshOut, const CMesh &inputMesh );
void DeIndexMesh( CMesh *pMeshOut, const CMesh &inputMesh );
bool ConcatMeshes( CMesh *pMeshOut, CMesh **ppMeshIn, int nInputMeshes,
CMeshVertexAttribute *pAttributeOverride = NULL, int nAttributeOverrideCount = 0, int nStrideOverride = 0 );
bool TessellateOnWrappedUV( CMesh *pMeshOut, const CMesh &inputMesh );
// returns the mesh containing the convex hull of the input mesh
void ConvexHull3D( CMesh *pOutMesh, const CMesh &inputMesh, float flCoplanarEpsilon = ONE_32ND_UNIT );
void FitOBBToMesh( matrix3x4_t *pCenter, Vector *pExtents, const CMesh &inputMesh, float flCoplanarEpsilon = ONE_32ND_UNIT );
// Builds a mesh of the hull defined by an array of planes (pointing out of the solid - plane eq is Ax + By + Cz = D). Optionally, outputs the index of the originating input plane for each triangle in the mesh.
void HullFromPlanes( CMesh *pOutMesh, CUtlVector<uint32> *pTrianglePlaneIndex, const float *pPlanes, int nPlaneCount, int nPlaneStrideFloats, float flCoplanarEpsilon = ONE_32ND_UNIT );
// Same as above but work on a temporary list of planes described as fltx4 (the buffer will be modified in place). No stride needed in this case, ABCD = XYZW
void HullFromPlanes_SIMD( CMesh *pOutMesh, CUtlVector<uint16> *pTrianglePlaneIndex, fltx4 *pPlanes, int nPlaneCount, float flCoplanarEpsilon = ONE_32ND_UNIT );
// TODO: Extreme welds can cause faces to flip/invert. Should we add an option to detect and avoid this?
bool WeldVertices( CMesh *pMeshOut, const CMesh &inputMesh, float *pEpsilons, int nEpsilons );
// This removes any degenerate triangles and puts the vertices in access order
void CleanMesh( CMesh *pMeshOut, const CMesh &inputMesh );
// partitions the mesh into an array of meshes, each with <= nMaxVertex vertices
void SplitMesh( CUtlVector<CMesh> &list, const CMesh &input, int nMaxVertex );
// Create a unique parameterization of the mesh
bool CreateUniqueUVParameterization( CMesh *pMeshOut, const CMesh &inputMesh, float flFaceAngleThreshold, int nAtlasTextureSizeX, int nAtlasTextureSizeY, float flGutterSize );
// Pack a set of charts into an atlas
int PackChartsIntoAtlas( AtlasChart_t *pCharts, int nCharts, int nAtlasTextureSizeX, int nAtlasTextureSizeY, int nAtlasGrow );
// Creates a list of AABBs for a give volume given a target AABB size
void CreateGridCellsForVolume( CUtlVector< GridVolume_t > &outputVolumes, const Vector &vMinBounds, const Vector &vMaxBounds, const Vector &vGridSize );
//--------------------------------------------------------------------------------------
// For a list of AABBs, find all indices in the mesh that belong to each AABB. Then
// coalesce those indices into a single buffer in order of the input AABBs and return a
// list of ranges of the indices in the output mesh that are needed in each input volume
//--------------------------------------------------------------------------------------
void CreatedGriddedIndexRangesFromMesh( CMesh *pOutputMesh, CUtlVector< IndexRange_t > &outputRanges, const CMesh &inputMesh, CUtlVector< GridVolume_t > &inputVolumes );
// Per-vertex ops
void CopyVertex( float *pOut, const float *pIn, int nFloats );
void SubtractVertex( float *pOutput, const float *pLeft, const float *pRight, int nFloats );
void AddVertex( float *pOutput, const float *pLeft, const float *pRight, int nFloats );
void MultiplyVertex( float *pOutput, const float *pLeft, const float* pRight, float nFloats );
void AddVertexInPlace( float *pLeft, const float *pRight, int nFloats );
void MultiplyVertexInPlace( float *pLeft, float flRight, int nFloats );
void LerpVertex( float *pOutput, const float *pLeft, const float *pRight, float flLerp, int nFloats );
void BaryCentricVertices( float *pOutput, float *p0, float *p1, float *p2, float flU, float flV, float flW, int nFloats );
// simple hashing function for edges
inline uint32 VertHashKey( int nV0, int nV1 )
{
uint32 nHash = ((uint32(nV0) >> 16) | (uint32(nV0)<<16)) ^ uint32(nV1);
nHash = ieqsel( nHash, uint32(~0), uint32( 0 ), nHash ); // don't ever return ~0
return nHash;
}
#endif // MESH_H