2020-07-04 21:40:54 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 1996, 2003 - 3D Realms Entertainment
|
|
|
|
|
|
|
|
This file is part of Duke Nukem 3D version 1.5 - Atomic Edition
|
|
|
|
|
|
|
|
Duke Nukem 3D 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
|
|
|
Original Source: 1996 - Todd Replogle
|
|
|
|
Prepared for public release: 03/21/2003 - Charlie Wiederhold, 3D Realms
|
|
|
|
Modifications for JonoF's port by Jonathon Fowler (jf@jonof.id.au)
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include "ns.h"
|
|
|
|
#include "duke3d.h"
|
2020-07-05 14:49:00 +00:00
|
|
|
#include "build.h"
|
|
|
|
#include "v_video.h"
|
2020-07-06 09:39:39 +00:00
|
|
|
#include "prediction.h"
|
2020-09-06 18:49:43 +00:00
|
|
|
#include "automap.h"
|
2020-11-04 06:01:25 +00:00
|
|
|
#include "dukeactor.h"
|
2020-11-26 15:03:40 +00:00
|
|
|
#include "interpolate.h"
|
2021-03-21 09:58:11 +00:00
|
|
|
#include "render.h"
|
2020-07-04 21:40:54 +00:00
|
|
|
|
2021-03-18 16:18:03 +00:00
|
|
|
// temporary hack to pass along RRRA's global fog. Needs to be done better later.
|
|
|
|
extern PalEntry GlobalMapFog;
|
|
|
|
extern float GlobalFogDensity;
|
|
|
|
|
2020-07-04 21:40:54 +00:00
|
|
|
BEGIN_DUKE_NS
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Floor Over Floor
|
|
|
|
|
|
|
|
// If standing in sector with SE42
|
|
|
|
// then draw viewing to SE41 and raise all =hi SE43 cielings.
|
|
|
|
|
|
|
|
// If standing in sector with SE43
|
|
|
|
// then draw viewing to SE40 and lower all =hi SE42 floors.
|
|
|
|
|
|
|
|
// If standing in sector with SE44
|
|
|
|
// then draw viewing to SE40.
|
|
|
|
|
|
|
|
// If standing in sector with SE45
|
|
|
|
// then draw viewing to SE41.
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-07-05 14:49:00 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2021-03-28 17:22:51 +00:00
|
|
|
void GameInterface::UpdateCameras(double smoothratio)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-12-05 19:04:31 +00:00
|
|
|
const int VIEWSCREEN_ACTIVE_DISTANCE = 1024;
|
2020-07-06 20:23:18 +00:00
|
|
|
|
2020-11-04 18:25:49 +00:00
|
|
|
if (camsprite == nullptr)
|
2020-07-05 14:49:00 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
auto p = &ps[screenpeek];
|
2020-11-04 18:25:49 +00:00
|
|
|
if (p->newOwner != nullptr) camsprite->SetOwner(p->newOwner);
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-09-11 18:46:21 +00:00
|
|
|
if (camsprite->GetOwner() && (p->GetActor()->spr.pos - camsprite->spr.pos).Length() < VIEWSCREEN_ACTIVE_DISTANCE)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-12-07 16:10:27 +00:00
|
|
|
auto tex = TexMan.FindGameTexture("VIEWSCR", ETextureType::Any);
|
|
|
|
if (!tex || !tex->GetTexture()->isCanvas()) return;
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-12-07 16:10:27 +00:00
|
|
|
auto canvas = static_cast<FCanvasTexture*>(tex->GetTexture());
|
2020-07-05 14:49:00 +00:00
|
|
|
|
|
|
|
screen->RenderTextureView(canvas, [=](IntRect& rect)
|
|
|
|
{
|
2021-12-22 22:27:32 +00:00
|
|
|
auto camera = camsprite->GetOwner();
|
2020-07-05 14:49:00 +00:00
|
|
|
display_mirror = 1; // should really be 'display external view'.
|
2022-08-03 13:50:27 +00:00
|
|
|
auto cstat = camera->spr.cstat;
|
|
|
|
camera->spr.cstat = CSTAT_SPRITE_INVISIBLE;
|
2022-12-05 05:09:43 +00:00
|
|
|
render_camtex(camera, camera->spr.pos, camera->sector(), DRotator(maphoriz(-camera->spr.shade), camera->interpolatedyaw(smoothratio), nullAngle), tex, rect, smoothratio);
|
2022-08-03 13:50:27 +00:00
|
|
|
camera->spr.cstat = cstat;
|
2020-07-05 14:49:00 +00:00
|
|
|
display_mirror = 0;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-22 22:27:32 +00:00
|
|
|
void GameInterface::EnterPortal(DCoreActor* viewer, int type)
|
2021-03-29 19:48:23 +00:00
|
|
|
{
|
|
|
|
if (type == PORTAL_WALL_MIRROR) display_mirror++;
|
|
|
|
}
|
|
|
|
|
2021-12-22 22:27:32 +00:00
|
|
|
void GameInterface::LeavePortal(DCoreActor* viewer, int type)
|
2021-03-29 19:48:23 +00:00
|
|
|
{
|
|
|
|
if (type == PORTAL_WALL_MIRROR) display_mirror--;
|
|
|
|
}
|
|
|
|
|
2021-11-18 18:33:32 +00:00
|
|
|
bool GameInterface::GetGeoEffect(GeoEffect* eff, sectortype* viewsector)
|
2021-04-02 20:52:46 +00:00
|
|
|
{
|
2021-11-18 18:33:32 +00:00
|
|
|
if (isRR() && viewsector->lotag == 848)
|
2021-04-02 20:52:46 +00:00
|
|
|
{
|
|
|
|
eff->geocnt = geocnt;
|
|
|
|
eff->geosector = geosector;
|
|
|
|
eff->geosectorwarp = geosectorwarp;
|
|
|
|
eff->geosectorwarp2 = geosectorwarp2;
|
|
|
|
eff->geox = geox;
|
|
|
|
eff->geoy = geoy;
|
|
|
|
eff->geox2 = geox2;
|
|
|
|
eff->geoy2 = geoy2;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-07-05 14:49:00 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
2020-07-05 19:21:39 +00:00
|
|
|
// RRRA's drug distortion effect
|
2020-07-05 14:49:00 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2020-11-05 06:25:44 +00:00
|
|
|
int DrugTimer;
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2020-11-06 09:52:43 +00:00
|
|
|
static int getdrugmode(player_struct *p, int oyrepeat)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2020-11-05 06:25:44 +00:00
|
|
|
int now = I_GetBuildTime() >> 1; // this function works on a 60 fps setup.
|
|
|
|
if (playrunning() && p->DrugMode > 0)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2020-11-05 06:25:44 +00:00
|
|
|
if (now - DrugTimer > 4 || now - DrugTimer < 0) DrugTimer = now - 1;
|
|
|
|
while (DrugTimer < now)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2020-11-05 06:25:44 +00:00
|
|
|
DrugTimer++;
|
2020-07-05 14:49:00 +00:00
|
|
|
int var_8c;
|
|
|
|
if (p->drug_stat[0] == 0)
|
|
|
|
{
|
|
|
|
p->drug_stat[1]++;
|
|
|
|
var_8c = oyrepeat + p->drug_stat[1] * 5000;
|
|
|
|
if (oyrepeat * 3 < var_8c)
|
|
|
|
{
|
|
|
|
p->drug_aspect = oyrepeat * 3;
|
|
|
|
p->drug_stat[0] = 2;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
p->drug_aspect = var_8c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (p->drug_stat[0] == 3)
|
|
|
|
{
|
|
|
|
p->drug_stat[1]--;
|
|
|
|
var_8c = oyrepeat + p->drug_stat[1] * 5000;
|
|
|
|
if (var_8c < oyrepeat)
|
|
|
|
{
|
|
|
|
p->DrugMode = 0;
|
|
|
|
p->drug_stat[0] = 0;
|
|
|
|
p->drug_stat[2] = 0;
|
|
|
|
p->drug_stat[1] = 0;
|
2020-11-06 09:52:43 +00:00
|
|
|
p->drug_aspect = oyrepeat;
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
p->drug_aspect = var_8c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (p->drug_stat[0] == 2)
|
|
|
|
{
|
|
|
|
if (p->drug_stat[2] > 30)
|
|
|
|
{
|
|
|
|
p->drug_stat[0] = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
p->drug_stat[2]++;
|
|
|
|
p->drug_aspect = oyrepeat * 3 + p->drug_stat[2] * 500;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (p->drug_stat[2] < 1)
|
|
|
|
{
|
|
|
|
p->drug_stat[0] = 2;
|
|
|
|
p->DrugMode--;
|
|
|
|
if (p->DrugMode == 1)
|
|
|
|
p->drug_stat[0] = 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
p->drug_stat[2]--;
|
|
|
|
p->drug_aspect = oyrepeat * 3 + p->drug_stat[2] * 500;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-11-06 09:52:43 +00:00
|
|
|
return p->drug_aspect;
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
2020-11-06 09:52:43 +00:00
|
|
|
else
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2020-11-06 09:52:43 +00:00
|
|
|
DrugTimer = now;
|
|
|
|
return oyrepeat;
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2022-09-08 11:16:27 +00:00
|
|
|
void displayrooms(int snum, double interpfrac, bool sceneonly)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-09-08 11:16:27 +00:00
|
|
|
DVector3 cpos;
|
2022-12-05 05:09:43 +00:00
|
|
|
DRotator cangles;
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-09-08 11:27:22 +00:00
|
|
|
player_struct* p = &ps[snum];
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-12-09 06:55:13 +00:00
|
|
|
// update render angles.
|
|
|
|
p->Angles.updateRenderAngles(interpfrac);
|
|
|
|
|
2021-11-21 07:45:07 +00:00
|
|
|
if (automapMode == am_full || !p->insector())
|
2020-07-05 14:49:00 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Do not light up the fog in RRRA's E2L1. Ideally this should apply to all foggy levels but all others use lookup table hacks for their fog.
|
|
|
|
if (isRRRA() && fogactive)
|
|
|
|
{
|
|
|
|
p->visibility = ud.const_visibility;
|
|
|
|
}
|
2022-01-08 11:23:05 +00:00
|
|
|
g_visibility = ud.const_visibility;
|
|
|
|
g_relvisibility = p->visibility - ud.const_visibility;
|
2021-03-18 16:18:03 +00:00
|
|
|
GlobalMapFog = fogactive ? 0x999999 : 0;
|
|
|
|
GlobalFogDensity = fogactive ? 350.f : 0.f;
|
2022-09-08 11:27:22 +00:00
|
|
|
|
2022-09-08 11:16:27 +00:00
|
|
|
DoInterpolations(interpfrac);
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2020-07-07 02:54:12 +00:00
|
|
|
setgamepalette(BASEPAL);
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-10-12 17:00:43 +00:00
|
|
|
float fov = (float)r_fov;
|
2022-09-08 11:27:22 +00:00
|
|
|
auto sect = p->cursector;
|
|
|
|
|
|
|
|
DDukeActor* viewer;
|
|
|
|
bool camview = false;
|
2022-08-03 17:38:39 +00:00
|
|
|
|
2020-11-02 22:53:55 +00:00
|
|
|
if (ud.cameraactor)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-09-08 11:27:22 +00:00
|
|
|
viewer = ud.cameraactor;
|
|
|
|
camview = true;
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-09-08 11:27:22 +00:00
|
|
|
if (viewer->spr.yint < 0) viewer->spr.yint = -100;
|
|
|
|
else if (viewer->spr.yint > 199) viewer->spr.yint = 300;
|
2021-12-27 09:15:42 +00:00
|
|
|
|
2022-09-08 11:27:22 +00:00
|
|
|
cpos = viewer->spr.pos.plusZ(-4);
|
2022-12-05 05:09:43 +00:00
|
|
|
cangles = DRotator(maphoriz(-viewer->spr.yint), viewer->interpolatedyaw(interpfrac), nullAngle);
|
2022-09-08 11:27:22 +00:00
|
|
|
sect = viewer->sector();
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-08-03 17:59:46 +00:00
|
|
|
if (isRRRA() && p->DrugMode)
|
|
|
|
{
|
|
|
|
double fovdelta = atan(getdrugmode(p, 65536) * (1. / 65536.)) * (360. / pi::pi()) - 90.;
|
|
|
|
fov = (float)clamp<double>(r_fov + fovdelta * 0.6, r_fov, 150.);
|
|
|
|
}
|
2022-08-03 17:38:39 +00:00
|
|
|
|
2020-11-05 06:25:44 +00:00
|
|
|
// The camera texture must be rendered with the base palette, so this is the only place where the current global palette can be set.
|
|
|
|
// The setting here will be carried over to the rendering of the weapon sprites, but other 2D content will always default to the main palette.
|
|
|
|
setgamepalette(setpal(p));
|
|
|
|
|
2022-11-21 04:08:33 +00:00
|
|
|
// use player's actor initially.
|
|
|
|
viewer = p->GetActor();
|
|
|
|
|
2020-07-05 14:49:00 +00:00
|
|
|
if ((snum == myconnectindex) && (numplayers > 1))
|
|
|
|
{
|
2022-09-08 11:16:27 +00:00
|
|
|
cpos = interpolatedvalue(omypos, mypos, interpfrac);
|
2022-12-05 05:09:43 +00:00
|
|
|
cangles = DRotator(interpolatedvalue(omyhoriz + omyhorizoff, myhoriz + myhorizoff, interpfrac), interpolatedvalue(omyang, myang, interpfrac), nullAngle);
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-11-21 04:24:25 +00:00
|
|
|
cpos = viewer->getRenderPos(interpfrac);
|
2022-12-05 05:09:43 +00:00
|
|
|
cangles = p->Angles.getRenderAngles(interpfrac);
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 23:20:51 +00:00
|
|
|
if (p->newOwner != nullptr)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-09-08 11:27:22 +00:00
|
|
|
viewer = p->newOwner;
|
|
|
|
cpos = viewer->spr.pos;
|
2022-12-05 05:09:43 +00:00
|
|
|
cangles = DRotator(maphoriz(-viewer->spr.shade), viewer->interpolatedyaw(interpfrac), nullAngle);
|
2022-09-08 11:27:22 +00:00
|
|
|
sect = viewer->sector();
|
2022-09-08 11:16:27 +00:00
|
|
|
interpfrac = 1.;
|
2021-12-27 09:15:42 +00:00
|
|
|
camview = true;
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
else if (p->over_shoulder_on == 0)
|
|
|
|
{
|
2022-09-08 11:16:27 +00:00
|
|
|
if (cl_viewbob) cpos.Z += interpolatedvalue(p->opyoff, p->pyoff, interpfrac);
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
2020-09-25 07:52:00 +00:00
|
|
|
else
|
|
|
|
{
|
2022-09-08 11:16:27 +00:00
|
|
|
auto adjustment = isRR() ? 15 : 12;
|
|
|
|
cpos.Z -= adjustment;
|
2020-09-25 07:52:00 +00:00
|
|
|
|
2022-12-05 05:09:43 +00:00
|
|
|
if (!calcChaseCamPos(cpos, viewer, §, cangles, interpfrac, 64.))
|
2020-09-25 07:52:00 +00:00
|
|
|
{
|
2022-09-08 11:16:27 +00:00
|
|
|
cpos.Z += adjustment;
|
2022-12-05 05:09:43 +00:00
|
|
|
calcChaseCamPos(cpos, viewer, §, cangles, interpfrac, 64.);
|
2020-09-25 07:52:00 +00:00
|
|
|
}
|
|
|
|
}
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-09-08 11:16:27 +00:00
|
|
|
double cz = p->GetActor()->ceilingz;
|
|
|
|
double fz = p->GetActor()->floorz;
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2022-11-30 16:15:44 +00:00
|
|
|
if (ud.earthquaketime > 0 && p->on_ground == 1)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-11-30 16:15:44 +00:00
|
|
|
cpos.Z += 1 - (((ud.earthquaketime) & 1) * 2.);
|
|
|
|
cangles.Yaw += DAngle::fromBuild((2 - ((ud.earthquaketime) & 2)) << 2);
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
|
2022-09-08 11:16:27 +00:00
|
|
|
if (p->GetActor()->spr.pal == 1) cpos.Z -= 18;
|
2020-07-05 14:49:00 +00:00
|
|
|
|
2020-11-02 23:20:51 +00:00
|
|
|
else if (p->spritebridge == 0 && p->newOwner == nullptr)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-09-20 00:36:32 +00:00
|
|
|
cpos.Z = min(max(cpos.Z, p->truecz + 4), p->truefz - 4);
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
|
2021-11-18 18:33:32 +00:00
|
|
|
if (sect)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
2022-10-25 18:12:51 +00:00
|
|
|
calcSlope(sect, cpos, &cz, &fz);
|
2022-09-20 00:36:32 +00:00
|
|
|
cpos.Z = min(max(cpos.Z, cz + 4), fz - 4);
|
2020-07-05 14:49:00 +00:00
|
|
|
}
|
|
|
|
}
|
2022-09-08 11:27:22 +00:00
|
|
|
|
|
|
|
auto cstat = viewer->spr.cstat;
|
|
|
|
if (camview) viewer->spr.cstat = CSTAT_SPRITE_INVISIBLE;
|
|
|
|
if (!sceneonly) drawweapon(interpfrac);
|
2022-12-05 05:09:43 +00:00
|
|
|
render_drawrooms(viewer, cpos, sect, cangles, interpfrac, fov);
|
2022-09-08 11:27:22 +00:00
|
|
|
viewer->spr.cstat = cstat;
|
|
|
|
|
2020-09-05 21:20:48 +00:00
|
|
|
//GLInterface.SetMapFog(false);
|
2020-11-26 15:03:40 +00:00
|
|
|
RestoreInterpolations();
|
2020-07-05 14:49:00 +00:00
|
|
|
|
|
|
|
if (!isRRRA() || !fogactive)
|
|
|
|
{
|
2021-02-18 10:46:36 +00:00
|
|
|
if (PlayClock < lastvisinc)
|
2020-07-05 14:49:00 +00:00
|
|
|
{
|
|
|
|
if (abs(p->visibility - ud.const_visibility) > 8)
|
|
|
|
p->visibility += (ud.const_visibility - p->visibility) >> 2;
|
|
|
|
}
|
|
|
|
else p->visibility = ud.const_visibility;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-05 19:21:39 +00:00
|
|
|
bool GameInterface::GenerateSavePic()
|
|
|
|
{
|
2022-09-11 10:04:17 +00:00
|
|
|
displayrooms(myconnectindex, 1., true);
|
2020-07-05 19:21:39 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-09-16 17:20:35 +00:00
|
|
|
void GameInterface::processSprites(tspriteArray& tsprites, const DVector3& view, DAngle viewang, double interpfrac)
|
2021-03-26 00:35:23 +00:00
|
|
|
{
|
2022-09-16 17:20:35 +00:00
|
|
|
fi.animatesprites(tsprites, view.XY(), viewang, interpfrac);
|
2021-03-26 00:35:23 +00:00
|
|
|
}
|
2020-07-05 19:21:39 +00:00
|
|
|
|
|
|
|
|
2020-07-04 21:40:54 +00:00
|
|
|
END_DUKE_NS
|