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.
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
2019-09-21 18:59:54 +00:00
|
|
|
#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
|
|
|
|
2020-12-09 14:56:32 +00:00
|
|
|
#include "blood.h"
|
2020-07-31 19:05:09 +00:00
|
|
|
#include "choke.h"
|
2019-11-10 10:42:25 +00:00
|
|
|
#include "zstring.h"
|
2020-10-04 16:31:48 +00:00
|
|
|
#include "razemenu.h"
|
2019-12-09 01:01:30 +00:00
|
|
|
#include "gstrings.h"
|
2019-12-30 18:29:32 +00:00
|
|
|
#include "v_2ddrawer.h"
|
2020-01-12 19:28:07 +00:00
|
|
|
#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"
|
2021-02-16 10:36:08 +00:00
|
|
|
#include "gamefuncs.h"
|
2020-10-04 16:31:48 +00:00
|
|
|
#include "v_draw.h"
|
2021-04-11 07:59:55 +00:00
|
|
|
#include "precache.h"
|
2021-03-20 22:01:16 +00:00
|
|
|
#include "render.h"
|
2021-05-30 21:00:06 +00:00
|
|
|
#include "razefont.h"
|
2019-09-19 22:42:45 +00:00
|
|
|
|
|
|
|
|
2019-09-22 06:39:22 +00:00
|
|
|
BEGIN_BLD_NS
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
VIEW gPrevView[kMaxPlayers];
|
|
|
|
VIEWPOS gViewPos;
|
|
|
|
int gViewIndex;
|
|
|
|
|
2020-08-02 22:50:48 +00:00
|
|
|
double gInterpolate;
|
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 viewBackupView(int nPlayer)
|
|
|
|
{
|
2021-12-29 21:56:21 +00:00
|
|
|
PLAYER* pPlayer = &gPlayer[nPlayer];
|
|
|
|
VIEW* pView = &gPrevView[nPlayer];
|
2022-08-27 13:09:22 +00:00
|
|
|
pView->angle = pPlayer->angle.ang;
|
2022-01-30 17:09:26 +00:00
|
|
|
pView->x = pPlayer->actor->int_pos().X;
|
|
|
|
pView->y = pPlayer->actor->int_pos().Y;
|
2021-12-29 21:56:21 +00:00
|
|
|
pView->viewz = pPlayer->zView;
|
|
|
|
pView->weaponZ = pPlayer->zWeapon - pPlayer->zView - 0xc00;
|
|
|
|
pView->horiz = pPlayer->horizon.horiz;
|
|
|
|
pView->horizoff = pPlayer->horizon.horizoff;
|
|
|
|
pView->at2c = pPlayer->slope;
|
|
|
|
pView->bobHeight = pPlayer->bobHeight;
|
|
|
|
pView->bobWidth = pPlayer->bobWidth;
|
|
|
|
pView->shakeBobY = pPlayer->swayHeight;
|
|
|
|
pView->shakeBobX = pPlayer->swayWidth;
|
2022-08-27 11:48:57 +00:00
|
|
|
pView->look_ang = pPlayer->angle.look_ang;
|
|
|
|
pView->rotscrnang = pPlayer->angle.rotscrnang;
|
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 viewCorrectViewOffsets(int nPlayer, vec3_t const* oldpos)
|
2019-09-07 16:39:26 +00:00
|
|
|
{
|
2021-12-29 21:56:21 +00:00
|
|
|
PLAYER* pPlayer = &gPlayer[nPlayer];
|
|
|
|
VIEW* pView = &gPrevView[nPlayer];
|
2022-01-30 17:09:26 +00:00
|
|
|
pView->x += pPlayer->actor->int_pos().X - oldpos->X;
|
|
|
|
pView->y += pPlayer->actor->int_pos().Y - oldpos->Y;
|
|
|
|
pView->viewz += pPlayer->actor->int_pos().Z - oldpos->Z;
|
2019-09-07 16:39:26 +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;
|
2020-07-14 13:25:35 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//y += pFont->yoff;
|
2020-07-14 13:25:35 +00:00
|
|
|
|
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
|
|
|
}
|
2021-05-29 09:47:05 +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
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2019-12-08 23:55:30 +00:00
|
|
|
GameStats GameInterface::getStats()
|
|
|
|
{
|
2020-09-02 05:47:26 +00:00
|
|
|
return { gKillMgr.Kills, gKillMgr.TotalKills, gSecretMgr.Founds, gSecretMgr.Total, gFrameCount / kTicsPerSec, gPlayer[myconnectindex].fragCount };
|
2019-12-08 23:55:30 +00:00
|
|
|
}
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2019-10-27 16:55:14 +00:00
|
|
|
void viewDrawAimedPlayerName(void)
|
|
|
|
{
|
2021-12-29 21:56:21 +00:00
|
|
|
if (!cl_idplayers || (gView->aim.dx == 0 && gView->aim.dy == 0))
|
|
|
|
return;
|
|
|
|
|
|
|
|
int hit = HitScan(gView->actor, gView->zView, gView->aim.dx, gView->aim.dy, gView->aim.dz, CLIPMASK0, 512);
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
2019-10-27 16:55:14 +00:00
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2022-08-30 18:49:04 +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-05-30 08:56:31 +00:00
|
|
|
|
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();
|
2021-12-14 11:08:48 +00:00
|
|
|
#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++)
|
|
|
|
{
|
2022-08-30 18:49:04 +00:00
|
|
|
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;
|
2021-02-16 10:36:08 +00:00
|
|
|
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);
|
2021-02-18 10:46:36 +00:00
|
|
|
int myclock = PlayClock + MulScale(4, smoothratio, 16);
|
2021-12-29 21:56:21 +00:00
|
|
|
othercameradist = ClipHigh(othercameradist + ((myclock - othercameraclock) << 10), 65536);
|
|
|
|
othercameraclock = myclock;
|
2021-11-09 08:06:54 +00:00
|
|
|
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
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
2019-10-07 19:29:52 +00:00
|
|
|
// by NoOne: show warning msgs in game instead of throwing errors (in some cases)
|
2021-12-29 21:56:21 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2019-10-07 19:29:52 +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
|
2019-10-07 19:29:52 +00:00
|
|
|
}
|
|
|
|
|
2022-01-22 08:38:01 +00:00
|
|
|
void viewSetMessage(const char* pMessage, const char* color, const MESSAGE_PRIORITY priority)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2020-08-23 20:39:53 +00:00
|
|
|
int printlevel = priority <= MESSAGE_PRIORITY_NORMAL ? PRINT_LOW : priority < MESSAGE_PRIORITY_SYSTEM ? PRINT_MEDIUM : PRINT_HIGH;
|
2022-01-22 08:38:01 +00:00
|
|
|
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)
|
|
|
|
{
|
2019-10-11 21:31:59 +00:00
|
|
|
// 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);
|
2019-10-11 21:31:59 +00:00
|
|
|
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);
|
2020-07-27 21:29:10 +00:00
|
|
|
}
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-07-27 21:29:10 +00:00
|
|
|
void UpdateBlend()
|
|
|
|
{
|
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
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
nRed += gView->pickupEffect;
|
|
|
|
nGreen += gView->pickupEffect;
|
|
|
|
nBlue -= gView->pickupEffect;
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
nRed += ClipHigh(gView->painEffect, 85) * 2;
|
|
|
|
nGreen -= ClipHigh(gView->painEffect, 85) * 3;
|
|
|
|
nBlue -= ClipHigh(gView->painEffect, 85) * 3;
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
nRed -= gView->blindEffect;
|
|
|
|
nGreen -= gView->blindEffect;
|
|
|
|
nBlue -= gView->blindEffect;
|
2020-07-27 21:29:10 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
nRed -= gView->chokeEffect >> 6;
|
|
|
|
nGreen -= gView->chokeEffect >> 5;
|
|
|
|
nBlue -= gView->chokeEffect >> 6;
|
2020-07-27 21:29:10 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
nRed = ClipRange(nRed, -255, 255);
|
|
|
|
nGreen = ClipRange(nGreen, -255, 255);
|
|
|
|
nBlue = ClipRange(nBlue, -255, 255);
|
2020-07-27 21:29:10 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
videoTintBlood(nRed, nGreen, nBlue);
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// int gVisibility;
|
|
|
|
|
2022-08-27 08:53:34 +00:00
|
|
|
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
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
void viewUpdateDelirium(void)
|
|
|
|
{
|
2021-12-29 21:56:21 +00:00
|
|
|
gScreenTiltO = gScreenTilt;
|
|
|
|
deliriumTurnO = deliriumTurn;
|
|
|
|
deliriumPitchO = deliriumPitch;
|
2019-09-19 22:42:45 +00:00
|
|
|
int powerCount;
|
2019-10-19 19:11:39 +00:00
|
|
|
if ((powerCount = powerupCheck(gView, kPwUpDeliriumShroom)) != 0)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
|
|
|
int tilt1 = 170, tilt2 = 170, pitch = 20;
|
2022-05-05 22:05:44 +00:00
|
|
|
int timer = PlayClock * 2;
|
2019-09-19 22:42:45 +00:00
|
|
|
if (powerCount < 512)
|
|
|
|
{
|
2020-09-01 13:00:35 +00:00
|
|
|
int powerScale = IntToFixed(powerCount) / 512;
|
2021-01-04 11:36:54 +00:00
|
|
|
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;
|
2022-08-27 08:53:34 +00:00
|
|
|
gScreenTilt = DAngle::fromBuild(MulScale(sin2 + sin3, tilt1, 30));
|
2021-12-29 21:56:21 +00:00
|
|
|
int sin4 = Sin(4 * timer) >> 1;
|
2022-08-27 08:53:34 +00:00
|
|
|
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;
|
|
|
|
}
|
2022-08-27 08:53:34 +00:00
|
|
|
gScreenTilt = gScreenTilt.Normalized180();
|
2022-08-28 00:40:50 +00:00
|
|
|
if (gScreenTilt > nullAngle)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2022-08-27 08:53:34 +00:00
|
|
|
gScreenTilt -= DAngle::fromBuild(8);
|
2022-08-28 00:40:50 +00:00
|
|
|
if (gScreenTilt < nullAngle)
|
|
|
|
gScreenTilt = nullAngle;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
2022-08-28 00:40:50 +00:00
|
|
|
else if (gScreenTilt < nullAngle)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2022-08-27 08:53:34 +00:00
|
|
|
gScreenTilt += DAngle::fromBuild(8);
|
2022-08-28 00:40:50 +00:00
|
|
|
if (gScreenTilt >= nullAngle)
|
|
|
|
gScreenTilt = nullAngle;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2022-08-27 08:53:34 +00:00
|
|
|
void viewUpdateShake(int& cX, int& cY, int& cZ, 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));
|
2022-08-27 08:53:34 +00:00
|
|
|
cA += DAngle::fromBuild(QRandom2(nValue >> 8));
|
2021-12-29 21:56:21 +00:00
|
|
|
cX += QRandom2(nValue >> 4);
|
|
|
|
cY += QRandom2(nValue >> 4);
|
|
|
|
cZ += QRandom2(nValue);
|
|
|
|
pshakeX += QRandom2(nValue);
|
|
|
|
pshakeY += QRandom2(nValue);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
doEffect(gView->flickerEffect);
|
|
|
|
doEffect(gView->quakeEffect);
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int gLastPal = 0;
|
|
|
|
|
|
|
|
int32_t g_frameRate;
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2021-12-23 09:55:13 +00:00
|
|
|
static void DrawMap(DBloodActor* view)
|
2020-09-06 08:59:45 +00:00
|
|
|
{
|
2021-12-29 21:56:21 +00:00
|
|
|
int tm = 0;
|
2022-08-04 21:47:01 +00:00
|
|
|
if (viewport3d.Left() > 0)
|
2021-12-29 21:56:21 +00:00
|
|
|
{
|
|
|
|
setViewport(Hud_Stbar);
|
|
|
|
tm = 1;
|
|
|
|
}
|
|
|
|
VIEW* pView = &gPrevView[gViewIndex];
|
2022-09-20 00:31:01 +00:00
|
|
|
auto xy = DVector2(interpolatedvalue(pView->x, view->int_pos().X, gInterpolate), interpolatedvalue(pView->y, view->int_pos().Y, gInterpolate)) * inttoworld;
|
2022-08-29 10:51:30 +00:00
|
|
|
auto ang = !SyncInput() ? gView->angle.sum() : gView->angle.interpolatedsum(gInterpolate);
|
2022-09-20 00:31:01 +00:00
|
|
|
DrawOverheadMap(xy, ang, gInterpolate);
|
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
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2022-08-27 08:53:34 +00:00
|
|
|
void SetupView(int& cX, int& cY, int& cZ, DAngle& cA, fixedhoriz& cH, sectortype*& pSector, double& zDelta, double& shakeX, double& shakeY, DAngle& rotscrnang)
|
2021-03-20 22:01:16 +00:00
|
|
|
{
|
2021-12-29 21:56:21 +00:00
|
|
|
int bobWidth, bobHeight;
|
|
|
|
|
2021-12-30 15:51:56 +00:00
|
|
|
pSector = gView->actor->sector();
|
2021-11-23 23:10:48 +00:00
|
|
|
#if 0
|
2021-12-29 21:56:21 +00:00
|
|
|
if (numplayers > 1 && gView == gMe && gPrediction && gMe->actor->xspr.health > 0)
|
|
|
|
{
|
|
|
|
nSectnum = predict.sectnum;
|
|
|
|
cX = interpolatedvalue(predictOld.x, predict.x, gInterpolate);
|
|
|
|
cY = interpolatedvalue(predictOld.y, predict.y, gInterpolate);
|
|
|
|
cZ = interpolatedvalue(predictOld.viewz, predict.viewz, gInterpolate);
|
|
|
|
zDelta = interpolatedvaluef(predictOld.weaponZ, predict.weaponZ, gInterpolate);
|
|
|
|
bobWidth = interpolatedvalue(predictOld.bobWidth, predict.bobWidth, gInterpolate);
|
|
|
|
bobHeight = interpolatedvalue(predictOld.bobHeight, predict.bobHeight, gInterpolate);
|
|
|
|
shakeX = interpolatedvaluef(predictOld.shakeBobX, predict.shakeBobX, gInterpolate);
|
|
|
|
shakeY = interpolatedvaluef(predictOld.shakeBobY, predict.shakeBobY, gInterpolate);
|
|
|
|
|
|
|
|
if (!SyncInput())
|
|
|
|
{
|
|
|
|
cA = bamang(predict.angle.asbam() + predict.look_ang.asbam());
|
|
|
|
cH = predict.horiz + predict.horizoff;
|
|
|
|
rotscrnang = predict.rotscrnang;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cA = interpolatedangle(predictOld.angle + predictOld.look_ang, predict.angle + predict.look_ang, gInterpolate);
|
|
|
|
cH = interpolatedhorizon(predictOld.horiz + predictOld.horizoff, predict.horiz + predict.horizoff, gInterpolate);
|
|
|
|
rotscrnang = interpolatedangle(predictOld.rotscrnang, predict.rotscrnang, gInterpolate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2021-11-23 23:10:48 +00:00
|
|
|
#endif
|
2021-12-29 21:56:21 +00:00
|
|
|
{
|
|
|
|
VIEW* pView = &gPrevView[gViewIndex];
|
2022-01-30 17:09:26 +00:00
|
|
|
cX = interpolatedvalue(pView->x, gView->actor->int_pos().X, gInterpolate);
|
|
|
|
cY = interpolatedvalue(pView->y, gView->actor->int_pos().Y, gInterpolate);
|
2021-12-29 21:56:21 +00:00
|
|
|
cZ = interpolatedvalue(pView->viewz, gView->zView, gInterpolate);
|
|
|
|
zDelta = interpolatedvaluef(pView->weaponZ, gView->zWeapon - gView->zView - (12 << 8), gInterpolate);
|
|
|
|
bobWidth = interpolatedvalue(pView->bobWidth, gView->bobWidth, gInterpolate);
|
|
|
|
bobHeight = interpolatedvalue(pView->bobHeight, gView->bobHeight, gInterpolate);
|
|
|
|
shakeX = interpolatedvaluef(pView->shakeBobX, gView->swayWidth, gInterpolate);
|
|
|
|
shakeY = interpolatedvaluef(pView->shakeBobY, gView->swayHeight, gInterpolate);
|
|
|
|
|
|
|
|
if (!SyncInput())
|
|
|
|
{
|
2022-08-27 13:09:22 +00:00
|
|
|
cA = gView->angle.sum();
|
2021-12-29 21:56:21 +00:00
|
|
|
cH = gView->horizon.sum();
|
2022-08-27 11:48:57 +00:00
|
|
|
rotscrnang = gView->angle.rotscrnang;
|
2021-12-29 21:56:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-08-27 13:09:22 +00:00
|
|
|
cA = gView->angle.interpolatedsum(gInterpolate);
|
2021-12-29 21:56:21 +00:00
|
|
|
cH = gView->horizon.interpolatedsum(gInterpolate);
|
2022-08-27 11:48:57 +00:00
|
|
|
rotscrnang = gView->angle.interpolatedrotscrn(gInterpolate);
|
2021-12-29 21:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
viewUpdateShake(cX, cY, cZ, cA, cH, shakeX, shakeY);
|
|
|
|
cH += buildhoriz(MulScale(0x40000000 - Cos(gView->tiltEffect << 2), 30, 30));
|
|
|
|
if (gViewPos == 0)
|
|
|
|
{
|
|
|
|
if (cl_viewhbob)
|
|
|
|
{
|
2022-08-27 08:53:34 +00:00
|
|
|
cX -= MulScale(bobWidth, Sin(cA.Buildang()), 30) >> 4;
|
|
|
|
cY += MulScale(bobWidth, Cos(cA.Buildang()), 30) >> 4;
|
2021-12-29 21:56:21 +00:00
|
|
|
}
|
|
|
|
if (cl_viewvbob)
|
|
|
|
{
|
|
|
|
cZ += bobHeight;
|
|
|
|
}
|
2022-07-21 10:18:40 +00:00
|
|
|
cZ += int(cH.asq16() * (1. / 6553.6));
|
2021-12-29 21:56:21 +00:00
|
|
|
cameradist = -1;
|
|
|
|
cameraclock = PlayClock + MulScale(4, (int)gInterpolate, 16);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-08-27 08:53:34 +00:00
|
|
|
calcChaseCamPos((int*)&cX, (int*)&cY, (int*)&cZ, gView->actor, &pSector, cA, cH, gInterpolate);
|
2021-12-29 21:56:21 +00:00
|
|
|
}
|
2022-10-02 07:52:13 +00:00
|
|
|
if (pSector != nullptr)
|
|
|
|
CheckLink((int*)&cX, (int*)&cY, (int*)&cZ, &pSector);
|
2021-03-20 22:01:16 +00:00
|
|
|
}
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2021-03-20 18:20:42 +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;
|
2022-08-27 12:57:06 +00:00
|
|
|
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();
|
2021-03-20 18:20:42 +00:00
|
|
|
#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
|
|
|
{
|
2021-05-09 07:05:42 +00:00
|
|
|
if (testgotpic(2342, true))
|
2020-09-04 18:46:44 +00:00
|
|
|
{
|
|
|
|
FireProcess();
|
|
|
|
}
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
if (!paused && (!M_Active() || gGameOptions.nGameType != 0))
|
|
|
|
{
|
|
|
|
gInterpolate = !cl_interpolate || cl_capfps ? MaxSmoothRatio : I_GetTimeFrac() * MaxSmoothRatio;
|
|
|
|
}
|
|
|
|
else gInterpolate = MaxSmoothRatio;
|
|
|
|
|
|
|
|
if (cl_interpolate)
|
|
|
|
{
|
|
|
|
DoInterpolations(gInterpolate / MaxSmoothRatio);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (automapMode != am_full)
|
|
|
|
{
|
|
|
|
DoSectorLighting();
|
|
|
|
}
|
|
|
|
if (automapMode == am_off)
|
|
|
|
{
|
|
|
|
int basepal = 0;
|
|
|
|
if (powerupCheck(gView, kPwUpDeathMask) > 0) basepal = 4;
|
|
|
|
else if (powerupCheck(gView, kPwUpReflectShots) > 0) basepal = 1;
|
|
|
|
else if (gView->isUnderwater) {
|
|
|
|
if (gView->nWaterPal) basepal = gView->nWaterPal;
|
|
|
|
else {
|
|
|
|
if (gView->actor->xspr.medium == kMediumWater) basepal = 1;
|
|
|
|
else if (gView->actor->xspr.medium == kMediumGoo) basepal = 3;
|
|
|
|
else basepal = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
UpdateDacs(basepal);
|
|
|
|
UpdateBlend();
|
|
|
|
|
|
|
|
int cX, cY, cZ;
|
2022-08-27 08:53:34 +00:00
|
|
|
DAngle cA, rotscrnang;
|
2021-12-29 21:56:21 +00:00
|
|
|
fixedhoriz cH;
|
|
|
|
sectortype* pSector;
|
|
|
|
double zDelta;
|
|
|
|
double shakeX, shakeY;
|
|
|
|
SetupView(cX, cY, cZ, cA, cH, pSector, zDelta, shakeX, shakeY, rotscrnang);
|
|
|
|
|
2022-08-27 08:53:34 +00:00
|
|
|
DAngle tilt = interpolatedangle(gScreenTiltO, gScreenTilt, gInterpolate);
|
2021-12-29 21:56:21 +00:00
|
|
|
bool bDelirium = powerupCheck(gView, kPwUpDeliriumShroom) > 0;
|
|
|
|
static bool bDeliriumOld = false;
|
|
|
|
//int tiltcs, tiltdim;
|
|
|
|
uint8_t otherview = powerupCheck(gView, kPwUpCrystalBall) > 0;
|
2022-08-27 08:53:34 +00:00
|
|
|
if (tilt.Degrees() || bDelirium)
|
2021-12-29 21:56:21 +00:00
|
|
|
{
|
|
|
|
rotscrnang = tilt;
|
|
|
|
}
|
|
|
|
else if (otherview && gNetPlayers > 1)
|
|
|
|
{
|
2021-03-20 18:20:42 +00:00
|
|
|
#if 0
|
2021-12-29 21:56:21 +00:00
|
|
|
renderCrystalBall();
|
2020-03-29 06:36:39 +00:00
|
|
|
#endif
|
2021-12-29 21:56:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
othercameraclock = PlayClock + MulScale(4, (int)gInterpolate, 16);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bDelirium)
|
|
|
|
{
|
|
|
|
deliriumTilt = 0;
|
2022-08-27 08:53:34 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
2022-01-08 11:23:05 +00:00
|
|
|
g_relvisibility = (int32_t)(ClipLow(gVisibility - 32 * gView->visibility - brightness, 0)) - g_visibility;
|
2022-08-27 08:53:34 +00:00
|
|
|
cA += interpolatedangle(deliriumTurnO, deliriumTurn, gInterpolate);
|
2021-12-29 21:56:21 +00:00
|
|
|
|
2022-10-02 07:52:13 +00:00
|
|
|
if (pSector != nullptr)
|
2021-12-29 21:56:21 +00:00
|
|
|
{
|
2022-10-02 07:52:13 +00:00
|
|
|
int ceilingZ, floorZ;
|
|
|
|
getzsofslopeptr(pSector, cX, cY, &ceilingZ, &floorZ);
|
|
|
|
if ((cZ > floorZ - (1 << 8)) && (pSector->upperLink == nullptr)) // clamp to floor
|
|
|
|
{
|
|
|
|
cZ = floorZ - (1 << 8);
|
|
|
|
}
|
|
|
|
if ((cZ < ceilingZ + (1 << 8)) && (pSector->lowerLink == nullptr)) // clamp to ceiling
|
|
|
|
{
|
|
|
|
cZ = ceilingZ + (1 << 8);
|
|
|
|
}
|
2021-12-29 21:56:21 +00:00
|
|
|
}
|
2022-10-02 07:52:13 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
cH = q16horiz(ClipRange(cH.asq16(), gi->playerHorizMin(), gi->playerHorizMax()));
|
|
|
|
|
2022-08-27 08:53:34 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-03 13:50:27 +00:00
|
|
|
if (!sceneonly) hudDraw(gView, pSector, shakeX, shakeY, zDelta, basepal, gInterpolate);
|
2022-08-29 01:18:54 +00:00
|
|
|
fixedhoriz deliriumPitchI = interpolatedhorizon(q16horiz(deliriumPitchO), q16horiz(deliriumPitch), gInterpolate);
|
2022-08-03 13:50:27 +00:00
|
|
|
auto bakCstat = gView->actor->spr.cstat;
|
|
|
|
gView->actor->spr.cstat |= (gViewPos == 0) ? CSTAT_SPRITE_INVISIBLE : CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_TRANS_FLIP;
|
2022-08-27 08:53:34 +00:00
|
|
|
render_drawrooms(gView->actor, { cX, cY, cZ }, sectnum(pSector), cA, cH + deliriumPitchI, rotscrnang, gInterpolate);
|
2022-08-03 13:50:27 +00:00
|
|
|
gView->actor->spr.cstat = bakCstat;
|
2021-12-29 21:56:21 +00:00
|
|
|
bDeliriumOld = bDelirium && gDeliriumBlur;
|
|
|
|
|
|
|
|
int nClipDist = gView->actor->spr.clipdist << 2;
|
|
|
|
int vec, vf4;
|
|
|
|
Collision c1, c2;
|
|
|
|
GetZRange(gView->actor, &vf4, &c1, &vec, &c2, nClipDist, 0);
|
|
|
|
if (sceneonly) return;
|
2022-01-08 11:09:04 +00:00
|
|
|
double look_anghalf = gView->angle.look_anghalf(gInterpolate);
|
|
|
|
DrawCrosshair(kCrosshairTile, gView->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.
|
2022-09-02 05:21:54 +00:00
|
|
|
|
|
|
|
MarkSectorSeen(gView->actor->sector());
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
if (automapMode != am_off)
|
|
|
|
{
|
|
|
|
DrawMap(gView->actor);
|
|
|
|
}
|
|
|
|
UpdateStatusBar();
|
|
|
|
|
|
|
|
viewDrawAimedPlayerName();
|
|
|
|
if (paused)
|
|
|
|
{
|
|
|
|
auto text = GStrings("TXTB_PAUSED");
|
|
|
|
viewDrawText(PickBigFont(text), text, 160, 10, 0, 0, 1, 0);
|
|
|
|
}
|
|
|
|
else if (gView != gMe)
|
|
|
|
{
|
|
|
|
FStringf gTempStr("] %s [", PlayerName(gView->nPlayer));
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2022-08-21 22:08:52 +00:00
|
|
|
std::pair<DVector3, DAngle> GameInterface::GetCoordinates()
|
2019-12-23 19:55:12 +00:00
|
|
|
{
|
2022-08-21 22:08:52 +00:00
|
|
|
if (!gMe || !gMe->actor) return std::make_pair(DVector3(DBL_MAX, 0, 0), nullAngle);
|
|
|
|
return std::make_pair(gMe->actor->spr.pos, gMe->actor->spr.angle);
|
2020-09-06 19:15:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2022-09-20 00:31:01 +00:00
|
|
|
bool GameInterface::DrawAutomapPlayer(const DVector2& mxy, const DVector2& cpos, const DAngle cang, const DVector2& xydim, const double czoom, double const smoothratio)
|
2020-09-06 19:15:59 +00:00
|
|
|
{
|
2022-09-20 00:07:15 +00:00
|
|
|
auto cangvect = cang.ToVector();
|
2022-09-20 00:31:01 +00:00
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
for (int i = connecthead; i >= 0; i = connectpoint2[i])
|
|
|
|
{
|
|
|
|
if (i == gView->nPlayer || gGameOptions.nGameType == 1)
|
|
|
|
{
|
2022-09-01 13:24:24 +00:00
|
|
|
auto actor = gPlayer[i].actor;
|
2022-09-20 00:07:15 +00:00
|
|
|
auto vect = OutAutomapVector(mxy - cpos, cangvect, czoom, xydim);
|
2022-09-01 13:24:24 +00:00
|
|
|
|
2022-09-20 00:31:01 +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,
|
2022-08-04 21:47:01 +00:00
|
|
|
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;
|
2019-12-23 19:55:12 +00:00
|
|
|
}
|
|
|
|
|
2021-12-29 21:56:21 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-11-22 15:47:08 +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
|
|
|
}
|
|
|
|
|
2019-09-22 06:39:22 +00:00
|
|
|
|
|
|
|
END_BLD_NS
|