2020-07-02 18:17:29 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 1996, 2003 - 3D Realms Entertainment
|
|
|
|
Copyright (C) 2000, 2003 - Matt Saettler (EDuke Enhancements)
|
|
|
|
Copyright (C) 2020 - Christoph Oelckers
|
|
|
|
|
|
|
|
This file is part of Enhanced 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
|
|
|
|
|
|
|
|
EDuke enhancements integrated: 04/13/2003 - Matt Saettler
|
|
|
|
|
|
|
|
Note: EDuke source was in transition. Changes are in-progress in the
|
|
|
|
source as it is released.
|
|
|
|
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
#include "ns.h" // Must come before everything else!
|
|
|
|
|
|
|
|
#include "v_font.h"
|
|
|
|
#include "duke3d.h"
|
|
|
|
#include "compat.h"
|
|
|
|
#include "sbar.h"
|
|
|
|
#include "v_draw.h"
|
|
|
|
#include "texturemanager.h"
|
2020-08-23 22:25:42 +00:00
|
|
|
#include "mapinfo.h"
|
2020-09-06 18:49:43 +00:00
|
|
|
#include "automap.h"
|
2020-08-23 22:25:42 +00:00
|
|
|
|
2020-07-02 18:17:29 +00:00
|
|
|
BEGIN_DUKE_NS
|
|
|
|
|
2020-10-28 19:04:53 +00:00
|
|
|
IMPLEMENT_CLASS(DDukeCommonStatusBar, true, true)
|
|
|
|
IMPLEMENT_POINTERS_START(DDukeCommonStatusBar)
|
|
|
|
IMPLEMENT_POINTER(miniFont)
|
|
|
|
IMPLEMENT_POINTER(numberFont)
|
|
|
|
IMPLEMENT_POINTER(digiFont)
|
|
|
|
IMPLEMENT_POINTER(indexFont)
|
|
|
|
IMPLEMENT_POINTERS_END
|
|
|
|
|
2020-07-02 18:17:29 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// very much a dummy to access the methods.
|
|
|
|
// The goal is to export this to a script.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
DDukeCommonStatusBar::DDukeCommonStatusBar()
|
|
|
|
{
|
|
|
|
drawOffset.Y = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Frag bar - todo
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
#if 0
|
|
|
|
void DDukeCommonStatusBar::displayfragbar(void)
|
|
|
|
{
|
|
|
|
short i, j;
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
|
|
|
|
for (i = connecthead; i >= 0; i = connectpoint2[i])
|
|
|
|
if (i > j) j = i;
|
|
|
|
|
2020-07-06 13:48:51 +00:00
|
|
|
auto tex = tileGetTexture(TILE_FRAGBAR);
|
|
|
|
for (int y = 0; y < 32; y += 8)
|
2020-08-24 18:25:53 +00:00
|
|
|
DrawTexture(twod, tex, 0, 0, DTA_FullscreenScale, FSMode_Fit320x200, DTA_ScaleX, 1.001, DTA_ScaleY, 1.001, TAG_Done);
|
2020-07-06 13:48:51 +00:00
|
|
|
|
2020-07-02 18:17:29 +00:00
|
|
|
for (i = connecthead; i >= 0; i = connectpoint2[i])
|
|
|
|
{
|
2020-10-23 19:40:49 +00:00
|
|
|
m initext(21 + (73 * (i & 3)), 2 + ((i & 28) << 1), &ud.user_name[i][0], ps[i].GetActor()->s.pal, 2 + 8 + 16 + 128);
|
2020-07-02 18:17:29 +00:00
|
|
|
sprintf(tempbuf, "%d", ps[i].frag - ps[i].fraggedself);
|
2020-10-23 19:40:49 +00:00
|
|
|
m initext(17 + 50 + (73 * (i & 3)), 2 + ((i & 28) << 1), tempbuf, ps[i].GetActor()->s.pal, 2 + 8 + 16 + 128);
|
2020-07-02 18:17:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Common inventory icon code for all styles
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2020-11-11 10:49:05 +00:00
|
|
|
std::pair<const char*, int> DDukeCommonStatusBar::ontext(struct player_struct *p)
|
2020-07-02 18:17:29 +00:00
|
|
|
{
|
2020-11-11 10:49:05 +00:00
|
|
|
std::pair<const char*, int> retval(nullptr, CR_RED);
|
2020-07-02 18:17:29 +00:00
|
|
|
|
|
|
|
int onstate = 0x80000000;
|
|
|
|
switch (p->inven_icon)
|
|
|
|
{
|
|
|
|
case ICON_HOLODUKE:
|
2020-10-21 08:48:38 +00:00
|
|
|
onstate = p->holoduke_on != nullptr;
|
2020-07-25 13:41:11 +00:00
|
|
|
break;
|
2020-07-02 18:17:29 +00:00
|
|
|
case ICON_JETPACK:
|
|
|
|
onstate = p->jetpack_on;
|
2020-07-25 13:41:11 +00:00
|
|
|
break;
|
2020-07-02 18:17:29 +00:00
|
|
|
case ICON_HEATS:
|
|
|
|
onstate = p->heat_on;
|
2020-07-25 13:41:11 +00:00
|
|
|
break;
|
2020-07-02 18:17:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Texts are intentionally not translated because the font is too small for making localization work and the translated words are too long.
|
|
|
|
if ((unsigned)onstate != 0x80000000 && !(g_gameType & (GAMEFLAG_WW2GI|GAMEFLAG_RRALL)))
|
|
|
|
{
|
2020-11-11 10:49:05 +00:00
|
|
|
retval.second = onstate > 0 ? 0 : 2;
|
2020-07-02 18:17:29 +00:00
|
|
|
retval.first = onstate > 0 ? "ON" : "OFF";
|
|
|
|
}
|
|
|
|
if (p->inven_icon >= ICON_SCUBA)
|
|
|
|
{
|
2020-11-11 10:49:05 +00:00
|
|
|
retval.second = 2;
|
2020-07-02 18:17:29 +00:00
|
|
|
retval.first = "AUTO";
|
|
|
|
}
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// draws the inventory selector
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2020-07-06 21:24:35 +00:00
|
|
|
void DDukeCommonStatusBar::DrawInventory(const struct player_struct* p, double x, double y, int align)
|
2020-07-02 18:17:29 +00:00
|
|
|
{
|
|
|
|
if (p->invdisptime <= 0)return;
|
|
|
|
|
|
|
|
int n = 0, j = 0;
|
2020-07-06 19:10:20 +00:00
|
|
|
if (p->firstaid_amount > 0) n |= 1, j++;
|
|
|
|
if (p->steroids_amount > 0) n |= 2, j++;
|
|
|
|
if (p->holoduke_amount > 0) n |= 4, j++;
|
|
|
|
if (p->jetpack_amount > 0) n |= 8, j++;
|
|
|
|
if (p->heat_amount > 0) n |= 16, j++;
|
|
|
|
if (p->scuba_amount > 0) n |= 32, j++;
|
|
|
|
if (p->boot_amount > 0) n |= 64, j++;
|
2020-07-02 18:17:29 +00:00
|
|
|
|
|
|
|
x -= (j * 11);
|
|
|
|
y -= 6;
|
|
|
|
|
|
|
|
; align |= DI_ITEM_CENTER;
|
|
|
|
for(int bit = 0; bit < 7; bit++)
|
|
|
|
{
|
|
|
|
int i = 1 << bit;
|
|
|
|
if (n & i)
|
|
|
|
{
|
|
|
|
int select = 1 << (p->inven_icon - 1);
|
|
|
|
double alpha = select == i ? 1.0 : 0.7;
|
|
|
|
DrawGraphic(tileGetTexture(item_icons[bit+1]), x, y, align, alpha, 0, 0, scale, scale);
|
2021-01-30 22:51:50 +00:00
|
|
|
if (select == i) DrawGraphic(tileGetTexture(TILE_ARROW), isWW2GI()? x + 7.5 : x, isWW2GI()? y + 0.5 : y, align, alpha, 0, 0, scale, scale);
|
2020-07-02 18:17:29 +00:00
|
|
|
x += 22;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Helper
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
PalEntry DDukeCommonStatusBar::LightForShade(int shade)
|
|
|
|
{
|
|
|
|
int ll = clamp((numshades - shade) * 255 / numshades, 0, 255);
|
|
|
|
return PalEntry(255, ll, ll, ll);
|
|
|
|
}
|
|
|
|
|
2020-07-05 22:15:22 +00:00
|
|
|
|
2020-07-02 21:56:22 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Statistics output
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void DDukeCommonStatusBar::PrintLevelStats(int bottomy)
|
|
|
|
{
|
2020-08-24 21:14:55 +00:00
|
|
|
FLevelStats stats{};
|
|
|
|
auto pp = &ps[myconnectindex];
|
|
|
|
stats.fontscale = isRR() ? 0.5 : 1.;
|
|
|
|
stats.screenbottomspace = bottomy;
|
|
|
|
stats.time = Scale(pp->player_par, 1000, REALGAMETICSPERSEC);
|
|
|
|
stats.font = SmallFont;
|
|
|
|
if (isNamWW2GI())
|
|
|
|
{
|
|
|
|
// The stock font of these games is totally unusable for this.
|
|
|
|
stats.font = ConFont;
|
|
|
|
stats.spacing = ConFont->GetHeight() + 1;
|
|
|
|
}
|
|
|
|
|
2020-08-24 17:31:43 +00:00
|
|
|
if (automapMode == am_full)
|
2020-08-23 22:25:42 +00:00
|
|
|
{
|
2020-09-29 19:47:32 +00:00
|
|
|
bool textfont = am_textfont;
|
2020-08-24 21:14:55 +00:00
|
|
|
if (!am_textfont)
|
2020-09-29 19:47:32 +00:00
|
|
|
{
|
|
|
|
// For non-English languages force use of the text font. The tiny one is simply too small to ever add localized characters to it.
|
|
|
|
auto p = GStrings["REQUIRED_CHARACTERS"];
|
|
|
|
if (p && *p) textfont = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!textfont)
|
2020-08-23 22:25:42 +00:00
|
|
|
{
|
2020-08-24 21:14:55 +00:00
|
|
|
stats.font = SmallFont2;
|
|
|
|
stats.spacing = 6;
|
2020-08-23 22:25:42 +00:00
|
|
|
}
|
2020-09-29 19:47:32 +00:00
|
|
|
else stats.spacing = stats.font->GetHeight() + 1;
|
2020-08-24 21:14:55 +00:00
|
|
|
stats.standardColor = (isNamWW2GI() && am_textfont)? CR_ORANGE : CR_UNTRANSLATED;
|
|
|
|
stats.letterColor = CR_GOLD;
|
2020-10-28 18:27:12 +00:00
|
|
|
DBaseStatusBar::PrintAutomapInfo(stats, textfont);
|
2020-08-23 22:25:42 +00:00
|
|
|
}
|
|
|
|
else if (hud_stats)
|
2020-07-02 21:56:22 +00:00
|
|
|
{
|
2020-08-23 22:25:42 +00:00
|
|
|
// JBF 20040124: display level stats in screen corner
|
2020-07-02 21:56:22 +00:00
|
|
|
|
2020-08-24 21:59:09 +00:00
|
|
|
stats.spacing = isRR() ? 10 : 7;
|
2020-07-02 21:56:22 +00:00
|
|
|
stats.kills = pp->actors_killed;
|
|
|
|
stats.maxkills = !isRR() && ud.player_skill > 3 ? -2 : pp->max_actors_killed;
|
|
|
|
stats.frags = ud.multimode > 1 && !ud.coop ? pp->frag - pp->fraggedself : -1;
|
|
|
|
stats.secrets = pp->secret_rooms;
|
|
|
|
stats.maxsecrets = pp->max_secret_rooms;
|
|
|
|
if (isNamWW2GI())
|
|
|
|
{
|
|
|
|
// The stock font of these games is totally unusable for this.
|
|
|
|
stats.letterColor = CR_ORANGE;
|
|
|
|
stats.standardColor = CR_YELLOW;
|
|
|
|
stats.completeColor = CR_FIRE;
|
|
|
|
}
|
|
|
|
else if (!isRR())
|
|
|
|
{
|
|
|
|
stats.letterColor = CR_ORANGE;
|
|
|
|
stats.standardColor = CR_CREAM;
|
|
|
|
stats.completeColor = CR_FIRE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
stats.letterColor = CR_ORANGE;
|
|
|
|
stats.standardColor =
|
|
|
|
stats.completeColor = CR_UNTRANSLATED;
|
|
|
|
}
|
2020-10-28 18:27:12 +00:00
|
|
|
DBaseStatusBar::PrintLevelStats(stats);
|
2020-07-02 21:56:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-05 22:15:22 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// 3D viewport size management
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void DrawBorder()
|
|
|
|
{
|
|
|
|
auto tex = tileGetTexture(TILE_SCREENBORDER);
|
|
|
|
if (tex != nullptr && tex->isValid())
|
|
|
|
{
|
|
|
|
if (windowxy1.y > 0)
|
|
|
|
{
|
|
|
|
twod->AddFlatFill(0, 0, twod->GetWidth(), windowxy1.y, tex, false, 1);
|
|
|
|
}
|
|
|
|
if (windowxy2.y + 1 < twod->GetHeight())
|
|
|
|
{
|
|
|
|
twod->AddFlatFill(0, windowxy2.y + 1, twod->GetWidth(), twod->GetHeight(), tex, false, 1);
|
|
|
|
}
|
|
|
|
if (windowxy1.x > 0)
|
|
|
|
{
|
|
|
|
twod->AddFlatFill(0, windowxy1.y, windowxy1.x, windowxy2.y + 1, tex, false, 1);
|
|
|
|
}
|
|
|
|
if (windowxy2.x + 1 < twod->GetWidth())
|
|
|
|
{
|
|
|
|
twod->AddFlatFill(windowxy2.x + 1, windowxy1.y, twod->GetWidth(), windowxy2.y + 1, tex, false, 1);
|
|
|
|
}
|
|
|
|
auto vb = tileGetTexture(TILE_VIEWBORDER);
|
|
|
|
auto ve = tileGetTexture(TILE_VIEWBORDER + 1);
|
|
|
|
int x1 = windowxy1.x - 4;
|
|
|
|
int y1 = windowxy1.y - 4;
|
|
|
|
int x2 = windowxy2.x + 5;
|
|
|
|
int y2 = windowxy2.y + 5;
|
|
|
|
twod->AddFlatFill(x1, y1, x2, y1 + 4, vb, 5);
|
|
|
|
twod->AddFlatFill(x1, y2 - 4, x2, y2, vb, 6);
|
|
|
|
twod->AddFlatFill(x1, y1, x1 + 4, y2, vb, 1);
|
|
|
|
twod->AddFlatFill(x2 - 4, y1, x2, y2, vb, 3);
|
|
|
|
twod->AddFlatFill(x1, y1, x1 + 4, y1 + 4, ve, 1);
|
|
|
|
twod->AddFlatFill(x2 - 4, y1, x2, y1 + 4, ve, 3);
|
|
|
|
twod->AddFlatFill(x1, y2 - 4, x1 + 4, y2, ve, 2);
|
|
|
|
twod->AddFlatFill(x2 - 4, y2 - 4, x2, y2, ve, 4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// view sizing game interface
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2020-07-15 17:48:04 +00:00
|
|
|
void GameInterface::PlayHudSound()
|
|
|
|
{
|
|
|
|
S_PlaySound(isRR() ? 341 : THUD, CHAN_AUTO, CHANF_UI);
|
|
|
|
}
|
|
|
|
|
2020-07-05 22:15:22 +00:00
|
|
|
|
2020-07-02 18:17:29 +00:00
|
|
|
END_DUKE_NS
|