/*
===========================================================================
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 .
===========================================================================
*/
#include "cg_headers.h"
#include "cg_media.h"
#include "../game/objectives.h"
// For printing objectives
static const short objectiveStartingYpos = 75; // Y starting position for objective text
static const short objectiveStartingXpos = 60; // X starting position for objective text
static const int objectiveTextBoxWidth = 500; // Width (in pixels) of text box
static const int objectiveTextBoxHeight = 300; // Height (in pixels) of text box
const char *showLoadPowersName[] =
{
"SP_INGAME_HEAL2",
"SP_INGAME_JUMP2",
"SP_INGAME_SPEED2",
"SP_INGAME_PUSH2",
"SP_INGAME_PULL2",
"SP_INGAME_MINDTRICK2",
"SP_INGAME_GRIP2",
"SP_INGAME_LIGHTNING2",
"SP_INGAME_SABER_THROW2",
"SP_INGAME_SABER_OFFENSE2",
"SP_INGAME_SABER_DEFENSE2",
NULL,
};
#define MAX_OBJ_GRAPHICS 4
#define OBJ_GRAPHIC_SIZE 240
int obj_graphics[MAX_OBJ_GRAPHICS];
qboolean CG_ForcePower_Valid(int forceKnownBits, int index);
/*
====================
ObjectivePrint_Line
Print a single mission objective
====================
*/
static void ObjectivePrint_Line(const int color, const int objectIndex, int &missionYcnt)
{
char *str,*strBegin;
int y,pixelLen,charLen,i;
const int maxHoldText = 1024;
char holdText[maxHoldText];
char finalText[2048];
qhandle_t graphic;
int iYPixelsPerLine = cgi_R_Font_HeightPixels(cgs.media.qhFontMedium, 1.0f);
cgi_SP_GetStringTextString( va("OBJECTIVES_%s",objectiveTable[objectIndex].name) , finalText, sizeof(finalText) );
// A hack to be able to count prisoners
if (objectIndex==T2_RANCOR_OBJ5)
{
char value[64];
int currTotal, minTotal;
gi.Cvar_VariableStringBuffer("ui_prisonerobj_currtotal",value,sizeof(value));
currTotal = atoi(value);
gi.Cvar_VariableStringBuffer("ui_prisonerobj_maxtotal",value,sizeof(value));
minTotal = atoi(value);
Q_strncpyz(finalText, va(finalText,currTotal,minTotal), sizeof(finalText));
}
pixelLen = cgi_R_Font_StrLenPixels(finalText, cgs.media.qhFontMedium, 1.0f);
str = finalText;
if (cgi_Language_IsAsian())
{
// this is execrable, and should NOT have had to've been done now, but...
//
extern const char *CG_DisplayBoxedText( int iBoxX, int iBoxY, int iBoxWidth, int iBoxHeight,
const char *psText, int iFontHandle, float fScale,
const vec4_t v4Color);
extern int giLinesOutput;
extern float gfAdvanceHack;
gfAdvanceHack = 1.0f; // override internal vertical advance
y = objectiveStartingYpos + (iYPixelsPerLine * missionYcnt);
// Advance line if a graphic has printed
for (i=0;i objectiveTextBoxWidth )
{ //Reached max length of this line
//step back until we find a space
while ((charLen>10) && (*str != ' ' ))
{
--str;
--charLen;
}
if (*str==' ')
{
++str; // To get past space
}
assert( charLengent->client->sess.mission_objectives[i].display)
{
// Calculate the Y position
totalY = objectiveStartingYpos + (iYPixelsPerLine * (missionYcnt))+(iYPixelsPerLine/2);
// Draw graphics that show if mission has been accomplished or not
cgi_R_SetColor(colorTable[CT_BLUE3]);
CG_DrawPic( (graphicXpos), (totalY-graphicYOffset), graphic_size, graphic_size, cgs.media.messageObjCircle); // Circle in front
if (cent->gent->client->sess.mission_objectives[i].status == OBJECTIVE_STAT_SUCCEEDED)
{
CG_DrawPic( (graphicXpos), (totalY-graphicYOffset), graphic_size, graphic_size, cgs.media.messageLitOn); // Center Dot
}
// Print current objective text
ObjectivePrint_Line(CT_WHITE, i, missionYcnt );
}
}
// No mission text?
if (!missionYcnt)
{
// Set the message a quarter of the way down and in the center of the text box
int messageYPosition = objectiveStartingYpos + (objectiveTextBoxHeight / 4);
cgi_SP_GetStringTextString( "SP_INGAME_OBJNONE", text, sizeof(text) );
int messageXPosition = objectiveStartingXpos + (objectiveTextBoxWidth/2) - (cgi_R_Font_StrLenPixels(text, cgs.media.qhFontMedium, 1.0f) /2);
cgi_R_Font_DrawString (
messageXPosition,
messageYPosition,
text,
colorTable[CT_WHITE],
cgs.media.qhFontMedium,
-1,
1.0f);
}
}
/*
//-------------------------------------------------------
static void CG_DrawForceCount( const int force, int x, float *y, const int pad,qboolean *hasForcePowers )
{
char s[MAX_STRING_CHARS];
int val, textColor;
char text[1024]={0};
gi.Cvar_VariableStringBuffer( va("playerfplvl%d", force ),s, sizeof(s) );
sscanf( s, "%d",&val );
if ((val<1) || (val> NUM_FORCE_POWERS))
{
return;
}
textColor = CT_ICON_BLUE;
// Draw title
cgi_SP_GetStringTextString( showLoadPowersName[force], text, sizeof(text) );
CG_DrawProportionalString( x, *y, text, CG_BIGFONT, colorTable[textColor] );
// Draw icons
cgi_R_SetColor( colorTable[CT_WHITE]);
const int iconSize = 30;
if ( val >= 0 )
{
x -= 10; // Back up from title a little
for ( int i = 0; i < val; i++ )
{
CG_DrawPic( x - iconSize - i * (iconSize + 10) , *y, iconSize, iconSize, force_icons[force] );
}
}
*y += pad;
*hasForcePowers = qtrue;
}
/ *
====================
CG_LoadScreen_PersonalInfo
====================
*/
/*
static void CG_LoadScreen_PersonalInfo(void)
{
float x, y;
int pad = 25;
char text[1024]={0};
qboolean hasForcePowers;
y = 65 + 30;
pad = 28;
x = 300;
hasForcePowers=qfalse;
CG_DrawForceCount( FP_HEAL, x, &y, pad,&hasForcePowers);
CG_DrawForceCount( FP_LEVITATION, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_SPEED, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_PUSH, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_PULL, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_TELEPATHY, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_GRIP, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_LIGHTNING, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_SABERTHROW, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_SABER_OFFENSE, x, &y, pad,&hasForcePowers );
CG_DrawForceCount( FP_SABER_DEFENSE, x, &y, pad,&hasForcePowers );
if (hasForcePowers)
{
cgi_SP_GetStringTextString( "SP_INGAME_CURRENTFORCEPOWERS", text, sizeof(text) );
CG_DrawProportionalString( 200, 65, text, CG_CENTER | CG_BIGFONT, colorTable[CT_WHITE] );
}
else
{ //you are only totally empty on the very first map?
// cgi_SP_GetStringTextString( "SP_INGAME_NONE", text, sizeof(text) );
// CG_DrawProportionalString( 320, y+30, text, CG_CENTER | CG_BIGFONT, colorTable[CT_ICON_BLUE] );
cgi_SP_GetStringTextString( "SP_INGAME_ALONGTIME", text, sizeof(text) );
int w = cgi_R_Font_StrLenPixels(text,cgs.media.qhFontMedium, 1.5f);
cgi_R_Font_DrawString((320)-(w/2), y+40, text, colorTable[CT_ICON_BLUE], cgs.media.qhFontMedium, -1, 1.5f);
}
}
*/
static void CG_LoadBar(void)
{
const int numticks = 9, tickwidth = 40, tickheight = 8;
const int tickpadx = 20, tickpady = 12;
const int capwidth = 8;
const int barwidth = numticks*tickwidth+tickpadx*2+capwidth*2, barleft = ((640-barwidth)/2);
const int barheight = tickheight + tickpady*2, bartop = 475-barheight;
const int capleft = barleft+tickpadx, tickleft = capleft+capwidth, ticktop = bartop+tickpady;
cgi_R_SetColor( colorTable[CT_WHITE]);
// Draw background
CG_DrawPic(barleft, bartop, barwidth, barheight, cgs.media.levelLoad);
// Draw left cap (backwards)
CG_DrawPic(tickleft, ticktop, -capwidth, tickheight, cgs.media.loadTickCap);
// Draw bar
CG_DrawPic(tickleft, ticktop, tickwidth*cg.loadLCARSStage, tickheight, cgs.media.loadTick);
// Draw right cap
CG_DrawPic(tickleft+tickwidth*cg.loadLCARSStage, ticktop, capwidth, tickheight, cgs.media.loadTickCap);
}
int CG_WeaponCheck( int weaponIndex );
// For printing load screen icons
const int MAXLOADICONSPERROW = 8; // Max icons displayed per row
const int MAXLOADWEAPONS = 16;
const int MAXLOAD_FORCEICONSIZE = 40; // Size of force power icons
const int MAXLOAD_FORCEICONPAD = 12; // Padding space between icons
static int CG_DrawLoadWeaponsPrintRow( const char *itemName, int weaponsBits,int rowIconCnt, int startIndex)
{
int i,endIndex=0, printedIconCnt=0;
int iconSize;
int holdX,x,y,pad;
int yOffset = 0;
int width,height;
vec4_t color;
qhandle_t background;
if (!cgi_UI_GetMenuItemInfo(
"loadScreen",
itemName,
&x,
&y,
&width,
&height,
color,
&background))
{
return(0);
}
cgi_R_SetColor( color );
iconSize = 60;
pad = 12;
// calculate placement of weapon icons
holdX = x + (width - ((iconSize*rowIconCnt) + (pad * (rowIconCnt-1))))/2;
for (i=startIndex;iweaponIconNoAmmo );
// }
// else
{
CG_DrawPic( holdX, y+yOffset, iconSize, iconSize, weaponInfo->weaponIcon );
}
printedIconCnt++;
if (printedIconCnt==MAXLOADICONSPERROW)
{
break;
}
holdX += (iconSize+pad);
}
}
return (endIndex);
}
// Print weapons the player is carrying
// Two rows print if there are too many
static void CG_DrawLoadWeapons( int weaponBits )
{
int i,endIndex=0;
int iconCnt,rowIconCnt;
// count the number of weapons owned
iconCnt = 0;
for ( i = 1 ; i < MAXLOADWEAPONS ; i++ )
{
if ( weaponBits & ( 1 << i ) )
{
iconCnt++;
}
}
if (!iconCnt) // If no weapons, don't display
{
return;
}
// Single line of icons
if (iconCnt<=MAXLOADICONSPERROW)
{
CG_DrawLoadWeaponsPrintRow("weaponicons_singlerow", weaponBits, iconCnt,0);
}
// Two lines of icons
else
{
// Print top row
endIndex = CG_DrawLoadWeaponsPrintRow("weaponicons_row1", weaponBits, MAXLOADICONSPERROW,0);
// Print second row
rowIconCnt = iconCnt - MAXLOADICONSPERROW;
CG_DrawLoadWeaponsPrintRow("weaponicons_row2", weaponBits, rowIconCnt,endIndex+1);
}
cgi_R_SetColor( NULL );
}
static int CG_DrawLoadForcePrintRow( const char *itemName, int forceBits,int rowIconCnt, int startIndex)
{
int i,endIndex=0, printedIconCnt=0;
int holdX,x,y;
int yOffset = 0;
int width,height;
vec4_t color;
qhandle_t background;
if (!cgi_UI_GetMenuItemInfo(
"loadScreen",
itemName,
&x,
&y,
&width,
&height,
color,
&background))
{
return(0);
}
cgi_R_SetColor( color );
// calculate placement of weapon icons
holdX = x + (width - ((MAXLOAD_FORCEICONSIZE*rowIconCnt) + (MAXLOAD_FORCEICONPAD * (rowIconCnt-1))))/2;
for (i=startIndex;ips.stats[STAT_HEALTH],
&iDummy, // &client->ps.stats[STAT_ARMOR],
&*weaponBits,// &client->ps.stats[STAT_WEAPONS],
&iDummy, // &client->ps.stats[STAT_ITEMS],
&iDummy, // &client->ps.weapon,
&iDummy, // &client->ps.weaponstate,
&iDummy, // &client->ps.batteryCharge,
&fDummy, // &client->ps.viewangles[0],
&fDummy, // &client->ps.viewangles[1],
&fDummy, // &client->ps.viewangles[2],
//force power data
&*forceBits, // &client->ps.forcePowersKnown,
&iDummy // &client->ps.forcePower,
);
}
// the new JK2 stuff - force powers, etc...
//
gi.Cvar_VariableStringBuffer( "playerfplvl", s, sizeof(s) );
i=0;
var = strtok( s, " " );
while( var != NULL )
{
/* While there are tokens in "s" */
loadForcePowerLevel[i++] = atoi(var);
/* Get next token: */
var = strtok( NULL, " " );
}
}
/*
====================
CG_DrawLoadingScreen
Load screen displays the map pic, the mission briefing and weapons/force powers
====================
*/
static void CG_DrawLoadingScreen( qhandle_t levelshot ,const char *mapName)
{
int xPos,yPos,width,height;
vec4_t color;
qhandle_t background;
int weapons=0, forcepowers=0;
// Get mission briefing for load screen
if (cgi_SP_GetStringTextString( va("BRIEFINGS_%s",mapName), NULL, 0 ) == 0)
{
cgi_Cvar_Set( "ui_missionbriefing", "@BRIEFINGS_NONE" );
}
else
{
cgi_Cvar_Set( "ui_missionbriefing", va("@BRIEFINGS_%s",mapName) );
}
// Print background
if (cgi_UI_GetMenuItemInfo(
"loadScreen",
"background",
&xPos,
&yPos,
&width,
&height,
color,
&background))
{
cgi_R_SetColor( color );
CG_DrawPic( xPos, yPos, width, height, background );
}
// Print level pic
if (cgi_UI_GetMenuItemInfo(
"loadScreen",
"mappic",
&xPos,
&yPos,
&width,
&height,
color,
&background))
{
//if (!levelshot)
//{// No level shot so use screenshot.
// CG_DrawPic( xPos, yPos, 1, 1, 0); //force the tess to flush
// cgi_R_DrawScreenShot( xPos, yPos+height, width, -height );
//}
//else
{
cgi_R_SetColor( color );
CG_DrawPic( xPos, yPos, width, height, levelshot );
}
}
// Get player weapons and force power info
CG_GetLoadScreenInfo(&weapons,&forcepowers);
// Print weapon icons
if (weapons)
{
CG_DrawLoadWeapons(weapons);
}
// Print force power icons
if (forcepowers)
{
CG_DrawLoadForcePowers(forcepowers);
}
}
/*
====================
CG_DrawInformation
Draw all the status / pacifier stuff during level loading
====================
*/
void CG_DrawInformation( void ) {
int y;
// draw the dialog background
const char *info = CG_ConfigString( CS_SERVERINFO );
const char *s = Info_ValueForKey( info, "mapname" );
qhandle_t levelshot;
extern SavedGameJustLoaded_e g_eSavedGameJustLoaded; // hack! (hey, it's the last week of coding, ok?
// if ( g_eSavedGameJustLoaded == eFULL )
// {
// levelshot = 0; //use the loaded thumbnail instead of the levelshot
// }
// else
{
levelshot = cgi_R_RegisterShaderNoMip( va( "levelshots/%s", s ) );
#ifndef FINAL_BUILD
if (!levelshot && !strncmp(s, "work/",5) )
{
levelshot = cgi_R_RegisterShaderNoMip( va( "levelshots/%s", s+5 ) );
}
#endif
if (!levelshot) {
levelshot = cgi_R_RegisterShaderNoMip( "menu/art/unknownmap" );
}
}
if ( g_eSavedGameJustLoaded != eFULL && !strcmp(s,"yavin1") )//special case for first map!
{
char text[1024]={0};
//
cgi_R_SetColor( colorTable[CT_BLACK] );
CG_DrawPic( 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, cgs.media.whiteShader );
cgi_SP_GetStringTextString( "SP_INGAME_ALONGTIME", text, sizeof(text) );
int w = cgi_R_Font_StrLenPixels(text,cgs.media.qhFontMedium, 1.0f);
cgi_R_Font_DrawString((320)-(w/2), 140, text, colorTable[CT_ICON_BLUE], cgs.media.qhFontMedium, -1, 1.0f);
}
else
{
CG_DrawLoadingScreen(levelshot, s);
cgi_UI_Menu_Paint( cgi_UI_GetMenuByName( "loadscreen" ), qtrue );
//cgi_UI_MenuPaintAll();
}
CG_LoadBar();
// the first 150 rows are reserved for the client connection
// screen to write into
// if ( cg.processedSnapshotNum == 0 )
{
// still loading
// print the current item being loaded
#ifdef _DEBUG
cgi_R_Font_DrawString( 40, 416, va("LOADING ... %s",cg.infoScreenText),colorTable[CT_LTGOLD1], cgs.media.qhFontSmall, -1, 1.0f );
#endif
}
// draw info string information
y = 20;
// map-specific message (long map name)
s = CG_ConfigString( CS_MESSAGE );
if ( s[0] )
{
if (s[0] == '@')
{
char text[1024]={0};
cgi_SP_GetStringTextString( s+1, text, sizeof(text) );
cgi_R_Font_DrawString( 15, y, va("\"%s\"",text),colorTable[CT_WHITE],cgs.media.qhFontMedium, -1, 1.0f );
}
else
{
cgi_R_Font_DrawString( 15, y, va("\"%s\"",s),colorTable[CT_WHITE],cgs.media.qhFontMedium, -1, 1.0f );
}
y += 20;
}
}