2000-08-02 15:20:15 +00:00
|
|
|
/*
|
|
|
|
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
|
|
|
|
|
|
|
|
$Id$
|
|
|
|
*/
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
#include <string.h>
|
2000-12-03 23:52:54 +00:00
|
|
|
#include <errno.h>
|
2000-10-06 16:30:37 +00:00
|
|
|
|
2000-08-22 17:53:48 +00:00
|
|
|
#include "bothdefs.h"
|
|
|
|
#include "cmd.h"
|
2000-10-06 16:30:37 +00:00
|
|
|
#include "client.h"
|
2000-08-02 22:50:28 +00:00
|
|
|
#include "teamplay.h"
|
2000-10-06 16:30:37 +00:00
|
|
|
#include "locs.h"
|
|
|
|
#include "sys.h"
|
2000-12-03 23:52:54 +00:00
|
|
|
#include "console.h"
|
|
|
|
#include "quakefs.h"
|
2000-08-02 22:50:28 +00:00
|
|
|
|
2000-10-06 19:33:55 +00:00
|
|
|
extern cvar_t *skin;
|
2000-08-02 22:50:28 +00:00
|
|
|
cvar_t *cl_deadbodyfilter;
|
|
|
|
cvar_t *cl_gibfilter;
|
2000-08-22 18:54:01 +00:00
|
|
|
cvar_t *cl_parsesay;
|
|
|
|
cvar_t *cl_nofake;
|
2000-10-06 19:33:55 +00:00
|
|
|
static qboolean died = false, recorded_location = false;
|
|
|
|
static vec3_t death_location, last_recorded_location;
|
2000-08-02 22:50:28 +00:00
|
|
|
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
void Team_BestWeaponImpulse (void)
|
2000-08-22 17:53:48 +00:00
|
|
|
{
|
|
|
|
int best, i, imp, items;
|
|
|
|
extern int in_impulse;
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
char *Team_ParseSay (char *s)
|
2000-08-22 18:54:01 +00:00
|
|
|
{
|
|
|
|
static char buf[1024];
|
2000-10-06 16:30:37 +00:00
|
|
|
int i, bracket;
|
|
|
|
char c, chr, *t1, t2[128], t3[128];
|
|
|
|
static location_t *location = NULL;
|
2000-08-22 18:54:01 +00:00
|
|
|
|
2000-10-17 03:17:42 +00:00
|
|
|
if (!cl_parsesay->int_val)
|
2000-08-22 18:54:01 +00:00
|
|
|
return s;
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
while (*s && (i <= sizeof(buf))) {
|
|
|
|
if (*s == '$') {
|
2000-08-22 18:54:01 +00:00
|
|
|
c = 0;
|
2000-10-06 16:30:37 +00:00
|
|
|
switch (s[1]) {
|
|
|
|
case '\\': c = 13; break; // fake message
|
|
|
|
case '[': c = 0x90; break; // colored brackets
|
|
|
|
case ']': c = 0x91; break;
|
|
|
|
case 'G': c = 0x86; break; // ocrana leds
|
|
|
|
case 'R': c = 0x87; break;
|
|
|
|
case 'Y': c = 0x88; break;
|
|
|
|
case 'B': c = 0x89; break;
|
2000-08-22 18:54:01 +00:00
|
|
|
}
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
if (c) {
|
2000-08-22 18:54:01 +00:00
|
|
|
buf[i++] = c;
|
|
|
|
s += 2;
|
|
|
|
continue;
|
|
|
|
}
|
2000-10-06 16:30:37 +00:00
|
|
|
} else if (*s == '%') {
|
|
|
|
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) {
|
2000-10-18 17:06:44 +00:00
|
|
|
case '%':
|
|
|
|
t2[0] = '%';
|
|
|
|
t2[1] = 0;
|
|
|
|
t1 = t2;
|
|
|
|
break;
|
2000-10-06 19:33:55 +00:00
|
|
|
case 's':
|
|
|
|
bracket = 0;
|
|
|
|
t1 = skin->string;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
bracket = 0;
|
|
|
|
if (died) {
|
|
|
|
location = locs_find(death_location);
|
|
|
|
if (location) {
|
|
|
|
recorded_location = true;
|
2000-10-24 17:03:09 +00:00
|
|
|
VectorCopy(death_location, last_recorded_location);
|
2000-10-06 19:33:55 +00:00
|
|
|
t1 = location->name;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
goto location;
|
|
|
|
case 'r':
|
|
|
|
bracket = 0;
|
|
|
|
if (recorded_location) {
|
|
|
|
location = locs_find(last_recorded_location);
|
|
|
|
if (location) {
|
|
|
|
t1 = location->name;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
goto location;
|
2000-10-06 16:30:37 +00:00
|
|
|
case 'l':
|
2000-10-06 19:33:55 +00:00
|
|
|
location:
|
2000-10-06 16:30:37 +00:00
|
|
|
bracket = 0;
|
2000-10-24 17:03:09 +00:00
|
|
|
location = locs_find(cl.simorg);
|
2000-10-06 16:30:37 +00:00
|
|
|
if (location) {
|
2000-10-06 19:33:55 +00:00
|
|
|
recorded_location = true;
|
2000-10-24 17:03:09 +00:00
|
|
|
VectorCopy(cl.simorg, last_recorded_location);
|
2000-10-06 16:30:37 +00:00
|
|
|
t1 = location->name;
|
|
|
|
} else
|
|
|
|
snprintf(t2, sizeof(t2), "Unknown!\n");
|
|
|
|
break;
|
|
|
|
case 'a':
|
|
|
|
if (bracket) {
|
|
|
|
if (cl.stats[STAT_ARMOR] > 50)
|
|
|
|
bracket = 0;
|
2000-08-22 18:54:01 +00:00
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
if (cl.stats[STAT_ITEMS] & IT_ARMOR3)
|
|
|
|
t3[0] = 'R' | 0x80;
|
|
|
|
else if (cl.stats[STAT_ITEMS] & IT_ARMOR2)
|
|
|
|
t3[0] = 'Y' | 0x80;
|
|
|
|
else if (cl.stats[STAT_ITEMS] & IT_ARMOR1)
|
|
|
|
t3[0] = 'G' | 0x80;
|
|
|
|
else {
|
|
|
|
t2[0] = 'N' | 0x80;
|
|
|
|
t2[1] = 'O' | 0x80;
|
|
|
|
t2[2] = 'N' | 0x80;
|
|
|
|
t2[3] = 'E' | 0x80;
|
|
|
|
t2[4] = '!' | 0x80;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2000-08-22 18:54:01 +00:00
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
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;
|
|
|
|
}
|
2000-08-22 18:54:01 +00:00
|
|
|
buf[i++] = *s++;
|
|
|
|
}
|
|
|
|
buf[i] = 0;
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
void Team_Dead ()
|
|
|
|
{
|
2000-10-06 19:33:55 +00:00
|
|
|
died = true;
|
2000-10-24 17:03:09 +00:00
|
|
|
VectorCopy(cl.simorg, death_location);
|
2000-10-06 16:30:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Team_NewMap ()
|
|
|
|
{
|
|
|
|
char *mapname, *t1, *t2;
|
|
|
|
|
2000-10-06 19:43:38 +00:00
|
|
|
died = false;
|
|
|
|
recorded_location = false;
|
|
|
|
|
2000-10-06 16:30:37 +00:00
|
|
|
mapname = strdup(cl.worldmodel->name);
|
|
|
|
if (!mapname)
|
|
|
|
Sys_Error("Can't duplicate mapname!");
|
|
|
|
t1 = strrchr(mapname, '/');
|
|
|
|
t2 = strrchr(mapname, '.');
|
|
|
|
if (!t1 || !t2)
|
|
|
|
Sys_Error("Can't find / or .!");
|
2000-10-10 23:35:09 +00:00
|
|
|
t1++; // skip over /
|
2000-10-06 16:30:37 +00:00
|
|
|
t2[0] = '\0';
|
|
|
|
|
|
|
|
locs_reset();
|
|
|
|
locs_load(t1);
|
|
|
|
free(mapname);
|
|
|
|
}
|
2000-08-22 18:54:01 +00:00
|
|
|
|
2000-10-29 15:35:24 +00:00
|
|
|
void Team_Init_Cvars (void)
|
2000-08-02 22:50:28 +00:00
|
|
|
{
|
2000-08-22 18:54:01 +00:00
|
|
|
cl_deadbodyfilter = Cvar_Get("cl_deadbodyfilter", "0", CVAR_NONE, "Hide dead player models");
|
|
|
|
cl_gibfilter = Cvar_Get("cl_gibfilter", "0", CVAR_NONE, "Hide gibs");
|
|
|
|
cl_parsesay = Cvar_Get("cl_parsesay", "0", CVAR_NONE, "None");
|
|
|
|
cl_nofake = Cvar_Get("cl_nofake", "0", CVAR_NONE, "Unhide fake messages");
|
2000-08-02 22:50:28 +00:00
|
|
|
}
|
2000-12-03 23:52:54 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* locs_markloc
|
|
|
|
*
|
|
|
|
* Record the current co-ords plus description into a loc file for current map
|
|
|
|
* */
|
|
|
|
|
|
|
|
// FIXME: No gzip'd loc file support
|
|
|
|
|
|
|
|
void locs_markloc()
|
|
|
|
{
|
|
|
|
vec3_t loc;
|
|
|
|
char *mapname, *t1;
|
|
|
|
QFile *locfd;
|
|
|
|
char locfile[MAX_OSPATH];
|
|
|
|
|
|
|
|
if (Cmd_Argc() != 2) {
|
|
|
|
Con_Printf("markloc <description> :marks the current location with the description and records the information into a loc file.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
VectorCopy(cl.simorg,loc);
|
|
|
|
locs_add(loc,Cmd_Argv(1));
|
|
|
|
loc[0] *= 8;
|
|
|
|
loc[1] *= 8;
|
|
|
|
loc[2] *= 8;
|
|
|
|
mapname = strdup(cl.worldmodel->name);
|
|
|
|
if (!mapname)
|
|
|
|
Sys_Error("Can't duplicate mapname!");
|
|
|
|
t1 = strrchr(mapname, '.');
|
|
|
|
if (!t1)
|
|
|
|
Sys_Error("Can't find / or .!");
|
|
|
|
t1++; // skip over /
|
|
|
|
t1[0] = 'l';
|
|
|
|
t1[1] = 'o';
|
|
|
|
t1[2] = 'c';
|
|
|
|
snprintf(locfile, sizeof(locfile), "%s/%s",com_gamedir,mapname);
|
|
|
|
locfd = Qopen(locfile,"a+");
|
|
|
|
if (locfd == 0) {
|
|
|
|
Qopen(locfile,"w+");
|
|
|
|
if (locfd == 0) {
|
|
|
|
Con_Printf("ERROR: Unable to open %s : %s\n",mapname,strerror(errno));
|
|
|
|
free(mapname);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Qprintf(locfd,"%.0f %.0f %.0f %s\n",loc[0],loc[1],loc[2],Cmd_Argv(1));
|
|
|
|
Qclose(locfd);
|
|
|
|
free(mapname);
|
|
|
|
}
|
|
|
|
void Locs_Init()
|
|
|
|
{
|
|
|
|
Cmd_AddCommand("markloc",locs_markloc);
|
|
|
|
}
|
|
|
|
|