hlsdk-portable/dlls/multiplay_gamerules.cpp

1692 lines
41 KiB
C++
Raw Normal View History

2016-06-04 13:24:23 +00:00
/***
*
* Copyright (c) 1996-2002, Valve LLC. All rights reserved.
*
* This product contains software technology licensed from Id
* Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc.
* All Rights Reserved.
*
* Use, distribution, and modification of this source code and/or resulting
* object code is restricted to non-commercial enhancements to products from
* Valve LLC. All other use, distribution, or modification is prohibited
* without written permission from Valve LLC.
*
****/
//
// teamplay_gamerules.cpp
//
2016-06-04 13:24:23 +00:00
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "player.h"
#include "weapons.h"
#include "gamerules.h"
#include "skill.h"
#include "game.h"
#include "items.h"
2016-08-01 17:18:54 +00:00
#ifndef NO_VOICEGAMEMGR
2016-06-04 13:24:23 +00:00
#include "voice_gamemgr.h"
2016-08-01 17:18:54 +00:00
#endif
2016-06-04 13:24:23 +00:00
#include "hltv.h"
2016-07-31 13:48:50 +00:00
extern DLL_GLOBAL CGameRules *g_pGameRules;
2016-06-04 13:24:23 +00:00
extern DLL_GLOBAL BOOL g_fGameOver;
extern int gmsgDeathMsg; // client dll messages
extern int gmsgScoreInfo;
extern int gmsgMOTD;
extern int gmsgServerName;
extern int g_teamplay;
#define ITEM_RESPAWN_TIME 30
#define WEAPON_RESPAWN_TIME 20
#define AMMO_RESPAWN_TIME 20
float g_flIntermissionStartTime = 0;
2016-06-12 20:43:32 +00:00
#ifndef NO_VOICEGAMEMGR
2016-06-04 13:24:23 +00:00
CVoiceGameMgr g_VoiceGameMgr;
class CMultiplayGameMgrHelper : public IVoiceGameMgrHelper
{
public:
2016-07-31 13:48:50 +00:00
virtual bool CanPlayerHearPlayer(CBasePlayer *pListener, CBasePlayer *pTalker)
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( g_teamplay )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( g_pGameRules->PlayerRelationship( pListener, pTalker ) != GR_TEAMMATE )
2016-06-04 13:24:23 +00:00
{
return false;
}
}
return true;
}
};
static CMultiplayGameMgrHelper g_GameMgrHelper;
2016-08-01 17:18:54 +00:00
#endif
2016-06-04 13:24:23 +00:00
//*********************************************************
// Rules for the half-life multiplayer game.
//*********************************************************
2016-07-31 13:48:50 +00:00
CHalfLifeMultiplay::CHalfLifeMultiplay()
2016-06-04 13:24:23 +00:00
{
2016-06-12 20:43:32 +00:00
#ifndef NO_VOICEGAMEMGR
2016-07-31 13:48:50 +00:00
g_VoiceGameMgr.Init( &g_GameMgrHelper, gpGlobals->maxClients );
2016-06-12 20:43:32 +00:00
#endif
2016-06-04 13:24:23 +00:00
RefreshSkillData();
m_flIntermissionEndTime = 0;
g_flIntermissionStartTime = 0;
// 11/8/98
// Modified by YWB: Server .cfg file is now a cvar, so that
// server ops can run multiple game servers, with different server .cfg files,
// from a single installed directory.
// Mapcyclefile is already a cvar.
// 3/31/99
// Added lservercfg file cvar, since listen and dedicated servers should not
// share a single config file. (sjb)
2016-07-31 13:48:50 +00:00
if( IS_DEDICATED_SERVER() )
2016-06-04 13:24:23 +00:00
{
// dedicated server
2017-07-23 21:24:55 +00:00
/*const char *servercfgfile = CVAR_GET_STRING( "servercfgfile" );
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( servercfgfile && servercfgfile[0] )
2016-06-04 13:24:23 +00:00
{
char szCommand[256];
ALERT( at_console, "Executing dedicated server config file\n" );
sprintf( szCommand, "exec %s\n", servercfgfile );
SERVER_COMMAND( szCommand );
}
*/
// this code has been moved into engine, to only run server.cfg once
2016-06-04 13:24:23 +00:00
}
else
{
// listen server
2017-07-23 21:24:55 +00:00
const char *lservercfgfile = CVAR_GET_STRING( "lservercfgfile" );
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( lservercfgfile && lservercfgfile[0] )
2016-06-04 13:24:23 +00:00
{
char szCommand[256];
ALERT( at_console, "Executing listen server config file\n" );
sprintf( szCommand, "exec %s\n", lservercfgfile );
SERVER_COMMAND( szCommand );
}
}
}
BOOL CHalfLifeMultiplay::ClientCommand( CBasePlayer *pPlayer, const char *pcmd )
{
2016-06-12 20:43:32 +00:00
#ifndef NO_VOICEGAMEMGR
2016-07-31 13:48:50 +00:00
if( g_VoiceGameMgr.ClientCommand( pPlayer, pcmd ) )
2016-06-04 13:24:23 +00:00
return TRUE;
2016-06-12 20:43:32 +00:00
#endif
2016-07-31 13:48:50 +00:00
return CGameRules::ClientCommand( pPlayer, pcmd );
2016-06-04 13:24:23 +00:00
}
//=========================================================
//=========================================================
void CHalfLifeMultiplay::RefreshSkillData( void )
{
// load all default values
2016-06-04 13:24:23 +00:00
CGameRules::RefreshSkillData();
// override some values for multiplay.
2016-06-04 13:24:23 +00:00
// suitcharger
gSkillData.suitchargerCapacity = 30;
// Crowbar whack
gSkillData.plrDmgCrowbar = 25;
// Glock Round
gSkillData.plrDmg9MM = 12;
// 357 Round
gSkillData.plrDmg357 = 40;
// MP5 Round
gSkillData.plrDmgMP5 = 12;
// M203 grenade
gSkillData.plrDmgM203Grenade = 100;
// Shotgun buckshot
gSkillData.plrDmgBuckshot = 20;// fewer pellets in deathmatch
// Crossbow
gSkillData.plrDmgCrossbowClient = 20;
// RPG
gSkillData.plrDmgRPG = 120;
// Egon
gSkillData.plrDmgEgonWide = 20;
gSkillData.plrDmgEgonNarrow = 10;
// Hand Grendade
gSkillData.plrDmgHandGrenade = 100;
// Satchel Charge
gSkillData.plrDmgSatchel = 120;
// Tripmine
gSkillData.plrDmgTripmine = 150;
// hornet
gSkillData.plrDmgHornet = 10;
}
// longest the intermission can last, in seconds
#define MAX_INTERMISSION_TIME 120
extern cvar_t timeleft, fragsleft;
extern cvar_t mp_chattime;
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::Think( void )
2016-06-04 13:24:23 +00:00
{
2016-06-12 20:43:32 +00:00
#ifndef NO_VOICEGAMEMGR
2016-07-31 13:48:50 +00:00
g_VoiceGameMgr.Update( gpGlobals->frametime );
2016-06-12 20:43:32 +00:00
#endif
2016-06-04 13:24:23 +00:00
///// Check game rules /////
static int last_frags;
static int last_time;
int frags_remaining = 0;
int time_remaining = 0;
2016-07-31 13:48:50 +00:00
if( g_fGameOver ) // someone else quit the game already
2016-06-04 13:24:23 +00:00
{
// bounds check
int time = (int)CVAR_GET_FLOAT( "mp_chattime" );
2016-07-31 13:48:50 +00:00
if( time < 1 )
2016-06-04 13:24:23 +00:00
CVAR_SET_STRING( "mp_chattime", "1" );
2016-07-31 13:48:50 +00:00
else if( time > MAX_INTERMISSION_TIME )
2016-06-04 13:24:23 +00:00
CVAR_SET_STRING( "mp_chattime", UTIL_dtos1( MAX_INTERMISSION_TIME ) );
m_flIntermissionEndTime = g_flIntermissionStartTime + mp_chattime.value;
// check to see if we should change levels now
2016-07-31 13:48:50 +00:00
if( m_flIntermissionEndTime < gpGlobals->time )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( m_iEndIntermissionButtonHit // check that someone has pressed a key, or the max intermission time is over
|| ( ( g_flIntermissionStartTime + MAX_INTERMISSION_TIME ) < gpGlobals->time ) )
2016-06-04 13:24:23 +00:00
ChangeLevel(); // intermission is over
}
return;
}
float flTimeLimit = timelimit.value * 60;
float flFragLimit = fraglimit.value;
2016-07-31 13:48:50 +00:00
time_remaining = (int)( flTimeLimit ? ( flTimeLimit - gpGlobals->time ) : 0);
if( flTimeLimit != 0 && gpGlobals->time >= flTimeLimit )
2016-06-04 13:24:23 +00:00
{
GoToIntermission();
return;
}
2016-07-31 13:48:50 +00:00
if( flFragLimit )
2016-06-04 13:24:23 +00:00
{
int bestfrags = 9999;
int remain;
// check if any player is over the frag limit
2016-07-31 13:48:50 +00:00
for( int i = 1; i <= gpGlobals->maxClients; i++ )
2016-06-04 13:24:23 +00:00
{
CBaseEntity *pPlayer = UTIL_PlayerByIndex( i );
2016-07-31 13:48:50 +00:00
if( pPlayer && pPlayer->pev->frags >= flFragLimit )
2016-06-04 13:24:23 +00:00
{
GoToIntermission();
return;
}
2016-07-31 13:48:50 +00:00
if( pPlayer )
2016-06-04 13:24:23 +00:00
{
2017-06-29 13:56:03 +00:00
remain = (int)( flFragLimit - pPlayer->pev->frags );
2016-07-31 13:48:50 +00:00
if( remain < bestfrags )
2016-06-04 13:24:23 +00:00
{
bestfrags = remain;
}
}
}
frags_remaining = bestfrags;
}
// Updates when frags change
2016-07-31 13:48:50 +00:00
if( frags_remaining != last_frags )
2016-06-04 13:24:23 +00:00
{
g_engfuncs.pfnCvar_DirectSet( &fragsleft, UTIL_VarArgs( "%i", frags_remaining ) );
}
// Updates once per second
2016-07-31 13:48:50 +00:00
if( timeleft.value != last_time )
2016-06-04 13:24:23 +00:00
{
g_engfuncs.pfnCvar_DirectSet( &timeleft, UTIL_VarArgs( "%i", time_remaining ) );
}
last_frags = frags_remaining;
2016-07-31 13:48:50 +00:00
last_time = time_remaining;
2016-06-04 13:24:23 +00:00
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::IsMultiplayer( void )
{
return TRUE;
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::IsDeathmatch( void )
{
return TRUE;
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::IsCoOp( void )
{
2017-06-29 13:56:03 +00:00
return gpGlobals->coop ? TRUE : FALSE;
2016-06-04 13:24:23 +00:00
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::FShouldSwitchWeapon( CBasePlayer *pPlayer, CBasePlayerItem *pWeapon )
{
2016-07-31 13:48:50 +00:00
if( !pWeapon->CanDeploy() )
2016-06-04 13:24:23 +00:00
{
// that weapon can't deploy anyway.
return FALSE;
}
2016-07-31 13:48:50 +00:00
if( !pPlayer->m_pActiveItem )
2016-06-04 13:24:23 +00:00
{
// player doesn't have an active item!
return TRUE;
}
2016-07-31 13:48:50 +00:00
if( !pPlayer->m_pActiveItem->CanHolster() )
2016-06-04 13:24:23 +00:00
{
// can't put away the active item.
return FALSE;
}
2016-07-31 13:48:50 +00:00
if( pWeapon->iWeight() > pPlayer->m_pActiveItem->iWeight() )
2016-06-04 13:24:23 +00:00
{
return TRUE;
}
return FALSE;
}
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::GetNextBestWeapon( CBasePlayer *pPlayer, CBasePlayerItem *pCurrentWeapon )
2016-06-04 13:24:23 +00:00
{
CBasePlayerItem *pCheck;
CBasePlayerItem *pBest;// this will be used in the event that we don't find a weapon in the same category.
int iBestWeight;
int i;
iBestWeight = -1;// no weapon lower than -1 can be autoswitched to
pBest = NULL;
2016-07-31 13:48:50 +00:00
if( !pCurrentWeapon->CanHolster() )
2016-06-04 13:24:23 +00:00
{
// can't put this gun away right now, so can't switch.
return FALSE;
}
2016-07-31 13:48:50 +00:00
for( i = 0; i < MAX_ITEM_TYPES; i++ )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
pCheck = pPlayer->m_rgpPlayerItems[i];
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
while( pCheck )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( pCheck->iWeight() > -1 && pCheck->iWeight() == pCurrentWeapon->iWeight() && pCheck != pCurrentWeapon )
2016-06-04 13:24:23 +00:00
{
// this weapon is from the same category.
if ( pCheck->CanDeploy() )
{
if ( pPlayer->SwitchWeapon( pCheck ) )
{
return TRUE;
}
}
}
2016-07-31 13:48:50 +00:00
else if( pCheck->iWeight() > iBestWeight && pCheck != pCurrentWeapon )// don't reselect the weapon we're trying to get rid of
2016-06-04 13:24:23 +00:00
{
//ALERT ( at_console, "Considering %s\n", STRING( pCheck->pev->classname ) );
// we keep updating the 'best' weapon just in case we can't find a weapon of the same weight
// that the player was using. This will end up leaving the player with his heaviest-weighted
// weapon.
2016-07-31 13:48:50 +00:00
if( pCheck->CanDeploy() )
2016-06-04 13:24:23 +00:00
{
// if this weapon is useable, flag it as the best
iBestWeight = pCheck->iWeight();
pBest = pCheck;
}
}
pCheck = pCheck->m_pNext;
}
}
// if we make it here, we've checked all the weapons and found no useable
// weapon in the same catagory as the current weapon.
// if pBest is null, we didn't find ANYTHING. Shouldn't be possible- should always
// at least get the crowbar, but ya never know.
2016-07-31 13:48:50 +00:00
if( !pBest )
2016-06-04 13:24:23 +00:00
{
return FALSE;
}
pPlayer->SwitchWeapon( pBest );
return TRUE;
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::ClientConnected( edict_t *pEntity, const char *pszName, const char *pszAddress, char szRejectReason[128] )
2016-06-04 13:24:23 +00:00
{
2016-06-12 20:43:32 +00:00
#ifndef NO_VOICEGAMEMGR
2016-07-31 13:48:50 +00:00
g_VoiceGameMgr.ClientConnected( pEntity );
2016-06-12 20:43:32 +00:00
#endif
2016-06-04 13:24:23 +00:00
return TRUE;
}
extern int gmsgSayText;
extern int gmsgGameMode;
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::UpdateGameMode( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
MESSAGE_BEGIN( MSG_ONE, gmsgGameMode, NULL, pPlayer->edict() );
WRITE_BYTE( 0 ); // game mode none
MESSAGE_END();
}
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::InitHUD( CBasePlayer *pl )
2016-06-04 13:24:23 +00:00
{
// notify other clients of player joining the game
UTIL_ClientPrintAll( HUD_PRINTNOTIFY, UTIL_VarArgs( "%s has joined the game\n",
2017-07-20 18:32:06 +00:00
( pl->pev->netname && ( STRING( pl->pev->netname ) )[0] != 0 ) ? STRING( pl->pev->netname ) : "unconnected" ) );
2016-06-04 13:24:23 +00:00
// team match?
2016-07-31 13:48:50 +00:00
if( g_teamplay )
2016-06-04 13:24:23 +00:00
{
UTIL_LogPrintf( "\"%s<%i><%s><%s>\" entered the game\n",
STRING( pl->pev->netname ),
GETPLAYERUSERID( pl->edict() ),
GETPLAYERAUTHID( pl->edict() ),
g_engfuncs.pfnInfoKeyValue( g_engfuncs.pfnGetInfoKeyBuffer( pl->edict() ), "model" ) );
}
else
{
UTIL_LogPrintf( "\"%s<%i><%s><%i>\" entered the game\n",
STRING( pl->pev->netname ),
GETPLAYERUSERID( pl->edict() ),
GETPLAYERAUTHID( pl->edict() ),
GETPLAYERUSERID( pl->edict() ) );
}
UpdateGameMode( pl );
// sending just one score makes the hud scoreboard active; otherwise
// it is just disabled for single play
MESSAGE_BEGIN( MSG_ONE, gmsgScoreInfo, NULL, pl->edict() );
WRITE_BYTE( ENTINDEX(pl->edict()) );
WRITE_SHORT( 0 );
WRITE_SHORT( 0 );
WRITE_SHORT( 0 );
WRITE_SHORT( 0 );
MESSAGE_END();
SendMOTDToClient( pl->edict() );
// loop through all active players and send their score info to the new client
2016-07-31 13:48:50 +00:00
for( int i = 1; i <= gpGlobals->maxClients; i++ )
2016-06-04 13:24:23 +00:00
{
// FIXME: Probably don't need to cast this just to read m_iDeaths
CBasePlayer *plr = (CBasePlayer *)UTIL_PlayerByIndex( i );
2016-07-31 13:48:50 +00:00
if( plr )
2016-06-04 13:24:23 +00:00
{
MESSAGE_BEGIN( MSG_ONE, gmsgScoreInfo, NULL, pl->edict() );
WRITE_BYTE( i ); // client number
2017-06-29 13:56:03 +00:00
WRITE_SHORT( (int)plr->pev->frags );
2016-06-04 13:24:23 +00:00
WRITE_SHORT( plr->m_iDeaths );
WRITE_SHORT( 0 );
WRITE_SHORT( GetTeamIndex( plr->m_szTeamName ) + 1 );
MESSAGE_END();
}
}
2016-07-31 13:48:50 +00:00
if( g_fGameOver )
2016-06-04 13:24:23 +00:00
{
MESSAGE_BEGIN( MSG_ONE, SVC_INTERMISSION, NULL, pl->edict() );
MESSAGE_END();
}
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::ClientDisconnected( edict_t *pClient )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( pClient )
2016-06-04 13:24:23 +00:00
{
CBasePlayer *pPlayer = (CBasePlayer *)CBaseEntity::Instance( pClient );
2016-07-31 13:48:50 +00:00
if( pPlayer )
2016-06-04 13:24:23 +00:00
{
FireTargets( "game_playerleave", pPlayer, pPlayer, USE_TOGGLE, 0 );
// team match?
2016-07-31 13:48:50 +00:00
if( g_teamplay )
2016-06-04 13:24:23 +00:00
{
UTIL_LogPrintf( "\"%s<%i><%s><%s>\" disconnected\n",
STRING( pPlayer->pev->netname ),
GETPLAYERUSERID( pPlayer->edict() ),
GETPLAYERAUTHID( pPlayer->edict() ),
g_engfuncs.pfnInfoKeyValue( g_engfuncs.pfnGetInfoKeyBuffer( pPlayer->edict() ), "model" ) );
}
else
{
UTIL_LogPrintf( "\"%s<%i><%s><%i>\" disconnected\n",
STRING( pPlayer->pev->netname ),
GETPLAYERUSERID( pPlayer->edict() ),
GETPLAYERAUTHID( pPlayer->edict() ),
GETPLAYERUSERID( pPlayer->edict() ) );
}
pPlayer->RemoveAllItems( TRUE );// destroy all of the players weapons and items
}
}
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
float CHalfLifeMultiplay::FlPlayerFallDamage( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
int iFallDamage = (int)falldamage.value;
2016-07-31 13:48:50 +00:00
switch( iFallDamage )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
case 1:
//progressive
2016-06-04 13:24:23 +00:00
pPlayer->m_flFallVelocity -= PLAYER_MAX_SAFE_FALL_SPEED;
return pPlayer->m_flFallVelocity * DAMAGE_FOR_FALL_SPEED;
break;
default:
2016-07-31 13:48:50 +00:00
case 0:
// fixed
2016-06-04 13:24:23 +00:00
return 10;
break;
}
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::FPlayerCanTakeDamage( CBasePlayer *pPlayer, CBaseEntity *pAttacker )
{
return TRUE;
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::PlayerThink( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( g_fGameOver )
2016-06-04 13:24:23 +00:00
{
// check for button presses
2016-07-31 13:48:50 +00:00
if( pPlayer->m_afButtonPressed & ( IN_DUCK | IN_ATTACK | IN_ATTACK2 | IN_USE | IN_JUMP ) )
2016-06-04 13:24:23 +00:00
m_iEndIntermissionButtonHit = TRUE;
// clear attack/use commands from player
pPlayer->m_afButtonPressed = 0;
pPlayer->pev->button = 0;
pPlayer->m_afButtonReleased = 0;
}
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::PlayerSpawn( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
BOOL addDefault;
CBaseEntity *pWeaponEntity = NULL;
2016-07-31 13:48:50 +00:00
pPlayer->pev->weapons |= ( 1 << WEAPON_SUIT );
2016-06-04 13:24:23 +00:00
addDefault = TRUE;
2016-07-31 13:48:50 +00:00
while( ( pWeaponEntity = UTIL_FindEntityByClassname( pWeaponEntity, "game_player_equip" ) ) )
2016-06-04 13:24:23 +00:00
{
pWeaponEntity->Touch( pPlayer );
addDefault = FALSE;
}
2016-07-31 13:48:50 +00:00
if( addDefault )
2016-06-04 13:24:23 +00:00
{
pPlayer->GiveNamedItem( "weapon_crowbar" );
pPlayer->GiveNamedItem( "weapon_9mmhandgun" );
pPlayer->GiveAmmo( 68, "9mm", _9MM_MAX_CARRY );// 4 full reloads
}
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::FPlayerCanRespawn( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
return TRUE;
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
float CHalfLifeMultiplay::FlPlayerSpawnTime( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
return gpGlobals->time;//now!
}
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::AllowAutoTargetCrosshair( void )
2016-06-04 13:24:23 +00:00
{
return ( aimcrosshair.value != 0 );
}
//=========================================================
// IPointsForKill - how many points awarded to anyone
// that kills this player?
//=========================================================
2016-07-31 13:48:50 +00:00
int CHalfLifeMultiplay::IPointsForKill( CBasePlayer *pAttacker, CBasePlayer *pKilled )
2016-06-04 13:24:23 +00:00
{
return 1;
}
//=========================================================
// PlayerKilled - someone/something killed this player
//=========================================================
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::PlayerKilled( CBasePlayer *pVictim, entvars_t *pKiller, entvars_t *pInflictor )
2016-06-04 13:24:23 +00:00
{
DeathNotice( pVictim, pKiller, pInflictor );
pVictim->m_iDeaths += 1;
FireTargets( "game_playerdie", pVictim, pVictim, USE_TOGGLE, 0 );
CBasePlayer *peKiller = NULL;
CBaseEntity *ktmp = CBaseEntity::Instance( pKiller );
2016-07-31 13:48:50 +00:00
if( ktmp && (ktmp->Classify() == CLASS_PLAYER ) )
2016-06-04 13:24:23 +00:00
peKiller = (CBasePlayer*)ktmp;
2016-07-31 13:48:50 +00:00
if( pVictim->pev == pKiller )
{
// killed self
2016-06-04 13:24:23 +00:00
pKiller->frags -= 1;
}
2016-07-31 13:48:50 +00:00
else if( ktmp && ktmp->IsPlayer() )
2016-06-04 13:24:23 +00:00
{
// if a player dies in a deathmatch game and the killer is a client, award the killer some points
pKiller->frags += IPointsForKill( peKiller, pVictim );
2016-07-31 13:48:50 +00:00
2016-06-04 13:24:23 +00:00
FireTargets( "game_playerkill", ktmp, ktmp, USE_TOGGLE, 0 );
}
else
{
// killed by the world
2016-06-04 13:24:23 +00:00
pKiller->frags -= 1;
}
// update the scores
// killed scores
MESSAGE_BEGIN( MSG_ALL, gmsgScoreInfo );
WRITE_BYTE( ENTINDEX(pVictim->edict()) );
2017-06-29 13:56:03 +00:00
WRITE_SHORT( (int)pVictim->pev->frags );
2016-06-04 13:24:23 +00:00
WRITE_SHORT( pVictim->m_iDeaths );
WRITE_SHORT( 0 );
WRITE_SHORT( GetTeamIndex( pVictim->m_szTeamName ) + 1 );
MESSAGE_END();
// killers score, if it's a player
CBaseEntity *ep = CBaseEntity::Instance( pKiller );
2016-07-31 13:48:50 +00:00
if( ep && ep->Classify() == CLASS_PLAYER )
2016-06-04 13:24:23 +00:00
{
CBasePlayer *PK = (CBasePlayer*)ep;
MESSAGE_BEGIN( MSG_ALL, gmsgScoreInfo );
2016-07-31 13:48:50 +00:00
WRITE_BYTE( ENTINDEX( PK->edict() ) );
2017-06-29 13:56:03 +00:00
WRITE_SHORT( (int)PK->pev->frags );
2016-06-04 13:24:23 +00:00
WRITE_SHORT( PK->m_iDeaths );
WRITE_SHORT( 0 );
2016-07-31 13:48:50 +00:00
WRITE_SHORT( GetTeamIndex( PK->m_szTeamName ) + 1 );
2016-06-04 13:24:23 +00:00
MESSAGE_END();
// let the killer paint another decal as soon as he'd like.
PK->m_flNextDecalTime = gpGlobals->time;
}
}
//=========================================================
// Deathnotice.
//=========================================================
void CHalfLifeMultiplay::DeathNotice( CBasePlayer *pVictim, entvars_t *pKiller, entvars_t *pevInflictor )
{
// Work out what killed the player, and send a message to all clients about it
2017-06-29 13:56:03 +00:00
CBaseEntity::Instance( pKiller );
2016-06-04 13:24:23 +00:00
const char *killer_weapon_name = "world"; // by default, the player is killed by the world
int killer_index = 0;
2016-07-31 13:48:50 +00:00
2016-06-04 13:24:23 +00:00
// Hack to fix name change
2017-06-29 13:56:03 +00:00
const char *tau = "tau_cannon";
const char *gluon = "gluon gun";
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( pKiller->flags & FL_CLIENT )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
killer_index = ENTINDEX( ENT( pKiller ) );
if( pevInflictor )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( pevInflictor == pKiller )
2016-06-04 13:24:23 +00:00
{
// If the inflictor is the killer, then it must be their current weapon doing the damage
CBasePlayer *pPlayer = (CBasePlayer*)CBaseEntity::Instance( pKiller );
2016-07-31 13:48:50 +00:00
if( pPlayer->m_pActiveItem )
2016-06-04 13:24:23 +00:00
{
killer_weapon_name = pPlayer->m_pActiveItem->pszName();
}
}
else
{
killer_weapon_name = STRING( pevInflictor->classname ); // it's just that easy
}
}
}
else
{
killer_weapon_name = STRING( pevInflictor->classname );
}
// strip the monster_* or weapon_* from the inflictor's classname
2016-07-31 13:48:50 +00:00
if( strncmp( killer_weapon_name, "weapon_", 7 ) == 0 )
2016-06-04 13:24:23 +00:00
killer_weapon_name += 7;
2016-07-31 13:48:50 +00:00
else if( strncmp( killer_weapon_name, "monster_", 8 ) == 0 )
2016-06-04 13:24:23 +00:00
killer_weapon_name += 8;
2016-07-31 13:48:50 +00:00
else if( strncmp( killer_weapon_name, "func_", 5 ) == 0 )
2016-06-04 13:24:23 +00:00
killer_weapon_name += 5;
MESSAGE_BEGIN( MSG_ALL, gmsgDeathMsg );
WRITE_BYTE( killer_index ); // the killer
2016-07-31 13:48:50 +00:00
WRITE_BYTE( ENTINDEX( pVictim->edict() ) ); // the victim
2016-06-04 13:24:23 +00:00
WRITE_STRING( killer_weapon_name ); // what they were killed by (should this be a string?)
MESSAGE_END();
// replace the code names with the 'real' names
2016-07-31 13:48:50 +00:00
if( !strcmp( killer_weapon_name, "egon" ) )
2016-06-04 13:24:23 +00:00
killer_weapon_name = gluon;
2016-07-31 13:48:50 +00:00
else if( !strcmp( killer_weapon_name, "gauss" ) )
2016-06-04 13:24:23 +00:00
killer_weapon_name = tau;
2016-07-31 13:48:50 +00:00
if( pVictim->pev == pKiller )
2016-06-04 13:24:23 +00:00
{
// killed self
// team match?
2016-07-31 13:48:50 +00:00
if( g_teamplay )
2016-06-04 13:24:23 +00:00
{
UTIL_LogPrintf( "\"%s<%i><%s><%s>\" committed suicide with \"%s\"\n",
STRING( pVictim->pev->netname ),
GETPLAYERUSERID( pVictim->edict() ),
GETPLAYERAUTHID( pVictim->edict() ),
g_engfuncs.pfnInfoKeyValue( g_engfuncs.pfnGetInfoKeyBuffer( pVictim->edict() ), "model" ),
killer_weapon_name );
}
else
{
UTIL_LogPrintf( "\"%s<%i><%s><%i>\" committed suicide with \"%s\"\n",
STRING( pVictim->pev->netname ),
GETPLAYERUSERID( pVictim->edict() ),
GETPLAYERAUTHID( pVictim->edict() ),
GETPLAYERUSERID( pVictim->edict() ),
killer_weapon_name );
}
}
2016-07-31 13:48:50 +00:00
else if( pKiller->flags & FL_CLIENT )
2016-06-04 13:24:23 +00:00
{
// team match?
2016-07-31 13:48:50 +00:00
if( g_teamplay )
2016-06-04 13:24:23 +00:00
{
UTIL_LogPrintf( "\"%s<%i><%s><%s>\" killed \"%s<%i><%s><%s>\" with \"%s\"\n",
STRING( pKiller->netname ),
GETPLAYERUSERID( ENT(pKiller) ),
GETPLAYERAUTHID( ENT(pKiller) ),
g_engfuncs.pfnInfoKeyValue( g_engfuncs.pfnGetInfoKeyBuffer( ENT(pKiller) ), "model" ),
STRING( pVictim->pev->netname ),
GETPLAYERUSERID( pVictim->edict() ),
GETPLAYERAUTHID( pVictim->edict() ),
g_engfuncs.pfnInfoKeyValue( g_engfuncs.pfnGetInfoKeyBuffer( pVictim->edict() ), "model" ),
killer_weapon_name );
}
else
{
UTIL_LogPrintf( "\"%s<%i><%s><%i>\" killed \"%s<%i><%s><%i>\" with \"%s\"\n",
STRING( pKiller->netname ),
GETPLAYERUSERID( ENT(pKiller) ),
GETPLAYERAUTHID( ENT(pKiller) ),
GETPLAYERUSERID( ENT(pKiller) ),
STRING( pVictim->pev->netname ),
GETPLAYERUSERID( pVictim->edict() ),
GETPLAYERAUTHID( pVictim->edict() ),
GETPLAYERUSERID( pVictim->edict() ),
killer_weapon_name );
}
}
else
{
// killed by the world
// team match?
2016-07-31 13:48:50 +00:00
if( g_teamplay )
2016-06-04 13:24:23 +00:00
{
UTIL_LogPrintf( "\"%s<%i><%s><%s>\" committed suicide with \"%s\" (world)\n",
STRING( pVictim->pev->netname ),
GETPLAYERUSERID( pVictim->edict() ),
GETPLAYERAUTHID( pVictim->edict() ),
g_engfuncs.pfnInfoKeyValue( g_engfuncs.pfnGetInfoKeyBuffer( pVictim->edict() ), "model" ),
killer_weapon_name );
}
else
{
UTIL_LogPrintf( "\"%s<%i><%s><%i>\" committed suicide with \"%s\" (world)\n",
STRING( pVictim->pev->netname ),
GETPLAYERUSERID( pVictim->edict() ),
GETPLAYERAUTHID( pVictim->edict() ),
GETPLAYERUSERID( pVictim->edict() ),
killer_weapon_name );
}
}
MESSAGE_BEGIN( MSG_SPEC, SVC_DIRECTOR );
2016-07-31 13:48:50 +00:00
WRITE_BYTE( 9 ); // command length in bytes
WRITE_BYTE( DRC_CMD_EVENT ); // player killed
WRITE_SHORT( ENTINDEX( pVictim->edict() ) ); // index number of primary entity
if( pevInflictor )
WRITE_SHORT( ENTINDEX( ENT( pevInflictor ) ) ); // index number of secondary entity
2016-06-04 13:24:23 +00:00
else
2016-07-31 13:48:50 +00:00
WRITE_SHORT( ENTINDEX( ENT( pKiller ) ) ); // index number of secondary entity
WRITE_LONG( 7 | DRC_FLAG_DRAMATIC ); // eventflags (priority and flags)
2016-06-04 13:24:23 +00:00
MESSAGE_END();
// Print a standard message
// TODO: make this go direct to console
return; // just remove for now
/*
2016-07-31 13:48:50 +00:00
char szText[128];
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( pKiller->flags & FL_MONSTER )
2016-06-04 13:24:23 +00:00
{
// killed by a monster
2016-07-31 13:48:50 +00:00
strcpy( szText, STRING( pVictim->pev->netname ) );
strcat( szText, " was killed by a monster.\n" );
2016-06-04 13:24:23 +00:00
return;
}
2016-07-31 13:48:50 +00:00
if( pKiller == pVictim->pev )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
strcpy( szText, STRING( pVictim->pev->netname ) );
strcat( szText, " commited suicide.\n" );
2016-06-04 13:24:23 +00:00
}
2016-07-31 13:48:50 +00:00
else if( pKiller->flags & FL_CLIENT )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
strcpy( szText, STRING( pKiller->netname ) );
2016-06-04 13:24:23 +00:00
strcat( szText, " : " );
strcat( szText, killer_weapon_name );
strcat( szText, " : " );
2016-07-31 13:48:50 +00:00
strcat( szText, STRING( pVictim->pev->netname ) );
strcat( szText, "\n" );
2016-06-04 13:24:23 +00:00
}
2016-07-31 13:48:50 +00:00
else if( FClassnameIs( pKiller, "worldspawn" ) )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
strcpy( szText, STRING( pVictim->pev->netname ) );
strcat( szText, " fell or drowned or something.\n" );
2016-06-04 13:24:23 +00:00
}
2016-07-31 13:48:50 +00:00
else if( pKiller->solid == SOLID_BSP )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
strcpy( szText, STRING( pVictim->pev->netname ) );
strcat( szText, " was mooshed.\n" );
2016-06-04 13:24:23 +00:00
}
else
{
2016-07-31 13:48:50 +00:00
strcpy( szText, STRING( pVictim->pev->netname ) );
strcat( szText, " died mysteriously.\n" );
2016-06-04 13:24:23 +00:00
}
UTIL_ClientPrintAll( szText );
*/
}
//=========================================================
// PlayerGotWeapon - player has grabbed a weapon that was
// sitting in the world
//=========================================================
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::PlayerGotWeapon( CBasePlayer *pPlayer, CBasePlayerItem *pWeapon )
2016-06-04 13:24:23 +00:00
{
}
//=========================================================
// FlWeaponRespawnTime - what is the time in the future
// at which this weapon may spawn?
//=========================================================
2016-07-31 13:48:50 +00:00
float CHalfLifeMultiplay::FlWeaponRespawnTime( CBasePlayerItem *pWeapon )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( weaponstay.value > 0 )
2016-06-04 13:24:23 +00:00
{
// make sure it's only certain weapons
2016-07-31 13:48:50 +00:00
if( !(pWeapon->iFlags() & ITEM_FLAG_LIMITINWORLD ) )
2016-06-04 13:24:23 +00:00
{
return gpGlobals->time + 0; // weapon respawns almost instantly
}
}
return gpGlobals->time + WEAPON_RESPAWN_TIME;
}
// when we are within this close to running out of entities, items
// marked with the ITEM_FLAG_LIMITINWORLD will delay their respawn
#define ENTITY_INTOLERANCE 100
//=========================================================
// FlWeaponRespawnTime - Returns 0 if the weapon can respawn
// now, otherwise it returns the time at which it can try
// to spawn again.
//=========================================================
2016-07-31 13:48:50 +00:00
float CHalfLifeMultiplay::FlWeaponTryRespawn( CBasePlayerItem *pWeapon )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( pWeapon && pWeapon->m_iId && ( pWeapon->iFlags() & ITEM_FLAG_LIMITINWORLD ) )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( NUMBER_OF_ENTITIES() < ( gpGlobals->maxEntities - ENTITY_INTOLERANCE ) )
2016-06-04 13:24:23 +00:00
return 0;
// we're past the entity tolerance level, so delay the respawn
return FlWeaponRespawnTime( pWeapon );
}
return 0;
}
//=========================================================
// VecWeaponRespawnSpot - where should this weapon spawn?
// Some game variations may choose to randomize spawn locations
//=========================================================
2016-07-31 13:48:50 +00:00
Vector CHalfLifeMultiplay::VecWeaponRespawnSpot( CBasePlayerItem *pWeapon )
2016-06-04 13:24:23 +00:00
{
return pWeapon->pev->origin;
}
//=========================================================
// WeaponShouldRespawn - any conditions inhibiting the
// respawning of this weapon?
//=========================================================
2016-07-31 13:48:50 +00:00
int CHalfLifeMultiplay::WeaponShouldRespawn( CBasePlayerItem *pWeapon )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( pWeapon->pev->spawnflags & SF_NORESPAWN )
2016-06-04 13:24:23 +00:00
{
return GR_WEAPON_RESPAWN_NO;
}
return GR_WEAPON_RESPAWN_YES;
}
//=========================================================
// CanHaveWeapon - returns FALSE if the player is not allowed
// to pick up this weapon
//=========================================================
BOOL CHalfLifeMultiplay::CanHavePlayerItem( CBasePlayer *pPlayer, CBasePlayerItem *pItem )
{
2016-07-31 13:48:50 +00:00
if( weaponstay.value > 0 )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( pItem->iFlags() & ITEM_FLAG_LIMITINWORLD )
2016-06-04 13:24:23 +00:00
return CGameRules::CanHavePlayerItem( pPlayer, pItem );
// check if the player already has this weapon
2016-07-31 13:48:50 +00:00
for( int i = 0; i < MAX_ITEM_TYPES; i++ )
2016-06-04 13:24:23 +00:00
{
CBasePlayerItem *it = pPlayer->m_rgpPlayerItems[i];
2016-07-31 13:48:50 +00:00
while( it != NULL )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( it->m_iId == pItem->m_iId )
2016-06-04 13:24:23 +00:00
{
return FALSE;
}
it = it->m_pNext;
}
}
}
return CGameRules::CanHavePlayerItem( pPlayer, pItem );
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::CanHaveItem( CBasePlayer *pPlayer, CItem *pItem )
{
return TRUE;
}
//=========================================================
//=========================================================
void CHalfLifeMultiplay::PlayerGotItem( CBasePlayer *pPlayer, CItem *pItem )
{
}
//=========================================================
//=========================================================
int CHalfLifeMultiplay::ItemShouldRespawn( CItem *pItem )
{
2016-07-31 13:48:50 +00:00
if( pItem->pev->spawnflags & SF_NORESPAWN )
2016-06-04 13:24:23 +00:00
{
return GR_ITEM_RESPAWN_NO;
}
return GR_ITEM_RESPAWN_YES;
}
//=========================================================
// At what time in the future may this Item respawn?
//=========================================================
float CHalfLifeMultiplay::FlItemRespawnTime( CItem *pItem )
{
return gpGlobals->time + ITEM_RESPAWN_TIME;
}
//=========================================================
// Where should this item respawn?
// Some game variations may choose to randomize spawn locations
//=========================================================
Vector CHalfLifeMultiplay::VecItemRespawnSpot( CItem *pItem )
{
return pItem->pev->origin;
}
//=========================================================
//=========================================================
void CHalfLifeMultiplay::PlayerGotAmmo( CBasePlayer *pPlayer, char *szName, int iCount )
{
}
//=========================================================
//=========================================================
BOOL CHalfLifeMultiplay::IsAllowedToSpawn( CBaseEntity *pEntity )
{
2016-07-31 13:48:50 +00:00
// if( pEntity->pev->flags & FL_MONSTER )
2016-06-04 13:24:23 +00:00
// return FALSE;
return TRUE;
}
//=========================================================
//=========================================================
int CHalfLifeMultiplay::AmmoShouldRespawn( CBasePlayerAmmo *pAmmo )
{
2016-07-31 13:48:50 +00:00
if( pAmmo->pev->spawnflags & SF_NORESPAWN )
2016-06-04 13:24:23 +00:00
{
return GR_AMMO_RESPAWN_NO;
}
return GR_AMMO_RESPAWN_YES;
}
//=========================================================
//=========================================================
float CHalfLifeMultiplay::FlAmmoRespawnTime( CBasePlayerAmmo *pAmmo )
{
return gpGlobals->time + AMMO_RESPAWN_TIME;
}
//=========================================================
//=========================================================
Vector CHalfLifeMultiplay::VecAmmoRespawnSpot( CBasePlayerAmmo *pAmmo )
{
return pAmmo->pev->origin;
}
//=========================================================
//=========================================================
float CHalfLifeMultiplay::FlHealthChargerRechargeTime( void )
{
return 60;
}
float CHalfLifeMultiplay::FlHEVChargerRechargeTime( void )
{
return 30;
}
//=========================================================
//=========================================================
int CHalfLifeMultiplay::DeadPlayerWeapons( CBasePlayer *pPlayer )
{
return GR_PLR_DROP_GUN_ACTIVE;
}
//=========================================================
//=========================================================
int CHalfLifeMultiplay::DeadPlayerAmmo( CBasePlayer *pPlayer )
{
return GR_PLR_DROP_AMMO_ACTIVE;
}
edict_t *CHalfLifeMultiplay::GetPlayerSpawnSpot( CBasePlayer *pPlayer )
{
edict_t *pentSpawnSpot = CGameRules::GetPlayerSpawnSpot( pPlayer );
2016-07-31 13:48:50 +00:00
if( IsMultiplayer() && pentSpawnSpot->v.target )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
FireTargets( STRING( pentSpawnSpot->v.target ), pPlayer, pPlayer, USE_TOGGLE, 0 );
2016-06-04 13:24:23 +00:00
}
return pentSpawnSpot;
}
//=========================================================
//=========================================================
int CHalfLifeMultiplay::PlayerRelationship( CBaseEntity *pPlayer, CBaseEntity *pTarget )
{
// half life deathmatch has only enemies
return GR_NOTTEAMMATE;
}
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::PlayFootstepSounds( CBasePlayer *pl, float fvol )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( g_footsteps && g_footsteps->value == 0 )
2016-06-04 13:24:23 +00:00
return FALSE;
2016-07-31 13:48:50 +00:00
if( pl->IsOnLadder() || pl->pev->velocity.Length2D() > 220 )
2016-06-04 13:24:23 +00:00
return TRUE; // only make step sounds in multiplayer if the player is moving fast enough
return FALSE;
}
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::FAllowFlashlight( void )
{
2016-06-04 13:24:23 +00:00
return flashlight.value != 0;
}
//=========================================================
//=========================================================
2016-07-31 13:48:50 +00:00
BOOL CHalfLifeMultiplay::FAllowMonsters( void )
2016-06-04 13:24:23 +00:00
{
return ( allowmonsters.value != 0 );
}
//=========================================================
//======== CHalfLifeMultiplay private functions ===========
#define INTERMISSION_TIME 6
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::GoToIntermission( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( g_fGameOver )
2016-06-04 13:24:23 +00:00
return; // intermission has already been triggered, so ignore.
2016-07-31 13:48:50 +00:00
MESSAGE_BEGIN( MSG_ALL, SVC_INTERMISSION );
2016-06-04 13:24:23 +00:00
MESSAGE_END();
// bounds check
int time = (int)CVAR_GET_FLOAT( "mp_chattime" );
2016-07-31 13:48:50 +00:00
if( time < 1 )
2016-06-04 13:24:23 +00:00
CVAR_SET_STRING( "mp_chattime", "1" );
2016-07-31 13:48:50 +00:00
else if( time > MAX_INTERMISSION_TIME )
2016-06-04 13:24:23 +00:00
CVAR_SET_STRING( "mp_chattime", UTIL_dtos1( MAX_INTERMISSION_TIME ) );
m_flIntermissionEndTime = gpGlobals->time + ( (int)mp_chattime.value );
g_flIntermissionStartTime = gpGlobals->time;
g_fGameOver = TRUE;
m_iEndIntermissionButtonHit = FALSE;
}
#define MAX_RULE_BUFFER 1024
typedef struct mapcycle_item_s
{
struct mapcycle_item_s *next;
2016-07-31 13:48:50 +00:00
char mapname[32];
int minplayers, maxplayers;
char rulebuffer[MAX_RULE_BUFFER];
2016-06-04 13:24:23 +00:00
} mapcycle_item_t;
typedef struct mapcycle_s
{
struct mapcycle_item_s *items;
struct mapcycle_item_s *next_item;
} mapcycle_t;
/*
==============
DestroyMapCycle
Clean up memory used by mapcycle when switching it
==============
*/
void DestroyMapCycle( mapcycle_t *cycle )
{
mapcycle_item_t *p, *n, *start;
p = cycle->items;
2016-07-31 13:48:50 +00:00
if( p )
2016-06-04 13:24:23 +00:00
{
start = p;
p = p->next;
2016-07-31 13:48:50 +00:00
while( p != start )
2016-06-04 13:24:23 +00:00
{
n = p->next;
delete p;
p = n;
}
delete cycle->items;
}
cycle->items = NULL;
cycle->next_item = NULL;
}
2016-07-31 13:48:50 +00:00
static char com_token[1500];
2016-06-04 13:24:23 +00:00
/*
==============
COM_Parse
Parse a token out of a string
==============
*/
2017-07-23 21:24:55 +00:00
const char *COM_Parse( const char *data )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
int c;
int len;
2016-06-04 13:24:23 +00:00
len = 0;
com_token[0] = 0;
2016-07-31 13:48:50 +00:00
if( !data )
2016-06-04 13:24:23 +00:00
return NULL;
2016-06-04 13:24:23 +00:00
// skip whitespace
skipwhite:
2016-07-31 13:48:50 +00:00
while( ( c = *data ) <= ' ')
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( c == 0 )
2016-06-04 13:24:23 +00:00
return NULL; // end of file;
data++;
}
// skip // comments
2016-07-31 13:48:50 +00:00
if( c=='/' && data[1] == '/' )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
while( *data && *data != '\n' )
2016-06-04 13:24:23 +00:00
data++;
goto skipwhite;
}
// handle quoted strings specially
2016-07-31 13:48:50 +00:00
if( c == '\"' )
2016-06-04 13:24:23 +00:00
{
data++;
2016-07-31 13:48:50 +00:00
while( 1 )
2016-06-04 13:24:23 +00:00
{
c = *data++;
2016-07-31 13:48:50 +00:00
if( c=='\"' || !c )
2016-06-04 13:24:23 +00:00
{
com_token[len] = 0;
return data;
}
com_token[len] = c;
len++;
}
}
// parse single characters
2016-07-31 13:48:50 +00:00
if( c=='{' || c=='}'|| c==')'|| c=='(' || c=='\'' || c == ',' )
2016-06-04 13:24:23 +00:00
{
com_token[len] = c;
len++;
com_token[len] = 0;
2016-07-31 13:48:50 +00:00
return data + 1;
2016-06-04 13:24:23 +00:00
}
// parse a regular word
2016-06-04 13:24:23 +00:00
do
{
com_token[len] = c;
data++;
len++;
c = *data;
2016-07-31 13:48:50 +00:00
if( c=='{' || c=='}'|| c==')'|| c=='(' || c=='\'' || c == ',' )
2016-06-04 13:24:23 +00:00
break;
2016-07-31 13:48:50 +00:00
} while ( c > 32 );
2016-06-04 13:24:23 +00:00
com_token[len] = 0;
return data;
}
/*
==============
COM_TokenWaiting
Returns 1 if additional data is waiting to be processed on this line
==============
*/
2017-07-23 21:24:55 +00:00
int COM_TokenWaiting( const char *buffer )
2016-06-04 13:24:23 +00:00
{
2017-07-23 21:24:55 +00:00
const char *p;
2016-06-04 13:24:23 +00:00
p = buffer;
2016-07-31 13:48:50 +00:00
while( *p && *p!='\n')
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( !isspace( *p ) || isalnum( *p ) )
2016-06-04 13:24:23 +00:00
return 1;
p++;
}
return 0;
}
/*
==============
ReloadMapCycleFile
Parses mapcycle.txt file into mapcycle_t structure
==============
*/
2017-07-23 21:24:55 +00:00
int ReloadMapCycleFile( const char *filename, mapcycle_t *cycle )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
char szMap[32];
2016-06-04 13:24:23 +00:00
int length;
2017-07-23 21:24:55 +00:00
const char *pFileList;
const char *aFileList = pFileList = (const char *)LOAD_FILE_FOR_ME( filename, &length );
2016-06-04 13:24:23 +00:00
int hasbuffer;
mapcycle_item_s *item, *newlist = NULL, *next;
2016-07-31 13:48:50 +00:00
if( pFileList && length )
2016-06-04 13:24:23 +00:00
{
// the first map name in the file becomes the default
2016-07-31 13:48:50 +00:00
while( 1 )
2016-06-04 13:24:23 +00:00
{
2016-08-02 11:59:22 +00:00
char szBuffer[MAX_RULE_BUFFER] = {0};
2016-06-04 13:24:23 +00:00
hasbuffer = 0;
pFileList = COM_Parse( pFileList );
2016-07-31 13:48:50 +00:00
if( strlen( com_token ) <= 0 )
2016-06-04 13:24:23 +00:00
break;
strcpy( szMap, com_token );
// Any more tokens on this line?
2016-07-31 13:48:50 +00:00
if( COM_TokenWaiting( pFileList ) )
2016-06-04 13:24:23 +00:00
{
pFileList = COM_Parse( pFileList );
2016-07-31 13:48:50 +00:00
if( strlen( com_token ) > 0 )
2016-06-04 13:24:23 +00:00
{
hasbuffer = 1;
strcpy( szBuffer, com_token );
}
}
// Check map
2016-07-31 13:48:50 +00:00
if( IS_MAP_VALID( szMap ) )
2016-06-04 13:24:23 +00:00
{
// Create entry
char *s;
item = new mapcycle_item_s;
strcpy( item->mapname, szMap );
item->minplayers = 0;
item->maxplayers = 0;
memset( item->rulebuffer, 0, MAX_RULE_BUFFER );
2016-07-31 13:48:50 +00:00
if( hasbuffer )
2016-06-04 13:24:23 +00:00
{
s = g_engfuncs.pfnInfoKeyValue( szBuffer, "minplayers" );
2016-07-31 13:48:50 +00:00
if( s && s[0] )
2016-06-04 13:24:23 +00:00
{
item->minplayers = atoi( s );
2017-10-14 19:57:55 +00:00
item->minplayers = Q_max( item->minplayers, 0 );
item->minplayers = Q_min( item->minplayers, gpGlobals->maxClients );
2016-06-04 13:24:23 +00:00
}
s = g_engfuncs.pfnInfoKeyValue( szBuffer, "maxplayers" );
2016-07-31 13:48:50 +00:00
if( s && s[0] )
2016-06-04 13:24:23 +00:00
{
item->maxplayers = atoi( s );
2017-10-14 19:57:55 +00:00
item->maxplayers = Q_max( item->maxplayers, 0 );
item->maxplayers = Q_min( item->maxplayers, gpGlobals->maxClients );
2016-06-04 13:24:23 +00:00
}
// Remove keys
//
g_engfuncs.pfnInfo_RemoveKey( szBuffer, "minplayers" );
g_engfuncs.pfnInfo_RemoveKey( szBuffer, "maxplayers" );
strcpy( item->rulebuffer, szBuffer );
}
item->next = cycle->items;
cycle->items = item;
}
else
{
ALERT( at_console, "Skipping %s from mapcycle, not a valid map\n", szMap );
}
}
2017-07-23 21:24:55 +00:00
FREE_FILE( (void*)aFileList );
2016-06-04 13:24:23 +00:00
}
// Fixup circular list pointer
item = cycle->items;
// Reverse it to get original order
2016-07-31 13:48:50 +00:00
while( item )
2016-06-04 13:24:23 +00:00
{
next = item->next;
item->next = newlist;
newlist = item;
item = next;
}
cycle->items = newlist;
item = cycle->items;
// Didn't parse anything
2016-07-31 13:48:50 +00:00
if( !item )
2016-06-04 13:24:23 +00:00
{
return 0;
}
2016-07-31 13:48:50 +00:00
while( item->next )
2016-06-04 13:24:23 +00:00
{
item = item->next;
}
item->next = cycle->items;
2016-06-04 13:24:23 +00:00
cycle->next_item = item->next;
return 1;
}
/*
==============
CountPlayers
Determine the current # of active players on the server for map cycling logic
==============
*/
int CountPlayers( void )
{
2016-07-31 13:48:50 +00:00
int num = 0;
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
for( int i = 1; i <= gpGlobals->maxClients; i++ )
2016-06-04 13:24:23 +00:00
{
CBaseEntity *pEnt = UTIL_PlayerByIndex( i );
2016-07-31 13:48:50 +00:00
if( pEnt )
2016-06-04 13:24:23 +00:00
{
num = num + 1;
}
}
return num;
}
/*
==============
ExtractCommandString
Parse commands/key value pairs to issue right after map xxx command is issued on server
level transition
==============
*/
void ExtractCommandString( char *s, char *szCommand )
{
// Now make rules happen
2016-07-31 13:48:50 +00:00
char pkey[512];
char value[512]; // use two buffers so compares
// work without stomping on each other
2016-07-31 13:48:50 +00:00
char *o;
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( *s == '\\' )
2016-06-04 13:24:23 +00:00
s++;
2016-07-31 13:48:50 +00:00
while( 1 )
2016-06-04 13:24:23 +00:00
{
o = pkey;
2016-07-31 13:48:50 +00:00
while( *s != '\\' )
2016-06-04 13:24:23 +00:00
{
if ( !*s )
return;
*o++ = *s++;
}
*o = 0;
s++;
o = value;
2016-07-31 13:48:50 +00:00
while( *s != '\\' && *s )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( !*s )
2016-06-04 13:24:23 +00:00
return;
*o++ = *s++;
}
*o = 0;
strcat( szCommand, pkey );
2016-07-31 13:48:50 +00:00
if( strlen( value ) > 0 )
2016-06-04 13:24:23 +00:00
{
strcat( szCommand, " " );
strcat( szCommand, value );
}
strcat( szCommand, "\n" );
2016-07-31 13:48:50 +00:00
if( !*s )
2016-06-04 13:24:23 +00:00
return;
s++;
}
}
/*
==============
ChangeLevel
Server is changing to a new level, check mapcycle.txt for map name and setup info
==============
*/
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::ChangeLevel( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
static char szPreviousMapCycleFile[256];
2016-06-04 13:24:23 +00:00
static mapcycle_t mapcycle;
char szNextMap[32];
char szFirstMapInList[32];
2016-07-31 13:48:50 +00:00
char szCommands[1500];
char szRules[1500];
2016-06-04 13:24:23 +00:00
int minplayers = 0, maxplayers = 0;
strcpy( szFirstMapInList, "hldm1" ); // the absolute default level is hldm1
2016-07-31 13:48:50 +00:00
int curplayers;
2016-06-04 13:24:23 +00:00
BOOL do_cycle = TRUE;
// find the map to change to
2017-07-23 21:24:55 +00:00
const char *mapcfile = CVAR_GET_STRING( "mapcyclefile" );
2016-06-04 13:24:23 +00:00
ASSERT( mapcfile != NULL );
2016-07-31 13:48:50 +00:00
szCommands[0] = '\0';
szRules[0] = '\0';
2016-06-04 13:24:23 +00:00
curplayers = CountPlayers();
// Has the map cycle filename changed?
2016-07-31 13:48:50 +00:00
if( stricmp( mapcfile, szPreviousMapCycleFile ) )
2016-06-04 13:24:23 +00:00
{
strcpy( szPreviousMapCycleFile, mapcfile );
DestroyMapCycle( &mapcycle );
2016-07-31 13:48:50 +00:00
if( !ReloadMapCycleFile( mapcfile, &mapcycle ) || ( !mapcycle.items ) )
2016-06-04 13:24:23 +00:00
{
ALERT( at_console, "Unable to load map cycle file %s\n", mapcfile );
do_cycle = FALSE;
}
}
2016-07-31 13:48:50 +00:00
if( do_cycle && mapcycle.items )
2016-06-04 13:24:23 +00:00
{
BOOL keeplooking = FALSE;
BOOL found = FALSE;
mapcycle_item_s *item;
// Assume current map
2016-07-31 13:48:50 +00:00
strcpy( szNextMap, STRING( gpGlobals->mapname ) );
strcpy( szFirstMapInList, STRING( gpGlobals->mapname ) );
2016-06-04 13:24:23 +00:00
// Traverse list
2016-07-31 13:48:50 +00:00
for( item = mapcycle.next_item; item->next != mapcycle.next_item; item = item->next )
2016-06-04 13:24:23 +00:00
{
keeplooking = FALSE;
ASSERT( item != NULL );
2016-07-31 13:48:50 +00:00
if( item->minplayers != 0 )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( curplayers >= item->minplayers )
2016-06-04 13:24:23 +00:00
{
found = TRUE;
minplayers = item->minplayers;
}
else
{
keeplooking = TRUE;
}
}
2016-07-31 13:48:50 +00:00
if( item->maxplayers != 0 )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( curplayers <= item->maxplayers )
2016-06-04 13:24:23 +00:00
{
found = TRUE;
maxplayers = item->maxplayers;
}
else
{
keeplooking = TRUE;
}
}
2016-07-31 13:48:50 +00:00
if( keeplooking )
2016-06-04 13:24:23 +00:00
continue;
found = TRUE;
break;
}
2016-07-31 13:48:50 +00:00
if( !found )
2016-06-04 13:24:23 +00:00
{
item = mapcycle.next_item;
2016-07-31 13:48:50 +00:00
}
2016-06-04 13:24:23 +00:00
// Increment next item pointer
mapcycle.next_item = item->next;
// Perform logic on current item
strcpy( szNextMap, item->mapname );
ExtractCommandString( item->rulebuffer, szCommands );
strcpy( szRules, item->rulebuffer );
}
2016-07-31 13:48:50 +00:00
if( !IS_MAP_VALID( szNextMap ) )
2016-06-04 13:24:23 +00:00
{
strcpy( szNextMap, szFirstMapInList );
}
g_fGameOver = TRUE;
ALERT( at_console, "CHANGE LEVEL: %s\n", szNextMap );
2016-07-31 13:48:50 +00:00
if( minplayers || maxplayers )
2016-06-04 13:24:23 +00:00
{
ALERT( at_console, "PLAYER COUNT: min %i max %i current %i\n", minplayers, maxplayers, curplayers );
}
2016-07-31 13:48:50 +00:00
if( strlen( szRules ) > 0 )
2016-06-04 13:24:23 +00:00
{
ALERT( at_console, "RULES: %s\n", szRules );
}
2016-07-31 13:48:50 +00:00
2016-06-04 13:24:23 +00:00
CHANGE_LEVEL( szNextMap, NULL );
2016-07-31 13:48:50 +00:00
if( strlen( szCommands ) > 0 )
2016-06-04 13:24:23 +00:00
{
SERVER_COMMAND( szCommands );
}
}
#define MAX_MOTD_CHUNK 60
#define MAX_MOTD_LENGTH 1536 // (MAX_MOTD_CHUNK * 4)
2016-07-31 13:48:50 +00:00
void CHalfLifeMultiplay::SendMOTDToClient( edict_t *client )
2016-06-04 13:24:23 +00:00
{
// read from the MOTD.txt file
int length, char_count = 0;
2018-04-10 23:48:42 +00:00
char *pFileList;
char *aFileList = pFileList = (char*)LOAD_FILE_FOR_ME( CVAR_GET_STRING( "motdfile" ), &length );
2016-06-04 13:24:23 +00:00
// send the server name
MESSAGE_BEGIN( MSG_ONE, gmsgServerName, NULL, client );
2016-07-31 13:48:50 +00:00
WRITE_STRING( CVAR_GET_STRING( "hostname" ) );
2016-06-04 13:24:23 +00:00
MESSAGE_END();
// Send the message of the day
// read it chunk-by-chunk, and send it in parts
2016-07-31 13:48:50 +00:00
while( pFileList && *pFileList && char_count < MAX_MOTD_LENGTH )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
char chunk[MAX_MOTD_CHUNK + 1];
2016-07-31 13:48:50 +00:00
if( strlen( pFileList ) < MAX_MOTD_CHUNK )
2016-06-04 13:24:23 +00:00
{
strcpy( chunk, pFileList );
}
else
{
strncpy( chunk, pFileList, MAX_MOTD_CHUNK );
chunk[MAX_MOTD_CHUNK] = 0; // strncpy doesn't always append the null terminator
}
char_count += strlen( chunk );
2016-07-31 13:48:50 +00:00
if( char_count < MAX_MOTD_LENGTH )
2016-06-04 13:24:23 +00:00
pFileList = aFileList + char_count;
else
2018-04-10 23:48:42 +00:00
*pFileList = 0;
2016-06-04 13:24:23 +00:00
MESSAGE_BEGIN( MSG_ONE, gmsgMOTD, NULL, client );
2018-04-10 23:48:42 +00:00
WRITE_BYTE( *pFileList ? FALSE : TRUE ); // FALSE means there is still more message to come
2016-06-04 13:24:23 +00:00
WRITE_STRING( chunk );
MESSAGE_END();
}
2017-07-23 21:24:55 +00:00
FREE_FILE( (void*)aFileList );
2016-06-04 13:24:23 +00:00
}