/*
===========================================================================
Copyright (C) 1999 - 2005, Id Software, Inc.
Copyright (C) 2000 - 2013, Raven Software, Inc.
Copyright (C) 2001 - 2013, Activision, Inc.
Copyright (C) 2013 - 2015, OpenJK contributors
This file is part of the OpenJK source code.
OpenJK is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, see .
===========================================================================
*/
// cg_draw.c -- draw all of the graphical elements during
// active (after loading) gameplay
#include "cg_local.h"
#include "game/bg_saga.h"
#include "ui/ui_shared.h"
#include "ui/ui_public.h"
extern float CG_RadiusForCent( centity_t *cent );
qboolean CG_WorldCoordToScreenCoordFloat(vec3_t worldCoord, float *x, float *y);
qboolean CG_CalcMuzzlePoint( int entityNum, vec3_t muzzle );
static void CG_DrawSiegeTimer(int timeRemaining, qboolean isMyTeam);
static void CG_DrawSiegeDeathTimer( int timeRemaining );
// nmckenzie: DUEL_HEALTH
void CG_DrawDuelistHealth ( float x, float y, float w, float h, int duelist );
// used for scoreboard
extern displayContextDef_t cgDC;
int sortedTeamPlayers[TEAM_MAXOVERLAY];
int numSortedTeamPlayers;
int lastvalidlockdif;
extern float zoomFov; //this has to be global client-side
char systemChat[256];
char teamChat1[256];
char teamChat2[256];
// The time at which you died and the time it will take for you to rejoin game.
int cg_siegeDeathTime = 0;
#define MAX_HUD_TICS 4
const char *armorTicName[MAX_HUD_TICS] =
{
"armor_tic1",
"armor_tic2",
"armor_tic3",
"armor_tic4",
};
const char *healthTicName[MAX_HUD_TICS] =
{
"health_tic1",
"health_tic2",
"health_tic3",
"health_tic4",
};
const char *forceTicName[MAX_HUD_TICS] =
{
"force_tic1",
"force_tic2",
"force_tic3",
"force_tic4",
};
const char *ammoTicName[MAX_HUD_TICS] =
{
"ammo_tic1",
"ammo_tic2",
"ammo_tic3",
"ammo_tic4",
};
char *showPowersName[] =
{
"HEAL2",//FP_HEAL
"JUMP2",//FP_LEVITATION
"SPEED2",//FP_SPEED
"PUSH2",//FP_PUSH
"PULL2",//FP_PULL
"MINDTRICK2",//FP_TELEPTAHY
"GRIP2",//FP_GRIP
"LIGHTNING2",//FP_LIGHTNING
"DARK_RAGE2",//FP_RAGE
"PROTECT2",//FP_PROTECT
"ABSORB2",//FP_ABSORB
"TEAM_HEAL2",//FP_TEAM_HEAL
"TEAM_REPLENISH2",//FP_TEAM_FORCE
"DRAIN2",//FP_DRAIN
"SEEING2",//FP_SEE
"SABER_OFFENSE2",//FP_SABER_OFFENSE
"SABER_DEFENSE2",//FP_SABER_DEFENSE
"SABER_THROW2",//FP_SABERTHROW
NULL
};
//Called from UI shared code. For now we'll just redirect to the normal anim load function.
int UI_ParseAnimationFile(const char *filename, animation_t *animset, qboolean isHumanoid)
{
return BG_ParseAnimationFile(filename, animset, isHumanoid);
}
int MenuFontToHandle(int iMenuFont)
{
switch (iMenuFont)
{
case FONT_SMALL: return cgDC.Assets.qhSmallFont;
case FONT_SMALL2: return cgDC.Assets.qhSmall2Font;
case FONT_MEDIUM: return cgDC.Assets.qhMediumFont;
case FONT_LARGE: return cgDC.Assets.qhMediumFont;//cgDC.Assets.qhBigFont;
//fixme? Big fonr isn't registered...?
}
return cgDC.Assets.qhMediumFont;
}
int CG_Text_Width(const char *text, float scale, int iMenuFont)
{
int iFontIndex = MenuFontToHandle(iMenuFont);
return trap->R_Font_StrLenPixels(text, iFontIndex, scale);
}
int CG_Text_Height(const char *text, float scale, int iMenuFont)
{
int iFontIndex = MenuFontToHandle(iMenuFont);
return trap->R_Font_HeightPixels(iFontIndex, scale);
}
#include "qcommon/qfiles.h" // for STYLE_BLINK etc
void CG_Text_Paint(float x, float y, float scale, vec4_t color, const char *text, float adjust, int limit, int style, int iMenuFont)
{
int iStyleOR = 0;
int iFontIndex = MenuFontToHandle(iMenuFont);
switch (style)
{
case ITEM_TEXTSTYLE_NORMAL: iStyleOR = 0;break; // JK2 normal text
case ITEM_TEXTSTYLE_BLINK: iStyleOR = STYLE_BLINK;break; // JK2 fast blinking
case ITEM_TEXTSTYLE_PULSE: iStyleOR = STYLE_BLINK;break; // JK2 slow pulsing
case ITEM_TEXTSTYLE_SHADOWED: iStyleOR = (int)STYLE_DROPSHADOW;break; // JK2 drop shadow ( need a color for this )
case ITEM_TEXTSTYLE_OUTLINED: iStyleOR = (int)STYLE_DROPSHADOW;break; // JK2 drop shadow ( need a color for this )
case ITEM_TEXTSTYLE_OUTLINESHADOWED: iStyleOR = (int)STYLE_DROPSHADOW;break; // JK2 drop shadow ( need a color for this )
case ITEM_TEXTSTYLE_SHADOWEDMORE: iStyleOR = (int)STYLE_DROPSHADOW;break; // JK2 drop shadow ( need a color for this )
}
trap->R_Font_DrawString( x, // int ox
y, // int oy
text, // const char *text
color, // paletteRGBA_c c
iStyleOR | iFontIndex, // const int iFontHandle
!limit?-1:limit, // iCharLimit (-1 = none)
scale // const float scale = 1.0f
);
}
/*
================
CG_DrawZoomMask
================
*/
static void CG_DrawZoomMask( void )
{
vec4_t color1;
float level;
static qboolean flip = qtrue;
// int ammo = cg_entities[0].gent->client->ps.ammo[weaponData[cent->currentState.weapon].ammoIndex];
float cx, cy;
// int val[5];
float max, fi;
// Check for Binocular specific zooming since we'll want to render different bits in each case
if ( cg.predictedPlayerState.zoomMode == 2 )
{
int val, i;
float off;
// zoom level
level = (float)(80.0f - cg.predictedPlayerState.zoomFov) / 80.0f;
// ...so we'll clamp it
if ( level < 0.0f )
{
level = 0.0f;
}
else if ( level > 1.0f )
{
level = 1.0f;
}
// Using a magic number to convert the zoom level to scale amount
level *= 162.0f;
// draw blue tinted distortion mask, trying to make it as small as is necessary to fill in the viewable area
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic( 34, 48, 570, 362, cgs.media.binocularStatic );
// Black out the area behind the numbers
trap->R_SetColor( colorTable[CT_BLACK]);
CG_DrawPic( 212, 367, 200, 40, cgs.media.whiteShader );
// Numbers should be kind of greenish
color1[0] = 0.2f;
color1[1] = 0.4f;
color1[2] = 0.2f;
color1[3] = 0.3f;
trap->R_SetColor( color1 );
// Draw scrolling numbers, use intervals 10 units apart--sorry, this section of code is just kind of hacked
// up with a bunch of magic numbers.....
val = ((int)((cg.refdef.viewangles[YAW] + 180) / 10)) * 10;
off = (cg.refdef.viewangles[YAW] + 180) - val;
for ( i = -10; i < 30; i += 10 )
{
val -= 10;
if ( val < 0 )
{
val += 360;
}
// we only want to draw the very far left one some of the time, if it's too far to the left it will
// poke outside the mask.
if (( off > 3.0f && i == -10 ) || i > -10 )
{
// draw the value, but add 200 just to bump the range up...arbitrary, so change it if you like
CG_DrawNumField( 155 + i * 10 + off * 10, 374, 3, val + 200, 24, 14, NUM_FONT_CHUNKY, qtrue );
CG_DrawPic( 245 + (i-1) * 10 + off * 10, 376, 6, 6, cgs.media.whiteShader );
}
}
CG_DrawPic( 212, 367, 200, 28, cgs.media.binocularOverlay );
color1[0] = sin( cg.time * 0.01f ) * 0.5f + 0.5f;
color1[0] = color1[0] * color1[0];
color1[1] = color1[0];
color1[2] = color1[0];
color1[3] = 1.0f;
trap->R_SetColor( color1 );
CG_DrawPic( 82, 94, 16, 16, cgs.media.binocularCircle );
// Flickery color
color1[0] = 0.7f + Q_flrand(-0.1f, 0.1f);
color1[1] = 0.8f + Q_flrand(-0.1f, 0.1f);
color1[2] = 0.7f + Q_flrand(-0.1f, 0.1f);
color1[3] = 1.0f;
trap->R_SetColor( color1 );
CG_DrawPic( 0, 0, 640, 480, cgs.media.binocularMask );
CG_DrawPic( 4, 282 - level, 16, 16, cgs.media.binocularArrow );
// The top triangle bit randomly flips
if ( flip )
{
CG_DrawPic( 330, 60, -26, -30, cgs.media.binocularTri );
}
else
{
CG_DrawPic( 307, 40, 26, 30, cgs.media.binocularTri );
}
if ( Q_flrand(0.0f, 1.0f) > 0.98f && ( cg.time & 1024 ))
{
flip = !flip;
}
}
else if ( cg.predictedPlayerState.zoomMode)
{
// disruptor zoom mode
level = (float)(50.0f - zoomFov) / 50.0f;//(float)(80.0f - zoomFov) / 80.0f;
// ...so we'll clamp it
if ( level < 0.0f )
{
level = 0.0f;
}
else if ( level > 1.0f )
{
level = 1.0f;
}
// Using a magic number to convert the zoom level to a rotation amount that correlates more or less with the zoom artwork.
level *= 103.0f;
// Draw target mask
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic( 0, 0, 640, 480, cgs.media.disruptorMask );
// apparently 99.0f is the full zoom level
if ( level >= 99 )
{
// Fully zoomed, so make the rotating insert pulse
color1[0] = 1.0f;
color1[1] = 1.0f;
color1[2] = 1.0f;
color1[3] = 0.7f + sin( cg.time * 0.01f ) * 0.3f;
trap->R_SetColor( color1 );
}
// Draw rotating insert
CG_DrawRotatePic2( 320, 240, 640, 480, -level, cgs.media.disruptorInsert );
// Increase the light levels under the center of the target
// CG_DrawPic( 198, 118, 246, 246, cgs.media.disruptorLight );
// weirdness.....converting ammo to a base five number scale just to be geeky.
/* val[0] = ammo % 5;
val[1] = (ammo / 5) % 5;
val[2] = (ammo / 25) % 5;
val[3] = (ammo / 125) % 5;
val[4] = (ammo / 625) % 5;
color1[0] = 0.2f;
color1[1] = 0.55f + Q_flrand(-1.0f, 1.0f) * 0.1f;
color1[2] = 0.5f + Q_flrand(-1.0f, 1.0f) * 0.1f;
color1[3] = 1.0f;
trap->R_SetColor( color1 );
for ( int t = 0; t < 5; t++ )
{
cx = 320 + sin( (t*10+45)/57.296f ) * 192;
cy = 240 + cos( (t*10+45)/57.296f ) * 192;
CG_DrawRotatePic2( cx, cy, 24, 38, 45 - t * 10, trap->R_RegisterShader( va("gfx/2d/char%d",val[4-t] )));
}
*/
//max = ( cg_entities[0].gent->health / 100.0f );
if ( (cg.snap->ps.eFlags & EF_DOUBLE_AMMO) )
{
max = cg.snap->ps.ammo[weaponData[WP_DISRUPTOR].ammoIndex] / ((float)ammoData[weaponData[WP_DISRUPTOR].ammoIndex].max*2.0f);
}
else
{
max = cg.snap->ps.ammo[weaponData[WP_DISRUPTOR].ammoIndex] / (float)ammoData[weaponData[WP_DISRUPTOR].ammoIndex].max;
}
if ( max > 1.0f )
{
max = 1.0f;
}
color1[0] = (1.0f - max) * 2.0f;
color1[1] = max * 1.5f;
color1[2] = 0.0f;
color1[3] = 1.0f;
// If we are low on health, make us flash
if ( max < 0.15f && ( cg.time & 512 ))
{
VectorClear( color1 );
}
if ( color1[0] > 1.0f )
{
color1[0] = 1.0f;
}
if ( color1[1] > 1.0f )
{
color1[1] = 1.0f;
}
trap->R_SetColor( color1 );
max *= 58.0f;
for (fi = 18.5f; fi <= 18.5f + max; fi+= 3 ) // going from 15 to 45 degrees, with 5 degree increments
{
cx = 320 + sin( (fi+90.0f)/57.296f ) * 190;
cy = 240 + cos( (fi+90.0f)/57.296f ) * 190;
CG_DrawRotatePic2( cx, cy, 12, 24, 90 - fi, cgs.media.disruptorInsertTick );
}
if ( cg.predictedPlayerState.weaponstate == WEAPON_CHARGING_ALT )
{
trap->R_SetColor( colorTable[CT_WHITE] );
// draw the charge level
max = ( cg.time - cg.predictedPlayerState.weaponChargeTime ) / ( 50.0f * 30.0f ); // bad hardcodedness 50 is disruptor charge unit and 30 is max charge units allowed.
if ( max > 1.0f )
{
max = 1.0f;
}
trap->R_DrawStretchPic(257, 435, 134*max, 34, 0, 0, max, 1, cgs.media.disruptorChargeShader);
}
// trap->R_SetColor( colorTable[CT_WHITE] );
// CG_DrawPic( 0, 0, 640, 480, cgs.media.disruptorMask );
}
}
/*
================
CG_Draw3DModel
================
*/
void CG_Draw3DModel( float x, float y, float w, float h, qhandle_t model, void *ghoul2, int g2radius, qhandle_t skin, vec3_t origin, vec3_t angles ) {
refdef_t refdef;
refEntity_t ent;
if ( !cg_draw3DIcons.integer || !cg_drawIcons.integer ) {
return;
}
memset( &refdef, 0, sizeof( refdef ) );
memset( &ent, 0, sizeof( ent ) );
AnglesToAxis( angles, ent.axis );
VectorCopy( origin, ent.origin );
ent.hModel = model;
ent.ghoul2 = ghoul2;
ent.radius = g2radius;
ent.customSkin = skin;
ent.renderfx = RF_NOSHADOW; // no stencil shadows
refdef.rdflags = RDF_NOWORLDMODEL;
AxisClear( refdef.viewaxis );
refdef.fov_x = 30;
refdef.fov_y = 30;
refdef.x = x;
refdef.y = y;
refdef.width = w;
refdef.height = h;
refdef.time = cg.time;
trap->R_ClearScene();
trap->R_AddRefEntityToScene( &ent );
trap->R_RenderScene( &refdef );
}
/*
================
CG_DrawHead
Used for both the status bar and the scoreboard
================
*/
void CG_DrawHead( float x, float y, float w, float h, int clientNum, vec3_t headAngles )
{
clientInfo_t *ci;
if (clientNum >= MAX_CLIENTS)
{ //npc?
return;
}
ci = &cgs.clientinfo[ clientNum ];
CG_DrawPic( x, y, w, h, ci->modelIcon );
// if they are deferred, draw a cross out
if ( ci->deferred )
{
CG_DrawPic( x, y, w, h, cgs.media.deferShader );
}
}
/*
================
CG_DrawFlagModel
Used for both the status bar and the scoreboard
================
*/
void CG_DrawFlagModel( float x, float y, float w, float h, int team, qboolean force2D ) {
qhandle_t cm;
float len;
vec3_t origin, angles;
vec3_t mins, maxs;
qhandle_t handle;
if ( !force2D && cg_draw3DIcons.integer ) {
x *= cgs.screenXScale;
y *= cgs.screenYScale;
w *= cgs.screenXScale;
h *= cgs.screenYScale;
VectorClear( angles );
cm = cgs.media.redFlagModel;
// offset the origin y and z to center the flag
trap->R_ModelBounds( cm, mins, maxs );
origin[2] = -0.5 * ( mins[2] + maxs[2] );
origin[1] = 0.5 * ( mins[1] + maxs[1] );
// calculate distance so the flag nearly fills the box
// assume heads are taller than wide
len = 0.5 * ( maxs[2] - mins[2] );
origin[0] = len / 0.268; // len / tan( fov/2 )
angles[YAW] = 60 * sin( cg.time / 2000.0 );;
if( team == TEAM_RED ) {
handle = cgs.media.redFlagModel;
} else if( team == TEAM_BLUE ) {
handle = cgs.media.blueFlagModel;
} else if( team == TEAM_FREE ) {
handle = 0;//cgs.media.neutralFlagModel;
} else {
return;
}
CG_Draw3DModel( x, y, w, h, handle, NULL, 0, 0, origin, angles );
} else if ( cg_drawIcons.integer ) {
gitem_t *item;
if( team == TEAM_RED ) {
item = BG_FindItemForPowerup( PW_REDFLAG );
} else if( team == TEAM_BLUE ) {
item = BG_FindItemForPowerup( PW_BLUEFLAG );
} else if( team == TEAM_FREE ) {
item = BG_FindItemForPowerup( PW_NEUTRALFLAG );
} else {
return;
}
if (item) {
CG_DrawPic( x, y, w, h, cg_items[ ITEM_INDEX(item) ].icon );
}
}
}
/*
================
CG_DrawHealth
================
*/
void CG_DrawHealth( menuDef_t *menuHUD )
{
vec4_t calcColor;
playerState_t *ps;
int healthAmt;
int i,currValue,inc;
itemDef_t *focusItem;
float percent;
// Can we find the menu?
if (!menuHUD)
{
return;
}
ps = &cg.snap->ps;
// What's the health?
healthAmt = ps->stats[STAT_HEALTH];
if (healthAmt > ps->stats[STAT_MAX_HEALTH])
{
healthAmt = ps->stats[STAT_MAX_HEALTH];
}
inc = (float) ps->stats[STAT_MAX_HEALTH] / MAX_HUD_TICS;
currValue = healthAmt;
// Print the health tics, fading out the one which is partial health
for (i=(MAX_HUD_TICS-1);i>=0;i--)
{
focusItem = Menu_FindItemByName(menuHUD, healthTicName[i]);
if (!focusItem) // This is bad
{
continue;
}
memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
percent = (float) currValue / inc;
calcColor[3] *= percent; // Fade it out
}
trap->R_SetColor( calcColor);
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
currValue -= inc;
}
// Print the mueric amount
focusItem = Menu_FindItemByName(menuHUD, "healthamount");
if (focusItem)
{
// Print health amount
trap->R_SetColor( focusItem->window.foreColor );
CG_DrawNumField (
focusItem->window.rect.x,
focusItem->window.rect.y,
3,
ps->stats[STAT_HEALTH],
focusItem->window.rect.w,
focusItem->window.rect.h,
NUM_FONT_SMALL,
qfalse);
}
}
/*
================
CG_DrawArmor
================
*/
void CG_DrawArmor( menuDef_t *menuHUD )
{
vec4_t calcColor;
playerState_t *ps;
int maxArmor;
itemDef_t *focusItem;
float percent,quarterArmor;
int i,currValue,inc;
//ps = &cg.snap->ps;
ps = &cg.predictedPlayerState;
// Can we find the menu?
if (!menuHUD)
{
return;
}
maxArmor = ps->stats[STAT_MAX_HEALTH];
currValue = ps->stats[STAT_ARMOR];
inc = (float) maxArmor / MAX_HUD_TICS;
memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
for (i=(MAX_HUD_TICS-1);i>=0;i--)
{
focusItem = Menu_FindItemByName(menuHUD, armorTicName[i]);
if (!focusItem) // This is bad
{
continue;
}
memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
percent = (float) currValue / inc;
calcColor[3] *= percent;
}
trap->R_SetColor( calcColor);
if ((i==(MAX_HUD_TICS-1)) && (currValue < inc))
{
if (cg.HUDArmorFlag)
{
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
}
else
{
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
currValue -= inc;
}
focusItem = Menu_FindItemByName(menuHUD, "armoramount");
if (focusItem)
{
// Print armor amount
trap->R_SetColor( focusItem->window.foreColor );
CG_DrawNumField (
focusItem->window.rect.x,
focusItem->window.rect.y,
3,
ps->stats[STAT_ARMOR],
focusItem->window.rect.w,
focusItem->window.rect.h,
NUM_FONT_SMALL,
qfalse);
}
// If armor is low, flash a graphic to warn the player
if (ps->stats[STAT_ARMOR]) // Is there armor? Draw the HUD Armor TIC
{
quarterArmor = (float) (ps->stats[STAT_MAX_HEALTH] / 4.0f);
// Make tic flash if armor is at 25% of full armor
if (ps->stats[STAT_ARMOR] < quarterArmor) // Do whatever the flash timer says
{
if (cg.HUDTickFlashTime < cg.time) // Flip at the same time
{
cg.HUDTickFlashTime = cg.time + 400;
if (cg.HUDArmorFlag)
{
cg.HUDArmorFlag = qfalse;
}
else
{
cg.HUDArmorFlag = qtrue;
}
}
}
else
{
cg.HUDArmorFlag=qtrue;
}
}
else // No armor? Don't show it.
{
cg.HUDArmorFlag=qfalse;
}
}
/*
================
CG_DrawSaberStyle
If the weapon is a light saber (which needs no ammo) then draw a graphic showing
the saber style (fast, medium, strong)
================
*/
static void CG_DrawSaberStyle( centity_t *cent, menuDef_t *menuHUD)
{
itemDef_t *focusItem;
if (!cent->currentState.weapon ) // We don't have a weapon right now
{
return;
}
if ( cent->currentState.weapon != WP_SABER )
{
return;
}
// Can we find the menu?
if (!menuHUD)
{
return;
}
// draw the current saber style in this window
switch ( cg.predictedPlayerState.fd.saberDrawAnimLevel )
{
case 1://FORCE_LEVEL_1:
case 5://FORCE_LEVEL_5://Tavion
focusItem = Menu_FindItemByName(menuHUD, "saberstyle_fast");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
break;
case 2://FORCE_LEVEL_2:
case 6://SS_DUAL
case 7://SS_STAFF
focusItem = Menu_FindItemByName(menuHUD, "saberstyle_medium");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
break;
case 3://FORCE_LEVEL_3:
case 4://FORCE_LEVEL_4://Desann
focusItem = Menu_FindItemByName(menuHUD, "saberstyle_strong");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
break;
}
}
/*
================
CG_DrawAmmo
================
*/
static void CG_DrawAmmo( centity_t *cent,menuDef_t *menuHUD)
{
playerState_t *ps;
int i;
vec4_t calcColor;
float value=0.0f,inc = 0.0f,percent;
itemDef_t *focusItem;
ps = &cg.snap->ps;
// Can we find the menu?
if (!menuHUD)
{
return;
}
if (!cent->currentState.weapon ) // We don't have a weapon right now
{
return;
}
value = ps->ammo[weaponData[cent->currentState.weapon].ammoIndex];
if (value < 0) // No ammo
{
return;
}
//
// ammo
//
if (cg.oldammo < value)
{
cg.oldAmmoTime = cg.time + 200;
}
cg.oldammo = value;
focusItem = Menu_FindItemByName(menuHUD, "ammoamount");
if (weaponData[cent->currentState.weapon].energyPerShot == 0 &&
weaponData[cent->currentState.weapon].altEnergyPerShot == 0)
{ //just draw "infinite"
inc = 8 / MAX_HUD_TICS;
value = 8;
focusItem = Menu_FindItemByName(menuHUD, "ammoinfinite");
trap->R_SetColor( colorTable[CT_YELLOW] );
if (focusItem)
{
CG_DrawProportionalString(focusItem->window.rect.x, focusItem->window.rect.y, "--", NUM_FONT_SMALL, focusItem->window.foreColor);
}
}
else
{
focusItem = Menu_FindItemByName(menuHUD, "ammoamount");
// Firing or reloading?
if (( cg.predictedPlayerState.weaponstate == WEAPON_FIRING
&& cg.predictedPlayerState.weaponTime > 100 ))
{
memcpy(calcColor, colorTable[CT_LTGREY], sizeof(vec4_t));
}
else
{
if ( value > 0 )
{
if (cg.oldAmmoTime > cg.time)
{
memcpy(calcColor, colorTable[CT_YELLOW], sizeof(vec4_t));
}
else
{
memcpy(calcColor, focusItem->window.foreColor, sizeof(vec4_t));
}
}
else
{
memcpy(calcColor, colorTable[CT_RED], sizeof(vec4_t));
}
}
trap->R_SetColor( calcColor );
if (focusItem)
{
if ( (cent->currentState.eFlags & EF_DOUBLE_AMMO) )
{
inc = (float) (ammoData[weaponData[cent->currentState.weapon].ammoIndex].max*2.0f) / MAX_HUD_TICS;
}
else
{
inc = (float) ammoData[weaponData[cent->currentState.weapon].ammoIndex].max / MAX_HUD_TICS;
}
value =ps->ammo[weaponData[cent->currentState.weapon].ammoIndex];
CG_DrawNumField (
focusItem->window.rect.x,
focusItem->window.rect.y,
3,
value,
focusItem->window.rect.w,
focusItem->window.rect.h,
NUM_FONT_SMALL,
qfalse);
}
}
trap->R_SetColor( colorTable[CT_WHITE] );
// Draw tics
for (i=MAX_HUD_TICS-1;i>=0;i--)
{
focusItem = Menu_FindItemByName(menuHUD, ammoTicName[i]);
if (!focusItem)
{
continue;
}
memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
if ( value <= 0 ) // done
{
break;
}
else if (value < inc) // partial tic
{
percent = value / inc;
calcColor[3] = percent;
}
trap->R_SetColor( calcColor);
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
value -= inc;
}
}
/*
================
CG_DrawForcePower
================
*/
void CG_DrawForcePower( menuDef_t *menuHUD )
{
int i;
vec4_t calcColor;
float value,inc,percent;
itemDef_t *focusItem;
const int maxForcePower = 100;
qboolean flash=qfalse;
// Can we find the menu?
if (!menuHUD)
{
return;
}
// Make the hud flash by setting forceHUDTotalFlashTime above cg.time
if (cg.forceHUDTotalFlashTime > cg.time )
{
flash = qtrue;
if (cg.forceHUDNextFlashTime < cg.time)
{
cg.forceHUDNextFlashTime = cg.time + 400;
trap->S_StartSound (NULL, 0, CHAN_LOCAL, cgs.media.noforceSound );
if (cg.forceHUDActive)
{
cg.forceHUDActive = qfalse;
}
else
{
cg.forceHUDActive = qtrue;
}
}
}
else // turn HUD back on if it had just finished flashing time.
{
cg.forceHUDNextFlashTime = 0;
cg.forceHUDActive = qtrue;
}
// if (!cg.forceHUDActive)
// {
// return;
// }
inc = (float) maxForcePower / MAX_HUD_TICS;
value = cg.snap->ps.fd.forcePower;
for (i=MAX_HUD_TICS-1;i>=0;i--)
{
focusItem = Menu_FindItemByName(menuHUD, forceTicName[i]);
if (!focusItem)
{
continue;
}
// memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
if ( value <= 0 ) // done
{
break;
}
else if (value < inc) // partial tic
{
if (flash)
{
memcpy(calcColor, colorTable[CT_RED], sizeof(vec4_t));
}
else
{
memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
}
percent = value / inc;
calcColor[3] = percent;
}
else
{
if (flash)
{
memcpy(calcColor, colorTable[CT_RED], sizeof(vec4_t));
}
else
{
memcpy(calcColor, colorTable[CT_WHITE], sizeof(vec4_t));
}
}
trap->R_SetColor( calcColor);
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
value -= inc;
}
focusItem = Menu_FindItemByName(menuHUD, "forceamount");
if (focusItem)
{
// Print force amount
if ( flash )
{
trap->R_SetColor( colorTable[CT_RED] );
}
else
{
trap->R_SetColor( focusItem->window.foreColor );
}
CG_DrawNumField (
focusItem->window.rect.x,
focusItem->window.rect.y,
3,
cg.snap->ps.fd.forcePower,
focusItem->window.rect.w,
focusItem->window.rect.h,
NUM_FONT_SMALL,
qfalse);
}
}
static void CG_DrawSimpleSaberStyle( const centity_t *cent )
{
uint32_t calcColor;
char num[7] = { 0 };
int weapX = 16;
if ( !cent->currentState.weapon ) // We don't have a weapon right now
{
return;
}
if ( cent->currentState.weapon != WP_SABER )
{
return;
}
switch ( cg.predictedPlayerState.fd.saberDrawAnimLevel )
{
default:
case SS_FAST:
Com_sprintf( num, sizeof( num ), "FAST" );
calcColor = CT_ICON_BLUE;
weapX = 0;
break;
case SS_MEDIUM:
Com_sprintf( num, sizeof( num ), "MEDIUM" );
calcColor = CT_YELLOW;
break;
case SS_STRONG:
Com_sprintf( num, sizeof( num ), "STRONG" );
calcColor = CT_HUD_RED;
break;
case SS_DESANN:
Com_sprintf( num, sizeof( num ), "DESANN" );
calcColor = CT_HUD_RED;
break;
case SS_TAVION:
Com_sprintf( num, sizeof( num ), "TAVION" );
calcColor = CT_ICON_BLUE;
break;
case SS_DUAL:
Com_sprintf( num, sizeof( num ), "AKIMBO" );
calcColor = CT_HUD_ORANGE;
break;
case SS_STAFF:
Com_sprintf( num, sizeof( num ), "STAFF" );
calcColor = CT_HUD_ORANGE;
break;
}
CG_DrawProportionalString( SCREEN_WIDTH - (weapX + 16 + 32), (SCREEN_HEIGHT - 80) + 40, num, UI_SMALLFONT | UI_DROPSHADOW, colorTable[calcColor] );
}
static void CG_DrawSimpleAmmo( const centity_t *cent )
{
playerState_t *ps;
uint32_t calcColor;
int currValue = 0;
char num[16] = { 0 };
if ( !cent->currentState.weapon ) // We don't have a weapon right now
{
return;
}
ps = &cg.snap->ps;
currValue = ps->ammo[weaponData[cent->currentState.weapon].ammoIndex];
// No ammo
if ( currValue < 0 || (weaponData[cent->currentState.weapon].energyPerShot == 0 && weaponData[cent->currentState.weapon].altEnergyPerShot == 0) )
{
CG_DrawProportionalString( SCREEN_WIDTH - (16 + 32), (SCREEN_HEIGHT - 80) + 40, "--", UI_SMALLFONT | UI_DROPSHADOW, colorTable[CT_HUD_ORANGE] );
return;
}
//
// ammo
//
if ( cg.oldammo < currValue )
{
cg.oldAmmoTime = cg.time + 200;
}
cg.oldammo = currValue;
// Determine the color of the numeric field
// Firing or reloading?
if ( (cg.predictedPlayerState.weaponstate == WEAPON_FIRING
&& cg.predictedPlayerState.weaponTime > 100) )
{
calcColor = CT_LTGREY;
}
else
{
if ( currValue > 0 )
{
if ( cg.oldAmmoTime > cg.time )
{
calcColor = CT_YELLOW;
}
else
{
calcColor = CT_HUD_ORANGE;
}
}
else
{
calcColor = CT_RED;
}
}
Com_sprintf( num, sizeof( num ), "%i", currValue );
CG_DrawProportionalString( SCREEN_WIDTH - (16 + 32), (SCREEN_HEIGHT - 80) + 40, num, UI_SMALLFONT | UI_DROPSHADOW, colorTable[calcColor] );
}
static void CG_DrawSimpleForcePower( const centity_t *cent )
{
uint32_t calcColor;
char num[16] = { 0 };
qboolean flash = qfalse;
if ( !cg.snap->ps.fd.forcePowersKnown )
{
return;
}
// Make the hud flash by setting forceHUDTotalFlashTime above cg.time
if ( cg.forceHUDTotalFlashTime > cg.time )
{
flash = qtrue;
if ( cg.forceHUDNextFlashTime < cg.time )
{
cg.forceHUDNextFlashTime = cg.time + 400;
trap->S_StartSound( NULL, 0, CHAN_LOCAL, cgs.media.noforceSound );
if ( cg.forceHUDActive )
{
cg.forceHUDActive = qfalse;
}
else
{
cg.forceHUDActive = qtrue;
}
}
}
else // turn HUD back on if it had just finished flashing time.
{
cg.forceHUDNextFlashTime = 0;
cg.forceHUDActive = qtrue;
}
// Determine the color of the numeric field
calcColor = flash ? CT_RED : CT_ICON_BLUE;
Com_sprintf( num, sizeof( num ), "%i", cg.snap->ps.fd.forcePower );
CG_DrawProportionalString( SCREEN_WIDTH - (18 + 14 + 32), (SCREEN_HEIGHT - 80) + 40 + 14, num, UI_SMALLFONT | UI_DROPSHADOW, colorTable[calcColor] );
}
/*
================
CG_DrawHUD
================
*/
void CG_DrawHUD(centity_t *cent)
{
menuDef_t *menuHUD = NULL;
itemDef_t *focusItem = NULL;
const char *scoreStr = NULL;
int scoreBias;
char scoreBiasStr[16];
if (cg_hudFiles.integer)
{
int x = 0;
int y = SCREEN_HEIGHT-80;
if (cg.predictedPlayerState.pm_type != PM_SPECTATOR)
{
CG_DrawProportionalString( x+16, y+40, va( "%i", cg.snap->ps.stats[STAT_HEALTH] ), UI_SMALLFONT|UI_DROPSHADOW, colorTable[CT_HUD_RED] );
CG_DrawProportionalString( x+18+14, y+40+14, va( "%i", cg.snap->ps.stats[STAT_ARMOR] ), UI_SMALLFONT|UI_DROPSHADOW, colorTable[CT_HUD_GREEN] );
CG_DrawSimpleForcePower( cent );
if ( cent->currentState.weapon == WP_SABER )
CG_DrawSimpleSaberStyle( cent );
else
CG_DrawSimpleAmmo( cent );
//TODO Add score line
}
return;
}
if (cg.predictedPlayerState.pm_type != PM_SPECTATOR)
{
// Draw the left HUD
menuHUD = Menus_FindByName("lefthud");
Menu_Paint( menuHUD, qtrue );
if (menuHUD)
{
itemDef_t *focusItem;
// Print scanline
focusItem = Menu_FindItemByName(menuHUD, "scanline");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
// Print frame
focusItem = Menu_FindItemByName(menuHUD, "frame");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
CG_DrawArmor(menuHUD);
CG_DrawHealth(menuHUD);
}
else
{
//trap->Error( ERR_DROP, "CG_ChatBox_ArrayInsert: unable to locate HUD menu file ");
}
//scoreStr = va("Score: %i", cgs.clientinfo[cg.snap->ps.clientNum].score);
if ( cgs.gametype == GT_DUEL )
{//A duel that requires more than one kill to knock the current enemy back to the queue
//show current kills out of how many needed
scoreStr = va("%s: %i/%i", CG_GetStringEdString("MP_INGAME", "SCORE"), cg.snap->ps.persistant[PERS_SCORE], cgs.fraglimit);
}
else if (0 && cgs.gametype < GT_TEAM )
{ // This is a teamless mode, draw the score bias.
scoreBias = cg.snap->ps.persistant[PERS_SCORE] - cgs.scores1;
if (scoreBias == 0)
{ // We are the leader!
if (cgs.scores2 <= 0)
{ // Nobody to be ahead of yet.
Com_sprintf(scoreBiasStr, sizeof(scoreBiasStr), "");
}
else
{
scoreBias = cg.snap->ps.persistant[PERS_SCORE] - cgs.scores2;
if (scoreBias == 0)
{
Com_sprintf(scoreBiasStr, sizeof(scoreBiasStr), " (Tie)");
}
else
{
Com_sprintf(scoreBiasStr, sizeof(scoreBiasStr), " (+%d)", scoreBias);
}
}
}
else // if (scoreBias < 0)
{ // We are behind!
Com_sprintf(scoreBiasStr, sizeof(scoreBiasStr), " (%d)", scoreBias);
}
scoreStr = va("%s: %i%s", CG_GetStringEdString("MP_INGAME", "SCORE"), cg.snap->ps.persistant[PERS_SCORE], scoreBiasStr);
}
else
{ // Don't draw a bias.
scoreStr = va("%s: %i", CG_GetStringEdString("MP_INGAME", "SCORE"), cg.snap->ps.persistant[PERS_SCORE]);
}
menuHUD = Menus_FindByName("righthud");
Menu_Paint( menuHUD, qtrue );
if (menuHUD)
{
if (cgs.gametype != GT_POWERDUEL)
{
focusItem = Menu_FindItemByName(menuHUD, "score_line");
if (focusItem)
{
CG_DrawScaledProportionalString(
focusItem->window.rect.x,
focusItem->window.rect.y,
scoreStr,
UI_RIGHT|UI_DROPSHADOW,
focusItem->window.foreColor,
0.7f);
}
}
// Print scanline
focusItem = Menu_FindItemByName(menuHUD, "scanline");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
focusItem = Menu_FindItemByName(menuHUD, "frame");
if (focusItem)
{
trap->R_SetColor( colorTable[CT_WHITE] );
CG_DrawPic(
focusItem->window.rect.x,
focusItem->window.rect.y,
focusItem->window.rect.w,
focusItem->window.rect.h,
focusItem->window.background
);
}
CG_DrawForcePower(menuHUD);
// Draw ammo tics or saber style
if ( cent->currentState.weapon == WP_SABER )
{
CG_DrawSaberStyle(cent,menuHUD);
}
else
{
CG_DrawAmmo(cent,menuHUD);
}
}
else
{
//trap->Error( ERR_DROP, "CG_ChatBox_ArrayInsert: unable to locate HUD menu file ");
}
}
}
#define MAX_SHOWPOWERS NUM_FORCE_POWERS
qboolean ForcePower_Valid(int i)
{
if (i == FP_LEVITATION ||
i == FP_SABER_OFFENSE ||
i == FP_SABER_DEFENSE ||
i == FP_SABERTHROW)
{
return qfalse;
}
if (cg.snap->ps.fd.forcePowersKnown & (1 << i))
{
return qtrue;
}
return qfalse;
}
/*
===================
CG_DrawForceSelect
===================
*/
void CG_DrawForceSelect( void )
{
int i;
int count;
int smallIconSize,bigIconSize;
int holdX, x, y, pad;
int sideLeftIconCnt,sideRightIconCnt;
int sideMax,holdCount,iconCnt;
int yOffset = 0;
// don't display if dead
if ( cg.snap->ps.stats[STAT_HEALTH] <= 0 )
{
return;
}
if ((cg.forceSelectTime+WEAPON_SELECT_TIME)ps.fd.forcePowerSelected;
return;
}
if (!cg.snap->ps.fd.forcePowersKnown)
{
return;
}
// count the number of powers owned
count = 0;
for (i=0;i < NUM_FORCE_POWERS;++i)
{
if (ForcePower_Valid(i))
{
count++;
}
}
if (count == 0) // If no force powers, don't display
{
return;
}
sideMax = 3; // Max number of icons on the side
// Calculate how many icons will appear to either side of the center one
holdCount = count - 1; // -1 for the center icon
if (holdCount == 0) // No icons to either side
{
sideLeftIconCnt = 0;
sideRightIconCnt = 0;
}
else if (count > (2*sideMax)) // Go to the max on each side
{
sideLeftIconCnt = sideMax;
sideRightIconCnt = sideMax;
}
else // Less than max, so do the calc
{
sideLeftIconCnt = holdCount/2;
sideRightIconCnt = holdCount - sideLeftIconCnt;
}
smallIconSize = 30;
bigIconSize = 60;
pad = 12;
x = 320;
y = 425;
i = BG_ProperForceIndex(cg.forceSelect) - 1;
if (i < 0)
{
i = MAX_SHOWPOWERS - 1;
}
trap->R_SetColor(NULL);
// Work backwards from current icon
holdX = x - ((bigIconSize/2) + pad + smallIconSize);
for (iconCnt=1;iconCnt<(sideLeftIconCnt+1);i--)
{
if (i < 0)
{
i = MAX_SHOWPOWERS - 1;
}
if (!ForcePower_Valid(forcePowerSorted[i])) // Does he have this power?
{
continue;
}
++iconCnt; // Good icon
if (cgs.media.forcePowerIcons[forcePowerSorted[i]])
{
CG_DrawPic( holdX, y + yOffset, smallIconSize, smallIconSize, cgs.media.forcePowerIcons[forcePowerSorted[i]] );
holdX -= (smallIconSize+pad);
}
}
if (ForcePower_Valid(cg.forceSelect))
{
// Current Center Icon
if (cgs.media.forcePowerIcons[cg.forceSelect])
{
CG_DrawPic( x-(bigIconSize/2), (y-((bigIconSize-smallIconSize)/2)) + yOffset, bigIconSize, bigIconSize, cgs.media.forcePowerIcons[cg.forceSelect] ); //only cache the icon for display
}
}
i = BG_ProperForceIndex(cg.forceSelect) + 1;
if (i>=MAX_SHOWPOWERS)
{
i = 0;
}
// Work forwards from current icon
holdX = x + (bigIconSize/2) + pad;
for (iconCnt=1;iconCnt<(sideRightIconCnt+1);i++)
{
if (i>=MAX_SHOWPOWERS)
{
i = 0;
}
if (!ForcePower_Valid(forcePowerSorted[i])) // Does he have this power?
{
continue;
}
++iconCnt; // Good icon
if (cgs.media.forcePowerIcons[forcePowerSorted[i]])
{
CG_DrawPic( holdX, y + yOffset, smallIconSize, smallIconSize, cgs.media.forcePowerIcons[forcePowerSorted[i]] ); //only cache the icon for display
holdX += (smallIconSize+pad);
}
}
if ( showPowersName[cg.forceSelect] )
{
CG_DrawProportionalString(320, y + 30 + yOffset, CG_GetStringEdString("SP_INGAME", showPowersName[cg.forceSelect]), UI_CENTER | UI_SMALLFONT, colorTable[CT_ICON_BLUE]);
}
}
/*
===================
CG_DrawInventorySelect
===================
*/
void CG_DrawInvenSelect( void )
{
int i;
int sideMax,holdCount,iconCnt;
int smallIconSize,bigIconSize;
int sideLeftIconCnt,sideRightIconCnt;
int count;
int holdX, x, y, y2, pad;
// float addX;
// don't display if dead
if ( cg.snap->ps.stats[STAT_HEALTH] <= 0 )
{
return;
}
if ((cg.invenSelectTime+WEAPON_SELECT_TIME)ps.stats[STAT_HOLDABLE_ITEM] || !cg.snap->ps.stats[STAT_HOLDABLE_ITEMS])
{
return;
}
if (cg.itemSelect == -1)
{
cg.itemSelect = bg_itemlist[cg.snap->ps.stats[STAT_HOLDABLE_ITEM]].giTag;
}
//const int bits = cg.snap->ps.stats[ STAT_ITEMS ];
// count the number of items owned
count = 0;
for ( i = 0 ; i < HI_NUM_HOLDABLE ; i++ )
{
if (/*CG_InventorySelectable(i) && inv_icons[i]*/
(cg.snap->ps.stats[STAT_HOLDABLE_ITEMS] & (1 << i)) )
{
count++;
}
}
if (!count)
{
y2 = 0; //err?
CG_DrawProportionalString(320, y2 + 22, "EMPTY INVENTORY", UI_CENTER | UI_SMALLFONT, colorTable[CT_ICON_BLUE]);
return;
}
sideMax = 3; // Max number of icons on the side
// Calculate how many icons will appear to either side of the center one
holdCount = count - 1; // -1 for the center icon
if (holdCount == 0) // No icons to either side
{
sideLeftIconCnt = 0;
sideRightIconCnt = 0;
}
else if (count > (2*sideMax)) // Go to the max on each side
{
sideLeftIconCnt = sideMax;
sideRightIconCnt = sideMax;
}
else // Less than max, so do the calc
{
sideLeftIconCnt = holdCount/2;
sideRightIconCnt = holdCount - sideLeftIconCnt;
}
i = cg.itemSelect - 1;
if (i<0)
{
i = HI_NUM_HOLDABLE-1;
}
smallIconSize = 40;
bigIconSize = 80;
pad = 16;
x = 320;
y = 410;
// Left side ICONS
// Work backwards from current icon
holdX = x - ((bigIconSize/2) + pad + smallIconSize);
// addX = (float) smallIconSize * .75;
for (iconCnt=0;iconCntps.stats[STAT_HOLDABLE_ITEMS] & (1 << i)) || i == cg.itemSelect )
{
continue;
}
++iconCnt; // Good icon
if (!BG_IsItemSelectable(&cg.predictedPlayerState, i))
{
continue;
}
if (cgs.media.invenIcons[i])
{
trap->R_SetColor(NULL);
CG_DrawPic( holdX, y+10, smallIconSize, smallIconSize, cgs.media.invenIcons[i] );
trap->R_SetColor(colorTable[CT_ICON_BLUE]);
/*CG_DrawNumField (holdX + addX, y + smallIconSize, 2, cg.snap->ps.inventory[i], 6, 12,
NUM_FONT_SMALL,qfalse);
*/
holdX -= (smallIconSize+pad);
}
}
// Current Center Icon
if (cgs.media.invenIcons[cg.itemSelect] && BG_IsItemSelectable(&cg.predictedPlayerState, cg.itemSelect))
{
int itemNdex;
trap->R_SetColor(NULL);
CG_DrawPic( x-(bigIconSize/2), (y-((bigIconSize-smallIconSize)/2))+10, bigIconSize, bigIconSize, cgs.media.invenIcons[cg.itemSelect] );
// addX = (float) bigIconSize * .75;
trap->R_SetColor(colorTable[CT_ICON_BLUE]);
/*CG_DrawNumField ((x-(bigIconSize/2)) + addX, y, 2, cg.snap->ps.inventory[cg.inventorySelect], 6, 12,
NUM_FONT_SMALL,qfalse);*/
itemNdex = BG_GetItemIndexByTag(cg.itemSelect, IT_HOLDABLE);
if (bg_itemlist[itemNdex].classname)
{
vec4_t textColor = { .312f, .75f, .621f, 1.0f };
char text[1024];
char upperKey[1024];
strcpy(upperKey, bg_itemlist[itemNdex].classname);
if ( trap->SE_GetStringTextString( va("SP_INGAME_%s",Q_strupr(upperKey)), text, sizeof( text )))
{
CG_DrawProportionalString(320, y+45, text, UI_CENTER | UI_SMALLFONT, textColor);
}
else
{
CG_DrawProportionalString(320, y+45, bg_itemlist[itemNdex].classname, UI_CENTER | UI_SMALLFONT, textColor);
}
}
}
i = cg.itemSelect + 1;
if (i> HI_NUM_HOLDABLE-1)
{
i = 0;
}
// Right side ICONS
// Work forwards from current icon
holdX = x + (bigIconSize/2) + pad;
// addX = (float) smallIconSize * .75;
for (iconCnt=0;iconCnt HI_NUM_HOLDABLE-1)
{
i = 0;
}
if ( !(cg.snap->ps.stats[STAT_HOLDABLE_ITEMS] & (1 << i)) || i == cg.itemSelect )
{
continue;
}
++iconCnt; // Good icon
if (!BG_IsItemSelectable(&cg.predictedPlayerState, i))
{
continue;
}
if (cgs.media.invenIcons[i])
{
trap->R_SetColor(NULL);
CG_DrawPic( holdX, y+10, smallIconSize, smallIconSize, cgs.media.invenIcons[i] );
trap->R_SetColor(colorTable[CT_ICON_BLUE]);
/*CG_DrawNumField (holdX + addX, y + smallIconSize, 2, cg.snap->ps.inventory[i], 6, 12,
NUM_FONT_SMALL,qfalse);*/
holdX += (smallIconSize+pad);
}
}
}
int cg_targVeh = ENTITYNUM_NONE;
int cg_targVehLastTime = 0;
qboolean CG_CheckTargetVehicle( centity_t **pTargetVeh, float *alpha )
{
int targetNum = ENTITYNUM_NONE;
centity_t *targetVeh = NULL;
if ( !pTargetVeh || !alpha )
{//hey, where are my pointers?
return qfalse;
}
*alpha = 1.0f;
//FIXME: need to clear all of these when you die?
if ( cg.predictedPlayerState.rocketLockIndex < ENTITYNUM_WORLD )
{
targetNum = cg.predictedPlayerState.rocketLockIndex;
}
else if ( cg.crosshairVehNum < ENTITYNUM_WORLD
&& cg.time - cg.crosshairVehTime < 3000 )
{//crosshair was on a vehicle in the last 3 seconds
targetNum = cg.crosshairVehNum;
}
else if ( cg.crosshairClientNum < ENTITYNUM_WORLD )
{
targetNum = cg.crosshairClientNum;
}
if ( targetNum < MAX_CLIENTS )
{//real client
if ( cg_entities[targetNum].currentState.m_iVehicleNum >= MAX_CLIENTS )
{//in a vehicle
targetNum = cg_entities[targetNum].currentState.m_iVehicleNum;
}
}
if ( targetNum < ENTITYNUM_WORLD
&& targetNum >= MAX_CLIENTS )
{
//centity_t *targetVeh = &cg_entities[targetNum];
targetVeh = &cg_entities[targetNum];
if ( targetVeh->currentState.NPC_class == CLASS_VEHICLE
&& targetVeh->m_pVehicle
&& targetVeh->m_pVehicle->m_pVehicleInfo
&& targetVeh->m_pVehicle->m_pVehicleInfo->type == VH_FIGHTER )
{//it's a vehicle
cg_targVeh = targetNum;
cg_targVehLastTime = cg.time;
*alpha = 1.0f;
}
else
{
targetVeh = NULL;
}
}
if ( targetVeh )
{
*pTargetVeh = targetVeh;
return qtrue;
}
if ( cg_targVehLastTime && cg.time - cg_targVehLastTime < 3000 )
{
targetVeh = &cg_entities[cg_targVeh];
//stay at full alpha for 1 sec after lose them from crosshair
if ( cg.time-cg_targVehLastTime < 1000 )
*alpha = 1.0f;
else //fade out over 2 secs
*alpha = 1.0f-((cg.time-cg_targVehLastTime-1000)/2000.0f);
}
return qfalse;
}
#define MAX_VHUD_SHIELD_TICS 12
#define MAX_VHUD_SPEED_TICS 5
#define MAX_VHUD_ARMOR_TICS 5
#define MAX_VHUD_AMMO_TICS 5
float CG_DrawVehicleShields( const menuDef_t *menuHUD, const centity_t *veh )
{
int i;
char itemName[64];
float inc, currValue,maxShields;
vec4_t calcColor;
itemDef_t *item;
float percShields;
item = Menu_FindItemByName((menuDef_t *) menuHUD, "armorbackground");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
maxShields = veh->m_pVehicle->m_pVehicleInfo->shields;
currValue = cg.predictedVehicleState.stats[STAT_ARMOR];
percShields = (float)currValue/(float)maxShields;
// Print all the tics of the shield graphic
// Look at the amount of health left and show only as much of the graphic as there is health.
// Use alpha to fade out partial section of health
inc = (float) maxShields / MAX_VHUD_ARMOR_TICS;
for (i=1;i<=MAX_VHUD_ARMOR_TICS;i++)
{
sprintf( itemName, "armor_tic%d", i );
item = Menu_FindItemByName((menuDef_t *) menuHUD, itemName);
if (!item)
{
continue;
}
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
float percent = currValue / inc;
calcColor[3] *= percent; // Fade it out
}
trap->R_SetColor( calcColor);
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
currValue -= inc;
}
return percShields;
}
int cg_vehicleAmmoWarning = 0;
int cg_vehicleAmmoWarningTime = 0;
void CG_DrawVehicleAmmo( const menuDef_t *menuHUD, const centity_t *veh )
{
int i;
char itemName[64];
float inc, currValue,maxAmmo;
vec4_t calcColor;
itemDef_t *item;
item = Menu_FindItemByName((menuDef_t *) menuHUD, "ammobackground");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
maxAmmo = veh->m_pVehicle->m_pVehicleInfo->weapon[0].ammoMax;
currValue = cg.predictedVehicleState.ammo[0];
inc = (float) maxAmmo / MAX_VHUD_AMMO_TICS;
for (i=1;i<=MAX_VHUD_AMMO_TICS;i++)
{
sprintf( itemName, "ammo_tic%d", i );
item = Menu_FindItemByName((menuDef_t *)menuHUD, itemName);
if (!item)
{
continue;
}
if ( cg_vehicleAmmoWarningTime > cg.time
&& cg_vehicleAmmoWarning == 0 )
{
memcpy(calcColor, g_color_table[ColorIndex(COLOR_RED)], sizeof(vec4_t));
calcColor[3] = sin(cg.time*0.005)*0.5f+0.5f;
}
else
{
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
float percent = currValue / inc;
calcColor[3] *= percent; // Fade it out
}
}
trap->R_SetColor( calcColor);
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
currValue -= inc;
}
}
void CG_DrawVehicleAmmoUpper( const menuDef_t *menuHUD, const centity_t *veh )
{
int i;
char itemName[64];
float inc, currValue,maxAmmo;
vec4_t calcColor;
itemDef_t *item;
item = Menu_FindItemByName((menuDef_t *)menuHUD, "ammoupperbackground");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
maxAmmo = veh->m_pVehicle->m_pVehicleInfo->weapon[0].ammoMax;
currValue = cg.predictedVehicleState.ammo[0];
inc = (float) maxAmmo / MAX_VHUD_AMMO_TICS;
for (i=1;i cg.time
&& cg_vehicleAmmoWarning == 0 )
{
memcpy(calcColor, g_color_table[ColorIndex(COLOR_RED)], sizeof(vec4_t));
calcColor[3] = sin(cg.time*0.005)*0.5f+0.5f;
}
else
{
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
float percent = currValue / inc;
calcColor[3] *= percent; // Fade it out
}
}
trap->R_SetColor( calcColor);
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
currValue -= inc;
}
}
void CG_DrawVehicleAmmoLower( const menuDef_t *menuHUD, const centity_t *veh )
{
int i;
char itemName[64];
float inc, currValue,maxAmmo;
vec4_t calcColor;
itemDef_t *item;
item = Menu_FindItemByName((menuDef_t *)menuHUD, "ammolowerbackground");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
maxAmmo = veh->m_pVehicle->m_pVehicleInfo->weapon[1].ammoMax;
currValue = cg.predictedVehicleState.ammo[1];
inc = (float) maxAmmo / MAX_VHUD_AMMO_TICS;
for (i=1;i cg.time
&& cg_vehicleAmmoWarning == 1 )
{
memcpy(calcColor, g_color_table[ColorIndex(COLOR_RED)], sizeof(vec4_t));
calcColor[3] = sin(cg.time*0.005)*0.5f+0.5f;
}
else
{
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
float percent = currValue / inc;
calcColor[3] *= percent; // Fade it out
}
}
trap->R_SetColor( calcColor);
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
currValue -= inc;
}
}
// The HUD.menu file has the graphic print with a negative height, so it will print from the bottom up.
void CG_DrawVehicleTurboRecharge( const menuDef_t *menuHUD, const centity_t *veh )
{
itemDef_t *item;
int height;
item = Menu_FindItemByName( (menuDef_t *) menuHUD, "turborecharge");
if (item)
{
float percent=0.0f;
int diff = ( cg.time - veh->m_pVehicle->m_iTurboTime );
height = item->window.rect.h;
if (diff > veh->m_pVehicle->m_pVehicleInfo->turboRecharge)
{
percent = 1.0f;
trap->R_SetColor( colorTable[CT_GREEN] );
}
else
{
percent = (float) diff / veh->m_pVehicle->m_pVehicleInfo->turboRecharge;
if (percent < 0.0f)
{
percent = 0.0f;
}
trap->R_SetColor( colorTable[CT_RED] );
}
height *= percent;
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
height,
cgs.media.whiteShader);
}
}
qboolean cg_drawLink = qfalse;
void CG_DrawVehicleWeaponsLinked( const menuDef_t *menuHUD, const centity_t *veh )
{
qboolean drawLink = qfalse;
if ( veh->m_pVehicle
&& veh->m_pVehicle->m_pVehicleInfo
&& (veh->m_pVehicle->m_pVehicleInfo->weapon[0].linkable == 2|| veh->m_pVehicle->m_pVehicleInfo->weapon[1].linkable == 2) )
{//weapon is always linked
drawLink = qtrue;
}
else
{
//MP way:
//must get sent over network
if ( cg.predictedVehicleState.vehWeaponsLinked )
{
drawLink = qtrue;
}
//NOTE: below is SP way
/*
//just cheat it
if ( veh->gent->m_pVehicle->weaponStatus[0].linked
|| veh->gent->m_pVehicle->weaponStatus[1].linked )
{
drawLink = qtrue;
}
*/
}
if ( cg_drawLink != drawLink )
{//state changed, play sound
cg_drawLink = drawLink;
trap->S_StartSound (NULL, cg.predictedPlayerState.clientNum, CHAN_LOCAL, trap->S_RegisterSound( "sound/vehicles/common/linkweaps.wav" ) );
}
if ( drawLink )
{
itemDef_t *item;
item = Menu_FindItemByName( (menuDef_t *) menuHUD, "weaponslinked");
if (item)
{
trap->R_SetColor( colorTable[CT_CYAN] );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
cgs.media.whiteShader);
}
}
}
void CG_DrawVehicleSpeed( const menuDef_t *menuHUD, const centity_t *veh )
{
int i;
char itemName[64];
float inc, currValue,maxSpeed;
vec4_t calcColor;
itemDef_t *item;
item = Menu_FindItemByName((menuDef_t *) menuHUD, "speedbackground");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
maxSpeed = veh->m_pVehicle->m_pVehicleInfo->speedMax;
currValue = cg.predictedVehicleState.speed;
// Print all the tics of the shield graphic
// Look at the amount of health left and show only as much of the graphic as there is health.
// Use alpha to fade out partial section of health
inc = (float) maxSpeed / MAX_VHUD_SPEED_TICS;
for (i=1;i<=MAX_VHUD_SPEED_TICS;i++)
{
sprintf( itemName, "speed_tic%d", i );
item = Menu_FindItemByName((menuDef_t *)menuHUD, itemName);
if (!item)
{
continue;
}
if ( cg.time > veh->m_pVehicle->m_iTurboTime )
{
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
}
else // In turbo mode
{
if (cg.VHUDFlashTime < cg.time)
{
cg.VHUDFlashTime = cg.time + 200;
if (cg.VHUDTurboFlag)
{
cg.VHUDTurboFlag = qfalse;
}
else
{
cg.VHUDTurboFlag = qtrue;
}
}
if (cg.VHUDTurboFlag)
{
memcpy(calcColor, colorTable[CT_LTRED1], sizeof(vec4_t));
}
else
{
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
}
}
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
float percent = currValue / inc;
calcColor[3] *= percent; // Fade it out
}
trap->R_SetColor( calcColor);
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
currValue -= inc;
}
}
void CG_DrawVehicleArmor( const menuDef_t *menuHUD, const centity_t *veh )
{
int i;
vec4_t calcColor;
char itemName[64];
float inc, currValue,maxArmor;
itemDef_t *item;
maxArmor = veh->m_pVehicle->m_pVehicleInfo->armor;
currValue = cg.predictedVehicleState.stats[STAT_HEALTH];
item = Menu_FindItemByName( (menuDef_t *) menuHUD, "shieldbackground");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
// Print all the tics of the shield graphic
// Look at the amount of health left and show only as much of the graphic as there is health.
// Use alpha to fade out partial section of health
inc = (float) maxArmor / MAX_VHUD_SHIELD_TICS;
for (i=1;i <= MAX_VHUD_SHIELD_TICS;i++)
{
sprintf( itemName, "shield_tic%d", i );
item = Menu_FindItemByName((menuDef_t *) menuHUD, itemName);
if (!item)
{
continue;
}
memcpy(calcColor, item->window.foreColor, sizeof(vec4_t));
if (currValue <= 0) // don't show tic
{
break;
}
else if (currValue < inc) // partial tic (alpha it out)
{
float percent = currValue / inc;
calcColor[3] *= percent; // Fade it out
}
trap->R_SetColor( calcColor);
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
currValue -= inc;
}
}
enum
{
VEH_DAMAGE_FRONT=0,
VEH_DAMAGE_BACK,
VEH_DAMAGE_LEFT,
VEH_DAMAGE_RIGHT,
};
typedef struct
{
const char *itemName;
short heavyDamage;
short lightDamage;
} veh_damage_t;
veh_damage_t vehDamageData[4] =
{
{ "vehicle_front",SHIPSURF_DAMAGE_FRONT_HEAVY,SHIPSURF_DAMAGE_FRONT_LIGHT },
{ "vehicle_back",SHIPSURF_DAMAGE_BACK_HEAVY,SHIPSURF_DAMAGE_BACK_LIGHT },
{ "vehicle_left",SHIPSURF_DAMAGE_LEFT_HEAVY,SHIPSURF_DAMAGE_LEFT_LIGHT },
{ "vehicle_right",SHIPSURF_DAMAGE_RIGHT_HEAVY,SHIPSURF_DAMAGE_RIGHT_LIGHT },
};
// Draw health graphic for given part of vehicle
void CG_DrawVehicleDamage(const centity_t *veh,int brokenLimbs,const menuDef_t *menuHUD,float alpha,int index)
{
itemDef_t *item;
int colorI;
vec4_t color;
int graphicHandle=0;
item = Menu_FindItemByName((menuDef_t *)menuHUD, vehDamageData[index].itemName);
if (item)
{
if (brokenLimbs & (1<R_SetColor( color );
switch ( index )
{
case VEH_DAMAGE_FRONT :
graphicHandle = veh->m_pVehicle->m_pVehicleInfo->iconFrontHandle;
break;
case VEH_DAMAGE_BACK :
graphicHandle = veh->m_pVehicle->m_pVehicleInfo->iconBackHandle;
break;
case VEH_DAMAGE_LEFT :
graphicHandle = veh->m_pVehicle->m_pVehicleInfo->iconLeftHandle;
break;
case VEH_DAMAGE_RIGHT :
graphicHandle = veh->m_pVehicle->m_pVehicleInfo->iconRightHandle;
break;
}
if (graphicHandle)
{
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
graphicHandle );
}
}
}
// Used on both damage indicators : player vehicle and the vehicle the player is locked on
void CG_DrawVehicleDamageHUD(const centity_t *veh,int brokenLimbs,float percShields,char *menuName, float alpha)
{
menuDef_t *menuHUD;
itemDef_t *item;
vec4_t color;
menuHUD = Menus_FindByName(menuName);
if ( !menuHUD )
{
return;
}
item = Menu_FindItemByName(menuHUD, "background");
if (item)
{
if (veh->m_pVehicle->m_pVehicleInfo->dmgIndicBackgroundHandle)
{
if ( veh->damageTime > cg.time )
{//ship shields currently taking damage
//NOTE: cent->damageAngle can be accessed to get the direction from the ship origin to the impact point (in 3-D space)
float perc = 1.0f - ((veh->damageTime - cg.time) / 2000.0f/*MIN_SHIELD_TIME*/);
if ( perc < 0.0f )
{
perc = 0.0f;
}
else if ( perc > 1.0f )
{
perc = 1.0f;
}
color[0] = item->window.foreColor[0];//flash red
color[1] = item->window.foreColor[1]*perc;//fade other colors back in over time
color[2] = item->window.foreColor[2]*perc;//fade other colors back in over time
color[3] = item->window.foreColor[3];//always normal alpha
trap->R_SetColor( color );
}
else
{
trap->R_SetColor( item->window.foreColor );
}
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
veh->m_pVehicle->m_pVehicleInfo->dmgIndicBackgroundHandle );
}
}
item = Menu_FindItemByName(menuHUD, "outer_frame");
if (item)
{
if (veh->m_pVehicle->m_pVehicleInfo->dmgIndicFrameHandle)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
veh->m_pVehicle->m_pVehicleInfo->dmgIndicFrameHandle );
}
}
item = Menu_FindItemByName(menuHUD, "shields");
if (item)
{
if (veh->m_pVehicle->m_pVehicleInfo->dmgIndicShieldHandle)
{
VectorCopy4 ( colorTable[CT_HUD_GREEN], color );
color[3] = percShields;
trap->R_SetColor( color );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
veh->m_pVehicle->m_pVehicleInfo->dmgIndicShieldHandle );
}
}
//TODO: if we check nextState.brokenLimbs & prevState.brokenLimbs, we can tell when a damage flag has been added and flash that part of the ship
//FIXME: when ship explodes, either stop drawing ship or draw all parts black
CG_DrawVehicleDamage(veh,brokenLimbs,menuHUD,alpha,VEH_DAMAGE_FRONT);
CG_DrawVehicleDamage(veh,brokenLimbs,menuHUD,alpha,VEH_DAMAGE_BACK);
CG_DrawVehicleDamage(veh,brokenLimbs,menuHUD,alpha,VEH_DAMAGE_LEFT);
CG_DrawVehicleDamage(veh,brokenLimbs,menuHUD,alpha,VEH_DAMAGE_RIGHT);
}
qboolean CG_DrawVehicleHud( const centity_t *cent )
{
itemDef_t *item;
menuDef_t *menuHUD;
playerState_t *ps;
centity_t *veh;
float shieldPerc,alpha;
menuHUD = Menus_FindByName("swoopvehiclehud");
if (!menuHUD)
{
return qtrue; // Draw player HUD
}
ps = &cg.predictedPlayerState;
if (!ps || !(ps->m_iVehicleNum))
{
return qtrue; // Draw player HUD
}
veh = &cg_entities[ps->m_iVehicleNum];
if ( !veh || !veh->m_pVehicle )
{
return qtrue; // Draw player HUD
}
CG_DrawVehicleTurboRecharge( menuHUD, veh );
CG_DrawVehicleWeaponsLinked( menuHUD, veh );
item = Menu_FindItemByName(menuHUD, "leftframe");
// Draw frame
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
item = Menu_FindItemByName(menuHUD, "rightframe");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
CG_DrawVehicleArmor( menuHUD, veh );
// Get animal hud for speed
// if (veh->m_pVehicle->m_pVehicleInfo->type == VH_ANIMAL)
// {
// menuHUD = Menus_FindByName("tauntaunhud");
// }
CG_DrawVehicleSpeed( menuHUD, veh );
// Revert to swoophud
// if (veh->m_pVehicle->m_pVehicleInfo->type == VH_ANIMAL)
// {
// menuHUD = Menus_FindByName("swoopvehiclehud");
// }
// if (veh->m_pVehicle->m_pVehicleInfo->type == VH_ANIMAL)
// {
shieldPerc = CG_DrawVehicleShields( menuHUD, veh );
// }
if (veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID && !veh->m_pVehicle->m_pVehicleInfo->weapon[1].ID)
{
CG_DrawVehicleAmmo( menuHUD, veh );
}
else if (veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID && veh->m_pVehicle->m_pVehicleInfo->weapon[1].ID)
{
CG_DrawVehicleAmmoUpper( menuHUD, veh );
CG_DrawVehicleAmmoLower( menuHUD, veh );
}
// If he's hidden, he must be in a vehicle
if (veh->m_pVehicle->m_pVehicleInfo->hideRider)
{
CG_DrawVehicleDamageHUD(veh,cg.predictedVehicleState.brokenLimbs,shieldPerc,"vehicledamagehud",1.0f);
// Has he targeted an enemy?
if (CG_CheckTargetVehicle( &veh, &alpha ))
{
CG_DrawVehicleDamageHUD(veh,veh->currentState.brokenLimbs,((float)veh->currentState.activeForcePass/10.0f),"enemyvehicledamagehud",alpha);
}
return qfalse; // Don't draw player HUD
}
return qtrue; // Draw player HUD
}
/*
================
CG_DrawStats
================
*/
static void CG_DrawStats( void )
{
centity_t *cent;
playerState_t *ps;
qboolean drawHUD = qtrue;
/* playerState_t *ps;
vec3_t angles;
// vec3_t origin;
if ( cg_drawStatus.integer == 0 ) {
return;
}
*/
cent = &cg_entities[cg.snap->ps.clientNum];
/* ps = &cg.snap->ps;
VectorClear( angles );
// Do start
if (!cg.interfaceStartupDone)
{
CG_InterfaceStartup();
}
cgi_UI_MenuPaintAll();*/
if ( cent )
{
ps = &cg.predictedPlayerState;
if ( (ps->m_iVehicleNum ) ) // In a vehicle???
{
drawHUD = CG_DrawVehicleHud( cent );
}
}
if (drawHUD)
{
CG_DrawHUD(cent);
}
/*CG_DrawArmor(cent);
CG_DrawHealth(cent);
CG_DrawAmmo(cent);
CG_DrawTalk(cent);*/
}
/*
===================
CG_DrawPickupItem
===================
*/
static void CG_DrawPickupItem( void ) {
int value;
float *fadeColor;
value = cg.itemPickup;
if ( value && cg_items[ value ].icon != -1 )
{
fadeColor = CG_FadeColor( cg.itemPickupTime, 3000 );
if ( fadeColor )
{
CG_RegisterItemVisuals( value );
trap->R_SetColor( fadeColor );
CG_DrawPic( 573, 320, ICON_SIZE, ICON_SIZE, cg_items[ value ].icon );
trap->R_SetColor( NULL );
}
}
}
/*
================
CG_DrawTeamBackground
================
*/
void CG_DrawTeamBackground( int x, int y, int w, int h, float alpha, int team )
{
vec4_t hcolor;
hcolor[3] = alpha;
if ( team == TEAM_RED ) {
hcolor[0] = 1;
hcolor[1] = .2f;
hcolor[2] = .2f;
} else if ( team == TEAM_BLUE ) {
hcolor[0] = .2f;
hcolor[1] = .2f;
hcolor[2] = 1;
} else {
return;
}
// trap->R_SetColor( hcolor );
CG_FillRect ( x, y, w, h, hcolor );
// CG_DrawPic( x, y, w, h, cgs.media.teamStatusBar );
trap->R_SetColor( NULL );
}
/*
===========================================================================================
UPPER RIGHT CORNER
===========================================================================================
*/
/*
================
CG_DrawMiniScoreboard
================
*/
static float CG_DrawMiniScoreboard ( float y )
{
char temp[MAX_QPATH];
int xOffset = 0;
if ( !cg_drawScores.integer )
{
return y;
}
if (cgs.gametype == GT_SIEGE)
{ //don't bother with this in siege
return y;
}
if ( cgs.gametype >= GT_TEAM )
{
Q_strncpyz( temp, va( "%s: ", CG_GetStringEdString( "MP_INGAME", "RED" ) ), sizeof( temp ) );
Q_strcat( temp, sizeof( temp ), cgs.scores1 == SCORE_NOT_PRESENT ? "-" : (va( "%i", cgs.scores1 )) );
Q_strcat( temp, sizeof( temp ), va( " %s: ", CG_GetStringEdString( "MP_INGAME", "BLUE" ) ) );
Q_strcat( temp, sizeof( temp ), cgs.scores2 == SCORE_NOT_PRESENT ? "-" : (va( "%i", cgs.scores2 )) );
CG_Text_Paint( 630 - CG_Text_Width ( temp, 0.7f, FONT_MEDIUM ) + xOffset, y, 0.7f, colorWhite, temp, 0, 0, ITEM_TEXTSTYLE_SHADOWEDMORE, FONT_MEDIUM );
y += 15;
}
else
{
/*
strcpy ( temp, "1st: " );
Q_strcat ( temp, MAX_QPATH, cgs.scores1==SCORE_NOT_PRESENT?"-":(va("%i",cgs.scores1)) );
Q_strcat ( temp, MAX_QPATH, " 2nd: " );
Q_strcat ( temp, MAX_QPATH, cgs.scores2==SCORE_NOT_PRESENT?"-":(va("%i",cgs.scores2)) );
CG_Text_Paint( 630 - CG_Text_Width ( temp, 0.7f, FONT_SMALL ), y, 0.7f, colorWhite, temp, 0, 0, ITEM_TEXTSTYLE_SHADOWEDMORE, FONT_MEDIUM );
y += 15;
*/
//rww - no longer doing this. Since the attacker now shows who is first, we print the score there.
}
return y;
}
/*
================
CG_DrawEnemyInfo
================
*/
static float CG_DrawEnemyInfo ( float y )
{
float size;
int clientNum;
const char *title;
clientInfo_t *ci;
int xOffset = 0;
if (!cg.snap)
{
return y;
}
if ( !cg_drawEnemyInfo.integer )
{
return y;
}
if ( cg.predictedPlayerState.stats[STAT_HEALTH] <= 0 )
{
return y;
}
if (cgs.gametype == GT_POWERDUEL)
{ //just get out of here then
return y;
}
if ( cgs.gametype == GT_JEDIMASTER )
{
//title = "Jedi Master";
title = CG_GetStringEdString("MP_INGAME", "MASTERY7");
clientNum = cgs.jediMaster;
if ( clientNum < 0 )
{
//return y;
// title = "Get Saber!";
title = CG_GetStringEdString("MP_INGAME", "GET_SABER");
size = ICON_SIZE * 1.25;
y += 5;
CG_DrawPic( 640 - size - 12 + xOffset, y, size, size, cgs.media.weaponIcons[WP_SABER] );
y += size;
/*
CG_Text_Paint( 630 - CG_Text_Width ( ci->name, 0.7f, FONT_MEDIUM ), y, 0.7f, colorWhite, ci->name, 0, 0, 0, FONT_MEDIUM );
y += 15;
*/
CG_Text_Paint( 630 - CG_Text_Width ( title, 0.7f, FONT_MEDIUM ) + xOffset, y, 0.7f, colorWhite, title, 0, 0, 0, FONT_MEDIUM );
return y + BIGCHAR_HEIGHT + 2;
}
}
else if ( cg.snap->ps.duelInProgress )
{
// title = "Dueling";
title = CG_GetStringEdString("MP_INGAME", "DUELING");
clientNum = cg.snap->ps.duelIndex;
}
else if ( cgs.gametype == GT_DUEL && cgs.clientinfo[cg.snap->ps.clientNum].team != TEAM_SPECTATOR)
{
title = CG_GetStringEdString("MP_INGAME", "DUELING");
if (cg.snap->ps.clientNum == cgs.duelist1)
{
clientNum = cgs.duelist2; //if power duel, should actually draw both duelists 2 and 3 I guess
}
else if (cg.snap->ps.clientNum == cgs.duelist2)
{
clientNum = cgs.duelist1;
}
else if (cg.snap->ps.clientNum == cgs.duelist3)
{
clientNum = cgs.duelist1;
}
else
{
return y;
}
}
else
{
/*
title = "Attacker";
clientNum = cg.predictedPlayerState.persistant[PERS_ATTACKER];
if ( clientNum < 0 || clientNum >= MAX_CLIENTS || clientNum == cg.snap->ps.clientNum )
{
return y;
}
if ( cg.time - cg.attackerTime > ATTACKER_HEAD_TIME )
{
cg.attackerTime = 0;
return y;
}
*/
//As of current, we don't want to draw the attacker. Instead, draw whoever is in first place.
if (cgs.duelWinner < 0 || cgs.duelWinner >= MAX_CLIENTS)
{
return y;
}
title = va("%s: %i",CG_GetStringEdString("MP_INGAME", "LEADER"), cgs.scores1);
/*
if (cgs.scores1 == 1)
{
title = va("%i kill", cgs.scores1);
}
else
{
title = va("%i kills", cgs.scores1);
}
*/
clientNum = cgs.duelWinner;
}
if ( clientNum >= MAX_CLIENTS || !(&cgs.clientinfo[ clientNum ]) )
{
return y;
}
ci = &cgs.clientinfo[ clientNum ];
size = ICON_SIZE * 1.25;
y += 5;
if ( ci->modelIcon )
{
CG_DrawPic( 640 - size - 5 + xOffset, y, size, size, ci->modelIcon );
}
y += size;
// CG_Text_Paint( 630 - CG_Text_Width ( ci->name, 0.7f, FONT_MEDIUM ) + xOffset, y, 0.7f, colorWhite, ci->name, 0, 0, 0, FONT_MEDIUM );
CG_Text_Paint( 630 - CG_Text_Width ( ci->name, 1.0f, FONT_SMALL2 ) + xOffset, y, 1.0f, colorWhite, ci->name, 0, 0, 0, FONT_SMALL2 );
y += 15;
// CG_Text_Paint( 630 - CG_Text_Width ( title, 0.7f, FONT_MEDIUM ) + xOffset, y, 0.7f, colorWhite, title, 0, 0, 0, FONT_MEDIUM );
CG_Text_Paint( 630 - CG_Text_Width ( title, 1.0f, FONT_SMALL2 ) + xOffset, y, 1.0f, colorWhite, title, 0, 0, 0, FONT_SMALL2 );
if ( (cgs.gametype == GT_DUEL || cgs.gametype == GT_POWERDUEL) && cgs.clientinfo[cg.snap->ps.clientNum].team != TEAM_SPECTATOR)
{//also print their score
char text[1024];
y += 15;
Com_sprintf(text, sizeof(text), "%i/%i", cgs.clientinfo[clientNum].score, cgs.fraglimit );
CG_Text_Paint( 630 - CG_Text_Width ( text, 0.7f, FONT_MEDIUM ) + xOffset, y, 0.7f, colorWhite, text, 0, 0, 0, FONT_MEDIUM );
}
// nmckenzie: DUEL_HEALTH - fixme - need checks and such here. And this is coded to duelist 1 right now, which is wrongly.
if ( cgs.showDuelHealths >= 2)
{
y += 15;
if ( cgs.duelist1 == clientNum )
{
CG_DrawDuelistHealth ( 640 - size - 5 + xOffset, y, 64, 8, 1 );
}
else if ( cgs.duelist2 == clientNum )
{
CG_DrawDuelistHealth ( 640 - size - 5 + xOffset, y, 64, 8, 2 );
}
}
return y + BIGCHAR_HEIGHT + 2;
}
/*
==================
CG_DrawSnapshot
==================
*/
static float CG_DrawSnapshot( float y ) {
char *s;
int w;
int xOffset = 0;
s = va( "time:%i snap:%i cmd:%i", cg.snap->serverTime,
cg.latestSnapshotNum, cgs.serverCommandSequence );
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString( 635 - w + xOffset, y + 2, s, 1.0F);
return y + BIGCHAR_HEIGHT + 4;
}
/*
==================
CG_DrawFPS
==================
*/
#define FPS_FRAMES 16
static float CG_DrawFPS( float y ) {
char *s;
int w;
static unsigned short previousTimes[FPS_FRAMES];
static unsigned short index;
static int previous, lastupdate;
int t, i, fps, total;
unsigned short frameTime;
const int xOffset = 0;
// don't use serverTime, because that will be drifting to
// correct for internet lag changes, timescales, timedemos, etc
t = trap->Milliseconds();
frameTime = t - previous;
previous = t;
if (t - lastupdate > 50) //don't sample faster than this
{
lastupdate = t;
previousTimes[index % FPS_FRAMES] = frameTime;
index++;
}
// average multiple frames together to smooth changes out a bit
total = 0;
for ( i = 0 ; i < FPS_FRAMES ; i++ ) {
total += previousTimes[i];
}
if ( !total ) {
total = 1;
}
fps = 1000 * FPS_FRAMES / total;
s = va( "%ifps", fps );
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString( 635 - w + xOffset, y + 2, s, 1.0F);
return y + BIGCHAR_HEIGHT + 4;
}
// nmckenzie: DUEL_HEALTH
#define MAX_HEALTH_FOR_IFACE 100
void CG_DrawHealthBarRough (float x, float y, int width, int height, float ratio, const float *color1, const float *color2)
{
float midpoint, remainder;
float color3[4] = {1, 0, 0, .7f};
midpoint = width * ratio - 1;
remainder = width - midpoint;
color3[0] = color1[0] * 0.5f;
assert(!(height%4));//this won't line up otherwise.
CG_DrawRect(x + 1, y + height/2-1, midpoint, 1, height/4+1, color1); // creme-y filling.
CG_DrawRect(x + midpoint, y + height/2-1, remainder, 1, height/4+1, color3); // used-up-ness.
CG_DrawRect(x, y, width, height, 1, color2); // hard crispy shell
}
void CG_DrawDuelistHealth ( float x, float y, float w, float h, int duelist )
{
float duelHealthColor[4] = {1, 0, 0, 0.7f};
float healthSrc = 0.0f;
float ratio;
if ( duelist == 1 )
{
healthSrc = cgs.duelist1health;
}
else if (duelist == 2 )
{
healthSrc = cgs.duelist2health;
}
ratio = healthSrc / MAX_HEALTH_FOR_IFACE;
if ( ratio > 1.0f )
{
ratio = 1.0f;
}
if ( ratio < 0.0f )
{
ratio = 0.0f;
}
duelHealthColor[0] = (ratio * 0.2f) + 0.5f;
CG_DrawHealthBarRough (x, y, w, h, ratio, duelHealthColor, colorTable[CT_WHITE]); // new art for this? I'm not crazy about how this looks.
}
/*
=====================
CG_DrawRadar
=====================
*/
float cg_radarRange = 2500.0f;
#define RADAR_RADIUS 60
#define RADAR_X (580 - RADAR_RADIUS)
#define RADAR_CHAT_DURATION 6000
static int radarLockSoundDebounceTime = 0;
static int impactSoundDebounceTime = 0;
#define RADAR_MISSILE_RANGE 3000.0f
#define RADAR_ASTEROID_RANGE 10000.0f
#define RADAR_MIN_ASTEROID_SURF_WARN_DIST 1200.0f
float CG_DrawRadar ( float y )
{
vec4_t color;
vec4_t teamColor;
float arrow_w;
float arrow_h;
clientInfo_t *cl;
clientInfo_t *local;
int i;
float arrowBaseScale;
float zScale;
int xOffset = 0;
if (!cg.snap)
{
return y;
}
// Make sure the radar should be showing
if ( cg.snap->ps.stats[STAT_HEALTH] <= 0 )
{
return y;
}
if ( (cg.predictedPlayerState.pm_flags & PMF_FOLLOW) || cg.predictedPlayerState.persistant[PERS_TEAM] == TEAM_SPECTATOR )
{
return y;
}
local = &cgs.clientinfo[ cg.snap->ps.clientNum ];
if ( !local->infoValid )
{
return y;
}
// Draw the radar background image
color[0] = color[1] = color[2] = 1.0f;
color[3] = 0.6f;
trap->R_SetColor ( color );
CG_DrawPic( RADAR_X + xOffset, y, RADAR_RADIUS*2, RADAR_RADIUS*2, cgs.media.radarShader );
//Always green for your own team.
VectorCopy ( g_color_table[ColorIndex(COLOR_GREEN)], teamColor );
teamColor[3] = 1.0f;
// Draw all of the radar entities. Draw them backwards so players are drawn last
for ( i = cg.radarEntityCount -1 ; i >= 0 ; i-- )
{
vec3_t dirLook;
vec3_t dirPlayer;
float angleLook;
float anglePlayer;
float angle;
float distance, actualDist;
centity_t* cent;
cent = &cg_entities[cg.radarEntities[i]];
// Get the distances first
VectorSubtract ( cg.predictedPlayerState.origin, cent->lerpOrigin, dirPlayer );
dirPlayer[2] = 0;
actualDist = distance = VectorNormalize ( dirPlayer );
if ( distance > cg_radarRange * 0.8f)
{
if ( (cent->currentState.eFlags & EF_RADAROBJECT)//still want to draw the direction
|| ( cent->currentState.eType==ET_NPC//FIXME: draw last, with players...
&& cent->currentState.NPC_class == CLASS_VEHICLE
&& cent->currentState.speed > 0 ) )//always draw vehicles
{
distance = cg_radarRange*0.8f;
}
else
{
continue;
}
}
distance = distance / cg_radarRange;
distance *= RADAR_RADIUS;
AngleVectors ( cg.predictedPlayerState.viewangles, dirLook, NULL, NULL );
dirLook[2] = 0;
anglePlayer = atan2(dirPlayer[0],dirPlayer[1]);
VectorNormalize ( dirLook );
angleLook = atan2(dirLook[0],dirLook[1]);
angle = angleLook - anglePlayer;
switch ( cent->currentState.eType )
{
default:
{
float x;
float ly;
qhandle_t shader;
vec4_t color;
x = (float)RADAR_X + (float)RADAR_RADIUS + (float)sin (angle) * distance;
ly = y + (float)RADAR_RADIUS + (float)cos (angle) * distance;
arrowBaseScale = 9.0f;
shader = 0;
zScale = 1.0f;
//we want to scale the thing up/down based on the relative Z (up/down) positioning
if (cent->lerpOrigin[2] > cg.predictedPlayerState.origin[2])
{ //higher, scale up (between 16 and 24)
float dif = (cent->lerpOrigin[2] - cg.predictedPlayerState.origin[2]);
//max out to 1.5x scale at 512 units above local player's height
dif /= 1024.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale += dif;
}
else if (cent->lerpOrigin[2] < cg.predictedPlayerState.origin[2])
{ //lower, scale down (between 16 and 8)
float dif = (cg.predictedPlayerState.origin[2] - cent->lerpOrigin[2]);
//half scale at 512 units below local player's height
dif /= 1024.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale -= dif;
}
arrowBaseScale *= zScale;
if (cent->currentState.brokenLimbs)
{ //slightly misleading to use this value, but don't want to add more to entstate.
//any ent with brokenLimbs non-0 and on radar is an objective ent.
//brokenLimbs is literal team value.
char objState[1024];
int complete;
//we only want to draw it if the objective for it is not complete.
//frame represents objective num.
trap->Cvar_VariableStringBuffer(va("team%i_objective%i", cent->currentState.brokenLimbs, cent->currentState.frame), objState, 1024);
complete = atoi(objState);
if (!complete)
{
// generic enemy index specifies a shader to use for the radar entity.
if ( cent->currentState.genericenemyindex && cent->currentState.genericenemyindex < MAX_ICONS )
{
color[0] = color[1] = color[2] = color[3] = 1.0f;
shader = cgs.gameIcons[cent->currentState.genericenemyindex];
}
else
{
if (cg.snap &&
cent->currentState.brokenLimbs == cg.snap->ps.persistant[PERS_TEAM])
{
VectorCopy ( g_color_table[ColorIndex(COLOR_RED)], color );
}
else
{
VectorCopy ( g_color_table[ColorIndex(COLOR_GREEN)], color );
}
shader = cgs.media.siegeItemShader;
}
}
}
else
{
color[0] = color[1] = color[2] = color[3] = 1.0f;
// generic enemy index specifies a shader to use for the radar entity.
if ( cent->currentState.genericenemyindex )
{
shader = cgs.gameIcons[cent->currentState.genericenemyindex];
}
else
{
shader = cgs.media.siegeItemShader;
}
}
if ( shader )
{
// Pulse the alpha if time2 is set. time2 gets set when the entity takes pain
if ( (cent->currentState.time2 && cg.time - cent->currentState.time2 < 5000) ||
(cent->currentState.time2 == 0xFFFFFFFF) )
{
if ( (cg.time / 200) & 1 )
{
color[3] = 0.1f + 0.9f * (float) (cg.time % 200) / 200.0f;
}
else
{
color[3] = 1.0f - 0.9f * (float) (cg.time % 200) / 200.0f;
}
}
trap->R_SetColor ( color );
CG_DrawPic ( x - 4 + xOffset, ly - 4, arrowBaseScale, arrowBaseScale, shader );
}
}
break;
case ET_NPC://FIXME: draw last, with players...
if ( cent->currentState.NPC_class == CLASS_VEHICLE
&& cent->currentState.speed > 0 )
{
if ( cent->m_pVehicle && cent->m_pVehicle->m_pVehicleInfo->radarIconHandle )
{
float x;
float ly;
x = (float)RADAR_X + (float)RADAR_RADIUS + (float)sin (angle) * distance;
ly = y + (float)RADAR_RADIUS + (float)cos (angle) * distance;
arrowBaseScale = 9.0f;
zScale = 1.0f;
//we want to scale the thing up/down based on the relative Z (up/down) positioning
if (cent->lerpOrigin[2] > cg.predictedPlayerState.origin[2])
{ //higher, scale up (between 16 and 24)
float dif = (cent->lerpOrigin[2] - cg.predictedPlayerState.origin[2]);
//max out to 1.5x scale at 512 units above local player's height
dif /= 4096.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale += dif;
}
else if (cent->lerpOrigin[2] < cg.predictedPlayerState.origin[2])
{ //lower, scale down (between 16 and 8)
float dif = (cg.predictedPlayerState.origin[2] - cent->lerpOrigin[2]);
//half scale at 512 units below local player's height
dif /= 4096.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale -= dif;
}
arrowBaseScale *= zScale;
if ( cent->currentState.m_iVehicleNum //vehicle has a driver
&& cgs.clientinfo[ cent->currentState.m_iVehicleNum-1 ].infoValid )
{
if ( cgs.clientinfo[ cent->currentState.m_iVehicleNum-1 ].team == local->team )
{
trap->R_SetColor ( teamColor );
}
else
{
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
}
}
else
{
trap->R_SetColor ( NULL );
}
CG_DrawPic ( x - 4 + xOffset, ly - 4, arrowBaseScale, arrowBaseScale, cent->m_pVehicle->m_pVehicleInfo->radarIconHandle );
}
}
break; //maybe do something?
case ET_MOVER:
if ( cent->currentState.speed//the mover's size, actually
&& actualDist < (cent->currentState.speed+RADAR_ASTEROID_RANGE)
&& cg.predictedPlayerState.m_iVehicleNum )
{//a mover that's close to me and I'm in a vehicle
qboolean mayImpact = qfalse;
float surfaceDist = (actualDist-cent->currentState.speed);
if ( surfaceDist < 0.0f )
{
surfaceDist = 0.0f;
}
if ( surfaceDist < RADAR_MIN_ASTEROID_SURF_WARN_DIST )
{//always warn!
mayImpact = qtrue;
}
else
{//not close enough to always warn, yet, so check its direction
vec3_t asteroidPos, myPos, moveDir;
int predictTime, timeStep = 500;
float newDist;
for ( predictTime = timeStep; predictTime < 5000; predictTime+=timeStep )
{
//asteroid dir, speed, size, + my dir & speed...
BG_EvaluateTrajectory( ¢->currentState.pos, cg.time+predictTime, asteroidPos );
//FIXME: I don't think it's calcing "myPos" correctly
AngleVectors( cg.predictedVehicleState.viewangles, moveDir, NULL, NULL );
VectorMA( cg.predictedVehicleState.origin, cg.predictedVehicleState.speed*predictTime/1000.0f, moveDir, myPos );
newDist = Distance( myPos, asteroidPos );
if ( (newDist-cent->currentState.speed) <= RADAR_MIN_ASTEROID_SURF_WARN_DIST )//200.0f )
{//heading for an impact within the next 5 seconds
mayImpact = qtrue;
break;
}
}
}
if ( mayImpact )
{//possible collision
vec4_t asteroidColor = {0.5f,0.5f,0.5f,1.0f};
float x;
float ly;
float asteroidScale = (cent->currentState.speed/2000.0f);//average asteroid radius?
if ( actualDist > RADAR_ASTEROID_RANGE )
{
actualDist = RADAR_ASTEROID_RANGE;
}
distance = (actualDist/RADAR_ASTEROID_RANGE)*RADAR_RADIUS;
x = (float)RADAR_X + (float)RADAR_RADIUS + (float)sin (angle) * distance;
ly = y + (float)RADAR_RADIUS + (float)cos (angle) * distance;
if ( asteroidScale > 3.0f )
{
asteroidScale = 3.0f;
}
else if ( asteroidScale < 0.2f )
{
asteroidScale = 0.2f;
}
arrowBaseScale = (9.0f*asteroidScale);
if ( impactSoundDebounceTime < cg.time )
{
vec3_t soundOrg;
if ( surfaceDist > RADAR_ASTEROID_RANGE*0.66f )
{
impactSoundDebounceTime = cg.time + 1000;
}
else if ( surfaceDist > RADAR_ASTEROID_RANGE/3.0f )
{
impactSoundDebounceTime = cg.time + 400;
}
else
{
impactSoundDebounceTime = cg.time + 100;
}
VectorMA( cg.refdef.vieworg, -500.0f*(surfaceDist/RADAR_ASTEROID_RANGE), dirPlayer, soundOrg );
trap->S_StartSound( soundOrg, ENTITYNUM_WORLD, CHAN_AUTO, trap->S_RegisterSound( "sound/vehicles/common/impactalarm.wav" ) );
}
//brighten it the closer it is
if ( surfaceDist > RADAR_ASTEROID_RANGE*0.66f )
{
asteroidColor[0] = asteroidColor[1] = asteroidColor[2] = 0.7f;
}
else if ( surfaceDist > RADAR_ASTEROID_RANGE/3.0f )
{
asteroidColor[0] = asteroidColor[1] = asteroidColor[2] = 0.85f;
}
else
{
asteroidColor[0] = asteroidColor[1] = asteroidColor[2] = 1.0f;
}
//alpha out the longer it's been since it was considered dangerous
if ( (cg.time-impactSoundDebounceTime) > 100 )
{
asteroidColor[3] = (float)((cg.time-impactSoundDebounceTime)-100)/900.0f;
}
trap->R_SetColor ( asteroidColor );
CG_DrawPic ( x - 4 + xOffset, ly - 4, arrowBaseScale, arrowBaseScale, trap->R_RegisterShaderNoMip( "gfx/menus/radar/asteroid" ) );
}
}
break;
case ET_MISSILE:
if ( //cent->currentState.weapon == WP_ROCKET_LAUNCHER &&//a rocket
cent->currentState.owner > MAX_CLIENTS //belongs to an NPC
&& cg_entities[cent->currentState.owner].currentState.NPC_class == CLASS_VEHICLE )
{//a rocket belonging to an NPC, FIXME: only tracking rockets!
float x;
float ly;
x = (float)RADAR_X + (float)RADAR_RADIUS + (float)sin (angle) * distance;
ly = y + (float)RADAR_RADIUS + (float)cos (angle) * distance;
arrowBaseScale = 3.0f;
if ( cg.predictedPlayerState.m_iVehicleNum )
{//I'm in a vehicle
//if it's targetting me, then play an alarm sound if I'm in a vehicle
if ( cent->currentState.otherEntityNum == cg.predictedPlayerState.clientNum || cent->currentState.otherEntityNum == cg.predictedPlayerState.m_iVehicleNum )
{
if ( radarLockSoundDebounceTime < cg.time )
{
vec3_t soundOrg;
int alarmSound;
if ( actualDist > RADAR_MISSILE_RANGE * 0.66f )
{
radarLockSoundDebounceTime = cg.time + 1000;
arrowBaseScale = 3.0f;
alarmSound = trap->S_RegisterSound( "sound/vehicles/common/lockalarm1.wav" );
}
else if ( actualDist > RADAR_MISSILE_RANGE/3.0f )
{
radarLockSoundDebounceTime = cg.time + 500;
arrowBaseScale = 6.0f;
alarmSound = trap->S_RegisterSound( "sound/vehicles/common/lockalarm2.wav" );
}
else
{
radarLockSoundDebounceTime = cg.time + 250;
arrowBaseScale = 9.0f;
alarmSound = trap->S_RegisterSound( "sound/vehicles/common/lockalarm3.wav" );
}
if ( actualDist > RADAR_MISSILE_RANGE )
{
actualDist = RADAR_MISSILE_RANGE;
}
VectorMA( cg.refdef.vieworg, -500.0f*(actualDist/RADAR_MISSILE_RANGE), dirPlayer, soundOrg );
trap->S_StartSound( soundOrg, ENTITYNUM_WORLD, CHAN_AUTO, alarmSound );
}
}
}
zScale = 1.0f;
//we want to scale the thing up/down based on the relative Z (up/down) positioning
if (cent->lerpOrigin[2] > cg.predictedPlayerState.origin[2])
{ //higher, scale up (between 16 and 24)
float dif = (cent->lerpOrigin[2] - cg.predictedPlayerState.origin[2]);
//max out to 1.5x scale at 512 units above local player's height
dif /= 1024.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale += dif;
}
else if (cent->lerpOrigin[2] < cg.predictedPlayerState.origin[2])
{ //lower, scale down (between 16 and 8)
float dif = (cg.predictedPlayerState.origin[2] - cent->lerpOrigin[2]);
//half scale at 512 units below local player's height
dif /= 1024.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale -= dif;
}
arrowBaseScale *= zScale;
if ( cent->currentState.owner >= MAX_CLIENTS//missile owned by an NPC
&& cg_entities[cent->currentState.owner].currentState.NPC_class == CLASS_VEHICLE//NPC is a vehicle
&& cg_entities[cent->currentState.owner].currentState.m_iVehicleNum <= MAX_CLIENTS//Vehicle has a player driver
&& cgs.clientinfo[cg_entities[cent->currentState.owner].currentState.m_iVehicleNum-1].infoValid ) //player driver is valid
{
cl = &cgs.clientinfo[cg_entities[cent->currentState.owner].currentState.m_iVehicleNum-1];
if ( cl->team == local->team )
{
trap->R_SetColor ( teamColor );
}
else
{
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
}
}
else
{
trap->R_SetColor ( NULL );
}
CG_DrawPic ( x - 4 + xOffset, ly - 4, arrowBaseScale, arrowBaseScale, cgs.media.mAutomapRocketIcon );
}
break;
case ET_PLAYER:
{
vec4_t color;
cl = &cgs.clientinfo[ cent->currentState.number ];
// not valid then dont draw it
if ( !cl->infoValid )
{
continue;
}
VectorCopy4 ( teamColor, color );
arrowBaseScale = 16.0f;
zScale = 1.0f;
// Pulse the radar icon after a voice message
if ( cent->vChatTime + 2000 > cg.time )
{
float f = (cent->vChatTime + 2000 - cg.time) / 3000.0f;
arrowBaseScale = 16.0f + 4.0f * f;
color[0] = teamColor[0] + (1.0f - teamColor[0]) * f;
color[1] = teamColor[1] + (1.0f - teamColor[1]) * f;
color[2] = teamColor[2] + (1.0f - teamColor[2]) * f;
}
trap->R_SetColor ( color );
//we want to scale the thing up/down based on the relative Z (up/down) positioning
if (cent->lerpOrigin[2] > cg.predictedPlayerState.origin[2])
{ //higher, scale up (between 16 and 32)
float dif = (cent->lerpOrigin[2] - cg.predictedPlayerState.origin[2]);
//max out to 2x scale at 1024 units above local player's height
dif /= 1024.0f;
if (dif > 1.0f)
{
dif = 1.0f;
}
zScale += dif;
}
else if (cent->lerpOrigin[2] < cg.predictedPlayerState.origin[2])
{ //lower, scale down (between 16 and 8)
float dif = (cg.predictedPlayerState.origin[2] - cent->lerpOrigin[2]);
//half scale at 512 units below local player's height
dif /= 1024.0f;
if (dif > 0.5f)
{
dif = 0.5f;
}
zScale -= dif;
}
arrowBaseScale *= zScale;
arrow_w = arrowBaseScale * RADAR_RADIUS / 128;
arrow_h = arrowBaseScale * RADAR_RADIUS / 128;
CG_DrawRotatePic2( RADAR_X + RADAR_RADIUS + sin (angle) * distance + xOffset,
y + RADAR_RADIUS + cos (angle) * distance,
arrow_w, arrow_h,
(360 - cent->lerpAngles[YAW]) + cg.predictedPlayerState.viewangles[YAW], cgs.media.mAutomapPlayerIcon );
break;
}
}
}
arrowBaseScale = 16.0f;
arrow_w = arrowBaseScale * RADAR_RADIUS / 128;
arrow_h = arrowBaseScale * RADAR_RADIUS / 128;
trap->R_SetColor ( colorWhite );
CG_DrawRotatePic2( RADAR_X + RADAR_RADIUS + xOffset, y + RADAR_RADIUS, arrow_w, arrow_h,
0, cgs.media.mAutomapPlayerIcon );
return y+(RADAR_RADIUS*2);
}
/*
=================
CG_DrawTimer
=================
*/
static float CG_DrawTimer( float y ) {
char *s;
int w;
int mins, seconds, tens;
int msec;
int xOffset = 0;
msec = cg.time - cgs.levelStartTime;
seconds = msec / 1000;
mins = seconds / 60;
seconds -= mins * 60;
tens = seconds / 10;
seconds -= tens * 10;
s = va( "%i:%i%i", mins, tens, seconds );
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString( 635 - w + xOffset, y + 2, s, 1.0F);
return y + BIGCHAR_HEIGHT + 4;
}
/*
=================
CG_DrawTeamOverlay
=================
*/
extern const char *CG_GetLocationString(const char *loc); //cg_main.c
static float CG_DrawTeamOverlay( float y, qboolean right, qboolean upper ) {
int x, w, h, xx;
int i, j, len;
const char *p;
vec4_t hcolor;
int pwidth, lwidth;
int plyrs;
char st[16];
clientInfo_t *ci;
gitem_t *item;
int ret_y, count;
int xOffset = 0;
if ( !cg_drawTeamOverlay.integer ) {
return y;
}
if ( cg.snap->ps.persistant[PERS_TEAM] != TEAM_RED && cg.snap->ps.persistant[PERS_TEAM] != TEAM_BLUE ) {
return y; // Not on any team
}
plyrs = 0;
//TODO: On basejka servers, we won't have valid teaminfo if we're spectating someone.
// Find a way to detect invalid info and return early?
// max player name width
pwidth = 0;
count = (numSortedTeamPlayers > 8) ? 8 : numSortedTeamPlayers;
for (i = 0; i < count; i++) {
ci = cgs.clientinfo + sortedTeamPlayers[i];
if ( ci->infoValid && ci->team == cg.snap->ps.persistant[PERS_TEAM]) {
plyrs++;
len = CG_DrawStrlen(ci->name);
if (len > pwidth)
pwidth = len;
}
}
if (!plyrs)
return y;
if (pwidth > TEAM_OVERLAY_MAXNAME_WIDTH)
pwidth = TEAM_OVERLAY_MAXNAME_WIDTH;
// max location name width
lwidth = 0;
for (i = 1; i < MAX_LOCATIONS; i++) {
p = CG_GetLocationString(CG_ConfigString(CS_LOCATIONS+i));
if (p && *p) {
len = CG_DrawStrlen(p);
if (len > lwidth)
lwidth = len;
}
}
if (lwidth > TEAM_OVERLAY_MAXLOCATION_WIDTH)
lwidth = TEAM_OVERLAY_MAXLOCATION_WIDTH;
w = (pwidth + lwidth + 4 + 7) * TINYCHAR_WIDTH;
if ( right )
x = 640 - w;
else
x = 0;
h = plyrs * TINYCHAR_HEIGHT;
if ( upper ) {
ret_y = y + h;
} else {
y -= h;
ret_y = y;
}
if ( cg.snap->ps.persistant[PERS_TEAM] == TEAM_RED ) {
hcolor[0] = 1.0f;
hcolor[1] = 0.0f;
hcolor[2] = 0.0f;
hcolor[3] = 0.33f;
} else { // if ( cg.snap->ps.persistant[PERS_TEAM] == TEAM_BLUE )
hcolor[0] = 0.0f;
hcolor[1] = 0.0f;
hcolor[2] = 1.0f;
hcolor[3] = 0.33f;
}
trap->R_SetColor( hcolor );
CG_DrawPic( x + xOffset, y, w, h, cgs.media.teamStatusBar );
trap->R_SetColor( NULL );
for (i = 0; i < count; i++) {
ci = cgs.clientinfo + sortedTeamPlayers[i];
if ( ci->infoValid && ci->team == cg.snap->ps.persistant[PERS_TEAM]) {
hcolor[0] = hcolor[1] = hcolor[2] = hcolor[3] = 1.0;
xx = x + TINYCHAR_WIDTH;
CG_DrawStringExt( xx + xOffset, y,
ci->name, hcolor, qfalse, qfalse,
TINYCHAR_WIDTH, TINYCHAR_HEIGHT, TEAM_OVERLAY_MAXNAME_WIDTH);
if (lwidth) {
p = CG_GetLocationString(CG_ConfigString(CS_LOCATIONS+ci->location));
if (!p || !*p)
p = "unknown";
len = CG_DrawStrlen(p);
if (len > lwidth)
len = lwidth;
// xx = x + TINYCHAR_WIDTH * 2 + TINYCHAR_WIDTH * pwidth +
// ((lwidth/2 - len/2) * TINYCHAR_WIDTH);
xx = x + TINYCHAR_WIDTH * 2 + TINYCHAR_WIDTH * pwidth;
CG_DrawStringExt( xx + xOffset, y,
p, hcolor, qfalse, qfalse, TINYCHAR_WIDTH, TINYCHAR_HEIGHT,
TEAM_OVERLAY_MAXLOCATION_WIDTH);
}
CG_GetColorForHealth( ci->health, ci->armor, hcolor );
Com_sprintf (st, sizeof(st), "%3i %3i", ci->health, ci->armor);
xx = x + TINYCHAR_WIDTH * 3 +
TINYCHAR_WIDTH * pwidth + TINYCHAR_WIDTH * lwidth;
CG_DrawStringExt( xx + xOffset, y,
st, hcolor, qfalse, qfalse,
TINYCHAR_WIDTH, TINYCHAR_HEIGHT, 0 );
// draw weapon icon
xx += TINYCHAR_WIDTH * 3;
if ( cg_weapons[ci->curWeapon].weaponIcon ) {
CG_DrawPic( xx + xOffset, y, TINYCHAR_WIDTH, TINYCHAR_HEIGHT,
cg_weapons[ci->curWeapon].weaponIcon );
} else {
CG_DrawPic( xx + xOffset, y, TINYCHAR_WIDTH, TINYCHAR_HEIGHT,
cgs.media.deferShader );
}
// Draw powerup icons
if (right) {
xx = x;
} else {
xx = x + w - TINYCHAR_WIDTH;
}
for (j = 0; j <= PW_NUM_POWERUPS; j++) {
if (ci->powerups & (1 << j)) {
item = BG_FindItemForPowerup( j );
if (item) {
CG_DrawPic( xx + xOffset, y, TINYCHAR_WIDTH, TINYCHAR_HEIGHT,
trap->R_RegisterShader( item->icon ) );
if (right) {
xx -= TINYCHAR_WIDTH;
} else {
xx += TINYCHAR_WIDTH;
}
}
}
}
y += TINYCHAR_HEIGHT;
}
}
return ret_y;
//#endif
}
static void CG_DrawPowerupIcons(int y)
{
int j;
int ico_size = 64;
//int y = ico_size/2;
int xOffset = 0;
gitem_t *item;
trap->R_SetColor( NULL );
if (!cg.snap)
{
return;
}
y += 16;
for (j = 0; j < PW_NUM_POWERUPS; j++)
{
if (cg.snap->ps.powerups[j] > cg.time)
{
int secondsleft = (cg.snap->ps.powerups[j] - cg.time)/1000;
item = BG_FindItemForPowerup( j );
if (item)
{
int icoShader = 0;
if (cgs.gametype == GT_CTY && (j == PW_REDFLAG || j == PW_BLUEFLAG))
{
if (j == PW_REDFLAG)
{
icoShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_rflag_ys" );
}
else
{
icoShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_bflag_ys" );
}
}
else
{
icoShader = trap->R_RegisterShader( item->icon );
}
CG_DrawPic( (640-(ico_size*1.1)) + xOffset, y, ico_size, ico_size, icoShader );
y += ico_size;
if (j != PW_REDFLAG && j != PW_BLUEFLAG && secondsleft < 999)
{
CG_DrawProportionalString((640-(ico_size*1.1))+(ico_size/2) + xOffset, y-8, va("%i", secondsleft), UI_CENTER | UI_BIGFONT | UI_DROPSHADOW, colorTable[CT_WHITE]);
}
y += (ico_size/3);
}
}
}
}
/*
=====================
CG_DrawUpperRight
=====================
*/
static void CG_DrawUpperRight( void ) {
float y = 0;
trap->R_SetColor( colorTable[CT_WHITE] );
if ( cgs.gametype >= GT_TEAM && cg_drawTeamOverlay.integer == 1 ) {
y = CG_DrawTeamOverlay( y, qtrue, qtrue );
}
if ( cg_drawSnapshot.integer ) {
y = CG_DrawSnapshot( y );
}
if ( cg_drawFPS.integer ) {
y = CG_DrawFPS( y );
}
if ( cg_drawTimer.integer ) {
y = CG_DrawTimer( y );
}
if ( ( cgs.gametype >= GT_TEAM || cg.predictedPlayerState.m_iVehicleNum )
&& cg_drawRadar.integer )
{//draw Radar in Siege mode or when in a vehicle of any kind
y = CG_DrawRadar ( y );
}
y = CG_DrawEnemyInfo ( y );
y = CG_DrawMiniScoreboard ( y );
CG_DrawPowerupIcons(y);
}
/*
===================
CG_DrawReward
===================
*/
#ifdef JK2AWARDS
static void CG_DrawReward( void ) {
float *color;
int i, count;
float x, y;
char buf[32];
if ( !cg_drawRewards.integer ) {
return;
}
color = CG_FadeColor( cg.rewardTime, REWARD_TIME );
if ( !color ) {
if (cg.rewardStack > 0) {
for(i = 0; i < cg.rewardStack; i++) {
cg.rewardSound[i] = cg.rewardSound[i+1];
cg.rewardShader[i] = cg.rewardShader[i+1];
cg.rewardCount[i] = cg.rewardCount[i+1];
}
cg.rewardTime = cg.time;
cg.rewardStack--;
color = CG_FadeColor( cg.rewardTime, REWARD_TIME );
trap->S_StartLocalSound(cg.rewardSound[0], CHAN_ANNOUNCER);
} else {
return;
}
}
trap->R_SetColor( color );
/*
count = cg.rewardCount[0]/10; // number of big rewards to draw
if (count) {
y = 4;
x = 320 - count * ICON_SIZE;
for ( i = 0 ; i < count ; i++ ) {
CG_DrawPic( x, y, (ICON_SIZE*2)-4, (ICON_SIZE*2)-4, cg.rewardShader[0] );
x += (ICON_SIZE*2);
}
}
count = cg.rewardCount[0] - count*10; // number of small rewards to draw
*/
if ( cg.rewardCount[0] >= 10 ) {
y = 56;
x = 320 - ICON_SIZE/2;
CG_DrawPic( x, y, ICON_SIZE-4, ICON_SIZE-4, cg.rewardShader[0] );
Com_sprintf(buf, sizeof(buf), "%d", cg.rewardCount[0]);
x = ( SCREEN_WIDTH - SMALLCHAR_WIDTH * CG_DrawStrlen( buf ) ) / 2;
CG_DrawStringExt( x, y+ICON_SIZE, buf, color, qfalse, qtrue,
SMALLCHAR_WIDTH, SMALLCHAR_HEIGHT, 0 );
}
else {
count = cg.rewardCount[0];
y = 56;
x = 320 - count * ICON_SIZE/2;
for ( i = 0 ; i < count ; i++ ) {
CG_DrawPic( x, y, ICON_SIZE-4, ICON_SIZE-4, cg.rewardShader[0] );
x += ICON_SIZE;
}
}
trap->R_SetColor( NULL );
}
#endif
/*
===============================================================================
LAGOMETER
===============================================================================
*/
#define LAG_SAMPLES 128
struct lagometer_s {
int frameSamples[LAG_SAMPLES];
int frameCount;
int snapshotFlags[LAG_SAMPLES];
int snapshotSamples[LAG_SAMPLES];
int snapshotCount;
} lagometer;
/*
==============
CG_AddLagometerFrameInfo
Adds the current interpolate / extrapolate bar for this frame
==============
*/
void CG_AddLagometerFrameInfo( void ) {
int offset;
offset = cg.time - cg.latestSnapshotTime;
lagometer.frameSamples[ lagometer.frameCount & ( LAG_SAMPLES - 1) ] = offset;
lagometer.frameCount++;
}
/*
==============
CG_AddLagometerSnapshotInfo
Each time a snapshot is received, log its ping time and
the number of snapshots that were dropped before it.
Pass NULL for a dropped packet.
==============
*/
void CG_AddLagometerSnapshotInfo( snapshot_t *snap ) {
// dropped packet
if ( !snap ) {
lagometer.snapshotSamples[ lagometer.snapshotCount & ( LAG_SAMPLES - 1) ] = -1;
lagometer.snapshotCount++;
return;
}
// add this snapshot's info
lagometer.snapshotSamples[ lagometer.snapshotCount & ( LAG_SAMPLES - 1) ] = snap->ping;
lagometer.snapshotFlags[ lagometer.snapshotCount & ( LAG_SAMPLES - 1) ] = snap->snapFlags;
lagometer.snapshotCount++;
}
/*
==============
CG_DrawDisconnect
Should we draw something different for long lag vs no packets?
==============
*/
static void CG_DrawDisconnect( void ) {
float x, y;
int cmdNum;
usercmd_t cmd;
const char *s;
int w; // bk010215 - FIXME char message[1024];
if (cg.mMapChange)
{
s = CG_GetStringEdString("MP_INGAME", "SERVER_CHANGING_MAPS"); // s = "Server Changing Maps";
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString( 320 - w/2, 100, s, 1.0F);
s = CG_GetStringEdString("MP_INGAME", "PLEASE_WAIT"); // s = "Please wait...";
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString( 320 - w/2, 200, s, 1.0F);
return;
}
// draw the phone jack if we are completely past our buffers
cmdNum = trap->GetCurrentCmdNumber() - CMD_BACKUP + 1;
trap->GetUserCmd( cmdNum, &cmd );
if ( cmd.serverTime <= cg.snap->ps.commandTime
|| cmd.serverTime > cg.time ) { // special check for map_restart // bk 0102165 - FIXME
return;
}
// also add text in center of screen
s = CG_GetStringEdString("MP_INGAME", "CONNECTION_INTERRUPTED"); // s = "Connection Interrupted"; // bk 010215 - FIXME
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString( 320 - w/2, 100, s, 1.0F);
// blink the icon
if ( ( cg.time >> 9 ) & 1 ) {
return;
}
x = 640 - 48;
y = 480 - 48;
CG_DrawPic( x, y, 48, 48, trap->R_RegisterShader("gfx/2d/net.tga" ) );
}
#define MAX_LAGOMETER_PING 900
#define MAX_LAGOMETER_RANGE 300
/*
==============
CG_DrawLagometer
==============
*/
static void CG_DrawLagometer( void ) {
int a, x, y, i;
float v;
float ax, ay, aw, ah, mid, range;
int color;
float vscale;
if ( !cg_lagometer.integer || cgs.localServer ) {
CG_DrawDisconnect();
return;
}
//
// draw the graph
//
x = 640 - 48;
y = 480 - 144;
trap->R_SetColor( NULL );
CG_DrawPic( x, y, 48, 48, cgs.media.lagometerShader );
ax = x;
ay = y;
aw = 48;
ah = 48;
color = -1;
range = ah / 3;
mid = ay + range;
vscale = range / MAX_LAGOMETER_RANGE;
// draw the frame interpoalte / extrapolate graph
for ( a = 0 ; a < aw ; a++ ) {
i = ( lagometer.frameCount - 1 - a ) & (LAG_SAMPLES - 1);
v = lagometer.frameSamples[i];
v *= vscale;
if ( v > 0 ) {
if ( color != 1 ) {
color = 1;
trap->R_SetColor( g_color_table[ColorIndex(COLOR_YELLOW)] );
}
if ( v > range ) {
v = range;
}
trap->R_DrawStretchPic ( ax + aw - a, mid - v, 1, v, 0, 0, 0, 0, cgs.media.whiteShader );
} else if ( v < 0 ) {
if ( color != 2 ) {
color = 2;
trap->R_SetColor( g_color_table[ColorIndex(COLOR_BLUE)] );
}
v = -v;
if ( v > range ) {
v = range;
}
trap->R_DrawStretchPic( ax + aw - a, mid, 1, v, 0, 0, 0, 0, cgs.media.whiteShader );
}
}
// draw the snapshot latency / drop graph
range = ah / 2;
vscale = range / MAX_LAGOMETER_PING;
for ( a = 0 ; a < aw ; a++ ) {
i = ( lagometer.snapshotCount - 1 - a ) & (LAG_SAMPLES - 1);
v = lagometer.snapshotSamples[i];
if ( v > 0 ) {
if ( lagometer.snapshotFlags[i] & SNAPFLAG_RATE_DELAYED ) {
if ( color != 5 ) {
color = 5; // YELLOW for rate delay
trap->R_SetColor( g_color_table[ColorIndex(COLOR_YELLOW)] );
}
} else {
if ( color != 3 ) {
color = 3;
trap->R_SetColor( g_color_table[ColorIndex(COLOR_GREEN)] );
}
}
v = v * vscale;
if ( v > range ) {
v = range;
}
trap->R_DrawStretchPic( ax + aw - a, ay + ah - v, 1, v, 0, 0, 0, 0, cgs.media.whiteShader );
} else if ( v < 0 ) {
if ( color != 4 ) {
color = 4; // RED for dropped snapshots
trap->R_SetColor( g_color_table[ColorIndex(COLOR_RED)] );
}
trap->R_DrawStretchPic( ax + aw - a, ay + ah - range, 1, range, 0, 0, 0, 0, cgs.media.whiteShader );
}
}
trap->R_SetColor( NULL );
if ( cg_noPredict.integer || g_synchronousClients.integer ) {
CG_DrawBigString( ax, ay, "snc", 1.0 );
}
CG_DrawDisconnect();
}
void CG_DrawSiegeMessage( const char *str, int objectiveScreen )
{
// if (!( trap->Key_GetCatcher() & KEYCATCH_UI ))
{
trap->OpenUIMenu(UIMENU_CLOSEALL);
trap->Cvar_Set("cg_siegeMessage", str);
if (objectiveScreen)
{
trap->OpenUIMenu(UIMENU_SIEGEOBJECTIVES);
}
else
{
trap->OpenUIMenu(UIMENU_SIEGEMESSAGE);
}
}
}
void CG_DrawSiegeMessageNonMenu( const char *str )
{
char text[1024];
if (str[0]=='@')
{
trap->SE_GetStringTextString(str+1, text, sizeof(text));
str = text;
}
CG_CenterPrint(str, SCREEN_HEIGHT * 0.30, BIGCHAR_WIDTH);
}
/*
===============================================================================
CENTER PRINTING
===============================================================================
*/
/*
==============
CG_CenterPrint
Called for important messages that should stay in the center of the screen
for a few moments
==============
*/
void CG_CenterPrint( const char *str, int y, int charWidth ) {
char *s;
//[BugFix19]
int i = 0;
//[/BugFix19]
Q_strncpyz( cg.centerPrint, str, sizeof(cg.centerPrint) );
cg.centerPrintTime = cg.time;
cg.centerPrintY = y;
cg.centerPrintCharWidth = charWidth;
// count the number of lines for centering
cg.centerPrintLines = 1;
s = cg.centerPrint;
while( *s )
{
//[BugFix19]
i++;
if(i >= 50)
{//maxed out a line of text, this will make the line spill over onto another line.
i = 0;
cg.centerPrintLines++;
}
else if (*s == '\n')
//if (*s == '\n')
//[/BugFix19]
cg.centerPrintLines++;
s++;
}
}
/*
===================
CG_DrawCenterString
===================
*/
qboolean BG_IsWhiteSpace( char c )
{//this function simply checks to see if the given character is whitespace.
if ( c == ' ' || c == '\n' || c == '\0' )
return qtrue;
return qfalse;
}
static void CG_DrawCenterString( void ) {
char *start;
int l;
int x, y, w;
int h;
float *color;
const float scale = 1.0; //0.5
if ( !cg.centerPrintTime ) {
return;
}
color = CG_FadeColor( cg.centerPrintTime, 1000 * cg_centerTime.value );
if ( !color ) {
return;
}
trap->R_SetColor( color );
start = cg.centerPrint;
y = cg.centerPrintY - cg.centerPrintLines * BIGCHAR_HEIGHT / 2;
while ( 1 ) {
char linebuffer[1024];
for ( l = 0; l < 50; l++ ) {
if ( !start[l] || start[l] == '\n' ) {
break;
}
linebuffer[l] = start[l];
}
linebuffer[l] = 0;
//[BugFix19]
if(!BG_IsWhiteSpace(start[l]) && !BG_IsWhiteSpace(linebuffer[l-1]) )
{//we might have cut a word off, attempt to find a spot where we won't cut words off at.
int savedL = l;
int counter = l-2;
for(; counter >= 0; counter--)
{
if(BG_IsWhiteSpace(start[counter]))
{//this location is whitespace, line break from this position
linebuffer[counter] = 0;
l = counter + 1;
break;
}
}
if(counter < 0)
{//couldn't find a break in the text, just go ahead and cut off the word mid-word.
l = savedL;
}
}
//[/BugFix19]
w = CG_Text_Width(linebuffer, scale, FONT_MEDIUM);
h = CG_Text_Height(linebuffer, scale, FONT_MEDIUM);
x = (SCREEN_WIDTH - w) / 2;
CG_Text_Paint(x, y + h, scale, color, linebuffer, 0, 0, ITEM_TEXTSTYLE_SHADOWEDMORE, FONT_MEDIUM);
y += h + 6;
//[BugFix19]
//this method of advancing to new line from the start of the array was causing long lines without
//new lines to be totally truncated.
if(start[l] && start[l] == '\n')
{//next char is a newline, advance past
l++;
}
if ( !start[l] )
{//end of string, we're done.
break;
}
//advance pointer to the last character that we didn't read in.
start = &start[l];
//[/BugFix19]
}
trap->R_SetColor( NULL );
}
/*
================================================================================
CROSSHAIR
================================================================================
*/
#define HEALTH_WIDTH 50.0f
#define HEALTH_HEIGHT 5.0f
//see if we can draw some extra info on this guy based on our class
void CG_DrawSiegeInfo(centity_t *cent, float chX, float chY, float chW, float chH)
{
siegeExtended_t *se = &cg_siegeExtendedData[cent->currentState.number];
clientInfo_t *ci;
const char *configstring, *v;
siegeClass_t *siegeClass;
vec4_t aColor;
vec4_t cColor;
float x;
float y;
float percent;
int ammoMax;
assert(cent->currentState.number < MAX_CLIENTS);
if (se->lastUpdated > cg.time)
{ //strange, shouldn't happen
return;
}
if ((cg.time - se->lastUpdated) > 10000)
{ //if you haven't received a status update on this guy in 10 seconds, forget about it
return;
}
if (cent->currentState.eFlags & EF_DEAD)
{ //he's dead, don't display info on him
return;
}
if (cent->currentState.weapon != se->weapon)
{ //data is invalidated until it syncs back again
return;
}
ci = &cgs.clientinfo[cent->currentState.number];
if (ci->team != cg.predictedPlayerState.persistant[PERS_TEAM])
{ //not on the same team
return;
}
configstring = CG_ConfigString( cg.predictedPlayerState.clientNum + CS_PLAYERS );
v = Info_ValueForKey( configstring, "siegeclass" );
if (!v || !v[0])
{ //don't have siege class in info?
return;
}
siegeClass = BG_SiegeFindClassByName(v);
if (!siegeClass)
{ //invalid
return;
}
if (!(siegeClass->classflags & (1<health/(float)se->maxhealth)*HEALTH_WIDTH;
//color of the bar
aColor[0] = 0.0f;
aColor[1] = 1.0f;
aColor[2] = 0.0f;
aColor[3] = 0.4f;
//color of greyed out "missing health"
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.4f;
//draw the background (black)
CG_DrawRect(x, y, HEALTH_WIDTH, HEALTH_HEIGHT, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f, percent-1.0f, HEALTH_HEIGHT-1.0f, aColor);
//then draw the other part greyed out
CG_FillRect(x+percent, y+1.0f, HEALTH_WIDTH-percent-1.0f, HEALTH_HEIGHT-1.0f, cColor);
//now draw his ammo
ammoMax = ammoData[weaponData[cent->currentState.weapon].ammoIndex].max;
if ( (cent->currentState.eFlags & EF_DOUBLE_AMMO) )
{
ammoMax *= 2;
}
x = chX+((chW/2)-(HEALTH_WIDTH/2));
y = (chY+chH) + HEALTH_HEIGHT + 10.0f;
if (!weaponData[cent->currentState.weapon].energyPerShot &&
!weaponData[cent->currentState.weapon].altEnergyPerShot)
{ //a weapon that takes no ammo, so show full
percent = HEALTH_WIDTH;
}
else
{
percent = ((float)se->ammo/(float)ammoMax)*HEALTH_WIDTH;
}
//color of the bar
aColor[0] = 1.0f;
aColor[1] = 1.0f;
aColor[2] = 0.0f;
aColor[3] = 0.4f;
//color of greyed out "missing health"
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.4f;
//draw the background (black)
CG_DrawRect(x, y, HEALTH_WIDTH, HEALTH_HEIGHT, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f, percent-1.0f, HEALTH_HEIGHT-1.0f, aColor);
//then draw the other part greyed out
CG_FillRect(x+percent, y+1.0f, HEALTH_WIDTH-percent-1.0f, HEALTH_HEIGHT-1.0f, cColor);
}
//draw the health bar based on current "health" and maxhealth
void CG_DrawHealthBar(centity_t *cent, float chX, float chY, float chW, float chH)
{
vec4_t aColor;
vec4_t cColor;
float x = chX+((chW/2)-(HEALTH_WIDTH/2));
float y = (chY+chH) + 8.0f;
float percent = ((float)cent->currentState.health/(float)cent->currentState.maxhealth)*HEALTH_WIDTH;
if (percent <= 0)
{
return;
}
//color of the bar
if (!cent->currentState.teamowner || cgs.gametype < GT_TEAM)
{ //not owned by a team or teamplay
aColor[0] = 1.0f;
aColor[1] = 1.0f;
aColor[2] = 0.0f;
aColor[3] = 0.4f;
}
else if (cent->currentState.teamowner == cg.predictedPlayerState.persistant[PERS_TEAM])
{ //owned by my team
aColor[0] = 0.0f;
aColor[1] = 1.0f;
aColor[2] = 0.0f;
aColor[3] = 0.4f;
}
else
{ //hostile
aColor[0] = 1.0f;
aColor[1] = 0.0f;
aColor[2] = 0.0f;
aColor[3] = 0.4f;
}
//color of greyed out "missing health"
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.4f;
//draw the background (black)
CG_DrawRect(x, y, HEALTH_WIDTH, HEALTH_HEIGHT, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f, percent-1.0f, HEALTH_HEIGHT-1.0f, aColor);
//then draw the other part greyed out
CG_FillRect(x+percent, y+1.0f, HEALTH_WIDTH-percent-1.0f, HEALTH_HEIGHT-1.0f, cColor);
}
//same routine (at least for now), draw progress of a "hack" or whatever
void CG_DrawHaqrBar(float chX, float chY, float chW, float chH)
{
vec4_t aColor;
vec4_t cColor;
float x = chX+((chW/2)-(HEALTH_WIDTH/2));
float y = (chY+chH) + 8.0f;
float percent = (((float)cg.predictedPlayerState.hackingTime-(float)cg.time)/(float)cg.predictedPlayerState.hackingBaseTime)*HEALTH_WIDTH;
if (percent > HEALTH_WIDTH ||
percent < 1.0f)
{
return;
}
//color of the bar
aColor[0] = 1.0f;
aColor[1] = 1.0f;
aColor[2] = 0.0f;
aColor[3] = 0.4f;
//color of greyed out done area
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.1f;
//draw the background (black)
CG_DrawRect(x, y, HEALTH_WIDTH, HEALTH_HEIGHT, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f, percent-1.0f, HEALTH_HEIGHT-1.0f, aColor);
//then draw the other part greyed out
CG_FillRect(x+percent, y+1.0f, HEALTH_WIDTH-percent-1.0f, HEALTH_HEIGHT-1.0f, cColor);
//draw the hacker icon
CG_DrawPic(x, y-HEALTH_WIDTH, HEALTH_WIDTH, HEALTH_WIDTH, cgs.media.hackerIconShader);
}
//generic timing bar
int cg_genericTimerBar = 0;
int cg_genericTimerDur = 0;
vec4_t cg_genericTimerColor;
#define CGTIMERBAR_H 50.0f
#define CGTIMERBAR_W 10.0f
#define CGTIMERBAR_X (SCREEN_WIDTH-CGTIMERBAR_W-120.0f)
#define CGTIMERBAR_Y (SCREEN_HEIGHT-CGTIMERBAR_H-20.0f)
void CG_DrawGenericTimerBar(void)
{
vec4_t aColor;
vec4_t cColor;
float x = CGTIMERBAR_X;
float y = CGTIMERBAR_Y;
float percent = ((float)(cg_genericTimerBar-cg.time)/(float)cg_genericTimerDur)*CGTIMERBAR_H;
if (percent > CGTIMERBAR_H)
{
return;
}
if (percent < 0.1f)
{
percent = 0.1f;
}
//color of the bar
aColor[0] = cg_genericTimerColor[0];
aColor[1] = cg_genericTimerColor[1];
aColor[2] = cg_genericTimerColor[2];
aColor[3] = cg_genericTimerColor[3];
//color of greyed out "missing fuel"
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.1f;
//draw the background (black)
CG_DrawRect(x, y, CGTIMERBAR_W, CGTIMERBAR_H, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f+(CGTIMERBAR_H-percent), CGTIMERBAR_W-2.0f, CGTIMERBAR_H-1.0f-(CGTIMERBAR_H-percent), aColor);
//then draw the other part greyed out
CG_FillRect(x+1.0f, y+1.0f, CGTIMERBAR_W-2.0f, CGTIMERBAR_H-percent, cColor);
}
/*
=================
CG_DrawCrosshair
=================
*/
float cg_crosshairPrevPosX = 0;
float cg_crosshairPrevPosY = 0;
#define CRAZY_CROSSHAIR_MAX_ERROR_X (100.0f*640.0f/480.0f)
#define CRAZY_CROSSHAIR_MAX_ERROR_Y (100.0f)
void CG_LerpCrosshairPos( float *x, float *y )
{
if ( cg_crosshairPrevPosX )
{//blend from old pos
float maxMove = 30.0f * ((float)cg.frametime/500.0f) * 640.0f/480.0f;
float xDiff = (*x - cg_crosshairPrevPosX);
if ( fabs(xDiff) > CRAZY_CROSSHAIR_MAX_ERROR_X )
{
maxMove = CRAZY_CROSSHAIR_MAX_ERROR_X;
}
if ( xDiff > maxMove )
{
*x = cg_crosshairPrevPosX + maxMove;
}
else if ( xDiff < -maxMove )
{
*x = cg_crosshairPrevPosX - maxMove;
}
}
cg_crosshairPrevPosX = *x;
if ( cg_crosshairPrevPosY )
{//blend from old pos
float maxMove = 30.0f * ((float)cg.frametime/500.0f);
float yDiff = (*y - cg_crosshairPrevPosY);
if ( fabs(yDiff) > CRAZY_CROSSHAIR_MAX_ERROR_Y )
{
maxMove = CRAZY_CROSSHAIR_MAX_ERROR_X;
}
if ( yDiff > maxMove )
{
*y = cg_crosshairPrevPosY + maxMove;
}
else if ( yDiff < -maxMove )
{
*y = cg_crosshairPrevPosY - maxMove;
}
}
cg_crosshairPrevPosY = *y;
}
vec3_t cg_crosshairPos={0,0,0};
static void CG_DrawCrosshair( vec3_t worldPoint, int chEntValid ) {
float w, h;
qhandle_t hShader = 0;
float f;
float x, y;
qboolean corona = qfalse;
vec4_t ecolor = {0,0,0,0};
centity_t *crossEnt = NULL;
float chX, chY;
if ( worldPoint )
{
VectorCopy( worldPoint, cg_crosshairPos );
}
if ( !cg_drawCrosshair.integer )
{
return;
}
if (cg.snap->ps.fallingToDeath)
{
return;
}
if ( cg.predictedPlayerState.zoomMode != 0 )
{//not while scoped
return;
}
if ( cg_crosshairHealth.integer )
{
vec4_t hcolor;
CG_ColorForHealth( hcolor );
trap->R_SetColor( hcolor );
}
else
{
//set color based on what kind of ent is under crosshair
if ( cg.crosshairClientNum >= ENTITYNUM_WORLD )
{
trap->R_SetColor( NULL );
}
//rwwFIXMEFIXME: Write this a different way, it's getting a bit too sloppy looking
else if (chEntValid &&
(cg_entities[cg.crosshairClientNum].currentState.number < MAX_CLIENTS ||
cg_entities[cg.crosshairClientNum].currentState.eType == ET_NPC ||
cg_entities[cg.crosshairClientNum].currentState.shouldtarget ||
cg_entities[cg.crosshairClientNum].currentState.health || //always show ents with health data under crosshair
(cg_entities[cg.crosshairClientNum].currentState.eType == ET_MOVER && cg_entities[cg.crosshairClientNum].currentState.bolt1 && cg.predictedPlayerState.weapon == WP_SABER) ||
(cg_entities[cg.crosshairClientNum].currentState.eType == ET_MOVER && cg_entities[cg.crosshairClientNum].currentState.teamowner)))
{
crossEnt = &cg_entities[cg.crosshairClientNum];
if ( crossEnt->currentState.powerups & (1 <currentState.number < MAX_CLIENTS )
{
if (cgs.gametype >= GT_TEAM &&
cgs.clientinfo[crossEnt->currentState.number].team == cgs.clientinfo[cg.snap->ps.clientNum].team )
{
//Allies are green
ecolor[0] = 0.0;//R
ecolor[1] = 1.0;//G
ecolor[2] = 0.0;//B
}
else
{
if (cgs.gametype == GT_POWERDUEL &&
cgs.clientinfo[crossEnt->currentState.number].duelTeam == cgs.clientinfo[cg.snap->ps.clientNum].duelTeam)
{ //on the same duel team in powerduel, so he's a friend
ecolor[0] = 0.0;//R
ecolor[1] = 1.0;//G
ecolor[2] = 0.0;//B
}
else
{ //Enemies are red
ecolor[0] = 1.0;//R
ecolor[1] = 0.0;//G
ecolor[2] = 0.0;//B
}
}
if (cg.snap->ps.duelInProgress)
{
if (crossEnt->currentState.number != cg.snap->ps.duelIndex)
{ //grey out crosshair for everyone but your foe if you're in a duel
ecolor[0] = 0.4f;
ecolor[1] = 0.4f;
ecolor[2] = 0.4f;
}
}
else if (crossEnt->currentState.bolt1)
{ //this fellow is in a duel. We just checked if we were in a duel above, so
//this means we aren't and he is. Which of course means our crosshair greys out over him.
ecolor[0] = 0.4f;
ecolor[1] = 0.4f;
ecolor[2] = 0.4f;
}
}
else if (crossEnt->currentState.shouldtarget || crossEnt->currentState.eType == ET_NPC)
{
//VectorCopy( crossEnt->startRGBA, ecolor );
if ( !ecolor[0] && !ecolor[1] && !ecolor[2] )
{
// We really don't want black, so set it to yellow
ecolor[0] = 1.0f;//R
ecolor[1] = 0.8f;//G
ecolor[2] = 0.3f;//B
}
if (crossEnt->currentState.eType == ET_NPC)
{
int plTeam;
if (cgs.gametype == GT_SIEGE)
{
plTeam = cg.predictedPlayerState.persistant[PERS_TEAM];
}
else
{
plTeam = NPCTEAM_PLAYER;
}
if ( crossEnt->currentState.powerups & (1 <currentState.teamowner )
{ //not on a team
if (!crossEnt->currentState.teamowner ||
crossEnt->currentState.NPC_class == CLASS_VEHICLE)
{ //neutral
if (crossEnt->currentState.owner < MAX_CLIENTS)
{ //base color on who is pilotting this thing
clientInfo_t *ci = &cgs.clientinfo[crossEnt->currentState.owner];
if (cgs.gametype >= GT_TEAM && ci->team == cg.predictedPlayerState.persistant[PERS_TEAM])
{ //friendly
ecolor[0] = 0.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
else
{ //hostile
ecolor[0] = 1.0f;//R
ecolor[1] = 0.0f;//G
ecolor[2] = 0.0f;//B
}
}
else
{ //unmanned
ecolor[0] = 1.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
}
else
{
ecolor[0] = 1.0f;//R
ecolor[1] = 0.0f;//G
ecolor[2] = 0.0f;//B
}
}
else if ( crossEnt->currentState.teamowner != plTeam )
{// on enemy team
ecolor[0] = 1.0f;//R
ecolor[1] = 0.0f;//G
ecolor[2] = 0.0f;//B
}
else
{ //a friend
ecolor[0] = 0.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
}
else if ( crossEnt->currentState.teamowner == TEAM_RED
|| crossEnt->currentState.teamowner == TEAM_BLUE )
{
if (cgs.gametype < GT_TEAM)
{ //not teamplay, just neutral then
ecolor[0] = 1.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
else if ( crossEnt->currentState.teamowner != cgs.clientinfo[cg.snap->ps.clientNum].team )
{ //on the enemy team
ecolor[0] = 1.0f;//R
ecolor[1] = 0.0f;//G
ecolor[2] = 0.0f;//B
}
else
{ //on my team
ecolor[0] = 0.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
}
else if (crossEnt->currentState.owner == cg.snap->ps.clientNum ||
(cgs.gametype >= GT_TEAM && crossEnt->currentState.teamowner == cgs.clientinfo[cg.snap->ps.clientNum].team))
{
ecolor[0] = 0.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
else if (crossEnt->currentState.teamowner == 16 ||
(cgs.gametype >= GT_TEAM && crossEnt->currentState.teamowner && crossEnt->currentState.teamowner != cgs.clientinfo[cg.snap->ps.clientNum].team))
{
ecolor[0] = 1.0f;//R
ecolor[1] = 0.0f;//G
ecolor[2] = 0.0f;//B
}
}
else if (crossEnt->currentState.eType == ET_MOVER && crossEnt->currentState.bolt1 && cg.predictedPlayerState.weapon == WP_SABER)
{ //can push/pull this mover. Only show it if we're using the saber.
ecolor[0] = 0.2f;
ecolor[1] = 0.5f;
ecolor[2] = 1.0f;
corona = qtrue;
}
else if (crossEnt->currentState.eType == ET_MOVER && crossEnt->currentState.teamowner)
{ //a team owns this - if it's my team green, if not red, if not teamplay then yellow
if (cgs.gametype < GT_TEAM)
{
ecolor[0] = 1.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
else if (cg.predictedPlayerState.persistant[PERS_TEAM] != crossEnt->currentState.teamowner)
{ //not my team
ecolor[0] = 1.0f;//R
ecolor[1] = 0.0f;//G
ecolor[2] = 0.0f;//B
}
else
{ //my team
ecolor[0] = 0.0f;//R
ecolor[1] = 1.0f;//G
ecolor[2] = 0.0f;//B
}
}
else if (crossEnt->currentState.health)
{
if (!crossEnt->currentState.teamowner || cgs.gametype < GT_TEAM)
{ //not owned by a team or teamplay
ecolor[0] = 1.0f;
ecolor[1] = 1.0f;
ecolor[2] = 0.0f;
}
else if (crossEnt->currentState.teamowner == cg.predictedPlayerState.persistant[PERS_TEAM])
{ //owned by my team
ecolor[0] = 0.0f;
ecolor[1] = 1.0f;
ecolor[2] = 0.0f;
}
else
{ //hostile
ecolor[0] = 1.0f;
ecolor[1] = 0.0f;
ecolor[2] = 0.0f;
}
}
ecolor[3] = 1.0f;
trap->R_SetColor( ecolor );
}
}
if ( cg.predictedPlayerState.m_iVehicleNum )
{//I'm in a vehicle
centity_t *vehCent = &cg_entities[cg.predictedPlayerState.m_iVehicleNum];
if ( vehCent
&& vehCent->m_pVehicle
&& vehCent->m_pVehicle->m_pVehicleInfo
&& vehCent->m_pVehicle->m_pVehicleInfo->crosshairShaderHandle )
{
hShader = vehCent->m_pVehicle->m_pVehicleInfo->crosshairShaderHandle;
}
//bigger by default
w = cg_crosshairSize.value*2.0f;
h = w;
}
else
{
w = h = cg_crosshairSize.value;
}
// pulse the size of the crosshair when picking up items
f = cg.time - cg.itemPickupBlendTime;
if ( f > 0 && f < ITEM_BLOB_TIME ) {
f /= ITEM_BLOB_TIME;
w *= ( 1 + f );
h *= ( 1 + f );
}
if ( worldPoint && VectorLength( worldPoint ) )
{
if ( !CG_WorldCoordToScreenCoordFloat( worldPoint, &x, &y ) )
{//off screen, don't draw it
return;
}
//CG_LerpCrosshairPos( &x, &y );
x -= 320;
y -= 240;
}
else
{
x = cg_crosshairX.integer;
y = cg_crosshairY.integer;
}
if ( !hShader )
{
hShader = cgs.media.crosshairShader[Com_Clampi( 1, NUM_CROSSHAIRS, cg_drawCrosshair.integer ) - 1];
}
chX = x + cg.refdef.x + 0.5 * (640 - w);
chY = y + cg.refdef.y + 0.5 * (480 - h);
trap->R_DrawStretchPic( chX, chY, w, h, 0, 0, 1, 1, hShader );
//draw a health bar directly under the crosshair if we're looking at something
//that takes damage
if (crossEnt &&
crossEnt->currentState.maxhealth)
{
CG_DrawHealthBar(crossEnt, chX, chY, w, h);
chY += HEALTH_HEIGHT*2;
}
else if (crossEnt && crossEnt->currentState.number < MAX_CLIENTS)
{
if (cgs.gametype == GT_SIEGE)
{
CG_DrawSiegeInfo(crossEnt, chX, chY, w, h);
chY += HEALTH_HEIGHT*4;
}
if (cg.crosshairVehNum && cg.time == cg.crosshairVehTime)
{ //it was in the crosshair this frame
centity_t *hisVeh = &cg_entities[cg.crosshairVehNum];
if (hisVeh->currentState.eType == ET_NPC &&
hisVeh->currentState.NPC_class == CLASS_VEHICLE &&
hisVeh->currentState.maxhealth &&
hisVeh->m_pVehicle)
{ //draw the health for this vehicle
CG_DrawHealthBar(hisVeh, chX, chY, w, h);
chY += HEALTH_HEIGHT*2;
}
}
}
if (cg.predictedPlayerState.hackingTime)
{ //hacking something
CG_DrawHaqrBar(chX, chY, w, h);
}
if (cg_genericTimerBar > cg.time)
{ //draw generic timing bar, can be used for whatever
CG_DrawGenericTimerBar();
}
if ( corona ) // drawing extra bits
{
ecolor[3] = 0.5f;
ecolor[0] = ecolor[1] = ecolor[2] = (1 - ecolor[3]) * ( sin( cg.time * 0.001f ) * 0.08f + 0.35f ); // don't draw full color
ecolor[3] = 1.0f;
trap->R_SetColor( ecolor );
w *= 2.0f;
h *= 2.0f;
trap->R_DrawStretchPic( x + cg.refdef.x + 0.5 * (640 - w),
y + cg.refdef.y + 0.5 * (480 - h),
w, h, 0, 0, 1, 1, cgs.media.forceCoronaShader );
}
trap->R_SetColor( NULL );
}
qboolean CG_WorldCoordToScreenCoordFloat(vec3_t worldCoord, float *x, float *y)
{
vec3_t trans;
float xc, yc;
float px, py;
float z;
px = tan(cg.refdef.fov_x * (M_PI / 360) );
py = tan(cg.refdef.fov_y * (M_PI / 360) );
VectorSubtract(worldCoord, cg.refdef.vieworg, trans);
xc = 640 / 2.0;
yc = 480 / 2.0;
// z = how far is the object in our forward direction
z = DotProduct(trans, cg.refdef.viewaxis[0]);
if (z <= 0.001)
return qfalse;
*x = xc - DotProduct(trans, cg.refdef.viewaxis[1])*xc/(z*px);
*y = yc - DotProduct(trans, cg.refdef.viewaxis[2])*yc/(z*py);
return qtrue;
}
qboolean CG_WorldCoordToScreenCoord( vec3_t worldCoord, int *x, int *y ) {
float xF, yF;
if ( CG_WorldCoordToScreenCoordFloat( worldCoord, &xF, &yF ) ) {
*x = (int)xF;
*y = (int)yF;
return qtrue;
}
return qfalse;
}
/*
====================
CG_SaberClashFlare
====================
*/
int cg_saberFlashTime = 0;
vec3_t cg_saberFlashPos = {0, 0, 0};
void CG_SaberClashFlare( void )
{
int t, maxTime = 150;
vec3_t dif;
vec4_t color;
int x,y;
float v, len;
trace_t tr;
t = cg.time - cg_saberFlashTime;
if ( t <= 0 || t >= maxTime )
{
return;
}
// Don't do clashes for things that are behind us
VectorSubtract( cg_saberFlashPos, cg.refdef.vieworg, dif );
if ( DotProduct( dif, cg.refdef.viewaxis[0] ) < 0.2 )
{
return;
}
CG_Trace( &tr, cg.refdef.vieworg, NULL, NULL, cg_saberFlashPos, -1, CONTENTS_SOLID );
if ( tr.fraction < 1.0f )
{
return;
}
len = VectorNormalize( dif );
// clamp to a known range
/*
if ( len > 800 )
{
len = 800;
}
*/
if ( len > 1200 )
{
return;
}
v = ( 1.0f - ((float)t / maxTime )) * ((1.0f - ( len / 800.0f )) * 2.0f + 0.35f);
if (v < 0.001f)
{
v = 0.001f;
}
if ( !CG_WorldCoordToScreenCoord( cg_saberFlashPos, &x, &y ) ) {
return;
}
VectorSet4( color, 0.8f, 0.8f, 0.8f, 1.0f );
trap->R_SetColor( color );
CG_DrawPic( x - ( v * 300 ), y - ( v * 300 ),
v * 600, v * 600,
trap->R_RegisterShader( "gfx/effects/saberFlare" ));
}
void CG_DottedLine( float x1, float y1, float x2, float y2, float dotSize, int numDots, vec4_t color, float alpha )
{
float x, y, xDiff, yDiff, xStep, yStep;
vec4_t colorRGBA;
int dotNum = 0;
VectorCopy4( color, colorRGBA );
colorRGBA[3] = alpha;
trap->R_SetColor( colorRGBA );
xDiff = x2-x1;
yDiff = y2-y1;
xStep = xDiff/(float)numDots;
yStep = yDiff/(float)numDots;
for ( dotNum = 0; dotNum < numDots; dotNum++ )
{
x = x1 + (xStep*dotNum) - (dotSize*0.5f);
y = y1 + (yStep*dotNum) - (dotSize*0.5f);
CG_DrawPic( x, y, dotSize, dotSize, cgs.media.whiteShader );
}
}
void CG_BracketEntity( centity_t *cent, float radius )
{
trace_t tr;
vec3_t dif;
float len, size, lineLength, lineWidth;
float x, y;
clientInfo_t *local;
qboolean isEnemy = qfalse;
VectorSubtract( cent->lerpOrigin, cg.refdef.vieworg, dif );
len = VectorNormalize( dif );
if ( cg.crosshairClientNum != cent->currentState.clientNum
&& (!cg.snap||cg.snap->ps.rocketLockIndex!= cent->currentState.clientNum) )
{//if they're the entity you're locking onto or under your crosshair, always draw bracket
//Hmm... for now, if they're closer than 2000, don't bracket?
if ( len < 2000.0f )
{
return;
}
CG_Trace( &tr, cg.refdef.vieworg, NULL, NULL, cent->lerpOrigin, -1, CONTENTS_OPAQUE );
//don't bracket if can't see them
if ( tr.fraction < 1.0f )
{
return;
}
}
if ( !CG_WorldCoordToScreenCoordFloat(cent->lerpOrigin, &x, &y) )
{//off-screen, don't draw it
return;
}
//just to see if it's centered
//CG_DrawPic( x-2, y-2, 4, 4, cgs.media.whiteShader );
local = &cgs.clientinfo[cg.snap->ps.clientNum];
if ( cent->currentState.m_iVehicleNum //vehicle has a driver
&& (cent->currentState.m_iVehicleNum-1) < MAX_CLIENTS
&& cgs.clientinfo[ cent->currentState.m_iVehicleNum-1 ].infoValid )
{
if ( cgs.gametype < GT_TEAM )
{//ffa?
isEnemy = qtrue;
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
}
else if ( cgs.clientinfo[ cent->currentState.m_iVehicleNum-1 ].team == local->team )
{
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_GREEN)] );
}
else
{
isEnemy = qtrue;
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
}
}
else if ( cent->currentState.teamowner )
{
if ( cgs.gametype < GT_TEAM )
{//ffa?
isEnemy = qtrue;
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
}
else if ( cent->currentState.teamowner != cg.predictedPlayerState.persistant[PERS_TEAM] )
{// on enemy team
isEnemy = qtrue;
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
}
else
{ //a friend
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_GREEN)] );
}
}
else
{//FIXME: if we want to ever bracket anything besides vehicles (like siege objectives we want to blow up), we should handle the coloring here
trap->R_SetColor ( NULL );
}
if ( len <= 1.0f )
{//super-close, max out at 400 times radius (which is HUGE)
size = radius*400.0f;
}
else
{//scale by dist
size = radius*(400.0f/len);
}
if ( size < 1.0f )
{
size = 1.0f;
}
//length scales with dist
lineLength = (size*0.1f);
if ( lineLength < 0.5f )
{//always visible
lineLength = 0.5f;
}
//always visible width
lineWidth = 1.0f;
x -= (size*0.5f);
y -= (size*0.5f);
/*
if ( x >= 0 && x <= 640
&& y >= 0 && y <= 480 )
*/
{//brackets would be drawn on the screen, so draw them
//upper left corner
//horz
CG_DrawPic( x, y, lineLength, lineWidth, cgs.media.whiteShader );
//vert
CG_DrawPic( x, y, lineWidth, lineLength, cgs.media.whiteShader );
//upper right corner
//horz
CG_DrawPic( x+size-lineLength, y, lineLength, lineWidth, cgs.media.whiteShader );
//vert
CG_DrawPic( x+size-lineWidth, y, lineWidth, lineLength, cgs.media.whiteShader );
//lower left corner
//horz
CG_DrawPic( x, y+size-lineWidth, lineLength, lineWidth, cgs.media.whiteShader );
//vert
CG_DrawPic( x, y+size-lineLength, lineWidth, lineLength, cgs.media.whiteShader );
//lower right corner
//horz
CG_DrawPic( x+size-lineLength, y+size-lineWidth, lineLength, lineWidth, cgs.media.whiteShader );
//vert
CG_DrawPic( x+size-lineWidth, y+size-lineLength, lineWidth, lineLength, cgs.media.whiteShader );
}
//Lead Indicator...
if ( cg_drawVehLeadIndicator.integer )
{//draw the lead indicator
if ( isEnemy )
{//an enemy object
if ( cent->currentState.NPC_class == CLASS_VEHICLE )
{//enemy vehicle
if ( !VectorCompare( cent->currentState.pos.trDelta, vec3_origin ) )
{//enemy vehicle is moving
if ( cg.predictedPlayerState.m_iVehicleNum )
{//I'm in a vehicle
centity_t *veh = &cg_entities[cg.predictedPlayerState.m_iVehicleNum];
if ( veh //vehicle cent
&& veh->m_pVehicle//vehicle
&& veh->m_pVehicle->m_pVehicleInfo//vehicle stats
&& veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID > VEH_WEAPON_BASE )//valid vehicle weapon
{
vehWeaponInfo_t *vehWeapon = &g_vehWeaponInfo[veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID];
if ( vehWeapon
&& vehWeapon->bIsProjectile//primary weapon's shot is a projectile
&& !vehWeapon->bHasGravity//primary weapon's shot is not affected by gravity
&& !vehWeapon->fHoming//primary weapon's shot is not homing
&& vehWeapon->fSpeed )//primary weapon's shot has speed
{//our primary weapon's projectile has a speed
vec3_t vehDiff, vehLeadPos;
float vehDist, eta;
float leadX, leadY;
VectorSubtract( cent->lerpOrigin, cg.predictedVehicleState.origin, vehDiff );
vehDist = VectorNormalize( vehDiff );
eta = (vehDist/vehWeapon->fSpeed);//how many seconds it would take for my primary weapon's projectile to get from my ship to theirs
//now extrapolate their position that number of seconds into the future based on their velocity
VectorMA( cent->lerpOrigin, eta, cent->currentState.pos.trDelta, vehLeadPos );
//now we have where we should be aiming at, project that onto the screen at a 2D co-ord
if ( !CG_WorldCoordToScreenCoordFloat(cent->lerpOrigin, &x, &y) )
{//off-screen, don't draw it
return;
}
if ( !CG_WorldCoordToScreenCoordFloat(vehLeadPos, &leadX, &leadY) )
{//off-screen, don't draw it
//just draw the line
CG_DottedLine( x, y, leadX, leadY, 1, 10, g_color_table[ColorIndex(COLOR_RED)], 0.5f );
return;
}
//draw a line from the ship's cur pos to the lead pos
CG_DottedLine( x, y, leadX, leadY, 1, 10, g_color_table[ColorIndex(COLOR_RED)], 0.5f );
//now draw the lead indicator
trap->R_SetColor ( g_color_table[ColorIndex(COLOR_RED)] );
CG_DrawPic( leadX-8, leadY-8, 16, 16, trap->R_RegisterShader( "gfx/menus/radar/lead" ) );
}
}
}
}
}
}
}
}
qboolean CG_InFighter( void )
{
if ( cg.predictedPlayerState.m_iVehicleNum )
{//I'm in a vehicle
centity_t *vehCent = &cg_entities[cg.predictedPlayerState.m_iVehicleNum];
if ( vehCent
&& vehCent->m_pVehicle
&& vehCent->m_pVehicle->m_pVehicleInfo
&& vehCent->m_pVehicle->m_pVehicleInfo->type == VH_FIGHTER )
{//I'm in a fighter
return qtrue;
}
}
return qfalse;
}
qboolean CG_InATST( void )
{
if ( cg.predictedPlayerState.m_iVehicleNum )
{//I'm in a vehicle
centity_t *vehCent = &cg_entities[cg.predictedPlayerState.m_iVehicleNum];
if ( vehCent
&& vehCent->m_pVehicle
&& vehCent->m_pVehicle->m_pVehicleInfo
&& vehCent->m_pVehicle->m_pVehicleInfo->type == VH_WALKER )
{//I'm in an atst
return qtrue;
}
}
return qfalse;
}
void CG_DrawBracketedEntities( void )
{
int i;
for ( i = 0; i < cg.bracketedEntityCount; i++ )
{
centity_t *cent = &cg_entities[cg.bracketedEntities[i]];
CG_BracketEntity( cent, CG_RadiusForCent( cent ) );
}
}
//--------------------------------------------------------------
static void CG_DrawHolocronIcons(void)
//--------------------------------------------------------------
{
int icon_size = 40;
int i = 0;
int startx = 10;
int starty = 10;//SCREEN_HEIGHT - icon_size*3;
int endx = icon_size;
int endy = icon_size;
if (cg.snap->ps.zoomMode)
{ //don't display over zoom mask
return;
}
if (cgs.clientinfo[cg.snap->ps.clientNum].team == TEAM_SPECTATOR)
{
return;
}
while (i < NUM_FORCE_POWERS)
{
if (cg.snap->ps.holocronBits & (1 << forcePowerSorted[i]))
{
CG_DrawPic( startx, starty, endx, endy, cgs.media.forcePowerIcons[forcePowerSorted[i]]);
starty += (icon_size+2); //+2 for spacing
if ((starty+icon_size) >= SCREEN_HEIGHT-80)
{
starty = 10;//SCREEN_HEIGHT - icon_size*3;
startx += (icon_size+2);
}
}
i++;
}
}
static qboolean CG_IsDurationPower(int power)
{
if (power == FP_HEAL ||
power == FP_SPEED ||
power == FP_TELEPATHY ||
power == FP_RAGE ||
power == FP_PROTECT ||
power == FP_ABSORB ||
power == FP_SEE)
{
return qtrue;
}
return qfalse;
}
//--------------------------------------------------------------
static void CG_DrawActivePowers(void)
//--------------------------------------------------------------
{
int icon_size = 40;
int i = 0;
int startx = icon_size*2+16;
int starty = SCREEN_HEIGHT - icon_size*2;
int endx = icon_size;
int endy = icon_size;
if (cg.snap->ps.zoomMode)
{ //don't display over zoom mask
return;
}
if (cgs.clientinfo[cg.snap->ps.clientNum].team == TEAM_SPECTATOR)
{
return;
}
trap->R_SetColor( NULL );
while (i < NUM_FORCE_POWERS)
{
if ((cg.snap->ps.fd.forcePowersActive & (1 << forcePowerSorted[i])) &&
CG_IsDurationPower(forcePowerSorted[i]))
{
CG_DrawPic( startx, starty, endx, endy, cgs.media.forcePowerIcons[forcePowerSorted[i]]);
startx += (icon_size+2); //+2 for spacing
if ((startx+icon_size) >= SCREEN_WIDTH-80)
{
startx = icon_size*2+16;
starty += (icon_size+2);
}
}
i++;
}
//additionally, draw an icon force force rage recovery
if (cg.snap->ps.fd.forceRageRecoveryTime > cg.time)
{
CG_DrawPic( startx, starty, endx, endy, cgs.media.rageRecShader);
}
}
//--------------------------------------------------------------
static void CG_DrawRocketLocking( int lockEntNum, int lockTime )
//--------------------------------------------------------------
{
int cx, cy;
vec3_t org;
static int oldDif = 0;
centity_t *cent = &cg_entities[lockEntNum];
vec4_t color={0.0f,0.0f,0.0f,0.0f};
float lockTimeInterval = ((cgs.gametype==GT_SIEGE)?2400.0f:1200.0f)/16.0f;
//FIXME: if in a vehicle, use the vehicle's lockOnTime...
int dif = (cg.time - cg.snap->ps.rocketLockTime)/lockTimeInterval;
int i;
if (!cg.snap->ps.rocketLockTime)
{
return;
}
if (cgs.clientinfo[cg.snap->ps.clientNum].team == TEAM_SPECTATOR)
{
return;
}
if ( cg.snap->ps.m_iVehicleNum )
{//driving a vehicle
centity_t *veh = &cg_entities[cg.snap->ps.m_iVehicleNum];
if ( veh->m_pVehicle )
{
vehWeaponInfo_t *vehWeapon = NULL;
if ( cg.predictedVehicleState.weaponstate == WEAPON_CHARGING_ALT )
{
if ( veh->m_pVehicle->m_pVehicleInfo->weapon[1].ID > VEH_WEAPON_BASE
&& veh->m_pVehicle->m_pVehicleInfo->weapon[1].ID < MAX_VEH_WEAPONS )
{
vehWeapon = &g_vehWeaponInfo[veh->m_pVehicle->m_pVehicleInfo->weapon[1].ID];
}
}
else
{
if ( veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID > VEH_WEAPON_BASE
&& veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID < MAX_VEH_WEAPONS )
{
vehWeapon = &g_vehWeaponInfo[veh->m_pVehicle->m_pVehicleInfo->weapon[0].ID];
}
}
if ( vehWeapon != NULL )
{//we are trying to lock on with a valid vehicle weapon, so use *its* locktime, not the hard-coded one
if ( !vehWeapon->iLockOnTime )
{//instant lock-on
dif = 10.0f;
}
else
{//use the custom vehicle lockOnTime
lockTimeInterval = (vehWeapon->iLockOnTime/16.0f);
dif = (cg.time - cg.snap->ps.rocketLockTime)/lockTimeInterval;
}
}
}
}
//We can't check to see in pmove if players are on the same team, so we resort
//to just not drawing the lock if a teammate is the locked on ent
if (cg.snap->ps.rocketLockIndex >= 0 &&
cg.snap->ps.rocketLockIndex < ENTITYNUM_NONE)
{
clientInfo_t *ci = NULL;
if (cg.snap->ps.rocketLockIndex < MAX_CLIENTS)
{
ci = &cgs.clientinfo[cg.snap->ps.rocketLockIndex];
}
else
{
ci = cg_entities[cg.snap->ps.rocketLockIndex].npcClient;
}
if (ci)
{
if (ci->team == cgs.clientinfo[cg.snap->ps.clientNum].team)
{
if (cgs.gametype >= GT_TEAM)
{
return;
}
}
else if (cgs.gametype >= GT_TEAM)
{
centity_t *hitEnt = &cg_entities[cg.snap->ps.rocketLockIndex];
if (hitEnt->currentState.eType == ET_NPC &&
hitEnt->currentState.NPC_class == CLASS_VEHICLE &&
hitEnt->currentState.owner < ENTITYNUM_WORLD)
{ //this is a vehicle, if it has a pilot and that pilot is on my team, then...
if (hitEnt->currentState.owner < MAX_CLIENTS)
{
ci = &cgs.clientinfo[hitEnt->currentState.owner];
}
else
{
ci = cg_entities[hitEnt->currentState.owner].npcClient;
}
if (ci && ci->team == cgs.clientinfo[cg.snap->ps.clientNum].team)
{
return;
}
}
}
}
}
if (cg.snap->ps.rocketLockTime != -1)
{
lastvalidlockdif = dif;
}
else
{
dif = lastvalidlockdif;
}
if ( !cent )
{
return;
}
VectorCopy( cent->lerpOrigin, org );
if ( CG_WorldCoordToScreenCoord( org, &cx, &cy ))
{
// we care about distance from enemy to eye, so this is good enough
float sz = Distance( cent->lerpOrigin, cg.refdef.vieworg ) / 1024.0f;
if ( sz > 1.0f )
{
sz = 1.0f;
}
else if ( sz < 0.0f )
{
sz = 0.0f;
}
sz = (1.0f - sz) * (1.0f - sz) * 32 + 6;
cy += sz * 0.5f;
if ( dif < 0 )
{
oldDif = 0;
return;
}
else if ( dif > 8 )
{
dif = 8;
}
// do sounds
if ( oldDif != dif )
{
if ( dif == 8 )
{
if ( cg.snap->ps.m_iVehicleNum )
{
trap->S_StartSound( org, 0, CHAN_AUTO, trap->S_RegisterSound( "sound/vehicles/weapons/common/lock.wav" ));
}
else
{
trap->S_StartSound( org, 0, CHAN_AUTO, trap->S_RegisterSound( "sound/weapons/rocket/lock.wav" ));
}
}
else
{
if ( cg.snap->ps.m_iVehicleNum )
{
trap->S_StartSound( org, 0, CHAN_AUTO, trap->S_RegisterSound( "sound/vehicles/weapons/common/tick.wav" ));
}
else
{
trap->S_StartSound( org, 0, CHAN_AUTO, trap->S_RegisterSound( "sound/weapons/rocket/tick.wav" ));
}
}
}
oldDif = dif;
for ( i = 0; i < dif; i++ )
{
color[0] = 1.0f;
color[1] = 0.0f;
color[2] = 0.0f;
color[3] = 0.1f * i + 0.2f;
trap->R_SetColor( color );
// our slices are offset by about 45 degrees.
CG_DrawRotatePic( cx - sz, cy - sz, sz, sz, i * 45.0f, trap->R_RegisterShaderNoMip( "gfx/2d/wedge" ));
}
// we are locked and loaded baby
if ( dif == 8 )
{
color[0] = color[1] = color[2] = sin( cg.time * 0.05f ) * 0.5f + 0.5f;
color[3] = 1.0f; // this art is additive, so the alpha value does nothing
trap->R_SetColor( color );
CG_DrawPic( cx - sz, cy - sz * 2, sz * 2, sz * 2, trap->R_RegisterShaderNoMip( "gfx/2d/lock" ));
}
}
}
extern void CG_CalcVehMuzzle(Vehicle_t *pVeh, centity_t *ent, int muzzleNum);
qboolean CG_CalcVehicleMuzzlePoint( int entityNum, vec3_t start, vec3_t d_f, vec3_t d_rt, vec3_t d_up)
{
centity_t *vehCent = &cg_entities[entityNum];
if ( vehCent->m_pVehicle && vehCent->m_pVehicle->m_pVehicleInfo->type == VH_WALKER )
{//draw from barrels
VectorCopy( vehCent->lerpOrigin, start );
start[2] += vehCent->m_pVehicle->m_pVehicleInfo->height-DEFAULT_MINS_2-48;
AngleVectors( vehCent->lerpAngles, d_f, d_rt, d_up );
/*
mdxaBone_t boltMatrix;
int bolt;
vec3_t yawOnlyAngles;
VectorSet( yawOnlyAngles, 0, vehCent->lerpAngles[YAW], 0 );
bolt = trap->G2API_AddBolt( vehCent->ghoul2, 0, "*flash1");
trap->G2API_GetBoltMatrix( vehCent->ghoul2, 0, bolt, &boltMatrix,
yawOnlyAngles, vehCent->lerpOrigin, cg.time,
NULL, vehCent->modelScale );
// work the matrix axis stuff into the original axis and origins used.
BG_GiveMeVectorFromMatrix( &boltMatrix, ORIGIN, start );
BG_GiveMeVectorFromMatrix( &boltMatrix, POSITIVE_X, d_f );
VectorClear( d_rt );//don't really need this, do we?
VectorClear( d_up );//don't really need this, do we?
*/
}
else
{
//check to see if we're a turret gunner on this vehicle
if ( cg.predictedPlayerState.generic1 )//as a passenger
{//passenger in a vehicle
if ( vehCent->m_pVehicle
&& vehCent->m_pVehicle->m_pVehicleInfo
&& vehCent->m_pVehicle->m_pVehicleInfo->maxPassengers )
{//a vehicle capable of carrying passengers
int turretNum;
for ( turretNum = 0; turretNum < MAX_VEHICLE_TURRETS; turretNum++ )
{
if ( vehCent->m_pVehicle->m_pVehicleInfo->turret[turretNum].iAmmoMax )
{// valid turret
if ( vehCent->m_pVehicle->m_pVehicleInfo->turret[turretNum].passengerNum == cg.predictedPlayerState.generic1 )
{//I control this turret
//Go through all muzzles, average their positions and directions and use the result for crosshair trace
int vehMuzzle, numMuzzles = 0;
vec3_t muzzlesAvgPos={0},muzzlesAvgDir={0};
int i;
for ( i = 0; i < MAX_VEHICLE_TURRET_MUZZLES; i++ )
{
vehMuzzle = vehCent->m_pVehicle->m_pVehicleInfo->turret[turretNum].iMuzzle[i];
if ( vehMuzzle )
{
vehMuzzle -= 1;
CG_CalcVehMuzzle( vehCent->m_pVehicle, vehCent, vehMuzzle );
VectorAdd( muzzlesAvgPos, vehCent->m_pVehicle->m_vMuzzlePos[vehMuzzle], muzzlesAvgPos );
VectorAdd( muzzlesAvgDir, vehCent->m_pVehicle->m_vMuzzleDir[vehMuzzle], muzzlesAvgDir );
numMuzzles++;
}
if ( numMuzzles )
{
VectorScale( muzzlesAvgPos, 1.0f/(float)numMuzzles, start );
VectorScale( muzzlesAvgDir, 1.0f/(float)numMuzzles, d_f );
VectorClear( d_rt );
VectorClear( d_up );
return qtrue;
}
}
}
}
}
}
}
VectorCopy( vehCent->lerpOrigin, start );
AngleVectors( vehCent->lerpAngles, d_f, d_rt, d_up );
}
return qfalse;
}
//calc the muzzle point from the e-web itself
void CG_CalcEWebMuzzlePoint(centity_t *cent, vec3_t start, vec3_t d_f, vec3_t d_rt, vec3_t d_up)
{
int bolt = trap->G2API_AddBolt(cent->ghoul2, 0, "*cannonflash");
assert(bolt != -1);
if (bolt != -1)
{
mdxaBone_t boltMatrix;
trap->G2API_GetBoltMatrix_NoRecNoRot(cent->ghoul2, 0, bolt, &boltMatrix, cent->lerpAngles, cent->lerpOrigin, cg.time, NULL, cent->modelScale);
BG_GiveMeVectorFromMatrix(&boltMatrix, ORIGIN, start);
BG_GiveMeVectorFromMatrix(&boltMatrix, NEGATIVE_X, d_f);
//these things start the shot a little inside the bbox to assure not starting in something solid
VectorMA(start, -16.0f, d_f, start);
//I guess
VectorClear( d_rt );//don't really need this, do we?
VectorClear( d_up );//don't really need this, do we?
}
}
/*
=================
CG_`Entity
=================
*/
#define MAX_XHAIR_DIST_ACCURACY 20000.0f
static void CG_ScanForCrosshairEntity( void ) {
trace_t trace;
vec3_t start, end;
int content;
int ignore;
qboolean bVehCheckTraceFromCamPos = qfalse;
ignore = cg.predictedPlayerState.clientNum;
if ( cg_dynamicCrosshair.integer )
{
vec3_t d_f, d_rt, d_up;
/*
if ( cg.snap->ps.weapon == WP_NONE ||
cg.snap->ps.weapon == WP_SABER ||
cg.snap->ps.weapon == WP_STUN_BATON)
{
VectorCopy( cg.refdef.vieworg, start );
AngleVectors( cg.refdef.viewangles, d_f, d_rt, d_up );
}
else
*/
//For now we still want to draw the crosshair in relation to the player's world coordinates
//even if we have a melee weapon/no weapon.
if ( cg.predictedPlayerState.m_iVehicleNum && (cg.predictedPlayerState.eFlags&EF_NODRAW) )
{//we're *inside* a vehicle
//do the vehicle's crosshair instead
centity_t *veh = &cg_entities[cg.predictedPlayerState.m_iVehicleNum];
qboolean gunner = qfalse;
//if (veh->currentState.owner == cg.predictedPlayerState.clientNum)
{ //the pilot
ignore = cg.predictedPlayerState.m_iVehicleNum;
gunner = CG_CalcVehicleMuzzlePoint(cg.predictedPlayerState.m_iVehicleNum, start, d_f, d_rt, d_up);
}
/*
else
{ //a passenger
ignore = cg.predictedPlayerState.m_iVehicleNum;
VectorCopy( veh->lerpOrigin, start );
AngleVectors( veh->lerpAngles, d_f, d_rt, d_up );
VectorMA(start, 32.0f, d_f, start); //super hack
}
*/
if ( veh->m_pVehicle
&& veh->m_pVehicle->m_pVehicleInfo
&& veh->m_pVehicle->m_pVehicleInfo->type == VH_FIGHTER
&& cg.distanceCull > MAX_XHAIR_DIST_ACCURACY
&& !gunner )
{
//NOTE: on huge maps, the crosshair gets inaccurate at close range,
// so we'll do an extra G2 trace from the cg.refdef.vieworg
// to see if we hit anything closer and auto-aim at it if so
bVehCheckTraceFromCamPos = qtrue;
}
}
else if (cg.snap && cg.snap->ps.weapon == WP_EMPLACED_GUN && cg.snap->ps.emplacedIndex &&
cg_entities[cg.snap->ps.emplacedIndex].ghoul2 && cg_entities[cg.snap->ps.emplacedIndex].currentState.weapon == WP_NONE)
{ //locked into our e-web, calc the muzzle from it
CG_CalcEWebMuzzlePoint(&cg_entities[cg.snap->ps.emplacedIndex], start, d_f, d_rt, d_up);
}
else
{
if (cg.snap && cg.snap->ps.weapon == WP_EMPLACED_GUN && cg.snap->ps.emplacedIndex)
{
vec3_t pitchConstraint;
ignore = cg.snap->ps.emplacedIndex;
VectorCopy(cg.refdef.viewangles, pitchConstraint);
if (cg.renderingThirdPerson)
{
VectorCopy(cg.predictedPlayerState.viewangles, pitchConstraint);
}
else
{
VectorCopy(cg.refdef.viewangles, pitchConstraint);
}
if (pitchConstraint[PITCH] > 40)
{
pitchConstraint[PITCH] = 40;
}
AngleVectors( pitchConstraint, d_f, d_rt, d_up );
}
else
{
vec3_t pitchConstraint;
if (cg.renderingThirdPerson)
{
VectorCopy(cg.predictedPlayerState.viewangles, pitchConstraint);
}
else
{
VectorCopy(cg.refdef.viewangles, pitchConstraint);
}
AngleVectors( pitchConstraint, d_f, d_rt, d_up );
}
CG_CalcMuzzlePoint(cg.snap->ps.clientNum, start);
}
VectorMA( start, cg.distanceCull, d_f, end );
}
else
{
VectorCopy( cg.refdef.vieworg, start );
VectorMA( start, 131072, cg.refdef.viewaxis[0], end );
}
if ( cg_dynamicCrosshair.integer && cg_dynamicCrosshairPrecision.integer )
{ //then do a trace with ghoul2 models in mind
CG_G2Trace( &trace, start, vec3_origin, vec3_origin, end,
ignore, CONTENTS_SOLID|CONTENTS_BODY );
if ( bVehCheckTraceFromCamPos )
{
//NOTE: this MUST stay up to date with the method used in WP_VehCheckTraceFromCamPos
centity_t *veh = &cg_entities[cg.predictedPlayerState.m_iVehicleNum];
trace_t extraTrace;
vec3_t viewDir2End, extraEnd;
float minAutoAimDist = Distance( veh->lerpOrigin, cg.refdef.vieworg ) + (veh->m_pVehicle->m_pVehicleInfo->length/2.0f) + 200.0f;
VectorSubtract( end, cg.refdef.vieworg, viewDir2End );
VectorNormalize( viewDir2End );
VectorMA( cg.refdef.vieworg, MAX_XHAIR_DIST_ACCURACY, viewDir2End, extraEnd );
CG_G2Trace( &extraTrace, cg.refdef.vieworg, vec3_origin, vec3_origin, extraEnd,
ignore, CONTENTS_SOLID|CONTENTS_BODY );
if ( !extraTrace.allsolid
&& !extraTrace.startsolid )
{
if ( extraTrace.fraction < 1.0f )
{
if ( (extraTrace.fraction*MAX_XHAIR_DIST_ACCURACY) > minAutoAimDist )
{
if ( ((extraTrace.fraction*MAX_XHAIR_DIST_ACCURACY)-Distance( veh->lerpOrigin, cg.refdef.vieworg )) < (trace.fraction*cg.distanceCull) )
{//this trace hit *something* that's closer than the thing the main trace hit, so use this result instead
memcpy( &trace, &extraTrace, sizeof( trace_t ) );
}
}
}
}
}
}
else
{
CG_Trace( &trace, start, vec3_origin, vec3_origin, end,
ignore, CONTENTS_SOLID|CONTENTS_BODY );
}
if (trace.entityNum < MAX_CLIENTS)
{
if (CG_IsMindTricked(cg_entities[trace.entityNum].currentState.trickedentindex,
cg_entities[trace.entityNum].currentState.trickedentindex2,
cg_entities[trace.entityNum].currentState.trickedentindex3,
cg_entities[trace.entityNum].currentState.trickedentindex4,
cg.snap->ps.clientNum))
{
if (cg.crosshairClientNum == trace.entityNum)
{
cg.crosshairClientNum = ENTITYNUM_NONE;
cg.crosshairClientTime = 0;
}
CG_DrawCrosshair(trace.endpos, 0);
return; //this entity is mind-tricking the current client, so don't render it
}
}
if (cg.snap->ps.persistant[PERS_TEAM] != TEAM_SPECTATOR)
{
if (trace.entityNum < /*MAX_CLIENTS*/ENTITYNUM_WORLD)
{
centity_t *veh = &cg_entities[trace.entityNum];
cg.crosshairClientNum = trace.entityNum;
cg.crosshairClientTime = cg.time;
if (veh->currentState.eType == ET_NPC &&
veh->currentState.NPC_class == CLASS_VEHICLE &&
veh->currentState.owner < MAX_CLIENTS)
{ //draw the name of the pilot then
cg.crosshairClientNum = veh->currentState.owner;
cg.crosshairVehNum = veh->currentState.number;
cg.crosshairVehTime = cg.time;
}
CG_DrawCrosshair(trace.endpos, 1);
}
else
{
CG_DrawCrosshair(trace.endpos, 0);
}
}
if ( trace.entityNum >= MAX_CLIENTS ) {
return;
}
// if the player is in fog, don't show it
content = CG_PointContents( trace.endpos, 0 );
if ( content & CONTENTS_FOG ) {
return;
}
// update the fade timer
cg.crosshairClientNum = trace.entityNum;
cg.crosshairClientTime = cg.time;
}
/*
=====================
CG_DrawCrosshairNames
=====================
*/
static void CG_DrawCrosshairNames( void ) {
float *color;
vec4_t tcolor;
char *name;
int baseColor;
qboolean isVeh = qfalse;
if ( !cg_drawCrosshair.integer ) {
return;
}
// scan the known entities to see if the crosshair is sighted on one
CG_ScanForCrosshairEntity();
if ( !cg_drawCrosshairNames.integer ) {
return;
}
//rww - still do the trace, our dynamic crosshair depends on it
if (cg.crosshairClientNum < ENTITYNUM_WORLD)
{
centity_t *veh = &cg_entities[cg.crosshairClientNum];
if (veh->currentState.eType == ET_NPC &&
veh->currentState.NPC_class == CLASS_VEHICLE &&
veh->currentState.owner < MAX_CLIENTS)
{ //draw the name of the pilot then
cg.crosshairClientNum = veh->currentState.owner;
cg.crosshairVehNum = veh->currentState.number;
cg.crosshairVehTime = cg.time;
isVeh = qtrue; //so we know we're drawing the pilot's name
}
}
if (cg.crosshairClientNum >= MAX_CLIENTS)
{
return;
}
if (cg_entities[cg.crosshairClientNum].currentState.powerups & (1 << PW_CLOAKED))
{
return;
}
// draw the name of the player being looked at
color = CG_FadeColor( cg.crosshairClientTime, 1000 );
if ( !color ) {
trap->R_SetColor( NULL );
return;
}
name = cgs.clientinfo[ cg.crosshairClientNum ].cleanname;
if (cgs.gametype >= GT_TEAM)
{
//if (cgs.gametype == GT_SIEGE)
if (1)
{ //instead of team-based we'll make it oriented based on which team we're on
if (cgs.clientinfo[cg.crosshairClientNum].team == cg.predictedPlayerState.persistant[PERS_TEAM])
{
baseColor = CT_GREEN;
}
else
{
baseColor = CT_RED;
}
}
else
{
if (cgs.clientinfo[cg.crosshairClientNum].team == TEAM_RED)
{
baseColor = CT_RED;
}
else
{
baseColor = CT_BLUE;
}
}
}
else
{
//baseColor = CT_WHITE;
if (cgs.gametype == GT_POWERDUEL &&
cgs.clientinfo[cg.snap->ps.clientNum].team != TEAM_SPECTATOR &&
cgs.clientinfo[cg.crosshairClientNum].duelTeam == cgs.clientinfo[cg.predictedPlayerState.clientNum].duelTeam)
{ //on the same duel team in powerduel, so he's a friend
baseColor = CT_GREEN;
}
else
{
baseColor = CT_RED; //just make it red in nonteam modes since everyone is hostile and crosshair will be red on them too
}
}
if (cg.snap->ps.duelInProgress)
{
if (cg.crosshairClientNum != cg.snap->ps.duelIndex)
{ //grey out crosshair for everyone but your foe if you're in a duel
baseColor = CT_BLACK;
}
}
else if (cg_entities[cg.crosshairClientNum].currentState.bolt1)
{ //this fellow is in a duel. We just checked if we were in a duel above, so
//this means we aren't and he is. Which of course means our crosshair greys out over him.
baseColor = CT_BLACK;
}
tcolor[0] = colorTable[baseColor][0];
tcolor[1] = colorTable[baseColor][1];
tcolor[2] = colorTable[baseColor][2];
tcolor[3] = color[3]*0.5f;
if (isVeh)
{
char str[MAX_STRING_CHARS];
Com_sprintf(str, MAX_STRING_CHARS, "%s (pilot)", name);
CG_DrawProportionalString(320, 170, str, UI_CENTER, tcolor);
}
else
{
CG_DrawProportionalString(320, 170, name, UI_CENTER, tcolor);
}
trap->R_SetColor( NULL );
}
//==============================================================================
/*
=================
CG_DrawSpectator
=================
*/
static void CG_DrawSpectator(void)
{
const char* s;
s = CG_GetStringEdString("MP_INGAME", "SPECTATOR");
if ((cgs.gametype == GT_DUEL || cgs.gametype == GT_POWERDUEL) &&
cgs.duelist1 != -1 &&
cgs.duelist2 != -1)
{
char text[1024];
int size = 64;
if (cgs.gametype == GT_POWERDUEL && cgs.duelist3 != -1)
{
Com_sprintf(text, sizeof(text), "%s^7 %s %s^7 %s %s", cgs.clientinfo[cgs.duelist1].name, CG_GetStringEdString("MP_INGAME", "SPECHUD_VERSUS"), cgs.clientinfo[cgs.duelist2].name, CG_GetStringEdString("MP_INGAME", "AND"), cgs.clientinfo[cgs.duelist3].name);
}
else
{
Com_sprintf(text, sizeof(text), "%s^7 %s %s", cgs.clientinfo[cgs.duelist1].name, CG_GetStringEdString("MP_INGAME", "SPECHUD_VERSUS"), cgs.clientinfo[cgs.duelist2].name);
}
CG_Text_Paint ( 320 - CG_Text_Width ( text, 1.0f, 3 ) / 2, 420, 1.0f, colorWhite, text, 0, 0, 0, 3 );
trap->R_SetColor( colorTable[CT_WHITE] );
if ( cgs.clientinfo[cgs.duelist1].modelIcon )
{
CG_DrawPic( 10, SCREEN_HEIGHT-(size*1.5), size, size, cgs.clientinfo[cgs.duelist1].modelIcon );
}
if ( cgs.clientinfo[cgs.duelist2].modelIcon )
{
CG_DrawPic( SCREEN_WIDTH-size-10, SCREEN_HEIGHT-(size*1.5), size, size, cgs.clientinfo[cgs.duelist2].modelIcon );
}
// nmckenzie: DUEL_HEALTH
if (cgs.gametype == GT_DUEL)
{
if ( cgs.showDuelHealths >= 1)
{ // draw the healths on the two guys - how does this interact with power duel, though?
CG_DrawDuelistHealth ( 10, SCREEN_HEIGHT-(size*1.5) - 12, 64, 8, 1 );
CG_DrawDuelistHealth ( SCREEN_WIDTH-size-10, SCREEN_HEIGHT-(size*1.5) - 12, 64, 8, 2 );
}
}
if (cgs.gametype != GT_POWERDUEL)
{
Com_sprintf(text, sizeof(text), "%i/%i", cgs.clientinfo[cgs.duelist1].score, cgs.fraglimit );
CG_Text_Paint( 42 - CG_Text_Width( text, 1.0f, 2 ) / 2, SCREEN_HEIGHT-(size*1.5) + 64, 1.0f, colorWhite, text, 0, 0, 0, 2 );
Com_sprintf(text, sizeof(text), "%i/%i", cgs.clientinfo[cgs.duelist2].score, cgs.fraglimit );
CG_Text_Paint( SCREEN_WIDTH-size+22 - CG_Text_Width( text, 1.0f, 2 ) / 2, SCREEN_HEIGHT-(size*1.5) + 64, 1.0f, colorWhite, text, 0, 0, 0, 2 );
}
if (cgs.gametype == GT_POWERDUEL && cgs.duelist3 != -1)
{
if ( cgs.clientinfo[cgs.duelist3].modelIcon )
{
CG_DrawPic( SCREEN_WIDTH-size-10, SCREEN_HEIGHT-(size*2.8), size, size, cgs.clientinfo[cgs.duelist3].modelIcon );
}
}
}
else
{
CG_Text_Paint ( 320 - CG_Text_Width ( s, 1.0f, 3 ) / 2, 420, 1.0f, colorWhite, s, 0, 0, 0, 3 );
}
if ( cgs.gametype == GT_DUEL || cgs.gametype == GT_POWERDUEL )
{
s = CG_GetStringEdString("MP_INGAME", "WAITING_TO_PLAY"); // "waiting to play";
CG_Text_Paint ( 320 - CG_Text_Width ( s, 1.0f, 3 ) / 2, 440, 1.0f, colorWhite, s, 0, 0, 0, 3 );
}
else //if ( cgs.gametype >= GT_TEAM )
{
//s = "press ESC and use the JOIN menu to play";
s = CG_GetStringEdString("MP_INGAME", "SPEC_CHOOSEJOIN");
CG_Text_Paint ( 320 - CG_Text_Width ( s, 1.0f, 3 ) / 2, 440, 1.0f, colorWhite, s, 0, 0, 0, 3 );
}
}
/*
=================
CG_DrawVote
=================
*/
static void CG_DrawVote(void) {
const char *s = NULL, *sParm = NULL;
int sec;
char sYes[20] = {0}, sNo[20] = {0}, sVote[20] = {0}, sCmd[100] = {0};
if ( !cgs.voteTime )
return;
// play a talk beep whenever it is modified
if ( cgs.voteModified ) {
cgs.voteModified = qfalse;
trap->S_StartLocalSound( cgs.media.talkSound, CHAN_LOCAL_SOUND );
}
sec = ( VOTE_TIME - ( cg.time - cgs.voteTime ) ) / 1000;
if ( sec < 0 ) {
sec = 0;
}
if ( !Q_strncmp( cgs.voteString, "map_restart", 11 ) )
trap->SE_GetStringTextString( "MENUS_RESTART_MAP", sCmd, sizeof( sCmd ) );
else if ( !Q_strncmp( cgs.voteString, "vstr nextmap", 12 ) )
trap->SE_GetStringTextString( "MENUS_NEXT_MAP", sCmd, sizeof( sCmd ) );
else if ( !Q_strncmp( cgs.voteString, "g_doWarmup", 10 ) )
trap->SE_GetStringTextString( "MENUS_WARMUP", sCmd, sizeof( sCmd ) );
else if ( !Q_strncmp( cgs.voteString, "g_gametype", 10 ) ) {
trap->SE_GetStringTextString( "MENUS_GAME_TYPE", sCmd, sizeof( sCmd ) );
if ( !Q_stricmp( "Free For All", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "FREE_FOR_ALL" );
else if ( !Q_stricmp( "Duel", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "DUEL" );
else if ( !Q_stricmp( "Holocron FFA", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "HOLOCRON_FFA" );
else if ( !Q_stricmp( "Power Duel", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "POWERDUEL" );
else if ( !Q_stricmp( "Team FFA", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "TEAM_FFA" );
else if ( !Q_stricmp( "Siege", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "SIEGE" );
else if ( !Q_stricmp( "Capture the Flag", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "CAPTURE_THE_FLAG" );
else if ( !Q_stricmp( "Capture the Ysalamiri", cgs.voteString+11 ) ) sParm = CG_GetStringEdString( "MENUS", "CAPTURE_THE_YSALIMARI" );
}
else if ( !Q_strncmp( cgs.voteString, "map", 3 ) ) {
trap->SE_GetStringTextString( "MENUS_NEW_MAP", sCmd, sizeof( sCmd ) );
sParm = cgs.voteString+4;
}
else if ( !Q_strncmp( cgs.voteString, "kick", 4 ) ) {
trap->SE_GetStringTextString( "MENUS_KICK_PLAYER", sCmd, sizeof( sCmd ) );
sParm = cgs.voteString+5;
}
else
{// custom votes like ampoll, cointoss, etc
sParm = cgs.voteString;
}
trap->SE_GetStringTextString( "MENUS_VOTE", sVote, sizeof( sVote ) );
trap->SE_GetStringTextString( "MENUS_YES", sYes, sizeof( sYes ) );
trap->SE_GetStringTextString( "MENUS_NO", sNo, sizeof( sNo ) );
if (sParm && sParm[0])
s = va( "%s(%i):<%s %s> %s:%i %s:%i", sVote, sec, sCmd, sParm, sYes, cgs.voteYes, sNo, cgs.voteNo);
else
s = va( "%s(%i):<%s> %s:%i %s:%i", sVote, sec, sCmd, sYes, cgs.voteYes, sNo, cgs.voteNo);
CG_DrawSmallString( 4, 58, s, 1.0F );
if ( cgs.clientinfo[cg.clientNum].team != TEAM_SPECTATOR ) {
s = CG_GetStringEdString( "MP_INGAME", "OR_PRESS_ESC_THEN_CLICK_VOTE" ); // s = "or press ESC then click Vote";
CG_DrawSmallString( 4, 58 + SMALLCHAR_HEIGHT + 2, s, 1.0F );
}
}
/*
=================
CG_DrawTeamVote
=================
*/
static void CG_DrawTeamVote(void) {
char *s;
int sec, cs_offset;
if ( cgs.clientinfo[cg.clientNum].team == TEAM_RED )
cs_offset = 0;
else if ( cgs.clientinfo[cg.clientNum].team == TEAM_BLUE )
cs_offset = 1;
else
return;
if ( !cgs.teamVoteTime[cs_offset] ) {
return;
}
// play a talk beep whenever it is modified
if ( cgs.teamVoteModified[cs_offset] ) {
cgs.teamVoteModified[cs_offset] = qfalse;
// trap->S_StartLocalSound( cgs.media.talkSound, CHAN_LOCAL_SOUND );
}
sec = ( VOTE_TIME - ( cg.time - cgs.teamVoteTime[cs_offset] ) ) / 1000;
if ( sec < 0 ) {
sec = 0;
}
if (strstr(cgs.teamVoteString[cs_offset], "leader"))
{
int i = 0;
while (cgs.teamVoteString[cs_offset][i] && cgs.teamVoteString[cs_offset][i] != ' ')
{
i++;
}
if (cgs.teamVoteString[cs_offset][i] == ' ')
{
int voteIndex = 0;
char voteIndexStr[256];
i++;
while (cgs.teamVoteString[cs_offset][i])
{
voteIndexStr[voteIndex] = cgs.teamVoteString[cs_offset][i];
voteIndex++;
i++;
}
voteIndexStr[voteIndex] = 0;
voteIndex = atoi(voteIndexStr);
s = va("TEAMVOTE(%i):(Make %s the new team leader) yes:%i no:%i", sec, cgs.clientinfo[voteIndex].name,
cgs.teamVoteYes[cs_offset], cgs.teamVoteNo[cs_offset] );
}
else
{
s = va("TEAMVOTE(%i):%s yes:%i no:%i", sec, cgs.teamVoteString[cs_offset],
cgs.teamVoteYes[cs_offset], cgs.teamVoteNo[cs_offset] );
}
}
else
{
s = va("TEAMVOTE(%i):%s yes:%i no:%i", sec, cgs.teamVoteString[cs_offset],
cgs.teamVoteYes[cs_offset], cgs.teamVoteNo[cs_offset] );
}
CG_DrawSmallString( 4, 90, s, 1.0F );
}
static qboolean CG_DrawScoreboard() {
return CG_DrawOldScoreboard();
}
/*
=================
CG_DrawIntermission
=================
*/
static void CG_DrawIntermission( void ) {
// int key;
//if (cg_singlePlayer.integer) {
// CG_DrawCenterString();
// return;
//}
cg.scoreFadeTime = cg.time;
cg.scoreBoardShowing = CG_DrawScoreboard();
}
/*
=================
CG_DrawFollow
=================
*/
static qboolean CG_DrawFollow( void )
{
const char *s;
if ( !(cg.snap->ps.pm_flags & PMF_FOLLOW) )
{
return qfalse;
}
// s = "following";
if (cgs.gametype == GT_POWERDUEL)
{
clientInfo_t *ci = &cgs.clientinfo[ cg.snap->ps.clientNum ];
if (ci->duelTeam == DUELTEAM_LONE)
{
s = CG_GetStringEdString("MP_INGAME", "FOLLOWINGLONE");
}
else if (ci->duelTeam == DUELTEAM_DOUBLE)
{
s = CG_GetStringEdString("MP_INGAME", "FOLLOWINGDOUBLE");
}
else
{
s = CG_GetStringEdString("MP_INGAME", "FOLLOWING");
}
}
else
{
s = CG_GetStringEdString("MP_INGAME", "FOLLOWING");
}
CG_Text_Paint ( 320 - CG_Text_Width ( s, 1.0f, FONT_MEDIUM ) / 2, 60, 1.0f, colorWhite, s, 0, 0, 0, FONT_MEDIUM );
s = cgs.clientinfo[ cg.snap->ps.clientNum ].name;
CG_Text_Paint ( 320 - CG_Text_Width ( s, 2.0f, FONT_MEDIUM ) / 2, 80, 2.0f, colorWhite, s, 0, 0, 0, FONT_MEDIUM );
return qtrue;
}
#if 0
static void CG_DrawTemporaryStats()
{ //placeholder for testing (draws ammo and force power)
char s[512];
if (!cg.snap)
{
return;
}
sprintf(s, "Force: %i", cg.snap->ps.fd.forcePower);
CG_DrawBigString(SCREEN_WIDTH-164, SCREEN_HEIGHT-dmgIndicSize, s, 1.0f);
sprintf(s, "Ammo: %i", cg.snap->ps.ammo[weaponData[cg.snap->ps.weapon].ammoIndex]);
CG_DrawBigString(SCREEN_WIDTH-164, SCREEN_HEIGHT-112, s, 1.0f);
sprintf(s, "Health: %i", cg.snap->ps.stats[STAT_HEALTH]);
CG_DrawBigString(8, SCREEN_HEIGHT-dmgIndicSize, s, 1.0f);
sprintf(s, "Armor: %i", cg.snap->ps.stats[STAT_ARMOR]);
CG_DrawBigString(8, SCREEN_HEIGHT-112, s, 1.0f);
}
#endif
/*
=================
CG_DrawAmmoWarning
=================
*/
static void CG_DrawAmmoWarning( void ) {
#if 0
const char *s;
int w;
if (!cg_drawStatus.integer)
{
return;
}
if ( cg_drawAmmoWarning.integer == 0 ) {
return;
}
if ( !cg.lowAmmoWarning ) {
return;
}
if ( cg.lowAmmoWarning == 2 ) {
s = "OUT OF AMMO";
} else {
s = "LOW AMMO WARNING";
}
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString(320 - w / 2, 64, s, 1.0F);
#endif
}
/*
=================
CG_DrawWarmup
=================
*/
static void CG_DrawWarmup( void ) {
int w, sec, i;
float scale;
const char *s;
sec = cg.warmup;
if ( !sec ) {
return;
}
if ( sec < 0 ) {
// s = "Waiting for players";
s = CG_GetStringEdString("MP_INGAME", "WAITING_FOR_PLAYERS");
w = CG_DrawStrlen( s ) * BIGCHAR_WIDTH;
CG_DrawBigString(320 - w / 2, 24, s, 1.0F);
cg.warmupCount = 0;
return;
}
if (cgs.gametype == GT_DUEL || cgs.gametype == GT_POWERDUEL)
{
// find the two active players
clientInfo_t *ci1, *ci2, *ci3;
ci1 = NULL;
ci2 = NULL;
ci3 = NULL;
if (cgs.gametype == GT_POWERDUEL)
{
if (cgs.duelist1 != -1)
{
ci1 = &cgs.clientinfo[cgs.duelist1];
}
if (cgs.duelist2 != -1)
{
ci2 = &cgs.clientinfo[cgs.duelist2];
}
if (cgs.duelist3 != -1)
{
ci3 = &cgs.clientinfo[cgs.duelist3];
}
}
else
{
for ( i = 0 ; i < cgs.maxclients ; i++ ) {
if ( cgs.clientinfo[i].infoValid && cgs.clientinfo[i].team == TEAM_FREE ) {
if ( !ci1 ) {
ci1 = &cgs.clientinfo[i];
} else {
ci2 = &cgs.clientinfo[i];
}
}
}
}
if ( ci1 && ci2 )
{
if (ci3)
{
s = va( "%s vs %s and %s", ci1->name, ci2->name, ci3->name );
}
else
{
s = va( "%s vs %s", ci1->name, ci2->name );
}
w = CG_Text_Width(s, 0.6f, FONT_MEDIUM);
CG_Text_Paint(320 - w / 2, 60, 0.6f, colorWhite, s, 0, 0, ITEM_TEXTSTYLE_SHADOWEDMORE,FONT_MEDIUM);
}
} else {
if ( cgs.gametype == GT_FFA ) s = CG_GetStringEdString("MENUS", "FREE_FOR_ALL");//"Free For All";
else if ( cgs.gametype == GT_HOLOCRON ) s = CG_GetStringEdString("MENUS", "HOLOCRON_FFA");//"Holocron FFA";
else if ( cgs.gametype == GT_JEDIMASTER ) s = "Jedi Master";
else if ( cgs.gametype == GT_TEAM ) s = CG_GetStringEdString("MENUS", "TEAM_FFA");//"Team FFA";
else if ( cgs.gametype == GT_SIEGE ) s = CG_GetStringEdString("MENUS", "SIEGE");//"Siege";
else if ( cgs.gametype == GT_CTF ) s = CG_GetStringEdString("MENUS", "CAPTURE_THE_FLAG");//"Capture the Flag";
else if ( cgs.gametype == GT_CTY ) s = CG_GetStringEdString("MENUS", "CAPTURE_THE_YSALIMARI");//"Capture the Ysalamiri";
else if ( cgs.gametype == GT_SINGLE_PLAYER ) s = "Cooperative";
else s = "";
w = CG_Text_Width(s, 1.5f, FONT_MEDIUM);
CG_Text_Paint(320 - w / 2, 90, 1.5f, colorWhite, s, 0, 0, ITEM_TEXTSTYLE_SHADOWEDMORE,FONT_MEDIUM);
}
sec = ( sec - cg.time ) / 1000;
if ( sec < 0 ) {
cg.warmup = 0;
sec = 0;
}
// s = va( "Starts in: %i", sec + 1 );
s = va( "%s: %i",CG_GetStringEdString("MP_INGAME", "STARTS_IN"), sec + 1 );
if ( sec != cg.warmupCount ) {
cg.warmupCount = sec;
if (cgs.gametype != GT_SIEGE)
{
switch ( sec ) {
case 0:
trap->S_StartLocalSound( cgs.media.count1Sound, CHAN_ANNOUNCER );
break;
case 1:
trap->S_StartLocalSound( cgs.media.count2Sound, CHAN_ANNOUNCER );
break;
case 2:
trap->S_StartLocalSound( cgs.media.count3Sound, CHAN_ANNOUNCER );
break;
default:
break;
}
}
}
scale = 0.45f;
switch ( cg.warmupCount ) {
case 0:
scale = 1.25f;
break;
case 1:
scale = 1.15f;
break;
case 2:
scale = 1.05f;
break;
default:
scale = 0.9f;
break;
}
w = CG_Text_Width(s, scale, FONT_MEDIUM);
CG_Text_Paint(320 - w / 2, 125, scale, colorWhite, s, 0, 0, ITEM_TEXTSTYLE_SHADOWEDMORE, FONT_MEDIUM);
}
//==================================================================================
/*
=================
CG_DrawTimedMenus
=================
*/
void CG_DrawTimedMenus() {
if (cg.voiceTime) {
int t = cg.time - cg.voiceTime;
if ( t > 2500 ) {
Menus_CloseByName("voiceMenu");
trap->Cvar_Set("cl_conXOffset", "0");
cg.voiceTime = 0;
}
}
}
void CG_DrawFlagStatus()
{
int myFlagTakenShader = 0;
int theirFlagShader = 0;
int team = 0;
int startDrawPos = 2;
int ico_size = 32;
//Raz: was missing this
trap->R_SetColor( NULL );
if (!cg.snap)
{
return;
}
if (cgs.gametype != GT_CTF && cgs.gametype != GT_CTY)
{
return;
}
team = cg.snap->ps.persistant[PERS_TEAM];
if (cgs.gametype == GT_CTY)
{
if (team == TEAM_RED)
{
myFlagTakenShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_rflag_x" );
theirFlagShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_bflag_ys" );
}
else
{
myFlagTakenShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_bflag_x" );
theirFlagShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_rflag_ys" );
}
}
else
{
if (team == TEAM_RED)
{
myFlagTakenShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_rflag_x" );
theirFlagShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_bflag" );
}
else
{
myFlagTakenShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_bflag_x" );
theirFlagShader = trap->R_RegisterShaderNoMip( "gfx/hud/mpi_rflag" );
}
}
if (CG_YourTeamHasFlag())
{
//CG_DrawPic( startDrawPos, 330, ico_size, ico_size, theirFlagShader );
CG_DrawPic( 2, 330-startDrawPos, ico_size, ico_size, theirFlagShader );
startDrawPos += ico_size+2;
}
if (CG_OtherTeamHasFlag())
{
//CG_DrawPic( startDrawPos, 330, ico_size, ico_size, myFlagTakenShader );
CG_DrawPic( 2, 330-startDrawPos, ico_size, ico_size, myFlagTakenShader );
}
}
//draw meter showing jetpack fuel when it's not full
#define JPFUELBAR_H 100.0f
#define JPFUELBAR_W 20.0f
#define JPFUELBAR_X (SCREEN_WIDTH-JPFUELBAR_W-8.0f)
#define JPFUELBAR_Y 260.0f
void CG_DrawJetpackFuel(void)
{
vec4_t aColor;
vec4_t cColor;
float x = JPFUELBAR_X;
float y = JPFUELBAR_Y;
float percent = ((float)cg.snap->ps.jetpackFuel/100.0f)*JPFUELBAR_H;
if (percent > JPFUELBAR_H)
{
return;
}
if (percent < 0.1f)
{
percent = 0.1f;
}
//color of the bar
aColor[0] = 0.5f;
aColor[1] = 0.0f;
aColor[2] = 0.0f;
aColor[3] = 0.8f;
//color of greyed out "missing fuel"
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.1f;
//draw the background (black)
CG_DrawRect(x, y, JPFUELBAR_W, JPFUELBAR_H, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f+(JPFUELBAR_H-percent), JPFUELBAR_W-1.0f, JPFUELBAR_H-1.0f-(JPFUELBAR_H-percent), aColor);
//then draw the other part greyed out
CG_FillRect(x+1.0f, y+1.0f, JPFUELBAR_W-1.0f, JPFUELBAR_H-percent, cColor);
}
//draw meter showing e-web health when it is in use
#define EWEBHEALTH_H 100.0f
#define EWEBHEALTH_W 20.0f
#define EWEBHEALTH_X (SCREEN_WIDTH-EWEBHEALTH_W-8.0f)
#define EWEBHEALTH_Y 290.0f
void CG_DrawEWebHealth(void)
{
vec4_t aColor;
vec4_t cColor;
float x = EWEBHEALTH_X;
float y = EWEBHEALTH_Y;
centity_t *eweb = &cg_entities[cg.predictedPlayerState.emplacedIndex];
float percent = ((float)eweb->currentState.health/eweb->currentState.maxhealth)*EWEBHEALTH_H;
if (percent > EWEBHEALTH_H)
{
return;
}
if (percent < 0.1f)
{
percent = 0.1f;
}
//kind of hacky, need to pass a coordinate in here
if (cg.snap->ps.jetpackFuel < 100)
{
x -= (JPFUELBAR_W+8.0f);
}
if (cg.snap->ps.cloakFuel < 100)
{
x -= (JPFUELBAR_W+8.0f);
}
//color of the bar
aColor[0] = 0.5f;
aColor[1] = 0.0f;
aColor[2] = 0.0f;
aColor[3] = 0.8f;
//color of greyed out "missing fuel"
cColor[0] = 0.5f;
cColor[1] = 0.5f;
cColor[2] = 0.5f;
cColor[3] = 0.1f;
//draw the background (black)
CG_DrawRect(x, y, EWEBHEALTH_W, EWEBHEALTH_H, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much health there is in the color specified
CG_FillRect(x+1.0f, y+1.0f+(EWEBHEALTH_H-percent), EWEBHEALTH_W-1.0f, EWEBHEALTH_H-1.0f-(EWEBHEALTH_H-percent), aColor);
//then draw the other part greyed out
CG_FillRect(x+1.0f, y+1.0f, EWEBHEALTH_W-1.0f, EWEBHEALTH_H-percent, cColor);
}
//draw meter showing cloak fuel when it's not full
#define CLFUELBAR_H 100.0f
#define CLFUELBAR_W 20.0f
#define CLFUELBAR_X (SCREEN_WIDTH-CLFUELBAR_W-8.0f)
#define CLFUELBAR_Y 260.0f
void CG_DrawCloakFuel(void)
{
vec4_t aColor;
vec4_t cColor;
float x = CLFUELBAR_X;
float y = CLFUELBAR_Y;
float percent = ((float)cg.snap->ps.cloakFuel/100.0f)*CLFUELBAR_H;
if (percent > CLFUELBAR_H)
{
return;
}
if ( cg.snap->ps.jetpackFuel < 100 )
{//if drawing jetpack fuel bar too, then move this over...?
x -= (JPFUELBAR_W+8.0f);
}
if (percent < 0.1f)
{
percent = 0.1f;
}
//color of the bar
aColor[0] = 0.0f;
aColor[1] = 0.0f;
aColor[2] = 0.6f;
aColor[3] = 0.8f;
//color of greyed out "missing fuel"
cColor[0] = 0.1f;
cColor[1] = 0.1f;
cColor[2] = 0.3f;
cColor[3] = 0.1f;
//draw the background (black)
CG_DrawRect(x, y, CLFUELBAR_W, CLFUELBAR_H, 1.0f, colorTable[CT_BLACK]);
//now draw the part to show how much fuel there is in the color specified
CG_FillRect(x+1.0f, y+1.0f+(CLFUELBAR_H-percent), CLFUELBAR_W-1.0f, CLFUELBAR_H-1.0f-(CLFUELBAR_H-percent), aColor);
//then draw the other part greyed out
CG_FillRect(x+1.0f, y+1.0f, CLFUELBAR_W-1.0f, CLFUELBAR_H-percent, cColor);
}
int cgRageTime = 0;
int cgRageFadeTime = 0;
float cgRageFadeVal = 0;
int cgRageRecTime = 0;
int cgRageRecFadeTime = 0;
float cgRageRecFadeVal = 0;
int cgAbsorbTime = 0;
int cgAbsorbFadeTime = 0;
float cgAbsorbFadeVal = 0;
int cgProtectTime = 0;
int cgProtectFadeTime = 0;
float cgProtectFadeVal = 0;
int cgYsalTime = 0;
int cgYsalFadeTime = 0;
float cgYsalFadeVal = 0;
qboolean gCGHasFallVector = qfalse;
vec3_t gCGFallVector;
/*
=================
CG_Draw2D
=================
*/
extern int cgSiegeRoundState;
extern int cgSiegeRoundTime;
extern int team1Timed;
extern int team2Timed;
int cg_beatingSiegeTime = 0;
int cgSiegeRoundBeganTime = 0;
int cgSiegeRoundCountTime = 0;
static void CG_DrawSiegeTimer(int timeRemaining, qboolean isMyTeam)
{ //rwwFIXMEFIXME: Make someone make assets and use them.
//this function is pretty much totally placeholder.
// int x = 0;
// int y = SCREEN_HEIGHT-160;
int fColor = 0;
int minutes = 0;
int seconds = 0;
char timeStr[1024];
menuDef_t *menuHUD = NULL;
itemDef_t *item = NULL;
menuHUD = Menus_FindByName("mp_timer");
if (!menuHUD)
{
return;
}
item = Menu_FindItemByName(menuHUD, "frame");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
seconds = timeRemaining;
while (seconds >= 60)
{
minutes++;
seconds -= 60;
}
strcpy(timeStr, va( "%i:%02i", minutes, seconds ));
if (isMyTeam)
{
fColor = CT_HUD_RED;
}
else
{
fColor = CT_HUD_GREEN;
}
// trap->Cvar_Set("ui_siegeTimer", timeStr);
// CG_DrawProportionalString( x+16, y+40, timeStr, UI_SMALLFONT|UI_DROPSHADOW, colorTable[fColor] );
item = Menu_FindItemByName(menuHUD, "timer");
if (item)
{
CG_DrawProportionalString(
item->window.rect.x,
item->window.rect.y,
timeStr,
UI_SMALLFONT|UI_DROPSHADOW,
colorTable[fColor] );
}
}
static void CG_DrawSiegeDeathTimer( int timeRemaining )
{
int minutes = 0;
int seconds = 0;
char timeStr[1024];
menuDef_t *menuHUD = NULL;
itemDef_t *item = NULL;
menuHUD = Menus_FindByName("mp_timer");
if (!menuHUD)
{
return;
}
item = Menu_FindItemByName(menuHUD, "frame");
if (item)
{
trap->R_SetColor( item->window.foreColor );
CG_DrawPic(
item->window.rect.x,
item->window.rect.y,
item->window.rect.w,
item->window.rect.h,
item->window.background );
}
seconds = timeRemaining;
while (seconds >= 60)
{
minutes++;
seconds -= 60;
}
if (seconds < 10)
{
strcpy(timeStr, va( "%i:0%i", minutes, seconds ));
}
else
{
strcpy(timeStr, va( "%i:%i", minutes, seconds ));
}
item = Menu_FindItemByName(menuHUD, "deathtimer");
if (item)
{
CG_DrawProportionalString(
item->window.rect.x,
item->window.rect.y,
timeStr,
UI_SMALLFONT|UI_DROPSHADOW,
item->window.foreColor );
}
}
int cgSiegeEntityRender = 0;
static void CG_DrawSiegeHUDItem(void)
{
void *g2;
qhandle_t handle;
vec3_t origin, angles;
vec3_t mins, maxs;
float len;
centity_t *cent = &cg_entities[cgSiegeEntityRender];
if (cent->ghoul2)
{
g2 = cent->ghoul2;
handle = 0;
}
else
{
handle = cgs.gameModels[cent->currentState.modelindex];
g2 = NULL;
}
if (handle)
{
trap->R_ModelBounds( handle, mins, maxs );
}
else
{
VectorSet(mins, -16, -16, -20);
VectorSet(maxs, 16, 16, 32);
}
origin[2] = -0.5 * ( mins[2] + maxs[2] );
origin[1] = 0.5 * ( mins[1] + maxs[1] );
len = 0.5 * ( maxs[2] - mins[2] );
origin[0] = len / 0.268;
VectorClear(angles);
angles[YAW] = cg.autoAngles[YAW];
CG_Draw3DModel( 8, 8, 64, 64, handle, g2, cent->currentState.g2radius, 0, origin, angles );
cgSiegeEntityRender = 0; //reset for next frame
}
/*====================================
chatbox functionality -rww
====================================*/
#define CHATBOX_CUTOFF_LEN 550
#define CHATBOX_FONT_HEIGHT 20
//utility func, insert a string into a string at the specified
//place (assuming this will not overflow the buffer)
void CG_ChatBox_StrInsert(char *buffer, int place, char *str)
{
int insLen = strlen(str);
int i = strlen(buffer);
int k = 0;
buffer[i+insLen+1] = 0; //terminate the string at its new length
while (i >= place)
{
buffer[i+insLen] = buffer[i];
i--;
}
i++;
while (k < insLen)
{
buffer[i] = str[k];
i++;
k++;
}
}
//add chatbox string
void CG_ChatBox_AddString(char *chatStr)
{
chatBoxItem_t *chat = &cg.chatItems[cg.chatItemActive];
float chatLen;
if (cg_chatBox.integer<=0)
{ //don't bother then.
return;
}
memset(chat, 0, sizeof(chatBoxItem_t));
if (strlen(chatStr) > sizeof(chat->string))
{ //too long, terminate at proper len.
chatStr[sizeof(chat->string)-1] = 0;
}
strcpy(chat->string, chatStr);
chat->time = cg.time + cg_chatBox.integer;
chat->lines = 1;
chatLen = CG_Text_Width(chat->string, 1.0f, FONT_SMALL);
if (chatLen > CHATBOX_CUTOFF_LEN)
{ //we have to break it into segments...
int i = 0;
int lastLinePt = 0;
char s[2];
chatLen = 0;
while (chat->string[i])
{
s[0] = chat->string[i];
s[1] = 0;
chatLen += CG_Text_Width(s, 0.65f, FONT_SMALL);
if (chatLen >= CHATBOX_CUTOFF_LEN)
{
int j = i;
while (j > 0 && j > lastLinePt)
{
if (chat->string[j] == ' ')
{
break;
}
j--;
}
if (chat->string[j] == ' ')
{
i = j;
}
chat->lines++;
CG_ChatBox_StrInsert(chat->string, i, "\n");
i++;
chatLen = 0;
lastLinePt = i+1;
}
i++;
}
}
cg.chatItemActive++;
if (cg.chatItemActive >= MAX_CHATBOX_ITEMS)
{
cg.chatItemActive = 0;
}
}
//insert item into array (rearranging the array if necessary)
void CG_ChatBox_ArrayInsert(chatBoxItem_t **array, int insPoint, int maxNum, chatBoxItem_t *item)
{
if (array[insPoint])
{ //recursively call, to move everything up to the top
if (insPoint+1 >= maxNum)
{
trap->Error( ERR_DROP, "CG_ChatBox_ArrayInsert: Exceeded array size");
}
CG_ChatBox_ArrayInsert(array, insPoint+1, maxNum, array[insPoint]);
}
//now that we have moved anything that would be in this slot up, insert what we want into the slot
array[insPoint] = item;
}
//go through all the chat strings and draw them if they are not yet expired
static QINLINE void CG_ChatBox_DrawStrings(void)
{
chatBoxItem_t *drawThese[MAX_CHATBOX_ITEMS];
int numToDraw = 0;
int linesToDraw = 0;
int i = 0;
int x = 30;
float y = cg.scoreBoardShowing ? 475 : cg_chatBoxHeight.integer;
float fontScale = 0.65f;
if (!cg_chatBox.integer)
{
return;
}
memset(drawThese, 0, sizeof(drawThese));
while (i < MAX_CHATBOX_ITEMS)
{
if (cg.chatItems[i].time >= cg.time)
{
int check = numToDraw;
int insertionPoint = numToDraw;
while (check >= 0)
{
if (drawThese[check] &&
cg.chatItems[i].time < drawThese[check]->time)
{ //insert here
insertionPoint = check;
}
check--;
}
CG_ChatBox_ArrayInsert(drawThese, insertionPoint, MAX_CHATBOX_ITEMS, &cg.chatItems[i]);
numToDraw++;
linesToDraw += cg.chatItems[i].lines;
}
i++;
}
if (!numToDraw)
{ //nothing, then, just get out of here now.
return;
}
//move initial point up so we draw bottom-up (visually)
y -= (CHATBOX_FONT_HEIGHT*fontScale)*linesToDraw;
//we have the items we want to draw, just quickly loop through them now
i = 0;
while (i < numToDraw)
{
CG_Text_Paint(x, y, fontScale, colorWhite, drawThese[i]->string, 0, 0, ITEM_TEXTSTYLE_OUTLINED, FONT_SMALL );
y += ((CHATBOX_FONT_HEIGHT*fontScale)*drawThese[i]->lines);
i++;
}
}
static void CG_Draw2DScreenTints( void )
{
float rageTime, rageRecTime, absorbTime, protectTime, ysalTime;
vec4_t hcolor;
if (cgs.clientinfo[cg.snap->ps.clientNum].team != TEAM_SPECTATOR)
{
if (cg.snap->ps.fd.forcePowersActive & (1 << FP_RAGE))
{
if (!cgRageTime)
{
cgRageTime = cg.time;
}
rageTime = (float)(cg.time - cgRageTime);
rageTime /= 9000;
if (rageTime < 0)
{
rageTime = 0;
}
if (rageTime > 0.15)
{
rageTime = 0.15f;
}
hcolor[3] = rageTime;
hcolor[0] = 0.7f;
hcolor[1] = 0;
hcolor[2] = 0;
if (!cg.renderingThirdPerson)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
cgRageFadeTime = 0;
cgRageFadeVal = 0;
}
else if (cgRageTime)
{
if (!cgRageFadeTime)
{
cgRageFadeTime = cg.time;
cgRageFadeVal = 0.15f;
}
rageTime = cgRageFadeVal;
cgRageFadeVal -= (cg.time - cgRageFadeTime)*0.000005;
if (rageTime < 0)
{
rageTime = 0;
}
if (rageTime > 0.15f)
{
rageTime = 0.15f;
}
if (cg.snap->ps.fd.forceRageRecoveryTime > cg.time)
{
float checkRageRecTime = rageTime;
if (checkRageRecTime < 0.15f)
{
checkRageRecTime = 0.15f;
}
hcolor[3] = checkRageRecTime;
hcolor[0] = rageTime*4;
if (hcolor[0] < 0.2f)
{
hcolor[0] = 0.2f;
}
hcolor[1] = 0.2f;
hcolor[2] = 0.2f;
}
else
{
hcolor[3] = rageTime;
hcolor[0] = 0.7f;
hcolor[1] = 0;
hcolor[2] = 0;
}
if (!cg.renderingThirdPerson && rageTime)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else
{
if (cg.snap->ps.fd.forceRageRecoveryTime > cg.time)
{
hcolor[3] = 0.15f;
hcolor[0] = 0.2f;
hcolor[1] = 0.2f;
hcolor[2] = 0.2f;
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
cgRageTime = 0;
}
}
else if (cg.snap->ps.fd.forceRageRecoveryTime > cg.time)
{
if (!cgRageRecTime)
{
cgRageRecTime = cg.time;
}
rageRecTime = (float)(cg.time - cgRageRecTime);
rageRecTime /= 9000;
if (rageRecTime < 0.15f)//0)
{
rageRecTime = 0.15f;//0;
}
if (rageRecTime > 0.15f)
{
rageRecTime = 0.15f;
}
hcolor[3] = rageRecTime;
hcolor[0] = 0.2f;
hcolor[1] = 0.2f;
hcolor[2] = 0.2f;
if (!cg.renderingThirdPerson)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
cgRageRecFadeTime = 0;
cgRageRecFadeVal = 0;
}
else if (cgRageRecTime)
{
if (!cgRageRecFadeTime)
{
cgRageRecFadeTime = cg.time;
cgRageRecFadeVal = 0.15f;
}
rageRecTime = cgRageRecFadeVal;
cgRageRecFadeVal -= (cg.time - cgRageRecFadeTime)*0.000005;
if (rageRecTime < 0)
{
rageRecTime = 0;
}
if (rageRecTime > 0.15f)
{
rageRecTime = 0.15f;
}
hcolor[3] = rageRecTime;
hcolor[0] = 0.2f;
hcolor[1] = 0.2f;
hcolor[2] = 0.2f;
if (!cg.renderingThirdPerson && rageRecTime)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else
{
cgRageRecTime = 0;
}
}
if (cg.snap->ps.fd.forcePowersActive & (1 << FP_ABSORB))
{
if (!cgAbsorbTime)
{
cgAbsorbTime = cg.time;
}
absorbTime = (float)(cg.time - cgAbsorbTime);
absorbTime /= 9000;
if (absorbTime < 0)
{
absorbTime = 0;
}
if (absorbTime > 0.15f)
{
absorbTime = 0.15f;
}
hcolor[3] = absorbTime/2;
hcolor[0] = 0;
hcolor[1] = 0;
hcolor[2] = 0.7f;
if (!cg.renderingThirdPerson)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
cgAbsorbFadeTime = 0;
cgAbsorbFadeVal = 0;
}
else if (cgAbsorbTime)
{
if (!cgAbsorbFadeTime)
{
cgAbsorbFadeTime = cg.time;
cgAbsorbFadeVal = 0.15f;
}
absorbTime = cgAbsorbFadeVal;
cgAbsorbFadeVal -= (cg.time - cgAbsorbFadeTime)*0.000005f;
if (absorbTime < 0)
{
absorbTime = 0;
}
if (absorbTime > 0.15f)
{
absorbTime = 0.15f;
}
hcolor[3] = absorbTime/2;
hcolor[0] = 0;
hcolor[1] = 0;
hcolor[2] = 0.7f;
if (!cg.renderingThirdPerson && absorbTime)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else
{
cgAbsorbTime = 0;
}
}
if (cg.snap->ps.fd.forcePowersActive & (1 << FP_PROTECT))
{
if (!cgProtectTime)
{
cgProtectTime = cg.time;
}
protectTime = (float)(cg.time - cgProtectTime);
protectTime /= 9000;
if (protectTime < 0)
{
protectTime = 0;
}
if (protectTime > 0.15f)
{
protectTime = 0.15f;
}
hcolor[3] = protectTime/2;
hcolor[0] = 0;
hcolor[1] = 0.7f;
hcolor[2] = 0;
if (!cg.renderingThirdPerson)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
cgProtectFadeTime = 0;
cgProtectFadeVal = 0;
}
else if (cgProtectTime)
{
if (!cgProtectFadeTime)
{
cgProtectFadeTime = cg.time;
cgProtectFadeVal = 0.15f;
}
protectTime = cgProtectFadeVal;
cgProtectFadeVal -= (cg.time - cgProtectFadeTime)*0.000005;
if (protectTime < 0)
{
protectTime = 0;
}
if (protectTime > 0.15f)
{
protectTime = 0.15f;
}
hcolor[3] = protectTime/2;
hcolor[0] = 0;
hcolor[1] = 0.7f;
hcolor[2] = 0;
if (!cg.renderingThirdPerson && protectTime)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else
{
cgProtectTime = 0;
}
}
if (cg.snap->ps.rocketLockIndex != ENTITYNUM_NONE && (cg.time - cg.snap->ps.rocketLockTime) > 0)
{
CG_DrawRocketLocking( cg.snap->ps.rocketLockIndex, cg.snap->ps.rocketLockTime );
}
if (BG_HasYsalamiri(cgs.gametype, &cg.snap->ps))
{
if (!cgYsalTime)
{
cgYsalTime = cg.time;
}
ysalTime = (float)(cg.time - cgYsalTime);
ysalTime /= 9000;
if (ysalTime < 0)
{
ysalTime = 0;
}
if (ysalTime > 0.15f)
{
ysalTime = 0.15f;
}
hcolor[3] = ysalTime/2;
hcolor[0] = 0.7f;
hcolor[1] = 0.7f;
hcolor[2] = 0;
if (!cg.renderingThirdPerson)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
cgYsalFadeTime = 0;
cgYsalFadeVal = 0;
}
else if (cgYsalTime)
{
if (!cgYsalFadeTime)
{
cgYsalFadeTime = cg.time;
cgYsalFadeVal = 0.15f;
}
ysalTime = cgYsalFadeVal;
cgYsalFadeVal -= (cg.time - cgYsalFadeTime)*0.000005f;
if (ysalTime < 0)
{
ysalTime = 0;
}
if (ysalTime > 0.15f)
{
ysalTime = 0.15f;
}
hcolor[3] = ysalTime/2;
hcolor[0] = 0.7f;
hcolor[1] = 0.7f;
hcolor[2] = 0;
if (!cg.renderingThirdPerson && ysalTime)
{
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else
{
cgYsalTime = 0;
}
}
}
if ( (cg.refdef.viewContents&CONTENTS_LAVA) )
{//tint screen red
float phase = cg.time / 1000.0 * WAVE_FREQUENCY * M_PI * 2;
hcolor[3] = 0.5 + (0.15f*sin( phase ));
hcolor[0] = 0.7f;
hcolor[1] = 0;
hcolor[2] = 0;
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else if ( (cg.refdef.viewContents&CONTENTS_SLIME) )
{//tint screen green
float phase = cg.time / 1000.0 * WAVE_FREQUENCY * M_PI * 2;
hcolor[3] = 0.4 + (0.1f*sin( phase ));
hcolor[0] = 0;
hcolor[1] = 0.7f;
hcolor[2] = 0;
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
else if ( (cg.refdef.viewContents&CONTENTS_WATER) )
{//tint screen light blue -- FIXME: don't do this if CONTENTS_FOG? (in case someone *does* make a water shader with fog in it?)
float phase = cg.time / 1000.0f * WAVE_FREQUENCY * M_PI * 2;
hcolor[3] = 0.3f + (0.05f*sinf( phase ));
hcolor[0] = 0;
hcolor[1] = 0.2f;
hcolor[2] = 0.8f;
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
}
}
static void CG_Draw2D( void ) {
float inTime = cg.invenSelectTime+WEAPON_SELECT_TIME;
float wpTime = cg.weaponSelectTime+WEAPON_SELECT_TIME;
float fallTime;
float bestTime;
int drawSelect = 0;
// if we are taking a levelshot for the menu, don't draw anything
if ( cg.levelShot ) {
return;
}
if (cgs.clientinfo[cg.snap->ps.clientNum].team == TEAM_SPECTATOR)
{
cgRageTime = 0;
cgRageFadeTime = 0;
cgRageFadeVal = 0;
cgRageRecTime = 0;
cgRageRecFadeTime = 0;
cgRageRecFadeVal = 0;
cgAbsorbTime = 0;
cgAbsorbFadeTime = 0;
cgAbsorbFadeVal = 0;
cgProtectTime = 0;
cgProtectFadeTime = 0;
cgProtectFadeVal = 0;
cgYsalTime = 0;
cgYsalFadeTime = 0;
cgYsalFadeVal = 0;
}
if ( !cg_draw2D.integer ) {
gCGHasFallVector = qfalse;
VectorClear( gCGFallVector );
return;
}
if ( cg.snap->ps.pm_type == PM_INTERMISSION ) {
CG_DrawIntermission();
CG_ChatBox_DrawStrings();
return;
}
CG_Draw2DScreenTints();
if (cg.snap->ps.rocketLockIndex != ENTITYNUM_NONE && (cg.time - cg.snap->ps.rocketLockTime) > 0)
{
CG_DrawRocketLocking( cg.snap->ps.rocketLockIndex, cg.snap->ps.rocketLockTime );
}
if (cg.snap->ps.holocronBits)
{
CG_DrawHolocronIcons();
}
if (cg.snap->ps.fd.forcePowersActive || cg.snap->ps.fd.forceRageRecoveryTime > cg.time)
{
CG_DrawActivePowers();
}
if (cg.snap->ps.jetpackFuel < 100)
{ //draw it as long as it isn't full
CG_DrawJetpackFuel();
}
if (cg.snap->ps.cloakFuel < 100)
{ //draw it as long as it isn't full
CG_DrawCloakFuel();
}
if (cg.predictedPlayerState.emplacedIndex > 0)
{
centity_t *eweb = &cg_entities[cg.predictedPlayerState.emplacedIndex];
if (eweb->currentState.weapon == WP_NONE)
{ //using an e-web, draw its health
CG_DrawEWebHealth();
}
}
// Draw this before the text so that any text won't get clipped off
CG_DrawZoomMask();
/*
if (cg.cameraMode) {
return;
}
*/
if ( cg.snap->ps.persistant[PERS_TEAM] == TEAM_SPECTATOR ) {
CG_DrawSpectator();
CG_DrawCrosshair(NULL, 0);
CG_DrawCrosshairNames();
CG_SaberClashFlare();
} else {
// don't draw any status if dead or the scoreboard is being explicitly shown
if ( !cg.showScores && cg.snap->ps.stats[STAT_HEALTH] > 0 ) {
if ( /*cg_drawStatus.integer*/0 ) {
//Reenable if stats are drawn with menu system again
Menu_PaintAll();
CG_DrawTimedMenus();
}
//CG_DrawTemporaryStats();
CG_DrawAmmoWarning();
CG_DrawCrosshairNames();
if (cg_drawStatus.integer)
{
CG_DrawIconBackground();
}
if (inTime > wpTime)
{
drawSelect = 1;
bestTime = cg.invenSelectTime;
}
else //only draw the most recent since they're drawn in the same place
{
drawSelect = 2;
bestTime = cg.weaponSelectTime;
}
if (cg.forceSelectTime > bestTime)
{
drawSelect = 3;
}
switch(drawSelect)
{
case 1:
CG_DrawInvenSelect();
break;
case 2:
CG_DrawWeaponSelect();
break;
case 3:
CG_DrawForceSelect();
break;
default:
break;
}
if (cg_drawStatus.integer)
{
//Powerups now done with upperright stuff
//CG_DrawPowerupIcons();
CG_DrawFlagStatus();
}
CG_SaberClashFlare();
if (cg_drawStatus.integer)
{
CG_DrawStats();
}
CG_DrawPickupItem();
//Do we want to use this system again at some point?
//CG_DrawReward();
}
}
if (cg.snap->ps.fallingToDeath)
{
vec4_t hcolor;
fallTime = (float)(cg.time - cg.snap->ps.fallingToDeath);
fallTime /= (FALL_FADE_TIME/2);
if (fallTime < 0)
{
fallTime = 0;
}
if (fallTime > 1)
{
fallTime = 1;
}
hcolor[3] = fallTime;
hcolor[0] = 0;
hcolor[1] = 0;
hcolor[2] = 0;
CG_FillRect( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, hcolor );
if (!gCGHasFallVector)
{
VectorCopy(cg.snap->ps.origin, gCGFallVector);
gCGHasFallVector = qtrue;
}
}
else
{
if (gCGHasFallVector)
{
gCGHasFallVector = qfalse;
VectorClear(gCGFallVector);
}
}
CG_DrawVote();
CG_DrawTeamVote();
CG_DrawLagometer();
if (!cl_paused.integer) {
CG_DrawBracketedEntities();
CG_DrawUpperRight();
}
if ( !CG_DrawFollow() ) {
CG_DrawWarmup();
}
if (cgSiegeRoundState)
{
char pStr[1024];
int rTime = 0;
//cgSiegeRoundBeganTime = 0;
switch (cgSiegeRoundState)
{
case 1:
CG_CenterPrint(CG_GetStringEdString("MP_INGAME", "WAITING_FOR_PLAYERS"), SCREEN_HEIGHT * 0.30, BIGCHAR_WIDTH);
break;
case 2:
rTime = (SIEGE_ROUND_BEGIN_TIME - (cg.time - cgSiegeRoundTime));
if (rTime < 0)
{
rTime = 0;
}
if (rTime > SIEGE_ROUND_BEGIN_TIME)
{
rTime = SIEGE_ROUND_BEGIN_TIME;
}
rTime /= 1000;
rTime += 1;
if (rTime < 1)
{
rTime = 1;
}
if (rTime <= 3 && rTime != cgSiegeRoundCountTime)
{
cgSiegeRoundCountTime = rTime;
switch (rTime)
{
case 1:
trap->S_StartLocalSound( cgs.media.count1Sound, CHAN_ANNOUNCER );
break;
case 2:
trap->S_StartLocalSound( cgs.media.count2Sound, CHAN_ANNOUNCER );
break;
case 3:
trap->S_StartLocalSound( cgs.media.count3Sound, CHAN_ANNOUNCER );
break;
default:
break;
}
}
Q_strncpyz(pStr, va("%s %i...", CG_GetStringEdString("MP_INGAME", "ROUNDBEGINSIN"), rTime), sizeof(pStr));
CG_CenterPrint(pStr, SCREEN_HEIGHT * 0.30, BIGCHAR_WIDTH);
//same
break;
default:
break;
}
cgSiegeEntityRender = 0;
}
else if (cgSiegeRoundTime)
{
CG_CenterPrint("", SCREEN_HEIGHT * 0.30, BIGCHAR_WIDTH);
cgSiegeRoundTime = 0;
//cgSiegeRoundBeganTime = cg.time;
cgSiegeEntityRender = 0;
}
else if (cgSiegeRoundBeganTime)
{ //Draw how much time is left in the round based on local info.
int timedTeam = TEAM_FREE;
int timedValue = 0;
if (cgSiegeEntityRender)
{ //render the objective item model since this client has it
CG_DrawSiegeHUDItem();
}
if (team1Timed)
{
timedTeam = TEAM_RED; //team 1
if (cg_beatingSiegeTime)
{
timedValue = cg_beatingSiegeTime;
}
else
{
timedValue = team1Timed;
}
}
else if (team2Timed)
{
timedTeam = TEAM_BLUE; //team 2
if (cg_beatingSiegeTime)
{
timedValue = cg_beatingSiegeTime;
}
else
{
timedValue = team2Timed;
}
}
if (timedTeam != TEAM_FREE)
{ //one of the teams has a timer
int timeRemaining;
qboolean isMyTeam = qfalse;
if (cgs.siegeTeamSwitch && !cg_beatingSiegeTime)
{ //in switchy mode but not beating a time, so count up.
timeRemaining = (cg.time-cgSiegeRoundBeganTime);
if (timeRemaining < 0)
{
timeRemaining = 0;
}
}
else
{
timeRemaining = (((cgSiegeRoundBeganTime)+timedValue) - cg.time);
}
if (timeRemaining > timedValue)
{
timeRemaining = timedValue;
}
else if (timeRemaining < 0)
{
timeRemaining = 0;
}
if (timeRemaining)
{
timeRemaining /= 1000;
}
if (cg.predictedPlayerState.persistant[PERS_TEAM] == timedTeam)
{ //the team that's timed is the one this client is on
isMyTeam = qtrue;
}
CG_DrawSiegeTimer(timeRemaining, isMyTeam);
}
}
else
{
cgSiegeEntityRender = 0;
}
if ( cg_siegeDeathTime )
{
int timeRemaining = ( cg_siegeDeathTime - cg.time );
if ( timeRemaining < 0 )
{
timeRemaining = 0;
cg_siegeDeathTime = 0;
}
if ( timeRemaining )
{
timeRemaining /= 1000;
}
CG_DrawSiegeDeathTimer( timeRemaining );
}
// don't draw center string if scoreboard is up
cg.scoreBoardShowing = CG_DrawScoreboard();
if ( !cg.scoreBoardShowing) {
CG_DrawCenterString();
}
// always draw chat
CG_ChatBox_DrawStrings();
}
qboolean CG_CullPointAndRadius( const vec3_t pt, float radius);
void CG_DrawMiscStaticModels( void ) {
int i, j;
refEntity_t ent;
vec3_t cullorg;
vec3_t diff;
memset( &ent, 0, sizeof( ent ) );
ent.reType = RT_MODEL;
ent.frame = 0;
ent.nonNormalizedAxes = qtrue;
// static models don't project shadows
ent.renderfx = RF_NOSHADOW;
for( i = 0; i < cgs.numMiscStaticModels; i++ ) {
VectorCopy(cgs.miscStaticModels[i].org, cullorg);
cullorg[2] += 1.0f;
if ( cgs.miscStaticModels[i].zoffset ) {
cullorg[2] += cgs.miscStaticModels[i].zoffset;
}
if( cgs.miscStaticModels[i].radius ) {
if( CG_CullPointAndRadius( cullorg, cgs.miscStaticModels[i].radius ) ) {
continue;
}
}
if( !trap->R_InPVS( cg.refdef.vieworg, cullorg, cg.refdef.areamask ) ) {
continue;
}
VectorCopy( cgs.miscStaticModels[i].org, ent.origin );
VectorCopy( cgs.miscStaticModels[i].org, ent.oldorigin );
VectorCopy( cgs.miscStaticModels[i].org, ent.lightingOrigin );
for( j = 0; j < 3; j++ ) {
VectorCopy( cgs.miscStaticModels[i].axes[j], ent.axis[j] );
}
ent.hModel = cgs.miscStaticModels[i].model;
VectorSubtract(ent.origin, cg.refdef.vieworg, diff);
if (VectorLength(diff)-(cgs.miscStaticModels[i].radius) <= cg.distanceCull) {
trap->R_AddRefEntityToScene( &ent );
}
}
}
static void CG_DrawTourneyScoreboard() {
}
/*
=====================
CG_DrawActive
Perform all drawing needed to completely fill the screen
=====================
*/
void CG_DrawActive( stereoFrame_t stereoView ) {
float separation;
vec3_t baseOrg;
// optionally draw the info screen instead
if ( !cg.snap ) {
CG_DrawInformation();
return;
}
// optionally draw the tournament scoreboard instead
if ( cg.snap->ps.persistant[PERS_TEAM] == TEAM_SPECTATOR &&
( cg.snap->ps.pm_flags & PMF_SCOREBOARD ) ) {
CG_DrawTourneyScoreboard();
return;
}
switch ( stereoView ) {
case STEREO_CENTER:
separation = 0;
break;
case STEREO_LEFT:
separation = -cg_stereoSeparation.value / 2;
break;
case STEREO_RIGHT:
separation = cg_stereoSeparation.value / 2;
break;
default:
separation = 0;
trap->Error( ERR_DROP, "CG_DrawActive: Undefined stereoView" );
}
// clear around the rendered view if sized down
CG_TileClear();
// offset vieworg appropriately if we're doing stereo separation
VectorCopy( cg.refdef.vieworg, baseOrg );
if ( separation != 0 ) {
VectorMA( cg.refdef.vieworg, -separation, cg.refdef.viewaxis[1], cg.refdef.vieworg );
}
if ( cg.snap->ps.fd.forcePowersActive & (1 << FP_SEE) )
cg.refdef.rdflags |= RDF_ForceSightOn;
cg.refdef.rdflags |= RDF_DRAWSKYBOX;
CG_DrawMiscStaticModels();
// draw 3D view
trap->R_RenderScene( &cg.refdef );
// restore original viewpoint if running stereo
if ( separation != 0 ) {
VectorCopy( baseOrg, cg.refdef.vieworg );
}
// draw status bar and other floating elements
CG_Draw2D();
}