yquake2remaster/src/client/cl_main.c

929 lines
18 KiB
C
Raw Normal View History

/*
* Copyright (C) 1997-2001 Id Software, Inc.
*
* 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 the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
2012-04-29 13:57:33 +00:00
*
* =======================================================================
*
* This is the clients main loop as well as some miscelangelous utility
* and support functions
*
2012-04-29 13:57:33 +00:00
* =======================================================================
*/
2009-03-03 13:43:32 +00:00
#include "header/client.h"
2012-07-22 13:34:45 +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);
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;
/* userinfo */
2012-07-22 13:34:45 +00:00
cvar_t *info_password;
cvar_t *info_spectator;
cvar_t *name;
cvar_t *skin;
cvar_t *rate;
cvar_t *fov;
cvar_t *horplus;
cvar_t *windowed_mouse;
2012-07-22 13:34:45 +00:00
cvar_t *msg;
cvar_t *hand;
cvar_t *gender;
cvar_t *gender_auto;
2012-07-22 13:34:45 +00:00
cvar_t *cl_vwep;
2012-07-22 13:34:45 +00:00
client_static_t cls;
client_state_t cl;
2012-07-22 13:34:45 +00:00
centity_t cl_entities[MAX_EDICTS];
2012-07-22 13:34:45 +00:00
entity_state_t cl_parse_entities[MAX_PARSE_ENTITIES];
2012-07-22 13:34:45 +00:00
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;
/*
* Dumps the current net message, prefixed by the length
*/
2012-07-22 13:34:45 +00:00
void
CL_WriteDemoMessage(void)
{
2012-07-22 13:34:45 +00:00
int len, swlen;
/* the first eight bytes are just packet sequencing stuff */
2012-07-22 13:34:45 +00:00
len = net_message.cursize - 8;
swlen = LittleLong(len);
2012-07-22 13:34:45 +00:00
fwrite(&swlen, 4, 1, cls.demofile);
fwrite(net_message.data + 8, len, 1, cls.demofile);
}
/*
* stop recording a demo
*/
2012-07-22 13:34:45 +00:00
void
CL_Stop_f(void)
{
2012-07-22 13:34:45 +00:00
int len;
if (!cls.demorecording)
{
2012-07-22 13:34:45 +00:00
Com_Printf("Not recording a demo.\n");
return;
}
len = -1;
2012-07-22 13:34:45 +00:00
fwrite(&len, 4, 1, cls.demofile);
fclose(cls.demofile);
cls.demofile = NULL;
cls.demorecording = false;
2012-07-22 13:34:45 +00:00
Com_Printf("Stopped demo.\n");
}
/*
* record <demoname>
* Begins recording a demo from the current position
*/
2012-07-22 13:34:45 +00:00
void
CL_Record_f(void)
{
2012-07-22 13:34:45 +00:00
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)
{
2012-07-22 13:34:45 +00:00
Com_Printf("record <demoname>\n");
return;
}
if (cls.demorecording)
{
2012-07-22 13:34:45 +00:00
Com_Printf("Already recording.\n");
return;
}
if (cls.state != ca_active)
{
2012-07-22 13:34:45 +00:00
Com_Printf("You must be in a level to record.\n");
return;
}
2012-07-22 13:34:45 +00:00
Com_sprintf(name, sizeof(name), "%s/demos/%s.dm2", FS_Gamedir(), Cmd_Argv(1));
2012-07-22 13:34:45 +00:00
Com_Printf("recording to %s.\n", name);
FS_CreatePath(name);
cls.demofile = fopen(name, "wb");
if (!cls.demofile)
{
2012-07-22 13:34:45 +00:00
Com_Printf("ERROR: couldn't open.\n");
return;
}
cls.demorecording = true;
/* don't start saving messages until a non-delta compressed message is received */
cls.demowaiting = true;
/* write out messages to hold the startup information */
2012-07-22 13:34:45 +00:00
SZ_Init(&buf, buf_data, sizeof(buf_data));
/* send the serverdata */
2012-07-22 13:34:45 +00:00
MSG_WriteByte(&buf, svc_serverdata);
MSG_WriteLong(&buf, PROTOCOL_VERSION);
MSG_WriteLong(&buf, 0x10000 + cl.servercount);
MSG_WriteByte(&buf, 1); /* demos are always attract loops */
MSG_WriteString(&buf, cl.gamedir);
MSG_WriteShort(&buf, cl.playernum);
2012-07-22 13:34:45 +00:00
MSG_WriteString(&buf, cl.configstrings[CS_NAME]);
/* configstrings */
2012-07-22 13:34:45 +00:00
for (i = 0; i < MAX_CONFIGSTRINGS; i++)
{
if (cl.configstrings[i][0])
{
2012-07-22 13:34:45 +00:00
if (buf.cursize + strlen(cl.configstrings[i]) + 32 > buf.maxsize)
{
2012-07-22 13:34:45 +00:00
len = LittleLong(buf.cursize);
fwrite(&len, 4, 1, cls.demofile);
fwrite(buf.data, buf.cursize, 1, cls.demofile);
buf.cursize = 0;
}
2012-07-22 13:34:45 +00:00
MSG_WriteByte(&buf, svc_configstring);
2012-07-22 13:34:45 +00:00
MSG_WriteShort(&buf, i);
MSG_WriteString(&buf, cl.configstrings[i]);
}
}
/* baselines */
2012-07-22 13:34:45 +00:00
memset(&nullstate, 0, sizeof(nullstate));
2012-07-22 13:34:45 +00:00
for (i = 0; i < MAX_EDICTS; i++)
{
ent = &cl_entities[i].baseline;
if (!ent->modelindex)
2012-07-22 13:34:45 +00:00
{
continue;
2012-07-22 13:34:45 +00:00
}
if (buf.cursize + 64 > buf.maxsize)
{
2012-07-22 13:34:45 +00:00
len = LittleLong(buf.cursize);
fwrite(&len, 4, 1, cls.demofile);
fwrite(buf.data, buf.cursize, 1, cls.demofile);
buf.cursize = 0;
}
2012-07-22 13:34:45 +00:00
MSG_WriteByte(&buf, svc_spawnbaseline);
2012-07-22 13:34:45 +00:00
MSG_WriteDeltaEntity(&nullstate, &cl_entities[i].baseline,
&buf, true, true);
}
2012-07-22 13:34:45 +00:00
MSG_WriteByte(&buf, svc_stufftext);
2012-07-22 13:34:45 +00:00
MSG_WriteString(&buf, "precache\n");
/* write it to the demo file */
2012-07-22 13:34:45 +00:00
len = LittleLong(buf.cursize);
fwrite(&len, 4, 1, cls.demofile);
fwrite(buf.data, buf.cursize, 1, cls.demofile);
}
2012-07-22 13:34:45 +00:00
void
CL_Setenv_f(void)
{
int argc = Cmd_Argc();
2012-07-22 13:34:45 +00:00
if (argc > 2)
{
char buffer[1000];
int i;
2012-07-22 13:34:45 +00:00
strcpy(buffer, Cmd_Argv(1));
strcat(buffer, "=");
2012-07-22 13:34:45 +00:00
for (i = 2; i < argc; i++)
{
2012-07-22 13:34:45 +00:00
strcat(buffer, Cmd_Argv(i));
strcat(buffer, " ");
}
2012-07-22 13:34:45 +00:00
putenv(buffer);
}
2012-07-22 13:34:45 +00:00
else if (argc == 2)
{
2012-07-22 13:34:45 +00:00
char *env = getenv(Cmd_Argv(1));
2012-07-22 13:34:45 +00:00
if (env)
{
2012-07-22 13:34:45 +00:00
Com_Printf("%s=%s\n", Cmd_Argv(1), env);
}
else
{
2012-07-22 13:34:45 +00:00
Com_Printf("%s undefined\n", Cmd_Argv(1), env);
}
}
}
2012-07-22 13:34:45 +00:00
void
CL_Pause_f(void)
{
/* never pause in multiplayer */
2012-07-22 13:34:45 +00:00
if ((Cvar_VariableValue("maxclients") > 1) || !Com_ServerState())
{
2012-07-22 13:34:45 +00:00
Cvar_SetValue("paused", 0);
return;
}
2012-07-22 13:34:45 +00:00
Cvar_SetValue("paused", !cl_paused->value);
}
2012-07-22 13:34:45 +00:00
void
CL_Quit_f(void)
{
2012-07-22 13:34:45 +00:00
CL_Disconnect();
Com_Quit();
}
2012-07-22 13:34:45 +00:00
void
CL_ClearState(void)
{
2012-07-22 13:34:45 +00:00
S_StopAllSounds();
CL_ClearEffects();
CL_ClearTEnts();
/* wipe the entire cl structure */
2012-07-22 13:34:45 +00:00
memset(&cl, 0, sizeof(cl));
memset(&cl_entities, 0, sizeof(cl_entities));
2012-07-22 13:34:45 +00:00
SZ_Clear(&cls.netchan.message);
}
/*
* Handle a reply from a ping
*/
2012-07-22 13:34:45 +00:00
void
CL_ParseStatusMessage(void)
{
2012-07-22 13:34:45 +00:00
char *s;
s = MSG_ReadString(&net_message);
2012-07-22 13:34:45 +00:00
Com_Printf("%s\n", s);
M_AddToServerList(net_from, s);
}
/*
* Load or download any custom player skins and models
*/
2012-07-22 13:34:45 +00:00
void
CL_Skins_f(void)
{
2012-07-22 13:34:45 +00:00
int i;
2012-07-22 13:34:45 +00:00
for (i = 0; i < MAX_CLIENTS; i++)
{
2012-07-22 13:34:45 +00:00
if (!cl.configstrings[CS_PLAYERSKINS + i][0])
{
continue;
2012-07-22 13:34:45 +00:00
}
2012-07-22 13:34:45 +00:00
Com_Printf("client %i: %s\n", i, cl.configstrings[CS_PLAYERSKINS + i]);
2012-07-22 13:34:45 +00:00
SCR_UpdateScreen();
2012-07-22 13:34:45 +00:00
Sys_SendKeyEvents(); /* pump message loop */
2012-07-22 13:34:45 +00:00
CL_ParseClientinfo(i);
}
}
/* This fixes some problems with wrong tagged models and skins */
2012-07-22 13:34:45 +00:00
void
CL_FixUpGender(void)
{
char *p;
char sk[80];
if (gender_auto->value)
{
if (gender->modified)
{
/* was set directly, don't override the user */
gender->modified = false;
return;
}
Q_strlcpy(sk, skin->string, sizeof(sk));
if ((p = strchr(sk, '/')) != NULL)
2012-07-22 13:34:45 +00:00
{
*p = 0;
2012-07-22 13:34:45 +00:00
}
2012-07-22 13:34:45 +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
2012-07-22 13:34:45 +00:00
{
Cvar_Set("gender", "none");
}
gender->modified = false;
}
}
2012-07-22 13:34:45 +00:00
void
CL_Userinfo_f(void)
{
2012-07-22 13:34:45 +00:00
Com_Printf("User info settings:\n");
Info_Print(Cvar_Userinfo());
}
/*
* Restart the sound subsystem so it can pick up
* new parameters and flush all sounds
*/
2012-07-22 13:34:45 +00:00
void
CL_Snd_Restart_f(void)
{
S_Shutdown();
2012-07-22 13:34:45 +00:00
S_Init();
CL_RegisterSounds();
}
int precache_check;
int precache_spawncount;
int precache_tex;
int precache_model_skin;
byte *precache_model;
/*
* The server will send this command right
* before allowing the client into the server
*/
2012-07-22 13:34:45 +00:00
void
CL_Precache_f(void)
{
/* Yet another hack to let old demos work */
if (Cmd_Argc() < 2)
{
2012-07-22 13:34:45 +00:00
unsigned map_checksum; /* for detecting cheater maps */
2012-07-22 13:34:45 +00:00
CM_LoadMap(cl.configstrings[CS_MODELS + 1], true, &map_checksum);
CL_RegisterSounds();
CL_PrepRefresh();
return;
}
precache_check = CS_MODELS;
2012-06-02 07:31:27 +00:00
precache_spawncount = (int)strtol(Cmd_Argv(1), (char **)NULL, 10);
precache_model = 0;
precache_model_skin = 0;
CL_RequestNextDownload();
}
2012-07-22 13:34:45 +00:00
void
CL_InitLocal(void)
{
cls.state = ca_disconnected;
2012-07-22 13:34:45 +00:00
cls.realtime = Sys_Milliseconds();
2012-07-22 13:34:45 +00:00
CL_InitInput();
2012-07-22 13:34:45 +00:00
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);
cin_force43 = Cvar_Get("cin_force43", "1", 0);
/* register our variables */
2012-07-22 13:34:45 +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);
cl_gun = Cvar_Get("cl_gun", "2", CVAR_ARCHIVE);
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);
2012-10-06 07:15:32 +00:00
cl_maxfps = Cvar_Get("cl_maxfps", "95", CVAR_ARCHIVE);
2012-07-22 13:34:45 +00:00
cl_drawfps = Cvar_Get("cl_drawfps", "0", CVAR_ARCHIVE);
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);
cl_run = Cvar_Get("cl_run", "0", CVAR_ARCHIVE);
freelook = Cvar_Get("freelook", "1", CVAR_ARCHIVE);
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("gl_lightlevel", "0", 0);
/* userinfo */
2012-07-22 13:34:45 +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);
rate = Cvar_Get("rate", "8000", CVAR_USERINFO | CVAR_ARCHIVE);
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);
horplus = Cvar_Get("horplus", "1", CVAR_ARCHIVE);
windowed_mouse = Cvar_Get("windowed_mouse", "1", CVAR_USERINFO | CVAR_ARCHIVE);
2012-07-22 13:34:45 +00:00
gender = Cvar_Get("gender", "male", CVAR_USERINFO | CVAR_ARCHIVE);
gender_auto = Cvar_Get("gender_auto", "1", CVAR_ARCHIVE);
gender->modified = false;
2012-07-22 13:34:45 +00:00
cl_vwep = Cvar_Get("cl_vwep", "1", CVAR_ARCHIVE);
/* register our commands */
2012-07-22 13:34:45 +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);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("userinfo", CL_Userinfo_f);
Cmd_AddCommand("snd_restart", CL_Snd_Restart_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("changing", CL_Changing_f);
Cmd_AddCommand("disconnect", CL_Disconnect_f);
Cmd_AddCommand("record", CL_Record_f);
Cmd_AddCommand("stop", CL_Stop_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("quit", CL_Quit_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("connect", CL_Connect_f);
Cmd_AddCommand("reconnect", CL_Reconnect_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("rcon", CL_Rcon_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("setenv", CL_Setenv_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("precache", CL_Precache_f);
2012-07-22 13:34:45 +00:00
Cmd_AddCommand("download", CL_Download_f);
/* forward to server commands
2012-07-22 13:34:45 +00:00
* the only thing this does is allow command completion
* to work -- all unknown commands are automatically
* forwarded to the server */
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);
}
/*
* Writes key bindings and archived cvars to config.cfg
*/
2012-07-22 13:34:45 +00:00
void
CL_WriteConfiguration(void)
{
2012-07-22 13:34:45 +00:00
FILE *f;
char path[MAX_OSPATH];
if (cls.state == ca_uninitialized)
2012-07-22 13:34:45 +00:00
{
return;
2012-07-22 13:34:45 +00:00
}
2012-07-22 13:34:45 +00:00
Com_sprintf(path, sizeof(path), "%s/config.cfg", FS_Gamedir());
2012-07-22 13:34:45 +00:00
f = fopen(path, "w");
if (!f)
{
2012-07-22 13:34:45 +00:00
Com_Printf("Couldn't write config.cfg.\n");
return;
}
2012-07-22 13:34:45 +00:00
fprintf(f, "// generated by quake, do not modify\n");
2012-07-22 13:34:45 +00:00
Key_WriteBindings(f);
fclose(f);
2012-07-22 13:34:45 +00:00
Cvar_WriteVariables(path);
}
typedef struct
{
2012-07-22 13:34:45 +00:00
char *name;
char *value;
cvar_t *var;
} cheatvar_t;
2012-07-22 13:34:45 +00:00
cheatvar_t cheatvars[] = {
{"timescale", "1"},
{"timedemo", "0"},
2010-10-24 08:50:01 +00:00
{"gl_drawworld", "1"},
{"cl_testlights", "0"},
2010-10-24 08:50:01 +00:00
{"gl_fullbright", "0"},
{"gl_drawflat", "0"},
{"paused", "0"},
{"fixedtime", "0"},
{"sw_draworder", "0"},
{"gl_lightmap", "0"},
{"gl_saturatelighting", "0"},
{NULL, NULL}
};
2012-07-22 13:34:45 +00:00
int numcheatvars;
2012-07-22 13:34:45 +00:00
void
CL_FixCvarCheats(void)
{
2012-07-22 13:34:45 +00:00
int i;
cheatvar_t *var;
2012-07-22 13:34:45 +00:00
if (!strcmp(cl.configstrings[CS_MAXCLIENTS], "1") ||
!cl.configstrings[CS_MAXCLIENTS][0])
{
return; /* single player can cheat */
2012-07-22 13:34:45 +00:00
}
/* find all the cvars if we haven't done it yet */
if (!numcheatvars)
{
while (cheatvars[numcheatvars].name)
{
2012-07-22 13:34:45 +00:00
cheatvars[numcheatvars].var = Cvar_Get(cheatvars[numcheatvars].name,
cheatvars[numcheatvars].value, 0);
numcheatvars++;
}
}
/* make sure they are all set to the proper values */
2012-07-22 13:34:45 +00:00
for (i = 0, var = cheatvars; i < numcheatvars; i++, var++)
{
2012-07-22 13:34:45 +00:00
if (strcmp(var->var->string, var->value))
{
2012-07-22 13:34:45 +00:00
Cvar_Set(var->name, var->value);
}
}
}
void
CL_UpdateWindowedMouse(void)
{
2012-11-13 21:37:17 +00:00
if (cls.disable_screen)
{
return;
}
if (cls.key_dest == key_menu || cls.key_dest == key_console ||
2012-11-13 21:37:17 +00:00
(cls.key_dest == key_game && (cls.state != ca_active || !cl.refresh_prepped)))
{
if (windowed_mouse->value)
{
Cvar_SetValue("windowed_mouse", 0);
}
}
else
{
if (!windowed_mouse->value)
{
Cvar_SetValue("windowed_mouse", 1);
}
}
}
2012-07-22 13:34:45 +00:00
void
CL_SendCommand(void)
{
/* update windowed_mouse cvar */
CL_UpdateWindowedMouse();
/* get new key events */
2012-07-22 13:34:45 +00:00
Sys_SendKeyEvents();
/* allow mice or other external controllers to add commands */
2012-07-22 13:34:45 +00:00
IN_Commands();
/* process console commands */
2012-07-22 13:34:45 +00:00
Cbuf_Execute();
/* fix any cheating cvars */
2012-07-22 13:34:45 +00:00
CL_FixCvarCheats();
/* send intentions now */
2012-07-22 13:34:45 +00:00
CL_SendCmd();
/* resend a connection request if necessary */
2012-07-22 13:34:45 +00:00
CL_CheckForResend();
}
2012-07-22 13:34:45 +00:00
void
CL_Frame(int msec)
{
2012-07-22 13:34:45 +00:00
static int extratime;
static int lasttimecalled;
if (dedicated->value)
2012-07-22 13:34:45 +00:00
{
return;
2012-07-22 13:34:45 +00:00
}
extratime += msec;
if (!cl_timedemo->value)
{
2012-07-22 13:34:45 +00:00
if ((cls.state == ca_connected) && (extratime < 100))
{
return; /* don't flood packets out while connecting */
2012-07-22 13:34:45 +00:00
}
2012-07-22 13:34:45 +00:00
if (extratime < 1000 / cl_maxfps->value)
{
return; /* framerate is too high */
2012-07-22 13:34:45 +00:00
}
}
/* decide the simulation time */
2012-07-22 13:34:45 +00:00
cls.frametime = extratime / 1000.0;
cl.time += extratime;
cls.realtime = curtime;
extratime = 0;
if (cls.frametime > (1.0 / 5))
2012-07-22 13:34:45 +00:00
{
cls.frametime = (1.0 / 5);
2012-07-22 13:34:45 +00:00
}
/* if in the debugger last frame, don't timeout */
if (msec > 5000)
2012-07-22 13:34:45 +00:00
{
cls.netchan.last_received = Sys_Milliseconds();
}
/* fetch results from server */
2012-07-22 13:34:45 +00:00
CL_ReadPackets();
/* send a new command message to the server */
2012-07-22 13:34:45 +00:00
CL_SendCommand();
/* predict all unacknowledged movements */
2012-07-22 13:34:45 +00:00
CL_PredictMovement();
/* allow renderer DLL change */
2012-07-22 13:34:45 +00:00
VID_CheckChanges();
2012-07-22 13:34:45 +00:00
if (!cl.refresh_prepped && (cls.state == ca_active))
{
CL_PrepRefresh();
}
/* update the screen */
if (host_speeds->value)
2012-07-22 13:34:45 +00:00
{
time_before_ref = Sys_Milliseconds();
}
2012-07-22 13:34:45 +00:00
SCR_UpdateScreen();
if (host_speeds->value)
2012-07-22 13:34:45 +00:00
{
time_after_ref = Sys_Milliseconds();
}
/* update audio */
2012-07-22 13:34:45 +00:00
S_Update(cl.refdef.vieworg, cl.v_forward, cl.v_right, cl.v_up);
2012-04-25 08:53:25 +00:00
#ifdef CDA
CDAudio_Update();
2012-04-25 08:53:25 +00:00
#endif
/* advance local effects for next frame */
2012-07-22 13:34:45 +00:00
CL_RunDLights();
2012-07-22 13:34:45 +00:00
CL_RunLightStyles();
2012-07-22 13:34:45 +00:00
SCR_RunCinematic();
2012-07-22 13:34:45 +00:00
SCR_RunConsole();
cls.framecount++;
2012-07-22 13:34:45 +00:00
if (log_stats->value)
{
2012-07-22 13:34:45 +00:00
if (cls.state == ca_active)
{
2012-07-22 13:34:45 +00:00
if (!lasttimecalled)
{
lasttimecalled = Sys_Milliseconds();
2012-07-22 13:34:45 +00:00
if (log_stats_file)
{
fprintf(log_stats_file, "0\n");
}
}
else
{
int now = Sys_Milliseconds();
2012-07-22 13:34:45 +00:00
if (log_stats_file)
{
fprintf(log_stats_file, "%d\n", now - lasttimecalled);
}
lasttimecalled = now;
}
}
}
}
2012-07-22 13:34:45 +00:00
void
CL_Init(void)
{
if (dedicated->value)
2012-07-22 13:34:45 +00:00
{
return; /* nothing running on the client */
2012-07-22 13:34:45 +00:00
}
/* all archived variables will now be loaded */
2012-07-22 13:34:45 +00:00
Con_Init();
2012-07-22 13:34:45 +00:00
S_Init();
2012-07-22 13:34:45 +00:00
VID_Init();
2012-07-22 13:34:45 +00:00
V_Init();
net_message.data = net_message_buffer;
net_message.maxsize = sizeof(net_message_buffer);
2012-07-22 13:34:45 +00:00
M_Init();
2012-07-22 13:34:45 +00:00
SCR_Init();
cls.disable_screen = true; /* don't draw yet */
2012-04-25 08:53:25 +00:00
#ifdef CDA
2012-07-22 13:34:45 +00:00
CDAudio_Init();
2012-04-25 08:53:25 +00:00
#endif
2012-07-22 13:34:45 +00:00
CL_InitLocal();
2012-07-22 13:34:45 +00:00
FS_ExecAutoexec();
2012-07-22 13:34:45 +00:00
Cbuf_Execute();
}
2012-07-22 13:34:45 +00:00
void
CL_Shutdown(void)
{
static qboolean isdown = false;
if (isdown)
{
2012-07-22 13:34:45 +00:00
printf("recursive shutdown\n");
return;
}
isdown = true;
2012-07-22 13:34:45 +00:00
CL_WriteConfiguration();
2012-04-25 08:53:25 +00:00
#ifdef CDA
2012-07-22 13:34:45 +00:00
CDAudio_Shutdown();
2012-04-25 08:53:25 +00:00
#endif
2012-04-25 08:24:38 +00:00
#ifdef OGG
OGG_Stop();
2012-04-25 08:24:38 +00:00
#endif
S_Shutdown();
VID_Shutdown();
}
2012-07-22 13:34:45 +00:00