2007-06-21 22:00:00 +02:00
|
|
|
|
//=======================================================================
|
|
|
|
|
// Copyright XashXT Group 2007 <20>
|
|
|
|
|
// r_sprite.c - render sprite models
|
|
|
|
|
//=======================================================================
|
|
|
|
|
|
|
|
|
|
#include "gl_local.h"
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
=============================================================
|
|
|
|
|
|
|
|
|
|
SPRITE MODELS
|
|
|
|
|
|
|
|
|
|
=============================================================
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
====================
|
|
|
|
|
Sprite model loader
|
|
|
|
|
====================
|
|
|
|
|
*/
|
|
|
|
|
void *R_SpriteLoadFrame (model_t *mod, void *pin, mspriteframe_t **ppframe, int framenum, byte *pal )
|
|
|
|
|
{
|
|
|
|
|
dspriteframe_t *pinframe;
|
|
|
|
|
mspriteframe_t *pspriteframe;
|
|
|
|
|
int width, height, size, origin[2];
|
|
|
|
|
char name[64];
|
2007-06-29 22:00:00 +02:00
|
|
|
|
rgbdata_t r_frame;
|
|
|
|
|
image_t *image;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
pinframe = (dspriteframe_t *)pin;
|
|
|
|
|
|
|
|
|
|
width = LittleLong (pinframe->width);
|
|
|
|
|
height = LittleLong (pinframe->height);
|
|
|
|
|
size = width * height;
|
|
|
|
|
|
|
|
|
|
pspriteframe = Mem_Alloc(mod->mempool, sizeof (mspriteframe_t));
|
|
|
|
|
memset (pspriteframe, 0, sizeof (mspriteframe_t));
|
|
|
|
|
|
|
|
|
|
*ppframe = pspriteframe;
|
|
|
|
|
|
2007-06-29 22:00:00 +02:00
|
|
|
|
pspriteframe->width = r_frame.width = width;
|
|
|
|
|
pspriteframe->height = r_frame.height = height;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
origin[0] = LittleLong (pinframe->origin[0]);
|
|
|
|
|
origin[1] = LittleLong (pinframe->origin[1]);
|
|
|
|
|
|
|
|
|
|
pspriteframe->up = origin[1];
|
|
|
|
|
pspriteframe->down = origin[1] - height;
|
|
|
|
|
pspriteframe->left = origin[0];
|
|
|
|
|
pspriteframe->right = width + origin[0];
|
|
|
|
|
pspriteframe->texnum = 0;
|
2007-06-29 22:00:00 +02:00
|
|
|
|
r_frame.compression = 0;
|
|
|
|
|
r_frame.bitsperpixel = 32;
|
|
|
|
|
r_frame.alpha = true;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
//extract sprite name from path
|
|
|
|
|
ri.FS_FileBase( mod->name, name );
|
|
|
|
|
strcat(name, va("_%i", framenum));
|
2007-06-29 22:00:00 +02:00
|
|
|
|
r_frame.palette = pal;
|
|
|
|
|
r_frame.buffer = (byte *)(pinframe + 1);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-29 22:00:00 +02:00
|
|
|
|
image = R_LoadImage( name, &r_frame, it_sprite );
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
if(image)
|
|
|
|
|
{
|
|
|
|
|
pspriteframe->texnum = image->texnum;
|
|
|
|
|
mod->skins[framenum] = image;
|
|
|
|
|
}
|
|
|
|
|
else Msg("Warning: %s has null frame %d\n", image->name, framenum );
|
|
|
|
|
|
|
|
|
|
return (void *)((byte *)(pinframe+1) + size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void R_SpriteLoadModel( model_t *mod, void *buffer )
|
|
|
|
|
{
|
2007-06-26 22:00:00 +02:00
|
|
|
|
int i, size, version, numframes;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
dsprite_t *pin;
|
2007-06-26 22:00:00 +02:00
|
|
|
|
short *numi;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
msprite_t *psprite;
|
2007-06-27 22:00:00 +02:00
|
|
|
|
frametype_t *pframetype;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
byte pal[256][4];
|
2007-06-26 22:00:00 +02:00
|
|
|
|
vec4_t rgbacolor;
|
|
|
|
|
float framerate;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
pin = (dsprite_t *)buffer;
|
|
|
|
|
version = LittleLong (pin->version);
|
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
switch( version )
|
|
|
|
|
{
|
|
|
|
|
case SPRITE_VERSION_HALF:
|
|
|
|
|
ResetRGBA( rgbacolor );
|
|
|
|
|
framerate = 0.0f;
|
|
|
|
|
break;
|
|
|
|
|
case SPRITE_VERSION_XASH:
|
|
|
|
|
//unpack rgba
|
|
|
|
|
rgbacolor[0] = LittleLong((pin->rgbacolor & 0x000000FF) >> 0);
|
|
|
|
|
rgbacolor[1] = LittleLong((pin->rgbacolor & 0x0000FF00) >> 8);
|
|
|
|
|
rgbacolor[2] = LittleLong((pin->rgbacolor & 0x00FF0000) >> 16);
|
|
|
|
|
rgbacolor[3] = LittleLong((pin->rgbacolor & 0xFF000000) >> 24);
|
|
|
|
|
|
|
|
|
|
TransformRGBA(rgbacolor, rgbacolor );//convert into float
|
|
|
|
|
framerate = LittleFloat (pin->framerate);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Msg("Warning: %s has wrong version number (%i should be %i or %i)", mod->name, version, SPRITE_VERSION_HALF, SPRITE_VERSION_XASH );
|
|
|
|
|
return;
|
|
|
|
|
}
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
numframes = LittleLong (pin->numframes);
|
|
|
|
|
size = sizeof (msprite_t) + (numframes - 1) * sizeof (psprite->frames);
|
|
|
|
|
|
|
|
|
|
psprite = Mem_Alloc(mod->mempool, size );
|
|
|
|
|
mod->extradata = psprite; //make link to extradata
|
|
|
|
|
|
|
|
|
|
psprite->type = LittleLong(pin->type);
|
|
|
|
|
psprite->maxwidth = LittleLong (pin->width);
|
|
|
|
|
psprite->maxheight = LittleLong (pin->height);
|
2007-06-26 22:00:00 +02:00
|
|
|
|
psprite->rendermode = LittleLong (pin->texFormat);
|
|
|
|
|
psprite->framerate = framerate;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
psprite->numframes = numframes;
|
2007-06-26 22:00:00 +02:00
|
|
|
|
memcpy(psprite->rgba, rgbacolor, sizeof(psprite->rgba));
|
|
|
|
|
numi = (short *)(pin + 1);
|
|
|
|
|
|
|
|
|
|
mod->mins[0] = mod->mins[1] = -psprite->maxwidth / 2;
|
|
|
|
|
mod->maxs[0] = mod->maxs[1] = psprite->maxwidth / 2;
|
|
|
|
|
mod->mins[2] = -psprite->maxheight / 2;
|
|
|
|
|
mod->maxs[2] = psprite->maxheight / 2;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
if (LittleShort(*numi) == 256)
|
2007-06-21 22:00:00 +02:00
|
|
|
|
{
|
2007-06-26 22:00:00 +02:00
|
|
|
|
byte *src = (byte *)(numi+1);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
switch( psprite->rendermode )
|
2007-06-21 22:00:00 +02:00
|
|
|
|
{
|
|
|
|
|
case SPR_NORMAL:
|
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
|
{
|
|
|
|
|
pal[i][0] = *src++;
|
|
|
|
|
pal[i][1] = *src++;
|
|
|
|
|
pal[i][2] = *src++;
|
|
|
|
|
pal[i][3] = 0;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2007-06-26 22:00:00 +02:00
|
|
|
|
case SPR_ADDGLOW:
|
2007-06-21 22:00:00 +02:00
|
|
|
|
case SPR_ADDITIVE:
|
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
|
{
|
|
|
|
|
pal[i][0] = *src++;
|
|
|
|
|
pal[i][1] = *src++;
|
|
|
|
|
pal[i][2] = *src++;
|
|
|
|
|
pal[i][3] = 255;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case SPR_INDEXALPHA:
|
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
|
{
|
|
|
|
|
pal[i][0] = *src++;
|
|
|
|
|
pal[i][1] = *src++;
|
|
|
|
|
pal[i][2] = *src++;
|
|
|
|
|
pal[i][3] = i;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case SPR_ALPHTEST:
|
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
|
{
|
|
|
|
|
pal[i][0] = *src++;
|
|
|
|
|
pal[i][1] = *src++;
|
|
|
|
|
pal[i][2] = *src++;
|
|
|
|
|
pal[i][3] = 255;
|
|
|
|
|
}
|
|
|
|
|
pal[255][0] = pal[255][1] = pal[255][2] = pal[255][3] = 0;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2007-06-26 22:00:00 +02:00
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
|
{
|
|
|
|
|
pal[i][0] = *src++;
|
|
|
|
|
pal[i][1] = *src++;
|
|
|
|
|
pal[i][2] = *src++;
|
|
|
|
|
pal[i][3] = 0;
|
|
|
|
|
}
|
|
|
|
|
Msg("Warning: %s has unknown texFormat (%i, should be in range 0-4 )\n", mod->name, psprite->rendermode );
|
|
|
|
|
break;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
}
|
2007-06-27 22:00:00 +02:00
|
|
|
|
pframetype = (frametype_t *)(src);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
}
|
2007-06-26 22:00:00 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Msg("Warning: %s has wrong number of palette colors %i (should be 256)\n", mod->name, numi);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if(numframes < 1)
|
|
|
|
|
{
|
|
|
|
|
Msg("Warning: %s has invalid # of frames: %d\n", mod->name, numframes );
|
|
|
|
|
return;
|
|
|
|
|
}
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
mod->numframes = mod->numtexinfo = numframes;
|
|
|
|
|
mod->registration_sequence = registration_sequence;
|
|
|
|
|
|
2007-06-21 22:00:00 +02:00
|
|
|
|
for (i = 0; i < numframes; i++ )
|
|
|
|
|
{
|
2007-06-27 22:00:00 +02:00
|
|
|
|
int frametype;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
frametype = LittleLong (pframetype->type);
|
2007-06-27 22:00:00 +02:00
|
|
|
|
psprite->frames[i].frametype = frametype;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-27 22:00:00 +02:00
|
|
|
|
if(frametype == 0)//SPR_SINGLE
|
2007-06-21 22:00:00 +02:00
|
|
|
|
{
|
2007-06-27 22:00:00 +02:00
|
|
|
|
pframetype = (frametype_t *)R_SpriteLoadFrame(mod, pframetype + 1, &psprite->frames[i].frameptr, i, (byte *)(&pal[0][0]));
|
2007-06-21 22:00:00 +02:00
|
|
|
|
}
|
2007-06-27 22:00:00 +02:00
|
|
|
|
else ri.Sys_Error(ERR_DROP, "R_SpriteLoadModel: group frames are not supported\n");
|
2007-06-21 22:00:00 +02:00
|
|
|
|
if(pframetype == NULL) break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
================
|
|
|
|
|
R_GetSpriteFrame
|
|
|
|
|
================
|
|
|
|
|
*/
|
|
|
|
|
mspriteframe_t *R_GetSpriteFrame (entity_t *currententity)
|
|
|
|
|
{
|
|
|
|
|
msprite_t *psprite;
|
|
|
|
|
mspriteframe_t *pspriteframe;
|
2007-06-27 22:00:00 +02:00
|
|
|
|
int frame;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
psprite = currententity->model->extradata;
|
|
|
|
|
frame = currententity->frame;
|
|
|
|
|
|
|
|
|
|
if ((frame >= psprite->numframes) || (frame < 0))
|
|
|
|
|
{
|
2007-06-27 22:00:00 +02:00
|
|
|
|
Com_Printf ("R_GetSpriteFrame: no such frame %d (%s)\n", frame, currententity->model->name);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
frame = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2007-06-27 22:00:00 +02:00
|
|
|
|
if (psprite->frames[frame].frametype == 0) //SPR_SINGLE
|
2007-06-21 22:00:00 +02:00
|
|
|
|
{
|
|
|
|
|
pspriteframe = psprite->frames[frame].frameptr;
|
|
|
|
|
}
|
2007-06-27 22:00:00 +02:00
|
|
|
|
else ri.Sys_Error(ERR_DROP, "R_GetSpriteFrame: group frames are not supported\n");
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
return pspriteframe;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2007-06-22 22:00:00 +02:00
|
|
|
|
void R_SpriteDrawModel( int passnum )
|
2007-06-21 22:00:00 +02:00
|
|
|
|
{
|
|
|
|
|
float alpha = 1.0F;
|
|
|
|
|
mspriteframe_t *frame;
|
|
|
|
|
vec3_t point, forward, right, up;
|
|
|
|
|
msprite_t *psprite;
|
2007-06-22 22:00:00 +02:00
|
|
|
|
entity_t *e = currententity;
|
2007-06-26 22:00:00 +02:00
|
|
|
|
model_t *mod = currentmodel;
|
|
|
|
|
float realtime = r_newrefdef.time / 1000.000f;
|
2007-06-22 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
if ( (e->flags & RF_TRANSLUCENT) && (passnum == RENDERPASS_SOLID)) return;// solid
|
|
|
|
|
if (!(e->flags & RF_TRANSLUCENT) && (passnum == RENDERPASS_ALPHA)) return;// solid
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
|
|
|
|
// don't even bother culling, because it's just a single
|
|
|
|
|
// polygon without a surface cache
|
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
psprite = (msprite_t *)mod->extradata;
|
|
|
|
|
|
|
|
|
|
//xash auto-animate
|
|
|
|
|
if(psprite->framerate > 0 && psprite->numframes > 1)
|
|
|
|
|
{
|
|
|
|
|
float frametime;
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
mod->prevanimtime = mod->animtime;
|
|
|
|
|
mod->animtime = realtime;
|
|
|
|
|
frametime = mod->animtime - mod->prevanimtime;
|
|
|
|
|
if(frametime == 0 ) return;//first call
|
|
|
|
|
|
|
|
|
|
mod->frame += psprite->framerate * frametime;
|
|
|
|
|
while (mod->frame > psprite->numframes) mod->frame -= psprite->numframes;
|
|
|
|
|
while (mod->frame < 0) mod->frame += psprite->numframes;
|
|
|
|
|
e->frame = mod->frame;//set properly frame
|
|
|
|
|
}
|
|
|
|
|
else e->frame = fmod(e->frame, psprite->numframes);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
frame = R_GetSpriteFrame(e);
|
|
|
|
|
|
|
|
|
|
switch( psprite->type )
|
|
|
|
|
{
|
|
|
|
|
case SPR_ORIENTED:
|
|
|
|
|
AngleVectors (e->angles, forward, right, up);
|
2007-06-26 22:00:00 +02:00
|
|
|
|
VectorScale(forward, 0.01, forward );//offset for decals
|
|
|
|
|
VectorSubtract(e->origin, forward, e->origin );
|
2007-06-21 22:00:00 +02:00
|
|
|
|
break;
|
|
|
|
|
case SPR_FACING_UPRIGHT:
|
|
|
|
|
up[0] = up[1] = 0;
|
|
|
|
|
up[2] = 1;
|
|
|
|
|
right[0] = e->origin[1] - r_origin[1];
|
|
|
|
|
right[1] = -(e->origin[0] - r_origin[0]);
|
|
|
|
|
right[2] = 0;
|
|
|
|
|
VectorNormalize (right);
|
|
|
|
|
VectorCopy (vforward, forward);
|
|
|
|
|
break;
|
|
|
|
|
case SPR_VP_PARALLEL_UPRIGHT:
|
|
|
|
|
up[0] = up[1] = 0;
|
|
|
|
|
up[2] = 1;
|
|
|
|
|
VectorCopy (vup, right);
|
|
|
|
|
VectorCopy (vforward, forward);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
// normal sprite
|
|
|
|
|
VectorCopy (vup, up);
|
|
|
|
|
VectorCopy (vright, right);
|
|
|
|
|
VectorCopy (vforward, forward);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( e->flags & RF_TRANSLUCENT ) alpha = e->alpha;
|
2007-06-26 22:00:00 +02:00
|
|
|
|
else alpha = psprite->rgba[3];
|
|
|
|
|
|
|
|
|
|
if ( e->scale == 0 ) e->scale = 1.0f;
|
|
|
|
|
if ( alpha != 1.0f ) qglEnable( GL_BLEND );
|
|
|
|
|
|
|
|
|
|
qglColor4f( psprite->rgba[0], psprite->rgba[1], psprite->rgba[2], alpha );
|
|
|
|
|
if(psprite->rendermode == SPR_ADDGLOW) qglDisable(GL_DEPTH_TEST);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
GL_Bind(currentmodel->skins[(int)e->frame]->texnum);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
GL_TexEnv( GL_MODULATE );
|
|
|
|
|
|
|
|
|
|
if ( alpha == 1.0 ) qglEnable (GL_ALPHA_TEST);
|
|
|
|
|
else qglDisable( GL_ALPHA_TEST );
|
2007-06-26 22:00:00 +02:00
|
|
|
|
|
2007-06-21 22:00:00 +02:00
|
|
|
|
qglDisable(GL_CULL_FACE);
|
|
|
|
|
qglBegin (GL_QUADS);
|
2007-06-26 22:00:00 +02:00
|
|
|
|
qglTexCoord2f (0, 0);
|
|
|
|
|
VectorMA (e->origin, frame->up * e->scale, up, point);
|
|
|
|
|
VectorMA (point, frame->left * e->scale, right, point);
|
|
|
|
|
qglVertex3fv (point);
|
|
|
|
|
|
|
|
|
|
qglTexCoord2f (1, 0);
|
|
|
|
|
VectorMA (e->origin, frame->up * e->scale, up, point);
|
|
|
|
|
VectorMA (point, frame->right * e->scale, right, point);
|
|
|
|
|
qglVertex3fv (point);
|
|
|
|
|
|
|
|
|
|
qglTexCoord2f (1, 1);
|
|
|
|
|
VectorMA (e->origin, frame->down * e->scale, up, point);
|
|
|
|
|
VectorMA (point, frame->right * e->scale, right, point);
|
|
|
|
|
qglVertex3fv (point);
|
|
|
|
|
|
|
|
|
|
qglTexCoord2f (0, 1);
|
|
|
|
|
VectorMA (e->origin, frame->down * e->scale, up, point);
|
|
|
|
|
VectorMA (point, frame->left * e->scale, right, point);
|
|
|
|
|
qglVertex3fv (point);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
qglEnd ();
|
|
|
|
|
|
|
|
|
|
qglDisable(GL_BLEND);
|
|
|
|
|
qglDisable (GL_ALPHA_TEST);
|
|
|
|
|
qglEnable(GL_DEPTH_TEST);
|
|
|
|
|
GL_TexEnv( GL_REPLACE );
|
|
|
|
|
|
2007-06-26 22:00:00 +02:00
|
|
|
|
if(psprite->rendermode == SPR_ADDGLOW) qglEnable(GL_DEPTH_TEST);
|
2007-06-21 22:00:00 +02:00
|
|
|
|
if ( alpha != 1.0F ) qglDisable( GL_BLEND );
|
|
|
|
|
qglColor4f( 1, 1, 1, 1 );
|
|
|
|
|
}
|
|
|
|
|
|