mirror of
https://github.com/nzp-team/fteqw.git
synced 2024-11-30 07:31:13 +00:00
fb86222fc7
tweaked shadowmaps. now seems faster than stencil shadows. cubemap orientation should now match other engines. tweaked terrain. rtlights work. added pvs tests for embedded terrain. sections are now saved in chunks instead, which should mean windows doesn't have a panic attack at 16 million files in a single directory. hurrah. first pass at realigning menu options to cope with variable-width fonts. still need to do pure-text items. git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@4514 fc73d0e0-1445-4013-8a0c-d673dee63da5
3322 lines
77 KiB
C
3322 lines
77 KiB
C
/*
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
|
|
|
This program 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.
|
|
|
|
*/
|
|
// sbar.c -- status bar code
|
|
|
|
#include "quakedef.h"
|
|
#include "shader.h"
|
|
|
|
extern cvar_t hud_tracking_show;
|
|
|
|
#define CON_ALTMASK (CON_2NDCHARSETTEXT|CON_WHITEMASK)
|
|
|
|
cvar_t scr_scoreboard_drawtitle = SCVAR("scr_scoreboard_drawtitle", "1");
|
|
cvar_t scr_scoreboard_forcecolors = SCVAR("scr_scoreboard_forcecolors", "0"); //damn americans
|
|
cvar_t scr_scoreboard_newstyle = SCVAR("scr_scoreboard_newstyle", "1"); // New scoreboard style ported from Electro, by Molgrum
|
|
cvar_t scr_scoreboard_showfrags = SCVAR("scr_scoreboard_showfrags", "0");
|
|
cvar_t scr_scoreboard_teamscores = SCVAR("scr_scoreboard_teamscores", "1");
|
|
cvar_t scr_scoreboard_titleseperator = SCVAR("scr_scoreboard_titleseperator", "1");
|
|
cvar_t sbar_teamstatus = SCVAR("sbar_teamstatus", "1");
|
|
|
|
//===========================================
|
|
//rogue changed and added defines
|
|
|
|
#define RIT_SHELLS 128
|
|
#define RIT_NAILS 256
|
|
#define RIT_ROCKETS 512
|
|
#define RIT_CELLS 1024
|
|
#define RIT_AXE 2048
|
|
#define RIT_LAVA_NAILGUN 4096
|
|
#define RIT_LAVA_SUPER_NAILGUN 8192
|
|
#define RIT_MULTI_GRENADE 16384
|
|
#define RIT_MULTI_ROCKET 32768
|
|
#define RIT_PLASMA_GUN 65536
|
|
#define RIT_ARMOR1 8388608
|
|
#define RIT_ARMOR2 16777216
|
|
#define RIT_ARMOR3 33554432
|
|
#define RIT_LAVA_NAILS 67108864
|
|
#define RIT_PLASMA_AMMO 134217728
|
|
#define RIT_MULTI_ROCKETS 268435456
|
|
#define RIT_SHIELD 536870912
|
|
#define RIT_ANTIGRAV 1073741824
|
|
#define RIT_SUPERHEALTH 2147483648
|
|
|
|
//===========================================
|
|
//hipnotic added defines
|
|
|
|
#define HIT_PROXIMITY_GUN_BIT 16
|
|
#define HIT_MJOLNIR_BIT 7
|
|
#define HIT_LASER_CANNON_BIT 23
|
|
#define HIT_PROXIMITY_GUN (1<<HIT_PROXIMITY_GUN_BIT)
|
|
#define HIT_MJOLNIR (1<<HIT_MJOLNIR_BIT)
|
|
#define HIT_LASER_CANNON (1<<HIT_LASER_CANNON_BIT)
|
|
#define HIT_WETSUIT (1<<(23+2))
|
|
#define HIT_EMPATHY_SHIELDS (1<<(23+3))
|
|
|
|
|
|
|
|
|
|
|
|
int sb_updates; // if >= vid.numpages, no update needed
|
|
|
|
qboolean sbar_parsingteamstatuses; //so we don't eat it if its not displayed
|
|
|
|
#define STAT_MINUS 10 // num frame for '-' stats digit
|
|
mpic_t *sb_nums[2][11];
|
|
mpic_t *sb_colon, *sb_slash;
|
|
mpic_t *sb_ibar;
|
|
mpic_t *sb_sbar;
|
|
mpic_t *sb_scorebar;
|
|
|
|
mpic_t *sb_weapons[7][8]; // 0 is active, 1 is owned, 2-5 are flashes
|
|
mpic_t *sb_ammo[4];
|
|
mpic_t *sb_sigil[4];
|
|
mpic_t *sb_armor[3];
|
|
mpic_t *sb_items[32];
|
|
|
|
mpic_t *sb_faces[7][2]; // 0 is gibbed, 1 is dead, 2-6 are alive
|
|
// 0 is static, 1 is temporary animation
|
|
mpic_t *sb_face_invis;
|
|
mpic_t *sb_face_quad;
|
|
mpic_t *sb_face_invuln;
|
|
mpic_t *sb_face_invis_invuln;
|
|
|
|
//rogue pictures.
|
|
mpic_t *rsb_invbar[2];
|
|
mpic_t *rsb_weapons[5];
|
|
mpic_t *rsb_items[2];
|
|
mpic_t *rsb_ammo[3];
|
|
mpic_t *rsb_teambord;
|
|
//all must be found for any to be used.
|
|
|
|
qboolean sb_showscores;
|
|
qboolean sb_showteamscores;
|
|
|
|
qboolean sbarfailed;
|
|
qboolean sbar_rogue;
|
|
qboolean sbar_hexen2;
|
|
|
|
vrect_t sbar_rect; //screen area that the sbar must fit.
|
|
|
|
int sb_lines; // scan lines to draw
|
|
|
|
void Sbar_DeathmatchOverlay (int start);
|
|
void Sbar_TeamOverlay (void);
|
|
static void Sbar_MiniDeathmatchOverlay (playerview_t *pv);
|
|
void Sbar_ChatModeOverlay(playerview_t *pv);
|
|
|
|
int Sbar_PlayerNum(playerview_t *pv)
|
|
{
|
|
int num;
|
|
num = cl.spectator?Cam_TrackNum(pv):-1;
|
|
if (num < 0)
|
|
num = pv->playernum;
|
|
return num;
|
|
}
|
|
|
|
int Sbar_TopColour(player_info_t *p)
|
|
{
|
|
if (scr_scoreboard_forcecolors.ival)
|
|
return p->ttopcolor;
|
|
else
|
|
return p->rtopcolor;
|
|
}
|
|
|
|
int Sbar_BottomColour(player_info_t *p)
|
|
{
|
|
if (scr_scoreboard_forcecolors.ival)
|
|
return p->tbottomcolor;
|
|
else
|
|
return p->rbottomcolor;
|
|
}
|
|
|
|
//Draws a pre-marked-up string with no width limit. doesn't support new lines
|
|
void Draw_ExpandedString(float x, float y, conchar_t *str)
|
|
{
|
|
int px, py;
|
|
Font_BeginString(font_conchar, x, y, &px, &py);
|
|
while(*str)
|
|
{
|
|
px = Font_DrawChar(px, py, *str++);
|
|
}
|
|
Font_EndString(font_conchar);
|
|
}
|
|
|
|
//Draws a marked-up string using the regular char set with no width limit. doesn't support new lines
|
|
void Draw_FunString(float x, float y, const void *str)
|
|
{
|
|
conchar_t buffer[2048];
|
|
COM_ParseFunString(CON_WHITEMASK, str, buffer, sizeof(buffer), false);
|
|
|
|
Draw_ExpandedString(x, y, buffer);
|
|
}
|
|
//Draws a marked up string using the alt char set (legacy mode would be |128)
|
|
void Draw_AltFunString(float x, float y, const void *str)
|
|
{
|
|
conchar_t buffer[2048];
|
|
COM_ParseFunString(CON_ALTMASK, str, buffer, sizeof(buffer), false);
|
|
|
|
Draw_ExpandedString(x, y, buffer);
|
|
}
|
|
|
|
//Draws a marked up string no wider than $width virtual pixels.
|
|
void Draw_FunStringWidth(float x, float y, const void *str, int width, qboolean rightalign, qboolean highlight)
|
|
{
|
|
conchar_t buffer[2048];
|
|
conchar_t *w;
|
|
int px, py;
|
|
int fw = 0;
|
|
|
|
width = (width*vid.rotpixelwidth)/vid.width;
|
|
|
|
COM_ParseFunString(highlight?CON_ALTMASK:CON_WHITEMASK, str, buffer, sizeof(buffer), false);
|
|
|
|
Font_BeginString(font_conchar, x, y, &px, &py);
|
|
if (rightalign)
|
|
{
|
|
for (w = buffer; *w; w++)
|
|
{
|
|
fw += Font_CharWidth(*w);
|
|
}
|
|
px += width;
|
|
if (fw > width)
|
|
fw = width;
|
|
px -= fw;
|
|
}
|
|
|
|
for (w = buffer; *w; w++)
|
|
{
|
|
width -= Font_CharWidth(*w);
|
|
if (width < 0)
|
|
return;
|
|
px = Font_DrawChar(px, py, *w);
|
|
}
|
|
Font_EndString(font_conchar);
|
|
}
|
|
|
|
static qboolean largegame = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef Q2CLIENT
|
|
static void DrawHUDString (char *string, float x, float y, int centerwidth, qboolean alt)
|
|
{
|
|
R_DrawTextField(x, y, centerwidth, 1024, string, alt?CON_ALTMASK:CON_WHITEMASK, CPRINT_TALIGN);
|
|
}
|
|
#define STAT_MINUS 10 // num frame for '-' stats digit
|
|
static char *q2sb_nums[2][11] =
|
|
{
|
|
{"num_0", "num_1", "num_2", "num_3", "num_4", "num_5",
|
|
"num_6", "num_7", "num_8", "num_9", "num_minus"},
|
|
{"anum_0", "anum_1", "anum_2", "anum_3", "anum_4", "anum_5",
|
|
"anum_6", "anum_7", "anum_8", "anum_9", "anum_minus"}
|
|
};
|
|
|
|
static mpic_t *Sbar_Q2CachePic(char *name)
|
|
{
|
|
return R2D_SafeCachePic(va("pics/%s.pcx", name));
|
|
}
|
|
|
|
#define ICON_WIDTH 24
|
|
#define ICON_HEIGHT 24
|
|
#define CHAR_WIDTH 16
|
|
#define ICON_SPACE 8
|
|
static void SCR_DrawField (float x, float y, int color, float width, int value)
|
|
{
|
|
char num[16], *ptr;
|
|
int l;
|
|
int frame;
|
|
mpic_t *p;
|
|
|
|
if (width < 1)
|
|
return;
|
|
|
|
// draw number string
|
|
if (width > 5)
|
|
width = 5;
|
|
|
|
snprintf (num, sizeof(num), "%i", value);
|
|
l = strlen(num);
|
|
if (l > width)
|
|
l = width;
|
|
x += 2 + CHAR_WIDTH*(width - l);
|
|
|
|
ptr = num;
|
|
while (*ptr && l)
|
|
{
|
|
if (*ptr == '-')
|
|
frame = STAT_MINUS;
|
|
else
|
|
frame = *ptr -'0';
|
|
|
|
p = Sbar_Q2CachePic(q2sb_nums[color][frame]);
|
|
if (p)
|
|
R2D_ScalePic (x,y,p->width, p->height, p);
|
|
x += CHAR_WIDTH;
|
|
ptr++;
|
|
l--;
|
|
}
|
|
}
|
|
|
|
char *Get_Q2ConfigString(int i)
|
|
{
|
|
if (i >= Q2CS_IMAGES && i < Q2CS_IMAGES + Q2MAX_IMAGES)
|
|
return cl.image_name [i-Q2CS_IMAGES];
|
|
if (i == Q2CS_STATUSBAR)
|
|
return cl.q2statusbar;
|
|
|
|
if (i >= Q2CS_MODELS && i < Q2CS_MODELS + Q2MAX_MODELS)
|
|
return cl.model_name [i-Q2CS_MODELS];
|
|
if (i >= Q2CS_SOUNDS && i < Q2CS_SOUNDS + Q2MAX_SOUNDS)
|
|
return cl.model_name [i-Q2CS_SOUNDS];
|
|
if (i == Q2CS_AIRACCEL)
|
|
return "4";
|
|
//#define Q2CS_LIGHTS (Q2CS_IMAGES +Q2MAX_IMAGES)
|
|
//#define Q2CS_ITEMS (Q2CS_LIGHTS +Q2MAX_LIGHTSTYLES)
|
|
//#define Q2CS_PLAYERSKINS (Q2CS_ITEMS +Q2MAX_ITEMS)
|
|
//#define Q2CS_GENERAL (Q2CS_PLAYERSKINS +Q2MAX_CLIENTS)
|
|
return "";
|
|
}
|
|
void Sbar_ExecuteLayoutString (char *s)
|
|
{
|
|
int x, y;
|
|
int value;
|
|
int width;
|
|
int index;
|
|
// q2clientinfo_t *ci;
|
|
mpic_t *p;
|
|
|
|
if (cls.state != ca_active)
|
|
return;
|
|
|
|
if (!s[0])
|
|
return;
|
|
|
|
x = sbar_rect.x;
|
|
y = sbar_rect.y;
|
|
width = 3;
|
|
|
|
while (s)
|
|
{
|
|
s = COM_Parse (s);
|
|
if (!strcmp(com_token, "xl"))
|
|
{
|
|
s = COM_Parse (s);
|
|
x = sbar_rect.x + atoi(com_token);
|
|
continue;
|
|
}
|
|
if (!strcmp(com_token, "xr"))
|
|
{
|
|
s = COM_Parse (s);
|
|
x = sbar_rect.x + sbar_rect.width + atoi(com_token);
|
|
continue;
|
|
}
|
|
if (!strcmp(com_token, "xv"))
|
|
{
|
|
s = COM_Parse (s);
|
|
x = sbar_rect.x + sbar_rect.width/2 - 160 + atoi(com_token);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "yt"))
|
|
{
|
|
s = COM_Parse (s);
|
|
y = sbar_rect.y + atoi(com_token);
|
|
continue;
|
|
}
|
|
if (!strcmp(com_token, "yb"))
|
|
{
|
|
s = COM_Parse (s);
|
|
y = sbar_rect.y + sbar_rect.height + atoi(com_token);
|
|
continue;
|
|
}
|
|
if (!strcmp(com_token, "yv"))
|
|
{
|
|
s = COM_Parse (s);
|
|
y = sbar_rect.y + sbar_rect.height/2 - 120 + atoi(com_token);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "pic"))
|
|
{ // draw a pic from a stat number
|
|
s = COM_Parse (s);
|
|
value = cl.q2frame.playerstate.stats[atoi(com_token)];
|
|
if (value >= Q2MAX_IMAGES || value < 0)
|
|
Host_EndGame ("Pic >= Q2MAX_IMAGES");
|
|
if (Get_Q2ConfigString(Q2CS_IMAGES+value))
|
|
{
|
|
// SCR_AddDirtyPoint (x, y);
|
|
// SCR_AddDirtyPoint (x+23, y+23);
|
|
p = Sbar_Q2CachePic(Get_Q2ConfigString(Q2CS_IMAGES+value));
|
|
if (p)
|
|
R2D_ScalePic (x, y, p->width, p->height, p);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "client"))
|
|
{ // draw a deathmatch client block
|
|
int score, ping, time;
|
|
|
|
s = COM_Parse (s);
|
|
x = sbar_rect.width/2 - 160 + atoi(com_token);
|
|
s = COM_Parse (s);
|
|
y = sbar_rect.height/2 - 120 + atoi(com_token);
|
|
// SCR_AddDirtyPoint (x, y);
|
|
// SCR_AddDirtyPoint (x+159, y+31);
|
|
|
|
s = COM_Parse (s);
|
|
value = atoi(com_token);
|
|
if (value >= MAX_CLIENTS || value < 0)
|
|
Host_EndGame ("client >= MAX_CLIENTS");
|
|
// ci = &cl.clientinfo[value];
|
|
|
|
s = COM_Parse (s);
|
|
score = atoi(com_token);
|
|
|
|
s = COM_Parse (s);
|
|
ping = atoi(com_token);
|
|
|
|
s = COM_Parse (s);
|
|
time = atoi(com_token);
|
|
|
|
Draw_AltFunString (x+32, y, cl.players[value].name);
|
|
Draw_FunString (x+32, y+8, "Score: ");
|
|
Draw_AltFunString (x+32+7*8, y+8, va("%i", score));
|
|
Draw_FunString (x+32, y+16, va("Ping: %i", ping));
|
|
Draw_FunString (x+32, y+24, va("Time: %i", time));
|
|
|
|
// if (!ci->icon)
|
|
// ci = &cl.baseclientinfo;
|
|
// Draw_Pic (x, y, R2D_SafeCachePic(ci->iconname));
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "ctf"))
|
|
{ // draw a ctf client block
|
|
int score, ping;
|
|
char block[80];
|
|
|
|
s = COM_Parse (s);
|
|
x = sbar_rect.width/2 - 160 + atoi(com_token);
|
|
s = COM_Parse (s);
|
|
y = sbar_rect.height/2 - 120 + atoi(com_token);
|
|
// SCR_AddDirtyPoint (x, y);
|
|
// SCR_AddDirtyPoint (x+159, y+31);
|
|
|
|
s = COM_Parse (s);
|
|
value = atoi(com_token);
|
|
if (value >= MAX_CLIENTS || value < 0)
|
|
Host_EndGame ("client >= MAX_CLIENTS");
|
|
// ci = &cl.clientinfo[value];
|
|
|
|
s = COM_Parse (s);
|
|
score = atoi(com_token);
|
|
|
|
s = COM_Parse (s);
|
|
ping = atoi(com_token);
|
|
if (ping > 999)
|
|
ping = 999;
|
|
|
|
sprintf(block, "%3d %3d %-12.12s", score, ping, "Player"/*ci->name*/);
|
|
|
|
// if (value == cl.playernum)
|
|
// Draw_Alt_String (x, y, block);
|
|
// else
|
|
Draw_FunString (x, y, block);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "picn"))
|
|
{ // draw a pic from a name
|
|
s = COM_Parse (s);
|
|
// SCR_AddDirtyPoint (x, y);
|
|
// SCR_AddDirtyPoint (x+23, y+23);
|
|
p = Sbar_Q2CachePic(com_token);
|
|
if (p)
|
|
R2D_ScalePic (x, y, p->width, p->height, p);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "num"))
|
|
{ // draw a number
|
|
s = COM_Parse (s);
|
|
width = atoi(com_token);
|
|
s = COM_Parse (s);
|
|
value = cl.q2frame.playerstate.stats[atoi(com_token)];
|
|
SCR_DrawField (x, y, 0, width, value);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "hnum"))
|
|
{ // health number
|
|
int color;
|
|
|
|
width = 3;
|
|
value = cl.q2frame.playerstate.stats[Q2STAT_HEALTH];
|
|
if (value > 25)
|
|
color = 0; // green
|
|
else if (value > 0)
|
|
color = (cl.q2frame.serverframe>>2) & 1; // flash
|
|
else
|
|
color = 1;
|
|
|
|
if (cl.q2frame.playerstate.stats[Q2STAT_FLASHES] & 1)
|
|
{
|
|
p = Sbar_Q2CachePic("field_3");
|
|
if (p)
|
|
R2D_ScalePic (x, y, p->width, p->height, p);
|
|
}
|
|
|
|
SCR_DrawField (x, y, color, width, value);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "anum"))
|
|
{ // ammo number
|
|
int color;
|
|
|
|
width = 3;
|
|
value = cl.q2frame.playerstate.stats[Q2STAT_AMMO];
|
|
if (value > 5)
|
|
color = 0; // green
|
|
else if (value >= 0)
|
|
color = (cl.q2frame.serverframe>>2) & 1; // flash
|
|
else
|
|
continue; // negative number = don't show
|
|
|
|
if (cl.q2frame.playerstate.stats[Q2STAT_FLASHES] & 4)
|
|
{
|
|
p = Sbar_Q2CachePic("field_3");
|
|
if (p)
|
|
R2D_ScalePic (x, y, p->width, p->height, p);
|
|
}
|
|
|
|
SCR_DrawField (x, y, color, width, value);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "rnum"))
|
|
{ // armor number
|
|
int color;
|
|
|
|
width = 3;
|
|
value = cl.q2frame.playerstate.stats[Q2STAT_ARMOR];
|
|
if (value < 1)
|
|
continue;
|
|
|
|
color = 0; // green
|
|
|
|
if (cl.q2frame.playerstate.stats[Q2STAT_FLASHES] & 2)
|
|
R2D_ScalePic (x, y, 64, 64, R2D_SafeCachePic("field_3"));
|
|
|
|
SCR_DrawField (x, y, color, width, value);
|
|
continue;
|
|
}
|
|
|
|
|
|
if (!strcmp(com_token, "stat_string"))
|
|
{
|
|
s = COM_Parse (s);
|
|
index = atoi(com_token);
|
|
if (index < 0 || index >= Q2MAX_CONFIGSTRINGS)
|
|
Host_EndGame ("Bad stat_string index");
|
|
index = cl.q2frame.playerstate.stats[index];
|
|
if (index < 0 || index >= Q2MAX_CONFIGSTRINGS)
|
|
Host_EndGame ("Bad stat_string index");
|
|
Draw_FunString (x, y, Get_Q2ConfigString(index));
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "cstring"))
|
|
{
|
|
s = COM_Parse (s);
|
|
DrawHUDString (com_token, x, y, 320, false);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "string"))
|
|
{
|
|
s = COM_Parse (s);
|
|
Draw_FunString (x, y, com_token);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "cstring2"))
|
|
{
|
|
s = COM_Parse (s);
|
|
DrawHUDString (com_token, x, y, 320, true);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "string2"))
|
|
{
|
|
s = COM_Parse (s);
|
|
Draw_AltFunString (x, y, com_token);
|
|
continue;
|
|
}
|
|
|
|
if (!strcmp(com_token, "if"))
|
|
{ // draw a number
|
|
s = COM_Parse (s);
|
|
value = cl.q2frame.playerstate.stats[atoi(com_token)];
|
|
if (!value)
|
|
{ // skip to endif
|
|
while (s && strcmp(com_token, "endif") )
|
|
{
|
|
s = COM_Parse (s);
|
|
}
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
===============
|
|
Sbar_ShowTeamScores
|
|
|
|
Tab key down
|
|
===============
|
|
*/
|
|
void Sbar_ShowTeamScores (void)
|
|
{
|
|
if (sb_showteamscores)
|
|
return;
|
|
|
|
#ifdef CSQC_DAT
|
|
if (CSQC_ConsoleCommand(Cmd_Argv(0)))
|
|
return;
|
|
#endif
|
|
|
|
sb_showteamscores = true;
|
|
sb_updates = 0;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Sbar_DontShowTeamScores
|
|
|
|
Tab key up
|
|
===============
|
|
*/
|
|
void Sbar_DontShowTeamScores (void)
|
|
{
|
|
sb_showteamscores = false;
|
|
sb_updates = 0;
|
|
|
|
#ifdef CSQC_DAT
|
|
if (CSQC_ConsoleCommand(Cmd_Argv(0)))
|
|
return;
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Sbar_ShowScores
|
|
|
|
Tab key down
|
|
===============
|
|
*/
|
|
void Sbar_ShowScores (void)
|
|
{
|
|
if (scr_scoreboard_teamscores.ival)
|
|
{
|
|
Sbar_ShowTeamScores();
|
|
return;
|
|
}
|
|
|
|
if (sb_showscores)
|
|
return;
|
|
|
|
#ifdef CSQC_DAT
|
|
if (CSQC_ConsoleCommand(Cmd_Argv(0)))
|
|
return;
|
|
#endif
|
|
|
|
sb_showscores = true;
|
|
sb_updates = 0;
|
|
}
|
|
|
|
void Sbar_Hexen2InvLeft_f(void)
|
|
{
|
|
if (cls.protocol == CP_QUAKE2)
|
|
{
|
|
CL_SendClientCommand(true, "invprev");
|
|
}
|
|
else
|
|
{
|
|
int tries = 15;
|
|
int pnum = CL_TargettedSplit(false);
|
|
playerview_t *pv = &cl.playerview[pnum];
|
|
pv->sb_hexen2_item_time = realtime;
|
|
while (tries-- > 0)
|
|
{
|
|
pv->sb_hexen2_cur_item--;
|
|
if (pv->sb_hexen2_cur_item < 0)
|
|
pv->sb_hexen2_cur_item = 14;
|
|
|
|
if (pv->stats[STAT_H2_CNT_TORCH+pv->sb_hexen2_cur_item] > 0)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
void Sbar_Hexen2InvRight_f(void)
|
|
{
|
|
if (cls.protocol == CP_QUAKE2)
|
|
{
|
|
CL_SendClientCommand(true, "invnext");
|
|
}
|
|
else
|
|
{
|
|
int tries = 15;
|
|
int pnum = CL_TargettedSplit(false);
|
|
playerview_t *pv = &cl.playerview[pnum];
|
|
pv->sb_hexen2_item_time = realtime;
|
|
while (tries-- > 0)
|
|
{
|
|
pv->sb_hexen2_cur_item++;
|
|
if (pv->sb_hexen2_cur_item > 14)
|
|
pv->sb_hexen2_cur_item = 0;
|
|
|
|
if (pv->stats[STAT_H2_CNT_TORCH+pv->sb_hexen2_cur_item] > 0)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
void Sbar_Hexen2InvUse_f(void)
|
|
{
|
|
if (cls.protocol == CP_QUAKE2)
|
|
{
|
|
CL_SendClientCommand(true, "invuse");
|
|
}
|
|
else
|
|
{
|
|
int pnum = CL_TargettedSplit(false);
|
|
playerview_t *pv = &cl.playerview[pnum];
|
|
Cmd_ExecuteString(va("impulse %d\n", 100+pv->sb_hexen2_cur_item), Cmd_ExecLevel);
|
|
}
|
|
}
|
|
void Sbar_Hexen2ShowInfo_f(void)
|
|
{
|
|
playerview_t *pv = &cl.playerview[CL_TargettedSplit(false)];
|
|
pv->sb_hexen2_extra_info = true;
|
|
}
|
|
void Sbar_Hexen2DontShowInfo_f(void)
|
|
{
|
|
playerview_t *pv = &cl.playerview[CL_TargettedSplit(false)];
|
|
pv->sb_hexen2_extra_info = false;
|
|
}
|
|
void Sbar_Hexen2PInfoPlaque_f(void)
|
|
{
|
|
playerview_t *pv = &cl.playerview[CL_TargettedSplit(false)];
|
|
pv->sb_hexen2_infoplaque = true;
|
|
}
|
|
void Sbar_Hexen2MInfoPlaque_f(void)
|
|
{
|
|
playerview_t *pv = &cl.playerview[CL_TargettedSplit(false)];
|
|
pv->sb_hexen2_infoplaque = false;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Sbar_DontShowScores
|
|
|
|
Tab key up
|
|
===============
|
|
*/
|
|
void Sbar_DontShowScores (void)
|
|
{
|
|
if (scr_scoreboard_teamscores.ival)
|
|
{
|
|
Sbar_DontShowTeamScores();
|
|
return;
|
|
}
|
|
|
|
sb_showscores = false;
|
|
sb_updates = 0;
|
|
|
|
#ifdef CSQC_DAT
|
|
if (CSQC_ConsoleCommand(Cmd_Argv(0)))
|
|
return;
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Sbar_Changed
|
|
===============
|
|
*/
|
|
void Sbar_Changed (void)
|
|
{
|
|
sb_updates = 0; // update next frame
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Sbar_Init
|
|
===============
|
|
*/
|
|
|
|
qboolean sbar_loaded;
|
|
|
|
char *failedpic;
|
|
mpic_t *Sbar_PicFromWad(char *name)
|
|
{
|
|
mpic_t *ret;
|
|
ret = R2D_SafePicFromWad(name);
|
|
|
|
if (ret)
|
|
return ret;
|
|
|
|
failedpic = name;
|
|
return NULL;
|
|
}
|
|
void Sbar_Flush (void)
|
|
{
|
|
sbar_loaded = false;
|
|
}
|
|
void Sbar_Start (void) //if one of these fails, skip the entire status bar.
|
|
{
|
|
int i;
|
|
if (sbar_loaded)
|
|
return;
|
|
|
|
sbar_loaded = true;
|
|
|
|
if (!wad_base) //the wad isn't loaded. This is an indication that it doesn't exist.
|
|
{
|
|
sbarfailed = true;
|
|
return;
|
|
}
|
|
failedpic = NULL;
|
|
|
|
sbarfailed = false;
|
|
|
|
for (i=0 ; i<10 ; i++)
|
|
{
|
|
sb_nums[0][i] = Sbar_PicFromWad (va("num_%i",i));
|
|
sb_nums[1][i] = Sbar_PicFromWad (va("anum_%i",i));
|
|
}
|
|
|
|
if (sb_nums[0][0] && sb_nums[0][0]->width < 13)
|
|
sbar_hexen2 = true;
|
|
|
|
sb_nums[0][10] = Sbar_PicFromWad ("num_minus");
|
|
sb_nums[1][10] = Sbar_PicFromWad ("anum_minus");
|
|
|
|
sb_colon = Sbar_PicFromWad ("num_colon");
|
|
sb_slash = Sbar_PicFromWad ("num_slash");
|
|
|
|
sb_weapons[0][0] = Sbar_PicFromWad ("inv_shotgun");
|
|
sb_weapons[0][1] = Sbar_PicFromWad ("inv_sshotgun");
|
|
sb_weapons[0][2] = Sbar_PicFromWad ("inv_nailgun");
|
|
sb_weapons[0][3] = Sbar_PicFromWad ("inv_snailgun");
|
|
sb_weapons[0][4] = Sbar_PicFromWad ("inv_rlaunch");
|
|
sb_weapons[0][5] = Sbar_PicFromWad ("inv_srlaunch");
|
|
sb_weapons[0][6] = Sbar_PicFromWad ("inv_lightng");
|
|
|
|
sb_weapons[1][0] = Sbar_PicFromWad ("inv2_shotgun");
|
|
sb_weapons[1][1] = Sbar_PicFromWad ("inv2_sshotgun");
|
|
sb_weapons[1][2] = Sbar_PicFromWad ("inv2_nailgun");
|
|
sb_weapons[1][3] = Sbar_PicFromWad ("inv2_snailgun");
|
|
sb_weapons[1][4] = Sbar_PicFromWad ("inv2_rlaunch");
|
|
sb_weapons[1][5] = Sbar_PicFromWad ("inv2_srlaunch");
|
|
sb_weapons[1][6] = Sbar_PicFromWad ("inv2_lightng");
|
|
|
|
for (i=0 ; i<5 ; i++)
|
|
{
|
|
sb_weapons[2+i][0] = Sbar_PicFromWad (va("inva%i_shotgun",i+1));
|
|
sb_weapons[2+i][1] = Sbar_PicFromWad (va("inva%i_sshotgun",i+1));
|
|
sb_weapons[2+i][2] = Sbar_PicFromWad (va("inva%i_nailgun",i+1));
|
|
sb_weapons[2+i][3] = Sbar_PicFromWad (va("inva%i_snailgun",i+1));
|
|
sb_weapons[2+i][4] = Sbar_PicFromWad (va("inva%i_rlaunch",i+1));
|
|
sb_weapons[2+i][5] = Sbar_PicFromWad (va("inva%i_srlaunch",i+1));
|
|
sb_weapons[2+i][6] = Sbar_PicFromWad (va("inva%i_lightng",i+1));
|
|
}
|
|
|
|
sb_ammo[0] = Sbar_PicFromWad ("sb_shells");
|
|
sb_ammo[1] = Sbar_PicFromWad ("sb_nails");
|
|
sb_ammo[2] = Sbar_PicFromWad ("sb_rocket");
|
|
sb_ammo[3] = Sbar_PicFromWad ("sb_cells");
|
|
|
|
sb_armor[0] = Sbar_PicFromWad ("sb_armor1");
|
|
sb_armor[1] = Sbar_PicFromWad ("sb_armor2");
|
|
sb_armor[2] = Sbar_PicFromWad ("sb_armor3");
|
|
|
|
sb_items[0] = Sbar_PicFromWad ("sb_key1");
|
|
sb_items[1] = Sbar_PicFromWad ("sb_key2");
|
|
sb_items[2] = Sbar_PicFromWad ("sb_invis");
|
|
sb_items[3] = Sbar_PicFromWad ("sb_invuln");
|
|
sb_items[4] = Sbar_PicFromWad ("sb_suit");
|
|
sb_items[5] = Sbar_PicFromWad ("sb_quad");
|
|
|
|
sb_sigil[0] = Sbar_PicFromWad ("sb_sigil1");
|
|
sb_sigil[1] = Sbar_PicFromWad ("sb_sigil2");
|
|
sb_sigil[2] = Sbar_PicFromWad ("sb_sigil3");
|
|
sb_sigil[3] = Sbar_PicFromWad ("sb_sigil4");
|
|
|
|
sb_faces[4][0] = Sbar_PicFromWad ("face1");
|
|
sb_faces[4][1] = Sbar_PicFromWad ("face_p1");
|
|
sb_faces[3][0] = Sbar_PicFromWad ("face2");
|
|
sb_faces[3][1] = Sbar_PicFromWad ("face_p2");
|
|
sb_faces[2][0] = Sbar_PicFromWad ("face3");
|
|
sb_faces[2][1] = Sbar_PicFromWad ("face_p3");
|
|
sb_faces[1][0] = Sbar_PicFromWad ("face4");
|
|
sb_faces[1][1] = Sbar_PicFromWad ("face_p4");
|
|
sb_faces[0][0] = Sbar_PicFromWad ("face5");
|
|
sb_faces[0][1] = Sbar_PicFromWad ("face_p5");
|
|
|
|
sb_face_invis = Sbar_PicFromWad ("face_invis");
|
|
sb_face_invuln = Sbar_PicFromWad ("face_invul2");
|
|
sb_face_invis_invuln = Sbar_PicFromWad ("face_inv2");
|
|
sb_face_quad = Sbar_PicFromWad ("face_quad");
|
|
|
|
sb_sbar = Sbar_PicFromWad ("sbar");
|
|
sb_ibar = Sbar_PicFromWad ("ibar");
|
|
sb_scorebar = Sbar_PicFromWad ("scorebar");
|
|
|
|
if (failedpic)
|
|
sbarfailed = true;
|
|
failedpic = NULL;
|
|
|
|
|
|
|
|
//try to detect rogue wads, and thus the stats we will be getting from the server.
|
|
failedpic = NULL;
|
|
|
|
rsb_invbar[0] = Sbar_PicFromWad ("r_invbar1");
|
|
rsb_invbar[1] = Sbar_PicFromWad ("r_invbar2");
|
|
|
|
rsb_weapons[0] = Sbar_PicFromWad ("r_lava");
|
|
rsb_weapons[1] = Sbar_PicFromWad ("r_superlava");
|
|
rsb_weapons[2] = Sbar_PicFromWad ("r_gren");
|
|
rsb_weapons[3] = Sbar_PicFromWad ("r_multirock");
|
|
rsb_weapons[4] = Sbar_PicFromWad ("r_plasma");
|
|
|
|
rsb_items[0] = Sbar_PicFromWad ("r_shield1");
|
|
rsb_items[1] = Sbar_PicFromWad ("r_agrav1");
|
|
|
|
rsb_teambord = Sbar_PicFromWad ("r_teambord");
|
|
|
|
rsb_ammo[0] = Sbar_PicFromWad ("r_ammolava");
|
|
rsb_ammo[1] = Sbar_PicFromWad ("r_ammomulti");
|
|
rsb_ammo[2] = Sbar_PicFromWad ("r_ammoplasma");
|
|
|
|
if (!failedpic || COM_CheckParm("-rogue"))
|
|
sbar_rogue = true;
|
|
else
|
|
sbar_rogue = false;
|
|
}
|
|
|
|
void Sbar_Init (void)
|
|
{
|
|
Cvar_Register(&scr_scoreboard_drawtitle, "Scoreboard settings");
|
|
Cvar_Register(&scr_scoreboard_forcecolors, "Scoreboard settings");
|
|
Cvar_Register(&scr_scoreboard_newstyle, "Scoreboard settings");
|
|
Cvar_Register(&scr_scoreboard_showfrags, "Scoreboard settings");
|
|
Cvar_Register(&scr_scoreboard_teamscores, "Scoreboard settings");
|
|
Cvar_Register(&scr_scoreboard_titleseperator, "Scoreboard settings");
|
|
|
|
Cvar_Register(&sbar_teamstatus, "Status bar settings");
|
|
|
|
Cmd_AddCommand ("+showscores", Sbar_ShowScores);
|
|
Cmd_AddCommand ("-showscores", Sbar_DontShowScores);
|
|
|
|
Cmd_AddCommand ("+showteamscores", Sbar_ShowTeamScores);
|
|
Cmd_AddCommand ("-showteamscores", Sbar_DontShowTeamScores);
|
|
|
|
//stuff to get hexen2 working out-of-the-box
|
|
Cmd_AddCommand ("invleft", Sbar_Hexen2InvLeft_f);
|
|
Cmd_AddCommand ("invright", Sbar_Hexen2InvRight_f);
|
|
Cmd_AddCommand ("invprev", Sbar_Hexen2InvLeft_f);
|
|
Cmd_AddCommand ("invnext", Sbar_Hexen2InvRight_f);
|
|
Cmd_AddCommand ("invuse", Sbar_Hexen2InvUse_f);
|
|
Cmd_AddCommand ("+showinfo", Sbar_Hexen2ShowInfo_f);
|
|
Cmd_AddCommand ("-showinfo", Sbar_Hexen2DontShowInfo_f);
|
|
Cmd_AddCommand ("+infoplaque", Sbar_Hexen2PInfoPlaque_f);
|
|
Cmd_AddCommand ("-infoplaque", Sbar_Hexen2MInfoPlaque_f);
|
|
Cbuf_AddText("alias +crouch \"impulse 22\"\n", RESTRICT_LOCAL);
|
|
Cbuf_AddText("alias -crouch \"impulse 22\"\n", RESTRICT_LOCAL);
|
|
}
|
|
|
|
|
|
//=============================================================================
|
|
|
|
// drawing routines are reletive to the status bar location
|
|
/*
|
|
=============
|
|
Sbar_DrawPic
|
|
=============
|
|
*/
|
|
void Sbar_DrawPic (float x, float y, float w, float h, mpic_t *pic)
|
|
{
|
|
R2D_ScalePic(sbar_rect.x + x /* + ((sbar_rect.width - 320)>>1) */, sbar_rect.y + y + (sbar_rect.height-SBAR_HEIGHT), w, h, pic);
|
|
}
|
|
|
|
/*
|
|
=============
|
|
Sbar_DrawSubPic
|
|
=============
|
|
JACK: Draws a portion of the picture in the status bar.
|
|
*/
|
|
|
|
void Sbar_DrawSubPic(float x, float y, float width, float height, mpic_t *pic, int srcx, int srcy, int srcwidth, int srcheight)
|
|
{
|
|
R2D_SubPic (sbar_rect.x + x, sbar_rect.y + y+(sbar_rect.height-SBAR_HEIGHT), width, height, pic, srcx, srcy, srcwidth, srcheight);
|
|
}
|
|
|
|
/*
|
|
================
|
|
Sbar_DrawCharacter
|
|
|
|
Draws one solid graphics character
|
|
================
|
|
*/
|
|
void Sbar_DrawCharacter (float x, float y, int num)
|
|
{
|
|
int px, py;
|
|
Font_BeginString(font_conchar, sbar_rect.x + x + 4, sbar_rect.y + y + sbar_rect.height-SBAR_HEIGHT, &px, &py);
|
|
Font_DrawChar(px, py, num | 0xe000 | CON_WHITEMASK);
|
|
Font_EndString(font_conchar);
|
|
}
|
|
|
|
/*
|
|
================
|
|
Sbar_DrawString
|
|
================
|
|
*/
|
|
void Sbar_DrawString (float x, float y, char *str)
|
|
{
|
|
Draw_FunString (sbar_rect.x + x /*+ ((sbar_rect.width - 320)>>1) */, sbar_rect.y + y+ sbar_rect.height-SBAR_HEIGHT, str);
|
|
}
|
|
|
|
void Sbar_DrawExpandedString (float x, float y, conchar_t *str)
|
|
{
|
|
Draw_ExpandedString (sbar_rect.x + x /*+ ((sbar_rect.width - 320)>>1) */, sbar_rect.y + y+ sbar_rect.height-SBAR_HEIGHT, str);
|
|
}
|
|
|
|
void Draw_TinyString (float x, float y, const qbyte *str)
|
|
{
|
|
float xstart;
|
|
int px, py;
|
|
|
|
if (!font_tiny)
|
|
{
|
|
font_tiny = Font_LoadFont(6, "gfx/tinyfont");
|
|
if (!font_tiny)
|
|
return;
|
|
}
|
|
|
|
Font_BeginString(font_tiny, x, y, &px, &py);
|
|
xstart = px;
|
|
|
|
while (*str)
|
|
{
|
|
if (*str == '\n')
|
|
{
|
|
px = xstart;
|
|
py += Font_CharHeight();
|
|
str++;
|
|
continue;
|
|
}
|
|
px = Font_DrawChar(px, py, CON_WHITEMASK|*str++);
|
|
}
|
|
Font_EndString(font_tiny);
|
|
}
|
|
void Sbar_DrawTinyString (float x, float y, char *str)
|
|
{
|
|
Draw_TinyString (sbar_rect.x + x /*+ ((sbar_rect.width - 320)>>1) */, sbar_rect.y + y+ sbar_rect.height-SBAR_HEIGHT, str);
|
|
}
|
|
|
|
void Sbar_FillPC (float x, float y, float w, float h, unsigned int pcolour)
|
|
{
|
|
if (pcolour >= 16)
|
|
{
|
|
R2D_ImageColours (((pcolour&0xff0000)>>16)/255.0f, ((pcolour&0xff00)>>8)/255.0f, (pcolour&0xff)/255.0f, 1.0);
|
|
R2D_FillBlock (x, y, w, h);
|
|
}
|
|
else
|
|
{
|
|
R2D_ImagePaletteColour(Sbar_ColorForMap(pcolour), 1.0);
|
|
R2D_FillBlock (x, y, w, h);
|
|
}
|
|
}
|
|
static void Sbar_FillPCDark (float x, float y, float w, float h, unsigned int pcolour)
|
|
{
|
|
if (pcolour >= 16)
|
|
{
|
|
R2D_ImageColours (((pcolour&0xff0000)>>16)/1024.0f, ((pcolour&0xff00)>>8)/1024.0f, (pcolour&0xff)/1024.0f, 1.0);
|
|
R2D_FillBlock (x, y, w, h);
|
|
}
|
|
else
|
|
{
|
|
R2D_ImagePaletteColour(Sbar_ColorForMap(pcolour)-1, 1.0);
|
|
R2D_FillBlock (x, y, w, h);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
=============
|
|
Sbar_itoa
|
|
=============
|
|
*/
|
|
int Sbar_itoa (int num, char *buf)
|
|
{
|
|
char *str;
|
|
int pow10;
|
|
int dig;
|
|
|
|
str = buf;
|
|
|
|
if (num < 0)
|
|
{
|
|
*str++ = '-';
|
|
num = -num;
|
|
}
|
|
|
|
for (pow10 = 10 ; num >= pow10 ; pow10 *= 10)
|
|
;
|
|
|
|
do
|
|
{
|
|
pow10 /= 10;
|
|
dig = num/pow10;
|
|
*str++ = '0'+dig;
|
|
num -= dig*pow10;
|
|
} while (pow10 != 1);
|
|
|
|
*str = 0;
|
|
|
|
return str-buf;
|
|
}
|
|
|
|
|
|
/*
|
|
=============
|
|
Sbar_DrawNum
|
|
=============
|
|
*/
|
|
void Sbar_DrawNum (float x, float y, int num, int digits, int color)
|
|
{
|
|
char str[12];
|
|
char *ptr;
|
|
int l, frame;
|
|
#undef small
|
|
int small=false;
|
|
|
|
if (digits < 0)
|
|
{
|
|
small = true;
|
|
digits*=-1;
|
|
}
|
|
|
|
l = Sbar_itoa (num, str);
|
|
ptr = str;
|
|
if (l > digits)
|
|
ptr += (l-digits);
|
|
|
|
if (small)
|
|
{
|
|
if (l < digits)
|
|
x += (digits-l)*8;
|
|
while (*ptr)
|
|
{
|
|
Sbar_DrawCharacter(x, y, *ptr+18 - '0');
|
|
ptr++;
|
|
x+=8;
|
|
}
|
|
return;
|
|
}
|
|
if (l < digits)
|
|
x += (digits-l)*24;
|
|
|
|
while (*ptr)
|
|
{
|
|
if (*ptr == '-')
|
|
frame = STAT_MINUS;
|
|
else
|
|
frame = *ptr -'0';
|
|
|
|
Sbar_DrawPic (x, y, 24, 24, sb_nums[color][frame]);
|
|
x += 24;
|
|
ptr++;
|
|
}
|
|
}
|
|
|
|
void Sbar_Hexen2DrawNum (float x, float y, int num, int digits)
|
|
{
|
|
char str[12];
|
|
char *ptr;
|
|
int l, frame;
|
|
|
|
l = Sbar_itoa (num, str);
|
|
ptr = str;
|
|
if (l > digits)
|
|
ptr += (l-digits);
|
|
|
|
//hexen2 hud has it centered
|
|
if (l < digits)
|
|
x += ((digits-l)*13)/2;
|
|
|
|
while (*ptr)
|
|
{
|
|
if (*ptr == '-')
|
|
frame = STAT_MINUS;
|
|
else
|
|
frame = *ptr -'0';
|
|
|
|
Sbar_DrawPic (x, y, 12, 16, sb_nums[0][frame]);
|
|
x += 13;
|
|
ptr++;
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
int fragsort[MAX_CLIENTS];
|
|
int playerteam[MAX_CLIENTS];
|
|
int scoreboardlines;
|
|
typedef struct {
|
|
char team[16+1];
|
|
int frags;
|
|
int players;
|
|
int plow, phigh, ptotal;
|
|
int topcolour, bottomcolour;
|
|
qboolean ownteam;
|
|
} team_t;
|
|
team_t teams[MAX_CLIENTS];
|
|
int teamsort[MAX_CLIENTS];
|
|
int scoreboardteams;
|
|
|
|
struct
|
|
{
|
|
unsigned char upper;
|
|
short frags;
|
|
} nqteam[14];
|
|
void Sbar_PQ_Team_New(unsigned int lower, unsigned int upper)
|
|
{
|
|
if (lower >= 14)
|
|
return;
|
|
nqteam[lower].upper = upper;
|
|
}
|
|
void Sbar_PQ_Team_Frags(unsigned int lower, int frags)
|
|
{
|
|
if (lower >= 14)
|
|
return;
|
|
nqteam[lower].frags = frags;
|
|
}
|
|
void Sbar_PQ_Team_Reset(void)
|
|
{
|
|
memset(nqteam, 0, sizeof(nqteam));
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Sbar_SortFrags
|
|
===============
|
|
*/
|
|
void Sbar_SortFrags (qboolean includespec, qboolean teamsort)
|
|
{
|
|
int i, j, k;
|
|
|
|
if (!cl.teamplay)
|
|
teamsort = false;
|
|
|
|
// sort by frags
|
|
scoreboardlines = 0;
|
|
for (i=0 ; i<MAX_CLIENTS ; i++)
|
|
{
|
|
if (cl.players[i].name[0] &&
|
|
(!cl.players[i].spectator || includespec))
|
|
{
|
|
fragsort[scoreboardlines] = i;
|
|
scoreboardlines++;
|
|
if (cl.players[i].spectator)
|
|
cl.players[i].frags = -999;
|
|
}
|
|
}
|
|
|
|
for (i=0 ; i<scoreboardlines ; i++)
|
|
for (j=0 ; j<scoreboardlines-1-i ; j++)
|
|
{
|
|
int t1 = playerteam[fragsort[j]];
|
|
int t2 = playerteam[fragsort[j+1]];
|
|
if (!teamsort || t1 == t2)
|
|
{
|
|
if (cl.players[fragsort[j]].frags < cl.players[fragsort[j+1]].frags)
|
|
{
|
|
k = fragsort[j];
|
|
fragsort[j] = fragsort[j+1];
|
|
fragsort[j+1] = k;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (t1 == -1)
|
|
t1 = MAX_CLIENTS;
|
|
if (t2 == -1)
|
|
t2 = MAX_CLIENTS;
|
|
if (t1 > t2)
|
|
{
|
|
k = fragsort[j];
|
|
fragsort[j] = fragsort[j+1];
|
|
fragsort[j+1] = k;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Sbar_SortTeams (playerview_t *pv)
|
|
{
|
|
int i, j, k;
|
|
player_info_t *s;
|
|
char t[16+1];
|
|
int ownnum;
|
|
|
|
// request new ping times every two second
|
|
scoreboardteams = 0;
|
|
|
|
if (!cl.teamplay)
|
|
return;
|
|
|
|
memset(teams, 0, sizeof(teams));
|
|
// sort the teams
|
|
for (i = 0; i < cl.allocated_client_slots; i++)
|
|
teams[i].plow = 999;
|
|
|
|
ownnum = Sbar_PlayerNum(pv);
|
|
|
|
for (i = 0; i < cl.allocated_client_slots; i++)
|
|
{
|
|
playerteam[i] = -1;
|
|
s = &cl.players[i];
|
|
if (!s->name[0] || s->spectator)
|
|
continue;
|
|
|
|
// find his team in the list
|
|
Q_strncpyz(t, s->team, sizeof(t));
|
|
if (!t[0])
|
|
continue; // not on team
|
|
if (cls.protocol == CP_NETQUAKE)
|
|
{
|
|
k = Sbar_BottomColour(s);
|
|
if (!k) //team 0 = spectator
|
|
continue;
|
|
for (j = 0; j < scoreboardteams; j++)
|
|
if (teams[j].bottomcolour == k)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (j = 0; j < scoreboardteams; j++)
|
|
if (!strcmp(teams[j].team, t))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*if (cl.teamfortress)
|
|
{
|
|
teams[j].topcolour = teams[j].bottomcolour = TF_TeamToColour(t);
|
|
}
|
|
else*/ if (j == scoreboardteams || i == ownnum)
|
|
{
|
|
teams[j].topcolour = Sbar_TopColour(s);
|
|
teams[j].bottomcolour = Sbar_BottomColour(s);
|
|
}
|
|
|
|
if (j == scoreboardteams)
|
|
{ // create a team for this player
|
|
scoreboardteams++;
|
|
strcpy(teams[j].team, t);
|
|
}
|
|
|
|
playerteam[i] = j;
|
|
teams[j].frags += s->frags;
|
|
teams[j].players++;
|
|
|
|
if (teams[j].plow > s->ping)
|
|
teams[j].plow = s->ping;
|
|
if (teams[j].phigh < s->ping)
|
|
teams[j].phigh = s->ping;
|
|
teams[j].ptotal += s->ping;
|
|
}
|
|
|
|
// sort
|
|
for (i = 0; i < scoreboardteams; i++)
|
|
teamsort[i] = i;
|
|
|
|
// good 'ol bubble sort
|
|
for (i = 0; i < scoreboardteams - 1; i++)
|
|
for (j = i + 1; j < scoreboardteams; j++)
|
|
if (teams[teamsort[i]].frags < teams[teamsort[j]].frags)
|
|
{
|
|
k = teamsort[i];
|
|
teamsort[i] = teamsort[j];
|
|
teamsort[j] = k;
|
|
}
|
|
}
|
|
|
|
unsigned int Sbar_ColorForMap (unsigned int m)
|
|
{
|
|
if (m >= 16)
|
|
return m;
|
|
|
|
m = (m < 0) ? 0 : ((m > 13) ? 13 : m);
|
|
|
|
m *= 16;
|
|
return m < 128 ? m + 8 : m + 8;
|
|
}
|
|
|
|
|
|
/*
|
|
===============
|
|
Sbar_SoloScoreboard
|
|
===============
|
|
*/
|
|
void Sbar_SoloScoreboard (void)
|
|
{
|
|
int l;
|
|
float time;
|
|
char str[80];
|
|
int minutes, seconds, tens, units;
|
|
|
|
Sbar_DrawPic (0, 0, 320, 24, sb_scorebar);
|
|
|
|
// time
|
|
time = cl.servertime;
|
|
minutes = time / 60;
|
|
seconds = time - 60*minutes;
|
|
tens = seconds / 10;
|
|
units = seconds - 10*tens;
|
|
sprintf (str,"Time :%3i:%i%i", minutes, tens, units);
|
|
Sbar_DrawString (184, 4, str);
|
|
|
|
// draw level name
|
|
l = strlen (cl.levelname);
|
|
Sbar_DrawString (232 - l*4, 12, cl.levelname);
|
|
}
|
|
|
|
void Sbar_CoopScoreboard (void)
|
|
{
|
|
float time;
|
|
char str[80];
|
|
int minutes, seconds, tens, units;
|
|
int l;
|
|
int pnum = 0; //doesn't matter, should all be the same
|
|
|
|
sprintf (str,"Monsters:%3i /%3i", cl.playerview[pnum].stats[STAT_MONSTERS], cl.playerview[pnum].stats[STAT_TOTALMONSTERS]);
|
|
Sbar_DrawString (8, 4, str);
|
|
|
|
sprintf (str,"Secrets :%3i /%3i", cl.playerview[pnum].stats[STAT_SECRETS], cl.playerview[pnum].stats[STAT_TOTALSECRETS]);
|
|
Sbar_DrawString (8, 12, str);
|
|
|
|
// time
|
|
time = cl.servertime;
|
|
minutes = time / 60;
|
|
seconds = time - 60*minutes;
|
|
tens = seconds / 10;
|
|
units = seconds - 10*tens;
|
|
sprintf (str,"Time :%3i:%i%i", minutes, tens, units);
|
|
Sbar_DrawString (184, 4, str);
|
|
|
|
// draw level name
|
|
l = strlen (cl.levelname);
|
|
Sbar_DrawString (232 - l*4, 12, cl.levelname);
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
/*
|
|
===============
|
|
Sbar_DrawInventory
|
|
===============
|
|
*/
|
|
void Sbar_DrawInventory (playerview_t *pv)
|
|
{
|
|
int i;
|
|
char num[6];
|
|
conchar_t numc[6];
|
|
float time;
|
|
int flashon;
|
|
qboolean headsup;
|
|
qboolean hudswap;
|
|
|
|
headsup = !(cl_sbar.value || (scr_viewsize.value<100&&cl.splitclients==1));
|
|
hudswap = cl_hudswap.value; // Get that nasty float out :)
|
|
|
|
if (!headsup)
|
|
{
|
|
if (sbar_rogue)
|
|
{
|
|
if ( pv->stats[STAT_ACTIVEWEAPON] >= RIT_LAVA_NAILGUN )
|
|
Sbar_DrawPic (0, -24, 320, 24, rsb_invbar[0]);
|
|
else
|
|
Sbar_DrawPic (0, -24, 320, 24, rsb_invbar[1]);
|
|
}
|
|
else
|
|
Sbar_DrawPic (0, -24, 320, 24, sb_ibar);
|
|
}
|
|
// weapons
|
|
for (i=0 ; i<7 ; i++)
|
|
{
|
|
if (pv->stats[STAT_ITEMS] & (IT_SHOTGUN<<i) )
|
|
{
|
|
time = pv->item_gettime[i];
|
|
flashon = (int)((cl.time - time)*10);
|
|
if (flashon < 0)
|
|
flashon = 0;
|
|
if (flashon >= 10)
|
|
{
|
|
if ( pv->stats[STAT_ACTIVEWEAPON] == (IT_SHOTGUN<<i) )
|
|
flashon = 1;
|
|
else
|
|
flashon = 0;
|
|
}
|
|
else
|
|
flashon = (flashon%5) + 2;
|
|
|
|
if (headsup)
|
|
{
|
|
if (i || sbar_rect.height>200)
|
|
Sbar_DrawSubPic ((hudswap) ? 0 : (sbar_rect.width-24),-68-(7-i)*16, 24,16, sb_weapons[flashon][i],0,0,(i==6)?48:24, 16);
|
|
}
|
|
else
|
|
{
|
|
Sbar_DrawPic (i*24, -16, (i==6)?48:24, 16, sb_weapons[flashon][i]);
|
|
}
|
|
|
|
if (flashon > 1)
|
|
sb_updates = 0; // force update to remove flash
|
|
}
|
|
}
|
|
|
|
if (sbar_rogue)
|
|
{
|
|
// check for powered up weapon.
|
|
if ( pv->stats[STAT_ACTIVEWEAPON] >= RIT_LAVA_NAILGUN )
|
|
{
|
|
for (i=0;i<5;i++)
|
|
{
|
|
if (pv->stats[STAT_ACTIVEWEAPON] == (RIT_LAVA_NAILGUN << i))
|
|
{
|
|
if (headsup)
|
|
{
|
|
if (sbar_rect.height>200)
|
|
Sbar_DrawSubPic ((hudswap) ? 0 : (sbar_rect.width-24),-68-(5-i)*16, 24, 16, rsb_weapons[i],0,0,((i==4)?48:24),16);
|
|
|
|
}
|
|
else
|
|
Sbar_DrawPic ((i+2)*24, -16, (i==4)?48:24, 16, rsb_weapons[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ammo counts
|
|
if (headsup)
|
|
{
|
|
for (i=0 ; i<4 ; i++)
|
|
{
|
|
Sbar_DrawSubPic((hudswap) ? 0 : (sbar_rect.width-42), -24 - (4-i)*11, 42, 11, sb_ibar, 3+(i*48), 0, 320, 24);
|
|
}
|
|
}
|
|
for (i=0 ; i<4 ; i++)
|
|
{
|
|
snprintf (num, sizeof(num), "%3i", pv->stats[STAT_SHELLS+i] );
|
|
numc[0] = CON_WHITEMASK|0xe000|((num[0]!=' ')?(num[0] + 18-'0'):' ');
|
|
numc[1] = CON_WHITEMASK|0xe000|((num[1]!=' ')?(num[1] + 18-'0'):' ');
|
|
numc[2] = CON_WHITEMASK|0xe000|((num[2]!=' ')?(num[2] + 18-'0'):' ');
|
|
numc[3] = 0;
|
|
if (headsup)
|
|
{
|
|
Sbar_DrawExpandedString((hudswap) ? 3 : (sbar_rect.width-39), -24 - (4-i)*11, numc);
|
|
}
|
|
else
|
|
{
|
|
Sbar_DrawExpandedString((6*i+1)*8 - 2, -24, numc);
|
|
}
|
|
}
|
|
|
|
flashon = 0;
|
|
// items
|
|
for (i=0 ; i<6 ; i++)
|
|
{
|
|
if (pv->stats[STAT_ITEMS] & (1<<(17+i)))
|
|
{
|
|
time = pv->item_gettime[17+i];
|
|
if (time && time > cl.time - 2 && flashon )
|
|
{ // flash frame
|
|
sb_updates = 0;
|
|
}
|
|
else
|
|
Sbar_DrawPic (192 + i*16, -16, 16, 16, sb_items[i]);
|
|
if (time && time > cl.time - 2)
|
|
sb_updates = 0;
|
|
}
|
|
}
|
|
|
|
if (sbar_rogue)
|
|
{
|
|
// new rogue items
|
|
for (i=0 ; i<2 ; i++)
|
|
{
|
|
if (pv->stats[STAT_ITEMS] & (1<<(29+i)))
|
|
{
|
|
time = pv->item_gettime[29+i];
|
|
|
|
if (time && time > cl.time - 2 && flashon )
|
|
{ // flash frame
|
|
sb_updates = 0;
|
|
}
|
|
else
|
|
{
|
|
Sbar_DrawPic (288 + i*16, -16, 16, 16, rsb_items[i]);
|
|
}
|
|
|
|
if (time && time > cl.time - 2)
|
|
sb_updates = 0;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// sigils
|
|
for (i=0 ; i<4 ; i++)
|
|
{
|
|
if (pv->stats[STAT_ITEMS] & (1<<(28+i)))
|
|
{
|
|
time = pv->item_gettime[28+i];
|
|
if (time && time > cl.time - 2 && flashon )
|
|
{ // flash frame
|
|
sb_updates = 0;
|
|
}
|
|
else
|
|
Sbar_DrawPic (320-32 + i*8, -16, 8, 16, sb_sigil[i]);
|
|
if (time && time > cl.time - 2)
|
|
sb_updates = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
/*
|
|
===============
|
|
Sbar_DrawFrags
|
|
===============
|
|
*/
|
|
void Sbar_DrawFrags (playerview_t *pv)
|
|
{
|
|
int i, k, l;
|
|
int top, bottom;
|
|
float x, y;
|
|
int f;
|
|
int ownnum;
|
|
char num[12];
|
|
player_info_t *s;
|
|
|
|
Sbar_SortFrags (false, false);
|
|
|
|
ownnum = Sbar_PlayerNum(pv);
|
|
|
|
// draw the text
|
|
l = scoreboardlines <= 4 ? scoreboardlines : 4;
|
|
|
|
x = 23;
|
|
// xofs = (sbar_rect.width - 320)>>1;
|
|
y = sbar_rect.height - SBAR_HEIGHT - 23;
|
|
|
|
for (i=0 ; i<l ; i++)
|
|
{
|
|
k = fragsort[i];
|
|
s = &cl.players[k];
|
|
if (!s->name[0])
|
|
continue;
|
|
if (s->spectator)
|
|
continue;
|
|
|
|
// draw background
|
|
top = Sbar_TopColour(s);
|
|
bottom = Sbar_BottomColour(s);
|
|
|
|
|
|
// Draw_Fill (xofs + x*8 + 10, y, 28, 4, top);
|
|
// Draw_Fill (xofs + x*8 + 10, y+4, 28, 3, bottom);
|
|
Sbar_FillPC (sbar_rect.x+x*8 + 10, sbar_rect.y+y, 28, 4, top);
|
|
Sbar_FillPC (sbar_rect.x+x*8 + 10, sbar_rect.y+y+4, 28, 3, bottom);
|
|
|
|
// draw number
|
|
f = s->frags;
|
|
sprintf (num, "%3i",f);
|
|
|
|
Sbar_DrawCharacter ( (x+1)*8 , -24, num[0]);
|
|
Sbar_DrawCharacter ( (x+2)*8 , -24, num[1]);
|
|
Sbar_DrawCharacter ( (x+3)*8 , -24, num[2]);
|
|
|
|
if (k == ownnum)
|
|
{
|
|
Sbar_DrawCharacter (x*8+2, -24, 16);
|
|
Sbar_DrawCharacter ( (x+4)*8-4, -24, 17);
|
|
}
|
|
x+=4;
|
|
}
|
|
R2D_ImageColours(1.0, 1.0, 1.0, 1.0);
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
|
|
/*
|
|
===============
|
|
Sbar_DrawFace
|
|
===============
|
|
*/
|
|
void Sbar_DrawFace (playerview_t *pv)
|
|
{
|
|
int f, anim;
|
|
|
|
if ( (pv->stats[STAT_ITEMS] & (IT_INVISIBILITY | IT_INVULNERABILITY) )
|
|
== (IT_INVISIBILITY | IT_INVULNERABILITY) )
|
|
{
|
|
Sbar_DrawPic (112, 0, 24, 24, sb_face_invis_invuln);
|
|
return;
|
|
}
|
|
if (pv->stats[STAT_ITEMS] & IT_QUAD)
|
|
{
|
|
Sbar_DrawPic (112, 0, 24, 24, sb_face_quad );
|
|
return;
|
|
}
|
|
if (pv->stats[STAT_ITEMS] & IT_INVISIBILITY)
|
|
{
|
|
Sbar_DrawPic (112, 0, 24, 24, sb_face_invis );
|
|
return;
|
|
}
|
|
if (pv->stats[STAT_ITEMS] & IT_INVULNERABILITY)
|
|
{
|
|
Sbar_DrawPic (112, 0, 24, 24, sb_face_invuln);
|
|
return;
|
|
}
|
|
|
|
if (pv->stats[STAT_HEALTH] >= 100)
|
|
f = 4;
|
|
else
|
|
f = pv->stats[STAT_HEALTH] / 20;
|
|
|
|
if (f < 0)
|
|
f=0;
|
|
|
|
if (cl.time <= pv->faceanimtime)
|
|
{
|
|
anim = 1;
|
|
sb_updates = 0; // make sure the anim gets drawn over
|
|
}
|
|
else
|
|
anim = 0;
|
|
Sbar_DrawPic (112, 0, 24, 24, sb_faces[f][anim]);
|
|
}
|
|
|
|
/*
|
|
=============
|
|
Sbar_DrawNormal
|
|
=============
|
|
*/
|
|
void Sbar_DrawNormal (playerview_t *pv)
|
|
{
|
|
if (cl_sbar.value || (scr_viewsize.value<100&&cl.splitclients==1))
|
|
Sbar_DrawPic (0, 0, 320, 24, sb_sbar);
|
|
|
|
// armor
|
|
if (pv->stats[STAT_ITEMS] & IT_INVULNERABILITY)
|
|
{
|
|
Sbar_DrawNum (24, 0, 666, 3, 1);
|
|
Sbar_DrawPic (0, 0, 24, 24, draw_disc);
|
|
}
|
|
else
|
|
{
|
|
if (sbar_rogue)
|
|
{
|
|
Sbar_DrawNum (24, 0, pv->stats[STAT_ARMOR], 3,
|
|
pv->stats[STAT_ARMOR] <= 25);
|
|
if (pv->stats[STAT_ITEMS] & RIT_ARMOR3)
|
|
Sbar_DrawPic (0, 0, 24, 24, sb_armor[2]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_ARMOR2)
|
|
Sbar_DrawPic (0, 0, 24, 24, sb_armor[1]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_ARMOR1)
|
|
Sbar_DrawPic (0, 0, 24, 24, sb_armor[0]);
|
|
}
|
|
else
|
|
{
|
|
Sbar_DrawNum (24, 0, pv->stats[STAT_ARMOR], 3,
|
|
pv->stats[STAT_ARMOR] <= 25);
|
|
if (pv->stats[STAT_ITEMS] & IT_ARMOR3)
|
|
Sbar_DrawPic (0, 0, 24, 24, sb_armor[2]);
|
|
else if (pv->stats[STAT_ITEMS] & IT_ARMOR2)
|
|
Sbar_DrawPic (0, 0, 24, 24, sb_armor[1]);
|
|
else if (pv->stats[STAT_ITEMS] & IT_ARMOR1)
|
|
Sbar_DrawPic (0, 0, 24, 24, sb_armor[0]);
|
|
}
|
|
}
|
|
|
|
// face
|
|
Sbar_DrawFace (pv);
|
|
|
|
// health
|
|
Sbar_DrawNum (136, 0, pv->stats[STAT_HEALTH], 3
|
|
, pv->stats[STAT_HEALTH] <= 25);
|
|
|
|
// ammo icon
|
|
if (sbar_rogue)
|
|
{
|
|
if (pv->stats[STAT_ITEMS] & RIT_SHELLS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[0]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_NAILS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[1]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_ROCKETS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[2]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_CELLS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[3]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_LAVA_NAILS)
|
|
Sbar_DrawPic (224, 0, 24, 24, rsb_ammo[0]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_PLASMA_AMMO)
|
|
Sbar_DrawPic (224, 0, 24, 24, rsb_ammo[1]);
|
|
else if (pv->stats[STAT_ITEMS] & RIT_MULTI_ROCKETS)
|
|
Sbar_DrawPic (224, 0, 24, 24, rsb_ammo[2]);
|
|
}
|
|
else
|
|
{
|
|
if (pv->stats[STAT_ITEMS] & IT_SHELLS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[0]);
|
|
else if (pv->stats[STAT_ITEMS] & IT_NAILS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[1]);
|
|
else if (pv->stats[STAT_ITEMS] & IT_ROCKETS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[2]);
|
|
else if (pv->stats[STAT_ITEMS] & IT_CELLS)
|
|
Sbar_DrawPic (224, 0, 24, 24, sb_ammo[3]);
|
|
}
|
|
|
|
Sbar_DrawNum (248, 0, pv->stats[STAT_AMMO], 3
|
|
, pv->stats[STAT_AMMO] <= 10);
|
|
}
|
|
|
|
qboolean Sbar_ShouldDraw (void)
|
|
{
|
|
#ifdef TEXTEDITOR
|
|
extern qboolean editoractive;
|
|
#endif
|
|
qboolean headsup;
|
|
|
|
if (scr_con_current == vid.height)
|
|
return false; // console is full screen
|
|
|
|
#ifdef TEXTEDITOR
|
|
if (editoractive)
|
|
return false;
|
|
#endif
|
|
|
|
#ifdef VM_UI
|
|
if (UI_DrawStatusBar((sb_showscores?1:0) + (sb_showteamscores?2:0))>0)
|
|
return false;
|
|
if (UI_MenuState())
|
|
return false;
|
|
#endif
|
|
|
|
headsup = !(cl_sbar.value || (scr_viewsize.value<100&&cl.splitclients==1));
|
|
if ((sb_updates >= vid.numpages) && !headsup)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void Sbar_DrawScoreboard (void)
|
|
{
|
|
int pnum;
|
|
int deadcount=0;
|
|
|
|
if (cls.protocol == CP_QUAKE2)
|
|
return;
|
|
|
|
if (Key_Dest_Has(~kdm_game))
|
|
return;
|
|
|
|
#ifndef CLIENTONLY
|
|
/*no scoreboard in single player (if you want bots, set deathmatch)*/
|
|
if (sv.state && !cls.deathmatch && sv.allocated_client_slots == 1)
|
|
{
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
for (pnum = 0; pnum < cl.splitclients; pnum++)
|
|
{
|
|
if (cl.playerview[pnum].stats[STAT_HEALTH] <= 0)
|
|
deadcount++;
|
|
}
|
|
|
|
if (deadcount == cl.splitclients && !cl.spectator)
|
|
{
|
|
if (cl.teamplay > 0 && !sb_showscores)
|
|
Sbar_TeamOverlay();
|
|
else
|
|
Sbar_DeathmatchOverlay (0);
|
|
}
|
|
else if (sb_showscores)
|
|
Sbar_DeathmatchOverlay (0);
|
|
else if (sb_showteamscores)
|
|
Sbar_TeamOverlay();
|
|
else
|
|
return;
|
|
|
|
sb_updates = 0;
|
|
}
|
|
|
|
|
|
static void Sbar_Hexen2DrawItem(playerview_t *pv, float x, float y, int itemnum)
|
|
{
|
|
int num;
|
|
Sbar_DrawPic(x, y, 29, 28, R2D_SafeCachePic(va("gfx/arti%02d.lmp", itemnum)));
|
|
|
|
num = pv->stats[STAT_H2_CNT_TORCH+itemnum];
|
|
if(num > 0)
|
|
{
|
|
if (num >= 10)
|
|
Sbar_DrawPic(x+20, y+21, 4, 6, R2D_SafeCachePic(va("gfx/artinum%d.lmp", num/10)));
|
|
Sbar_DrawPic(x+20+4, y+21, 4, 6, R2D_SafeCachePic(va("gfx/artinum%d.lmp", num%10)));
|
|
}
|
|
}
|
|
|
|
static void Sbar_Hexen2DrawInventory(playerview_t *pv)
|
|
{
|
|
int i;
|
|
int x, y=-37;
|
|
int activeleft = 0;
|
|
int activeright = 0;
|
|
|
|
/*always select an artifact that we actually have whether we are drawing the full bar or not.*/
|
|
for (i = 0; i < 15; i++)
|
|
{
|
|
if (pv->stats[STAT_H2_CNT_TORCH+(i+pv->sb_hexen2_cur_item)%15])
|
|
{
|
|
pv->sb_hexen2_cur_item = (pv->sb_hexen2_cur_item + i)%15;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pv->sb_hexen2_item_time+3 < realtime)
|
|
return;
|
|
|
|
for (i = pv->sb_hexen2_cur_item; i < 15; i++)
|
|
if (pv->sb_hexen2_cur_item == i || pv->stats[STAT_H2_CNT_TORCH+i] > 0)
|
|
activeright++;
|
|
for (i = pv->sb_hexen2_cur_item-1; i >= 0; i--)
|
|
if (pv->sb_hexen2_cur_item == i || pv->stats[STAT_H2_CNT_TORCH+i] > 0)
|
|
activeleft++;
|
|
|
|
if (activeleft > 3 + (activeright<=3?(4-activeright):0))
|
|
activeleft = 3 + (activeright<=3?(4-activeright):0);
|
|
x=320/2-114 + (activeleft-1)*33;
|
|
for (i = pv->sb_hexen2_cur_item-1; x>=320/2-114; i--)
|
|
{
|
|
if (!pv->stats[STAT_H2_CNT_TORCH+i])
|
|
continue;
|
|
|
|
if (i == pv->sb_hexen2_cur_item)
|
|
Sbar_DrawPic(x+9, y-12, 11, 11, R2D_SafeCachePic("gfx/artisel.lmp"));
|
|
Sbar_Hexen2DrawItem(pv, x, y, i);
|
|
x -= 33;
|
|
}
|
|
|
|
x=320/2-114 + activeleft*33;
|
|
for (i = pv->sb_hexen2_cur_item; i < 15 && x < 320/2-114+7*33; i++)
|
|
{
|
|
if (i != pv->sb_hexen2_cur_item && !pv->stats[STAT_H2_CNT_TORCH+i])
|
|
continue;
|
|
if (i == pv->sb_hexen2_cur_item)
|
|
Sbar_DrawPic(x+9, y-12, 11, 11, R2D_SafeCachePic("gfx/artisel.lmp"));
|
|
Sbar_Hexen2DrawItem(pv, x, y, i);
|
|
x+=33;
|
|
}
|
|
}
|
|
|
|
static void Sbar_Hexen2DrawExtra (playerview_t *pv)
|
|
{
|
|
unsigned int i, slot;
|
|
unsigned int pclass;
|
|
int ringpos[] = {6, 44, 81, 119};
|
|
//char *ringimages[] = {"gfx/ring_f.lmp", "gfx/ring_w.lmp", "gfx/ring_t.lmp", "gfx/ring_r.lmp"}; //unused variable
|
|
float val;
|
|
char *pclassname[] = {
|
|
"Unknown",
|
|
"Paladin",
|
|
"Crusader",
|
|
"Necromancer",
|
|
"Assasin",
|
|
"Demoness"
|
|
};
|
|
|
|
if (pv->sb_hexen2_infoplaque)
|
|
{
|
|
int i;
|
|
Con_Printf("Objectives:\n");
|
|
for (i = 0; i < 64; i++)
|
|
{
|
|
if (pv->stats[STAT_H2_OBJECTIVE1 + i/32] & (1<<(i&31)))
|
|
Con_Printf("%s\n", T_GetInfoString(i));
|
|
}
|
|
pv->sb_hexen2_infoplaque = false;
|
|
}
|
|
|
|
if (!pv->sb_hexen2_extra_info)
|
|
{
|
|
sbar_rect.y -= 46-SBAR_HEIGHT;
|
|
return;
|
|
}
|
|
|
|
pclass = cl.players[pv->playernum].h2playerclass;
|
|
if (pclass >= sizeof(pclassname)/sizeof(pclassname[0]))
|
|
pclass = sizeof(pclassname)/sizeof(pclassname[0]) - 1;
|
|
|
|
|
|
//adjust it so there's space
|
|
sbar_rect.y -= 46+98-SBAR_HEIGHT;
|
|
|
|
Sbar_DrawPic(0, 46, 160, 98, R2D_SafeCachePic("gfx/btmbar1.lmp"));
|
|
Sbar_DrawPic(160, 46, 160, 98, R2D_SafeCachePic("gfx/btmbar2.lmp"));
|
|
|
|
Sbar_DrawTinyString (11, 48, pclassname[pclass]);
|
|
|
|
Sbar_DrawTinyString (11, 58, va("int"));
|
|
Sbar_DrawTinyString (33, 58, va("%02d", pv->stats[STAT_H2_INTELLIGENCE]));
|
|
|
|
Sbar_DrawTinyString (11, 64, va("wis"));
|
|
Sbar_DrawTinyString (33, 64, va("%02d", pv->stats[STAT_H2_WISDOM]));
|
|
|
|
Sbar_DrawTinyString (11, 70, va("dex"));
|
|
Sbar_DrawTinyString (33, 70, va("%02d", pv->stats[STAT_H2_DEXTERITY]));
|
|
|
|
|
|
Sbar_DrawTinyString (58, 58, va("str"));
|
|
Sbar_DrawTinyString (80, 58, va("%02d", pv->stats[STAT_H2_STRENGTH]));
|
|
|
|
Sbar_DrawTinyString (58, 64, va("lvl"));
|
|
Sbar_DrawTinyString (80, 64, va("%02d", pv->stats[STAT_H2_LEVEL]));
|
|
|
|
Sbar_DrawTinyString (58, 70, va("exp"));
|
|
Sbar_DrawTinyString (80, 70, va("%06d", pv->stats[STAT_H2_EXPERIENCE]));
|
|
|
|
Sbar_DrawTinyString (11, 79, va("abilities"));
|
|
if (pv->stats[STAT_H2_FLAGS] & (1<<22))
|
|
Sbar_DrawTinyString (8, 89, T_GetString(400 + 2*(pclass-1) + 0));
|
|
if (pv->stats[STAT_H2_FLAGS] & (1<<23))
|
|
Sbar_DrawTinyString (8, 96, T_GetString(400 + 2*(pclass-1) + 1));
|
|
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
if (pv->stats[STAT_H2_ARMOUR1+i] > 0)
|
|
{
|
|
Sbar_DrawPic (164+i*40, 115, 28, 19, R2D_SafeCachePic(va("gfx/armor%d.lmp", i+1)));
|
|
Sbar_DrawTinyString (168+i*40, 136, va("+%d", pv->stats[STAT_H2_ARMOUR1+i]));
|
|
}
|
|
}
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
if (pv->stats[STAT_H2_FLIGHT_T+i] > 0)
|
|
{
|
|
Sbar_DrawPic (ringpos[i], 119, 32, 22, R2D_SafeCachePic(va("gfx/ring_f.lmp")));
|
|
val = pv->stats[STAT_H2_FLIGHT_T+i];
|
|
if (val > 100)
|
|
val = 100;
|
|
if (val < 0)
|
|
val = 0;
|
|
Sbar_DrawPic(ringpos[i]+29 - (int)(26 * (val/(float)100)),142, 26, 1, R2D_SafeCachePic("gfx/ringhlth.lmp"));
|
|
Sbar_DrawPic(ringpos[i]+29, 142, 26, 1, R2D_SafeCachePic("gfx/rhlthcvr.lmp"));
|
|
}
|
|
}
|
|
|
|
slot = 0;
|
|
for (i = 0; i < 8; i++)
|
|
{
|
|
if (pv->statsstr[STAT_H2_PUZZLE1+i])
|
|
{
|
|
Sbar_DrawPic (194+(slot%4)*31, slot<4?51:82, 26, 26, R2D_SafeCachePic(va("gfx/puzzle/%s.lmp", pv->statsstr[STAT_H2_PUZZLE1+i])));
|
|
slot++;
|
|
}
|
|
}
|
|
|
|
Sbar_DrawPic(134, 50, 49, 56, R2D_SafeCachePic(va("gfx/cport%d.lmp", pclass)));
|
|
}
|
|
|
|
static int Sbar_Hexen2ArmourValue(playerview_t *pv)
|
|
{
|
|
int i;
|
|
float ac = 0;
|
|
/*
|
|
WARNING: these values match the engine - NOT the gamecode!
|
|
Even the gamecode's values are misleading due to an indexing bug.
|
|
*/
|
|
static int acv[5][4] =
|
|
{
|
|
{8, 6, 2, 4},
|
|
{4, 8, 6, 2},
|
|
{2, 4, 8, 6},
|
|
{6, 2, 4, 8},
|
|
{6, 2, 4, 8}
|
|
};
|
|
|
|
int classno;
|
|
classno = cl.players[pv->playernum].h2playerclass;
|
|
if (classno >= 1 && classno <= 5)
|
|
{
|
|
classno--;
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
if (pv->stats[STAT_H2_ARMOUR1+i])
|
|
{
|
|
ac += acv[classno][i];
|
|
ac += pv->stats[STAT_H2_ARMOUR1+i]/5.0;
|
|
}
|
|
}
|
|
}
|
|
return ac;
|
|
}
|
|
|
|
static void Sbar_Hexen2DrawBasic(playerview_t *pv)
|
|
{
|
|
int chainpos;
|
|
int val, maxval;
|
|
Sbar_DrawPic(0, 0, 160, 46, R2D_SafeCachePic("gfx/topbar1.lmp"));
|
|
Sbar_DrawPic(160, 0, 160, 46, R2D_SafeCachePic("gfx/topbar2.lmp"));
|
|
Sbar_DrawPic(0, -23, 51, 23, R2D_SafeCachePic("gfx/topbumpl.lmp"));
|
|
Sbar_DrawPic(138, -8, 39, 8, R2D_SafeCachePic("gfx/topbumpm.lmp"));
|
|
Sbar_DrawPic(269, -23, 51, 23, R2D_SafeCachePic("gfx/topbumpr.lmp"));
|
|
|
|
//mana1
|
|
maxval = pv->stats[STAT_H2_MAXMANA];
|
|
val = pv->stats[STAT_H2_BLUEMANA];
|
|
val = bound(0, val, maxval);
|
|
Sbar_DrawTinyString(201, 22, va("%03d", val));
|
|
if(val)
|
|
{
|
|
Sbar_DrawPic(190, 26-(int)((val*18.0)/(float)maxval+0.5), 3, 19, R2D_SafeCachePic("gfx/bmana.lmp"));
|
|
Sbar_DrawPic(190, 27, 3, 19, R2D_SafeCachePic("gfx/bmanacov.lmp"));
|
|
}
|
|
|
|
//mana2
|
|
maxval = pv->stats[STAT_H2_MAXMANA];
|
|
val = pv->stats[STAT_H2_GREENMANA];
|
|
val = bound(0, val, maxval);
|
|
Sbar_DrawTinyString(243, 22, va("%03d", val));
|
|
if(val)
|
|
{
|
|
Sbar_DrawPic(232, 26-(int)((val*18.0)/(float)maxval+0.5), 3, 19, R2D_SafeCachePic("gfx/gmana.lmp"));
|
|
Sbar_DrawPic(232, 27, 3, 19, R2D_SafeCachePic("gfx/gmanacov.lmp"));
|
|
}
|
|
|
|
|
|
//health
|
|
val = pv->stats[STAT_HEALTH];
|
|
if (val < -99)
|
|
val = -99;
|
|
Sbar_Hexen2DrawNum(58, 14, val, 3);
|
|
|
|
//armour
|
|
val = Sbar_Hexen2ArmourValue(pv);
|
|
Sbar_Hexen2DrawNum(105, 14, val, 2);
|
|
|
|
// SetChainPosition(cl.v.health, cl.v.max_health);
|
|
chainpos = (195.0f*pv->stats[STAT_HEALTH]) / pv->stats[STAT_H2_MAXHEALTH];
|
|
if (chainpos < 0)
|
|
chainpos = 0;
|
|
Sbar_DrawPic(45+((int)chainpos&7), 38, 222, 5, R2D_SafeCachePic("gfx/hpchain.lmp"));
|
|
Sbar_DrawPic(45+(int)chainpos, 36, 35, 9, R2D_SafeCachePic("gfx/hpgem.lmp"));
|
|
Sbar_DrawPic(43, 36, 10, 10, R2D_SafeCachePic("gfx/chnlcov.lmp"));
|
|
Sbar_DrawPic(267, 36, 10, 10, R2D_SafeCachePic("gfx/chnrcov.lmp"));
|
|
|
|
|
|
Sbar_Hexen2DrawItem(pv, 144, 3, pv->sb_hexen2_cur_item);
|
|
}
|
|
|
|
static void Sbar_Hexen2DrawMinimal(playerview_t *pv)
|
|
{
|
|
int y;
|
|
y = -16;
|
|
Sbar_DrawPic(3, y, 31, 17, R2D_SafeCachePic("gfx/bmmana.lmp"));
|
|
Sbar_DrawPic(3, y+18, 31, 17, R2D_SafeCachePic("gfx/gmmana.lmp"));
|
|
|
|
Sbar_DrawTinyString(10, y+6, va("%03d", pv->stats[STAT_H2_BLUEMANA]));
|
|
Sbar_DrawTinyString(10, y+18+6, va("%03d", pv->stats[STAT_H2_GREENMANA]));
|
|
|
|
Sbar_Hexen2DrawNum(38, y+18, pv->stats[STAT_HEALTH], 3);
|
|
}
|
|
|
|
|
|
static void Sbar_DrawTeamStatus(playerview_t *pv)
|
|
{
|
|
int p;
|
|
int y;
|
|
int track;
|
|
|
|
if (!sbar_teamstatus.ival)
|
|
return;
|
|
y = -32;
|
|
|
|
track = Cam_TrackNum(pv);
|
|
if (track == -1 || !cl.spectator)
|
|
track = pv->playernum;
|
|
|
|
for (p = 0; p < cl.allocated_client_slots; p++)
|
|
{
|
|
if (pv->playernum == p) //self is not shown
|
|
continue;
|
|
if (track == p) //nor is the person you are tracking
|
|
continue;
|
|
|
|
if (cl.players[p].teamstatustime < realtime)
|
|
continue;
|
|
|
|
if (!*cl.players[p].teamstatus) //only show them if they have something. no blank lines thanks
|
|
continue;
|
|
if (strcmp(cl.players[p].team, cl.players[track].team))
|
|
continue;
|
|
|
|
if (*cl.players[p].name)
|
|
{
|
|
Sbar_DrawString (0, y, cl.players[p].teamstatus);
|
|
y-=8;
|
|
}
|
|
}
|
|
sbar_parsingteamstatuses = true;
|
|
}
|
|
|
|
qboolean Sbar_UpdateTeamStatus(player_info_t *player, char *status)
|
|
{
|
|
qboolean aswhite = false;
|
|
char *outb;
|
|
int outlen;
|
|
char *msgstart;
|
|
char *ledstatus;
|
|
|
|
if (*status != '\r')// && !(strchr(status, 0x86) || strchr(status, 0x87) || strchr(status, 0x88) || strchr(status, 0x89)))
|
|
{
|
|
if (*status != 'x' || status[1] != '\r')
|
|
return false;
|
|
status++;
|
|
}
|
|
|
|
if (*status == '\r')
|
|
{
|
|
while (*status == ' ' || *status == '\r')
|
|
status++;
|
|
ledstatus = status;
|
|
if (*(unsigned char*)ledstatus >= 0x86 && *(unsigned char*)ledstatus <= 0x89)
|
|
{
|
|
msgstart = strchr(status, ':');
|
|
if (!status)
|
|
return false;
|
|
if (msgstart)
|
|
status = msgstart+1;
|
|
else
|
|
ledstatus = NULL;
|
|
}
|
|
else
|
|
ledstatus = NULL;
|
|
}
|
|
else
|
|
ledstatus = NULL;
|
|
|
|
while (*status == ' ' || *status == '\r')
|
|
status++;
|
|
|
|
//fixme: handle { and } stuff (assume red?)
|
|
outb = player->teamstatus;
|
|
outlen = sizeof(player->teamstatus)-1;
|
|
if (ledstatus)
|
|
{
|
|
*outb++ = *ledstatus;
|
|
outlen--;
|
|
}
|
|
|
|
while(outlen>0 && *status)
|
|
{
|
|
if (*status == '{')
|
|
{
|
|
aswhite=true;
|
|
status++;
|
|
continue;
|
|
}
|
|
if (aswhite)
|
|
{
|
|
if (*status == '}')
|
|
{
|
|
aswhite = false;
|
|
status++;
|
|
continue;
|
|
}
|
|
*outb++ = *status++;
|
|
}
|
|
else
|
|
*outb++ = *status++|128;
|
|
outlen--;
|
|
}
|
|
|
|
player->teamstatustime = realtime + 10;
|
|
|
|
*outb = '\0';
|
|
|
|
if (sbar_teamstatus.value == 2)
|
|
return sbar_parsingteamstatuses;
|
|
return false;
|
|
}
|
|
|
|
|
|
static void Sbar_Voice(int y)
|
|
{
|
|
#ifdef VOICECHAT
|
|
int loudness;
|
|
if (!snd_voip_showmeter.ival)
|
|
return;
|
|
loudness = S_Voip_Loudness(snd_voip_showmeter.ival==2);
|
|
if (loudness >= 0)
|
|
{
|
|
int w;
|
|
int x=0;
|
|
int s, i;
|
|
float range = loudness/100.0f;
|
|
w = 0;
|
|
Font_BeginString(font_conchar, sbar_rect.x + sbar_rect.width/2, sbar_rect.y + y + sbar_rect.height-SBAR_HEIGHT, &x, &y);
|
|
w += Font_CharWidth(0xe080 | CON_WHITEMASK);
|
|
w += Font_CharWidth(0xe081 | CON_WHITEMASK)*16;
|
|
w += Font_CharWidth(0xe082 | CON_WHITEMASK);
|
|
w += Font_CharWidth('M' | CON_WHITEMASK);
|
|
w += Font_CharWidth('i' | CON_WHITEMASK);
|
|
w += Font_CharWidth('c' | CON_WHITEMASK);
|
|
w += Font_CharWidth(' ' | CON_WHITEMASK);
|
|
x -= w/2;
|
|
x = Font_DrawChar(x, y, 'M' | CON_WHITEMASK);
|
|
x = Font_DrawChar(x, y, 'i' | CON_WHITEMASK);
|
|
x = Font_DrawChar(x, y, 'c' | CON_WHITEMASK);
|
|
x = Font_DrawChar(x, y, ' ' | CON_WHITEMASK);
|
|
x = Font_DrawChar(x, y, 0xe080 | CON_WHITEMASK);
|
|
s = x;
|
|
for (i=0 ; i<16 ; i++)
|
|
x = Font_DrawChar(x, y, 0xe081 | CON_WHITEMASK);
|
|
Font_DrawChar(x, y, 0xe082 | CON_WHITEMASK);
|
|
Font_DrawChar(s + (x-s) * range - Font_CharWidth(0xe083 | CON_WHITEMASK)/2, y, 0xe083 | CON_WHITEMASK);
|
|
Font_EndString(font_conchar);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
===============
|
|
Sbar_Draw
|
|
===============
|
|
*/
|
|
void Sbar_Draw (playerview_t *pv)
|
|
{
|
|
qboolean headsup;
|
|
char st[512];
|
|
int sbarwidth;
|
|
qboolean minidmoverlay;
|
|
extern cvar_t scr_centersbar;
|
|
|
|
headsup = !(cl_sbar.value || (scr_viewsize.value<100&&cl.splitclients==1));
|
|
if ((sb_updates >= vid.numpages) && !headsup)
|
|
return;
|
|
|
|
sbar_parsingteamstatuses = false;
|
|
|
|
#ifdef HLCLIENT
|
|
if (CLHL_DrawHud())
|
|
return;
|
|
#endif
|
|
|
|
#ifdef Q2CLIENT
|
|
if (cls.protocol == CP_QUAKE2)
|
|
{
|
|
sbar_rect = r_refdef.grect;
|
|
R2D_ImageColours(1, 1, 1, 1);
|
|
if (*cl.q2statusbar)
|
|
Sbar_ExecuteLayoutString(cl.q2statusbar);
|
|
if (*cl.q2layout)
|
|
{
|
|
if (cl.q2frame.playerstate.stats[Q2STAT_LAYOUTS] & 1)
|
|
Sbar_ExecuteLayoutString(cl.q2layout);
|
|
}
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
Sbar_Start();
|
|
|
|
R2D_ImageColours(1, 1, 1, 1);
|
|
|
|
minidmoverlay = cl.deathmatch;
|
|
sbar_rect = r_refdef.grect;
|
|
|
|
sbarwidth = 320;
|
|
if (minidmoverlay && r_refdef.grect.width >= 640 && cl.teamplay)
|
|
sbarwidth += 320;
|
|
else if (minidmoverlay && r_refdef.grect.width >= 512)
|
|
sbarwidth += 192;
|
|
else
|
|
minidmoverlay = 0;
|
|
|
|
if (scr_centersbar.ival)
|
|
{
|
|
float ofs = (sbar_rect.width - sbarwidth)/2;
|
|
sbar_rect.x += ofs;
|
|
sbar_rect.width -= ofs;
|
|
}
|
|
|
|
sb_updates++;
|
|
|
|
|
|
if (sbar_hexen2)
|
|
{
|
|
//hexen2 hud
|
|
if (sb_lines > 24 || pv->sb_hexen2_extra_info)
|
|
{
|
|
Sbar_Hexen2DrawExtra(pv);
|
|
Sbar_Hexen2DrawBasic(pv);
|
|
}
|
|
else if (sb_lines > 0)
|
|
Sbar_Hexen2DrawMinimal(pv);
|
|
Sbar_Hexen2DrawInventory(pv);
|
|
|
|
if (cl.deathmatch)
|
|
Sbar_MiniDeathmatchOverlay (pv);
|
|
}
|
|
else if (sbarfailed) //files failed to load.
|
|
{
|
|
//fallback hud
|
|
if (pv->stats[STAT_HEALTH] > 0) //when dead, show nothing
|
|
{
|
|
// if (scr_viewsize.value != 120)
|
|
// Cvar_Set(&scr_viewsize, "120");
|
|
|
|
Sbar_DrawString (0, -8, va("Health: %i", pv->stats[STAT_HEALTH]));
|
|
Sbar_DrawString (0, -16, va(" Armor: %i", pv->stats[STAT_ARMOR]));
|
|
|
|
Sbar_Voice(-24);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//standard quake(world) hud.
|
|
// top line
|
|
if (sb_lines > 24)
|
|
{
|
|
if (!cl.spectator || pv->cam_auto == CAM_TRACK)
|
|
Sbar_DrawInventory (pv);
|
|
if ((!headsup || sbar_rect.width<512) && cl.deathmatch)
|
|
Sbar_DrawFrags (pv);
|
|
}
|
|
|
|
// main area
|
|
if (sb_lines > 0)
|
|
{
|
|
if (cl.spectator)
|
|
{
|
|
if (pv->cam_auto != CAM_TRACK)
|
|
{
|
|
Sbar_DrawPic (0, 0, 320, 24, sb_scorebar);
|
|
Sbar_DrawString (160-7*8,4, "SPECTATOR MODE");
|
|
Sbar_DrawString(160-14*8+4, 12, "Press [ATTACK] for AutoCamera");
|
|
}
|
|
else
|
|
{
|
|
if (sb_showscores || sb_showteamscores || pv->stats[STAT_HEALTH] <= 0)
|
|
Sbar_SoloScoreboard ();
|
|
// else if (cls.gamemode != GAME_DEATHMATCH)
|
|
// Sbar_CoopScoreboard ();
|
|
else
|
|
Sbar_DrawNormal (pv);
|
|
|
|
if (hud_tracking_show.ival)
|
|
{
|
|
Q_snprintfz(st, sizeof(st), "Tracking %-.64s",
|
|
cl.players[pv->cam_spec_track].name);
|
|
Sbar_DrawString(0, -8, st);
|
|
}
|
|
}
|
|
}
|
|
else if (sb_showscores || sb_showteamscores || (pv->stats[STAT_HEALTH] <= 0 && cl.splitclients == 1))
|
|
{
|
|
if (pv == cl.playerview)
|
|
{
|
|
if (!cls.deathmatch)
|
|
Sbar_CoopScoreboard ();
|
|
else
|
|
Sbar_SoloScoreboard ();
|
|
}
|
|
}
|
|
else
|
|
Sbar_DrawNormal (pv);
|
|
}
|
|
|
|
if (minidmoverlay)
|
|
Sbar_MiniDeathmatchOverlay (pv);
|
|
|
|
if (sb_lines > 0)
|
|
Sbar_DrawTeamStatus(pv);
|
|
R2D_ImageColours (1, 1, 1, 1);
|
|
}
|
|
|
|
if (cl_sbar.value == 1 || scr_viewsize.value<100)
|
|
{
|
|
if (sbar_rect.x>r_refdef.grect.x)
|
|
{ // left
|
|
R2D_TileClear (r_refdef.grect.x, r_refdef.grect.y+sbar_rect.height - sb_lines, sbar_rect.x - r_refdef.grect.x, sb_lines);
|
|
}
|
|
if (sbar_rect.x + 320 <= r_refdef.grect.x + sbar_rect.width && !headsup)
|
|
R2D_TileClear (sbar_rect.x + 320, r_refdef.grect.y+sbar_rect.height - sb_lines, sbar_rect.width - (320), sb_lines);
|
|
}
|
|
|
|
if (sb_lines > 24)
|
|
Sbar_Voice(-32);
|
|
else if (sb_lines > 0)
|
|
Sbar_Voice(-8);
|
|
else
|
|
Sbar_Voice(16);
|
|
|
|
{
|
|
extern int scr_chatmode;
|
|
if (scr_chatmode)
|
|
Sbar_ChatModeOverlay(pv);
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
/*
|
|
==================
|
|
Sbar_IntermissionNumber
|
|
|
|
==================
|
|
*/
|
|
void Sbar_IntermissionNumber (float x, float y, int num, int digits, int color, qboolean left)
|
|
{
|
|
char str[12];
|
|
char *ptr;
|
|
int l, frame;
|
|
|
|
l = Sbar_itoa (num, str);
|
|
ptr = str;
|
|
if (l > digits)
|
|
ptr += (l-digits);
|
|
if (!left)
|
|
if (l < digits)
|
|
x += (digits-l)*24;
|
|
|
|
while (*ptr)
|
|
{
|
|
if (*ptr == '-')
|
|
frame = STAT_MINUS;
|
|
else
|
|
frame = *ptr -'0';
|
|
|
|
R2D_ScalePic (x,y, 16, 24, sb_nums[color][frame]);
|
|
x += 24;
|
|
ptr++;
|
|
}
|
|
}
|
|
|
|
#define COL_TEAM_LOWAVGHIGH COLUMN("low/avg/high", 12*8, {sprintf (num, "%3i/%3i/%3i", plow, pavg, phigh); Draw_FunString ( x, y, num); })
|
|
#define COL_TEAM_TEAM COLUMN("team", 4*8, {Draw_FunStringWidth ( x, y, tm->team, 4*8, false, false); \
|
|
if (!strncmp(cl.players[pv->playernum].team, tm->team, 16))\
|
|
{\
|
|
Draw_FunString ( x - 1*8, y, "^Ue010");\
|
|
Draw_FunString ( x + 4*8, y, "^Ue011");\
|
|
}\
|
|
})
|
|
#define COL_TEAM_TOTAL COLUMN("total", 5*8, {Draw_FunString ( x, y, va("%5i", tm->frags)); })
|
|
#define COL_TEAM_PLAYERS COLUMN("players", 7*8, {Draw_FunString ( x, y, va("%5i", tm->players)); })
|
|
#define ALL_TEAM_COLUMNS COL_TEAM_LOWAVGHIGH COL_TEAM_TEAM COL_TEAM_TOTAL COL_TEAM_PLAYERS
|
|
|
|
|
|
/*
|
|
==================
|
|
Sbar_TeamOverlay
|
|
|
|
team frags
|
|
added by Zoid
|
|
==================
|
|
*/
|
|
void Sbar_TeamOverlay (void)
|
|
{
|
|
mpic_t *pic;
|
|
int i, k;
|
|
int x, y, l;
|
|
char num[12];
|
|
team_t *tm;
|
|
int plow, phigh, pavg;
|
|
playerview_t *pv = r_refdef.playerview;
|
|
|
|
if (!pv)
|
|
pv = &cl.playerview[0];
|
|
|
|
// request new ping times every two second
|
|
if (!cl.teamplay)
|
|
{
|
|
Sbar_DeathmatchOverlay(0);
|
|
return;
|
|
}
|
|
|
|
y = 0;
|
|
|
|
if (scr_scoreboard_drawtitle.ival)
|
|
{
|
|
pic = R2D_SafeCachePic ("gfx/ranking.lmp");
|
|
if (pic)
|
|
{
|
|
k = (pic->width * 24) / pic->height;
|
|
R2D_ScalePic ((vid.width-k)/2, 0, k, 24, pic);
|
|
}
|
|
y += 24;
|
|
}
|
|
|
|
x = l = (vid.width - 320)/2 + 36;
|
|
|
|
#define COLUMN(title, cwidth, code) Draw_FunString(x, y, title), x+=cwidth + 8;
|
|
ALL_TEAM_COLUMNS
|
|
// if (rank_width+(cwidth)+8 <= vid.width) {showcolumns |= (1<<COLUMN##title); rank_width += cwidth+8;}
|
|
|
|
// Draw_FunString(x, y, "low/avg/high");
|
|
// Draw_FunString(x+13*8, y, "team");
|
|
// Draw_FunString(x+18*8, y, "total");
|
|
// Draw_FunString(x+24*8, y, "players");
|
|
y += 8;
|
|
// Draw_String(x, y, "------------ ---- ----- -------");
|
|
x = l;
|
|
#undef COLUMN
|
|
#define COLUMN(title, cwidth, code) {char buf[64*6]; int t = (cwidth)/8; int c=0; while (t-->0) {buf[c++] = '^'; buf[c++] = 'U'; buf[c++] = 'e'; buf[c++] = '0'; buf[c++] = '1'; buf[c] = (c==5?'d':(!t?'f':'e')); c++;} buf[c] = 0; Draw_FunString(x, y, buf); x += cwidth + 8;}
|
|
ALL_TEAM_COLUMNS
|
|
// Draw_FunString(x, y, "^Ue01d^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01f ^Ue01d^Ue01e^Ue01e^Ue01f ^Ue01d^Ue01e^Ue01e^Ue01e^Ue01f ^Ue01d^Ue01e^Ue01e^Ue01e^Ue01e^Ue01e^Ue01f");
|
|
y += 8;
|
|
|
|
#undef COLUMN
|
|
|
|
// sort the teams
|
|
Sbar_SortTeams(pv);
|
|
|
|
// draw the text
|
|
for (i=0 ; i < scoreboardteams && y <= vid.height-10 ; i++)
|
|
{
|
|
k = teamsort[i];
|
|
tm = teams + k;
|
|
|
|
// draw pings
|
|
plow = tm->plow;
|
|
if (plow < 0 || plow > 999)
|
|
plow = 999;
|
|
phigh = tm->phigh;
|
|
if (phigh < 0 || phigh > 999)
|
|
phigh = 999;
|
|
if (!tm->players)
|
|
pavg = 999;
|
|
else
|
|
pavg = tm->ptotal / tm->players;
|
|
if (pavg < 0 || pavg > 999)
|
|
pavg = 999;
|
|
|
|
x = l;
|
|
|
|
#if 1
|
|
#define COLUMN(title, cwidth, code) code; x+=cwidth + 8;
|
|
ALL_TEAM_COLUMNS
|
|
#undef COLUMN
|
|
#else
|
|
sprintf (num, "%3i/%3i/%3i", plow, pavg, phigh);
|
|
Draw_FunString ( x, y, num);
|
|
|
|
// draw team
|
|
Q_strncpyz (team, tm->team, sizeof(team));
|
|
Draw_FunString (x + 104, y, team);
|
|
|
|
// draw total
|
|
sprintf (num, "%5i", tm->frags);
|
|
Draw_FunString (x + 104 + 40, y, num);
|
|
|
|
// draw players
|
|
sprintf (num, "%5i", tm->players);
|
|
Draw_FunString (x + 104 + 88, y, num);
|
|
|
|
if (!strncmp(cl.players[cl.playernum[0]].team, tm->team, 16))
|
|
{
|
|
Draw_FunString ( x + 104 - 8, y, "^Ue010");
|
|
Draw_FunString ( x + 104 + 32, y, "^Ue011");
|
|
}
|
|
#endif
|
|
y += 8;
|
|
}
|
|
y += 8;
|
|
Sbar_DeathmatchOverlay(y);
|
|
}
|
|
|
|
/*
|
|
==================
|
|
Sbar_DeathmatchOverlay
|
|
|
|
ping time frags name
|
|
==================
|
|
*/
|
|
|
|
//for reference:
|
|
//define COLUMN(title, width, code)
|
|
|
|
#define COLUMN_PING COLUMN(ping, 4*8, \
|
|
{ \
|
|
int p = s->ping; \
|
|
if (p < 0 || p > 999) p = 999; \
|
|
sprintf(num, "%4i", p); \
|
|
Draw_FunStringWidth(x, y, num, 4*8, false, false); \
|
|
})
|
|
|
|
#define COLUMN_PL COLUMN(pl, 2*8, \
|
|
{ \
|
|
int p = s->pl; \
|
|
sprintf(num, "%2i", p); \
|
|
Draw_FunStringWidth(x, y, num, 2*8, false, false); \
|
|
})
|
|
#define COLUMN_TIME COLUMN(time, 4*8, \
|
|
{ \
|
|
if (cl.intermission) \
|
|
total = cl.completed_time - s->entertime; \
|
|
else \
|
|
total = cl.servertime - s->entertime; \
|
|
minutes = (int)total/60; \
|
|
sprintf (num, "%4i", minutes); \
|
|
Draw_FunStringWidth(x, y, num, 4*8, false, false); \
|
|
})
|
|
#define COLUMN_FRAGS COLUMN(frags, 5*8, \
|
|
{ \
|
|
int cx; int cy; \
|
|
if (s->spectator) \
|
|
{ \
|
|
Draw_FunStringWidth(x, y, "spectator", 5*8, false, false); \
|
|
} \
|
|
else \
|
|
{ \
|
|
if (largegame) \
|
|
Sbar_FillPC(x, y+1, 40, 3, top); \
|
|
else \
|
|
Sbar_FillPC(x, y, 40, 4, top); \
|
|
Sbar_FillPC(x, y+4, 40, 4, bottom); \
|
|
\
|
|
f = s->frags; \
|
|
sprintf(num, "%3i",f); \
|
|
\
|
|
Font_BeginString(font_conchar, x+8, y, &cx, &cy); \
|
|
Font_DrawChar(cx, cy, num[0] | 0xe000 | CON_WHITEMASK); \
|
|
Font_BeginString(font_conchar, x+16, y, &cx, &cy); \
|
|
Font_DrawChar(cx, cy, num[1] | 0xe000 | CON_WHITEMASK); \
|
|
Font_BeginString(font_conchar, x+24, y, &cx, &cy); \
|
|
Font_DrawChar(cx, cy, num[2] | 0xe000 | CON_WHITEMASK); \
|
|
\
|
|
if ((cl.spectator && k == Cam_TrackNum(pv)) ||\
|
|
(!cl.spectator && k == pv->playernum)) \
|
|
{ \
|
|
Font_BeginString(font_conchar, x, y, &cx, &cy); \
|
|
Font_DrawChar(cx, cy, 16 | 0xe000 | CON_WHITEMASK); \
|
|
Font_BeginString(font_conchar, x+32, y, &cx, &cy); \
|
|
Font_DrawChar(cx, cy, 17 | 0xe000 | CON_WHITEMASK); \
|
|
} \
|
|
Font_EndString(font_conchar); \
|
|
} \
|
|
})
|
|
#define COLUMN_TEAMNAME COLUMN(team, 4*8, \
|
|
{ \
|
|
if (!s->spectator) \
|
|
{ \
|
|
Draw_FunStringWidth(x, y, s->team, 4*8, false, false); \
|
|
} \
|
|
})
|
|
#define COLUMN_NAME COLUMN(name, (cl.teamplay ? 12*8 : 16*8), {Draw_FunStringWidth(x, y, s->name, (cl.teamplay ? 12*8 : 16*8), false, false);})
|
|
#define COLUMN_KILLS COLUMN(kils, 4*8, {Draw_FunStringWidth(x, y, va("%4i", Stats_GetKills(k)), 4*8, false, false);})
|
|
#define COLUMN_TKILLS COLUMN(tkil, 4*8, {Draw_FunStringWidth(x, y, va("%4i", Stats_GetTKills(k)), 4*8, false, false);})
|
|
#define COLUMN_DEATHS COLUMN(dths, 4*8, {Draw_FunStringWidth(x, y, va("%4i", Stats_GetDeaths(k)), 4*8, false, false);})
|
|
#define COLUMN_TOUCHES COLUMN(tchs, 4*8, {Draw_FunStringWidth(x, y, va("%4i", Stats_GetTouches(k)), 4*8, false, false);})
|
|
#define COLUMN_CAPS COLUMN(caps, 4*8, {Draw_FunStringWidth(x, y, va("%4i", Stats_GetCaptures(k)), 4*8, false, false);})
|
|
|
|
|
|
|
|
//columns are listed here in display order
|
|
#define ALLCOLUMNS COLUMN_PING COLUMN_PL COLUMN_TIME COLUMN_FRAGS COLUMN_TEAMNAME COLUMN_NAME COLUMN_KILLS COLUMN_TKILLS COLUMN_DEATHS COLUMN_TOUCHES COLUMN_CAPS
|
|
|
|
enum
|
|
{
|
|
#define COLUMN(title, width, code) COLUMN##title,
|
|
ALLCOLUMNS
|
|
#undef COLUMN
|
|
COLUMN_MAX
|
|
};
|
|
|
|
#define ADDCOLUMN(id) showcolumns |= (1<<id)
|
|
void Sbar_DeathmatchOverlay (int start)
|
|
{
|
|
mpic_t *pic;
|
|
int i, k;
|
|
int x, y, f;
|
|
char num[12];
|
|
player_info_t *s;
|
|
int total;
|
|
int minutes;
|
|
int skip = 10;
|
|
int showcolumns;
|
|
int startx, rank_width;
|
|
playerview_t *pv = r_refdef.playerview;
|
|
|
|
if (!pv)
|
|
return;
|
|
|
|
if (largegame)
|
|
skip = 8;
|
|
|
|
// request new ping times every two second
|
|
if (realtime - cl.last_ping_request > 2 && cls.demoplayback != DPB_EZTV)
|
|
{
|
|
if (cls.protocol == CP_QUAKEWORLD)
|
|
{
|
|
cl.last_ping_request = realtime;
|
|
CL_SendClientCommand(true, "pings");
|
|
}
|
|
else if (cls.protocol == CP_NETQUAKE)
|
|
{
|
|
cl.last_ping_request = realtime;
|
|
CL_SendClientCommand(true, "ping");
|
|
}
|
|
}
|
|
|
|
if (start)
|
|
y = start;
|
|
else
|
|
{
|
|
y = 0;
|
|
if (scr_scoreboard_drawtitle.ival)
|
|
{
|
|
pic = R2D_SafeCachePic ("gfx/ranking.lmp");
|
|
if (pic)
|
|
{
|
|
k = (pic->width * 24) / pic->height;
|
|
R2D_ScalePic ((vid.width-k)/2, 0, k, 24, pic);
|
|
}
|
|
y += 24;
|
|
}
|
|
}
|
|
|
|
// scores
|
|
Sbar_SortFrags(true, true);
|
|
|
|
// draw the text
|
|
if (start)
|
|
y = start;
|
|
else
|
|
y = 24;
|
|
|
|
if (scr_scoreboard_newstyle.ival)
|
|
{
|
|
// Electro's scoreboard eyecandy: Increase to fit the new scoreboard
|
|
y += 8;
|
|
}
|
|
|
|
showcolumns = 0;
|
|
|
|
rank_width = 0;
|
|
|
|
#define COLUMN(title, cwidth, code) if (rank_width+(cwidth)+8 <= vid.width) {showcolumns |= (1<<COLUMN##title); rank_width += cwidth+8;}
|
|
//columns are listed here in priority order (if the screen is too narrow, later ones will be hidden)
|
|
COLUMN_NAME
|
|
COLUMN_PING
|
|
if (cls.protocol == CP_QUAKEWORLD)
|
|
{
|
|
COLUMN_PL
|
|
COLUMN_TIME
|
|
}
|
|
COLUMN_FRAGS
|
|
if (cl.teamplay)
|
|
{
|
|
COLUMN_TEAMNAME
|
|
}
|
|
if (cl.teamplay && Stats_HaveFlags())
|
|
{
|
|
COLUMN_CAPS
|
|
}
|
|
if (scr_scoreboard_showfrags.value && Stats_HaveKills())
|
|
{
|
|
COLUMN_KILLS
|
|
COLUMN_DEATHS
|
|
if (cl.teamplay)
|
|
{
|
|
COLUMN_TKILLS
|
|
}
|
|
}
|
|
if (cl.teamplay && Stats_HaveFlags())
|
|
{
|
|
COLUMN_TOUCHES
|
|
}
|
|
#undef COLUMN
|
|
|
|
startx = (vid.width-rank_width)/2;
|
|
|
|
if (scr_scoreboard_newstyle.ival)
|
|
{
|
|
// Electro's scoreboard eyecandy: Draw top border
|
|
R2D_ImagePaletteColour (0, 1.0);
|
|
R2D_FillBlock(startx - 3, y - 1, rank_width - 1, 1);
|
|
|
|
// Electro's scoreboard eyecandy: Draw the title row background
|
|
R2D_ImagePaletteColour (1, 1.0);
|
|
R2D_FillBlock(startx - 2, y, rank_width - 3, 9);
|
|
}
|
|
|
|
x = startx;
|
|
#define COLUMN(title, width, code) if (showcolumns & (1<<COLUMN##title)) {Draw_FunString(x, y, #title); x += width+8;}
|
|
ALLCOLUMNS
|
|
#undef COLUMN
|
|
|
|
|
|
y += 8;
|
|
|
|
if (scr_scoreboard_titleseperator.ival && !scr_scoreboard_newstyle.ival)
|
|
{
|
|
x = startx;
|
|
#define COLUMN(title, width, code) \
|
|
if (showcolumns & (1<<COLUMN##title)) \
|
|
{ \
|
|
Draw_FunString(x, y, "^Ue01d"); \
|
|
for (i = 8; i < width-8; i+= 8) \
|
|
Draw_FunString(x+i, y, "^Ue01e"); \
|
|
Draw_FunString(x+i, y, "^Ue01f"); \
|
|
x += width+8; \
|
|
}
|
|
ALLCOLUMNS
|
|
#undef COLUMN
|
|
y += 8;
|
|
}
|
|
|
|
if (scr_scoreboard_newstyle.ival)
|
|
{
|
|
// Electro's scoreboard eyecandy: Draw top border (under header)
|
|
R2D_ImagePaletteColour (0, 1.0);
|
|
R2D_FillBlock (startx - 3, y + 1, rank_width - 1, 1);
|
|
// Electro's scoreboard eyecandy: Don't go over the black border, move the rest down
|
|
y += 2;
|
|
// Electro's scoreboard eyecandy: Draw left border
|
|
R2D_FillBlock (startx - 3, y - 10, 1, 9);
|
|
// Electro's scoreboard eyecandy: Draw right border
|
|
R2D_FillBlock (startx - 3 + rank_width - 2, y - 10, 1, 9);
|
|
}
|
|
|
|
y -= skip;
|
|
|
|
for (i = 0; i < scoreboardlines; i++)
|
|
{
|
|
char team[5];
|
|
unsigned int top, bottom;
|
|
|
|
// TODO: Sort players so that the leading teams are drawn first
|
|
k = fragsort[i];
|
|
s = &cl.players[k];
|
|
if (!s->name[0])
|
|
continue;
|
|
|
|
y += skip;
|
|
if (y > vid.height-10)
|
|
break;
|
|
|
|
// Electro's scoreboard eyecandy: Moved this up here for usage with the row background color
|
|
top = Sbar_TopColour(s);
|
|
bottom = Sbar_BottomColour(s);
|
|
|
|
if (scr_scoreboard_newstyle.ival)
|
|
{
|
|
// Electro's scoreboard eyecandy: Render the main background transparencies behind players row
|
|
// TODO: Alpha values on the background
|
|
if ((cl.teamplay) && (!s->spectator))
|
|
{
|
|
int background_color;
|
|
// Electro's scoreboard eyecandy: red vs blue are common teams, force the colours
|
|
Q_strncpyz (team, Info_ValueForKey(s->userinfo, "team"), sizeof(team));
|
|
|
|
if (S_Voip_Speaking(k))
|
|
background_color = 0x00ff00;
|
|
else if (!(strcmp("red", team)))
|
|
background_color = 4; // forced red
|
|
else if (!(strcmp("blue", team)))
|
|
background_color = 13; // forced blue
|
|
else
|
|
background_color = bottom;
|
|
|
|
Sbar_FillPCDark (startx - 2, y, rank_width - 3, skip, background_color);
|
|
}
|
|
else if (S_Voip_Speaking(k))
|
|
Sbar_FillPCDark (startx - 2, y, rank_width - 3, skip, 0x00ff00);
|
|
else
|
|
{
|
|
R2D_ImagePaletteColour (2, 1.0);
|
|
R2D_FillBlock (startx - 2, y, rank_width - 3, skip);
|
|
}
|
|
|
|
R2D_ImagePaletteColour (0, 1.0);
|
|
R2D_FillBlock (startx - 3, y, 1, skip); // Electro - Border - Left
|
|
R2D_FillBlock (startx - 3 + rank_width - 2, y, 1, skip); // Electro - Border - Right
|
|
}
|
|
|
|
x = startx;
|
|
#define COLUMN(title, width, code) \
|
|
if (showcolumns & (1<<COLUMN##title)) \
|
|
{ \
|
|
code \
|
|
x += width+8; \
|
|
}
|
|
ALLCOLUMNS
|
|
#undef COLUMN
|
|
}
|
|
|
|
if (scr_scoreboard_newstyle.ival)
|
|
{
|
|
R2D_ImagePaletteColour (0, 1.0);
|
|
R2D_FillBlock (startx - 3, y + skip, rank_width - 1, 1); // Electro - Border - Bottom
|
|
}
|
|
|
|
if (y >= vid.height-10) // we ran over the screen size, squish
|
|
largegame = true;
|
|
|
|
R2D_ImageColours(1.0, 1.0, 1.0, 1.0);
|
|
}
|
|
|
|
void Sbar_ChatModeOverlay(playerview_t *pv)
|
|
{
|
|
int start =0;
|
|
int i, k, l;
|
|
int top, bottom;
|
|
int x, y;
|
|
player_info_t *s;
|
|
char team[5];
|
|
int skip = 10;
|
|
|
|
if (largegame)
|
|
skip = 8;
|
|
|
|
// request new ping times every two second
|
|
if (realtime - cl.last_ping_request > 2 && cls.protocol == CP_QUAKEWORLD && cls.demoplayback != DPB_EZTV)
|
|
{
|
|
cl.last_ping_request = realtime;
|
|
CL_SendClientCommand(true, "pings");
|
|
}
|
|
|
|
// scores
|
|
Sbar_SortFrags (true, false);
|
|
|
|
if (Cam_TrackNum(pv)>=0)
|
|
Q_strncpyz (team, cl.players[Cam_TrackNum(pv)].team, sizeof(team));
|
|
else if (pv->playernum>=0 && pv->playernum<MAX_CLIENTS)
|
|
Q_strncpyz (team, cl.players[pv->playernum].team, sizeof(team));
|
|
else
|
|
*team = '\0';
|
|
|
|
// draw the text
|
|
l = scoreboardlines;
|
|
|
|
if (start)
|
|
y = start;
|
|
else
|
|
y = 24;
|
|
y = vid.height/2;
|
|
|
|
x = 4;
|
|
Draw_FunString ( x , y, "name");
|
|
y += 8;
|
|
Draw_FunString ( x , y, "\x1d\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1e\x1f");
|
|
y += 8;
|
|
|
|
for (i=0 ; i<l && y <= vid.height-10 ; i++)
|
|
{
|
|
k = fragsort[i];
|
|
s = &cl.players[k];
|
|
if (!s->name[0])
|
|
continue;
|
|
|
|
// draw background
|
|
top = Sbar_TopColour(s);
|
|
bottom = Sbar_BottomColour(s);
|
|
|
|
if (largegame)
|
|
Sbar_FillPC ( x, y+1, 8*4, 3, top);
|
|
else
|
|
Sbar_FillPC ( x, y, 8*4, 4, top);
|
|
Sbar_FillPC ( x, y+4, 8*4, 4, bottom);
|
|
/*
|
|
if (cl.spectator && k == Cam_TrackNum(pv))
|
|
{
|
|
Draw_Character ( x, y, 16);
|
|
Draw_Character ( x+8*3, y, 17);
|
|
}
|
|
else if (!cl.spectator && k == pv->cl.playernum)
|
|
{
|
|
Draw_Character ( x, y, 16);
|
|
Draw_Character ( x+8*3, y, 17);
|
|
}
|
|
else if (cl.teamplay)
|
|
{
|
|
if (!stricmp(s->team, team))
|
|
{
|
|
Draw_Character ( x, y, '[');
|
|
Draw_Character ( x+8*3, y, ']');
|
|
}
|
|
}
|
|
*/
|
|
// draw name
|
|
if (cl.teamplay)
|
|
Draw_FunString (x+8*4, y, s->name);
|
|
else
|
|
Draw_FunString (x+8*4, y, s->name);
|
|
|
|
y += skip;
|
|
}
|
|
|
|
if (y >= vid.height-10) // we ran over the screen size, squish
|
|
largegame = true;
|
|
}
|
|
|
|
/*
|
|
==================
|
|
Sbar_MiniDeathmatchOverlay
|
|
|
|
frags name
|
|
frags team name
|
|
displayed to right of status bar if there's room
|
|
==================
|
|
*/
|
|
static void Sbar_MiniDeathmatchOverlay (playerview_t *pv)
|
|
{
|
|
int i, k;
|
|
int top, bottom;
|
|
int x, y, f, px, py;
|
|
char num[12];
|
|
player_info_t *s;
|
|
int numlines;
|
|
char name[64+1];
|
|
team_t *tm;
|
|
|
|
// scores
|
|
Sbar_SortFrags (false, false);
|
|
if (sbar_rect.width >= 640)
|
|
Sbar_SortTeams(pv);
|
|
|
|
if (!scoreboardlines)
|
|
return; // no one there?
|
|
|
|
// draw the text
|
|
y = sbar_rect.y + sbar_rect.height - sb_lines - 1;
|
|
numlines = sb_lines/8;
|
|
if (numlines < 3)
|
|
return; // not enough room
|
|
|
|
// find us
|
|
for (i=0 ; i < scoreboardlines; i++)
|
|
if (fragsort[i] == pv->playernum)
|
|
break;
|
|
|
|
if (i == scoreboardlines) // we're not there, we are probably a spectator, just display top
|
|
i = 0;
|
|
else // figure out start
|
|
i = i - numlines/2;
|
|
|
|
if (i > scoreboardlines - numlines)
|
|
i = scoreboardlines - numlines;
|
|
if (i < 0)
|
|
i = 0;
|
|
|
|
x = sbar_rect.x + 320 + 4;
|
|
|
|
for (/* */ ; i < scoreboardlines && y < sbar_rect.y + sbar_rect.height - 8 + 1; i++)
|
|
{
|
|
k = fragsort[i];
|
|
s = &cl.players[k];
|
|
if (!s->name[0])
|
|
continue;
|
|
|
|
// draw ping
|
|
top = Sbar_TopColour(s);
|
|
bottom = Sbar_BottomColour(s);
|
|
|
|
Sbar_FillPC ( x, y+1, 40, 3, top);
|
|
Sbar_FillPC ( x, y+4, 40, 4, bottom);
|
|
|
|
// draw number
|
|
f = s->frags;
|
|
sprintf (num, "%3i",f);
|
|
|
|
Font_BeginString(font_conchar, x+8, y, &px, &py);
|
|
Font_DrawChar ( px, py, num[0] | 0xe000 | CON_WHITEMASK);
|
|
Font_BeginString(font_conchar, x+16, y, &px, &py);
|
|
Font_DrawChar ( px, py, num[1] | 0xe000 | CON_WHITEMASK);
|
|
Font_BeginString(font_conchar, x+24, y, &px, &py);
|
|
Font_DrawChar ( px, py, num[2] | 0xe000 | CON_WHITEMASK);
|
|
|
|
if ((cl.spectator && k == pv->cam_spec_track) ||
|
|
(!cl.spectator && k == pv->playernum))
|
|
{
|
|
Font_BeginString(font_conchar, x, y, &px, &py);
|
|
Font_DrawChar ( px, py, 16 | 0xe000 | CON_WHITEMASK);
|
|
Font_BeginString(font_conchar, x+32, y, &px, &py);
|
|
Font_DrawChar ( px, py, 17 | 0xe000 | CON_WHITEMASK);
|
|
}
|
|
|
|
Q_strncpyz(name, s->name, sizeof(name));
|
|
// team and name
|
|
if (cl.teamplay)
|
|
{
|
|
Draw_FunStringWidth (x+48, y, s->team, 32, false, false);
|
|
Draw_FunStringWidth (x+48+40, y, name, MAX_DISPLAYEDNAME*8, false, false);
|
|
}
|
|
else
|
|
Draw_FunStringWidth (x+48, y, name, MAX_DISPLAYEDNAME*8, false, false);
|
|
y += 8;
|
|
}
|
|
|
|
// draw teams if room
|
|
if (sbar_rect.width < 640 || !cl.teamplay)
|
|
return;
|
|
|
|
// draw seperator
|
|
x += 208;
|
|
// for (y = sbar_rect.height - sb_lines; y < sbar_rect.height - 6; y += 2)
|
|
// Draw_ColouredCharacter(x, y, CON_WHITEMASK|14);
|
|
|
|
x += 16;
|
|
|
|
y = sbar_rect.height - sb_lines;
|
|
for (i=0 ; i < scoreboardteams && y <= sbar_rect.height; i++)
|
|
{
|
|
k = teamsort[i];
|
|
tm = teams + k;
|
|
|
|
// draw pings
|
|
Draw_FunStringWidth (x, y, tm->team, 32, false, false);
|
|
|
|
// draw total
|
|
sprintf (num, "%5i", tm->frags);
|
|
Draw_FunString(x + 40, y, num);
|
|
|
|
if (!strncmp(cl.players[pv->playernum].team, tm->team, 16))
|
|
{
|
|
Font_BeginString(font_conchar, x-8, y, &px, &py);
|
|
Font_DrawChar(px, py, 16|0xe000|CON_WHITEMASK);
|
|
Font_BeginString(font_conchar, x+32, y, &px, &py);
|
|
Font_DrawChar(px, py, 17|0xe000|CON_WHITEMASK);
|
|
Font_EndString(font_conchar);
|
|
}
|
|
|
|
y += 8;
|
|
}
|
|
|
|
}
|
|
|
|
void Sbar_CoopIntermission (void)
|
|
{
|
|
mpic_t *pic;
|
|
int dig;
|
|
int num;
|
|
int pnum = 0; //should be the same for all players.
|
|
|
|
sbar_rect.width = vid.width;
|
|
sbar_rect.height = vid.height;
|
|
sbar_rect.x = 0;
|
|
sbar_rect.y = 0;
|
|
|
|
pic = R2D_SafeCachePic ("gfx/complete.lmp");
|
|
if (!pic)
|
|
return;
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 64, (sbar_rect.height - 200)/2 + 24, 192, 24, pic);
|
|
|
|
pic = R2D_SafeCachePic ("gfx/inter.lmp");
|
|
if (pic)
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 0, (sbar_rect.height - 200)/2 + 56, 160, 144, pic);
|
|
|
|
// time
|
|
dig = cl.completed_time/60;
|
|
Sbar_IntermissionNumber ((sbar_rect.width - 320)/2 + 230 - 24*4, (sbar_rect.height - 200)/2 + 64, dig, 4, 0, false);
|
|
num = cl.completed_time - dig*60;
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 230,(sbar_rect.height - 200)/2 + 64, 16, 24, sb_colon);
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 254,(sbar_rect.height - 200)/2 + 64, 16, 26, sb_nums[0][num/10]);
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 278,(sbar_rect.height - 200)/2 + 64, 16, 24, sb_nums[0][num%10]);
|
|
|
|
//it is assumed that secrits/monsters are going to be constant for any player...
|
|
Sbar_IntermissionNumber ((sbar_rect.width - 320)/2 + 230 - 24*4, (sbar_rect.height - 200)/2 + 104, cl.playerview[pnum].stats[STAT_SECRETS], 4, 0, false);
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 230, (sbar_rect.height - 200)/2 + 104, 16, 24, sb_slash);
|
|
Sbar_IntermissionNumber ((sbar_rect.width - 320)/2 + 254, (sbar_rect.height - 200)/2 + 104, cl.playerview[pnum].stats[STAT_TOTALSECRETS], 4, 0, true);
|
|
|
|
Sbar_IntermissionNumber ((sbar_rect.width - 320)/2 + 230 - 24*4, (sbar_rect.height - 200)/2 + 144, cl.playerview[pnum].stats[STAT_MONSTERS], 4, 0, false);
|
|
R2D_ScalePic ((sbar_rect.width - 320)/2 + 230,(sbar_rect.height - 200)/2 + 144, 16, 24, sb_slash);
|
|
Sbar_IntermissionNumber ((sbar_rect.width - 320)/2 + 254, (sbar_rect.height - 200)/2 + 144, cl.playerview[pnum].stats[STAT_TOTALMONSTERS], 4, 0, true);
|
|
}
|
|
/*
|
|
==================
|
|
Sbar_IntermissionOverlay
|
|
|
|
==================
|
|
*/
|
|
void Sbar_IntermissionOverlay (void)
|
|
{
|
|
#ifdef VM_UI
|
|
if (UI_DrawIntermission()>0)
|
|
return;
|
|
#endif
|
|
|
|
Sbar_Start();
|
|
|
|
if (!cls.deathmatch)
|
|
Sbar_CoopIntermission();
|
|
else if (cl.teamplay > 0 && !sb_showscores)
|
|
Sbar_TeamOverlay ();
|
|
else
|
|
Sbar_DeathmatchOverlay (0);
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Sbar_FinaleOverlay
|
|
|
|
==================
|
|
*/
|
|
void Sbar_FinaleOverlay (void)
|
|
{
|
|
#ifdef VM_UI
|
|
if (UI_DrawFinale()>0)
|
|
return;
|
|
#endif
|
|
}
|
|
|