2014-09-30 04:18:43 +00:00
|
|
|
//--------------------------------------- VOX LIBRARY BEGINS ---------------------------------------
|
|
|
|
|
|
|
|
#ifdef USE_OPENGL
|
|
|
|
|
|
|
|
#include "compat.h"
|
|
|
|
#include "build.h"
|
|
|
|
#include "engine_priv.h"
|
|
|
|
#include "polymost.h"
|
|
|
|
#include "mdsprite.h"
|
2020-01-18 21:41:08 +00:00
|
|
|
#include "v_video.h"
|
|
|
|
#include "flatvertices.h"
|
2020-05-31 08:24:04 +00:00
|
|
|
#include "hw_renderstate.h"
|
2020-05-29 22:30:56 +00:00
|
|
|
#include "texturemanager.h"
|
2021-03-01 19:48:03 +00:00
|
|
|
#include "voxels.h"
|
|
|
|
#include "glbackend/gl_models.h"
|
2021-03-15 18:39:58 +00:00
|
|
|
#include "printf.h"
|
2019-12-17 22:25:07 +00:00
|
|
|
|
2016-06-21 00:33:06 +00:00
|
|
|
#include "palette.h"
|
2019-09-16 19:08:42 +00:00
|
|
|
#include "../../glbackend/glbackend.h"
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2021-03-15 18:39:58 +00:00
|
|
|
using namespace Polymost;
|
|
|
|
|
2014-09-30 04:18:43 +00:00
|
|
|
void voxfree(voxmodel_t *m)
|
|
|
|
{
|
2014-12-26 17:29:48 +00:00
|
|
|
if (!m)
|
|
|
|
return;
|
|
|
|
|
2019-06-25 11:29:08 +00:00
|
|
|
Xfree(m);
|
2014-09-30 04:18:43 +00:00
|
|
|
}
|
|
|
|
|
2021-03-01 19:48:03 +00:00
|
|
|
voxmodel_t *voxload(int lumpnum)
|
2014-09-30 04:18:43 +00:00
|
|
|
{
|
2021-03-01 19:48:03 +00:00
|
|
|
FVoxel* voxel = R_LoadKVX(lumpnum);
|
|
|
|
if (voxel != nullptr)
|
2014-09-30 04:18:43 +00:00
|
|
|
{
|
2021-03-01 19:48:03 +00:00
|
|
|
voxmodel_t* vm = (voxmodel_t*)Xmalloc(sizeof(voxmodel_t));
|
|
|
|
memset(vm, 0, sizeof(voxmodel_t));
|
|
|
|
auto pivot = voxel->Mips[0].Pivot;
|
2014-09-30 04:18:43 +00:00
|
|
|
vm->mdnum = 1; //VOXel model id
|
|
|
|
vm->scale = vm->bscale = 1.f;
|
2021-03-01 19:48:03 +00:00
|
|
|
vm->piv.x = float(pivot.X);
|
|
|
|
vm->piv.y = float(pivot.Y);
|
|
|
|
vm->piv.z = float(pivot.Z);
|
|
|
|
vm->siz.x = voxel->Mips[0].SizeX;
|
|
|
|
vm->siz.y = voxel->Mips[0].SizeY;
|
|
|
|
vm->siz.z = voxel->Mips[0].SizeZ;
|
|
|
|
vm->is8bit = true;
|
|
|
|
voxel->Mips[0].Pivot.Zero(); // Needs to be taken out of the voxel data because it gets baked into the vertex buffer which we cannot use here.
|
|
|
|
vm->model = new FVoxelModel(voxel, true);
|
|
|
|
return vm;
|
2014-09-30 04:18:43 +00:00
|
|
|
}
|
2021-03-01 19:48:03 +00:00
|
|
|
return nullptr;
|
2019-09-19 20:02:45 +00:00
|
|
|
}
|
2019-09-16 19:08:42 +00:00
|
|
|
|
2014-09-30 04:18:43 +00:00
|
|
|
//Draw voxel model as perfect cubes
|
2019-09-16 19:08:42 +00:00
|
|
|
// Note: This is a hopeless mess that totally forfeits any chance of using a vertex buffer with its messy coordinate adjustments. :(
|
2020-09-20 18:39:11 +00:00
|
|
|
int32_t polymost_voxdraw(voxmodel_t* m, tspriteptr_t const tspr)
|
2014-09-30 04:18:43 +00:00
|
|
|
{
|
2019-07-13 03:20:17 +00:00
|
|
|
float f, g, k0, zoff;
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2014-12-26 17:29:48 +00:00
|
|
|
if ((intptr_t)m == (intptr_t)(-1)) // hackhackhack
|
|
|
|
return 0;
|
|
|
|
|
2020-09-20 18:39:11 +00:00
|
|
|
if ((tspr->cstat & 48) == 32)
|
2014-09-30 04:18:43 +00:00
|
|
|
return 0;
|
|
|
|
|
2014-12-26 17:29:48 +00:00
|
|
|
vec3f_t m0 = { m->scale, m->scale, m->scale };
|
2019-06-25 11:30:27 +00:00
|
|
|
vec3f_t a0 = { 0, 0, m->zadd*m->scale };
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2014-12-06 21:05:36 +00:00
|
|
|
k0 = m->bscale / 64.f;
|
2014-09-30 04:18:43 +00:00
|
|
|
f = (float) tspr->xrepeat * (256.f/320.f) * k0;
|
|
|
|
if ((sprite[tspr->owner].cstat&48)==16)
|
2019-06-25 11:30:27 +00:00
|
|
|
{
|
2014-09-30 04:18:43 +00:00
|
|
|
f *= 1.25f;
|
2020-11-14 09:00:37 +00:00
|
|
|
a0.y -= tspr->xoffset * bcosf(spriteext[tspr->owner].angoff, -20);
|
|
|
|
a0.x += tspr->xoffset * bsinf(spriteext[tspr->owner].angoff, -20);
|
2019-06-25 11:30:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (globalorientation&8) { m0.z = -m0.z; a0.z = -a0.z; } //y-flipping
|
|
|
|
if (globalorientation&4) { m0.x = -m0.x; a0.x = -a0.x; a0.y = -a0.y; } //x-flipping
|
2014-09-30 04:18:43 +00:00
|
|
|
|
|
|
|
m0.x *= f; a0.x *= f; f = -f;
|
|
|
|
m0.y *= f; a0.y *= f;
|
|
|
|
f = (float) tspr->yrepeat * k0;
|
|
|
|
m0.z *= f; a0.z *= f;
|
|
|
|
|
2019-12-31 19:35:28 +00:00
|
|
|
k0 = (float) (tspr->z+spriteext[tspr->owner].position_offset.z);
|
2019-07-09 05:42:15 +00:00
|
|
|
f = ((globalorientation&8) && (sprite[tspr->owner].cstat&48)!=0) ? -4.f : 4.f;
|
2019-06-25 11:30:27 +00:00
|
|
|
k0 -= (tspr->yoffset*tspr->yrepeat)*f*m->bscale;
|
2019-07-13 03:20:17 +00:00
|
|
|
zoff = m->siz.z*.5f;
|
|
|
|
if (!(tspr->cstat&128))
|
|
|
|
zoff += m->piv.z;
|
|
|
|
else if ((tspr->cstat&48) != 48)
|
|
|
|
{
|
|
|
|
zoff += m->piv.z;
|
|
|
|
zoff -= m->siz.z*.5f;
|
|
|
|
}
|
2020-03-12 00:58:45 +00:00
|
|
|
if (globalorientation&8) zoff = m->siz.z-zoff;
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2014-12-26 17:29:48 +00:00
|
|
|
f = (65536.f*512.f) / ((float)xdimen*viewingrange);
|
|
|
|
g = 32.f / ((float)xdimen*gxyaspect);
|
|
|
|
|
2019-12-26 06:28:03 +00:00
|
|
|
int const shadowHack = !!(tspr->clipdist & TSPR_FLAGS_MDHACK);
|
2019-07-06 19:04:54 +00:00
|
|
|
|
2019-12-31 19:35:28 +00:00
|
|
|
m0.y *= f; a0.y = (((float)(tspr->x+spriteext[tspr->owner].position_offset.x-globalposx)) * (1.f/1024.f) + a0.y) * f;
|
|
|
|
m0.x *=-f; a0.x = (((float)(tspr->y+spriteext[tspr->owner].position_offset.y-globalposy)) * -(1.f/1024.f) + a0.x) * -f;
|
2019-07-06 19:04:54 +00:00
|
|
|
m0.z *= g; a0.z = (((float)(k0 -globalposz - shadowHack)) * -(1.f/16384.f) + a0.z) * g;
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2014-12-26 17:29:48 +00:00
|
|
|
float mat[16];
|
2019-09-30 06:35:33 +00:00
|
|
|
md3_vox_calcmat_common(tspr, &a0, f, mat);
|
2014-09-30 04:18:43 +00:00
|
|
|
|
|
|
|
//Mirrors
|
2014-12-26 17:29:48 +00:00
|
|
|
if (grhalfxdown10x < 0)
|
|
|
|
{
|
|
|
|
mat[0] = -mat[0];
|
|
|
|
mat[4] = -mat[4];
|
|
|
|
mat[8] = -mat[8];
|
|
|
|
mat[12] = -mat[12];
|
|
|
|
}
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2019-07-06 19:04:54 +00:00
|
|
|
if (shadowHack)
|
2014-09-30 04:18:43 +00:00
|
|
|
{
|
2020-06-11 16:40:53 +00:00
|
|
|
GLInterface.SetDepthFunc(DF_LEqual);
|
2019-10-04 16:44:16 +00:00
|
|
|
}
|
2014-12-26 17:29:48 +00:00
|
|
|
|
2015-02-11 05:22:48 +00:00
|
|
|
|
2019-10-04 19:13:04 +00:00
|
|
|
int winding = ((grhalfxdown10x >= 0) ^ ((globalorientation & 8) != 0) ^ ((globalorientation & 4) != 0)) ? Winding_CW : Winding_CCW;
|
|
|
|
GLInterface.SetCull(Cull_Back, winding);
|
2014-09-30 04:18:43 +00:00
|
|
|
|
2014-12-26 17:29:48 +00:00
|
|
|
float pc[4];
|
|
|
|
|
2020-01-22 20:34:18 +00:00
|
|
|
pc[0] = pc[1] = pc[2] = 1.f;
|
2019-10-17 18:29:58 +00:00
|
|
|
|
2020-01-11 21:39:53 +00:00
|
|
|
|
2019-07-06 19:04:54 +00:00
|
|
|
if (!shadowHack)
|
|
|
|
{
|
|
|
|
pc[3] = (tspr->cstat & 2) ? glblend[tspr->blend].def[!!(tspr->cstat & 512)].alpha : 1.0f;
|
|
|
|
pc[3] *= 1.0f - spriteext[tspr->owner].alpha;
|
2014-12-26 17:29:48 +00:00
|
|
|
|
2020-05-23 12:40:54 +00:00
|
|
|
SetRenderStyleFromBlend(!!(tspr->cstat & 2), tspr->blend, !!(tspr->cstat & 512));
|
2016-10-09 07:55:23 +00:00
|
|
|
|
2019-07-06 19:04:54 +00:00
|
|
|
if (!(tspr->cstat & 2) || spriteext[tspr->owner].alpha > 0.f || pc[3] < 1.0f)
|
2019-10-04 16:12:03 +00:00
|
|
|
GLInterface.EnableBlend(true); // else GLInterface.EnableBlend(false);
|
2019-07-06 19:04:54 +00:00
|
|
|
}
|
|
|
|
else pc[3] = 1.f;
|
2019-12-19 16:51:33 +00:00
|
|
|
GLInterface.SetShade(std::max(0, globalshade), numshades);
|
2014-09-30 04:18:43 +00:00
|
|
|
//------------
|
|
|
|
|
|
|
|
//transform to Build coords
|
2014-12-26 17:29:48 +00:00
|
|
|
float omat[16];
|
2020-09-08 16:48:18 +00:00
|
|
|
memcpy(omat, mat, sizeof(omat));
|
2014-12-26 17:29:48 +00:00
|
|
|
|
2014-09-30 04:18:43 +00:00
|
|
|
f = 1.f/64.f;
|
|
|
|
g = m0.x*f; mat[0] *= g; mat[1] *= g; mat[2] *= g;
|
|
|
|
g = m0.y*f; mat[4] = omat[8]*g; mat[5] = omat[9]*g; mat[6] = omat[10]*g;
|
|
|
|
g =-m0.z*f; mat[8] = omat[4]*g; mat[9] = omat[5]*g; mat[10] = omat[6]*g;
|
2014-12-26 17:29:48 +00:00
|
|
|
//
|
2019-07-13 03:20:17 +00:00
|
|
|
mat[12] -= (m->piv.x*mat[0] + m->piv.y*mat[4] + zoff*mat[8]);
|
|
|
|
mat[13] -= (m->piv.x*mat[1] + m->piv.y*mat[5] + zoff*mat[9]);
|
|
|
|
mat[14] -= (m->piv.x*mat[2] + m->piv.y*mat[6] + zoff*mat[10]);
|
2014-12-26 17:29:48 +00:00
|
|
|
//
|
2019-10-04 16:12:03 +00:00
|
|
|
//Let OpenGL (and perhaps hardware :) handle the matrix rotation
|
2014-09-30 04:18:43 +00:00
|
|
|
mat[3] = mat[7] = mat[11] = 0.f; mat[15] = 1.f;
|
|
|
|
|
2020-06-01 07:55:28 +00:00
|
|
|
for (int i = 0; i < 15; i++) mat[i] *= 1024.f;
|
|
|
|
|
2021-03-01 19:48:03 +00:00
|
|
|
// Adjust to backend coordinate system being used by the vertex buffer.
|
|
|
|
for (int i = 4; i < 8; i++)
|
2020-02-09 19:14:26 +00:00
|
|
|
{
|
2021-03-01 19:48:03 +00:00
|
|
|
float f = mat[i];
|
|
|
|
mat[i] = -mat[i + 4];
|
|
|
|
mat[i + 4] = -f;
|
2020-02-09 19:14:26 +00:00
|
|
|
}
|
2019-09-18 18:44:21 +00:00
|
|
|
|
2021-03-01 19:48:03 +00:00
|
|
|
GLInterface.SetMatrix(Matrix_Model, mat);
|
|
|
|
|
|
|
|
int palId = TRANSLATION(Translation_Remap + curbasepal, globalpal);
|
2020-02-10 21:05:53 +00:00
|
|
|
GLInterface.SetPalswap(globalpal);
|
2020-09-20 18:39:11 +00:00
|
|
|
GLInterface.SetFade(sector[tspr->sectnum].floorpal);
|
2014-12-26 17:29:48 +00:00
|
|
|
|
2021-03-01 19:48:03 +00:00
|
|
|
auto tex = TexMan.GetGameTexture(m->model->GetPaletteTexture());
|
|
|
|
GLInterface.SetTexture(tex, TRANSLATION(Translation_Remap + curbasepal, globalpal), CLAMP_NOFILTER_XY, true);
|
|
|
|
GLInterface.SetModel(m->model, 0, 0, 0);
|
|
|
|
GLInterface.Draw(DT_Triangles, 0, 0);
|
|
|
|
GLInterface.SetModel(nullptr, 0, 0, 0);
|
2019-10-04 16:12:03 +00:00
|
|
|
GLInterface.SetCull(Cull_None);
|
|
|
|
|
2019-07-06 19:04:54 +00:00
|
|
|
if (shadowHack)
|
2014-09-30 04:18:43 +00:00
|
|
|
{
|
2020-06-11 16:40:53 +00:00
|
|
|
GLInterface.SetDepthFunc(DF_Less);
|
2019-10-04 16:44:16 +00:00
|
|
|
}
|
2020-06-04 19:34:27 +00:00
|
|
|
GLInterface.SetIdentityMatrix(Matrix_Model);
|
2020-01-18 15:13:39 +00:00
|
|
|
return 1;
|
2014-09-30 04:18:43 +00:00
|
|
|
}
|
2021-03-15 18:39:58 +00:00
|
|
|
|
|
|
|
extern char* voxfilenames[MAXVOXELS];
|
|
|
|
void (*PolymostProcessVoxels_Callback)(void) = NULL;
|
|
|
|
void PolymostProcessVoxels(void)
|
|
|
|
{
|
|
|
|
if (PolymostProcessVoxels_Callback)
|
|
|
|
PolymostProcessVoxels_Callback();
|
|
|
|
|
|
|
|
if (g_haveVoxels != 1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_haveVoxels = 2;
|
|
|
|
|
|
|
|
Printf(PRINT_NONOTIFY, "Generating voxel models for Polymost. This may take a while...\n");
|
|
|
|
|
|
|
|
for (bssize_t i = 0; i < MAXVOXELS; i++)
|
|
|
|
{
|
|
|
|
if (voxfilenames[i])
|
|
|
|
{
|
|
|
|
int lumpnum = fileSystem.FindFile(voxfilenames[i]);
|
|
|
|
if (lumpnum >= 0)
|
|
|
|
{
|
|
|
|
voxmodels[i] = voxload(lumpnum);
|
|
|
|
voxmodels[i]->scale = voxscale[i] * (1.f / 65536.f);
|
|
|
|
}
|
|
|
|
DO_FREE_AND_NULL(voxfilenames[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-30 04:18:43 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//---------------------------------------- VOX LIBRARY ENDS ----------------------------------------
|