2009-02-28 14:41:18 +00:00
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* Copyright (C) 1997-2001 Id Software, Inc.
|
|
|
|
*
|
2010-07-13 18:19:42 +00:00
|
|
|
* 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.
|
2010-06-18 08:33:12 +00:00
|
|
|
*
|
2010-07-13 18:19:42 +00:00
|
|
|
* 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.
|
2010-06-18 08:33:12 +00:00
|
|
|
*
|
|
|
|
* See the GNU General Public License for more details.
|
|
|
|
*
|
2010-07-13 18:19:42 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
2010-06-18 15:47:54 +00:00
|
|
|
*
|
|
|
|
* =======================================================================
|
2010-06-18 08:33:12 +00:00
|
|
|
*
|
2010-07-13 18:19:42 +00:00
|
|
|
* This is the clients main loop as well as some miscelangelous utility
|
|
|
|
* and support functions
|
2010-06-18 15:47:54 +00:00
|
|
|
*
|
|
|
|
* =======================================================================
|
2010-06-18 08:33:12 +00:00
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
|
2009-03-03 13:43:32 +00:00
|
|
|
#include "header/client.h"
|
2009-02-28 14:41:18 +00:00
|
|
|
|
2010-06-18 16:07:51 +00:00
|
|
|
void CL_ForwardToServer_f (void);
|
|
|
|
void CL_Changing_f (void);
|
|
|
|
void CL_Reconnect_f (void);
|
|
|
|
void CL_Connect_f (void);
|
|
|
|
void CL_Rcon_f (void);
|
|
|
|
void CL_CheckForResend (void);
|
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
cvar_t *freelook;
|
|
|
|
|
|
|
|
cvar_t *adr0;
|
|
|
|
cvar_t *adr1;
|
|
|
|
cvar_t *adr2;
|
|
|
|
cvar_t *adr3;
|
|
|
|
cvar_t *adr4;
|
|
|
|
cvar_t *adr5;
|
|
|
|
cvar_t *adr6;
|
|
|
|
cvar_t *adr7;
|
|
|
|
cvar_t *adr8;
|
|
|
|
|
|
|
|
cvar_t *rcon_client_password;
|
|
|
|
cvar_t *rcon_address;
|
|
|
|
|
|
|
|
cvar_t *cl_noskins;
|
|
|
|
cvar_t *cl_autoskins;
|
|
|
|
cvar_t *cl_footsteps;
|
|
|
|
cvar_t *cl_timeout;
|
|
|
|
cvar_t *cl_predict;
|
|
|
|
cvar_t *cl_maxfps;
|
|
|
|
cvar_t *cl_drawfps;
|
|
|
|
cvar_t *cl_gun;
|
|
|
|
cvar_t *cl_add_particles;
|
|
|
|
cvar_t *cl_add_lights;
|
|
|
|
cvar_t *cl_add_entities;
|
|
|
|
cvar_t *cl_add_blend;
|
|
|
|
|
|
|
|
cvar_t *cl_shownet;
|
|
|
|
cvar_t *cl_showmiss;
|
|
|
|
cvar_t *cl_showclamp;
|
|
|
|
|
|
|
|
cvar_t *cl_paused;
|
|
|
|
cvar_t *cl_timedemo;
|
|
|
|
|
|
|
|
cvar_t *lookspring;
|
|
|
|
cvar_t *lookstrafe;
|
|
|
|
cvar_t *sensitivity;
|
|
|
|
|
|
|
|
cvar_t *m_pitch;
|
|
|
|
cvar_t *m_yaw;
|
|
|
|
cvar_t *m_forward;
|
|
|
|
cvar_t *m_side;
|
|
|
|
|
|
|
|
cvar_t *cl_lightlevel;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* userinfo */
|
2009-02-28 14:41:18 +00:00
|
|
|
cvar_t *info_password;
|
|
|
|
cvar_t *info_spectator;
|
|
|
|
cvar_t *name;
|
|
|
|
cvar_t *skin;
|
|
|
|
cvar_t *rate;
|
|
|
|
cvar_t *fov;
|
|
|
|
cvar_t *msg;
|
|
|
|
cvar_t *hand;
|
|
|
|
cvar_t *gender;
|
|
|
|
cvar_t *gender_auto;
|
|
|
|
|
|
|
|
cvar_t *cl_vwep;
|
|
|
|
|
|
|
|
client_static_t cls;
|
|
|
|
client_state_t cl;
|
|
|
|
|
|
|
|
centity_t cl_entities[MAX_EDICTS];
|
|
|
|
|
|
|
|
entity_state_t cl_parse_entities[MAX_PARSE_ENTITIES];
|
|
|
|
|
|
|
|
extern cvar_t *allow_download;
|
|
|
|
extern cvar_t *allow_download_players;
|
|
|
|
extern cvar_t *allow_download_models;
|
|
|
|
extern cvar_t *allow_download_sounds;
|
|
|
|
extern cvar_t *allow_download_maps;
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* Dumps the current net message, prefixed by the length
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_WriteDemoMessage (void)
|
|
|
|
{
|
|
|
|
int len, swlen;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* the first eight bytes are just packet sequencing stuff */
|
2009-02-28 14:41:18 +00:00
|
|
|
len = net_message.cursize-8;
|
|
|
|
swlen = LittleLong(len);
|
|
|
|
fwrite (&swlen, 4, 1, cls.demofile);
|
|
|
|
fwrite (net_message.data+8, len, 1, cls.demofile);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* stop recording a demo
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_Stop_f (void)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
|
|
|
|
if (!cls.demorecording)
|
|
|
|
{
|
|
|
|
Com_Printf ("Not recording a demo.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = -1;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
fwrite (&len, 4, 1, cls.demofile);
|
|
|
|
fclose (cls.demofile);
|
|
|
|
cls.demofile = NULL;
|
|
|
|
cls.demorecording = false;
|
|
|
|
Com_Printf ("Stopped demo.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* record <demoname>
|
|
|
|
* Begins recording a demo from the current position
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_Record_f (void)
|
|
|
|
{
|
|
|
|
char name[MAX_OSPATH];
|
|
|
|
byte buf_data[MAX_MSGLEN];
|
|
|
|
sizebuf_t buf;
|
|
|
|
int i;
|
|
|
|
int len;
|
|
|
|
entity_state_t *ent;
|
|
|
|
entity_state_t nullstate;
|
|
|
|
|
|
|
|
if (Cmd_Argc() != 2)
|
|
|
|
{
|
|
|
|
Com_Printf ("record <demoname>\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cls.demorecording)
|
|
|
|
{
|
|
|
|
Com_Printf ("Already recording.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cls.state != ca_active)
|
|
|
|
{
|
|
|
|
Com_Printf ("You must be in a level to record.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Com_sprintf (name, sizeof(name), "%s/demos/%s.dm2", FS_Gamedir(), Cmd_Argv(1));
|
|
|
|
|
|
|
|
Com_Printf ("recording to %s.\n", name);
|
|
|
|
FS_CreatePath (name);
|
|
|
|
cls.demofile = fopen (name, "wb");
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (!cls.demofile)
|
|
|
|
{
|
|
|
|
Com_Printf ("ERROR: couldn't open.\n");
|
|
|
|
return;
|
|
|
|
}
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
cls.demorecording = true;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* don't start saving messages until a non-delta compressed message is received */
|
2009-02-28 14:41:18 +00:00
|
|
|
cls.demowaiting = true;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* write out messages to hold the startup information */
|
2009-02-28 14:41:18 +00:00
|
|
|
SZ_Init (&buf, buf_data, sizeof(buf_data));
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* send the serverdata */
|
2009-02-28 14:41:18 +00:00
|
|
|
MSG_WriteByte (&buf, svc_serverdata);
|
|
|
|
MSG_WriteLong (&buf, PROTOCOL_VERSION);
|
|
|
|
MSG_WriteLong (&buf, 0x10000 + cl.servercount);
|
2010-06-18 08:33:12 +00:00
|
|
|
MSG_WriteByte (&buf, 1); /* demos are always attract loops */
|
2009-02-28 14:41:18 +00:00
|
|
|
MSG_WriteString (&buf, cl.gamedir);
|
|
|
|
MSG_WriteShort (&buf, cl.playernum);
|
|
|
|
|
|
|
|
MSG_WriteString (&buf, cl.configstrings[CS_NAME]);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* configstrings */
|
2009-02-28 14:41:18 +00:00
|
|
|
for (i=0 ; i<MAX_CONFIGSTRINGS ; i++)
|
|
|
|
{
|
|
|
|
if (cl.configstrings[i][0])
|
|
|
|
{
|
|
|
|
if (buf.cursize + strlen (cl.configstrings[i]) + 32 > buf.maxsize)
|
2010-06-18 08:33:12 +00:00
|
|
|
{
|
2009-02-28 14:41:18 +00:00
|
|
|
len = LittleLong (buf.cursize);
|
|
|
|
fwrite (&len, 4, 1, cls.demofile);
|
|
|
|
fwrite (buf.data, buf.cursize, 1, cls.demofile);
|
|
|
|
buf.cursize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
MSG_WriteByte (&buf, svc_configstring);
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
MSG_WriteShort (&buf, i);
|
|
|
|
MSG_WriteString (&buf, cl.configstrings[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* baselines */
|
2009-02-28 14:41:18 +00:00
|
|
|
memset (&nullstate, 0, sizeof(nullstate));
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
for (i=0; i<MAX_EDICTS ; i++)
|
|
|
|
{
|
|
|
|
ent = &cl_entities[i].baseline;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (!ent->modelindex)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (buf.cursize + 64 > buf.maxsize)
|
2010-06-18 08:33:12 +00:00
|
|
|
{
|
2009-02-28 14:41:18 +00:00
|
|
|
len = LittleLong (buf.cursize);
|
|
|
|
fwrite (&len, 4, 1, cls.demofile);
|
|
|
|
fwrite (buf.data, buf.cursize, 1, cls.demofile);
|
|
|
|
buf.cursize = 0;
|
|
|
|
}
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
MSG_WriteByte (&buf, svc_spawnbaseline);
|
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
MSG_WriteDeltaEntity (&nullstate, &cl_entities[i].baseline, &buf, true, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
MSG_WriteByte (&buf, svc_stufftext);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
MSG_WriteString (&buf, "precache\n");
|
2009-02-28 14:41:18 +00:00
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* write it to the demo file */
|
2009-02-28 14:41:18 +00:00
|
|
|
len = LittleLong (buf.cursize);
|
|
|
|
fwrite (&len, 4, 1, cls.demofile);
|
|
|
|
fwrite (buf.data, buf.cursize, 1, cls.demofile);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Setenv_f( void )
|
|
|
|
{
|
|
|
|
int argc = Cmd_Argc();
|
|
|
|
|
|
|
|
if ( argc > 2 )
|
|
|
|
{
|
|
|
|
char buffer[1000];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
strcpy( buffer, Cmd_Argv(1) );
|
|
|
|
strcat( buffer, "=" );
|
|
|
|
|
|
|
|
for ( i = 2; i < argc; i++ )
|
|
|
|
{
|
|
|
|
strcat( buffer, Cmd_Argv( i ) );
|
|
|
|
strcat( buffer, " " );
|
|
|
|
}
|
|
|
|
|
|
|
|
putenv( buffer );
|
|
|
|
}
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
else if ( argc == 2 )
|
|
|
|
{
|
|
|
|
char *env = getenv( Cmd_Argv(1) );
|
|
|
|
|
|
|
|
if ( env )
|
|
|
|
{
|
|
|
|
Com_Printf( "%s=%s\n", Cmd_Argv(1), env );
|
|
|
|
}
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
Com_Printf( "%s undefined\n", Cmd_Argv(1), env );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Pause_f (void)
|
|
|
|
{
|
2010-06-18 08:33:12 +00:00
|
|
|
/* never pause in multiplayer */
|
2009-02-28 14:41:18 +00:00
|
|
|
if (Cvar_VariableValue ("maxclients") > 1 || !Com_ServerState ())
|
|
|
|
{
|
|
|
|
Cvar_SetValue ("paused", 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Cvar_SetValue ("paused", !cl_paused->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Quit_f (void)
|
|
|
|
{
|
|
|
|
CL_Disconnect ();
|
|
|
|
Com_Quit ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_ClearState (void)
|
|
|
|
{
|
|
|
|
S_StopAllSounds ();
|
|
|
|
CL_ClearEffects ();
|
|
|
|
CL_ClearTEnts ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* wipe the entire cl structure */
|
2009-02-28 14:41:18 +00:00
|
|
|
memset (&cl, 0, sizeof(cl));
|
|
|
|
memset (&cl_entities, 0, sizeof(cl_entities));
|
|
|
|
|
|
|
|
SZ_Clear (&cls.netchan.message);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* Handle a reply from a ping
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_ParseStatusMessage (void)
|
|
|
|
{
|
|
|
|
char *s;
|
|
|
|
|
|
|
|
s = MSG_ReadString(&net_message);
|
|
|
|
|
|
|
|
Com_Printf ("%s\n", s);
|
|
|
|
M_AddToServerList (net_from, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* Load or download any custom player skins and models
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_Skins_f (void)
|
|
|
|
{
|
2010-06-18 08:33:12 +00:00
|
|
|
int i;
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
for (i=0 ; i<MAX_CLIENTS ; i++)
|
|
|
|
{
|
|
|
|
if (!cl.configstrings[CS_PLAYERSKINS+i][0])
|
|
|
|
continue;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
|
|
|
Com_Printf ("client %i: %s\n", i, cl.configstrings[CS_PLAYERSKINS+i]);
|
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
SCR_UpdateScreen ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
|
|
|
Sys_SendKeyEvents (); /* pump message loop */
|
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_ParseClientinfo (i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* This fixes some problems with wrong tagged models and skins */
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_FixUpGender(void)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
char sk[80];
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
if (gender_auto->value)
|
|
|
|
{
|
2009-02-28 14:41:18 +00:00
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
if (gender->modified)
|
|
|
|
{
|
|
|
|
/* was set directly, don't override the user */
|
2009-02-28 14:41:18 +00:00
|
|
|
gender->modified = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(sk, skin->string, sizeof(sk) - 1);
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if ((p = strchr(sk, '/')) != NULL)
|
|
|
|
*p = 0;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (Q_stricmp(sk, "male") == 0 || Q_stricmp(sk, "cyborg") == 0)
|
|
|
|
Cvar_Set ("gender", "male");
|
|
|
|
else if (Q_stricmp(sk, "female") == 0 || Q_stricmp(sk, "crackhor") == 0)
|
|
|
|
Cvar_Set ("gender", "female");
|
|
|
|
else
|
|
|
|
Cvar_Set ("gender", "none");
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
gender->modified = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Userinfo_f (void)
|
|
|
|
{
|
|
|
|
Com_Printf ("User info settings:\n");
|
|
|
|
Info_Print (Cvar_Userinfo());
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* Restart the sound subsystem so it can pick up
|
|
|
|
* new parameters and flush all sounds
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_Snd_Restart_f (void)
|
|
|
|
{
|
|
|
|
S_Shutdown ();
|
|
|
|
S_Init ();
|
|
|
|
CL_RegisterSounds ();
|
|
|
|
}
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
int precache_check;
|
2009-02-28 14:41:18 +00:00
|
|
|
int precache_spawncount;
|
|
|
|
int precache_tex;
|
|
|
|
int precache_model_skin;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
byte *precache_model;
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* The server will send this command right
|
|
|
|
* before allowing the client into the server
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_Precache_f (void)
|
|
|
|
{
|
2010-06-18 08:33:12 +00:00
|
|
|
/* Yet another hack to let old demos work */
|
|
|
|
if (Cmd_Argc() < 2)
|
|
|
|
{
|
|
|
|
unsigned map_checksum; /* for detecting cheater maps */
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
CM_LoadMap (cl.configstrings[CS_MODELS+1], true, &map_checksum);
|
|
|
|
CL_RegisterSounds ();
|
|
|
|
CL_PrepRefresh ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
precache_check = CS_MODELS;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
precache_spawncount = atoi(Cmd_Argv(1));
|
|
|
|
precache_model = 0;
|
|
|
|
precache_model_skin = 0;
|
|
|
|
|
|
|
|
CL_RequestNextDownload();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_InitLocal (void)
|
|
|
|
{
|
|
|
|
cls.state = ca_disconnected;
|
|
|
|
cls.realtime = Sys_Milliseconds ();
|
|
|
|
|
|
|
|
CL_InitInput ();
|
|
|
|
|
|
|
|
adr0 = Cvar_Get( "adr0", "", CVAR_ARCHIVE );
|
|
|
|
adr1 = Cvar_Get( "adr1", "", CVAR_ARCHIVE );
|
|
|
|
adr2 = Cvar_Get( "adr2", "", CVAR_ARCHIVE );
|
|
|
|
adr3 = Cvar_Get( "adr3", "", CVAR_ARCHIVE );
|
|
|
|
adr4 = Cvar_Get( "adr4", "", CVAR_ARCHIVE );
|
|
|
|
adr5 = Cvar_Get( "adr5", "", CVAR_ARCHIVE );
|
|
|
|
adr6 = Cvar_Get( "adr6", "", CVAR_ARCHIVE );
|
|
|
|
adr7 = Cvar_Get( "adr7", "", CVAR_ARCHIVE );
|
|
|
|
adr8 = Cvar_Get( "adr8", "", CVAR_ARCHIVE );
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* register our variables */
|
2009-02-28 14:41:18 +00:00
|
|
|
cl_add_blend = Cvar_Get ("cl_blend", "1", 0);
|
|
|
|
cl_add_lights = Cvar_Get ("cl_lights", "1", 0);
|
|
|
|
cl_add_particles = Cvar_Get ("cl_particles", "1", 0);
|
|
|
|
cl_add_entities = Cvar_Get ("cl_entities", "1", 0);
|
2009-10-04 12:32:44 +00:00
|
|
|
cl_gun = Cvar_Get ("cl_gun", "2", CVAR_ARCHIVE);
|
2009-02-28 14:41:18 +00:00
|
|
|
cl_footsteps = Cvar_Get ("cl_footsteps", "1", 0);
|
|
|
|
cl_noskins = Cvar_Get ("cl_noskins", "0", 0);
|
|
|
|
cl_autoskins = Cvar_Get ("cl_autoskins", "0", 0);
|
|
|
|
cl_predict = Cvar_Get ("cl_predict", "1", 0);
|
2010-07-13 18:42:12 +00:00
|
|
|
cl_maxfps = Cvar_Get ("cl_maxfps", "60", 0);
|
2010-06-18 08:33:12 +00:00
|
|
|
cl_drawfps = Cvar_Get("cl_drawfps","0",CVAR_ARCHIVE);
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
cl_upspeed = Cvar_Get ("cl_upspeed", "200", 0);
|
|
|
|
cl_forwardspeed = Cvar_Get ("cl_forwardspeed", "200", 0);
|
|
|
|
cl_sidespeed = Cvar_Get ("cl_sidespeed", "200", 0);
|
|
|
|
cl_yawspeed = Cvar_Get ("cl_yawspeed", "140", 0);
|
|
|
|
cl_pitchspeed = Cvar_Get ("cl_pitchspeed", "150", 0);
|
|
|
|
cl_anglespeedkey = Cvar_Get ("cl_anglespeedkey", "1.5", 0);
|
|
|
|
|
2010-02-01 15:23:57 +00:00
|
|
|
cl_run = Cvar_Get ("cl_run", "0", CVAR_ARCHIVE);
|
2010-02-01 15:10:50 +00:00
|
|
|
freelook = Cvar_Get( "freelook", "1", CVAR_ARCHIVE );
|
2009-02-28 14:41:18 +00:00
|
|
|
lookspring = Cvar_Get ("lookspring", "0", CVAR_ARCHIVE);
|
|
|
|
lookstrafe = Cvar_Get ("lookstrafe", "0", CVAR_ARCHIVE);
|
|
|
|
sensitivity = Cvar_Get ("sensitivity", "3", CVAR_ARCHIVE);
|
|
|
|
|
|
|
|
m_pitch = Cvar_Get ("m_pitch", "0.022", CVAR_ARCHIVE);
|
|
|
|
m_yaw = Cvar_Get ("m_yaw", "0.022", 0);
|
|
|
|
m_forward = Cvar_Get ("m_forward", "1", 0);
|
|
|
|
m_side = Cvar_Get ("m_side", "1", 0);
|
|
|
|
|
|
|
|
cl_shownet = Cvar_Get ("cl_shownet", "0", 0);
|
|
|
|
cl_showmiss = Cvar_Get ("cl_showmiss", "0", 0);
|
|
|
|
cl_showclamp = Cvar_Get ("showclamp", "0", 0);
|
|
|
|
cl_timeout = Cvar_Get ("cl_timeout", "120", 0);
|
|
|
|
cl_paused = Cvar_Get ("paused", "0", 0);
|
|
|
|
cl_timedemo = Cvar_Get ("timedemo", "0", 0);
|
|
|
|
|
|
|
|
rcon_client_password = Cvar_Get ("rcon_password", "", 0);
|
|
|
|
rcon_address = Cvar_Get ("rcon_address", "", 0);
|
|
|
|
|
|
|
|
cl_lightlevel = Cvar_Get ("r_lightlevel", "0", 0);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* userinfo */
|
2009-02-28 14:41:18 +00:00
|
|
|
info_password = Cvar_Get ("password", "", CVAR_USERINFO);
|
|
|
|
info_spectator = Cvar_Get ("spectator", "0", CVAR_USERINFO);
|
|
|
|
name = Cvar_Get ("name", "unnamed", CVAR_USERINFO | CVAR_ARCHIVE);
|
|
|
|
skin = Cvar_Get ("skin", "male/grunt", CVAR_USERINFO | CVAR_ARCHIVE);
|
2010-07-13 18:42:12 +00:00
|
|
|
rate = Cvar_Get ("rate", "8000", CVAR_USERINFO | CVAR_ARCHIVE);
|
2009-02-28 14:41:18 +00:00
|
|
|
msg = Cvar_Get ("msg", "1", CVAR_USERINFO | CVAR_ARCHIVE);
|
|
|
|
hand = Cvar_Get ("hand", "0", CVAR_USERINFO | CVAR_ARCHIVE);
|
|
|
|
fov = Cvar_Get ("fov", "90", CVAR_USERINFO | CVAR_ARCHIVE);
|
|
|
|
gender = Cvar_Get ("gender", "male", CVAR_USERINFO | CVAR_ARCHIVE);
|
|
|
|
gender_auto = Cvar_Get ("gender_auto", "1", CVAR_ARCHIVE);
|
2010-06-18 08:33:12 +00:00
|
|
|
gender->modified = false;
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
cl_vwep = Cvar_Get ("cl_vwep", "1", CVAR_ARCHIVE);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* register our commands */
|
2009-02-28 14:41:18 +00:00
|
|
|
Cmd_AddCommand ("cmd", CL_ForwardToServer_f);
|
|
|
|
Cmd_AddCommand ("pause", CL_Pause_f);
|
|
|
|
Cmd_AddCommand ("pingservers", CL_PingServers_f);
|
|
|
|
Cmd_AddCommand ("skins", CL_Skins_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("userinfo", CL_Userinfo_f);
|
|
|
|
Cmd_AddCommand ("snd_restart", CL_Snd_Restart_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("changing", CL_Changing_f);
|
|
|
|
Cmd_AddCommand ("disconnect", CL_Disconnect_f);
|
|
|
|
Cmd_AddCommand ("record", CL_Record_f);
|
|
|
|
Cmd_AddCommand ("stop", CL_Stop_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("quit", CL_Quit_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("connect", CL_Connect_f);
|
|
|
|
Cmd_AddCommand ("reconnect", CL_Reconnect_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("rcon", CL_Rcon_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("setenv", CL_Setenv_f );
|
|
|
|
|
|
|
|
Cmd_AddCommand ("precache", CL_Precache_f);
|
|
|
|
|
|
|
|
Cmd_AddCommand ("download", CL_Download_f);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* forward to server commands
|
|
|
|
the only thing this does is allow command completion
|
|
|
|
to work -- all unknown commands are automatically
|
|
|
|
forwarded to the server */
|
2009-02-28 14:41:18 +00:00
|
|
|
Cmd_AddCommand ("wave", NULL);
|
|
|
|
Cmd_AddCommand ("inven", NULL);
|
|
|
|
Cmd_AddCommand ("kill", NULL);
|
|
|
|
Cmd_AddCommand ("use", NULL);
|
|
|
|
Cmd_AddCommand ("drop", NULL);
|
|
|
|
Cmd_AddCommand ("say", NULL);
|
|
|
|
Cmd_AddCommand ("say_team", NULL);
|
|
|
|
Cmd_AddCommand ("info", NULL);
|
|
|
|
Cmd_AddCommand ("prog", NULL);
|
|
|
|
Cmd_AddCommand ("give", NULL);
|
|
|
|
Cmd_AddCommand ("god", NULL);
|
|
|
|
Cmd_AddCommand ("notarget", NULL);
|
|
|
|
Cmd_AddCommand ("noclip", NULL);
|
|
|
|
Cmd_AddCommand ("invuse", NULL);
|
|
|
|
Cmd_AddCommand ("invprev", NULL);
|
|
|
|
Cmd_AddCommand ("invnext", NULL);
|
|
|
|
Cmd_AddCommand ("invdrop", NULL);
|
|
|
|
Cmd_AddCommand ("weapnext", NULL);
|
|
|
|
Cmd_AddCommand ("weapprev", NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-06-18 08:33:12 +00:00
|
|
|
* Writes key bindings and archived cvars to config.cfg
|
|
|
|
*/
|
2009-02-28 14:41:18 +00:00
|
|
|
void CL_WriteConfiguration (void)
|
|
|
|
{
|
|
|
|
FILE *f;
|
|
|
|
char path[MAX_QPATH];
|
|
|
|
|
|
|
|
if (cls.state == ca_uninitialized)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Com_sprintf (path, sizeof(path),"%s/config.cfg",FS_Gamedir());
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
f = fopen (path, "w");
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (!f)
|
|
|
|
{
|
|
|
|
Com_Printf ("Couldn't write config.cfg.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf (f, "// generated by quake, do not modify\n");
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
Key_WriteBindings (f);
|
|
|
|
fclose (f);
|
|
|
|
|
|
|
|
Cvar_WriteVariables (path);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
char *value;
|
|
|
|
cvar_t *var;
|
|
|
|
} cheatvar_t;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
cheatvar_t cheatvars[] =
|
|
|
|
{
|
2009-02-28 14:41:18 +00:00
|
|
|
{"timescale", "1"},
|
|
|
|
{"timedemo", "0"},
|
|
|
|
{"r_drawworld", "1"},
|
|
|
|
{"cl_testlights", "0"},
|
|
|
|
{"r_fullbright", "0"},
|
|
|
|
{"r_drawflat", "0"},
|
|
|
|
{"paused", "0"},
|
|
|
|
{"fixedtime", "0"},
|
|
|
|
{"sw_draworder", "0"},
|
|
|
|
{"gl_lightmap", "0"},
|
|
|
|
{"gl_saturatelighting", "0"},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
int numcheatvars;
|
|
|
|
|
|
|
|
void CL_FixCvarCheats (void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
cheatvar_t *var;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
if ( !strcmp(cl.configstrings[CS_MAXCLIENTS], "1")
|
|
|
|
|| !cl.configstrings[CS_MAXCLIENTS][0] )
|
|
|
|
return; /* single player can cheat */
|
2009-02-28 14:41:18 +00:00
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* find all the cvars if we haven't done it yet */
|
2009-02-28 14:41:18 +00:00
|
|
|
if (!numcheatvars)
|
|
|
|
{
|
|
|
|
while (cheatvars[numcheatvars].name)
|
|
|
|
{
|
|
|
|
cheatvars[numcheatvars].var = Cvar_Get (cheatvars[numcheatvars].name,
|
2010-06-18 08:33:12 +00:00
|
|
|
cheatvars[numcheatvars].value, 0);
|
2009-02-28 14:41:18 +00:00
|
|
|
numcheatvars++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* make sure they are all set to the proper values */
|
2009-02-28 14:41:18 +00:00
|
|
|
for (i=0, var = cheatvars ; i<numcheatvars ; i++, var++)
|
|
|
|
{
|
|
|
|
if ( strcmp (var->var->string, var->value) )
|
|
|
|
{
|
|
|
|
Cvar_Set (var->name, var->value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_SendCommand (void)
|
|
|
|
{
|
2010-06-18 08:33:12 +00:00
|
|
|
/* get new key events */
|
2009-02-28 14:41:18 +00:00
|
|
|
Sys_SendKeyEvents ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* allow mice or other external controllers to add commands */
|
2009-02-28 14:41:18 +00:00
|
|
|
IN_Commands ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* process console commands */
|
2009-02-28 14:41:18 +00:00
|
|
|
Cbuf_Execute ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* fix any cheating cvars */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_FixCvarCheats ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* send intentions now */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_SendCmd ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* resend a connection request if necessary */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_CheckForResend ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Frame (int msec)
|
|
|
|
{
|
|
|
|
static int extratime;
|
|
|
|
static int lasttimecalled;
|
|
|
|
|
|
|
|
if (dedicated->value)
|
|
|
|
return;
|
|
|
|
|
|
|
|
extratime += msec;
|
|
|
|
|
|
|
|
if (!cl_timedemo->value)
|
|
|
|
{
|
|
|
|
if (cls.state == ca_connected && extratime < 100)
|
2010-06-18 08:33:12 +00:00
|
|
|
return; /* don't flood packets out while connecting */
|
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (extratime < 1000/cl_maxfps->value)
|
2010-06-18 08:33:12 +00:00
|
|
|
return; /* framerate is too high */
|
2009-02-28 14:41:18 +00:00
|
|
|
}
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* let the mouse activate or deactivate */
|
2009-02-28 14:41:18 +00:00
|
|
|
IN_Frame ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* decide the simulation time */
|
2009-02-28 14:41:18 +00:00
|
|
|
cls.frametime = extratime/1000.0;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
cl.time += extratime;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
cls.realtime = curtime;
|
|
|
|
|
|
|
|
extratime = 0;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (cls.frametime > (1.0 / 5))
|
|
|
|
cls.frametime = (1.0 / 5);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* if in the debugger last frame, don't timeout */
|
2009-02-28 14:41:18 +00:00
|
|
|
if (msec > 5000)
|
|
|
|
cls.netchan.last_received = Sys_Milliseconds ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* fetch results from server */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_ReadPackets ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* send a new command message to the server */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_SendCommand ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* predict all unacknowledged movements */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_PredictMovement ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* allow renderer DLL change */
|
2009-02-28 14:41:18 +00:00
|
|
|
VID_CheckChanges ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (!cl.refresh_prepped && cls.state == ca_active)
|
|
|
|
CL_PrepRefresh ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* update the screen */
|
2009-02-28 14:41:18 +00:00
|
|
|
if (host_speeds->value)
|
|
|
|
time_before_ref = Sys_Milliseconds ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
SCR_UpdateScreen ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (host_speeds->value)
|
|
|
|
time_after_ref = Sys_Milliseconds ();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* update audio */
|
2009-02-28 14:41:18 +00:00
|
|
|
S_Update (cl.refdef.vieworg, cl.v_forward, cl.v_right, cl.v_up);
|
|
|
|
CDAudio_Update();
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* advance local effects for next frame */
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_RunDLights ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_RunLightStyles ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
SCR_RunCinematic ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
SCR_RunConsole ();
|
|
|
|
|
|
|
|
cls.framecount++;
|
|
|
|
|
|
|
|
if ( log_stats->value )
|
|
|
|
{
|
|
|
|
if ( cls.state == ca_active )
|
|
|
|
{
|
|
|
|
if ( !lasttimecalled )
|
|
|
|
{
|
|
|
|
lasttimecalled = Sys_Milliseconds();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if ( log_stats_file )
|
|
|
|
fprintf( log_stats_file, "0\n" );
|
|
|
|
}
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
int now = Sys_Milliseconds();
|
|
|
|
|
|
|
|
if ( log_stats_file )
|
|
|
|
fprintf( log_stats_file, "%d\n", now - lasttimecalled );
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
lasttimecalled = now;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Init (void)
|
|
|
|
{
|
|
|
|
if (dedicated->value)
|
2010-06-18 08:33:12 +00:00
|
|
|
return; /* nothing running on the client */
|
2009-02-28 14:41:18 +00:00
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
/* all archived variables will now be loaded */
|
|
|
|
Con_Init ();
|
|
|
|
|
|
|
|
S_Init ();
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
VID_Init ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
V_Init ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
net_message.data = net_message_buffer;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
net_message.maxsize = sizeof(net_message_buffer);
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
M_Init ();
|
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
SCR_Init ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
|
|
|
cls.disable_screen = true; /* don't draw yet */
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
CDAudio_Init ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
CL_InitLocal ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
FS_ExecAutoexec ();
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
Cbuf_Execute ();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void CL_Shutdown(void)
|
|
|
|
{
|
|
|
|
static qboolean isdown = false;
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
if (isdown)
|
|
|
|
{
|
|
|
|
printf ("recursive shutdown\n");
|
|
|
|
return;
|
|
|
|
}
|
2010-06-18 08:33:12 +00:00
|
|
|
|
2009-02-28 14:41:18 +00:00
|
|
|
isdown = true;
|
|
|
|
|
2010-06-18 08:33:12 +00:00
|
|
|
CL_WriteConfiguration ();
|
2009-02-28 14:41:18 +00:00
|
|
|
|
|
|
|
CDAudio_Shutdown ();
|
2009-10-03 16:06:45 +00:00
|
|
|
OGG_Stop();
|
2009-02-28 14:41:18 +00:00
|
|
|
S_Shutdown();
|
|
|
|
IN_Shutdown ();
|
|
|
|
VID_Shutdown();
|
|
|
|
}
|
2010-06-18 16:28:25 +00:00
|
|
|
|