2016-09-14 18:01:13 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Copyright(C) 2006-2016 Christoph Oelckers
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program 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 Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with this program. If not, see http://www.gnu.org/licenses/
|
|
|
|
//
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
//
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "cmdlib.h"
|
2017-11-25 12:19:00 +00:00
|
|
|
#include "r_data/models/models.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
#define MAX_QPATH 64
|
|
|
|
|
2017-11-25 12:00:44 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(disable:4244) // warning C4244: conversion from 'double' to 'float', possible loss of data
|
|
|
|
#endif
|
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// decode the lat/lng normal to a 3 float normal
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
static void UnpackVector(unsigned short packed, float & nx, float & ny, float & nz)
|
|
|
|
{
|
|
|
|
double lat = ( packed >> 8 ) & 0xff;
|
|
|
|
double lng = ( packed & 0xff );
|
2016-04-28 15:55:58 +00:00
|
|
|
lat *= M_PI/128;
|
|
|
|
lng *= M_PI/128;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
nx = cos(lat) * sin(lng);
|
|
|
|
ny = sin(lat) * sin(lng);
|
|
|
|
nz = cos(lng);
|
|
|
|
}
|
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// MD3 File structure
|
|
|
|
//
|
|
|
|
//===========================================================================
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
#pragma pack(4)
|
|
|
|
struct md3_header_t
|
|
|
|
{
|
2016-05-01 11:09:13 +00:00
|
|
|
uint32_t Magic;
|
|
|
|
uint32_t Version;
|
2015-04-12 17:42:03 +00:00
|
|
|
char Name[MAX_QPATH];
|
2016-05-01 11:09:13 +00:00
|
|
|
uint32_t Flags;
|
|
|
|
uint32_t Num_Frames;
|
|
|
|
uint32_t Num_Tags;
|
|
|
|
uint32_t Num_Surfaces;
|
|
|
|
uint32_t Num_Skins;
|
|
|
|
uint32_t Ofs_Frames;
|
|
|
|
uint32_t Ofs_Tags;
|
|
|
|
uint32_t Ofs_Surfaces;
|
|
|
|
uint32_t Ofs_Eof;
|
2015-04-12 17:42:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct md3_surface_t
|
|
|
|
{
|
2016-05-01 11:09:13 +00:00
|
|
|
uint32_t Magic;
|
2015-04-12 17:42:03 +00:00
|
|
|
char Name[MAX_QPATH];
|
2016-05-01 11:09:13 +00:00
|
|
|
uint32_t Flags;
|
|
|
|
uint32_t Num_Frames;
|
|
|
|
uint32_t Num_Shaders;
|
|
|
|
uint32_t Num_Verts;
|
|
|
|
uint32_t Num_Triangles;
|
|
|
|
uint32_t Ofs_Triangles;
|
|
|
|
uint32_t Ofs_Shaders;
|
|
|
|
uint32_t Ofs_Texcoord;
|
|
|
|
uint32_t Ofs_XYZNormal;
|
|
|
|
uint32_t Ofs_End;
|
2015-04-12 17:42:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct md3_triangle_t
|
|
|
|
{
|
2016-05-01 11:09:13 +00:00
|
|
|
uint32_t vt_index[3];
|
2015-04-12 17:42:03 +00:00
|
|
|
};
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
struct md3_shader_t
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2015-04-12 17:42:03 +00:00
|
|
|
char Name[MAX_QPATH];
|
2016-05-01 11:09:13 +00:00
|
|
|
uint32_t index;
|
2015-04-12 17:42:03 +00:00
|
|
|
};
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
struct md3_texcoord_t
|
|
|
|
{
|
|
|
|
float s, t;
|
|
|
|
};
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
struct md3_vertex_t
|
|
|
|
{
|
|
|
|
short x, y, z, n;
|
|
|
|
};
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
struct md3_frame_t
|
|
|
|
{
|
|
|
|
float min_Bounds[3];
|
|
|
|
float max_Bounds[3];
|
|
|
|
float localorigin[3];
|
|
|
|
float radius;
|
|
|
|
char Name[16];
|
|
|
|
};
|
|
|
|
#pragma pack()
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
bool FMD3Model::Load(const char * path, int lumpnum, const char * buffer, int length)
|
|
|
|
{
|
|
|
|
md3_header_t * hdr = (md3_header_t *)buffer;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
numFrames = LittleLong(hdr->Num_Frames);
|
|
|
|
numTags = LittleLong(hdr->Num_Tags);
|
|
|
|
numSurfaces = LittleLong(hdr->Num_Surfaces);
|
|
|
|
|
|
|
|
md3_frame_t * frm = (md3_frame_t*)(buffer + LittleLong(hdr->Ofs_Frames));
|
|
|
|
|
|
|
|
frames = new MD3Frame[numFrames];
|
2015-04-12 17:42:03 +00:00
|
|
|
for (int i = 0; i < numFrames; i++)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
strncpy(frames[i].Name, frm[i].Name, 16);
|
2015-04-12 17:42:03 +00:00
|
|
|
for (int j = 0; j < 3; j++) frames[i].origin[j] = frm[i].localorigin[j];
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
md3_surface_t * surf = (md3_surface_t*)(buffer + LittleLong(hdr->Ofs_Surfaces));
|
|
|
|
|
|
|
|
surfaces = new MD3Surface[numSurfaces];
|
2015-04-12 17:42:03 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < numSurfaces; i++)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
MD3Surface * s = &surfaces[i];
|
|
|
|
md3_surface_t * ss = surf;
|
|
|
|
|
|
|
|
surf = (md3_surface_t *)(((char*)surf) + LittleLong(surf->Ofs_End));
|
|
|
|
|
|
|
|
s->numSkins = LittleLong(ss->Num_Shaders);
|
|
|
|
s->numTriangles = LittleLong(ss->Num_Triangles);
|
|
|
|
s->numVertices = LittleLong(ss->Num_Verts);
|
|
|
|
|
|
|
|
// copy shaders (skins)
|
2015-04-12 17:42:03 +00:00
|
|
|
md3_shader_t * shader = (md3_shader_t*)(((char*)ss) + LittleLong(ss->Ofs_Shaders));
|
2016-05-03 13:45:21 +00:00
|
|
|
s->skins = new FTextureID[s->numSkins];
|
2013-06-23 07:49:34 +00:00
|
|
|
|
2015-04-12 17:42:03 +00:00
|
|
|
for (int i = 0; i < s->numSkins; i++)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
// [BB] According to the MD3 spec, Name is supposed to include the full path.
|
2017-12-03 19:02:55 +00:00
|
|
|
// ... and since some tools seem to output backslashes, these need to be replaced with forward slashes to work.
|
|
|
|
FixPathSeperator(shader[i].Name);
|
2013-06-23 07:49:34 +00:00
|
|
|
s->skins[i] = LoadSkin("", shader[i].Name);
|
|
|
|
// [BB] Fall back and check if Name is relative.
|
2016-05-03 13:45:21 +00:00
|
|
|
if (!s->skins[i].isValid())
|
2013-06-23 07:49:34 +00:00
|
|
|
s->skins[i] = LoadSkin(path, shader[i].Name);
|
|
|
|
}
|
2015-04-12 17:42:03 +00:00
|
|
|
}
|
|
|
|
mLumpNum = lumpnum;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void FMD3Model::LoadGeometry()
|
|
|
|
{
|
|
|
|
FMemLump lumpdata = Wads.ReadLump(mLumpNum);
|
|
|
|
const char *buffer = (const char *)lumpdata.GetMem();
|
|
|
|
md3_header_t * hdr = (md3_header_t *)buffer;
|
|
|
|
md3_surface_t * surf = (md3_surface_t*)(buffer + LittleLong(hdr->Ofs_Surfaces));
|
|
|
|
|
|
|
|
for(int i=0;i<numSurfaces;i++)
|
|
|
|
{
|
|
|
|
MD3Surface * s = &surfaces[i];
|
|
|
|
md3_surface_t * ss = surf;
|
|
|
|
|
|
|
|
surf = (md3_surface_t *)(((char*)surf) + LittleLong(surf->Ofs_End));
|
|
|
|
|
|
|
|
// copy triangle indices
|
|
|
|
md3_triangle_t * tris = (md3_triangle_t*)(((char*)ss)+LittleLong(ss->Ofs_Triangles));
|
|
|
|
s->tris = new MD3Triangle[s->numTriangles];
|
|
|
|
|
|
|
|
for(int i=0;i<s->numTriangles;i++) for (int j=0;j<3;j++)
|
|
|
|
{
|
|
|
|
s->tris[i].VertIndex[j]=LittleLong(tris[i].vt_index[j]);
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
// Load texture coordinates
|
|
|
|
md3_texcoord_t * tc = (md3_texcoord_t*)(((char*)ss)+LittleLong(ss->Ofs_Texcoord));
|
|
|
|
s->texcoords = new MD3TexCoord[s->numVertices];
|
|
|
|
|
|
|
|
for(int i=0;i<s->numVertices;i++)
|
|
|
|
{
|
|
|
|
s->texcoords[i].s = tc[i].s;
|
|
|
|
s->texcoords[i].t = tc[i].t;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load vertices and texture coordinates
|
|
|
|
md3_vertex_t * vt = (md3_vertex_t*)(((char*)ss)+LittleLong(ss->Ofs_XYZNormal));
|
|
|
|
s->vertices = new MD3Vertex[s->numVertices * numFrames];
|
|
|
|
|
|
|
|
for(int i=0;i<s->numVertices * numFrames;i++)
|
|
|
|
{
|
|
|
|
s->vertices[i].x = LittleShort(vt[i].x)/64.f;
|
|
|
|
s->vertices[i].y = LittleShort(vt[i].y)/64.f;
|
2014-12-31 11:53:29 +00:00
|
|
|
s->vertices[i].z = LittleShort(vt[i].z)/64.f;
|
2013-06-23 07:49:34 +00:00
|
|
|
UnpackVector( LittleShort(vt[i].n), s->vertices[i].nx, s->vertices[i].ny, s->vertices[i].nz);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2017-11-23 23:38:52 +00:00
|
|
|
void FMD3Model::BuildVertexBuffer(FModelRenderer *renderer)
|
2014-06-19 20:24:33 +00:00
|
|
|
{
|
2018-05-21 15:52:03 +00:00
|
|
|
if (!GetVertexBuffer(renderer))
|
2014-06-19 20:24:33 +00:00
|
|
|
{
|
2015-04-12 17:42:03 +00:00
|
|
|
LoadGeometry();
|
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
unsigned int vbufsize = 0;
|
|
|
|
unsigned int ibufsize = 0;
|
2014-06-19 20:24:33 +00:00
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
for (int i = 0; i < numSurfaces; i++)
|
2014-06-19 20:24:33 +00:00
|
|
|
{
|
2014-10-24 09:43:25 +00:00
|
|
|
MD3Surface * surf = &surfaces[i];
|
|
|
|
vbufsize += numFrames * surf->numVertices;
|
|
|
|
ibufsize += 3 * surf->numTriangles;
|
|
|
|
}
|
2014-06-19 20:24:33 +00:00
|
|
|
|
2018-05-21 15:52:03 +00:00
|
|
|
auto vbuf = renderer->CreateVertexBuffer(true, numFrames == 1);
|
|
|
|
SetVertexBuffer(renderer, vbuf);
|
|
|
|
|
|
|
|
FModelVertex *vertptr = vbuf->LockVertexBuffer(vbufsize);
|
|
|
|
unsigned int *indxptr = vbuf->LockIndexBuffer(ibufsize);
|
2014-06-19 20:24:33 +00:00
|
|
|
|
2016-05-03 11:10:00 +00:00
|
|
|
assert(vertptr != nullptr && indxptr != nullptr);
|
2014-06-19 20:24:33 +00:00
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
unsigned int vindex = 0, iindex = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < numSurfaces; i++)
|
2014-06-19 20:24:33 +00:00
|
|
|
{
|
2014-10-24 09:43:25 +00:00
|
|
|
MD3Surface * surf = &surfaces[i];
|
|
|
|
|
|
|
|
surf->vindex = vindex;
|
|
|
|
surf->iindex = iindex;
|
|
|
|
for (int j = 0; j < numFrames * surf->numVertices; j++)
|
|
|
|
{
|
|
|
|
MD3Vertex* vert = surf->vertices + j;
|
|
|
|
|
|
|
|
FModelVertex *bvert = &vertptr[vindex++];
|
|
|
|
|
|
|
|
int tc = j % surf->numVertices;
|
|
|
|
bvert->Set(vert->x, vert->z, vert->y, surf->texcoords[tc].s, surf->texcoords[tc].t);
|
|
|
|
bvert->SetNormal(vert->nx, vert->nz, vert->ny);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int k = 0; k < surf->numTriangles; k++)
|
2014-06-19 20:24:33 +00:00
|
|
|
{
|
2014-10-24 09:43:25 +00:00
|
|
|
for (int l = 0; l < 3; l++)
|
|
|
|
{
|
|
|
|
indxptr[iindex++] = surf->tris[k].VertIndex[l];
|
|
|
|
}
|
2014-06-19 20:24:33 +00:00
|
|
|
}
|
2015-04-12 17:42:03 +00:00
|
|
|
surf->UnloadGeometry();
|
2014-06-19 20:24:33 +00:00
|
|
|
}
|
2018-05-21 15:52:03 +00:00
|
|
|
vbuf->UnlockVertexBuffer();
|
|
|
|
vbuf->UnlockIndexBuffer();
|
2014-06-19 20:24:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-03 16:14:16 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// for skin precaching
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
void FMD3Model::AddSkins(uint8_t *hitlist)
|
2016-05-03 16:14:16 +00:00
|
|
|
{
|
|
|
|
for (int i = 0; i < numSurfaces; i++)
|
|
|
|
{
|
2016-07-14 15:35:30 +00:00
|
|
|
if (curSpriteMDLFrame->surfaceskinIDs[curMDLIndex][i].isValid())
|
|
|
|
{
|
2018-03-25 18:26:16 +00:00
|
|
|
hitlist[curSpriteMDLFrame->surfaceskinIDs[curMDLIndex][i].GetIndex()] |= FTextureManager::HIT_Flat;
|
2016-07-14 15:35:30 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 16:14:16 +00:00
|
|
|
MD3Surface * surf = &surfaces[i];
|
|
|
|
for (int j = 0; j < surf->numSkins; j++)
|
|
|
|
{
|
|
|
|
if (surf->skins[j].isValid())
|
|
|
|
{
|
2018-03-25 18:26:16 +00:00
|
|
|
hitlist[surf->skins[j].GetIndex()] |= FTextureManager::HIT_Flat;
|
2016-05-03 16:14:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
int FMD3Model::FindFrame(const char * name)
|
|
|
|
{
|
|
|
|
for (int i=0;i<numFrames;i++)
|
|
|
|
{
|
|
|
|
if (!stricmp(name, frames[i].Name)) return i;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2017-11-23 23:38:52 +00:00
|
|
|
void FMD3Model::RenderFrame(FModelRenderer *renderer, FTexture * skin, int frameno, int frameno2, double inter, int translation)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
if (frameno>=numFrames || frameno2>=numFrames) return;
|
|
|
|
|
2017-11-23 23:38:52 +00:00
|
|
|
renderer->SetInterpolation(inter);
|
2013-06-23 07:49:34 +00:00
|
|
|
for(int i=0;i<numSurfaces;i++)
|
|
|
|
{
|
|
|
|
MD3Surface * surf = &surfaces[i];
|
|
|
|
|
|
|
|
// [BB] In case no skin is specified via MODELDEF, check if the MD3 has a skin for the current surface.
|
|
|
|
// Note: Each surface may have a different skin.
|
|
|
|
FTexture *surfaceSkin = skin;
|
|
|
|
if (!surfaceSkin)
|
|
|
|
{
|
2016-07-14 15:35:30 +00:00
|
|
|
if (curSpriteMDLFrame->surfaceskinIDs[curMDLIndex][i].isValid())
|
|
|
|
{
|
|
|
|
surfaceSkin = TexMan(curSpriteMDLFrame->surfaceskinIDs[curMDLIndex][i]);
|
|
|
|
}
|
|
|
|
else if(surf->numSkins > 0 && surf->skins[0].isValid())
|
|
|
|
{
|
|
|
|
surfaceSkin = TexMan(surf->skins[0]);
|
|
|
|
}
|
|
|
|
|
2018-10-09 17:16:15 +00:00
|
|
|
if (!surfaceSkin)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
2017-11-25 12:00:44 +00:00
|
|
|
renderer->SetMaterial(surfaceSkin, false, translation);
|
2018-05-21 15:52:03 +00:00
|
|
|
GetVertexBuffer(renderer)->SetupFrame(renderer, surf->vindex + frameno * surf->numVertices, surf->vindex + frameno2 * surf->numVertices, surf->numVertices);
|
2017-11-25 12:00:44 +00:00
|
|
|
renderer->DrawElements(surf->numTriangles * 3, surf->iindex * sizeof(unsigned int));
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2017-11-23 23:38:52 +00:00
|
|
|
renderer->SetInterpolation(0.f);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
2014-10-24 09:43:25 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
FMD3Model::~FMD3Model()
|
|
|
|
{
|
|
|
|
if (frames) delete [] frames;
|
|
|
|
if (surfaces) delete [] surfaces;
|
2016-05-03 11:10:00 +00:00
|
|
|
frames = nullptr;
|
|
|
|
surfaces = nullptr;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|