mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-07 09:40:43 +00:00
240 lines
6 KiB
C++
240 lines
6 KiB
C++
//
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
// Copyright(C) 2005-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 2 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/
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
//
|
|
/*
|
|
** gl_models.cpp
|
|
**
|
|
** General model handling code
|
|
**
|
|
**/
|
|
#include <stddef.h> // offsetof() macro.
|
|
|
|
#include "filesystem.h"
|
|
#include "cmdlib.h"
|
|
#include "sc_man.h"
|
|
#include "m_crc32.h"
|
|
#include "printf.h"
|
|
#include "model_ue1.h"
|
|
#include "model_obj.h"
|
|
#include "model_md2.h"
|
|
#include "model_md3.h"
|
|
#include "model_kvx.h"
|
|
#include "model_iqm.h"
|
|
#include "i_time.h"
|
|
#include "voxels.h"
|
|
#include "texturemanager.h"
|
|
#include "modelrenderer.h"
|
|
|
|
TDeletingArray<FModel*> Models;
|
|
TArray<FSpriteModelFrame> SpriteModelFrames;
|
|
TMap<void*, FSpriteModelFrame> BaseSpriteModelFrames;
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
void FlushModels()
|
|
{
|
|
for (int i = Models.Size() - 1; i >= 0; i--)
|
|
{
|
|
Models[i]->DestroyVertexBuffer();
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
FModel::FModel()
|
|
{
|
|
for (int i = 0; i < NumModelRendererTypes; i++)
|
|
mVBuf[i] = nullptr;
|
|
}
|
|
|
|
FModel::~FModel()
|
|
{
|
|
DestroyVertexBuffer();
|
|
}
|
|
|
|
void FModel::DestroyVertexBuffer()
|
|
{
|
|
for (int i = 0; i < NumModelRendererTypes; i++)
|
|
{
|
|
delete mVBuf[i];
|
|
mVBuf[i] = nullptr;
|
|
}
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
// FindGFXFile
|
|
//
|
|
//===========================================================================
|
|
|
|
static int FindGFXFile(FString & fn)
|
|
{
|
|
int lump = fileSystem.CheckNumForFullName(fn.GetChars()); // if we find something that matches the name plus the extension, return it and do not enter the substitution logic below.
|
|
if (lump != -1) return lump;
|
|
|
|
int best = -1;
|
|
auto dot = fn.LastIndexOf('.');
|
|
auto slash = fn.LastIndexOf('/');
|
|
if (dot > slash) fn.Truncate(dot);
|
|
|
|
static const char * extensions[] = { ".png", ".jpg", ".tga", ".pcx", nullptr };
|
|
|
|
for (const char ** extp=extensions; *extp; extp++)
|
|
{
|
|
lump = fileSystem.CheckNumForFullName((fn + *extp).GetChars());
|
|
if (lump >= best) best = lump;
|
|
}
|
|
return best;
|
|
}
|
|
|
|
|
|
//===========================================================================
|
|
//
|
|
// LoadSkin
|
|
//
|
|
//===========================================================================
|
|
|
|
FTextureID LoadSkin(const char * path, const char * fn)
|
|
{
|
|
FString buffer;
|
|
|
|
buffer.Format("%s%s", path, fn);
|
|
|
|
int texlump = FindGFXFile(buffer);
|
|
const char * const texname = texlump < 0 ? fn : fileSystem.GetFileFullName(texlump);
|
|
return TexMan.CheckForTexture(texname, ETextureType::Any, FTextureManager::TEXMAN_TryAny | FTextureManager::TEXMAN_ForceLookup);
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
// ModelFrameHash
|
|
//
|
|
//===========================================================================
|
|
|
|
int ModelFrameHash(FSpriteModelFrame * smf)
|
|
{
|
|
return crc32(0, (const unsigned char *)(&smf->type), offsetof(FSpriteModelFrame, hashnext) - offsetof(FSpriteModelFrame, type));
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
// FindModel
|
|
//
|
|
//===========================================================================
|
|
|
|
unsigned FindModel(const char * path, const char * modelfile, bool silent)
|
|
{
|
|
FModel * model = nullptr;
|
|
FString fullname;
|
|
|
|
if (path) fullname.Format("%s%s", path, modelfile);
|
|
else fullname = modelfile;
|
|
int lump = fileSystem.CheckNumForFullName(fullname.GetChars());
|
|
|
|
if (lump<0)
|
|
{
|
|
Printf(PRINT_HIGH, "FindModel: '%s' not found\n", fullname.GetChars());
|
|
return -1;
|
|
}
|
|
|
|
for(unsigned i = 0; i< Models.Size(); i++)
|
|
{
|
|
if (Models[i]->mFileName.CompareNoCase(fullname) == 0) return i;
|
|
}
|
|
|
|
auto len = fileSystem.FileLength(lump);
|
|
if (len >= 0x80000000ll)
|
|
{
|
|
Printf(PRINT_HIGH, "LoadModel: File to large: '%s'\n", fullname.GetChars());
|
|
return -1;
|
|
}
|
|
|
|
auto lumpd = fileSystem.ReadFile(lump);
|
|
const char * buffer = lumpd.string();
|
|
|
|
if ( (size_t)fullname.LastIndexOf("_d.3d") == fullname.Len()-5 )
|
|
{
|
|
FString anivfile = fullname.GetChars();
|
|
anivfile.Substitute("_d.3d","_a.3d");
|
|
if ( fileSystem.CheckNumForFullName(anivfile.GetChars()) > 0 )
|
|
{
|
|
model = new FUE1Model;
|
|
}
|
|
}
|
|
else if ( (size_t)fullname.LastIndexOf("_a.3d") == fullname.Len()-5 )
|
|
{
|
|
FString datafile = fullname.GetChars();
|
|
datafile.Substitute("_a.3d","_d.3d");
|
|
if ( fileSystem.CheckNumForFullName(datafile.GetChars()) > 0 )
|
|
{
|
|
model = new FUE1Model;
|
|
}
|
|
}
|
|
else if ( (size_t)fullname.LastIndexOf(".obj") == fullname.Len() - 4 )
|
|
{
|
|
model = new FOBJModel;
|
|
}
|
|
else if (!memcmp(buffer, "DMDM", 4))
|
|
{
|
|
model = new FDMDModel;
|
|
}
|
|
else if (!memcmp(buffer, "IDP2", 4))
|
|
{
|
|
model = new FMD2Model;
|
|
}
|
|
else if (!memcmp(buffer, "IDP3", 4))
|
|
{
|
|
model = new FMD3Model;
|
|
}
|
|
else if (!memcmp(buffer, "INTERQUAKEMODEL\0", 16))
|
|
{
|
|
model = new IQMModel;
|
|
}
|
|
|
|
if (model != nullptr)
|
|
{
|
|
if (!model->Load(path, lump, buffer, (int)len))
|
|
{
|
|
delete model;
|
|
return -1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// try loading as a voxel
|
|
FVoxel *voxel = R_LoadKVX(lump);
|
|
if (voxel != nullptr)
|
|
{
|
|
model = new FVoxelModel(voxel, true);
|
|
}
|
|
else
|
|
{
|
|
Printf(PRINT_HIGH, "LoadModel: Unknown model format in '%s'\n", fullname.GetChars());
|
|
return -1;
|
|
}
|
|
}
|
|
// The vertex buffer cannot be initialized here because this gets called before OpenGL is initialized
|
|
model->mFileName = fullname;
|
|
model->mFilePath = {path, modelfile};
|
|
return Models.Push(model);
|
|
}
|
|
|