mirror of
https://github.com/DrBeef/Doom3Quest.git
synced 2024-12-11 20:50:52 +00:00
b2b8f43c9d
Builds, runs, no stereo or much else is working, menus work ok though
2338 lines
62 KiB
C++
2338 lines
62 KiB
C++
/*
|
|
===========================================================================
|
|
|
|
Doom 3 GPL Source Code
|
|
Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company.
|
|
|
|
This file is part of the Doom 3 GPL Source Code ("Doom 3 Source Code").
|
|
|
|
Doom 3 Source Code is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Doom 3 Source Code is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Doom 3 Source Code. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below.
|
|
|
|
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
|
|
|
|
===========================================================================
|
|
*/
|
|
|
|
#include "sys/platform.h"
|
|
#include "idlib/containers/VectorSet.h"
|
|
#include "framework/DemoFile.h"
|
|
#include "renderer/tr_local.h"
|
|
#include "renderer/Model_local.h"
|
|
#include "renderer/Model_ase.h"
|
|
#include "renderer/Model_lwo.h"
|
|
#include "renderer/Model_ma.h"
|
|
#include "renderer/VertexCache.h"
|
|
|
|
#include "renderer/Model.h"
|
|
|
|
idCVar idRenderModelStatic::r_mergeModelSurfaces( "r_mergeModelSurfaces", "1", CVAR_BOOL|CVAR_RENDERER, "combine model surfaces with the same material" );
|
|
idCVar idRenderModelStatic::r_slopVertex( "r_slopVertex", "0.01", CVAR_RENDERER, "merge xyz coordinates this far apart" );
|
|
idCVar idRenderModelStatic::r_slopTexCoord( "r_slopTexCoord", "0.001", CVAR_RENDERER, "merge texture coordinates this far apart" );
|
|
idCVar idRenderModelStatic::r_slopNormal( "r_slopNormal", "0.02", CVAR_RENDERER, "merge normals that dot less than this" );
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::idRenderModelStatic
|
|
================
|
|
*/
|
|
idRenderModelStatic::idRenderModelStatic() {
|
|
name = "<undefined>";
|
|
bounds.Clear();
|
|
lastModifiedFrame = 0;
|
|
lastArchivedFrame = 0;
|
|
overlaysAdded = 0;
|
|
shadowHull = NULL;
|
|
isStaticWorldModel = false;
|
|
defaulted = false;
|
|
purged = false;
|
|
fastLoad = false;
|
|
reloadable = true;
|
|
levelLoadReferenced = false;
|
|
timeStamp = 0;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::~idRenderModelStatic
|
|
================
|
|
*/
|
|
idRenderModelStatic::~idRenderModelStatic() {
|
|
PurgeModel();
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idRenderModelStatic::Print
|
|
==============
|
|
*/
|
|
void idRenderModelStatic::Print() const {
|
|
common->Printf( "%s\n", name.c_str() );
|
|
common->Printf( "Static model.\n" );
|
|
common->Printf( "bounds: (%f %f %f) to (%f %f %f)\n",
|
|
bounds[0][0], bounds[0][1], bounds[0][2],
|
|
bounds[1][0], bounds[1][1], bounds[1][2] );
|
|
|
|
common->Printf( " verts tris material\n" );
|
|
for ( int i = 0 ; i < NumSurfaces() ; i++ ) {
|
|
const modelSurface_t *surf = Surface( i );
|
|
|
|
srfTriangles_t *tri = surf->geometry;
|
|
const idMaterial *material = surf->shader;
|
|
|
|
if ( !tri ) {
|
|
common->Printf( "%2i: %s, NULL surface geometry\n", i, material->GetName() );
|
|
continue;
|
|
}
|
|
|
|
common->Printf( "%2i: %5i %5i %s", i, tri->numVerts, tri->numIndexes / 3, material->GetName() );
|
|
if ( tri->generateNormals ) {
|
|
common->Printf( " (smoothed)\n" );
|
|
} else {
|
|
common->Printf( "\n" );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idRenderModelStatic::Memory
|
|
==============
|
|
*/
|
|
int idRenderModelStatic::Memory() const {
|
|
int totalBytes = 0;
|
|
|
|
totalBytes += sizeof( *this );
|
|
totalBytes += name.DynamicMemoryUsed();
|
|
totalBytes += surfaces.MemoryUsed();
|
|
|
|
if ( shadowHull ) {
|
|
totalBytes += R_TriSurfMemory( shadowHull );
|
|
}
|
|
|
|
for ( int j = 0 ; j < NumSurfaces() ; j++ ) {
|
|
const modelSurface_t *surf = Surface( j );
|
|
if ( !surf->geometry ) {
|
|
continue;
|
|
}
|
|
totalBytes += R_TriSurfMemory( surf->geometry );
|
|
}
|
|
|
|
return totalBytes;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idRenderModelStatic::List
|
|
==============
|
|
*/
|
|
void idRenderModelStatic::List() const {
|
|
int totalTris = 0;
|
|
int totalVerts = 0;
|
|
int totalBytes = 0;
|
|
|
|
totalBytes = Memory();
|
|
|
|
char closed = 'C';
|
|
for ( int j = 0 ; j < NumSurfaces() ; j++ ) {
|
|
const modelSurface_t *surf = Surface( j );
|
|
if ( !surf->geometry ) {
|
|
continue;
|
|
}
|
|
if ( !surf->geometry->perfectHull ) {
|
|
closed = ' ';
|
|
}
|
|
totalTris += surf->geometry->numIndexes / 3;
|
|
totalVerts += surf->geometry->numVerts;
|
|
}
|
|
common->Printf( "%c%4ik %3i %4i %4i %s", closed, totalBytes/1024, NumSurfaces(), totalVerts, totalTris, Name() );
|
|
|
|
if ( IsDynamicModel() == DM_CACHED ) {
|
|
common->Printf( " (DM_CACHED)" );
|
|
}
|
|
if ( IsDynamicModel() == DM_CONTINUOUS ) {
|
|
common->Printf( " (DM_CONTINUOUS)" );
|
|
}
|
|
if ( defaulted ) {
|
|
common->Printf( " (DEFAULTED)" );
|
|
}
|
|
if ( bounds[0][0] >= bounds[1][0] ) {
|
|
common->Printf( " (EMPTY BOUNDS)" );
|
|
}
|
|
if ( bounds[1][0] - bounds[0][0] > 100000 ) {
|
|
common->Printf( " (HUGE BOUNDS)" );
|
|
}
|
|
|
|
common->Printf( "\n" );
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::IsDefaultModel
|
|
================
|
|
*/
|
|
bool idRenderModelStatic::IsDefaultModel() const {
|
|
return defaulted;
|
|
}
|
|
|
|
/*
|
|
================
|
|
AddCubeFace
|
|
================
|
|
*/
|
|
static void AddCubeFace( srfTriangles_t *tri, idVec3 v1, idVec3 v2, idVec3 v3, idVec3 v4 ) {
|
|
tri->verts[tri->numVerts+0].Clear();
|
|
tri->verts[tri->numVerts+0].xyz = v1 * 8;
|
|
tri->verts[tri->numVerts+0].st[0] = 0;
|
|
tri->verts[tri->numVerts+0].st[1] = 0;
|
|
|
|
tri->verts[tri->numVerts+1].Clear();
|
|
tri->verts[tri->numVerts+1].xyz = v2 * 8;
|
|
tri->verts[tri->numVerts+1].st[0] = 1;
|
|
tri->verts[tri->numVerts+1].st[1] = 0;
|
|
|
|
tri->verts[tri->numVerts+2].Clear();
|
|
tri->verts[tri->numVerts+2].xyz = v3 * 8;
|
|
tri->verts[tri->numVerts+2].st[0] = 1;
|
|
tri->verts[tri->numVerts+2].st[1] = 1;
|
|
|
|
tri->verts[tri->numVerts+3].Clear();
|
|
tri->verts[tri->numVerts+3].xyz = v4 * 8;
|
|
tri->verts[tri->numVerts+3].st[0] = 0;
|
|
tri->verts[tri->numVerts+3].st[1] = 1;
|
|
|
|
tri->indexes[tri->numIndexes+0] = tri->numVerts + 0;
|
|
tri->indexes[tri->numIndexes+1] = tri->numVerts + 1;
|
|
tri->indexes[tri->numIndexes+2] = tri->numVerts + 2;
|
|
tri->indexes[tri->numIndexes+3] = tri->numVerts + 0;
|
|
tri->indexes[tri->numIndexes+4] = tri->numVerts + 2;
|
|
tri->indexes[tri->numIndexes+5] = tri->numVerts + 3;
|
|
|
|
tri->numVerts += 4;
|
|
tri->numIndexes += 6;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::MakeDefaultModel
|
|
================
|
|
*/
|
|
void idRenderModelStatic::MakeDefaultModel() {
|
|
|
|
defaulted = true;
|
|
|
|
// throw out any surfaces we already have
|
|
PurgeModel();
|
|
|
|
// create one new surface
|
|
modelSurface_t surf;
|
|
|
|
srfTriangles_t *tri = R_AllocStaticTriSurf();
|
|
|
|
surf.shader = tr.defaultMaterial;
|
|
surf.geometry = tri;
|
|
|
|
R_AllocStaticTriSurfVerts( tri, 24 );
|
|
R_AllocStaticTriSurfIndexes( tri, 36 );
|
|
|
|
AddCubeFace( tri, idVec3(-1, 1, 1), idVec3(1, 1, 1), idVec3(1, -1, 1), idVec3(-1, -1, 1) );
|
|
AddCubeFace( tri, idVec3(-1, 1, -1), idVec3(-1, -1, -1), idVec3(1, -1, -1), idVec3(1, 1, -1) );
|
|
|
|
AddCubeFace( tri, idVec3(1, -1, 1), idVec3(1, 1, 1), idVec3(1, 1, -1), idVec3(1, -1, -1) );
|
|
AddCubeFace( tri, idVec3(-1, -1, 1), idVec3(-1, -1, -1), idVec3(-1, 1, -1), idVec3(-1, 1, 1) );
|
|
|
|
AddCubeFace( tri, idVec3(-1, -1, 1), idVec3(1, -1, 1), idVec3(1, -1, -1), idVec3(-1, -1, -1) );
|
|
AddCubeFace( tri, idVec3(-1, 1, 1), idVec3(-1, 1, -1), idVec3(1, 1, -1), idVec3(1, 1, 1) );
|
|
|
|
tri->generateNormals = true;
|
|
|
|
AddSurface( surf );
|
|
FinishSurfaces();
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::PartialInitFromFile
|
|
================
|
|
*/
|
|
void idRenderModelStatic::PartialInitFromFile( const char *fileName ) {
|
|
fastLoad = true;
|
|
InitFromFile( fileName );
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::InitFromFile
|
|
================
|
|
*/
|
|
void idRenderModelStatic::InitFromFile( const char *fileName ) {
|
|
bool loaded;
|
|
idStr extension;
|
|
|
|
InitEmpty( fileName );
|
|
|
|
// FIXME: load new .proc map format
|
|
|
|
name.ExtractFileExtension( extension );
|
|
|
|
if ( extension.Icmp( "ase" ) == 0 ) {
|
|
loaded = LoadASE( name );
|
|
reloadable = true;
|
|
} else if ( extension.Icmp( "lwo" ) == 0 ) {
|
|
loaded = LoadLWO( name );
|
|
reloadable = true;
|
|
} else if ( extension.Icmp( "flt" ) == 0 ) {
|
|
loaded = LoadFLT( name );
|
|
reloadable = true;
|
|
} else if ( extension.Icmp( "ma" ) == 0 ) {
|
|
loaded = LoadMA( name );
|
|
reloadable = true;
|
|
} else {
|
|
common->Warning( "idRenderModelStatic::InitFromFile: unknown type for model: \'%s\'", name.c_str() );
|
|
loaded = false;
|
|
}
|
|
|
|
if ( !loaded ) {
|
|
common->Warning( "Couldn't load model: '%s'", name.c_str() );
|
|
MakeDefaultModel();
|
|
return;
|
|
}
|
|
|
|
// it is now available for use
|
|
purged = false;
|
|
|
|
// create the bounds for culling and dynamic surface creation
|
|
FinishSurfaces();
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::LoadModel
|
|
================
|
|
*/
|
|
void idRenderModelStatic::LoadModel() {
|
|
PurgeModel();
|
|
InitFromFile( name );
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::InitEmpty
|
|
================
|
|
*/
|
|
void idRenderModelStatic::InitEmpty( const char *fileName ) {
|
|
// model names of the form _area* are static parts of the
|
|
// world, and have already been considered for optimized shadows
|
|
// other model names are inline entity models, and need to be
|
|
// shadowed normally
|
|
if ( !idStr::Cmpn( fileName, "_area", 5 ) ) {
|
|
isStaticWorldModel = true;
|
|
} else {
|
|
isStaticWorldModel = false;
|
|
}
|
|
|
|
name = fileName;
|
|
reloadable = false; // if it didn't come from a file, we can't reload it
|
|
PurgeModel();
|
|
purged = false;
|
|
bounds.Zero();
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::AddSurface
|
|
================
|
|
*/
|
|
void idRenderModelStatic::AddSurface( modelSurface_t surface ) {
|
|
surfaces.Append( surface );
|
|
if ( surface.geometry ) {
|
|
bounds += surface.geometry->bounds;
|
|
}
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::Name
|
|
================
|
|
*/
|
|
const char *idRenderModelStatic::Name() const {
|
|
return name;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::Timestamp
|
|
================
|
|
*/
|
|
ID_TIME_T idRenderModelStatic::Timestamp() const {
|
|
return timeStamp;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::NumSurfaces
|
|
================
|
|
*/
|
|
int idRenderModelStatic::NumSurfaces() const {
|
|
return surfaces.Num();
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::NumBaseSurfaces
|
|
================
|
|
*/
|
|
int idRenderModelStatic::NumBaseSurfaces() const {
|
|
return surfaces.Num() - overlaysAdded;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::Surface
|
|
================
|
|
*/
|
|
const modelSurface_t *idRenderModelStatic::Surface( int surfaceNum ) const {
|
|
return &surfaces[surfaceNum];
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::AllocSurfaceTriangles
|
|
================
|
|
*/
|
|
srfTriangles_t *idRenderModelStatic::AllocSurfaceTriangles( int numVerts, int numIndexes ) const {
|
|
srfTriangles_t *tri = R_AllocStaticTriSurf();
|
|
R_AllocStaticTriSurfVerts( tri, numVerts );
|
|
R_AllocStaticTriSurfIndexes( tri, numIndexes );
|
|
return tri;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::FreeSurfaceTriangles
|
|
================
|
|
*/
|
|
void idRenderModelStatic::FreeSurfaceTriangles( srfTriangles_t *tris ) const {
|
|
R_FreeStaticTriSurf( tris );
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::ShadowHull
|
|
================
|
|
*/
|
|
srfTriangles_t *idRenderModelStatic::ShadowHull() const {
|
|
return shadowHull;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::IsStaticWorldModel
|
|
================
|
|
*/
|
|
bool idRenderModelStatic::IsStaticWorldModel() const {
|
|
return isStaticWorldModel;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::IsDynamicModel
|
|
================
|
|
*/
|
|
dynamicModel_t idRenderModelStatic::IsDynamicModel() const {
|
|
// dynamic subclasses will override this
|
|
return DM_STATIC;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::IsReloadable
|
|
================
|
|
*/
|
|
bool idRenderModelStatic::IsReloadable() const {
|
|
return reloadable;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::Bounds
|
|
================
|
|
*/
|
|
idBounds idRenderModelStatic::Bounds( const struct renderEntity_s *mdef ) const {
|
|
return bounds;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::DepthHack
|
|
================
|
|
*/
|
|
float idRenderModelStatic::DepthHack() const {
|
|
return 0.0f;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::InstantiateDynamicModel
|
|
================
|
|
*/
|
|
idRenderModel *idRenderModelStatic::InstantiateDynamicModel( const struct renderEntity_s *ent, const struct viewDef_s *view, idRenderModel *cachedModel ) {
|
|
if ( cachedModel ) {
|
|
delete cachedModel;
|
|
cachedModel = NULL;
|
|
}
|
|
common->Error( "InstantiateDynamicModel called on static model '%s'", name.c_str() );
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::NumJoints
|
|
================
|
|
*/
|
|
int idRenderModelStatic::NumJoints( void ) const {
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::GetJoints
|
|
================
|
|
*/
|
|
const idMD5Joint *idRenderModelStatic::GetJoints( void ) const {
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::GetJointHandle
|
|
================
|
|
*/
|
|
jointHandle_t idRenderModelStatic::GetJointHandle( const char *name ) const {
|
|
return INVALID_JOINT;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::GetJointName
|
|
================
|
|
*/
|
|
const char * idRenderModelStatic::GetJointName( jointHandle_t handle ) const {
|
|
return "";
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::GetDefaultPose
|
|
================
|
|
*/
|
|
const idJointQuat *idRenderModelStatic::GetDefaultPose( void ) const {
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::NearestJoint
|
|
================
|
|
*/
|
|
int idRenderModelStatic::NearestJoint( int surfaceNum, int a, int b, int c ) const {
|
|
return INVALID_JOINT;
|
|
}
|
|
|
|
|
|
//=====================================================================
|
|
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::FinishSurfaces
|
|
|
|
The mergeShadows option allows surfaces with different textures to share
|
|
silhouette edges for shadow calculation, instead of leaving shared edges
|
|
hanging.
|
|
|
|
If any of the original shaders have the noSelfShadow flag set, the surfaces
|
|
can't be merged, because they will need to be drawn in different order.
|
|
|
|
If there is only one surface, a separate merged surface won't be generated.
|
|
|
|
A model with multiple surfaces can't later have a skinned shader change the
|
|
state of the noSelfShadow flag.
|
|
|
|
-----------------
|
|
|
|
Creates mirrored copies of two sided surfaces with normal maps, which would
|
|
otherwise light funny.
|
|
|
|
Extends the bounds of deformed surfaces so they don't cull incorrectly at screen edges.
|
|
|
|
================
|
|
*/
|
|
void idRenderModelStatic::FinishSurfaces() {
|
|
int i;
|
|
int totalVerts, totalIndexes;
|
|
|
|
purged = false;
|
|
|
|
// make sure we don't have a huge bounds even if we don't finish everything
|
|
bounds.Zero();
|
|
|
|
if ( surfaces.Num() == 0 ) {
|
|
return;
|
|
}
|
|
|
|
// renderBump doesn't care about most of this
|
|
if ( fastLoad ) {
|
|
bounds.Zero();
|
|
for ( i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
|
|
R_BoundTriSurf( surf->geometry );
|
|
bounds.AddBounds( surf->geometry->bounds );
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// cleanup all the final surfaces, but don't create sil edges
|
|
totalVerts = 0;
|
|
totalIndexes = 0;
|
|
|
|
// decide if we are going to merge all the surfaces into one shadower
|
|
int numOriginalSurfaces = surfaces.Num();
|
|
|
|
// make sure there aren't any NULL shaders or geometry
|
|
for ( i = 0 ; i < numOriginalSurfaces ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
|
|
if ( surf->geometry == NULL || surf->shader == NULL ) {
|
|
MakeDefaultModel();
|
|
common->Error( "Model %s, surface %i had NULL geometry", name.c_str(), i );
|
|
}
|
|
if ( surf->shader == NULL ) {
|
|
MakeDefaultModel();
|
|
common->Error( "Model %s, surface %i had NULL shader", name.c_str(), i );
|
|
}
|
|
}
|
|
|
|
// duplicate and reverse triangles for two sided bump mapped surfaces
|
|
// note that this won't catch surfaces that have their shaders dynamically
|
|
// changed, and won't work with animated models.
|
|
// It is better to create completely separate surfaces, rather than
|
|
// add vertexes and indexes to the existing surface, because the
|
|
// tangent generation wouldn't like the acute shared edges
|
|
for ( i = 0 ; i < numOriginalSurfaces ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
|
|
if ( surf->shader->ShouldCreateBackSides() ) {
|
|
srfTriangles_t *newTri;
|
|
|
|
newTri = R_CopyStaticTriSurf( surf->geometry );
|
|
R_ReverseTriangles( newTri );
|
|
|
|
modelSurface_t newSurf;
|
|
|
|
newSurf.shader = surf->shader;
|
|
newSurf.geometry = newTri;
|
|
|
|
AddSurface( newSurf );
|
|
}
|
|
}
|
|
|
|
// clean the surfaces
|
|
for ( i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
|
|
R_CleanupTriangles( surf->geometry, surf->geometry->generateNormals, true, surf->shader->UseUnsmoothedTangents() );
|
|
if ( surf->shader->SurfaceCastsShadow() ) {
|
|
totalVerts += surf->geometry->numVerts;
|
|
totalIndexes += surf->geometry->numIndexes;
|
|
}
|
|
}
|
|
|
|
// add up the total surface area for development information
|
|
for ( i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
srfTriangles_t *tri = surf->geometry;
|
|
|
|
for ( int j = 0 ; j < tri->numIndexes ; j += 3 ) {
|
|
float area = idWinding::TriangleArea( tri->verts[tri->indexes[j]].xyz,
|
|
tri->verts[tri->indexes[j+1]].xyz, tri->verts[tri->indexes[j+2]].xyz );
|
|
const_cast<idMaterial *>(surf->shader)->AddToSurfaceArea( area );
|
|
}
|
|
}
|
|
|
|
// calculate the bounds
|
|
if ( surfaces.Num() == 0 ) {
|
|
bounds.Zero();
|
|
} else {
|
|
bounds.Clear();
|
|
for ( i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
modelSurface_t *surf = &surfaces[i];
|
|
|
|
// if the surface has a deformation, increase the bounds
|
|
// the amount here is somewhat arbitrary, designed to handle
|
|
// autosprites and flares, but could be done better with exact
|
|
// deformation information.
|
|
// Note that this doesn't handle deformations that are skinned in
|
|
// at run time...
|
|
if ( surf->shader->Deform() != DFRM_NONE ) {
|
|
srfTriangles_t *tri = surf->geometry;
|
|
idVec3 mid = ( tri->bounds[1] + tri->bounds[0] ) * 0.5f;
|
|
float radius = ( tri->bounds[0] - mid ).Length();
|
|
radius += 20.0f;
|
|
|
|
tri->bounds[0][0] = mid[0] - radius;
|
|
tri->bounds[0][1] = mid[1] - radius;
|
|
tri->bounds[0][2] = mid[2] - radius;
|
|
|
|
tri->bounds[1][0] = mid[0] + radius;
|
|
tri->bounds[1][1] = mid[1] + radius;
|
|
tri->bounds[1][2] = mid[2] + radius;
|
|
}
|
|
|
|
// add to the model bounds
|
|
bounds.AddBounds( surf->geometry->bounds );
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::ConvertASEToModelSurfaces
|
|
=================
|
|
*/
|
|
typedef struct matchVert_s {
|
|
struct matchVert_s *next;
|
|
int v, tv;
|
|
byte color[4];
|
|
idVec3 normal;
|
|
} matchVert_t;
|
|
|
|
bool idRenderModelStatic::ConvertASEToModelSurfaces( const struct aseModel_s *ase ) {
|
|
aseObject_t * object;
|
|
aseMesh_t * mesh;
|
|
aseMaterial_t * material;
|
|
const idMaterial *im1, *im2;
|
|
srfTriangles_t *tri;
|
|
int objectNum;
|
|
int i, j, k;
|
|
int v, tv;
|
|
int * vRemap;
|
|
int * tvRemap;
|
|
matchVert_t * mvTable; // all of the match verts
|
|
matchVert_t ** mvHash; // points inside mvTable for each xyz index
|
|
matchVert_t * lastmv;
|
|
matchVert_t * mv;
|
|
idVec3 normal;
|
|
float uOffset, vOffset, textureSin, textureCos;
|
|
float uTiling, vTiling;
|
|
int * mergeTo;
|
|
byte * color;
|
|
static byte identityColor[4] = { 255, 255, 255, 255 };
|
|
modelSurface_t surf, *modelSurf;
|
|
|
|
if ( !ase ) {
|
|
return false;
|
|
}
|
|
if ( ase->objects.Num() < 1 ) {
|
|
return false;
|
|
}
|
|
|
|
timeStamp = ase->timeStamp;
|
|
|
|
// the modeling programs can save out multiple surfaces with a common
|
|
// material, but we would like to mege them together where possible
|
|
// meaning that this->NumSurfaces() <= ase->objects.currentElements
|
|
mergeTo = (int *)_alloca( ase->objects.Num() * sizeof( *mergeTo ) );
|
|
surf.geometry = NULL;
|
|
if ( ase->materials.Num() == 0 ) {
|
|
// if we don't have any materials, dump everything into a single surface
|
|
surf.shader = tr.defaultMaterial;
|
|
surf.id = 0;
|
|
this->AddSurface( surf );
|
|
for ( i = 0 ; i < ase->objects.Num() ; i++ ) {
|
|
mergeTo[i] = 0;
|
|
}
|
|
} else if ( !r_mergeModelSurfaces.GetBool() ) {
|
|
// don't merge any
|
|
for ( i = 0 ; i < ase->objects.Num() ; i++ ) {
|
|
mergeTo[i] = i;
|
|
object = ase->objects[i];
|
|
material = ase->materials[object->materialRef];
|
|
surf.shader = declManager->FindMaterial( material->name );
|
|
surf.id = this->NumSurfaces();
|
|
this->AddSurface( surf );
|
|
}
|
|
} else {
|
|
// search for material matches
|
|
for ( i = 0 ; i < ase->objects.Num() ; i++ ) {
|
|
object = ase->objects[i];
|
|
material = ase->materials[object->materialRef];
|
|
im1 = declManager->FindMaterial( material->name );
|
|
if ( im1->IsDiscrete() ) {
|
|
// flares, autosprites, etc
|
|
j = this->NumSurfaces();
|
|
} else {
|
|
for ( j = 0 ; j < this->NumSurfaces() ; j++ ) {
|
|
modelSurf = &this->surfaces[j];
|
|
im2 = modelSurf->shader;
|
|
if ( im1 == im2 ) {
|
|
// merge this
|
|
mergeTo[i] = j;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if ( j == this->NumSurfaces() ) {
|
|
// didn't merge
|
|
mergeTo[i] = j;
|
|
surf.shader = im1;
|
|
surf.id = this->NumSurfaces();
|
|
this->AddSurface( surf );
|
|
}
|
|
}
|
|
}
|
|
|
|
idVectorSubset<idVec3, 3> vertexSubset;
|
|
idVectorSubset<idVec2, 2> texCoordSubset;
|
|
|
|
// build the surfaces
|
|
for ( objectNum = 0 ; objectNum < ase->objects.Num() ; objectNum++ ) {
|
|
object = ase->objects[objectNum];
|
|
mesh = &object->mesh;
|
|
material = ase->materials[object->materialRef];
|
|
im1 = declManager->FindMaterial( material->name );
|
|
|
|
bool normalsParsed = mesh->normalsParsed;
|
|
|
|
// completely ignore any explict normals on surfaces with a renderbump command
|
|
// which will guarantee the best contours and least vertexes.
|
|
const char *rb = im1->GetRenderBump();
|
|
if ( rb && rb[0] ) {
|
|
normalsParsed = false;
|
|
}
|
|
|
|
// It seems like the tools our artists are using often generate
|
|
// verts and texcoords slightly separated that should be merged
|
|
// note that we really should combine the surfaces with common materials
|
|
// before doing this operation, because we can miss a slop combination
|
|
// if they are in different surfaces
|
|
|
|
vRemap = (int *)R_StaticAlloc( mesh->numVertexes * sizeof( vRemap[0] ) );
|
|
|
|
if ( fastLoad ) {
|
|
// renderbump doesn't care about vertex count
|
|
for ( j = 0; j < mesh->numVertexes; j++ ) {
|
|
vRemap[j] = j;
|
|
}
|
|
} else {
|
|
float vertexEpsilon = r_slopVertex.GetFloat();
|
|
float expand = 2 * 32 * vertexEpsilon;
|
|
idVec3 mins, maxs;
|
|
|
|
SIMDProcessor->MinMax( mins, maxs, mesh->vertexes, mesh->numVertexes );
|
|
mins -= idVec3( expand, expand, expand );
|
|
maxs += idVec3( expand, expand, expand );
|
|
vertexSubset.Init( mins, maxs, 32, 1024 );
|
|
for ( j = 0; j < mesh->numVertexes; j++ ) {
|
|
vRemap[j] = vertexSubset.FindVector( mesh->vertexes, j, vertexEpsilon );
|
|
}
|
|
}
|
|
|
|
tvRemap = (int *)R_StaticAlloc( mesh->numTVertexes * sizeof( tvRemap[0] ) );
|
|
|
|
if ( fastLoad ) {
|
|
// renderbump doesn't care about vertex count
|
|
for ( j = 0; j < mesh->numTVertexes; j++ ) {
|
|
tvRemap[j] = j;
|
|
}
|
|
} else {
|
|
float texCoordEpsilon = r_slopTexCoord.GetFloat();
|
|
float expand = 2 * 32 * texCoordEpsilon;
|
|
idVec2 mins, maxs;
|
|
|
|
SIMDProcessor->MinMax( mins, maxs, mesh->tvertexes, mesh->numTVertexes );
|
|
mins -= idVec2( expand, expand );
|
|
maxs += idVec2( expand, expand );
|
|
texCoordSubset.Init( mins, maxs, 32, 1024 );
|
|
for ( j = 0; j < mesh->numTVertexes; j++ ) {
|
|
tvRemap[j] = texCoordSubset.FindVector( mesh->tvertexes, j, texCoordEpsilon );
|
|
}
|
|
}
|
|
|
|
// we need to find out how many unique vertex / texcoord combinations
|
|
// there are, because ASE tracks them separately but we need them unified
|
|
|
|
// the maximum possible number of combined vertexes is the number of indexes
|
|
mvTable = (matchVert_t *)R_ClearedStaticAlloc( mesh->numFaces * 3 * sizeof( mvTable[0] ) );
|
|
|
|
// we will have a hash chain based on the xyz values
|
|
mvHash = (matchVert_t **)R_ClearedStaticAlloc( mesh->numVertexes * sizeof( mvHash[0] ) );
|
|
|
|
// allocate triangle surface
|
|
tri = R_AllocStaticTriSurf();
|
|
tri->numVerts = 0;
|
|
tri->numIndexes = 0;
|
|
R_AllocStaticTriSurfIndexes( tri, mesh->numFaces * 3 );
|
|
tri->generateNormals = !normalsParsed;
|
|
|
|
// init default normal, color and tex coord index
|
|
normal.Zero();
|
|
color = identityColor;
|
|
tv = 0;
|
|
|
|
// find all the unique combinations
|
|
float normalEpsilon = 1.0f - r_slopNormal.GetFloat();
|
|
for ( j = 0; j < mesh->numFaces; j++ ) {
|
|
for ( k = 0; k < 3; k++ ) {
|
|
v = mesh->faces[j].vertexNum[k];
|
|
|
|
if ( v < 0 || v >= mesh->numVertexes ) {
|
|
common->Error( "ConvertASEToModelSurfaces: bad vertex index in ASE file %s", name.c_str() );
|
|
}
|
|
|
|
// collapse the position if it was slightly offset
|
|
v = vRemap[v];
|
|
|
|
// we may or may not have texcoords to compare
|
|
if ( mesh->numTVFaces == mesh->numFaces && mesh->numTVertexes != 0 ) {
|
|
tv = mesh->faces[j].tVertexNum[k];
|
|
if ( tv < 0 || tv >= mesh->numTVertexes ) {
|
|
common->Error( "ConvertASEToModelSurfaces: bad tex coord index in ASE file %s", name.c_str() );
|
|
}
|
|
// collapse the tex coord if it was slightly offset
|
|
tv = tvRemap[tv];
|
|
}
|
|
|
|
// we may or may not have normals to compare
|
|
if ( normalsParsed ) {
|
|
normal = mesh->faces[j].vertexNormals[k];
|
|
}
|
|
|
|
// we may or may not have colors to compare
|
|
if ( mesh->colorsParsed ) {
|
|
color = mesh->faces[j].vertexColors[k];
|
|
}
|
|
|
|
// find a matching vert
|
|
for ( lastmv = NULL, mv = mvHash[v]; mv != NULL; lastmv = mv, mv = mv->next ) {
|
|
if ( mv->tv != tv ) {
|
|
continue;
|
|
}
|
|
if ( *(unsigned *)mv->color != *(unsigned *)color ) {
|
|
continue;
|
|
}
|
|
if ( !normalsParsed ) {
|
|
// if we are going to create the normals, just
|
|
// matching texcoords is enough
|
|
break;
|
|
}
|
|
if ( mv->normal * normal > normalEpsilon ) {
|
|
break; // we already have this one
|
|
}
|
|
}
|
|
if ( !mv ) {
|
|
// allocate a new match vert and link to hash chain
|
|
mv = &mvTable[ tri->numVerts ];
|
|
mv->v = v;
|
|
mv->tv = tv;
|
|
mv->normal = normal;
|
|
*(unsigned *)mv->color = *(unsigned *)color;
|
|
mv->next = NULL;
|
|
if ( lastmv ) {
|
|
lastmv->next = mv;
|
|
} else {
|
|
mvHash[v] = mv;
|
|
}
|
|
tri->numVerts++;
|
|
}
|
|
|
|
tri->indexes[tri->numIndexes] = mv - mvTable;
|
|
tri->numIndexes++;
|
|
}
|
|
}
|
|
|
|
// allocate space for the indexes and copy them
|
|
if ( tri->numIndexes > mesh->numFaces * 3 ) {
|
|
common->FatalError( "ConvertASEToModelSurfaces: index miscount in ASE file %s", name.c_str() );
|
|
}
|
|
if ( tri->numVerts > mesh->numFaces * 3 ) {
|
|
common->FatalError( "ConvertASEToModelSurfaces: vertex miscount in ASE file %s", name.c_str() );
|
|
}
|
|
|
|
// an ASE allows the texture coordinates to be scaled, translated, and rotated
|
|
if ( ase->materials.Num() == 0 ) {
|
|
uOffset = vOffset = 0.0f;
|
|
uTiling = vTiling = 1.0f;
|
|
textureSin = 0.0f;
|
|
textureCos = 1.0f;
|
|
} else {
|
|
material = ase->materials[object->materialRef];
|
|
uOffset = -material->uOffset;
|
|
vOffset = material->vOffset;
|
|
uTiling = material->uTiling;
|
|
vTiling = material->vTiling;
|
|
textureSin = idMath::Sin( material->angle );
|
|
textureCos = idMath::Cos( material->angle );
|
|
}
|
|
|
|
// now allocate and generate the combined vertexes
|
|
R_AllocStaticTriSurfVerts( tri, tri->numVerts );
|
|
for ( j = 0; j < tri->numVerts; j++ ) {
|
|
mv = &mvTable[j];
|
|
tri->verts[ j ].Clear();
|
|
tri->verts[ j ].xyz = mesh->vertexes[ mv->v ];
|
|
tri->verts[ j ].normal = mv->normal;
|
|
*(unsigned *)tri->verts[j].color = *(unsigned *)mv->color;
|
|
if ( mesh->numTVFaces == mesh->numFaces && mesh->numTVertexes != 0 ) {
|
|
const idVec2 &tv = mesh->tvertexes[ mv->tv ];
|
|
float u = tv.x * uTiling + uOffset;
|
|
float v = tv.y * vTiling + vOffset;
|
|
tri->verts[ j ].st[0] = u * textureCos + v * textureSin;
|
|
tri->verts[ j ].st[1] = u * -textureSin + v * textureCos;
|
|
}
|
|
}
|
|
|
|
R_StaticFree( mvTable );
|
|
R_StaticFree( mvHash );
|
|
R_StaticFree( tvRemap );
|
|
R_StaticFree( vRemap );
|
|
|
|
// see if we need to merge with a previous surface of the same material
|
|
modelSurf = &this->surfaces[mergeTo[ objectNum ]];
|
|
srfTriangles_t *mergeTri = modelSurf->geometry;
|
|
if ( !mergeTri ) {
|
|
modelSurf->geometry = tri;
|
|
} else {
|
|
modelSurf->geometry = R_MergeTriangles( mergeTri, tri );
|
|
R_FreeStaticTriSurf( tri );
|
|
R_FreeStaticTriSurf( mergeTri );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::ConvertLWOToModelSurfaces
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::ConvertLWOToModelSurfaces( const struct st_lwObject *lwo ) {
|
|
const idMaterial *im1, *im2;
|
|
srfTriangles_t *tri;
|
|
lwSurface * lwoSurf;
|
|
int numTVertexes;
|
|
int i, j, k;
|
|
int v, tv;
|
|
idVec3 * vList;
|
|
int * vRemap;
|
|
idVec2 * tvList;
|
|
int * tvRemap;
|
|
matchVert_t * mvTable; // all of the match verts
|
|
matchVert_t ** mvHash; // points inside mvTable for each xyz index
|
|
matchVert_t * lastmv;
|
|
matchVert_t * mv;
|
|
idVec3 normal;
|
|
int * mergeTo;
|
|
byte color[4];
|
|
modelSurface_t surf, *modelSurf;
|
|
|
|
if ( !lwo ) {
|
|
return false;
|
|
}
|
|
if ( lwo->surf == NULL ) {
|
|
return false;
|
|
}
|
|
|
|
timeStamp = lwo->timeStamp;
|
|
|
|
// count the number of surfaces
|
|
i = 0;
|
|
for ( lwoSurf = lwo->surf; lwoSurf; lwoSurf = lwoSurf->next ) {
|
|
i++;
|
|
}
|
|
|
|
// the modeling programs can save out multiple surfaces with a common
|
|
// material, but we would like to merge them together where possible
|
|
mergeTo = (int *)_alloca( i * sizeof( mergeTo[0] ) );
|
|
memset( &surf, 0, sizeof( surf ) );
|
|
|
|
if ( !r_mergeModelSurfaces.GetBool() ) {
|
|
// don't merge any
|
|
for ( lwoSurf = lwo->surf, i = 0; lwoSurf; lwoSurf = lwoSurf->next, i++ ) {
|
|
mergeTo[i] = i;
|
|
surf.shader = declManager->FindMaterial( lwoSurf->name );
|
|
surf.id = this->NumSurfaces();
|
|
this->AddSurface( surf );
|
|
}
|
|
} else {
|
|
// search for material matches
|
|
for ( lwoSurf = lwo->surf, i = 0; lwoSurf; lwoSurf = lwoSurf->next, i++ ) {
|
|
im1 = declManager->FindMaterial( lwoSurf->name );
|
|
if ( im1->IsDiscrete() ) {
|
|
// flares, autosprites, etc
|
|
j = this->NumSurfaces();
|
|
} else {
|
|
for ( j = 0 ; j < this->NumSurfaces() ; j++ ) {
|
|
modelSurf = &this->surfaces[j];
|
|
im2 = modelSurf->shader;
|
|
if ( im1 == im2 ) {
|
|
// merge this
|
|
mergeTo[i] = j;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if ( j == this->NumSurfaces() ) {
|
|
// didn't merge
|
|
mergeTo[i] = j;
|
|
surf.shader = im1;
|
|
surf.id = this->NumSurfaces();
|
|
this->AddSurface( surf );
|
|
}
|
|
}
|
|
}
|
|
|
|
idVectorSubset<idVec3, 3> vertexSubset;
|
|
idVectorSubset<idVec2, 2> texCoordSubset;
|
|
|
|
// we only ever use the first layer
|
|
lwLayer *layer = lwo->layer;
|
|
|
|
// vertex positions
|
|
if ( layer->point.count <= 0 ) {
|
|
common->Warning( "ConvertLWOToModelSurfaces: model \'%s\' has bad or missing vertex data", name.c_str() );
|
|
return false;
|
|
}
|
|
|
|
vList = (idVec3 *)R_StaticAlloc( layer->point.count * sizeof( vList[0] ) );
|
|
for ( j = 0; j < layer->point.count; j++ ) {
|
|
vList[j].x = layer->point.pt[j].pos[0];
|
|
vList[j].y = layer->point.pt[j].pos[2];
|
|
vList[j].z = layer->point.pt[j].pos[1];
|
|
}
|
|
|
|
// vertex texture coords
|
|
numTVertexes = 0;
|
|
|
|
if ( layer->nvmaps ) {
|
|
for( lwVMap *vm = layer->vmap; vm; vm = vm->next ) {
|
|
if ( vm->type == LWID_('T','X','U','V') ) {
|
|
numTVertexes += vm->nverts;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( numTVertexes ) {
|
|
tvList = (idVec2 *)Mem_Alloc( numTVertexes * sizeof( tvList[0] ) );
|
|
int offset = 0;
|
|
for( lwVMap *vm = layer->vmap; vm; vm = vm->next ) {
|
|
if ( vm->type == LWID_('T','X','U','V') ) {
|
|
vm->offset = offset;
|
|
for ( k = 0; k < vm->nverts; k++ ) {
|
|
tvList[k + offset].x = vm->val[k][0];
|
|
tvList[k + offset].y = 1.0f - vm->val[k][1]; // invert the t
|
|
}
|
|
offset += vm->nverts;
|
|
}
|
|
}
|
|
} else {
|
|
common->Warning( "ConvertLWOToModelSurfaces: model \'%s\' has bad or missing uv data", name.c_str() );
|
|
numTVertexes = 1;
|
|
tvList = (idVec2 *)Mem_ClearedAlloc( numTVertexes * sizeof( tvList[0] ) );
|
|
}
|
|
|
|
// It seems like the tools our artists are using often generate
|
|
// verts and texcoords slightly separated that should be merged
|
|
// note that we really should combine the surfaces with common materials
|
|
// before doing this operation, because we can miss a slop combination
|
|
// if they are in different surfaces
|
|
|
|
vRemap = (int *)R_StaticAlloc( layer->point.count * sizeof( vRemap[0] ) );
|
|
|
|
if ( fastLoad ) {
|
|
// renderbump doesn't care about vertex count
|
|
for ( j = 0; j < layer->point.count; j++ ) {
|
|
vRemap[j] = j;
|
|
}
|
|
} else {
|
|
float vertexEpsilon = r_slopVertex.GetFloat();
|
|
float expand = 2 * 32 * vertexEpsilon;
|
|
idVec3 mins, maxs;
|
|
|
|
SIMDProcessor->MinMax( mins, maxs, vList, layer->point.count );
|
|
mins -= idVec3( expand, expand, expand );
|
|
maxs += idVec3( expand, expand, expand );
|
|
vertexSubset.Init( mins, maxs, 32, 1024 );
|
|
for ( j = 0; j < layer->point.count; j++ ) {
|
|
vRemap[j] = vertexSubset.FindVector( vList, j, vertexEpsilon );
|
|
}
|
|
}
|
|
|
|
tvRemap = (int *)R_StaticAlloc( numTVertexes * sizeof( tvRemap[0] ) );
|
|
|
|
if ( fastLoad ) {
|
|
// renderbump doesn't care about vertex count
|
|
for ( j = 0; j < numTVertexes; j++ ) {
|
|
tvRemap[j] = j;
|
|
}
|
|
} else {
|
|
float texCoordEpsilon = r_slopTexCoord.GetFloat();
|
|
float expand = 2 * 32 * texCoordEpsilon;
|
|
idVec2 mins, maxs;
|
|
|
|
SIMDProcessor->MinMax( mins, maxs, tvList, numTVertexes );
|
|
mins -= idVec2( expand, expand );
|
|
maxs += idVec2( expand, expand );
|
|
texCoordSubset.Init( mins, maxs, 32, 1024 );
|
|
for ( j = 0; j < numTVertexes; j++ ) {
|
|
tvRemap[j] = texCoordSubset.FindVector( tvList, j, texCoordEpsilon );
|
|
}
|
|
}
|
|
|
|
// build the surfaces
|
|
for ( lwoSurf = lwo->surf, i = 0; lwoSurf; lwoSurf = lwoSurf->next, i++ ) {
|
|
im1 = declManager->FindMaterial( lwoSurf->name );
|
|
|
|
bool normalsParsed = true;
|
|
|
|
// completely ignore any explict normals on surfaces with a renderbump command
|
|
// which will guarantee the best contours and least vertexes.
|
|
const char *rb = im1->GetRenderBump();
|
|
if ( rb && rb[0] ) {
|
|
normalsParsed = false;
|
|
}
|
|
|
|
// we need to find out how many unique vertex / texcoord combinations there are
|
|
|
|
// the maximum possible number of combined vertexes is the number of indexes
|
|
mvTable = (matchVert_t *)R_ClearedStaticAlloc( layer->polygon.count * 3 * sizeof( mvTable[0] ) );
|
|
|
|
// we will have a hash chain based on the xyz values
|
|
mvHash = (matchVert_t **)R_ClearedStaticAlloc( layer->point.count * sizeof( mvHash[0] ) );
|
|
|
|
// allocate triangle surface
|
|
tri = R_AllocStaticTriSurf();
|
|
tri->numVerts = 0;
|
|
tri->numIndexes = 0;
|
|
R_AllocStaticTriSurfIndexes( tri, layer->polygon.count * 3 );
|
|
tri->generateNormals = !normalsParsed;
|
|
|
|
// find all the unique combinations
|
|
float normalEpsilon;
|
|
if ( fastLoad ) {
|
|
normalEpsilon = 1.0f; // don't merge unless completely exact
|
|
} else {
|
|
normalEpsilon = 1.0f - r_slopNormal.GetFloat();
|
|
}
|
|
for ( j = 0; j < layer->polygon.count; j++ ) {
|
|
lwPolygon *poly = &layer->polygon.pol[j];
|
|
|
|
if ( poly->surf != lwoSurf ) {
|
|
continue;
|
|
}
|
|
|
|
if ( poly->nverts != 3 ) {
|
|
common->Warning( "ConvertLWOToModelSurfaces: model %s has too many verts for a poly! Make sure you triplet it down", name.c_str() );
|
|
continue;
|
|
}
|
|
|
|
for ( k = 0; k < 3; k++ ) {
|
|
|
|
v = vRemap[poly->v[k].index];
|
|
|
|
normal.x = poly->v[k].norm[0];
|
|
normal.y = poly->v[k].norm[2];
|
|
normal.z = poly->v[k].norm[1];
|
|
|
|
// LWO models aren't all that pretty when it comes down to the floating point values they store
|
|
normal.FixDegenerateNormal();
|
|
|
|
tv = 0;
|
|
|
|
color[0] = lwoSurf->color.rgb[0] * 255;
|
|
color[1] = lwoSurf->color.rgb[1] * 255;
|
|
color[2] = lwoSurf->color.rgb[2] * 255;
|
|
color[3] = 255;
|
|
|
|
// first set attributes from the vertex
|
|
lwPoint *pt = &layer->point.pt[poly->v[k].index];
|
|
int nvm;
|
|
for ( nvm = 0; nvm < pt->nvmaps; nvm++ ) {
|
|
lwVMapPt *vm = &pt->vm[nvm];
|
|
|
|
if ( vm->vmap->type == LWID_('T','X','U','V') ) {
|
|
tv = tvRemap[vm->index + vm->vmap->offset];
|
|
}
|
|
if ( vm->vmap->type == LWID_('R','G','B','A') ) {
|
|
for ( int chan = 0; chan < 4; chan++ ) {
|
|
color[chan] = 255 * vm->vmap->val[vm->index][chan];
|
|
}
|
|
}
|
|
}
|
|
|
|
// then override with polygon attributes
|
|
for ( nvm = 0; nvm < poly->v[k].nvmaps; nvm++ ) {
|
|
lwVMapPt *vm = &poly->v[k].vm[nvm];
|
|
|
|
if ( vm->vmap->type == LWID_('T','X','U','V') ) {
|
|
tv = tvRemap[vm->index + vm->vmap->offset];
|
|
}
|
|
if ( vm->vmap->type == LWID_('R','G','B','A') ) {
|
|
for ( int chan = 0; chan < 4; chan++ ) {
|
|
color[chan] = 255 * vm->vmap->val[vm->index][chan];
|
|
}
|
|
}
|
|
}
|
|
|
|
// find a matching vert
|
|
for ( lastmv = NULL, mv = mvHash[v]; mv != NULL; lastmv = mv, mv = mv->next ) {
|
|
if ( mv->tv != tv ) {
|
|
continue;
|
|
}
|
|
if ( *(unsigned *)mv->color != *(unsigned *)color ) {
|
|
continue;
|
|
}
|
|
if ( !normalsParsed ) {
|
|
// if we are going to create the normals, just
|
|
// matching texcoords is enough
|
|
break;
|
|
}
|
|
if ( mv->normal * normal > normalEpsilon ) {
|
|
break; // we already have this one
|
|
}
|
|
}
|
|
if ( !mv ) {
|
|
// allocate a new match vert and link to hash chain
|
|
mv = &mvTable[ tri->numVerts ];
|
|
mv->v = v;
|
|
mv->tv = tv;
|
|
mv->normal = normal;
|
|
*(unsigned *)mv->color = *(unsigned *)color;
|
|
mv->next = NULL;
|
|
if ( lastmv ) {
|
|
lastmv->next = mv;
|
|
} else {
|
|
mvHash[v] = mv;
|
|
}
|
|
tri->numVerts++;
|
|
}
|
|
|
|
tri->indexes[tri->numIndexes] = mv - mvTable;
|
|
tri->numIndexes++;
|
|
}
|
|
}
|
|
|
|
// allocate space for the indexes and copy them
|
|
if ( tri->numIndexes > layer->polygon.count * 3 ) {
|
|
common->FatalError( "ConvertLWOToModelSurfaces: index miscount in LWO file %s", name.c_str() );
|
|
}
|
|
if ( tri->numVerts > layer->polygon.count * 3 ) {
|
|
common->FatalError( "ConvertLWOToModelSurfaces: vertex miscount in LWO file %s", name.c_str() );
|
|
}
|
|
|
|
// now allocate and generate the combined vertexes
|
|
R_AllocStaticTriSurfVerts( tri, tri->numVerts );
|
|
for ( j = 0; j < tri->numVerts; j++ ) {
|
|
mv = &mvTable[j];
|
|
tri->verts[ j ].Clear();
|
|
tri->verts[ j ].xyz = vList[ mv->v ];
|
|
tri->verts[ j ].st = tvList[ mv->tv ];
|
|
tri->verts[ j ].normal = mv->normal;
|
|
*(unsigned *)tri->verts[j].color = *(unsigned *)mv->color;
|
|
}
|
|
|
|
R_StaticFree( mvTable );
|
|
R_StaticFree( mvHash );
|
|
|
|
// see if we need to merge with a previous surface of the same material
|
|
modelSurf = &this->surfaces[mergeTo[ i ]];
|
|
srfTriangles_t *mergeTri = modelSurf->geometry;
|
|
if ( !mergeTri ) {
|
|
modelSurf->geometry = tri;
|
|
} else {
|
|
modelSurf->geometry = R_MergeTriangles( mergeTri, tri );
|
|
R_FreeStaticTriSurf( tri );
|
|
R_FreeStaticTriSurf( mergeTri );
|
|
}
|
|
}
|
|
|
|
R_StaticFree( tvRemap );
|
|
R_StaticFree( vRemap );
|
|
R_StaticFree( tvList );
|
|
R_StaticFree( vList );
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::ConvertLWOToASE
|
|
=================
|
|
*/
|
|
struct aseModel_s *idRenderModelStatic::ConvertLWOToASE( const struct st_lwObject *obj, const char *fileName ) {
|
|
int j, k;
|
|
aseModel_t *ase;
|
|
|
|
if ( !obj ) {
|
|
return NULL;
|
|
}
|
|
|
|
// NOTE: using new operator because aseModel_t contains idList class objects
|
|
ase = new aseModel_t;
|
|
ase->timeStamp = obj->timeStamp;
|
|
ase->objects.Resize( obj->nlayers, obj->nlayers );
|
|
|
|
int materialRef = 0;
|
|
|
|
for ( lwSurface *surf = obj->surf; surf; surf = surf->next ) {
|
|
|
|
aseMaterial_t *mat = (aseMaterial_t *)Mem_ClearedAlloc( sizeof( *mat ) );
|
|
strcpy( mat->name, surf->name );
|
|
mat->uTiling = mat->vTiling = 1;
|
|
mat->angle = mat->uOffset = mat->vOffset = 0;
|
|
ase->materials.Append( mat );
|
|
|
|
lwLayer *layer = obj->layer;
|
|
|
|
aseObject_t *object = (aseObject_t *)Mem_ClearedAlloc( sizeof( *object ) );
|
|
object->materialRef = materialRef++;
|
|
|
|
aseMesh_t *mesh = &object->mesh;
|
|
ase->objects.Append( object );
|
|
|
|
mesh->numFaces = layer->polygon.count;
|
|
mesh->numTVFaces = mesh->numFaces;
|
|
mesh->faces = (aseFace_t *)Mem_Alloc( mesh->numFaces * sizeof( mesh->faces[0] ) );
|
|
|
|
mesh->numVertexes = layer->point.count;
|
|
mesh->vertexes = (idVec3 *)Mem_Alloc( mesh->numVertexes * sizeof( mesh->vertexes[0] ) );
|
|
|
|
// vertex positions
|
|
if ( layer->point.count <= 0 ) {
|
|
common->Warning( "ConvertLWOToASE: model \'%s\' has bad or missing vertex data", name.c_str() );
|
|
}
|
|
|
|
for ( j = 0; j < layer->point.count; j++ ) {
|
|
mesh->vertexes[j].x = layer->point.pt[j].pos[0];
|
|
mesh->vertexes[j].y = layer->point.pt[j].pos[2];
|
|
mesh->vertexes[j].z = layer->point.pt[j].pos[1];
|
|
}
|
|
|
|
// vertex texture coords
|
|
mesh->numTVertexes = 0;
|
|
|
|
if ( layer->nvmaps ) {
|
|
for( lwVMap *vm = layer->vmap; vm; vm = vm->next ) {
|
|
if ( vm->type == LWID_('T','X','U','V') ) {
|
|
mesh->numTVertexes += vm->nverts;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( mesh->numTVertexes ) {
|
|
mesh->tvertexes = (idVec2 *)Mem_Alloc( mesh->numTVertexes * sizeof( mesh->tvertexes[0] ) );
|
|
int offset = 0;
|
|
for( lwVMap *vm = layer->vmap; vm; vm = vm->next ) {
|
|
if ( vm->type == LWID_('T','X','U','V') ) {
|
|
vm->offset = offset;
|
|
for ( k = 0; k < vm->nverts; k++ ) {
|
|
mesh->tvertexes[k + offset].x = vm->val[k][0];
|
|
mesh->tvertexes[k + offset].y = 1.0f - vm->val[k][1]; // invert the t
|
|
}
|
|
offset += vm->nverts;
|
|
}
|
|
}
|
|
} else {
|
|
common->Warning( "ConvertLWOToASE: model \'%s\' has bad or missing uv data", fileName );
|
|
mesh->numTVertexes = 1;
|
|
mesh->tvertexes = (idVec2 *)Mem_ClearedAlloc( mesh->numTVertexes * sizeof( mesh->tvertexes[0] ) );
|
|
}
|
|
|
|
mesh->normalsParsed = true;
|
|
mesh->colorsParsed = true; // because we are falling back to the surface color
|
|
|
|
// triangles
|
|
int faceIndex = 0;
|
|
for ( j = 0; j < layer->polygon.count; j++ ) {
|
|
lwPolygon *poly = &layer->polygon.pol[j];
|
|
|
|
if ( poly->surf != surf ) {
|
|
continue;
|
|
}
|
|
|
|
if ( poly->nverts != 3 ) {
|
|
common->Warning( "ConvertLWOToASE: model %s has too many verts for a poly! Make sure you triplet it down", fileName );
|
|
continue;
|
|
}
|
|
|
|
mesh->faces[faceIndex].faceNormal.x = poly->norm[0];
|
|
mesh->faces[faceIndex].faceNormal.y = poly->norm[2];
|
|
mesh->faces[faceIndex].faceNormal.z = poly->norm[1];
|
|
|
|
for ( k = 0; k < 3; k++ ) {
|
|
|
|
mesh->faces[faceIndex].vertexNum[k] = poly->v[k].index;
|
|
|
|
mesh->faces[faceIndex].vertexNormals[k].x = poly->v[k].norm[0];
|
|
mesh->faces[faceIndex].vertexNormals[k].y = poly->v[k].norm[2];
|
|
mesh->faces[faceIndex].vertexNormals[k].z = poly->v[k].norm[1];
|
|
|
|
// complete fallbacks
|
|
mesh->faces[faceIndex].tVertexNum[k] = 0;
|
|
|
|
mesh->faces[faceIndex].vertexColors[k][0] = surf->color.rgb[0] * 255;
|
|
mesh->faces[faceIndex].vertexColors[k][1] = surf->color.rgb[1] * 255;
|
|
mesh->faces[faceIndex].vertexColors[k][2] = surf->color.rgb[2] * 255;
|
|
mesh->faces[faceIndex].vertexColors[k][3] = 255;
|
|
|
|
// first set attributes from the vertex
|
|
lwPoint *pt = &layer->point.pt[poly->v[k].index];
|
|
int nvm;
|
|
for ( nvm = 0; nvm < pt->nvmaps; nvm++ ) {
|
|
lwVMapPt *vm = &pt->vm[nvm];
|
|
|
|
if ( vm->vmap->type == LWID_('T','X','U','V') ) {
|
|
mesh->faces[faceIndex].tVertexNum[k] = vm->index + vm->vmap->offset;
|
|
}
|
|
if ( vm->vmap->type == LWID_('R','G','B','A') ) {
|
|
for ( int chan = 0; chan < 4; chan++ ) {
|
|
mesh->faces[faceIndex].vertexColors[k][chan] = 255 * vm->vmap->val[vm->index][chan];
|
|
}
|
|
}
|
|
}
|
|
|
|
// then override with polygon attributes
|
|
for ( nvm = 0; nvm < poly->v[k].nvmaps; nvm++ ) {
|
|
lwVMapPt *vm = &poly->v[k].vm[nvm];
|
|
|
|
if ( vm->vmap->type == LWID_('T','X','U','V') ) {
|
|
mesh->faces[faceIndex].tVertexNum[k] = vm->index + vm->vmap->offset;
|
|
}
|
|
if ( vm->vmap->type == LWID_('R','G','B','A') ) {
|
|
for ( int chan = 0; chan < 4; chan++ ) {
|
|
mesh->faces[faceIndex].vertexColors[k][chan] = 255 * vm->vmap->val[vm->index][chan];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
faceIndex++;
|
|
}
|
|
|
|
mesh->numFaces = faceIndex;
|
|
mesh->numTVFaces = faceIndex;
|
|
|
|
aseFace_t *newFaces = ( aseFace_t* )Mem_Alloc( mesh->numFaces * sizeof ( mesh->faces[0] ) );
|
|
memcpy( newFaces, mesh->faces, sizeof( mesh->faces[0] ) * mesh->numFaces );
|
|
Mem_Free( mesh->faces );
|
|
mesh->faces = newFaces;
|
|
}
|
|
|
|
return ase;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::ConvertMAToModelSurfaces
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::ConvertMAToModelSurfaces (const struct maModel_s *ma ) {
|
|
|
|
maObject_t * object;
|
|
maMesh_t * mesh;
|
|
maMaterial_t * material;
|
|
|
|
const idMaterial *im1, *im2;
|
|
srfTriangles_t *tri;
|
|
int objectNum;
|
|
int i, j, k;
|
|
int v, tv;
|
|
int * vRemap;
|
|
int * tvRemap;
|
|
matchVert_t * mvTable; // all of the match verts
|
|
matchVert_t ** mvHash; // points inside mvTable for each xyz index
|
|
matchVert_t * lastmv;
|
|
matchVert_t * mv;
|
|
idVec3 normal;
|
|
float uOffset, vOffset, textureSin, textureCos;
|
|
float uTiling, vTiling;
|
|
int * mergeTo;
|
|
byte * color;
|
|
static byte identityColor[4] = { 255, 255, 255, 255 };
|
|
modelSurface_t surf, *modelSurf;
|
|
|
|
if ( !ma ) {
|
|
return false;
|
|
}
|
|
if ( ma->objects.Num() < 1 ) {
|
|
return false;
|
|
}
|
|
|
|
timeStamp = ma->timeStamp;
|
|
|
|
// the modeling programs can save out multiple surfaces with a common
|
|
// material, but we would like to mege them together where possible
|
|
// meaning that this->NumSurfaces() <= ma->objects.currentElements
|
|
mergeTo = (int *)_alloca( ma->objects.Num() * sizeof( *mergeTo ) );
|
|
|
|
surf.geometry = NULL;
|
|
if ( ma->materials.Num() == 0 ) {
|
|
// if we don't have any materials, dump everything into a single surface
|
|
surf.shader = tr.defaultMaterial;
|
|
surf.id = 0;
|
|
this->AddSurface( surf );
|
|
for ( i = 0 ; i < ma->objects.Num() ; i++ ) {
|
|
mergeTo[i] = 0;
|
|
}
|
|
} else if ( !r_mergeModelSurfaces.GetBool() ) {
|
|
// don't merge any
|
|
for ( i = 0 ; i < ma->objects.Num() ; i++ ) {
|
|
mergeTo[i] = i;
|
|
object = ma->objects[i];
|
|
if(object->materialRef >= 0) {
|
|
material = ma->materials[object->materialRef];
|
|
surf.shader = declManager->FindMaterial( material->name );
|
|
} else {
|
|
surf.shader = tr.defaultMaterial;
|
|
}
|
|
surf.id = this->NumSurfaces();
|
|
this->AddSurface( surf );
|
|
}
|
|
} else {
|
|
// search for material matches
|
|
for ( i = 0 ; i < ma->objects.Num() ; i++ ) {
|
|
object = ma->objects[i];
|
|
if(object->materialRef >= 0) {
|
|
material = ma->materials[object->materialRef];
|
|
im1 = declManager->FindMaterial( material->name );
|
|
} else {
|
|
im1 = tr.defaultMaterial;
|
|
}
|
|
if ( im1->IsDiscrete() ) {
|
|
// flares, autosprites, etc
|
|
j = this->NumSurfaces();
|
|
} else {
|
|
for ( j = 0 ; j < this->NumSurfaces() ; j++ ) {
|
|
modelSurf = &this->surfaces[j];
|
|
im2 = modelSurf->shader;
|
|
if ( im1 == im2 ) {
|
|
// merge this
|
|
mergeTo[i] = j;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if ( j == this->NumSurfaces() ) {
|
|
// didn't merge
|
|
mergeTo[i] = j;
|
|
surf.shader = im1;
|
|
surf.id = this->NumSurfaces();
|
|
this->AddSurface( surf );
|
|
}
|
|
}
|
|
}
|
|
|
|
idVectorSubset<idVec3, 3> vertexSubset;
|
|
idVectorSubset<idVec2, 2> texCoordSubset;
|
|
|
|
// build the surfaces
|
|
for ( objectNum = 0 ; objectNum < ma->objects.Num() ; objectNum++ ) {
|
|
object = ma->objects[objectNum];
|
|
mesh = &object->mesh;
|
|
if(object->materialRef >= 0) {
|
|
material = ma->materials[object->materialRef];
|
|
im1 = declManager->FindMaterial( material->name );
|
|
} else {
|
|
im1 = tr.defaultMaterial;
|
|
}
|
|
|
|
bool normalsParsed = mesh->normalsParsed;
|
|
|
|
// completely ignore any explict normals on surfaces with a renderbump command
|
|
// which will guarantee the best contours and least vertexes.
|
|
const char *rb = im1->GetRenderBump();
|
|
if ( rb && rb[0] ) {
|
|
normalsParsed = false;
|
|
}
|
|
|
|
// It seems like the tools our artists are using often generate
|
|
// verts and texcoords slightly separated that should be merged
|
|
// note that we really should combine the surfaces with common materials
|
|
// before doing this operation, because we can miss a slop combination
|
|
// if they are in different surfaces
|
|
|
|
vRemap = (int *)R_StaticAlloc( mesh->numVertexes * sizeof( vRemap[0] ) );
|
|
|
|
if ( fastLoad ) {
|
|
// renderbump doesn't care about vertex count
|
|
for ( j = 0; j < mesh->numVertexes; j++ ) {
|
|
vRemap[j] = j;
|
|
}
|
|
} else {
|
|
float vertexEpsilon = r_slopVertex.GetFloat();
|
|
float expand = 2 * 32 * vertexEpsilon;
|
|
idVec3 mins, maxs;
|
|
|
|
SIMDProcessor->MinMax( mins, maxs, mesh->vertexes, mesh->numVertexes );
|
|
mins -= idVec3( expand, expand, expand );
|
|
maxs += idVec3( expand, expand, expand );
|
|
vertexSubset.Init( mins, maxs, 32, 1024 );
|
|
for ( j = 0; j < mesh->numVertexes; j++ ) {
|
|
vRemap[j] = vertexSubset.FindVector( mesh->vertexes, j, vertexEpsilon );
|
|
}
|
|
}
|
|
|
|
tvRemap = (int *)R_StaticAlloc( mesh->numTVertexes * sizeof( tvRemap[0] ) );
|
|
|
|
if ( fastLoad ) {
|
|
// renderbump doesn't care about vertex count
|
|
for ( j = 0; j < mesh->numTVertexes; j++ ) {
|
|
tvRemap[j] = j;
|
|
}
|
|
} else {
|
|
float texCoordEpsilon = r_slopTexCoord.GetFloat();
|
|
float expand = 2 * 32 * texCoordEpsilon;
|
|
idVec2 mins, maxs;
|
|
|
|
SIMDProcessor->MinMax( mins, maxs, mesh->tvertexes, mesh->numTVertexes );
|
|
mins -= idVec2( expand, expand );
|
|
maxs += idVec2( expand, expand );
|
|
texCoordSubset.Init( mins, maxs, 32, 1024 );
|
|
for ( j = 0; j < mesh->numTVertexes; j++ ) {
|
|
tvRemap[j] = texCoordSubset.FindVector( mesh->tvertexes, j, texCoordEpsilon );
|
|
}
|
|
}
|
|
|
|
// we need to find out how many unique vertex / texcoord / color combinations
|
|
// there are, because MA tracks them separately but we need them unified
|
|
|
|
// the maximum possible number of combined vertexes is the number of indexes
|
|
mvTable = (matchVert_t *)R_ClearedStaticAlloc( mesh->numFaces * 3 * sizeof( mvTable[0] ) );
|
|
|
|
// we will have a hash chain based on the xyz values
|
|
mvHash = (matchVert_t **)R_ClearedStaticAlloc( mesh->numVertexes * sizeof( mvHash[0] ) );
|
|
|
|
// allocate triangle surface
|
|
tri = R_AllocStaticTriSurf();
|
|
tri->numVerts = 0;
|
|
tri->numIndexes = 0;
|
|
R_AllocStaticTriSurfIndexes( tri, mesh->numFaces * 3 );
|
|
tri->generateNormals = !normalsParsed;
|
|
|
|
// init default normal, color and tex coord index
|
|
normal.Zero();
|
|
color = identityColor;
|
|
tv = 0;
|
|
|
|
// find all the unique combinations
|
|
float normalEpsilon = 1.0f - r_slopNormal.GetFloat();
|
|
for ( j = 0; j < mesh->numFaces; j++ ) {
|
|
for ( k = 0; k < 3; k++ ) {
|
|
v = mesh->faces[j].vertexNum[k];
|
|
|
|
if ( v < 0 || v >= mesh->numVertexes ) {
|
|
common->Error( "ConvertMAToModelSurfaces: bad vertex index in MA file %s", name.c_str() );
|
|
}
|
|
|
|
// collapse the position if it was slightly offset
|
|
v = vRemap[v];
|
|
|
|
// we may or may not have texcoords to compare
|
|
if ( mesh->numTVertexes != 0 ) {
|
|
tv = mesh->faces[j].tVertexNum[k];
|
|
if ( tv < 0 || tv >= mesh->numTVertexes ) {
|
|
common->Error( "ConvertMAToModelSurfaces: bad tex coord index in MA file %s", name.c_str() );
|
|
}
|
|
// collapse the tex coord if it was slightly offset
|
|
tv = tvRemap[tv];
|
|
}
|
|
|
|
// we may or may not have normals to compare
|
|
if ( normalsParsed ) {
|
|
normal = mesh->faces[j].vertexNormals[k];
|
|
}
|
|
|
|
//BSM: Todo: Fix the vertex colors
|
|
// we may or may not have colors to compare
|
|
if ( mesh->faces[j].vertexColors[k] != -1 && mesh->faces[j].vertexColors[k] != -999 ) {
|
|
|
|
color = &mesh->colors[mesh->faces[j].vertexColors[k]*4];
|
|
}
|
|
|
|
// find a matching vert
|
|
for ( lastmv = NULL, mv = mvHash[v]; mv != NULL; lastmv = mv, mv = mv->next ) {
|
|
if ( mv->tv != tv ) {
|
|
continue;
|
|
}
|
|
if ( *(unsigned *)mv->color != *(unsigned *)color ) {
|
|
continue;
|
|
}
|
|
if ( !normalsParsed ) {
|
|
// if we are going to create the normals, just
|
|
// matching texcoords is enough
|
|
break;
|
|
}
|
|
if ( mv->normal * normal > normalEpsilon ) {
|
|
break; // we already have this one
|
|
}
|
|
}
|
|
if ( !mv ) {
|
|
// allocate a new match vert and link to hash chain
|
|
mv = &mvTable[ tri->numVerts ];
|
|
mv->v = v;
|
|
mv->tv = tv;
|
|
mv->normal = normal;
|
|
*(unsigned *)mv->color = *(unsigned *)color;
|
|
mv->next = NULL;
|
|
if ( lastmv ) {
|
|
lastmv->next = mv;
|
|
} else {
|
|
mvHash[v] = mv;
|
|
}
|
|
tri->numVerts++;
|
|
}
|
|
|
|
tri->indexes[tri->numIndexes] = mv - mvTable;
|
|
tri->numIndexes++;
|
|
}
|
|
}
|
|
|
|
// allocate space for the indexes and copy them
|
|
if ( tri->numIndexes > mesh->numFaces * 3 ) {
|
|
common->FatalError( "ConvertMAToModelSurfaces: index miscount in MA file %s", name.c_str() );
|
|
}
|
|
if ( tri->numVerts > mesh->numFaces * 3 ) {
|
|
common->FatalError( "ConvertMAToModelSurfaces: vertex miscount in MA file %s", name.c_str() );
|
|
}
|
|
|
|
// an MA allows the texture coordinates to be scaled, translated, and rotated
|
|
//BSM: Todo: Does Maya support this and if so how
|
|
//if ( ase->materials.Num() == 0 ) {
|
|
uOffset = vOffset = 0.0f;
|
|
uTiling = vTiling = 1.0f;
|
|
textureSin = 0.0f;
|
|
textureCos = 1.0f;
|
|
//} else {
|
|
// material = ase->materials[object->materialRef];
|
|
// uOffset = -material->uOffset;
|
|
// vOffset = material->vOffset;
|
|
// uTiling = material->uTiling;
|
|
// vTiling = material->vTiling;
|
|
// textureSin = idMath::Sin( material->angle );
|
|
// textureCos = idMath::Cos( material->angle );
|
|
//}
|
|
|
|
// now allocate and generate the combined vertexes
|
|
R_AllocStaticTriSurfVerts( tri, tri->numVerts );
|
|
for ( j = 0; j < tri->numVerts; j++ ) {
|
|
mv = &mvTable[j];
|
|
tri->verts[ j ].Clear();
|
|
tri->verts[ j ].xyz = mesh->vertexes[ mv->v ];
|
|
tri->verts[ j ].normal = mv->normal;
|
|
*(unsigned *)tri->verts[j].color = *(unsigned *)mv->color;
|
|
if ( mesh->numTVertexes != 0 ) {
|
|
const idVec2 &tv = mesh->tvertexes[ mv->tv ];
|
|
float u = tv.x * uTiling + uOffset;
|
|
float v = tv.y * vTiling + vOffset;
|
|
tri->verts[ j ].st[0] = u * textureCos + v * textureSin;
|
|
tri->verts[ j ].st[1] = u * -textureSin + v * textureCos;
|
|
}
|
|
}
|
|
|
|
R_StaticFree( mvTable );
|
|
R_StaticFree( mvHash );
|
|
R_StaticFree( tvRemap );
|
|
R_StaticFree( vRemap );
|
|
|
|
// see if we need to merge with a previous surface of the same material
|
|
modelSurf = &this->surfaces[mergeTo[ objectNum ]];
|
|
srfTriangles_t *mergeTri = modelSurf->geometry;
|
|
if ( !mergeTri ) {
|
|
modelSurf->geometry = tri;
|
|
} else {
|
|
modelSurf->geometry = R_MergeTriangles( mergeTri, tri );
|
|
R_FreeStaticTriSurf( tri );
|
|
R_FreeStaticTriSurf( mergeTri );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::LoadASE
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::LoadASE( const char *fileName ) {
|
|
aseModel_t *ase;
|
|
|
|
ase = ASE_Load( fileName );
|
|
if ( ase == NULL ) {
|
|
return false;
|
|
}
|
|
|
|
ConvertASEToModelSurfaces( ase );
|
|
|
|
ASE_Free( ase );
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::LoadLWO
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::LoadLWO( const char *fileName ) {
|
|
unsigned int failID;
|
|
int failPos;
|
|
lwObject *lwo;
|
|
|
|
lwo = lwGetObject( fileName, &failID, &failPos );
|
|
if ( lwo == NULL ) {
|
|
return false;
|
|
}
|
|
|
|
ConvertLWOToModelSurfaces( lwo );
|
|
|
|
lwFreeObject( lwo );
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::LoadMA
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::LoadMA( const char *fileName ) {
|
|
maModel_t *ma;
|
|
|
|
ma = MA_Load( fileName );
|
|
if ( ma == NULL ) {
|
|
return false;
|
|
}
|
|
|
|
ConvertMAToModelSurfaces( ma );
|
|
|
|
MA_Free( ma );
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::LoadFLT
|
|
|
|
USGS height map data for megaTexture experiments
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::LoadFLT( const char *fileName ) {
|
|
float *data;
|
|
int len;
|
|
|
|
len = fileSystem->ReadFile( fileName, (void **)&data );
|
|
if ( len <= 0 ) {
|
|
return false;
|
|
}
|
|
int size = sqrt( len / 4.0f );
|
|
|
|
// bound the altitudes
|
|
float min = 9999999;
|
|
float max = -9999999;
|
|
for ( int i = 0 ; i < len/4 ; i++ ) {
|
|
data[i] = BigFloat( data[i] );
|
|
if ( data[i] == -9999 ) {
|
|
data[i] = 0; // unscanned areas
|
|
}
|
|
|
|
if ( data[i] < min ) {
|
|
min = data[i];
|
|
}
|
|
if ( data[i] > max ) {
|
|
max = data[i];
|
|
}
|
|
}
|
|
#if 1
|
|
// write out a gray scale height map
|
|
byte *image = (byte *)R_StaticAlloc( len );
|
|
byte *image_p = image;
|
|
for ( int i = 0 ; i < len/4 ; i++ ) {
|
|
float v = ( data[i] - min ) / ( max - min );
|
|
image_p[0] =
|
|
image_p[1] =
|
|
image_p[2] = v * 255;
|
|
image_p[3] = 255;
|
|
image_p += 4;
|
|
}
|
|
idStr tgaName = fileName;
|
|
tgaName.StripFileExtension();
|
|
tgaName += ".tga";
|
|
R_WriteTGA( tgaName.c_str(), image, size, size, false );
|
|
R_StaticFree( image );
|
|
//return false;
|
|
#endif
|
|
|
|
// find the island above sea level
|
|
int minX, maxX, minY, maxY;
|
|
{
|
|
int i;
|
|
for ( minX = 0 ; minX < size ; minX++ ) {
|
|
for ( i = 0 ; i < size ; i++ ) {
|
|
if ( data[i*size + minX] > 1.0 ) {
|
|
break;
|
|
}
|
|
}
|
|
if ( i != size ) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
for ( maxX = size-1 ; maxX > 0 ; maxX-- ) {
|
|
for ( i = 0 ; i < size ; i++ ) {
|
|
if ( data[i*size + maxX] > 1.0 ) {
|
|
break;
|
|
}
|
|
}
|
|
if ( i != size ) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
for ( minY = 0 ; minY < size ; minY++ ) {
|
|
for ( i = 0 ; i < size ; i++ ) {
|
|
if ( data[minY*size + i] > 1.0 ) {
|
|
break;
|
|
}
|
|
}
|
|
if ( i != size ) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
for ( maxY = size-1 ; maxY < size ; maxY-- ) {
|
|
for ( i = 0 ; i < size ; i++ ) {
|
|
if ( data[maxY*size + i] > 1.0 ) {
|
|
break;
|
|
}
|
|
}
|
|
if ( i != size ) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
int width = maxX - minX + 1;
|
|
int height = maxY - minY + 1;
|
|
|
|
//width /= 2;
|
|
// allocate triangle surface
|
|
srfTriangles_t *tri = R_AllocStaticTriSurf();
|
|
tri->numVerts = width * height;
|
|
tri->numIndexes = (width-1) * (height-1) * 6;
|
|
|
|
fastLoad = true; // don't do all the sil processing
|
|
|
|
R_AllocStaticTriSurfIndexes( tri, tri->numIndexes );
|
|
R_AllocStaticTriSurfVerts( tri, tri->numVerts );
|
|
|
|
for ( int i = 0 ; i < height ; i++ ) {
|
|
for ( int j = 0; j < width ; j++ ) {
|
|
int v = i * width + j;
|
|
tri->verts[ v ].Clear();
|
|
tri->verts[ v ].xyz[0] = j * 10; // each sample is 10 meters
|
|
tri->verts[ v ].xyz[1] = -i * 10;
|
|
tri->verts[ v ].xyz[2] = data[(minY+i)*size+minX+j]; // height is in meters
|
|
tri->verts[ v ].st[0] = (float) j / (width-1);
|
|
tri->verts[ v ].st[1] = 1.0 - ( (float) i / (height-1) );
|
|
}
|
|
}
|
|
|
|
for ( int i = 0 ; i < height-1 ; i++ ) {
|
|
for ( int j = 0; j < width-1 ; j++ ) {
|
|
int v = ( i * (width-1) + j ) * 6;
|
|
#if 0
|
|
tri->indexes[ v + 0 ] = i * width + j;
|
|
tri->indexes[ v + 1 ] = (i+1) * width + j;
|
|
tri->indexes[ v + 2 ] = (i+1) * width + j + 1;
|
|
tri->indexes[ v + 3 ] = i * width + j;
|
|
tri->indexes[ v + 4 ] = (i+1) * width + j + 1;
|
|
tri->indexes[ v + 5 ] = i * width + j + 1;
|
|
#else
|
|
tri->indexes[ v + 0 ] = i * width + j;
|
|
tri->indexes[ v + 1 ] = i * width + j + 1;
|
|
tri->indexes[ v + 2 ] = (i+1) * width + j + 1;
|
|
tri->indexes[ v + 3 ] = i * width + j;
|
|
tri->indexes[ v + 4 ] = (i+1) * width + j + 1;
|
|
tri->indexes[ v + 5 ] = (i+1) * width + j;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
fileSystem->FreeFile( data );
|
|
|
|
modelSurface_t surface;
|
|
|
|
surface.geometry = tri;
|
|
surface.id = 0;
|
|
surface.shader = tr.defaultMaterial; // declManager->FindMaterial( "shaderDemos/megaTexture" );
|
|
|
|
this->AddSurface( surface );
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
//=============================================================================
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::PurgeModel
|
|
================
|
|
*/
|
|
void idRenderModelStatic::PurgeModel() {
|
|
int i;
|
|
modelSurface_t *surf;
|
|
|
|
for ( i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
surf = &surfaces[i];
|
|
|
|
if ( surf->geometry ) {
|
|
R_FreeStaticTriSurf( surf->geometry );
|
|
}
|
|
}
|
|
surfaces.Clear();
|
|
|
|
purged = true;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idRenderModelStatic::FreeVertexCache
|
|
|
|
We are about to restart the vertex cache, so dump everything
|
|
==============
|
|
*/
|
|
void idRenderModelStatic::FreeVertexCache( void ) {
|
|
for ( int j = 0 ; j < surfaces.Num() ; j++ ) {
|
|
srfTriangles_t *tri = surfaces[j].geometry;
|
|
if ( !tri ) {
|
|
continue;
|
|
}
|
|
|
|
// Free the vertex caches
|
|
// NB: these are always private, so we can free them without worrying
|
|
if ( tri->ambientCache ) {
|
|
vertexCache.Free( tri->ambientCache );
|
|
tri->ambientCache = NULL;
|
|
}
|
|
if ( tri->indexCache ) {
|
|
vertexCache.Free( tri->indexCache );
|
|
tri->indexCache = NULL;
|
|
}
|
|
// static shadows may be present
|
|
if ( tri->shadowCache ) {
|
|
vertexCache.Free( tri->shadowCache );
|
|
tri->shadowCache = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::ReadFromDemoFile
|
|
================
|
|
*/
|
|
void idRenderModelStatic::ReadFromDemoFile( class idDemoFile *f ) {
|
|
PurgeModel();
|
|
|
|
InitEmpty( f->ReadHashString() );
|
|
|
|
int i, j, numSurfaces;
|
|
f->ReadInt( numSurfaces );
|
|
|
|
for ( i = 0 ; i < numSurfaces ; i++ ) {
|
|
modelSurface_t surf;
|
|
|
|
surf.shader = declManager->FindMaterial( f->ReadHashString() );
|
|
|
|
srfTriangles_t *tri = R_AllocStaticTriSurf();
|
|
|
|
f->ReadInt( tri->numIndexes );
|
|
R_AllocStaticTriSurfIndexes( tri, tri->numIndexes );
|
|
for ( j = 0; j < tri->numIndexes; ++j )
|
|
f->ReadInt( (int&)tri->indexes[j] );
|
|
|
|
f->ReadInt( tri->numVerts );
|
|
R_AllocStaticTriSurfVerts( tri, tri->numVerts );
|
|
for ( j = 0; j < tri->numVerts; ++j ) {
|
|
f->ReadVec3( tri->verts[j].xyz );
|
|
f->ReadVec2( tri->verts[j].st );
|
|
f->ReadVec3( tri->verts[j].normal );
|
|
f->ReadVec3( tri->verts[j].tangents[0] );
|
|
f->ReadVec3( tri->verts[j].tangents[1] );
|
|
f->ReadUnsignedChar( tri->verts[j].color[0] );
|
|
f->ReadUnsignedChar( tri->verts[j].color[1] );
|
|
f->ReadUnsignedChar( tri->verts[j].color[2] );
|
|
f->ReadUnsignedChar( tri->verts[j].color[3] );
|
|
}
|
|
|
|
surf.geometry = tri;
|
|
|
|
this->AddSurface( surf );
|
|
}
|
|
this->FinishSurfaces();
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::WriteToDemoFile
|
|
================
|
|
*/
|
|
void idRenderModelStatic::WriteToDemoFile( class idDemoFile *f ) {
|
|
int data[1];
|
|
|
|
// note that it has been updated
|
|
lastArchivedFrame = tr.frameCount;
|
|
|
|
data[0] = DC_DEFINE_MODEL;
|
|
f->WriteInt( data[0] );
|
|
f->WriteHashString( this->Name() );
|
|
|
|
int i, j, iData = surfaces.Num();
|
|
f->WriteInt( iData );
|
|
|
|
for ( i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
|
|
f->WriteHashString( surf->shader->GetName() );
|
|
|
|
srfTriangles_t *tri = surf->geometry;
|
|
f->WriteInt( tri->numIndexes );
|
|
for ( j = 0; j < tri->numIndexes; ++j )
|
|
f->WriteInt( (int&)tri->indexes[j] );
|
|
f->WriteInt( tri->numVerts );
|
|
for ( j = 0; j < tri->numVerts; ++j ) {
|
|
f->WriteVec3( tri->verts[j].xyz );
|
|
f->WriteVec2( tri->verts[j].st );
|
|
f->WriteVec3( tri->verts[j].normal );
|
|
f->WriteVec3( tri->verts[j].tangents[0] );
|
|
f->WriteVec3( tri->verts[j].tangents[1] );
|
|
f->WriteUnsignedChar( tri->verts[j].color[0] );
|
|
f->WriteUnsignedChar( tri->verts[j].color[1] );
|
|
f->WriteUnsignedChar( tri->verts[j].color[2] );
|
|
f->WriteUnsignedChar( tri->verts[j].color[3] );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::IsLoaded
|
|
================
|
|
*/
|
|
bool idRenderModelStatic::IsLoaded( void ) const {
|
|
return !purged;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::SetLevelLoadReferenced
|
|
================
|
|
*/
|
|
void idRenderModelStatic::SetLevelLoadReferenced( bool referenced ) {
|
|
levelLoadReferenced = referenced;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idRenderModelStatic::IsLevelLoadReferenced
|
|
================
|
|
*/
|
|
bool idRenderModelStatic::IsLevelLoadReferenced( void ) {
|
|
return levelLoadReferenced;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::TouchData
|
|
=================
|
|
*/
|
|
void idRenderModelStatic::TouchData( void ) {
|
|
for ( int i = 0 ; i < surfaces.Num() ; i++ ) {
|
|
const modelSurface_t *surf = &surfaces[i];
|
|
|
|
// re-find the material to make sure it gets added to the
|
|
// level keep list
|
|
declManager->FindMaterial( surf->shader->GetName() );
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::DeleteSurfaceWithId
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::DeleteSurfaceWithId( int id ) {
|
|
int i;
|
|
|
|
for ( i = 0; i < surfaces.Num(); i++ ) {
|
|
if ( surfaces[i].id == id ) {
|
|
R_FreeStaticTriSurf( surfaces[i].geometry );
|
|
surfaces.RemoveIndex( i );
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::DeleteSurfacesWithNegativeId
|
|
=================
|
|
*/
|
|
void idRenderModelStatic::DeleteSurfacesWithNegativeId( void ) {
|
|
int i;
|
|
|
|
for ( i = 0; i < surfaces.Num(); i++ ) {
|
|
if ( surfaces[i].id < 0 ) {
|
|
R_FreeStaticTriSurf( surfaces[i].geometry );
|
|
surfaces.RemoveIndex( i );
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idRenderModelStatic::FindSurfaceWithId
|
|
=================
|
|
*/
|
|
bool idRenderModelStatic::FindSurfaceWithId( int id, int &surfaceNum ) {
|
|
int i;
|
|
|
|
for ( i = 0; i < surfaces.Num(); i++ ) {
|
|
if ( surfaces[i].id == id ) {
|
|
surfaceNum = i;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|