raze/source/games/blood/src/view.cpp

874 lines
24 KiB
C++
Raw Normal View History

2019-09-19 22:42:45 +00:00
//-------------------------------------------------------------------------
/*
Copyright (C) 2010-2019 EDuke32 developers and contributors
Copyright (C) 2019 Nuke.YKT
This file is part of NBlood.
NBlood is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License version 2
as published by the Free Software Foundation.
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.
*/
//-------------------------------------------------------------------------
#include "ns.h" // Must come before everything else!
2019-09-19 22:42:45 +00:00
#include <stdlib.h>
#include <string.h>
#include "build.h"
2020-07-31 19:05:09 +00:00
#include "v_font.h"
2019-09-19 22:42:45 +00:00
#include "blood.h"
2020-07-31 19:05:09 +00:00
#include "choke.h"
#include "zstring.h"
#include "razemenu.h"
#include "gstrings.h"
#include "v_2ddrawer.h"
#include "v_video.h"
2020-07-31 19:05:09 +00:00
#include "v_font.h"
2020-09-06 08:59:45 +00:00
#include "statusbar.h"
2020-09-06 10:44:58 +00:00
#include "automap.h"
#include "gamefuncs.h"
#include "v_draw.h"
#include "precache.h"
2021-03-20 22:01:16 +00:00
#include "render.h"
#include "razefont.h"
2019-09-19 22:42:45 +00:00
BEGIN_BLD_NS
2019-09-19 22:42:45 +00:00
VIEWPOS gViewPos;
int gViewIndex;
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2019-09-19 22:42:45 +00:00
void viewBackupView(int nPlayer)
{
2021-12-29 21:56:21 +00:00
PLAYER* pPlayer = &gPlayer[nPlayer];
2022-09-10 12:15:33 +00:00
pPlayer->ozView = pPlayer->zView;
2022-09-10 12:20:04 +00:00
pPlayer->ozWeapon = pPlayer->zWeapon - pPlayer->zView - 0xc00;
pPlayer->obobHeight = pPlayer->bobHeight;
pPlayer->obobWidth = pPlayer->bobWidth;
pPlayer->oswayHeight = pPlayer->swayHeight;
pPlayer->oswayWidth = pPlayer->swayWidth;
2021-12-29 21:56:21 +00:00
pPlayer->angle.backup();
pPlayer->horizon.backup();
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void viewDrawText(FFont* pFont, const char* pString, int x, int y, int nShade, int nPalette, int position, bool shadow)
2019-09-19 22:42:45 +00:00
{
2021-12-29 21:56:21 +00:00
if (!pString) return;
2021-12-29 21:56:21 +00:00
//y += pFont->yoff;
2020-08-10 19:14:42 +00:00
if (position == 1) x -= pFont->StringWidth(pString) / 2;
2021-12-29 21:56:21 +00:00
if (position == 2) x -= pFont->StringWidth(pString);
2020-08-10 19:14:42 +00:00
2020-07-31 19:05:09 +00:00
if (shadow)
{
2021-12-29 21:56:21 +00:00
DrawText(twod, pFont, CR_UNTRANSLATED, x + 1, y + 1, pString, DTA_FullscreenScale, FSMode_Fit320x200, DTA_Color, 0xff000000, DTA_Alpha, 0.5, TAG_DONE);
2020-07-31 19:05:09 +00:00
}
DrawText(twod, pFont, CR_NATIVEPAL, x, y, pString, DTA_FullscreenScale, FSMode_Fit320x200, DTA_TranslationIndex, TRANSLATION(Translation_Remap, nPalette),
2021-12-29 21:56:21 +00:00
DTA_Color, shadeToLight(nShade), TAG_DONE);
2020-08-01 18:07:32 +00:00
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
GameStats GameInterface::getStats()
{
return { gKillMgr.Kills, gKillMgr.TotalKills, gSecretMgr.Founds, gSecretMgr.Total, gFrameCount / kTicsPerSec, gPlayer[myconnectindex].fragCount };
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void viewDrawAimedPlayerName(PLAYER* pPlayer)
{
if (!cl_idplayers || (pPlayer->aim.dx == 0 && pPlayer->aim.dy == 0))
2021-12-29 21:56:21 +00:00
return;
int hit = HitScan(pPlayer->actor, pPlayer->zView, pPlayer->aim.dx, pPlayer->aim.dy, pPlayer->aim.dz, CLIPMASK0, 512);
2021-12-29 21:56:21 +00:00
if (hit == 3)
{
auto actor = gHitInfo.actor();
if (actor && actor->IsPlayerActor())
{
int nPlayer = actor->spr.type - kDudePlayer1;
const char* szName = PlayerName(nPlayer);
int nPalette = (gPlayer[nPlayer].teamId & 3) + 11;
viewDrawText(DigiFont, szName, 160, 125, -128, nPalette, 1, 1);
}
}
}
2020-04-11 21:54:33 +00:00
static TArray<uint8_t> lensdata;
2021-12-29 21:56:21 +00:00
int* lensTable;
2019-09-19 22:42:45 +00:00
extern DAngle random_angles[16][3];
2019-09-19 22:42:45 +00:00
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2019-09-19 22:42:45 +00:00
void viewInit(void)
{
2021-12-29 21:56:21 +00:00
Printf("Initializing status bar\n");
2021-12-29 21:56:21 +00:00
lensdata = fileSystem.LoadFile("lens.dat");
assert(lensdata.Size() == kLensSize * kLensSize * sizeof(int));
2019-09-19 22:42:45 +00:00
2021-12-29 21:56:21 +00:00
lensTable = (int*)lensdata.Data();
#if WORDS_BIGENDIAN
2021-12-29 21:56:21 +00:00
for (int i = 0; i < kLensSize * kLensSize; i++)
{
lensTable[i] = LittleLong(lensTable[i]);
}
2019-09-19 22:42:45 +00:00
#endif
2021-12-29 21:56:21 +00:00
uint8_t* data = TileFiles.tileCreate(4077, kLensSize, kLensSize);
memset(data, TRANSPARENT_INDEX, kLensSize * kLensSize);
for (int i = 0; i < 16; i++)
{
random_angles[i][0] = randomAngle();
random_angles[i][1] = randomAngle();
random_angles[i][2] = randomAngle();
2021-12-29 21:56:21 +00:00
}
2019-09-19 22:42:45 +00:00
}
int othercameradist = 1280;
int othercameraclock;
2019-09-19 22:42:45 +00:00
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2021-11-24 00:30:05 +00:00
#if 0
2021-12-29 21:56:21 +00:00
void CalcOtherPosition(DBloodActor* actor, int* pX, int* pY, int* pZ, sectortype** vsectnum, int nAng, fixed_t zm, int smoothratio) // currently unused
2019-09-19 22:42:45 +00:00
{
2021-12-29 21:56:21 +00:00
int vX = MulScale(-Cos(nAng), 1280, 30);
int vY = MulScale(-Sin(nAng), 1280, 30);
int vZ = FixedToInt(MulScale(zm, 1280, 3)) - (16 << 8);
int bakCstat = pSprite->cstat;
pSprite->cstat &= ~CSTAT_SPRITE_BLOCK_HITSCAN;
assert(validSectorIndex(*vsectnum));
FindSector(*pX, *pY, *pZ, vsectnum);
int nHSector;
int hX, hY;
vec3_t pos = { *pX, *pY, *pZ };
hitscan(&pos, *vsectnum, vX, vY, vZ, &hitdata, CLIPMASK1);
nHSector = hitdata.sect;
hX = hitdata.pos.x;
hY = hitdata.pos.y;
int dX = hX - *pX;
int dY = hY - *pY;
if (abs(vX) + abs(vY) > abs(dX) + abs(dY))
{
*vsectnum = nHSector;
dX -= Sgn(vX) << 6;
dY -= Sgn(vY) << 6;
int nDist;
if (abs(vX) > abs(vY))
{
nDist = ClipHigh(DivScale(dX, vX, 16), othercameradist);
}
else
{
nDist = ClipHigh(DivScale(dY, vY, 16), othercameradist);
}
othercameradist = nDist;
}
*pX += MulScale(vX, othercameradist, 16);
*pY += MulScale(vY, othercameradist, 16);
*pZ += MulScale(vZ, othercameradist, 16);
int myclock = PlayClock + MulScale(4, smoothratio, 16);
2021-12-29 21:56:21 +00:00
othercameradist = ClipHigh(othercameradist + ((myclock - othercameraclock) << 10), 65536);
othercameraclock = myclock;
assert(validSectorIndex(*vsectnum));
2021-12-29 21:56:21 +00:00
FindSector(*pX, *pY, *pZ, vsectnum);
pSprite->cstat = bakCstat;
2019-09-19 22:42:45 +00:00
}
2021-11-24 00:30:05 +00:00
#endif
2019-09-19 22:42:45 +00:00
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
// by NoOne: show warning msgs in game instead of throwing errors (in some cases)
2021-12-29 21:56:21 +00:00
//
//---------------------------------------------------------------------------
void viewSetSystemMessage(const char* pMessage, ...) {
2021-12-29 21:56:21 +00:00
char buffer[1024]; va_list args; va_start(args, pMessage);
vsprintf(buffer, pMessage, args);
Printf(PRINT_HIGH | PRINT_NOTIFY, "%s\n", buffer); // print it also in console
}
void viewSetMessage(const char* pMessage, const char* color, const MESSAGE_PRIORITY priority)
2019-09-19 22:42:45 +00:00
{
int printlevel = priority <= MESSAGE_PRIORITY_NORMAL ? PRINT_LOW : priority < MESSAGE_PRIORITY_SYSTEM ? PRINT_MEDIUM : PRINT_HIGH;
Printf(printlevel | PRINT_NOTIFY, "%s%s\n", color? color : TEXTCOLOR_TAN, pMessage);
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
void viewSetErrorMessage(const char* pMessage)
2019-09-19 22:42:45 +00:00
{
2021-12-29 21:56:21 +00:00
Printf(PRINT_BOLD | PRINT_NOTIFY, "%s\n", pMessage);
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2019-09-19 22:42:45 +00:00
void DoLensEffect(void)
{
// To investigate whether this can be implemented as a shader effect.
2021-12-29 21:56:21 +00:00
auto d = tileData(4077);
assert(d != NULL);
auto s = tilePtr(4079);
2021-12-29 21:56:21 +00:00
assert(s != NULL);
for (int i = 0; i < kLensSize * kLensSize; i++, d++)
if (lensTable[i] >= 0)
*d = s[lensTable[i]];
TileFiles.InvalidateTile(4077);
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2019-09-19 22:42:45 +00:00
void UpdateDacs(int nPalette, bool bNoTint)
{
2021-12-29 21:56:21 +00:00
gLastPal = 0;
auto& tint = lookups.tables[MAXPALOOKUPS - 1];
tint.tintFlags = 0;
switch (nPalette)
{
case 0:
default:
tint.tintColor.r = 255;
tint.tintColor.g = 255;
tint.tintColor.b = 255;
break;
case 1:
tint.tintColor.r = 132;
tint.tintColor.g = 164;
tint.tintColor.b = 255;
break;
case 2:
tint.tintColor.r = 255;
tint.tintColor.g = 126;
tint.tintColor.b = 105;
break;
case 3:
tint.tintColor.r = 162;
tint.tintColor.g = 186;
tint.tintColor.b = 15;
break;
case 4:
tint.tintColor.r = 255;
tint.tintColor.g = 255;
tint.tintColor.b = 255;
break;
}
videoSetPalette(nPalette);
}
2019-09-19 22:42:45 +00:00
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void UpdateBlend(PLAYER* pPlayer)
{
2021-12-29 21:56:21 +00:00
int nRed = 0;
int nGreen = 0;
int nBlue = 0;
2019-09-19 22:42:45 +00:00
nRed += pPlayer->pickupEffect;
nGreen += pPlayer->pickupEffect;
nBlue -= pPlayer->pickupEffect;
2019-09-19 22:42:45 +00:00
nRed += ClipHigh(pPlayer->painEffect, 85) * 2;
nGreen -= ClipHigh(pPlayer->painEffect, 85) * 3;
nBlue -= ClipHigh(pPlayer->painEffect, 85) * 3;
2019-09-19 22:42:45 +00:00
nRed -= pPlayer->blindEffect;
nGreen -= pPlayer->blindEffect;
nBlue -= pPlayer->blindEffect;
nRed -= pPlayer->chokeEffect >> 6;
nGreen -= pPlayer->chokeEffect >> 5;
nBlue -= pPlayer->chokeEffect >> 6;
2021-12-29 21:56:21 +00:00
nRed = ClipRange(nRed, -255, 255);
nGreen = ClipRange(nGreen, -255, 255);
nBlue = ClipRange(nBlue, -255, 255);
2021-12-29 21:56:21 +00:00
videoTintBlood(nRed, nGreen, nBlue);
2019-09-19 22:42:45 +00:00
}
// int gVisibility;
int deliriumTilt, deliriumPitch;
int deliriumPitchO;
DAngle deliriumTurnO, deliriumTurn;
DAngle gScreenTiltO, gScreenTilt;
2019-09-19 22:42:45 +00:00
int gShowFrameRate = 1;
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void viewUpdateDelirium(PLAYER* pPlayer)
2019-09-19 22:42:45 +00:00
{
2021-12-29 21:56:21 +00:00
gScreenTiltO = gScreenTilt;
deliriumTurnO = deliriumTurn;
deliriumPitchO = deliriumPitch;
2019-09-19 22:42:45 +00:00
int powerCount;
if ((powerCount = powerupCheck(pPlayer, kPwUpDeliriumShroom)) != 0)
2019-09-19 22:42:45 +00:00
{
int tilt1 = 170, tilt2 = 170, pitch = 20;
int timer = PlayClock * 2;
2019-09-19 22:42:45 +00:00
if (powerCount < 512)
{
int powerScale = IntToFixed(powerCount) / 512;
tilt1 = MulScale(tilt1, powerScale, 16);
tilt2 = MulScale(tilt2, powerScale, 16);
pitch = MulScale(pitch, powerScale, 16);
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
int sin2 = Sin(2 * timer) >> 1;
int sin3 = Sin(3 * timer) >> 1;
gScreenTilt = DAngle::fromBuild(MulScale(sin2 + sin3, tilt1, 30));
2021-12-29 21:56:21 +00:00
int sin4 = Sin(4 * timer) >> 1;
deliriumTurn = DAngle::fromBuild(MulScale(sin3 + sin4, tilt2, 30));
2021-12-29 21:56:21 +00:00
int sin5 = Sin(5 * timer) >> 1;
deliriumPitch = MulScale(sin4 + sin5, pitch, 30);
2019-09-19 22:42:45 +00:00
return;
}
gScreenTilt = gScreenTilt.Normalized180();
if (gScreenTilt > nullAngle)
2019-09-19 22:42:45 +00:00
{
gScreenTilt -= DAngle::fromBuild(8);
if (gScreenTilt < nullAngle)
gScreenTilt = nullAngle;
2019-09-19 22:42:45 +00:00
}
else if (gScreenTilt < nullAngle)
2019-09-19 22:42:45 +00:00
{
gScreenTilt += DAngle::fromBuild(8);
if (gScreenTilt >= nullAngle)
gScreenTilt = nullAngle;
2019-09-19 22:42:45 +00:00
}
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void viewUpdateShake(PLAYER* pPlayer, DVector3& cPos, DAngle& cA, fixedhoriz& cH, double& pshakeX, double& pshakeY)
2019-09-19 22:42:45 +00:00
{
2021-12-29 21:56:21 +00:00
auto doEffect = [&](const int& effectType)
{
if (effectType)
{
int nValue = ClipHigh(effectType * 8, 2000);
cH += buildhoriz(QRandom2(nValue >> 8));
cA += DAngle::fromBuild(QRandom2(nValue >> 8));
cPos.X += QRandom2F(nValue * inttoworld) * inttoworld;
cPos.Y += QRandom2F(nValue * inttoworld) * inttoworld;
cPos.Z += QRandom2F(nValue) * zinttoworld;
2021-12-29 21:56:21 +00:00
pshakeX += QRandom2(nValue);
pshakeY += QRandom2(nValue);
}
};
doEffect(pPlayer->flickerEffect);
doEffect(pPlayer->quakeEffect);
2019-09-19 22:42:45 +00:00
}
int gLastPal = 0;
int32_t g_frameRate;
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void DrawMap(PLAYER* pPlayer, const double interpfrac)
2020-09-06 08:59:45 +00:00
{
2021-12-29 21:56:21 +00:00
int tm = 0;
if (viewport3d.Left() > 0)
2021-12-29 21:56:21 +00:00
{
setViewport(Hud_Stbar);
tm = 1;
}
auto ang = !SyncInput() ? pPlayer->angle.sum() : pPlayer->angle.interpolatedsum(interpfrac);
DrawOverheadMap(pPlayer->actor->interpolatedpos(interpfrac).XY(), ang, interpfrac);
2021-12-29 21:56:21 +00:00
if (tm)
setViewport(hud_size);
2020-09-06 08:59:45 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void SetupView(PLAYER* pPlayer, DVector3& cPos, DAngle& cA, fixedhoriz& cH, sectortype*& pSector, double& zDelta, double& shakeX, double& shakeY, DAngle& rotscrnang, const double interpfrac)
2021-03-20 22:01:16 +00:00
{
double bobWidth, bobHeight;
2021-12-29 21:56:21 +00:00
pSector = pPlayer->actor->sector();
2021-11-23 23:10:48 +00:00
#if 0
if (numplayers > 1 && pPlayer == gMe && gPrediction && gMe->actor->xspr.health > 0)
2021-12-29 21:56:21 +00:00
{
nSectnum = predict.sectnum;
cX = interpolatedvalue(predictOld.x, predict.x, interpfrac);
cY = interpolatedvalue(predictOld.y, predict.y, interpfrac);
cZ = interpolatedvalue(predictOld.viewz, predict.viewz, interpfrac);
zDelta = interpolatedvalue(predictOld.weaponZ, predict.weaponZ, interpfrac);
bobWidth = interpolatedvalue(predictOld.bobWidth, predict.bobWidth, interpfrac);
bobHeight = interpolatedvalue(predictOld.bobHeight, predict.bobHeight, interpfrac);
shakeX = interpolatedvalue(predictOld.shakeBobX, predict.shakeBobX, interpfrac);
shakeY = interpolatedvalue(predictOld.shakeBobY, predict.shakeBobY, interpfrac);
2021-12-29 21:56:21 +00:00
if (!SyncInput())
{
cA = bamang(predict.angle.asbam() + predict.look_ang.asbam());
cH = predict.horiz + predict.horizoff;
rotscrnang = predict.rotscrnang;
}
else
{
cA = interpolatedvalue(predictOld.angle + predictOld.look_ang, predict.angle + predict.look_ang, interpfrac);
cH = interpolatedvalue(predictOld.horiz + predictOld.horizoff, predict.horiz + predict.horizoff, interpfrac);
rotscrnang = interpolatedvalue(predictOld.rotscrnang, predict.rotscrnang, interpfrac);
2021-12-29 21:56:21 +00:00
}
}
else
2021-11-23 23:10:48 +00:00
#endif
2021-12-29 21:56:21 +00:00
{
cPos.XY() = pPlayer->actor->interpolatedpos(interpfrac).XY();
cPos.Z = interpolatedvalue(pPlayer->ozView, pPlayer->zView, interpfrac) * zinttoworld;
zDelta = interpolatedvalue<double>(pPlayer->ozWeapon, pPlayer->zWeapon - pPlayer->zView - (12 << 8), interpfrac);
bobWidth = interpolatedvalue<double>(pPlayer->obobWidth, pPlayer->bobWidth, interpfrac);
bobHeight = interpolatedvalue<double>(pPlayer->obobHeight, pPlayer->bobHeight, interpfrac);
shakeX = interpolatedvalue<double>(pPlayer->oswayWidth, pPlayer->swayWidth, interpfrac);
shakeY = interpolatedvalue<double>(pPlayer->oswayHeight, pPlayer->swayHeight, interpfrac);
2021-12-29 21:56:21 +00:00
if (!SyncInput())
{
cA = pPlayer->angle.sum();
cH = pPlayer->horizon.sum();
rotscrnang = pPlayer->angle.rotscrnang;
2021-12-29 21:56:21 +00:00
}
else
{
cA = pPlayer->angle.interpolatedsum(interpfrac);
cH = pPlayer->horizon.interpolatedsum(interpfrac);
rotscrnang = pPlayer->angle.interpolatedrotscrn(interpfrac);
2021-12-29 21:56:21 +00:00
}
}
viewUpdateShake(pPlayer, cPos, cA, cH, shakeX, shakeY);
cH += buildhoriz(MulScale(0x40000000 - Cos(pPlayer->tiltEffect << 2), 30, 30));
2021-12-29 21:56:21 +00:00
if (gViewPos == 0)
{
if (cl_viewhbob)
{
cPos.X -= bobWidth * cA.Sin() * (1. / 256.);
cPos.Y += bobWidth * cA.Cos() * (1. / 256.);
2021-12-29 21:56:21 +00:00
}
if (cl_viewvbob)
{
cPos.Z += bobHeight * zinttoworld;
2021-12-29 21:56:21 +00:00
}
cPos.Z += FixedToFloat<24>(cH.asq16() * 10);
2021-12-29 21:56:21 +00:00
}
else
{
calcChaseCamPos(cPos, pPlayer->actor, &pSector, cA, cH, interpfrac);
2021-12-29 21:56:21 +00:00
}
if (pSector != nullptr)
CheckLink(cPos, &pSector);
2021-03-20 22:01:16 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void renderCrystalBall()
{
#if 0
2021-12-29 21:56:21 +00:00
int tmp = (PlayClock / 240) % (gNetPlayers - 1);
int i = connecthead;
while (1)
{
if (i == gViewIndex)
i = connectpoint2[i];
if (tmp == 0)
break;
i = connectpoint2[i];
tmp--;
}
PLAYER* pOther = &gPlayer[i];
//othercameraclock = PlayClock + MulScale(4, (int)gInterpolate, 16);;
if (!tileData(4079))
{
TileFiles.tileCreate(4079, 128, 128);
}
//renderSetTarget(4079, 128, 128);
renderSetAspect(65536, 78643);
int vd8 = pOther->actor->spr.x;
int vd4 = pOther->actor->spr.y;
int vd0 = pOther->zView;
int vcc = pOther->actor->spr.sectnum;
int v50 = pOther->actor->spr.angle;
2021-12-29 21:56:21 +00:00
int v54 = 0;
if (pOther->flickerEffect)
{
int nValue = ClipHigh(pOther->flickerEffect * 8, 2000);
v54 += QRandom2(nValue >> 8);
v50 += QRandom2(nValue >> 8);
vd8 += QRandom2(nValue >> 4);
vd4 += QRandom2(nValue >> 4);
vd0 += QRandom2(nValue);
}
if (pOther->quakeEffect)
{
int nValue = ClipHigh(pOther->quakeEffect * 8, 2000);
v54 += QRandom2(nValue >> 8);
v50 += QRandom2(nValue >> 8);
vd8 += QRandom2(nValue >> 4);
vd4 += QRandom2(nValue >> 4);
vd0 += QRandom2(nValue);
}
CalcOtherPosition(pOther->actor, &vd8, &vd4, &vd0, &vcc, v50, 0, (int)gInterpolate);
CheckLink(&vd8, &vd4, &vd0, &vcc);
uint8_t v14 = 0;
if (IsUnderwaterSector(vcc))
{
v14 = 10;
}
drawrooms(vd8, vd4, vd0, v50, v54, vcc);
viewProcessSprites(vd8, vd4, vd0, v50, gInterpolate);
renderDrawMasks();
#endif
}
2020-09-06 08:59:45 +00:00
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2020-01-14 20:20:46 +00:00
void viewDrawScreen(bool sceneonly)
2019-09-19 22:42:45 +00:00
{
PLAYER* pPlayer = &gPlayer[gViewIndex];
2021-05-09 07:05:42 +00:00
if (testgotpic(2342, true))
{
FireProcess();
}
double interpfrac;
if (!paused && (!M_Active() || gGameOptions.nGameType != 0))
{
interpfrac = !cl_interpolate || cl_capfps ? 1. : I_GetTimeFrac() * 1.;
}
else interpfrac = 1.;
2021-12-29 21:56:21 +00:00
if (cl_interpolate)
{
DoInterpolations(interpfrac);
2021-12-29 21:56:21 +00:00
}
if (automapMode != am_full)
{
DoSectorLighting();
}
if (automapMode == am_off)
{
int basepal = 0;
if (powerupCheck(pPlayer, kPwUpDeathMask) > 0) basepal = 4;
else if (powerupCheck(pPlayer, kPwUpReflectShots) > 0) basepal = 1;
else if (pPlayer->isUnderwater) {
if (pPlayer->nWaterPal) basepal = pPlayer->nWaterPal;
2021-12-29 21:56:21 +00:00
else {
if (pPlayer->actor->xspr.medium == kMediumWater) basepal = 1;
else if (pPlayer->actor->xspr.medium == kMediumGoo) basepal = 3;
2021-12-29 21:56:21 +00:00
else basepal = 2;
}
}
UpdateDacs(basepal);
UpdateBlend(pPlayer);
2021-12-29 21:56:21 +00:00
DVector3 cPos;
DAngle cA, rotscrnang;
2021-12-29 21:56:21 +00:00
fixedhoriz cH;
sectortype* pSector;
double zDelta;
double shakeX, shakeY;
SetupView(pPlayer, cPos, cA, cH, pSector, zDelta, shakeX, shakeY, rotscrnang, interpfrac);
2021-12-29 21:56:21 +00:00
DAngle tilt = interpolatedvalue(gScreenTiltO, gScreenTilt, interpfrac);
bool bDelirium = powerupCheck(pPlayer, kPwUpDeliriumShroom) > 0;
2021-12-29 21:56:21 +00:00
static bool bDeliriumOld = false;
//int tiltcs, tiltdim;
uint8_t otherview = powerupCheck(pPlayer, kPwUpCrystalBall) > 0;
if (tilt.Degrees() || bDelirium)
2021-12-29 21:56:21 +00:00
{
rotscrnang = tilt;
}
else if (otherview && gNetPlayers > 1)
{
#if 0
2021-12-29 21:56:21 +00:00
renderCrystalBall();
#endif
2021-12-29 21:56:21 +00:00
}
else
{
othercameraclock = PlayClock + int(4 * interpfrac);
2021-12-29 21:56:21 +00:00
}
if (!bDelirium)
{
deliriumTilt = 0;
deliriumTurn = DAngle::fromDeg(0.);
2021-12-29 21:56:21 +00:00
deliriumPitch = 0;
}
int brightness = 0;
BloodStatIterator it(kStatExplosion);
while (auto actor = it.Next())
{
2021-12-30 15:39:43 +00:00
if (actor->hasX() && gotsector[actor->sectno()])
2021-12-29 21:56:21 +00:00
{
brightness += actor->xspr.data3 * 32;
}
}
it.Reset(kStatProjectile);
while (auto actor = it.Next())
{
switch (actor->spr.type) {
case kMissileFlareRegular:
case kMissileTeslaAlt:
case kMissileFlareAlt:
case kMissileTeslaRegular:
2021-12-30 15:39:43 +00:00
if (gotsector[actor->sectno()]) brightness += 256;
2021-12-29 21:56:21 +00:00
break;
}
}
g_relvisibility = (int32_t)(ClipLow(gVisibility - 32 * pPlayer->visibility - brightness, 0)) - g_visibility;
cA += interpolatedvalue(deliriumTurnO, deliriumTurn, interpfrac);
2021-12-29 21:56:21 +00:00
if (pSector != nullptr)
2021-12-29 21:56:21 +00:00
{
double ceilingZ, floorZ;
getzsofslopeptr(pSector, cPos, &ceilingZ, &floorZ);
if ((cPos.Z > floorZ - 1) && (pSector->upperLink == nullptr)) // clamp to floor
{
cPos.Z = floorZ - 1;
}
if ((cPos.Z < ceilingZ + 1) && (pSector->lowerLink == nullptr)) // clamp to ceiling
{
cPos.Z = ceilingZ + 1;
}
2021-12-29 21:56:21 +00:00
}
cH = q16horiz(ClipRange(cH.asq16(), gi->playerHorizMin(), gi->playerHorizMax()));
if ((tilt.Degrees() || bDelirium) && !sceneonly)
2021-12-29 21:56:21 +00:00
{
if (gDeliriumBlur)
{
// todo: Set up a blurring postprocessing shader.
//const float fBlur = pow(1.f/3.f, 30.f/g_frameRate);
//g lAccum(GL _MULT, fBlur);
//g lAccum(GL _ACCUM, 1.f-fBlur);
//g lAccum(GL _RETURN, 1.f);
}
}
if (!sceneonly) hudDraw(pPlayer, pSector, shakeX, shakeY, zDelta, basepal, interpfrac);
fixedhoriz deliriumPitchI = interpolatedvalue(q16horiz(deliriumPitchO), q16horiz(deliriumPitch), interpfrac);
auto bakCstat = pPlayer->actor->spr.cstat;
pPlayer->actor->spr.cstat |= (gViewPos == 0) ? CSTAT_SPRITE_INVISIBLE : CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_TRANS_FLIP;
render_drawrooms(pPlayer->actor, cPos, sectnum(pSector), cA, cH + deliriumPitchI, rotscrnang, interpfrac);
pPlayer->actor->spr.cstat = bakCstat;
2021-12-29 21:56:21 +00:00
bDeliriumOld = bDelirium && gDeliriumBlur;
int nClipDist = pPlayer->actor->int_clipdist();
2021-12-29 21:56:21 +00:00
int vec, vf4;
Collision c1, c2;
GetZRange(pPlayer->actor, &vf4, &c1, &vec, &c2, nClipDist, 0);
2021-12-29 21:56:21 +00:00
if (sceneonly) return;
double look_anghalf = pPlayer->angle.look_anghalf(interpfrac);
DrawCrosshair(kCrosshairTile, pPlayer->actor->xspr.health >> 4, -look_anghalf, 0, 2);
#if 0 // This currently does not work. May have to be redone as a hardware effect.
if (v4 && gNetPlayers > 1)
{
DoLensEffect();
r otatesprite(IntToFixed(280), IntToFixed(35), 53248, 512, 4077, v10, v14, 512 + 6, gViewX0, gViewY0, gViewX1, gViewY1);
r otatesprite(IntToFixed(280), IntToFixed(35), 53248, 0, 1683, v10, 0, 512 + 35, gViewX0, gViewY0, gViewX1, gViewY1);
}
#endif
2019-09-19 22:42:45 +00:00
#if 0
2021-12-29 21:56:21 +00:00
int tmpSect = nSectnum;
if ((vf0 & 0xc000) == 0x4000)
{
tmpSect = vf0 & (kMaxWalls - 1);
}
int v8 = byte_1CE5C2 > 0 && (sector[tmpSect].ceilingstat & CSTAT_SECTOR_SKY);
if (gWeather.at12d8 > 0 || v8)
{
gWeather.Draw(cX, cY, cZ, cA.asq16(), cH.asq16() + deliriumPitch, gWeather.at12d8);
if (v8)
{
gWeather.at12d8 = ClipRange(delta * 8 + gWeather.at12d8, 0, 4095);
}
else
{
gWeather.at12d8 = ClipRange(gWeather.at12d8 - delta * 64, 0, 4095);
}
}
2019-09-19 22:42:45 +00:00
#endif
2021-12-29 21:56:21 +00:00
}
UpdateDacs(0, true); // keep the view palette active only for the actual 3D view and its overlays.
MarkSectorSeen(pPlayer->actor->sector());
2021-12-29 21:56:21 +00:00
if (automapMode != am_off)
{
DrawMap(pPlayer, interpfrac);
2021-12-29 21:56:21 +00:00
}
UpdateStatusBar(pPlayer);
2021-12-29 21:56:21 +00:00
viewDrawAimedPlayerName(pPlayer);
2021-12-29 21:56:21 +00:00
if (paused)
{
auto text = GStrings("TXTB_PAUSED");
viewDrawText(PickBigFont(text), text, 160, 10, 0, 0, 1, 0);
}
else if (pPlayer->nPlayer != myconnectindex)
2021-12-29 21:56:21 +00:00
{
FStringf gTempStr("] %s [", PlayerName(pPlayer->nPlayer));
2021-12-29 21:56:21 +00:00
viewDrawText(OriginalSmallFont, gTempStr, 160, 10, 0, 0, 1, 0);
}
if (cl_interpolate)
{
RestoreInterpolations();
}
2019-09-19 22:42:45 +00:00
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2020-01-14 20:20:46 +00:00
bool GameInterface::GenerateSavePic()
{
2021-12-29 21:56:21 +00:00
viewDrawScreen(true);
return true;
2020-01-14 20:20:46 +00:00
}
std::pair<DVector3, DAngle> GameInterface::GetCoordinates()
{
PLAYER* pPlayer = &gPlayer[myconnectindex];
if (!pPlayer->actor) return std::make_pair(DVector3(DBL_MAX, 0, 0), nullAngle);
return std::make_pair(pPlayer->actor->spr.pos, pPlayer->actor->spr.angle);
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
bool GameInterface::DrawAutomapPlayer(const DVector2& mxy, const DVector2& cpos, const DAngle cang, const DVector2& xydim, const double czoom, double const interpfrac)
{
auto cangvect = cang.ToVector();
2021-12-29 21:56:21 +00:00
for (int i = connecthead; i >= 0; i = connectpoint2[i])
{
if (i == gViewIndex || gGameOptions.nGameType == 1)
2021-12-29 21:56:21 +00:00
{
2022-09-01 13:24:24 +00:00
auto actor = gPlayer[i].actor;
auto vect = OutAutomapVector(mxy - cpos, cangvect, czoom, xydim);
2022-09-01 13:24:24 +00:00
DrawTexture(twod, tileGetTexture(actor->spr.picnum, true), vect.X, vect.Y, DTA_ClipLeft, viewport3d.Left(), DTA_ClipTop, viewport3d.Top(), DTA_ScaleX, czoom * (2. / 3.), DTA_ScaleY, czoom * (2. / 3.), DTA_CenterOffset, true,
DTA_ClipRight, viewport3d.Right(), DTA_ClipBottom, viewport3d.Bottom(), DTA_Alpha, (actor->spr.cstat & CSTAT_SPRITE_TRANSLUCENT ? 0.5 : 1.), TAG_DONE);
2021-12-29 21:56:21 +00:00
}
}
return true;
}
2021-12-29 21:56:21 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void SerializeView(FSerializer& arc)
2019-09-19 22:42:45 +00:00
{
2021-12-29 21:56:21 +00:00
if (arc.BeginObject("view"))
{
arc("screentilt", gScreenTilt)
("deliriumtilt", deliriumTilt)
("deliriumturn", deliriumTurn)
("deliriumpitch", deliriumPitch)
.EndObject();
}
2019-09-19 22:42:45 +00:00
}
END_BLD_NS