2020-05-01 09:28:49 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 1997, 2005 - 3D Realms Entertainment
|
|
|
|
|
|
|
|
This file is part of Shadow Warrior version 1.2
|
|
|
|
|
|
|
|
Shadow Warrior 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 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
|
|
|
Original Source: 1997 - Frank Maddin and Jim Norwood
|
|
|
|
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
|
2020-05-21 14:25:41 +00:00
|
|
|
#include "ns.h"
|
|
|
|
|
2020-05-01 09:28:49 +00:00
|
|
|
#include "compat.h"
|
|
|
|
#include "pragmas.h"
|
|
|
|
|
|
|
|
#include "game.h"
|
|
|
|
#include "interpso.h"
|
2020-05-21 18:42:48 +00:00
|
|
|
#include "names2.h"
|
2020-05-01 09:28:49 +00:00
|
|
|
|
2020-05-21 14:25:41 +00:00
|
|
|
BEGIN_SW_NS
|
|
|
|
|
2020-11-26 17:05:49 +00:00
|
|
|
#define SO_MAXINTERPOLATIONS 1024
|
2020-05-01 09:28:49 +00:00
|
|
|
|
|
|
|
static struct so_interp
|
|
|
|
{
|
|
|
|
struct interp_data
|
|
|
|
{
|
|
|
|
void *curipos;
|
|
|
|
int32_t oldipos;
|
|
|
|
int32_t bakipos;
|
|
|
|
int32_t lastipos;
|
|
|
|
int32_t lastoldipos;
|
|
|
|
int32_t lastangdiff;
|
|
|
|
int32_t spriteofang;
|
|
|
|
} data[SO_MAXINTERPOLATIONS];
|
|
|
|
|
|
|
|
int32_t numinterpolations;
|
|
|
|
int32_t tic, lasttic;
|
2020-09-09 18:32:24 +00:00
|
|
|
bool hasvator;
|
2020-05-01 09:28:49 +00:00
|
|
|
} so_interpdata[MAX_SECTOR_OBJECTS];
|
|
|
|
|
|
|
|
static void so_setpointinterpolation(so_interp *interp, int32_t *posptr)
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
if (interp->numinterpolations >= SO_MAXINTERPOLATIONS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < interp->numinterpolations; i++)
|
|
|
|
if (interp->data[i].curipos == posptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
so_interp::interp_data *data = &interp->data[interp->numinterpolations++];
|
|
|
|
|
|
|
|
data->curipos = posptr;
|
|
|
|
data->oldipos = *posptr;
|
|
|
|
data->lastipos = *posptr;
|
|
|
|
data->lastoldipos = *posptr;
|
|
|
|
data->spriteofang = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void so_setspriteanginterpolation(so_interp *interp, int16_t *posptr, int32_t spritenum)
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
if (interp->numinterpolations >= SO_MAXINTERPOLATIONS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < interp->numinterpolations; i++)
|
|
|
|
if (interp->data[i].curipos == posptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
so_interp::interp_data *data = &interp->data[interp->numinterpolations++];
|
|
|
|
|
|
|
|
data->curipos = posptr;
|
|
|
|
data->oldipos = *posptr;
|
|
|
|
data->lastipos = *posptr;
|
|
|
|
data->lastoldipos = *posptr;
|
|
|
|
data->lastangdiff = 0;
|
|
|
|
data->spriteofang = spritenum;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Covers points and angles altogether
|
|
|
|
static void so_stopdatainterpolation(so_interp *interp, void *posptr)
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < interp->numinterpolations; i++)
|
|
|
|
if (interp->data[i].curipos == posptr)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (i == interp->numinterpolations)
|
|
|
|
return;
|
|
|
|
|
|
|
|
interp->data[i] = interp->data[--(interp->numinterpolations)];
|
|
|
|
}
|
|
|
|
|
|
|
|
void so_addinterpolation(SECTOR_OBJECTp sop)
|
|
|
|
{
|
|
|
|
SECTORp *sectp;
|
|
|
|
int32_t startwall, endwall;
|
|
|
|
int32_t i;
|
|
|
|
|
|
|
|
so_interp *interp = &so_interpdata[sop - SectorObject];
|
|
|
|
interp->numinterpolations = 0;
|
2020-09-09 17:52:52 +00:00
|
|
|
interp->hasvator = false;
|
2020-05-01 09:28:49 +00:00
|
|
|
|
|
|
|
for (sectp = sop->sectp; *sectp; sectp++)
|
|
|
|
{
|
|
|
|
startwall = (*sectp)->wallptr;
|
|
|
|
endwall = startwall + (*sectp)->wallnum - 1;
|
|
|
|
|
|
|
|
for (i = startwall; i <= endwall; i++)
|
|
|
|
{
|
|
|
|
int32_t nextwall = wall[i].nextwall;
|
|
|
|
|
|
|
|
so_setpointinterpolation(interp, &wall[i].x);
|
|
|
|
so_setpointinterpolation(interp, &wall[i].y);
|
|
|
|
|
|
|
|
if (nextwall >= 0)
|
|
|
|
{
|
|
|
|
so_setpointinterpolation(interp, &wall[wall[nextwall].point2].x);
|
|
|
|
so_setpointinterpolation(interp, &wall[wall[nextwall].point2].y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-15 14:59:43 +00:00
|
|
|
int i;
|
|
|
|
SectIterator it(*sectp - sector);
|
|
|
|
while ((i = it.NextIndex()) >= 0)
|
2020-05-21 18:42:48 +00:00
|
|
|
if (sprite[i].statnum == STAT_VATOR && SP_TAG1(sprite+i) == SECT_VATOR)
|
|
|
|
break;
|
|
|
|
interp->hasvator |= (i >= 0);
|
2020-05-01 09:28:49 +00:00
|
|
|
}
|
|
|
|
|
2020-05-21 18:42:48 +00:00
|
|
|
if (!interp->hasvator)
|
|
|
|
for (sectp = sop->sectp; *sectp; sectp++)
|
|
|
|
{
|
|
|
|
so_setpointinterpolation(interp, &(*sectp)->ceilingz);
|
|
|
|
so_setpointinterpolation(interp, &(*sectp)->floorz);
|
|
|
|
}
|
|
|
|
|
2020-05-01 09:28:49 +00:00
|
|
|
// interpolate midpoint, for aiming at a remote controlled SO
|
|
|
|
so_setpointinterpolation(interp, &sop->xmid);
|
|
|
|
so_setpointinterpolation(interp, &sop->ymid);
|
|
|
|
so_setpointinterpolation(interp, &sop->zmid);
|
|
|
|
|
|
|
|
interp->tic = 0;
|
|
|
|
interp->lasttic = synctics;
|
|
|
|
}
|
|
|
|
|
|
|
|
void so_setspriteinterpolation(SECTOR_OBJECTp sop, spritetype *sp)
|
|
|
|
{
|
|
|
|
so_interp *interp = &so_interpdata[sop - SectorObject];
|
|
|
|
|
|
|
|
so_setpointinterpolation(interp, &sp->x);
|
|
|
|
so_setpointinterpolation(interp, &sp->y);
|
2020-05-21 18:42:48 +00:00
|
|
|
if (!interp->hasvator)
|
|
|
|
so_setpointinterpolation(interp, &sp->z);
|
2020-05-01 09:28:49 +00:00
|
|
|
so_setspriteanginterpolation(interp, &sp->ang, sp - sprite);
|
|
|
|
}
|
|
|
|
|
|
|
|
void so_stopspriteinterpolation(SECTOR_OBJECTp sop, spritetype *sp)
|
|
|
|
{
|
|
|
|
so_interp *interp = &so_interpdata[sop - SectorObject];
|
|
|
|
|
|
|
|
so_stopdatainterpolation(interp, &sp->x);
|
|
|
|
so_stopdatainterpolation(interp, &sp->y);
|
2020-05-21 18:42:48 +00:00
|
|
|
if (!interp->hasvator)
|
|
|
|
so_stopdatainterpolation(interp, &sp->z);
|
2020-05-01 09:28:49 +00:00
|
|
|
so_stopdatainterpolation(interp, &sp->ang);
|
|
|
|
}
|
|
|
|
|
|
|
|
void so_setinterpolationtics(SECTOR_OBJECTp sop, int16_t locktics)
|
|
|
|
{
|
|
|
|
so_interp *interp = &so_interpdata[sop - SectorObject];
|
|
|
|
|
|
|
|
interp->tic = 0;
|
|
|
|
interp->lasttic = locktics;
|
|
|
|
}
|
|
|
|
|
|
|
|
void so_updateinterpolations(void) // Stick at beginning of domovethings
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
SECTOR_OBJECTp sop;
|
|
|
|
so_interp *interp;
|
|
|
|
so_interp::interp_data *data;
|
2020-09-09 18:32:24 +00:00
|
|
|
bool interpolating = cl_sointerpolation && !CommEnabled; // If changing from menu
|
2020-05-01 09:28:49 +00:00
|
|
|
|
|
|
|
for (sop = SectorObject, interp = so_interpdata;
|
|
|
|
sop < &SectorObject[MAX_SECTOR_OBJECTS]; sop++, interp++)
|
|
|
|
{
|
2020-09-10 09:56:13 +00:00
|
|
|
bool skip = !cl_syncinput && (sop->track == SO_TURRET);
|
|
|
|
if (SO_EMPTY(sop) || skip)
|
2020-05-01 09:28:49 +00:00
|
|
|
continue;
|
|
|
|
if (interp->tic < interp->lasttic)
|
|
|
|
interp->tic += synctics;
|
|
|
|
for (i = 0, data = interp->data; i < interp->numinterpolations; i++, data++)
|
|
|
|
{
|
|
|
|
if (data->spriteofang >= 0)
|
|
|
|
{
|
|
|
|
USERp u = User[data->spriteofang];
|
|
|
|
if (u)
|
|
|
|
u->oangdiff = 0;
|
|
|
|
if (!interpolating)
|
|
|
|
data->lastangdiff = 0;
|
|
|
|
data->oldipos = *(int16_t *)(data->curipos);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
data->oldipos = *(int32_t *)(data->curipos);
|
|
|
|
|
|
|
|
if (!interpolating)
|
|
|
|
data->lastipos = data->lastoldipos = data->oldipos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// must call restore for every do interpolations
|
|
|
|
// make sure you don't exit
|
|
|
|
void so_dointerpolations(int32_t smoothratio) // Stick at beginning of drawscreen
|
|
|
|
{
|
|
|
|
int32_t i, delta;
|
|
|
|
SECTOR_OBJECTp sop;
|
|
|
|
so_interp *interp;
|
|
|
|
so_interp::interp_data *data;
|
|
|
|
|
|
|
|
// Set the bakipos values separately, in case a point is shared.
|
|
|
|
// Also set lastipos if there's been an actual change in a point.
|
|
|
|
for (sop = SectorObject, interp = so_interpdata;
|
|
|
|
sop < &SectorObject[MAX_SECTOR_OBJECTS]; sop++, interp++)
|
|
|
|
{
|
2020-09-10 09:56:13 +00:00
|
|
|
bool skip = !cl_syncinput && (sop->track == SO_TURRET);
|
|
|
|
if (SO_EMPTY(sop) || skip)
|
2020-05-01 09:28:49 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
for (i = 0; i < interp->numinterpolations; i++)
|
|
|
|
interp->data[i].bakipos = (interp->data[i].spriteofang >= 0) ?
|
|
|
|
*(int16_t *)(interp->data[i].curipos) :
|
|
|
|
*(int32_t *)(interp->data[i].curipos);
|
|
|
|
|
|
|
|
if (interp->tic == 0) // Only if the SO has just moved
|
|
|
|
{
|
|
|
|
for (i = 0, data = interp->data; i < interp->numinterpolations; i++, data++)
|
|
|
|
{
|
|
|
|
data->lastipos = data->bakipos;
|
|
|
|
data->lastoldipos = data->oldipos;
|
|
|
|
if (data->spriteofang >= 0)
|
|
|
|
{
|
|
|
|
USERp u = User[data->spriteofang];
|
|
|
|
data->lastangdiff = u ? u->oangdiff : 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (sop = SectorObject, interp = so_interpdata;
|
|
|
|
sop < &SectorObject[MAX_SECTOR_OBJECTS]; sop++, interp++)
|
|
|
|
{
|
2020-09-10 09:56:13 +00:00
|
|
|
bool skip = !cl_syncinput && (sop->track == SO_TURRET);
|
|
|
|
if (SO_EMPTY(sop) || skip)
|
2020-05-01 09:28:49 +00:00
|
|
|
continue;
|
|
|
|
|
2020-05-22 11:07:32 +00:00
|
|
|
// Check if interpolation has been explicitly disabled
|
|
|
|
if (interp->lasttic == 0)
|
|
|
|
continue;
|
|
|
|
|
2020-05-02 21:20:48 +00:00
|
|
|
// Unfortunately, interpolating over less samples doesn't work well
|
|
|
|
// in multiplayer. We also skip any sector object not
|
|
|
|
// remotely controlled by some player.
|
2020-05-01 15:10:48 +00:00
|
|
|
if (CommEnabled &&
|
|
|
|
((interp->lasttic != synctics) ||
|
2020-05-02 21:20:48 +00:00
|
|
|
!(sop->controller) ||
|
2020-05-01 15:10:48 +00:00
|
|
|
((Player[screenpeek].sop_control == sop) &&
|
|
|
|
!Player[screenpeek].sop_remote)))
|
2020-05-01 13:35:46 +00:00
|
|
|
continue;
|
|
|
|
|
2020-05-26 18:27:14 +00:00
|
|
|
int32_t ratio = smoothratio * synctics + 65536 * interp->tic;
|
|
|
|
ratio /= interp->lasttic;
|
|
|
|
ratio = (interp->tic == interp->lasttic) ? 65536 : ratio;
|
|
|
|
|
2020-05-01 09:28:49 +00:00
|
|
|
for (i = 0, data = interp->data; i < interp->numinterpolations; i++, data++)
|
|
|
|
{
|
2020-05-26 19:09:31 +00:00
|
|
|
// Hack for jittery coolies in level 1's train.
|
|
|
|
// Based in idea on code from draw.cpp:analyzesprites.
|
|
|
|
// TODO: It could be better. In particular, it could be better
|
|
|
|
// to conditionally disable the interpolation from analyzesprites
|
|
|
|
// instead, using TSPRITE info if possible.
|
|
|
|
if (((uintptr_t)(data->curipos) >= (uintptr_t)sprite) &&
|
|
|
|
((uintptr_t)(data->curipos) < (uintptr_t)(sprite + Numsprites)))
|
|
|
|
{
|
|
|
|
int32_t sprnum = ((char *)data->curipos - (char *)sprite) / sizeof(*sprite);
|
|
|
|
USERp u = User[sprnum];
|
|
|
|
if (u && (sprite[sprnum].statnum != STAT_DEFAULT) &&
|
|
|
|
((TEST(u->Flags, SPR_SKIP4) && (sprite[sprnum].statnum <= STAT_SKIP4_INTERP_END)) ||
|
|
|
|
(TEST(u->Flags, SPR_SKIP2) && (sprite[sprnum].statnum <= STAT_SKIP2_INTERP_END))))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-05-01 09:28:49 +00:00
|
|
|
if (data->spriteofang >= 0)
|
|
|
|
*(int16_t *)(data->curipos) = NORM_ANGLE(data->lastoldipos + mulscale16(data->lastangdiff, ratio));
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delta = data->lastipos - data->lastoldipos;
|
|
|
|
*(int32_t *)(data->curipos) = data->lastoldipos + mulscale16(delta, ratio);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void so_restoreinterpolations(void) // Stick at end of drawscreen
|
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
SECTOR_OBJECTp sop;
|
|
|
|
so_interp *interp;
|
|
|
|
so_interp::interp_data *data;
|
|
|
|
|
|
|
|
for (sop = SectorObject, interp = so_interpdata;
|
|
|
|
sop < &SectorObject[MAX_SECTOR_OBJECTS]; sop++, interp++)
|
|
|
|
{
|
2020-09-10 09:56:13 +00:00
|
|
|
bool skip = !cl_syncinput && (sop->track == SO_TURRET);
|
|
|
|
if (SO_EMPTY(sop) || skip)
|
2020-05-01 09:28:49 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
for (i = 0, data = interp->data; i < interp->numinterpolations; i++, data++)
|
|
|
|
if (data->spriteofang >= 0)
|
|
|
|
*(int16_t *)(data->curipos) = data->bakipos;
|
|
|
|
else
|
|
|
|
*(int32_t *)(data->curipos) = data->bakipos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int SaveSymDataInfo(MFILE_WRITE fil, void *ptr);
|
|
|
|
|
2020-09-09 18:32:24 +00:00
|
|
|
int so_writeinterpolations(MFILE_WRITE fil)
|
2020-05-01 09:28:49 +00:00
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
SECTOR_OBJECTp sop;
|
2020-05-21 14:25:41 +00:00
|
|
|
so_interp *interp;
|
2020-09-09 18:28:05 +00:00
|
|
|
int saveisshot = 0;
|
2020-05-01 09:28:49 +00:00
|
|
|
|
|
|
|
for (sop = SectorObject, interp = so_interpdata;
|
|
|
|
sop < &SectorObject[MAX_SECTOR_OBJECTS]; sop++, interp++)
|
|
|
|
{
|
2020-05-21 14:25:41 +00:00
|
|
|
so_interp::interp_data *data = interp->data;
|
2020-05-01 09:28:49 +00:00
|
|
|
MWRITE(&interp->numinterpolations,sizeof(interp->numinterpolations),1,fil);
|
2020-05-21 18:42:48 +00:00
|
|
|
MWRITE(&interp->hasvator,sizeof(interp->hasvator),1,fil);
|
2020-05-01 09:28:49 +00:00
|
|
|
for (i = 0; i < interp->numinterpolations; i++, data++)
|
|
|
|
{
|
|
|
|
saveisshot |= SaveSymDataInfo(fil, data->curipos);
|
|
|
|
MWRITE(&data->oldipos,sizeof(data->oldipos),1,fil);
|
|
|
|
MWRITE(&data->spriteofang,sizeof(data->spriteofang),1,fil);
|
|
|
|
}
|
|
|
|
}
|
2020-09-09 18:32:24 +00:00
|
|
|
return saveisshot;
|
2020-05-01 09:28:49 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 18:28:05 +00:00
|
|
|
int LoadSymDataInfo(MFILE_READ fil, void** ptr);
|
2020-05-01 09:28:49 +00:00
|
|
|
|
2020-09-09 18:32:24 +00:00
|
|
|
int so_readinterpolations(MFILE_READ fil)
|
2020-05-01 09:28:49 +00:00
|
|
|
{
|
|
|
|
int32_t i;
|
|
|
|
SECTOR_OBJECTp sop;
|
|
|
|
so_interp *interp;
|
2020-09-09 18:28:05 +00:00
|
|
|
int saveisshot = 0;
|
2020-05-01 09:28:49 +00:00
|
|
|
|
|
|
|
for (sop = SectorObject, interp = so_interpdata;
|
|
|
|
sop < &SectorObject[MAX_SECTOR_OBJECTS]; sop++, interp++)
|
|
|
|
{
|
|
|
|
so_interp::interp_data *data = interp->data;
|
|
|
|
MREAD(&interp->numinterpolations,sizeof(interp->numinterpolations),1,fil);
|
2020-05-21 18:42:48 +00:00
|
|
|
MREAD(&interp->hasvator,sizeof(interp->hasvator),1,fil);
|
2020-05-01 09:28:49 +00:00
|
|
|
for (i = 0; i < interp->numinterpolations; i++, data++)
|
|
|
|
{
|
|
|
|
saveisshot |= LoadSymDataInfo(fil, (void **)&data->curipos);
|
|
|
|
MREAD(&data->oldipos,sizeof(data->oldipos),1,fil);
|
|
|
|
MREAD(&data->spriteofang,sizeof(data->spriteofang),1,fil);
|
|
|
|
data->lastipos = data->lastoldipos = data->oldipos;
|
|
|
|
data->lastangdiff = 0;
|
|
|
|
}
|
|
|
|
interp->tic = 0;
|
|
|
|
interp->lasttic = synctics;
|
|
|
|
}
|
2020-09-09 18:32:24 +00:00
|
|
|
return saveisshot;
|
2020-05-01 09:28:49 +00:00
|
|
|
}
|
2020-05-21 14:25:41 +00:00
|
|
|
|
|
|
|
END_SW_NS
|