raze/source/games/duke/src/render.cpp

441 lines
12 KiB
C++
Raw Normal View History

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"
#include "prediction.h"
#include "automap.h"
2020-11-04 06:01:25 +00:00
#include "dukeactor.h"
#include "interpolate.h"
#include "render.h"
#include "glbackend/glbackend.h"
#include "_polymost.cpp"
2020-07-04 21:40:54 +00:00
// temporary hack to pass along RRRA's global fog. Needs to be done better later.
extern PalEntry GlobalMapFog;
extern float GlobalFogDensity;
EXTERN_CVAR(Bool, testnewrenderer)
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.
//
//---------------------------------------------------------------------------
2021-12-01 23:13:07 +00:00
void renderView(spritetype* playersprite, sectortype* sect, int x, int y, int z, binangle a, fixedhoriz h, binangle rotscrnang, int smoothratio)
2020-07-04 21:40:54 +00:00
{
if (!testnewrenderer)
2020-07-05 14:49:00 +00:00
{
// do screen rotation.
2021-05-12 15:57:36 +00:00
renderSetRollAngle((float)rotscrnang.asbuildf());
2020-07-05 14:49:00 +00:00
se40code(x, y, z, a, h, smoothratio);
renderMirror(x, y, z, a, h, smoothratio);
2021-12-01 23:13:07 +00:00
renderDrawRoomsQ16(x, y, z, a.asq16(), h.asq16(), sect, false);
fi.animatesprites(pm_tsprite, pm_spritesortcnt, x, y, a.asbuild(), smoothratio);
renderDrawMasks();
2020-07-05 14:49:00 +00:00
}
else
2020-07-05 14:49:00 +00:00
{
2021-12-01 23:13:07 +00:00
render_drawrooms(playersprite, { x, y, z }, sectnum(sect), a, h, rotscrnang, smoothratio);
2020-07-05 14:49:00 +00:00
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void GameInterface::UpdateCameras(double smoothratio)
2020-07-05 14:49:00 +00:00
{
2020-07-06 20:23:18 +00:00
const int VIEWSCREEN_ACTIVE_DISTANCE = 8192;
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
2020-11-04 18:25:49 +00:00
if (camsprite->GetOwner() && dist(p->GetActor(), camsprite) < VIEWSCREEN_ACTIVE_DISTANCE)
2020-07-05 14:49:00 +00:00
{
2021-12-21 20:35:07 +00:00
auto tex = tileGetTexture(camsprite->spr.picnum);
2021-05-12 15:57:36 +00:00
TileFiles.MakeCanvas(TILE_VIEWSCR, (int)tex->GetDisplayWidth(), (int)tex->GetDisplayHeight());
2020-07-05 14:49:00 +00:00
auto canvas = renderSetTarget(TILE_VIEWSCR);
if (!canvas) return;
screen->RenderTextureView(canvas, [=](IntRect& rect)
{
2021-12-21 20:35:07 +00:00
auto camera = &camsprite->GetOwner()->spr;
auto ang = buildang(camera->interpolatedang(smoothratio));
2020-07-05 14:49:00 +00:00
display_mirror = 1; // should really be 'display external view'.
if (!testnewrenderer)
{
// Note: no ROR or camera here - Polymost has no means to detect these things before rendering the scene itself.
2021-12-01 23:13:07 +00:00
renderDrawRoomsQ16(camera->x, camera->y, camera->z, ang.asq16(), IntToFixed(camera->shade), camera->sector(), false); // why 'shade'...?
2021-05-12 15:57:36 +00:00
fi.animatesprites(pm_tsprite, pm_spritesortcnt, camera->x, camera->y, ang.asbuild(), (int)smoothratio);
renderDrawMasks();
}
else
{
render_camtex(camera, camera->pos, camera->sector(), ang, buildhoriz(camera->shade), buildang(0), tex, rect, smoothratio);
}
2020-07-05 14:49:00 +00:00
display_mirror = 0;
});
renderRestoreTarget();
}
}
2021-03-29 19:48:23 +00:00
void GameInterface::EnterPortal(spritetype* viewer, int type)
{
if (type == PORTAL_WALL_MIRROR) display_mirror++;
}
void GameInterface::LeavePortal(spritetype* viewer, int type)
{
if (type == PORTAL_WALL_MIRROR) display_mirror--;
}
2021-11-18 18:33:32 +00:00
bool GameInterface::GetGeoEffect(GeoEffect* eff, sectortype* viewsector)
{
2021-11-18 18:33:32 +00:00
if (isRR() && viewsector->lotag == 848)
{
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
//---------------------------------------------------------------------------
//
// RRRA's drug distortion effect
2020-07-05 14:49:00 +00:00
//
//---------------------------------------------------------------------------
int DrugTimer;
2020-07-05 14:49:00 +00:00
static int getdrugmode(player_struct *p, int oyrepeat)
2020-07-05 14:49:00 +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
{
if (now - DrugTimer > 4 || now - DrugTimer < 0) DrugTimer = now - 1;
while (DrugTimer < now)
2020-07-05 14:49:00 +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;
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;
}
}
}
return p->drug_aspect;
2020-07-05 14:49:00 +00:00
}
else
2020-07-05 14:49:00 +00:00
{
DrugTimer = now;
return oyrepeat;
2020-07-05 14:49:00 +00:00
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void displayrooms(int snum, double smoothratio)
2020-07-05 14:49:00 +00:00
{
2020-07-05 19:24:55 +00:00
int cposx, cposy, cposz, fz, cz;
binangle cang, rotscrnang;
fixedhoriz choriz;
2020-07-05 14:49:00 +00:00
struct player_struct* p;
p = &ps[snum];
pm_smoothratio = (int)smoothratio;
2020-07-05 14:49:00 +00:00
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;
}
g_visibility = p->visibility;
videoSetCorrectedAspect();
2020-07-05 14:49:00 +00:00
auto sect = p->cursector;
2020-07-05 14:49:00 +00:00
GlobalMapFog = fogactive ? 0x999999 : 0;
GlobalFogDensity = fogactive ? 350.f : 0.f;
GLInterface.SetMapFog(fogactive != 0);
DoInterpolations(smoothratio / 65536.);
2020-07-05 14:49:00 +00:00
setgamepalette(BASEPAL);
if (!testnewrenderer) gi->UpdateCameras(smoothratio); // Only Polymost does this here. The new renderer calls this internally.
2020-07-05 14:49:00 +00:00
2020-11-02 22:53:55 +00:00
if (ud.cameraactor)
2020-07-05 14:49:00 +00:00
{
2021-12-21 20:35:07 +00:00
spritetype* s = &ud.cameraactor->spr;
2020-07-05 14:49:00 +00:00
if (s->yvel < 0) s->yvel = -100;
else if (s->yvel > 199) s->yvel = 300;
cang = buildang(interpolatedangle(ud.cameraactor->tempang, s->ang, smoothratio));
2020-07-05 14:49:00 +00:00
auto bh = buildhoriz(s->yvel);
2021-12-01 23:13:07 +00:00
renderView(s, s->sector(), s->x, s->y, s->z - (4 << 8), cang, bh, buildang(0), (int)smoothratio);
2020-07-05 14:49:00 +00:00
}
else
{
// Fixme: This should get the aspect ratio from the backend, not the current viewport size.
int i = DivScale(1, isRR() ? 64 : p->GetActor()->spr.yrepeat + 28, 22);
int viewingaspect = !isRRRA() || !p->DrugMode ? xs_CRoundToInt(double(i) * tan(r_fov * (pi::pi() / 360.))) : getdrugmode(p, i);
renderSetAspect(MulScale(viewingaspect, viewingrange, 16), yxaspect);
2020-07-05 14:49:00 +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));
// set screen rotation.
rotscrnang = !SyncInput() ? p->angle.rotscrnang : p->angle.interpolatedrotscrn(smoothratio);
2020-07-05 14:49:00 +00:00
2021-11-18 18:33:32 +00:00
#if 0
2020-07-05 14:49:00 +00:00
if ((snum == myconnectindex) && (numplayers > 1))
{
cposx = interpolatedvalue(omyx, myx, smoothratio);
cposy = interpolatedvalue(omyy, myy, smoothratio);
cposz = interpolatedvalue(omyz, myz, smoothratio);
if (SyncInput())
{
choriz = interpolatedhorizon(omyhoriz + omyhorizoff, myhoriz + myhorizoff, smoothratio);
cang = interpolatedangle(omyang, myang, smoothratio);
}
else
{
cang = myang;
choriz = myhoriz + myhorizoff;
}
2020-07-05 14:49:00 +00:00
sect = mycursectnum;
}
else
2021-11-18 18:33:32 +00:00
#endif
2020-07-05 14:49:00 +00:00
{
2021-12-22 09:36:09 +00:00
cposx = interpolatedvalue(p->oposx, p->pos.X, smoothratio);
2021-12-22 09:40:26 +00:00
cposy = interpolatedvalue(p->oposy, p->pos.Y, smoothratio);
cposz = interpolatedvalue(p->oposz, p->pos.z, smoothratio);;
if (SyncInput())
{
// Original code for when the values are passed through the sync struct
choriz = p->horizon.interpolatedsum(smoothratio);
cang = p->angle.interpolatedsum(smoothratio);
}
else
{
// This is for real time updating of the view direction.
cang = p->angle.sum();
choriz = p->horizon.sum();
}
2020-07-05 14:49:00 +00:00
}
spritetype* viewer;
2020-11-02 23:20:51 +00:00
if (p->newOwner != nullptr)
2020-07-05 14:49:00 +00:00
{
2021-12-21 20:35:07 +00:00
auto spr = &p->newOwner->spr;
cang = buildang(spr->interpolatedang(smoothratio));
2020-11-02 18:23:15 +00:00
choriz = buildhoriz(spr->shade);
2021-12-22 09:36:09 +00:00
cposx = spr->pos.X;
2021-12-22 09:40:26 +00:00
cposy = spr->pos.Y;
2020-11-02 18:23:15 +00:00
cposz = spr->pos.z;
2021-11-18 18:33:32 +00:00
sect = spr->sector();
rotscrnang = buildang(0);
smoothratio = MaxSmoothRatio;
viewer = spr;
2020-07-05 14:49:00 +00:00
}
else if (p->over_shoulder_on == 0)
{
if (cl_viewbob) cposz += interpolatedvalue(p->opyoff, p->pyoff, smoothratio);
2021-12-21 20:35:07 +00:00
viewer = &p->GetActor()->spr;
2020-07-05 14:49:00 +00:00
}
else
{
cposz -= isRR() ? 3840 : 3072;
2021-12-21 20:35:07 +00:00
viewer = &p->GetActor()->spr;
if (!calcChaseCamPos(&cposx, &cposy, &cposz, viewer, &sect, cang, choriz, smoothratio))
{
cposz += isRR() ? 3840 : 3072;
2021-12-21 20:35:07 +00:00
calcChaseCamPos(&cposx, &cposy, &cposz, viewer, &sect, cang, choriz, smoothratio);
}
}
2020-07-05 14:49:00 +00:00
2020-11-02 19:23:05 +00:00
cz = p->GetActor()->ceilingz;
fz = p->GetActor()->floorz;
2020-07-05 14:49:00 +00:00
if (earthquaketime > 0 && p->on_ground == 1)
{
cposz += 256 - (((earthquaketime) & 1) << 9);
cang += buildang((2 - ((earthquaketime) & 2)) << 2);
2020-07-05 14:49:00 +00:00
}
if (p->GetActor()->spr.pal == 1) cposz -= (18 << 8);
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
{
if (cposz < (p->truecz + (4 << 8))) cposz = cz + (4 << 8);
else if (cposz > (p->truefz - (4 << 8))) cposz = fz - (4 << 8);
}
2021-11-18 18:33:32 +00:00
if (sect)
2020-07-05 14:49:00 +00:00
{
2021-11-18 18:33:32 +00:00
getzsofslopeptr(sect, cposx, cposy, &cz, &fz);
2020-07-05 14:49:00 +00:00
if (cposz < cz + (4 << 8)) cposz = cz + (4 << 8);
if (cposz > fz - (4 << 8)) cposz = fz - (4 << 8);
}
choriz = clamp(choriz, q16horiz(gi->playerHorizMin()), q16horiz(gi->playerHorizMax()));
2020-07-05 14:49:00 +00:00
2021-11-18 18:33:32 +00:00
if (isRR() && sect->lotag == 848 && !testnewrenderer)
2020-07-05 14:49:00 +00:00
{
2021-05-12 15:57:36 +00:00
renderSetRollAngle((float)rotscrnang.asbuildf());
2021-11-18 18:33:32 +00:00
geometryEffect(cposx, cposy, cposz, cang, choriz, sectnum(sect), (int)smoothratio);
2020-07-05 14:49:00 +00:00
}
else
{
2021-12-01 23:13:07 +00:00
renderView(viewer, sect, cposx, cposy, cposz, cang, choriz, rotscrnang, (int)smoothratio);
2020-07-05 14:49:00 +00:00
}
}
2020-09-05 21:20:48 +00:00
//GLInterface.SetMapFog(false);
RestoreInterpolations();
2020-07-05 14:49:00 +00:00
if (!isRRRA() || !fogactive)
{
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;
}
}
bool GameInterface::GenerateSavePic()
{
displayrooms(myconnectindex, MaxSmoothRatio);
return true;
}
void GameInterface::processSprites(tspritetype* tsprite, int& spritesortcnt, int viewx, int viewy, int viewz, binangle viewang, double smoothRatio)
{
fi.animatesprites(tsprite, spritesortcnt, viewx, viewy, viewang.asbuild(), int(smoothRatio));
}
2020-07-04 21:40:54 +00:00
END_DUKE_NS