2004-08-23 00:15:46 +00:00
/*
Copyright ( C ) 1996 - 1997 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
2005-09-26 03:40:09 +00:00
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
2004-08-23 00:15:46 +00:00
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 .
*/
// cl_parse.c -- parse a message received from the server
# include "quakedef.h"
2005-11-30 01:20:53 +00:00
# include "cl_ignore.h"
2004-08-23 00:15:46 +00:00
void CL_GetNumberedEntityInfo ( int num , float * org , float * ang ) ;
2004-11-27 08:16:25 +00:00
void CLNQ_ParseDarkPlaces5Entities ( void ) ;
2008-05-25 22:23:43 +00:00
void CL_SetStatInt ( int pnum , int stat , int value ) ;
2008-11-14 16:43:28 +00:00
static qboolean CL_CheckModelResources ( char * name ) ;
2004-08-23 00:15:46 +00:00
2005-09-07 14:55:25 +00:00
int msgflags ;
2004-08-23 00:15:46 +00:00
2008-11-09 22:29:28 +00:00
char cl_dp_csqc_progsname [ 128 ] ;
2007-06-20 00:02:54 +00:00
int cl_dp_csqc_progssize ;
int cl_dp_csqc_progscrc ;
2008-12-03 02:42:05 +00:00
int cl_dp_serverextension_download ;
2007-06-20 00:02:54 +00:00
2004-08-23 00:15:46 +00:00
char * svc_strings [ ] =
{
" svc_bad " ,
" svc_nop " ,
" svc_disconnect " ,
" svc_updatestat " ,
" svc_version " , // [long] server version
" svc_setview " , // [short] entity number
" svc_sound " , // <see code>
" svc_time " , // [float] server time
" svc_print " , // [string] null terminated string
" svc_stufftext " , // [string] stuffed into client's console buffer
// the string should be \n terminated
" svc_setangle " , // [vec3] set the view angle to this absolute value
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
" svc_serverdata " , // [long] version ...
" svc_lightstyle " , // [qbyte] [string]
" svc_updatename " , // [qbyte] [string]
" svc_updatefrags " , // [qbyte] [short]
" svc_clientdata " , // <shortbits + data>
" svc_stopsound " , // <see code>
" svc_updatecolors " , // [qbyte] [qbyte]
" svc_particle " , // [vec3] <variable>
" svc_damage " , // [qbyte] impact [qbyte] blood [vec3] from
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
" svc_spawnstatic " ,
" svc_spawnstatic2 " ,
" svc_spawnbaseline " ,
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
" svc_temp_entity " , // <variable>
" svc_setpause " ,
" svc_signonnum " ,
" svc_centerprint " ,
" svc_killedmonster " ,
" svc_foundsecret " ,
" svc_spawnstaticsound " ,
" svc_intermission " ,
" svc_finale " ,
" svc_cdtrack " ,
" svc_sellscreen " ,
" svc_smallkick " ,
" svc_bigkick " ,
" svc_updateping " ,
" svc_updateentertime " ,
" svc_updatestatlong " ,
" svc_muzzleflash " ,
" svc_updateuserinfo " ,
" svc_download " ,
" svc_playerinfo " ,
" svc_nails " ,
" svc_choke " ,
" svc_modellist " ,
" svc_soundlist " ,
" svc_packetentities " ,
" svc_deltapacketentities " ,
" svc_maxspeed " ,
" svc_entgravity " ,
" svc_setinfo " ,
" svc_serverinfo " ,
" svc_updatepl " ,
2006-03-14 01:21:18 +00:00
" MVD svc_nails2 " ,
" BAD svc_unused " ,
" FTE svc_view2 " ,
" FTE svc_lightstylecol " ,
" FTE svc_bulletentext " ,
" FTE svc_lightnings " ,
" FTE svc_modellistshort " ,
" FTE svc_ftesetclientpersist " ,
" FTE svc_setportalstate " ,
" FTE svc_particle2 " ,
" FTE svc_particle3 " ,
" FTE svc_particle4 " ,
" FTE svc_spawnbaseline2 " ,
" FTE svc_customtempent " ,
" FTE svc_choosesplitclient " ,
" FTE svc_showpic " ,
" FTE svc_hidepic " ,
" FTE svc_movepic " ,
" FTE svc_updatepic " ,
" FTE svcqw_effect " ,
" FTE svcqw_effect2 " ,
" FTE svc_csqcentities " ,
" FTE svc_precache " ,
" FTE svc_choosesplitclient " ,
2004-08-23 00:15:46 +00:00
} ;
char * svc_nqstrings [ ] =
{
" nqsvc_bad " ,
" nqsvc_nop " ,
" nqsvc_disconnect " ,
" nqsvc_updatestat " ,
" nqsvc_version " , // [long] server version
" nqsvc_setview " , // [short] entity number
" nqsvc_sound " , // <see code>
" nqsvc_time " , // [float] server time
" nqsvc_print " , // [string] null terminated string
" nqsvc_stufftext " , // [string] stuffed into client's console buffer
// the string should be \n terminated
" nqsvc_setangle " , // [vec3] set the view angle to this absolute value
" nqsvc_serverinfo " , // [long] version
// [string] signon string
// [string]..[0]model cache [string]...[0]sounds cache
// [string]..[0]item cache
" nqsvc_lightstyle " , // [qbyte] [string]
" nqsvc_updatename " , // [qbyte] [string]
" nqsvc_updatefrags " , // [qbyte] [short]
" nqsvc_clientdata " , // <shortbits + data>
" nqsvc_stopsound " , // <see code>
" nqsvc_updatecolors " , // [qbyte] [qbyte]
" nqsvc_particle " , // [vec3] <variable>
" nqsvc_damage " , // [qbyte] impact [qbyte] blood [vec3] from
" nqsvc_spawnstatic " ,
" nqOBSOLETE svc_spawnbinary " ,
" nqsvc_spawnbaseline " ,
" nqsvc_temp_entity " , // <variable>
" nqsvc_setpause " ,
" nqsvc_signonnum " ,
" nqsvc_centerprint " ,
" nqsvc_killedmonster " ,
" nqsvc_foundsecret " ,
" nqsvc_spawnstaticsound " ,
" nqsvc_intermission " ,
" nqsvc_finale " , // [string] music [string] text
" nqsvc_cdtrack " , // [qbyte] track [qbyte] looptrack
" nqsvc_sellscreen " ,
2005-09-26 03:40:09 +00:00
" nqsvc_cutscene " , //34
2004-08-23 00:15:46 +00:00
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
2005-09-26 03:40:09 +00:00
" NEW PROTOCOL " , //40
2004-08-23 00:15:46 +00:00
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
" NEW PROTOCOL " ,
2005-09-26 03:40:09 +00:00
" NEW PROTOCOL " ,
2007-06-20 00:02:54 +00:00
" dpsvc_downloaddata " , //50
2008-11-09 22:29:28 +00:00
" dpsvc_updatestatubyte " , //51
" dpsvc_effect " , //52
" dpsvc_effect2 " , //53
" dp6svc_precache/dp5svc_sound2 " , //54
" dpsvc_spawnbaseline2 " , //55
" dpsvc_spawnstatic2 " , //56 obsolete
" dpsvc_entities " , //57
" NEW PROTOCOL " , //58
" dpsvc_spawnstaticsound2 " , //59
" dpsvc_trailparticles " , //60
" dpsvc_pointparticles " , //61
" dpsvc_pointparticles1 " //62
2004-08-23 00:15:46 +00:00
} ;
2010-03-25 22:56:11 +00:00
extern cvar_t requiredownloads , cl_standardchat , msg_filter , cl_countpendingpl , cl_download_mapsrc ;
2004-08-23 00:15:46 +00:00
int oldparsecountmod ;
int parsecountmod ;
double parsecounttime ;
2005-01-18 20:15:20 +00:00
int cl_spikeindex , cl_playerindex , cl_h_playerindex , cl_flagindex , cl_rocketindex , cl_grenadeindex , cl_gib1index , cl_gib2index , cl_gib3index ;
2004-08-23 00:15:46 +00:00
//=============================================================================
int packet_latency [ NET_TIMINGS ] ;
int CL_CalcNet ( void )
{
int a , i ;
frame_t * frame ;
int lost ;
2006-04-02 23:25:03 +00:00
int percent ;
int sent ;
int pending ;
2004-08-23 00:15:46 +00:00
// char st[80];
2006-04-02 23:25:03 +00:00
sent = NET_TIMINGS ;
2004-08-23 00:15:46 +00:00
for ( i = cls . netchan . outgoing_sequence - UPDATE_BACKUP + 1
; i < = cls . netchan . outgoing_sequence
; i + + )
{
frame = & cl . frames [ i & UPDATE_MASK ] ;
if ( frame - > receivedtime = = - 1 )
packet_latency [ i & NET_TIMINGSMASK ] = 9999 ; // dropped
else if ( frame - > receivedtime = = - 2 )
packet_latency [ i & NET_TIMINGSMASK ] = 10000 ; // choked
2006-04-02 23:25:03 +00:00
else if ( frame - > receivedtime = = - 3 )
{
packet_latency [ i & NET_TIMINGSMASK ] = 9997 ; // c2spps
sent - - ;
}
2004-08-23 00:15:46 +00:00
else if ( frame - > invalid )
packet_latency [ i & NET_TIMINGSMASK ] = 9998 ; // invalid delta
else
packet_latency [ i & NET_TIMINGSMASK ] = ( frame - > receivedtime - frame - > senttime ) * 20 ;
}
lost = 0 ;
for ( a = 0 ; a < NET_TIMINGS ; a + + )
{
i = ( cls . netchan . outgoing_sequence - a ) & NET_TIMINGSMASK ;
if ( packet_latency [ i ] = = 9999 )
lost + + ;
}
2006-04-02 23:25:03 +00:00
2009-11-04 21:16:50 +00:00
if ( ! cl_countpendingpl . ival )
2006-04-02 23:25:03 +00:00
{
pending = cls . netchan . outgoing_sequence - cls . netchan . incoming_sequence - 1 ;
lost - = pending ;
sent - = pending ;
if ( sent < 1 )
percent = 100 ;
else
percent = lost * 100 / sent ;
if ( lost & & ! percent ) //if they have any confirmed lost packets, report at least 1%
percent = 1 ;
}
else
2006-04-02 23:52:05 +00:00
{
if ( sent < 1 )
percent = 100 ; //shouldn't ever happen.
else
percent = lost * 100 / sent ;
}
2006-04-02 23:25:03 +00:00
return percent ;
2004-08-23 00:15:46 +00:00
}
//=============================================================================
2005-01-13 16:29:20 +00:00
//note: this will overwrite existing files.
//returns true if the download is going to be downloaded after the call.
2008-11-09 22:29:28 +00:00
qboolean CL_EnqueDownload ( char * filename , char * localname , unsigned int flags )
2005-01-13 16:29:20 +00:00
{
downloadlist_t * dl ;
2010-03-25 22:56:11 +00:00
qboolean webdl = false ;
if ( localname & & ! strncmp ( filename , " http:// " , 7 ) )
{
webdl = true ;
}
else
{
if ( ! localname )
localname = filename ;
if ( cls . demoplayback & & cls . demoplayback ! = DPB_EZTV )
return false ;
}
2005-08-11 04:14:33 +00:00
if ( strchr ( localname , ' \\ ' ) | | strchr ( localname , ' : ' ) | | strstr ( localname , " .. " ) )
2005-01-13 16:29:20 +00:00
{
Con_Printf ( " Denying download of \" %s \" \n " , filename ) ;
return false ;
}
2008-11-09 22:29:28 +00:00
if ( ! ( flags & DLLF_IGNOREFAILED ) )
2005-01-13 16:29:20 +00:00
{
2008-12-03 02:42:05 +00:00
# ifdef NQPROT
2010-03-25 22:56:11 +00:00
if ( ! webdl & & cls . protocol = = CP_NETQUAKE )
2008-12-03 02:42:05 +00:00
if ( ! cl_dp_serverextension_download )
return false ;
# endif
2005-01-13 16:29:20 +00:00
for ( dl = cl . faileddownloads ; dl ; dl = dl - > next ) //yeah, so it failed... Ignore it.
{
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( dl - > rname , filename ) )
2005-01-13 16:29:20 +00:00
{
2008-11-09 22:29:28 +00:00
if ( flags & DLLF_VERBOSE )
2005-01-13 16:29:20 +00:00
Con_Printf ( " We've failed to download \" %s \" already \n " , filename ) ;
return false ;
}
}
}
for ( dl = cl . downloadlist ; dl ; dl = dl - > next ) //It's already on our list. Ignore it.
{
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( dl - > rname , filename ) )
2005-01-13 16:29:20 +00:00
{
2008-11-09 22:29:28 +00:00
if ( flags & DLLF_VERBOSE )
2005-01-13 16:29:20 +00:00
Con_Printf ( " Already waiting for \" %s \" \n " , filename ) ;
return true ;
}
}
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( cls . downloadremotename , filename ) )
2005-01-13 16:29:20 +00:00
{
2008-11-09 22:29:28 +00:00
if ( flags & DLLF_VERBOSE )
2005-01-13 16:29:20 +00:00
Con_Printf ( " Already downloading \" %s \" \n " , filename ) ;
return true ;
}
2008-11-09 22:29:28 +00:00
if ( ! * filename )
{
Con_Printf ( " Download \" \" ? Huh? \n " ) ;
return true ;
}
2005-01-13 16:29:20 +00:00
dl = Z_Malloc ( sizeof ( downloadlist_t ) ) ;
2009-05-24 10:11:17 +00:00
Q_strncpyz ( dl - > rname , filename , sizeof ( dl - > rname ) ) ;
2005-08-11 04:14:33 +00:00
Q_strncpyz ( dl - > localname , localname , sizeof ( dl - > localname ) ) ;
2005-01-13 16:29:20 +00:00
dl - > next = cl . downloadlist ;
cl . downloadlist = dl ;
2008-11-09 22:29:28 +00:00
dl - > size = 0 ;
dl - > flags = flags | DLLF_SIZEUNKNOWN ;
2005-01-13 16:29:20 +00:00
2010-03-25 22:56:11 +00:00
if ( ! webdl & & ( cls . fteprotocolextensions & ( PEXT_CHUNKEDDOWNLOADS
2008-11-09 22:29:28 +00:00
# ifdef PEXT_PK3DOWNLOADS
| PEXT_PK3DOWNLOADS
# endif
2010-03-25 22:56:11 +00:00
) ) )
2009-05-24 10:11:17 +00:00
CL_SendClientCommand ( true , " dlsize \" %s \" " , dl - > rname ) ;
2008-11-09 22:29:28 +00:00
if ( flags & DLLF_VERBOSE )
2005-01-13 16:29:20 +00:00
Con_Printf ( " Enqued download of \" %s \" \n " , filename ) ;
return true ;
}
2008-11-09 22:29:28 +00:00
# pragma message("fix this")
int downloadsize ;
void CL_GetDownloadSizes ( unsigned int * filecount , unsigned int * totalsize , qboolean * somesizesunknown )
{
downloadlist_t * dl ;
* filecount = 0 ;
* totalsize = 0 ;
* somesizesunknown = false ;
for ( dl = cl . downloadlist ; dl ; dl = dl - > next )
{
2009-04-06 00:34:32 +00:00
* filecount + = 1 ;
2008-11-09 22:29:28 +00:00
if ( dl - > flags & DLLF_SIZEUNKNOWN )
* somesizesunknown = true ;
else
* totalsize + = dl - > size ;
}
if ( cls . downloadmethod = = DL_QW )
{
* totalsize + = downloadsize ;
* somesizesunknown = true ;
}
if ( cls . downloadmethod = = DL_QWCHUNKS )
* totalsize + = downloadsize ;
}
2005-01-16 02:25:35 +00:00
void CL_DisenqueDownload ( char * filename )
2005-01-13 16:29:20 +00:00
{
downloadlist_t * dl , * nxt ;
if ( cl . downloadlist ) //remove from enqued download list
{
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( cl . downloadlist - > rname , filename ) )
2005-01-13 16:29:20 +00:00
{
dl = cl . downloadlist ;
cl . downloadlist = cl . downloadlist - > next ;
Z_Free ( dl ) ;
}
else
{
2009-04-06 00:34:32 +00:00
for ( dl = cl . downloadlist ; dl - > next ; dl = dl - > next )
2005-01-13 16:29:20 +00:00
{
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( dl - > next - > rname , filename ) )
2005-01-13 16:29:20 +00:00
{
nxt = dl - > next - > next ;
Z_Free ( dl - > next ) ;
dl - > next = nxt ;
break ;
}
}
}
}
2005-01-13 23:33:00 +00:00
}
2005-01-13 16:29:20 +00:00
2010-07-11 02:22:39 +00:00
# ifdef WEBCLIENT
2010-03-25 22:56:11 +00:00
void CL_WebDownloadFinished ( struct dl_download * dl )
{
if ( dl - > status = = DL_FAILED )
CL_DownloadFailed ( dl - > url ) ;
else if ( dl - > status = = DL_FINISHED )
{
if ( dl - > file )
VFS_CLOSE ( dl - > file ) ;
dl - > file = NULL ;
CL_DownloadFinished ( ) ;
}
}
2010-07-11 02:22:39 +00:00
# endif
2010-03-25 22:56:11 +00:00
2009-05-24 10:11:17 +00:00
void CL_SendDownloadStartRequest ( char * filename , char * localname )
2005-01-16 02:25:35 +00:00
{
2009-05-24 10:11:17 +00:00
strcpy ( cls . downloadremotename , filename ) ;
strcpy ( cls . downloadlocalname , localname ) ;
Con_TPrintf ( TL_DOWNLOADINGFILE , cls . downloadlocalname ) ;
2005-01-16 02:25:35 +00:00
// download to a temp name, and only rename
// to the real name when done, so if interrupted
// a runt file wont be left
2006-03-11 03:12:10 +00:00
COM_StripExtension ( localname , cls . downloadtempname , sizeof ( cls . downloadtempname ) - 5 ) ;
2005-01-16 02:25:35 +00:00
strcat ( cls . downloadtempname , " .tmp " ) ;
2010-07-11 02:22:39 +00:00
# ifdef WEBCLIENT
2010-03-25 22:56:11 +00:00
if ( ! strncmp ( cls . downloadremotename , " http:// " , 7 ) )
{
cls . downloadmethod = DL_HTTP ;
cls . downloadpercent = 0 ;
if ( ! HTTP_CL_Get ( cls . downloadremotename , cls . downloadtempname , CL_WebDownloadFinished ) )
CL_DownloadFailed ( cls . downloadremotename ) ;
}
else
2010-07-11 02:22:39 +00:00
# endif
2010-03-25 22:56:11 +00:00
{
CL_SendClientCommand ( true , " download %s " , filename ) ;
2005-01-16 02:25:35 +00:00
2010-03-25 22:56:11 +00:00
//prevent ftp/http from changing stuff
cls . downloadmethod = DL_QWPENDING ;
cls . downloadpercent = 0 ;
}
2005-01-16 02:25:35 +00:00
CL_DisenqueDownload ( filename ) ;
}
2005-01-13 23:33:00 +00:00
//Do any reloading for the file that just reloaded.
2009-05-24 10:11:17 +00:00
void CL_DownloadFinished ( void )
2005-01-13 23:33:00 +00:00
{
int i ;
extern int mod_numknown ;
2008-11-09 22:29:28 +00:00
char * ext ;
2005-01-13 23:33:00 +00:00
extern model_t mod_known [ ] ;
2005-01-13 16:29:20 +00:00
2009-05-24 10:11:17 +00:00
char * filename = cls . downloadlocalname ;
char * tempname = cls . downloadtempname ;
2005-01-16 02:25:35 +00:00
COM_RefreshFSCache_f ( ) ;
cls . downloadmethod = DL_NONE ;
// rename the temp file to it's final name
if ( tempname )
{
if ( strcmp ( tempname , filename ) )
{
if ( strncmp ( tempname , " skins/ " , 6 ) )
{
2009-07-27 07:34:10 +00:00
if ( FS_Rename ( tempname , filename , FS_GAME ) )
2009-07-05 18:45:53 +00:00
{
char nativetmp [ MAX_OSPATH ] , nativefinal [ MAX_OSPATH ] ; ;
FS_NativePath ( tempname , FS_GAME , nativetmp , sizeof ( nativetmp ) ) ;
FS_NativePath ( filename , FS_GAME , nativefinal , sizeof ( nativefinal ) ) ;
Con_Printf ( " Couldn't rename %s to %s \n " , nativetmp , nativefinal ) ;
}
2005-01-16 02:25:35 +00:00
}
else
{
2009-07-27 07:34:10 +00:00
if ( FS_Rename ( tempname + 6 , filename + 6 , FS_SKINS ) )
2009-07-05 18:45:53 +00:00
{
char nativetmp [ MAX_OSPATH ] , nativefinal [ MAX_OSPATH ] ; ;
FS_NativePath ( tempname + 6 , FS_SKINS , nativetmp , sizeof ( nativetmp ) ) ;
FS_NativePath ( filename + 6 , FS_SKINS , nativefinal , sizeof ( nativefinal ) ) ;
Con_Printf ( " Couldn't rename %s to %s \n " , nativetmp , nativefinal ) ;
}
2005-01-16 02:25:35 +00:00
}
}
}
2008-11-09 22:29:28 +00:00
ext = COM_FileExtension ( filename ) ;
if ( ! strcmp ( ext , " pk3 " ) | | ! strcmp ( ext , " pak " ) )
FS_ReloadPackFiles ( ) ;
else if ( ! strcmp ( filename , " gfx/palette.lmp " ) )
2005-01-13 23:33:00 +00:00
{
Cbuf_AddText ( " vid_restart \n " , RESTRICT_LOCAL ) ;
}
else
{
2008-11-28 20:34:51 +00:00
CL_CheckModelResources ( filename ) ;
2008-11-09 22:29:28 +00:00
if ( ! cl . sendprespawn )
2005-01-13 23:33:00 +00:00
{
2008-11-09 22:29:28 +00:00
for ( i = 0 ; i < mod_numknown ; i + + ) //go and load this model now.
2005-01-13 23:33:00 +00:00
{
2008-11-09 22:29:28 +00:00
if ( ! strcmp ( mod_known [ i ] . name , filename ) )
{
Mod_ForName ( mod_known [ i ] . name , false ) ; //throw away result.
break ;
}
2005-01-13 23:33:00 +00:00
}
2008-11-09 22:29:28 +00:00
for ( i = 0 ; i < MAX_MODELS ; i + + ) //go and load this model now.
2006-01-04 00:44:34 +00:00
{
2008-11-09 22:29:28 +00:00
if ( ! strcmp ( cl . model_name [ i ] , filename ) )
{
cl . model_precache [ i ] = Mod_ForName ( cl . model_name [ i ] , false ) ; //throw away result.
if ( i = = 1 )
cl . worldmodel = cl . model_precache [ i ] ;
break ;
}
2006-01-04 00:44:34 +00:00
}
2009-07-06 01:20:20 +00:00
for ( i = 0 ; i < MAX_VWEP_MODELS ; i + + )
{
if ( ! strcmp ( cl . model_name_vwep [ i ] , filename ) )
{
cl . model_precache_vwep [ i ] = Mod_ForName ( cl . model_name_vwep [ i ] , false ) ;
break ;
}
}
2006-01-04 00:44:34 +00:00
}
2006-10-05 22:11:17 +00:00
S_ResetFailedLoad ( ) ; //okay, so this can still get a little spammy in bad places...
2006-01-04 00:44:34 +00:00
//this'll do the magic for us
2005-01-13 23:33:00 +00:00
Skin_FlushSkin ( filename ) ;
}
2005-01-13 16:29:20 +00:00
}
2007-06-20 00:02:54 +00:00
qboolean CL_CheckFile ( char * filename )
{
if ( strstr ( filename , " .. " ) )
{
Con_TPrintf ( TL_NORELATIVEPATHS ) ;
return true ;
}
if ( COM_FCheckExists ( filename ) )
{ // it exists, no need to download
return true ;
}
return false ;
}
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = =
2006-01-04 00:44:34 +00:00
CL_CheckOrEnqueDownloadFile
2004-08-23 00:15:46 +00:00
2010-03-25 22:56:11 +00:00
Returns true if the file exists , returns false if it triggered a download .
2004-08-23 00:15:46 +00:00
= = = = = = = = = = = = = = =
*/
2007-06-20 00:02:54 +00:00
2008-11-09 22:29:28 +00:00
qboolean CL_CheckOrEnqueDownloadFile ( char * filename , char * localname , unsigned int flags )
2006-01-04 00:44:34 +00:00
{ //returns false if we don't have the file yet.
2005-08-11 04:14:33 +00:00
if ( ! localname )
localname = filename ;
2005-09-26 03:40:09 +00:00
2009-03-03 01:52:30 +00:00
# ifndef CLIENTONLY
if ( sv . state )
return true ;
# endif
2010-03-25 22:56:11 +00:00
if ( ! ( flags & DLLF_OVERWRITE ) & & CL_CheckFile ( localname ) )
return true ;
2004-08-23 00:15:46 +00:00
//ZOID - can't download when recording
2005-01-13 16:29:20 +00:00
if ( cls . demorecording )
{
2004-08-23 00:15:46 +00:00
Con_TPrintf ( TL_NODOWNLOADINDEMO , filename ) ;
return true ;
}
//ZOID - can't download when playback
2010-03-25 22:56:11 +00:00
// if (cls.demoplayback && cls.demoplayback != DPB_EZTV)
// return true;
2004-08-23 00:15:46 +00:00
2004-11-27 08:16:25 +00:00
SCR_EndLoadingPlaque ( ) ; //release console.
2010-03-25 22:56:11 +00:00
if ( * cl_download_mapsrc . string )
if ( ! strncmp ( filename , " maps/ " , 5 ) )
if ( ! strcmp ( filename + strlen ( filename ) - 4 , " .bsp " ) )
2005-06-04 04:20:20 +00:00
{
char base [ MAX_QPATH ] ;
2010-03-25 22:56:11 +00:00
COM_FileBase ( filename , base , sizeof ( base ) ) ;
filename = va ( " %s%s.bsp " , cl_download_mapsrc . string , base ) ;
2005-06-04 04:20:20 +00:00
}
2010-03-25 22:56:11 +00:00
if ( ! CL_EnqueDownload ( filename , localname , flags ) )
return true ; /*don't stall waiting for it if it failed*/
2008-11-09 22:29:28 +00:00
if ( ! ( flags & DLLF_IGNOREFAILED ) )
{
downloadlist_t * dl ;
for ( dl = cl . faileddownloads ; dl ; dl = dl - > next )
{
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( dl - > rname , filename ) )
2008-11-09 22:29:28 +00:00
{
//if its on the failed list, don't block waiting for it to download
return true ;
}
}
}
2006-01-04 00:44:34 +00:00
return false ;
2004-08-23 00:15:46 +00:00
}
2008-11-09 22:29:28 +00:00
static qboolean CL_CheckMD2Skins ( qbyte * precache_model )
2004-08-23 00:15:46 +00:00
{
2008-11-09 22:29:28 +00:00
qboolean ret = false ;
2004-08-23 00:15:46 +00:00
md2_t * pheader ;
2008-11-09 22:29:28 +00:00
int skin = 1 ;
2005-03-18 06:14:07 +00:00
char * str ;
2004-08-23 00:15:46 +00:00
pheader = ( md2_t * ) precache_model ;
2008-11-09 22:29:28 +00:00
if ( LittleLong ( pheader - > version ) ! = MD2ALIAS_VERSION )
{
//bad version.
2004-08-23 00:15:46 +00:00
return false ;
}
pheader = ( md2_t * ) precache_model ;
2008-11-09 22:29:28 +00:00
for ( skin = 0 ; skin < LittleLong ( pheader - > num_skins ) ; skin + + )
2005-03-18 06:14:07 +00:00
{
str = ( char * ) precache_model +
2005-09-26 03:40:09 +00:00
LittleLong ( pheader - > ofs_skins ) +
2008-11-09 22:29:28 +00:00
skin * MD2MAX_SKINNAME ;
2005-03-18 06:14:07 +00:00
COM_CleanUpPath ( str ) ;
2008-11-09 22:29:28 +00:00
if ( ! CL_CheckOrEnqueDownloadFile ( str , str , 0 ) )
ret = true ;
2004-08-23 00:15:46 +00:00
}
2008-11-09 22:29:28 +00:00
return ret ;
}
qboolean CL_CheckHLBspWads ( char * file )
{
lump_t lump ;
dheader_t * dh ;
char * s ;
char * w ;
char key [ 256 ] ;
char wads [ 4096 ] ;
dh = ( dheader_t * ) file ;
lump . fileofs = LittleLong ( dh - > lumps [ LUMP_ENTITIES ] . fileofs ) ;
lump . filelen = LittleLong ( dh - > lumps [ LUMP_ENTITIES ] . filelen ) ;
s = file + lump . fileofs ;
2005-09-26 03:40:09 +00:00
2008-11-09 22:29:28 +00:00
s = COM_Parse ( s ) ;
if ( strcmp ( com_token , " { " ) )
return false ;
while ( * s )
{
s = COM_ParseOut ( s , key , sizeof ( key ) ) ;
if ( ! strcmp ( key , " } " ) )
break ;
s = COM_ParseOut ( s , wads , sizeof ( wads ) ) ;
if ( ! strcmp ( key , " wad " ) )
{
s = wads ;
while ( s = COM_ParseToken ( s , " ; " ) )
{
if ( ! strcmp ( com_token , " ; " ) )
continue ;
while ( w = strchr ( com_token , ' \\ ' ) )
* w = ' / ' ;
w = COM_SkipPath ( com_token ) ;
Con_Printf ( " wads: %s \n " , w ) ;
if ( ! CL_CheckFile ( w ) )
CL_CheckOrEnqueDownloadFile ( va ( " textures/%s " , w ) , NULL , DLLF_REQUIRED ) ;
}
return false ;
}
}
2004-08-23 00:15:46 +00:00
return false ;
}
2006-06-04 16:02:03 +00:00
2008-11-09 22:29:28 +00:00
qboolean CL_CheckQ2BspWals ( char * file )
{
qboolean gotone = false ;
q2dheader_t * dh ;
lump_t lump ;
q2texinfo_t * tinf ;
unsigned int i , j , count ;
dh = ( q2dheader_t * ) file ;
if ( LittleLong ( dh - > version ) ! = Q2BSPVERSION )
{
//quake3? unknown?
return false ;
}
lump . fileofs = LittleLong ( dh - > lumps [ Q2LUMP_TEXINFO ] . fileofs ) ;
lump . filelen = LittleLong ( dh - > lumps [ Q2LUMP_TEXINFO ] . filelen ) ;
count = lump . filelen / sizeof ( * tinf ) ;
if ( lump . filelen ! = count * sizeof ( * tinf ) )
return false ;
tinf = ( q2texinfo_t * ) ( file + lump . fileofs ) ;
for ( i = 0 ; i < count ; i + + )
{
//ignore duplicate files (to save filesystem hits)
for ( j = 0 ; j < i ; j + + )
if ( ! strcmp ( tinf [ i ] . texture , tinf [ j ] . texture ) )
break ;
if ( i = = j )
if ( ! CL_CheckOrEnqueDownloadFile ( tinf [ i ] . texture , NULL , 0 ) )
gotone = true ;
}
return gotone ;
}
2008-11-14 16:43:28 +00:00
static qboolean CL_CheckModelResources ( char * name )
2008-11-09 22:29:28 +00:00
{
//returns true if we triggered a download
qboolean ret ;
qbyte * file ;
if ( ! ( strstr ( name , " .md2 " ) | | strstr ( name , " .bsp " ) ) )
return false ;
// checking for skins in the model
2009-05-24 10:11:17 +00:00
FS_LoadFile ( name , & file ) ;
2008-11-09 22:29:28 +00:00
if ( ! file )
{
return false ; // couldn't load it
}
if ( LittleLong ( * ( unsigned * ) file ) = = MD2IDALIASHEADER )
ret = CL_CheckMD2Skins ( file ) ;
else if ( LittleLong ( * ( unsigned * ) file ) = = BSPVERSIONHL )
ret = CL_CheckHLBspWads ( file ) ;
else if ( LittleLong ( * ( unsigned * ) file ) = = IDBSPHEADER )
ret = CL_CheckQ2BspWals ( file ) ;
else
ret = false ;
2009-05-24 10:11:17 +00:00
FS_FreeFile ( file ) ;
2008-11-09 22:29:28 +00:00
return ret ;
}
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = = = =
Model_NextDownload
= = = = = = = = = = = = = = = = =
*/
2009-04-06 00:34:32 +00:00
void Model_CheckDownloads ( void )
2004-08-23 00:15:46 +00:00
{
// char *twf;
char * s ;
int i ;
2006-02-17 19:54:47 +00:00
// extern char gamedirfile[];
2004-08-23 00:15:46 +00:00
2009-03-03 01:52:30 +00:00
// Con_TPrintf (TLC_CHECKINGMODELS);
2006-01-04 00:44:34 +00:00
2010-07-11 02:22:39 +00:00
R_SetSky ( cl . skyname ) ;
2004-08-23 00:15:46 +00:00
# ifdef Q2CLIENT
2005-05-26 12:55:34 +00:00
if ( cls . protocol = = CP_QUAKE2 )
2004-08-23 00:15:46 +00:00
{
for ( i = 0 ; i < Q2MAX_IMAGES ; i + + )
{
char picname [ 256 ] ;
if ( ! * cl . image_name [ i ] )
continue ;
sprintf ( picname , " pics/%s.pcx " , cl . image_name [ i ] ) ;
2008-11-09 22:29:28 +00:00
CL_CheckOrEnqueDownloadFile ( picname , picname , 0 ) ;
2004-08-23 00:15:46 +00:00
}
if ( ! CLQ2_RegisterTEntModels ( ) )
return ;
}
# endif
2004-11-17 17:58:22 +00:00
2006-03-26 17:08:19 +00:00
for ( i = 1 ; cl . model_name [ i ] [ 0 ] ; i + + )
2004-08-23 00:15:46 +00:00
{
2006-01-04 00:44:34 +00:00
s = cl . model_name [ i ] ;
2004-08-23 00:15:46 +00:00
if ( s [ 0 ] = = ' * ' )
continue ; // inline brush model
if ( ! stricmp ( COM_FileExtension ( s ) , " dsp " ) ) //doom sprites are weird, and not really downloadable via this system
continue ;
2004-11-17 17:58:22 +00:00
2005-01-27 22:20:16 +00:00
# ifdef Q2CLIENT
2005-05-26 12:55:34 +00:00
if ( cls . protocol = = CP_QUAKE2 & & s [ 0 ] = = ' # ' ) //this is a vweap
2005-01-07 02:41:46 +00:00
continue ;
2005-01-27 22:20:16 +00:00
# endif
2005-01-07 02:41:46 +00:00
2008-11-09 22:29:28 +00:00
CL_CheckOrEnqueDownloadFile ( s , s , ( i = = 1 ) ? DLLF_REQUIRED : 0 ) ; //world is required to be loaded.
CL_CheckModelResources ( s ) ;
2004-08-23 00:15:46 +00:00
}
2009-07-06 01:20:20 +00:00
for ( i = 0 ; i < MAX_VWEP_MODELS ; i + + )
{
s = cl . model_name_vwep [ i ] ;
if ( ! stricmp ( COM_FileExtension ( s ) , " dsp " ) ) //doom sprites are weird, and not really downloadable via this system
continue ;
2010-03-25 22:56:11 +00:00
if ( ! * s )
continue ;
2009-07-06 01:20:20 +00:00
CL_CheckOrEnqueDownloadFile ( s , s , 0 ) ;
CL_CheckModelResources ( s ) ;
}
2006-01-04 00:44:34 +00:00
}
2009-04-06 00:34:32 +00:00
int CL_LoadModels ( int stage , qboolean dontactuallyload )
2006-01-04 00:44:34 +00:00
{
extern model_t * loadmodel ;
int i ;
2006-01-11 22:24:08 +00:00
float giveuptime = Sys_DoubleTime ( ) + 0.1 ; //small things get padded into a single frame
2009-04-06 00:34:32 +00:00
# define atstage() ((cl.contentstage == stage++ && !dontactuallyload)?++cl.contentstage:false)
2010-07-11 02:22:39 +00:00
# define endstage() if (!cls.timedemo && giveuptime<Sys_DoubleTime()) return -1;
2006-01-04 00:44:34 +00:00
2006-02-27 00:42:25 +00:00
pmove . numphysent = 0 ;
2007-06-20 00:02:54 +00:00
# ifdef PEXT_CSQC
if ( atstage ( ) )
{
2010-07-11 02:22:39 +00:00
extern cvar_t cl_nocsqc ;
if ( cls . protocol = = CP_NETQUAKE & & ! cl_nocsqc . ival & & ! cls . demoplayback )
2007-06-20 00:02:54 +00:00
{
char * s ;
s = Info_ValueForKey ( cl . serverinfo , " *csprogs " ) ;
2010-07-11 02:22:39 +00:00
if ( * s ) //only allow csqc if the server says so, and the 'checksum' matches.
2007-06-20 00:02:54 +00:00
{
extern cvar_t allow_download_csprogs ;
unsigned int chksum = strtoul ( s , NULL , 0 ) ;
2009-11-04 21:16:50 +00:00
if ( allow_download_csprogs . ival )
2007-06-20 00:02:54 +00:00
{
char * str = va ( " csprogsvers/%x.dat " , chksum ) ;
2008-11-09 22:29:28 +00:00
if ( CL_CheckOrEnqueDownloadFile ( " csprogs.dat " , str , DLLF_REQUIRED ) )
return stage ; //its kinda required
2007-06-20 00:02:54 +00:00
}
else
{
Con_Printf ( " Not downloading csprogs.dat due to allow_download_csprogs \n " ) ;
}
}
}
endstage ( ) ;
}
# endif
2009-04-01 22:03:56 +00:00
# ifdef HLCLIENT
if ( atstage ( ) )
{
CLHL_LoadClientGame ( ) ;
endstage ( ) ;
}
# endif
2006-01-13 06:27:18 +00:00
# ifdef PEXT_CSQC
2006-01-04 00:44:34 +00:00
if ( atstage ( ) )
{
2008-11-09 22:29:28 +00:00
char * s ;
2009-11-04 21:16:50 +00:00
qboolean anycsqc ;
#if 0 //ndef FTE_DEBUG
anycsqc = true ;
# else
anycsqc = atoi ( Info_ValueForKey ( cl . serverinfo , " anycsqc " ) ) ;
2008-11-09 22:29:28 +00:00
# endif
2009-11-04 21:16:50 +00:00
s = Info_ValueForKey ( cl . serverinfo , " *csprogs " ) ;
if ( anycsqc | | * s | | cls . demoplayback ) //only allow csqc if the server says so, and the 'checksum' matches.
2006-01-04 00:44:34 +00:00
{
2009-11-04 21:16:50 +00:00
unsigned int chksum = anycsqc ? 0 : strtoul ( s , NULL , 0 ) ;
2008-11-09 22:29:28 +00:00
if ( CSQC_Init ( chksum ) )
2006-01-04 00:44:34 +00:00
{
2008-11-09 22:29:28 +00:00
CL_SendClientCommand ( true , " enablecsqc " ) ;
}
else
{
CL_SendClientCommand ( true , " disablecsqc " ) ;
Sbar_Start ( ) ; //try and start this before we're actually on the server,
//this'll stop the mod from sending so much stuffed data at us, whilst we're frozen while trying to load.
//hopefully this'll make it more robust.
//csqc is expected to use it's own huds, or to run on decent servers. :p
2006-01-04 00:44:34 +00:00
}
}
2006-01-11 22:24:08 +00:00
endstage ( ) ;
2006-01-04 00:44:34 +00:00
}
2006-01-13 06:27:18 +00:00
# endif
2006-01-04 00:44:34 +00:00
2008-01-19 05:59:00 +00:00
if ( atstage ( ) )
{
loadmodel = cl . worldmodel ;
if ( R_PreNewMap )
R_PreNewMap ( ) ;
endstage ( ) ;
}
2004-08-23 00:15:46 +00:00
if ( cl . playernum [ 0 ] = = - 1 )
{ //q2 cinematic - don't load the models.
cl . worldmodel = cl . model_precache [ 1 ] = Mod_ForName ( " " , false ) ;
}
2006-01-04 00:44:34 +00:00
else
2004-08-23 00:15:46 +00:00
{
for ( i = 1 ; i < MAX_MODELS ; i + + )
{
if ( ! cl . model_name [ i ] [ 0 ] )
2006-01-04 00:44:34 +00:00
continue ;
2005-02-28 07:16:19 +00:00
2006-01-04 00:44:34 +00:00
if ( atstage ( ) )
{
2008-11-09 22:29:28 +00:00
# ifdef CSQC_DAT
if ( i = = 1 )
CSQC_LoadResource ( cl . model_name [ i ] , " map " ) ;
else
CSQC_LoadResource ( cl . model_name [ i ] , " model " ) ;
# endif
2009-04-01 22:03:56 +00:00
# ifdef Q2CLIENT
if ( cls . protocol = = CP_QUAKE2 & & * cl . model_name [ i ] = = ' # ' )
cl . model_precache [ i ] = NULL ;
else
# endif
cl . model_precache [ i ] = Mod_ForName ( cl . model_name [ i ] , false ) ;
2006-01-04 00:44:34 +00:00
Hunk_Check ( ) ;
2004-08-23 00:15:46 +00:00
2006-01-04 00:44:34 +00:00
S_ExtraUpdate ( ) ;
2004-08-23 00:15:46 +00:00
2006-01-11 22:24:08 +00:00
endstage ( ) ;
2006-01-04 00:44:34 +00:00
}
2004-08-23 00:15:46 +00:00
}
2009-07-06 01:20:20 +00:00
for ( i = 0 ; i < MAX_VWEP_MODELS ; i + + )
{
if ( ! cl . model_name_vwep [ i ] [ 0 ] )
continue ;
if ( atstage ( ) )
{
2009-07-09 22:12:29 +00:00
# ifdef CSQC_DAT
CSQC_LoadResource ( cl . model_name_vwep [ i ] , " vwep " ) ;
# endif
2009-07-06 01:20:20 +00:00
cl . model_precache_vwep [ i ] = Mod_ForName ( cl . model_name_vwep [ i ] , false ) ;
endstage ( ) ;
}
}
2006-01-04 00:44:34 +00:00
}
2004-08-23 00:15:46 +00:00
2008-05-25 22:23:43 +00:00
if ( atstage ( ) )
{
cl . worldmodel = cl . model_precache [ 1 ] ;
if ( ! cl . worldmodel | | cl . worldmodel - > type = = mod_dummy )
2008-11-09 22:29:28 +00:00
{
if ( ! cl . model_name [ 1 ] [ 0 ] )
Host_EndGame ( " Worldmodel name wasn't sent \n " ) ;
2009-03-03 01:52:30 +00:00
// else
// return stage;
// Host_EndGame("Worldmodel wasn't loaded\n");
2008-11-09 22:29:28 +00:00
}
2008-05-25 22:23:43 +00:00
2008-06-08 20:46:58 +00:00
# ifdef CSQC_DAT
2008-05-25 22:23:43 +00:00
CSQC_WorldLoaded ( ) ;
2008-06-08 20:46:58 +00:00
# endif
2008-05-25 22:23:43 +00:00
endstage ( ) ;
}
2006-01-11 22:24:08 +00:00
for ( i = 1 ; i < MAX_CSQCMODELS ; i + + )
{
if ( ! cl . model_csqcname [ i ] [ 0 ] )
continue ;
if ( atstage ( ) )
{
2008-11-09 22:29:28 +00:00
# ifdef CSQC_DAT
if ( i = = 1 )
CSQC_LoadResource ( cl . model_csqcname [ i ] , " map " ) ;
else
CSQC_LoadResource ( cl . model_csqcname [ i ] , " model " ) ;
# endif
2006-01-11 22:24:08 +00:00
cl . model_csqcprecache [ i ] = Mod_ForName ( cl . model_csqcname [ i ] , false ) ;
Hunk_Check ( ) ;
S_ExtraUpdate ( ) ;
endstage ( ) ;
}
}
2006-01-04 00:44:34 +00:00
if ( atstage ( ) )
{
Wad_NextDownload ( ) ;
2006-01-11 22:24:08 +00:00
endstage ( ) ;
2006-01-04 00:44:34 +00:00
}
2004-08-23 00:15:46 +00:00
2006-01-04 00:44:34 +00:00
if ( atstage ( ) )
{
loadmodel = cl . worldmodel ;
2009-03-03 01:52:30 +00:00
// if (!loadmodel || loadmodel->type == mod_dummy)
// Host_EndGame("No worldmodel was loaded\n");
2004-08-23 00:15:46 +00:00
Mod_NowLoadExternal ( ) ;
2006-01-04 00:44:34 +00:00
2006-01-11 22:24:08 +00:00
endstage ( ) ;
2004-08-23 00:15:46 +00:00
}
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
// all done
2006-01-04 00:44:34 +00:00
if ( atstage ( ) )
2004-08-23 00:15:46 +00:00
{
2006-01-04 00:44:34 +00:00
loadmodel = cl . worldmodel ;
2009-03-03 01:52:30 +00:00
// if (!loadmodel || loadmodel->type == mod_dummy)
// Host_EndGame("No worldmodel was loaded\n");
cl . model_precaches_added = false ;
2006-01-04 00:44:34 +00:00
R_NewMap ( ) ;
2006-01-28 19:04:13 +00:00
pmove . physents [ 0 ] . model = cl . worldmodel ;
2006-01-11 22:24:08 +00:00
endstage ( ) ;
2004-08-23 00:15:46 +00:00
}
2006-01-04 00:44:34 +00:00
return stage ;
2004-08-23 00:15:46 +00:00
}
2009-04-06 00:34:32 +00:00
int CL_LoadSounds ( int stage , qboolean dontactuallyload )
2008-11-09 22:29:28 +00:00
{
int i ;
float giveuptime = Sys_DoubleTime ( ) + 0.1 ; //small things get padded into a single frame
2009-04-06 00:34:32 +00:00
//#define atstage() ((cl.contentstage == stage++)?++cl.contentstage:false)
//#define endstage() if (giveuptime<Sys_DoubleTime()) return -1;
2008-11-09 22:29:28 +00:00
for ( i = 1 ; i < MAX_SOUNDS ; i + + )
{
if ( ! cl . sound_name [ i ] [ 0 ] )
break ;
if ( atstage ( ) )
{
# ifdef CSQC_DAT
CSQC_LoadResource ( cl . sound_name [ i ] , " sound " ) ;
# endif
cl . sound_precache [ i ] = S_PrecacheSound ( cl . sound_name [ i ] ) ;
S_ExtraUpdate ( ) ;
endstage ( ) ;
}
}
return stage ;
}
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = = = =
Sound_NextDownload
= = = = = = = = = = = = = = = = =
*/
2009-04-06 00:34:32 +00:00
void Sound_CheckDownloads ( void )
2004-08-23 00:15:46 +00:00
{
2007-06-20 00:02:54 +00:00
char mangled [ 512 ] ;
2004-08-23 00:15:46 +00:00
char * s ;
int i ;
2005-08-11 04:14:33 +00:00
2009-03-03 01:52:30 +00:00
// Con_TPrintf (TLC_CHECKINGSOUNDS);
2004-08-23 00:15:46 +00:00
2005-08-11 04:14:33 +00:00
# ifdef CSQC_DAT
2009-05-24 10:11:17 +00:00
// if (cls.fteprotocolextensions & PEXT_CSQC)
2005-08-11 04:14:33 +00:00
{
s = Info_ValueForKey ( cl . serverinfo , " *csprogs " ) ;
2009-05-24 10:11:17 +00:00
if ( * s ) //only allow csqc if the server says so, and the 'checksum' matches.
2005-08-11 04:14:33 +00:00
{
2010-07-11 02:22:39 +00:00
extern cvar_t allow_download_csprogs , cl_nocsqc ;
2005-08-11 04:14:33 +00:00
unsigned int chksum = strtoul ( s , NULL , 0 ) ;
2010-07-11 02:22:39 +00:00
if ( cl_nocsqc . ival | | cls . demoplayback )
{
}
else if ( allow_download_csprogs . ival )
2006-01-04 00:44:34 +00:00
{
char * str = va ( " csprogsvers/%x.dat " , chksum ) ;
2008-11-09 22:29:28 +00:00
CL_CheckOrEnqueDownloadFile ( " csprogs.dat " , str , DLLF_REQUIRED ) ;
2006-01-04 00:44:34 +00:00
}
2005-10-01 03:09:17 +00:00
else
2006-01-04 00:44:34 +00:00
{
Con_Printf ( " Not downloading csprogs.dat \n " ) ;
}
2005-08-11 04:14:33 +00:00
}
}
# endif
2006-01-04 00:44:34 +00:00
for ( i = 1 ; cl . sound_name [ i ] [ 0 ]
; i + + )
2004-08-23 00:15:46 +00:00
{
2006-01-04 00:44:34 +00:00
s = cl . sound_name [ i ] ;
2009-04-06 00:34:32 +00:00
if ( * s = = ' * ' ) //q2 sexed sound
continue ;
if ( ! S_HaveOutput ( ) )
2004-08-23 00:15:46 +00:00
continue ;
2007-06-20 00:02:54 +00:00
2008-11-09 22:29:28 +00:00
//check without the sound/ prefix
2007-06-20 00:02:54 +00:00
if ( CL_CheckFile ( s ) )
continue ; //we have it already
//the things I do for nexuiz... *sigh*
COM_StripExtension ( s , mangled , sizeof ( mangled ) ) ;
COM_DefaultExtension ( mangled , " .ogg " , sizeof ( mangled ) ) ;
if ( CL_CheckFile ( mangled ) )
continue ;
2008-11-09 22:29:28 +00:00
//check with the sound/ prefix
s = va ( " sound/%s " , s ) ;
2004-08-23 00:15:46 +00:00
2008-11-09 22:29:28 +00:00
if ( CL_CheckFile ( s ) )
continue ; //we have it already
//the things I do for nexuiz... *sigh*
COM_StripExtension ( s , mangled , sizeof ( mangled ) ) ;
COM_DefaultExtension ( mangled , " .ogg " , sizeof ( mangled ) ) ;
if ( CL_CheckFile ( mangled ) )
continue ;
2004-08-23 00:15:46 +00:00
2008-11-09 22:29:28 +00:00
//download the one the server said.
CL_CheckOrEnqueDownloadFile ( s , NULL , 0 ) ;
2004-08-23 00:15:46 +00:00
}
}
/*
= = = = = = = = = = = = = = = = = = = = = =
CL_RequestNextDownload
= = = = = = = = = = = = = = = = = = = = = =
*/
void CL_RequestNextDownload ( void )
{
2008-11-09 22:29:28 +00:00
int stage ;
2006-01-04 00:44:34 +00:00
if ( cls . downloadmethod )
return ;
if ( cl . sendprespawn | | cls . state = = ca_active )
if ( cl . downloadlist )
{
2008-11-09 22:29:28 +00:00
downloadlist_t * dl ;
2009-04-06 00:34:32 +00:00
//download required downloads first
2008-11-09 22:29:28 +00:00
for ( dl = cl . downloadlist ; dl ; dl = dl - > next )
{
if ( dl - > flags & DLLF_REQUIRED )
break ;
}
if ( ! dl )
dl = cl . downloadlist ;
if ( ( dl - > flags & DLLF_OVERWRITE ) | | ! COM_FCheckExists ( dl - > localname ) )
2009-05-24 10:11:17 +00:00
CL_SendDownloadStartRequest ( dl - > rname , dl - > localname ) ;
2006-01-04 00:44:34 +00:00
else
{
2008-11-09 22:29:28 +00:00
Con_Printf ( " Already have %s \n " , dl - > localname ) ;
2009-05-24 10:11:17 +00:00
CL_DisenqueDownload ( dl - > rname ) ;
2008-11-09 22:29:28 +00:00
//recurse a bit.
CL_RequestNextDownload ( ) ;
return ;
2006-01-04 00:44:34 +00:00
}
2008-11-09 22:29:28 +00:00
if ( requiredownloads . value | | ( dl - > flags & DLLF_REQUIRED ) )
2006-01-04 00:44:34 +00:00
return ;
}
if ( cl . sendprespawn )
{ // get next signon phase
2009-04-06 00:34:32 +00:00
extern int total_loading_size , current_loading_size ;
if ( ! cl . contentstage )
{
stage = 0 ;
stage = CL_LoadModels ( stage , true ) ;
stage = CL_LoadSounds ( stage , true ) ;
total_loading_size = stage ;
cl . contentstage = 0 ;
}
2008-11-09 22:29:28 +00:00
stage = 0 ;
2009-04-06 00:34:32 +00:00
stage = CL_LoadModels ( stage , false ) ;
current_loading_size = cl . contentstage ;
2008-11-09 22:29:28 +00:00
if ( stage < 0 )
2006-01-04 00:44:34 +00:00
return ; //not yet
2009-04-06 00:34:32 +00:00
stage = CL_LoadSounds ( stage , false ) ;
current_loading_size = cl . contentstage ;
2008-11-09 22:29:28 +00:00
if ( stage < 0 )
return ;
2006-01-04 00:44:34 +00:00
cl . sendprespawn = false ;
2008-01-09 00:52:31 +00:00
# ifdef _MSC_VER
//FIXME: timedemo timer should start here.
# else
# warning timedemo timer should start here
# endif
2006-01-04 00:44:34 +00:00
2009-04-06 00:34:32 +00:00
if ( ! cl . worldmodel | | cl . worldmodel - > needload )
{
Con_Printf ( " \n \n ------------- \n Couldn't download %s - cannot fully connect \n " , cl . worldmodel - > name ) ;
SCR_SetLoadingStage ( LS_NONE ) ;
return ;
}
2006-01-04 00:44:34 +00:00
# ifdef Q2CLIENT
if ( cls . protocol = = CP_QUAKE2 )
{
Skin_NextDownload ( ) ;
2009-04-06 00:34:32 +00:00
SCR_SetLoadingStage ( LS_NONE ) ;
2006-01-06 23:58:32 +00:00
CL_SendClientCommand ( true , " begin %i \n " , cl . servercount ) ;
2006-01-04 00:44:34 +00:00
}
2005-01-16 02:25:35 +00:00
else
2006-01-04 00:44:34 +00:00
# endif
2005-01-16 02:25:35 +00:00
{
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_EZTV )
{
if ( CL_RemoveClientCommands ( " qtvspawn " ) )
Con_Printf ( " Multiple prespawns \n " ) ;
2009-11-04 21:16:50 +00:00
CL_SendClientCommand ( true , " qtvspawn %i 0 %i " , cl . servercount , COM_RemapMapChecksum ( LittleLong ( cl . worldmodel - > checksum2 ) ) ) ;
2009-04-06 00:34:32 +00:00
SCR_SetLoadingStage ( LS_NONE ) ;
2008-01-09 00:52:31 +00:00
}
else
{
2006-01-04 00:44:34 +00:00
// done with modellist, request first of static signon messages
2008-01-09 00:52:31 +00:00
if ( CL_RemoveClientCommands ( " prespawn " ) )
Con_Printf ( " Multiple prespawns \n " ) ;
// CL_SendClientCommand("prespawn %i 0 %i", cl.servercount, cl.worldmodel->checksum2);
2009-11-04 21:16:50 +00:00
CL_SendClientCommand ( true , prespawn_name , cl . servercount , COM_RemapMapChecksum ( LittleLong ( cl . worldmodel - > checksum2 ) ) ) ;
2008-01-09 00:52:31 +00:00
}
2005-01-16 02:25:35 +00:00
}
2006-01-04 00:44:34 +00:00
2004-08-23 00:15:46 +00:00
}
}
2005-01-16 02:25:35 +00:00
int CL_RequestADownloadChunk ( void ) ;
void CL_SendDownloadReq ( sizebuf_t * msg )
2005-01-13 23:33:00 +00:00
{
2008-01-30 02:32:00 +00:00
if ( cls . demoplayback = = DPB_EZTV )
return ; //tcp connection, so no need to constantly ask
2005-01-13 23:33:00 +00:00
if ( cl . downloadlist & & ! cls . downloadmethod )
{
CL_RequestNextDownload ( ) ;
return ;
}
# ifdef PEXT_CHUNKEDDOWNLOADS
if ( cls . downloadmethod = = DL_QWCHUNKS )
{
2008-11-09 22:29:28 +00:00
extern int download_file_number ;
int p ;
for ( p = 0 ; p < 8 ; p + + )
2005-01-16 02:25:35 +00:00
{
2008-11-09 22:29:28 +00:00
int i = CL_RequestADownloadChunk ( ) ;
if ( i > = 0 )
2008-12-23 02:55:20 +00:00
{
char * cmd = va ( " nextdl %i - %i \n " , i , download_file_number ) ;
CL_RemoveClientCommands ( cmd ) ;
CL_SendClientCommand ( false , " %s " , cmd ) ;
}
2008-11-09 22:29:28 +00:00
else
break ; //we can stop downloading now.
2005-01-16 02:25:35 +00:00
}
2005-01-13 23:33:00 +00:00
return ;
}
# endif
}
2004-08-23 00:15:46 +00:00
# ifdef PEXT_ZLIBDL
# ifdef _WIN32
# define ZEXPORT VARGS
# include "../../zip/zlib.h"
2005-09-26 03:40:09 +00:00
//# pragma comment (lib, "zip/zlib.lib")
2004-08-23 00:15:46 +00:00
# else
# include <zlib.h>
# endif
char * ZLibDownloadDecode ( int * messagesize , char * input , int finalsize )
{
char * outbuf = Hunk_TempAlloc ( finalsize ) ;
z_stream zs ;
* messagesize = ( * ( short * ) input ) ;
input + = 2 ;
if ( ! * messagesize )
{
* messagesize = finalsize + 2 ;
return input ;
}
memset ( & zs , 0 , sizeof ( zs ) ) ;
zs . next_in = input ;
zs . avail_in = * messagesize ; //tell it that it has a lot. Possibly a bad idea.
zs . total_in = 0 ;
zs . next_out = outbuf ;
zs . avail_out = finalsize ; //this is the limiter.
zs . total_out = 0 ;
zs . data_type = Z_BINARY ;
inflateInit ( & zs ) ;
inflate ( & zs , Z_FINISH ) ; //decompress it in one go.
inflateEnd ( & zs ) ;
* messagesize = zs . total_in + 2 ;
return outbuf ;
}
# endif
2008-11-09 22:29:28 +00:00
downloadlist_t * CL_DownloadFailed ( char * name )
2005-01-16 02:25:35 +00:00
{
//add this to our failed list. (so we don't try downloading it again...)
2008-11-09 22:29:28 +00:00
downloadlist_t * failed , * * link , * dl ;
2005-01-16 02:25:35 +00:00
failed = Z_Malloc ( sizeof ( downloadlist_t ) ) ;
failed - > next = cl . faileddownloads ;
cl . faileddownloads = failed ;
2009-05-24 10:11:17 +00:00
Q_strncpyz ( failed - > rname , name , sizeof ( failed - > rname ) ) ;
2005-01-16 02:25:35 +00:00
2008-11-09 22:29:28 +00:00
//if this is what we're currently downloading, close it up now.
2009-05-24 10:11:17 +00:00
if ( ! stricmp ( cls . downloadremotename , name ) | | ! * name )
2008-11-09 22:29:28 +00:00
{
cls . downloadmethod = DL_NONE ;
if ( cls . downloadqw )
{
VFS_CLOSE ( cls . downloadqw ) ;
cls . downloadqw = NULL ;
CL_SendClientCommand ( true , " stopdownload " ) ;
}
2009-05-24 10:11:17 +00:00
* cls . downloadlocalname = 0 ;
* cls . downloadremotename = 0 ;
2008-11-09 22:29:28 +00:00
}
link = & cl . downloadlist ;
while ( * link )
{
dl = * link ;
2009-05-24 10:11:17 +00:00
if ( ! strcmp ( dl - > rname , name ) )
2008-11-09 22:29:28 +00:00
{
* link = dl - > next ;
failed - > flags | = dl - > flags ;
Z_Free ( dl ) ;
}
else
link = & ( * link ) - > next ;
}
return failed ;
2005-01-16 02:25:35 +00:00
}
2005-01-13 16:29:20 +00:00
2005-01-16 02:25:35 +00:00
float downloadstarttime ;
2005-01-13 16:29:20 +00:00
# ifdef PEXT_CHUNKEDDOWNLOADS
2005-01-16 02:25:35 +00:00
# define MAXBLOCKS 64 //must be power of 2
# define DLBLOCKSIZE 1024
int downloadsize ;
int receivedbytes ;
2005-01-13 23:33:00 +00:00
int recievedblock [ MAXBLOCKS ] ;
2005-01-16 02:25:35 +00:00
int firstblock ;
int blockcycle ;
2008-11-09 22:29:28 +00:00
int download_file_number ;
int CL_DownloadRate ( void )
{
return receivedbytes / ( Sys_DoubleTime ( ) - downloadstarttime ) ;
}
2005-01-13 16:29:20 +00:00
void CL_ParseChunkedDownload ( void )
{
2005-01-17 17:40:21 +00:00
qbyte * svname ;
2006-01-27 08:12:14 +00:00
//qbyte osname[MAX_OSPATH]; //unreferenced
2005-01-13 16:29:20 +00:00
int totalsize ;
int chunknum ;
2005-01-16 02:25:35 +00:00
char data [ DLBLOCKSIZE ] ;
2005-01-13 16:29:20 +00:00
chunknum = MSG_ReadLong ( ) ;
if ( chunknum < 0 )
{
totalsize = MSG_ReadLong ( ) ;
2005-01-17 17:40:21 +00:00
svname = MSG_ReadString ( ) ;
2005-01-13 16:29:20 +00:00
if ( cls . demoplayback )
return ;
2008-11-09 22:29:28 +00:00
if ( ! * svname )
{
//stupid mvdsv.
/*if (totalsize < 0)
svname = cls . downloadname ;
else */
{
Con_Printf ( " ignoring nameless download \n " ) ;
return ;
}
}
2005-01-13 16:29:20 +00:00
if ( totalsize < 0 )
{
2008-11-09 22:29:28 +00:00
if ( totalsize = = - 3 )
Con_Printf ( " Server reported an error when downloading file \" %s \" \n " , svname ) ;
else if ( totalsize = = - 2 )
Con_Printf ( " Server permissions deny downloading file \" %s \" \n " , svname ) ;
2005-01-13 16:29:20 +00:00
else
2008-11-09 22:29:28 +00:00
Con_Printf ( " Couldn't find file \" %s \" on the server \n " , svname ) ;
2005-01-13 16:29:20 +00:00
2009-05-24 10:11:17 +00:00
cls . downloadmethod = 0 ;
2005-01-17 17:40:21 +00:00
CL_DownloadFailed ( svname ) ;
2005-01-13 16:29:20 +00:00
CL_RequestNextDownload ( ) ;
return ;
}
if ( cls . downloadmethod = = DL_QWCHUNKS )
2005-01-17 17:40:21 +00:00
Host_EndGame ( " Received second download - \" %s \" \n " , svname ) ;
2005-01-13 16:29:20 +00:00
2009-05-24 10:11:17 +00:00
if ( stricmp ( cls . downloadremotename , svname ) )
Host_EndGame ( " Server sent the wrong download - \" %s \" instead of \" %s \" \n " , svname , cls . downloadremotename ) ;
2006-01-04 00:44:34 +00:00
2005-12-13 02:31:57 +00:00
2005-01-13 16:29:20 +00:00
//start the new download
cls . downloadmethod = DL_QWCHUNKS ;
cls . downloadpercent = 0 ;
2005-01-16 02:25:35 +00:00
downloadsize = totalsize ;
downloadstarttime = Sys_DoubleTime ( ) ;
2005-01-13 16:29:20 +00:00
2006-01-10 18:48:03 +00:00
/*
2005-01-17 17:40:21 +00:00
strcpy ( cls . downloadname , svname ) ;
COM_StripExtension ( svname , cls . downloadtempname ) ;
2005-01-13 16:29:20 +00:00
COM_DefaultExtension ( cls . downloadtempname , " .tmp " ) ;
2006-01-10 18:48:03 +00:00
*/
2005-01-13 16:29:20 +00:00
2005-01-16 02:25:35 +00:00
if ( ! strncmp ( cls . downloadtempname , " skins/ " , 6 ) )
2006-01-02 23:46:44 +00:00
{
2008-12-23 02:55:20 +00:00
FS_CreatePath ( va ( " qw/%s " , cls . downloadtempname ) , FS_ROOT ) ;
cls . downloadqw = FS_OpenVFS ( va ( " qw/%s " , cls . downloadtempname ) , " wb " , FS_ROOT ) ;
2006-01-02 23:46:44 +00:00
}
2005-01-16 02:25:35 +00:00
else
2006-01-02 23:46:44 +00:00
{
FS_CreatePath ( cls . downloadtempname , FS_GAME ) ;
cls . downloadqw = FS_OpenVFS ( cls . downloadtempname , " wb " , FS_GAME ) ;
}
2005-01-16 02:25:35 +00:00
2005-12-13 02:31:57 +00:00
if ( ! cls . downloadqw )
{
CL_DownloadFailed ( svname ) ;
return ;
}
2008-11-09 22:29:28 +00:00
download_file_number + + ;
2005-01-16 02:25:35 +00:00
firstblock = 0 ;
receivedbytes = 0 ;
blockcycle = - 1 ; //so it requests 0 first. :)
memset ( recievedblock , 0 , sizeof ( recievedblock ) ) ;
2005-01-13 16:29:20 +00:00
return ;
}
2005-01-17 17:40:21 +00:00
// Con_Printf("Received dl block %i: ", chunknum);
2005-01-16 02:25:35 +00:00
MSG_ReadData ( data , DLBLOCKSIZE ) ;
2006-01-13 20:30:01 +00:00
if ( ! cls . downloadqw )
return ;
2005-01-13 16:29:20 +00:00
if ( cls . demoplayback )
2005-01-16 02:25:35 +00:00
{ //err, yeah, when playing demos we don't actually pay any attention to this.
return ;
}
if ( chunknum < firstblock )
2005-01-13 16:29:20 +00:00
{
2005-01-17 17:40:21 +00:00
// Con_Printf("too old\n", chunknum);
2005-01-13 16:29:20 +00:00
return ;
}
2005-01-16 02:25:35 +00:00
if ( chunknum - firstblock > = MAXBLOCKS )
{
2005-01-17 17:40:21 +00:00
// Con_Printf("^1too new!\n", chunknum);
2005-01-16 02:25:35 +00:00
return ;
}
2005-09-26 03:40:09 +00:00
2005-01-16 02:25:35 +00:00
if ( recievedblock [ chunknum & ( MAXBLOCKS - 1 ) ] )
{
2005-01-17 17:40:21 +00:00
// Con_Printf("duplicated\n", chunknum);
2005-01-16 02:25:35 +00:00
return ;
}
2005-01-17 17:40:21 +00:00
// Con_Printf("usable\n", chunknum);
2005-01-16 02:25:35 +00:00
receivedbytes + = DLBLOCKSIZE ;
recievedblock [ chunknum & ( MAXBLOCKS - 1 ) ] = true ;
2005-01-13 23:33:00 +00:00
2005-01-16 02:25:35 +00:00
while ( recievedblock [ firstblock & ( MAXBLOCKS - 1 ) ] )
{
recievedblock [ firstblock & ( MAXBLOCKS - 1 ) ] = false ;
firstblock + + ;
}
2006-01-02 23:00:10 +00:00
VFS_SEEK ( cls . downloadqw , chunknum * DLBLOCKSIZE ) ;
2005-01-16 02:25:35 +00:00
if ( downloadsize - chunknum * DLBLOCKSIZE < DLBLOCKSIZE ) //final block is actually meant to be smaller than we recieve.
2006-01-02 23:00:10 +00:00
VFS_WRITE ( cls . downloadqw , data , downloadsize - chunknum * DLBLOCKSIZE ) ;
2005-01-16 02:25:35 +00:00
else
2006-01-02 23:00:10 +00:00
VFS_WRITE ( cls . downloadqw , data , DLBLOCKSIZE ) ;
2005-01-16 02:25:35 +00:00
cls . downloadpercent = receivedbytes / ( float ) downloadsize * 100 ;
2005-01-13 23:33:00 +00:00
}
2006-01-04 00:44:34 +00:00
int CL_CountQueuedDownloads ( void )
{
int count = 0 ;
downloadlist_t * dl ;
for ( dl = cl . downloadlist ; dl ; dl = dl - > next )
count + + ;
return count ;
}
2005-01-16 02:25:35 +00:00
int CL_RequestADownloadChunk ( void )
2005-01-13 23:33:00 +00:00
{
2005-01-16 02:25:35 +00:00
int i ;
int b ;
if ( cls . downloadmethod ! = DL_QWCHUNKS )
{
Con_Printf ( " download not initiated \n " ) ;
2008-11-09 22:29:28 +00:00
return - 1 ;
2005-01-16 02:25:35 +00:00
}
blockcycle + + ;
for ( i = 0 ; i < MAXBLOCKS ; i + + )
{
b = ( ( i + blockcycle ) & ( MAXBLOCKS - 1 ) )
+ firstblock ;
if ( ! recievedblock [ b & ( MAXBLOCKS - 1 ) ] ) //don't ask for ones we've already got.
{
if ( b > = ( downloadsize + DLBLOCKSIZE - 1 ) / DLBLOCKSIZE ) //don't ask for blocks that are over the size of the file.
continue ;
2005-01-17 17:40:21 +00:00
// Con_Printf("Requesting block %i\n", b);
2005-01-16 02:25:35 +00:00
return b ;
}
}
2005-01-17 17:40:21 +00:00
// Con_Printf("^1 EOF?\n");
2005-01-16 02:25:35 +00:00
2006-01-02 23:00:10 +00:00
VFS_CLOSE ( cls . downloadqw ) ;
2008-11-09 22:29:28 +00:00
cls . downloadqw = NULL ;
2007-06-20 00:02:54 +00:00
CL_SendClientCommand ( true , " stopdownload " ) ;
2009-05-24 10:11:17 +00:00
CL_DownloadFinished ( ) ;
2005-01-16 02:25:35 +00:00
2006-01-04 00:44:34 +00:00
Con_Printf ( " Download took %i seconds (%i more) \n " , ( int ) ( Sys_DoubleTime ( ) - downloadstarttime ) , CL_CountQueuedDownloads ( ) ) ;
2005-01-17 17:40:21 +00:00
2009-05-24 10:11:17 +00:00
* cls . downloadlocalname = ' \0 ' ;
* cls . downloadremotename = ' \0 ' ;
2005-01-16 02:25:35 +00:00
cls . downloadpercent = 0 ;
return - 1 ;
2005-01-13 23:33:00 +00:00
}
2005-01-16 02:25:35 +00:00
2005-01-13 16:29:20 +00:00
# endif
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = = = = = = = =
CL_ParseDownload
A download message has been received from the server
= = = = = = = = = = = = = = = = = = = = =
*/
void CL_ParseDownload ( void )
{
2006-08-02 05:34:17 +00:00
extern cvar_t cl_dlemptyterminate ;
2004-08-23 00:15:46 +00:00
int size , percent ;
qbyte name [ 1024 ] ;
2005-01-13 16:29:20 +00:00
# ifdef PEXT_CHUNKEDDOWNLOADS
if ( cls . fteprotocolextensions & PEXT_CHUNKEDDOWNLOADS )
{
2008-01-30 02:32:00 +00:00
if ( cls . demoplayback = = DPB_EZTV )
Host_EndGame ( " CL_ParseDownload: chunked download on qtv proxy. " ) ;
2005-01-13 16:29:20 +00:00
CL_ParseChunkedDownload ( ) ;
return ;
}
# endif
2004-08-23 00:15:46 +00:00
// read the data
size = MSG_ReadShort ( ) ;
percent = MSG_ReadByte ( ) ;
2008-01-30 02:32:00 +00:00
if ( cls . demoplayback & & cls . demoplayback ! = DPB_EZTV )
2004-08-23 00:15:46 +00:00
{
if ( size > 0 )
msg_readcount + = size ;
return ; // not in demo playback
}
2009-05-24 10:11:17 +00:00
if ( ! * cls . downloadlocalname ) //huh... that's not right...
2004-12-05 11:37:39 +00:00
{
2007-09-23 15:28:06 +00:00
Con_Printf ( CON_WARNING " Warning: Server sending unknown file. \n " ) ;
2009-05-24 10:11:17 +00:00
strcpy ( cls . downloadlocalname , " unknown.txt " ) ;
2004-12-05 11:37:39 +00:00
strcpy ( cls . downloadtempname , " unknown.tmp " ) ;
}
2009-05-24 10:11:17 +00:00
if ( ! * cls . downloadremotename )
strcpy ( cls . downloadremotename , " unknown.txt " ) ;
2004-12-05 11:37:39 +00:00
2005-01-13 16:29:20 +00:00
if ( size < 0 )
2004-08-23 00:15:46 +00:00
{
Con_TPrintf ( TL_FILENOTFOUND ) ;
if ( cls . downloadqw )
{
Con_TPrintf ( TL_CLS_DOWNLOAD_ISSET ) ;
2006-01-02 23:00:10 +00:00
VFS_CLOSE ( cls . downloadqw ) ;
2004-08-23 00:15:46 +00:00
cls . downloadqw = NULL ;
}
2005-01-16 02:25:35 +00:00
2009-05-24 10:11:17 +00:00
CL_DownloadFailed ( cls . downloadremotename ) ;
2004-08-23 00:15:46 +00:00
CL_RequestNextDownload ( ) ;
return ;
}
// open the file if not opened yet
if ( ! cls . downloadqw )
{
if ( strncmp ( cls . downloadtempname , " skins/ " , 6 ) )
2006-01-04 00:44:34 +00:00
{
sprintf ( name , " %s " , cls . downloadtempname ) ;
FS_CreatePath ( name , FS_GAME ) ;
cls . downloadqw = FS_OpenVFS ( name , " wb " , FS_GAME ) ;
}
2004-08-23 00:15:46 +00:00
else
2006-01-04 00:44:34 +00:00
{
sprintf ( name , " %s " , cls . downloadtempname + 6 ) ;
FS_CreatePath ( name , FS_SKINS ) ;
cls . downloadqw = FS_OpenVFS ( name , " wb " , FS_SKINS ) ;
}
2004-08-23 00:15:46 +00:00
if ( ! cls . downloadqw )
{
msg_readcount + = size ;
Con_TPrintf ( TL_FAILEDTOOPEN , cls . downloadtempname ) ;
2009-05-24 10:11:17 +00:00
CL_DownloadFailed ( cls . downloadremotename ) ;
2004-08-23 00:15:46 +00:00
CL_RequestNextDownload ( ) ;
return ;
}
2005-01-16 02:25:35 +00:00
downloadstarttime = Sys_DoubleTime ( ) ;
2008-11-09 22:29:28 +00:00
receivedbytes = 0 ;
2004-08-23 00:15:46 +00:00
SCR_EndLoadingPlaque ( ) ;
}
# ifdef PEXT_ZLIBDL
if ( percent > = 101 & & percent < = 201 ) // && cls.fteprotocolextensions & PEXT_ZLIBDL)
{
int compsize ;
percent = percent - 101 ;
2006-01-02 23:00:10 +00:00
VFS_WRITE ( cls . download , ZLibDownloadDecode ( & compsize , net_message . data + msg_readcount , size ) , size ) ;
2004-08-23 00:15:46 +00:00
msg_readcount + = compsize ;
}
else
# endif
{
2006-01-02 23:00:10 +00:00
VFS_WRITE ( cls . downloadqw , net_message . data + msg_readcount , size ) ;
2004-08-23 00:15:46 +00:00
msg_readcount + = size ;
}
2008-11-09 22:29:28 +00:00
receivedbytes + = size ;
if ( cls . downloadpercent ! = percent ) //try and guess the size (its most acurate when the percent value changes)
downloadsize = ( ( float ) receivedbytes * 100 ) / percent ;
2005-01-13 23:33:00 +00:00
if ( cls . downloadmethod = = DL_QWPENDING )
2004-08-23 00:15:46 +00:00
cls . downloadmethod = DL_QW ;
2009-11-04 21:16:50 +00:00
if ( percent ! = 100 & & size = = 0 & & cl_dlemptyterminate . ival )
2006-08-02 05:34:17 +00:00
{
2007-09-23 15:28:06 +00:00
Con_Printf ( CON_WARNING " WARNING: Client received empty svc_download, assuming EOF \n " ) ;
2006-08-02 05:34:17 +00:00
percent = 100 ;
}
2004-08-23 00:15:46 +00:00
if ( percent ! = 100 )
{
// change display routines by zoid
// request next block
cls . downloadpercent = percent ;
2005-03-23 22:14:08 +00:00
CL_SendClientCommand ( true , " nextdl " ) ;
2004-08-23 00:15:46 +00:00
}
else
{
2006-01-02 23:00:10 +00:00
VFS_CLOSE ( cls . downloadqw ) ;
2004-08-23 00:15:46 +00:00
2009-05-24 10:11:17 +00:00
CL_DownloadFinished ( ) ;
* cls . downloadlocalname = ' \0 ' ;
* cls . downloadremotename = ' \0 ' ;
2004-08-23 00:15:46 +00:00
cls . downloadqw = NULL ;
cls . downloadpercent = 0 ;
2005-01-17 17:40:21 +00:00
Con_Printf ( " Download took %i seconds \n " , ( int ) ( Sys_DoubleTime ( ) - downloadstarttime ) ) ;
2005-01-16 02:25:35 +00:00
2004-08-23 00:15:46 +00:00
// get another file if needed
CL_RequestNextDownload ( ) ;
}
}
2008-11-09 22:29:28 +00:00
qboolean CL_ParseOOBDownload ( void )
{
if ( MSG_ReadLong ( ) ! = download_file_number )
return false ;
if ( MSG_ReadChar ( ) ! = svc_download )
return false ;
CL_ParseDownload ( ) ;
return true ;
}
2007-06-20 00:02:54 +00:00
void CLDP_ParseDownloadData ( void )
{
unsigned char buffer [ 1 < < 16 ] ;
int start ;
int size ;
start = MSG_ReadLong ( ) ;
size = ( unsigned short ) MSG_ReadShort ( ) ;
MSG_ReadData ( buffer , size ) ;
2009-04-12 16:57:30 +00:00
if ( cls . downloadqw )
{
VFS_SEEK ( cls . downloadqw , start ) ;
VFS_WRITE ( cls . downloadqw , buffer , size ) ;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . downloadpercent = start / ( float ) VFS_GETLEN ( cls . downloadqw ) * 100 ;
2009-04-12 16:57:30 +00:00
}
2007-06-20 00:02:54 +00:00
//this is only reliable because I'm lazy
MSG_WriteByte ( & cls . netchan . message , clcdp_ackdownloaddata ) ;
MSG_WriteLong ( & cls . netchan . message , start ) ;
MSG_WriteShort ( & cls . netchan . message , size ) ;
}
void CLDP_ParseDownloadBegin ( char * s )
{
char buffer [ 8192 ] ;
unsigned int size , pos , chunk ;
char * fname ;
Cmd_TokenizeString ( s + 1 , false , false ) ;
size = ( unsigned int ) atoi ( Cmd_Argv ( 1 ) ) ;
fname = Cmd_Argv ( 2 ) ;
COM_StripExtension ( fname , cls . downloadtempname , sizeof ( cls . downloadtempname ) - 5 ) ;
strcat ( cls . downloadtempname , " .tmp " ) ;
CL_SendClientCommand ( true , " sv_startdownload " ) ;
if ( cls . downloadqw )
{
Con_Printf ( " Warning: cl_begindownload while already downloading \n " ) ;
VFS_CLOSE ( cls . downloadqw ) ;
}
FS_CreatePath ( cls . downloadtempname , FS_GAME ) ;
cls . downloadqw = FS_OpenVFS ( cls . downloadtempname , " wb " , FS_GAME ) ;
cls . downloadmethod = DL_DARKPLACES ;
2009-04-12 16:57:30 +00:00
if ( cls . downloadqw )
2007-06-20 00:02:54 +00:00
{
2009-04-12 16:57:30 +00:00
//fill the file with 0 bytes
memset ( buffer , 0 , sizeof ( buffer ) ) ;
for ( pos = 0 , chunk = 1 ; chunk ; pos + = chunk )
{
chunk = size - pos ;
if ( chunk > sizeof ( buffer ) )
chunk = sizeof ( buffer ) ;
VFS_WRITE ( cls . downloadqw , buffer , chunk ) ;
}
2007-06-20 00:02:54 +00:00
}
2009-04-12 16:57:30 +00:00
else
2009-05-24 10:11:17 +00:00
CL_DownloadFailed ( cls . downloadremotename ) ;
2007-06-20 00:02:54 +00:00
downloadstarttime = Sys_DoubleTime ( ) ;
}
void CLDP_ParseDownloadFinished ( char * s )
{
2007-08-08 03:02:36 +00:00
unsigned short runningcrc = 0 ;
2007-06-20 00:02:54 +00:00
char buffer [ 8192 ] ;
int size , pos , chunk ;
if ( ! cls . downloadqw )
return ;
Cmd_TokenizeString ( s + 1 , false , false ) ;
VFS_CLOSE ( cls . downloadqw ) ;
cls . downloadqw = FS_OpenVFS ( cls . downloadtempname , " rb " , FS_GAME ) ;
if ( cls . downloadqw )
{
size = VFS_GETLEN ( cls . downloadqw ) ;
QCRC_Init ( & runningcrc ) ;
for ( pos = 0 , chunk = 1 ; chunk ; pos + = chunk )
{
chunk = size - pos ;
if ( chunk > sizeof ( buffer ) )
chunk = sizeof ( buffer ) ;
VFS_READ ( cls . downloadqw , buffer , chunk ) ;
QCRC_AddBlock ( & runningcrc , buffer , chunk ) ;
}
VFS_CLOSE ( cls . downloadqw ) ;
2007-08-08 03:02:36 +00:00
}
else
{
Con_Printf ( " Download failed: unable to check CRC of download \n " ) ;
2009-05-24 10:11:17 +00:00
CL_DownloadFailed ( cls . downloadremotename ) ;
2007-08-08 03:02:36 +00:00
return ;
}
2007-06-20 00:02:54 +00:00
2007-08-08 03:02:36 +00:00
Cmd_TokenizeString ( s + 1 , false , false ) ;
if ( size ! = atoi ( Cmd_Argv ( 1 ) ) )
{
Con_Printf ( " Download failed: wrong file size \n " ) ;
2009-05-24 10:11:17 +00:00
CL_DownloadFailed ( cls . downloadremotename ) ;
2007-08-08 03:02:36 +00:00
return ;
}
if ( runningcrc ! = atoi ( Cmd_Argv ( 2 ) ) )
{
Con_Printf ( " Download failed: wrong crc \n " ) ;
2009-05-24 10:11:17 +00:00
CL_DownloadFailed ( cls . downloadremotename ) ;
2007-08-08 03:02:36 +00:00
return ;
2007-06-20 00:02:54 +00:00
}
2009-05-24 10:11:17 +00:00
CL_DownloadFinished ( ) ;
* cls . downloadlocalname = ' \0 ' ;
* cls . downloadremotename = ' \0 ' ;
2007-06-20 00:02:54 +00:00
cls . downloadqw = NULL ;
cls . downloadpercent = 0 ;
Con_Printf ( " Download took %i seconds \n " , ( int ) ( Sys_DoubleTime ( ) - downloadstarttime ) ) ;
// get another file if needed
CL_RequestNextDownload ( ) ;
}
2006-01-12 22:56:54 +00:00
static vfsfile_t * upload_file ;
2004-08-23 00:15:46 +00:00
static qbyte * upload_data ;
static int upload_pos ;
static int upload_size ;
void CL_NextUpload ( void )
{
qbyte buffer [ 1024 ] ;
int r ;
int percent ;
int size ;
r = upload_size - upload_pos ;
if ( r > 768 )
r = 768 ;
2006-01-12 22:56:54 +00:00
if ( upload_data )
{
memcpy ( buffer , upload_data + upload_pos , r ) ;
}
else if ( upload_file )
{
r = VFS_READ ( upload_file , buffer , r ) ;
if ( r = = 0 )
{
CL_StopUpload ( ) ;
return ;
}
}
else
return ;
2004-08-23 00:15:46 +00:00
MSG_WriteByte ( & cls . netchan . message , clc_upload ) ;
MSG_WriteShort ( & cls . netchan . message , r ) ;
upload_pos + = r ;
size = upload_size ;
if ( ! size )
size = 1 ;
percent = upload_pos * 100 / size ;
MSG_WriteByte ( & cls . netchan . message , percent ) ;
SZ_Write ( & cls . netchan . message , buffer , r ) ;
Con_DPrintf ( " UPLOAD: %6d: %d written \n " , upload_pos - r , r ) ;
if ( upload_pos ! = upload_size )
return ;
Con_TPrintf ( TL_UPLOADCOMPLEATE ) ;
2006-01-12 22:56:54 +00:00
CL_StopUpload ( ) ;
2004-08-23 00:15:46 +00:00
}
void CL_StartUpload ( qbyte * data , int size )
{
if ( cls . state < ca_onserver )
return ; // gotta be connected
// override
2006-01-12 22:56:54 +00:00
CL_StopUpload ( ) ;
2004-08-23 00:15:46 +00:00
Con_DPrintf ( " Upload starting of %d... \n " , size ) ;
upload_data = BZ_Malloc ( size ) ;
memcpy ( upload_data , data , size ) ;
upload_size = size ;
upload_pos = 0 ;
CL_NextUpload ( ) ;
2005-09-26 03:40:09 +00:00
}
2004-08-23 00:15:46 +00:00
qboolean CL_IsUploading ( void )
{
2006-01-12 22:56:54 +00:00
if ( upload_data | | upload_file )
2004-08-23 00:15:46 +00:00
return true ;
return false ;
}
void CL_StopUpload ( void )
{
if ( upload_data )
BZ_Free ( upload_data ) ;
2006-01-12 22:56:54 +00:00
if ( upload_file )
VFS_CLOSE ( upload_file ) ;
upload_file = NULL ;
2004-08-23 00:15:46 +00:00
upload_data = NULL ;
2006-01-12 22:56:54 +00:00
upload_pos = upload_size = 0 ;
}
qboolean CL_StartUploadFile ( char * filename )
{
if ( ! COM_CheckParm ( " -fileul " ) )
{
Con_Printf ( " You must currently use the -fileul commandline parameter in order to use this functionality \n " ) ;
return false ;
}
if ( cls . state < ca_onserver )
return false ; // gotta be connected
CL_StopUpload ( ) ;
2008-12-23 02:55:20 +00:00
upload_file = FS_OpenVFS ( filename , " rb " , FS_ROOT ) ;
2006-01-12 22:56:54 +00:00
upload_size = VFS_GETLEN ( upload_file ) ;
upload_pos = 0 ;
if ( upload_file )
{
CL_NextUpload ( ) ;
return true ;
}
return false ;
2004-08-23 00:15:46 +00:00
}
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
SERVER CONNECTING MESSAGES
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
2004-12-08 07:55:43 +00:00
# ifdef CLIENTONLY
float nextdemotime ;
# endif
2004-08-23 00:15:46 +00:00
2009-04-06 00:34:32 +00:00
void CL_ClearParseState ( void )
{
// done with sounds, request models now
memset ( cl . model_precache , 0 , sizeof ( cl . model_precache ) ) ;
cl_playerindex = - 1 ;
cl_h_playerindex = - 1 ;
cl_spikeindex = - 1 ;
cl_flagindex = - 1 ;
cl_rocketindex = - 1 ;
cl_grenadeindex = - 1 ;
cl_gib1index = - 1 ;
cl_gib2index = - 1 ;
cl_gib3index = - 1 ;
}
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = = = = =
CL_ParseServerData
= = = = = = = = = = = = = = = = = =
*/
void CL_ParseServerData ( void )
{
int pnum ;
int clnum ;
char * str ;
int protover , svcnt ;
float maxspeed , entgrav ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
Con_DPrintf ( " Serverdata packet received. \n " ) ;
//
// wipe the client_state_t struct
2005-09-26 03:40:09 +00:00
//
2004-08-23 00:15:46 +00:00
2009-04-06 00:34:32 +00:00
SCR_SetLoadingStage ( LS_CLIENT ) ;
2004-08-23 00:15:46 +00:00
SCR_BeginLoadingPlaque ( ) ;
// parse protocol version number
// allow 2.2 and 2.29 demos to play
# ifdef PROTOCOL_VERSION_FTE
cls . fteprotocolextensions = 0 ;
2009-11-04 21:16:50 +00:00
cls . fteprotocolextensions2 = 0 ;
2004-08-23 00:15:46 +00:00
for ( ; ; )
{
protover = MSG_ReadLong ( ) ;
if ( protover = = PROTOCOL_VERSION_FTE )
{
2010-07-18 08:42:59 +00:00
cls . fteprotocolextensions = MSG_ReadLong ( ) ;
2009-11-04 21:16:50 +00:00
continue ;
}
if ( protover = = PROTOCOL_VERSION_FTE2 )
{
2010-07-18 08:42:59 +00:00
cls . fteprotocolextensions2 = MSG_ReadLong ( ) ;
2004-08-23 00:15:46 +00:00
continue ;
}
2010-07-18 08:42:59 +00:00
if ( protover = = PROTOCOL_VERSION_VARLENGTH )
{
int ident ;
int len ;
char data [ 1024 ] ;
ident = MSG_ReadLong ( ) ;
len = MSG_ReadLong ( ) ;
if ( len < = sizeof ( data ) )
{
MSG_ReadData ( data , len ) ;
switch ( ident )
{
default :
break ;
}
continue ;
}
}
2008-04-12 23:24:19 +00:00
if ( protover = = PROTOCOL_VERSION_QW ) //this ends the version info
2004-08-23 00:15:46 +00:00
break ;
if ( cls . demoplayback & & ( protover = = 26 | | protover = = 27 | | protover = = 28 ) ) //older versions, maintain demo compatability.
break ;
2008-04-12 23:24:19 +00:00
Host_EndGame ( " Server returned version %i, not %i \n " , protover , PROTOCOL_VERSION_QW ) ;
2004-08-23 00:15:46 +00:00
}
# else
2005-09-26 03:40:09 +00:00
protover = MSG_ReadLong ( ) ;
2008-04-12 23:24:19 +00:00
if ( protover ! = PROTOCOL_VERSION_QW & &
2004-08-23 00:15:46 +00:00
! ( cls . demoplayback & & ( protover = = 26 | | protover = = 27 | | protover = = 28 ) ) )
2008-04-12 23:24:19 +00:00
Host_EndGame ( " Server returned version %i, not %i \n " , protover , PROTOCOL_VERSION_QW ) ;
2004-08-23 00:15:46 +00:00
# endif
2009-11-04 21:16:50 +00:00
if ( cls . fteprotocolextensions2 | | cls . fteprotocolextensions )
if ( developer . ival | | cl_shownet . ival )
Con_TPrintf ( TL_FTEEXTENSIONS , cls . fteprotocolextensions2 , cls . fteprotocolextensions ) ;
2004-11-27 08:16:25 +00:00
if ( cls . fteprotocolextensions & PEXT_FLOATCOORDS )
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . netchan . netprim . coordsize = 4 ;
cls . netchan . netprim . anglesize = 2 ;
2004-11-27 08:16:25 +00:00
}
else
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . netchan . netprim . coordsize = 2 ;
cls . netchan . netprim . anglesize = 1 ;
2004-11-27 08:16:25 +00:00
}
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . netchan . message . prim = cls . netchan . netprim ;
MSG_ChangePrimitives ( cls . netchan . netprim ) ;
2004-11-27 08:16:25 +00:00
2005-09-08 01:58:25 +00:00
svcnt = MSG_ReadLong ( ) ;
2004-08-23 00:15:46 +00:00
// game directory
str = MSG_ReadString ( ) ;
2006-02-22 23:35:04 +00:00
if ( ! * str )
str = " qw " ;
2004-08-23 00:15:46 +00:00
# ifndef CLIENTONLY
if ( ! sv . state )
# endif
{
COM_FlushTempoaryPacks ( ) ;
COM_Gamedir ( str ) ;
2004-11-29 01:21:00 +00:00
# ifndef CLIENTONLY
2004-08-23 00:15:46 +00:00
Info_SetValueForStarKey ( svs . info , " *gamedir " , str , MAX_SERVERINFO_STRING ) ;
2004-11-29 01:21:00 +00:00
# endif
2004-08-23 00:15:46 +00:00
COM_FlushFSCache ( ) ;
}
CL_ClearState ( ) ;
2008-11-09 22:29:28 +00:00
Cvar_ForceCallback ( Cvar_FindVar ( " r_particlesdesc " ) ) ;
2004-08-23 00:15:46 +00:00
Stats_NewMap ( ) ;
cl . servercount = svcnt ;
2004-12-29 03:24:21 +00:00
cl . teamfortress = ! Q_strcasecmp ( str , " fortress " ) ;
2004-08-23 00:15:46 +00:00
if ( cl . gamedirchanged )
{
cl . gamedirchanged = false ;
# ifndef CLIENTONLY
if ( ! sv . state )
# endif
Wads_Flush ( ) ;
}
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
2004-08-23 00:15:46 +00:00
{
int i ;
2008-01-30 02:32:00 +00:00
MSG_ReadFloat ( ) ;
2004-08-23 00:15:46 +00:00
cl . playernum [ 0 ] = MAX_CLIENTS - 1 ;
2009-05-24 10:11:17 +00:00
cl . playernum [ 1 ] = MAX_CLIENTS - 2 ;
cl . playernum [ 2 ] = MAX_CLIENTS - 3 ;
cl . playernum [ 3 ] = MAX_CLIENTS - 4 ;
2004-08-23 00:15:46 +00:00
cl . spectator = true ;
for ( i = 0 ; i < UPDATE_BACKUP ; i + + )
cl . frames [ i ] . playerstate [ cl . playernum [ 0 ] ] . pm_type = PM_SPECTATOR ;
cl . splitclients = 1 ;
}
else
{
// parse player slot, high bit means spectator
pnum = MSG_ReadByte ( ) ;
for ( clnum = 0 ; ; clnum + + )
{
cl . playernum [ clnum ] = pnum ;
if ( cl . playernum [ clnum ] & 128 )
{
cl . spectator = true ;
cl . playernum [ clnum ] & = ~ 128 ;
}
if ( ! ( cls . fteprotocolextensions & PEXT_SPLITSCREEN ) )
break ;
pnum = MSG_ReadByte ( ) ;
if ( pnum = = 128 )
break ;
if ( clnum = = MAX_SPLITS )
Host_EndGame ( " Server sent us too many alternate clients \n " ) ;
}
cl . splitclients = clnum + 1 ;
}
// get the full level name
str = MSG_ReadString ( ) ;
Q_strncpyz ( cl . levelname , str , sizeof ( cl . levelname ) ) ;
// get the movevars
movevars . gravity = MSG_ReadFloat ( ) ;
movevars . stopspeed = MSG_ReadFloat ( ) ;
maxspeed = MSG_ReadFloat ( ) ;
movevars . spectatormaxspeed = MSG_ReadFloat ( ) ;
movevars . accelerate = MSG_ReadFloat ( ) ;
movevars . airaccelerate = MSG_ReadFloat ( ) ;
movevars . wateraccelerate = MSG_ReadFloat ( ) ;
movevars . friction = MSG_ReadFloat ( ) ;
movevars . waterfriction = MSG_ReadFloat ( ) ;
entgrav = MSG_ReadFloat ( ) ;
for ( clnum = 0 ; clnum < cl . splitclients ; clnum + + )
{
cl . maxspeed [ clnum ] = maxspeed ;
cl . entgravity [ clnum ] = entgrav ;
}
// seperate the printfs so the server message can have a color
2009-07-05 18:45:53 +00:00
# if 1
{
int i ;
Con_Printf ( " \n \n " ) ;
Con_Printf ( " ^Ue01d " ) ;
for ( i = 34 ; i - - > 0 ; i - - )
Con_Printf ( " ^Ue01e " ) ;
Con_Printf ( " ^Ue01f " ) ;
Con_Printf ( " \n \n " ) ;
}
Con_Printf ( " \1 %s \n " , str ) ;
# else
2004-08-23 00:15:46 +00:00
Con_TPrintf ( TLC_LINEBREAK_NEWLEVEL ) ;
Con_TPrintf ( TLC_PC_PS_NL , 2 , str ) ;
2009-07-05 18:45:53 +00:00
# endif
2004-08-23 00:15:46 +00:00
2005-06-04 04:20:20 +00:00
if ( CL_RemoveClientCommands ( " new " ) ) //mvdsv is really appaling some times.
2005-10-01 13:47:47 +00:00
{
// Con_Printf("Multiple 'new' commands?!?!? This server needs reinstalling!\n");
}
2005-06-04 04:20:20 +00:00
2004-08-23 00:15:46 +00:00
memset ( cl . sound_name , 0 , sizeof ( cl . sound_name ) ) ;
2005-01-24 23:47:32 +00:00
# ifdef PEXT_PK3DOWNLOADS
if ( cls . fteprotocolextensions & PEXT_PK3DOWNLOADS ) //instead of going for a soundlist, go for the pk3 list instead. The server will make us go for the soundlist after.
2004-11-17 17:58:22 +00:00
{
2006-01-02 23:00:10 +00:00
if ( CL_RemoveClientCommands ( " pk3list " ) )
Con_Printf ( " Multiple pk3lists \n " ) ;
2005-02-28 07:16:19 +00:00
CL_SendClientCommand ( " pk3list %i 0 " , cl . servercount , 0 ) ;
2004-11-17 17:58:22 +00:00
}
else
2005-01-24 23:47:32 +00:00
# endif
2004-11-17 17:58:22 +00:00
{
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_EZTV )
{
if ( CL_RemoveClientCommands ( " qtvsoundlist " ) )
Con_Printf ( " Multiple soundlists \n " ) ;
CL_SendClientCommand ( true , " qtvsoundlist %i 0 " , cl . servercount ) ;
}
else
{
if ( CL_RemoveClientCommands ( " soundlist " ) )
Con_Printf ( " Multiple soundlists \n " ) ;
// ask for the sound list next
// CL_SendClientCommand ("soundlist %i 0", cl.servercount);
CL_SendClientCommand ( true , soundlist_name , cl . servercount , 0 ) ;
}
2004-11-17 17:58:22 +00:00
}
2004-08-23 00:15:46 +00:00
// now waiting for downloads, etc
cls . state = ca_onserver ;
2006-01-10 18:50:23 +00:00
cl . sendprespawn = false ;
2004-08-23 00:15:46 +00:00
# ifdef VM_CG
CG_Stop ( ) ;
# endif
2005-02-09 19:32:09 +00:00
# ifdef CSQC_DAT
2005-03-10 03:55:18 +00:00
CSQC_Shutdown ( ) ; //revive it when we get the serverinfo saying the checksum.
2005-02-09 19:32:09 +00:00
# endif
2004-08-23 00:15:46 +00:00
}
2010-08-11 09:31:24 +00:00
# ifdef Q2CLIENT
2004-08-23 00:15:46 +00:00
void CLQ2_ParseServerData ( void )
{
char * str ;
int i ;
int svcnt ;
// int cflag;
2004-11-27 08:16:25 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . netchan . netprim . coordsize = 2 ;
cls . netchan . netprim . anglesize = 1 ;
MSG_ChangePrimitives ( cls . netchan . netprim ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
Con_DPrintf ( " Serverdata packet received. \n " ) ;
//
// wipe the client_state_t struct
//
2009-04-06 00:34:32 +00:00
SCR_SetLoadingStage ( LS_CLIENT ) ;
2004-08-23 00:15:46 +00:00
SCR_BeginLoadingPlaque ( ) ;
// CL_ClearState ();
cls . state = ca_onserver ;
// parse protocol version number
i = MSG_ReadLong ( ) ;
// cls.serverProtocol = i;
if ( i > PROTOCOL_VERSION_Q2 | | i < PROTOCOL_VERSION_Q2_MIN )
Host_EndGame ( " Server returned version %i, not %i " , i , PROTOCOL_VERSION_Q2 ) ;
svcnt = MSG_ReadLong ( ) ;
/*cl.attractloop =*/ MSG_ReadByte ( ) ;
// game directory
str = MSG_ReadString ( ) ;
// strncpy (cl.gamedir, str, sizeof(cl.gamedir)-1);
// set gamedir
if ( ! * str )
COM_Gamedir ( " baseq2 " ) ;
else
COM_Gamedir ( str ) ;
COM_FlushFSCache ( ) ;
// if ((*str && (!fs_gamedirvar->string || !*fs_gamedirvar->string || strcmp(fs_gamedirvar->string, str))) || (!*str && (fs_gamedirvar->string || *fs_gamedirvar->string)))
// Cvar_Set("game", str);
2006-04-11 20:09:39 +00:00
Cvar_Get ( " timescale " , " 1 " , 0 , " Q2Admin hacks " ) ; //Q2Admin will kick players who have a timescale set to something other than 1
//FTE doesn't actually have a timescale cvar, so create one to fool q2admin.
//I can't really blame q2admin for rejecting engines that don't have this cvar, as it could have been renamed via a hex-edit.
2004-08-23 00:15:46 +00:00
CL_ClearState ( ) ;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
CLQ2_ClearState ( ) ;
2004-08-23 00:15:46 +00:00
cl . minpitch = - 89 ;
cl . maxpitch = 89 ;
cl . servercount = svcnt ;
Stats_NewMap ( ) ;
// parse player entity number
cl . playernum [ 0 ] = MSG_ReadShort ( ) ;
cl . splitclients = 1 ;
cl . spectator = false ;
2006-04-11 20:09:39 +00:00
cl . numq2visibleweapons = 1 ; //give it a default.
cl . q2visibleweapons [ 0 ] = " weapon.md2 " ;
2004-08-23 00:15:46 +00:00
// get the full level name
str = MSG_ReadString ( ) ;
Q_strncpyz ( cl . levelname , str , sizeof ( cl . levelname ) ) ;
if ( cl . playernum [ 0 ] = = - 1 )
{ // playing a cinematic or showing a pic, not a level
SCR_EndLoadingPlaque ( ) ;
if ( ! Media_PlayFilm ( str ) )
Con_TPrintf ( TLC_NOQ2CINEMATICSSUPPORT , cl . servercount ) ;
2004-08-27 00:48:03 +00:00
else
2005-07-14 01:57:34 +00:00
CL_MakeActive ( " Quake2 " ) ;
2004-08-23 00:15:46 +00:00
}
else
{
// seperate the printfs so the server message can have a color
Con_TPrintf ( TLC_LINEBREAK_NEWLEVEL ) ;
Con_TPrintf ( TLC_PC_PS_NL , 2 , str ) ;
2004-08-27 00:48:03 +00:00
Media_PlayFilm ( " " ) ;
2004-08-23 00:15:46 +00:00
// need to prep refresh at next oportunity
//cl.refresh_prepped = false;
}
2008-11-09 22:29:28 +00:00
Cvar_ForceCallback ( Cvar_FindVar ( " r_particlesdesc " ) ) ;
2004-08-23 00:15:46 +00:00
if ( R_PreNewMap )
R_PreNewMap ( ) ;
}
2010-08-11 09:31:24 +00:00
# endif
2004-08-23 00:15:46 +00:00
2008-12-03 02:42:05 +00:00
void CL_ParseEstablished ( void )
{
# ifdef NQPROT
cl_dp_serverextension_download = false ;
cl_dp_csqc_progscrc = 0 ;
cl_dp_csqc_progssize = 0 ;
# endif
}
2004-08-23 00:15:46 +00:00
# ifdef NQPROT
//FIXME: move to header
void CL_KeepaliveMessage ( void ) { }
void CLNQ_ParseServerData ( void ) //Doesn't change gamedir - use with caution.
{
2008-11-09 22:29:28 +00:00
int nummodels , numsounds ;
2004-08-23 00:15:46 +00:00
char * str ;
2008-11-09 22:29:28 +00:00
int gametype ;
2004-08-23 00:15:46 +00:00
int protover ;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
struct netprim_s netprim ;
2009-11-04 21:16:50 +00:00
if ( developer . ival )
2004-08-23 00:15:46 +00:00
Con_TPrintf ( TLC_GOTSVDATAPACKET ) ;
2009-04-06 00:34:32 +00:00
SCR_SetLoadingStage ( LS_CLIENT ) ;
2004-08-23 00:15:46 +00:00
CL_ClearState ( ) ;
Stats_NewMap ( ) ;
2008-11-09 22:29:28 +00:00
Cvar_ForceCallback ( Cvar_FindVar ( " r_particlesdesc " ) ) ;
2007-06-20 00:02:54 +00:00
2005-09-26 03:40:09 +00:00
protover = MSG_ReadLong ( ) ;
2004-08-23 00:15:46 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
netprim . coordsize = 2 ;
netprim . anglesize = 1 ;
2004-11-27 08:16:25 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . protocol_nq = 0 ;
2004-12-05 08:19:54 +00:00
cls . z_ext = 0 ;
2004-11-27 08:16:25 +00:00
2010-08-16 02:03:02 +00:00
if ( protover = = NEHD_PROTOCOL_VERSION )
2004-08-23 00:15:46 +00:00
Host_EndGame ( " Nehahra demo net protocol is not supported \n " ) ;
2010-08-16 02:03:02 +00:00
else if ( protover = = FITZ_PROTOCOL_VERSION )
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
{
//fitzquake 0.85
cls . protocol_nq = CPNQ_FITZ666 ;
Con_DPrintf ( " FitzQuake 666 protocol \n " ) ;
}
2010-08-16 02:03:02 +00:00
else if ( protover = = DP5_PROTOCOL_VERSION )
2004-11-27 08:16:25 +00:00
{
//darkplaces5
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . protocol_nq = CPNQ_DP5 ;
netprim . coordsize = 4 ;
netprim . anglesize = 2 ;
2005-09-26 03:40:09 +00:00
Con_DPrintf ( " DP5 protocols \n " ) ;
2004-11-27 08:16:25 +00:00
}
2005-05-27 05:41:07 +00:00
else if ( protover = = DP6_PROTOCOL_VERSION )
2004-12-05 08:19:54 +00:00
{
//darkplaces6 (it's a small difference from dp5)
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . protocol_nq = CPNQ_DP6 ;
netprim . coordsize = 4 ;
netprim . anglesize = 2 ;
2004-12-05 08:19:54 +00:00
cls . z_ext = Z_EXT_VIEWHEIGHT ;
2005-09-26 03:40:09 +00:00
Con_DPrintf ( " DP6 protocols \n " ) ;
2004-12-05 08:19:54 +00:00
}
2005-05-27 05:41:07 +00:00
else if ( protover = = DP7_PROTOCOL_VERSION )
2005-05-26 12:55:34 +00:00
{
//darkplaces7 (it's a small difference from dp5)
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . protocol_nq = CPNQ_DP7 ;
netprim . coordsize = 4 ;
netprim . anglesize = 2 ;
2005-05-26 12:55:34 +00:00
cls . z_ext = Z_EXT_VIEWHEIGHT ;
2005-09-26 03:40:09 +00:00
Con_DPrintf ( " DP7 protocols \n " ) ;
2005-05-26 12:55:34 +00:00
}
2010-08-16 02:03:02 +00:00
else if ( protover = = H2_PROTOCOL_VERSION )
{
Host_EndGame ( " \n Unable to connect to Hexen2 servers. \n " ) ;
}
2004-08-23 00:15:46 +00:00
else if ( protover ! = NQ_PROTOCOL_VERSION )
{
Host_EndGame ( " Server returned version %i, not %i \n You will need to use a different client. " , protover , NQ_PROTOCOL_VERSION ) ;
}
2005-09-26 03:40:09 +00:00
else
{
Con_DPrintf ( " Standard NQ protocols \n " ) ;
}
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cls . netchan . message . prim = cls . netchan . netprim = netprim ;
MSG_ChangePrimitives ( netprim ) ;
2005-09-26 03:40:09 +00:00
2005-05-17 02:36:54 +00:00
if ( MSG_ReadByte ( ) > MAX_CLIENTS )
2004-08-23 00:15:46 +00:00
{
2006-01-28 19:04:13 +00:00
Con_Printf ( " \n Warning, this server supports more than %i clients, additional clients will do bad things \n " , MAX_CLIENTS ) ;
2004-08-23 00:15:46 +00:00
}
cl . splitclients = 1 ;
2008-11-09 22:29:28 +00:00
gametype = MSG_ReadByte ( ) ;
2004-08-23 00:15:46 +00:00
str = MSG_ReadString ( ) ;
Q_strncpyz ( cl . levelname , str , sizeof ( cl . levelname ) ) ;
// seperate the printfs so the server message can have a color
Con_TPrintf ( TLC_LINEBREAK_NEWLEVEL ) ;
Con_TPrintf ( TLC_PC_PS_NL , 2 , str ) ;
SCR_BeginLoadingPlaque ( ) ;
if ( R_PreNewMap )
R_PreNewMap ( ) ;
memset ( cl . model_name , 0 , sizeof ( cl . model_name ) ) ;
for ( nummodels = 1 ; ; nummodels + + )
{
str = MSG_ReadString ( ) ;
if ( ! str [ 0 ] )
break ;
if ( nummodels = = MAX_MODELS )
{
Con_TPrintf ( TLC_TOOMANYMODELPRECACHES ) ;
return ;
}
strcpy ( cl . model_name [ nummodels ] , str ) ;
2008-12-03 02:42:05 +00:00
if ( * str ! = ' * ' ) //not inline models!
CL_CheckOrEnqueDownloadFile ( str , NULL , 0 ) ;
2004-08-23 00:15:46 +00:00
Mod_TouchModel ( str ) ;
}
memset ( cl . sound_name , 0 , sizeof ( cl . sound_name ) ) ;
for ( numsounds = 1 ; ; numsounds + + )
{
2005-09-26 03:40:09 +00:00
str = MSG_ReadString ( ) ;
2004-08-23 00:15:46 +00:00
if ( ! str [ 0 ] )
break ;
if ( numsounds = = MAX_SOUNDS )
{
Con_TPrintf ( TLC_TOOMANYSOUNDPRECACHES ) ;
return ;
}
strcpy ( cl . sound_name [ numsounds ] , str ) ;
2008-12-03 02:42:05 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
# pragma message("CLNQ_ParseServerData: no sound autodownloads")
2008-12-03 02:42:05 +00:00
//CL_CheckOrEnqueDownloadFile(str, NULL, 0);
2004-08-23 00:15:46 +00:00
S_TouchSound ( str ) ;
}
2008-11-09 22:29:28 +00:00
cls . signon = 0 ;
cls . state = ca_onserver ;
2004-08-23 00:15:46 +00:00
2008-11-09 22:29:28 +00:00
//fill in the csqc stuff
2009-03-03 01:52:30 +00:00
if ( ! cl_dp_csqc_progscrc )
{
Info_RemoveKey ( cl . serverinfo , " *csprogs " ) ;
Info_RemoveKey ( cl . serverinfo , " *csprogssize " ) ;
Info_RemoveKey ( cl . serverinfo , " *csprogsname " ) ;
}
else
{
Info_SetValueForStarKey ( cl . serverinfo , " *csprogs " , va ( " %i " , cl_dp_csqc_progscrc ) , sizeof ( cl . serverinfo ) ) ;
Info_SetValueForStarKey ( cl . serverinfo , " *csprogssize " , va ( " %i " , cl_dp_csqc_progssize ) , sizeof ( cl . serverinfo ) ) ;
2009-11-04 21:16:50 +00:00
Info_SetValueForStarKey ( cl . serverinfo , " *csprogsname " , va ( " %s " , cl_dp_csqc_progsname ) , sizeof ( cl . serverinfo ) ) ;
2009-03-03 01:52:30 +00:00
}
2004-08-23 00:15:46 +00:00
2008-11-09 22:29:28 +00:00
//update gamemode
if ( gametype = = 1 )
Info_SetValueForStarKey ( cl . serverinfo , " deathmatch " , " 1 " , sizeof ( cl . serverinfo ) ) ;
else
Info_SetValueForStarKey ( cl . serverinfo , " deathmatch " , " 0 " , sizeof ( cl . serverinfo ) ) ;
Info_SetValueForStarKey ( cl . serverinfo , " teamplay " , " 0 " , sizeof ( cl . serverinfo ) ) ;
2004-08-23 00:15:46 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
//allow some things by default that quakeworld bans by default
Info_SetValueForStarKey ( cl . serverinfo , " watervis " , " 1 " , sizeof ( cl . serverinfo ) ) ;
2005-05-19 02:53:03 +00:00
2008-11-09 22:29:28 +00:00
//pretend it came from the server, and update cheat/permissions/etc
CL_CheckServerInfo ( ) ;
2004-08-23 00:15:46 +00:00
2009-07-18 20:46:42 +00:00
# ifdef PEXT_CSQC
2008-11-09 22:29:28 +00:00
CSQC_Shutdown ( ) ;
if ( cls . demoplayback )
CSQC_Init ( 0 ) ;
2009-07-18 20:46:42 +00:00
# endif
2004-08-23 00:15:46 +00:00
}
void CLNQ_SignonReply ( void )
2005-02-28 07:16:19 +00:00
{
2004-08-23 00:15:46 +00:00
extern cvar_t topcolor ;
extern cvar_t bottomcolor ;
2005-10-01 03:09:17 +00:00
extern cvar_t rate ;
extern cvar_t model ;
extern cvar_t skin ;
2004-08-23 00:15:46 +00:00
Con_DPrintf ( " CL_SignonReply: %i \n " , cls . signon ) ;
switch ( cls . signon )
{
case 1 :
2007-06-20 00:02:54 +00:00
cl . sendprespawn = true ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
case 2 :
2005-03-23 22:14:08 +00:00
CL_SendClientCommand ( true , " name \" %s \" \n " , name . string ) ;
2005-09-26 03:40:09 +00:00
2009-11-04 21:16:50 +00:00
CL_SendClientCommand ( true , " color %i %i \n " , topcolor . ival , bottomcolor . ival ) ;
2005-09-26 03:40:09 +00:00
2005-03-23 22:14:08 +00:00
CL_SendClientCommand ( true , " spawn %s " , " " ) ;
2005-10-01 03:09:17 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( CPNQ_IS_DP ) //dp needs a couple of extras to work properly.
2005-10-01 03:09:17 +00:00
{
CL_SendClientCommand ( true , " rate %s " , rate . string ) ;
CL_SendClientCommand ( true , " playermodel %s " , model . string ) ;
CL_SendClientCommand ( true , " playerskin %s " , skin . string ) ;
2009-07-18 20:46:42 +00:00
# ifdef PEXT_CSQC
2007-06-20 00:02:54 +00:00
{
char * s ;
s = Info_ValueForKey ( cl . serverinfo , " *csprogs " ) ;
if ( * s )
CSQC_Init ( atoi ( s ) ) ;
else
CSQC_Shutdown ( ) ;
}
2009-07-18 20:46:42 +00:00
# endif
2005-10-01 03:09:17 +00:00
}
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
case 3 :
2005-03-23 22:14:08 +00:00
CL_SendClientCommand ( true , " begin " ) ;
2004-08-23 00:15:46 +00:00
Cache_Report ( ) ; // print remaining memory
2004-08-23 04:43:03 +00:00
# ifdef VM_CG
2004-08-23 00:15:46 +00:00
CG_Start ( ) ;
2004-08-23 04:43:03 +00:00
# endif
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case 4 :
2005-09-26 03:40:09 +00:00
SCR_EndLoadingPlaque ( ) ; // allow normal screen updates
2009-04-19 00:50:42 +00:00
SCR_SetLoadingStage ( LS_NONE ) ;
2004-08-23 00:15:46 +00:00
break ;
}
}
# define SU_VIEWHEIGHT (1<<0)
# define SU_IDEALPITCH (1<<1)
# define SU_PUNCH1 (1<<2)
# define SU_PUNCH2 (1<<3)
# define SU_PUNCH3 (1<<4)
# define SU_VELOCITY1 (1<<5)
# define SU_VELOCITY2 (1<<6)
# define SU_VELOCITY3 (1<<7)
//define SU_AIMENT (1<<8) AVAILABLE BIT
# define SU_ITEMS (1<<9)
# define SU_ONGROUND (1<<10) // no data follows, the bit is it
# define SU_INWATER (1<<11) // no data follows, the bit is it
# define SU_WEAPONFRAME (1<<12)
# define SU_ARMOR (1<<13)
# define SU_WEAPON (1<<14)
2004-11-27 08:16:25 +00:00
# define DPSU_EXTEND1 (1<<15)
// first extend byte
# define DPSU_PUNCHVEC1 (1<<16)
# define DPSU_PUNCHVEC2 (1<<17)
# define DPSU_PUNCHVEC3 (1<<18)
# define DPSU_VIEWZOOM (1<<19) // byte factor (0 = 0.0 (not valid), 255 = 1.0)
# define DPSU_UNUSED20 (1<<20)
# define DPSU_UNUSED21 (1<<21)
# define DPSU_UNUSED22 (1<<22)
# define DPSU_EXTEND2 (1<<23) // another byte to follow, future expansion
// second extend byte
# define DPSU_UNUSED24 (1<<24)
# define DPSU_UNUSED25 (1<<25)
# define DPSU_UNUSED26 (1<<26)
# define DPSU_UNUSED27 (1<<27)
# define DPSU_UNUSED28 (1<<28)
# define DPSU_UNUSED29 (1<<29)
# define DPSU_UNUSED30 (1<<30)
# define DPSU_EXTEND3 (1<<31) // another byte to follow, future expansion
2004-08-23 00:15:46 +00:00
# define DEFAULT_VIEWHEIGHT 22
2005-09-26 03:40:09 +00:00
void CLNQ_ParseClientdata ( void )
2004-08-23 00:15:46 +00:00
{
2004-12-05 08:19:54 +00:00
int i ;
2004-11-27 08:16:25 +00:00
2005-09-26 03:40:09 +00:00
unsigned int bits ;
2006-02-12 20:20:01 +00:00
bits = ( unsigned short ) MSG_ReadShort ( ) ;
2004-11-27 08:16:25 +00:00
if ( bits & DPSU_EXTEND1 )
bits | = ( MSG_ReadByte ( ) < < 16 ) ;
if ( bits & DPSU_EXTEND2 )
bits | = ( MSG_ReadByte ( ) < < 24 ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
if ( bits & SU_VIEWHEIGHT )
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_VIEWHEIGHT , MSG_ReadChar ( ) ) ;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
else if ( CPNQ_IS_DP | | cls . protocol_nq = = CPNQ_DP5 )
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_VIEWHEIGHT , DEFAULT_VIEWHEIGHT ) ;
2004-08-23 00:15:46 +00:00
if ( bits & SU_IDEALPITCH )
/*cl.idealpitch =*/ MSG_ReadChar ( ) ;
/*else
cl . idealpitch = 0 ; */
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
// VectorCopy (cl.mvelocity[0], cl.mvelocity[1]);
for ( i = 0 ; i < 3 ; i + + )
{
if ( bits & ( SU_PUNCH1 < < i ) )
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
/*cl.punchangle[i] =*/ CPNQ_IS_DP ? MSG_ReadAngle16 ( ) : MSG_ReadChar ( ) ;
2004-08-23 00:15:46 +00:00
// else
// cl.punchangle[i] = 0;
2004-11-27 08:16:25 +00:00
if ( bits & ( DPSU_PUNCHVEC1 < < i ) )
{
/*cl.punchvector[i] =*/ MSG_ReadCoord ( ) ;
}
// else
// cl.punchvector[i] = 0;
2004-08-23 00:15:46 +00:00
if ( bits & ( SU_VELOCITY1 < < i ) )
2004-11-27 08:16:25 +00:00
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( CPNQ_IS_DP )
2004-11-27 08:16:25 +00:00
/*cl.simvel[0][i] =*/ MSG_ReadFloat ( ) ;
else
2004-08-23 00:15:46 +00:00
/*cl.mvelocity[0][i] =*/ MSG_ReadChar ( ) /**16*/ ;
2004-11-27 08:16:25 +00:00
}
2004-08-23 00:15:46 +00:00
// else
// cl.mvelocity[0][i] = 0;
}
2004-12-05 08:19:54 +00:00
if ( bits & SU_ITEMS )
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_ITEMS , MSG_ReadLong ( ) ) ;
2004-08-23 00:15:46 +00:00
// cl.onground = (bits & SU_ONGROUND) != 0;
// cl.inwater = (bits & SU_INWATER) != 0;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( cls . protocol_nq = = CPNQ_DP5 )
2004-12-05 08:19:54 +00:00
{
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_WEAPONFRAME , ( bits & SU_WEAPONFRAME ) ? ( unsigned short ) MSG_ReadShort ( ) : 0 ) ;
CL_SetStatInt ( 0 , STAT_ARMOR , ( bits & SU_ARMOR ) ? MSG_ReadShort ( ) : 0 ) ;
CL_SetStatInt ( 0 , STAT_WEAPON , ( bits & SU_WEAPON ) ? MSG_ReadShort ( ) : 0 ) ;
2004-11-27 08:16:25 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_HEALTH , MSG_ReadShort ( ) ) ;
2004-11-27 08:16:25 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_AMMO , MSG_ReadShort ( ) ) ;
2004-11-27 08:16:25 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_SHELLS , MSG_ReadShort ( ) ) ;
CL_SetStatInt ( 0 , STAT_NAILS , MSG_ReadShort ( ) ) ;
CL_SetStatInt ( 0 , STAT_ROCKETS , MSG_ReadShort ( ) ) ;
CL_SetStatInt ( 0 , STAT_CELLS , MSG_ReadShort ( ) ) ;
2004-08-23 00:15:46 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_ACTIVEWEAPON , ( unsigned short ) MSG_ReadShort ( ) ) ;
2004-08-23 00:15:46 +00:00
}
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
else if ( CPNQ_IS_DP )
{
/*nothing*/
}
2004-11-27 08:16:25 +00:00
else
2004-08-23 00:15:46 +00:00
{
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_WEAPONFRAME , ( bits & SU_WEAPONFRAME ) ? ( unsigned char ) MSG_ReadByte ( ) : 0 ) ;
CL_SetStatInt ( 0 , STAT_ARMOR , ( bits & SU_ARMOR ) ? MSG_ReadByte ( ) : 0 ) ;
CL_SetStatInt ( 0 , STAT_WEAPON , ( bits & SU_WEAPON ) ? MSG_ReadByte ( ) : 0 ) ;
2004-08-23 00:15:46 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_HEALTH , MSG_ReadShort ( ) ) ;
2004-08-23 00:15:46 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_AMMO , MSG_ReadByte ( ) ) ;
2004-11-27 08:16:25 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_SHELLS , MSG_ReadByte ( ) ) ;
CL_SetStatInt ( 0 , STAT_NAILS , MSG_ReadByte ( ) ) ;
CL_SetStatInt ( 0 , STAT_ROCKETS , MSG_ReadByte ( ) ) ;
CL_SetStatInt ( 0 , STAT_CELLS , MSG_ReadByte ( ) ) ;
2004-08-23 00:15:46 +00:00
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , STAT_ACTIVEWEAPON , MSG_ReadByte ( ) ) ;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( cls . protocol_nq = = CPNQ_FITZ666 )
{
# define FITZSU_WEAPON2 (1<<16) // 1 byte, this is .weaponmodel & 0xFF00 (second byte)
# define FITZSU_ARMOR2 (1<<17) // 1 byte, this is .armorvalue & 0xFF00 (second byte)
# define FITZSU_AMMO2 (1<<18) // 1 byte, this is .currentammo & 0xFF00 (second byte)
# define FITZSU_SHELLS2 (1<<19) // 1 byte, this is .ammo_shells & 0xFF00 (second byte)
# define FITZSU_NAILS2 (1<<20) // 1 byte, this is .ammo_nails & 0xFF00 (second byte)
# define FITZSU_ROCKETS2 (1<<21) // 1 byte, this is .ammo_rockets & 0xFF00 (second byte)
# define FITZSU_CELLS2 (1<<22) // 1 byte, this is .ammo_cells & 0xFF00 (second byte)
# define FITZSU_WEAPONFRAME2 (1<<24) // 1 byte, this is .weaponframe & 0xFF00 (second byte)
# define FITZSU_WEAPONALPHA (1<<25) // 1 byte, this is alpha for weaponmodel, uses ENTALPHA_ENCODE, not sent if ENTALPHA_DEFAULT
if ( bits & FITZSU_WEAPON2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_ARMOR2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_AMMO2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_SHELLS2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_NAILS2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_ROCKETS2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_CELLS2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_WEAPONFRAME2 )
MSG_ReadByte ( ) ;
if ( bits & FITZSU_WEAPONALPHA )
MSG_ReadByte ( ) ;
}
2004-08-23 00:15:46 +00:00
}
2004-11-27 08:16:25 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( CPNQ_IS_DP | | cls . protocol_nq = = CPNQ_DP5 )
2004-11-27 08:16:25 +00:00
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( bits & DPSU_VIEWZOOM )
{
if ( cls . protocol_nq )
i = ( unsigned short ) MSG_ReadShort ( ) ;
else
i = MSG_ReadByte ( ) ;
if ( i < 2 )
i = 2 ;
CL_SetStatInt ( 0 , STAT_VIEWZOOM , i ) ;
}
2004-11-27 08:16:25 +00:00
else
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
CL_SetStatInt ( 0 , STAT_VIEWZOOM , 255 ) ;
2004-11-27 08:16:25 +00:00
}
2004-08-23 00:15:46 +00:00
}
# endif
/*
= = = = = = = = = = = = = = = = = =
CL_ParseSoundlist
= = = = = = = = = = = = = = = = = =
*/
2008-11-28 20:34:51 +00:00
void CL_ParseSoundlist ( qboolean lots )
2004-08-23 00:15:46 +00:00
{
int numsounds ;
char * str ;
int n ;
// precache sounds
// memset (cl.sound_precache, 0, sizeof(cl.sound_precache));
2008-11-28 20:34:51 +00:00
if ( lots )
numsounds = MSG_ReadShort ( ) ;
else
numsounds = MSG_ReadByte ( ) ;
2004-08-23 00:15:46 +00:00
2005-02-28 07:16:19 +00:00
for ( ; ; )
{
2004-08-23 00:15:46 +00:00
str = MSG_ReadString ( ) ;
if ( ! str [ 0 ] )
break ;
numsounds + + ;
2005-08-07 18:08:13 +00:00
if ( numsounds > = MAX_SOUNDS )
2004-08-23 00:15:46 +00:00
Host_EndGame ( " Server sent too many sound_precache " ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
// if (strlen(str)>4)
// if (!strcmp(str+strlen(str)-4, ".mp3")) //don't let the server send us a specific mp3. convert it to wav and this way we know not to look outside the quake path for it.
// strcpy(str+strlen(str)-4, ".wav");
2005-09-26 03:40:09 +00:00
strcpy ( cl . sound_name [ numsounds ] , str ) ;
2004-08-23 00:15:46 +00:00
}
n = MSG_ReadByte ( ) ;
2005-02-28 07:16:19 +00:00
if ( n )
{
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback ! = DPB_EZTV )
{
if ( CL_RemoveClientCommands ( " soundlist " ) )
Con_Printf ( " Multiple soundlists \n " ) ;
// CL_SendClientCommand("soundlist %i %i", cl.servercount, n);
2008-11-28 20:34:51 +00:00
CL_SendClientCommand ( true , soundlist_name , cl . servercount , ( numsounds & 0xff00 ) + n ) ;
2008-01-09 00:52:31 +00:00
}
2004-08-23 00:15:46 +00:00
return ;
}
2009-04-06 00:34:32 +00:00
# ifdef Q2CLIENT
if ( cls . protocol = = CP_QUAKE2 )
{
CL_AllowIndependantSendCmd ( false ) ; //stop it now, the indep stuff *could* require model tracing.
Hunk_Check ( ) ; // make sure nothing is hurt
cl . sendprespawn = true ;
}
else
# endif
{
if ( cls . demoplayback = = DPB_EZTV )
{
if ( CL_RemoveClientCommands ( " qtvmodellist " ) )
Con_Printf ( " Multiple modellists \n " ) ;
CL_SendClientCommand ( true , " qtvmodellist %i 0 " , cl . servercount ) ;
}
else
{
if ( CL_RemoveClientCommands ( " modellist " ) )
Con_Printf ( " Multiple modellists \n " ) ;
// CL_SendClientCommand ("modellist %i 0", cl.servercount);
CL_SendClientCommand ( true , modellist_name , cl . servercount , 0 ) ;
}
}
2004-08-23 00:15:46 +00:00
}
/*
= = = = = = = = = = = = = = = = = =
CL_ParseModellist
= = = = = = = = = = = = = = = = = =
*/
void CL_ParseModellist ( qboolean lots )
{
int nummodels ;
char * str ;
int n ;
// precache models and note certain default indexes
if ( lots )
nummodels = MSG_ReadShort ( ) ;
else
nummodels = MSG_ReadByte ( ) ;
for ( ; ; )
{
str = MSG_ReadString ( ) ;
if ( ! str [ 0 ] )
break ;
nummodels + + ;
2005-08-07 18:08:13 +00:00
if ( nummodels > = MAX_MODELS )
2004-08-23 00:15:46 +00:00
Host_EndGame ( " Server sent too many model_precache " ) ;
strcpy ( cl . model_name [ nummodels ] , str ) ;
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/spike.mdl " ) )
cl_spikeindex = nummodels ;
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/player.mdl " ) )
cl_playerindex = nummodels ;
2005-01-18 20:15:20 +00:00
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/h_player.mdl " ) )
cl_h_playerindex = nummodels ;
2004-08-23 00:15:46 +00:00
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/flag.mdl " ) )
cl_flagindex = nummodels ;
2005-01-17 17:40:21 +00:00
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/missile.mdl " ) )
cl_rocketindex = nummodels ;
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/grenade.mdl " ) )
cl_grenadeindex = nummodels ;
2005-01-18 20:15:20 +00:00
2005-09-26 03:40:09 +00:00
2005-01-18 20:15:20 +00:00
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/gib1.mdl " ) )
cl_gib1index = nummodels ;
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/gib2.mdl " ) )
cl_gib2index = nummodels ;
if ( ! strcmp ( cl . model_name [ nummodels ] , " progs/gib3.mdl " ) )
cl_gib3index = nummodels ;
2004-08-23 00:15:46 +00:00
}
if ( nummodels )
SCR_ImageName ( cl . model_name [ 1 ] ) ;
n = MSG_ReadByte ( ) ;
2005-02-28 07:16:19 +00:00
if ( n )
{
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback ! = DPB_EZTV )
{
if ( CL_RemoveClientCommands ( " modellist " ) )
Con_Printf ( " Multiple modellists \n " ) ;
// CL_SendClientCommand("modellist %i %i", cl.servercount, n);
CL_SendClientCommand ( true , modellist_name , cl . servercount , ( nummodels & 0xff00 ) + n ) ;
}
2004-08-23 00:15:46 +00:00
return ;
}
2009-05-24 10:11:17 +00:00
Sound_CheckDownloads ( ) ;
2009-04-06 00:34:32 +00:00
Model_CheckDownloads ( ) ;
CL_AllowIndependantSendCmd ( false ) ; //stop it now, the indep stuff *could* require model tracing.
Hunk_Check ( ) ; // make sure nothing is hurt
//set the flag to load models and send prespawn
cl . sendprespawn = true ;
2004-08-23 00:15:46 +00:00
}
void CL_ProcessUserInfo ( int slot , player_info_t * player ) ;
2006-05-28 21:56:04 +00:00
# ifdef Q2CLIENT
2004-08-23 00:15:46 +00:00
void CLQ2_ParseClientinfo ( int i , char * s )
{
2005-03-20 02:57:11 +00:00
char * model , * name ;
2004-08-23 00:15:46 +00:00
player_info_t * player ;
//s contains "name\model/skin"
player = & cl . players [ i ] ;
* player - > userinfo = ' \0 ' ;
model = strchr ( s , ' \\ ' ) ;
if ( model )
{
* model = ' \0 ' ;
model + + ;
name = s ;
}
else
{
name = " Unnammed " ;
model = " male " ;
}
2005-03-20 02:57:11 +00:00
#if 0
2004-08-23 00:15:46 +00:00
skin = strchr ( model , ' / ' ) ;
if ( skin )
{
* skin = ' \0 ' ;
skin + + ;
}
else
skin = " " ;
Info_SetValueForKey ( player - > userinfo , " model " , model , MAX_INFO_STRING ) ;
Info_SetValueForKey ( player - > userinfo , " skin " , skin , MAX_INFO_STRING ) ;
2005-03-20 02:57:11 +00:00
# else
Info_SetValueForKey ( player - > userinfo , " skin " , model , MAX_INFO_STRING ) ;
# endif
Info_SetValueForKey ( player - > userinfo , " name " , name , MAX_INFO_STRING ) ;
cl . players [ i ] . userid = i ;
2007-07-23 18:52:11 +00:00
cl . players [ i ] . rbottomcolor = 1 ;
cl . players [ i ] . rtopcolor = 1 ;
2004-08-23 00:15:46 +00:00
CL_ProcessUserInfo ( i , player ) ;
}
void CLQ2_ParseConfigString ( void )
{
int i ;
char * s ;
// char olds[MAX_QPATH];
i = MSG_ReadShort ( ) ;
if ( i < 0 | | i > = Q2MAX_CONFIGSTRINGS )
Host_EndGame ( " configstring > Q2MAX_CONFIGSTRINGS " ) ;
s = MSG_ReadString ( ) ;
// strncpy (olds, cl.configstrings[i], sizeof(olds));
// olds[sizeof(olds) - 1] = 0;
// strcpy (cl.configstrings[i], s);
2005-09-26 03:40:09 +00:00
// do something apropriate
2004-08-23 00:15:46 +00:00
if ( i = = Q2CS_SKY )
{
Q_strncpyz ( cl . skyname , s , sizeof ( cl . skyname ) ) ;
}
2005-10-16 03:53:31 +00:00
else if ( i = = Q2CS_SKYAXIS )
{
s = COM_Parse ( s ) ;
2006-01-09 01:49:06 +00:00
if ( s )
{
cl . skyaxis [ 0 ] = atof ( com_token ) ;
s = COM_Parse ( s ) ;
if ( s )
{
cl . skyaxis [ 1 ] = atof ( com_token ) ;
s = COM_Parse ( s ) ;
if ( s )
cl . skyaxis [ 2 ] = atof ( com_token ) ;
}
}
2005-10-16 03:53:31 +00:00
}
else if ( i = = Q2CS_SKYROTATE )
cl . skyrotate = atof ( s ) ;
2004-08-23 00:15:46 +00:00
else if ( i = = Q2CS_STATUSBAR )
{
Q_strncpyz ( cl . q2statusbar , s , sizeof ( cl . q2statusbar ) ) ;
}
else if ( i > = Q2CS_LIGHTS & & i < Q2CS_LIGHTS + Q2MAX_LIGHTSTYLES )
{
cl_lightstyle [ i - Q2CS_LIGHTS ] . colour = 7 ; //white
Q_strncpyz ( cl_lightstyle [ i - Q2CS_LIGHTS ] . map , s , sizeof ( cl_lightstyle [ i - Q2CS_LIGHTS ] . map ) ) ;
cl_lightstyle [ i - Q2CS_LIGHTS ] . length = Q_strlen ( cl_lightstyle [ i - Q2CS_LIGHTS ] . map ) ;
}
else if ( i = = Q2CS_CDTRACK )
{
// if (cl.refresh_prepped)
CDAudio_Play ( atoi ( s ) , true ) ;
}
else if ( i > = Q2CS_MODELS & & i < Q2CS_MODELS + Q2MAX_MODELS )
{
// if (cl.refresh_prepped)
{
Q_strncpyz ( cl . model_name [ i - Q2CS_MODELS ] , s , MAX_QPATH ) ;
2006-04-11 20:09:39 +00:00
if ( cl . model_name [ i - Q2CS_MODELS ] [ 0 ] = = ' # ' )
{
if ( cl . numq2visibleweapons < Q2MAX_VISIBLE_WEAPONS )
{
cl . q2visibleweapons [ cl . numq2visibleweapons ] = cl . model_name [ i - Q2CS_MODELS ] + 1 ;
cl . numq2visibleweapons + + ;
}
cl . model_precache [ i - Q2CS_MODELS ] = NULL ;
}
else
cl . model_precache [ i - Q2CS_MODELS ] = Mod_ForName ( cl . model_name [ i - Q2CS_MODELS ] , false ) ;
2004-08-23 00:15:46 +00:00
}
}
else if ( i > = Q2CS_SOUNDS & & i < Q2CS_SOUNDS + Q2MAX_MODELS )
{
// if (cl.refresh_prepped)
Q_strncpyz ( cl . sound_name [ i - Q2CS_SOUNDS ] , s , MAX_QPATH ) ;
cl . sound_precache [ i - Q2CS_SOUNDS ] = S_PrecacheSound ( s ) ;
}
else if ( i > = Q2CS_IMAGES & & i < Q2CS_IMAGES + Q2MAX_MODELS )
{ //ignore
Q_strncpyz ( cl . image_name [ i - Q2CS_IMAGES ] , s , MAX_QPATH ) ;
}
else if ( i > = Q2CS_PLAYERSKINS & & i < Q2CS_PLAYERSKINS + Q2MAX_CLIENTS )
{
// if (cl.refresh_prepped && strcmp(olds, s))
CLQ2_ParseClientinfo ( i - Q2CS_PLAYERSKINS , s ) ;
}
2006-04-08 05:43:53 +00:00
else if ( i = = Q2CS_MAPCHECKSUM )
{
extern int map_checksum ;
int serverchecksum = atoi ( s ) ;
if ( cl . worldmodel & & ( cl . worldmodel - > fromgame = = fg_quake2 | | cl . worldmodel - > fromgame = = fg_quake3 ) )
{
// the Q2 client normally exits here, however for our purposes we might as well ignore it
if ( map_checksum ! = serverchecksum )
2007-09-23 15:28:06 +00:00
Con_Printf ( CON_WARNING " WARNING: Client checksum does not match server checksum (%i != %i) " , map_checksum , serverchecksum ) ;
2006-04-08 05:43:53 +00:00
}
}
2004-08-23 00:15:46 +00:00
2005-09-09 23:40:55 +00:00
# ifdef VM_UI
2004-08-23 00:15:46 +00:00
UI_StringChanged ( i ) ;
2005-09-09 23:40:55 +00:00
# endif
2004-08-23 00:15:46 +00:00
}
2006-05-28 21:56:04 +00:00
# endif
2004-08-23 00:15:46 +00:00
2006-07-24 04:24:41 +00:00
qboolean CL_CheckBaselines ( int size )
{
int i ;
if ( size < 0 )
return false ;
if ( size > MAX_EDICTS )
return false ;
size = ( size + 64 ) & ~ 63 ; // round up to next 64
2009-03-03 01:52:30 +00:00
if ( size < = cl_baselines_count )
2006-07-24 04:24:41 +00:00
return true ;
2008-05-09 14:22:37 +00:00
cl_baselines = BZ_Realloc ( cl_baselines , sizeof ( * cl_baselines ) * size ) ;
2006-07-24 04:24:41 +00:00
for ( i = cl_baselines_count ; i < size ; i + + )
{
memcpy ( cl_baselines + i , & nullentitystate , sizeof ( * cl_baselines ) ) ;
}
cl_baselines_count = size ;
return true ;
}
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = = = = =
CL_ParseBaseline
= = = = = = = = = = = = = = = = = =
*/
void CL_ParseBaseline ( entity_state_t * es )
{
int i ;
2006-02-27 00:42:25 +00:00
memcpy ( es , & nullentitystate , sizeof ( entity_state_t ) ) ;
2005-09-26 03:40:09 +00:00
2006-02-27 00:42:25 +00:00
es - > modelindex = MSG_ReadByte ( ) ;
2004-08-23 00:15:46 +00:00
es - > frame = MSG_ReadByte ( ) ;
es - > colormap = MSG_ReadByte ( ) ;
es - > skinnum = MSG_ReadByte ( ) ;
for ( i = 0 ; i < 3 ; i + + )
{
es - > origin [ i ] = MSG_ReadCoord ( ) ;
es - > angles [ i ] = MSG_ReadAngle ( ) ;
}
}
void CL_ParseBaseline2 ( void )
{
2006-02-27 00:42:25 +00:00
entity_state_t es ;
2004-08-23 00:15:46 +00:00
2006-02-27 00:42:25 +00:00
CL_ParseDelta ( & nullentitystate , & es , MSG_ReadShort ( ) , true ) ;
2006-07-24 04:24:41 +00:00
if ( ! CL_CheckBaselines ( es . number ) )
Host_EndGame ( " CL_ParseBaseline2: check baselines failed with size %i " , es . number ) ;
memcpy ( cl_baselines + es . number , & es , sizeof ( es ) ) ;
2004-08-23 00:15:46 +00:00
}
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
void CLFitz_ParseBaseline2 ( entity_state_t * es )
{
int i ;
int bits ;
memcpy ( es , & nullentitystate , sizeof ( entity_state_t ) ) ;
bits = MSG_ReadByte ( ) ;
es - > modelindex = ( bits & FITZB_LARGEMODEL ) ? MSG_ReadShort ( ) : MSG_ReadByte ( ) ;
es - > frame = ( bits & FITZB_LARGEFRAME ) ? MSG_ReadShort ( ) : MSG_ReadByte ( ) ;
es - > colormap = MSG_ReadByte ( ) ;
es - > skinnum = MSG_ReadByte ( ) ;
for ( i = 0 ; i < 3 ; i + + )
{
es - > origin [ i ] = MSG_ReadCoord ( ) ;
es - > angles [ i ] = MSG_ReadAngle ( ) ;
}
es - > trans = ( bits & FITZB_ALPHA ) ? MSG_ReadByte ( ) : 255 ;
}
2004-08-23 00:15:46 +00:00
void CLQ2_Precache_f ( void )
{
2009-04-06 00:34:32 +00:00
Model_CheckDownloads ( ) ;
Sound_CheckDownloads ( ) ;
cl . contentstage = 0 ;
2006-01-06 23:58:32 +00:00
cl . sendprespawn = true ;
2009-04-06 00:34:32 +00:00
2004-08-23 00:15:46 +00:00
# ifdef VM_CG
CG_Start ( ) ;
# endif
}
/*
= = = = = = = = = = = = = = = = = = = = =
CL_ParseStatic
Static entities are non - interactive world objects
like torches
= = = = = = = = = = = = = = = = = = = = =
*/
void CL_ParseStatic ( int version )
{
entity_t * ent ;
int i ;
2006-02-27 00:42:25 +00:00
entity_state_t es ;
2010-07-11 02:22:39 +00:00
vec3_t mins , maxs ;
2004-08-23 00:15:46 +00:00
if ( version = = 1 )
{
CL_ParseBaseline ( & es ) ;
i = cl . num_statics ;
cl . num_statics + + ;
}
else
{
2006-02-27 00:42:25 +00:00
CL_ParseDelta ( & nullentitystate , & es , MSG_ReadShort ( ) , true ) ;
2004-08-23 00:15:46 +00:00
es . number + = MAX_EDICTS ;
for ( i = 0 ; i < cl . num_statics ; i + + )
2010-07-11 02:22:39 +00:00
if ( cl_static_entities [ i ] . ent . keynum = = es . number )
2004-08-23 00:15:46 +00:00
{
2010-07-11 02:22:39 +00:00
pe - > DelinkTrailstate ( & cl_static_entities [ i ] . emit ) ;
2004-08-23 00:15:46 +00:00
break ;
}
if ( i = = cl . num_statics )
cl . num_statics + + ;
}
2005-09-26 03:40:09 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( i = = cl_max_static_entities )
2004-08-23 00:15:46 +00:00
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cl_max_static_entities + = 16 ;
cl_static_entities = BZ_Realloc ( cl_static_entities , sizeof ( * cl_static_entities ) * cl_max_static_entities ) ;
2004-08-23 00:15:46 +00:00
}
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
cl_static_entities [ i ] . mdlidx = es . modelindex ;
cl_static_entities [ i ] . emit = NULL ;
2010-07-11 02:22:39 +00:00
ent = & cl_static_entities [ i ] . ent ;
2006-03-06 21:11:50 +00:00
memset ( ent , 0 , sizeof ( * ent ) ) ;
2004-08-23 00:15:46 +00:00
ent - > keynum = es . number ;
// copy it to the current state
ent - > model = cl . model_precache [ es . modelindex ] ;
2008-12-23 02:55:20 +00:00
ent - > framestate . g [ FS_REG ] . frame [ 0 ] = ent - > framestate . g [ FS_REG ] . frame [ 1 ] = es . frame ;
2004-08-23 00:15:46 +00:00
ent - > skinnum = es . skinnum ;
2005-05-15 18:49:04 +00:00
ent - > drawflags = es . hexen2flags ;
2004-08-23 00:15:46 +00:00
# ifdef PEXT_SCALE
2005-07-01 19:23:00 +00:00
ent - > scale = es . scale / 16.0 ;
2004-08-23 00:15:46 +00:00
# endif
2006-02-27 00:42:25 +00:00
ent - > shaderRGBAf [ 0 ] = ( 8.0f / 255.0f ) * es . colormod [ 0 ] ;
ent - > shaderRGBAf [ 1 ] = ( 8.0f / 255.0f ) * es . colormod [ 1 ] ;
ent - > shaderRGBAf [ 2 ] = ( 8.0f / 255.0f ) * es . colormod [ 2 ] ;
ent - > shaderRGBAf [ 3 ] = es . trans / 255 ;
2009-07-05 18:45:53 +00:00
ent - > fatness = es . fatness / 16.0 ;
2004-08-23 00:15:46 +00:00
ent - > abslight = es . abslight ;
VectorCopy ( es . origin , ent - > origin ) ;
VectorCopy ( es . angles , ent - > angles ) ;
2004-11-17 17:58:22 +00:00
es . angles [ 0 ] * = - 1 ;
AngleVectors ( es . angles , ent - > axis [ 0 ] , ent - > axis [ 1 ] , ent - > axis [ 2 ] ) ;
2004-11-22 00:00:54 +00:00
VectorInverse ( ent - > axis [ 1 ] ) ;
2004-08-23 00:15:46 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( ! cl . worldmodel | | cl . worldmodel - > needload )
2004-08-23 00:15:46 +00:00
{
Con_TPrintf ( TLC_PARSESTATICWITHNOMAP ) ;
return ;
}
2010-07-11 02:22:39 +00:00
if ( ent - > model )
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
/*FIXME: compensate for angle*/
2010-07-11 02:22:39 +00:00
VectorAdd ( es . origin , ent - > model - > mins , mins ) ;
VectorAdd ( es . origin , ent - > model - > maxs , maxs ) ;
cl . worldmodel - > funcs . FindTouchedLeafs ( cl . worldmodel , & cl_static_entities [ i ] . pvscache , mins , maxs ) ;
}
2004-08-23 00:15:46 +00:00
}
/*
= = = = = = = = = = = = = = = = = = =
CL_ParseStaticSound
= = = = = = = = = = = = = = = = = = =
*/
void CL_ParseStaticSound ( void )
{
2005-01-13 16:29:20 +00:00
extern cvar_t cl_staticsounds ;
2004-08-23 00:15:46 +00:00
vec3_t org ;
int sound_num , vol , atten ;
int i ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
for ( i = 0 ; i < 3 ; i + + )
org [ i ] = MSG_ReadCoord ( ) ;
sound_num = MSG_ReadByte ( ) ;
vol = MSG_ReadByte ( ) ;
atten = MSG_ReadByte ( ) ;
2005-01-13 16:29:20 +00:00
2009-11-04 21:16:50 +00:00
vol * = cl_staticsounds . value ;
if ( vol < 0 )
2005-01-13 16:29:20 +00:00
return ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
S_StaticSound ( cl . sound_precache [ sound_num ] , org , vol , atten ) ;
}
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
ACTION MESSAGES
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
/*
= = = = = = = = = = = = = = = = = =
CL_ParseStartSoundPacket
= = = = = = = = = = = = = = = = = =
*/
void CL_ParseStartSoundPacket ( void )
{
vec3_t pos ;
int channel , ent ;
int sound_num ;
int volume ;
2005-09-26 03:40:09 +00:00
float attenuation ;
2004-08-23 00:15:46 +00:00
int i ;
2005-09-26 03:40:09 +00:00
channel = MSG_ReadShort ( ) ;
2004-08-23 00:15:46 +00:00
if ( channel & SND_VOLUME )
volume = MSG_ReadByte ( ) ;
else
volume = DEFAULT_SOUND_PACKET_VOLUME ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
if ( channel & SND_ATTENUATION )
attenuation = MSG_ReadByte ( ) / 64.0 ;
else
attenuation = DEFAULT_SOUND_PACKET_ATTENUATION ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
sound_num = MSG_ReadByte ( ) ;
for ( i = 0 ; i < 3 ; i + + )
pos [ i ] = MSG_ReadCoord ( ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
ent = ( channel > > 3 ) & 1023 ;
channel & = 7 ;
if ( ent > MAX_EDICTS )
Host_EndGame ( " CL_ParseStartSoundPacket: ent = %i " , ent ) ;
2005-09-26 03:40:09 +00:00
2005-08-03 23:14:59 +00:00
# ifdef PEXT_CSQC
2005-08-11 04:14:33 +00:00
if ( ! CSQC_StartSound ( ent , channel , cl . sound_name [ sound_num ] , pos , volume / 255.0 , attenuation ) )
2005-08-03 23:14:59 +00:00
# endif
2010-08-21 13:31:39 +00:00
{
if ( ! sound_num )
S_StopSound ( ent , channel ) ;
else
S_StartSound ( ent , channel , cl . sound_precache [ sound_num ] , pos , volume / 255.0 , attenuation ) ;
}
2004-08-23 00:15:46 +00:00
2005-06-04 04:20:20 +00:00
if ( ent = = cl . playernum [ 0 ] + 1 )
TP_CheckPickupSound ( cl . sound_name [ sound_num ] , pos ) ;
2004-08-23 00:15:46 +00:00
}
# ifdef Q2CLIENT
void CLQ2_ParseStartSoundPacket ( void )
{
vec3_t pos_v ;
float * pos ;
int channel , ent ;
int sound_num ;
float volume ;
2005-09-26 03:40:09 +00:00
float attenuation ;
2004-08-23 00:15:46 +00:00
int flags ;
float ofs ;
flags = MSG_ReadByte ( ) ;
sound_num = MSG_ReadByte ( ) ;
if ( flags & Q2SND_VOLUME )
volume = MSG_ReadByte ( ) / 255.0 ;
else
volume = Q2DEFAULT_SOUND_PACKET_VOLUME ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
if ( flags & Q2SND_ATTENUATION )
attenuation = MSG_ReadByte ( ) / 64.0 ;
else
2005-09-26 03:40:09 +00:00
attenuation = Q2DEFAULT_SOUND_PACKET_ATTENUATION ;
2004-08-23 00:15:46 +00:00
if ( flags & Q2SND_OFFSET )
ofs = MSG_ReadByte ( ) / 1000.0 ;
else
ofs = 0 ;
if ( flags & Q2SND_ENT )
{ // entity reletive
2005-09-26 03:40:09 +00:00
channel = MSG_ReadShort ( ) ;
2004-08-23 00:15:46 +00:00
ent = channel > > 3 ;
if ( ent > MAX_EDICTS )
Host_EndGame ( " CL_ParseStartSoundPacket: ent = %i " , ent ) ;
channel & = 7 ;
}
else
{
ent = 0 ;
channel = 0 ;
}
if ( flags & Q2SND_POS )
{ // positioned in space
MSG_ReadPos ( pos_v ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
pos = pos_v ;
}
else // use entity number
{
CL_GetNumberedEntityInfo ( ent , pos_v , NULL ) ;
pos = pos_v ;
// pos = NULL;
}
if ( ! cl . sound_precache [ sound_num ] )
return ;
2006-04-11 20:09:39 +00:00
if ( cl . sound_precache [ sound_num ] - > name [ 0 ] = = ' * ' & & ent > 0 & & ent < = MAX_CLIENTS )
{ //a 'sexed' sound
char * model = Info_ValueForKey ( cl . players [ ent - 1 ] . userinfo , " skin " ) ;
char * skin ;
skin = strchr ( model , ' / ' ) ;
if ( skin )
* skin = ' \0 ' ;
if ( * model )
{
S_StartSound ( ent , channel , S_PrecacheSound ( va ( " players/%s/%s " , model , cl . sound_precache [ sound_num ] - > name + 1 ) ) , pos , volume , attenuation ) ;
return ;
}
}
2004-08-23 00:15:46 +00:00
S_StartSound ( ent , channel , cl . sound_precache [ sound_num ] , pos , volume , attenuation ) ;
}
# endif
2008-12-01 02:56:32 +00:00
# if defined(NQPROT) || defined(PEXT_SOUNDDBL)
2004-08-23 00:15:46 +00:00
void CLNQ_ParseStartSoundPacket ( void )
{
vec3_t pos ;
int channel , ent ;
int sound_num ;
int volume ;
int field_mask ;
2005-09-26 03:40:09 +00:00
float attenuation ;
2004-08-23 00:15:46 +00:00
int i ;
2005-09-26 03:40:09 +00:00
field_mask = MSG_ReadByte ( ) ;
2004-08-23 00:15:46 +00:00
if ( field_mask & NQSND_VOLUME )
volume = MSG_ReadByte ( ) ;
else
volume = DEFAULT_SOUND_PACKET_VOLUME ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
if ( field_mask & NQSND_ATTENUATION )
attenuation = MSG_ReadByte ( ) / 64.0 ;
else
attenuation = DEFAULT_SOUND_PACKET_ATTENUATION ;
2005-09-26 03:40:09 +00:00
2008-11-09 22:29:28 +00:00
if ( field_mask & DPSND_LARGEENTITY )
{
ent = ( unsigned short ) MSG_ReadShort ( ) ;
channel = MSG_ReadByte ( ) ;
}
else
{ //regular
channel = MSG_ReadShort ( ) ;
ent = channel > > 3 ;
channel & = 7 ;
}
2004-08-23 00:15:46 +00:00
2008-11-09 22:29:28 +00:00
if ( field_mask & DPSND_LARGESOUND )
sound_num = ( unsigned short ) MSG_ReadShort ( ) ;
else
sound_num = MSG_ReadByte ( ) ;
2004-08-23 00:15:46 +00:00
if ( ent > MAX_EDICTS )
Host_EndGame ( " CL_ParseStartSoundPacket: ent = %i " , ent ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
for ( i = 0 ; i < 3 ; i + + )
pos [ i ] = MSG_ReadCoord ( ) ;
2005-09-26 03:40:09 +00:00
2010-08-21 13:31:39 +00:00
# ifdef PEXT_CSQC
if ( ! CSQC_StartSound ( ent , channel , cl . sound_name [ sound_num ] , pos , volume / 255.0 , attenuation ) )
# endif
{
if ( ! sound_num )
S_StopSound ( ent , channel ) ;
else
S_StartSound ( ent , channel , cl . sound_precache [ sound_num ] , pos , volume / 255.0 , attenuation ) ;
}
2005-06-04 04:20:20 +00:00
if ( ent = = cl . playernum [ 0 ] + 1 )
TP_CheckPickupSound ( cl . sound_name [ sound_num ] , pos ) ;
2005-09-26 03:40:09 +00:00
}
2004-08-23 00:15:46 +00:00
# endif
/*
= = = = = = = = = = = = = = = = = =
CL_ParseClientdata
Server information pertaining to this client only , sent every frame
= = = = = = = = = = = = = = = = = =
*/
void CL_ParseClientdata ( void )
{
int i ;
float latency ;
frame_t * frame ;
// calculate simulated time of message
oldparsecountmod = parsecountmod ;
i = cls . netchan . incoming_acknowledged ;
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
2004-08-27 00:48:03 +00:00
cl . oldparsecount = i - 1 ;
2004-08-23 00:15:46 +00:00
cl . parsecount = i ;
i & = UPDATE_MASK ;
parsecountmod = i ;
frame = & cl . frames [ i ] ;
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
2004-08-27 00:48:03 +00:00
frame - > senttime = realtime - host_frametime ;
2004-08-23 00:15:46 +00:00
parsecounttime = cl . frames [ i ] . senttime ;
2005-10-01 03:09:17 +00:00
frame - > receivedtime = ( cl . gametimemark - cl . oldgametimemark ) * 20 ;
2004-08-23 00:15:46 +00:00
// calculate latency
latency = frame - > receivedtime - frame - > senttime ;
if ( latency < 0 | | latency > 1.0 )
{
// Con_Printf ("Odd latency: %5.2f\n", latency);
}
else
{
// drift the average latency towards the observed latency
if ( latency < cls . latency )
cls . latency = latency ;
else
cls . latency + = 0.001 ; // drift up, so correction are needed
2005-09-26 03:40:09 +00:00
}
2004-08-23 00:15:46 +00:00
}
/*
= = = = = = = = = = = = = = = = = = = = =
CL_NewTranslation
= = = = = = = = = = = = = = = = = = = = =
*/
void CL_NewTranslation ( int slot )
{
int top , bottom ;
2005-11-30 01:20:53 +00:00
int local ;
2004-08-23 00:15:46 +00:00
2004-12-29 03:24:21 +00:00
char * s ;
player_info_t * player ;
2005-01-23 17:47:04 +00:00
if ( slot > = MAX_CLIENTS )
2006-03-12 22:01:49 +00:00
Host_Error ( " CL_NewTranslation: slot > MAX_CLIENTS " ) ;
2004-12-29 03:24:21 +00:00
player = & cl . players [ slot ] ;
s = Skin_FindName ( player ) ;
2006-03-11 03:12:10 +00:00
COM_StripExtension ( s , s , MAX_QPATH ) ;
2004-12-29 03:24:21 +00:00
if ( player - > skin & & ! stricmp ( s , player - > skin - > name ) )
player - > skin = NULL ;
2007-07-23 18:52:11 +00:00
top = player - > rtopcolor ;
bottom = player - > rbottomcolor ;
if ( cl . splitclients < 2 & & ! ( cl . fpd & FPD_NO_FORCE_COLOR ) ) //no colour/skin forcing in splitscreen.
2004-08-23 00:15:46 +00:00
{
2007-07-23 18:52:11 +00:00
if ( cl . teamplay & & cl . spectator )
2004-09-26 00:30:42 +00:00
{
2007-07-23 18:52:11 +00:00
local = Cam_TrackNum ( 0 ) ;
if ( local < 0 )
2005-06-14 04:52:10 +00:00
local = cl . playernum [ 0 ] ;
2004-09-26 00:30:42 +00:00
}
2007-07-23 18:52:11 +00:00
else
local = cl . playernum [ 0 ] ;
2009-04-01 22:03:56 +00:00
if ( ( cl . teamplay | | cls . protocol = = CP_NETQUAKE ) & & ! strcmp ( player - > team , cl . players [ local ] . team ) )
2007-07-23 18:52:11 +00:00
{
2008-06-01 22:06:22 +00:00
if ( cl_teamtopcolor ! = ~ 0 )
2007-07-23 18:52:11 +00:00
top = cl_teamtopcolor ;
2008-06-01 22:06:22 +00:00
if ( cl_teambottomcolor ! = ~ 0 )
2007-07-23 18:52:11 +00:00
bottom = cl_teambottomcolor ;
2008-06-05 07:45:34 +00:00
if ( player - > colourised )
{
if ( player - > colourised - > topcolour ! = ~ 0 )
top = player - > colourised - > topcolour ;
if ( player - > colourised - > bottomcolour ! = ~ 0 )
bottom = player - > colourised - > bottomcolour ;
}
2007-07-23 18:52:11 +00:00
}
else
{
2008-06-01 22:06:22 +00:00
if ( cl_enemytopcolor ! = ~ 0 )
2007-07-23 18:52:11 +00:00
top = cl_enemytopcolor ;
2008-06-01 22:06:22 +00:00
if ( cl_enemybottomcolor ! = ~ 0 )
2007-07-23 18:52:11 +00:00
bottom = cl_enemybottomcolor ;
}
}
2008-06-01 22:06:22 +00:00
/*
2007-07-23 18:52:11 +00:00
if ( top > 13 | | top < 0 )
top = 13 ;
if ( bottom > 13 | | bottom < 0 )
bottom = 13 ;
2008-06-01 22:06:22 +00:00
*/
2007-07-23 18:52:11 +00:00
//other renderers still need the team stuff set, but that's all
player - > ttopcolor = top ;
player - > tbottomcolor = bottom ;
2004-08-23 00:15:46 +00:00
}
/*
= = = = = = = = = = = = = =
CL_UpdateUserinfo
= = = = = = = = = = = = = =
*/
void CL_ProcessUserInfo ( int slot , player_info_t * player )
{
2008-06-01 22:06:22 +00:00
char * col ;
2004-08-23 00:15:46 +00:00
Q_strncpyz ( player - > name , Info_ValueForKey ( player - > userinfo , " name " ) , sizeof ( player - > name ) ) ;
Q_strncpyz ( player - > team , Info_ValueForKey ( player - > userinfo , " team " ) , sizeof ( player - > team ) ) ;
2008-06-01 22:06:22 +00:00
col = Info_ValueForKey ( player - > userinfo , " topcolor " ) ;
if ( ! strncmp ( col , " 0x " , 2 ) )
player - > rtopcolor = 0xff000000 | strtoul ( col + 2 , NULL , 16 ) ;
else
player - > rtopcolor = atoi ( col ) ;
col = Info_ValueForKey ( player - > userinfo , " bottomcolor " ) ;
if ( ! strncmp ( col , " 0x " , 2 ) )
player - > rbottomcolor = 0xff000000 | strtoul ( col + 2 , NULL , 16 ) ;
else
player - > rbottomcolor = atoi ( col ) ;
2005-01-17 17:40:21 +00:00
if ( atoi ( Info_ValueForKey ( player - > userinfo , " *spectator " ) ) )
2004-08-23 00:15:46 +00:00
player - > spectator = true ;
else
player - > spectator = false ;
2008-06-01 22:06:22 +00:00
/*
if ( player - > rtopcolor > 13 )
player - > rtopcolor = 13 ;
if ( player - > rbottomcolor > 13 )
player - > rbottomcolor = 13 ;
*/
2008-01-28 23:03:42 +00:00
player - > model = NULL ;
2008-06-05 07:45:34 +00:00
player - > colourised = TP_FindColours ( player - > name ) ;
2007-11-23 18:27:30 +00:00
// If it's us
2004-08-23 00:15:46 +00:00
if ( slot = = cl . playernum [ 0 ] & & player - > name [ 0 ] )
2007-11-23 18:27:30 +00:00
{
2004-08-23 00:15:46 +00:00
cl . spectator = player - > spectator ;
2007-11-23 18:27:30 +00:00
// Update the rules since spectators can bypass everything but players can't
CL_CheckServerInfo ( ) ;
2008-01-28 23:03:42 +00:00
Skin_FlushPlayers ( ) ;
}
else if ( cls . state = = ca_active )
2004-08-23 00:15:46 +00:00
Skin_Find ( player ) ;
Sbar_Changed ( ) ;
CL_NewTranslation ( slot ) ;
}
/*
= = = = = = = = = = = = = =
CL_UpdateUserinfo
= = = = = = = = = = = = = =
*/
void CL_UpdateUserinfo ( void )
{
int slot ;
player_info_t * player ;
slot = MSG_ReadByte ( ) ;
if ( slot > = MAX_CLIENTS )
Host_EndGame ( " CL_ParseServerMessage: svc_updateuserinfo > MAX_SCOREBOARD " ) ;
player = & cl . players [ slot ] ;
player - > userid = MSG_ReadLong ( ) ;
Q_strncpyz ( player - > userinfo , MSG_ReadString ( ) , sizeof ( player - > userinfo ) ) ;
CL_ProcessUserInfo ( slot , player ) ;
2006-02-22 23:35:04 +00:00
if ( slot = = cl . playernum [ 0 ] & & player - > name [ 0 ] )
{
char * qz ;
qz = Info_ValueForKey ( player - > userinfo , " Qizmo " ) ;
if ( * qz )
TP_ExecTrigger ( " f_qizmoconnect " ) ;
}
2004-08-23 00:15:46 +00:00
}
/*
= = = = = = = = = = = = = =
CL_SetInfo
= = = = = = = = = = = = = =
*/
2007-02-23 00:21:33 +00:00
void CL_ParseSetInfo ( void )
2004-08-23 00:15:46 +00:00
{
int slot ;
player_info_t * player ;
char key [ MAX_QWMSGLEN ] ;
char value [ MAX_QWMSGLEN ] ;
slot = MSG_ReadByte ( ) ;
if ( slot > = MAX_CLIENTS )
Host_EndGame ( " CL_ParseServerMessage: svc_setinfo > MAX_SCOREBOARD " ) ;
player = & cl . players [ slot ] ;
Q_strncpyz ( key , MSG_ReadString ( ) , sizeof ( key ) ) ;
Q_strncpyz ( value , MSG_ReadString ( ) , sizeof ( value ) ) ;
Con_DPrintf ( " SETINFO %s: %s=%s \n " , player - > name , key , value ) ;
2005-03-07 08:58:26 +00:00
Info_SetValueForStarKey ( player - > userinfo , key , value , sizeof ( player - > userinfo ) ) ;
2004-08-23 00:15:46 +00:00
CL_ProcessUserInfo ( slot , player ) ;
}
/*
= = = = = = = = = = = = = =
CL_ServerInfo
= = = = = = = = = = = = = =
*/
void CL_ServerInfo ( void )
{
// int slot;
// player_info_t *player;
char key [ MAX_QWMSGLEN ] ;
char value [ MAX_QWMSGLEN ] ;
Q_strncpyz ( key , MSG_ReadString ( ) , sizeof ( key ) ) ;
Q_strncpyz ( value , MSG_ReadString ( ) , sizeof ( value ) ) ;
Con_DPrintf ( " SERVERINFO: %s=%s \n " , key , value ) ;
Info_SetValueForKey ( cl . serverinfo , key , value , MAX_SERVERINFO_STRING ) ;
CL_CheckServerInfo ( ) ;
}
/*
= = = = = = = = = = = = = = = = = = = = =
CL_SetStat
= = = = = = = = = = = = = = = = = = = = =
*/
2008-01-09 00:52:31 +00:00
static void CL_SetStat_Internal ( int pnum , int stat , int value )
2004-08-23 00:15:46 +00:00
{
int j ;
2004-12-05 08:19:54 +00:00
if ( cl . stats [ pnum ] [ stat ] ! = value )
Sbar_Changed ( ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
if ( stat = = STAT_ITEMS )
{ // set flash times
for ( j = 0 ; j < 32 ; j + + )
if ( ( value & ( 1 < < j ) ) & & ! ( cl . stats [ pnum ] [ stat ] & ( 1 < < j ) ) )
cl . item_gettime [ pnum ] [ j ] = cl . time ;
}
if ( stat = = STAT_VIEWHEIGHT & & cls . z_ext & Z_EXT_VIEWHEIGHT )
cl . viewheight [ pnum ] = value ;
2005-07-14 01:57:34 +00:00
if ( stat = = STAT_WEAPON )
{
if ( cl . stats [ pnum ] [ stat ] ! = value )
{
if ( value = = 0 )
TP_ExecTrigger ( " f_reloadstart " ) ;
else if ( cl . stats [ pnum ] [ stat ] = = 0 )
TP_ExecTrigger ( " f_reloadend " ) ;
}
}
2004-08-23 00:15:46 +00:00
cl . stats [ pnum ] [ stat ] = value ;
2004-09-30 22:42:34 +00:00
if ( pnum = = 0 )
TP_StatChanged ( stat , value ) ;
2004-08-23 00:15:46 +00:00
}
2008-05-25 22:23:43 +00:00
void CL_SetStatInt ( int pnum , int stat , int value )
2008-01-09 00:52:31 +00:00
{
if ( stat < 0 | | stat > = MAX_CL_STATS )
return ;
// Host_EndGame ("CL_SetStat: %i is invalid", stat);
if ( stat = = STAT_TIME & & ( cls . fteprotocolextensions & PEXT_ACCURATETIMINGS ) )
{
cl . oldgametime = cl . gametime ;
cl . oldgametimemark = cl . gametimemark ;
cl . gametime = value * 0.001 ;
cl . gametimemark = realtime ;
}
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
{
extern int cls_lastto ;
cl . players [ cls_lastto ] . stats [ stat ] = value ;
for ( pnum = 0 ; pnum < cl . splitclients ; pnum + + )
if ( spec_track [ pnum ] = = cls_lastto )
CL_SetStat_Internal ( pnum , stat , value ) ;
}
else
CL_SetStat_Internal ( pnum , stat , value ) ;
}
2008-05-25 22:23:43 +00:00
void CL_SetStatFloat ( int pnum , int stat , float value )
{
if ( stat < 0 | | stat > = MAX_CL_STATS )
return ;
// Host_EndGame ("CL_SetStat: %i is invalid", stat);
2008-01-09 00:52:31 +00:00
2008-05-25 22:23:43 +00:00
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
{
extern int cls_lastto ;
cl . players [ cls_lastto ] . statsf [ stat ] = value ;
for ( pnum = 0 ; pnum < cl . splitclients ; pnum + + )
if ( spec_track [ pnum ] = = cls_lastto )
cl . statsf [ pnum ] [ stat ] = value ;
}
else
cl . statsf [ pnum ] [ stat ] = value ;
}
void CL_SetStatString ( int pnum , int stat , char * value )
{
if ( stat < 0 | | stat > = MAX_CL_STATS )
return ;
// Host_EndGame ("CL_SetStat: %i is invalid", stat);
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
{
/* extern int cls_lastto;
cl . players [ cls_lastto ] . statsstr [ stat ] = value ;
for ( pnum = 0 ; pnum < cl . splitclients ; pnum + + )
if ( spec_track [ pnum ] = = cls_lastto )
cl . statsstr [ pnum ] [ stat ] = value ; */
}
else
{
if ( cl . statsstr [ pnum ] [ stat ] )
Z_Free ( cl . statsstr [ pnum ] [ stat ] ) ;
2008-05-31 10:33:03 +00:00
cl . statsstr [ pnum ] [ stat ] = Z_Malloc ( strlen ( value ) + 1 ) ;
strcpy ( cl . statsstr [ pnum ] [ stat ] , value ) ;
2008-05-25 22:23:43 +00:00
}
}
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = =
CL_MuzzleFlash
= = = = = = = = = = = = = =
*/
2005-06-04 12:40:12 +00:00
void CL_MuzzleFlash ( int destsplit )
2004-08-23 00:15:46 +00:00
{
vec3_t fv , rv , uv ;
dlight_t * dl = NULL ;
int i ;
player_state_t * pl ;
packet_entities_t * pack ;
entity_state_t * s1 ;
int pnum ;
2005-01-13 16:29:20 +00:00
extern cvar_t cl_muzzleflash ;
2004-08-23 00:15:46 +00:00
i = MSG_ReadShort ( ) ;
2005-01-13 16:29:20 +00:00
//was it us?
2009-11-04 21:16:50 +00:00
if ( ! cl_muzzleflash . ival ) // remove all muzzleflashes
2005-06-04 12:40:12 +00:00
return ;
2004-08-23 00:15:46 +00:00
2005-06-04 12:40:12 +00:00
if ( i - 1 = = cl . playernum [ destsplit ] & & cl_muzzleflash . value = = 2 )
return ;
2004-08-23 00:15:46 +00:00
2005-09-26 03:40:09 +00:00
pack = & cl . frames [ cls . netchan . incoming_sequence & UPDATE_MASK ] . packet_entities ;
2004-08-23 00:15:46 +00:00
for ( pnum = 0 ; pnum < pack - > num_entities ; pnum + + ) //try looking for an entity with that id first
{
s1 = & pack - > entities [ pnum ] ;
if ( s1 - > number = = i )
{
2008-01-24 01:24:10 +00:00
dl = CL_AllocDlight ( - i ) ;
2004-08-23 00:15:46 +00:00
VectorCopy ( s1 - > origin , dl - > origin ) ;
2009-07-05 18:45:53 +00:00
AngleVectors ( s1 - > angles , dl - > axis [ 0 ] , dl - > axis [ 1 ] , dl - > axis [ 2 ] ) ;
2004-08-23 00:15:46 +00:00
break ;
}
}
if ( pnum = = pack - > num_entities )
{ //that ent number doesn't exist, go for a player with that number
if ( ( unsigned ) ( i ) < = MAX_CLIENTS & & i > 0 )
{
pl = & cl . frames [ parsecountmod ] . playerstate [ i - 1 ] ;
2008-01-24 01:24:10 +00:00
dl = CL_AllocDlight ( - i ) ;
2004-08-23 00:15:46 +00:00
VectorCopy ( pl - > origin , dl - > origin ) ; //set it's origin
2010-08-14 00:15:07 +00:00
if ( pl - > hullnum & 0x80 ) /*hull is 0-based, so origin is bottom of model, move the light up slightly*/
dl - > origin [ 2 ] + = 24 ;
2009-07-05 18:45:53 +00:00
AngleVectors ( pl - > viewangles , dl - > axis [ 0 ] , dl - > axis [ 1 ] , dl - > axis [ 2 ] ) ;
2004-08-23 00:15:46 +00:00
AngleVectors ( pl - > viewangles , fv , rv , uv ) ; //shift it up a little
2009-07-05 18:45:53 +00:00
VectorMA ( dl - > origin , 15 , fv , dl - > origin ) ;
2004-08-23 00:15:46 +00:00
}
else
return ;
}
dl - > radius = 200 + ( rand ( ) & 31 ) ;
dl - > minlight = 32 ;
dl - > die = cl . time + 0.1334 ;
dl - > color [ 0 ] = 0.2 ;
dl - > color [ 1 ] = 0.1 ;
dl - > color [ 2 ] = 0.05 ;
2005-06-04 04:20:20 +00:00
dl - > channelfade [ 0 ] = 1.5 ;
dl - > channelfade [ 1 ] = 0.75 ;
2004-08-23 00:15:46 +00:00
dl - > channelfade [ 2 ] = 0.375 ;
}
# ifdef Q2CLIENT
void Q2S_StartSound ( vec3_t origin , int entnum , int entchannel , sfx_t * sfx , float fvol , float attenuation , float timeofs ) ;
void CLQ2_ParseMuzzleFlash ( void )
{
vec3_t fv , rv , dummy ;
dlight_t * dl ;
int i , weapon ;
vec3_t org , ang ;
int silenced ;
float volume ;
char soundname [ 64 ] ;
i = MSG_ReadShort ( ) ;
if ( i < 1 | | i > = Q2MAX_EDICTS )
Host_Error ( " CL_ParseMuzzleFlash: bad entity " ) ;
weapon = MSG_ReadByte ( ) ;
silenced = weapon & Q2MZ_SILENCED ;
weapon & = ~ Q2MZ_SILENCED ;
CL_GetNumberedEntityInfo ( i , org , ang ) ;
dl = CL_AllocDlight ( i ) ;
VectorCopy ( org , dl - > origin ) ;
AngleVectors ( ang , fv , rv , dummy ) ;
VectorMA ( dl - > origin , 18 , fv , dl - > origin ) ;
VectorMA ( dl - > origin , 16 , rv , dl - > origin ) ;
if ( silenced )
dl - > radius = 100 + ( rand ( ) & 31 ) ;
else
dl - > radius = 200 + ( rand ( ) & 31 ) ;
dl - > minlight = 32 ;
dl - > die = cl . time + 0.05 ; //+ 0.1;
dl - > decay = 1 ;
dl - > channelfade [ 0 ] = 2 ;
dl - > channelfade [ 1 ] = 2 ;
dl - > channelfade [ 2 ] = 2 ;
if ( silenced )
volume = 0.2 ;
else
volume = 1 ;
switch ( weapon )
{
case Q2MZ_BLASTER :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/blastf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_BLUEHYPERBLASTER :
dl - > color [ 0 ] = 0 ; dl - > color [ 1 ] = 0 ; dl - > color [ 2 ] = 0.2 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/hyprbf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_HYPERBLASTER :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/hyprbf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_MACHINEGUN :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2004-08-23 00:15:46 +00:00
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0 ) ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case Q2MZ_SHOTGUN :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/shotgf1b.wav " ) , volume , ATTN_NORM , 0 ) ;
Q2S_StartSound ( NULL , i , CHAN_AUTO , S_PrecacheSound ( " weapons/shotgr1b.wav " ) , volume , ATTN_NORM , 0.1 ) ;
break ;
case Q2MZ_SSHOTGUN :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/sshotf1b.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_CHAINGUN1 :
dl - > radius = 200 + ( rand ( ) & 31 ) ;
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.05 ; dl - > color [ 2 ] = 0 ;
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2004-08-23 00:15:46 +00:00
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_CHAINGUN2 :
dl - > radius = 225 + ( rand ( ) & 31 ) ;
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.1 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 0.1 ; // long delay
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2004-08-23 00:15:46 +00:00
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0 ) ;
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2006-05-10 07:35:19 +00:00
Q2S_StartSound ( NULL , i , CHAN_AUTO , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0.05 ) ;
2004-08-23 00:15:46 +00:00
break ;
case Q2MZ_CHAINGUN3 :
dl - > radius = 250 + ( rand ( ) & 31 ) ;
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 0.1 ; // long delay
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2004-08-23 00:15:46 +00:00
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0 ) ;
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2006-05-10 07:35:19 +00:00
Q2S_StartSound ( NULL , i , CHAN_AUTO , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0.033 ) ;
2006-03-06 01:41:09 +00:00
snprintf ( soundname , sizeof ( soundname ) , " weapons/machgf%ib.wav " , ( rand ( ) % 5 ) + 1 ) ;
2006-05-10 07:35:19 +00:00
Q2S_StartSound ( NULL , i , CHAN_AUTO , S_PrecacheSound ( soundname ) , volume , ATTN_NORM , 0.066 ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case Q2MZ_RAILGUN :
dl - > color [ 0 ] = 0.1 ; dl - > color [ 1 ] = 0.1 ; dl - > color [ 2 ] = 0.2 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/railgf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_ROCKET :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.1 ; dl - > color [ 2 ] = 0.04 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/rocklf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
Q2S_StartSound ( NULL , i , CHAN_AUTO , S_PrecacheSound ( " weapons/rocklr1b.wav " ) , volume , ATTN_NORM , 0.1 ) ;
break ;
case Q2MZ_GRENADE :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.1 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/grenlf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
Q2S_StartSound ( NULL , i , CHAN_AUTO , S_PrecacheSound ( " weapons/grenlr1b.wav " ) , volume , ATTN_NORM , 0.1 ) ;
break ;
case Q2MZ_BFG :
dl - > color [ 0 ] = 0 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/bfg__f1y.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_LOGIN :
dl - > color [ 0 ] = 0 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 1.0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/grenlf1a.wav " ) , 1 , ATTN_NORM , 0 ) ;
// CL_LogoutEffect (pl->current.origin, weapon);
break ;
case Q2MZ_LOGOUT :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 1.0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/grenlf1a.wav " ) , 1 , ATTN_NORM , 0 ) ;
// CL_LogoutEffect (pl->current.origin, weapon);
break ;
case Q2MZ_RESPAWN :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 1.0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/grenlf1a.wav " ) , 1 , ATTN_NORM , 0 ) ;
// CL_LogoutEffect (pl->current.origin, weapon);
break ;
// RAFAEL
case Q2MZ_PHALANX :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.1 ; dl - > color [ 2 ] = 0.1 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/plasshot.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
// RAFAEL
2005-09-26 03:40:09 +00:00
case Q2MZ_IONRIPPER :
2004-08-23 00:15:46 +00:00
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.1 ; dl - > color [ 2 ] = 0.1 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/rippfire.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
// ======================
// PGM
case Q2MZ_ETF_RIFLE :
dl - > color [ 0 ] = 0.18 ; dl - > color [ 1 ] = 0.14 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/nail1.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_SHOTGUN2 :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/shotg2.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_HEATBEAM :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 100 ;
// Q2S_StartSound (NULL, i, CHAN_WEAPON, S_PrecacheSound("weapons/bfg__l1a.wav"), volume, ATTN_NORM, 0);
break ;
case Q2MZ_BLASTER2 :
dl - > color [ 0 ] = 0 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
// FIXME - different sound for blaster2 ??
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/blastf1a.wav " ) , volume , ATTN_NORM , 0 ) ;
break ;
case Q2MZ_TRACKER :
// negative flashes handled the same in gl/soft until CL_AddDLights
dl - > color [ 0 ] = - 0.2 ; dl - > color [ 1 ] = - 0.2 ; dl - > color [ 2 ] = - 0.2 ;
Q2S_StartSound ( NULL , i , CHAN_WEAPON , S_PrecacheSound ( " weapons/disint2.wav " ) , volume , ATTN_NORM , 0 ) ;
2005-09-26 03:40:09 +00:00
break ;
2004-08-23 00:15:46 +00:00
case Q2MZ_NUKE1 :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 100 ;
break ;
case Q2MZ_NUKE2 :
dl - > color [ 0 ] = 0.2 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0 ;
dl - > die = cl . time + 100 ;
break ;
case Q2MZ_NUKE4 :
dl - > color [ 0 ] = 0 ; dl - > color [ 1 ] = 0 ; dl - > color [ 2 ] = 0.2 ;
dl - > die = cl . time + 100 ;
break ;
case Q2MZ_NUKE8 :
dl - > color [ 0 ] = 0 ; dl - > color [ 1 ] = 0.2 ; dl - > color [ 2 ] = 0.2 ;
dl - > die = cl . time + 100 ;
break ;
// PGM
// ======================
}
}
2005-09-26 03:40:09 +00:00
void CLQ2_ParseMuzzleFlash2 ( void )
2004-08-23 00:15:46 +00:00
{
int ent ;
int flash_number ;
ent = MSG_ReadShort ( ) ;
if ( ent < 1 | | ent > = Q2MAX_EDICTS )
Host_EndGame ( " CL_ParseMuzzleFlash2: bad entity " ) ;
flash_number = MSG_ReadByte ( ) ;
2005-03-18 06:14:07 +00:00
CLQ2_RunMuzzleFlash2 ( ent , flash_number ) ;
2004-08-23 00:15:46 +00:00
}
2005-09-04 05:48:26 +00:00
void CLQ2_ParseInventory ( void )
{
int i ;
2005-09-07 14:55:25 +00:00
// TODO: finish this properly
2005-09-04 05:48:26 +00:00
for ( i = 0 ; i < Q2MAX_ITEMS ; i + + )
// cl.inventory[i] = MSG_ReadShort (&net_message);
MSG_ReadShort ( ) ; // just ignore everything for now
}
2004-08-23 00:15:46 +00:00
# endif
2004-09-30 22:42:34 +00:00
//return if we want to print the message.
2005-09-07 14:55:25 +00:00
char * CL_ParseChat ( char * text , player_info_t * * player )
2005-09-26 03:40:09 +00:00
{
2005-12-13 02:31:57 +00:00
extern cvar_t cl_chatsound , cl_nofake , cl_teamchatsound , cl_enemychatsound ;
2005-09-07 14:55:25 +00:00
int flags ;
int offset = 0 ;
qboolean suppress_talksound ;
char * p ;
2004-08-23 00:15:46 +00:00
char * s ;
2005-11-30 01:20:53 +00:00
int check_flood ;
2004-08-30 07:08:13 +00:00
2005-09-07 14:55:25 +00:00
flags = TP_CategorizeMessage ( text , & offset , player ) ;
s = text + offset ;
2004-08-30 07:08:13 +00:00
2005-09-07 14:55:25 +00:00
if ( flags )
2004-08-23 00:15:46 +00:00
{
2005-09-07 14:55:25 +00:00
if ( ! cls . demoplayback )
Sys_ServerActivity ( ) ; //chat always flashes the screen..
2007-09-17 20:35:39 +00:00
2005-09-07 14:55:25 +00:00
//check f_ stuff
2007-09-17 20:35:39 +00:00
if ( * player & & ! strncmp ( s , " f_ " , 2 ) )
2004-09-30 22:42:34 +00:00
{
2007-09-17 20:35:39 +00:00
Validation_Auto_Response ( * player - cl . players , s ) ;
2005-09-08 08:10:06 +00:00
return s ;
2005-09-07 14:55:25 +00:00
}
2007-09-17 20:35:39 +00:00
2005-09-07 14:55:25 +00:00
Validation_CheckIfResponse ( text ) ;
2004-09-30 22:42:34 +00:00
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( ! Plug_ChatMessage ( text + offset , * player ? ( int ) ( * player - cl . players ) : - 1 , flags ) )
return NULL ;
2006-01-13 06:27:18 +00:00
# endif
2006-01-01 09:01:15 +00:00
2008-11-09 22:29:28 +00:00
if ( flags & ( TPM_TEAM | TPM_OBSERVEDTEAM ) & & ! TP_FilterMessage ( text + offset ) )
return NULL ;
if ( flags & ( TPM_TEAM | TPM_OBSERVEDTEAM ) & & Sbar_UpdateTeamStatus ( * player , text + offset ) )
2005-09-07 14:55:25 +00:00
return NULL ;
2005-11-30 01:20:53 +00:00
2008-11-09 22:29:28 +00:00
2005-11-30 01:20:53 +00:00
if ( ( int ) msg_filter . value & flags )
return NULL ; //filter chat
2005-12-13 02:31:57 +00:00
check_flood = Ignore_Check_Flood ( s , flags , offset ) ;
if ( check_flood = = IGNORE_NO_ADD )
2005-11-30 01:20:53 +00:00
return NULL ;
2005-12-13 02:31:57 +00:00
else if ( check_flood = = NO_IGNORE_ADD )
2005-11-30 01:20:53 +00:00
Ignore_Flood_Add ( s ) ;
2005-09-07 14:55:25 +00:00
}
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
else
{
if ( ! Plug_ServerMessage ( text + offset , PRINT_CHAT ) )
return NULL ;
}
2006-01-13 06:27:18 +00:00
# endif
2004-09-30 22:42:34 +00:00
2005-09-07 14:55:25 +00:00
suppress_talksound = false ;
2004-09-30 22:42:34 +00:00
2005-09-07 14:55:25 +00:00
if ( flags = = 2 | | ( ! cl . teamplay & & flags ) )
suppress_talksound = TP_CheckSoundTrigger ( text + offset ) ;
2004-09-30 22:42:34 +00:00
2005-09-07 14:55:25 +00:00
if ( ! cl_chatsound . value | | // no sound at all
( cl_chatsound . value = = 2 & & flags ! = 2 ) ) // only play sound in mm2
suppress_talksound = true ;
2004-09-30 22:42:34 +00:00
2005-09-07 14:55:25 +00:00
if ( ! suppress_talksound )
2005-12-13 02:31:57 +00:00
{
2008-11-09 22:29:28 +00:00
if ( flags & ( TPM_OBSERVEDTEAM | TPM_TEAM ) & & cl . teamplay )
2005-12-13 02:33:03 +00:00
S_LocalSound ( cl_teamchatsound . string ) ;
2005-12-13 02:31:57 +00:00
else
2005-12-13 02:33:03 +00:00
S_LocalSound ( cl_enemychatsound . string ) ;
2005-12-13 02:31:57 +00:00
}
2004-09-30 22:42:34 +00:00
2005-09-07 14:55:25 +00:00
if ( cl_nofake . value = = 1 | | ( cl_nofake . value = = 2 & & flags ! = 2 ) ) {
for ( p = s ; * p ; p + + )
if ( * p = = 13 | | ( * p = = 10 & & p [ 1 ] ) )
2005-09-26 03:40:09 +00:00
* p = ' ' ;
2004-09-30 22:42:34 +00:00
}
2005-09-07 14:55:25 +00:00
msgflags = flags ;
2005-09-08 08:10:06 +00:00
2005-09-07 14:55:25 +00:00
return s ;
2004-08-23 00:15:46 +00:00
}
2005-04-26 16:04:12 +00:00
char printtext [ 4096 ] ;
2004-09-30 22:42:34 +00:00
void CL_ParsePrint ( char * msg , int level )
2004-08-23 00:15:46 +00:00
{
2005-03-01 15:36:23 +00:00
if ( strlen ( printtext ) + strlen ( msg ) > = sizeof ( printtext ) )
{
Con_Printf ( " %s " , printtext ) ;
Q_strncpyz ( printtext , msg , sizeof ( printtext ) ) ;
}
else
strcat ( printtext , msg ) ; //safe due to size on if.
2004-09-13 04:16:52 +00:00
while ( ( msg = strchr ( printtext , ' \n ' ) ) )
2004-08-23 00:15:46 +00:00
{
* msg = ' \0 ' ;
2004-09-30 22:42:34 +00:00
if ( level ! = PRINT_CHAT )
Stats_ParsePrintLine ( printtext ) ;
2005-09-07 14:55:25 +00:00
2005-07-02 00:32:09 +00:00
TP_SearchForMsgTriggers ( printtext , level ) ;
2004-08-23 00:15:46 +00:00
msg + + ;
memmove ( printtext , msg , strlen ( msg ) + 1 ) ;
}
}
2005-09-26 08:07:26 +00:00
// CL_PlayerColor: returns color and mask for player_info_t
2009-05-24 10:11:17 +00:00
int CL_PlayerColor ( player_info_t * plr , qboolean * name_coloured )
2005-09-26 08:07:26 +00:00
{
char * t ;
int c ;
2009-05-24 10:11:17 +00:00
* name_coloured = false ;
2005-09-26 08:07:26 +00:00
if ( cl . teamfortress ) //override based on team
{
// TODO: needs some work
2007-07-23 18:52:11 +00:00
switch ( plr - > rbottomcolor )
2005-09-26 08:07:26 +00:00
{ //translate q1 skin colours to console colours
case 10 :
case 1 :
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
case 4 : //red
c = 1 ;
break ;
case 11 :
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
case 3 : // green
c = 2 ;
break ;
case 5 :
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
case 12 :
c = 3 ;
break ;
case 6 :
case 7 :
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
case 8 :
case 9 :
2006-01-02 22:05:47 +00:00
c = 6 ;
2005-09-26 08:07:26 +00:00
break ;
case 2 : // light blue
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
case 13 : //blue
case 14 : //blue
2006-01-02 22:05:47 +00:00
c = 5 ;
2005-09-26 08:07:26 +00:00
break ;
default :
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
case 0 : // white
c = 7 ;
break ;
}
}
2005-11-01 23:25:15 +00:00
else if ( cl . teamplay )
{
// team name hacks
if ( ! strcmp ( plr - > team , " red " ) )
c = 1 ;
else if ( ! strcmp ( plr - > team , " blue " ) )
2006-01-02 22:05:47 +00:00
c = 5 ;
2005-11-01 23:25:15 +00:00
else
{
char * t ;
t = plr - > team ;
c = 0 ;
for ( t = plr - > team ; * t ; t + + )
{
c > > = 1 ;
c ^ = * t ; // TODO: very weak hash, replace
}
if ( ( c / 7 ) & 1 )
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-11-01 23:25:15 +00:00
c = 1 + ( c % 7 ) ;
}
}
2005-09-26 08:07:26 +00:00
else
{
// override chat color with tc infokey
// 0-6 is standard colors (red to white)
// 7-13 is using secondard charactermask
// 14 and afterwards repeats
t = Info_ValueForKey ( plr - > userinfo , " tc " ) ;
if ( * t )
c = atoi ( t ) ;
else
2005-10-08 22:35:20 +00:00
c = plr - > userid ; // Quake2 can start from 0
2005-09-26 08:07:26 +00:00
if ( ( c / 7 ) & 1 )
2009-05-24 10:11:17 +00:00
* name_coloured = true ;
2005-09-26 08:07:26 +00:00
c = 1 + ( c % 7 ) ;
}
return c ;
}
2006-01-01 09:01:15 +00:00
// CL_PrintChat: takes chat strings and performs name coloring and cl_parsewhitetext parsing
2005-09-08 08:10:06 +00:00
// NOTE: text in rawmsg/msg is assumed destroyable and should not be used afterwards
void CL_PrintChat ( player_info_t * plr , char * rawmsg , char * msg , int plrflags )
2005-09-07 14:55:25 +00:00
{
2005-09-08 08:10:06 +00:00
char * name = NULL ;
2005-09-07 14:55:25 +00:00
int c ;
2009-05-24 10:11:17 +00:00
qboolean name_coloured = false ;
2005-09-07 14:55:25 +00:00
extern cvar_t cl_parsewhitetext ;
qboolean memessage = false ;
2009-05-24 10:11:17 +00:00
char fullchatmessage [ 2048 ] ;
fullchatmessage [ 0 ] = 0 ;
2005-09-08 08:10:06 +00:00
if ( plrflags & TPM_FAKED )
{
name = rawmsg ; // use rawmsg pointer and msg modification to generate null-terminated string
if ( msg )
* ( msg - 2 ) = 0 ; // it's assumed that msg has 2 chars before it due to strstr
}
2005-09-07 14:55:25 +00:00
if ( msg [ 0 ] = = ' / ' & & msg [ 1 ] = = ' m ' & & msg [ 2 ] = = ' e ' & & msg [ 3 ] = = ' ' )
{
msg + = 4 ;
memessage = true ; // special /me formatting
}
if ( plr ) // use special formatting with a real chat message
2005-09-08 08:10:06 +00:00
name = plr - > name ; // use player's name
2009-11-04 21:16:50 +00:00
if ( cl_standardchat . ival )
2005-09-07 14:55:25 +00:00
{
2009-05-24 10:11:17 +00:00
name_coloured = true ;
2005-09-08 08:10:06 +00:00
c = 7 ;
}
else
{
if ( plrflags & TPM_SPECTATOR ) // is an observer
2005-09-07 14:55:25 +00:00
{
// TODO: we don't even check for this yet...
2008-11-09 22:29:28 +00:00
if ( plrflags & ( TPM_TEAM | TPM_OBSERVEDTEAM ) ) // is on team
2005-09-07 14:55:25 +00:00
c = 0 ; // blacken () on observers
else
{
2009-05-24 10:11:17 +00:00
name_coloured = true ;
2005-09-07 14:55:25 +00:00
c = 7 ;
}
}
2005-09-08 08:10:06 +00:00
else if ( plr )
2009-05-24 10:11:17 +00:00
c = CL_PlayerColor ( plr , & name_coloured ) ;
2005-09-08 08:10:06 +00:00
else
{
// defaults for fake clients
2009-05-24 10:11:17 +00:00
name_coloured = true ;
2005-09-08 08:10:06 +00:00
c = 7 ;
2005-09-07 14:55:25 +00:00
}
2005-09-08 08:10:06 +00:00
}
2005-09-07 14:55:25 +00:00
2005-09-08 08:10:06 +00:00
c = ' 0 ' + c ;
2005-09-07 14:55:25 +00:00
2005-09-08 08:10:06 +00:00
if ( name )
{
2005-09-07 14:55:25 +00:00
if ( memessage )
{
2005-09-08 08:10:06 +00:00
if ( ! cl_standardchat . value & & ( plrflags & TPM_SPECTATOR ) )
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , " ^m^0*^7 " , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
else
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , " ^m* " , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
2008-11-09 22:29:28 +00:00
if ( plrflags & ( TPM_TEAM | TPM_OBSERVEDTEAM ) ) // for team chat don't highlight the name, just the brackets
2005-09-07 14:55:25 +00:00
{
// color is reset every printf so we're safe here
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , va ( " \1 %s^%c( " , name_coloured ? " " : " ^m " , c ) , sizeof ( fullchatmessage ) ) ;
Q_strncatz ( fullchatmessage , va ( " %s%s^d " , name_coloured ? " ^m " : " " , name ) , sizeof ( fullchatmessage ) ) ;
Q_strncatz ( fullchatmessage , va ( " %s^%c) " , name_coloured ? " ^m " : " " , c ) , sizeof ( fullchatmessage ) ) ;
}
2009-11-04 21:16:50 +00:00
else if ( cl_standardchat . ival )
2009-05-24 10:11:17 +00:00
{
Q_strncatz ( fullchatmessage , va ( " \1 %s " , name ) , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
else
{
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , va ( " \1 %s^%c%s^d " , name_coloured ? " " : " ^m " , c , name ) , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
if ( ! memessage )
{
// only print seperator with an actual player name
2005-09-08 08:10:06 +00:00
if ( ! cl_standardchat . value & & ( plrflags & TPM_SPECTATOR ) )
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , " ^0: ^d " , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
else
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , " : " , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
else
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , " " , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
// print message
2008-11-09 22:29:28 +00:00
if ( cl_parsewhitetext . value & & ( cl_parsewhitetext . value = = 1 | | ( plrflags & ( TPM_TEAM | TPM_OBSERVEDTEAM ) ) ) )
2005-09-07 14:55:25 +00:00
{
2005-09-26 08:07:26 +00:00
char * t , * u ;
2005-09-07 14:55:25 +00:00
2006-02-17 19:54:47 +00:00
while ( ( t = strchr ( msg , ' { ' ) ) )
2005-09-07 14:55:25 +00:00
{
u = strchr ( msg , ' } ' ) ;
if ( u )
{
* t = 0 ;
* u = 0 ;
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , va ( " %s " , msg ) , sizeof ( fullchatmessage ) ) ;
Q_strncatz ( fullchatmessage , va ( " ^m%s^m " , t + 1 ) , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
msg = u + 1 ;
}
else
break ;
}
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , va ( " %s " , msg ) , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
else
{
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullchatmessage , va ( " %s " , msg ) , sizeof ( fullchatmessage ) ) ;
2005-09-07 14:55:25 +00:00
}
2009-05-24 10:28:26 +00:00
# ifdef CSQC_DAT
if ( CSQC_ParsePrint ( fullchatmessage , PRINT_CHAT ) )
return ;
# endif
Con_Printf ( " %s " , fullchatmessage ) ;
2005-09-07 14:55:25 +00:00
}
2005-09-26 08:07:26 +00:00
// CL_PrintStandardMessage: takes non-chat net messages and performs name coloring
// NOTE: msg is considered destroyable
2005-10-16 03:47:32 +00:00
char acceptedchars [ ] = { ' . ' , ' ? ' , ' ! ' , ' \' ' , ' , ' , ' : ' , ' ' , ' \0 ' } ;
2009-05-24 10:11:17 +00:00
void CL_PrintStandardMessage ( char * msg , int printlevel )
2005-09-26 08:07:26 +00:00
{
int i ;
player_info_t * p ;
extern cvar_t cl_standardmsg ;
2005-10-16 03:47:32 +00:00
char * begin = msg ;
2009-05-24 10:11:17 +00:00
char fullmessage [ 2048 ] ;
fullmessage [ 0 ] = 0 ;
2005-09-26 08:07:26 +00:00
// search for player names in message
for ( i = 0 , p = cl . players ; i < MAX_CLIENTS ; p + + , i + + )
{
char * v ;
char * name ;
int len ;
2009-05-24 10:11:17 +00:00
qboolean coloured ;
2005-09-26 08:07:26 +00:00
char c ;
2005-11-01 23:25:15 +00:00
name = p - > name ;
if ( ! ( * name ) )
2005-09-26 08:07:26 +00:00
continue ;
len = strlen ( name ) ;
v = strstr ( msg , name ) ;
2005-10-16 03:47:32 +00:00
while ( v )
2005-09-26 08:07:26 +00:00
{
2005-10-16 03:47:32 +00:00
// name parsing rules
if ( v ! = begin & & * ( v - 1 ) ! = ' ' ) // must be space before name
{
2005-10-16 19:35:21 +00:00
v = strstr ( v + len , name ) ;
2005-10-16 03:47:32 +00:00
continue ;
}
2005-11-26 03:02:55 +00:00
2005-10-16 03:47:32 +00:00
{
int i ;
char aftername = * ( v + len ) ;
2005-11-26 03:02:55 +00:00
2005-10-16 03:47:32 +00:00
// search for accepted chars in char after name in msg
for ( i = 0 ; i < sizeof ( acceptedchars ) ; i + + )
{
if ( acceptedchars [ i ] = = aftername )
break ;
}
if ( sizeof ( acceptedchars ) = = i )
{
2005-10-16 19:35:21 +00:00
v = strstr ( v + len , name ) ;
2005-10-16 03:47:32 +00:00
continue ; // no accepted char found
}
}
2005-09-26 08:07:26 +00:00
* v = 0 ; // cut off message
2009-05-24 10:11:17 +00:00
2005-09-26 08:07:26 +00:00
// print msg chunk
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullmessage , msg , sizeof ( fullmessage ) ) ;
2005-09-26 08:07:26 +00:00
msg = v + len ; // update search point
// get name color
2009-11-04 21:16:50 +00:00
if ( p - > spectator | | cl_standardmsg . ival )
2005-09-26 08:07:26 +00:00
{
2009-05-24 10:11:17 +00:00
coloured = false ;
2005-09-26 08:07:26 +00:00
c = ' 7 ' ;
}
else
2009-05-24 10:11:17 +00:00
c = ' 0 ' + CL_PlayerColor ( p , & coloured ) ;
2005-09-26 08:07:26 +00:00
// print name
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullmessage , va ( " %s^%c%s^7 " , coloured ? " \1 " : " " , c , name ) , sizeof ( fullmessage ) ) ;
2005-10-16 03:47:32 +00:00
break ;
2005-09-26 08:07:26 +00:00
}
}
// print final chunk
2009-05-24 10:11:17 +00:00
Q_strncatz ( fullmessage , msg , sizeof ( fullmessage ) ) ;
2009-05-24 10:28:26 +00:00
# ifdef CSQC_DAT
if ( CSQC_ParsePrint ( fullmessage , printlevel ) )
return ;
# endif
Con_Printf ( " %s " , fullmessage ) ;
2005-09-26 08:07:26 +00:00
}
2004-08-23 00:15:46 +00:00
char stufftext [ 4096 ] ;
void CL_ParseStuffCmd ( char * msg , int destsplit ) //this protects stuffcmds from network segregation.
{
strncat ( stufftext , msg , sizeof ( stufftext ) - 1 ) ;
2004-09-13 04:16:52 +00:00
while ( ( msg = strchr ( stufftext , ' \n ' ) ) )
2004-08-23 00:15:46 +00:00
{
* msg = ' \0 ' ;
Con_DPrintf ( " stufftext: %s \n " , stufftext ) ;
if ( ! strncmp ( stufftext , " fullserverinfo " , 15 ) )
2005-02-12 18:56:04 +00:00
Cmd_ExecuteString ( stufftext , RESTRICT_SERVER + destsplit ) ; //do this NOW so that it's done before any models or anything are loaded
2004-08-23 00:15:46 +00:00
else
{
2006-11-03 15:53:04 +00:00
if ( ! strncmp ( stufftext , " //querycmd " , 11 ) )
{
COM_Parse ( stufftext + 11 ) ;
if ( Cmd_Exists ( com_token ) )
{
Cbuf_AddText ( " cmd cmdsupported " , RESTRICT_SERVER + destsplit ) ;
Cbuf_AddText ( com_token , RESTRICT_SERVER + destsplit ) ;
Cbuf_AddText ( " \n " , RESTRICT_SERVER + destsplit ) ;
}
}
2009-07-06 01:20:20 +00:00
else if ( ! strncmp ( stufftext , " //vwep " , 7 ) )
2009-06-21 17:45:33 +00:00
{
2009-07-06 01:20:20 +00:00
int i ;
char * mname ;
Cmd_TokenizeString ( stufftext + 7 , false , false ) ;
for ( i = 0 ; i < Cmd_Argc ( ) ; i + + )
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
mname = Cmd_Argv ( i ) ;
if ( strcmp ( mname , " - " ) )
2009-07-06 01:20:20 +00:00
{
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
mname = va ( " progs/%s.mdl " , Cmd_Argv ( i ) ) ;
Q_strncpyz ( cl . model_name_vwep [ i ] , mname , sizeof ( cl . model_name_vwep [ i ] ) ) ;
if ( cls . state = = ca_active )
{
CL_CheckOrEnqueDownloadFile ( mname , NULL , 0 ) ;
cl . model_precache_vwep [ i ] = Mod_ForName ( mname , false ) ;
}
2009-07-06 01:20:20 +00:00
}
}
2009-06-21 17:45:33 +00:00
}
2007-01-06 09:36:38 +00:00
else if ( ! strncmp ( stufftext , " //exectrigger " , 14 ) )
{
COM_Parse ( stufftext + 14 ) ;
if ( Cmd_AliasExist ( com_token , RESTRICT_SERVER ) )
Cmd_ExecuteString ( com_token , RESTRICT_SERVER ) ; //do this NOW so that it's done before any models or anything are loaded
}
else if ( ! strncmp ( stufftext , " //set " , 6 ) )
{
Cmd_ExecuteString ( stufftext + 2 , RESTRICT_SERVER + destsplit ) ; //do this NOW so that it's done before any models or anything are loaded
}
2008-01-30 02:32:00 +00:00
else if ( ! strncmp ( stufftext , " //at " , 5 ) )
{
Cam_SetAutoTrack ( atoi ( stufftext + 5 ) ) ;
}
2007-10-05 14:11:17 +00:00
# ifdef PLUGINS
2007-10-05 11:45:18 +00:00
else if ( ! strncmp ( stufftext , " //tinfo " , 8 ) )
{
Cmd_TokenizeString ( stufftext + 2 , false , false ) ;
Plug_Command_f ( ) ;
}
2008-01-30 02:32:00 +00:00
else if ( ! strncmp ( stufftext , " //sn " , 5 ) )
{
Cmd_TokenizeString ( stufftext + 2 , false , false ) ;
Plug_Command_f ( ) ;
}
2007-10-05 14:11:17 +00:00
# endif
2005-02-12 18:56:04 +00:00
# ifdef CSQC_DAT
2010-07-11 02:22:39 +00:00
else if ( CSQC_StuffCmd ( destsplit , stufftext , msg ) )
{
}
2005-02-12 18:56:04 +00:00
# endif
2008-01-30 02:32:00 +00:00
else
2005-02-12 18:56:04 +00:00
{
Cbuf_AddText ( stufftext , RESTRICT_SERVER + destsplit ) ;
Cbuf_AddText ( " \n " , RESTRICT_SERVER + destsplit ) ;
}
2004-08-23 00:15:46 +00:00
}
msg + + ;
memmove ( stufftext , msg , strlen ( msg ) + 1 ) ;
}
}
2005-05-26 12:55:34 +00:00
void CL_ParsePrecache ( void )
{
int i = ( unsigned short ) MSG_ReadShort ( ) ;
char * s = MSG_ReadString ( ) ;
if ( i < 32768 )
{
if ( i > = 1 & & i < MAX_MODELS )
{
model_t * model ;
2008-11-09 22:29:28 +00:00
CL_CheckOrEnqueDownloadFile ( s , s , 0 ) ;
2005-05-26 12:55:34 +00:00
model = Mod_ForName ( s , i = = 1 ) ;
if ( ! model )
Con_Printf ( " svc_precache: Mod_ForName( \" %s \" ) failed \n " , s ) ;
cl . model_precache [ i ] = model ;
strcpy ( cl . model_name [ i ] , s ) ;
2009-03-03 01:52:30 +00:00
cl . model_precaches_added = true ;
2005-05-26 12:55:34 +00:00
}
else
Con_Printf ( " svc_precache: model index %i outside range %i...%i \n " , i , 1 , MAX_MODELS ) ;
}
else
{
i - = 32768 ;
if ( i > = 1 & & i < MAX_SOUNDS )
{
sfx_t * sfx ;
2009-04-06 00:34:32 +00:00
if ( S_HaveOutput ( ) )
CL_CheckOrEnqueDownloadFile ( va ( " sound/%s " , s ) , NULL , 0 ) ;
2005-05-26 12:55:34 +00:00
sfx = S_PrecacheSound ( s ) ;
if ( ! sfx )
Con_Printf ( " svc_precache: S_PrecacheSound( \" %s \" ) failed \n " , s ) ;
cl . sound_precache [ i ] = sfx ;
strcpy ( cl . sound_name [ i ] , s ) ;
}
else
Con_Printf ( " svc_precache: sound index %i outside range %i...%i \n " , i , 1 , MAX_SOUNDS ) ;
}
}
2006-01-11 22:24:08 +00:00
void CL_DumpPacket ( void )
{
int i ;
char * packet = net_message . data ;
int pos ;
pos = 0 ;
while ( pos < net_message . cursize )
{
Con_Printf ( " %5i " , pos ) ;
for ( i = 0 ; i < 16 ; i + + )
{
if ( pos > = net_message . cursize )
2010-08-16 02:03:02 +00:00
Con_Printf ( " - " ) ;
2006-01-11 22:24:08 +00:00
else
Con_Printf ( " %2x " , ( unsigned char ) packet [ pos ] ) ;
pos + + ;
}
pos - = 16 ;
for ( i = 0 ; i < 16 ; i + + )
{
if ( pos > = net_message . cursize )
Con_Printf ( " X " ) ;
2008-11-09 22:29:28 +00:00
else if ( packet [ pos ] = = 0 | | packet [ pos ] = = ' \n ' )
2006-01-11 22:24:08 +00:00
Con_Printf ( " . " ) ;
else
Con_Printf ( " %c " , ( unsigned char ) packet [ pos ] ) ;
pos + + ;
}
Con_Printf ( " \n " ) ;
}
}
2005-05-26 12:55:34 +00:00
2004-08-23 00:15:46 +00:00
# define SHOWNET(x) if(cl_shownet.value==2)Con_Printf ("%3i:%s\n", msg_readcount-1, x);
2004-11-27 08:16:25 +00:00
# define SHOWNET2(x, y) if(cl_shownet.value==2)Con_Printf ("%3i:%3i:%s\n", msg_readcount-1, y, x);
2004-08-23 00:15:46 +00:00
/*
= = = = = = = = = = = = = = = = = = = = =
CL_ParseServerMessage
= = = = = = = = = = = = = = = = = = = = =
*/
int received_framecount ;
void CL_ParseServerMessage ( void )
{
int cmd ;
char * s ;
int i , j ;
int destsplit ;
2008-05-25 22:23:43 +00:00
float f ;
2004-08-23 00:15:46 +00:00
received_framecount = host_framecount ;
cl . last_servermessage = realtime ;
CL_ClearProjectiles ( ) ;
cl . fixangle = false ;
//
// if recording demos, copy the message out
2005-03-12 23:40:42 +00:00
//
2004-08-23 00:15:46 +00:00
if ( cl_shownet . value = = 1 )
Con_TPrintf ( TL_INT_SPACE , net_message . cursize ) ;
else if ( cl_shownet . value = = 2 )
Con_TPrintf ( TLC_LINEBREAK_MINUS ) ;
CL_ParseClientdata ( ) ;
//
// parse the message
//
while ( 1 )
{
if ( msg_badread )
{
2006-02-12 20:20:01 +00:00
CL_DumpPacket ( ) ;
2004-08-23 00:15:46 +00:00
Host_EndGame ( " CL_ParseServerMessage: Bad server message " ) ;
break ;
}
cmd = MSG_ReadByte ( ) ;
2008-05-25 22:23:43 +00:00
if ( cmd = = svcfte_choosesplitclient )
2004-08-23 00:15:46 +00:00
{
SHOWNET ( svc_strings [ cmd ] ) ;
2010-07-11 02:22:39 +00:00
destsplit = MSG_ReadByte ( ) % MAX_SPLITS ;
2004-08-23 00:15:46 +00:00
cmd = MSG_ReadByte ( ) ;
}
else
destsplit = 0 ;
if ( cmd = = - 1 )
{
msg_readcount + + ; // so the EOM showner has the right value
SHOWNET ( " END OF MESSAGE " ) ;
break ;
}
SHOWNET ( svc_strings [ cmd ] ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
// other commands
switch ( cmd )
{
default :
2006-01-11 22:24:08 +00:00
CL_DumpPacket ( ) ;
2006-02-06 01:06:17 +00:00
Host_EndGame ( " CL_ParseServerMessage: Illegible server message (%i) " , cmd ) ;
2004-08-23 00:15:46 +00:00
return ;
case svc_time :
2005-05-26 12:55:34 +00:00
cl . oldgametime = cl . gametime ;
2004-08-23 00:15:46 +00:00
cl . gametime = MSG_ReadFloat ( ) ;
cl . gametimemark = realtime ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_nop :
// Con_Printf ("svc_nop\n");
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_disconnect :
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_EZTV ) //eztv fails to detect the end of demos.
MSG_ReadString ( ) ;
else if ( cls . state = = ca_connected )
2004-08-23 00:15:46 +00:00
{
Host_EndGame ( " Server disconnected \n "
" Server version may not be compatible " ) ;
}
else
Host_EndGame ( " Server disconnected " ) ;
break ;
case svc_print :
i = MSG_ReadByte ( ) ;
s = MSG_ReadString ( ) ;
2005-11-30 02:26:12 +00:00
2004-08-23 00:15:46 +00:00
if ( i = = PRINT_CHAT )
{
2005-09-07 14:55:25 +00:00
char * msg ;
player_info_t * plr = NULL ;
2005-04-26 16:04:12 +00:00
if ( TP_SuppressMessage ( s ) )
break ; //if this was unseen-sent from us, ignore it.
2006-02-17 19:54:47 +00:00
if ( ( msg = CL_ParseChat ( s , & plr ) ) )
2004-08-23 00:15:46 +00:00
{
2004-09-30 22:42:34 +00:00
CL_ParsePrint ( s , i ) ;
2005-09-08 08:10:06 +00:00
CL_PrintChat ( plr , s , msg , msgflags ) ;
2004-08-23 00:15:46 +00:00
}
}
else
{
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( Plug_ServerMessage ( s , i ) )
2006-01-13 06:27:18 +00:00
# endif
2006-01-01 09:01:15 +00:00
{
CL_ParsePrint ( s , i ) ;
2009-05-24 10:11:17 +00:00
CL_PrintStandardMessage ( s , i ) ;
2006-01-01 09:01:15 +00:00
}
2004-08-23 00:15:46 +00:00
}
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_centerprint :
2006-01-01 09:01:15 +00:00
s = MSG_ReadString ( ) ;
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( Plug_CenterPrintMessage ( s , destsplit ) )
2006-01-13 06:27:18 +00:00
# endif
2008-11-09 22:29:28 +00:00
SCR_CenterPrint ( destsplit , s , false ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_stufftext :
s = MSG_ReadString ( ) ;
2005-09-26 03:40:09 +00:00
CL_ParseStuffCmd ( s , destsplit ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_damage :
V_ParseDamage ( destsplit ) ;
break ;
case svc_serverdata :
Cbuf_Execute ( ) ; // make sure any stuffed commands are done
CL_ParseServerData ( ) ;
vid . recalc_refdef = true ; // leave full screen intermission
break ;
# ifdef PEXT_SETVIEW
case svc_setview :
if ( ! ( cls . fteprotocolextensions & PEXT_SETVIEW ) )
2008-11-09 22:29:28 +00:00
Con_Printf ( " ^1PEXT_SETVIEW is meant to be disabled \n " ) ;
2005-09-26 03:40:09 +00:00
cl . viewentity [ destsplit ] = MSG_ReadShort ( ) ;
2008-11-09 22:29:28 +00:00
if ( cl . viewentity [ destsplit ] = = cl . playernum [ destsplit ] + 1 )
cl . viewentity [ destsplit ] = 0 ;
2004-08-23 00:15:46 +00:00
break ;
# endif
case svc_setangle :
2008-01-09 00:52:31 +00:00
if ( cls . demoplayback = = DPB_MVD | | cls . demoplayback = = DPB_EZTV )
2004-08-23 00:15:46 +00:00
{
i = MSG_ReadByte ( ) ;
if ( i ! = spec_track [ 0 ] | | ! autocam [ 0 ] )
{ //this wasn't for us.
for ( i = 0 ; i < 3 ; i + + )
MSG_ReadAngle ( ) ;
break ;
}
cl . fixangle = true ;
for ( i = 0 ; i < 3 ; i + + )
cl . simangles [ destsplit ] [ i ] = cl . viewangles [ destsplit ] [ i ] = MSG_ReadAngle ( ) ;
break ;
}
cl . fixangle = true ;
for ( i = 0 ; i < 3 ; i + + )
cl . viewangles [ destsplit ] [ i ] = MSG_ReadAngle ( ) ;
// cl.viewangles[PITCH] = cl.viewangles[ROLL] = 0;
break ;
case svc_lightstyle :
i = MSG_ReadByte ( ) ;
if ( i > = MAX_LIGHTSTYLES )
Host_EndGame ( " svc_lightstyle > MAX_LIGHTSTYLES " ) ;
cl_lightstyle [ i ] . colour = 7 ; //white
Q_strncpyz ( cl_lightstyle [ i ] . map , MSG_ReadString ( ) , sizeof ( cl_lightstyle [ i ] . map ) ) ;
cl_lightstyle [ i ] . length = Q_strlen ( cl_lightstyle [ i ] . map ) ;
break ;
# ifdef PEXT_LIGHTSTYLECOL
2008-05-25 22:23:43 +00:00
case svcfte_lightstylecol :
2004-08-23 00:15:46 +00:00
if ( ! ( cls . fteprotocolextensions & PEXT_LIGHTSTYLECOL ) )
Host_EndGame ( " PEXT_LIGHTSTYLECOL is meant to be disabled \n " ) ;
i = MSG_ReadByte ( ) ;
if ( i > = MAX_LIGHTSTYLES )
2005-09-18 02:12:30 +00:00
Host_EndGame ( " svc_lightstyle > MAX_LIGHTSTYLES " ) ;
2004-08-23 00:15:46 +00:00
cl_lightstyle [ i ] . colour = MSG_ReadByte ( ) ;
Q_strncpyz ( cl_lightstyle [ i ] . map , MSG_ReadString ( ) , sizeof ( cl_lightstyle [ i ] . map ) ) ;
cl_lightstyle [ i ] . length = Q_strlen ( cl_lightstyle [ i ] . map ) ;
break ;
# endif
case svc_sound :
CL_ParseStartSoundPacket ( ) ;
break ;
2008-11-28 20:34:51 +00:00
# ifdef PEXT_SOUNDDBL
case svcfte_soundextended :
CLNQ_ParseStartSoundPacket ( ) ;
break ;
# endif
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_stopsound :
i = MSG_ReadShort ( ) ;
S_StopSound ( i > > 3 , i & 7 ) ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_updatefrags :
Sbar_Changed ( ) ;
i = MSG_ReadByte ( ) ;
if ( i > = MAX_CLIENTS )
Host_EndGame ( " CL_ParseServerMessage: svc_updatefrags > MAX_SCOREBOARD " ) ;
cl . players [ i ] . frags = MSG_ReadShort ( ) ;
2005-09-26 03:40:09 +00:00
break ;
2004-08-23 00:15:46 +00:00
case svc_updateping :
i = MSG_ReadByte ( ) ;
if ( i > = MAX_CLIENTS )
Host_EndGame ( " CL_ParseServerMessage: svc_updateping > MAX_SCOREBOARD " ) ;
cl . players [ i ] . ping = MSG_ReadShort ( ) ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_updatepl :
i = MSG_ReadByte ( ) ;
if ( i > = MAX_CLIENTS )
Host_EndGame ( " CL_ParseServerMessage: svc_updatepl > MAX_SCOREBOARD " ) ;
cl . players [ i ] . pl = MSG_ReadByte ( ) ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_updateentertime :
// time is sent over as seconds ago
i = MSG_ReadByte ( ) ;
if ( i > = MAX_CLIENTS )
Host_EndGame ( " CL_ParseServerMessage: svc_updateentertime > MAX_SCOREBOARD " ) ;
2005-11-26 03:02:55 +00:00
cl . players [ i ] . entertime = cl . servertime - MSG_ReadFloat ( ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_spawnbaseline :
i = MSG_ReadShort ( ) ;
2006-07-24 04:24:41 +00:00
if ( ! CL_CheckBaselines ( i ) )
Host_EndGame ( " CL_ParseServerMessage: svc_spawnbaseline failed with size %i " , i ) ;
CL_ParseBaseline ( cl_baselines + i ) ;
2004-08-23 00:15:46 +00:00
break ;
2008-05-25 22:23:43 +00:00
case svcfte_spawnbaseline2 :
2004-08-23 00:15:46 +00:00
CL_ParseBaseline2 ( ) ;
break ;
case svc_spawnstatic :
CL_ParseStatic ( 1 ) ;
break ;
case svc_spawnstatic2 :
CL_ParseStatic ( 2 ) ;
break ;
case svc_temp_entity :
# ifdef NQPROT
CL_ParseTEnt ( false ) ;
# else
CL_ParseTEnt ( ) ;
# endif
break ;
2008-05-25 22:23:43 +00:00
case svcfte_customtempent :
2004-08-23 00:15:46 +00:00
CL_ParseCustomTEnt ( ) ;
break ;
case svc_particle :
2005-03-10 03:55:18 +00:00
CLNQ_ParseParticleEffect ( ) ;
2004-08-23 00:15:46 +00:00
break ;
2008-05-25 22:23:43 +00:00
case svcfte_particle2 :
2005-03-10 03:55:18 +00:00
CL_ParseParticleEffect2 ( ) ;
2004-08-23 00:15:46 +00:00
break ;
2008-05-25 22:23:43 +00:00
case svcfte_particle3 :
2005-03-10 03:55:18 +00:00
CL_ParseParticleEffect3 ( ) ;
2004-08-23 00:15:46 +00:00
break ;
2008-05-25 22:23:43 +00:00
case svcfte_particle4 :
2005-03-10 03:55:18 +00:00
CL_ParseParticleEffect4 ( ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_killedmonster :
2010-07-11 02:22:39 +00:00
cl . stats [ destsplit ] [ STAT_MONSTERS ] + + ;
2004-08-23 00:15:46 +00:00
break ;
case svc_foundsecret :
2010-07-11 02:22:39 +00:00
cl . stats [ destsplit ] [ STAT_SECRETS ] + + ;
2004-08-23 00:15:46 +00:00
break ;
case svc_updatestat :
i = MSG_ReadByte ( ) ;
j = MSG_ReadByte ( ) ;
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( destsplit , i , j ) ;
CL_SetStatFloat ( destsplit , i , j ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_updatestatlong :
i = MSG_ReadByte ( ) ;
j = MSG_ReadLong ( ) ; //make qbyte if nq compatability?
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( destsplit , i , j ) ;
CL_SetStatFloat ( destsplit , i , j ) ;
break ;
case svcfte_updatestatstring :
i = MSG_ReadByte ( ) ;
s = MSG_ReadString ( ) ;
CL_SetStatString ( destsplit , i , s ) ;
break ;
case svcfte_updatestatfloat :
i = MSG_ReadByte ( ) ;
f = MSG_ReadFloat ( ) ;
CL_SetStatInt ( destsplit , i , f ) ;
CL_SetStatFloat ( destsplit , i , f ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_spawnstaticsound :
CL_ParseStaticSound ( ) ;
break ;
case svc_cdtrack :
cl . cdtrack = MSG_ReadByte ( ) ;
CDAudio_Play ( ( qbyte ) cl . cdtrack , true ) ;
break ;
case svc_intermission :
2005-07-14 01:57:34 +00:00
if ( ! cl . intermission )
TP_ExecTrigger ( " f_mapend " ) ;
2004-08-23 00:15:46 +00:00
cl . intermission = 1 ;
2005-11-26 03:02:55 +00:00
cl . completed_time = cl . servertime ;
2004-08-23 00:15:46 +00:00
vid . recalc_refdef = true ; // go to full screen
for ( i = 0 ; i < 3 ; i + + )
2005-09-26 03:40:09 +00:00
cl . simorg [ 0 ] [ i ] = MSG_ReadCoord ( ) ;
2004-08-23 00:15:46 +00:00
for ( i = 0 ; i < 3 ; i + + )
cl . simangles [ 0 ] [ i ] = MSG_ReadAngle ( ) ;
2006-05-29 04:50:24 +00:00
VectorClear ( cl . simvel [ 0 ] ) ;
2004-08-23 00:15:46 +00:00
VectorCopy ( cl . simvel [ 0 ] , cl . simvel [ 1 ] ) ;
VectorCopy ( cl . simangles [ 0 ] , cl . simangles [ 1 ] ) ;
VectorCopy ( cl . simorg [ 0 ] , cl . simorg [ 1 ] ) ;
break ;
case svc_finale :
cl . intermission = 2 ;
2005-11-26 03:02:55 +00:00
cl . completed_time = cl . servertime ;
2004-08-23 00:15:46 +00:00
vid . recalc_refdef = true ; // go to full screen
2008-11-09 22:29:28 +00:00
SCR_CenterPrint ( destsplit , MSG_ReadString ( ) , false ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_sellscreen :
Cmd_ExecuteString ( " help " , RESTRICT_RCON ) ;
break ;
case svc_smallkick :
cl . punchangle [ destsplit ] = - 2 ;
break ;
case svc_bigkick :
cl . punchangle [ destsplit ] = - 4 ;
break ;
case svc_muzzleflash :
2005-06-04 12:40:12 +00:00
CL_MuzzleFlash ( destsplit ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_updateuserinfo :
CL_UpdateUserinfo ( ) ;
break ;
case svc_setinfo :
2007-02-23 00:21:33 +00:00
CL_ParseSetInfo ( ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_serverinfo :
CL_ServerInfo ( ) ;
break ;
case svc_download :
CL_ParseDownload ( ) ;
break ;
case svc_playerinfo :
CL_ParsePlayerinfo ( ) ;
break ;
case svc_nails :
2004-12-08 04:14:52 +00:00
CL_ParseProjectiles ( cl_spikeindex , false ) ;
break ;
case svc_nails2 :
CL_ParseProjectiles ( cl_spikeindex , true ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_chokecount : // some preceding packets were choked
i = MSG_ReadByte ( ) ;
for ( j = 0 ; j < i ; j + + )
cl . frames [ ( cls . netchan . incoming_acknowledged - 1 - j ) & UPDATE_MASK ] . receivedtime = - 2 ;
break ;
case svc_modellist :
CL_ParseModellist ( false ) ;
break ;
2008-05-25 22:23:43 +00:00
case svcfte_modellistshort :
2004-08-23 00:15:46 +00:00
CL_ParseModellist ( true ) ;
break ;
case svc_soundlist :
2008-11-28 20:34:51 +00:00
CL_ParseSoundlist ( false ) ;
break ;
# ifdef PEXT_SOUNDDBL
case svcfte_soundlistshort :
CL_ParseSoundlist ( true ) ;
2004-08-23 00:15:46 +00:00
break ;
2008-11-28 20:34:51 +00:00
# endif
2004-08-23 00:15:46 +00:00
case svc_packetentities :
CL_ParsePacketEntities ( false ) ;
2007-06-20 00:02:54 +00:00
cl . ackedinputsequence = cl . validsequence ;
2004-08-23 00:15:46 +00:00
break ;
case svc_deltapacketentities :
CL_ParsePacketEntities ( true ) ;
2007-06-20 00:02:54 +00:00
cl . ackedinputsequence = cl . validsequence ;
2004-08-23 00:15:46 +00:00
break ;
case svc_maxspeed :
cl . maxspeed [ destsplit ] = MSG_ReadFloat ( ) ;
break ;
case svc_entgravity :
cl . entgravity [ destsplit ] = MSG_ReadFloat ( ) ;
break ;
case svc_setpause :
cl . paused = MSG_ReadByte ( ) ;
if ( cl . paused )
CDAudio_Pause ( ) ;
else
CDAudio_Resume ( ) ;
break ;
# ifdef PEXT_BULLETENS
2008-05-25 22:23:43 +00:00
case svcfte_bulletentext :
2004-08-23 00:15:46 +00:00
if ( ! ( cls . fteprotocolextensions & PEXT_BULLETENS ) )
Host_EndGame ( " PEXT_BULLETENS is meant to be disabled \n " ) ;
Bul_ParseMessage ( ) ;
break ;
# endif
case svc_ftesetclientpersist :
CL_ParseClientPersist ( ) ;
break ;
# ifdef Q2BSPS
case svc_setportalstate :
i = MSG_ReadByte ( ) ;
j = MSG_ReadByte ( ) ;
i * = j & 127 ;
j & = ~ 128 ;
CMQ2_SetAreaPortalState ( i , j ! = 0 ) ;
break ;
# endif
2004-09-01 00:01:08 +00:00
2008-05-25 22:23:43 +00:00
case svcfte_showpic :
2004-09-01 00:01:08 +00:00
SCR_ShowPic_Create ( ) ;
break ;
2008-05-25 22:23:43 +00:00
case svcfte_hidepic :
2004-09-01 00:01:08 +00:00
SCR_ShowPic_Hide ( ) ;
break ;
2008-05-25 22:23:43 +00:00
case svcfte_movepic :
2004-09-01 00:01:08 +00:00
SCR_ShowPic_Move ( ) ;
break ;
2008-05-25 22:23:43 +00:00
case svcfte_updatepic :
2004-09-01 00:01:08 +00:00
SCR_ShowPic_Update ( ) ;
break ;
2004-10-10 06:32:29 +00:00
2008-05-25 22:23:43 +00:00
case svcfte_effect :
2004-11-22 00:00:54 +00:00
CL_ParseEffect ( false ) ;
break ;
2008-05-25 22:23:43 +00:00
case svcfte_effect2 :
2004-11-22 00:00:54 +00:00
CL_ParseEffect ( true ) ;
break ;
2005-02-28 07:16:19 +00:00
# ifdef PEXT_CSQC
2008-05-25 22:23:43 +00:00
case svcfte_csqcentities :
2005-02-28 07:16:19 +00:00
CSQC_ParseEntities ( ) ;
break ;
# endif
2008-05-25 22:23:43 +00:00
case svcfte_precache :
2005-05-26 12:55:34 +00:00
CL_ParsePrecache ( ) ;
break ;
2008-11-09 22:29:28 +00:00
case svcfte_trailparticles :
2008-11-28 20:34:51 +00:00
CLDP_ParseTrailParticles ( ) ;
2008-11-09 22:29:28 +00:00
break ;
case svcfte_pointparticles :
CLDP_ParsePointParticles ( false ) ;
break ;
case svcfte_pointparticles1 :
CLDP_ParsePointParticles ( true ) ;
break ;
2009-04-01 22:03:56 +00:00
case svcfte_cgamepacket :
# ifdef HLCLIENT
if ( CLHL_ParseGamePacket ( ) ) ;
break ;
# endif
# ifdef CSQC_DAT
if ( CSQC_ParseGamePacket ( ) ) ;
break ;
# endif
Con_Printf ( " Unable to parse gamecode packet \n " ) ;
break ;
2004-08-23 00:15:46 +00:00
}
}
}
# ifdef Q2CLIENT
void CLQ2_ParseServerMessage ( void )
{
int cmd ;
char * s ;
int i ;
// int j;
received_framecount = host_framecount ;
cl . last_servermessage = realtime ;
CL_ClearProjectiles ( ) ;
//
// if recording demos, copy the message out
//
if ( cl_shownet . value = = 1 )
Con_TPrintf ( TL_INT_SPACE , net_message . cursize ) ;
else if ( cl_shownet . value = = 2 )
Con_TPrintf ( TLC_LINEBREAK_MINUS ) ;
CL_ParseClientdata ( ) ;
//
// parse the message
//
while ( 1 )
{
if ( msg_badread )
{
2005-03-18 06:14:07 +00:00
Host_EndGame ( " CLQ2_ParseServerMessage: Bad server message " ) ;
2004-08-23 00:15:46 +00:00
break ;
}
cmd = MSG_ReadByte ( ) ;
if ( cmd = = - 1 )
{
msg_readcount + + ; // so the EOM showner has the right value
SHOWNET ( " END OF MESSAGE " ) ;
break ;
}
SHOWNET ( va ( " %i " , cmd ) ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
// other commands
switch ( cmd )
{
default :
2009-11-04 21:16:50 +00:00
Host_EndGame ( " CLQ2_ParseServerMessage: Illegible server message (%i) " , cmd ) ;
2004-08-23 00:15:46 +00:00
return ;
//known to game
case svcq2_muzzleflash :
2005-09-26 03:40:09 +00:00
CLQ2_ParseMuzzleFlash ( ) ;
2004-08-23 00:15:46 +00:00
break ;
case svcq2_muzzleflash2 :
CLQ2_ParseMuzzleFlash2 ( ) ;
return ;
case svcq2_temp_entity :
CLQ2_ParseTEnt ( ) ;
break ;
case svcq2_layout :
s = MSG_ReadString ( ) ;
Q_strncpyz ( cl . q2layout , s , sizeof ( cl . q2layout ) ) ;
2005-09-09 23:40:55 +00:00
# ifdef VM_UI
2004-08-23 00:15:46 +00:00
UI_Q2LayoutChanged ( ) ;
2005-09-09 23:40:55 +00:00
# endif
2004-08-23 00:15:46 +00:00
break ;
case svcq2_inventory :
2005-09-04 05:48:26 +00:00
CLQ2_ParseInventory ( ) ;
break ;
2004-08-23 00:15:46 +00:00
// the rest are private to the client and server
case svcq2_nop : //6
Host_EndGame ( " CL_ParseServerMessage: svcq2_nop not implemented " ) ;
return ;
case svcq2_disconnect :
if ( cls . state = = ca_connected )
Host_EndGame ( " Server disconnected \n "
" Server version may not be compatible " ) ;
else
Host_EndGame ( " Server disconnected " ) ;
return ;
case svcq2_reconnect : //8
Con_TPrintf ( TLC_RECONNECTING ) ;
2005-03-23 22:14:08 +00:00
CL_SendClientCommand ( true , " new " ) ;
2004-08-23 00:15:46 +00:00
break ;
case svcq2_sound : //9 // <see code>
CLQ2_ParseStartSoundPacket ( ) ;
break ;
case svcq2_print : //10 // [qbyte] id [string] null terminated string
i = MSG_ReadByte ( ) ;
s = MSG_ReadString ( ) ;
2005-11-30 02:26:12 +00:00
2004-08-23 00:15:46 +00:00
if ( i = = PRINT_CHAT )
{
2005-09-07 14:55:25 +00:00
char * msg ;
player_info_t * plr = NULL ;
2006-02-17 19:54:47 +00:00
if ( ( msg = CL_ParseChat ( s , & plr ) ) )
2004-09-30 22:42:34 +00:00
{
CL_ParsePrint ( s , i ) ;
2005-09-08 08:10:06 +00:00
CL_PrintChat ( plr , s , msg , msgflags ) ;
2004-09-30 22:42:34 +00:00
}
2004-08-23 00:15:46 +00:00
}
else
{
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( Plug_ServerMessage ( s , i ) )
2006-01-13 06:27:18 +00:00
# endif
2006-01-01 09:01:15 +00:00
{
CL_ParsePrint ( s , i ) ;
2009-05-24 10:11:17 +00:00
CL_PrintStandardMessage ( s , i ) ;
2006-01-01 09:01:15 +00:00
}
2004-08-23 00:15:46 +00:00
}
break ;
case svcq2_stufftext : //11 // [string] stuffed into client's console buffer, should be \n terminated
s = MSG_ReadString ( ) ;
Con_DPrintf ( " stufftext: %s \n " , s ) ;
if ( ! strncmp ( s , " precache " , 8 ) ) //big major hack. Q2 uses a command that q1 has as a cvar.
{ //call the q2 precache function.
CLQ2_Precache_f ( ) ;
}
else
Cbuf_AddText ( s , RESTRICT_SERVER ) ; //don't let the local user cheat
break ;
case svcq2_serverdata : //12 // [long] protocol ...
Cbuf_Execute ( ) ; // make sure any stuffed commands are done
CLQ2_ParseServerData ( ) ;
break ;
case svcq2_configstring : //13 // [short] [string]
CLQ2_ParseConfigString ( ) ;
break ;
2005-09-26 03:40:09 +00:00
case svcq2_spawnbaseline : //14
2004-08-23 00:15:46 +00:00
CLQ2_ParseBaseline ( ) ;
break ;
case svcq2_centerprint : //15 // [string] to put in center of the screen
2006-01-01 09:01:15 +00:00
s = MSG_ReadString ( ) ;
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( Plug_CenterPrintMessage ( s , 0 ) )
2006-01-13 06:27:18 +00:00
# endif
2008-11-09 22:29:28 +00:00
SCR_CenterPrint ( 0 , s , false ) ;
2004-08-23 00:15:46 +00:00
break ;
case svcq2_download : //16 // [short] size [size bytes]
CL_ParseDownload ( ) ;
break ;
case svcq2_playerinfo : //17 // variable
Host_EndGame ( " CL_ParseServerMessage: svcq2_playerinfo not implemented " ) ;
return ;
case svcq2_packetentities : //18 // [...]
Host_EndGame ( " CL_ParseServerMessage: svcq2_packetentities not implemented " ) ;
return ;
case svcq2_deltapacketentities : //19 // [...]
Host_EndGame ( " CL_ParseServerMessage: svcq2_deltapacketentities not implemented " ) ;
return ;
case svcq2_frame : //20 (the bastard to implement.)
CLQ2_ParseFrame ( ) ;
break ;
}
}
CL_SetSolidEntities ( ) ;
}
# endif
# ifdef NQPROT
2005-10-16 03:53:31 +00:00
//Proquake specific stuff
# define pqc_nop 1
# define pqc_new_team 2
# define pqc_erase_team 3
# define pqc_team_frags 4
# define pqc_match_time 5
# define pqc_match_reset 6
# define pqc_ping_times 7
int MSG_ReadBytePQ ( char * * s )
{
int ret = ( * s ) [ 0 ] * 16 + ( * s ) [ 1 ] - 272 ;
* s + = 2 ;
return ret ;
}
int MSG_ReadShortPQ ( char * * s )
{
return MSG_ReadBytePQ ( s ) * 256 + MSG_ReadBytePQ ( s ) ;
}
void CLNQ_ParseProQuakeMessage ( char * s )
{
2005-11-30 01:20:53 +00:00
int cmd ;
int ping ;
// int team, shirt, frags, i, j;
2005-10-16 03:53:31 +00:00
s + + ;
cmd = * s + + ;
2005-11-26 03:02:55 +00:00
2005-10-16 03:53:31 +00:00
switch ( cmd )
{
default :
Con_DPrintf ( " Unrecognised ProQuake Message %i \n " , cmd ) ;
break ;
/* case pqc_new_team:
Sbar_Changed ( ) ;
team = MSG_ReadByte ( ) - 16 ;
if ( team < 0 | | team > 13 )
Host_Error ( " CL_ParseProQuakeMessage: pqc_new_team invalid team " ) ;
shirt = MSG_ReadByte ( ) - 16 ;
cl . teamgame = true ;
// cl.teamscores[team].frags = 0; // JPG 3.20 - removed this
cl . teamscores [ team ] . colors = 16 * shirt + team ;
//Con_Printf("pqc_new_team %d %d\n", team, shirt);
break ;
case pqc_erase_team :
Sbar_Changed ( ) ;
team = MSG_ReadByte ( ) - 16 ;
if ( team < 0 | | team > 13 )
Host_Error ( " CL_ParseProQuakeMessage: pqc_erase_team invalid team " ) ;
cl . teamscores [ team ] . colors = 0 ;
cl . teamscores [ team ] . frags = 0 ; // JPG 3.20 - added this
//Con_Printf("pqc_erase_team %d\n", team);
break ;
case pqc_team_frags :
Sbar_Changed ( ) ;
team = MSG_ReadByte ( ) - 16 ;
if ( team < 0 | | team > 13 )
Host_Error ( " CL_ParseProQuakeMessage: pqc_team_frags invalid team " ) ;
frags = MSG_ReadShortPQ ( ) ; ;
if ( frags & 32768 )
frags = frags - 65536 ;
cl . teamscores [ team ] . frags = frags ;
//Con_Printf("pqc_team_frags %d %d\n", team, frags);
2005-11-26 03:02:55 +00:00
break ;
2005-10-16 03:53:31 +00:00
case pqc_match_time :
Sbar_Changed ( ) ;
cl . minutes = MSG_ReadBytePQ ( ) ;
cl . seconds = MSG_ReadBytePQ ( ) ;
cl . last_match_time = cl . time ;
//Con_Printf("pqc_match_time %d %d\n", cl.minutes, cl.seconds);
break ;
case pqc_match_reset :
Sbar_Changed ( ) ;
for ( i = 0 ; i < 14 ; i + + )
{
cl . teamscores [ i ] . colors = 0 ;
cl . teamscores [ i ] . frags = 0 ; // JPG 3.20 - added this
}
//Con_Printf("pqc_match_reset\n");
break ;
*/
case pqc_ping_times :
while ( ( ping = MSG_ReadShortPQ ( & s ) ) )
{
if ( ( ping / 4096 ) > = MAX_CLIENTS )
Host_Error ( " CL_ParseProQuakeMessage: pqc_ping_times > MAX_CLIENTS " ) ;
cl . players [ ping / 4096 ] . ping = ping & 4095 ;
}
break ;
}
}
2010-08-16 02:03:02 +00:00
static enum {
CLNQPP_NONE ,
CLNQPP_PINGS
} cl_nqparseprint ;
qboolean CLNQ_ParseNQPrints ( char * s )
{
int i ;
char * start = s ;
if ( cl_nqparseprint = = CLNQPP_PINGS )
{
char * pingstart ;
cl_nqparseprint = CLNQPP_NONE ;
while ( * s = = ' ' )
s + + ;
pingstart = s ;
if ( * s = = ' - ' )
s + + ;
if ( * s > = ' 0 ' & & * s < = ' 9 ' )
{
while ( * s > = ' 0 ' & & * s < = ' 9 ' )
s + + ;
if ( * s = = ' ' & & s - start > = 4 )
{
s + + ;
start = s ;
s = strchr ( s , ' \n ' ) ;
if ( ! s )
return false ;
* s = 0 ;
for ( i = 0 ; i < MAX_CLIENTS ; i + + )
{
if ( ! strcmp ( start , cl . players [ i ] . name ) )
break ;
}
if ( i = = MAX_CLIENTS )
{
}
if ( i ! = MAX_CLIENTS )
{
cl . players [ i ] . ping = atoi ( pingstart ) ;
}
cl_nqparseprint = CLNQPP_PINGS ;
return true ;
}
}
s = start ;
}
if ( ! strcmp ( s , " Client ping times: \n " ) )
{
cl_nqparseprint = CLNQPP_PINGS ;
return true ;
}
return false ;
}
2005-10-16 03:53:31 +00:00
2004-08-23 00:15:46 +00:00
void CLNQ_ParseServerMessage ( void )
{
int cmd ;
char * s ;
int i , j ;
// received_framecount = host_framecount;
// cl.last_servermessage = realtime;
CL_ClearProjectiles ( ) ;
cl . fixangle = false ;
2005-05-26 12:55:34 +00:00
cl . allowsendpacket = true ;
2004-08-23 00:15:46 +00:00
//
// if recording demos, copy the message out
//
if ( cl_shownet . value = = 1 )
Con_TPrintf ( TL_INT_SPACE , net_message . cursize ) ;
else if ( cl_shownet . value = = 2 )
Con_TPrintf ( TLC_LINEBREAK_MINUS ) ;
CL_ParseClientdata ( ) ;
//
// parse the message
//
while ( 1 )
{
if ( msg_badread )
{
2006-02-12 20:20:01 +00:00
CL_DumpPacket ( ) ;
2004-08-23 00:15:46 +00:00
Host_EndGame ( " CL_ParseServerMessage: Bad server message " ) ;
break ;
}
cmd = MSG_ReadByte ( ) ;
if ( cmd = = - 1 )
{
msg_readcount + + ; // so the EOM showner has the right value
SHOWNET ( " END OF MESSAGE " ) ;
break ;
}
if ( cmd & 128 )
{
SHOWNET ( " fast update " ) ;
CLNQ_ParseEntity ( cmd & 127 ) ;
continue ;
}
2004-11-27 08:16:25 +00:00
SHOWNET2 ( svc_nqstrings [ cmd > ( sizeof ( svc_nqstrings ) / sizeof ( char * ) ) ? 0 : cmd ] , cmd ) ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
// other commands
switch ( cmd )
{
default :
2006-02-12 20:20:01 +00:00
CL_DumpPacket ( ) ;
2004-11-20 01:21:17 +00:00
Host_EndGame ( " CLNQ_ParseServerMessage: Illegible server message (%i) " , cmd ) ;
2004-08-23 00:15:46 +00:00
return ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_nop :
// Con_Printf ("svc_nop\n");
2005-09-26 03:40:09 +00:00
break ;
2004-08-23 00:15:46 +00:00
case svc_print :
s = MSG_ReadString ( ) ;
2005-11-30 02:26:12 +00:00
2004-09-30 22:42:34 +00:00
if ( * s = = 1 | | * s = = 2 )
{
2005-09-07 14:55:25 +00:00
char * msg ;
player_info_t * plr = NULL ;
2006-02-17 19:54:47 +00:00
if ( ( msg = CL_ParseChat ( s + 1 , & plr ) ) )
2004-09-30 22:42:34 +00:00
{
2005-09-07 14:55:25 +00:00
CL_ParsePrint ( s + 1 , PRINT_CHAT ) ;
2005-09-08 08:10:06 +00:00
CL_PrintChat ( plr , s + 1 , msg , msgflags ) ;
2004-09-30 22:42:34 +00:00
}
}
else
{
2010-08-16 02:03:02 +00:00
if ( CLNQ_ParseNQPrints ( s ) )
break ;
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( Plug_ServerMessage ( s , PRINT_HIGH ) )
2006-01-13 06:27:18 +00:00
# endif
2006-01-01 09:01:15 +00:00
{
CL_ParsePrint ( s , PRINT_HIGH ) ;
2009-05-24 10:11:17 +00:00
CL_PrintStandardMessage ( s , PRINT_HIGH ) ;
2006-01-01 09:01:15 +00:00
}
2004-09-30 22:42:34 +00:00
}
2004-08-23 00:15:46 +00:00
break ;
case svc_disconnect :
2005-05-26 12:55:34 +00:00
CL_Disconnect ( ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_centerprint :
2006-01-01 09:01:15 +00:00
s = MSG_ReadString ( ) ;
2006-01-13 06:27:18 +00:00
# ifdef PLUGINS
2006-01-01 09:01:15 +00:00
if ( Plug_CenterPrintMessage ( s , 0 ) )
2006-01-13 06:27:18 +00:00
# endif
2008-11-09 22:29:28 +00:00
SCR_CenterPrint ( 0 , s , false ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_stufftext :
s = MSG_ReadString ( ) ;
2005-10-16 03:53:31 +00:00
if ( * s = = 1 )
{
Con_DPrintf ( " Proquake: %s \n " , s ) ;
CLNQ_ParseProQuakeMessage ( s ) ;
}
else
{
Con_DPrintf ( " stufftext: %s \n " , s ) ;
2007-06-20 00:02:54 +00:00
if ( ! strncmp ( s , " cl_serverextension_download " , 14 ) )
{
2008-12-03 02:42:05 +00:00
cl_dp_serverextension_download = true ;
2007-06-20 00:02:54 +00:00
}
else if ( ! strncmp ( s , " \n cl_downloadbegin " , 17 ) )
CLDP_ParseDownloadBegin ( s ) ;
else if ( ! strncmp ( s , " \n cl_downloadfinished " , 17 ) )
CLDP_ParseDownloadFinished ( s ) ;
else if ( ! strncmp ( s , " csqc_progname " , 14 ) )
2008-11-09 22:29:28 +00:00
COM_ParseOut ( s + 14 , cl_dp_csqc_progsname , sizeof ( cl_dp_csqc_progsname ) ) ;
2007-06-20 00:02:54 +00:00
else if ( ! strncmp ( s , " csqc_progsize " , 14 ) )
cl_dp_csqc_progssize = atoi ( s + 14 ) ;
else if ( ! strncmp ( s , " csqc_progcrc " , 13 ) )
cl_dp_csqc_progscrc = atoi ( s + 13 ) ;
2009-03-03 01:52:30 +00:00
else if ( ! strncmp ( s , " cl_fullpitch " , 13 ) | | ! strncmp ( s , " pq_fullpitch " , 13 ) )
{
//
}
2007-06-20 00:02:54 +00:00
else
{
Cbuf_AddText ( s , RESTRICT_SERVER ) ; //no cheating here...
}
2005-10-16 03:53:31 +00:00
}
2004-08-23 00:15:46 +00:00
break ;
case svc_serverdata :
Cbuf_Execute ( ) ; // make sure any stuffed commands are done
CLNQ_ParseServerData ( ) ;
2005-09-26 03:40:09 +00:00
vid . recalc_refdef = true ; // leave full screen intermission
2004-08-23 00:15:46 +00:00
break ;
2005-05-26 12:55:34 +00:00
case svcdp_precache :
CL_ParsePrecache ( ) ;
2004-12-05 08:19:54 +00:00
break ;
2004-08-23 00:15:46 +00:00
case svc_cdtrack :
cl . cdtrack = MSG_ReadByte ( ) ;
MSG_ReadByte ( ) ;
2005-09-26 03:40:09 +00:00
CDAudio_Play ( ( qbyte ) cl . cdtrack , true ) ;
break ;
2004-08-23 00:15:46 +00:00
case svc_setview :
if ( ! cl . viewentity [ 0 ] )
2006-01-28 19:04:13 +00:00
{
2004-08-23 00:15:46 +00:00
cl . playernum [ 0 ] = ( cl . viewentity [ 0 ] = MSG_ReadShort ( ) ) - 1 ;
2006-01-28 19:04:13 +00:00
if ( cl . playernum [ 0 ] > = MAX_CLIENTS )
{
2009-11-04 21:16:50 +00:00
Con_Printf ( CON_WARNING " WARNING: Server put us in slot %i. We are not on the scoreboard. \n " , cl . playernum [ 0 ] ) ;
cl . playernum [ 0 ] = MAX_CLIENTS ; //pretend it's an mvd (we have that spare slot)
2006-01-28 19:04:13 +00:00
}
}
2004-08-23 00:15:46 +00:00
else
cl . viewentity [ 0 ] = MSG_ReadShort ( ) ;
break ;
case svc_signonnum :
i = MSG_ReadByte ( ) ;
if ( i < = cls . signon )
Host_EndGame ( " Received signon %i when at %i " , i , cls . signon ) ;
cls . signon = i ;
CLNQ_SignonReply ( ) ;
break ;
case svc_setpause :
cl . paused = MSG_ReadByte ( ) ;
if ( cl . paused )
CDAudio_Pause ( ) ;
else
CDAudio_Resume ( ) ;
break ;
case svc_spawnstaticsound :
CL_ParseStaticSound ( ) ;
break ;
case svc_spawnstatic :
CL_ParseStatic ( 1 ) ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_spawnbaseline :
i = MSG_ReadShort ( ) ;
2006-07-24 04:24:41 +00:00
if ( ! CL_CheckBaselines ( i ) )
Host_EndGame ( " CLNQ_ParseServerMessage: svc_spawnbaseline failed with size %i " , i ) ;
CL_ParseBaseline ( cl_baselines + i ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_time :
2005-10-01 03:09:17 +00:00
cls . netchan . outgoing_sequence + + ;
cls . netchan . incoming_sequence = cls . netchan . outgoing_sequence - 1 ;
cl . validsequence = cls . netchan . incoming_sequence - 1 ;
2004-08-23 00:15:46 +00:00
received_framecount = host_framecount ;
cl . last_servermessage = realtime ;
2005-10-01 03:09:17 +00:00
2005-05-26 12:55:34 +00:00
cl . oldgametime = cl . gametime ;
2005-06-04 04:20:20 +00:00
cl . oldgametimemark = cl . gametimemark ;
2004-08-23 00:15:46 +00:00
cl . gametime = MSG_ReadFloat ( ) ;
cl . gametimemark = realtime ;
2005-10-01 03:09:17 +00:00
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
if ( ! CPNQ_IS_DP )
2004-11-27 08:16:25 +00:00
{
// cl.frames[(cls.netchan.incoming_sequence-1)&UPDATE_MASK].packet_entities = cl.frames[cls.netchan.incoming_sequence&UPDATE_MASK].packet_entities;
cl . frames [ cls . netchan . incoming_sequence & UPDATE_MASK ] . packet_entities . num_entities = 0 ;
2005-11-26 03:02:55 +00:00
cl . frames [ cls . netchan . incoming_sequence & UPDATE_MASK ] . packet_entities . servertime = cl . gametime ;
2004-11-27 08:16:25 +00:00
}
2004-08-23 00:15:46 +00:00
break ;
case svc_updatename :
Sbar_Changed ( ) ;
i = MSG_ReadByte ( ) ;
2005-05-17 02:36:54 +00:00
if ( i > = MAX_CLIENTS )
2006-01-28 19:04:13 +00:00
MSG_ReadString ( ) ;
else
2009-03-03 01:52:30 +00:00
{
2006-01-28 19:04:13 +00:00
strcpy ( cl . players [ i ] . name , MSG_ReadString ( ) ) ;
2009-03-03 01:52:30 +00:00
if ( * cl . players [ i ] . name )
cl . players [ i ] . userid = i + 1 ;
}
2004-08-23 00:15:46 +00:00
break ;
case svc_updatefrags :
Sbar_Changed ( ) ;
i = MSG_ReadByte ( ) ;
2005-05-17 02:36:54 +00:00
if ( i > = MAX_CLIENTS )
2006-01-28 19:04:13 +00:00
MSG_ReadShort ( ) ;
else
cl . players [ i ] . frags = MSG_ReadShort ( ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_updatecolors :
{
2009-03-03 01:52:30 +00:00
int a ;
i = MSG_ReadByte ( ) ;
a = MSG_ReadByte ( ) ;
if ( i < MAX_CLIENTS )
{
cl . players [ i ] . rtopcolor = a & 0x0f ;
cl . players [ i ] . rbottomcolor = ( a & 0xf0 ) > > 4 ;
2004-08-23 00:15:46 +00:00
2009-03-03 01:52:30 +00:00
sprintf ( cl . players [ i ] . team , " %2d " , cl . players [ i ] . rbottomcolor ) ;
2004-08-23 00:15:46 +00:00
2009-03-03 01:52:30 +00:00
if ( cls . state = = ca_active )
Skin_Find ( & cl . players [ i ] ) ;
2004-08-23 00:15:46 +00:00
2009-04-01 22:03:56 +00:00
if ( i = = cl . playernum [ 0 ] )
Skin_FlushPlayers ( ) ;
2009-03-03 01:52:30 +00:00
Sbar_Changed ( ) ;
CL_NewTranslation ( i ) ;
}
2004-08-23 00:15:46 +00:00
}
break ;
case svc_lightstyle :
i = MSG_ReadByte ( ) ;
if ( i > = MAX_LIGHTSTYLES )
2005-10-01 03:09:17 +00:00
{
Con_Printf ( " svc_lightstyle: %i >= MAX_LIGHTSTYLES \n " , i ) ;
MSG_ReadString ( ) ;
break ;
}
2004-08-23 00:15:46 +00:00
cl_lightstyle [ i ] . colour = 7 ; //white
Q_strncpyz ( cl_lightstyle [ i ] . map , MSG_ReadString ( ) , sizeof ( cl_lightstyle [ i ] . map ) ) ;
cl_lightstyle [ i ] . length = Q_strlen ( cl_lightstyle [ i ] . map ) ;
break ;
case svc_updatestat :
2005-09-26 03:40:09 +00:00
i = MSG_ReadByte ( ) ;
2004-08-23 00:15:46 +00:00
j = MSG_ReadLong ( ) ;
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , i , j ) ;
CL_SetStatFloat ( 0 , i , j ) ;
2004-08-23 00:15:46 +00:00
break ;
2005-09-26 03:40:09 +00:00
case svcdp_updatestatbyte :
i = MSG_ReadByte ( ) ;
2004-12-05 08:19:54 +00:00
j = MSG_ReadByte ( ) ;
2008-05-25 22:23:43 +00:00
CL_SetStatInt ( 0 , i , j ) ;
CL_SetStatFloat ( 0 , i , j ) ;
2004-12-05 08:19:54 +00:00
break ;
2004-08-23 00:15:46 +00:00
case svc_setangle :
for ( i = 0 ; i < 3 ; i + + )
cl . viewangles [ 0 ] [ i ] = MSG_ReadAngle ( ) ;
// cl.viewangles[PITCH] = cl.viewangles[ROLL] = 0;
break ;
case svc_clientdata :
2005-09-26 03:40:09 +00:00
CLNQ_ParseClientdata ( ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_sound :
CLNQ_ParseStartSoundPacket ( ) ;
break ;
case svc_temp_entity :
CL_ParseTEnt ( true ) ;
break ;
2005-09-26 03:40:09 +00:00
2004-08-23 00:15:46 +00:00
case svc_particle :
2005-03-10 03:55:18 +00:00
CLNQ_ParseParticleEffect ( ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_killedmonster :
cl . stats [ 0 ] [ STAT_MONSTERS ] + + ;
break ;
case svc_foundsecret :
cl . stats [ 0 ] [ STAT_SECRETS ] + + ;
break ;
case svc_intermission :
2005-07-14 01:57:34 +00:00
if ( ! cl . intermission )
TP_ExecTrigger ( " f_mapend " ) ;
2004-08-23 00:15:46 +00:00
cl . intermission = 1 ;
2005-11-26 03:02:55 +00:00
cl . completed_time = cl . servertime ;
2004-08-23 00:15:46 +00:00
vid . recalc_refdef = true ; // go to full screen
break ;
case svc_finale :
cl . intermission = 2 ;
2005-11-26 03:02:55 +00:00
cl . completed_time = cl . servertime ;
2004-08-23 00:15:46 +00:00
vid . recalc_refdef = true ; // go to full screen
2008-11-09 22:29:28 +00:00
SCR_CenterPrint ( 0 , MSG_ReadString ( ) , false ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_cutscene :
cl . intermission = 3 ;
2005-11-26 03:02:55 +00:00
cl . completed_time = cl . servertime ;
2004-08-23 00:15:46 +00:00
vid . recalc_refdef = true ; // go to full screen
2008-11-09 22:29:28 +00:00
SCR_CenterPrint ( 0 , MSG_ReadString ( ) , false ) ;
2004-08-23 00:15:46 +00:00
break ;
case svc_sellscreen : //pantsie
Cmd_ExecuteString ( " help 0 " , RESTRICT_RCON ) ;
break ;
case svc_damage :
V_ParseDamage ( 0 ) ;
break ;
Fixes, workarounds, and breakages. Hexen2 should work much better (-hexen2 says no mission pack, -portals says h2mp). Started working on splitting bigcoords per client, far too much work still to go on that. Removed gl_ztrick entirely. Enabled csprogs download by default. Added client support for fitzquake's 666 protocol, needs testing, some cleanup for dp protocols too, no server support, couldn't selectively enable it anyway. Now attempting to cache shadow meshes for explosions and stuff. Played with lightmaps a little, should potentially run a little faster on certain (intel?) cards. Tweeked npfte a little to try to avoid deadlocks and crashes. Fixed sky worldspawn parsing. Added h2mp's model format. Fixed baseline issue in q2 client, made servers generate q2 baselines. MOVETYPE_PUSH will not rotate extra if rotation is forced. Made status command show allowed client types. Changed lighting on weapons - should now be shaded.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/branches/wip@3572 fc73d0e0-1445-4013-8a0c-d673dee63da5
2010-08-11 03:36:31 +00:00
case svcfitz_skybox :
{
extern cvar_t r_skyboxname ;
Cvar_Set ( & r_skyboxname , MSG_ReadString ( ) ) ;
}
break ;
case svcfitz_bf :
Cmd_ExecuteString ( " bf " , RESTRICT_RCON ) ;
break ;
case svcfitz_fog :
/*density =*/ MSG_ReadByte ( ) ;
/*red =*/ MSG_ReadByte ( ) ;
/*green =*/ MSG_ReadByte ( ) ;
/*blue =*/ MSG_ReadByte ( ) ;
/*time =*/ MSG_ReadShort ( ) ;
break ;
case svcfitz_spawnbaseline2 :
i = MSG_ReadShort ( ) ;
if ( ! CL_CheckBaselines ( i ) )
Host_EndGame ( " CLNQ_ParseServerMessage: svcfitz_spawnbaseline2 failed with ent %i " , i ) ;
CLFitz_ParseBaseline2 ( cl_baselines + i ) ;
break ;
// case svcfitz_spawnstatic2:
// break;
// case svcfitz_spawnstaticsound2:
// break;
2004-11-20 01:21:17 +00:00
case svcnq_effect :
CL_ParseEffect ( false ) ;
break ;
case svcnq_effect2 :
CL_ParseEffect ( true ) ;
break ;
2004-11-27 08:16:25 +00:00
2007-06-20 00:02:54 +00:00
case svcdp_entities :
2004-12-05 08:19:54 +00:00
if ( cls . signon = = 4 - 1 )
{ // first update is the final signon stage
cls . signon = 4 ;
CLNQ_SignonReply ( ) ;
2005-09-26 03:40:09 +00:00
}
2004-11-27 08:16:25 +00:00
//well, it's really any protocol, but we're only going to support version 5.
CLNQ_ParseDarkPlaces5Entities ( ) ;
break ;
2007-06-20 00:02:54 +00:00
2009-07-18 20:46:42 +00:00
# ifdef PEXT_CSQC
2007-06-20 00:02:54 +00:00
case svcdp_csqcentities :
CSQC_ParseEntities ( ) ;
break ;
2009-07-18 20:46:42 +00:00
# endif
2007-06-20 00:02:54 +00:00
case svcdp_downloaddata :
CLDP_ParseDownloadData ( ) ;
break ;
2008-11-09 22:29:28 +00:00
case svcdp_trailparticles :
CLDP_ParseTrailParticles ( ) ;
break ;
case svcdp_pointparticles :
CLDP_ParsePointParticles ( false ) ;
break ;
case svcdp_pointparticles1 :
CLDP_ParsePointParticles ( true ) ;
break ;
2004-08-23 00:15:46 +00:00
}
2007-06-20 00:02:54 +00:00
2004-08-23 00:15:46 +00:00
}
}
# endif