jkxr/Projects/Android/jni/OpenJK/codemp/cgame/cg_saga.c

1101 lines
24 KiB
C

/*
===========================================================================
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 <http://www.gnu.org/licenses/>.
===========================================================================
*/
/*****************************************************************************
* name: cg_siege.c
*
* desc: Clientgame-side module for Siege gametype.
*
* $Author: osman $
* $Revision: 1.5 $
*
*****************************************************************************/
#include "cg_local.h"
#include "game/bg_saga.h"
int cgSiegeRoundState = 0;
int cgSiegeRoundTime = 0;
static char team1[512];
static char team2[512];
int team1Timed = 0;
int team2Timed = 0;
int cgSiegeTeam1PlShader = 0;
int cgSiegeTeam2PlShader = 0;
static char cgParseObjectives[MAX_SIEGE_INFO_SIZE];
extern void CG_LoadCISounds(clientInfo_t *ci, qboolean modelloaded); //cg_players.c
void CG_DrawSiegeMessage( const char *str, int objectiveScreen );
void CG_DrawSiegeMessageNonMenu( const char *str );
void CG_SiegeBriefingDisplay(int team, int dontshow);
void CG_PrecacheSiegeObjectiveAssetsForTeam(int myTeam)
{
char teamstr[64];
char objstr[256];
char foundobjective[MAX_SIEGE_INFO_SIZE];
if (!siege_valid)
{
trap->Error( ERR_DROP, "Siege data does not exist on client!\n");
return;
}
if (myTeam == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{
int i = 1;
while (i < 32)
{ //eh, just try 32 I guess
Com_sprintf(objstr, sizeof(objstr), "Objective%i", i);
if (BG_SiegeGetValueGroup(cgParseObjectives, objstr, foundobjective))
{
char str[MAX_QPATH];
if (BG_SiegeGetPairedValue(foundobjective, "sound_team1", str))
{
trap->S_RegisterSound(str);
}
if (BG_SiegeGetPairedValue(foundobjective, "sound_team2", str))
{
trap->S_RegisterSound(str);
}
if (BG_SiegeGetPairedValue(foundobjective, "objgfx", str))
{
trap->R_RegisterShaderNoMip(str);
}
if (BG_SiegeGetPairedValue(foundobjective, "mapicon", str))
{
trap->R_RegisterShaderNoMip(str);
}
if (BG_SiegeGetPairedValue(foundobjective, "litmapicon", str))
{
trap->R_RegisterShaderNoMip(str);
}
if (BG_SiegeGetPairedValue(foundobjective, "donemapicon", str))
{
trap->R_RegisterShaderNoMip(str);
}
}
else
{ //no more
break;
}
i++;
}
}
}
void CG_PrecachePlayersForSiegeTeam(int team)
{
siegeTeam_t *stm;
int i = 0;
stm = BG_SiegeFindThemeForTeam(team);
if (!stm)
{ //invalid team/no theme for team?
return;
}
while (i < stm->numClasses)
{
siegeClass_t *scl = stm->classes[i];
if (scl->forcedModel[0])
{
clientInfo_t fake;
memset(&fake, 0, sizeof(fake));
Q_strncpyz(fake.modelName, scl->forcedModel, sizeof(fake.modelName));
trap->R_RegisterModel(va("models/players/%s/model.glm", scl->forcedModel));
if (scl->forcedSkin[0])
{
trap->R_RegisterSkin(va("models/players/%s/model_%s.skin", scl->forcedModel, scl->forcedSkin));
Q_strncpyz(fake.skinName, scl->forcedSkin, sizeof(fake.modelName));
}
else
{
Q_strncpyz(fake.skinName, "default", sizeof(fake.skinName));
}
//precache the sounds for the model...
CG_LoadCISounds(&fake, qtrue);
}
i++;
}
}
void CG_InitSiegeMode(void)
{
char levelname[MAX_QPATH];
char btime[1024];
char teams[2048];
char teamInfo[MAX_SIEGE_INFO_SIZE];
int len = 0;
int i = 0;
int j = 0;
siegeClass_t *cl;
siegeTeam_t *sTeam;
fileHandle_t f;
char teamIcon[128];
if (cgs.gametype != GT_SIEGE)
{
goto failure;
}
Com_sprintf(levelname, sizeof(levelname), "%s.siege", cgs.rawmapname);
if (!levelname[0])
{
goto failure;
}
len = trap->FS_Open(levelname, &f, FS_READ);
if ( !f ) {
goto failure;
}
if ( len >= MAX_SIEGE_INFO_SIZE ) {
trap->FS_Close( f );
goto failure;
}
trap->FS_Read(siege_info, len, f);
trap->FS_Close(f);
siege_valid = 1;
if (BG_SiegeGetValueGroup(siege_info, "Teams", teams))
{
char buf[1024];
trap->Cvar_VariableStringBuffer("cg_siegeTeam1", buf, 1024);
if (buf[0] && Q_stricmp(buf, "none"))
{
Q_strncpyz(team1, buf, sizeof(team1));
}
else
{
BG_SiegeGetPairedValue(teams, "team1", team1);
}
if (team1[0] == '@')
{ //it's a damn stringed reference.
char b[256];
trap->SE_GetStringTextString(team1+1, b, 256);
trap->Cvar_Set("cg_siegeTeam1Name", b);
}
else
{
trap->Cvar_Set("cg_siegeTeam1Name", team1);
}
trap->Cvar_VariableStringBuffer("cg_siegeTeam2", buf, 1024);
if (buf[0] && Q_stricmp(buf, "none"))
{
Q_strncpyz(team2, buf, sizeof(team2));
}
else
{
BG_SiegeGetPairedValue(teams, "team2", team2);
}
if (team2[0] == '@')
{ //it's a damn stringed reference.
char b[256];
trap->SE_GetStringTextString(team2+1, b, 256);
trap->Cvar_Set("cg_siegeTeam2Name", b);
}
else
{
trap->Cvar_Set("cg_siegeTeam2Name", team2);
}
}
else
{
trap->Error( ERR_DROP, "Siege teams not defined");
}
if (BG_SiegeGetValueGroup(siege_info, team1, teamInfo))
{
if (BG_SiegeGetPairedValue(teamInfo, "TeamIcon", teamIcon))
{
trap->Cvar_Set( "team1_icon", teamIcon);
}
if (BG_SiegeGetPairedValue(teamInfo, "Timed", btime))
{
team1Timed = atoi(btime)*1000;
CG_SetSiegeTimerCvar ( team1Timed );
}
else
{
team1Timed = 0;
}
}
else
{
trap->Error( ERR_DROP, "No team entry for '%s'\n", team1);
}
if (BG_SiegeGetPairedValue(siege_info, "mapgraphic", teamInfo))
{
trap->Cvar_Set("siege_mapgraphic", teamInfo);
}
else
{
trap->Cvar_Set("siege_mapgraphic", "gfx/mplevels/siege1_hoth");
}
if (BG_SiegeGetPairedValue(siege_info, "missionname", teamInfo))
{
trap->Cvar_Set("siege_missionname", teamInfo);
}
else
{
trap->Cvar_Set("siege_missionname", " ");
}
if (BG_SiegeGetValueGroup(siege_info, team2, teamInfo))
{
if (BG_SiegeGetPairedValue(teamInfo, "TeamIcon", teamIcon))
{
trap->Cvar_Set( "team2_icon", teamIcon);
}
if (BG_SiegeGetPairedValue(teamInfo, "Timed", btime))
{
team2Timed = atoi(btime)*1000;
CG_SetSiegeTimerCvar ( team2Timed );
}
else
{
team2Timed = 0;
}
}
else
{
trap->Error( ERR_DROP, "No team entry for '%s'\n", team2);
}
//Load the player class types
BG_SiegeLoadClasses(NULL);
if (!bgNumSiegeClasses)
{ //We didn't find any?!
trap->Error( ERR_DROP, "Couldn't find any player classes for Siege");
}
//Now load the teams since we have class data.
BG_SiegeLoadTeams();
if (!bgNumSiegeTeams)
{ //React same as with classes.
trap->Error( ERR_DROP, "Couldn't find any player teams for Siege");
}
//Get and set the team themes for each team. This will control which classes can be
//used on each team.
if (BG_SiegeGetValueGroup(siege_info, team1, teamInfo))
{
if (BG_SiegeGetPairedValue(teamInfo, "UseTeam", btime))
{
BG_SiegeSetTeamTheme(SIEGETEAM_TEAM1, btime);
}
if (BG_SiegeGetPairedValue(teamInfo, "FriendlyShader", btime))
{
cgSiegeTeam1PlShader = trap->R_RegisterShaderNoMip(btime);
}
else
{
cgSiegeTeam1PlShader = 0;
}
}
if (BG_SiegeGetValueGroup(siege_info, team2, teamInfo))
{
if (BG_SiegeGetPairedValue(teamInfo, "UseTeam", btime))
{
BG_SiegeSetTeamTheme(SIEGETEAM_TEAM2, btime);
}
if (BG_SiegeGetPairedValue(teamInfo, "FriendlyShader", btime))
{
cgSiegeTeam2PlShader = trap->R_RegisterShaderNoMip(btime);
}
else
{
cgSiegeTeam2PlShader = 0;
}
}
//Now go through the classes used by the loaded teams and try to precache
//any forced models or forced skins.
i = SIEGETEAM_TEAM1;
while (i <= SIEGETEAM_TEAM2)
{
j = 0;
sTeam = BG_SiegeFindThemeForTeam(i);
if (!sTeam)
{
i++;
continue;
}
//Get custom team shaders while we're at it.
if (i == SIEGETEAM_TEAM1)
{
cgSiegeTeam1PlShader = sTeam->friendlyShader;
}
else if (i == SIEGETEAM_TEAM2)
{
cgSiegeTeam2PlShader = sTeam->friendlyShader;
}
while (j < sTeam->numClasses)
{
cl = sTeam->classes[j];
if (cl->forcedModel[0])
{ //This class has a forced model, so precache it.
trap->R_RegisterModel(va("models/players/%s/model.glm", cl->forcedModel));
if (cl->forcedSkin[0])
{ //also has a forced skin, precache it.
char *useSkinName;
if (strchr(cl->forcedSkin, '|'))
{//three part skin
useSkinName = va("models/players/%s/|%s", cl->forcedModel, cl->forcedSkin);
}
else
{
useSkinName = va("models/players/%s/model_%s.skin", cl->forcedModel, cl->forcedSkin);
}
trap->R_RegisterSkin(useSkinName);
}
}
j++;
}
i++;
}
//precache saber data for classes that use sabers on both teams
BG_PrecacheSabersForSiegeTeam(SIEGETEAM_TEAM1);
BG_PrecacheSabersForSiegeTeam(SIEGETEAM_TEAM2);
CG_PrecachePlayersForSiegeTeam(SIEGETEAM_TEAM1);
CG_PrecachePlayersForSiegeTeam(SIEGETEAM_TEAM2);
CG_PrecachePlayersForSiegeTeam(SIEGETEAM_TEAM1);
CG_PrecachePlayersForSiegeTeam(SIEGETEAM_TEAM2);
CG_PrecacheSiegeObjectiveAssetsForTeam(SIEGETEAM_TEAM1);
CG_PrecacheSiegeObjectiveAssetsForTeam(SIEGETEAM_TEAM2);
return;
failure:
siege_valid = 0;
}
static char QINLINE *CG_SiegeObjectiveBuffer(int team, int objective)
{
static char buf[8192];
char teamstr[1024];
if (team == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{ //found the team group
if (BG_SiegeGetValueGroup(cgParseObjectives, va("Objective%i", objective), buf))
{ //found the objective group
return buf;
}
}
return NULL;
}
void CG_ParseSiegeObjectiveStatus(const char *str)
{
int i = 0;
int team = SIEGETEAM_TEAM1;
char *cvarName;
char *s;
int objectiveNum = 0;
if (!str || !str[0])
{
return;
}
while (str[i])
{
if (str[i] == '|')
{ //switch over to team2, this is the next section
team = SIEGETEAM_TEAM2;
objectiveNum = 0;
}
else if (str[i] == '-')
{
objectiveNum++;
i++;
cvarName = va("team%i_objective%i", team, objectiveNum);
if (str[i] == '1')
{ //it's completed
trap->Cvar_Set(cvarName, "1");
}
else
{ //otherwise assume it is not
trap->Cvar_Set(cvarName, "0");
}
s = CG_SiegeObjectiveBuffer(team, objectiveNum);
if (s && s[0])
{ //now set the description and graphic cvars to by read by the menu
char buffer[8192];
cvarName = va("team%i_objective%i_longdesc", team, objectiveNum);
if (BG_SiegeGetPairedValue(s, "objdesc", buffer))
{
trap->Cvar_Set(cvarName, buffer);
}
else
{
trap->Cvar_Set(cvarName, "UNSPECIFIED");
}
cvarName = va("team%i_objective%i_gfx", team, objectiveNum);
if (BG_SiegeGetPairedValue(s, "objgfx", buffer))
{
trap->Cvar_Set(cvarName, buffer);
}
else
{
trap->Cvar_Set(cvarName, "UNSPECIFIED");
}
cvarName = va("team%i_objective%i_mapicon", team, objectiveNum);
if (BG_SiegeGetPairedValue(s, "mapicon", buffer))
{
trap->Cvar_Set(cvarName, buffer);
}
else
{
trap->Cvar_Set(cvarName, "UNSPECIFIED");
}
cvarName = va("team%i_objective%i_litmapicon", team, objectiveNum);
if (BG_SiegeGetPairedValue(s, "litmapicon", buffer))
{
trap->Cvar_Set(cvarName, buffer);
}
else
{
trap->Cvar_Set(cvarName, "UNSPECIFIED");
}
cvarName = va("team%i_objective%i_donemapicon", team, objectiveNum);
if (BG_SiegeGetPairedValue(s, "donemapicon", buffer))
{
trap->Cvar_Set(cvarName, buffer);
}
else
{
trap->Cvar_Set(cvarName, "UNSPECIFIED");
}
cvarName = va("team%i_objective%i_mappos", team, objectiveNum);
if (BG_SiegeGetPairedValue(s, "mappos", buffer))
{
trap->Cvar_Set(cvarName, buffer);
}
else
{
trap->Cvar_Set(cvarName, "0 0 32 32");
}
}
}
i++;
}
if (cg.predictedPlayerState.persistant[PERS_TEAM] != TEAM_SPECTATOR)
{ //update menu cvars
CG_SiegeBriefingDisplay(cg.predictedPlayerState.persistant[PERS_TEAM], 1);
}
}
void CG_SiegeRoundOver(centity_t *ent, int won)
{
int myTeam;
char teamstr[64];
char appstring[1024];
char soundstr[1024];
int success = 0;
playerState_t *ps = NULL;
if (!siege_valid)
{
trap->Error( ERR_DROP, "ERROR: Siege data does not exist on client!\n");
return;
}
if (cg.snap)
{ //this should always be true, if it isn't though use the predicted ps as a fallback
ps = &cg.snap->ps;
}
else
{
ps = &cg.predictedPlayerState;
}
if (!ps)
{
assert(0);
return;
}
myTeam = ps->persistant[PERS_TEAM];
if (myTeam == TEAM_SPECTATOR)
{
return;
}
if (myTeam == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{
if (won == myTeam)
{
success = BG_SiegeGetPairedValue(cgParseObjectives, "wonround", appstring);
}
else
{
success = BG_SiegeGetPairedValue(cgParseObjectives, "lostround", appstring);
}
if (success)
{
CG_DrawSiegeMessage(appstring, 0);
}
appstring[0] = 0;
soundstr[0] = 0;
if (myTeam == won)
{
Com_sprintf(teamstr, sizeof(teamstr), "roundover_sound_wewon");
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), "roundover_sound_welost");
}
if (BG_SiegeGetPairedValue(cgParseObjectives, teamstr, appstring))
{
Com_sprintf(soundstr, sizeof(soundstr), appstring);
}
/*
else
{
if (myTeam != won)
{
Com_sprintf(soundstr, sizeof(soundstr), DEFAULT_LOSE_ROUND);
}
else
{
Com_sprintf(soundstr, sizeof(soundstr), DEFAULT_WIN_ROUND);
}
}
*/
if (soundstr[0])
{
trap->S_StartLocalSound(trap->S_RegisterSound(soundstr), CHAN_ANNOUNCER);
}
}
}
void CG_SiegeGetObjectiveDescription(int team, int objective, char *buffer)
{
char teamstr[1024];
char objectiveStr[8192];
buffer[0] = 0; //set to 0 ahead of time in case we fail to find the objective group/name
if (team == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{ //found the team group
if (BG_SiegeGetValueGroup(cgParseObjectives, va("Objective%i", objective), objectiveStr))
{ //found the objective group
//Parse the name right into the buffer.
BG_SiegeGetPairedValue(objectiveStr, "goalname", buffer);
}
}
}
int CG_SiegeGetObjectiveFinal(int team, int objective )
{
char finalStr[64];
char teamstr[1024];
char objectiveStr[8192];
if (team == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{ //found the team group
if (BG_SiegeGetValueGroup(cgParseObjectives, va("Objective%i", objective), objectiveStr))
{ //found the objective group
//Parse the name right into the buffer.
BG_SiegeGetPairedValue(objectiveStr, "final", finalStr);
return (atoi( finalStr ));
}
}
return 0;
}
void CG_SiegeBriefingDisplay(int team, int dontshow)
{
char teamstr[64];
char briefing[8192];
char properValue[1024];
char objectiveDesc[1024];
int i = 1;
int useTeam = team;
qboolean primary = qfalse;
if (!siege_valid)
{
return;
}
if (team == TEAM_SPECTATOR)
{
return;
}
if (team == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (useTeam != SIEGETEAM_TEAM1 && useTeam != SIEGETEAM_TEAM2)
{ //This shouldn't be happening. But just fall back to team 2 anyway.
useTeam = SIEGETEAM_TEAM2;
}
trap->Cvar_Set(va("siege_primobj_inuse"), "0");
while (i < 16)
{ //do up to 16 objectives I suppose
//Get the value for this objective on this team
//Now set the cvar for the menu to display.
//primary = (CG_SiegeGetObjectiveFinal(useTeam, i)>-1)?qtrue:qfalse;
primary = (CG_SiegeGetObjectiveFinal(useTeam, i)>0)?qtrue:qfalse;
properValue[0] = 0;
trap->Cvar_VariableStringBuffer(va("team%i_objective%i", useTeam, i), properValue, 1024);
if (primary)
{
trap->Cvar_Set(va("siege_primobj"), properValue);
}
else
{
trap->Cvar_Set(va("siege_objective%i", i), properValue);
}
//Now set the long desc cvar for the menu to display.
properValue[0] = 0;
trap->Cvar_VariableStringBuffer(va("team%i_objective%i_longdesc", useTeam, i), properValue, 1024);
if (primary)
{
trap->Cvar_Set(va("siege_primobj_longdesc"), properValue);
}
else
{
trap->Cvar_Set(va("siege_objective%i_longdesc", i), properValue);
}
//Now set the gfx cvar for the menu to display.
properValue[0] = 0;
trap->Cvar_VariableStringBuffer(va("team%i_objective%i_gfx", useTeam, i), properValue, 1024);
if (primary)
{
trap->Cvar_Set(va("siege_primobj_gfx"), properValue);
}
else
{
trap->Cvar_Set(va("siege_objective%i_gfx", i), properValue);
}
//Now set the mapicon cvar for the menu to display.
properValue[0] = 0;
trap->Cvar_VariableStringBuffer(va("team%i_objective%i_mapicon", useTeam, i), properValue, 1024);
if (primary)
{
trap->Cvar_Set(va("siege_primobj_mapicon"), properValue);
}
else
{
trap->Cvar_Set(va("siege_objective%i_mapicon", i), properValue);
}
//Now set the mappos cvar for the menu to display.
properValue[0] = 0;
trap->Cvar_VariableStringBuffer(va("team%i_objective%i_mappos", useTeam, i), properValue, 1024);
if (primary)
{
trap->Cvar_Set(va("siege_primobj_mappos"), properValue);
}
else
{
trap->Cvar_Set(va("siege_objective%i_mappos", i), properValue);
}
//Now set the description cvar for the objective
CG_SiegeGetObjectiveDescription(useTeam, i, objectiveDesc);
if (objectiveDesc[0])
{ //found a valid objective description
if ( primary )
{
trap->Cvar_Set(va("siege_primobj_desc"), objectiveDesc);
//this one is marked not in use because it gets primobj
trap->Cvar_Set(va("siege_objective%i_inuse", i), "0");
trap->Cvar_Set(va("siege_primobj_inuse"), "1");
trap->Cvar_Set(va("team%i_objective%i_inuse", useTeam, i), "1");
}
else
{
trap->Cvar_Set(va("siege_objective%i_desc", i), objectiveDesc);
trap->Cvar_Set(va("siege_objective%i_inuse", i), "2");
trap->Cvar_Set(va("team%i_objective%i_inuse", useTeam, i), "2");
}
}
else
{ //didn't find one, so set the "inuse" cvar to 0 for the objective and mark it non-complete.
trap->Cvar_Set(va("siege_objective%i_inuse", i), "0");
trap->Cvar_Set(va("siege_objective%i", i), "0");
trap->Cvar_Set(va("team%i_objective%i_inuse", useTeam, i), "0");
trap->Cvar_Set(va("team%i_objective%i", useTeam, i), "0");
trap->Cvar_Set(va("siege_objective%i_mappos", i), "");
trap->Cvar_Set(va("team%i_objective%i_mappos", useTeam, i), "");
trap->Cvar_Set(va("siege_objective%i_gfx", i), "");
trap->Cvar_Set(va("team%i_objective%i_gfx", useTeam, i), "");
trap->Cvar_Set(va("siege_objective%i_mapicon", i), "");
trap->Cvar_Set(va("team%i_objective%i_mapicon", useTeam, i), "");
}
i++;
}
if (dontshow)
{
return;
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{
if (BG_SiegeGetPairedValue(cgParseObjectives, "briefing", briefing))
{
CG_DrawSiegeMessage(briefing, 1);
}
}
}
void CG_SiegeObjectiveCompleted(centity_t *ent, int won, int objectivenum)
{
int myTeam;
char teamstr[64];
char objstr[256];
char foundobjective[MAX_SIEGE_INFO_SIZE];
char appstring[1024];
char soundstr[1024];
int success = 0;
playerState_t *ps = NULL;
if (!siege_valid)
{
trap->Error( ERR_DROP, "Siege data does not exist on client!\n");
return;
}
if (cg.snap)
{ //this should always be true, if it isn't though use the predicted ps as a fallback
ps = &cg.snap->ps;
}
else
{
ps = &cg.predictedPlayerState;
}
if (!ps)
{
assert(0);
return;
}
myTeam = ps->persistant[PERS_TEAM];
if (myTeam == TEAM_SPECTATOR)
{
return;
}
if (won == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), team1);
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), team2);
}
if (BG_SiegeGetValueGroup(siege_info, teamstr, cgParseObjectives))
{
Com_sprintf(objstr, sizeof(objstr), "Objective%i", objectivenum);
if (BG_SiegeGetValueGroup(cgParseObjectives, objstr, foundobjective))
{
if (myTeam == SIEGETEAM_TEAM1)
{
success = BG_SiegeGetPairedValue(foundobjective, "message_team1", appstring);
}
else
{
success = BG_SiegeGetPairedValue(foundobjective, "message_team2", appstring);
}
if (success)
{
CG_DrawSiegeMessageNonMenu(appstring);
}
appstring[0] = 0;
soundstr[0] = 0;
if (myTeam == SIEGETEAM_TEAM1)
{
Com_sprintf(teamstr, sizeof(teamstr), "sound_team1");
}
else
{
Com_sprintf(teamstr, sizeof(teamstr), "sound_team2");
}
if (BG_SiegeGetPairedValue(foundobjective, teamstr, appstring))
{
Com_sprintf(soundstr, sizeof(soundstr), appstring);
}
/*
else
{
if (myTeam != won)
{
Com_sprintf(soundstr, sizeof(soundstr), DEFAULT_LOSE_OBJECTIVE);
}
else
{
Com_sprintf(soundstr, sizeof(soundstr), DEFAULT_WIN_OBJECTIVE);
}
}
*/
if (soundstr[0])
{
trap->S_StartLocalSound(trap->S_RegisterSound(soundstr), CHAN_ANNOUNCER);
}
}
}
}
siegeExtended_t cg_siegeExtendedData[MAX_CLIENTS];
//parse a single extended siege data entry
void CG_ParseSiegeExtendedDataEntry(const char *conStr)
{
char s[MAX_STRING_CHARS];
char *str = (char *)conStr;
int argParses = 0;
int i;
int maxAmmo = 0, clNum = -1, health = 1, maxhealth = 1, ammo = 1;
centity_t *cent;
if (!conStr || !conStr[0])
{
return;
}
while (*str && argParses < 4)
{
i = 0;
while (*str && *str != '|')
{
s[i] = *str;
i++;
str++;
}
s[i] = 0;
switch (argParses)
{
case 0:
clNum = atoi(s);
break;
case 1:
health = atoi(s);
break;
case 2:
maxhealth = atoi(s);
break;
case 3:
ammo = atoi(s);
break;
default:
break;
}
argParses++;
str++;
}
if (clNum < 0 || clNum >= MAX_CLIENTS)
{
return;
}
cg_siegeExtendedData[clNum].health = health;
cg_siegeExtendedData[clNum].maxhealth = maxhealth;
cg_siegeExtendedData[clNum].ammo = ammo;
cent = &cg_entities[clNum];
maxAmmo = ammoData[weaponData[cent->currentState.weapon].ammoIndex].max;
if ( (cent->currentState.eFlags & EF_DOUBLE_AMMO) )
{
maxAmmo *= 2.0f;
}
if (ammo >= 0 && ammo <= maxAmmo )
{ //assure the weapon number is valid and not over max
//keep the weapon so if it changes before our next ext data update we'll know
//that the ammo is not applicable.
cg_siegeExtendedData[clNum].weapon = cent->currentState.weapon;
}
else
{ //not valid? Oh well, just invalidate the weapon too then so we don't display ammo
cg_siegeExtendedData[clNum].weapon = -1;
}
cg_siegeExtendedData[clNum].lastUpdated = cg.time;
}
//parse incoming siege data, see counterpart in g_saga.c
void CG_ParseSiegeExtendedData(void)
{
int numEntries = trap->Cmd_Argc();
int i = 0;
if (numEntries < 1)
{
assert(!"Bad numEntries for sxd");
return;
}
while (i < numEntries)
{
CG_ParseSiegeExtendedDataEntry(CG_Argv(i+1));
i++;
}
}
void CG_SetSiegeTimerCvar ( int msec )
{
int seconds;
int mins;
int tens;
seconds = msec / 1000;
mins = seconds / 60;
seconds -= mins * 60;
tens = seconds / 10;
seconds -= tens * 10;
trap->Cvar_Set("ui_siegeTimer", va( "%i:%i%i", mins, tens, seconds ) );
}