quakeforge/qw/source/teamplay.c

481 lines
10 KiB
C
Raw Normal View History

/*
teamplay.c
Teamplay enhancements ("proxy features")
Copyright (C) 2000 Anton Gavrilov (tonik@quake.ru)
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to:
Free Software Foundation, Inc.
59 Temple Place - Suite 330
Boston, MA 02111-1307, USA
*/
2001-09-28 06:26:31 +00:00
static const char rcsid[] =
"$Id$";
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#ifdef HAVE_STRING_H
# include <string.h>
#endif
#ifdef HAVE_STRINGS_H
# include <strings.h>
#endif
#include <errno.h>
#include <ctype.h>
#include "QF/console.h"
#include "QF/cmd.h"
2001-05-31 03:41:35 +00:00
#include "QF/cvar.h"
2001-05-09 17:57:57 +00:00
#include "QF/locs.h"
#include "QF/model.h"
#include "QF/sys.h"
#include "QF/teamplay.h"
#include "QF/va.h"
#include "QF/skin.h"
2001-05-09 17:57:57 +00:00
#include "bothdefs.h"
2001-10-29 17:46:03 +00:00
#include "cl_input.h"
2001-05-09 17:57:57 +00:00
#include "client.h"
2001-08-28 03:47:10 +00:00
#include "compat.h"
static qboolean died = false, recorded_location = false;
2001-08-28 03:47:10 +00:00
static vec3_t death_location, last_recorded_location;
cvar_t *cl_deadbodyfilter;
cvar_t *cl_gibfilter;
cvar_t *cl_parsesay;
cvar_t *cl_nofake;
cvar_t *cl_freply;
2001-08-28 03:47:10 +00:00
void
Team_BestWeaponImpulse (void)
{
int best, i, imp, items;
2001-08-28 03:47:10 +00:00
items = cl.stats[STAT_ITEMS];
best = 0;
for (i = Cmd_Argc () - 1; i > 0; i--) {
imp = atoi (Cmd_Argv (i));
if (imp < 1 || imp > 8)
continue;
switch (imp) {
2001-08-28 03:47:10 +00:00
case 1:
if (items & IT_AXE)
best = 1;
break;
case 2:
if (items & IT_SHOTGUN && cl.stats[STAT_SHELLS] >= 1)
best = 2;
break;
case 3:
if (items & IT_SUPER_SHOTGUN && cl.stats[STAT_SHELLS] >= 2)
best = 3;
break;
case 4:
if (items & IT_NAILGUN && cl.stats[STAT_NAILS] >= 1)
best = 4;
break;
case 5:
if (items & IT_SUPER_NAILGUN && cl.stats[STAT_NAILS] >= 2)
best = 5;
break;
case 6:
if (items & IT_GRENADE_LAUNCHER && cl.stats[STAT_ROCKETS] >= 1)
best = 6;
break;
case 7:
if (items & IT_ROCKET_LAUNCHER && cl.stats[STAT_ROCKETS] >= 1)
best = 7;
break;
case 8:
if (items & IT_LIGHTNING && cl.stats[STAT_CELLS] >= 1)
best = 8;
}
}
if (best)
in_impulse = best;
}
2001-07-15 07:04:17 +00:00
const char *
Team_ParseSay (const char *s)
{
char chr, t2[128], t3[128];
2001-07-15 07:04:17 +00:00
const char *t1;
2001-08-28 03:47:10 +00:00
static char buf[1024];
int i, bracket;
static location_t *location = NULL;
if (!cl_parsesay->int_val)
return s;
i = 0;
while (*s && (i <= sizeof (buf))) {
if ((*s == '%') && (s[1] != '\0')) {
t1 = NULL;
memset (t2, '\0', sizeof (t2));
memset (t3, '\0', sizeof (t3));
if ((s[1] == '[') && (s[3] == ']')) {
bracket = 1;
chr = s[2];
s += 4;
} else {
bracket = 0;
chr = s[1];
s += 2;
}
switch (chr) {
2001-08-28 03:47:10 +00:00
case '%':
t2[0] = '%';
t2[1] = 0;
t1 = t2;
break;
case 's':
bracket = 0;
t1 = skin->string;
break;
case 'd':
bracket = 0;
if (died) {
location = locs_find (death_location);
if (location) {
recorded_location = true;
2001-08-28 03:47:10 +00:00
VectorCopy (death_location, last_recorded_location);
t1 = location->name;
2001-08-28 03:47:10 +00:00
break;
}
}
goto location;
case 'r':
bracket = 0;
if (recorded_location) {
location = locs_find (last_recorded_location);
if (location) {
t1 = location->name;
break;
}
}
goto location;
case 'l':
location:
bracket = 0;
location = locs_find (cl.simorg);
if (location) {
recorded_location = true;
VectorCopy (cl.simorg, last_recorded_location);
t1 = location->name;
} else
snprintf (t2, sizeof (t2), "Unknown!\n");
break;
case 'a':
if (bracket) {
if (cl.stats[STAT_ARMOR] > 50)
bracket = 0;
if (cl.stats[STAT_ITEMS] & IT_ARMOR3)
2001-08-28 03:47:10 +00:00
t3[0] = 'R' | 0x80;
else if (cl.stats[STAT_ITEMS] & IT_ARMOR2)
2001-08-28 03:47:10 +00:00
t3[0] = 'Y' | 0x80;
else if (cl.stats[STAT_ITEMS] & IT_ARMOR1)
2001-08-28 03:47:10 +00:00
t3[0] = 'G' | 0x80;
else {
t2[0] = 'N' | 0x80;
t2[1] = 'O' | 0x80;
t2[2] = 'N' | 0x80;
t2[3] = 'E' | 0x80;
t2[4] = '!' | 0x80;
}
2001-08-28 03:47:10 +00:00
snprintf (t2, sizeof (t2), "%sa:%i", t3,
cl.stats[STAT_ARMOR]);
} else
snprintf (t2, sizeof (t2), "%i", cl.stats[STAT_ARMOR]);
break;
case 'A':
bracket = 0;
if (cl.stats[STAT_ITEMS] & IT_ARMOR3)
t2[0] = 'R' | 0x80;
else if (cl.stats[STAT_ITEMS] & IT_ARMOR2)
t2[0] = 'Y' | 0x80;
else if (cl.stats[STAT_ITEMS] & IT_ARMOR1)
t2[0] = 'G' | 0x80;
else {
t2[0] = 'N' | 0x80;
t2[1] = 'O' | 0x80;
t2[2] = 'N' | 0x80;
t2[3] = 'E' | 0x80;
t2[4] = '!' | 0x80;
}
break;
case 'h':
if (bracket) {
if (cl.stats[STAT_HEALTH] > 50)
bracket = 0;
snprintf (t2, sizeof (t2), "h:%i",
cl.stats[STAT_HEALTH]);
} else
snprintf (t2, sizeof (t2), "%i", cl.stats[STAT_HEALTH]);
break;
default:
bracket = 0;
}
if (!t1) {
if (!t2[0]) {
t2[0] = '%';
t2[1] = chr;
}
t1 = t2;
}
if (bracket)
buf[i++] = 0x90; // '['
if (t1) {
int len;
len = strlen (t1);
if (i + len >= sizeof (buf))
continue; // No more space in buffer, icky.
strncpy (buf + i, t1, len);
i += len;
}
if (bracket)
buf[i++] = 0x91; // ']'
continue;
}
buf[i++] = *s++;
}
buf[i] = 0;
return buf;
}
void
Team_Dead (void)
{
died = true;
VectorCopy (cl.simorg, death_location);
}
void
Team_NewMap (void)
{
char *mapname, *t1, *t2;
died = false;
recorded_location = false;
2001-08-28 03:47:10 +00:00
mapname = strdup (cl.worldmodel->name);
t2 = malloc (sizeof(cl.worldmodel->name));
if (!mapname || !t2)
Sys_Error ("Can't duplicate mapname!");
2001-08-28 03:47:10 +00:00
map_to_loc (mapname,t2);
t1 = strrchr (t2, '/');
if (!t1)
Sys_Error ("Can't find /!");
t1++; // skip over /
locs_reset ();
locs_load (t1);
free (mapname);
free (t2);
}
void
Team_Init_Cvars (void)
{
2001-08-28 03:47:10 +00:00
cl_deadbodyfilter = Cvar_Get ("cl_deadbodyfilter", "0", CVAR_NONE, NULL,
"Hide dead player models");
cl_gibfilter = Cvar_Get ("cl_gibfilter", "0", CVAR_NONE, NULL,
"Hide gibs");
cl_parsesay = Cvar_Get ("cl_parsesay", "0", CVAR_NONE, NULL,
2001-08-28 03:47:10 +00:00
"Use .loc files to find your present location "
"when you put %l in messages");
cl_nofake = Cvar_Get ("cl_nofake", "0", CVAR_NONE, NULL,
2001-08-28 03:47:10 +00:00
"Unhide fake messages");
cl_freply = Cvar_Get ("cl_freply", "20", CVAR_NONE, NULL,
"Delay between replies to f_*. Set to zero to disable.");
}
/*
2001-08-28 03:47:10 +00:00
locs_loc
Location marker manipulation
*/
void
locs_loc (void)
{
2001-08-28 03:47:10 +00:00
char locfile[MAX_OSPATH];
char *mapname;
2001-07-15 07:04:17 +00:00
const char *desc = NULL;
2001-08-28 03:47:10 +00:00
// FIXME: need to check to ensure you are actually in the game and alive.
if (Cmd_Argc () == 1) {
2001-08-28 03:47:10 +00:00
Con_Printf ("loc <add|delete|rename|move|save|zsave> [<description>] "
":Modifies location data, add|rename take <description> "
"parameter\n");
return;
}
if (Cmd_Argc () >= 3)
desc = Cmd_Args (2);
2001-08-28 03:47:10 +00:00
mapname = malloc (sizeof(cl.worldmodel->name));
if (!mapname)
Sys_Error ("Can't duplicate mapname!");
2001-08-28 03:47:10 +00:00
map_to_loc (cl.worldmodel->name,mapname);
snprintf (locfile, sizeof (locfile), "%s/%s", com_gamedir, mapname);
free(mapname);
2001-02-21 22:00:52 +00:00
if (strcasecmp(Cmd_Argv(1),"save") == 0) {
if (Cmd_Argc () == 2) {
locs_save(locfile, false);
} else
2001-08-28 03:47:10 +00:00
Con_Printf ("loc save :saves locs from memory into a .loc file\n");
}
2001-02-21 22:00:52 +00:00
if (strcasecmp(Cmd_Argv(1),"zsave") == 0) {
if (Cmd_Argc () == 2) {
locs_save(locfile, true);
} else
2001-08-28 03:47:10 +00:00
Con_Printf ("loc save :saves locs from memory into a .loc file\n");
}
2001-02-21 22:00:52 +00:00
if (strcasecmp(Cmd_Argv(1),"add") == 0) {
if (Cmd_Argc () >= 3)
locs_mark(cl.simorg,desc);
else
2001-08-28 03:47:10 +00:00
Con_Printf ("loc add <description> :marks the current location "
"with the description and records the information "
"into a loc file.\n");
}
2001-02-21 22:00:52 +00:00
if (strcasecmp(Cmd_Argv(1),"rename") == 0) {
if (Cmd_Argc () >= 3)
locs_edit(cl.simorg,desc);
else
2001-08-28 03:47:10 +00:00
Con_Printf ("loc rename <description> :changes the description of "
"the nearest location marker\n");
}
2001-02-21 22:00:52 +00:00
if (strcasecmp(Cmd_Argv(1),"delete") == 0) {
if (Cmd_Argc () == 2)
locs_del(cl.simorg);
else
2001-08-28 03:47:10 +00:00
Con_Printf ("loc delete :removes nearest location marker\n");
}
2001-02-21 22:00:52 +00:00
if (strcasecmp(Cmd_Argv(1),"move") == 0) {
if (Cmd_Argc () == 2)
locs_edit(cl.simorg,NULL);
else
2001-08-28 03:47:10 +00:00
Con_Printf ("loc move :moves the nearest location marker to your "
"current location\n");
}
}
void
Locs_Init (void)
{
2001-08-28 03:47:10 +00:00
Cmd_AddCommand ("loc", locs_loc, "Location marker editing commands: 'loc "
"help' for more");
}
char *
Team_F_Version (char *args)
{
return va("say %s %s", PROGRAM, VERSION);
}
char *
Team_F_Skins (char *args)
{
int totalfb, i, l;
skin_t *skin;
while(isspace(*args))
args++;
for (l = 0;args[l] && !isspace(args[l]);l++);
if (l == 0) {
for (i = 0, totalfb = 0; i < numskins; i++)
totalfb += skin_cache[i].numfb;
return va("say Average percent fullbright for all loaded skins is %.1f", (float)totalfb/(float)(numskins * fullfb)*100.0);
}
for (i = 0, skin = 0; i < numskins; i++) {
if (!strncmp(skin_cache[i].name, args, l)) {
skin = &skin_cache[i];
break;
}
}
if (skin)
return va("say \"Skin %s is %.1f%% fullbright\"",
skin->name,
(float)skin->numfb/(float)fullfb*100.0);
else
return ("say \"Skin not currently loaded.\"");
}
freply_t f_replies[] = {
{"f_version", Team_F_Version, 0},
{"f_skins", Team_F_Skins, 0},
{0, 0}
};
void
Team_ParseChat (const char *string)
{
char *s;
int i;
s = strchr(string, ':') + 1;
while (isspace(*s))
s++;
if (s && cl_freply->value) {
for (i = 0; f_replies[i].name; i++) {
if (!strncmp(f_replies[i].name, s, strlen(f_replies[i].name)) && cl_freply->value) {
while (*s && !isspace(*s))
s++;
Cbuf_AddText(f_replies[i].func(s));
f_replies[i].lasttime = realtime;
}
}
}
}
void
Team_ResetTimers (void)
{
int i;
for (i = 0; f_replies[i].name; i++)
f_replies[i].lasttime = realtime - cl_freply->value;
return;
}